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