blob: 5a68960dfb43241d9e4e8c57db01abe7854c5b3a [file] [log] [blame]
Eric Laurent1c333e22014-05-20 10:48:17 -07001/*
2**
3** Copyright 2014, The Android Open Source Project
4**
5** Licensed under the Apache License, Version 2.0 (the "License");
6** you may not use this file except in compliance with the License.
7** You may obtain a copy of the License at
8**
9** http://www.apache.org/licenses/LICENSE-2.0
10**
11** Unless required by applicable law or agreed to in writing, software
12** distributed under the License is distributed on an "AS IS" BASIS,
13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14** See the License for the specific language governing permissions and
15** limitations under the License.
16*/
17
18#ifndef INCLUDING_FROM_AUDIOFLINGER_H
19 #error This header file should only be included from AudioFlinger.h
20#endif
21
Mikhail Naganovdea53042018-04-26 13:10:21 -070022// PatchPanel is concealed within AudioFlinger, their lifetimes are the same.
23class PatchPanel {
Eric Laurent1c333e22014-05-20 10:48:17 -070024public:
Mikhail Naganovdea53042018-04-26 13:10:21 -070025 explicit PatchPanel(AudioFlinger* audioFlinger) : mAudioFlinger(*audioFlinger) {}
Eric Laurent1c333e22014-05-20 10:48:17 -070026
27 /* List connected audio ports and their attributes */
28 status_t listAudioPorts(unsigned int *num_ports,
29 struct audio_port *ports);
30
31 /* Get supported attributes for a given audio port */
32 status_t getAudioPort(struct audio_port *port);
33
34 /* Create a patch between several source and sink ports */
35 status_t createAudioPatch(const struct audio_patch *patch,
36 audio_patch_handle_t *handle);
37
38 /* Release a patch */
39 status_t releaseAudioPatch(audio_patch_handle_t handle);
40
41 /* List connected audio devices and they attributes */
42 status_t listAudioPatches(unsigned int *num_patches,
43 struct audio_patch *patches);
44
Mikhail Naganovdea53042018-04-26 13:10:21 -070045private:
Mikhail Naganov444ecc32018-05-01 17:40:05 -070046 template<typename ThreadType, typename TrackType>
47 class Endpoint {
48 public:
49 status_t checkTrack(TrackType *trackOrNull) const {
50 if (trackOrNull == nullptr) return NO_MEMORY;
51 return trackOrNull->initCheck();
52 }
53 audio_patch_handle_t handle() const { return mHandle; }
54 sp<ThreadType> thread() { return mThread; }
55 sp<TrackType> track() { return mTrack; }
56
57 void closeConnections(PatchPanel *panel) {
58 if (mHandle != AUDIO_PATCH_HANDLE_NONE) {
59 panel->releaseAudioPatch(mHandle);
60 mHandle = AUDIO_PATCH_HANDLE_NONE;
61 }
62 if (mThread != 0) {
63 if (mTrack != 0) {
64 mThread->deletePatchTrack(mTrack);
65 }
66 if (mCloseThread) {
67 panel->mAudioFlinger.closeThreadInternal_l(mThread);
68 }
69 }
70 }
71 audio_patch_handle_t* handlePtr() { return &mHandle; }
72 void setThread(const sp<ThreadType>& thread, bool closeThread = true) {
73 mThread = thread;
74 mCloseThread = closeThread;
75 }
76 void setTrackAndPeer(const sp<TrackType>& track,
77 ThreadBase::PatchProxyBufferProvider *peer) {
78 mTrack = track;
79 mThread->addPatchTrack(mTrack);
80 mTrack->setPeerProxy(peer);
81 }
82 void stopTrack() { if (mTrack) mTrack->stop(); }
83
84 private:
85 sp<ThreadType> mThread;
86 bool mCloseThread = true;
87 audio_patch_handle_t mHandle = AUDIO_PATCH_HANDLE_NONE;
88 sp<TrackType> mTrack;
89 };
90
Eric Laurent1c333e22014-05-20 10:48:17 -070091 class Patch {
92 public:
Mikhail Naganovdea53042018-04-26 13:10:21 -070093 explicit Patch(const struct audio_patch &patch) : mAudioPatch(patch) {}
Mikhail Naganov444ecc32018-05-01 17:40:05 -070094 ~Patch();
Eric Laurent1c333e22014-05-20 10:48:17 -070095
Mikhail Naganovdea53042018-04-26 13:10:21 -070096 status_t createConnections(PatchPanel *panel);
97 void clearConnections(PatchPanel *panel);
Mikhail Naganov444ecc32018-05-01 17:40:05 -070098 bool isSoftware() const {
99 return mRecord.handle() != AUDIO_PATCH_HANDLE_NONE ||
100 mPlayback.handle() != AUDIO_PATCH_HANDLE_NONE; }
Mikhail Naganovdea53042018-04-26 13:10:21 -0700101
102 // Note that audio_patch::id is only unique within a HAL module
Eric Laurent83b88082014-06-20 18:31:16 -0700103 struct audio_patch mAudioPatch;
Eric Laurentb997d3a2016-06-07 18:23:45 -0700104 // handle for audio HAL patch handle present only when the audio HAL version is >= 3.0
Mikhail Naganovdea53042018-04-26 13:10:21 -0700105 audio_patch_handle_t mHalHandle = AUDIO_PATCH_HANDLE_NONE;
Eric Laurentb997d3a2016-06-07 18:23:45 -0700106 // below members are used by a software audio patch connecting a source device from a
107 // given audio HW module to a sink device on an other audio HW module.
Mikhail Naganovdea53042018-04-26 13:10:21 -0700108 // the objects are created by createConnections() and released by clearConnections()
109 // playback thread is created if no existing playback thread can be used
Mikhail Naganov444ecc32018-05-01 17:40:05 -0700110 // connects playback thread output to sink device
111 Endpoint<PlaybackThread, PlaybackThread::PatchTrack> mPlayback;
112 // connects source device to record thread input
113 Endpoint<RecordThread, RecordThread::PatchRecord> mRecord;
Eric Laurent1c333e22014-05-20 10:48:17 -0700114 };
Eric Laurent83b88082014-06-20 18:31:16 -0700115
Mikhail Naganov444ecc32018-05-01 17:40:05 -0700116 sp<DeviceHalInterface> findHwDeviceByModule(audio_module_handle_t module);
117
Mikhail Naganovdea53042018-04-26 13:10:21 -0700118 AudioFlinger &mAudioFlinger;
119 std::map<audio_patch_handle_t, Patch> mPatches;
Eric Laurent1c333e22014-05-20 10:48:17 -0700120};