VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostCoreAudio.cpp@ 62448

Last change on this file since 62448 was 62337, checked in by vboxsync, 9 years ago

Build fix.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 107.2 KB
Line 
1/* $Id: DrvHostCoreAudio.cpp 62337 2016-07-19 18:59:29Z vboxsync $ */
2/** @file
3 * VBox audio devices: Mac OS X CoreAudio audio driver.
4 */
5
6/*
7 * Copyright (C) 2010-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
18#include <VBox/log.h>
19#include <VBox/vmm/pdmaudioifs.h>
20
21#include "DrvAudio.h"
22#include "AudioMixBuffer.h"
23
24#include "VBoxDD.h"
25
26#include <iprt/asm.h>
27#include <iprt/cdefs.h>
28#include <iprt/circbuf.h>
29#include <iprt/mem.h>
30
31#include <iprt/uuid.h>
32
33#include <CoreAudio/CoreAudio.h>
34#include <CoreServices/CoreServices.h>
35#include <AudioUnit/AudioUnit.h>
36#include <AudioToolbox/AudioConverter.h>
37
38#if 0
39# include <iprt/file.h>
40# define DEBUG_DUMP_PCM_DATA
41# ifdef RT_OS_WINDOWS
42# define DEBUG_DUMP_PCM_DATA_PATH "c:\\temp\\"
43# else
44# define DEBUG_DUMP_PCM_DATA_PATH "/tmp/"
45# endif
46#endif
47
48/* TODO:
49 * - Maybe make sure the threads are immediately stopped if playing/recording stops.
50 */
51
52/*
53 * Most of this is based on:
54 * http://developer.apple.com/mac/library/technotes/tn2004/tn2097.html
55 * http://developer.apple.com/mac/library/technotes/tn2002/tn2091.html
56 * http://developer.apple.com/mac/library/qa/qa2007/qa1533.html
57 * http://developer.apple.com/mac/library/qa/qa2001/qa1317.html
58 * http://developer.apple.com/mac/library/documentation/AudioUnit/Reference/AUComponentServicesReference/Reference/reference.html
59 */
60
61/**
62 * Host Coreaudio driver instance data.
63 * @implements PDMIAUDIOCONNECTOR
64 */
65typedef struct DRVHOSTCOREAUDIO
66{
67 /** Pointer to the driver instance structure. */
68 PPDMDRVINS pDrvIns;
69 /** Pointer to host audio interface. */
70 PDMIHOSTAUDIO IHostAudio;
71} DRVHOSTCOREAUDIO, *PDRVHOSTCOREAUDIO;
72
73/*******************************************************************************
74 *
75 * Helper function section
76 *
77 ******************************************************************************/
78
79static void coreAudioPrintASBD(const char *pszDesc, const AudioStreamBasicDescription *pASBD)
80{
81 char pszSampleRate[32];
82 LogRel2(("CoreAudio: %s description:\n", pszDesc));
83 LogRel2(("CoreAudio: Format ID: %RU32 (%c%c%c%c)\n", pASBD->mFormatID,
84 RT_BYTE4(pASBD->mFormatID), RT_BYTE3(pASBD->mFormatID),
85 RT_BYTE2(pASBD->mFormatID), RT_BYTE1(pASBD->mFormatID)));
86 LogRel2(("CoreAudio: Flags: %RU32", pASBD->mFormatFlags));
87 if (pASBD->mFormatFlags & kAudioFormatFlagIsFloat)
88 LogRel2((" Float"));
89 if (pASBD->mFormatFlags & kAudioFormatFlagIsBigEndian)
90 LogRel2((" BigEndian"));
91 if (pASBD->mFormatFlags & kAudioFormatFlagIsSignedInteger)
92 LogRel2((" SignedInteger"));
93 if (pASBD->mFormatFlags & kAudioFormatFlagIsPacked)
94 LogRel2((" Packed"));
95 if (pASBD->mFormatFlags & kAudioFormatFlagIsAlignedHigh)
96 LogRel2((" AlignedHigh"));
97 if (pASBD->mFormatFlags & kAudioFormatFlagIsNonInterleaved)
98 LogRel2((" NonInterleaved"));
99 if (pASBD->mFormatFlags & kAudioFormatFlagIsNonMixable)
100 LogRel2((" NonMixable"));
101 if (pASBD->mFormatFlags & kAudioFormatFlagsAreAllClear)
102 LogRel2((" AllClear"));
103 LogRel2(("\n"));
104 snprintf(pszSampleRate, 32, "%.2f", (float)pASBD->mSampleRate); /** @todo r=andy Use RTStrPrint*. */
105 LogRel2(("CoreAudio: SampleRate : %s\n", pszSampleRate));
106 LogRel2(("CoreAudio: ChannelsPerFrame: %RU32\n", pASBD->mChannelsPerFrame));
107 LogRel2(("CoreAudio: FramesPerPacket : %RU32\n", pASBD->mFramesPerPacket));
108 LogRel2(("CoreAudio: BitsPerChannel : %RU32\n", pASBD->mBitsPerChannel));
109 LogRel2(("CoreAudio: BytesPerFrame : %RU32\n", pASBD->mBytesPerFrame));
110 LogRel2(("CoreAudio: BytesPerPacket : %RU32\n", pASBD->mBytesPerPacket));
111}
112
113static void coreAudioPCMPropsToASBD(PDMPCMPROPS *pPCMProps, AudioStreamBasicDescription *pASBD)
114{
115 AssertPtrReturnVoid(pPCMProps);
116 AssertPtrReturnVoid(pASBD);
117
118 RT_BZERO(pASBD, sizeof(AudioStreamBasicDescription));
119
120 pASBD->mFormatID = kAudioFormatLinearPCM;
121 pASBD->mFormatFlags = kAudioFormatFlagIsPacked;
122 pASBD->mFramesPerPacket = 1; /* For uncompressed audio, set this to 1. */
123 pASBD->mSampleRate = (Float64)pPCMProps->uHz;
124 pASBD->mChannelsPerFrame = pPCMProps->cChannels;
125 pASBD->mBitsPerChannel = pPCMProps->cBits;
126 if (pPCMProps->fSigned)
127 pASBD->mFormatFlags |= kAudioFormatFlagIsSignedInteger;
128 pASBD->mBytesPerFrame = pASBD->mChannelsPerFrame * (pASBD->mBitsPerChannel / 8);
129 pASBD->mBytesPerPacket = pASBD->mFramesPerPacket * pASBD->mBytesPerFrame;
130}
131
132static OSStatus coreAudioSetFrameBufferSize(AudioDeviceID deviceID, bool fInput, UInt32 cReqSize, UInt32 *pcActSize)
133{
134 AudioObjectPropertyScope propScope = fInput
135 ? kAudioDevicePropertyScopeInput : kAudioDevicePropertyScopeOutput;
136 AudioObjectPropertyAddress propAdr = { kAudioDevicePropertyBufferFrameSize, propScope,
137 kAudioObjectPropertyElementMaster };
138
139 /* First try to set the new frame buffer size. */
140 OSStatus err = AudioObjectSetPropertyData(deviceID, &propAdr, 0, NULL, sizeof(cReqSize), &cReqSize);
141
142 /* Check if it really was set. */
143 UInt32 cSize = sizeof(*pcActSize);
144 err = AudioObjectGetPropertyData(deviceID, &propAdr, 0, NULL, &cSize, pcActSize);
145 if (RT_UNLIKELY(err != noErr))
146 return err;
147
148 /* If both sizes are the same, we are done. */
149 if (cReqSize == *pcActSize)
150 return noErr;
151
152 /* If not we have to check the limits of the device. First get the size of
153 the buffer size range property. */
154 propAdr.mSelector = kAudioDevicePropertyBufferSizeRange;
155 err = AudioObjectGetPropertyDataSize(deviceID, &propAdr, 0, NULL, &cSize);
156 if (RT_UNLIKELY(err != noErr))
157 return err;
158
159 Assert(cSize);
160 AudioValueRange *pRange = (AudioValueRange *)RTMemAllocZ(cSize);
161 if (pRange)
162 {
163 err = AudioObjectGetPropertyData(deviceID, &propAdr, 0, NULL, &cSize, pRange);
164 if (err == noErr)
165 {
166 Float64 cMin = -1;
167 Float64 cMax = -1;
168 for (size_t a = 0; a < cSize / sizeof(AudioValueRange); a++)
169 {
170 /* Search for the absolute minimum. */
171 if ( pRange[a].mMinimum < cMin
172 || cMin == -1)
173 cMin = pRange[a].mMinimum;
174
175 /* Search for the best maximum which isn't bigger than cReqSize. */
176 if (pRange[a].mMaximum < cReqSize)
177 {
178 if (pRange[a].mMaximum > cMax)
179 cMax = pRange[a].mMaximum;
180 }
181 }
182 if (cMax == -1)
183 cMax = cMin;
184 cReqSize = cMax;
185
186 /* First try to set the new frame buffer size. */
187 propAdr.mSelector = kAudioDevicePropertyBufferFrameSize;
188 err = AudioObjectSetPropertyData(deviceID, &propAdr, 0, NULL, sizeof(cReqSize), &cReqSize);
189 if (err == noErr)
190 {
191 /* Check if it really was set. */
192 cSize = sizeof(*pcActSize);
193 err = AudioObjectGetPropertyData(deviceID, &propAdr, 0, NULL, &cSize, pcActSize);
194 }
195 }
196
197 RTMemFree(pRange);
198 }
199 else
200 err = notEnoughMemoryErr;
201
202 return err;
203}
204
205DECL_FORCE_INLINE(bool) coreAudioIsRunning(AudioDeviceID deviceID)
206{
207 AudioObjectPropertyAddress propAdr = { kAudioDevicePropertyDeviceIsRunning, kAudioObjectPropertyScopeGlobal,
208 kAudioObjectPropertyElementMaster };
209 UInt32 uFlag = 0;
210 UInt32 uSize = sizeof(uFlag);
211 OSStatus err = AudioObjectGetPropertyData(deviceID, &propAdr, 0, NULL, &uSize, &uFlag);
212 if (err != kAudioHardwareNoError)
213 LogRel(("CoreAudio: Could not determine whether the device is running (%RI32)\n", err));
214
215 return (uFlag >= 1);
216}
217
218static int coreAudioCFStringToCString(const CFStringRef pCFString, char **ppszString)
219{
220 CFIndex cLen = CFStringGetLength(pCFString) + 1;
221 char *pszResult = (char *)RTMemAllocZ(cLen * sizeof(char));
222 if (!CFStringGetCString(pCFString, pszResult, cLen, kCFStringEncodingUTF8))
223 {
224 RTMemFree(pszResult);
225 return VERR_NOT_FOUND;
226 }
227
228 *ppszString = pszResult;
229 return VINF_SUCCESS;
230}
231
232static AudioDeviceID coreAudioDeviceUIDtoID(const char* pszUID)
233{
234 /* Create a CFString out of our CString. */
235 CFStringRef strUID = CFStringCreateWithCString(NULL, pszUID, kCFStringEncodingMacRoman);
236
237 /* Fill the translation structure. */
238 AudioDeviceID deviceID;
239
240 AudioValueTranslation translation;
241 translation.mInputData = &strUID;
242 translation.mInputDataSize = sizeof(CFStringRef);
243 translation.mOutputData = &deviceID;
244 translation.mOutputDataSize = sizeof(AudioDeviceID);
245
246 /* Fetch the translation from the UID to the device ID. */
247 AudioObjectPropertyAddress propAdr = { kAudioHardwarePropertyDeviceForUID, kAudioObjectPropertyScopeGlobal,
248 kAudioObjectPropertyElementMaster };
249
250 UInt32 uSize = sizeof(AudioValueTranslation);
251 OSStatus err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &propAdr, 0, NULL, &uSize, &translation);
252
253 /* Release the temporary CFString */
254 CFRelease(strUID);
255
256 if (RT_LIKELY(err == noErr))
257 return deviceID;
258
259 /* Return the unknown device on error. */
260 return kAudioDeviceUnknown;
261}
262
263/*******************************************************************************
264 *
265 * Global structures section
266 *
267 ******************************************************************************/
268
269/* Initialization status indicator used for the recreation of the AudioUnits. */
270#define CA_STATUS_UNINIT UINT32_C(0) /* The device is uninitialized */
271#define CA_STATUS_IN_INIT UINT32_C(1) /* The device is currently initializing */
272#define CA_STATUS_INIT UINT32_C(2) /* The device is initialized */
273#define CA_STATUS_IN_UNINIT UINT32_C(3) /* The device is currently uninitializing */
274#define CA_STATUS_REINIT UINT32_C(4) /* The device has to be reinitialized */
275
276/* Error code which indicates "End of data" */
277static const OSStatus caConverterEOFDErr = 0x656F6664; /* 'eofd' */
278
279/* Prototypes needed for COREAUDIOSTREAMCBCTX. */
280struct COREAUDIOSTREAMIN;
281typedef struct COREAUDIOSTREAMIN *PCOREAUDIOSTREAMIN;
282struct COREAUDIOSTREAMOUT;
283typedef struct COREAUDIOSTREAMOUT *PCOREAUDIOSTREAMOUT;
284
285/**
286 * Simple structure for maintaining a stream's callback context.
287 ** @todo Remove this as soon as we have unified input/output streams in this backend.
288 */
289typedef struct COREAUDIOSTREAMCBCTX
290{
291 /** Pointer to driver instance. */
292 PDRVHOSTCOREAUDIO pThis;
293 /** The stream's direction. */
294 PDMAUDIODIR enmDir;
295 union
296 {
297 /** Pointer to self, if it's an input stream. */
298 PCOREAUDIOSTREAMIN pIn;
299 /** Pointer to self, if it's an output stream. */
300 PCOREAUDIOSTREAMOUT pOut;
301 };
302} COREAUDIOSTREAMCBCTX, *PCOREAUDIOSTREAMCBCTX;
303
304/**
305 * Structure for keeping a conversion callback context.
306 * This is needed when using an audio converter during input/output processing.
307 */
308typedef struct COREAUDIOCONVCBCTX
309{
310 /** Pointer to stream context this converter callback context
311 * is bound to. */
312 /** @todo Remove this as soon as we have unified input/output streams in this backend. */
313 COREAUDIOSTREAMCBCTX pStream;
314 /** Source stream description. */
315 AudioStreamBasicDescription asbdSrc;
316 /** Destination stream description. */
317 AudioStreamBasicDescription asbdDst;
318 /** Native buffer list used for rendering the source audio data into. */
319 AudioBufferList bufLstSrc;
320 /** Total packet conversion count. */
321 UInt32 uPacketCnt;
322 /** Current packet conversion index. */
323 UInt32 uPacketIdx;
324} COREAUDIOCONVCBCTX, *PCOREAUDIOCONVCBCTX;
325
326/** @todo Unify COREAUDIOSTREAMOUT / COREAUDIOSTREAMIN. */
327typedef struct COREAUDIOSTREAMOUT
328{
329 /** Host output stream.
330 * Note: Always must come first in this structure! */
331 PDMAUDIOSTREAM Stream;
332 /** Stream description which is default on the device. */
333 AudioStreamBasicDescription deviceFormat;
334 /** Stream description which is selected for using with VBox. */
335 AudioStreamBasicDescription streamFormat;
336 /** The audio device ID of the currently used device. */
337 AudioDeviceID deviceID;
338 /** The AudioUnit being used. */
339 AudioUnit audioUnit;
340 /** A ring buffer for transferring data to the playback thread. */
341 PRTCIRCBUF pCircBuf;
342 /** Initialization status tracker. Used when some of the device parameters
343 * or the device itself is changed during the runtime. */
344 volatile uint32_t status;
345 /** Flag whether the "default device changed" listener was registered. */
346 bool fDefDevChgListReg;
347 /** Flag whether the "device state changed" listener was registered. */
348 bool fDevStateChgListReg;
349 /** Callback context for this stream for handing this stream in to
350 * a CoreAudio callback.
351 ** @todo Remove this as soon as we have unified input/output streams in this backend. */
352 COREAUDIOSTREAMCBCTX cbCtx;
353} COREAUDIOSTREAMOUT, *PCOREAUDIOSTREAMOUT;
354
355typedef struct COREAUDIOSTREAMIN
356{
357 /** Host input stream.
358 * Note: Always must come first in this structure! */
359 PDMAUDIOSTREAM Stream;
360 /** Stream description which is default on the device. */
361 AudioStreamBasicDescription deviceFormat;
362 /** Stream description which is selected for using with VBox. */
363 AudioStreamBasicDescription streamFormat;
364 /** The audio device ID of the currently used device. */
365 AudioDeviceID deviceID;
366 /** The AudioUnit used. */
367 AudioUnit audioUnit;
368 /** A ring buffer for transferring data from the capturing thread. */
369 PRTCIRCBUF pCircBuf;
370 /** The audio converter if necessary. NULL if no converter is being used. */
371 AudioConverterRef pConverter;
372 /** Callback context for the audio converter. */
373 COREAUDIOCONVCBCTX convCbCtx;
374 /** The ratio between the device & the stream sample rate. */
375 Float64 sampleRatio;
376 /** Initialization status tracker. Used when some of the device parameters
377 * or the device itself is changed during the runtime. */
378 volatile uint32_t status;
379 /** Flag whether the "default device changed" listener was registered. */
380 bool fDefDevChgListReg;
381 /** Flag whether the "device state changed" listener was registered. */
382 bool fDevStateChgListReg;
383 /** Callback context for this stream for handing this stream in to
384 * a CoreAudio callback.
385 ** @todo Remove this as soon as we have unified input/output streams in this backend. */
386 COREAUDIOSTREAMCBCTX cbCtx;
387} COREAUDIOSTREAMIN, *PCOREAUDIOSTREAMIN;
388
389
390static int coreAudioInitIn(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream, uint32_t *pcSamples);
391static int coreAudioInitOut(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream, uint32_t *pcSamples);
392static int coreAudioControlStreamIn(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd);
393static int coreAudioControlStreamOut(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd);
394static int coreAudioCreateStreamIn(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples);
395static int coreAudioCreateStreamOut(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples);
396static int coreAudioDestroyStreamIn(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream);
397static int coreAudioDestroyStreamOut(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream);
398
399static OSStatus coreAudioPlaybackAudioDevicePropertyChanged(AudioObjectID propertyID, UInt32 nAddresses, const AudioObjectPropertyAddress properties[], void *pvUser);
400static OSStatus coreAudioPlaybackCb(void *pvUser, AudioUnitRenderActionFlags *pActionFlags, const AudioTimeStamp *pAudioTS, UInt32 uBusID, UInt32 cFrames, AudioBufferList* pBufData);
401
402/**
403 * Initializes a conversion callback context.
404 *
405 * @return IPRT status code.
406 * @param pConvCbCtx Conversion callback context to initialize.
407 * @param pASBDSrc Input (source) stream description to use.
408 * @param pASBDDst Output (destination) stream description to use.
409 */
410static int coreAudioInitConvCbCtx(PCOREAUDIOCONVCBCTX pConvCbCtx,
411 AudioStreamBasicDescription *pASBDSrc, AudioStreamBasicDescription *pASBDDst)
412{
413 AssertPtrReturn(pConvCbCtx, VERR_INVALID_POINTER);
414 AssertPtrReturn(pASBDSrc, VERR_INVALID_POINTER);
415 AssertPtrReturn(pASBDDst, VERR_INVALID_POINTER);
416
417 memcpy(&pConvCbCtx->asbdSrc, pASBDSrc, sizeof(AudioStreamBasicDescription));
418 memcpy(&pConvCbCtx->asbdDst, pASBDDst, sizeof(AudioStreamBasicDescription));
419
420 /* Create the AudioBufferList structure with one buffer. */
421 pConvCbCtx->bufLstSrc.mNumberBuffers = 1;
422
423 /* Initialize the conversion buffer. */
424 pConvCbCtx->bufLstSrc.mBuffers[0].mNumberChannels = pConvCbCtx->asbdSrc.mChannelsPerFrame;
425 pConvCbCtx->bufLstSrc.mBuffers[0].mDataByteSize = 0;
426 pConvCbCtx->bufLstSrc.mBuffers[0].mData = NULL;
427
428 return VINF_SUCCESS;
429}
430
431/**
432 * Uninitializes a conversion callback context.
433 *
434 * @return IPRT status code.
435 * @param pConvCbCtx Conversion callback context to uninitialize.
436 */
437static void coreAudioUninitConvCbCtx(PCOREAUDIOCONVCBCTX pConvCbCtx)
438{
439 AssertPtrReturnVoid(pConvCbCtx);
440
441 RT_ZERO(pConvCbCtx->asbdSrc);
442 RT_ZERO(pConvCbCtx->asbdDst);
443
444 pConvCbCtx->bufLstSrc.mNumberBuffers = 0;
445}
446
447/**
448 * Does a (Re-)enumeration of the host's playback + recording devices.
449 *
450 * @return IPRT status code.
451 * @param pThis Host audio driver instance.
452 * @param pCfg Where to store the enumeration results.
453 * @param fEnum Enumeration flags.
454 */
455static int coreAudioDevicesEnumerate(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOBACKENDCFG pCfg, bool fIn, uint32_t fEnum)
456{
457 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
458 /* pCfg is optional. */
459
460 int rc = VINF_SUCCESS;
461
462 uint8_t cDevs = 0;
463
464 do
465 {
466 AudioObjectPropertyAddress propAdrDevList = { kAudioHardwarePropertyDevices, kAudioObjectPropertyScopeGlobal,
467 kAudioObjectPropertyElementMaster };
468 UInt32 uSize = 0;
469 OSStatus err = AudioObjectGetPropertyDataSize(kAudioObjectSystemObject, &propAdrDevList, 0, NULL, &uSize);
470 if (err != kAudioHardwareNoError)
471 break;
472
473 AudioDeviceID *pDevIDs = (AudioDeviceID *)alloca(uSize);
474 if (pDevIDs == NULL)
475 break;
476
477 err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &propAdrDevList, 0, NULL, &uSize, pDevIDs);
478 if (err != kAudioHardwareNoError)
479 break;
480
481 UInt32 cDevices = uSize / sizeof (AudioDeviceID);
482 for (UInt32 i = 0; i < cDevices; i++)
483 {
484 AudioDeviceID curDevID = pDevIDs[i];
485
486 /* Check if the device is valid. */
487 AudioObjectPropertyAddress propAddrCfg = { kAudioDevicePropertyStreamConfiguration,
488 fIn ? kAudioDevicePropertyScopeInput : kAudioDevicePropertyScopeOutput,
489 kAudioObjectPropertyElementMaster };
490
491 err = AudioObjectGetPropertyDataSize(curDevID, &propAddrCfg, 0, NULL, &uSize);
492 if (err != noErr)
493 continue;
494
495 AudioBufferList *pBufList = (AudioBufferList *)RTMemAlloc(uSize);
496 if (!pBufList)
497 continue;
498
499 bool fIsValid = false;
500
501 err = AudioObjectGetPropertyData(curDevID, &propAddrCfg, 0, NULL, &uSize, pBufList);
502 if (err == noErr)
503 {
504 for (UInt32 a = 0; a < pBufList->mNumberBuffers; a++)
505 {
506 fIsValid = pBufList->mBuffers[a].mNumberChannels > 0;
507 if (fIsValid)
508 break;
509 }
510 }
511
512 if (pBufList)
513 {
514 RTMemFree(pBufList);
515 pBufList = NULL;
516 }
517
518 if (!fIsValid)
519 continue;
520
521 /* Resolve the device's name. */
522 AudioObjectPropertyAddress propAddrName = { kAudioObjectPropertyName,
523 fIn ? kAudioDevicePropertyScopeInput : kAudioDevicePropertyScopeOutput,
524 kAudioObjectPropertyElementMaster };
525 uSize = sizeof(CFStringRef);
526 CFStringRef pcfstrName = NULL;
527
528 err = AudioObjectGetPropertyData(curDevID, &propAddrName, 0, NULL, &uSize, &pcfstrName);
529 if (err != kAudioHardwareNoError)
530 continue;
531
532 CFIndex uMax = CFStringGetMaximumSizeForEncoding(CFStringGetLength(pcfstrName), kCFStringEncodingUTF8) + 1;
533 if (uMax)
534 {
535 char *pszName = (char *)RTStrAlloc(uMax);
536 if ( pszName
537 && CFStringGetCString(pcfstrName, pszName, uMax, kCFStringEncodingUTF8))
538 {
539 LogRel2(("CoreAudio: Found %s device '%s'\n", fIn ? "recording" : "playback", pszName));
540 cDevs++;
541 }
542
543 if (pszName)
544 {
545 RTStrFree(pszName);
546 pszName = NULL;
547 }
548 }
549
550 CFRelease(pcfstrName);
551 }
552
553 } while (0);
554
555 if (fIn)
556 LogRel2(("CoreAudio: Found %RU8 recording device(s)\n", cDevs));
557 else
558 LogRel2(("CoreAudio: Found %RU8 playback device(s)\n", cDevs));
559
560 if (pCfg)
561 {
562 if (fIn)
563 pCfg->cSources = cDevs;
564 else
565 pCfg->cSinks = cDevs;
566 }
567
568 LogFlowFuncLeaveRC(rc);
569 return rc;
570}
571
572/**
573 * Updates this host driver's internal status, according to the global, overall input/output
574 * state and all connected (native) audio streams.
575 *
576 * @param pThis Host audio driver instance.
577 * @param pCfg Where to store the backend configuration. Optional.
578 * @param fEnum Enumeration flags.
579 */
580int coreAudioUpdateStatusInternalEx(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOBACKENDCFG pCfg, uint32_t fEnum)
581{
582 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
583 /* pCfg is optional. */
584
585 PDMAUDIOBACKENDCFG Cfg;
586 RT_ZERO(Cfg);
587
588 Cfg.cbStreamOut = sizeof(COREAUDIOSTREAMOUT);
589 Cfg.cbStreamIn = sizeof(COREAUDIOSTREAMIN);
590 Cfg.cMaxStreamsIn = UINT32_MAX;
591 Cfg.cMaxStreamsOut = UINT32_MAX;
592
593 int rc = coreAudioDevicesEnumerate(pThis, &Cfg, false /* fIn */, 0 /* fEnum */);
594 AssertRC(rc);
595 rc = coreAudioDevicesEnumerate(pThis, &Cfg, true /* fIn */, 0 /* fEnum */);
596 AssertRC(rc);
597
598 if (pCfg)
599 memcpy(pCfg, &Cfg, sizeof(PDMAUDIOBACKENDCFG));
600
601 LogFlowFuncLeaveRC(rc);
602 return rc;
603}
604
605static DECLCALLBACK(OSStatus) drvHostCoreAudioDeviceStateChanged(AudioObjectID propertyID,
606 UInt32 nAddresses,
607 const AudioObjectPropertyAddress properties[],
608 void *pvUser)
609{
610 LogFlowFunc(("propertyID=%u nAddresses=%u pvUser=%p\n", propertyID, nAddresses, pvUser));
611
612 PCOREAUDIOSTREAMCBCTX pCbCtx = (PCOREAUDIOSTREAMCBCTX)pvUser;
613 AssertPtr(pCbCtx);
614
615 UInt32 uAlive = 1;
616 UInt32 uSize = sizeof(UInt32);
617
618 AudioObjectPropertyAddress propAdr = { kAudioDevicePropertyDeviceIsAlive, kAudioObjectPropertyScopeGlobal,
619 kAudioObjectPropertyElementMaster };
620
621 AudioDeviceID deviceID = pCbCtx->enmDir == PDMAUDIODIR_IN
622 ? pCbCtx->pIn->deviceID : pCbCtx->pOut->deviceID;
623
624 OSStatus err = AudioObjectGetPropertyData(deviceID, &propAdr, 0, NULL, &uSize, &uAlive);
625
626 bool fIsDead = false;
627
628 if (err == kAudioHardwareBadDeviceError)
629 fIsDead = true; /* Unplugged. */
630 else if ((err == kAudioHardwareNoError) && (!RT_BOOL(uAlive)))
631 fIsDead = true; /* Something else happened. */
632
633 if (fIsDead)
634 {
635 switch (pCbCtx->enmDir)
636 {
637 case PDMAUDIODIR_IN:
638 {
639 PCOREAUDIOSTREAMIN pStreamIn = pCbCtx->pIn;
640
641 /* We move the reinitialization to the next output event.
642 * This make sure this thread isn't blocked and the
643 * reinitialization is done when necessary only. */
644 ASMAtomicXchgU32(&pStreamIn->status, CA_STATUS_REINIT);
645
646 LogRel(("CoreAudio: Recording device stopped functioning\n"));
647 break;
648 }
649
650 case PDMAUDIODIR_OUT:
651 {
652 PCOREAUDIOSTREAMOUT pStreamOut = pCbCtx->pOut;
653
654 /* We move the reinitialization to the next output event.
655 * This make sure this thread isn't blocked and the
656 * reinitialization is done when necessary only. */
657 ASMAtomicXchgU32(&pStreamOut->status, CA_STATUS_REINIT);
658
659 LogRel(("CoreAudio: Playback device stopped functioning\n"));
660 break;
661 }
662
663 default:
664 AssertMsgFailed(("Not implemented\n"));
665 break;
666 }
667 }
668
669 int rc2 = coreAudioDevicesEnumerate(pCbCtx->pThis, NULL /* pCfg */, false /* fIn */, 0 /* fEnum */);
670 AssertRC(rc2);
671 rc2 = coreAudioDevicesEnumerate(pCbCtx->pThis, NULL /* pCfg */, true /* fIn */, 0 /* fEnum */);
672 AssertRC(rc2);
673
674 return noErr;
675}
676
677/* Callback for getting notified when the default recording/playback device has been changed. */
678static DECLCALLBACK(OSStatus) coreAudioDefaultDeviceChanged(AudioObjectID propertyID,
679 UInt32 nAddresses,
680 const AudioObjectPropertyAddress properties[],
681 void *pvUser)
682{
683 OSStatus err = noErr;
684
685 LogFlowFunc(("propertyID=%u nAddresses=%u pvUser=%p\n", propertyID, nAddresses, pvUser));
686
687 PCOREAUDIOSTREAMCBCTX pCbCtx = (PCOREAUDIOSTREAMCBCTX)pvUser;
688 AssertPtr(pCbCtx);
689
690 for (UInt32 idxAddress = 0; idxAddress < nAddresses; idxAddress++)
691 {
692 const AudioObjectPropertyAddress *pProperty = &properties[idxAddress];
693
694 switch (pProperty->mSelector)
695 {
696 case kAudioHardwarePropertyDefaultInputDevice:
697 {
698 PCOREAUDIOSTREAMIN pStreamIn = pCbCtx->pIn;
699 AssertPtr(pStreamIn);
700
701 /* This listener is called on every change of the hardware
702 * device. So check if the default device has really changed. */
703 UInt32 uSize = sizeof(pStreamIn->deviceID);
704 UInt32 uResp;
705 err = AudioObjectGetPropertyData(kAudioObjectSystemObject, pProperty, 0, NULL, &uSize, &uResp);
706
707 if (err == noErr)
708 {
709 if (pStreamIn->deviceID != uResp)
710 {
711 LogRel(("CoreAudio: Default device for recording has changed\n"));
712
713 /* We move the reinitialization to the next input event.
714 * This make sure this thread isn't blocked and the
715 * reinitialization is done when necessary only. */
716 ASMAtomicXchgU32(&pStreamIn->status, CA_STATUS_REINIT);
717 }
718 }
719 break;
720 }
721
722 case kAudioHardwarePropertyDefaultOutputDevice:
723 {
724 PCOREAUDIOSTREAMOUT pStreamOut = pCbCtx->pOut;
725 AssertPtr(pStreamOut);
726
727 /* This listener is called on every change of the hardware
728 * device. So check if the default device has really changed. */
729 AudioObjectPropertyAddress propAdr = { kAudioHardwarePropertyDefaultOutputDevice,
730 kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMaster };
731
732 UInt32 uSize = sizeof(pStreamOut->deviceID);
733 UInt32 uResp;
734 err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &propAdr, 0, NULL, &uSize, &uResp);
735
736 if (err == noErr)
737 {
738 if (pStreamOut->deviceID != uResp)
739 {
740 LogRel(("CoreAudio: Default device for playback has changed\n"));
741
742 /* We move the reinitialization to the next input event.
743 * This make sure this thread isn't blocked and the
744 * reinitialization is done when necessary only. */
745 ASMAtomicXchgU32(&pStreamOut->status, CA_STATUS_REINIT);
746 }
747 }
748 break;
749 }
750
751 default:
752 break;
753 }
754 }
755
756 int rc2 = coreAudioDevicesEnumerate(pCbCtx->pThis, NULL /* pCfg */, false /* fIn */, 0 /* fEnum */);
757 AssertRC(rc2);
758 rc2 = coreAudioDevicesEnumerate(pCbCtx->pThis, NULL /* pCfg */, true /* fIn */, 0 /* fEnum */);
759 AssertRC(rc2);
760
761 /** @todo Implement callback notification here to let the audio connector / device emulation
762 * know that something has changed. */
763
764 return noErr;
765}
766
767/* Callback for getting notified when some of the properties of an audio device has changed. */
768static DECLCALLBACK(OSStatus) coreAudioRecordingAudioDevicePropertyChanged(AudioObjectID propertyID,
769 UInt32 cAdresses,
770 const AudioObjectPropertyAddress aProperties[],
771 void *pvUser)
772{
773 PCOREAUDIOSTREAMIN pStreamIn = (PCOREAUDIOSTREAMIN)pvUser;
774
775 switch (propertyID)
776 {
777#ifdef DEBUG
778 case kAudioDeviceProcessorOverload:
779 {
780 LogFunc(("Processor overload detected!\n"));
781 break;
782 }
783#endif /* DEBUG */
784 case kAudioDevicePropertyNominalSampleRate:
785 {
786 LogRel(("CoreAudio: Recording sample rate changed\n"));
787
788 /* We move the reinitialization to the next input event.
789 * This make sure this thread isn't blocked and the
790 * reinitialization is done when necessary only. */
791 ASMAtomicXchgU32(&pStreamIn->status, CA_STATUS_REINIT);
792 break;
793 }
794
795 default:
796 break;
797 }
798
799 return noErr;
800}
801
802/* Callback to convert audio input data from one format to another. */
803static DECLCALLBACK(OSStatus) coreAudioConverterCallback(AudioConverterRef inAudioConverter,
804 UInt32 *ioNumberDataPackets,
805 AudioBufferList *ioData,
806 AudioStreamPacketDescription **ppASPD,
807 void *pvUser)
808{
809 AssertPtrReturn(ioNumberDataPackets, caConverterEOFDErr);
810 AssertPtrReturn(ioData, caConverterEOFDErr);
811
812 PCOREAUDIOCONVCBCTX pConvCbCtx = (PCOREAUDIOCONVCBCTX)pvUser;
813 AssertPtr(pConvCbCtx);
814
815 /* Initialize values. */
816 ioData->mBuffers[0].mNumberChannels = 0;
817 ioData->mBuffers[0].mDataByteSize = 0;
818 ioData->mBuffers[0].mData = NULL;
819
820 /** @todo Check converter ID? */
821
822 /** @todo Handled non-interleaved data by going through the full buffer list,
823 * not only through the first buffer like we do now. */
824
825 Log3Func(("ioNumberDataPackets=%RU32\n", *ioNumberDataPackets));
826
827 if (pConvCbCtx->uPacketIdx + *ioNumberDataPackets > pConvCbCtx->uPacketCnt)
828 {
829 Log3Func(("Limiting ioNumberDataPackets to %RU32\n", pConvCbCtx->uPacketCnt - pConvCbCtx->uPacketIdx));
830 *ioNumberDataPackets = pConvCbCtx->uPacketCnt - pConvCbCtx->uPacketIdx;
831 }
832
833 if (*ioNumberDataPackets)
834 {
835 Assert(pConvCbCtx->bufLstSrc.mNumberBuffers == 1); /* Only one buffer for the source supported atm. */
836
837 size_t cbOff = pConvCbCtx->uPacketIdx * pConvCbCtx->asbdSrc.mBytesPerPacket;
838
839 size_t cbAvail = RT_MIN(*ioNumberDataPackets * pConvCbCtx->asbdSrc.mBytesPerPacket,
840 pConvCbCtx->bufLstSrc.mBuffers[0].mDataByteSize - cbOff);
841
842 void *pvAvail = (uint8_t *)pConvCbCtx->bufLstSrc.mBuffers[0].mData + cbOff;
843
844 Log3Func(("cbOff=%zu, cbAvail=%zu\n", cbOff, cbAvail));
845
846 /* Set input data for the converter to use.
847 * Note: For VBR (Variable Bit Rates) or interleaved data handling we need multiple buffers here. */
848 ioData->mNumberBuffers = 1;
849
850 ioData->mBuffers[0].mNumberChannels = pConvCbCtx->bufLstSrc.mBuffers[0].mNumberChannels;
851 ioData->mBuffers[0].mDataByteSize = cbAvail;
852 ioData->mBuffers[0].mData = pvAvail;
853
854#ifdef DEBUG_DUMP_PCM_DATA
855 RTFILE fh;
856 int rc = RTFileOpen(&fh, DEBUG_DUMP_PCM_DATA_PATH "ca-converter-cb-input.pcm",
857 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
858 if (RT_SUCCESS(rc))
859 {
860 RTFileWrite(fh, pvAvail, cbAvail, NULL);
861 RTFileClose(fh);
862 }
863 else
864 AssertFailed();
865#endif
866
867 pConvCbCtx->uPacketIdx += *ioNumberDataPackets;
868 Assert(pConvCbCtx->uPacketIdx <= pConvCbCtx->uPacketCnt);
869 }
870
871 Log3Func(("%RU32 / %RU32 -> ioNumberDataPackets=%RU32\n",
872 pConvCbCtx->uPacketIdx, pConvCbCtx->uPacketCnt, *ioNumberDataPackets));
873
874 return noErr;
875}
876
877/* Callback to feed audio input buffer. */
878static DECLCALLBACK(OSStatus) coreAudioRecordingCb(void *pvUser,
879 AudioUnitRenderActionFlags *pActionFlags,
880 const AudioTimeStamp *pAudioTS,
881 UInt32 uBusID,
882 UInt32 cFrames,
883 AudioBufferList *pBufData)
884{
885 /* If nothing is pending return immediately. */
886 if (cFrames == 0)
887 return noErr;
888
889 PCOREAUDIOSTREAMIN pStreamIn = (PCOREAUDIOSTREAMIN)pvUser;
890
891 if (ASMAtomicReadU32(&pStreamIn->status) != CA_STATUS_INIT)
892 return noErr;
893
894 PCOREAUDIOCONVCBCTX pConvCbCtx = &pStreamIn->convCbCtx;
895
896 OSStatus err = noErr;
897 int rc = VINF_SUCCESS;
898
899 Assert(pConvCbCtx->bufLstSrc.mNumberBuffers == 1);
900 AudioBuffer *pSrcBuf = &pConvCbCtx->bufLstSrc.mBuffers[0];
901
902 pConvCbCtx->uPacketCnt = cFrames / pConvCbCtx->asbdSrc.mFramesPerPacket;
903 pConvCbCtx->uPacketIdx = 0;
904
905 AudioConverterReset(pStreamIn->pConverter);
906
907 Log3Func(("cFrames=%RU32 (%RU32 frames per packet) -> %RU32 packets\n",
908 cFrames, pConvCbCtx->asbdSrc.mFramesPerPacket, pConvCbCtx->uPacketCnt));
909
910 do
911 {
912 /* Are we using a converter? */
913 if (pStreamIn->pConverter)
914 {
915 pSrcBuf->mNumberChannels = pConvCbCtx->asbdSrc.mChannelsPerFrame;
916 pSrcBuf->mDataByteSize = pConvCbCtx->asbdSrc.mBytesPerFrame * cFrames;
917 pSrcBuf->mData = RTMemAllocZ(pSrcBuf->mDataByteSize);
918 if (!pSrcBuf->mData)
919 {
920 rc = VERR_NO_MEMORY;
921 break;
922 }
923
924 /* First, render the source data as usual. */
925 err = AudioUnitRender(pStreamIn->audioUnit, pActionFlags, pAudioTS, uBusID, cFrames, &pConvCbCtx->bufLstSrc);
926 if (err != noErr)
927 {
928 LogRel2(("CoreAudio: Failed rendering converted audio input data (%RI32)\n", err));
929 rc = VERR_IO_GEN_FAILURE; /** @todo Improve this. */
930 break;
931 }
932
933 Log3Func(("cbSrcBufSize=%RU32\n", pSrcBuf->mDataByteSize));
934
935#ifdef DEBUG_DUMP_PCM_DATA
936 RTFILE fh;
937 rc = RTFileOpen(&fh, DEBUG_DUMP_PCM_DATA_PATH "ca-recording-cb-src.pcm",
938 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
939 if (RT_SUCCESS(rc))
940 {
941 RTFileWrite(fh, pSrcBuf->mData, pSrcBuf->mDataByteSize, NULL);
942 RTFileClose(fh);
943 }
944 else
945 AssertFailed();
946#endif
947 AudioBufferList dstBufList;
948
949 dstBufList.mNumberBuffers = 1; /* We only use one buffer at once. */
950
951 AudioBuffer *pDstBuf = &dstBufList.mBuffers[0];
952 pDstBuf->mDataByteSize = pConvCbCtx->asbdDst.mBytesPerFrame * cFrames;
953 pDstBuf->mData = RTMemAllocZ(pDstBuf->mDataByteSize);
954 if (!pDstBuf->mData)
955 {
956 rc = VERR_NO_MEMORY;
957 break;
958 }
959
960 UInt32 cPacketsToWriteAndWritten = pConvCbCtx->uPacketCnt;
961 Assert(cPacketsToWriteAndWritten);
962
963 Log3Func(("cPacketsToWrite=%RU32\n", cPacketsToWriteAndWritten));
964
965 do
966 {
967 err = AudioConverterFillComplexBuffer(pStreamIn->pConverter,
968 coreAudioConverterCallback, pConvCbCtx /* pvData */,
969 &cPacketsToWriteAndWritten, &dstBufList, NULL);
970
971 Log3Func(("cPacketsWritten=%RU32 (%zu bytes), err=%RI32\n",
972 cPacketsToWriteAndWritten, cPacketsToWriteAndWritten * pConvCbCtx->asbdDst.mBytesPerPacket, err));
973
974 if (err != noErr)
975 {
976 LogFlowFunc(("Failed to convert audio data (%RI32:%c%c%c%c)\n", err,
977 RT_BYTE4(err), RT_BYTE3(err), RT_BYTE2(err), RT_BYTE1(err)));
978 rc = VERR_IO_GEN_FAILURE;
979 break;
980 }
981
982 if (cPacketsToWriteAndWritten == 0)
983 break;
984
985 size_t cbDst = cPacketsToWriteAndWritten * pConvCbCtx->asbdDst.mBytesPerPacket;
986
987#ifdef DEBUG_DUMP_PCM_DATA
988 rc = RTFileOpen(&fh, DEBUG_DUMP_PCM_DATA_PATH "ca-recording-cb-dst.pcm",
989 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
990 if (RT_SUCCESS(rc))
991 {
992 RTFileWrite(fh, pDstBuf->mData, cbDst, NULL);
993 RTFileClose(fh);
994 }
995 else
996 AssertFailed();
997#endif
998 size_t cbFree = RTCircBufFree(pStreamIn->pCircBuf);
999 if (cbFree < cbDst)
1000 {
1001 LogRel2(("CoreAudio: Recording is lagging behind (%zu bytes available but only %zu bytes free)\n",
1002 cbDst, cbFree));
1003 break;
1004 }
1005
1006 size_t cbDstChunk;
1007 void *puDst;
1008 RTCircBufAcquireWriteBlock(pStreamIn->pCircBuf, cbDst, (void **)&puDst, &cbDstChunk);
1009
1010 if (cbDstChunk)
1011 memcpy(puDst, pDstBuf->mData, cbDstChunk);
1012
1013 RTCircBufReleaseWriteBlock(pStreamIn->pCircBuf, cbDstChunk);
1014
1015 } while (1);
1016
1017 if (pDstBuf->mData)
1018 {
1019 RTMemFree(pDstBuf->mData);
1020 pDstBuf->mData = NULL;
1021 }
1022 }
1023 else /* No converter being used. */
1024 {
1025 AssertBreakStmt(pStreamIn->streamFormat.mChannelsPerFrame >= 1, rc = VERR_INVALID_PARAMETER);
1026 AssertBreakStmt(pStreamIn->streamFormat.mBytesPerFrame >= 1, rc = VERR_INVALID_PARAMETER);
1027
1028 AssertBreakStmt(pSrcBuf->mNumberChannels, rc = VERR_INVALID_PARAMETER);
1029
1030 pSrcBuf->mNumberChannels = pStreamIn->streamFormat.mChannelsPerFrame;
1031 pSrcBuf->mDataByteSize = pStreamIn->streamFormat.mBytesPerFrame * cFrames;
1032 pSrcBuf->mData = RTMemAlloc(pSrcBuf->mDataByteSize);
1033 if (!pSrcBuf->mData)
1034 {
1035 rc = VERR_NO_MEMORY;
1036 break;
1037 }
1038
1039 err = AudioUnitRender(pStreamIn->audioUnit, pActionFlags, pAudioTS, uBusID, cFrames, &pConvCbCtx->bufLstSrc);
1040 if (err != noErr)
1041 {
1042 LogRel2(("CoreAudio: Failed rendering non-coverted audio input data (%RI32)\n", err));
1043 rc = VERR_IO_GEN_FAILURE; /** @todo Improve this. */
1044 break;
1045 }
1046
1047 const uint32_t cbDataSize = pSrcBuf->mDataByteSize;
1048 const size_t cbBufFree = RTCircBufFree(pStreamIn->pCircBuf);
1049 size_t cbAvail = RT_MIN(cbDataSize, cbBufFree);
1050
1051 Log3Func(("cbDataSize=%RU32, cbBufFree=%zu, cbAvail=%zu\n", cbDataSize, cbBufFree, cbAvail));
1052
1053 /* Iterate as long as data is available. */
1054 uint8_t *puDst = NULL;
1055 uint32_t cbWrittenTotal = 0;
1056 while (cbAvail)
1057 {
1058 /* Try to acquire the necessary space from the ring buffer. */
1059 size_t cbToWrite = 0;
1060 RTCircBufAcquireWriteBlock(pStreamIn->pCircBuf, cbAvail, (void **)&puDst, &cbToWrite);
1061 if (!cbToWrite)
1062 break;
1063
1064 /* Copy the data from the Core Audio buffer to the ring buffer. */
1065 memcpy(puDst, (uint8_t *)pSrcBuf->mData + cbWrittenTotal, cbToWrite);
1066
1067 /* Release the ring buffer, so the main thread could start reading this data. */
1068 RTCircBufReleaseWriteBlock(pStreamIn->pCircBuf, cbToWrite);
1069
1070 cbWrittenTotal += cbToWrite;
1071
1072 Assert(cbAvail >= cbToWrite);
1073 cbAvail -= cbToWrite;
1074 }
1075
1076 Log3Func(("cbWrittenTotal=%RU32, cbLeft=%zu\n", cbWrittenTotal, cbAvail));
1077 }
1078
1079 } while (0);
1080
1081 if (pSrcBuf->mData)
1082 {
1083 RTMemFree(pSrcBuf->mData);
1084 pSrcBuf->mData = NULL;
1085 }
1086
1087 return err;
1088}
1089
1090# define CA_BREAK_STMT(stmt) \
1091 stmt; \
1092 break;
1093
1094/** @todo Eventually split up this function, as this already is huge! */
1095static int coreAudioInitIn(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream, uint32_t *pcSamples)
1096{
1097 int rc = VINF_SUCCESS;
1098
1099 PCOREAUDIOSTREAMIN pStreamIn = (PCOREAUDIOSTREAMIN)pStream;
1100 UInt32 cSamples = 0;
1101
1102 OSStatus err = noErr;
1103 AudioDeviceID deviceID = pStreamIn->deviceID;
1104
1105 UInt32 uSize = 0;
1106 if (deviceID == kAudioDeviceUnknown)
1107 {
1108 /* Fetch the default audio recording device currently in use. */
1109 AudioObjectPropertyAddress propAdr = { kAudioHardwarePropertyDefaultInputDevice,
1110 kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMaster };
1111 uSize = sizeof(deviceID);
1112 err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &propAdr, 0, NULL, &uSize, &deviceID);
1113 if (err != noErr)
1114 {
1115 LogFlowFunc(("CoreAudio: Unable to determine default recording device (%RI32)\n", err));
1116 return VERR_AUDIO_NO_FREE_INPUT_STREAMS;
1117 }
1118 }
1119
1120 if (deviceID == kAudioDeviceUnknown)
1121 {
1122 LogFlowFunc(("No default recording device found\n"));
1123 return VERR_AUDIO_NO_FREE_INPUT_STREAMS;
1124 }
1125
1126 do
1127 {
1128 ASMAtomicXchgU32(&pStreamIn->status, CA_STATUS_IN_INIT);
1129
1130 /* Assign device ID. */
1131 pStreamIn->deviceID = deviceID;
1132
1133 /*
1134 * Try to get the name of the recording device and log it. It's not fatal if it fails.
1135 */
1136 CFStringRef strTemp;
1137
1138 AudioObjectPropertyAddress propAdr = { kAudioObjectPropertyName, kAudioObjectPropertyScopeGlobal,
1139 kAudioObjectPropertyElementMaster };
1140 uSize = sizeof(CFStringRef);
1141 err = AudioObjectGetPropertyData(pStreamIn->deviceID, &propAdr, 0, NULL, &uSize, &strTemp);
1142 if (err == noErr)
1143 {
1144 char *pszDevName = NULL;
1145 err = coreAudioCFStringToCString(strTemp, &pszDevName);
1146 if (err == noErr)
1147 {
1148 CFRelease(strTemp);
1149
1150 /* Get the device' UUID. */
1151 propAdr.mSelector = kAudioDevicePropertyDeviceUID;
1152 err = AudioObjectGetPropertyData(pStreamIn->deviceID, &propAdr, 0, NULL, &uSize, &strTemp);
1153 if (err == noErr)
1154 {
1155 char *pszUID = NULL;
1156
1157 err = coreAudioCFStringToCString(strTemp, &pszUID);
1158 if (err == noErr)
1159 {
1160 CFRelease(strTemp);
1161 LogRel(("CoreAudio: Using recording device: %s (UID: %s)\n", pszDevName, pszUID));
1162
1163 RTMemFree(pszUID);
1164 }
1165 }
1166
1167 RTMemFree(pszDevName);
1168 }
1169 }
1170 else
1171 {
1172 /* This is not fatal, can happen for some Macs. */
1173 LogRel2(("CoreAudio: Unable to determine recording device name (%RI32)\n", err));
1174 }
1175
1176 /* Get the default frames buffer size, so that we can setup our internal buffers. */
1177 UInt32 cFrames;
1178 uSize = sizeof(cFrames);
1179 propAdr.mSelector = kAudioDevicePropertyBufferFrameSize;
1180 propAdr.mScope = kAudioDevicePropertyScopeInput;
1181 err = AudioObjectGetPropertyData(pStreamIn->deviceID, &propAdr, 0, NULL, &uSize, &cFrames);
1182 if (err != noErr)
1183 {
1184 /* Can happen if no recording device is available by default. Happens on some Macs,
1185 * so don't log this by default to not scare people. */
1186 LogRel2(("CoreAudio: Failed to determine frame buffer size of the audio recording device (%RI32)\n", err));
1187 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1188 }
1189
1190 /* Set the frame buffer size and honor any minimum/maximum restrictions on the device. */
1191 err = coreAudioSetFrameBufferSize(pStreamIn->deviceID, true /* fInput */, cFrames, &cFrames);
1192 if (err != noErr)
1193 {
1194 LogRel(("CoreAudio: Failed to set frame buffer size for the audio recording device (%RI32)\n", err));
1195 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1196 }
1197
1198 LogFlowFunc(("cFrames=%RU32\n", cFrames));
1199
1200 ComponentDescription cd;
1201 RT_ZERO(cd);
1202 cd.componentType = kAudioUnitType_Output;
1203 cd.componentSubType = kAudioUnitSubType_HALOutput;
1204 cd.componentManufacturer = kAudioUnitManufacturer_Apple;
1205
1206 /* Try to find the default HAL output component. */
1207 Component cp = FindNextComponent(NULL, &cd);
1208 if (cp == 0)
1209 {
1210 LogRel(("CoreAudio: Failed to find HAL output component\n")); /** @todo Return error value? */
1211 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1212 }
1213
1214 /* Open the default HAL output component. */
1215 err = OpenAComponent(cp, &pStreamIn->audioUnit);
1216 if (err != noErr)
1217 {
1218 LogRel(("CoreAudio: Failed to open output component (%RI32)\n", err));
1219 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1220 }
1221
1222 /* Switch the I/O mode for input to on. */
1223 UInt32 uFlag = 1;
1224 err = AudioUnitSetProperty(pStreamIn->audioUnit, kAudioOutputUnitProperty_EnableIO, kAudioUnitScope_Input,
1225 1, &uFlag, sizeof(uFlag));
1226 if (err != noErr)
1227 {
1228 LogRel(("CoreAudio: Failed to disable input I/O mode for input stream (%RI32)\n", err));
1229 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1230 }
1231
1232 /* Switch the I/O mode for output to off. This is important, as this is a pure input stream. */
1233 uFlag = 0;
1234 err = AudioUnitSetProperty(pStreamIn->audioUnit, kAudioOutputUnitProperty_EnableIO, kAudioUnitScope_Output,
1235 0, &uFlag, sizeof(uFlag));
1236 if (err != noErr)
1237 {
1238 LogRel(("CoreAudio: Failed to disable output I/O mode for input stream (%RI32)\n", err));
1239 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1240 }
1241
1242 /* Set the default audio recording device as the device for the new AudioUnit. */
1243 err = AudioUnitSetProperty(pStreamIn->audioUnit, kAudioOutputUnitProperty_CurrentDevice, kAudioUnitScope_Global,
1244 0, &pStreamIn->deviceID, sizeof(pStreamIn->deviceID));
1245 if (err != noErr)
1246 {
1247 LogRel(("CoreAudio: Failed to set current device (%RI32)\n", err));
1248 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1249 }
1250
1251 /*
1252 * CoreAudio will inform us on a second thread for new incoming audio data.
1253 * Therefore register a callback function which will process the new data.
1254 */
1255 AURenderCallbackStruct cb;
1256 RT_ZERO(cb);
1257 cb.inputProc = coreAudioRecordingCb;
1258 cb.inputProcRefCon = pStreamIn;
1259
1260 err = AudioUnitSetProperty(pStreamIn->audioUnit, kAudioOutputUnitProperty_SetInputCallback, kAudioUnitScope_Global,
1261 0, &cb, sizeof(cb));
1262 if (err != noErr)
1263 {
1264 LogRel(("CoreAudio: Failed to register input callback (%RI32)\n", err));
1265 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1266 }
1267
1268 /* Fetch the current stream format of the device. */
1269 RT_ZERO(pStreamIn->deviceFormat);
1270 uSize = sizeof(pStreamIn->deviceFormat);
1271 err = AudioUnitGetProperty(pStreamIn->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input,
1272 1, &pStreamIn->deviceFormat, &uSize);
1273 if (err != noErr)
1274 {
1275 LogRel(("CoreAudio: Failed to get device format (%RI32)\n", err));
1276 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1277 }
1278
1279 /* Create an AudioStreamBasicDescription based on our required audio settings. */
1280 RT_ZERO(pStreamIn->streamFormat);
1281 coreAudioPCMPropsToASBD(&pStreamIn->Stream.Props, &pStreamIn->streamFormat);
1282
1283 coreAudioPrintASBD("Recording device", &pStreamIn->deviceFormat);
1284 coreAudioPrintASBD("Recording stream", &pStreamIn->streamFormat);
1285
1286 /* If the frequency of the device is different from the requested one we
1287 * need a converter. The same count if the number of channels is different. */
1288 if ( pStreamIn->deviceFormat.mSampleRate != pStreamIn->streamFormat.mSampleRate
1289 || pStreamIn->deviceFormat.mChannelsPerFrame != pStreamIn->streamFormat.mChannelsPerFrame)
1290 {
1291 LogRel2(("CoreAudio: Input converter is active\n"));
1292
1293 err = AudioConverterNew(&pStreamIn->deviceFormat, &pStreamIn->streamFormat, &pStreamIn->pConverter);
1294 if (RT_UNLIKELY(err != noErr))
1295 {
1296 LogRel(("CoreAudio: Failed to create the audio converter (%RI32)\n", err));
1297 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1298 }
1299
1300 if ( pStreamIn->deviceFormat.mChannelsPerFrame == 1 /* Mono */
1301 && pStreamIn->streamFormat.mChannelsPerFrame == 2 /* Stereo */)
1302 {
1303 LogRel2(("CoreAudio: Mono to stereo conversion active\n"));
1304
1305 /*
1306 * If the channel count is different we have to tell this the converter
1307 * and supply a channel mapping. For now we only support mapping
1308 * from mono to stereo. For all other cases the core audio defaults
1309 * are used, which means dropping additional channels in most
1310 * cases.
1311 */
1312 const SInt32 channelMap[2] = {0, 0}; /* Channel map for mono -> stereo. */
1313
1314 err = AudioConverterSetProperty(pStreamIn->pConverter, kAudioConverterChannelMap, sizeof(channelMap), channelMap);
1315 if (err != noErr)
1316 {
1317 LogRel(("CoreAudio: Failed to set channel mapping (mono -> stereo) for the audio input converter (%RI32)\n", err));
1318 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1319 }
1320 }
1321
1322 /* Set sample rate converter quality to maximum. */
1323 uFlag = kAudioConverterQuality_Max;
1324 err = AudioConverterSetProperty(pStreamIn->pConverter, kAudioConverterSampleRateConverterQuality,
1325 sizeof(uFlag), &uFlag);
1326 if (err != noErr)
1327 LogRel2(("CoreAudio: Failed to set input audio converter quality to the maximum (%RI32)\n", err));
1328
1329 uSize = sizeof(UInt32);
1330 UInt32 maxOutputSize;
1331 err = AudioConverterGetProperty(pStreamIn->pConverter, kAudioConverterPropertyMaximumOutputPacketSize,
1332 &uSize, &maxOutputSize);
1333 if (RT_UNLIKELY(err != noErr))
1334 {
1335 LogRel(("CoreAudio: Failed to retrieve converter's maximum output size (%RI32)\n", err));
1336 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1337 }
1338
1339 LogFunc(("Maximum converter packet output size is: %RI32\n", maxOutputSize));
1340
1341 /* Set the input (source) format, that is, the format the device is recording data with. */
1342 err = AudioUnitSetProperty(pStreamIn->audioUnit,
1343 kAudioUnitProperty_StreamFormat,
1344 kAudioUnitScope_Input,
1345 1,
1346 &pStreamIn->deviceFormat,
1347 sizeof(pStreamIn->deviceFormat));
1348 if (RT_UNLIKELY(err != noErr))
1349 {
1350 LogRel(("CoreAudio: Failed to set device input format (%RI32)\n", err));
1351 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1352 }
1353
1354 /* Set the output (target) format, that is, the format we created the input stream with. */
1355 err = AudioUnitSetProperty(pStreamIn->audioUnit,
1356 kAudioUnitProperty_StreamFormat,
1357 kAudioUnitScope_Output,
1358 1,
1359 &pStreamIn->deviceFormat,
1360 sizeof(pStreamIn->deviceFormat));
1361 if (RT_UNLIKELY(err != noErr))
1362 {
1363 LogRel(("CoreAudio: Failed to set stream output format (%RI32)\n", err));
1364 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1365 }
1366 }
1367 else
1368 {
1369 /* Set the new output format description for the input stream. */
1370 err = AudioUnitSetProperty(pStreamIn->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Output,
1371 1, &pStreamIn->streamFormat, sizeof(pStreamIn->streamFormat));
1372 if (err != noErr)
1373 {
1374 LogRel(("CoreAudio: Failed to set output format for input stream (%RI32)\n", err));
1375 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1376 }
1377 }
1378
1379 /*
1380 * Also set the frame buffer size off the device on our AudioUnit. This
1381 * should make sure that the frames count which we receive in the render
1382 * thread is as we like.
1383 */
1384 err = AudioUnitSetProperty(pStreamIn->audioUnit, kAudioUnitProperty_MaximumFramesPerSlice, kAudioUnitScope_Global,
1385 1, &cFrames, sizeof(cFrames));
1386 if (err != noErr)
1387 {
1388 LogRel(("CoreAudio: Failed to set maximum frame buffer size for input stream (%RI32)\n", err));
1389 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1390 }
1391
1392 /* Finally initialize the new AudioUnit. */
1393 err = AudioUnitInitialize(pStreamIn->audioUnit);
1394 if (err != noErr)
1395 {
1396 LogRel(("CoreAudio: Failed to initialize audio unit for input stream (%RI32)\n", err));
1397 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1398 }
1399
1400 uSize = sizeof(pStreamIn->deviceFormat);
1401 err = AudioUnitGetProperty(pStreamIn->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Output,
1402 1, &pStreamIn->deviceFormat, &uSize);
1403 if (err != noErr)
1404 {
1405 LogRel(("CoreAudio: Failed to get recording device format (%RI32)\n", err));
1406 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1407 }
1408
1409 /*
1410 * There are buggy devices (e.g. my Bluetooth headset) which doesn't honor
1411 * the frame buffer size set in the previous calls. So finally get the
1412 * frame buffer size after the AudioUnit was initialized.
1413 */
1414 uSize = sizeof(cFrames);
1415 err = AudioUnitGetProperty(pStreamIn->audioUnit, kAudioUnitProperty_MaximumFramesPerSlice, kAudioUnitScope_Global,
1416 0, &cFrames, &uSize);
1417 if (err != noErr)
1418 {
1419 LogRel(("CoreAudio: Failed to get maximum frame buffer size from input audio device (%RI32)\n", err));
1420 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1421 }
1422
1423 /* Destroy any former internal ring buffer. */
1424 if (pStreamIn->pCircBuf)
1425 {
1426 RTCircBufDestroy(pStreamIn->pCircBuf);
1427 pStreamIn->pCircBuf = NULL;
1428 }
1429
1430 coreAudioUninitConvCbCtx(&pStreamIn->convCbCtx);
1431
1432 /* Calculate the ratio between the device and the stream sample rate. */
1433 pStreamIn->sampleRatio = pStreamIn->streamFormat.mSampleRate / pStreamIn->deviceFormat.mSampleRate;
1434
1435 /*
1436 * Make sure that the ring buffer is big enough to hold the recording
1437 * data. Compare the maximum frames per slice value with the frames
1438 * necessary when using the converter where the sample rate could differ.
1439 * The result is always multiplied by the channels per frame to get the
1440 * samples count.
1441 */
1442 cSamples = RT_MAX(cFrames,
1443 (cFrames * pStreamIn->deviceFormat.mBytesPerFrame * pStreamIn->sampleRatio)
1444 / pStreamIn->streamFormat.mBytesPerFrame)
1445 * pStreamIn->streamFormat.mChannelsPerFrame;
1446 if (!cSamples)
1447 {
1448 LogRel(("CoreAudio: Failed to determine samples buffer count input stream\n"));
1449 CA_BREAK_STMT(rc = VERR_INVALID_PARAMETER);
1450 }
1451
1452 if (RT_SUCCESS(rc))
1453 rc = RTCircBufCreate(&pStreamIn->pCircBuf, cSamples << pStreamIn->Stream.Props.cShift);
1454
1455 /* Init the converter callback context. */
1456 if (RT_SUCCESS(rc))
1457 rc = coreAudioInitConvCbCtx(&pStreamIn->convCbCtx,
1458 &pStreamIn->deviceFormat /* Source */, &pStreamIn->streamFormat /* Dest */);
1459
1460 if (RT_SUCCESS(rc))
1461 {
1462#ifdef DEBUG
1463 propAdr.mSelector = kAudioDeviceProcessorOverload;
1464 propAdr.mScope = kAudioUnitScope_Global;
1465 err = AudioObjectAddPropertyListener(pStreamIn->deviceID, &propAdr,
1466 coreAudioRecordingAudioDevicePropertyChanged, (void *)pStreamIn);
1467 if (RT_UNLIKELY(err != noErr))
1468 LogRel2(("CoreAudio: Failed to add the processor overload listener for input stream (%RI32)\n", err));
1469#endif /* DEBUG */
1470 propAdr.mSelector = kAudioDevicePropertyNominalSampleRate;
1471 propAdr.mScope = kAudioUnitScope_Global;
1472 err = AudioObjectAddPropertyListener(pStreamIn->deviceID, &propAdr,
1473 coreAudioRecordingAudioDevicePropertyChanged, (void *)pStreamIn);
1474 /* Not fatal. */
1475 if (RT_UNLIKELY(err != noErr))
1476 LogRel2(("CoreAudio: Failed to register sample rate changed listener for input stream (%RI32)\n", err));
1477 }
1478
1479 } while (0);
1480
1481 if (RT_SUCCESS(rc))
1482 {
1483 ASMAtomicXchgU32(&pStreamIn->status, CA_STATUS_INIT);
1484
1485 LogFunc(("cSamples=%RU32\n", cSamples));
1486
1487 if (pcSamples)
1488 *pcSamples = cSamples;
1489 }
1490 else
1491 {
1492 AudioUnitUninitialize(pStreamIn->audioUnit);
1493
1494 if (pStreamIn->pCircBuf)
1495 {
1496 RTCircBufDestroy(pStreamIn->pCircBuf);
1497 pStreamIn->pCircBuf = NULL;
1498 }
1499
1500 coreAudioUninitConvCbCtx(&pStreamIn->convCbCtx);
1501
1502 ASMAtomicXchgU32(&pStreamIn->status, CA_STATUS_UNINIT);
1503 }
1504
1505 LogFunc(("rc=%Rrc\n", rc));
1506 return rc;
1507}
1508
1509/** @todo Eventually split up this function, as this already is huge! */
1510static int coreAudioInitOut(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream, uint32_t *pcSamples)
1511{
1512 int rc = VINF_SUCCESS;
1513
1514 PCOREAUDIOSTREAMOUT pStreamOut = (PCOREAUDIOSTREAMOUT)pStream;
1515 UInt32 cSamples = 0;
1516
1517 OSStatus err = noErr;
1518 AudioDeviceID deviceID = pStreamOut->deviceID;
1519
1520 UInt32 uSize = 0;
1521 if (pStreamOut->deviceID == kAudioDeviceUnknown)
1522 {
1523 /* Fetch the default audio recording device currently in use. */
1524 AudioObjectPropertyAddress propAdr = { kAudioHardwarePropertyDefaultOutputDevice,
1525 kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMaster };
1526 uSize = sizeof(pStreamOut->deviceID);
1527 err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &propAdr, 0, NULL, &uSize, &pStreamOut->deviceID);
1528 if (err != noErr)
1529 {
1530 LogRel(("CoreAudio: Unable to determine default playback device (%RI32)\n", err));
1531 return VERR_NOT_FOUND;
1532 }
1533 }
1534
1535 if (deviceID == kAudioDeviceUnknown)
1536 {
1537 LogFlowFunc(("No default playback device found\n"));
1538 return VERR_NOT_FOUND;
1539 }
1540
1541 do
1542 {
1543 ASMAtomicXchgU32(&pStreamOut->status, CA_STATUS_IN_INIT);
1544
1545 /* Assign device ID. */
1546 pStreamOut->deviceID = deviceID;
1547
1548 /*
1549 * Try to get the name of the playback device and log it. It's not fatal if it fails.
1550 */
1551 CFStringRef strTemp;
1552
1553 AudioObjectPropertyAddress propAdr = { kAudioObjectPropertyName, kAudioObjectPropertyScopeGlobal,
1554 kAudioObjectPropertyElementMaster };
1555 uSize = sizeof(CFStringRef);
1556 err = AudioObjectGetPropertyData(pStreamOut->deviceID, &propAdr, 0, NULL, &uSize, &strTemp);
1557 if (err == noErr)
1558 {
1559 char *pszDevName = NULL;
1560 err = coreAudioCFStringToCString(strTemp, &pszDevName);
1561 if (err == noErr)
1562 {
1563 CFRelease(strTemp);
1564
1565 /* Get the device' UUID. */
1566 propAdr.mSelector = kAudioDevicePropertyDeviceUID;
1567 err = AudioObjectGetPropertyData(pStreamOut->deviceID, &propAdr, 0, NULL, &uSize, &strTemp);
1568 if (err == noErr)
1569 {
1570 char *pszUID = NULL;
1571 err = coreAudioCFStringToCString(strTemp, &pszUID);
1572 if (err == noErr)
1573 {
1574 CFRelease(strTemp);
1575 LogRel(("CoreAudio: Using playback device: %s (UID: %s)\n", pszDevName, pszUID));
1576
1577 RTMemFree(pszUID);
1578 }
1579 }
1580
1581 RTMemFree(pszDevName);
1582 }
1583 }
1584 else
1585 LogRel(("CoreAudio: Unable to determine playback device name (%RI32)\n", err));
1586
1587 /* Get the default frames buffer size, so that we can setup our internal buffers. */
1588 UInt32 cFrames;
1589 uSize = sizeof(cFrames);
1590 propAdr.mSelector = kAudioDevicePropertyBufferFrameSize;
1591 propAdr.mScope = kAudioDevicePropertyScopeInput;
1592 err = AudioObjectGetPropertyData(pStreamOut->deviceID, &propAdr, 0, NULL, &uSize, &cFrames);
1593 if (err != noErr)
1594 {
1595 LogRel(("CoreAudio: Failed to determine frame buffer size of the audio playback device (%RI32)\n", err));
1596 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1597 }
1598
1599 /* Set the frame buffer size and honor any minimum/maximum restrictions on the device. */
1600 err = coreAudioSetFrameBufferSize(pStreamOut->deviceID, false /* fInput */, cFrames, &cFrames);
1601 if (err != noErr)
1602 {
1603 LogRel(("CoreAudio: Failed to set frame buffer size for the audio playback device (%RI32)\n", err));
1604 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1605 }
1606
1607 ComponentDescription cd;
1608 RT_ZERO(cd);
1609 cd.componentType = kAudioUnitType_Output;
1610 cd.componentSubType = kAudioUnitSubType_HALOutput;
1611 cd.componentManufacturer = kAudioUnitManufacturer_Apple;
1612
1613 /* Try to find the default HAL output component. */
1614 Component cp = FindNextComponent(NULL, &cd);
1615 if (cp == 0)
1616 {
1617 LogRel(("CoreAudio: Failed to find HAL output component\n")); /** @todo Return error value? */
1618 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1619 }
1620
1621 /* Open the default HAL output component. */
1622 err = OpenAComponent(cp, &pStreamOut->audioUnit);
1623 if (err != noErr)
1624 {
1625 LogRel(("CoreAudio: Failed to open output component (%RI32)\n", err));
1626 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1627 }
1628
1629 /* Switch the I/O mode for output to on. */
1630 UInt32 uFlag = 1;
1631 err = AudioUnitSetProperty(pStreamOut->audioUnit, kAudioOutputUnitProperty_EnableIO, kAudioUnitScope_Output,
1632 0, &uFlag, sizeof(uFlag));
1633 if (err != noErr)
1634 {
1635 LogRel(("CoreAudio: Failed to disable I/O mode for output stream (%RI32)\n", err));
1636 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1637 }
1638
1639 /* Set the default audio playback device as the device for the new AudioUnit. */
1640 err = AudioUnitSetProperty(pStreamOut->audioUnit, kAudioOutputUnitProperty_CurrentDevice, kAudioUnitScope_Global,
1641 0, &pStreamOut->deviceID, sizeof(pStreamOut->deviceID));
1642 if (err != noErr)
1643 {
1644 LogRel(("CoreAudio: Failed to set current device for output stream (%RI32)\n", err));
1645 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1646 }
1647
1648 /*
1649 * CoreAudio will inform us on a second thread for new incoming audio data.
1650 * Therefor register a callback function which will process the new data.
1651 */
1652 AURenderCallbackStruct cb;
1653 RT_ZERO(cb);
1654 cb.inputProc = coreAudioPlaybackCb; /* pvUser */
1655 cb.inputProcRefCon = pStreamOut;
1656
1657 err = AudioUnitSetProperty(pStreamOut->audioUnit, kAudioUnitProperty_SetRenderCallback, kAudioUnitScope_Input,
1658 0, &cb, sizeof(cb));
1659 if (err != noErr)
1660 {
1661 LogRel(("CoreAudio: Failed to register output callback (%RI32)\n", err));
1662 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1663 }
1664
1665 /* Fetch the current stream format of the device. */
1666 uSize = sizeof(pStreamOut->deviceFormat);
1667 err = AudioUnitGetProperty(pStreamOut->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input,
1668 0, &pStreamOut->deviceFormat, &uSize);
1669 if (err != noErr)
1670 {
1671 LogRel(("CoreAudio: Failed to get device format (%RI32)\n", err));
1672 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1673 }
1674
1675 /* Create an AudioStreamBasicDescription based on our required audio settings. */
1676 coreAudioPCMPropsToASBD(&pStreamOut->Stream.Props, &pStreamOut->streamFormat);
1677
1678 coreAudioPrintASBD("Playback device", &pStreamOut->deviceFormat);
1679 coreAudioPrintASBD("Playback format", &pStreamOut->streamFormat);
1680
1681 /* Set the new output format description for the stream. */
1682 err = AudioUnitSetProperty(pStreamOut->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input,
1683 0, &pStreamOut->streamFormat, sizeof(pStreamOut->streamFormat));
1684 if (err != noErr)
1685 {
1686 LogRel(("CoreAudio: Failed to set stream format for output stream (%RI32)\n", err));
1687 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1688 }
1689
1690 uSize = sizeof(pStreamOut->deviceFormat);
1691 err = AudioUnitGetProperty(pStreamOut->audioUnit, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input,
1692 0, &pStreamOut->deviceFormat, &uSize);
1693 if (err != noErr)
1694 {
1695 LogRel(("CoreAudio: Failed to retrieve device format for output stream (%RI32)\n", err));
1696 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1697 }
1698
1699 /*
1700 * Also set the frame buffer size off the device on our AudioUnit. This
1701 * should make sure that the frames count which we receive in the render
1702 * thread is as we like.
1703 */
1704 err = AudioUnitSetProperty(pStreamOut->audioUnit, kAudioUnitProperty_MaximumFramesPerSlice, kAudioUnitScope_Global,
1705 0, &cFrames, sizeof(cFrames));
1706 if (err != noErr)
1707 {
1708 LogRel(("CoreAudio: Failed to set maximum frame buffer size for output AudioUnit (%RI32)\n", err));
1709 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1710 }
1711
1712 /* Finally initialize the new AudioUnit. */
1713 err = AudioUnitInitialize(pStreamOut->audioUnit);
1714 if (err != noErr)
1715 {
1716 LogRel(("CoreAudio: Failed to initialize the output audio device (%RI32)\n", err));
1717 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1718 }
1719
1720 /*
1721 * There are buggy devices (e.g. my Bluetooth headset) which doesn't honor
1722 * the frame buffer size set in the previous calls. So finally get the
1723 * frame buffer size after the AudioUnit was initialized.
1724 */
1725 uSize = sizeof(cFrames);
1726 err = AudioUnitGetProperty(pStreamOut->audioUnit, kAudioUnitProperty_MaximumFramesPerSlice, kAudioUnitScope_Global,
1727 0, &cFrames, &uSize);
1728 if (err != noErr)
1729 {
1730 LogRel(("CoreAudio: Failed to get maximum frame buffer size from output audio device (%RI32)\n", err));
1731
1732 AudioUnitUninitialize(pStreamOut->audioUnit);
1733 CA_BREAK_STMT(rc = VERR_AUDIO_BACKEND_INIT_FAILED);
1734 }
1735
1736 /*
1737 * Make sure that the ring buffer is big enough to hold the recording
1738 * data. Compare the maximum frames per slice value with the frames
1739 * necessary when using the converter where the sample rate could differ.
1740 * The result is always multiplied by the channels per frame to get the
1741 * samples count.
1742 */
1743 cSamples = cFrames * pStreamOut->streamFormat.mChannelsPerFrame;
1744 if (!cSamples)
1745 {
1746 LogRel(("CoreAudio: Failed to determine samples buffer count output stream\n"));
1747 CA_BREAK_STMT(rc = VERR_INVALID_PARAMETER);
1748 }
1749
1750 /* Destroy any former internal ring buffer. */
1751 if (pStreamOut->pCircBuf)
1752 {
1753 RTCircBufDestroy(pStreamOut->pCircBuf);
1754 pStreamOut->pCircBuf = NULL;
1755 }
1756
1757 /* Create the internal ring buffer. */
1758 rc = RTCircBufCreate(&pStreamOut->pCircBuf, cSamples << pStream->Props.cShift);
1759 if (RT_SUCCESS(rc))
1760 {
1761 /*
1762 * Register callbacks.
1763 */
1764#ifdef DEBUG
1765 propAdr.mSelector = kAudioDeviceProcessorOverload;
1766 propAdr.mScope = kAudioUnitScope_Global;
1767 err = AudioObjectAddPropertyListener(pStreamOut->deviceID, &propAdr,
1768 coreAudioPlaybackAudioDevicePropertyChanged, (void *)pStreamOut);
1769 if (err != noErr)
1770 LogRel(("CoreAudio: Failed to register processor overload listener for output stream (%RI32)\n", err));
1771#endif /* DEBUG */
1772
1773 propAdr.mSelector = kAudioDevicePropertyNominalSampleRate;
1774 propAdr.mScope = kAudioUnitScope_Global;
1775 err = AudioObjectAddPropertyListener(pStreamOut->deviceID, &propAdr,
1776 coreAudioPlaybackAudioDevicePropertyChanged, (void *)pStreamOut);
1777 /* Not fatal. */
1778 if (err != noErr)
1779 LogRel(("CoreAudio: Failed to register sample rate changed listener for output stream (%RI32)\n", err));
1780 }
1781
1782 } while (0);
1783
1784 if (RT_SUCCESS(rc))
1785 {
1786 ASMAtomicXchgU32(&pStreamOut->status, CA_STATUS_INIT);
1787
1788 LogFunc(("cSamples=%RU32\n", cSamples));
1789
1790 if (pcSamples)
1791 *pcSamples = cSamples;
1792 }
1793 else
1794 {
1795 AudioUnitUninitialize(pStreamOut->audioUnit);
1796
1797 if (pStreamOut->pCircBuf)
1798 {
1799 RTCircBufDestroy(pStreamOut->pCircBuf);
1800 pStreamOut->pCircBuf = NULL;
1801 }
1802
1803 ASMAtomicXchgU32(&pStreamOut->status, CA_STATUS_UNINIT);
1804 }
1805
1806 LogFunc(("cSamples=%RU32, rc=%Rrc\n", cSamples, rc));
1807 return rc;
1808}
1809
1810/* Callback for getting notified when some of the properties of an audio device has changed. */
1811static DECLCALLBACK(OSStatus) coreAudioPlaybackAudioDevicePropertyChanged(AudioObjectID propertyID,
1812 UInt32 nAddresses,
1813 const AudioObjectPropertyAddress properties[],
1814 void *pvUser)
1815{
1816 switch (propertyID)
1817 {
1818#ifdef DEBUG
1819#endif /* DEBUG */
1820 default:
1821 break;
1822 }
1823
1824 return noErr;
1825}
1826
1827/* Callback to feed audio output buffer. */
1828static DECLCALLBACK(OSStatus) coreAudioPlaybackCb(void *pvUser,
1829 AudioUnitRenderActionFlags *pActionFlags,
1830 const AudioTimeStamp *pAudioTS,
1831 UInt32 uBusID,
1832 UInt32 cFrames,
1833 AudioBufferList *pBufData)
1834{
1835 PCOREAUDIOSTREAMOUT pStreamOut = (PCOREAUDIOSTREAMOUT)pvUser;
1836 PPDMAUDIOSTREAM pStream = &pStreamOut->Stream;
1837
1838 if (ASMAtomicReadU32(&pStreamOut->status) != CA_STATUS_INIT)
1839 {
1840 pBufData->mBuffers[0].mDataByteSize = 0;
1841 return noErr;
1842 }
1843
1844 /* How much space is used in the ring buffer? */
1845 size_t cbToRead = RT_MIN(RTCircBufUsed(pStreamOut->pCircBuf), pBufData->mBuffers[0].mDataByteSize);
1846 if (!cbToRead)
1847 {
1848 pBufData->mBuffers[0].mDataByteSize = 0;
1849 return noErr;
1850 }
1851
1852 uint8_t *pbSrc = NULL;
1853 size_t cbRead = 0;
1854
1855 size_t cbLeft = cbToRead;
1856 while (cbLeft)
1857 {
1858 /* Try to acquire the necessary block from the ring buffer. */
1859 RTCircBufAcquireReadBlock(pStreamOut->pCircBuf, cbLeft, (void **)&pbSrc, &cbToRead);
1860
1861 /* Break if nothing is used anymore. */
1862 if (!cbToRead)
1863 break;
1864
1865 /* Copy the data from our ring buffer to the core audio buffer. */
1866 memcpy((uint8_t *)pBufData->mBuffers[0].mData + cbRead, pbSrc, cbToRead);
1867
1868 /* Release the read buffer, so it could be used for new data. */
1869 RTCircBufReleaseReadBlock(pStreamOut->pCircBuf, cbToRead);
1870
1871 /* Move offset. */
1872 cbRead += cbToRead;
1873
1874 /* Check if we're lagging behind. */
1875 if (cbRead > pBufData->mBuffers[0].mDataByteSize)
1876 {
1877 LogRel2(("CoreAudio: Host output lagging behind, expect stuttering guest audio output\n"));
1878 cbRead = pBufData->mBuffers[0].mDataByteSize;
1879 break;
1880 }
1881
1882 Assert(cbToRead <= cbLeft);
1883 cbLeft -= cbToRead;
1884 }
1885
1886 /* Write the bytes to the core audio buffer which were really written. */
1887 Assert(pBufData->mBuffers[0].mDataByteSize >= cbRead);
1888 pBufData->mBuffers[0].mDataByteSize = cbRead;
1889
1890 Log3Func(("Read %zu / %zu bytes\n", cbRead, cbToRead));
1891
1892 return noErr;
1893}
1894
1895static DECLCALLBACK(int) drvHostCoreAudioInit(PPDMIHOSTAUDIO pInterface)
1896{
1897 NOREF(pInterface);
1898
1899 LogFlowFuncEnter();
1900
1901 return VINF_SUCCESS;
1902}
1903
1904static DECLCALLBACK(int) drvHostCoreAudioStreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
1905 uint32_t *pcSamplesCaptured)
1906{
1907 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
1908 PDRVHOSTCOREAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTCOREAUDIO);
1909
1910 PCOREAUDIOSTREAMIN pStreamIn = (PCOREAUDIOSTREAMIN)pStream;
1911
1912 size_t csReads = 0;
1913 char *pcSrc;
1914 PPDMAUDIOSAMPLE psDst;
1915
1916 if (ASMAtomicReadU32(&pStreamIn->status) != CA_STATUS_INIT)
1917 {
1918 if (pcSamplesCaptured)
1919 *pcSamplesCaptured = 0;
1920 return VINF_SUCCESS;
1921 }
1922
1923 int rc = VINF_SUCCESS;
1924 uint32_t cbWrittenTotal = 0;
1925
1926 do
1927 {
1928 size_t cbBuf = AudioMixBufSizeBytes(&pStream->MixBuf);
1929 size_t cbToWrite = RT_MIN(cbBuf, RTCircBufUsed(pStreamIn->pCircBuf));
1930
1931 uint32_t cWritten, cbWritten;
1932 uint8_t *puBuf;
1933 size_t cbToRead;
1934
1935 Log3Func(("cbBuf=%zu, cbToWrite=%zu/%zu\n", cbBuf, cbToWrite, RTCircBufSize(pStreamIn->pCircBuf)));
1936
1937 while (cbToWrite)
1938 {
1939 /* Try to acquire the necessary block from the ring buffer. */
1940 RTCircBufAcquireReadBlock(pStreamIn->pCircBuf, cbToWrite, (void **)&puBuf, &cbToRead);
1941 if (!cbToRead)
1942 {
1943 RTCircBufReleaseReadBlock(pStreamIn->pCircBuf, cbToRead);
1944 break;
1945 }
1946
1947#ifdef DEBUG_DUMP_PCM_DATA
1948 RTFILE fh;
1949 rc = RTFileOpen(&fh, DEBUG_DUMP_PCM_DATA_PATH "ca-capture.pcm",
1950 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
1951 if (RT_SUCCESS(rc))
1952 {
1953 RTFileWrite(fh, puBuf + cbWrittenTotal, cbToRead, NULL);
1954 RTFileClose(fh);
1955 }
1956 else
1957 AssertFailed();
1958#endif
1959 rc = AudioMixBufWriteCirc(&pStream->MixBuf, puBuf, cbToRead, &cWritten);
1960
1961 /* Release the read buffer, so it could be used for new data. */
1962 RTCircBufReleaseReadBlock(pStreamIn->pCircBuf, cbToRead);
1963
1964 if ( RT_FAILURE(rc)
1965 || !cWritten)
1966 {
1967 RTCircBufReleaseReadBlock(pStreamIn->pCircBuf, cbToRead);
1968 break;
1969 }
1970
1971 cbWritten = AUDIOMIXBUF_S2B(&pStream->MixBuf, cWritten);
1972
1973 /* Release the read buffer, so it could be used for new data. */
1974 RTCircBufReleaseReadBlock(pStreamIn->pCircBuf, cbWritten);
1975
1976 Assert(cbToWrite >= cbWritten);
1977 cbToWrite -= cbWritten;
1978 cbWrittenTotal += cbWritten;
1979 }
1980
1981 Log3Func(("cbToWrite=%zu, cbToRead=%zu, cbWrittenTotal=%RU32, rc=%Rrc\n", cbToWrite, cbToRead, cbWrittenTotal, rc));
1982 }
1983 while (0);
1984
1985 if (RT_SUCCESS(rc))
1986 {
1987 uint32_t cCaptured = 0;
1988 uint32_t cWrittenTotal = AUDIOMIXBUF_B2S(&pStream->MixBuf, cbWrittenTotal);
1989 if (cWrittenTotal)
1990 rc = AudioMixBufMixToParent(&pStream->MixBuf, cWrittenTotal, &cCaptured);
1991
1992 Log3Func(("cWrittenTotal=%RU32 (%RU32 bytes), cCaptured=%RU32, rc=%Rrc\n", cWrittenTotal, cbWrittenTotal, cCaptured, rc));
1993
1994 if (cCaptured)
1995 LogFlowFunc(("%RU32 samples captured\n", cCaptured));
1996
1997 if (pcSamplesCaptured)
1998 *pcSamplesCaptured = cCaptured;
1999 }
2000
2001 if (RT_FAILURE(rc))
2002 LogFunc(("Failed with rc=%Rrc\n", rc));
2003
2004 return rc;
2005}
2006
2007static DECLCALLBACK(int) drvHostCoreAudioStreamPlay(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
2008 uint32_t *pcSamplesPlayed)
2009{
2010 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
2011 PDRVHOSTCOREAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTCOREAUDIO);
2012
2013 PCOREAUDIOSTREAMOUT pStreamOut = (PCOREAUDIOSTREAMOUT)pStream;
2014
2015 int rc = VINF_SUCCESS;
2016
2017 uint32_t cLive = AudioMixBufLive(&pStream->MixBuf);
2018 if (!cLive) /* Not live samples to play? Bail out. */
2019 {
2020 if (pcSamplesPlayed)
2021 *pcSamplesPlayed = 0;
2022 return VINF_SUCCESS;
2023 }
2024
2025 size_t cbLive = AUDIOMIXBUF_S2B(&pStream->MixBuf, cLive);
2026
2027 uint32_t cbReadTotal = 0;
2028
2029 size_t cbToRead = RT_MIN(cbLive, RTCircBufFree(pStreamOut->pCircBuf));
2030 Log3Func(("cbToRead=%zu\n", cbToRead));
2031
2032 while (cbToRead)
2033 {
2034 uint32_t cRead, cbRead;
2035 uint8_t *puBuf;
2036 size_t cbCopy;
2037
2038 /* Try to acquire the necessary space from the ring buffer. */
2039 RTCircBufAcquireWriteBlock(pStreamOut->pCircBuf, cbToRead, (void **)&puBuf, &cbCopy);
2040 if (!cbCopy)
2041 {
2042 RTCircBufReleaseWriteBlock(pStreamOut->pCircBuf, cbCopy);
2043 break;
2044 }
2045
2046 Assert(cbCopy <= cbToRead);
2047
2048 rc = AudioMixBufReadCirc(&pStream->MixBuf,
2049 puBuf, cbCopy, &cRead);
2050
2051 if ( RT_FAILURE(rc)
2052 || !cRead)
2053 {
2054 RTCircBufReleaseWriteBlock(pStreamOut->pCircBuf, 0);
2055 break;
2056 }
2057
2058 cbRead = AUDIOMIXBUF_S2B(&pStream->MixBuf, cRead);
2059
2060 /* Release the ring buffer, so the read thread could start reading this data. */
2061 RTCircBufReleaseWriteBlock(pStreamOut->pCircBuf, cbRead);
2062
2063 Assert(cbToRead >= cbRead);
2064 cbToRead -= cbRead;
2065 cbReadTotal += cbRead;
2066 }
2067
2068 if (RT_SUCCESS(rc))
2069 {
2070 uint32_t cReadTotal = AUDIOMIXBUF_B2S(&pStream->MixBuf, cbReadTotal);
2071 if (cReadTotal)
2072 AudioMixBufFinish(&pStream->MixBuf, cReadTotal);
2073
2074 Log3Func(("cReadTotal=%RU32 (%RU32 bytes)\n", cReadTotal, cbReadTotal));
2075
2076 if (pcSamplesPlayed)
2077 *pcSamplesPlayed = cReadTotal;
2078 }
2079
2080 return rc;
2081}
2082
2083static DECLCALLBACK(int) coreAudioControlStreamOut(PDRVHOSTCOREAUDIO pThis,
2084 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
2085{
2086 PCOREAUDIOSTREAMOUT pStreamOut = (PCOREAUDIOSTREAMOUT)pStream;
2087
2088 LogFlowFunc(("enmStreamCmd=%RU32\n", enmStreamCmd));
2089
2090 uint32_t uStatus = ASMAtomicReadU32(&pStreamOut->status);
2091 if (!( uStatus == CA_STATUS_INIT
2092 || uStatus == CA_STATUS_REINIT))
2093 {
2094 return VINF_SUCCESS;
2095 }
2096
2097 int rc = VINF_SUCCESS;
2098
2099 OSStatus err;
2100 switch (enmStreamCmd)
2101 {
2102 case PDMAUDIOSTREAMCMD_ENABLE:
2103 case PDMAUDIOSTREAMCMD_RESUME:
2104 {
2105 /* Only start the device if it is actually stopped */
2106 if (!coreAudioIsRunning(pStreamOut->deviceID))
2107 {
2108 err = AudioUnitReset(pStreamOut->audioUnit, kAudioUnitScope_Input, 0);
2109 if (err != noErr)
2110 {
2111 LogRel(("CoreAudio: Failed to reset AudioUnit (%RI32)\n", err));
2112 /* Keep going. */
2113 }
2114 RTCircBufReset(pStreamOut->pCircBuf);
2115
2116 err = AudioOutputUnitStart(pStreamOut->audioUnit);
2117 if (err != noErr)
2118 {
2119 LogRel(("CoreAudio: Failed to start playback (%RI32)\n", err));
2120 rc = VERR_GENERAL_FAILURE; /** @todo Fudge! */
2121 }
2122 }
2123 break;
2124 }
2125
2126 case PDMAUDIOSTREAMCMD_DISABLE:
2127 case PDMAUDIOSTREAMCMD_PAUSE:
2128 {
2129 /* Only stop the device if it is actually running */
2130 if (coreAudioIsRunning(pStreamOut->deviceID))
2131 {
2132 err = AudioOutputUnitStop(pStreamOut->audioUnit);
2133 if (err != noErr)
2134 {
2135 LogRel(("CoreAudio: Failed to stop playback (%RI32)\n", err));
2136 rc = VERR_GENERAL_FAILURE; /** @todo Fudge! */
2137 break;
2138 }
2139
2140 err = AudioUnitReset(pStreamOut->audioUnit, kAudioUnitScope_Input, 0);
2141 if (err != noErr)
2142 {
2143 LogRel(("CoreAudio: Failed to reset AudioUnit (%RI32)\n", err));
2144 rc = VERR_GENERAL_FAILURE; /** @todo Fudge! */
2145 }
2146 }
2147 break;
2148 }
2149
2150 default:
2151 rc = VERR_NOT_SUPPORTED;
2152 break;
2153 }
2154
2155 LogFlowFuncLeaveRC(rc);
2156 return rc;
2157}
2158
2159static int coreAudioControlStreamIn(PDRVHOSTCOREAUDIO pThis,
2160 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
2161{
2162 PCOREAUDIOSTREAMIN pStreamIn = (PCOREAUDIOSTREAMIN)pStream;
2163
2164 LogFlowFunc(("enmStreamCmd=%RU32\n", enmStreamCmd));
2165
2166 uint32_t uStatus = ASMAtomicReadU32(&pStreamIn->status);
2167 if (!( uStatus == CA_STATUS_INIT
2168 || uStatus == CA_STATUS_REINIT))
2169 {
2170 return VINF_SUCCESS;
2171 }
2172
2173 int rc = VINF_SUCCESS;
2174
2175 OSStatus err;
2176 switch (enmStreamCmd)
2177 {
2178 case PDMAUDIOSTREAMCMD_ENABLE:
2179 case PDMAUDIOSTREAMCMD_RESUME:
2180 {
2181 /* Only start the device if it is actually stopped */
2182 if (!coreAudioIsRunning(pStreamIn->deviceID))
2183 {
2184 RTCircBufReset(pStreamIn->pCircBuf);
2185 err = AudioOutputUnitStart(pStreamIn->audioUnit);
2186 if (err != noErr)
2187 {
2188 LogRel(("CoreAudio: Failed to start recording (%RI32)\n", err));
2189 rc = VERR_GENERAL_FAILURE; /** @todo Fudge! */
2190 break;
2191 }
2192 }
2193
2194 break;
2195 }
2196
2197 case PDMAUDIOSTREAMCMD_DISABLE:
2198 case PDMAUDIOSTREAMCMD_PAUSE:
2199 {
2200 /* Only stop the device if it is actually running */
2201 if (coreAudioIsRunning(pStreamIn->deviceID))
2202 {
2203 err = AudioOutputUnitStop(pStreamIn->audioUnit);
2204 if (err != noErr)
2205 {
2206 LogRel(("CoreAudio: Failed to stop recording (%RI32)\n", err));
2207 rc = VERR_GENERAL_FAILURE; /** @todo Fudge! */
2208 break;
2209 }
2210
2211 err = AudioUnitReset(pStreamIn->audioUnit, kAudioUnitScope_Input, 0);
2212 if (err != noErr)
2213 {
2214 LogRel(("CoreAudio: Failed to reset AudioUnit (%RI32)\n", err));
2215 rc = VERR_GENERAL_FAILURE; /** @todo Fudge! */
2216 break;
2217 }
2218 }
2219 break;
2220 }
2221
2222 default:
2223 rc = VERR_NOT_SUPPORTED;
2224 break;
2225 }
2226
2227 LogFlowFuncLeaveRC(rc);
2228 return rc;
2229}
2230
2231static int coreAudioDestroyStreamIn(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream)
2232{
2233 PCOREAUDIOSTREAMIN pStreamIn = (PCOREAUDIOSTREAMIN)pStream;
2234
2235 LogFlowFuncEnter();
2236
2237 uint32_t status = ASMAtomicReadU32(&pStreamIn->status);
2238 if (!( status == CA_STATUS_INIT
2239 || status == CA_STATUS_REINIT))
2240 {
2241 return VINF_SUCCESS;
2242 }
2243
2244 OSStatus err = noErr;
2245
2246 int rc = coreAudioControlStreamIn(pThis, &pStreamIn->Stream, PDMAUDIOSTREAMCMD_DISABLE);
2247 if (RT_SUCCESS(rc))
2248 {
2249 ASMAtomicXchgU32(&pStreamIn->status, CA_STATUS_IN_UNINIT);
2250
2251 /*
2252 * Unregister recording device callbacks.
2253 */
2254 AudioObjectPropertyAddress propAdr = { kAudioDeviceProcessorOverload, kAudioObjectPropertyScopeGlobal,
2255 kAudioObjectPropertyElementMaster };
2256#ifdef DEBUG
2257 err = AudioObjectRemovePropertyListener(pStreamIn->deviceID, &propAdr,
2258 coreAudioRecordingAudioDevicePropertyChanged, &pStreamIn->cbCtx);
2259 if ( err != noErr
2260 && err != kAudioHardwareBadObjectError)
2261 {
2262 LogRel(("CoreAudio: Failed to remove the recording processor overload listener (%RI32)\n", err));
2263 }
2264#endif /* DEBUG */
2265
2266 propAdr.mSelector = kAudioDevicePropertyNominalSampleRate;
2267 err = AudioObjectRemovePropertyListener(pStreamIn->deviceID, &propAdr,
2268 coreAudioRecordingAudioDevicePropertyChanged, &pStreamIn->cbCtx);
2269 if ( err != noErr
2270 && err != kAudioHardwareBadObjectError)
2271 {
2272 LogRel(("CoreAudio: Failed to remove the recording sample rate changed listener (%RI32)\n", err));
2273 }
2274
2275 if (pStreamIn->fDefDevChgListReg)
2276 {
2277 propAdr.mSelector = kAudioHardwarePropertyDefaultInputDevice;
2278 err = AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &propAdr,
2279 coreAudioDefaultDeviceChanged, &pStreamIn->cbCtx);
2280 if ( err != noErr
2281 && err != kAudioHardwareBadObjectError)
2282 {
2283 LogRel(("CoreAudio: Failed to remove the default recording device changed listener (%RI32)\n", err));
2284 }
2285
2286 pStreamIn->fDefDevChgListReg = false;
2287 }
2288
2289 if (pStreamIn->fDevStateChgListReg)
2290 {
2291 Assert(pStreamIn->deviceID != kAudioDeviceUnknown);
2292
2293 AudioObjectPropertyAddress propAdr2 = { kAudioDevicePropertyDeviceIsAlive, kAudioObjectPropertyScopeGlobal,
2294 kAudioObjectPropertyElementMaster };
2295 err = AudioObjectRemovePropertyListener(pStreamIn->deviceID, &propAdr2,
2296 drvHostCoreAudioDeviceStateChanged, &pStreamIn->cbCtx);
2297 if ( err != noErr
2298 && err != kAudioHardwareBadObjectError)
2299 {
2300 LogRel(("CoreAudio: Failed to remove the recording device state changed listener (%RI32)\n", err));
2301 }
2302
2303 pStreamIn->fDevStateChgListReg = false;
2304 }
2305
2306 if (pStreamIn->pConverter)
2307 {
2308 AudioConverterDispose(pStreamIn->pConverter);
2309 pStreamIn->pConverter = NULL;
2310 }
2311
2312 err = AudioUnitUninitialize(pStreamIn->audioUnit);
2313 if (err == noErr)
2314 err = CloseComponent(pStreamIn->audioUnit);
2315
2316 if ( err != noErr
2317 && err != kAudioHardwareBadObjectError)
2318 {
2319 LogRel(("CoreAudio: Failed to uninit the recording device (%RI32)\n", err));
2320 }
2321
2322 pStreamIn->deviceID = kAudioDeviceUnknown;
2323 pStreamIn->audioUnit = NULL;
2324 pStreamIn->sampleRatio = 1;
2325
2326 coreAudioUninitConvCbCtx(&pStreamIn->convCbCtx);
2327
2328 if (pStreamIn->pCircBuf)
2329 {
2330 RTCircBufDestroy(pStreamIn->pCircBuf);
2331 pStreamIn->pCircBuf = NULL;
2332 }
2333
2334 ASMAtomicXchgU32(&pStreamIn->status, CA_STATUS_UNINIT);
2335 }
2336 else
2337 {
2338 LogRel(("CoreAudio: Failed to stop recording on uninit (%RI32)\n", err));
2339 rc = VERR_GENERAL_FAILURE; /** @todo Fudge! */
2340 }
2341
2342 LogFlowFuncLeaveRC(rc);
2343 return rc;
2344}
2345
2346static int coreAudioDestroyStreamOut(PDRVHOSTCOREAUDIO pThis, PPDMAUDIOSTREAM pStream)
2347{
2348 PCOREAUDIOSTREAMOUT pStreamOut = (PCOREAUDIOSTREAMOUT)pStream;
2349
2350 LogFlowFuncEnter();
2351
2352 uint32_t status = ASMAtomicReadU32(&pStreamOut->status);
2353 if (!( status == CA_STATUS_INIT
2354 || status == CA_STATUS_REINIT))
2355 {
2356 return VINF_SUCCESS;
2357 }
2358
2359 int rc = coreAudioControlStreamOut(pThis, &pStreamOut->Stream, PDMAUDIOSTREAMCMD_DISABLE);
2360 if (RT_SUCCESS(rc))
2361 {
2362 ASMAtomicXchgU32(&pStreamOut->status, CA_STATUS_IN_UNINIT);
2363
2364 OSStatus err;
2365
2366 /*
2367 * Unregister playback device callbacks.
2368 */
2369 AudioObjectPropertyAddress propAdr = { kAudioDeviceProcessorOverload, kAudioObjectPropertyScopeGlobal,
2370 kAudioObjectPropertyElementMaster };
2371#ifdef DEBUG
2372 err = AudioObjectRemovePropertyListener(pStreamOut->deviceID, &propAdr,
2373 coreAudioPlaybackAudioDevicePropertyChanged, &pStreamOut->cbCtx);
2374 if ( err != noErr
2375 && err != kAudioHardwareBadObjectError)
2376 {
2377 LogRel(("CoreAudio: Failed to remove the playback processor overload listener (%RI32)\n", err));
2378 }
2379#endif /* DEBUG */
2380
2381 propAdr.mSelector = kAudioDevicePropertyNominalSampleRate;
2382 err = AudioObjectRemovePropertyListener(pStreamOut->deviceID, &propAdr,
2383 coreAudioPlaybackAudioDevicePropertyChanged, &pStreamOut->cbCtx);
2384 if ( err != noErr
2385 && err != kAudioHardwareBadObjectError)
2386 {
2387 LogRel(("CoreAudio: Failed to remove the playback sample rate changed listener (%RI32)\n", err));
2388 }
2389
2390 if (pStreamOut->fDefDevChgListReg)
2391 {
2392 propAdr.mSelector = kAudioHardwarePropertyDefaultOutputDevice;
2393 propAdr.mScope = kAudioObjectPropertyScopeGlobal;
2394 propAdr.mElement = kAudioObjectPropertyElementMaster;
2395 err = AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &propAdr,
2396 coreAudioDefaultDeviceChanged, &pStreamOut->cbCtx);
2397 if ( err != noErr
2398 && err != kAudioHardwareBadObjectError)
2399 {
2400 LogRel(("CoreAudio: Failed to remove the default playback device changed listener (%RI32)\n", err));
2401 }
2402
2403 pStreamOut->fDefDevChgListReg = false;
2404 }
2405
2406 if (pStreamOut->fDevStateChgListReg)
2407 {
2408 Assert(pStreamOut->deviceID != kAudioDeviceUnknown);
2409
2410 AudioObjectPropertyAddress propAdr2 = { kAudioDevicePropertyDeviceIsAlive, kAudioObjectPropertyScopeGlobal,
2411 kAudioObjectPropertyElementMaster };
2412 err = AudioObjectRemovePropertyListener(pStreamOut->deviceID, &propAdr2,
2413 drvHostCoreAudioDeviceStateChanged, &pStreamOut->cbCtx);
2414 if ( err != noErr
2415 && err != kAudioHardwareBadObjectError)
2416 {
2417 LogRel(("CoreAudio: Failed to remove the playback device state changed listener (%RI32)\n", err));
2418 }
2419
2420 pStreamOut->fDevStateChgListReg = false;
2421 }
2422
2423 err = AudioUnitUninitialize(pStreamOut->audioUnit);
2424 if (err == noErr)
2425 err = CloseComponent(pStreamOut->audioUnit);
2426
2427 if ( err != noErr
2428 && err != kAudioHardwareBadObjectError)
2429 {
2430 LogRel(("CoreAudio: Failed to uninit the playback device (%RI32)\n", err));
2431 }
2432
2433 pStreamOut->deviceID = kAudioDeviceUnknown;
2434 pStreamOut->audioUnit = NULL;
2435 if (pStreamOut->pCircBuf)
2436 {
2437 RTCircBufDestroy(pStreamOut->pCircBuf);
2438 pStreamOut->pCircBuf = NULL;
2439 }
2440
2441 ASMAtomicXchgU32(&pStreamOut->status, CA_STATUS_UNINIT);
2442 }
2443 else
2444 LogRel(("CoreAudio: Failed to stop playback on uninit, rc=%Rrc\n", rc));
2445
2446 LogFlowFuncLeaveRC(rc);
2447 return rc;
2448}
2449
2450static int coreAudioCreateStreamIn(PDRVHOSTCOREAUDIO pThis,
2451 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
2452{
2453 PCOREAUDIOSTREAMIN pStreamIn = (PCOREAUDIOSTREAMIN)pStream;
2454
2455 LogFlowFunc(("enmRecSource=%RU32\n", pCfg->DestSource.Source));
2456
2457 pStreamIn->deviceID = kAudioDeviceUnknown;
2458 pStreamIn->audioUnit = NULL;
2459 pStreamIn->pConverter = NULL;
2460 pStreamIn->sampleRatio = 1;
2461 pStreamIn->pCircBuf = NULL;
2462 pStreamIn->status = CA_STATUS_UNINIT;
2463 pStreamIn->fDefDevChgListReg = false;
2464 pStreamIn->fDevStateChgListReg = false;
2465
2466 /* Set callback context. */
2467 pStreamIn->cbCtx.pThis = pThis;
2468 pStreamIn->cbCtx.enmDir = PDMAUDIODIR_IN;
2469 pStreamIn->cbCtx.pIn = pStreamIn;
2470
2471 bool fDeviceByUser = false; /* Do we use a device which was set by the user? */
2472
2473 /* Initialize the hardware info section with the audio settings */
2474 int rc = DrvAudioHlpStreamCfgToProps(pCfg, &pStreamIn->Stream.Props);
2475 if (RT_SUCCESS(rc))
2476 {
2477#if 0
2478 /* Try to find the audio device set by the user */
2479 if (DeviceUID.pszInputDeviceUID)
2480 {
2481 pStreamIn->deviceID = drvHostCoreAudioDeviceUIDtoID(DeviceUID.pszInputDeviceUID);
2482 /* Not fatal */
2483 if (pStreamIn->deviceID == kAudioDeviceUnknown)
2484 LogRel(("CoreAudio: Unable to find recording device %s. Falling back to the default audio device. \n", DeviceUID.pszInputDeviceUID));
2485 else
2486 fDeviceByUser = true;
2487 }
2488#endif
2489 rc = coreAudioInitIn(pThis, &pStreamIn->Stream, pcSamples);
2490 }
2491
2492 OSStatus err;
2493
2494 /* When the devices isn't forced by the user, we want default device change notifications. */
2495 if (!fDeviceByUser)
2496 {
2497 if (!pStreamIn->fDefDevChgListReg)
2498 {
2499 AudioObjectPropertyAddress propAdr = { kAudioHardwarePropertyDefaultInputDevice, kAudioObjectPropertyScopeGlobal,
2500 kAudioObjectPropertyElementMaster };
2501 err = AudioObjectAddPropertyListener(kAudioObjectSystemObject, &propAdr,
2502 coreAudioDefaultDeviceChanged, &pStreamIn->cbCtx);
2503 if ( err == noErr
2504 || err == kAudioHardwareIllegalOperationError)
2505 {
2506 pStreamIn->fDefDevChgListReg = true;
2507 }
2508 else
2509 LogRel(("CoreAudio: Failed to add the default recording device changed listener (%RI32)\n", err));
2510 }
2511 }
2512
2513 if ( !pStreamIn->fDevStateChgListReg
2514 && (pStreamIn->deviceID != kAudioDeviceUnknown))
2515 {
2516 /* Register callback for being notified if the device stops being alive. */
2517 AudioObjectPropertyAddress propAdr = { kAudioDevicePropertyDeviceIsAlive, kAudioObjectPropertyScopeGlobal,
2518 kAudioObjectPropertyElementMaster };
2519 err = AudioObjectAddPropertyListener(pStreamIn->deviceID, &propAdr, drvHostCoreAudioDeviceStateChanged,
2520 &pStreamIn->cbCtx);
2521 if (err == noErr)
2522 {
2523 pStreamIn->fDevStateChgListReg = true;
2524 }
2525 else
2526 LogRel(("CoreAudio: Failed to add the recording device state changed listener (%RI32)\n", err));
2527 }
2528
2529 LogFlowFuncLeaveRC(rc);
2530 return rc;
2531}
2532
2533static int coreAudioCreateStreamOut(PDRVHOSTCOREAUDIO pThis,
2534 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg,
2535 uint32_t *pcSamples)
2536{
2537 PCOREAUDIOSTREAMOUT pStreamOut = (PCOREAUDIOSTREAMOUT)pStream;
2538
2539 LogFlowFuncEnter();
2540
2541 pStreamOut->deviceID = kAudioDeviceUnknown;
2542 pStreamOut->audioUnit = NULL;
2543 pStreamOut->pCircBuf = NULL;
2544 pStreamOut->status = CA_STATUS_UNINIT;
2545 pStreamOut->fDefDevChgListReg = false;
2546 pStreamOut->fDevStateChgListReg = false;
2547
2548 /* Set callback context. */
2549 pStreamOut->cbCtx.pThis = pThis;
2550 pStreamOut->cbCtx.enmDir = PDMAUDIODIR_OUT;
2551 pStreamOut->cbCtx.pOut = pStreamOut;
2552
2553 bool fDeviceByUser = false; /* Do we use a device which was set by the user? */
2554
2555 /* If a stream configuration is given, apply that to the stream. */
2556 int rc = DrvAudioHlpStreamCfgToProps(pCfg, &pStreamOut->Stream.Props);
2557 if (RT_SUCCESS(rc))
2558 {
2559#if 0
2560 /* Try to find the audio device set by the user. Use
2561 * export VBOX_COREAUDIO_OUTPUT_DEVICE_UID=AppleHDAEngineOutput:0
2562 * to set it. */
2563 if (DeviceUID.pszOutputDeviceUID)
2564 {
2565 pStreamOut->audioDeviceId = drvHostCoreAudioDeviceUIDtoID(DeviceUID.pszOutputDeviceUID);
2566 /* Not fatal */
2567 if (pStreamOut->audioDeviceId == kAudioDeviceUnknown)
2568 LogRel(("CoreAudio: Unable to find playback device %s. Falling back to the default audio device. \n", DeviceUID.pszOutputDeviceUID));
2569 else
2570 fDeviceByUser = true;
2571 }
2572#endif
2573 rc = coreAudioInitOut(pThis, pStream, pcSamples);
2574 }
2575
2576 OSStatus err;
2577
2578 /* When the devices isn't forced by the user, we want default device change notifications. */
2579 if (!fDeviceByUser)
2580 {
2581 AudioObjectPropertyAddress propAdr = { kAudioHardwarePropertyDefaultOutputDevice, kAudioObjectPropertyScopeGlobal,
2582 kAudioObjectPropertyElementMaster };
2583 err = AudioObjectAddPropertyListener(kAudioObjectSystemObject, &propAdr,
2584 coreAudioDefaultDeviceChanged, &pStreamOut->cbCtx);
2585 if (err == noErr)
2586 {
2587 pStreamOut->fDefDevChgListReg = true;
2588 }
2589 else
2590 LogRel(("CoreAudio: Failed to add the default playback device changed listener (%RI32)\n", err));
2591 }
2592
2593 if ( !pStreamOut->fDevStateChgListReg
2594 && (pStreamOut->deviceID != kAudioDeviceUnknown))
2595 {
2596 /* Register callback for being notified if the device stops being alive. */
2597 AudioObjectPropertyAddress propAdr = { kAudioDevicePropertyDeviceIsAlive, kAudioObjectPropertyScopeGlobal,
2598 kAudioObjectPropertyElementMaster };
2599 err = AudioObjectAddPropertyListener(pStreamOut->deviceID, &propAdr, drvHostCoreAudioDeviceStateChanged,
2600 (void *)&pStreamOut->cbCtx);
2601 if (err == noErr)
2602 {
2603 pStreamOut->fDevStateChgListReg = true;
2604 }
2605 else
2606 LogRel(("CoreAudio: Failed to add the playback device state changed listener (%RI32)\n", err));
2607 }
2608
2609 LogFlowFuncLeaveRC(rc);
2610 return rc;
2611}
2612
2613static DECLCALLBACK(int) drvHostCoreAudioGetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pCfg)
2614{
2615 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2616 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
2617
2618 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
2619 PDRVHOSTCOREAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTCOREAUDIO);
2620
2621 return coreAudioUpdateStatusInternalEx(pThis, pCfg, 0 /* fEnum */);
2622}
2623
2624static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvHostCoreAudioGetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
2625{
2626 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
2627
2628 return PDMAUDIOBACKENDSTS_RUNNING;
2629}
2630
2631static DECLCALLBACK(int) drvHostCoreAudioStreamCreate(PPDMIHOSTAUDIO pInterface,
2632 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
2633{
2634 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2635 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2636 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
2637
2638 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
2639 PDRVHOSTCOREAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTCOREAUDIO);
2640
2641 int rc;
2642 if (pCfg->enmDir == PDMAUDIODIR_IN)
2643 rc = coreAudioCreateStreamIn(pThis, pStream, pCfg, pcSamples);
2644 else
2645 rc = coreAudioCreateStreamOut(pThis, pStream, pCfg, pcSamples);
2646
2647 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
2648 return rc;
2649}
2650
2651static DECLCALLBACK(int) drvHostCoreAudioStreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
2652{
2653 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2654 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2655
2656 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
2657 PDRVHOSTCOREAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTCOREAUDIO);
2658
2659 int rc;
2660 if (pStream->enmDir == PDMAUDIODIR_IN)
2661 rc = coreAudioDestroyStreamIn(pThis, pStream);
2662 else
2663 rc = coreAudioDestroyStreamOut(pThis, pStream);
2664
2665 return rc;
2666}
2667
2668static DECLCALLBACK(int) drvHostCoreAudioStreamControl(PPDMIHOSTAUDIO pInterface,
2669 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
2670{
2671 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2672 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2673
2674 Assert(pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
2675
2676 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
2677 PDRVHOSTCOREAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTCOREAUDIO);
2678
2679 int rc;
2680 if (pStream->enmDir == PDMAUDIODIR_IN)
2681 rc = coreAudioControlStreamIn(pThis, pStream, enmStreamCmd);
2682 else
2683 rc = coreAudioControlStreamOut(pThis, pStream, enmStreamCmd);
2684
2685 return rc;
2686}
2687
2688static DECLCALLBACK(PDMAUDIOSTRMSTS) drvHostCoreAudioStreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
2689{
2690 NOREF(pInterface);
2691 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2692
2693 Assert(pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
2694
2695 PDMAUDIOSTRMSTS strmSts = PDMAUDIOSTRMSTS_FLAG_NONE;
2696
2697 if (pStream->enmDir == PDMAUDIODIR_IN)
2698 {
2699 PCOREAUDIOSTREAMIN pStreamIn = (PCOREAUDIOSTREAMIN)pStream;
2700
2701 if (ASMAtomicReadU32(&pStreamIn->status) == CA_STATUS_INIT)
2702 strmSts |= PDMAUDIOSTRMSTS_FLAG_INITIALIZED | PDMAUDIOSTRMSTS_FLAG_ENABLED;
2703
2704 if (strmSts & PDMAUDIOSTRMSTS_FLAG_ENABLED)
2705 strmSts |= PDMAUDIOSTRMSTS_FLAG_DATA_READABLE;
2706 }
2707 else if (pStream->enmDir == PDMAUDIODIR_OUT)
2708 {
2709 PCOREAUDIOSTREAMOUT pStreamOut = (PCOREAUDIOSTREAMOUT)pStream;
2710
2711 if (ASMAtomicReadU32(&pStreamOut->status) == CA_STATUS_INIT)
2712 strmSts |= PDMAUDIOSTRMSTS_FLAG_INITIALIZED | PDMAUDIOSTRMSTS_FLAG_ENABLED;
2713
2714 if (strmSts & PDMAUDIOSTRMSTS_FLAG_ENABLED)
2715 strmSts |= PDMAUDIOSTRMSTS_FLAG_DATA_WRITABLE;
2716 }
2717 else
2718 AssertFailed();
2719
2720 return strmSts;
2721}
2722
2723static DECLCALLBACK(int) drvHostCoreAudioStreamIterate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
2724{
2725 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2726 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2727
2728 /* Nothing to do here for Core Audio. */
2729 return VINF_SUCCESS;
2730}
2731
2732static DECLCALLBACK(void) drvHostCoreAudioShutdown(PPDMIHOSTAUDIO pInterface)
2733{
2734 NOREF(pInterface);
2735}
2736
2737static DECLCALLBACK(void *) drvHostCoreAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
2738{
2739 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
2740 PDRVHOSTCOREAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTCOREAUDIO);
2741
2742 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
2743 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
2744
2745 return NULL;
2746}
2747
2748 /* Construct a DirectSound Audio driver instance.
2749 *
2750 * @copydoc FNPDMDRVCONSTRUCT
2751 */
2752static DECLCALLBACK(int) drvHostCoreAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
2753{
2754 PDRVHOSTCOREAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTCOREAUDIO);
2755 LogRel(("Audio: Initializing Core Audio driver\n"));
2756
2757 /*
2758 * Init the static parts.
2759 */
2760 pThis->pDrvIns = pDrvIns;
2761 /* IBase */
2762 pDrvIns->IBase.pfnQueryInterface = drvHostCoreAudioQueryInterface;
2763 /* IHostAudio */
2764 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvHostCoreAudio);
2765
2766 return VINF_SUCCESS;
2767}
2768
2769/**
2770 * Char driver registration record.
2771 */
2772const PDMDRVREG g_DrvHostCoreAudio =
2773{
2774 /* u32Version */
2775 PDM_DRVREG_VERSION,
2776 /* szName */
2777 "CoreAudio",
2778 /* szRCMod */
2779 "",
2780 /* szR0Mod */
2781 "",
2782 /* pszDescription */
2783 "Core Audio host driver",
2784 /* fFlags */
2785 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
2786 /* fClass. */
2787 PDM_DRVREG_CLASS_AUDIO,
2788 /* cMaxInstances */
2789 ~0U,
2790 /* cbInstance */
2791 sizeof(DRVHOSTCOREAUDIO),
2792 /* pfnConstruct */
2793 drvHostCoreAudioConstruct,
2794 /* pfnDestruct */
2795 NULL,
2796 /* pfnRelocate */
2797 NULL,
2798 /* pfnIOCtl */
2799 NULL,
2800 /* pfnPowerOn */
2801 NULL,
2802 /* pfnReset */
2803 NULL,
2804 /* pfnSuspend */
2805 NULL,
2806 /* pfnResume */
2807 NULL,
2808 /* pfnAttach */
2809 NULL,
2810 /* pfnDetach */
2811 NULL,
2812 /* pfnPowerOff */
2813 NULL,
2814 /* pfnSoftReset */
2815 NULL,
2816 /* u32EndVersion */
2817 PDM_DRVREG_VERSION
2818};
2819
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette