VirtualBox

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

Last change on this file since 64402 was 64379, checked in by vboxsync, 8 years ago

Audio/DrvAudio.cpp: Don't create internal stream objects if backend failed to initialize.

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