coreaudio.c 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810
  1. /**
  2. * OpenAL cross platform audio library
  3. * Copyright (C) 1999-2007 by authors.
  4. * This library is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU Library General Public
  6. * License as published by the Free Software Foundation; either
  7. * version 2 of the License, or (at your option) any later version.
  8. *
  9. * This library is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  12. * Library General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU Library General Public
  15. * License along with this library; if not, write to the
  16. * Free Software Foundation, Inc.,
  17. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  18. * Or go to http://www.gnu.org/copyleft/lgpl.html
  19. */
  20. #include "config.h"
  21. #include <stdio.h>
  22. #include <stdlib.h>
  23. #include <string.h>
  24. #include "alMain.h"
  25. #include "alu.h"
  26. #include "ringbuffer.h"
  27. #include <CoreServices/CoreServices.h>
  28. #include <unistd.h>
  29. #include <AudioUnit/AudioUnit.h>
  30. #include <AudioToolbox/AudioToolbox.h>
  31. #include "backends/base.h"
  32. static const ALCchar ca_device[] = "CoreAudio Default";
  33. typedef struct ALCcoreAudioPlayback {
  34. DERIVE_FROM_TYPE(ALCbackend);
  35. AudioUnit audioUnit;
  36. ALuint frameSize;
  37. AudioStreamBasicDescription format; // This is the OpenAL format as a CoreAudio ASBD
  38. } ALCcoreAudioPlayback;
  39. static void ALCcoreAudioPlayback_Construct(ALCcoreAudioPlayback *self, ALCdevice *device);
  40. static void ALCcoreAudioPlayback_Destruct(ALCcoreAudioPlayback *self);
  41. static ALCenum ALCcoreAudioPlayback_open(ALCcoreAudioPlayback *self, const ALCchar *name);
  42. static ALCboolean ALCcoreAudioPlayback_reset(ALCcoreAudioPlayback *self);
  43. static ALCboolean ALCcoreAudioPlayback_start(ALCcoreAudioPlayback *self);
  44. static void ALCcoreAudioPlayback_stop(ALCcoreAudioPlayback *self);
  45. static DECLARE_FORWARD2(ALCcoreAudioPlayback, ALCbackend, ALCenum, captureSamples, void*, ALCuint)
  46. static DECLARE_FORWARD(ALCcoreAudioPlayback, ALCbackend, ALCuint, availableSamples)
  47. static DECLARE_FORWARD(ALCcoreAudioPlayback, ALCbackend, ClockLatency, getClockLatency)
  48. static DECLARE_FORWARD(ALCcoreAudioPlayback, ALCbackend, void, lock)
  49. static DECLARE_FORWARD(ALCcoreAudioPlayback, ALCbackend, void, unlock)
  50. DECLARE_DEFAULT_ALLOCATORS(ALCcoreAudioPlayback)
  51. DEFINE_ALCBACKEND_VTABLE(ALCcoreAudioPlayback);
  52. static void ALCcoreAudioPlayback_Construct(ALCcoreAudioPlayback *self, ALCdevice *device)
  53. {
  54. ALCbackend_Construct(STATIC_CAST(ALCbackend, self), device);
  55. SET_VTABLE2(ALCcoreAudioPlayback, ALCbackend, self);
  56. self->frameSize = 0;
  57. memset(&self->format, 0, sizeof(self->format));
  58. }
  59. static void ALCcoreAudioPlayback_Destruct(ALCcoreAudioPlayback *self)
  60. {
  61. AudioUnitUninitialize(self->audioUnit);
  62. AudioComponentInstanceDispose(self->audioUnit);
  63. ALCbackend_Destruct(STATIC_CAST(ALCbackend, self));
  64. }
  65. static OSStatus ALCcoreAudioPlayback_MixerProc(void *inRefCon,
  66. AudioUnitRenderActionFlags* UNUSED(ioActionFlags), const AudioTimeStamp* UNUSED(inTimeStamp),
  67. UInt32 UNUSED(inBusNumber), UInt32 UNUSED(inNumberFrames), AudioBufferList *ioData)
  68. {
  69. ALCcoreAudioPlayback *self = inRefCon;
  70. ALCdevice *device = STATIC_CAST(ALCbackend,self)->mDevice;
  71. ALCcoreAudioPlayback_lock(self);
  72. aluMixData(device, ioData->mBuffers[0].mData,
  73. ioData->mBuffers[0].mDataByteSize / self->frameSize);
  74. ALCcoreAudioPlayback_unlock(self);
  75. return noErr;
  76. }
  77. static ALCenum ALCcoreAudioPlayback_open(ALCcoreAudioPlayback *self, const ALCchar *name)
  78. {
  79. ALCdevice *device = STATIC_CAST(ALCbackend,self)->mDevice;
  80. AudioComponentDescription desc;
  81. AudioComponent comp;
  82. OSStatus err;
  83. if(!name)
  84. name = ca_device;
  85. else if(strcmp(name, ca_device) != 0)
  86. return ALC_INVALID_VALUE;
  87. /* open the default output unit */
  88. desc.componentType = kAudioUnitType_Output;
  89. desc.componentSubType = kAudioUnitSubType_DefaultOutput;
  90. desc.componentManufacturer = kAudioUnitManufacturer_Apple;
  91. desc.componentFlags = 0;
  92. desc.componentFlagsMask = 0;
  93. comp = AudioComponentFindNext(NULL, &desc);
  94. if(comp == NULL)
  95. {
  96. ERR("AudioComponentFindNext failed\n");
  97. return ALC_INVALID_VALUE;
  98. }
  99. err = AudioComponentInstanceNew(comp, &self->audioUnit);
  100. if(err != noErr)
  101. {
  102. ERR("AudioComponentInstanceNew failed\n");
  103. return ALC_INVALID_VALUE;
  104. }
  105. /* init and start the default audio unit... */
  106. err = AudioUnitInitialize(self->audioUnit);
  107. if(err != noErr)
  108. {
  109. ERR("AudioUnitInitialize failed\n");
  110. AudioComponentInstanceDispose(self->audioUnit);
  111. return ALC_INVALID_VALUE;
  112. }
  113. alstr_copy_cstr(&device->DeviceName, name);
  114. return ALC_NO_ERROR;
  115. }
  116. static ALCboolean ALCcoreAudioPlayback_reset(ALCcoreAudioPlayback *self)
  117. {
  118. ALCdevice *device = STATIC_CAST(ALCbackend,self)->mDevice;
  119. AudioStreamBasicDescription streamFormat;
  120. AURenderCallbackStruct input;
  121. OSStatus err;
  122. UInt32 size;
  123. err = AudioUnitUninitialize(self->audioUnit);
  124. if(err != noErr)
  125. ERR("-- AudioUnitUninitialize failed.\n");
  126. /* retrieve default output unit's properties (output side) */
  127. size = sizeof(AudioStreamBasicDescription);
  128. err = AudioUnitGetProperty(self->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Output, 0, &streamFormat, &size);
  129. if(err != noErr || size != sizeof(AudioStreamBasicDescription))
  130. {
  131. ERR("AudioUnitGetProperty failed\n");
  132. return ALC_FALSE;
  133. }
  134. #if 0
  135. TRACE("Output streamFormat of default output unit -\n");
  136. TRACE(" streamFormat.mFramesPerPacket = %d\n", streamFormat.mFramesPerPacket);
  137. TRACE(" streamFormat.mChannelsPerFrame = %d\n", streamFormat.mChannelsPerFrame);
  138. TRACE(" streamFormat.mBitsPerChannel = %d\n", streamFormat.mBitsPerChannel);
  139. TRACE(" streamFormat.mBytesPerPacket = %d\n", streamFormat.mBytesPerPacket);
  140. TRACE(" streamFormat.mBytesPerFrame = %d\n", streamFormat.mBytesPerFrame);
  141. TRACE(" streamFormat.mSampleRate = %5.0f\n", streamFormat.mSampleRate);
  142. #endif
  143. /* set default output unit's input side to match output side */
  144. err = AudioUnitSetProperty(self->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input, 0, &streamFormat, size);
  145. if(err != noErr)
  146. {
  147. ERR("AudioUnitSetProperty failed\n");
  148. return ALC_FALSE;
  149. }
  150. if(device->Frequency != streamFormat.mSampleRate)
  151. {
  152. device->NumUpdates = (ALuint)((ALuint64)device->NumUpdates *
  153. streamFormat.mSampleRate /
  154. device->Frequency);
  155. device->Frequency = streamFormat.mSampleRate;
  156. }
  157. /* FIXME: How to tell what channels are what in the output device, and how
  158. * to specify what we're giving? eg, 6.0 vs 5.1 */
  159. switch(streamFormat.mChannelsPerFrame)
  160. {
  161. case 1:
  162. device->FmtChans = DevFmtMono;
  163. break;
  164. case 2:
  165. device->FmtChans = DevFmtStereo;
  166. break;
  167. case 4:
  168. device->FmtChans = DevFmtQuad;
  169. break;
  170. case 6:
  171. device->FmtChans = DevFmtX51;
  172. break;
  173. case 7:
  174. device->FmtChans = DevFmtX61;
  175. break;
  176. case 8:
  177. device->FmtChans = DevFmtX71;
  178. break;
  179. default:
  180. ERR("Unhandled channel count (%d), using Stereo\n", streamFormat.mChannelsPerFrame);
  181. device->FmtChans = DevFmtStereo;
  182. streamFormat.mChannelsPerFrame = 2;
  183. break;
  184. }
  185. SetDefaultWFXChannelOrder(device);
  186. /* use channel count and sample rate from the default output unit's current
  187. * parameters, but reset everything else */
  188. streamFormat.mFramesPerPacket = 1;
  189. streamFormat.mFormatFlags = 0;
  190. switch(device->FmtType)
  191. {
  192. case DevFmtUByte:
  193. device->FmtType = DevFmtByte;
  194. /* fall-through */
  195. case DevFmtByte:
  196. streamFormat.mFormatFlags = kLinearPCMFormatFlagIsSignedInteger;
  197. streamFormat.mBitsPerChannel = 8;
  198. break;
  199. case DevFmtUShort:
  200. device->FmtType = DevFmtShort;
  201. /* fall-through */
  202. case DevFmtShort:
  203. streamFormat.mFormatFlags = kLinearPCMFormatFlagIsSignedInteger;
  204. streamFormat.mBitsPerChannel = 16;
  205. break;
  206. case DevFmtUInt:
  207. device->FmtType = DevFmtInt;
  208. /* fall-through */
  209. case DevFmtInt:
  210. streamFormat.mFormatFlags = kLinearPCMFormatFlagIsSignedInteger;
  211. streamFormat.mBitsPerChannel = 32;
  212. break;
  213. case DevFmtFloat:
  214. streamFormat.mFormatFlags = kLinearPCMFormatFlagIsFloat;
  215. streamFormat.mBitsPerChannel = 32;
  216. break;
  217. }
  218. streamFormat.mBytesPerFrame = streamFormat.mChannelsPerFrame *
  219. streamFormat.mBitsPerChannel / 8;
  220. streamFormat.mBytesPerPacket = streamFormat.mBytesPerFrame;
  221. streamFormat.mFormatID = kAudioFormatLinearPCM;
  222. streamFormat.mFormatFlags |= kAudioFormatFlagsNativeEndian |
  223. kLinearPCMFormatFlagIsPacked;
  224. err = AudioUnitSetProperty(self->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input, 0, &streamFormat, sizeof(AudioStreamBasicDescription));
  225. if(err != noErr)
  226. {
  227. ERR("AudioUnitSetProperty failed\n");
  228. return ALC_FALSE;
  229. }
  230. /* setup callback */
  231. self->frameSize = FrameSizeFromDevFmt(device->FmtChans, device->FmtType, device->AmbiOrder);
  232. input.inputProc = ALCcoreAudioPlayback_MixerProc;
  233. input.inputProcRefCon = self;
  234. err = AudioUnitSetProperty(self->audioUnit, kAudioUnitProperty_SetRenderCallback, kAudioUnitScope_Input, 0, &input, sizeof(AURenderCallbackStruct));
  235. if(err != noErr)
  236. {
  237. ERR("AudioUnitSetProperty failed\n");
  238. return ALC_FALSE;
  239. }
  240. /* init the default audio unit... */
  241. err = AudioUnitInitialize(self->audioUnit);
  242. if(err != noErr)
  243. {
  244. ERR("AudioUnitInitialize failed\n");
  245. return ALC_FALSE;
  246. }
  247. return ALC_TRUE;
  248. }
  249. static ALCboolean ALCcoreAudioPlayback_start(ALCcoreAudioPlayback *self)
  250. {
  251. OSStatus err = AudioOutputUnitStart(self->audioUnit);
  252. if(err != noErr)
  253. {
  254. ERR("AudioOutputUnitStart failed\n");
  255. return ALC_FALSE;
  256. }
  257. return ALC_TRUE;
  258. }
  259. static void ALCcoreAudioPlayback_stop(ALCcoreAudioPlayback *self)
  260. {
  261. OSStatus err = AudioOutputUnitStop(self->audioUnit);
  262. if(err != noErr)
  263. ERR("AudioOutputUnitStop failed\n");
  264. }
  265. typedef struct ALCcoreAudioCapture {
  266. DERIVE_FROM_TYPE(ALCbackend);
  267. AudioUnit audioUnit;
  268. ALuint frameSize;
  269. ALdouble sampleRateRatio; // Ratio of hardware sample rate / requested sample rate
  270. AudioStreamBasicDescription format; // This is the OpenAL format as a CoreAudio ASBD
  271. AudioConverterRef audioConverter; // Sample rate converter if needed
  272. AudioBufferList *bufferList; // Buffer for data coming from the input device
  273. ALCvoid *resampleBuffer; // Buffer for returned RingBuffer data when resampling
  274. ll_ringbuffer_t *ring;
  275. } ALCcoreAudioCapture;
  276. static void ALCcoreAudioCapture_Construct(ALCcoreAudioCapture *self, ALCdevice *device);
  277. static void ALCcoreAudioCapture_Destruct(ALCcoreAudioCapture *self);
  278. static ALCenum ALCcoreAudioCapture_open(ALCcoreAudioCapture *self, const ALCchar *name);
  279. static DECLARE_FORWARD(ALCcoreAudioCapture, ALCbackend, ALCboolean, reset)
  280. static ALCboolean ALCcoreAudioCapture_start(ALCcoreAudioCapture *self);
  281. static void ALCcoreAudioCapture_stop(ALCcoreAudioCapture *self);
  282. static ALCenum ALCcoreAudioCapture_captureSamples(ALCcoreAudioCapture *self, ALCvoid *buffer, ALCuint samples);
  283. static ALCuint ALCcoreAudioCapture_availableSamples(ALCcoreAudioCapture *self);
  284. static DECLARE_FORWARD(ALCcoreAudioCapture, ALCbackend, ClockLatency, getClockLatency)
  285. static DECLARE_FORWARD(ALCcoreAudioCapture, ALCbackend, void, lock)
  286. static DECLARE_FORWARD(ALCcoreAudioCapture, ALCbackend, void, unlock)
  287. DECLARE_DEFAULT_ALLOCATORS(ALCcoreAudioCapture)
  288. DEFINE_ALCBACKEND_VTABLE(ALCcoreAudioCapture);
  289. static AudioBufferList *allocate_buffer_list(UInt32 channelCount, UInt32 byteSize)
  290. {
  291. AudioBufferList *list;
  292. list = calloc(1, FAM_SIZE(AudioBufferList, mBuffers, 1) + byteSize);
  293. if(list)
  294. {
  295. list->mNumberBuffers = 1;
  296. list->mBuffers[0].mNumberChannels = channelCount;
  297. list->mBuffers[0].mDataByteSize = byteSize;
  298. list->mBuffers[0].mData = &list->mBuffers[1];
  299. }
  300. return list;
  301. }
  302. static void destroy_buffer_list(AudioBufferList *list)
  303. {
  304. free(list);
  305. }
  306. static void ALCcoreAudioCapture_Construct(ALCcoreAudioCapture *self, ALCdevice *device)
  307. {
  308. ALCbackend_Construct(STATIC_CAST(ALCbackend, self), device);
  309. SET_VTABLE2(ALCcoreAudioCapture, ALCbackend, self);
  310. self->audioUnit = 0;
  311. self->audioConverter = NULL;
  312. self->bufferList = NULL;
  313. self->resampleBuffer = NULL;
  314. self->ring = NULL;
  315. }
  316. static void ALCcoreAudioCapture_Destruct(ALCcoreAudioCapture *self)
  317. {
  318. ll_ringbuffer_free(self->ring);
  319. self->ring = NULL;
  320. free(self->resampleBuffer);
  321. self->resampleBuffer = NULL;
  322. destroy_buffer_list(self->bufferList);
  323. self->bufferList = NULL;
  324. if(self->audioConverter)
  325. AudioConverterDispose(self->audioConverter);
  326. self->audioConverter = NULL;
  327. if(self->audioUnit)
  328. AudioComponentInstanceDispose(self->audioUnit);
  329. self->audioUnit = 0;
  330. ALCbackend_Destruct(STATIC_CAST(ALCbackend, self));
  331. }
  332. static OSStatus ALCcoreAudioCapture_RecordProc(void *inRefCon,
  333. AudioUnitRenderActionFlags* UNUSED(ioActionFlags),
  334. const AudioTimeStamp *inTimeStamp, UInt32 UNUSED(inBusNumber),
  335. UInt32 inNumberFrames, AudioBufferList* UNUSED(ioData))
  336. {
  337. ALCcoreAudioCapture *self = inRefCon;
  338. AudioUnitRenderActionFlags flags = 0;
  339. OSStatus err;
  340. // fill the bufferList with data from the input device
  341. err = AudioUnitRender(self->audioUnit, &flags, inTimeStamp, 1, inNumberFrames, self->bufferList);
  342. if(err != noErr)
  343. {
  344. ERR("AudioUnitRender error: %d\n", err);
  345. return err;
  346. }
  347. ll_ringbuffer_write(self->ring, self->bufferList->mBuffers[0].mData, inNumberFrames);
  348. return noErr;
  349. }
  350. static OSStatus ALCcoreAudioCapture_ConvertCallback(AudioConverterRef UNUSED(inAudioConverter),
  351. UInt32 *ioNumberDataPackets, AudioBufferList *ioData,
  352. AudioStreamPacketDescription** UNUSED(outDataPacketDescription),
  353. void *inUserData)
  354. {
  355. ALCcoreAudioCapture *self = inUserData;
  356. // Read from the ring buffer and store temporarily in a large buffer
  357. ll_ringbuffer_read(self->ring, self->resampleBuffer, *ioNumberDataPackets);
  358. // Set the input data
  359. ioData->mNumberBuffers = 1;
  360. ioData->mBuffers[0].mNumberChannels = self->format.mChannelsPerFrame;
  361. ioData->mBuffers[0].mData = self->resampleBuffer;
  362. ioData->mBuffers[0].mDataByteSize = (*ioNumberDataPackets) * self->format.mBytesPerFrame;
  363. return noErr;
  364. }
  365. static ALCenum ALCcoreAudioCapture_open(ALCcoreAudioCapture *self, const ALCchar *name)
  366. {
  367. ALCdevice *device = STATIC_CAST(ALCbackend,self)->mDevice;
  368. AudioStreamBasicDescription requestedFormat; // The application requested format
  369. AudioStreamBasicDescription hardwareFormat; // The hardware format
  370. AudioStreamBasicDescription outputFormat; // The AudioUnit output format
  371. AURenderCallbackStruct input;
  372. AudioComponentDescription desc;
  373. AudioDeviceID inputDevice;
  374. UInt32 outputFrameCount;
  375. UInt32 propertySize;
  376. AudioObjectPropertyAddress propertyAddress;
  377. UInt32 enableIO;
  378. AudioComponent comp;
  379. OSStatus err;
  380. if(!name)
  381. name = ca_device;
  382. else if(strcmp(name, ca_device) != 0)
  383. return ALC_INVALID_VALUE;
  384. desc.componentType = kAudioUnitType_Output;
  385. desc.componentSubType = kAudioUnitSubType_HALOutput;
  386. desc.componentManufacturer = kAudioUnitManufacturer_Apple;
  387. desc.componentFlags = 0;
  388. desc.componentFlagsMask = 0;
  389. // Search for component with given description
  390. comp = AudioComponentFindNext(NULL, &desc);
  391. if(comp == NULL)
  392. {
  393. ERR("AudioComponentFindNext failed\n");
  394. return ALC_INVALID_VALUE;
  395. }
  396. // Open the component
  397. err = AudioComponentInstanceNew(comp, &self->audioUnit);
  398. if(err != noErr)
  399. {
  400. ERR("AudioComponentInstanceNew failed\n");
  401. goto error;
  402. }
  403. // Turn off AudioUnit output
  404. enableIO = 0;
  405. err = AudioUnitSetProperty(self->audioUnit, kAudioOutputUnitProperty_EnableIO, kAudioUnitScope_Output, 0, &enableIO, sizeof(ALuint));
  406. if(err != noErr)
  407. {
  408. ERR("AudioUnitSetProperty failed\n");
  409. goto error;
  410. }
  411. // Turn on AudioUnit input
  412. enableIO = 1;
  413. err = AudioUnitSetProperty(self->audioUnit, kAudioOutputUnitProperty_EnableIO, kAudioUnitScope_Input, 1, &enableIO, sizeof(ALuint));
  414. if(err != noErr)
  415. {
  416. ERR("AudioUnitSetProperty failed\n");
  417. goto error;
  418. }
  419. // Get the default input device
  420. propertySize = sizeof(AudioDeviceID);
  421. propertyAddress.mSelector = kAudioHardwarePropertyDefaultInputDevice;
  422. propertyAddress.mScope = kAudioObjectPropertyScopeGlobal;
  423. propertyAddress.mElement = kAudioObjectPropertyElementMaster;
  424. err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &propertyAddress, 0, NULL, &propertySize, &inputDevice);
  425. if(err != noErr)
  426. {
  427. ERR("AudioObjectGetPropertyData failed\n");
  428. goto error;
  429. }
  430. if(inputDevice == kAudioDeviceUnknown)
  431. {
  432. ERR("No input device found\n");
  433. goto error;
  434. }
  435. // Track the input device
  436. err = AudioUnitSetProperty(self->audioUnit, kAudioOutputUnitProperty_CurrentDevice, kAudioUnitScope_Global, 0, &inputDevice, sizeof(AudioDeviceID));
  437. if(err != noErr)
  438. {
  439. ERR("AudioUnitSetProperty failed\n");
  440. goto error;
  441. }
  442. // set capture callback
  443. input.inputProc = ALCcoreAudioCapture_RecordProc;
  444. input.inputProcRefCon = self;
  445. err = AudioUnitSetProperty(self->audioUnit, kAudioOutputUnitProperty_SetInputCallback, kAudioUnitScope_Global, 0, &input, sizeof(AURenderCallbackStruct));
  446. if(err != noErr)
  447. {
  448. ERR("AudioUnitSetProperty failed\n");
  449. goto error;
  450. }
  451. // Initialize the device
  452. err = AudioUnitInitialize(self->audioUnit);
  453. if(err != noErr)
  454. {
  455. ERR("AudioUnitInitialize failed\n");
  456. goto error;
  457. }
  458. // Get the hardware format
  459. propertySize = sizeof(AudioStreamBasicDescription);
  460. err = AudioUnitGetProperty(self->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input, 1, &hardwareFormat, &propertySize);
  461. if(err != noErr || propertySize != sizeof(AudioStreamBasicDescription))
  462. {
  463. ERR("AudioUnitGetProperty failed\n");
  464. goto error;
  465. }
  466. // Set up the requested format description
  467. switch(device->FmtType)
  468. {
  469. case DevFmtUByte:
  470. requestedFormat.mBitsPerChannel = 8;
  471. requestedFormat.mFormatFlags = kAudioFormatFlagIsPacked;
  472. break;
  473. case DevFmtShort:
  474. requestedFormat.mBitsPerChannel = 16;
  475. requestedFormat.mFormatFlags = kAudioFormatFlagIsSignedInteger | kAudioFormatFlagsNativeEndian | kAudioFormatFlagIsPacked;
  476. break;
  477. case DevFmtInt:
  478. requestedFormat.mBitsPerChannel = 32;
  479. requestedFormat.mFormatFlags = kAudioFormatFlagIsSignedInteger | kAudioFormatFlagsNativeEndian | kAudioFormatFlagIsPacked;
  480. break;
  481. case DevFmtFloat:
  482. requestedFormat.mBitsPerChannel = 32;
  483. requestedFormat.mFormatFlags = kAudioFormatFlagIsPacked;
  484. break;
  485. case DevFmtByte:
  486. case DevFmtUShort:
  487. case DevFmtUInt:
  488. ERR("%s samples not supported\n", DevFmtTypeString(device->FmtType));
  489. goto error;
  490. }
  491. switch(device->FmtChans)
  492. {
  493. case DevFmtMono:
  494. requestedFormat.mChannelsPerFrame = 1;
  495. break;
  496. case DevFmtStereo:
  497. requestedFormat.mChannelsPerFrame = 2;
  498. break;
  499. case DevFmtQuad:
  500. case DevFmtX51:
  501. case DevFmtX51Rear:
  502. case DevFmtX61:
  503. case DevFmtX71:
  504. case DevFmtAmbi3D:
  505. ERR("%s not supported\n", DevFmtChannelsString(device->FmtChans));
  506. goto error;
  507. }
  508. requestedFormat.mBytesPerFrame = requestedFormat.mChannelsPerFrame * requestedFormat.mBitsPerChannel / 8;
  509. requestedFormat.mBytesPerPacket = requestedFormat.mBytesPerFrame;
  510. requestedFormat.mSampleRate = device->Frequency;
  511. requestedFormat.mFormatID = kAudioFormatLinearPCM;
  512. requestedFormat.mReserved = 0;
  513. requestedFormat.mFramesPerPacket = 1;
  514. // save requested format description for later use
  515. self->format = requestedFormat;
  516. self->frameSize = FrameSizeFromDevFmt(device->FmtChans, device->FmtType, device->AmbiOrder);
  517. // Use intermediate format for sample rate conversion (outputFormat)
  518. // Set sample rate to the same as hardware for resampling later
  519. outputFormat = requestedFormat;
  520. outputFormat.mSampleRate = hardwareFormat.mSampleRate;
  521. // Determine sample rate ratio for resampling
  522. self->sampleRateRatio = outputFormat.mSampleRate / device->Frequency;
  523. // The output format should be the requested format, but using the hardware sample rate
  524. // This is because the AudioUnit will automatically scale other properties, except for sample rate
  525. err = AudioUnitSetProperty(self->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Output, 1, (void *)&outputFormat, sizeof(outputFormat));
  526. if(err != noErr)
  527. {
  528. ERR("AudioUnitSetProperty failed\n");
  529. goto error;
  530. }
  531. // Set the AudioUnit output format frame count
  532. outputFrameCount = device->UpdateSize * self->sampleRateRatio;
  533. err = AudioUnitSetProperty(self->audioUnit, kAudioUnitProperty_MaximumFramesPerSlice, kAudioUnitScope_Output, 0, &outputFrameCount, sizeof(outputFrameCount));
  534. if(err != noErr)
  535. {
  536. ERR("AudioUnitSetProperty failed: %d\n", err);
  537. goto error;
  538. }
  539. // Set up sample converter
  540. err = AudioConverterNew(&outputFormat, &requestedFormat, &self->audioConverter);
  541. if(err != noErr)
  542. {
  543. ERR("AudioConverterNew failed: %d\n", err);
  544. goto error;
  545. }
  546. // Create a buffer for use in the resample callback
  547. self->resampleBuffer = malloc(device->UpdateSize * self->frameSize * self->sampleRateRatio);
  548. // Allocate buffer for the AudioUnit output
  549. self->bufferList = allocate_buffer_list(outputFormat.mChannelsPerFrame, device->UpdateSize * self->frameSize * self->sampleRateRatio);
  550. if(self->bufferList == NULL)
  551. goto error;
  552. self->ring = ll_ringbuffer_create(
  553. (size_t)ceil(device->UpdateSize*self->sampleRateRatio*device->NumUpdates),
  554. self->frameSize, false
  555. );
  556. if(!self->ring) goto error;
  557. alstr_copy_cstr(&device->DeviceName, name);
  558. return ALC_NO_ERROR;
  559. error:
  560. ll_ringbuffer_free(self->ring);
  561. self->ring = NULL;
  562. free(self->resampleBuffer);
  563. self->resampleBuffer = NULL;
  564. destroy_buffer_list(self->bufferList);
  565. self->bufferList = NULL;
  566. if(self->audioConverter)
  567. AudioConverterDispose(self->audioConverter);
  568. self->audioConverter = NULL;
  569. if(self->audioUnit)
  570. AudioComponentInstanceDispose(self->audioUnit);
  571. self->audioUnit = 0;
  572. return ALC_INVALID_VALUE;
  573. }
  574. static ALCboolean ALCcoreAudioCapture_start(ALCcoreAudioCapture *self)
  575. {
  576. OSStatus err = AudioOutputUnitStart(self->audioUnit);
  577. if(err != noErr)
  578. {
  579. ERR("AudioOutputUnitStart failed\n");
  580. return ALC_FALSE;
  581. }
  582. return ALC_TRUE;
  583. }
  584. static void ALCcoreAudioCapture_stop(ALCcoreAudioCapture *self)
  585. {
  586. OSStatus err = AudioOutputUnitStop(self->audioUnit);
  587. if(err != noErr)
  588. ERR("AudioOutputUnitStop failed\n");
  589. }
  590. static ALCenum ALCcoreAudioCapture_captureSamples(ALCcoreAudioCapture *self, ALCvoid *buffer, ALCuint samples)
  591. {
  592. union {
  593. ALbyte _[sizeof(AudioBufferList) + sizeof(AudioBuffer)];
  594. AudioBufferList list;
  595. } audiobuf = { { 0 } };
  596. UInt32 frameCount;
  597. OSStatus err;
  598. // If no samples are requested, just return
  599. if(samples == 0) return ALC_NO_ERROR;
  600. // Point the resampling buffer to the capture buffer
  601. audiobuf.list.mNumberBuffers = 1;
  602. audiobuf.list.mBuffers[0].mNumberChannels = self->format.mChannelsPerFrame;
  603. audiobuf.list.mBuffers[0].mDataByteSize = samples * self->frameSize;
  604. audiobuf.list.mBuffers[0].mData = buffer;
  605. // Resample into another AudioBufferList
  606. frameCount = samples;
  607. err = AudioConverterFillComplexBuffer(self->audioConverter,
  608. ALCcoreAudioCapture_ConvertCallback, self, &frameCount, &audiobuf.list, NULL
  609. );
  610. if(err != noErr)
  611. {
  612. ERR("AudioConverterFillComplexBuffer error: %d\n", err);
  613. return ALC_INVALID_VALUE;
  614. }
  615. return ALC_NO_ERROR;
  616. }
  617. static ALCuint ALCcoreAudioCapture_availableSamples(ALCcoreAudioCapture *self)
  618. {
  619. return ll_ringbuffer_read_space(self->ring) / self->sampleRateRatio;
  620. }
  621. typedef struct ALCcoreAudioBackendFactory {
  622. DERIVE_FROM_TYPE(ALCbackendFactory);
  623. } ALCcoreAudioBackendFactory;
  624. #define ALCCOREAUDIOBACKENDFACTORY_INITIALIZER { { GET_VTABLE2(ALCcoreAudioBackendFactory, ALCbackendFactory) } }
  625. ALCbackendFactory *ALCcoreAudioBackendFactory_getFactory(void);
  626. static ALCboolean ALCcoreAudioBackendFactory_init(ALCcoreAudioBackendFactory *self);
  627. static DECLARE_FORWARD(ALCcoreAudioBackendFactory, ALCbackendFactory, void, deinit)
  628. static ALCboolean ALCcoreAudioBackendFactory_querySupport(ALCcoreAudioBackendFactory *self, ALCbackend_Type type);
  629. static void ALCcoreAudioBackendFactory_probe(ALCcoreAudioBackendFactory *self, enum DevProbe type);
  630. static ALCbackend* ALCcoreAudioBackendFactory_createBackend(ALCcoreAudioBackendFactory *self, ALCdevice *device, ALCbackend_Type type);
  631. DEFINE_ALCBACKENDFACTORY_VTABLE(ALCcoreAudioBackendFactory);
  632. ALCbackendFactory *ALCcoreAudioBackendFactory_getFactory(void)
  633. {
  634. static ALCcoreAudioBackendFactory factory = ALCCOREAUDIOBACKENDFACTORY_INITIALIZER;
  635. return STATIC_CAST(ALCbackendFactory, &factory);
  636. }
  637. static ALCboolean ALCcoreAudioBackendFactory_init(ALCcoreAudioBackendFactory* UNUSED(self))
  638. {
  639. return ALC_TRUE;
  640. }
  641. static ALCboolean ALCcoreAudioBackendFactory_querySupport(ALCcoreAudioBackendFactory* UNUSED(self), ALCbackend_Type type)
  642. {
  643. if(type == ALCbackend_Playback || ALCbackend_Capture)
  644. return ALC_TRUE;
  645. return ALC_FALSE;
  646. }
  647. static void ALCcoreAudioBackendFactory_probe(ALCcoreAudioBackendFactory* UNUSED(self), enum DevProbe type)
  648. {
  649. switch(type)
  650. {
  651. case ALL_DEVICE_PROBE:
  652. AppendAllDevicesList(ca_device);
  653. break;
  654. case CAPTURE_DEVICE_PROBE:
  655. AppendCaptureDeviceList(ca_device);
  656. break;
  657. }
  658. }
  659. static ALCbackend* ALCcoreAudioBackendFactory_createBackend(ALCcoreAudioBackendFactory* UNUSED(self), ALCdevice *device, ALCbackend_Type type)
  660. {
  661. if(type == ALCbackend_Playback)
  662. {
  663. ALCcoreAudioPlayback *backend;
  664. NEW_OBJ(backend, ALCcoreAudioPlayback)(device);
  665. if(!backend) return NULL;
  666. return STATIC_CAST(ALCbackend, backend);
  667. }
  668. if(type == ALCbackend_Capture)
  669. {
  670. ALCcoreAudioCapture *backend;
  671. NEW_OBJ(backend, ALCcoreAudioCapture)(device);
  672. if(!backend) return NULL;
  673. return STATIC_CAST(ALCbackend, backend);
  674. }
  675. return NULL;
  676. }