| Glenn Kasten | 99e53b8 | 2012-01-19 08:59:58 -0800 | [diff] [blame] | 1 | /* | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 2 | ** | 
 | 3 | ** Copyright 2007, 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 ANDROID_AUDIO_FLINGER_H | 
 | 19 | #define ANDROID_AUDIO_FLINGER_H | 
 | 20 |  | 
 | 21 | #include <stdint.h> | 
 | 22 | #include <sys/types.h> | 
 | 23 | #include <limits.h> | 
 | 24 |  | 
| John Grossman | 4ff14ba | 2012-02-08 16:37:41 -0800 | [diff] [blame] | 25 | #include <common_time/cc_helper.h> | 
 | 26 |  | 
| Mathias Agopian | e762be9 | 2013-05-09 16:26:45 -0700 | [diff] [blame] | 27 | #include <cutils/compiler.h> | 
 | 28 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 29 | #include <media/IAudioFlinger.h> | 
 | 30 | #include <media/IAudioFlingerClient.h> | 
 | 31 | #include <media/IAudioTrack.h> | 
 | 32 | #include <media/IAudioRecord.h> | 
| Glenn Kasten | 335787f | 2012-01-20 17:00:00 -0800 | [diff] [blame] | 33 | #include <media/AudioSystem.h> | 
| John Grossman | 4ff14ba | 2012-02-08 16:37:41 -0800 | [diff] [blame] | 34 | #include <media/AudioTrack.h> | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 35 |  | 
 | 36 | #include <utils/Atomic.h> | 
 | 37 | #include <utils/Errors.h> | 
 | 38 | #include <utils/threads.h> | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 39 | #include <utils/SortedVector.h> | 
| Dima Zavin | 799a70e | 2011-04-18 16:57:27 -0700 | [diff] [blame] | 40 | #include <utils/TypeHelpers.h> | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 41 | #include <utils/Vector.h> | 
 | 42 |  | 
| Mathias Agopian | 5462fc9 | 2010-07-14 18:41:18 -0700 | [diff] [blame] | 43 | #include <binder/BinderService.h> | 
 | 44 | #include <binder/MemoryDealer.h> | 
 | 45 |  | 
| Dima Zavin | 6476024 | 2011-05-11 14:15:23 -0700 | [diff] [blame] | 46 | #include <system/audio.h> | 
| Dima Zavin | 7394a4f | 2011-06-13 18:16:26 -0700 | [diff] [blame] | 47 | #include <hardware/audio.h> | 
| Eric Laurent | a4c5a55 | 2012-03-29 10:12:40 -0700 | [diff] [blame] | 48 | #include <hardware/audio_policy.h> | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 49 |  | 
| Glenn Kasten | 2dd4bdd | 2012-08-29 11:10:32 -0700 | [diff] [blame] | 50 | #include <media/AudioBufferProvider.h> | 
 | 51 | #include <media/ExtendedAudioBufferProvider.h> | 
| Glenn Kasten | 5891256 | 2012-04-03 10:45:00 -0700 | [diff] [blame] | 52 | #include "FastMixer.h" | 
| Glenn Kasten | 2dd4bdd | 2012-08-29 11:10:32 -0700 | [diff] [blame] | 53 | #include <media/nbaio/NBAIO.h> | 
| Glenn Kasten | c15d665 | 2012-05-30 14:52:57 -0700 | [diff] [blame] | 54 | #include "AudioWatchdog.h" | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 55 |  | 
| Eric Laurent | feb0db6 | 2011-07-22 09:04:31 -0700 | [diff] [blame] | 56 | #include <powermanager/IPowerManager.h> | 
 | 57 |  | 
| Glenn Kasten | 9e58b55 | 2013-01-18 15:09:48 -0800 | [diff] [blame] | 58 | #include <media/nbaio/NBLog.h> | 
| Glenn Kasten | 9f80dd2 | 2012-12-18 15:57:32 -0800 | [diff] [blame] | 59 | #include <private/media/AudioTrackShared.h> | 
| Glenn Kasten | 9e58b55 | 2013-01-18 15:09:48 -0800 | [diff] [blame] | 60 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 61 | namespace android { | 
 | 62 |  | 
 | 63 | class audio_track_cblk_t; | 
 | 64 | class effect_param_cblk_t; | 
 | 65 | class AudioMixer; | 
 | 66 | class AudioBuffer; | 
 | 67 | class AudioResampler; | 
| Glenn Kasten | 5891256 | 2012-04-03 10:45:00 -0700 | [diff] [blame] | 68 | class FastMixer; | 
| Glenn Kasten | e3aa659 | 2012-12-04 12:22:46 -0800 | [diff] [blame] | 69 | class ServerProxy; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 70 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 71 | // ---------------------------------------------------------------------------- | 
 | 72 |  | 
| Glenn Kasten | 53d76db | 2012-03-08 12:32:47 -0800 | [diff] [blame] | 73 | // AudioFlinger has a hard-coded upper limit of 2 channels for capture and playback. | 
 | 74 | // There is support for > 2 channel tracks down-mixed to 2 channel output via a down-mix effect. | 
 | 75 | // Adding full support for > 2 channel capture or playback would require more than simply changing | 
 | 76 | // this #define.  There is an independent hard-coded upper limit in AudioMixer; | 
 | 77 | // removing that AudioMixer limit would be necessary but insufficient to support > 2 channels. | 
 | 78 | // The macro FCC_2 highlights some (but not all) places where there is are 2-channel assumptions. | 
 | 79 | // Search also for "2", "left", "right", "[0]", "[1]", ">> 16", "<< 16", etc. | 
 | 80 | #define FCC_2 2     // FCC_2 = Fixed Channel Count 2 | 
 | 81 |  | 
| John Grossman | 4ff14ba | 2012-02-08 16:37:41 -0800 | [diff] [blame] | 82 | static const nsecs_t kDefaultStandbyTimeInNsecs = seconds(3); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 83 |  | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 84 | #define MAX_GAIN 4096.0f | 
 | 85 | #define MAX_GAIN_INT 0x1000 | 
 | 86 |  | 
 | 87 | #define INCLUDING_FROM_AUDIOFLINGER_H | 
 | 88 |  | 
| Mathias Agopian | 5462fc9 | 2010-07-14 18:41:18 -0700 | [diff] [blame] | 89 | class AudioFlinger : | 
 | 90 |     public BinderService<AudioFlinger>, | 
 | 91 |     public BnAudioFlinger | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 92 | { | 
| Glenn Kasten | 1998661 | 2012-03-09 12:07:30 -0800 | [diff] [blame] | 93 |     friend class BinderService<AudioFlinger>;   // for AudioFlinger() | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 94 | public: | 
| Mathias Agopian | e762be9 | 2013-05-09 16:26:45 -0700 | [diff] [blame] | 95 |     static const char* getServiceName() ANDROID_API { return "media.audio_flinger"; } | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 96 |  | 
 | 97 |     virtual     status_t    dump(int fd, const Vector<String16>& args); | 
 | 98 |  | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 99 |     // IAudioFlinger interface, in binder opcode order | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 100 |     virtual sp<IAudioTrack> createTrack( | 
| Glenn Kasten | fff6d71 | 2012-01-12 16:38:12 -0800 | [diff] [blame] | 101 |                                 audio_stream_type_t streamType, | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 102 |                                 uint32_t sampleRate, | 
| Glenn Kasten | 58f3021 | 2012-01-12 12:27:51 -0800 | [diff] [blame] | 103 |                                 audio_format_t format, | 
| Glenn Kasten | dd8104c | 2012-07-02 12:42:44 -0700 | [diff] [blame] | 104 |                                 audio_channel_mask_t channelMask, | 
| Glenn Kasten | e33054e | 2012-11-14 12:54:39 -0800 | [diff] [blame] | 105 |                                 size_t frameCount, | 
| Glenn Kasten | e0b0717 | 2012-11-06 15:03:34 -0800 | [diff] [blame] | 106 |                                 IAudioFlinger::track_flags_t *flags, | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 107 |                                 const sp<IMemory>& sharedBuffer, | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 108 |                                 audio_io_handle_t output, | 
| Glenn Kasten | 3acbd05 | 2012-02-28 10:39:56 -0800 | [diff] [blame] | 109 |                                 pid_t tid, | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 110 |                                 int *sessionId, | 
| Glenn Kasten | d054c32 | 2013-07-12 12:59:20 -0700 | [diff] [blame] | 111 |                                 String8& name, | 
| Marco Nelissen | 462fd2f | 2013-01-14 14:12:05 -0800 | [diff] [blame^] | 112 |                                 int clientUid, | 
| Glenn Kasten | 9156ef3 | 2013-08-06 15:39:08 -0700 | [diff] [blame] | 113 |                                 status_t *status /*non-NULL*/); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 114 |  | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 115 |     virtual sp<IAudioRecord> openRecord( | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 116 |                                 audio_io_handle_t input, | 
 | 117 |                                 uint32_t sampleRate, | 
 | 118 |                                 audio_format_t format, | 
| Glenn Kasten | dd8104c | 2012-07-02 12:42:44 -0700 | [diff] [blame] | 119 |                                 audio_channel_mask_t channelMask, | 
| Glenn Kasten | e33054e | 2012-11-14 12:54:39 -0800 | [diff] [blame] | 120 |                                 size_t frameCount, | 
| Glenn Kasten | eeca326 | 2013-07-31 16:12:48 -0700 | [diff] [blame] | 121 |                                 IAudioFlinger::track_flags_t *flags, | 
| Glenn Kasten | 1879fff | 2012-07-11 15:36:59 -0700 | [diff] [blame] | 122 |                                 pid_t tid, | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 123 |                                 int *sessionId, | 
| Glenn Kasten | 9156ef3 | 2013-08-06 15:39:08 -0700 | [diff] [blame] | 124 |                                 status_t *status /*non-NULL*/); | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 125 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 126 |     virtual     uint32_t    sampleRate(audio_io_handle_t output) const; | 
 | 127 |     virtual     int         channelCount(audio_io_handle_t output) const; | 
 | 128 |     virtual     audio_format_t format(audio_io_handle_t output) const; | 
 | 129 |     virtual     size_t      frameCount(audio_io_handle_t output) const; | 
 | 130 |     virtual     uint32_t    latency(audio_io_handle_t output) const; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 131 |  | 
 | 132 |     virtual     status_t    setMasterVolume(float value); | 
 | 133 |     virtual     status_t    setMasterMute(bool muted); | 
 | 134 |  | 
 | 135 |     virtual     float       masterVolume() const; | 
 | 136 |     virtual     bool        masterMute() const; | 
 | 137 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 138 |     virtual     status_t    setStreamVolume(audio_stream_type_t stream, float value, | 
 | 139 |                                             audio_io_handle_t output); | 
| Glenn Kasten | fff6d71 | 2012-01-12 16:38:12 -0800 | [diff] [blame] | 140 |     virtual     status_t    setStreamMute(audio_stream_type_t stream, bool muted); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 141 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 142 |     virtual     float       streamVolume(audio_stream_type_t stream, | 
 | 143 |                                          audio_io_handle_t output) const; | 
| Glenn Kasten | fff6d71 | 2012-01-12 16:38:12 -0800 | [diff] [blame] | 144 |     virtual     bool        streamMute(audio_stream_type_t stream) const; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 145 |  | 
| Glenn Kasten | f78aee7 | 2012-01-04 11:00:47 -0800 | [diff] [blame] | 146 |     virtual     status_t    setMode(audio_mode_t mode); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 147 |  | 
 | 148 |     virtual     status_t    setMicMute(bool state); | 
 | 149 |     virtual     bool        getMicMute() const; | 
 | 150 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 151 |     virtual     status_t    setParameters(audio_io_handle_t ioHandle, const String8& keyValuePairs); | 
 | 152 |     virtual     String8     getParameters(audio_io_handle_t ioHandle, const String8& keys) const; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 153 |  | 
 | 154 |     virtual     void        registerClient(const sp<IAudioFlingerClient>& client); | 
 | 155 |  | 
| Glenn Kasten | dd8104c | 2012-07-02 12:42:44 -0700 | [diff] [blame] | 156 |     virtual     size_t      getInputBufferSize(uint32_t sampleRate, audio_format_t format, | 
 | 157 |                                                audio_channel_mask_t channelMask) const; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 158 |  | 
| Eric Laurent | a4c5a55 | 2012-03-29 10:12:40 -0700 | [diff] [blame] | 159 |     virtual audio_io_handle_t openOutput(audio_module_handle_t module, | 
 | 160 |                                          audio_devices_t *pDevices, | 
 | 161 |                                          uint32_t *pSamplingRate, | 
 | 162 |                                          audio_format_t *pFormat, | 
 | 163 |                                          audio_channel_mask_t *pChannelMask, | 
 | 164 |                                          uint32_t *pLatencyMs, | 
| Richard Fitzgerald | ad3af33 | 2013-03-25 16:54:37 +0000 | [diff] [blame] | 165 |                                          audio_output_flags_t flags, | 
 | 166 |                                          const audio_offload_info_t *offloadInfo); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 167 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 168 |     virtual audio_io_handle_t openDuplicateOutput(audio_io_handle_t output1, | 
 | 169 |                                                   audio_io_handle_t output2); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 170 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 171 |     virtual status_t closeOutput(audio_io_handle_t output); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 172 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 173 |     virtual status_t suspendOutput(audio_io_handle_t output); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 174 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 175 |     virtual status_t restoreOutput(audio_io_handle_t output); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 176 |  | 
| Eric Laurent | a4c5a55 | 2012-03-29 10:12:40 -0700 | [diff] [blame] | 177 |     virtual audio_io_handle_t openInput(audio_module_handle_t module, | 
 | 178 |                                         audio_devices_t *pDevices, | 
 | 179 |                                         uint32_t *pSamplingRate, | 
 | 180 |                                         audio_format_t *pFormat, | 
 | 181 |                                         audio_channel_mask_t *pChannelMask); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 182 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 183 |     virtual status_t closeInput(audio_io_handle_t input); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 184 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 185 |     virtual status_t setStreamOutput(audio_stream_type_t stream, audio_io_handle_t output); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 186 |  | 
 | 187 |     virtual status_t setVoiceVolume(float volume); | 
 | 188 |  | 
| Glenn Kasten | 26c7755 | 2012-11-16 12:01:44 -0800 | [diff] [blame] | 189 |     virtual status_t getRenderPosition(size_t *halFrames, size_t *dspFrames, | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 190 |                                        audio_io_handle_t output) const; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 191 |  | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 192 |     virtual     unsigned int  getInputFramesLost(audio_io_handle_t ioHandle) const; | 
 | 193 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 194 |     virtual int newAudioSessionId(); | 
 | 195 |  | 
| Marco Nelissen | 3a34bef | 2011-08-02 13:33:41 -0700 | [diff] [blame] | 196 |     virtual void acquireAudioSessionId(int audioSession); | 
 | 197 |  | 
 | 198 |     virtual void releaseAudioSessionId(int audioSession); | 
 | 199 |  | 
| Glenn Kasten | f587ba5 | 2012-01-26 16:25:10 -0800 | [diff] [blame] | 200 |     virtual status_t queryNumberEffects(uint32_t *numEffects) const; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 201 |  | 
| Glenn Kasten | f587ba5 | 2012-01-26 16:25:10 -0800 | [diff] [blame] | 202 |     virtual status_t queryEffect(uint32_t index, effect_descriptor_t *descriptor) const; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 203 |  | 
| Glenn Kasten | 5e92a78 | 2012-01-30 07:40:52 -0800 | [diff] [blame] | 204 |     virtual status_t getEffectDescriptor(const effect_uuid_t *pUuid, | 
| Glenn Kasten | f587ba5 | 2012-01-26 16:25:10 -0800 | [diff] [blame] | 205 |                                          effect_descriptor_t *descriptor) const; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 206 |  | 
| Glenn Kasten | 8d6cc84 | 2012-02-03 11:06:53 -0800 | [diff] [blame] | 207 |     virtual sp<IEffect> createEffect( | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 208 |                         effect_descriptor_t *pDesc, | 
 | 209 |                         const sp<IEffectClient>& effectClient, | 
 | 210 |                         int32_t priority, | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 211 |                         audio_io_handle_t io, | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 212 |                         int sessionId, | 
| Glenn Kasten | 9156ef3 | 2013-08-06 15:39:08 -0700 | [diff] [blame] | 213 |                         status_t *status /*non-NULL*/, | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 214 |                         int *id, | 
 | 215 |                         int *enabled); | 
 | 216 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 217 |     virtual status_t moveEffects(int sessionId, audio_io_handle_t srcOutput, | 
 | 218 |                         audio_io_handle_t dstOutput); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 219 |  | 
| Eric Laurent | a4c5a55 | 2012-03-29 10:12:40 -0700 | [diff] [blame] | 220 |     virtual audio_module_handle_t loadHwModule(const char *name); | 
 | 221 |  | 
| Glenn Kasten | 3b16c76 | 2012-11-14 08:44:39 -0800 | [diff] [blame] | 222 |     virtual uint32_t getPrimaryOutputSamplingRate(); | 
| Glenn Kasten | e33054e | 2012-11-14 12:54:39 -0800 | [diff] [blame] | 223 |     virtual size_t getPrimaryOutputFrameCount(); | 
| Glenn Kasten | cc0f1cf | 2012-09-24 11:27:18 -0700 | [diff] [blame] | 224 |  | 
| Glenn Kasten | 4182c4e | 2013-07-15 14:45:07 -0700 | [diff] [blame] | 225 |     virtual status_t setLowRamDevice(bool isLowRamDevice); | 
 | 226 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 227 |     virtual     status_t    onTransact( | 
 | 228 |                                 uint32_t code, | 
 | 229 |                                 const Parcel& data, | 
 | 230 |                                 Parcel* reply, | 
 | 231 |                                 uint32_t flags); | 
 | 232 |  | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 233 |     // end of IAudioFlinger interface | 
 | 234 |  | 
| Glenn Kasten | 9e58b55 | 2013-01-18 15:09:48 -0800 | [diff] [blame] | 235 |     sp<NBLog::Writer>   newWriter_l(size_t size, const char *name); | 
 | 236 |     void                unregisterWriter(const sp<NBLog::Writer>& writer); | 
 | 237 | private: | 
| Glenn Kasten | 7f5d335 | 2013-02-15 23:55:04 +0000 | [diff] [blame] | 238 |     static const size_t kLogMemorySize = 10 * 1024; | 
| Glenn Kasten | 9e58b55 | 2013-01-18 15:09:48 -0800 | [diff] [blame] | 239 |     sp<MemoryDealer>    mLogMemoryDealer;   // == 0 when NBLog is disabled | 
 | 240 | public: | 
 | 241 |  | 
| Eric Laurent | a011e35 | 2012-03-29 15:51:43 -0700 | [diff] [blame] | 242 |     class SyncEvent; | 
 | 243 |  | 
 | 244 |     typedef void (*sync_event_callback_t)(const wp<SyncEvent>& event) ; | 
 | 245 |  | 
 | 246 |     class SyncEvent : public RefBase { | 
 | 247 |     public: | 
 | 248 |         SyncEvent(AudioSystem::sync_event_t type, | 
 | 249 |                   int triggerSession, | 
 | 250 |                   int listenerSession, | 
 | 251 |                   sync_event_callback_t callBack, | 
 | 252 |                   void *cookie) | 
 | 253 |         : mType(type), mTriggerSession(triggerSession), mListenerSession(listenerSession), | 
 | 254 |           mCallback(callBack), mCookie(cookie) | 
 | 255 |         {} | 
 | 256 |  | 
 | 257 |         virtual ~SyncEvent() {} | 
 | 258 |  | 
 | 259 |         void trigger() { Mutex::Autolock _l(mLock); if (mCallback) mCallback(this); } | 
| Glenn Kasten | 106e8a4 | 2012-08-02 13:37:12 -0700 | [diff] [blame] | 260 |         bool isCancelled() const { Mutex::Autolock _l(mLock); return (mCallback == NULL); } | 
 | 261 |         void cancel() { Mutex::Autolock _l(mLock); mCallback = NULL; } | 
| Eric Laurent | a011e35 | 2012-03-29 15:51:43 -0700 | [diff] [blame] | 262 |         AudioSystem::sync_event_t type() const { return mType; } | 
 | 263 |         int triggerSession() const { return mTriggerSession; } | 
 | 264 |         int listenerSession() const { return mListenerSession; } | 
 | 265 |         void *cookie() const { return mCookie; } | 
 | 266 |  | 
 | 267 |     private: | 
 | 268 |           const AudioSystem::sync_event_t mType; | 
 | 269 |           const int mTriggerSession; | 
 | 270 |           const int mListenerSession; | 
 | 271 |           sync_event_callback_t mCallback; | 
 | 272 |           void * const mCookie; | 
| Glenn Kasten | 106e8a4 | 2012-08-02 13:37:12 -0700 | [diff] [blame] | 273 |           mutable Mutex mLock; | 
| Eric Laurent | a011e35 | 2012-03-29 15:51:43 -0700 | [diff] [blame] | 274 |     }; | 
 | 275 |  | 
 | 276 |     sp<SyncEvent> createSyncEvent(AudioSystem::sync_event_t type, | 
 | 277 |                                         int triggerSession, | 
 | 278 |                                         int listenerSession, | 
 | 279 |                                         sync_event_callback_t callBack, | 
 | 280 |                                         void *cookie); | 
| Eric Laurent | 717e128 | 2012-06-29 16:36:52 -0700 | [diff] [blame] | 281 |  | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 282 | private: | 
| John Grossman | ee578c0 | 2012-07-23 17:05:46 -0700 | [diff] [blame] | 283 |     class AudioHwDevice;    // fwd declaration for findSuitableHwDev_l | 
 | 284 |  | 
| Glenn Kasten | f78aee7 | 2012-01-04 11:00:47 -0800 | [diff] [blame] | 285 |                audio_mode_t getMode() const { return mMode; } | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 286 |  | 
| Glenn Kasten | c59c004 | 2012-02-02 14:06:11 -0800 | [diff] [blame] | 287 |                 bool        btNrecIsOff() const { return mBtNrecIsOff; } | 
| Eric Laurent | 59bd0da | 2011-08-01 09:52:20 -0700 | [diff] [blame] | 288 |  | 
| Mathias Agopian | e762be9 | 2013-05-09 16:26:45 -0700 | [diff] [blame] | 289 |                             AudioFlinger() ANDROID_API; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 290 |     virtual                 ~AudioFlinger(); | 
 | 291 |  | 
| Glenn Kasten | 2b213bc | 2012-02-02 14:05:20 -0800 | [diff] [blame] | 292 |     // call in any IAudioFlinger method that accesses mPrimaryHardwareDev | 
| Glenn Kasten | 85ab62c | 2012-11-01 11:11:38 -0700 | [diff] [blame] | 293 |     status_t                initCheck() const { return mPrimaryHardwareDev == NULL ? | 
 | 294 |                                                         NO_INIT : NO_ERROR; } | 
| Glenn Kasten | 2b213bc | 2012-02-02 14:05:20 -0800 | [diff] [blame] | 295 |  | 
| Glenn Kasten | 000f0e3 | 2012-03-01 17:10:56 -0800 | [diff] [blame] | 296 |     // RefBase | 
| Dima Zavin | 5a61d2f | 2011-04-19 19:04:32 -0700 | [diff] [blame] | 297 |     virtual     void        onFirstRef(); | 
| Glenn Kasten | 000f0e3 | 2012-03-01 17:10:56 -0800 | [diff] [blame] | 298 |  | 
| Glenn Kasten | 85ab62c | 2012-11-01 11:11:38 -0700 | [diff] [blame] | 299 |     AudioHwDevice*          findSuitableHwDev_l(audio_module_handle_t module, | 
 | 300 |                                                 audio_devices_t devices); | 
| Marco Nelissen | 3a34bef | 2011-08-02 13:33:41 -0700 | [diff] [blame] | 301 |     void                    purgeStaleEffects_l(); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 302 |  | 
| Glenn Kasten | 66fcab9 | 2012-02-24 14:59:21 -0800 | [diff] [blame] | 303 |     // standby delay for MIXER and DUPLICATING playback threads is read from property | 
 | 304 |     // ro.audio.flinger_standbytime_ms or defaults to kDefaultStandbyTimeInNsecs | 
| John Grossman | 4ff14ba | 2012-02-08 16:37:41 -0800 | [diff] [blame] | 305 |     static nsecs_t          mStandbyTimeInNsecs; | 
 | 306 |  | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 307 |     // incremented by 2 when screen state changes, bit 0 == 1 means "off" | 
 | 308 |     // AudioFlinger::setParameters() updates, other threads read w/o lock | 
 | 309 |     static uint32_t         mScreenState; | 
 | 310 |  | 
| Glenn Kasten | be5f05e | 2012-07-18 15:24:02 -0700 | [diff] [blame] | 311 |     // Internal dump utilities. | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 312 |     static const int kDumpLockRetries = 50; | 
 | 313 |     static const int kDumpLockSleepUs = 20000; | 
 | 314 |     static bool dumpTryLock(Mutex& mutex); | 
| Glenn Kasten | be5f05e | 2012-07-18 15:24:02 -0700 | [diff] [blame] | 315 |     void dumpPermissionDenial(int fd, const Vector<String16>& args); | 
 | 316 |     void dumpClients(int fd, const Vector<String16>& args); | 
 | 317 |     void dumpInternals(int fd, const Vector<String16>& args); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 318 |  | 
 | 319 |     // --- Client --- | 
 | 320 |     class Client : public RefBase { | 
 | 321 |     public: | 
 | 322 |                             Client(const sp<AudioFlinger>& audioFlinger, pid_t pid); | 
 | 323 |         virtual             ~Client(); | 
| Glenn Kasten | 435dbe6 | 2012-01-30 10:15:48 -0800 | [diff] [blame] | 324 |         sp<MemoryDealer>    heap() const; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 325 |         pid_t               pid() const { return mPid; } | 
| Glenn Kasten | c59c004 | 2012-02-02 14:06:11 -0800 | [diff] [blame] | 326 |         sp<AudioFlinger>    audioFlinger() const { return mAudioFlinger; } | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 327 |  | 
| John Grossman | 4ff14ba | 2012-02-08 16:37:41 -0800 | [diff] [blame] | 328 |         bool reserveTimedTrack(); | 
 | 329 |         void releaseTimedTrack(); | 
 | 330 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 331 |     private: | 
 | 332 |                             Client(const Client&); | 
 | 333 |                             Client& operator = (const Client&); | 
| Glenn Kasten | 84afa3b | 2012-01-25 15:28:08 -0800 | [diff] [blame] | 334 |         const sp<AudioFlinger> mAudioFlinger; | 
 | 335 |         const sp<MemoryDealer> mMemoryDealer; | 
 | 336 |         const pid_t         mPid; | 
| John Grossman | 4ff14ba | 2012-02-08 16:37:41 -0800 | [diff] [blame] | 337 |  | 
 | 338 |         Mutex               mTimedTrackLock; | 
 | 339 |         int                 mTimedTrackCount; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 340 |     }; | 
 | 341 |  | 
 | 342 |     // --- Notification Client --- | 
 | 343 |     class NotificationClient : public IBinder::DeathRecipient { | 
 | 344 |     public: | 
 | 345 |                             NotificationClient(const sp<AudioFlinger>& audioFlinger, | 
 | 346 |                                                 const sp<IAudioFlingerClient>& client, | 
 | 347 |                                                 pid_t pid); | 
 | 348 |         virtual             ~NotificationClient(); | 
 | 349 |  | 
| Glenn Kasten | 84afa3b | 2012-01-25 15:28:08 -0800 | [diff] [blame] | 350 |                 sp<IAudioFlingerClient> audioFlingerClient() const { return mAudioFlingerClient; } | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 351 |  | 
 | 352 |                 // IBinder::DeathRecipient | 
 | 353 |                 virtual     void        binderDied(const wp<IBinder>& who); | 
 | 354 |  | 
 | 355 |     private: | 
 | 356 |                             NotificationClient(const NotificationClient&); | 
 | 357 |                             NotificationClient& operator = (const NotificationClient&); | 
 | 358 |  | 
| Glenn Kasten | 84afa3b | 2012-01-25 15:28:08 -0800 | [diff] [blame] | 359 |         const sp<AudioFlinger>  mAudioFlinger; | 
 | 360 |         const pid_t             mPid; | 
 | 361 |         const sp<IAudioFlingerClient> mAudioFlingerClient; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 362 |     }; | 
 | 363 |  | 
 | 364 |     class TrackHandle; | 
 | 365 |     class RecordHandle; | 
 | 366 |     class RecordThread; | 
 | 367 |     class PlaybackThread; | 
 | 368 |     class MixerThread; | 
 | 369 |     class DirectOutputThread; | 
| Eric Laurent | bfb1b83 | 2013-01-07 09:53:42 -0800 | [diff] [blame] | 370 |     class OffloadThread; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 371 |     class DuplicatingThread; | 
| Eric Laurent | bfb1b83 | 2013-01-07 09:53:42 -0800 | [diff] [blame] | 372 |     class AsyncCallbackThread; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 373 |     class Track; | 
 | 374 |     class RecordTrack; | 
 | 375 |     class EffectModule; | 
 | 376 |     class EffectHandle; | 
 | 377 |     class EffectChain; | 
| Dima Zavin | 799a70e | 2011-04-18 16:57:27 -0700 | [diff] [blame] | 378 |     struct AudioStreamOut; | 
 | 379 |     struct AudioStreamIn; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 380 |  | 
| Glenn Kasten | b7bf796 | 2012-02-08 12:36:25 -0800 | [diff] [blame] | 381 |     struct  stream_type_t { | 
 | 382 |         stream_type_t() | 
 | 383 |             :   volume(1.0f), | 
| Eric Laurent | 1a9ed11 | 2012-03-20 18:36:01 -0700 | [diff] [blame] | 384 |                 mute(false) | 
| Glenn Kasten | b7bf796 | 2012-02-08 12:36:25 -0800 | [diff] [blame] | 385 |         { | 
 | 386 |         } | 
 | 387 |         float       volume; | 
 | 388 |         bool        mute; | 
| Glenn Kasten | b7bf796 | 2012-02-08 12:36:25 -0800 | [diff] [blame] | 389 |     }; | 
 | 390 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 391 |     // --- PlaybackThread --- | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 392 |  | 
 | 393 | #include "Threads.h" | 
 | 394 |  | 
 | 395 | #include "Effects.h" | 
 | 396 |  | 
 | 397 |     // server side of the client's IAudioTrack | 
 | 398 |     class TrackHandle : public android::BnAudioTrack { | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 399 |     public: | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 400 |                             TrackHandle(const sp<PlaybackThread::Track>& track); | 
 | 401 |         virtual             ~TrackHandle(); | 
 | 402 |         virtual sp<IMemory> getCblk() const; | 
 | 403 |         virtual status_t    start(); | 
 | 404 |         virtual void        stop(); | 
 | 405 |         virtual void        flush(); | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 406 |         virtual void        pause(); | 
 | 407 |         virtual status_t    attachAuxEffect(int effectId); | 
 | 408 |         virtual status_t    allocateTimedBuffer(size_t size, | 
 | 409 |                                                 sp<IMemory>* buffer); | 
 | 410 |         virtual status_t    queueTimedBuffer(const sp<IMemory>& buffer, | 
 | 411 |                                              int64_t pts); | 
 | 412 |         virtual status_t    setMediaTimeTransform(const LinearTransform& xform, | 
 | 413 |                                                   int target); | 
| Glenn Kasten | 3dcd00d | 2013-07-17 10:10:23 -0700 | [diff] [blame] | 414 |         virtual status_t    setParameters(const String8& keyValuePairs); | 
| Glenn Kasten | 53cec22 | 2013-08-29 09:01:02 -0700 | [diff] [blame] | 415 |         virtual status_t    getTimestamp(AudioTimestamp& timestamp); | 
| Eric Laurent | 59fe010 | 2013-09-27 18:48:26 -0700 | [diff] [blame] | 416 |         virtual void        signal(); // signal playback thread for a change in control block | 
| Glenn Kasten | 3dcd00d | 2013-07-17 10:10:23 -0700 | [diff] [blame] | 417 |  | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 418 |         virtual status_t onTransact( | 
 | 419 |             uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags); | 
| Richard Fitzgerald | ad3af33 | 2013-03-25 16:54:37 +0000 | [diff] [blame] | 420 |  | 
| Glenn Kasten | 9806710 | 2011-12-13 11:47:54 -0800 | [diff] [blame] | 421 |     private: | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 422 |         const sp<PlaybackThread::Track> mTrack; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 423 |     }; | 
 | 424 |  | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 425 |     // server side of the client's IAudioRecord | 
 | 426 |     class RecordHandle : public android::BnAudioRecord { | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 427 |     public: | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 428 |         RecordHandle(const sp<RecordThread::RecordTrack>& recordTrack); | 
 | 429 |         virtual             ~RecordHandle(); | 
 | 430 |         virtual sp<IMemory> getCblk() const; | 
 | 431 |         virtual status_t    start(int /*AudioSystem::sync_event_t*/ event, int triggerSession); | 
 | 432 |         virtual void        stop(); | 
 | 433 |         virtual status_t onTransact( | 
 | 434 |             uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags); | 
| Glenn Kasten | 5891256 | 2012-04-03 10:45:00 -0700 | [diff] [blame] | 435 |     private: | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 436 |         const sp<RecordThread::RecordTrack> mRecordTrack; | 
| Glenn Kasten | 5891256 | 2012-04-03 10:45:00 -0700 | [diff] [blame] | 437 |  | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 438 |         // for use from destructor | 
 | 439 |         void                stop_nonvirtual(); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 440 |     }; | 
 | 441 |  | 
| Eric Laurent | bfb1b83 | 2013-01-07 09:53:42 -0800 | [diff] [blame] | 442 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 443 |               PlaybackThread *checkPlaybackThread_l(audio_io_handle_t output) const; | 
 | 444 |               MixerThread *checkMixerThread_l(audio_io_handle_t output) const; | 
 | 445 |               RecordThread *checkRecordThread_l(audio_io_handle_t input) const; | 
| Glenn Kasten | 6637baa | 2012-01-09 09:40:36 -0800 | [diff] [blame] | 446 |               // no range check, AudioFlinger::mLock held | 
 | 447 |               bool streamMute_l(audio_stream_type_t stream) const | 
 | 448 |                                 { return mStreamTypes[stream].mute; } | 
 | 449 |               // no range check, doesn't check per-thread stream volume, AudioFlinger::mLock held | 
 | 450 |               float streamVolume_l(audio_stream_type_t stream) const | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 451 |                                 { return mStreamTypes[stream].volume; } | 
| Glenn Kasten | b81cc8c | 2012-03-01 09:14:51 -0800 | [diff] [blame] | 452 |               void audioConfigChanged_l(int event, audio_io_handle_t ioHandle, const void *param2); | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 453 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 454 |               // allocate an audio_io_handle_t, session ID, or effect ID | 
| Eric Laurent | 7c7f10b | 2011-06-17 21:29:58 -0700 | [diff] [blame] | 455 |               uint32_t nextUniqueId(); | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 456 |  | 
| Eric Laurent | 59255e4 | 2011-07-27 19:49:51 -0700 | [diff] [blame] | 457 |               status_t moveEffectChain_l(int sessionId, | 
| Glenn Kasten | a111792 | 2012-01-26 10:53:32 -0800 | [diff] [blame] | 458 |                                      PlaybackThread *srcThread, | 
 | 459 |                                      PlaybackThread *dstThread, | 
| Eric Laurent | 39e94f8 | 2010-07-28 01:32:47 -0700 | [diff] [blame] | 460 |                                      bool reRegister); | 
| Glenn Kasten | 02fe1bf | 2012-02-24 15:42:17 -0800 | [diff] [blame] | 461 |               // return thread associated with primary hardware device, or NULL | 
 | 462 |               PlaybackThread *primaryPlaybackThread_l() const; | 
| Glenn Kasten | bb4350d | 2012-07-03 15:56:38 -0700 | [diff] [blame] | 463 |               audio_devices_t primaryOutputDevice_l() const; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 464 |  | 
| Eric Laurent | 717e128 | 2012-06-29 16:36:52 -0700 | [diff] [blame] | 465 |               sp<PlaybackThread> getEffectThread_l(int sessionId, int EffectId); | 
 | 466 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 467 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 468 |                 void        removeClient_l(pid_t pid); | 
 | 469 |                 void        removeNotificationClient(pid_t pid); | 
 | 470 |  | 
| Eric Laurent | 5baf2af | 2013-09-12 17:37:00 -0700 | [diff] [blame] | 471 |                 bool isNonOffloadableGlobalEffectEnabled_l(); | 
 | 472 |                 void onNonOffloadableGlobalEffectEnable(); | 
| Eric Laurent | 813e2a7 | 2013-08-31 12:59:48 -0700 | [diff] [blame] | 473 |  | 
| John Grossman | ee578c0 | 2012-07-23 17:05:46 -0700 | [diff] [blame] | 474 |     class AudioHwDevice { | 
 | 475 |     public: | 
 | 476 |         enum Flags { | 
 | 477 |             AHWD_CAN_SET_MASTER_VOLUME  = 0x1, | 
 | 478 |             AHWD_CAN_SET_MASTER_MUTE    = 0x2, | 
 | 479 |         }; | 
 | 480 |  | 
 | 481 |         AudioHwDevice(const char *moduleName, | 
 | 482 |                       audio_hw_device_t *hwDevice, | 
 | 483 |                       Flags flags) | 
 | 484 |             : mModuleName(strdup(moduleName)) | 
 | 485 |             , mHwDevice(hwDevice) | 
 | 486 |             , mFlags(flags) { } | 
 | 487 |         /*virtual*/ ~AudioHwDevice() { free((void *)mModuleName); } | 
 | 488 |  | 
 | 489 |         bool canSetMasterVolume() const { | 
 | 490 |             return (0 != (mFlags & AHWD_CAN_SET_MASTER_VOLUME)); | 
 | 491 |         } | 
 | 492 |  | 
 | 493 |         bool canSetMasterMute() const { | 
 | 494 |             return (0 != (mFlags & AHWD_CAN_SET_MASTER_MUTE)); | 
 | 495 |         } | 
 | 496 |  | 
 | 497 |         const char *moduleName() const { return mModuleName; } | 
 | 498 |         audio_hw_device_t *hwDevice() const { return mHwDevice; } | 
 | 499 |     private: | 
 | 500 |         const char * const mModuleName; | 
 | 501 |         audio_hw_device_t * const mHwDevice; | 
 | 502 |         Flags mFlags; | 
 | 503 |     }; | 
 | 504 |  | 
| Glenn Kasten | aed850d | 2012-01-26 09:46:34 -0800 | [diff] [blame] | 505 |     // AudioStreamOut and AudioStreamIn are immutable, so their fields are const. | 
 | 506 |     // For emphasis, we could also make all pointers to them be "const *", | 
 | 507 |     // but that would clutter the code unnecessarily. | 
 | 508 |  | 
| Dima Zavin | 799a70e | 2011-04-18 16:57:27 -0700 | [diff] [blame] | 509 |     struct AudioStreamOut { | 
| John Grossman | ee578c0 | 2012-07-23 17:05:46 -0700 | [diff] [blame] | 510 |         AudioHwDevice* const audioHwDev; | 
| Glenn Kasten | aed850d | 2012-01-26 09:46:34 -0800 | [diff] [blame] | 511 |         audio_stream_out_t* const stream; | 
| Eric Laurent | bfb1b83 | 2013-01-07 09:53:42 -0800 | [diff] [blame] | 512 |         audio_output_flags_t flags; | 
| Dima Zavin | 799a70e | 2011-04-18 16:57:27 -0700 | [diff] [blame] | 513 |  | 
| John Grossman | ee578c0 | 2012-07-23 17:05:46 -0700 | [diff] [blame] | 514 |         audio_hw_device_t* hwDev() const { return audioHwDev->hwDevice(); } | 
 | 515 |  | 
| Eric Laurent | bfb1b83 | 2013-01-07 09:53:42 -0800 | [diff] [blame] | 516 |         AudioStreamOut(AudioHwDevice *dev, audio_stream_out_t *out, audio_output_flags_t flags) : | 
 | 517 |             audioHwDev(dev), stream(out), flags(flags) {} | 
| Dima Zavin | 799a70e | 2011-04-18 16:57:27 -0700 | [diff] [blame] | 518 |     }; | 
 | 519 |  | 
 | 520 |     struct AudioStreamIn { | 
| John Grossman | ee578c0 | 2012-07-23 17:05:46 -0700 | [diff] [blame] | 521 |         AudioHwDevice* const audioHwDev; | 
| Glenn Kasten | aed850d | 2012-01-26 09:46:34 -0800 | [diff] [blame] | 522 |         audio_stream_in_t* const stream; | 
| Dima Zavin | 799a70e | 2011-04-18 16:57:27 -0700 | [diff] [blame] | 523 |  | 
| John Grossman | ee578c0 | 2012-07-23 17:05:46 -0700 | [diff] [blame] | 524 |         audio_hw_device_t* hwDev() const { return audioHwDev->hwDevice(); } | 
 | 525 |  | 
 | 526 |         AudioStreamIn(AudioHwDevice *dev, audio_stream_in_t *in) : | 
 | 527 |             audioHwDev(dev), stream(in) {} | 
| Dima Zavin | 799a70e | 2011-04-18 16:57:27 -0700 | [diff] [blame] | 528 |     }; | 
 | 529 |  | 
| Glenn Kasten | 99e53b8 | 2012-01-19 08:59:58 -0800 | [diff] [blame] | 530 |     // for mAudioSessionRefs only | 
| Marco Nelissen | 3a34bef | 2011-08-02 13:33:41 -0700 | [diff] [blame] | 531 |     struct AudioSessionRef { | 
| Glenn Kasten | 012ca6b | 2012-03-06 11:22:01 -0800 | [diff] [blame] | 532 |         AudioSessionRef(int sessionid, pid_t pid) : | 
 | 533 |             mSessionid(sessionid), mPid(pid), mCnt(1) {} | 
 | 534 |         const int   mSessionid; | 
 | 535 |         const pid_t mPid; | 
 | 536 |         int         mCnt; | 
| Marco Nelissen | 3a34bef | 2011-08-02 13:33:41 -0700 | [diff] [blame] | 537 |     }; | 
 | 538 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 539 |     mutable     Mutex                               mLock; | 
 | 540 |  | 
| Glenn Kasten | 98ec94c | 2012-01-25 14:28:29 -0800 | [diff] [blame] | 541 |                 DefaultKeyedVector< pid_t, wp<Client> >     mClients;   // see ~Client() | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 542 |  | 
 | 543 |                 mutable     Mutex                   mHardwareLock; | 
| Eric Laurent | a4c5a55 | 2012-03-29 10:12:40 -0700 | [diff] [blame] | 544 |                 // NOTE: If both mLock and mHardwareLock mutexes must be held, | 
 | 545 |                 // always take mLock before mHardwareLock | 
| Glenn Kasten | 2b213bc | 2012-02-02 14:05:20 -0800 | [diff] [blame] | 546 |  | 
 | 547 |                 // These two fields are immutable after onFirstRef(), so no lock needed to access | 
| John Grossman | ee578c0 | 2012-07-23 17:05:46 -0700 | [diff] [blame] | 548 |                 AudioHwDevice*                      mPrimaryHardwareDev; // mAudioHwDevs[0] or NULL | 
| Eric Laurent | a4c5a55 | 2012-03-29 10:12:40 -0700 | [diff] [blame] | 549 |                 DefaultKeyedVector<audio_module_handle_t, AudioHwDevice*>  mAudioHwDevs; | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 550 |  | 
| Glenn Kasten | 8abf44d | 2012-02-02 14:16:03 -0800 | [diff] [blame] | 551 |     // for dump, indicates which hardware operation is currently in progress (but not stream ops) | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 552 |     enum hardware_call_state { | 
| Glenn Kasten | 8abf44d | 2012-02-02 14:16:03 -0800 | [diff] [blame] | 553 |         AUDIO_HW_IDLE = 0,              // no operation in progress | 
 | 554 |         AUDIO_HW_INIT,                  // init_check | 
 | 555 |         AUDIO_HW_OUTPUT_OPEN,           // open_output_stream | 
 | 556 |         AUDIO_HW_OUTPUT_CLOSE,          // unused | 
 | 557 |         AUDIO_HW_INPUT_OPEN,            // unused | 
 | 558 |         AUDIO_HW_INPUT_CLOSE,           // unused | 
 | 559 |         AUDIO_HW_STANDBY,               // unused | 
 | 560 |         AUDIO_HW_SET_MASTER_VOLUME,     // set_master_volume | 
 | 561 |         AUDIO_HW_GET_ROUTING,           // unused | 
 | 562 |         AUDIO_HW_SET_ROUTING,           // unused | 
 | 563 |         AUDIO_HW_GET_MODE,              // unused | 
 | 564 |         AUDIO_HW_SET_MODE,              // set_mode | 
 | 565 |         AUDIO_HW_GET_MIC_MUTE,          // get_mic_mute | 
 | 566 |         AUDIO_HW_SET_MIC_MUTE,          // set_mic_mute | 
 | 567 |         AUDIO_HW_SET_VOICE_VOLUME,      // set_voice_volume | 
 | 568 |         AUDIO_HW_SET_PARAMETER,         // set_parameters | 
 | 569 |         AUDIO_HW_GET_INPUT_BUFFER_SIZE, // get_input_buffer_size | 
 | 570 |         AUDIO_HW_GET_MASTER_VOLUME,     // get_master_volume | 
 | 571 |         AUDIO_HW_GET_PARAMETER,         // get_parameters | 
| John Grossman | d8f178d | 2012-07-20 14:51:35 -0700 | [diff] [blame] | 572 |         AUDIO_HW_SET_MASTER_MUTE,       // set_master_mute | 
 | 573 |         AUDIO_HW_GET_MASTER_MUTE,       // get_master_mute | 
| Glenn Kasten | 2f732eb | 2012-01-26 09:48:03 -0800 | [diff] [blame] | 574 |     }; | 
 | 575 |  | 
| Glenn Kasten | a4454b4 | 2012-01-04 11:02:33 -0800 | [diff] [blame] | 576 |     mutable     hardware_call_state                 mHardwareStatus;    // for dump only | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 577 |  | 
 | 578 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 579 |                 DefaultKeyedVector< audio_io_handle_t, sp<PlaybackThread> >  mPlaybackThreads; | 
| Glenn Kasten | b7bf796 | 2012-02-08 12:36:25 -0800 | [diff] [blame] | 580 |                 stream_type_t                       mStreamTypes[AUDIO_STREAM_CNT]; | 
| Glenn Kasten | 9806710 | 2011-12-13 11:47:54 -0800 | [diff] [blame] | 581 |  | 
| Glenn Kasten | c3ae93f | 2012-07-30 10:59:30 -0700 | [diff] [blame] | 582 |                 // member variables below are protected by mLock | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 583 |                 float                               mMasterVolume; | 
 | 584 |                 bool                                mMasterMute; | 
| Glenn Kasten | c3ae93f | 2012-07-30 10:59:30 -0700 | [diff] [blame] | 585 |                 // end of variables protected by mLock | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 586 |  | 
| Glenn Kasten | 72ef00d | 2012-01-17 11:09:42 -0800 | [diff] [blame] | 587 |                 DefaultKeyedVector< audio_io_handle_t, sp<RecordThread> >    mRecordThreads; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 588 |  | 
 | 589 |                 DefaultKeyedVector< pid_t, sp<NotificationClient> >    mNotificationClients; | 
| Glenn Kasten | 99e53b8 | 2012-01-19 08:59:58 -0800 | [diff] [blame] | 590 |                 volatile int32_t                    mNextUniqueId;  // updated by android_atomic_inc | 
| Glenn Kasten | f78aee7 | 2012-01-04 11:00:47 -0800 | [diff] [blame] | 591 |                 audio_mode_t                        mMode; | 
| Eric Laurent | bee5337 | 2011-08-29 12:42:48 -0700 | [diff] [blame] | 592 |                 bool                                mBtNrecIsOff; | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 593 |  | 
| Glenn Kasten | 99e53b8 | 2012-01-19 08:59:58 -0800 | [diff] [blame] | 594 |                 // protected by mLock | 
| Marco Nelissen | 3a34bef | 2011-08-02 13:33:41 -0700 | [diff] [blame] | 595 |                 Vector<AudioSessionRef*> mAudioSessionRefs; | 
| Glenn Kasten | 9806710 | 2011-12-13 11:47:54 -0800 | [diff] [blame] | 596 |  | 
| John Grossman | 4ff14ba | 2012-02-08 16:37:41 -0800 | [diff] [blame] | 597 |                 float       masterVolume_l() const; | 
| John Grossman | d8f178d | 2012-07-20 14:51:35 -0700 | [diff] [blame] | 598 |                 bool        masterMute_l() const; | 
| Eric Laurent | a4c5a55 | 2012-03-29 10:12:40 -0700 | [diff] [blame] | 599 |                 audio_module_handle_t loadHwModule_l(const char *name); | 
| Glenn Kasten | 98ec94c | 2012-01-25 14:28:29 -0800 | [diff] [blame] | 600 |  | 
| Eric Laurent | a011e35 | 2012-03-29 15:51:43 -0700 | [diff] [blame] | 601 |                 Vector < sp<SyncEvent> > mPendingSyncEvents; // sync events awaiting for a session | 
 | 602 |                                                              // to be created | 
 | 603 |  | 
| Glenn Kasten | 98ec94c | 2012-01-25 14:28:29 -0800 | [diff] [blame] | 604 | private: | 
 | 605 |     sp<Client>  registerPid_l(pid_t pid);    // always returns non-0 | 
 | 606 |  | 
| Glenn Kasten | d96c572 | 2012-04-25 13:44:49 -0700 | [diff] [blame] | 607 |     // for use from destructor | 
 | 608 |     status_t    closeOutput_nonvirtual(audio_io_handle_t output); | 
 | 609 |     status_t    closeInput_nonvirtual(audio_io_handle_t input); | 
| Glenn Kasten | d06785b | 2012-09-30 12:29:28 -0700 | [diff] [blame] | 610 |  | 
| Glenn Kasten | 0d61251 | 2013-07-15 16:09:26 -0700 | [diff] [blame] | 611 | #ifdef TEE_SINK | 
| Glenn Kasten | d06785b | 2012-09-30 12:29:28 -0700 | [diff] [blame] | 612 |     // all record threads serially share a common tee sink, which is re-created on format change | 
 | 613 |     sp<NBAIO_Sink>   mRecordTeeSink; | 
 | 614 |     sp<NBAIO_Source> mRecordTeeSource; | 
| Glenn Kasten | 0d61251 | 2013-07-15 16:09:26 -0700 | [diff] [blame] | 615 | #endif | 
| Glenn Kasten | d06785b | 2012-09-30 12:29:28 -0700 | [diff] [blame] | 616 |  | 
 | 617 | public: | 
| Glenn Kasten | 46909e7 | 2013-02-26 09:20:22 -0800 | [diff] [blame] | 618 |  | 
 | 619 | #ifdef TEE_SINK | 
| Glenn Kasten | da6ef13 | 2013-01-10 12:31:01 -0800 | [diff] [blame] | 620 |     // tee sink, if enabled by property, allows dumpsys to write most recent audio to .wav file | 
| Glenn Kasten | d06785b | 2012-09-30 12:29:28 -0700 | [diff] [blame] | 621 |     static void dumpTee(int fd, const sp<NBAIO_Source>& source, audio_io_handle_t id = 0); | 
| Glenn Kasten | da6ef13 | 2013-01-10 12:31:01 -0800 | [diff] [blame] | 622 |  | 
 | 623 |     // whether tee sink is enabled by property | 
 | 624 |     static bool mTeeSinkInputEnabled; | 
 | 625 |     static bool mTeeSinkOutputEnabled; | 
 | 626 |     static bool mTeeSinkTrackEnabled; | 
 | 627 |  | 
 | 628 |     // runtime configured size of each tee sink pipe, in frames | 
 | 629 |     static size_t mTeeSinkInputFrames; | 
 | 630 |     static size_t mTeeSinkOutputFrames; | 
 | 631 |     static size_t mTeeSinkTrackFrames; | 
 | 632 |  | 
 | 633 |     // compile-time default size of tee sink pipes, in frames | 
 | 634 |     // 0x200000 stereo 16-bit PCM frames = 47.5 seconds at 44.1 kHz, 8 megabytes | 
 | 635 |     static const size_t kTeeSinkInputFramesDefault = 0x200000; | 
 | 636 |     static const size_t kTeeSinkOutputFramesDefault = 0x200000; | 
 | 637 |     static const size_t kTeeSinkTrackFramesDefault = 0x1000; | 
| Glenn Kasten | 46909e7 | 2013-02-26 09:20:22 -0800 | [diff] [blame] | 638 | #endif | 
 | 639 |  | 
| Glenn Kasten | 4182c4e | 2013-07-15 14:45:07 -0700 | [diff] [blame] | 640 |     // This method reads from a variable without mLock, but the variable is updated under mLock.  So | 
 | 641 |     // we might read a stale value, or a value that's inconsistent with respect to other variables. | 
 | 642 |     // In this case, it's safe because the return value isn't used for making an important decision. | 
 | 643 |     // The reason we don't want to take mLock is because it could block the caller for a long time. | 
 | 644 |     bool    isLowRamDevice() const { return mIsLowRamDevice; } | 
 | 645 |  | 
 | 646 | private: | 
 | 647 |     bool    mIsLowRamDevice; | 
 | 648 |     bool    mIsDeviceTypeKnown; | 
| Eric Laurent | 813e2a7 | 2013-08-31 12:59:48 -0700 | [diff] [blame] | 649 |     nsecs_t mGlobalEffectEnableTime;  // when a global effect was last enabled | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 650 | }; | 
 | 651 |  | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 652 | #undef INCLUDING_FROM_AUDIOFLINGER_H | 
| Dima Zavin | 799a70e | 2011-04-18 16:57:27 -0700 | [diff] [blame] | 653 |  | 
| Mathias Agopian | 65ab471 | 2010-07-14 17:59:35 -0700 | [diff] [blame] | 654 | // ---------------------------------------------------------------------------- | 
 | 655 |  | 
 | 656 | }; // namespace android | 
 | 657 |  | 
 | 658 | #endif // ANDROID_AUDIO_FLINGER_H |