audio_device_mac.h 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. /*
  2. * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
  3. *
  4. * Use of this source code is governed by a BSD-style license
  5. * that can be found in the LICENSE file in the root of the source
  6. * tree. An additional intellectual property rights grant can be found
  7. * in the file PATENTS. All contributing project authors may
  8. * be found in the AUTHORS file in the root of the source tree.
  9. */
  10. #ifndef AUDIO_DEVICE_AUDIO_DEVICE_MAC_H_
  11. #define AUDIO_DEVICE_AUDIO_DEVICE_MAC_H_
  12. #include <AudioToolbox/AudioConverter.h>
  13. #include <CoreAudio/CoreAudio.h>
  14. #include <mach/semaphore.h>
  15. #include <memory>
  16. #include "modules/audio_device/audio_device_generic.h"
  17. #include "modules/audio_device/mac/audio_mixer_manager_mac.h"
  18. #include "rtc_base/event.h"
  19. #include "rtc_base/logging.h"
  20. #include "rtc_base/synchronization/mutex.h"
  21. #include "rtc_base/thread_annotations.h"
  22. struct PaUtilRingBuffer;
  23. namespace rtc {
  24. class PlatformThread;
  25. } // namespace rtc
  26. namespace webrtc {
  27. const uint32_t N_REC_SAMPLES_PER_SEC = 48000;
  28. const uint32_t N_PLAY_SAMPLES_PER_SEC = 48000;
  29. const uint32_t N_REC_CHANNELS = 1; // default is mono recording
  30. const uint32_t N_PLAY_CHANNELS = 2; // default is stereo playout
  31. const uint32_t N_DEVICE_CHANNELS = 64;
  32. const int kBufferSizeMs = 10;
  33. const uint32_t ENGINE_REC_BUF_SIZE_IN_SAMPLES =
  34. N_REC_SAMPLES_PER_SEC * kBufferSizeMs / 1000;
  35. const uint32_t ENGINE_PLAY_BUF_SIZE_IN_SAMPLES =
  36. N_PLAY_SAMPLES_PER_SEC * kBufferSizeMs / 1000;
  37. const int N_BLOCKS_IO = 2;
  38. const int N_BUFFERS_IN = 2; // Must be at least N_BLOCKS_IO.
  39. const int N_BUFFERS_OUT = 3; // Must be at least N_BLOCKS_IO.
  40. const uint32_t TIMER_PERIOD_MS = 2 * 10 * N_BLOCKS_IO * 1000000;
  41. const uint32_t REC_BUF_SIZE_IN_SAMPLES =
  42. ENGINE_REC_BUF_SIZE_IN_SAMPLES * N_DEVICE_CHANNELS * N_BUFFERS_IN;
  43. const uint32_t PLAY_BUF_SIZE_IN_SAMPLES =
  44. ENGINE_PLAY_BUF_SIZE_IN_SAMPLES * N_PLAY_CHANNELS * N_BUFFERS_OUT;
  45. const int kGetMicVolumeIntervalMs = 1000;
  46. class AudioDeviceMac : public AudioDeviceGeneric {
  47. public:
  48. AudioDeviceMac();
  49. ~AudioDeviceMac();
  50. // Retrieve the currently utilized audio layer
  51. virtual int32_t ActiveAudioLayer(
  52. AudioDeviceModule::AudioLayer& audioLayer) const;
  53. // Main initializaton and termination
  54. virtual InitStatus Init() RTC_LOCKS_EXCLUDED(mutex_);
  55. virtual int32_t Terminate() RTC_LOCKS_EXCLUDED(mutex_);
  56. virtual bool Initialized() const;
  57. // Device enumeration
  58. virtual int16_t PlayoutDevices();
  59. virtual int16_t RecordingDevices();
  60. virtual int32_t PlayoutDeviceName(uint16_t index,
  61. char name[kAdmMaxDeviceNameSize],
  62. char guid[kAdmMaxGuidSize]);
  63. virtual int32_t RecordingDeviceName(uint16_t index,
  64. char name[kAdmMaxDeviceNameSize],
  65. char guid[kAdmMaxGuidSize]);
  66. // Device selection
  67. virtual int32_t SetPlayoutDevice(uint16_t index) RTC_LOCKS_EXCLUDED(mutex_);
  68. virtual int32_t SetPlayoutDevice(AudioDeviceModule::WindowsDeviceType device);
  69. virtual int32_t SetRecordingDevice(uint16_t index);
  70. virtual int32_t SetRecordingDevice(
  71. AudioDeviceModule::WindowsDeviceType device);
  72. // Audio transport initialization
  73. virtual int32_t PlayoutIsAvailable(bool& available);
  74. virtual int32_t InitPlayout() RTC_LOCKS_EXCLUDED(mutex_);
  75. virtual bool PlayoutIsInitialized() const;
  76. virtual int32_t RecordingIsAvailable(bool& available);
  77. virtual int32_t InitRecording() RTC_LOCKS_EXCLUDED(mutex_);
  78. virtual bool RecordingIsInitialized() const;
  79. // Audio transport control
  80. virtual int32_t StartPlayout() RTC_LOCKS_EXCLUDED(mutex_);
  81. virtual int32_t StopPlayout() RTC_LOCKS_EXCLUDED(mutex_);
  82. virtual bool Playing() const;
  83. virtual int32_t StartRecording() RTC_LOCKS_EXCLUDED(mutex_);
  84. virtual int32_t StopRecording() RTC_LOCKS_EXCLUDED(mutex_);
  85. virtual bool Recording() const;
  86. // Audio mixer initialization
  87. virtual int32_t InitSpeaker() RTC_LOCKS_EXCLUDED(mutex_);
  88. virtual bool SpeakerIsInitialized() const;
  89. virtual int32_t InitMicrophone() RTC_LOCKS_EXCLUDED(mutex_);
  90. virtual bool MicrophoneIsInitialized() const;
  91. // Speaker volume controls
  92. virtual int32_t SpeakerVolumeIsAvailable(bool& available)
  93. RTC_LOCKS_EXCLUDED(mutex_);
  94. virtual int32_t SetSpeakerVolume(uint32_t volume);
  95. virtual int32_t SpeakerVolume(uint32_t& volume) const;
  96. virtual int32_t MaxSpeakerVolume(uint32_t& maxVolume) const;
  97. virtual int32_t MinSpeakerVolume(uint32_t& minVolume) const;
  98. // Microphone volume controls
  99. virtual int32_t MicrophoneVolumeIsAvailable(bool& available)
  100. RTC_LOCKS_EXCLUDED(mutex_);
  101. virtual int32_t SetMicrophoneVolume(uint32_t volume);
  102. virtual int32_t MicrophoneVolume(uint32_t& volume) const;
  103. virtual int32_t MaxMicrophoneVolume(uint32_t& maxVolume) const;
  104. virtual int32_t MinMicrophoneVolume(uint32_t& minVolume) const;
  105. // Microphone mute control
  106. virtual int32_t MicrophoneMuteIsAvailable(bool& available)
  107. RTC_LOCKS_EXCLUDED(mutex_);
  108. virtual int32_t SetMicrophoneMute(bool enable);
  109. virtual int32_t MicrophoneMute(bool& enabled) const;
  110. // Speaker mute control
  111. virtual int32_t SpeakerMuteIsAvailable(bool& available)
  112. RTC_LOCKS_EXCLUDED(mutex_);
  113. virtual int32_t SetSpeakerMute(bool enable);
  114. virtual int32_t SpeakerMute(bool& enabled) const;
  115. // Stereo support
  116. virtual int32_t StereoPlayoutIsAvailable(bool& available)
  117. RTC_LOCKS_EXCLUDED(mutex_);
  118. virtual int32_t SetStereoPlayout(bool enable);
  119. virtual int32_t StereoPlayout(bool& enabled) const;
  120. virtual int32_t StereoRecordingIsAvailable(bool& available);
  121. virtual int32_t SetStereoRecording(bool enable);
  122. virtual int32_t StereoRecording(bool& enabled) const;
  123. // Delay information and control
  124. virtual int32_t PlayoutDelay(uint16_t& delayMS) const;
  125. virtual void AttachAudioBuffer(AudioDeviceBuffer* audioBuffer)
  126. RTC_LOCKS_EXCLUDED(mutex_);
  127. private:
  128. int32_t InitSpeakerLocked() RTC_EXCLUSIVE_LOCKS_REQUIRED(mutex_);
  129. int32_t InitMicrophoneLocked() RTC_EXCLUSIVE_LOCKS_REQUIRED(mutex_);
  130. virtual int32_t MicrophoneIsAvailable(bool& available)
  131. RTC_LOCKS_EXCLUDED(mutex_);
  132. virtual int32_t MicrophoneIsAvailableLocked(bool& available)
  133. RTC_EXCLUSIVE_LOCKS_REQUIRED(mutex_);
  134. virtual int32_t SpeakerIsAvailable(bool& available)
  135. RTC_LOCKS_EXCLUDED(mutex_);
  136. virtual int32_t SpeakerIsAvailableLocked(bool& available)
  137. RTC_EXCLUSIVE_LOCKS_REQUIRED(mutex_);
  138. static void AtomicSet32(int32_t* theValue, int32_t newValue);
  139. static int32_t AtomicGet32(int32_t* theValue);
  140. static void logCAMsg(const rtc::LoggingSeverity sev,
  141. const char* msg,
  142. const char* err);
  143. int32_t GetNumberDevices(const AudioObjectPropertyScope scope,
  144. AudioDeviceID scopedDeviceIds[],
  145. const uint32_t deviceListLength);
  146. int32_t GetDeviceName(const AudioObjectPropertyScope scope,
  147. const uint16_t index,
  148. char* name);
  149. int32_t InitDevice(uint16_t userDeviceIndex,
  150. AudioDeviceID& deviceId,
  151. bool isInput);
  152. // Always work with our preferred playout format inside VoE.
  153. // Then convert the output to the OS setting using an AudioConverter.
  154. OSStatus SetDesiredPlayoutFormat();
  155. static OSStatus objectListenerProc(
  156. AudioObjectID objectId,
  157. UInt32 numberAddresses,
  158. const AudioObjectPropertyAddress addresses[],
  159. void* clientData);
  160. OSStatus implObjectListenerProc(AudioObjectID objectId,
  161. UInt32 numberAddresses,
  162. const AudioObjectPropertyAddress addresses[]);
  163. int32_t HandleDeviceChange();
  164. int32_t HandleStreamFormatChange(AudioObjectID objectId,
  165. AudioObjectPropertyAddress propertyAddress);
  166. int32_t HandleDataSourceChange(AudioObjectID objectId,
  167. AudioObjectPropertyAddress propertyAddress);
  168. int32_t HandleProcessorOverload(AudioObjectPropertyAddress propertyAddress);
  169. static OSStatus deviceIOProc(AudioDeviceID device,
  170. const AudioTimeStamp* now,
  171. const AudioBufferList* inputData,
  172. const AudioTimeStamp* inputTime,
  173. AudioBufferList* outputData,
  174. const AudioTimeStamp* outputTime,
  175. void* clientData);
  176. static OSStatus outConverterProc(
  177. AudioConverterRef audioConverter,
  178. UInt32* numberDataPackets,
  179. AudioBufferList* data,
  180. AudioStreamPacketDescription** dataPacketDescription,
  181. void* userData);
  182. static OSStatus inDeviceIOProc(AudioDeviceID device,
  183. const AudioTimeStamp* now,
  184. const AudioBufferList* inputData,
  185. const AudioTimeStamp* inputTime,
  186. AudioBufferList* outputData,
  187. const AudioTimeStamp* outputTime,
  188. void* clientData);
  189. static OSStatus inConverterProc(
  190. AudioConverterRef audioConverter,
  191. UInt32* numberDataPackets,
  192. AudioBufferList* data,
  193. AudioStreamPacketDescription** dataPacketDescription,
  194. void* inUserData);
  195. OSStatus implDeviceIOProc(const AudioBufferList* inputData,
  196. const AudioTimeStamp* inputTime,
  197. AudioBufferList* outputData,
  198. const AudioTimeStamp* outputTime)
  199. RTC_LOCKS_EXCLUDED(mutex_);
  200. OSStatus implOutConverterProc(UInt32* numberDataPackets,
  201. AudioBufferList* data);
  202. OSStatus implInDeviceIOProc(const AudioBufferList* inputData,
  203. const AudioTimeStamp* inputTime)
  204. RTC_LOCKS_EXCLUDED(mutex_);
  205. OSStatus implInConverterProc(UInt32* numberDataPackets,
  206. AudioBufferList* data);
  207. static void RunCapture(void*);
  208. static void RunRender(void*);
  209. bool CaptureWorkerThread();
  210. bool RenderWorkerThread();
  211. bool KeyPressed();
  212. AudioDeviceBuffer* _ptrAudioBuffer;
  213. Mutex mutex_;
  214. rtc::Event _stopEventRec;
  215. rtc::Event _stopEvent;
  216. // TODO(pbos): Replace with direct members, just start/stop, no need to
  217. // recreate the thread.
  218. // Only valid/running between calls to StartRecording and StopRecording.
  219. std::unique_ptr<rtc::PlatformThread> capture_worker_thread_;
  220. // Only valid/running between calls to StartPlayout and StopPlayout.
  221. std::unique_ptr<rtc::PlatformThread> render_worker_thread_;
  222. AudioMixerManagerMac _mixerManager;
  223. uint16_t _inputDeviceIndex;
  224. uint16_t _outputDeviceIndex;
  225. AudioDeviceID _inputDeviceID;
  226. AudioDeviceID _outputDeviceID;
  227. #if __MAC_OS_X_VERSION_MAX_ALLOWED >= 1050
  228. AudioDeviceIOProcID _inDeviceIOProcID;
  229. AudioDeviceIOProcID _deviceIOProcID;
  230. #endif
  231. bool _inputDeviceIsSpecified;
  232. bool _outputDeviceIsSpecified;
  233. uint8_t _recChannels;
  234. uint8_t _playChannels;
  235. Float32* _captureBufData;
  236. SInt16* _renderBufData;
  237. SInt16 _renderConvertData[PLAY_BUF_SIZE_IN_SAMPLES];
  238. bool _initialized;
  239. bool _isShutDown;
  240. bool _recording;
  241. bool _playing;
  242. bool _recIsInitialized;
  243. bool _playIsInitialized;
  244. // Atomically set varaibles
  245. int32_t _renderDeviceIsAlive;
  246. int32_t _captureDeviceIsAlive;
  247. bool _twoDevices;
  248. bool _doStop; // For play if not shared device or play+rec if shared device
  249. bool _doStopRec; // For rec if not shared device
  250. bool _macBookPro;
  251. bool _macBookProPanRight;
  252. AudioConverterRef _captureConverter;
  253. AudioConverterRef _renderConverter;
  254. AudioStreamBasicDescription _outStreamFormat;
  255. AudioStreamBasicDescription _outDesiredFormat;
  256. AudioStreamBasicDescription _inStreamFormat;
  257. AudioStreamBasicDescription _inDesiredFormat;
  258. uint32_t _captureLatencyUs;
  259. uint32_t _renderLatencyUs;
  260. // Atomically set variables
  261. mutable int32_t _captureDelayUs;
  262. mutable int32_t _renderDelayUs;
  263. int32_t _renderDelayOffsetSamples;
  264. PaUtilRingBuffer* _paCaptureBuffer;
  265. PaUtilRingBuffer* _paRenderBuffer;
  266. semaphore_t _renderSemaphore;
  267. semaphore_t _captureSemaphore;
  268. int _captureBufSizeSamples;
  269. int _renderBufSizeSamples;
  270. // Typing detection
  271. // 0x5c is key "9", after that comes function keys.
  272. bool prev_key_state_[0x5d];
  273. };
  274. } // namespace webrtc
  275. #endif // MODULES_AUDIO_DEVICE_MAIN_SOURCE_MAC_AUDIO_DEVICE_MAC_H_