coreaudio.c 23 KB

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