coreaudio.c 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704
  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., 59 Temple Place - Suite 330,
  17. * Boston, MA 02111-1307, 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 <CoreServices/CoreServices.h>
  27. #include <unistd.h>
  28. #include <AudioUnit/AudioUnit.h>
  29. #include <AudioToolbox/AudioToolbox.h>
  30. typedef struct {
  31. AudioUnit audioUnit;
  32. ALuint frameSize;
  33. ALdouble sampleRateRatio; // Ratio of hardware sample rate / requested sample rate
  34. AudioStreamBasicDescription format; // This is the OpenAL format as a CoreAudio ASBD
  35. AudioConverterRef audioConverter; // Sample rate converter if needed
  36. AudioBufferList *bufferList; // Buffer for data coming from the input device
  37. ALCvoid *resampleBuffer; // Buffer for returned RingBuffer data when resampling
  38. RingBuffer *ring;
  39. } ca_data;
  40. static const ALCchar ca_device[] = "CoreAudio Default";
  41. static void destroy_buffer_list(AudioBufferList* list)
  42. {
  43. if(list)
  44. {
  45. UInt32 i;
  46. for(i = 0;i < list->mNumberBuffers;i++)
  47. free(list->mBuffers[i].mData);
  48. free(list);
  49. }
  50. }
  51. static AudioBufferList* allocate_buffer_list(UInt32 channelCount, UInt32 byteSize)
  52. {
  53. AudioBufferList *list;
  54. list = calloc(1, sizeof(AudioBufferList) + sizeof(AudioBuffer));
  55. if(list)
  56. {
  57. list->mNumberBuffers = 1;
  58. list->mBuffers[0].mNumberChannels = channelCount;
  59. list->mBuffers[0].mDataByteSize = byteSize;
  60. list->mBuffers[0].mData = malloc(byteSize);
  61. if(list->mBuffers[0].mData == NULL)
  62. {
  63. free(list);
  64. list = NULL;
  65. }
  66. }
  67. return list;
  68. }
  69. static OSStatus ca_callback(void *inRefCon, AudioUnitRenderActionFlags *ioActionFlags, const AudioTimeStamp *inTimeStamp,
  70. UInt32 inBusNumber, UInt32 inNumberFrames, AudioBufferList *ioData)
  71. {
  72. ALCdevice *device = (ALCdevice*)inRefCon;
  73. ca_data *data = (ca_data*)device->ExtraData;
  74. aluMixData(device, ioData->mBuffers[0].mData,
  75. ioData->mBuffers[0].mDataByteSize / data->frameSize);
  76. return noErr;
  77. }
  78. static OSStatus ca_capture_conversion_callback(AudioConverterRef inAudioConverter, UInt32 *ioNumberDataPackets,
  79. AudioBufferList *ioData, AudioStreamPacketDescription **outDataPacketDescription, void* inUserData)
  80. {
  81. ALCdevice *device = (ALCdevice*)inUserData;
  82. ca_data *data = (ca_data*)device->ExtraData;
  83. // Read from the ring buffer and store temporarily in a large buffer
  84. ReadRingBuffer(data->ring, data->resampleBuffer, (ALsizei)(*ioNumberDataPackets));
  85. // Set the input data
  86. ioData->mNumberBuffers = 1;
  87. ioData->mBuffers[0].mNumberChannels = data->format.mChannelsPerFrame;
  88. ioData->mBuffers[0].mData = data->resampleBuffer;
  89. ioData->mBuffers[0].mDataByteSize = (*ioNumberDataPackets) * data->format.mBytesPerFrame;
  90. return noErr;
  91. }
  92. static OSStatus ca_capture_callback(void *inRefCon, AudioUnitRenderActionFlags *ioActionFlags,
  93. const AudioTimeStamp *inTimeStamp, UInt32 inBusNumber,
  94. UInt32 inNumberFrames, AudioBufferList *ioData)
  95. {
  96. ALCdevice *device = (ALCdevice*)inRefCon;
  97. ca_data *data = (ca_data*)device->ExtraData;
  98. AudioUnitRenderActionFlags flags = 0;
  99. OSStatus err;
  100. // fill the bufferList with data from the input device
  101. err = AudioUnitRender(data->audioUnit, &flags, inTimeStamp, 1, inNumberFrames, data->bufferList);
  102. if(err != noErr)
  103. {
  104. ERR("AudioUnitRender error: %d\n", err);
  105. return err;
  106. }
  107. WriteRingBuffer(data->ring, data->bufferList->mBuffers[0].mData, inNumberFrames);
  108. return noErr;
  109. }
  110. static ALCenum ca_open_playback(ALCdevice *device, const ALCchar *deviceName)
  111. {
  112. ComponentDescription desc;
  113. Component comp;
  114. ca_data *data;
  115. OSStatus err;
  116. if(!deviceName)
  117. deviceName = ca_device;
  118. else if(strcmp(deviceName, ca_device) != 0)
  119. return ALC_INVALID_VALUE;
  120. /* open the default output unit */
  121. desc.componentType = kAudioUnitType_Output;
  122. desc.componentSubType = kAudioUnitSubType_DefaultOutput;
  123. desc.componentManufacturer = kAudioUnitManufacturer_Apple;
  124. desc.componentFlags = 0;
  125. desc.componentFlagsMask = 0;
  126. comp = FindNextComponent(NULL, &desc);
  127. if(comp == NULL)
  128. {
  129. ERR("FindNextComponent failed\n");
  130. return ALC_INVALID_VALUE;
  131. }
  132. data = calloc(1, sizeof(*data));
  133. err = OpenAComponent(comp, &data->audioUnit);
  134. if(err != noErr)
  135. {
  136. ERR("OpenAComponent failed\n");
  137. free(data);
  138. return ALC_INVALID_VALUE;
  139. }
  140. /* init and start the default audio unit... */
  141. err = AudioUnitInitialize(data->audioUnit);
  142. if(err != noErr)
  143. {
  144. ERR("AudioUnitInitialize failed\n");
  145. CloseComponent(data->audioUnit);
  146. free(data);
  147. return ALC_INVALID_VALUE;
  148. }
  149. device->DeviceName = strdup(deviceName);
  150. device->ExtraData = data;
  151. return ALC_NO_ERROR;
  152. }
  153. static void ca_close_playback(ALCdevice *device)
  154. {
  155. ca_data *data = (ca_data*)device->ExtraData;
  156. AudioUnitUninitialize(data->audioUnit);
  157. CloseComponent(data->audioUnit);
  158. free(data);
  159. device->ExtraData = NULL;
  160. }
  161. static ALCboolean ca_reset_playback(ALCdevice *device)
  162. {
  163. ca_data *data = (ca_data*)device->ExtraData;
  164. AudioStreamBasicDescription streamFormat;
  165. AURenderCallbackStruct input;
  166. OSStatus err;
  167. UInt32 size;
  168. err = AudioUnitUninitialize(data->audioUnit);
  169. if(err != noErr)
  170. ERR("-- AudioUnitUninitialize failed.\n");
  171. /* retrieve default output unit's properties (output side) */
  172. size = sizeof(AudioStreamBasicDescription);
  173. err = AudioUnitGetProperty(data->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Output, 0, &streamFormat, &size);
  174. if(err != noErr || size != sizeof(AudioStreamBasicDescription))
  175. {
  176. ERR("AudioUnitGetProperty failed\n");
  177. return ALC_FALSE;
  178. }
  179. #if 0
  180. TRACE("Output streamFormat of default output unit -\n");
  181. TRACE(" streamFormat.mFramesPerPacket = %d\n", streamFormat.mFramesPerPacket);
  182. TRACE(" streamFormat.mChannelsPerFrame = %d\n", streamFormat.mChannelsPerFrame);
  183. TRACE(" streamFormat.mBitsPerChannel = %d\n", streamFormat.mBitsPerChannel);
  184. TRACE(" streamFormat.mBytesPerPacket = %d\n", streamFormat.mBytesPerPacket);
  185. TRACE(" streamFormat.mBytesPerFrame = %d\n", streamFormat.mBytesPerFrame);
  186. TRACE(" streamFormat.mSampleRate = %5.0f\n", streamFormat.mSampleRate);
  187. #endif
  188. /* set default output unit's input side to match output side */
  189. err = AudioUnitSetProperty(data->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input, 0, &streamFormat, size);
  190. if(err != noErr)
  191. {
  192. ERR("AudioUnitSetProperty failed\n");
  193. return ALC_FALSE;
  194. }
  195. if(device->Frequency != streamFormat.mSampleRate)
  196. {
  197. device->UpdateSize = (ALuint)((ALuint64)device->UpdateSize *
  198. streamFormat.mSampleRate /
  199. device->Frequency);
  200. device->Frequency = streamFormat.mSampleRate;
  201. }
  202. /* FIXME: How to tell what channels are what in the output device, and how
  203. * to specify what we're giving? eg, 6.0 vs 5.1 */
  204. switch(streamFormat.mChannelsPerFrame)
  205. {
  206. case 1:
  207. device->FmtChans = DevFmtMono;
  208. break;
  209. case 2:
  210. device->FmtChans = DevFmtStereo;
  211. break;
  212. case 4:
  213. device->FmtChans = DevFmtQuad;
  214. break;
  215. case 6:
  216. device->FmtChans = DevFmtX51;
  217. break;
  218. case 7:
  219. device->FmtChans = DevFmtX61;
  220. break;
  221. case 8:
  222. device->FmtChans = DevFmtX71;
  223. break;
  224. default:
  225. ERR("Unhandled channel count (%d), using Stereo\n", streamFormat.mChannelsPerFrame);
  226. device->FmtChans = DevFmtStereo;
  227. streamFormat.mChannelsPerFrame = 2;
  228. break;
  229. }
  230. SetDefaultWFXChannelOrder(device);
  231. /* use channel count and sample rate from the default output unit's current
  232. * parameters, but reset everything else */
  233. streamFormat.mFramesPerPacket = 1;
  234. streamFormat.mFormatFlags = 0;
  235. switch(device->FmtType)
  236. {
  237. case DevFmtUByte:
  238. device->FmtType = DevFmtByte;
  239. /* fall-through */
  240. case DevFmtByte:
  241. streamFormat.mFormatFlags = kLinearPCMFormatFlagIsSignedInteger;
  242. streamFormat.mBitsPerChannel = 8;
  243. break;
  244. case DevFmtUShort:
  245. device->FmtType = DevFmtShort;
  246. /* fall-through */
  247. case DevFmtShort:
  248. streamFormat.mFormatFlags = kLinearPCMFormatFlagIsSignedInteger;
  249. streamFormat.mBitsPerChannel = 16;
  250. break;
  251. case DevFmtUInt:
  252. device->FmtType = DevFmtInt;
  253. /* fall-through */
  254. case DevFmtInt:
  255. streamFormat.mFormatFlags = kLinearPCMFormatFlagIsSignedInteger;
  256. streamFormat.mBitsPerChannel = 32;
  257. break;
  258. case DevFmtFloat:
  259. streamFormat.mFormatFlags = kLinearPCMFormatFlagIsFloat;
  260. streamFormat.mBitsPerChannel = 32;
  261. break;
  262. }
  263. streamFormat.mBytesPerFrame = streamFormat.mChannelsPerFrame *
  264. streamFormat.mBitsPerChannel / 8;
  265. streamFormat.mBytesPerPacket = streamFormat.mBytesPerFrame;
  266. streamFormat.mFormatID = kAudioFormatLinearPCM;
  267. streamFormat.mFormatFlags |= kAudioFormatFlagsNativeEndian |
  268. kLinearPCMFormatFlagIsPacked;
  269. err = AudioUnitSetProperty(data->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input, 0, &streamFormat, sizeof(AudioStreamBasicDescription));
  270. if(err != noErr)
  271. {
  272. ERR("AudioUnitSetProperty failed\n");
  273. return ALC_FALSE;
  274. }
  275. /* setup callback */
  276. data->frameSize = FrameSizeFromDevFmt(device->FmtChans, device->FmtType);
  277. input.inputProc = ca_callback;
  278. input.inputProcRefCon = device;
  279. err = AudioUnitSetProperty(data->audioUnit, kAudioUnitProperty_SetRenderCallback, kAudioUnitScope_Input, 0, &input, sizeof(AURenderCallbackStruct));
  280. if(err != noErr)
  281. {
  282. ERR("AudioUnitSetProperty failed\n");
  283. return ALC_FALSE;
  284. }
  285. /* init the default audio unit... */
  286. err = AudioUnitInitialize(data->audioUnit);
  287. if(err != noErr)
  288. {
  289. ERR("AudioUnitInitialize failed\n");
  290. return ALC_FALSE;
  291. }
  292. return ALC_TRUE;
  293. }
  294. static ALCboolean ca_start_playback(ALCdevice *device)
  295. {
  296. ca_data *data = (ca_data*)device->ExtraData;
  297. OSStatus err;
  298. err = AudioOutputUnitStart(data->audioUnit);
  299. if(err != noErr)
  300. {
  301. ERR("AudioOutputUnitStart failed\n");
  302. return ALC_FALSE;
  303. }
  304. return ALC_TRUE;
  305. }
  306. static void ca_stop_playback(ALCdevice *device)
  307. {
  308. ca_data *data = (ca_data*)device->ExtraData;
  309. OSStatus err;
  310. err = AudioOutputUnitStop(data->audioUnit);
  311. if(err != noErr)
  312. ERR("AudioOutputUnitStop failed\n");
  313. }
  314. static ALCenum ca_open_capture(ALCdevice *device, const ALCchar *deviceName)
  315. {
  316. AudioStreamBasicDescription requestedFormat; // The application requested format
  317. AudioStreamBasicDescription hardwareFormat; // The hardware format
  318. AudioStreamBasicDescription outputFormat; // The AudioUnit output format
  319. AURenderCallbackStruct input;
  320. ComponentDescription desc;
  321. AudioDeviceID inputDevice;
  322. UInt32 outputFrameCount;
  323. UInt32 propertySize;
  324. UInt32 enableIO;
  325. Component comp;
  326. ca_data *data;
  327. OSStatus err;
  328. desc.componentType = kAudioUnitType_Output;
  329. desc.componentSubType = kAudioUnitSubType_HALOutput;
  330. desc.componentManufacturer = kAudioUnitManufacturer_Apple;
  331. desc.componentFlags = 0;
  332. desc.componentFlagsMask = 0;
  333. // Search for component with given description
  334. comp = FindNextComponent(NULL, &desc);
  335. if(comp == NULL)
  336. {
  337. ERR("FindNextComponent failed\n");
  338. return ALC_INVALID_VALUE;
  339. }
  340. data = calloc(1, sizeof(*data));
  341. device->ExtraData = data;
  342. // Open the component
  343. err = OpenAComponent(comp, &data->audioUnit);
  344. if(err != noErr)
  345. {
  346. ERR("OpenAComponent failed\n");
  347. goto error;
  348. }
  349. // Turn off AudioUnit output
  350. enableIO = 0;
  351. err = AudioUnitSetProperty(data->audioUnit, kAudioOutputUnitProperty_EnableIO, kAudioUnitScope_Output, 0, &enableIO, sizeof(ALuint));
  352. if(err != noErr)
  353. {
  354. ERR("AudioUnitSetProperty failed\n");
  355. goto error;
  356. }
  357. // Turn on AudioUnit input
  358. enableIO = 1;
  359. err = AudioUnitSetProperty(data->audioUnit, kAudioOutputUnitProperty_EnableIO, kAudioUnitScope_Input, 1, &enableIO, sizeof(ALuint));
  360. if(err != noErr)
  361. {
  362. ERR("AudioUnitSetProperty failed\n");
  363. goto error;
  364. }
  365. // Get the default input device
  366. propertySize = sizeof(AudioDeviceID);
  367. err = AudioHardwareGetProperty(kAudioHardwarePropertyDefaultInputDevice, &propertySize, &inputDevice);
  368. if(err != noErr)
  369. {
  370. ERR("AudioHardwareGetProperty failed\n");
  371. goto error;
  372. }
  373. if(inputDevice == kAudioDeviceUnknown)
  374. {
  375. ERR("No input device found\n");
  376. goto error;
  377. }
  378. // Track the input device
  379. err = AudioUnitSetProperty(data->audioUnit, kAudioOutputUnitProperty_CurrentDevice, kAudioUnitScope_Global, 0, &inputDevice, sizeof(AudioDeviceID));
  380. if(err != noErr)
  381. {
  382. ERR("AudioUnitSetProperty failed\n");
  383. goto error;
  384. }
  385. // set capture callback
  386. input.inputProc = ca_capture_callback;
  387. input.inputProcRefCon = device;
  388. err = AudioUnitSetProperty(data->audioUnit, kAudioOutputUnitProperty_SetInputCallback, kAudioUnitScope_Global, 0, &input, sizeof(AURenderCallbackStruct));
  389. if(err != noErr)
  390. {
  391. ERR("AudioUnitSetProperty failed\n");
  392. goto error;
  393. }
  394. // Initialize the device
  395. err = AudioUnitInitialize(data->audioUnit);
  396. if(err != noErr)
  397. {
  398. ERR("AudioUnitInitialize failed\n");
  399. goto error;
  400. }
  401. // Get the hardware format
  402. propertySize = sizeof(AudioStreamBasicDescription);
  403. err = AudioUnitGetProperty(data->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input, 1, &hardwareFormat, &propertySize);
  404. if(err != noErr || propertySize != sizeof(AudioStreamBasicDescription))
  405. {
  406. ERR("AudioUnitGetProperty failed\n");
  407. goto error;
  408. }
  409. // Set up the requested format description
  410. switch(device->FmtType)
  411. {
  412. case DevFmtUByte:
  413. requestedFormat.mBitsPerChannel = 8;
  414. requestedFormat.mFormatFlags = kAudioFormatFlagIsPacked;
  415. break;
  416. case DevFmtShort:
  417. requestedFormat.mBitsPerChannel = 16;
  418. requestedFormat.mFormatFlags = kAudioFormatFlagIsSignedInteger | kAudioFormatFlagsNativeEndian | kAudioFormatFlagIsPacked;
  419. break;
  420. case DevFmtInt:
  421. requestedFormat.mBitsPerChannel = 32;
  422. requestedFormat.mFormatFlags = kAudioFormatFlagIsSignedInteger | kAudioFormatFlagsNativeEndian | kAudioFormatFlagIsPacked;
  423. break;
  424. case DevFmtFloat:
  425. requestedFormat.mBitsPerChannel = 32;
  426. requestedFormat.mFormatFlags = kAudioFormatFlagIsPacked;
  427. break;
  428. case DevFmtByte:
  429. case DevFmtUShort:
  430. case DevFmtUInt:
  431. ERR("%s samples not supported\n", DevFmtTypeString(device->FmtType));
  432. goto error;
  433. }
  434. switch(device->FmtChans)
  435. {
  436. case DevFmtMono:
  437. requestedFormat.mChannelsPerFrame = 1;
  438. break;
  439. case DevFmtStereo:
  440. requestedFormat.mChannelsPerFrame = 2;
  441. break;
  442. case DevFmtQuad:
  443. case DevFmtX51:
  444. case DevFmtX51Side:
  445. case DevFmtX61:
  446. case DevFmtX71:
  447. ERR("%s not supported\n", DevFmtChannelsString(device->FmtChans));
  448. goto error;
  449. }
  450. requestedFormat.mBytesPerFrame = requestedFormat.mChannelsPerFrame * requestedFormat.mBitsPerChannel / 8;
  451. requestedFormat.mBytesPerPacket = requestedFormat.mBytesPerFrame;
  452. requestedFormat.mSampleRate = device->Frequency;
  453. requestedFormat.mFormatID = kAudioFormatLinearPCM;
  454. requestedFormat.mReserved = 0;
  455. requestedFormat.mFramesPerPacket = 1;
  456. // save requested format description for later use
  457. data->format = requestedFormat;
  458. data->frameSize = FrameSizeFromDevFmt(device->FmtChans, device->FmtType);
  459. // Use intermediate format for sample rate conversion (outputFormat)
  460. // Set sample rate to the same as hardware for resampling later
  461. outputFormat = requestedFormat;
  462. outputFormat.mSampleRate = hardwareFormat.mSampleRate;
  463. // Determine sample rate ratio for resampling
  464. data->sampleRateRatio = outputFormat.mSampleRate / device->Frequency;
  465. // The output format should be the requested format, but using the hardware sample rate
  466. // This is because the AudioUnit will automatically scale other properties, except for sample rate
  467. err = AudioUnitSetProperty(data->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Output, 1, (void *)&outputFormat, sizeof(outputFormat));
  468. if(err != noErr)
  469. {
  470. ERR("AudioUnitSetProperty failed\n");
  471. goto error;
  472. }
  473. // Set the AudioUnit output format frame count
  474. outputFrameCount = device->UpdateSize * data->sampleRateRatio;
  475. err = AudioUnitSetProperty(data->audioUnit, kAudioUnitProperty_MaximumFramesPerSlice, kAudioUnitScope_Output, 0, &outputFrameCount, sizeof(outputFrameCount));
  476. if(err != noErr)
  477. {
  478. ERR("AudioUnitSetProperty failed: %d\n", err);
  479. goto error;
  480. }
  481. // Set up sample converter
  482. err = AudioConverterNew(&outputFormat, &requestedFormat, &data->audioConverter);
  483. if(err != noErr)
  484. {
  485. ERR("AudioConverterNew failed: %d\n", err);
  486. goto error;
  487. }
  488. // Create a buffer for use in the resample callback
  489. data->resampleBuffer = malloc(device->UpdateSize * data->frameSize * data->sampleRateRatio);
  490. // Allocate buffer for the AudioUnit output
  491. data->bufferList = allocate_buffer_list(outputFormat.mChannelsPerFrame, device->UpdateSize * data->frameSize * data->sampleRateRatio);
  492. if(data->bufferList == NULL)
  493. goto error;
  494. data->ring = CreateRingBuffer(data->frameSize, (device->UpdateSize * data->sampleRateRatio) * device->NumUpdates);
  495. if(data->ring == NULL)
  496. goto error;
  497. return ALC_NO_ERROR;
  498. error:
  499. DestroyRingBuffer(data->ring);
  500. free(data->resampleBuffer);
  501. destroy_buffer_list(data->bufferList);
  502. if(data->audioConverter)
  503. AudioConverterDispose(data->audioConverter);
  504. if(data->audioUnit)
  505. CloseComponent(data->audioUnit);
  506. free(data);
  507. device->ExtraData = NULL;
  508. return ALC_INVALID_VALUE;
  509. }
  510. static void ca_close_capture(ALCdevice *device)
  511. {
  512. ca_data *data = (ca_data*)device->ExtraData;
  513. DestroyRingBuffer(data->ring);
  514. free(data->resampleBuffer);
  515. destroy_buffer_list(data->bufferList);
  516. AudioConverterDispose(data->audioConverter);
  517. CloseComponent(data->audioUnit);
  518. free(data);
  519. device->ExtraData = NULL;
  520. }
  521. static void ca_start_capture(ALCdevice *device)
  522. {
  523. ca_data *data = (ca_data*)device->ExtraData;
  524. OSStatus err = AudioOutputUnitStart(data->audioUnit);
  525. if(err != noErr)
  526. ERR("AudioOutputUnitStart failed\n");
  527. }
  528. static void ca_stop_capture(ALCdevice *device)
  529. {
  530. ca_data *data = (ca_data*)device->ExtraData;
  531. OSStatus err = AudioOutputUnitStop(data->audioUnit);
  532. if(err != noErr)
  533. ERR("AudioOutputUnitStop failed\n");
  534. }
  535. static ALCenum ca_capture_samples(ALCdevice *device, ALCvoid *buffer, ALCuint samples)
  536. {
  537. ca_data *data = (ca_data*)device->ExtraData;
  538. AudioBufferList *list;
  539. UInt32 frameCount;
  540. OSStatus err;
  541. // If no samples are requested, just return
  542. if(samples == 0)
  543. return ALC_NO_ERROR;
  544. // Allocate a temporary AudioBufferList to use as the return resamples data
  545. list = alloca(sizeof(AudioBufferList) + sizeof(AudioBuffer));
  546. // Point the resampling buffer to the capture buffer
  547. list->mNumberBuffers = 1;
  548. list->mBuffers[0].mNumberChannels = data->format.mChannelsPerFrame;
  549. list->mBuffers[0].mDataByteSize = samples * data->frameSize;
  550. list->mBuffers[0].mData = buffer;
  551. // Resample into another AudioBufferList
  552. frameCount = samples;
  553. err = AudioConverterFillComplexBuffer(data->audioConverter, ca_capture_conversion_callback,
  554. device, &frameCount, list, NULL);
  555. if(err != noErr)
  556. {
  557. ERR("AudioConverterFillComplexBuffer error: %d\n", err);
  558. return ALC_INVALID_VALUE;
  559. }
  560. return ALC_NO_ERROR;
  561. }
  562. static ALCuint ca_available_samples(ALCdevice *device)
  563. {
  564. ca_data *data = device->ExtraData;
  565. return RingBufferSize(data->ring) / data->sampleRateRatio;
  566. }
  567. static const BackendFuncs ca_funcs = {
  568. ca_open_playback,
  569. ca_close_playback,
  570. ca_reset_playback,
  571. ca_start_playback,
  572. ca_stop_playback,
  573. ca_open_capture,
  574. ca_close_capture,
  575. ca_start_capture,
  576. ca_stop_capture,
  577. ca_capture_samples,
  578. ca_available_samples,
  579. ALCdevice_GetLatencyDefault
  580. };
  581. ALCboolean alc_ca_init(BackendFuncs *func_list)
  582. {
  583. *func_list = ca_funcs;
  584. return ALC_TRUE;
  585. }
  586. void alc_ca_deinit(void)
  587. {
  588. }
  589. void alc_ca_probe(enum DevProbe type)
  590. {
  591. switch(type)
  592. {
  593. case ALL_DEVICE_PROBE:
  594. AppendAllDevicesList(ca_device);
  595. break;
  596. case CAPTURE_DEVICE_PROBE:
  597. AppendCaptureDeviceList(ca_device);
  598. break;
  599. }
  600. }