VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvAudio.cpp@ 67075

Last change on this file since 67075 was 65742, checked in by vboxsync, 8 years ago

Build fix.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 108.4 KB
Line 
1/* $Id: DrvAudio.cpp 65742 2017-02-10 16:21:55Z vboxsync $ */
2/** @file
3 * Intermediate audio driver header.
4 *
5 * @remarks Intermediate audio driver for connecting the audio device emulation
6 * with the host backend.
7 */
8
9/*
10 * Copyright (C) 2006-2017 Oracle Corporation
11 *
12 * This file is part of VirtualBox Open Source Edition (OSE), as
13 * available from http://www.virtualbox.org. This file is free software;
14 * you can redistribute it and/or modify it under the terms of the GNU
15 * General Public License (GPL) as published by the Free Software
16 * Foundation, in version 2 as it comes in the "COPYING" file of the
17 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
18 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
19 * --------------------------------------------------------------------
20 */
21#define LOG_GROUP LOG_GROUP_DRV_AUDIO
22#include <VBox/log.h>
23#include <VBox/vmm/pdm.h>
24#include <VBox/err.h>
25#include <VBox/vmm/mm.h>
26#include <VBox/vmm/pdmaudioifs.h>
27
28#include <iprt/alloc.h>
29#include <iprt/asm-math.h>
30#include <iprt/assert.h>
31#include <iprt/circbuf.h>
32#include <iprt/string.h>
33#include <iprt/uuid.h>
34
35#include "VBoxDD.h"
36
37#include <ctype.h>
38#include <stdlib.h>
39
40#include "DrvAudio.h"
41#include "AudioMixBuffer.h"
42
43#ifdef VBOX_WITH_AUDIO_ENUM
44static int drvAudioDevicesEnumerateInternal(PDRVAUDIO pThis, bool fLog, PPDMAUDIODEVICEENUM pDevEnum);
45#endif
46
47static DECLCALLBACK(int) drvAudioStreamDestroy(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream);
48static int drvAudioStreamControlInternalBackend(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd);
49static int drvAudioStreamControlInternal(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd);
50static int drvAudioStreamCreateInternalBackend(PDRVAUDIO pThis, PPDMAUDIOSTREAM pHstStream, PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq);
51static int drvAudioStreamDestroyInternalBackend(PDRVAUDIO pThis, PPDMAUDIOSTREAM pHstStream);
52static void drvAudioStreamFree(PPDMAUDIOSTREAM pStream);
53static int drvAudioStreamUninitInternal(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream);
54static int drvAudioStreamInitInternal(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfgHost, PPDMAUDIOSTREAMCFG pCfgGuest);
55static int drvAudioStreamIterateInternal(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream);
56static int drvAudioStreamReInitInternal(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream);
57static int drvAudioStreamLinkToInternal(PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAM pPair);
58
59#ifndef VBOX_AUDIO_TESTCASE
60
61# if 0 /* unused */
62
63static PDMAUDIOFMT drvAudioGetConfFormat(PCFGMNODE pCfgHandle, const char *pszKey,
64 PDMAUDIOFMT enmDefault, bool *pfDefault)
65{
66 if ( pCfgHandle == NULL
67 || pszKey == NULL)
68 {
69 *pfDefault = true;
70 return enmDefault;
71 }
72
73 char *pszValue = NULL;
74 int rc = CFGMR3QueryStringAlloc(pCfgHandle, pszKey, &pszValue);
75 if (RT_FAILURE(rc))
76 {
77 *pfDefault = true;
78 return enmDefault;
79 }
80
81 PDMAUDIOFMT fmt = DrvAudioHlpStrToAudFmt(pszValue);
82 if (fmt == PDMAUDIOFMT_INVALID)
83 {
84 *pfDefault = true;
85 return enmDefault;
86 }
87
88 *pfDefault = false;
89 return fmt;
90}
91
92static int drvAudioGetConfInt(PCFGMNODE pCfgHandle, const char *pszKey,
93 int iDefault, bool *pfDefault)
94{
95
96 if ( pCfgHandle == NULL
97 || pszKey == NULL)
98 {
99 *pfDefault = true;
100 return iDefault;
101 }
102
103 uint64_t u64Data = 0;
104 int rc = CFGMR3QueryInteger(pCfgHandle, pszKey, &u64Data);
105 if (RT_FAILURE(rc))
106 {
107 *pfDefault = true;
108 return iDefault;
109
110 }
111
112 *pfDefault = false;
113 return u64Data;
114}
115
116static const char *drvAudioGetConfStr(PCFGMNODE pCfgHandle, const char *pszKey,
117 const char *pszDefault, bool *pfDefault)
118{
119 if ( pCfgHandle == NULL
120 || pszKey == NULL)
121 {
122 *pfDefault = true;
123 return pszDefault;
124 }
125
126 char *pszValue = NULL;
127 int rc = CFGMR3QueryStringAlloc(pCfgHandle, pszKey, &pszValue);
128 if (RT_FAILURE(rc))
129 {
130 *pfDefault = true;
131 return pszDefault;
132 }
133
134 *pfDefault = false;
135 return pszValue;
136}
137
138# endif /* unused */
139
140#ifdef LOG_ENABLED
141/**
142 * Converts an audio stream status to a string.
143 *
144 * @returns Stringified stream status flags. Must be free'd with RTStrFree().
145 * "NONE" if no flags set.
146 * @param fFlags Stream status flags to convert.
147 */
148static char *dbgAudioStreamStatusToStr(PDMAUDIOSTRMSTS fStatus)
149{
150#define APPEND_FLAG_TO_STR(_aFlag) \
151 if (fStatus & PDMAUDIOSTRMSTS_FLAG_##_aFlag) \
152 { \
153 if (pszFlags) \
154 { \
155 rc2 = RTStrAAppend(&pszFlags, " "); \
156 if (RT_FAILURE(rc2)) \
157 break; \
158 } \
159 \
160 rc2 = RTStrAAppend(&pszFlags, #_aFlag); \
161 if (RT_FAILURE(rc2)) \
162 break; \
163 } \
164
165 char *pszFlags = NULL;
166 int rc2 = VINF_SUCCESS;
167
168 do
169 {
170 APPEND_FLAG_TO_STR(INITIALIZED );
171 APPEND_FLAG_TO_STR(ENABLED );
172 APPEND_FLAG_TO_STR(PAUSED );
173 APPEND_FLAG_TO_STR(PENDING_DISABLE);
174 APPEND_FLAG_TO_STR(PENDING_REINIT );
175 } while (0);
176
177 if (!pszFlags)
178 rc2 = RTStrAAppend(&pszFlags, "NONE");
179
180 if ( RT_FAILURE(rc2)
181 && pszFlags)
182 {
183 RTStrFree(pszFlags);
184 pszFlags = NULL;
185 }
186
187#undef APPEND_FLAG_TO_STR
188
189 return pszFlags;
190}
191#endif /* LOG_ENABLED */
192
193/**
194 * Returns the host stream part of an audio stream pair, or NULL
195 * if no host stream has been assigned / is not available.
196 *
197 * @returns IPRT status code.
198 * @param pStream Audio stream to retrieve host stream part for.
199 */
200DECLINLINE(PPDMAUDIOSTREAM) drvAudioGetHostStream(PPDMAUDIOSTREAM pStream)
201{
202 AssertPtrReturn(pStream, NULL);
203
204 if (!pStream)
205 return NULL;
206
207 PPDMAUDIOSTREAM pHstStream = pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST
208 ? pStream
209 : pStream->pPair;
210 if (pHstStream)
211 {
212 AssertReleaseMsg(pHstStream->enmCtx == PDMAUDIOSTREAMCTX_HOST,
213 ("Stream '%s' resolved as host part is not marked as such (enmCtx=%RU32)\n",
214 pHstStream->szName, pHstStream->enmCtx));
215
216 AssertReleaseMsg(pHstStream->pPair != NULL,
217 ("Stream '%s' resolved as host part has no guest part (anymore)\n", pHstStream->szName));
218 }
219 else
220 LogRel(("Audio: Warning: Stream '%s' does not have a host stream (anymore)\n", pStream->szName));
221
222 return pHstStream;
223}
224
225# if 0 /* unused */
226static int drvAudioProcessOptions(PCFGMNODE pCfgHandle, const char *pszPrefix, audio_option *paOpts, size_t cOpts)
227{
228 AssertPtrReturn(pCfgHandle, VERR_INVALID_POINTER);
229 AssertPtrReturn(pszPrefix, VERR_INVALID_POINTER);
230 /* oaOpts and cOpts are optional. */
231
232 PCFGMNODE pCfgChildHandle = NULL;
233 PCFGMNODE pCfgChildChildHandle = NULL;
234
235 /* If pCfgHandle is NULL, let NULL be passed to get int and get string functions..
236 * The getter function will return default values.
237 */
238 if (pCfgHandle != NULL)
239 {
240 /* If its audio general setting, need to traverse to one child node.
241 * /Devices/ichac97/0/LUN#0/Config/Audio
242 */
243 if(!strncmp(pszPrefix, "AUDIO", 5)) /** @todo Use a \#define */
244 {
245 pCfgChildHandle = CFGMR3GetFirstChild(pCfgHandle);
246 if(pCfgChildHandle)
247 pCfgHandle = pCfgChildHandle;
248 }
249 else
250 {
251 /* If its driver specific configuration , then need to traverse two level deep child
252 * child nodes. for eg. in case of DirectSoundConfiguration item
253 * /Devices/ichac97/0/LUN#0/Config/Audio/DirectSoundConfig
254 */
255 pCfgChildHandle = CFGMR3GetFirstChild(pCfgHandle);
256 if (pCfgChildHandle)
257 {
258 pCfgChildChildHandle = CFGMR3GetFirstChild(pCfgChildHandle);
259 if (pCfgChildChildHandle)
260 pCfgHandle = pCfgChildChildHandle;
261 }
262 }
263 }
264
265 for (size_t i = 0; i < cOpts; i++)
266 {
267 audio_option *pOpt = &paOpts[i];
268 if (!pOpt->valp)
269 {
270 LogFlowFunc(("Option value pointer for `%s' is not set\n", pOpt->name));
271 continue;
272 }
273
274 bool fUseDefault;
275
276 switch (pOpt->tag)
277 {
278 case AUD_OPT_BOOL:
279 case AUD_OPT_INT:
280 {
281 int *intp = (int *)pOpt->valp;
282 *intp = drvAudioGetConfInt(pCfgHandle, pOpt->name, *intp, &fUseDefault);
283
284 break;
285 }
286
287 case AUD_OPT_FMT:
288 {
289 PDMAUDIOFMT *fmtp = (PDMAUDIOFMT *)pOpt->valp;
290 *fmtp = drvAudioGetConfFormat(pCfgHandle, pOpt->name, *fmtp, &fUseDefault);
291
292 break;
293 }
294
295 case AUD_OPT_STR:
296 {
297 const char **strp = (const char **)pOpt->valp;
298 *strp = drvAudioGetConfStr(pCfgHandle, pOpt->name, *strp, &fUseDefault);
299
300 break;
301 }
302
303 default:
304 LogFlowFunc(("Bad value tag for option `%s' - %d\n", pOpt->name, pOpt->tag));
305 fUseDefault = false;
306 break;
307 }
308
309 if (!pOpt->overridenp)
310 pOpt->overridenp = &pOpt->overriden;
311
312 *pOpt->overridenp = !fUseDefault;
313 }
314
315 return VINF_SUCCESS;
316}
317# endif /* unused */
318#endif /* !VBOX_AUDIO_TESTCASE */
319
320/**
321 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamControl}
322 */
323static DECLCALLBACK(int) drvAudioStreamControl(PPDMIAUDIOCONNECTOR pInterface,
324 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
325{
326 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
327
328 if (!pStream)
329 return VINF_SUCCESS;
330
331 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
332
333 int rc = RTCritSectEnter(&pThis->CritSect);
334 if (RT_FAILURE(rc))
335 return rc;
336
337 LogFlowFunc(("[%s] enmStreamCmd=%s\n", pStream->szName, DrvAudioHlpStreamCmdToStr(enmStreamCmd)));
338
339 rc = drvAudioStreamControlInternal(pThis, pStream, enmStreamCmd);
340
341 int rc2 = RTCritSectLeave(&pThis->CritSect);
342 if (RT_SUCCESS(rc))
343 rc = rc2;
344
345 return rc;
346}
347
348/**
349 * Controls an audio stream.
350 *
351 * @returns IPRT status code.
352 * @param pThis Pointer to driver instance.
353 * @param pStream Stream to control.
354 * @param enmStreamCmd Control command.
355 */
356static int drvAudioStreamControlInternal(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
357{
358 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
359 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
360
361 LogFunc(("[%s] enmStreamCmd=%s\n", pStream->szName, DrvAudioHlpStreamCmdToStr(enmStreamCmd)));
362
363 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
364 PPDMAUDIOSTREAM pGstStream = pHstStream ? pHstStream->pPair : pStream;
365 AssertPtr(pGstStream);
366
367#ifdef LOG_ENABLED
368 char *pszHstSts = dbgAudioStreamStatusToStr(pHstStream->fStatus);
369 char *pszGstSts = dbgAudioStreamStatusToStr(pHstStream->fStatus);
370 LogFlowFunc(("Status host=%s, guest=%s\n", pszHstSts, pszGstSts));
371 RTStrFree(pszGstSts);
372 RTStrFree(pszHstSts);
373#endif /* LOG_ENABLED */
374
375 int rc = VINF_SUCCESS;
376
377 switch (enmStreamCmd)
378 {
379 case PDMAUDIOSTREAMCMD_ENABLE:
380 {
381 if (!(pGstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_ENABLED))
382 {
383 if (pHstStream)
384 {
385 /* Is a pending disable outstanding? Then disable first. */
386 if (pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_PENDING_DISABLE)
387 rc = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_DISABLE);
388
389 if (RT_SUCCESS(rc))
390 rc = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_ENABLE);
391 }
392
393 if (RT_SUCCESS(rc))
394 pGstStream->fStatus |= PDMAUDIOSTRMSTS_FLAG_ENABLED;
395 }
396 break;
397 }
398
399 case PDMAUDIOSTREAMCMD_DISABLE:
400 {
401 if (pGstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_ENABLED)
402 {
403 if (pHstStream)
404 {
405 /*
406 * For playback (output) streams first mark the host stream as pending disable,
407 * so that the rest of the remaining audio data will be played first before
408 * closing the stream.
409 */
410 if (pHstStream->enmDir == PDMAUDIODIR_OUT)
411 {
412 LogFunc(("[%s] Pending disable/pause\n", pHstStream->szName));
413 pHstStream->fStatus |= PDMAUDIOSTRMSTS_FLAG_PENDING_DISABLE;
414 }
415
416 /* Can we close the host stream as well (not in pending disable mode)? */
417 if (!(pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_PENDING_DISABLE))
418 rc = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_DISABLE);
419 }
420
421 if (RT_SUCCESS(rc))
422 pGstStream->fStatus &= ~PDMAUDIOSTRMSTS_FLAG_ENABLED;
423 }
424 break;
425 }
426
427 case PDMAUDIOSTREAMCMD_PAUSE:
428 {
429 if (!(pGstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_PAUSED))
430 {
431 if (pHstStream)
432 rc = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_PAUSE);
433
434 if (RT_SUCCESS(rc))
435 pGstStream->fStatus |= PDMAUDIOSTRMSTS_FLAG_PAUSED;
436 }
437 break;
438 }
439
440 case PDMAUDIOSTREAMCMD_RESUME:
441 {
442 if (pGstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_PAUSED)
443 {
444 if (pHstStream)
445 rc = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_RESUME);
446
447 if (RT_SUCCESS(rc))
448 pGstStream->fStatus &= ~PDMAUDIOSTRMSTS_FLAG_PAUSED;
449 }
450 break;
451 }
452
453 default:
454 AssertMsgFailed(("Command %s (%RU32) not implemented\n", DrvAudioHlpStreamCmdToStr(enmStreamCmd), enmStreamCmd));
455 rc = VERR_NOT_IMPLEMENTED;
456 break;
457 }
458
459 if (RT_FAILURE(rc))
460 LogFunc(("[%s] Failed with %Rrc\n", pStream->szName, rc));
461
462 return rc;
463}
464
465/**
466 * Controls a stream's backend.
467 * If the stream has no backend available, VERR_NOT_FOUND is returned.
468 *
469 * @returns IPRT status code.
470 * @param pThis Pointer to driver instance.
471 * @param pStream Stream to control.
472 * @param enmStreamCmd Control command.
473 */
474static int drvAudioStreamControlInternalBackend(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
475{
476 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
477 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
478
479 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
480 if (!pHstStream) /* Stream does not have a host backend? Bail out. */
481 return VERR_NOT_FOUND;
482
483#ifdef LOG_ENABLED
484 char *pszHstSts = dbgAudioStreamStatusToStr(pHstStream->fStatus);
485 LogFlowFunc(("[%s] enmStreamCmd=%s, fStatus=%s\n", pHstStream->szName, DrvAudioHlpStreamCmdToStr(enmStreamCmd), pszHstSts));
486 RTStrFree(pszHstSts);
487#endif /* LOG_ENABLED */
488
489 AssertPtr(pThis->pHostDrvAudio);
490
491 int rc = VINF_SUCCESS;
492
493 switch (enmStreamCmd)
494 {
495 case PDMAUDIOSTREAMCMD_ENABLE:
496 {
497 if (!(pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_ENABLED))
498 {
499 LogRel2(("Audio: Enabling stream '%s'\n", pHstStream->szName));
500 rc = pThis->pHostDrvAudio->pfnStreamControl(pThis->pHostDrvAudio, pHstStream->pvBackend,
501 PDMAUDIOSTREAMCMD_ENABLE);
502 if (RT_SUCCESS(rc))
503 {
504 pHstStream->fStatus |= PDMAUDIOSTRMSTS_FLAG_ENABLED;
505 }
506 else
507 LogRel2(("Audio: Disabling stream '%s' failed with %Rrc\n", pHstStream->szName, rc));
508 }
509 break;
510 }
511
512 case PDMAUDIOSTREAMCMD_DISABLE:
513 {
514 if (pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_ENABLED)
515 {
516 LogRel2(("Audio: Disabling stream '%s'\n", pHstStream->szName));
517 rc = pThis->pHostDrvAudio->pfnStreamControl(pThis->pHostDrvAudio, pHstStream->pvBackend,
518 PDMAUDIOSTREAMCMD_DISABLE);
519 if (RT_SUCCESS(rc))
520 {
521 pHstStream->fStatus &= ~PDMAUDIOSTRMSTS_FLAG_ENABLED;
522 pHstStream->fStatus &= ~PDMAUDIOSTRMSTS_FLAG_PENDING_DISABLE;
523 AudioMixBufReset(&pHstStream->MixBuf);
524 }
525 else
526 LogRel2(("Audio: Disabling stream '%s' failed with %Rrc\n", pHstStream->szName, rc));
527 }
528 break;
529 }
530
531 case PDMAUDIOSTREAMCMD_PAUSE:
532 {
533 /* Only pause if the stream is enabled. */
534 if (!(pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_ENABLED))
535 break;
536
537 if (!(pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_PAUSED))
538 {
539 LogRel2(("Audio: Pausing stream '%s'\n", pHstStream->szName));
540 rc = pThis->pHostDrvAudio->pfnStreamControl(pThis->pHostDrvAudio, pHstStream->pvBackend,
541 PDMAUDIOSTREAMCMD_PAUSE);
542 if (RT_SUCCESS(rc))
543 {
544 pHstStream->fStatus |= PDMAUDIOSTRMSTS_FLAG_PAUSED;
545 }
546 else
547 LogRel2(("Audio: Pausing stream '%s' failed with %Rrc\n", pHstStream->szName, rc));
548 }
549 break;
550 }
551
552 case PDMAUDIOSTREAMCMD_RESUME:
553 {
554 /* Only need to resume if the stream is enabled. */
555 if (!(pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_ENABLED))
556 break;
557
558 if (pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_PAUSED)
559 {
560 LogRel2(("Audio: Resuming stream '%s'\n", pHstStream->szName));
561 rc = pThis->pHostDrvAudio->pfnStreamControl(pThis->pHostDrvAudio, pHstStream->pvBackend,
562 PDMAUDIOSTREAMCMD_RESUME);
563 if (RT_SUCCESS(rc))
564 {
565 pHstStream->fStatus &= ~PDMAUDIOSTRMSTS_FLAG_PAUSED;
566 }
567 else
568 LogRel2(("Audio: Resuming stream '%s' failed with %Rrc\n", pHstStream->szName, rc));
569 }
570 break;
571 }
572
573 default:
574 {
575 AssertMsgFailed(("Command %RU32 not implemented\n", enmStreamCmd));
576 rc = VERR_NOT_IMPLEMENTED;
577 break;
578 }
579 }
580
581 if (RT_FAILURE(rc))
582 LogFunc(("[%s] Failed with %Rrc\n", pStream->szName, rc));
583
584 return rc;
585}
586
587/**
588 * Initializes an audio stream with a given host and guest stream configuration.
589 *
590 * @returns IPRT status code.
591 * @param pThis Pointer to driver instance.
592 * @param pStream Stream to initialize.
593 * @param pCfgHost Stream configuration to use for the host side (backend).
594 * @param pCfgGuest Stream configuration to use for the guest side.
595 */
596static int drvAudioStreamInitInternal(PDRVAUDIO pThis,
597 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfgHost, PPDMAUDIOSTREAMCFG pCfgGuest)
598{
599 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
600 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
601 AssertPtrReturn(pCfgHost, VERR_INVALID_POINTER);
602 AssertPtrReturn(pCfgGuest, VERR_INVALID_POINTER);
603
604 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
605 PPDMAUDIOSTREAM pGstStream = pHstStream ? pHstStream->pPair : pStream;
606 AssertPtr(pGstStream);
607
608 /*
609 * Init host stream.
610 */
611
612 /* Set the host's default audio data layout. */
613 pCfgHost->enmLayout = PDMAUDIOSTREAMLAYOUT_NON_INTERLEAVED;
614
615#ifdef DEBUG
616 LogFunc(("[%s] Requested host format:\n", pStream->szName));
617 DrvAudioHlpStreamCfgPrint(pCfgHost);
618#else
619 LogRel2(("Audio: Creating stream '%s'\n", pStream->szName));
620 LogRel2(("Audio: Requested %s host format for '%s': %RU32Hz, %RU8%s, %RU8 %s\n",
621 pCfgGuest->enmDir == PDMAUDIODIR_IN ? "recording" : "playback", pStream->szName,
622 pCfgHost->Props.uHz, pCfgHost->Props.cBits, pCfgHost->Props.fSigned ? "S" : "U",
623 pCfgHost->Props.cChannels, pCfgHost->Props.cChannels == 0 ? "Channel" : "Channels"));
624#endif
625
626 PDMAUDIOSTREAMCFG CfgHostAcq;
627 int rc = drvAudioStreamCreateInternalBackend(pThis, pHstStream, pCfgHost, &CfgHostAcq);
628 if (RT_FAILURE(rc))
629 return rc;
630
631#ifdef DEBUG
632 LogFunc(("[%s] Acquired host format:\n", pStream->szName));
633 DrvAudioHlpStreamCfgPrint(&CfgHostAcq);
634#else
635 LogRel2(("Audio: Acquired %s host format for '%s': %RU32Hz, %RU8%s, %RU8 %s\n",
636 CfgHostAcq.enmDir == PDMAUDIODIR_IN ? "recording" : "playback", pStream->szName,
637 CfgHostAcq.Props.uHz, CfgHostAcq.Props.cBits, CfgHostAcq.Props.fSigned ? "S" : "U",
638 CfgHostAcq.Props.cChannels, CfgHostAcq.Props.cChannels == 0 ? "Channel" : "Channels"));
639#endif
640
641 /* No sample buffer size hint given by the backend? Default to some sane value. */
642 if (!CfgHostAcq.cSampleBufferHint)
643 {
644 CfgHostAcq.cSampleBufferHint = _1K; /** @todo Make this configurable? */
645 }
646
647 /* Destroy any former mixing buffer. */
648 AudioMixBufDestroy(&pHstStream->MixBuf);
649
650 /* Make sure to (re-)set the host buffer's shift size. */
651 CfgHostAcq.Props.cShift = PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(CfgHostAcq.Props.cBits, CfgHostAcq.Props.cChannels);
652
653 /* Set set host buffer size multiplicator. */
654 const unsigned cSampleBufferHostFactor = 10; /** @todo Make this configurable. */
655
656 LogFunc(("[%s] cSamples=%RU32 (x %u)\n", pHstStream->szName, CfgHostAcq.cSampleBufferHint, cSampleBufferHostFactor));
657
658 int rc2 = AudioMixBufInit(&pHstStream->MixBuf, pHstStream->szName, &CfgHostAcq.Props,
659 CfgHostAcq.cSampleBufferHint * cSampleBufferHostFactor);
660 AssertRC(rc2);
661
662 /* Make a copy of the acquired host stream configuration. */
663 rc2 = DrvAudioHlpStreamCfgCopy(&pHstStream->Cfg, &CfgHostAcq);
664 AssertRC(rc2);
665
666 /*
667 * Init guest stream.
668 */
669
670 /* Destroy any former mixing buffer. */
671 AudioMixBufDestroy(&pGstStream->MixBuf);
672
673 /* Set the guests's default audio data layout. */
674 pCfgHost->enmLayout = PDMAUDIOSTREAMLAYOUT_NON_INTERLEAVED;
675
676 /* Make sure to (re-)set the guest buffer's shift size. */
677 pCfgGuest->Props.cShift = PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(pCfgGuest->Props.cBits, pCfgGuest->Props.cChannels);
678
679 /* Set set guest buffer size multiplicator. */
680 const unsigned cSampleBufferGuestFactor = 10; /** @todo Make this configurable. */
681
682 LogFunc(("[%s] cSamples=%RU32 (x %u)\n", pGstStream->szName, CfgHostAcq.cSampleBufferHint, cSampleBufferGuestFactor));
683
684 rc2 = AudioMixBufInit(&pGstStream->MixBuf, pGstStream->szName, &pCfgGuest->Props,
685 CfgHostAcq.cSampleBufferHint * cSampleBufferGuestFactor);
686 AssertRC(rc2);
687
688 if (pCfgGuest->enmDir == PDMAUDIODIR_IN)
689 {
690 /* Host (Parent) -> Guest (Child). */
691 rc2 = AudioMixBufLinkTo(&pHstStream->MixBuf, &pGstStream->MixBuf);
692 AssertRC(rc2);
693 }
694 else
695 {
696 /* Guest (Parent) -> Host (Child). */
697 rc2 = AudioMixBufLinkTo(&pGstStream->MixBuf, &pHstStream->MixBuf);
698 AssertRC(rc2);
699 }
700
701 /* Make a copy of the guest stream configuration. */
702 rc2 = DrvAudioHlpStreamCfgCopy(&pGstStream->Cfg, pCfgGuest);
703 AssertRC(rc2);
704
705 if (RT_FAILURE(rc))
706 LogRel2(("Audio: Creating stream '%s' failed with %Rrc\n", pStream->szName, rc));
707
708 LogFlowFunc(("[%s] Returning %Rrc\n", pStream->szName, rc));
709 return rc;
710}
711
712/**
713 * Frees an audio stream and its allocated resources.
714 *
715 * @param pStream Audio stream to free.
716 * After this call the pointer will not be valid anymore.
717 */
718static void drvAudioStreamFree(PPDMAUDIOSTREAM pStream)
719{
720 if (!pStream)
721 return;
722
723 if (pStream->pvBackend)
724 {
725 Assert(pStream->cbBackend);
726 RTMemFree(pStream->pvBackend);
727 pStream->pvBackend = NULL;
728 }
729
730 RTMemFree(pStream);
731 pStream = NULL;
732}
733
734#ifdef VBOX_WITH_AUDIO_CALLBACKS
735/**
736 * Schedules a re-initialization of all current audio streams.
737 * The actual re-initialization will happen at some later point in time.
738 *
739 * @returns IPRT status code.
740 * @param pThis Pointer to driver instance.
741 */
742static int drvAudioScheduleReInitInternal(PDRVAUDIO pThis)
743{
744 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
745
746 LogFunc(("\n"));
747
748 /* Mark all host streams to re-initialize. */
749 PPDMAUDIOSTREAM pHstStream;
750 RTListForEach(&pThis->lstHstStreams, pHstStream, PDMAUDIOSTREAM, Node)
751 pHstStream->fStatus |= PDMAUDIOSTRMSTS_FLAG_PENDING_REINIT;
752
753# ifdef VBOX_WITH_AUDIO_ENUM
754 /* Re-enumerate all host devices as soon as possible. */
755 pThis->fEnumerateDevices = true;
756# endif
757
758 return VINF_SUCCESS;
759}
760#endif /* VBOX_WITH_AUDIO_CALLBACKS */
761
762/**
763 * Re-initializes an audio stream with its existing host and guest stream configuration.
764 * This might be the case if the backend told us we need to re-initialize because something
765 * on the host side has changed.
766 *
767 * @returns IPRT status code.
768 * @param pThis Pointer to driver instance.
769 * @param pStream Stream to re-initialize.
770 */
771static int drvAudioStreamReInitInternal(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream)
772{
773 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
774 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
775
776 LogFlowFunc(("[%s]\n", pStream->szName));
777
778 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
779 AssertPtr(pHstStream);
780
781 /*
782 * Gather current stream status.
783 */
784 bool fIsEnabled = RT_BOOL(pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_ENABLED); /* Stream is enabled? */
785
786 /*
787 * Destroy and re-create stream on backend side.
788 */
789 int rc = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_DISABLE);
790 if (RT_SUCCESS(rc))
791 {
792 rc = drvAudioStreamDestroyInternalBackend(pThis, pHstStream);
793 if (RT_SUCCESS(rc))
794 {
795 rc = drvAudioStreamCreateInternalBackend(pThis, pHstStream, &pHstStream->Cfg, NULL /* pCfgAcq */);
796 /** @todo Validate (re-)acquired configuration with pHstStream->Cfg? */
797 }
798 }
799
800 /*
801 * Restore previous stream state.
802 */
803 if (RT_SUCCESS(rc))
804 {
805 PPDMAUDIOSTREAM pGstStream = pHstStream->pPair;
806
807 if (fIsEnabled)
808 {
809 rc = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_ENABLE);
810 if (RT_SUCCESS(rc))
811 {
812 if (pGstStream)
813 {
814 /* Also reset the guest stream mixing buffer. */
815 AudioMixBufReset(&pGstStream->MixBuf);
816 }
817 }
818 }
819
820#ifdef VBOX_WITH_STATISTICS
821 /*
822 * Reset statistics.
823 */
824 if (RT_SUCCESS(rc))
825 {
826 if (pHstStream->enmDir == PDMAUDIODIR_IN)
827 {
828 STAM_COUNTER_RESET(&pHstStream->In.StatBytesElapsed);
829 STAM_COUNTER_RESET(&pHstStream->In.StatBytesTotalRead);
830 STAM_COUNTER_RESET(&pHstStream->In.StatSamplesCaptured);
831
832 if (pGstStream)
833 {
834 Assert(pGstStream->enmDir == pHstStream->enmDir);
835
836 STAM_COUNTER_RESET(&pGstStream->In.StatBytesElapsed);
837 STAM_COUNTER_RESET(&pGstStream->In.StatBytesTotalRead);
838 STAM_COUNTER_RESET(&pGstStream->In.StatSamplesCaptured);
839 }
840 }
841 else if (pHstStream->enmDir == PDMAUDIODIR_OUT)
842 {
843 STAM_COUNTER_RESET(&pHstStream->Out.StatBytesElapsed);
844 STAM_COUNTER_RESET(&pHstStream->Out.StatBytesTotalWritten);
845 STAM_COUNTER_RESET(&pHstStream->Out.StatSamplesPlayed);
846
847 if (pGstStream)
848 {
849 Assert(pGstStream->enmDir == pHstStream->enmDir);
850
851 STAM_COUNTER_RESET(&pGstStream->Out.StatBytesElapsed);
852 STAM_COUNTER_RESET(&pGstStream->Out.StatBytesTotalWritten);
853 STAM_COUNTER_RESET(&pGstStream->Out.StatSamplesPlayed);
854 }
855 }
856 else
857 AssertFailed();
858 }
859#endif
860 }
861
862 if (RT_FAILURE(rc))
863 LogRel2(("Audio: Re-initializing stream '%s' failed with %Rrc\n", pStream->szName, rc));
864
865 LogFunc(("[%s] Returning %Rrc\n", pStream->szName, rc));
866 return rc;
867}
868
869/**
870 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamWrite}
871 */
872static DECLCALLBACK(int) drvAudioStreamWrite(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream,
873 const void *pvBuf, uint32_t cbBuf, uint32_t *pcbWritten)
874{
875 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
876 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
877 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
878 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
879 /* pcbWritten is optional. */
880
881 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
882
883 AssertMsg(pStream->enmDir == PDMAUDIODIR_OUT,
884 ("Stream '%s' is not an output stream and therefore cannot be written to (direction is 0x%x)\n",
885 pStream->szName, pStream->enmDir));
886
887 uint32_t cbWritten = 0;
888
889 int rc = RTCritSectEnter(&pThis->CritSect);
890 if (RT_FAILURE(rc))
891 return rc;
892
893#ifdef VBOX_WITH_STATISTICS
894 STAM_PROFILE_ADV_START(&pThis->Stats.DelayOut, out);
895#endif
896
897#ifdef LOG_ENABLED
898 char *pszGstSts = NULL;
899 char *pszHstSts = NULL;
900#endif
901
902 do
903 {
904 if ( pThis->pHostDrvAudio
905 && pThis->pHostDrvAudio->pfnGetStatus
906 && pThis->pHostDrvAudio->pfnGetStatus(pThis->pHostDrvAudio, PDMAUDIODIR_OUT) != PDMAUDIOBACKENDSTS_RUNNING)
907 {
908 rc = VERR_NOT_AVAILABLE;
909 break;
910 }
911
912 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
913 if (!pHstStream)
914 {
915 rc = VERR_NOT_AVAILABLE;
916 break;
917 }
918
919#ifdef LOG_ENABLED
920 pszHstSts = dbgAudioStreamStatusToStr(pHstStream->fStatus);
921 AssertPtr(pszHstSts);
922#endif
923 PPDMAUDIOSTREAM pGstStream = pHstStream->pPair;
924 AssertPtr(pGstStream);
925
926#ifdef LOG_ENABLED
927 pszGstSts = dbgAudioStreamStatusToStr(pGstStream->fStatus);
928 AssertPtr(pszGstSts);
929#endif
930
931#ifdef LOG_ENABLED
932 AssertMsg(pGstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_ENABLED,
933 ("Writing to disabled guest output stream \"%s\" not possible (status is %s, host status %s)\n",
934 pGstStream->szName, pszGstSts, pszHstSts));
935#endif
936 pGstStream->Out.tsLastWriteMS = RTTimeMilliTS();
937
938 if (!AudioMixBufFreeBytes(&pGstStream->MixBuf))
939 {
940 LogRel2(("Audio: Guest stream '%s' full, expect stuttering audio output\n", pGstStream->szName));
941
942#ifdef DEBUG_andy
943 AssertMsgFailed(("[%s] Failed because guest stream full (guest status %s, host status %s): cbBuf=%RU32\n",
944 pGstStream->szName, pszGstSts, pszHstSts, cbBuf));
945#endif
946 break;
947 }
948
949 uint32_t csWritten = 0;
950 rc = AudioMixBufWriteCirc(&pGstStream->MixBuf, pvBuf, cbBuf, &csWritten);
951 if (RT_FAILURE(rc))
952 {
953 LogRel2(("Audio: Lost audio samples due to full guest stream '%s', expect stuttering audio output\n",
954 pGstStream->szName));
955 rc = VINF_SUCCESS; /* Continue. */
956 }
957
958#ifdef DEBUG_andy
959 if ( RT_FAILURE(rc)
960 || !csWritten)
961 {
962 AssertMsgFailed(("[%s] Write failed (guest status %s, host status %s): cbBuf=%RU32, csWritten=%RU32, rc=%Rrc\n",
963 pGstStream->szName, pszGstSts, pszHstSts, cbBuf, csWritten, rc));
964 }
965#endif
966
967#ifdef VBOX_WITH_STATISTICS
968 STAM_COUNTER_ADD(&pThis->Stats.TotalSamplesWritten, csWritten);
969#endif
970 uint32_t csMixed = 0;
971 if (csWritten)
972 {
973 int rc2 = AudioMixBufMixToParent(&pGstStream->MixBuf, csWritten, &csMixed);
974 if (RT_FAILURE(rc2))
975 {
976 LogRel2(("Audio: Lost audio samples (%RU32) due to full host stream '%s', expect stuttering audio output\n",
977 csWritten - csMixed, pHstStream->szName));
978 }
979
980#ifdef DEBUG_andy
981 if ( RT_FAILURE(rc2)
982 || csMixed < csWritten)
983 {
984 AssertMsgFailed(("[%s] Mixing failed (guest status %s, host status %s): cbBuf=%RU32, csWritten=%RU32, csMixed=%RU32, rc=%Rrc\n",
985 pGstStream->szName, pszGstSts, pszHstSts, cbBuf, csWritten, csMixed, rc2));
986 }
987#endif
988 if (RT_SUCCESS(rc))
989 rc = rc2;
990
991 cbWritten = AUDIOMIXBUF_S2B(&pGstStream->MixBuf, csWritten);
992
993#ifdef VBOX_WITH_STATISTICS
994 STAM_COUNTER_ADD(&pThis->Stats.TotalSamplesMixedOut, csMixed);
995 Assert(csWritten >= csMixed);
996 STAM_COUNTER_ADD(&pThis->Stats.TotalSamplesLostOut, csWritten - csMixed);
997 STAM_COUNTER_ADD(&pThis->Stats.TotalBytesWritten, cbWritten);
998 STAM_COUNTER_ADD(&pGstStream->Out.StatBytesTotalWritten, cbWritten);
999#endif
1000 }
1001
1002 Log3Func(("[%s] cbBuf=%RU32, cUsed=%RU32, cLive=%RU32, cWritten=%RU32, cMixed=%RU32, rc=%Rrc\n",
1003 pGstStream->szName,
1004 cbBuf, AudioMixBufUsed(&pGstStream->MixBuf), AudioMixBufLive(&pGstStream->MixBuf), csWritten, csMixed, rc));
1005
1006 } while (0);
1007
1008#ifdef LOG_ENABLED
1009 RTStrFree(pszHstSts);
1010 RTStrFree(pszGstSts);
1011#endif
1012
1013 int rc2 = RTCritSectLeave(&pThis->CritSect);
1014 if (RT_SUCCESS(rc))
1015 rc = rc2;
1016
1017 if (RT_SUCCESS(rc))
1018 {
1019 if (pcbWritten)
1020 *pcbWritten = cbWritten;
1021 }
1022
1023 return rc;
1024}
1025
1026/**
1027 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamRetain}
1028 */
1029static DECLCALLBACK(uint32_t) drvAudioStreamRetain(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream)
1030{
1031 AssertPtrReturn(pInterface, UINT32_MAX);
1032 AssertPtrReturn(pStream, UINT32_MAX);
1033
1034 NOREF(pInterface);
1035
1036 return ++pStream->cRefs;
1037}
1038
1039/**
1040 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamRelease}
1041 */
1042static DECLCALLBACK(uint32_t) drvAudioStreamRelease(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream)
1043{
1044 AssertPtrReturn(pInterface, UINT32_MAX);
1045 AssertPtrReturn(pStream, UINT32_MAX);
1046
1047 NOREF(pInterface);
1048
1049 if (pStream->cRefs > 1) /* 1 reference always is kept by this audio driver. */
1050 pStream->cRefs--;
1051
1052 return pStream->cRefs;
1053}
1054
1055/**
1056 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamIterate}
1057 */
1058static DECLCALLBACK(int) drvAudioStreamIterate(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream)
1059{
1060 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1061 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1062 /* pcData is optional. */
1063
1064 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1065
1066 int rc = RTCritSectEnter(&pThis->CritSect);
1067 if (RT_FAILURE(rc))
1068 return rc;
1069
1070 rc = drvAudioStreamIterateInternal(pThis, pStream);
1071
1072 int rc2 = RTCritSectLeave(&pThis->CritSect);
1073 if (RT_SUCCESS(rc))
1074 rc = rc2;
1075
1076 if (RT_FAILURE(rc))
1077 LogFlowFuncLeaveRC(rc);
1078
1079 return rc;
1080}
1081
1082/**
1083 * Does one iteration of an audio stream.
1084 * This function gives the backend the chance of iterating / altering data and
1085 * does the actual mixing between the guest <-> host mixing buffers.
1086 *
1087 * @returns IPRT status code.
1088 * @param pThis Pointer to driver instance.
1089 * @param pStream Stream to iterate.
1090 */
1091static int drvAudioStreamIterateInternal(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream)
1092{
1093 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1094
1095 if (!pStream)
1096 return VINF_SUCCESS;
1097
1098 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
1099 AssertPtr(pHstStream);
1100 PPDMAUDIOSTREAM pGstStream = pHstStream ? pHstStream->pPair : NULL;
1101 AssertPtr(pGstStream);
1102
1103 int rc;
1104
1105 /* Is the stream scheduled for re-initialization? Do so now. */
1106 if (pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_PENDING_REINIT)
1107 {
1108#ifdef VBOX_WITH_AUDIO_ENUM
1109 if (pThis->fEnumerateDevices)
1110 {
1111 /* Re-enumerate all host devices. */
1112 drvAudioDevicesEnumerateInternal(pThis, true /* fLog */, NULL /* pDevEnum */);
1113
1114 pThis->fEnumerateDevices = false;
1115 }
1116#endif /* VBOX_WITH_AUDIO_ENUM */
1117
1118 /* Remove the pending re-init flag in any case, regardless whether the actual re-initialization succeeded
1119 * or not. If it failed, the backend needs to notify us again to try again at some later point in time. */
1120 pHstStream->fStatus &= ~PDMAUDIOSTRMSTS_FLAG_PENDING_REINIT;
1121
1122 rc = drvAudioStreamReInitInternal(pThis, pStream);
1123 if (RT_FAILURE(rc))
1124 return rc;
1125 }
1126
1127#ifdef LOG_ENABLED
1128 char *pszHstSts = dbgAudioStreamStatusToStr(pHstStream->fStatus);
1129 Log3Func(("[%s] fStatus=%s\n", pHstStream->szName, pszHstSts));
1130 RTStrFree(pszHstSts);
1131#endif /* LOG_ENABLED */
1132
1133 /* Not enabled or paused? Skip iteration. */
1134 if ( !(pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_ENABLED)
1135 || (pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_PAUSED))
1136 {
1137 return VINF_SUCCESS;
1138 }
1139
1140 /* Whether to try closing a pending to close stream. */
1141 bool fTryClosePending = false;
1142
1143 do
1144 {
1145 uint32_t csMixed = 0;
1146
1147 rc = pThis->pHostDrvAudio->pfnStreamIterate(pThis->pHostDrvAudio, pHstStream->pvBackend);
1148 if (RT_FAILURE(rc))
1149 break;
1150
1151 if (pHstStream->enmDir == PDMAUDIODIR_IN)
1152 {
1153 /* Has the host captured any samples which were not mixed to the guest side yet? */
1154 uint32_t csCaptured = AudioMixBufUsed(&pHstStream->MixBuf);
1155 if (csCaptured)
1156 {
1157 /* When capturing samples, the guest is the parent while the host is the child.
1158 * So try mixing not yet mixed host-side samples to the guest-side buffer. */
1159 rc = AudioMixBufMixToParent(&pHstStream->MixBuf, csCaptured, &csMixed);
1160 if (RT_FAILURE(rc))
1161 {
1162 if (rc == VERR_BUFFER_OVERFLOW)
1163 LogRel2(("Audio: Guest input stream '%s' full, expect stuttering audio capture\n", pGstStream->szName));
1164 else
1165 LogRel2(("Audio: Mixing to guest input stream '%s' failed: %Rrc\n", pGstStream->szName, rc));
1166#ifdef DEBUG_andy_disabled
1167 AssertFailed();
1168#endif
1169 }
1170
1171#ifdef VBOX_WITH_STATISTICS
1172 STAM_COUNTER_ADD(&pThis->Stats.TotalSamplesMixedIn, csMixed);
1173 Assert(csCaptured >= csMixed);
1174 STAM_COUNTER_ADD(&pThis->Stats.TotalSamplesLostIn, csCaptured - csMixed);
1175#endif
1176 Log3Func(("[%s] %RU32/%RU32 input samples mixed, rc=%Rrc\n", pHstStream->szName, csMixed, csCaptured, rc));
1177 }
1178 else
1179 {
1180 fTryClosePending = true;
1181 }
1182 }
1183 else if (pHstStream->enmDir == PDMAUDIODIR_OUT)
1184 {
1185 /* Nothing to do here (yet). */
1186 }
1187 else
1188 AssertFailedStmt(rc = VERR_NOT_IMPLEMENTED);
1189
1190 if (fTryClosePending)
1191 {
1192 /* Has the host stream marked as disabled but there still were guest streams relying
1193 * on it? Check if the stream now can be closed and do so, if possible. */
1194 if (pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_PENDING_DISABLE)
1195 {
1196 LogFunc(("[%s] Closing pending stream\n", pHstStream->szName));
1197 rc = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_DISABLE);
1198 if (RT_SUCCESS(rc))
1199 {
1200 pHstStream->fStatus &= ~PDMAUDIOSTRMSTS_FLAG_PENDING_DISABLE;
1201 }
1202 else
1203 LogFunc(("[%s] Backend vetoed against closing pending output stream, rc=%Rrc\n", pHstStream->szName, rc));
1204 }
1205 }
1206
1207 } while (0);
1208
1209 /* Update timestamps. */
1210 pHstStream->tsLastIterateMS = RTTimeMilliTS();
1211 pGstStream->tsLastIterateMS = RTTimeMilliTS();
1212
1213 if (RT_FAILURE(rc))
1214 LogFunc(("[%s] Failed with %Rrc\n", pHstStream->szName, rc));
1215
1216 return rc;
1217}
1218
1219/**
1220 * Links an audio stream to another audio stream (pair).
1221 *
1222 * @returns IPRT status code.
1223 * @param pStream Stream to handle linking for.
1224 * @param pPair Stream to link pStream to. Specify NULL to unlink pStream from a former linked stream.
1225 */
1226static int drvAudioStreamLinkToInternal(PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAM pPair)
1227{
1228 if (pPair) /* Link. */
1229 {
1230 pStream->pPair = pPair;
1231 pPair->pPair = pStream;
1232
1233 LogRel2(("Audio: Linked audio stream '%s' to '%s'\n", pStream->szName, pPair->szName));
1234 }
1235 else /* Unlink. */
1236 {
1237 if (pStream->pPair)
1238 {
1239 LogRel2(("Audio: Unlinked pair '%s' from stream '%s'\n", pStream->pPair->szName, pStream->szName));
1240
1241 AssertMsg(pStream->pPair->pPair == pStream,
1242 ("Pair '%s' is not linked to '%s' (linked to '%s')\n",
1243 pStream->pPair->szName, pStream->szName, pStream->pPair->pPair ? pStream->pPair->pPair->szName : "<NONE>"));
1244
1245 pStream->pPair->pPair = NULL; /* Also make sure to unlink the pair from pStream */
1246 pStream->pPair = NULL;
1247 }
1248 }
1249
1250 return VINF_SUCCESS;
1251}
1252
1253/**
1254 * Plays an audio host output stream which has been configured for non-interleaved (layout) data.
1255 *
1256 * @return IPRT status code.
1257 * @param pThis Pointer to driver instance.
1258 * @param pHstStream Host stream to play.
1259 * @param csToPlay Number of audio samples to play.
1260 * @param pcsPlayed Returns number of audio samples played. Optional.
1261 */
1262static int drvAudioStreamPlayNonInterleaved(PDRVAUDIO pThis,
1263 PPDMAUDIOSTREAM pHstStream, uint32_t csToPlay, uint32_t *pcsPlayed)
1264{
1265 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1266 AssertPtrReturn(pHstStream, VERR_INVALID_POINTER);
1267 /* pcsPlayed is optional. */
1268
1269 /* Sanity. */
1270 Assert(pHstStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
1271 Assert(pHstStream->enmDir == PDMAUDIODIR_OUT);
1272 Assert(pHstStream->Cfg.enmLayout == PDMAUDIOSTREAMLAYOUT_NON_INTERLEAVED);
1273
1274 if (!csToPlay)
1275 {
1276 if (pcsPlayed)
1277 *pcsPlayed = 0;
1278 return VINF_SUCCESS;
1279 }
1280
1281 int rc = VINF_SUCCESS;
1282
1283 uint32_t csPlayedTotal = 0;
1284
1285 AssertPtr(pThis->pHostDrvAudio->pfnStreamGetWritable);
1286 uint32_t cbWritable = pThis->pHostDrvAudio->pfnStreamGetWritable(pThis->pHostDrvAudio, pHstStream->pvBackend);
1287 if (cbWritable)
1288 {
1289 if (csToPlay > AUDIOMIXBUF_B2S(&pHstStream->MixBuf, cbWritable)) /* More samples available than we can write? Limit. */
1290 csToPlay = AUDIOMIXBUF_B2S(&pHstStream->MixBuf, cbWritable);
1291
1292 if (csToPlay)
1293 {
1294 uint8_t auBuf[256]; /** @todo Get rid of this here. */
1295
1296 uint32_t cbLeft = AUDIOMIXBUF_S2B(&pHstStream->MixBuf, csToPlay);
1297 uint32_t cbChunk = sizeof(auBuf);
1298
1299 while (cbLeft)
1300 {
1301 uint32_t csRead = 0;
1302 rc = AudioMixBufReadCirc(&pHstStream->MixBuf, auBuf, RT_MIN(cbChunk, cbLeft), &csRead);
1303 if ( !csRead
1304 || RT_FAILURE(rc))
1305 {
1306 break;
1307 }
1308
1309 uint32_t cbRead = AUDIOMIXBUF_S2B(&pHstStream->MixBuf, csRead);
1310 Assert(cbRead <= cbChunk);
1311
1312 uint32_t cbPlayed = 0;
1313 rc = pThis->pHostDrvAudio->pfnStreamPlay(pThis->pHostDrvAudio, pHstStream->pvBackend,
1314 auBuf, cbRead, &cbPlayed);
1315 if (RT_FAILURE(rc))
1316 break;
1317
1318 AssertMsg(cbPlayed <= cbRead, ("Played more than available (%RU32 available but got %RU32)\n", cbRead, cbPlayed));
1319 AssertMsg(cbPlayed % 2 == 0,
1320 ("Backend for stream '%s' returned uneven played bytes count (csRead=%RU32, cbPlayed=%RU32)\n",
1321 pHstStream->szName, csRead, cbPlayed));
1322
1323 csPlayedTotal += AUDIOMIXBUF_B2S(&pHstStream->MixBuf, cbPlayed);
1324 Assert(cbLeft >= cbPlayed);
1325 cbLeft -= cbPlayed;
1326 }
1327 }
1328 }
1329
1330 Log3Func(("[%s] Played %RU32/%RU32 samples, rc=%Rrc\n", pHstStream->szName, csPlayedTotal, csToPlay, rc));
1331
1332 if (RT_SUCCESS(rc))
1333 {
1334 if (pcsPlayed)
1335 *pcsPlayed = csPlayedTotal;
1336 }
1337
1338 return rc;
1339}
1340
1341/**
1342 * Plays an audio host output stream which has been configured for raw audio (layout) data.
1343 *
1344 * @return IPRT status code.
1345 * @param pThis Pointer to driver instance.
1346 * @param pHstStream Host stream to play.
1347 * @param csToPlay Number of audio samples to play.
1348 * @param pcsPlayed Returns number of audio samples played. Optional.
1349 */
1350static int drvAudioStreamPlayRaw(PDRVAUDIO pThis,
1351 PPDMAUDIOSTREAM pHstStream, uint32_t csToPlay, uint32_t *pcsPlayed)
1352{
1353 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1354 AssertPtrReturn(pHstStream, VERR_INVALID_POINTER);
1355 /* pcsPlayed is optional. */
1356
1357 /* Sanity. */
1358 Assert(pHstStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
1359 Assert(pHstStream->enmDir == PDMAUDIODIR_OUT);
1360 Assert(pHstStream->Cfg.enmLayout == PDMAUDIOSTREAMLAYOUT_RAW);
1361
1362 if (!csToPlay)
1363 {
1364 if (pcsPlayed)
1365 *pcsPlayed = 0;
1366 return VINF_SUCCESS;
1367 }
1368
1369 int rc = VINF_SUCCESS;
1370
1371 uint32_t csPlayedTotal = 0;
1372
1373 AssertPtr(pThis->pHostDrvAudio->pfnStreamGetWritable);
1374 uint32_t csWritable = pThis->pHostDrvAudio->pfnStreamGetWritable(pThis->pHostDrvAudio, pHstStream->pvBackend);
1375 if (csWritable)
1376 {
1377 if (csToPlay > csWritable) /* More samples available than we can write? Limit. */
1378 csToPlay = csWritable;
1379
1380 PDMAUDIOSAMPLE aSampleBuf[256]; /** @todo Get rid of this here. */
1381
1382 uint32_t csLeft = csToPlay;
1383 while (csLeft)
1384 {
1385 uint32_t csRead = 0;
1386 rc = AudioMixBufPeek(&pHstStream->MixBuf, csLeft, aSampleBuf,
1387 RT_MIN(csLeft, RT_ELEMENTS(aSampleBuf)), &csRead);
1388
1389 if ( RT_SUCCESS(rc)
1390 && csRead)
1391 {
1392 uint32_t csPlayedChunk;
1393
1394 Assert(csRead <= RT_ELEMENTS(aSampleBuf));
1395 rc = pThis->pHostDrvAudio->pfnStreamPlay(pThis->pHostDrvAudio, pHstStream->pvBackend,
1396 aSampleBuf, csRead, &csPlayedChunk);
1397 if (RT_FAILURE(rc))
1398 break;
1399
1400 csPlayedTotal += csPlayedChunk;
1401 Assert(csPlayedTotal <= csToPlay);
1402
1403 Assert(csLeft >= csRead);
1404 csLeft -= csRead;
1405 }
1406 else if (RT_FAILURE(rc))
1407 break;
1408 }
1409 }
1410
1411 Log3Func(("[%s] Played %RU32/%RU32 samples, rc=%Rrc\n", pHstStream->szName, csPlayedTotal, csToPlay, rc));
1412
1413 if (RT_SUCCESS(rc))
1414 {
1415 if (pcsPlayed)
1416 *pcsPlayed = csPlayedTotal;
1417
1418 }
1419
1420 return rc;
1421}
1422
1423/**
1424 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamPlay}
1425 */
1426static DECLCALLBACK(int) drvAudioStreamPlay(PPDMIAUDIOCONNECTOR pInterface,
1427 PPDMAUDIOSTREAM pStream, uint32_t *pcSamplesPlayed)
1428{
1429 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1430 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1431 /* pcSamplesPlayed is optional. */
1432
1433 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1434
1435 int rc = RTCritSectEnter(&pThis->CritSect);
1436 if (RT_FAILURE(rc))
1437 return rc;
1438
1439 AssertMsg(pStream->enmDir == PDMAUDIODIR_OUT,
1440 ("Stream '%s' is not an output stream and therefore cannot be played back (direction is 0x%x)\n",
1441 pStream->szName, pStream->enmDir));
1442
1443 uint32_t csPlayedTotal = 0;
1444
1445 do
1446 {
1447 if (!pThis->pHostDrvAudio)
1448 {
1449 rc = VERR_NOT_AVAILABLE;
1450 break;
1451 }
1452
1453 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
1454 AssertPtr(pHstStream);
1455 PPDMAUDIOSTREAM pGstStream = pHstStream ? pHstStream->pPair : NULL;
1456 AssertPtr(pGstStream);
1457
1458 AssertReleaseMsgBreakStmt(pHstStream != NULL,
1459 ("[%s] Host stream is NULL (cRefs=%RU32, fStatus=%x, enmCtx=%ld)\n",
1460 pStream->szName, pStream->cRefs, pStream->fStatus, pStream->enmCtx),
1461 rc = VERR_NOT_AVAILABLE);
1462 AssertReleaseMsgBreakStmt(pGstStream != NULL,
1463 ("[%s] Guest stream is NULL (cRefs=%RU32, fStatus=%x, enmCtx=%ld)\n",
1464 pStream->szName, pStream->cRefs, pStream->fStatus, pStream->enmCtx),
1465 rc = VERR_NOT_AVAILABLE);
1466
1467 /*
1468 * Check if the backend is ready to operate.
1469 */
1470
1471 AssertPtr(pThis->pHostDrvAudio->pfnStreamGetStatus);
1472 PDMAUDIOSTRMSTS stsBackend = pThis->pHostDrvAudio->pfnStreamGetStatus(pThis->pHostDrvAudio, pHstStream->pvBackend);
1473#ifdef LOG_ENABLED
1474 char *pszBackendSts = dbgAudioStreamStatusToStr(stsBackend);
1475 Log3Func(("[%s] Start: stsBackend=%s\n", pHstStream->szName, pszBackendSts));
1476 RTStrFree(pszBackendSts);
1477#endif /* LOG_ENABLED */
1478 if (!(stsBackend & PDMAUDIOSTRMSTS_FLAG_ENABLED)) /* Backend disabled? Bail out. */
1479 break;
1480
1481 uint32_t csToPlay = AudioMixBufLive(&pHstStream->MixBuf);
1482
1483 if (RT_LIKELY(pHstStream->Cfg.enmLayout == PDMAUDIOSTREAMLAYOUT_NON_INTERLEAVED))
1484 {
1485 rc = drvAudioStreamPlayNonInterleaved(pThis, pHstStream, csToPlay, &csPlayedTotal);
1486 }
1487 else if (pHstStream->Cfg.enmLayout == PDMAUDIOSTREAMLAYOUT_RAW)
1488 {
1489 rc = drvAudioStreamPlayRaw(pThis, pHstStream, csToPlay, &csPlayedTotal);
1490 }
1491 else
1492 AssertFailedStmt(rc = VERR_NOT_IMPLEMENTED);
1493
1494 uint32_t csLive = 0;
1495
1496 if (RT_SUCCESS(rc))
1497 {
1498 AudioMixBufFinish(&pHstStream->MixBuf, csPlayedTotal);
1499
1500#ifdef VBOX_WITH_STATISTICS
1501 STAM_COUNTER_ADD(&pThis->Stats.TotalSamplesOut, csPlayedTotal);
1502 STAM_PROFILE_ADV_STOP(&pThis->Stats.DelayOut, out);
1503 STAM_COUNTER_ADD(&pHstStream->Out.StatSamplesPlayed, csPlayedTotal);
1504#endif
1505 csLive = AudioMixBufLive(&pHstStream->MixBuf);
1506 }
1507
1508#ifdef LOG_ENABLED
1509 pszBackendSts = dbgAudioStreamStatusToStr(stsBackend);
1510 Log3Func(("[%s] End: stsBackend=%s, csLive=%RU32, csPlayedTotal=%RU32, rc=%Rrc\n",
1511 pHstStream->szName, pszBackendSts, csLive, csPlayedTotal, rc));
1512 RTStrFree(pszBackendSts);
1513#endif /* LOG_ENABLED */
1514
1515 if (!csLive)
1516 {
1517 /* Has the host stream marked as disabled but there still were guest streams relying
1518 * on it? Check if the stream now can be closed and do so, if possible. */
1519 if (pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_PENDING_DISABLE)
1520 {
1521 LogFunc(("[%s] Closing pending stream\n", pHstStream->szName));
1522 rc = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_DISABLE);
1523 if (RT_SUCCESS(rc))
1524 {
1525 pHstStream->fStatus &= ~PDMAUDIOSTRMSTS_FLAG_PENDING_DISABLE;
1526 }
1527 else
1528 LogFunc(("[%s] Backend vetoed against closing output stream, rc=%Rrc\n", pHstStream->szName, rc));
1529 }
1530 }
1531
1532 } while (0);
1533
1534 int rc2 = RTCritSectLeave(&pThis->CritSect);
1535 if (RT_SUCCESS(rc))
1536 rc = rc2;
1537
1538 if (RT_SUCCESS(rc))
1539 {
1540 if (pcSamplesPlayed)
1541 *pcSamplesPlayed = csPlayedTotal;
1542 }
1543
1544 if (RT_FAILURE(rc))
1545 LogFlowFunc(("[%s] Failed with %Rrc\n", pStream->szName, rc));
1546
1547 return rc;
1548}
1549
1550static int drvAudioStreamCaptureNonInterleaved(PDRVAUDIO pThis, PPDMAUDIOSTREAM pHstStream, uint32_t *pcsCaptured)
1551{
1552 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1553 AssertPtrReturn(pHstStream, VERR_INVALID_POINTER);
1554 /* pcsCaptured is optional. */
1555
1556 /* Sanity. */
1557 Assert(pHstStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
1558 Assert(pHstStream->enmDir == PDMAUDIODIR_IN);
1559 Assert(pHstStream->Cfg.enmLayout == PDMAUDIOSTREAMLAYOUT_NON_INTERLEAVED);
1560
1561 int rc = VINF_SUCCESS;
1562
1563 uint32_t csCapturedTotal = 0;
1564
1565 AssertPtr(pThis->pHostDrvAudio->pfnStreamGetReadable);
1566
1567 uint8_t auBuf[_1K]; /** @todo Get rid of this. */
1568 size_t cbBuf = sizeof(auBuf);
1569
1570 for (;;)
1571 {
1572 uint32_t cbReadable = pThis->pHostDrvAudio->pfnStreamGetReadable(pThis->pHostDrvAudio, pHstStream->pvBackend);
1573 if (!cbReadable)
1574 break;
1575
1576 uint32_t cbFree = AUDIOMIXBUF_S2B(&pHstStream->MixBuf, AudioMixBufFree(&pHstStream->MixBuf));
1577 if (!cbFree)
1578 break;
1579
1580 if (cbFree < cbReadable) /* More data captured than we can read? */
1581 {
1582 /** @todo Warn? */
1583 }
1584
1585 if (cbFree > cbBuf) /* Limit to buffer size. */
1586 cbFree = (uint32_t)cbBuf;
1587
1588 uint32_t cbCaptured;
1589 rc = pThis->pHostDrvAudio->pfnStreamCapture(pThis->pHostDrvAudio, pHstStream->pvBackend,
1590 auBuf, cbFree, &cbCaptured);
1591 if (RT_FAILURE(rc))
1592 {
1593 int rc2 = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_DISABLE);
1594 AssertRC(rc2);
1595 }
1596 else if (cbCaptured)
1597 {
1598 Assert(cbCaptured <= cbBuf);
1599 if (cbCaptured > cbBuf) /* Paranoia. */
1600 cbCaptured = (uint32_t)cbBuf;
1601
1602 uint32_t csCaptured = 0;
1603 rc = AudioMixBufWriteCirc(&pHstStream->MixBuf, auBuf, cbCaptured, &csCaptured);
1604 if (RT_SUCCESS(rc))
1605 csCapturedTotal += csCaptured;
1606 }
1607 else /* Nothing captured -- bail out. */
1608 {
1609#ifdef VBOX_STRICT
1610 AssertMsgFailed(("[%s] Captured anything, even if announced readable data (%RU32 captured samples so far) -- backend buggy?\n",
1611 pHstStream->szName, csCapturedTotal));
1612#endif
1613 break;
1614 }
1615
1616 if (RT_FAILURE(rc))
1617 break;
1618 }
1619
1620 Log2Func(("[%s] %RU32 samples captured, rc=%Rrc\n", pHstStream->szName, csCapturedTotal, rc));
1621
1622 if (pcsCaptured)
1623 *pcsCaptured = csCapturedTotal;
1624
1625 return rc;
1626}
1627
1628static int drvAudioStreamCaptureRaw(PDRVAUDIO pThis, PPDMAUDIOSTREAM pHstStream, uint32_t *pcsCaptured)
1629{
1630 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1631 AssertPtrReturn(pHstStream, VERR_INVALID_POINTER);
1632 /* pcsCaptured is optional. */
1633
1634 /* Sanity. */
1635 Assert(pHstStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
1636 Assert(pHstStream->enmDir == PDMAUDIODIR_IN);
1637 Assert(pHstStream->Cfg.enmLayout == PDMAUDIOSTREAMLAYOUT_RAW);
1638
1639 int rc = VINF_SUCCESS;
1640
1641 uint32_t csCapturedTotal = 0;
1642
1643 AssertPtr(pThis->pHostDrvAudio->pfnStreamGetReadable);
1644
1645 for (;;)
1646 {
1647 uint32_t csReadable = pThis->pHostDrvAudio->pfnStreamGetReadable(pThis->pHostDrvAudio, pHstStream->pvBackend);
1648 if (!csReadable)
1649 break;
1650
1651 uint32_t csFree = AudioMixBufFree(&pHstStream->MixBuf);
1652 if (!csFree)
1653 break;
1654
1655 if (csFree < csReadable) /* More data captured than we can read? */
1656 {
1657 /** @todo Warn? */
1658 }
1659
1660 PPDMAUDIOSAMPLE paSamples;
1661 uint32_t csWritable;
1662 rc = AudioMixBufPeekMutable(&pHstStream->MixBuf, csReadable, &paSamples, &csWritable);
1663 if (RT_FAILURE(rc))
1664 break;
1665
1666 uint32_t csCaptured;
1667 rc = pThis->pHostDrvAudio->pfnStreamCapture(pThis->pHostDrvAudio, pHstStream->pvBackend,
1668 paSamples, csWritable, &csCaptured);
1669 if (RT_FAILURE(rc))
1670 {
1671 int rc2 = drvAudioStreamControlInternalBackend(pThis, pHstStream, PDMAUDIOSTREAMCMD_DISABLE);
1672 AssertRC(rc2);
1673 }
1674 else if (csCaptured)
1675 {
1676 Assert(csCaptured <= csWritable);
1677 if (csCaptured > csWritable) /* Paranoia. */
1678 csCaptured = csWritable;
1679
1680 csCapturedTotal += csCaptured;
1681 }
1682 else /* Nothing captured -- bail out. */
1683 {
1684#ifdef VBOX_STRICT
1685 AssertMsgFailed(("[%s] Captured anything, even if announced readable data (%RU32 captured samples so far) -- backend buggy?\n",
1686 pHstStream->szName, csCapturedTotal));
1687#endif
1688 break;
1689 }
1690
1691 if (RT_FAILURE(rc))
1692 break;
1693 }
1694
1695 Log2Func(("[%s] %RU32 samples captured, rc=%Rrc\n", pHstStream->szName, csCapturedTotal, rc));
1696
1697 if (pcsCaptured)
1698 *pcsCaptured = csCapturedTotal;
1699
1700 return rc;
1701}
1702
1703/**
1704 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamCapture}
1705 */
1706static DECLCALLBACK(int) drvAudioStreamCapture(PPDMIAUDIOCONNECTOR pInterface,
1707 PPDMAUDIOSTREAM pStream, uint32_t *pcSamplesCaptured)
1708{
1709 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1710
1711 int rc = RTCritSectEnter(&pThis->CritSect);
1712 if (RT_FAILURE(rc))
1713 return rc;
1714
1715 AssertMsg(pStream->enmDir == PDMAUDIODIR_IN,
1716 ("Stream '%s' is not an input stream and therefore cannot be captured (direction is 0x%x)\n",
1717 pStream->szName, pStream->enmDir));
1718
1719 uint32_t csCaptured = 0;
1720
1721 do
1722 {
1723 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
1724 AssertPtr(pHstStream);
1725 PPDMAUDIOSTREAM pGstStream = pHstStream ? pHstStream->pPair : NULL;
1726 AssertPtr(pGstStream);
1727
1728 AssertReleaseMsgBreakStmt(pHstStream != NULL,
1729 ("[%s] Host stream is NULL (cRefs=%RU32, fStatus=%x, enmCtx=%ld)\n",
1730 pStream->szName, pStream->cRefs, pStream->fStatus, pStream->enmCtx),
1731 rc = VERR_NOT_AVAILABLE);
1732 AssertReleaseMsgBreakStmt(pGstStream != NULL,
1733 ("[%s] Guest stream is NULL (cRefs=%RU32, fStatus=%x, enmCtx=%ld)\n",
1734 pStream->szName, pStream->cRefs, pStream->fStatus, pStream->enmCtx),
1735 rc = VERR_NOT_AVAILABLE);
1736
1737 /*
1738 * Check if the backend is ready to operate.
1739 */
1740
1741 AssertPtr(pThis->pHostDrvAudio->pfnStreamGetStatus);
1742 PDMAUDIOSTRMSTS stsBackend = pThis->pHostDrvAudio->pfnStreamGetStatus(pThis->pHostDrvAudio, pHstStream->pvBackend);
1743#ifdef LOG_ENABLED
1744 char *pszBackendSts = dbgAudioStreamStatusToStr(stsBackend);
1745 Log3Func(("[%s] Start: stsBackend=%s\n", pHstStream->szName, pszBackendSts));
1746 RTStrFree(pszBackendSts);
1747#endif /* LOG_ENABLED */
1748 if (!(stsBackend & PDMAUDIOSTRMSTS_FLAG_ENABLED)) /* Backend disabled? Bail out. */
1749 break;
1750
1751 /*
1752 * Do the actual capturing.
1753 */
1754
1755 if (RT_LIKELY(pHstStream->Cfg.enmLayout == PDMAUDIOSTREAMLAYOUT_NON_INTERLEAVED))
1756 {
1757 rc = drvAudioStreamCaptureNonInterleaved(pThis, pHstStream, &csCaptured);
1758 }
1759 else if (pHstStream->Cfg.enmLayout == PDMAUDIOSTREAMLAYOUT_RAW)
1760 {
1761 rc = drvAudioStreamCaptureRaw(pThis, pHstStream, &csCaptured);
1762 }
1763 else
1764 AssertFailedStmt(rc = VERR_NOT_IMPLEMENTED);
1765
1766#ifdef LOG_ENABLED
1767 pszBackendSts = dbgAudioStreamStatusToStr(stsBackend);
1768 Log3Func(("[%s] End: stsBackend=%s, csCaptured=%RU32, rc=%Rrc\n",
1769 pHstStream->szName, pszBackendSts, csCaptured, rc));
1770 RTStrFree(pszBackendSts);
1771#endif /* LOG_ENABLED */
1772
1773 if (RT_SUCCESS(rc))
1774 {
1775#ifdef VBOX_WITH_STATISTICS
1776 STAM_COUNTER_ADD(&pThis->Stats.TotalSamplesIn, csCaptured);
1777 STAM_COUNTER_ADD(&pHstStream->In.StatSamplesCaptured, csCaptured);
1778#endif
1779 }
1780 else
1781 LogRel2(("Audio: Capturing stream '%s' failed with %Rrc\n", pHstStream->szName, rc));
1782
1783 } while (0);
1784
1785 if (pcSamplesCaptured)
1786 *pcSamplesCaptured = csCaptured;
1787
1788 int rc2 = RTCritSectLeave(&pThis->CritSect);
1789 if (RT_SUCCESS(rc))
1790 rc = rc2;
1791
1792 if (RT_FAILURE(rc))
1793 LogFlowFuncLeaveRC(rc);
1794
1795 return rc;
1796}
1797
1798#ifdef VBOX_WITH_AUDIO_DEVICE_CALLBACKS
1799/**
1800 * Duplicates an audio callback.
1801 *
1802 * @returns Pointer to duplicated callback, or NULL on failure.
1803 * @param pCB Callback to duplicate.
1804 */
1805static PPDMAUDIOCALLBACK drvAudioCallbackDuplicate(PPDMAUDIOCALLBACK pCB)
1806{
1807 AssertPtrReturn(pCB, NULL);
1808
1809 PPDMAUDIOCALLBACK pCBCopy = (PPDMAUDIOCALLBACK)RTMemDup((void *)pCB, sizeof(PDMAUDIOCALLBACK));
1810 if (!pCBCopy)
1811 return NULL;
1812
1813 if (pCB->pvCtx)
1814 {
1815 pCBCopy->pvCtx = RTMemDup(pCB->pvCtx, pCB->cbCtx);
1816 if (!pCBCopy->pvCtx)
1817 {
1818 RTMemFree(pCBCopy);
1819 return NULL;
1820 }
1821
1822 pCBCopy->cbCtx = pCB->cbCtx;
1823 }
1824
1825 return pCBCopy;
1826}
1827
1828/**
1829 * Destroys a given callback.
1830 *
1831 * @param pCB Callback to destroy.
1832 */
1833static void drvAudioCallbackDestroy(PPDMAUDIOCALLBACK pCB)
1834{
1835 if (!pCB)
1836 return;
1837
1838 RTListNodeRemove(&pCB->Node);
1839 if (pCB->pvCtx)
1840 {
1841 Assert(pCB->cbCtx);
1842 RTMemFree(pCB->pvCtx);
1843 }
1844 RTMemFree(pCB);
1845}
1846
1847/**
1848 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnRegisterCallbacks}
1849 */
1850static DECLCALLBACK(int) drvAudioRegisterCallbacks(PPDMIAUDIOCONNECTOR pInterface,
1851 PPDMAUDIOCALLBACK paCallbacks, size_t cCallbacks)
1852{
1853 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1854 AssertPtrReturn(paCallbacks, VERR_INVALID_POINTER);
1855 AssertReturn(cCallbacks, VERR_INVALID_PARAMETER);
1856
1857 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1858
1859 int rc = RTCritSectEnter(&pThis->CritSect);
1860 if (RT_FAILURE(rc))
1861 return rc;
1862
1863 for (size_t i = 0; i < cCallbacks; i++)
1864 {
1865 PPDMAUDIOCALLBACK pCB = drvAudioCallbackDuplicate(&paCallbacks[i]);
1866 if (!pCB)
1867 {
1868 rc = VERR_NO_MEMORY;
1869 break;
1870 }
1871
1872 switch (pCB->enmType)
1873 {
1874 case PDMAUDIOCBTYPE_DATA_INPUT:
1875 RTListAppend(&pThis->lstCBIn, &pCB->Node);
1876 break;
1877
1878 case PDMAUDIOCBTYPE_DATA_OUTPUT:
1879 RTListAppend(&pThis->lstCBOut, &pCB->Node);
1880 break;
1881
1882 default:
1883 AssertMsgFailed(("Not supported\n"));
1884 break;
1885 }
1886 }
1887
1888 /** @todo Undo allocations on error. */
1889
1890 int rc2 = RTCritSectLeave(&pThis->CritSect);
1891 if (RT_SUCCESS(rc))
1892 rc = rc2;
1893
1894 return rc;
1895}
1896
1897/**
1898 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnCallback}
1899 */
1900static DECLCALLBACK(int) drvAudioCallback(PPDMIAUDIOCONNECTOR pInterface, PDMAUDIOCBTYPE enmType,
1901 void *pvUser, size_t cbUser)
1902{
1903 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1904 AssertPtrReturn(pvUser, VERR_INVALID_POINTER);
1905 AssertReturn(cbUser, VERR_INVALID_PARAMETER);
1906
1907 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1908 PRTLISTANCHOR pListAnchor = NULL;
1909
1910 switch (enmType)
1911 {
1912 case PDMAUDIOCBTYPE_DATA_INPUT:
1913 pListAnchor = &pThis->lstCBIn;
1914 break;
1915
1916 case PDMAUDIOCBTYPE_DATA_OUTPUT:
1917 pListAnchor = &pThis->lstCBOut;
1918 break;
1919
1920 default:
1921 AssertMsgFailed(("Not supported\n"));
1922 break;
1923 }
1924
1925 if (pListAnchor)
1926 {
1927 PPDMAUDIOCALLBACK pCB;
1928 RTListForEach(pListAnchor, pCB, PDMAUDIOCALLBACK, Node)
1929 {
1930 Assert(pCB->enmType == enmType);
1931 int rc2 = pCB->pfnCallback(enmType, pCB->pvCtx, pCB->cbCtx, pvUser, cbUser);
1932 if (RT_FAILURE(rc2))
1933 LogFunc(("Failed with %Rrc\n", rc2));
1934 }
1935
1936 return VINF_SUCCESS;
1937 }
1938
1939 return VERR_NOT_SUPPORTED;
1940}
1941#endif /* VBOX_WITH_AUDIO_DEVICE_CALLBACKS */
1942
1943#ifdef VBOX_WITH_AUDIO_CALLBACKS
1944/**
1945 * Backend callback implementation.
1946 *
1947 * Important: No calls back to the backend within this function, as the backend
1948 * might hold any locks / critical sections while executing this callback.
1949 * Will result in some ugly deadlocks (or at least locking order violations) then.
1950 *
1951 * @copydoc FNPDMHOSTAUDIOCALLBACK
1952 */
1953static DECLCALLBACK(int) drvAudioBackendCallback(PPDMDRVINS pDrvIns,
1954 PDMAUDIOCBTYPE enmType, void *pvUser, size_t cbUser)
1955{
1956 AssertPtrReturn(pDrvIns, VERR_INVALID_POINTER);
1957 RT_NOREF(pvUser, cbUser);
1958 /* pvUser and cbUser are optional. */
1959
1960 /* Get the upper driver (PDMIAUDIOCONNECTOR). */
1961 AssertPtr(pDrvIns->pUpBase);
1962 PPDMIAUDIOCONNECTOR pInterface = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIAUDIOCONNECTOR);
1963 AssertPtr(pInterface);
1964 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1965
1966 int rc = RTCritSectEnter(&pThis->CritSect);
1967 AssertRCReturn(rc, rc);
1968
1969 LogFunc(("pThis=%p, enmType=%RU32, pvUser=%p, cbUser=%zu\n", pThis, enmType, pvUser, cbUser));
1970
1971 switch (enmType)
1972 {
1973 case PDMAUDIOCBTYPE_DEVICES_CHANGED:
1974 LogRel(("Audio: Host audio device configuration has changed\n"));
1975 rc = drvAudioScheduleReInitInternal(pThis);
1976 break;
1977
1978 default:
1979 AssertMsgFailed(("Not supported\n"));
1980 break;
1981 }
1982
1983 int rc2 = RTCritSectLeave(&pThis->CritSect);
1984 if (RT_SUCCESS(rc))
1985 rc = rc2;
1986
1987 LogFlowFunc(("Returning %Rrc\n", rc));
1988 return rc;
1989}
1990#endif /* VBOX_WITH_AUDIO_CALLBACKS */
1991
1992#ifdef VBOX_WITH_AUDIO_ENUM
1993/**
1994 * Enumerates all host audio devices.
1995 * This functionality might not be implemented by all backends and will return VERR_NOT_SUPPORTED
1996 * if not being supported.
1997 *
1998 * @returns IPRT status code.
1999 * @param pThis Driver instance to be called.
2000 * @param fLog Whether to print the enumerated device to the release log or not.
2001 * @param pDevEnum Where to store the device enumeration.
2002 */
2003static int drvAudioDevicesEnumerateInternal(PDRVAUDIO pThis, bool fLog, PPDMAUDIODEVICEENUM pDevEnum)
2004{
2005 int rc;
2006
2007 /*
2008 * If the backend supports it, do a device enumeration.
2009 */
2010 if (pThis->pHostDrvAudio->pfnGetDevices)
2011 {
2012 PDMAUDIODEVICEENUM DevEnum;
2013 rc = pThis->pHostDrvAudio->pfnGetDevices(pThis->pHostDrvAudio, &DevEnum);
2014 if (RT_SUCCESS(rc))
2015 {
2016 if (fLog)
2017 LogRel(("Audio: Found %RU16 devices\n", DevEnum.cDevices));
2018
2019 PPDMAUDIODEVICE pDev;
2020 RTListForEach(&DevEnum.lstDevices, pDev, PDMAUDIODEVICE, Node)
2021 {
2022 if (fLog)
2023 {
2024 char *pszFlags = DrvAudioHlpAudDevFlagsToStrA(pDev->fFlags);
2025
2026 LogRel(("Audio: Device '%s':\n", pDev->szName));
2027 LogRel(("Audio: \tUsage = %s\n", DrvAudioHlpAudDirToStr(pDev->enmUsage)));
2028 LogRel(("Audio: \tFlags = %s\n", pszFlags ? pszFlags : "<NONE>"));
2029 LogRel(("Audio: \tInput channels = %RU8\n", pDev->cMaxInputChannels));
2030 LogRel(("Audio: \tOutput channels = %RU8\n", pDev->cMaxOutputChannels));
2031
2032 if (pszFlags)
2033 RTStrFree(pszFlags);
2034 }
2035 }
2036
2037 if (pDevEnum)
2038 rc = DrvAudioHlpDeviceEnumCopy(pDevEnum, &DevEnum);
2039
2040 DrvAudioHlpDeviceEnumFree(&DevEnum);
2041 }
2042 else
2043 {
2044 if (fLog)
2045 LogRel(("Audio: Device enumeration failed with %Rrc\n", rc));
2046 /* Not fatal. */
2047 }
2048 }
2049 else
2050 {
2051 rc = VERR_NOT_SUPPORTED;
2052
2053 if (fLog)
2054 LogRel3(("Audio: Host audio backend does not support audio device enumeration, skipping\n"));
2055 }
2056
2057 LogFunc(("Returning %Rrc\n", rc));
2058 return rc;
2059}
2060#endif /* VBOX_WITH_AUDIO_ENUM */
2061
2062/**
2063 * Initializes the host backend and queries its initial configuration.
2064 * If the host backend fails, VERR_AUDIO_BACKEND_INIT_FAILED will be returned.
2065 *
2066 * Note: As this routine is called when attaching to the device LUN in the
2067 * device emulation, we either check for success or VERR_AUDIO_BACKEND_INIT_FAILED.
2068 * Everything else is considered as fatal and must be handled separately in
2069 * the device emulation!
2070 *
2071 * @return IPRT status code.
2072 * @param pThis Driver instance to be called.
2073 * @param pCfgHandle CFGM configuration handle to use for this driver.
2074 */
2075static int drvAudioHostInit(PDRVAUDIO pThis, PCFGMNODE pCfgHandle)
2076{
2077 /* pCfgHandle is optional. */
2078 NOREF(pCfgHandle);
2079 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2080
2081 LogFlowFuncEnter();
2082
2083 AssertPtr(pThis->pHostDrvAudio);
2084 int rc = pThis->pHostDrvAudio->pfnInit(pThis->pHostDrvAudio);
2085 if (RT_FAILURE(rc))
2086 {
2087 LogRel(("Audio: Initialization of host backend failed with %Rrc\n", rc));
2088 return VERR_AUDIO_BACKEND_INIT_FAILED;
2089 }
2090
2091 /*
2092 * Get the backend configuration.
2093 */
2094 rc = pThis->pHostDrvAudio->pfnGetConfig(pThis->pHostDrvAudio, &pThis->BackendCfg);
2095 if (RT_FAILURE(rc))
2096 {
2097 LogRel(("Audio: Getting host backend configuration failed with %Rrc\n", rc));
2098 return VERR_AUDIO_BACKEND_INIT_FAILED;
2099 }
2100
2101 pThis->cStreamsFreeIn = pThis->BackendCfg.cMaxStreamsIn;
2102 pThis->cStreamsFreeOut = pThis->BackendCfg.cMaxStreamsOut;
2103
2104 LogFlowFunc(("cStreamsFreeIn=%RU8, cStreamsFreeOut=%RU8\n", pThis->cStreamsFreeIn, pThis->cStreamsFreeOut));
2105
2106 LogRel2(("Audio: Host audio backend supports %RU32 input streams and %RU32 output streams at once\n",
2107 /* Clamp for logging. Unlimited streams are defined by UINT32_MAX. */
2108 RT_MIN(64, pThis->cStreamsFreeIn), RT_MIN(64, pThis->cStreamsFreeOut)));
2109
2110#ifdef VBOX_WITH_AUDIO_ENUM
2111 int rc2 = drvAudioDevicesEnumerateInternal(pThis, true /* fLog */, NULL /* pDevEnum */);
2112 if (rc2 != VERR_NOT_SUPPORTED) /* Some backends don't implement device enumeration. */
2113 AssertRC(rc2);
2114
2115 RT_NOREF(rc2);
2116 /* Ignore rc. */
2117#endif
2118
2119#ifdef VBOX_WITH_AUDIO_CALLBACKS
2120 /*
2121 * If the backend supports it, offer a callback to this connector.
2122 */
2123 if (pThis->pHostDrvAudio->pfnSetCallback)
2124 {
2125 int rc2 = pThis->pHostDrvAudio->pfnSetCallback(pThis->pHostDrvAudio, drvAudioBackendCallback);
2126 if (RT_FAILURE(rc2))
2127 LogRel(("Audio: Error registering backend callback, rc=%Rrc\n", rc2));
2128 /* Not fatal. */
2129 }
2130#endif
2131
2132 LogFlowFuncLeave();
2133 return VINF_SUCCESS;
2134}
2135
2136/**
2137 * Handles state changes for all audio streams.
2138 *
2139 * @param pDrvIns Pointer to driver instance.
2140 * @param enmCmd Stream command to set for all streams.
2141 */
2142static void drvAudioStateHandler(PPDMDRVINS pDrvIns, PDMAUDIOSTREAMCMD enmCmd)
2143{
2144 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
2145 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
2146
2147 LogFlowFunc(("enmCmd=%s\n", DrvAudioHlpStreamCmdToStr(enmCmd)));
2148
2149 if (!pThis->pHostDrvAudio)
2150 return;
2151
2152 PPDMAUDIOSTREAM pHstStream;
2153 RTListForEach(&pThis->lstHstStreams, pHstStream, PDMAUDIOSTREAM, Node)
2154 drvAudioStreamControlInternalBackend(pThis, pHstStream, enmCmd);
2155}
2156
2157/**
2158 * Intializes an audio driver instance.
2159 *
2160 * @returns IPRT status code.
2161 * @param pDrvIns Pointer to driver instance.
2162 * @param pCfgHandle CFGM handle to use for configuration.
2163 */
2164static int drvAudioInit(PPDMDRVINS pDrvIns, PCFGMNODE pCfgHandle)
2165{
2166 AssertPtrReturn(pCfgHandle, VERR_INVALID_POINTER);
2167 AssertPtrReturn(pDrvIns, VERR_INVALID_POINTER);
2168
2169 LogRel2(("Audio: Verbose logging enabled\n"));
2170
2171 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
2172 LogFlowFunc(("pThis=%p, pDrvIns=%p\n", pThis, pDrvIns));
2173
2174 int rc = RTCritSectInit(&pThis->CritSect);
2175 AssertRCReturn(rc, rc);
2176
2177 /** @todo Add audio driver options. */
2178
2179 /*
2180 * If everything went well, initialize the lower driver.
2181 */
2182 rc = drvAudioHostInit(pThis, pCfgHandle);
2183
2184 LogFlowFuncLeaveRC(rc);
2185 return rc;
2186}
2187
2188/**
2189 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamRead}
2190 */
2191static DECLCALLBACK(int) drvAudioStreamRead(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream,
2192 void *pvBuf, uint32_t cbBuf, uint32_t *pcbRead)
2193{
2194 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
2195 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2196
2197 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2198 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
2199 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
2200 /* pcbWritten is optional. */
2201
2202 AssertMsg(pStream->enmDir == PDMAUDIODIR_IN,
2203 ("Stream '%s' is not an input stream and therefore cannot be read from (direction is 0x%x)\n",
2204 pStream->szName, pStream->enmDir));
2205
2206 uint32_t cbRead = 0;
2207
2208 int rc = RTCritSectEnter(&pThis->CritSect);
2209 if (RT_FAILURE(rc))
2210 return rc;
2211
2212 do
2213 {
2214 if ( pThis->pHostDrvAudio
2215 && pThis->pHostDrvAudio->pfnGetStatus
2216 && pThis->pHostDrvAudio->pfnGetStatus(pThis->pHostDrvAudio, PDMAUDIODIR_IN) != PDMAUDIOBACKENDSTS_RUNNING)
2217 {
2218 rc = VERR_NOT_AVAILABLE;
2219 break;
2220 }
2221
2222 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
2223 if (!pHstStream)
2224 {
2225 rc = VERR_NOT_AVAILABLE;
2226 break;
2227 }
2228
2229 PPDMAUDIOSTREAM pGstStream = pHstStream->pPair;
2230 AssertPtr(pGstStream);
2231
2232 pGstStream->In.tsLastReadMS = RTTimeMilliTS();
2233
2234 /*
2235 * Read from the parent buffer (that is, the guest buffer) which
2236 * should have the audio data in the format the guest needs.
2237 */
2238 uint32_t cRead;
2239 rc = AudioMixBufReadCirc(&pGstStream->MixBuf, pvBuf, cbBuf, &cRead);
2240 if (RT_SUCCESS(rc))
2241 {
2242 if (cRead)
2243 {
2244 cbRead = AUDIOMIXBUF_S2B(&pGstStream->MixBuf, cRead);
2245
2246#ifdef VBOX_WITH_STATISTICS
2247 STAM_COUNTER_ADD(&pThis->Stats.TotalBytesRead, cbRead);
2248 STAM_COUNTER_ADD(&pGstStream->In.StatBytesTotalRead, cbRead);
2249#endif
2250 AudioMixBufFinish(&pGstStream->MixBuf, cRead);
2251 }
2252 }
2253
2254 } while (0);
2255
2256 Log3Func(("[%s] cbRead=%RU32, rc=%Rrc\n", pStream->szName, cbRead, rc));
2257
2258 int rc2 = RTCritSectLeave(&pThis->CritSect);
2259 if (RT_SUCCESS(rc))
2260 rc = rc2;
2261
2262 if (RT_SUCCESS(rc))
2263 {
2264 if (pcbRead)
2265 *pcbRead = cbRead;
2266 }
2267
2268 return rc;
2269}
2270
2271/**
2272 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamCreate}
2273 */
2274static DECLCALLBACK(int) drvAudioStreamCreate(PPDMIAUDIOCONNECTOR pInterface,
2275 PPDMAUDIOSTREAMCFG pCfgHost, PPDMAUDIOSTREAMCFG pCfgGuest,
2276 PPDMAUDIOSTREAM *ppStream)
2277{
2278 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2279 AssertPtrReturn(pCfgHost, VERR_INVALID_POINTER);
2280 AssertPtrReturn(pCfgGuest, VERR_INVALID_POINTER);
2281 AssertPtrReturn(ppStream, VERR_INVALID_POINTER);
2282
2283 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
2284
2285 int rc = RTCritSectEnter(&pThis->CritSect);
2286 if (RT_FAILURE(rc))
2287 return rc;
2288
2289 LogFlowFunc(("Host=%s, Guest=%s\n", pCfgHost->szName, pCfgGuest->szName));
2290#ifdef DEBUG
2291 DrvAudioHlpStreamCfgPrint(pCfgHost);
2292 DrvAudioHlpStreamCfgPrint(pCfgGuest);
2293#endif
2294
2295 /*
2296 * The guest stream always will get the audio stream configuration told
2297 * by the device emulation (which in turn was/could be set by the guest OS).
2298 */
2299 PPDMAUDIOSTREAM pGstStrm = NULL;
2300
2301 /** @todo Docs! */
2302 PPDMAUDIOSTREAM pHstStrm = NULL;
2303
2304#define RC_BREAK(x) { rc = x; break; }
2305
2306 do
2307 {
2308 if ( !DrvAudioHlpStreamCfgIsValid(pCfgHost)
2309 || !DrvAudioHlpStreamCfgIsValid(pCfgGuest))
2310 {
2311 RC_BREAK(VERR_INVALID_PARAMETER);
2312 }
2313
2314 /* Make sure that both configurations actually intend the same thing. */
2315 if (pCfgHost->enmDir != pCfgGuest->enmDir)
2316 {
2317 AssertMsgFailed(("Stream configuration directions do not match\n"));
2318 RC_BREAK(VERR_INVALID_PARAMETER);
2319 }
2320
2321 /* Note: cbHstStrm will contain the size of the data the backend needs to operate on. */
2322 size_t cbHstStrm = 0;
2323 if (pCfgHost->enmDir == PDMAUDIODIR_IN)
2324 {
2325 if (!pThis->cStreamsFreeIn)
2326 LogFunc(("Warning: No more input streams free to use\n"));
2327
2328 cbHstStrm = pThis->BackendCfg.cbStreamIn;
2329 }
2330 else /* Out */
2331 {
2332 if (!pThis->cStreamsFreeOut)
2333 {
2334 LogFlowFunc(("Maximum number of host output streams reached\n"));
2335 RC_BREAK(VERR_AUDIO_NO_FREE_OUTPUT_STREAMS);
2336 }
2337
2338 cbHstStrm = pThis->BackendCfg.cbStreamOut;
2339 }
2340
2341 pHstStrm = (PPDMAUDIOSTREAM)RTMemAllocZ(sizeof(PDMAUDIOSTREAM));
2342 AssertPtrBreakStmt(pHstStrm, rc = VERR_NO_MEMORY);
2343
2344 if (cbHstStrm) /* High unlikely that backends do not have an own space for data, but better check. */
2345 {
2346 pHstStrm->pvBackend = RTMemAllocZ(cbHstStrm);
2347 AssertPtrBreakStmt(pHstStrm->pvBackend, rc = VERR_NO_MEMORY);
2348
2349 pHstStrm->cbBackend = cbHstStrm;
2350 }
2351
2352 pHstStrm->enmCtx = PDMAUDIOSTREAMCTX_HOST;
2353 pHstStrm->enmDir = pCfgHost->enmDir;
2354
2355 pGstStrm = (PPDMAUDIOSTREAM)RTMemAllocZ(sizeof(PDMAUDIOSTREAM));
2356 AssertPtrBreakStmt(pGstStrm, rc = VERR_NO_MEMORY);
2357
2358 pGstStrm->enmCtx = PDMAUDIOSTREAMCTX_GUEST;
2359 pGstStrm->enmDir = pCfgGuest->enmDir;
2360
2361 /*
2362 * Init host stream.
2363 */
2364 RTStrPrintf(pHstStrm->szName, RT_ELEMENTS(pHstStrm->szName), "%s (Host)",
2365 strlen(pCfgHost->szName) ? pCfgHost->szName : "<Untitled>");
2366
2367 rc = drvAudioStreamLinkToInternal(pHstStrm, pGstStrm);
2368 AssertRCBreak(rc);
2369
2370 /*
2371 * Init guest stream.
2372 */
2373 RTStrPrintf(pGstStrm->szName, RT_ELEMENTS(pGstStrm->szName), "%s (Guest)",
2374 strlen(pCfgGuest->szName) ? pCfgGuest->szName : "<Untitled>");
2375
2376 pGstStrm->fStatus = pHstStrm->fStatus; /* Reflect the host stream's status. */
2377
2378 rc = drvAudioStreamLinkToInternal(pGstStrm, pHstStrm);
2379 AssertRCBreak(rc);
2380
2381 /*
2382 * Try to init the rest.
2383 */
2384 rc = drvAudioStreamInitInternal(pThis, pHstStrm, pCfgHost, pCfgGuest);
2385 if (RT_FAILURE(rc))
2386 break;
2387
2388#ifdef VBOX_WITH_STATISTICS
2389 char szStatName[255];
2390
2391 if (pCfgGuest->enmDir == PDMAUDIODIR_IN)
2392 {
2393 RTStrPrintf(szStatName, sizeof(szStatName), "Guest/%s/BytesElapsed", pGstStrm->szName);
2394 PDMDrvHlpSTAMRegCounterEx(pThis->pDrvIns, &pGstStrm->In.StatBytesElapsed,
2395 szStatName, STAMUNIT_BYTES, "Elapsed bytes read.");
2396
2397 RTStrPrintf(szStatName, sizeof(szStatName), "Guest/%s/BytesRead", pGstStrm->szName);
2398 PDMDrvHlpSTAMRegCounterEx(pThis->pDrvIns, &pGstStrm->In.StatBytesTotalRead,
2399 szStatName, STAMUNIT_BYTES, "Total bytes read.");
2400
2401 RTStrPrintf(szStatName, sizeof(szStatName), "Host/%s/SamplesCaptured", pHstStrm->szName);
2402 PDMDrvHlpSTAMRegCounterEx(pThis->pDrvIns, &pHstStrm->In.StatSamplesCaptured,
2403 szStatName, STAMUNIT_COUNT, "Total samples captured.");
2404 }
2405 else if (pCfgGuest->enmDir == PDMAUDIODIR_OUT)
2406 {
2407 RTStrPrintf(szStatName, sizeof(szStatName), "Guest/%s/BytesElapsed", pGstStrm->szName);
2408 PDMDrvHlpSTAMRegCounterEx(pThis->pDrvIns, &pGstStrm->Out.StatBytesElapsed,
2409 szStatName, STAMUNIT_BYTES, "Elapsed bytes written.");
2410
2411 RTStrPrintf(szStatName, sizeof(szStatName), "Guest/%s/BytesWritten", pGstStrm->szName);
2412 PDMDrvHlpSTAMRegCounterEx(pThis->pDrvIns, &pGstStrm->Out.StatBytesTotalWritten,
2413 szStatName, STAMUNIT_BYTES, "Total bytes written.");
2414
2415 RTStrPrintf(szStatName, sizeof(szStatName), "Host/%s/SamplesPlayed", pHstStrm->szName);
2416 PDMDrvHlpSTAMRegCounterEx(pThis->pDrvIns, &pHstStrm->Out.StatSamplesPlayed,
2417 szStatName, STAMUNIT_COUNT, "Total samples played.");
2418 }
2419 else
2420 AssertFailed();
2421#endif
2422
2423 } while (0);
2424
2425#undef RC_BREAK
2426
2427 if (RT_FAILURE(rc))
2428 {
2429 if (pGstStrm)
2430 {
2431 int rc2 = drvAudioStreamUninitInternal(pThis, pGstStrm);
2432 if (RT_SUCCESS(rc2))
2433 {
2434 RTMemFree(pGstStrm);
2435 pGstStrm = NULL;
2436 }
2437 }
2438
2439 if (pHstStrm)
2440 {
2441 int rc2 = drvAudioStreamUninitInternal(pThis, pHstStrm);
2442 if (RT_SUCCESS(rc2))
2443 {
2444 drvAudioStreamFree(pHstStrm);
2445 pHstStrm = NULL;
2446 }
2447 }
2448 }
2449 else
2450 {
2451 /* Set initial reference counts. */
2452 RTListAppend(&pThis->lstGstStreams, &pGstStrm->Node);
2453 pGstStrm->cRefs = 1;
2454
2455 RTListAppend(&pThis->lstHstStreams, &pHstStrm->Node);
2456 pHstStrm->cRefs = 1;
2457
2458 if (pCfgHost->enmDir == PDMAUDIODIR_IN)
2459 {
2460 if (pThis->cStreamsFreeIn)
2461 pThis->cStreamsFreeIn--;
2462 }
2463 else /* Out */
2464 {
2465 if (pThis->cStreamsFreeOut)
2466 pThis->cStreamsFreeOut--;
2467 }
2468
2469#ifdef VBOX_WITH_STATISTICS
2470 STAM_COUNTER_ADD(&pThis->Stats.TotalStreamsCreated, 1);
2471#endif
2472 /* Always return the guest-side part to the device emulation. */
2473 *ppStream = pGstStrm;
2474 }
2475
2476 int rc2 = RTCritSectLeave(&pThis->CritSect);
2477 if (RT_SUCCESS(rc))
2478 rc = rc2;
2479
2480 LogFlowFuncLeaveRC(rc);
2481 return rc;
2482}
2483
2484/**
2485 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnGetConfig}
2486 */
2487static DECLCALLBACK(int) drvAudioGetConfig(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOBACKENDCFG pCfg)
2488{
2489 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2490 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
2491
2492 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
2493
2494 int rc = RTCritSectEnter(&pThis->CritSect);
2495 if (RT_FAILURE(rc))
2496 return rc;
2497
2498 if (pThis->pHostDrvAudio)
2499 {
2500 if (pThis->pHostDrvAudio->pfnGetConfig)
2501 rc = pThis->pHostDrvAudio->pfnGetConfig(pThis->pHostDrvAudio, pCfg);
2502 else
2503 rc = VERR_NOT_SUPPORTED;
2504 }
2505 else
2506 AssertFailed();
2507
2508 int rc2 = RTCritSectLeave(&pThis->CritSect);
2509 if (RT_SUCCESS(rc))
2510 rc = rc2;
2511
2512 LogFlowFuncLeaveRC(rc);
2513 return rc;
2514}
2515
2516/**
2517 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnGetStatus}
2518 */
2519static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvAudioGetStatus(PPDMIAUDIOCONNECTOR pInterface, PDMAUDIODIR enmDir)
2520{
2521 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
2522
2523 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
2524
2525 PDMAUDIOBACKENDSTS backendSts = PDMAUDIOBACKENDSTS_UNKNOWN;
2526
2527 int rc = RTCritSectEnter(&pThis->CritSect);
2528 if (RT_SUCCESS(rc))
2529 {
2530 if ( pThis->pHostDrvAudio
2531 && pThis->pHostDrvAudio->pfnGetStatus)
2532 {
2533 backendSts = pThis->pHostDrvAudio->pfnGetStatus(pThis->pHostDrvAudio, enmDir);
2534 }
2535
2536 int rc2 = RTCritSectLeave(&pThis->CritSect);
2537 if (RT_SUCCESS(rc))
2538 rc = rc2;
2539 }
2540
2541 LogFlowFuncLeaveRC(rc);
2542 return backendSts;
2543}
2544
2545/**
2546 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamGetReadable}
2547 */
2548static DECLCALLBACK(uint32_t) drvAudioStreamGetReadable(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream)
2549{
2550 AssertPtrReturn(pInterface, 0);
2551 AssertPtrReturn(pStream, 0);
2552
2553 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
2554
2555 int rc2 = RTCritSectEnter(&pThis->CritSect);
2556 AssertRC(rc2);
2557
2558 AssertMsg(pStream->enmDir == PDMAUDIODIR_IN, ("Can't read from a non-input stream\n"));
2559
2560 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
2561 if (!pHstStream) /* No host stream available? Bail out early. */
2562 {
2563 rc2 = RTCritSectLeave(&pThis->CritSect);
2564 AssertRC(rc2);
2565
2566 return 0;
2567 }
2568
2569 uint32_t cReadable = 0;
2570
2571 PPDMAUDIOSTREAM pGstStream = pHstStream->pPair;
2572 if (pGstStream)
2573 cReadable = AudioMixBufLive(&pGstStream->MixBuf);
2574
2575 Log3Func(("[%s] cbReadable=%RU32 (%zu bytes)\n", pHstStream->szName, cReadable,
2576 AUDIOMIXBUF_S2B(&pGstStream->MixBuf, cReadable)));
2577
2578 uint32_t cbReadable = AUDIOMIXBUF_S2B(&pGstStream->MixBuf, cReadable);
2579
2580 rc2 = RTCritSectLeave(&pThis->CritSect);
2581 AssertRC(rc2);
2582
2583 /* Return bytes instead of audio samples. */
2584 return cbReadable;
2585}
2586
2587/**
2588 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamGetWritable}
2589 */
2590static DECLCALLBACK(uint32_t) drvAudioStreamGetWritable(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream)
2591{
2592 AssertPtrReturn(pInterface, 0);
2593 AssertPtrReturn(pStream, 0);
2594
2595 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
2596
2597 int rc2 = RTCritSectEnter(&pThis->CritSect);
2598 AssertRC(rc2);
2599
2600 AssertMsg(pStream->enmDir == PDMAUDIODIR_OUT, ("Can't write to a non-output stream\n"));
2601
2602 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
2603 if (!pHstStream) /* No host stream available? Bail out early. */
2604 {
2605 rc2 = RTCritSectLeave(&pThis->CritSect);
2606 AssertRC(rc2);
2607
2608 AssertMsgFailed(("Guest stream '%s' does not have a host stream attached\n", pStream->szName));
2609 return 0;
2610 }
2611
2612 PPDMAUDIOSTREAM pGstStream = pHstStream->pPair;
2613 AssertPtr(pGstStream);
2614
2615 uint32_t cWritable = AudioMixBufFree(&pGstStream->MixBuf);
2616
2617 Log3Func(("[%s] cWritable=%RU32 (%zu bytes)\n", pHstStream->szName, cWritable,
2618 AUDIOMIXBUF_S2B(&pGstStream->MixBuf, cWritable)));
2619
2620 uint32_t cbWritable = AUDIOMIXBUF_S2B(&pGstStream->MixBuf, cWritable);
2621
2622 rc2 = RTCritSectLeave(&pThis->CritSect);
2623 AssertRC(rc2);
2624
2625 /* Return bytes instead of audio samples. */
2626 return cbWritable;
2627}
2628
2629/**
2630 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamGetStatus}
2631 */
2632static DECLCALLBACK(PDMAUDIOSTRMSTS) drvAudioStreamGetStatus(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream)
2633{
2634 AssertPtrReturn(pInterface, false);
2635
2636 if (!pStream)
2637 return PDMAUDIOSTRMSTS_FLAG_NONE;
2638
2639 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
2640
2641 int rc2 = RTCritSectEnter(&pThis->CritSect);
2642 AssertRC(rc2);
2643
2644 PDMAUDIOSTRMSTS strmSts = PDMAUDIOSTRMSTS_FLAG_NONE;
2645
2646 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
2647 if (pHstStream)
2648 {
2649 strmSts = pHstStream->fStatus;
2650#ifdef LOG_ENABLED
2651 char *pszHstSts = dbgAudioStreamStatusToStr(pHstStream->fStatus);
2652 Log3Func(("[%s] %s\n", pHstStream->szName, pszHstSts));
2653 RTStrFree(pszHstSts);
2654#endif
2655 }
2656
2657 rc2 = RTCritSectLeave(&pThis->CritSect);
2658 AssertRC(rc2);
2659
2660 return strmSts;
2661}
2662
2663/**
2664 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamSetVolume}
2665 */
2666static DECLCALLBACK(int) drvAudioStreamSetVolume(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream, PPDMAUDIOVOLUME pVol)
2667{
2668 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2669 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2670 AssertPtrReturn(pVol, VERR_INVALID_POINTER);
2671
2672 LogFlowFunc(("[%s] volL=%RU32, volR=%RU32, fMute=%RTbool\n", pStream->szName, pVol->uLeft, pVol->uRight, pVol->fMuted));
2673
2674 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
2675 PPDMAUDIOSTREAM pGstStream = pHstStream ? pHstStream->pPair : pStream;
2676
2677 AudioMixBufSetVolume(&pHstStream->MixBuf, pVol);
2678 AudioMixBufSetVolume(&pGstStream->MixBuf, pVol);
2679 return VINF_SUCCESS;
2680}
2681
2682/**
2683 * @interface_method_impl{PDMIAUDIOCONNECTOR,pfnStreamDestroy}
2684 */
2685static DECLCALLBACK(int) drvAudioStreamDestroy(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOSTREAM pStream)
2686{
2687 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2688 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2689
2690 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
2691
2692 int rc = RTCritSectEnter(&pThis->CritSect);
2693 AssertRC(rc);
2694
2695 PDMAUDIODIR enmDir = pStream->enmDir;
2696
2697 LogFlowFunc(("[%s] cRefs=%RU32\n", pStream->szName, pStream->cRefs));
2698 if (pStream->cRefs > 1)
2699 rc = VERR_WRONG_ORDER;
2700
2701 if (RT_SUCCESS(rc))
2702 {
2703 PPDMAUDIOSTREAM pHstStream = drvAudioGetHostStream(pStream);
2704 PPDMAUDIOSTREAM pGstStream = pHstStream ? pHstStream->pPair : pStream;
2705
2706 LogRel2(("Audio: Destroying host stream '%s' (guest stream '%s')\n",
2707 pHstStream ? pHstStream->szName : "<None>",
2708 pGstStream ? pGstStream->szName : "<None>"));
2709
2710 /* Should prevent double frees. */
2711 Assert(pHstStream != pGstStream);
2712
2713 if (pHstStream)
2714 {
2715 rc = drvAudioStreamUninitInternal(pThis, pHstStream);
2716 if (RT_SUCCESS(rc))
2717 {
2718#ifdef VBOX_WITH_STATISTICS
2719 if (pHstStream->enmDir == PDMAUDIODIR_IN)
2720 {
2721 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pHstStream->In.StatSamplesCaptured);
2722 }
2723 else if (pHstStream->enmDir == PDMAUDIODIR_OUT)
2724 {
2725 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pHstStream->Out.StatSamplesPlayed);
2726 }
2727 else
2728 AssertFailed();
2729#endif
2730 RTListNodeRemove(&pHstStream->Node);
2731
2732 drvAudioStreamFree(pHstStream);
2733 pHstStream = NULL;
2734 }
2735 else
2736 LogRel2(("Audio: Uninitializing host stream '%s' failed with %Rrc\n", pHstStream->szName, rc));
2737 }
2738
2739 if ( RT_SUCCESS(rc)
2740 && pGstStream)
2741 {
2742 rc = drvAudioStreamUninitInternal(pThis, pGstStream);
2743 if (RT_SUCCESS(rc))
2744 {
2745#ifdef VBOX_WITH_STATISTICS
2746 if (pGstStream->enmDir == PDMAUDIODIR_IN)
2747 {
2748 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pGstStream->In.StatBytesElapsed);
2749 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pGstStream->In.StatBytesTotalRead);
2750 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pGstStream->In.StatSamplesCaptured);
2751 }
2752 else if (pGstStream->enmDir == PDMAUDIODIR_OUT)
2753 {
2754 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pGstStream->Out.StatBytesElapsed);
2755 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pGstStream->Out.StatBytesTotalWritten);
2756 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pGstStream->Out.StatSamplesPlayed);
2757 }
2758 else
2759 AssertFailed();
2760#endif
2761 RTListNodeRemove(&pGstStream->Node);
2762
2763 RTMemFree(pGstStream);
2764 pGstStream = NULL;
2765 }
2766 else
2767 LogRel2(("Audio: Uninitializing guest stream '%s' failed with %Rrc\n", pGstStream->szName, rc));
2768 }
2769 }
2770
2771 if (RT_SUCCESS(rc))
2772 {
2773 if (enmDir == PDMAUDIODIR_IN)
2774 {
2775 pThis->cStreamsFreeIn++;
2776 }
2777 else /* Out */
2778 {
2779 pThis->cStreamsFreeOut++;
2780 }
2781 }
2782
2783 int rc2 = RTCritSectLeave(&pThis->CritSect);
2784 if (RT_SUCCESS(rc))
2785 rc = rc2;
2786
2787 LogFlowFuncLeaveRC(rc);
2788 return rc;
2789}
2790
2791/**
2792 * Creates an audio stream on the backend side.
2793 *
2794 * @returns IPRT status code.
2795 * @param pThis Pointer to driver instance.
2796 * @param pHstStream (Host) audio stream to use for creating the stream on the backend side.
2797 * @param pCfgReq Requested audio stream configuration to use for stream creation.
2798 * @param pCfgAcq Acquired audio stream configuration returned by the backend. Optional, can be NULL.
2799 */
2800static int drvAudioStreamCreateInternalBackend(PDRVAUDIO pThis,
2801 PPDMAUDIOSTREAM pHstStream, PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
2802{
2803 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2804 AssertPtrReturn(pHstStream, VERR_INVALID_POINTER);
2805 AssertPtrReturn(pCfgReq, VERR_INVALID_POINTER);
2806 /* pCfgAcq is optional. */
2807
2808 AssertMsg(pHstStream->enmCtx == PDMAUDIOSTREAMCTX_HOST,
2809 ("Stream '%s' is not a host stream and therefore has no backend\n", pHstStream->szName));
2810
2811 AssertMsg((pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_INITIALIZED) == 0,
2812 ("Stream '%s' already initialized in backend\n", pHstStream->szName));
2813
2814 /* Make the acquired host configuration the requested host configuration initially,
2815 * in case the backend does not report back an acquired configuration. */
2816 PDMAUDIOSTREAMCFG CfgAcq;
2817 int rc = DrvAudioHlpStreamCfgCopy(&CfgAcq, pCfgReq);
2818 if (RT_FAILURE(rc))
2819 {
2820 LogRel(("Audio: Creating stream '%s' with an invalid backend configuration not possible, skipping\n",
2821 pHstStream->szName));
2822 return rc;
2823 }
2824
2825 rc = pThis->pHostDrvAudio->pfnStreamCreate(pThis->pHostDrvAudio, pHstStream->pvBackend, pCfgReq, &CfgAcq);
2826 if (RT_FAILURE(rc))
2827 {
2828 LogRel(("Audio: Creating stream '%s' in backend failed with %Rrc\n", pHstStream->szName, rc));
2829 return rc;
2830 }
2831
2832 /* Validate acquired configuration. */
2833 if (!DrvAudioHlpStreamCfgIsValid(&CfgAcq))
2834 {
2835 LogRel(("Audio: Creating stream '%s' returned an invalid backend configuration, skipping\n", pHstStream->szName));
2836 return VERR_INVALID_PARAMETER;
2837 }
2838
2839 /* Only set the host's stream to initialized if we were able create the stream
2840 * in the host backend. This is necessary for trying to re-initialize the stream
2841 * at some later point in time. */
2842 pHstStream->fStatus |= PDMAUDIOSTRMSTS_FLAG_INITIALIZED;
2843
2844 if (pCfgAcq)
2845 {
2846 int rc2 = DrvAudioHlpStreamCfgCopy(pCfgAcq, &CfgAcq);
2847 AssertRC(rc2);
2848 }
2849
2850 return VINF_SUCCESS;
2851}
2852
2853/**
2854 * Calls the backend to give it the chance to destroy its part of the audio stream.
2855 *
2856 * @returns IPRT status code.
2857 * @param pThis Pointer to driver instance.
2858 * @param pHstStream Host audio stream to call the backend destruction for.
2859 */
2860static int drvAudioStreamDestroyInternalBackend(PDRVAUDIO pThis, PPDMAUDIOSTREAM pHstStream)
2861{
2862 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2863 AssertPtrReturn(pHstStream, VERR_INVALID_POINTER);
2864
2865 AssertMsg(pHstStream->enmCtx == PDMAUDIOSTREAMCTX_HOST,
2866 ("Stream '%s' is not a host stream and therefore has no backend\n", pHstStream->szName));
2867
2868 int rc = VINF_SUCCESS;
2869
2870#ifdef LOG_ENABLED
2871 char *pszHstSts = dbgAudioStreamStatusToStr(pHstStream->fStatus);
2872 LogFunc(("[%s] fStatus=%s\n", pHstStream->szName, pszHstSts));
2873 RTStrFree(pszHstSts);
2874#endif /* LOG_ENABLED */
2875
2876 if (pHstStream->fStatus & PDMAUDIOSTRMSTS_FLAG_INITIALIZED)
2877 {
2878 /* Check if the pointer to the host audio driver is still valid.
2879 * It can be NULL if we were called in drvAudioDestruct, for example. */
2880 if (pThis->pHostDrvAudio)
2881 rc = pThis->pHostDrvAudio->pfnStreamDestroy(pThis->pHostDrvAudio, pHstStream->pvBackend);
2882 if (RT_SUCCESS(rc))
2883 {
2884 pHstStream->fStatus &= ~PDMAUDIOSTRMSTS_FLAG_INITIALIZED;
2885#if 0 /** @todo r=andy Disabled for now -- need to test this on Windows hosts. */
2886 Assert(pHstStream->fStatus == PDMAUDIOSTRMSTS_FLAG_NONE);
2887#endif
2888 }
2889 }
2890
2891 LogFlowFunc(("[%s] Returning %Rrc\n", pHstStream->szName, rc));
2892 return rc;
2893}
2894
2895/**
2896 * Uninitializes an audio stream.
2897 *
2898 * @returns IPRT status code.
2899 * @param pThis Pointer to driver instance.
2900 * @param pStream Pointer to audio stream to uninitialize.
2901 */
2902static int drvAudioStreamUninitInternal(PDRVAUDIO pThis, PPDMAUDIOSTREAM pStream)
2903{
2904 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2905 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2906
2907 LogFlowFunc(("[%s] cRefs=%RU32\n", pStream->szName, pStream->cRefs));
2908
2909 if (pStream->cRefs > 1)
2910 return VERR_WRONG_ORDER;
2911
2912 int rc = VINF_SUCCESS;
2913
2914 if (pStream->enmCtx == PDMAUDIOSTREAMCTX_GUEST)
2915 {
2916 if (pStream->fStatus & PDMAUDIOSTRMSTS_FLAG_INITIALIZED)
2917 {
2918 rc = drvAudioStreamControlInternal(pThis, pStream, PDMAUDIOSTREAMCMD_DISABLE);
2919 if (RT_SUCCESS(rc))
2920 {
2921 pStream->fStatus &= ~PDMAUDIOSTRMSTS_FLAG_INITIALIZED;
2922 Assert(pStream->fStatus == PDMAUDIOSTRMSTS_FLAG_NONE);
2923 }
2924 }
2925 }
2926 else if (pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST)
2927 {
2928 rc = drvAudioStreamDestroyInternalBackend(pThis, pStream);
2929 }
2930 else
2931 AssertFailedReturn(VERR_NOT_IMPLEMENTED);
2932
2933 if (RT_SUCCESS(rc))
2934 {
2935 /* Make sure that the pair (if any) knows that we're not valid anymore. */
2936 int rc2 = drvAudioStreamLinkToInternal(pStream, NULL);
2937 AssertRC(rc2);
2938
2939 /* Reset status. */
2940 pStream->fStatus = PDMAUDIOSTRMSTS_FLAG_NONE;
2941
2942 /* Destroy mixing buffer. */
2943 AudioMixBufDestroy(&pStream->MixBuf);
2944 }
2945
2946 LogFlowFunc(("Returning %Rrc\n", rc));
2947 return rc;
2948}
2949
2950/********************************************************************/
2951
2952/**
2953 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
2954 */
2955static DECLCALLBACK(void *) drvAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
2956{
2957 LogFlowFunc(("pInterface=%p, pszIID=%s\n", pInterface, pszIID));
2958
2959 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
2960 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
2961
2962 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
2963 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIAUDIOCONNECTOR, &pThis->IAudioConnector);
2964
2965 return NULL;
2966}
2967
2968/**
2969 * Power Off notification.
2970 *
2971 * @param pDrvIns The driver instance data.
2972 */
2973static DECLCALLBACK(void) drvAudioPowerOff(PPDMDRVINS pDrvIns)
2974{
2975 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
2976
2977 LogFlowFuncEnter();
2978
2979 /* Just destroy the host stream on the backend side.
2980 * The rest will either be destructed by the device emulation or
2981 * in drvAudioDestruct(). */
2982 PPDMAUDIOSTREAM pStream;
2983 RTListForEach(&pThis->lstHstStreams, pStream, PDMAUDIOSTREAM, Node)
2984 drvAudioStreamDestroyInternalBackend(pThis, pStream);
2985
2986 /*
2987 * Last call for the driver below us.
2988 * Let it know that we reached end of life.
2989 */
2990 if (pThis->pHostDrvAudio->pfnShutdown)
2991 pThis->pHostDrvAudio->pfnShutdown(pThis->pHostDrvAudio);
2992
2993 pThis->pHostDrvAudio = NULL;
2994
2995 LogFlowFuncLeave();
2996}
2997
2998/**
2999 * Constructs an audio driver instance.
3000 *
3001 * @copydoc FNPDMDRVCONSTRUCT
3002 */
3003static DECLCALLBACK(int) drvAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
3004{
3005 LogFlowFunc(("pDrvIns=%#p, pCfgHandle=%#p, fFlags=%x\n", pDrvIns, pCfg, fFlags));
3006
3007 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
3008 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
3009
3010 RTListInit(&pThis->lstHstStreams);
3011 RTListInit(&pThis->lstGstStreams);
3012#ifdef VBOX_WITH_AUDIO_DEVICE_CALLBACKS
3013 RTListInit(&pThis->lstCBIn);
3014 RTListInit(&pThis->lstCBOut);
3015#endif
3016
3017 /*
3018 * Init the static parts.
3019 */
3020 pThis->pDrvIns = pDrvIns;
3021 /* IBase. */
3022 pDrvIns->IBase.pfnQueryInterface = drvAudioQueryInterface;
3023 /* IAudioConnector. */
3024 pThis->IAudioConnector.pfnGetConfig = drvAudioGetConfig;
3025 pThis->IAudioConnector.pfnGetStatus = drvAudioGetStatus;
3026 pThis->IAudioConnector.pfnStreamCreate = drvAudioStreamCreate;
3027 pThis->IAudioConnector.pfnStreamDestroy = drvAudioStreamDestroy;
3028 pThis->IAudioConnector.pfnStreamRetain = drvAudioStreamRetain;
3029 pThis->IAudioConnector.pfnStreamRelease = drvAudioStreamRelease;
3030 pThis->IAudioConnector.pfnStreamControl = drvAudioStreamControl;
3031 pThis->IAudioConnector.pfnStreamRead = drvAudioStreamRead;
3032 pThis->IAudioConnector.pfnStreamWrite = drvAudioStreamWrite;
3033 pThis->IAudioConnector.pfnStreamIterate = drvAudioStreamIterate;
3034 pThis->IAudioConnector.pfnStreamGetReadable = drvAudioStreamGetReadable;
3035 pThis->IAudioConnector.pfnStreamGetWritable = drvAudioStreamGetWritable;
3036 pThis->IAudioConnector.pfnStreamGetStatus = drvAudioStreamGetStatus;
3037 pThis->IAudioConnector.pfnStreamSetVolume = drvAudioStreamSetVolume;
3038 pThis->IAudioConnector.pfnStreamPlay = drvAudioStreamPlay;
3039 pThis->IAudioConnector.pfnStreamCapture = drvAudioStreamCapture;
3040#ifdef VBOX_WITH_AUDIO_DEVICE_CALLBACKS
3041 pThis->IAudioConnector.pfnRegisterCallbacks = drvAudioRegisterCallbacks;
3042 pThis->IAudioConnector.pfnCallback = drvAudioCallback;
3043#endif
3044
3045 /*
3046 * Attach driver below and query its connector interface.
3047 */
3048 PPDMIBASE pDownBase;
3049 int rc = PDMDrvHlpAttach(pDrvIns, fFlags, &pDownBase);
3050 if (RT_FAILURE(rc))
3051 {
3052 LogRel(("Audio: Failed to attach to driver %p below (flags=0x%x), rc=%Rrc\n",
3053 pDrvIns, fFlags, rc));
3054 return rc;
3055 }
3056
3057 pThis->pHostDrvAudio = PDMIBASE_QUERY_INTERFACE(pDownBase, PDMIHOSTAUDIO);
3058 if (!pThis->pHostDrvAudio)
3059 {
3060 LogRel(("Audio: Failed to query interface for underlying host driver\n"));
3061 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_MISSING_INTERFACE_BELOW,
3062 N_("Host audio backend missing or invalid"));
3063 }
3064
3065 rc = drvAudioInit(pDrvIns, pCfg);
3066 if (RT_SUCCESS(rc))
3067 {
3068 pThis->fTerminate = false;
3069 pThis->pDrvIns = pDrvIns;
3070
3071#ifdef VBOX_WITH_STATISTICS
3072 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalStreamsActive, "TotalStreamsActive",
3073 STAMUNIT_COUNT, "Total active audio streams.");
3074 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalStreamsCreated, "TotalStreamsCreated",
3075 STAMUNIT_COUNT, "Total created audio streams.");
3076 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalSamplesRead, "TotalSamplesRead",
3077 STAMUNIT_COUNT, "Total samples read by device emulation.");
3078 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalSamplesWritten, "TotalSamplesWritten",
3079 STAMUNIT_COUNT, "Total samples written by device emulation ");
3080 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalSamplesMixedIn, "TotalSamplesMixedIn",
3081 STAMUNIT_COUNT, "Total input samples mixed.");
3082 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalSamplesMixedOut, "TotalSamplesMixedOut",
3083 STAMUNIT_COUNT, "Total output samples mixed.");
3084 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalSamplesLostIn, "TotalSamplesLostIn",
3085 STAMUNIT_COUNT, "Total input samples lost.");
3086 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalSamplesLostOut, "TotalSamplesLostOut",
3087 STAMUNIT_COUNT, "Total output samples lost.");
3088 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalSamplesOut, "TotalSamplesPlayed",
3089 STAMUNIT_COUNT, "Total samples played by backend.");
3090 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalSamplesIn, "TotalSamplesCaptured",
3091 STAMUNIT_COUNT, "Total samples captured by backend.");
3092 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalBytesRead, "TotalBytesRead",
3093 STAMUNIT_BYTES, "Total bytes read.");
3094 PDMDrvHlpSTAMRegCounterEx(pDrvIns, &pThis->Stats.TotalBytesWritten, "TotalBytesWritten",
3095 STAMUNIT_BYTES, "Total bytes written.");
3096
3097 PDMDrvHlpSTAMRegProfileAdvEx(pDrvIns, &pThis->Stats.DelayIn, "DelayIn",
3098 STAMUNIT_NS_PER_CALL, "Profiling of input data processing.");
3099 PDMDrvHlpSTAMRegProfileAdvEx(pDrvIns, &pThis->Stats.DelayOut, "DelayOut",
3100 STAMUNIT_NS_PER_CALL, "Profiling of output data processing.");
3101#endif
3102 }
3103
3104 LogFlowFuncLeaveRC(rc);
3105 return rc;
3106}
3107
3108/**
3109 * Destructs an audio driver instance.
3110 *
3111 * @copydoc FNPDMDRVDESTRUCT
3112 */
3113static DECLCALLBACK(void) drvAudioDestruct(PPDMDRVINS pDrvIns)
3114{
3115 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
3116 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
3117
3118 LogFlowFuncEnter();
3119
3120 int rc2 = RTCritSectEnter(&pThis->CritSect);
3121 AssertRC(rc2);
3122
3123 /*
3124 * Note: No calls here to the driver below us anymore,
3125 * as PDM already has destroyed it.
3126 * If you need to call something from the host driver,
3127 * do this in drvAudioPowerOff() instead.
3128 */
3129
3130 /* Thus, NULL the pointer to the host audio driver first,
3131 * so that routines like drvAudioStreamDestroyInternal() don't call the driver(s) below us anymore. */
3132 pThis->pHostDrvAudio = NULL;
3133
3134 PPDMAUDIOSTREAM pStream, pStreamNext;
3135 RTListForEachSafe(&pThis->lstHstStreams, pStream, pStreamNext, PDMAUDIOSTREAM, Node)
3136 {
3137 rc2 = drvAudioStreamUninitInternal(pThis, pStream);
3138 if (RT_SUCCESS(rc2))
3139 {
3140 RTListNodeRemove(&pStream->Node);
3141
3142 drvAudioStreamFree(pStream);
3143 pStream = NULL;
3144 }
3145 }
3146
3147 /* Sanity. */
3148 Assert(RTListIsEmpty(&pThis->lstHstStreams));
3149
3150 RTListForEachSafe(&pThis->lstGstStreams, pStream, pStreamNext, PDMAUDIOSTREAM, Node)
3151 {
3152 rc2 = drvAudioStreamUninitInternal(pThis, pStream);
3153 if (RT_SUCCESS(rc2))
3154 {
3155 RTListNodeRemove(&pStream->Node);
3156
3157 RTMemFree(pStream);
3158 pStream = NULL;
3159 }
3160 }
3161
3162 /* Sanity. */
3163 Assert(RTListIsEmpty(&pThis->lstGstStreams));
3164
3165#ifdef VBOX_WITH_AUDIO_DEVICE_CALLBACKS
3166 /*
3167 * Destroy device callbacks, if any.
3168 */
3169 PPDMAUDIOCALLBACK pCB, pCBNext;
3170 RTListForEachSafe(&pThis->lstCBIn, pCB, pCBNext, PDMAUDIOCALLBACK, Node)
3171 drvAudioCallbackDestroy(pCB);
3172
3173 RTListForEachSafe(&pThis->lstCBOut, pCB, pCBNext, PDMAUDIOCALLBACK, Node)
3174 drvAudioCallbackDestroy(pCB);
3175#endif
3176
3177 rc2 = RTCritSectLeave(&pThis->CritSect);
3178 AssertRC(rc2);
3179
3180 rc2 = RTCritSectDelete(&pThis->CritSect);
3181 AssertRC(rc2);
3182
3183#ifdef VBOX_WITH_STATISTICS
3184 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalStreamsActive);
3185 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalStreamsCreated);
3186 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalSamplesRead);
3187 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalSamplesWritten);
3188 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalSamplesMixedIn);
3189 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalSamplesMixedOut);
3190 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalSamplesLostIn);
3191 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalSamplesLostOut);
3192 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalSamplesOut);
3193 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalSamplesIn);
3194 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalBytesRead);
3195 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.TotalBytesWritten);
3196 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.DelayIn);
3197 PDMDrvHlpSTAMDeregister(pThis->pDrvIns, &pThis->Stats.DelayOut);
3198#endif
3199
3200 LogFlowFuncLeave();
3201}
3202
3203/**
3204 * Suspend notification.
3205 *
3206 * @param pDrvIns The driver instance data.
3207 */
3208static DECLCALLBACK(void) drvAudioSuspend(PPDMDRVINS pDrvIns)
3209{
3210 drvAudioStateHandler(pDrvIns, PDMAUDIOSTREAMCMD_PAUSE);
3211}
3212
3213/**
3214 * Resume notification.
3215 *
3216 * @param pDrvIns The driver instance data.
3217 */
3218static DECLCALLBACK(void) drvAudioResume(PPDMDRVINS pDrvIns)
3219{
3220 drvAudioStateHandler(pDrvIns, PDMAUDIOSTREAMCMD_RESUME);
3221}
3222
3223/**
3224 * Audio driver registration record.
3225 */
3226const PDMDRVREG g_DrvAUDIO =
3227{
3228 /* u32Version */
3229 PDM_DRVREG_VERSION,
3230 /* szName */
3231 "AUDIO",
3232 /* szRCMod */
3233 "",
3234 /* szR0Mod */
3235 "",
3236 /* pszDescription */
3237 "Audio connector driver",
3238 /* fFlags */
3239 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
3240 /* fClass */
3241 PDM_DRVREG_CLASS_AUDIO,
3242 /* cMaxInstances */
3243 UINT32_MAX,
3244 /* cbInstance */
3245 sizeof(DRVAUDIO),
3246 /* pfnConstruct */
3247 drvAudioConstruct,
3248 /* pfnDestruct */
3249 drvAudioDestruct,
3250 /* pfnRelocate */
3251 NULL,
3252 /* pfnIOCtl */
3253 NULL,
3254 /* pfnPowerOn */
3255 NULL,
3256 /* pfnReset */
3257 NULL,
3258 /* pfnSuspend */
3259 drvAudioSuspend,
3260 /* pfnResume */
3261 drvAudioResume,
3262 /* pfnAttach */
3263 NULL,
3264 /* pfnDetach */
3265 NULL,
3266 /* pfnPowerOff */
3267 drvAudioPowerOff,
3268 /* pfnSoftReset */
3269 NULL,
3270 /* u32EndVersion */
3271 PDM_DRVREG_VERSION
3272};
3273
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