VirtualBox

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

Last change on this file since 57146 was 57054, checked in by vboxsync, 9 years ago

Devices/Audio/DrvAudio.cpp: Fix cleanup logic, as having it in the destruct callback is too late (it uses the driver below, which is already destructed at this point). Kind of a use after free, but for VM poweroff the memory is freed later (it crashed when constructing failed as then the driver below is freed before destructing this one).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 58.6 KB
Line 
1/* $Id: DrvAudio.cpp 57054 2015-07-22 11:38:49Z vboxsync $ */
2/** @file
3 * Intermediate audio driver header.
4 *
5 * @remarks Intermediate audio driver having audio device as one of the sink and
6 * host backend as other.
7 */
8
9/*
10 * Copyright (C) 2006-2015 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 * This code is based on: audio.c from QEMU AUDIO subsystem.
22 *
23 * QEMU Audio subsystem
24 *
25 * Copyright (c) 2003-2005 Vassili Karpov (malc)
26 *
27 * Permission is hereby granted, free of charge, to any person obtaining a copy
28 * of this software and associated documentation files (the "Software"), to deal
29 * in the Software without restriction, including without limitation the rights
30 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
31 * copies of the Software, and to permit persons to whom the Software is
32 * furnished to do so, subject to the following conditions:
33 *
34 * The above copyright notice and this permission notice shall be included in
35 * all copies or substantial portions of the Software.
36 *
37 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
38 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
39 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
40 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
41 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
42 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
43 * THE SOFTWARE.
44 */
45#define LOG_GROUP LOG_GROUP_DRV_AUDIO
46#include <VBox/log.h>
47#include <VBox/vmm/pdm.h>
48#include <VBox/err.h>
49#include <VBox/vmm/mm.h>
50#include <VBox/vmm/pdmaudioifs.h>
51
52#include <iprt/alloc.h>
53#include <iprt/asm-math.h>
54#include <iprt/assert.h>
55#include <iprt/circbuf.h>
56#include <iprt/string.h>
57#include <iprt/uuid.h>
58
59#include "VBoxDD.h"
60
61#include <ctype.h>
62#include <stdlib.h>
63
64#include "DrvAudio.h"
65#include "AudioMixBuffer.h"
66
67static int drvAudioDestroyGstIn(PDRVAUDIO pThis, PPDMAUDIOGSTSTRMIN pGstStrmIn);
68
69static int drvAudioAllocHstIn(PDRVAUDIO pThis, const char *pszName, PPDMAUDIOSTREAMCFG pCfg, PDMAUDIORECSOURCE enmRecSource, PPDMAUDIOHSTSTRMIN *ppHstStrmIn);
70static int drvAudioDestroyHstIn(PDRVAUDIO pThis, PPDMAUDIOHSTSTRMIN pHstStrmIn);
71
72int drvAudioAddHstOut(PDRVAUDIO pThis, const char *pszName, PPDMAUDIOSTREAMCFG pCfg, PPDMAUDIOHSTSTRMOUT *ppHstStrmOut)
73{
74 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
75 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
76 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
77
78 PPDMAUDIOHSTSTRMOUT pHstStrmOut;
79
80 int rc;
81 if ( conf.fixed_out.enabled /** @todo Get rid of these settings! */
82 && conf.fixed_out.greedy)
83 {
84 rc = drvAudioAllocHstOut(pThis, pszName, pCfg, &pHstStrmOut);
85 }
86 else
87 rc = VERR_NOT_FOUND;
88
89 if (RT_FAILURE(rc))
90 {
91 pHstStrmOut = drvAudioFindSpecificOut(pThis, NULL, pCfg);
92 if (!pHstStrmOut)
93 {
94 rc = drvAudioAllocHstOut(pThis, pszName, pCfg, &pHstStrmOut);
95 if (RT_FAILURE(rc))
96 pHstStrmOut = drvAudioFindAnyHstOut(pThis, NULL /* pHstStrmOut */);
97 }
98
99 rc = pHstStrmOut ? VINF_SUCCESS : rc;
100 }
101
102 if (RT_SUCCESS(rc))
103 *ppHstStrmOut = pHstStrmOut;
104
105 return rc;
106}
107
108static PDMAUDIOFMT drvAudioGetConfFormat(PCFGMNODE pCfgHandle, const char *pszKey,
109 PDMAUDIOFMT enmDefault, bool *pfDefault)
110{
111 if ( pCfgHandle == NULL
112 || pszKey == NULL)
113 {
114 *pfDefault = true;
115 return enmDefault;
116 }
117
118 char *pszValue = NULL;
119 int rc = CFGMR3QueryStringAlloc(pCfgHandle, pszKey, &pszValue);
120 if (RT_FAILURE(rc))
121 {
122 *pfDefault = true;
123 return enmDefault;
124 }
125
126 PDMAUDIOFMT fmt = drvAudioHlpStringToFormat(pszValue);
127 if (fmt == AUD_FMT_INVALID)
128 {
129 *pfDefault = true;
130 return enmDefault;
131 }
132
133 *pfDefault = false;
134 return fmt;
135}
136
137static int drvAudioGetConfInt(PCFGMNODE pCfgHandle, const char *pszKey,
138 int iDefault, bool *pfDefault)
139{
140
141 if ( pCfgHandle == NULL
142 || pszKey == NULL)
143 {
144 *pfDefault = true;
145 return iDefault;
146 }
147
148 uint64_t u64Data = 0;
149 int rc = CFGMR3QueryInteger(pCfgHandle, pszKey, &u64Data);
150 if (RT_FAILURE(rc))
151 {
152 *pfDefault = true;
153 return iDefault;
154
155 }
156
157 *pfDefault = false;
158 return u64Data;
159}
160
161static const char *drvAudioGetConfStr(PCFGMNODE pCfgHandle, const char *pszKey,
162 const char *pszDefault, bool *pfDefault)
163{
164 if ( pCfgHandle == NULL
165 || pszKey == NULL)
166 {
167 *pfDefault = true;
168 return pszDefault;
169 }
170
171 char *pszValue = NULL;
172 int rc = CFGMR3QueryStringAlloc(pCfgHandle, pszKey, &pszValue);
173 if (RT_FAILURE(rc))
174 {
175 *pfDefault = true;
176 return pszDefault;
177 }
178
179 *pfDefault = false;
180 return pszValue;
181}
182
183static int drvAudioProcessOptions(PCFGMNODE pCfgHandle, const char *pszPrefix, struct audio_option *opt)
184{
185 AssertPtrReturn(pCfgHandle, VERR_INVALID_POINTER);
186 AssertPtrReturn(pszPrefix, VERR_INVALID_POINTER);
187 AssertPtrReturn(opt, VERR_INVALID_POINTER);
188
189 PCFGMNODE pCfgChildHandle = NULL;
190 PCFGMNODE pCfgChildChildHandle = NULL;
191
192 /* If pCfgHandle is NULL, let NULL be passed to get int and get string functions..
193 * The getter function will return default values.
194 */
195 if (pCfgHandle != NULL)
196 {
197 /* If its audio general setting, need to traverse to one child node.
198 * /Devices/ichac97/0/LUN#0/Config/Audio
199 */
200 if(!strncmp(pszPrefix, "AUDIO", 5)) /** @todo Use a #define */
201 {
202 pCfgChildHandle = CFGMR3GetFirstChild(pCfgHandle);
203 if(pCfgChildHandle)
204 pCfgHandle = pCfgChildHandle;
205 }
206 else
207 {
208 /* If its driver specific configuration , then need to traverse two level deep child
209 * child nodes. for eg. in case of DirectSoundConfiguration item
210 * /Devices/ichac97/0/LUN#0/Config/Audio/DirectSoundConfig
211 */
212 pCfgChildHandle = CFGMR3GetFirstChild(pCfgHandle);
213 if (pCfgChildHandle)
214 {
215 pCfgChildChildHandle = CFGMR3GetFirstChild(pCfgChildHandle);
216 if (pCfgChildChildHandle)
217 pCfgHandle = pCfgChildChildHandle;
218 }
219 }
220 }
221
222 for (; opt->name; opt++)
223 {
224 LogFlowFunc(("Option value pointer for `%s' is not set\n",
225 opt->name));
226 if (!opt->valp) {
227 LogFlowFunc(("Option value pointer for `%s' is not set\n",
228 opt->name));
229 continue;
230 }
231
232 bool fUseDefault;
233
234 switch (opt->tag)
235 {
236 case AUD_OPT_BOOL:
237 case AUD_OPT_INT:
238 {
239 int *intp = (int *)opt->valp;
240 *intp = drvAudioGetConfInt(pCfgHandle, opt->name, *intp, &fUseDefault);
241
242 break;
243 }
244
245 case AUD_OPT_FMT:
246 {
247 PDMAUDIOFMT *fmtp = (PDMAUDIOFMT *)opt->valp;
248 *fmtp = drvAudioGetConfFormat(pCfgHandle, opt->name, *fmtp, &fUseDefault);
249
250 break;
251 }
252
253 case AUD_OPT_STR:
254 {
255 const char **strp = (const char **)opt->valp;
256 *strp = drvAudioGetConfStr(pCfgHandle, opt->name, *strp, &fUseDefault);
257
258 break;
259 }
260
261 default:
262 LogFlowFunc(("Bad value tag for option `%s' - %d\n", opt->name, opt->tag));
263 fUseDefault = false;
264 break;
265 }
266
267 if (!opt->overridenp)
268 opt->overridenp = &opt->overriden;
269
270 *opt->overridenp = !fUseDefault;
271 }
272
273 return VINF_SUCCESS;
274}
275
276static bool drvAudioStreamCfgIsValid(PPDMAUDIOSTREAMCFG pCfg)
277{
278 bool fValid = ( pCfg->cChannels == 1
279 || pCfg->cChannels == 2); /* Either stereo (2) or mono (1), per stream. */
280
281 fValid |= ( pCfg->enmEndianness == PDMAUDIOENDIANNESS_LITTLE
282 || pCfg->enmEndianness == PDMAUDIOENDIANNESS_BIG);
283
284 if (fValid)
285 {
286 switch (pCfg->enmFormat)
287 {
288 case AUD_FMT_S8:
289 case AUD_FMT_U8:
290 case AUD_FMT_S16:
291 case AUD_FMT_U16:
292 case AUD_FMT_S32:
293 case AUD_FMT_U32:
294 break;
295 default:
296 fValid = false;
297 break;
298 }
299 }
300
301 /** @todo Check for defined frequencies supported. */
302 fValid |= pCfg->uHz > 0;
303
304#ifdef DEBUG
305 drvAudioStreamCfgPrint(pCfg);
306#endif
307
308 LogFlowFunc(("pCfg=%p, fValid=%RTbool\n", pCfg, fValid));
309 return fValid;
310}
311
312void audio_pcm_info_clear_buf(PPDMPCMPROPS pPCMInfo, void *pvBuf, int len)
313{
314 if (!len)
315 return;
316
317 if (pPCMInfo->fSigned)
318 {
319 memset (pvBuf, 0, len << pPCMInfo->cShift);
320 }
321 else
322 {
323 switch (pPCMInfo->cBits)
324 {
325
326 case 8:
327 memset (pvBuf, 0x80, len << pPCMInfo->cShift);
328 break;
329
330 case 16:
331 {
332 int i;
333 uint16_t *p = (uint16_t *)pvBuf;
334 int shift = pPCMInfo->cChannels - 1;
335 short s = INT16_MAX;
336
337 if (pPCMInfo->fSwapEndian)
338 s = RT_BSWAP_U16(s);
339
340 for (i = 0; i < len << shift; i++)
341 p[i] = s;
342 }
343 break;
344
345 case 32:
346 {
347 int i;
348 uint32_t *p = (uint32_t *)pvBuf;
349 int shift = pPCMInfo->cChannels - 1;
350 int32_t s = INT32_MAX;
351
352 if (pPCMInfo->fSwapEndian)
353 s = RT_BSWAP_U32(s);
354
355 for (i = 0; i < len << shift; i++)
356 p[i] = s;
357 }
358 break;
359
360 default:
361 LogFlowFunc(("audio_pcm_info_clear_buf: invalid bits %d\n", pPCMInfo->cBits));
362 break;
363 }
364 }
365}
366
367int drvAudioDestroyHstOut(PDRVAUDIO pThis, PPDMAUDIOHSTSTRMOUT pHstStrmOut)
368{
369 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
370 AssertPtrReturn(pHstStrmOut, VERR_INVALID_POINTER);
371
372 LogFlowFunc(("%s\n", pHstStrmOut->MixBuf.pszName));
373
374 int rc;
375 if (RTListIsEmpty(&pHstStrmOut->lstGstStrmOut))
376 {
377 rc = pThis->pHostDrvAudio->pfnFiniOut(pThis->pHostDrvAudio, pHstStrmOut);
378 if (RT_SUCCESS(rc))
379 {
380 drvAudioHstOutFreeRes(pHstStrmOut);
381
382 /* Remove from driver instance list. */
383 RTListNodeRemove(&pHstStrmOut->Node);
384
385 RTMemFree(pHstStrmOut);
386 pThis->cFreeOutputStreams++;
387 return VINF_SUCCESS;
388 }
389 }
390 else
391 {
392 rc = VERR_ACCESS_DENIED;
393 LogFlowFunc(("[%s] Still is being used, rc=%Rrc\n", pHstStrmOut->MixBuf.pszName, rc));
394 }
395
396 return rc;
397}
398
399int drvAudioDestroyGstOut(PDRVAUDIO pThis, PPDMAUDIOGSTSTRMOUT pGstStrmOut)
400{
401 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
402
403 if (pGstStrmOut)
404 {
405 drvAudioGstOutFreeRes(pGstStrmOut);
406
407 if (pGstStrmOut->pHstStrmOut)
408 {
409 /* Unregister from parent first. */
410 RTListNodeRemove(&pGstStrmOut->Node);
411
412 /* Try destroying the associated host output stream. This could
413 * be skipped if there are other guest output streams with this
414 * host stream. */
415 drvAudioDestroyHstOut(pThis, pGstStrmOut->pHstStrmOut);
416 }
417
418 RTMemFree(pGstStrmOut);
419 }
420
421 return VINF_SUCCESS;
422}
423
424PPDMAUDIOHSTSTRMIN drvAudioFindNextHstIn(PDRVAUDIO pThis, PPDMAUDIOHSTSTRMIN pHstStrmIn)
425{
426 if (pHstStrmIn)
427 {
428 if (RTListNodeIsLast(&pThis->lstHstStrmIn, &pHstStrmIn->Node))
429 return NULL;
430
431 return RTListNodeGetNext(&pHstStrmIn->Node, PDMAUDIOHSTSTRMIN, Node);
432 }
433
434 return RTListGetFirst(&pThis->lstHstStrmIn, PDMAUDIOHSTSTRMIN, Node);
435}
436
437PPDMAUDIOHSTSTRMIN drvAudioFindNextEnabledHstIn(PDRVAUDIO pThis, PPDMAUDIOHSTSTRMIN pHstStrmIn)
438{
439 while ((pHstStrmIn = drvAudioFindNextHstIn(pThis, pHstStrmIn)))
440 if (pHstStrmIn->fEnabled)
441 return pHstStrmIn;
442
443 return NULL;
444}
445
446PPDMAUDIOHSTSTRMIN drvAudioFindNextEqHstIn(PDRVAUDIO pThis, PPDMAUDIOHSTSTRMIN pHstStrmIn,
447 PPDMAUDIOSTREAMCFG pCfg)
448{
449 while ((pHstStrmIn = drvAudioFindNextHstIn(pThis, pHstStrmIn)))
450 if (drvAudioPCMPropsAreEqual(&pHstStrmIn->Props, pCfg))
451 return pHstStrmIn;
452
453 return NULL;
454}
455
456static int drvAudioHstInAdd(PDRVAUDIO pThis, const char *pszName, PPDMAUDIOSTREAMCFG pCfg, PDMAUDIORECSOURCE enmRecSource,
457 PPDMAUDIOHSTSTRMIN *ppHstStrmIn)
458{
459 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
460 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
461 AssertPtrReturn(ppHstStrmIn, VERR_INVALID_POINTER);
462
463 PPDMAUDIOHSTSTRMIN pHstStrmIn;
464 int rc = drvAudioAllocHstIn(pThis, pszName, pCfg, enmRecSource, &pHstStrmIn);
465 if (RT_SUCCESS(rc))
466 *ppHstStrmIn = pHstStrmIn;
467
468 LogFlowFuncLeaveRC(rc);
469 return rc;
470}
471
472int drvAudioGstOutInit(PPDMAUDIOGSTSTRMOUT pGstStrmOut, PPDMAUDIOHSTSTRMOUT pHostStrmOut,
473 const char *pszName, PPDMAUDIOSTREAMCFG pCfg)
474{
475 AssertPtrReturn(pGstStrmOut, VERR_INVALID_POINTER);
476 AssertPtrReturn(pHostStrmOut, VERR_INVALID_POINTER);
477 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
478 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
479
480 int rc = drvAudioStreamCfgToProps(pCfg, &pGstStrmOut->Props);
481 if (RT_SUCCESS(rc))
482 {
483 char *pszTemp;
484 if (RTStrAPrintf(&pszTemp, "%s (Guest)", pszName) <= 0)
485 return VERR_NO_MEMORY;
486
487 rc = AudioMixBufInit(&pGstStrmOut->MixBuf, pszTemp, &pGstStrmOut->Props, AudioMixBufSize(&pHostStrmOut->MixBuf));
488 if (RT_SUCCESS(rc))
489 rc = AudioMixBufLinkTo(&pGstStrmOut->MixBuf, &pHostStrmOut->MixBuf);
490
491 RTStrFree(pszTemp);
492
493 if (RT_SUCCESS(rc))
494 {
495 pGstStrmOut->State.fActive = false;
496 pGstStrmOut->State.fEmpty = true;
497
498 pGstStrmOut->State.pszName = RTStrDup(pszName);
499 if (!pGstStrmOut->State.pszName)
500 return VERR_NO_MEMORY;
501
502 pGstStrmOut->pHstStrmOut = pHostStrmOut;
503 }
504 }
505
506 LogFlowFunc(("pszName=%s, rc=%Rrc\n", pszName, rc));
507 return rc;
508}
509
510int drvAudioAllocHstOut(PDRVAUDIO pThis, const char *pszName, PPDMAUDIOSTREAMCFG pCfg, PPDMAUDIOHSTSTRMOUT *ppHstStrmOut)
511{
512 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
513 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
514 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
515
516 if (!pThis->cFreeOutputStreams)
517 {
518 LogFlowFunc(("Maximum number of host output streams reached\n"));
519 return VERR_NO_MORE_HANDLES;
520 }
521
522 /* Validate backend configuration. */
523 if (!pThis->BackendCfg.cbStreamOut)
524 {
525 LogFlowFunc(("Backend output configuration not valid, bailing out\n"));
526 return VERR_INVALID_PARAMETER;
527 }
528
529 PPDMAUDIOHSTSTRMOUT pHstStrmOut = (PPDMAUDIOHSTSTRMOUT)RTMemAllocZ(pThis->BackendCfg.cbStreamOut);
530 if (!pHstStrmOut)
531 {
532 LogFlowFunc(("Error allocating host output stream with %zu bytes\n",
533 pThis->BackendCfg.cbStreamOut));
534 return VERR_NO_MEMORY;
535 }
536
537 int rc;
538 bool fInitialized = false;
539
540 do
541 {
542 RTListInit(&pHstStrmOut->lstGstStrmOut);
543
544 uint32_t cSamples;
545 rc = pThis->pHostDrvAudio->pfnInitOut(pThis->pHostDrvAudio, pHstStrmOut, pCfg, &cSamples);
546 if (RT_FAILURE(rc))
547 {
548 LogFlowFunc(("Initializing host backend failed with rc=%Rrc\n", rc));
549 break;
550 }
551
552 fInitialized = true;
553
554 char *pszTemp;
555 if (RTStrAPrintf(&pszTemp, "%s (Host)", pszName) <= 0)
556 {
557 rc = VERR_NO_MEMORY;
558 break;
559 }
560
561 rc = AudioMixBufInit(&pHstStrmOut->MixBuf, pszTemp, &pHstStrmOut->Props, cSamples);
562 if (RT_SUCCESS(rc))
563 {
564 RTListPrepend(&pThis->lstHstStrmOut, &pHstStrmOut->Node);
565 pThis->cFreeOutputStreams--;
566 }
567
568 RTStrFree(pszTemp);
569
570 } while (0);
571
572 if (RT_FAILURE(rc))
573 {
574 if (fInitialized)
575 {
576 int rc2 = pThis->pHostDrvAudio->pfnFiniOut(pThis->pHostDrvAudio, pHstStrmOut);
577 AssertRC(rc2);
578 }
579
580 drvAudioHstOutFreeRes(pHstStrmOut);
581 RTMemFree(pHstStrmOut);
582 }
583 else
584 *ppHstStrmOut = pHstStrmOut;
585
586 LogFlowFuncLeaveRC(rc);
587 return rc;
588}
589
590int drvAudioCreateStreamPairOut(PDRVAUDIO pThis, const char *pszName,
591 PPDMAUDIOSTREAMCFG pCfg, PPDMAUDIOGSTSTRMOUT *ppGstStrmOut)
592{
593 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
594 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
595 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
596
597 /*
598 * Try figuring out which audio stream configuration this backend
599 * should use. If fixed output is enabled the backend will be tied
600 * to a fixed rate (in Hz, among other parameters), regardless of
601 * what the backend could do else.
602 */
603 PPDMAUDIOSTREAMCFG pBackendCfg;
604 if (conf.fixed_out.enabled)
605 pBackendCfg = &conf.fixed_out.settings;
606 else
607 pBackendCfg = pCfg;
608
609 AssertPtrReturn(pBackendCfg, VERR_INVALID_POINTER);
610
611 LogFlowFunc(("Using fixed audio output settings: %RTbool\n",
612 RT_BOOL(conf.fixed_out.enabled)));
613
614 PPDMAUDIOGSTSTRMOUT pGstStrmOut =
615 (PPDMAUDIOGSTSTRMOUT)RTMemAllocZ(sizeof(PDMAUDIOGSTSTRMOUT));
616 if (!pGstStrmOut)
617 {
618 LogFlowFunc(("Failed to allocate memory for guest output stream \"%s\"\n", pszName));
619 return VERR_NO_MEMORY;
620 }
621
622 /*
623 * The host stream always will get the backend audio stream configuration.
624 */
625 PPDMAUDIOHSTSTRMOUT pHstStrmOut;
626 int rc = drvAudioAddHstOut(pThis, pszName, pBackendCfg, &pHstStrmOut);
627 if (RT_FAILURE(rc))
628 {
629 LogFlowFunc(("Error adding host output stream \"%s\", rc=%Rrc\n", pszName, rc));
630
631 RTMemFree(pGstStrmOut);
632 return rc;
633 }
634
635 /*
636 * The guest stream always will get the audio stream configuration told
637 * by the device emulation (which in turn was/could be set by the guest OS).
638 */
639 rc = drvAudioGstOutInit(pGstStrmOut, pHstStrmOut, pszName, pCfg);
640 if (RT_SUCCESS(rc))
641 {
642 RTListPrepend(&pHstStrmOut->lstGstStrmOut, &pGstStrmOut->Node);
643
644 if (ppGstStrmOut)
645 *ppGstStrmOut = pGstStrmOut;
646 }
647
648 if (RT_FAILURE(rc))
649 drvAudioDestroyGstOut(pThis, pGstStrmOut);
650
651 LogFlowFuncLeaveRC(rc);
652 return rc;
653}
654
655static int drvAudioCreateStreamPairIn(PDRVAUDIO pThis, const char *pszName, PDMAUDIORECSOURCE enmRecSource,
656 PPDMAUDIOSTREAMCFG pCfg, PPDMAUDIOGSTSTRMIN *ppGstStrmIn)
657{
658 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
659 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
660
661/*
662 * Try figuring out which audio stream configuration this backend
663 * should use for the audio input data. If fixed input is enabled
664 * the backend will be tied to a fixed rate (in Hz, among other parameters),
665 * regardless of what the backend initially wanted to use.
666 */
667 PPDMAUDIOSTREAMCFG pBackendCfg;
668 if (conf.fixed_in.enabled)
669 pBackendCfg = &conf.fixed_in.settings;
670 else
671 pBackendCfg = pCfg;
672
673 AssertPtrReturn(pBackendCfg, VERR_INVALID_POINTER);
674
675 LogFlowFunc(("Using fixed audio input settings: %RTbool\n",
676 RT_BOOL(conf.fixed_in.enabled)));
677
678 PPDMAUDIOGSTSTRMIN pGstStrmIn = (PPDMAUDIOGSTSTRMIN)RTMemAllocZ(sizeof(PDMAUDIOGSTSTRMIN));
679 if (!pGstStrmIn)
680 return VERR_NO_MEMORY;
681
682 /*
683 * The host stream always will get the backend audio stream configuration.
684 */
685 PPDMAUDIOHSTSTRMIN pHstStrmIn;
686 int rc = drvAudioHstInAdd(pThis, pszName, pBackendCfg, enmRecSource, &pHstStrmIn);
687 if (RT_FAILURE(rc))
688 {
689 LogFunc(("Failed to add host audio input stream \"%s\", rc=%Rrc\n", pszName, rc));
690
691 RTMemFree(pGstStrmIn);
692 return rc;
693 }
694
695 /*
696 * The guest stream always will get the audio stream configuration told
697 * by the device emulation (which in turn was/could be set by the guest OS).
698 */
699 rc = drvAudioGstInInit(pGstStrmIn, pHstStrmIn, pszName, pCfg);
700 if (RT_SUCCESS(rc))
701 {
702 pHstStrmIn->pGstStrmIn = pGstStrmIn;
703
704 if (ppGstStrmIn)
705 *ppGstStrmIn = pGstStrmIn;
706 }
707 else
708 drvAudioDestroyGstIn(pThis, pGstStrmIn);
709
710 LogFlowFuncLeaveRC(rc);
711 return rc;
712}
713
714/**
715 * Initializes a guest input stream.
716 *
717 * @return IPRT status code.
718 * @param pGstStrmIn Pointer to guest stream to initialize.
719 * @param pHstStrmIn Pointer to host input stream to associate this guest
720 * stream with.
721 * @param pszName Pointer to stream name to use for this stream.
722 * @param pCfg Pointer to stream configuration to use.
723 */
724int drvAudioGstInInit(PPDMAUDIOGSTSTRMIN pGstStrmIn, PPDMAUDIOHSTSTRMIN pHstStrmIn,
725 const char *pszName, PPDMAUDIOSTREAMCFG pCfg)
726{
727 AssertPtrReturn(pGstStrmIn, VERR_INVALID_POINTER);
728 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
729 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
730 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
731
732 int rc = drvAudioStreamCfgToProps(pCfg, &pGstStrmIn->Props);
733 if (RT_SUCCESS(rc))
734 {
735 char *pszTemp;
736 if (RTStrAPrintf(&pszTemp, "%s (Guest)", pszName) <= 0)
737 return VERR_NO_MEMORY;
738
739 rc = AudioMixBufInit(&pGstStrmIn->MixBuf, pszTemp, &pGstStrmIn->Props, AudioMixBufSize(&pHstStrmIn->MixBuf));
740 if (RT_SUCCESS(rc))
741 rc = AudioMixBufLinkTo(&pHstStrmIn->MixBuf, &pGstStrmIn->MixBuf);
742
743 RTStrFree(pszTemp);
744
745 if (RT_SUCCESS(rc))
746 {
747#ifdef DEBUG
748 drvAudioStreamCfgPrint(pCfg);
749#endif
750 pGstStrmIn->State.fActive = false;
751 pGstStrmIn->State.fEmpty = true;
752
753 pGstStrmIn->State.pszName = RTStrDup(pszName);
754 if (!pGstStrmIn->State.pszName)
755 return VERR_NO_MEMORY;
756
757 pGstStrmIn->pHstStrmIn = pHstStrmIn;
758 }
759 }
760
761 LogFlowFunc(("pszName=%s, rc=%Rrc\n", pszName, rc));
762 return rc;
763}
764
765static int drvAudioAllocHstIn(PDRVAUDIO pThis, const char *pszName, PPDMAUDIOSTREAMCFG pCfg,
766 PDMAUDIORECSOURCE enmRecSource, PPDMAUDIOHSTSTRMIN *ppHstStrmIn)
767{
768 if (!pThis->cFreeInputStreams)
769 {
770 LogFlowFunc(("No more input streams free to use, bailing out\n"));
771 return VERR_NO_MORE_HANDLES;
772 }
773
774 /* Validate backend configuration. */
775 if (!pThis->BackendCfg.cbStreamIn)
776 {
777 LogFlowFunc(("Backend input configuration not valid, bailing out\n"));
778 return VERR_INVALID_PARAMETER;
779 }
780
781 PPDMAUDIOHSTSTRMIN pHstStrmIn =
782 (PPDMAUDIOHSTSTRMIN)RTMemAllocZ(pThis->BackendCfg.cbStreamIn);
783 if (!pHstStrmIn)
784 {
785 LogFlowFunc(("Error allocating host innput stream with %RU32 bytes\n",
786 pThis->BackendCfg.cbStreamOut));
787 return VERR_NO_MEMORY;
788 }
789
790 int rc;
791 bool fInitialized = false;
792
793 do
794 {
795 uint32_t cSamples;
796 rc = pThis->pHostDrvAudio->pfnInitIn(pThis->pHostDrvAudio, pHstStrmIn,
797 pCfg, enmRecSource, &cSamples);
798 if (RT_FAILURE(rc))
799 {
800 LogFlowFunc(("Initializing host backend failed with rc=%Rrc\n", rc));
801 break;
802 }
803
804 fInitialized = true;
805
806 char *pszTemp;
807 if (RTStrAPrintf(&pszTemp, "%s (Host)", pszName) <= 0)
808 {
809 rc = VERR_NO_MEMORY;
810 break;
811 }
812
813 rc = AudioMixBufInit(&pHstStrmIn->MixBuf, pszTemp, &pHstStrmIn->Props, cSamples);
814 if (RT_SUCCESS(rc))
815 {
816 RTListPrepend(&pThis->lstHstStrmIn, &pHstStrmIn->Node);
817 pThis->cFreeInputStreams--;
818 }
819
820 RTStrFree(pszTemp);
821
822 } while (0);
823
824 if (RT_FAILURE(rc))
825 {
826 if (fInitialized)
827 {
828 int rc2 = pThis->pHostDrvAudio->pfnFiniIn(pThis->pHostDrvAudio,
829 pHstStrmIn);
830 AssertRC(rc2);
831 }
832
833 drvAudioHstInFreeRes(pHstStrmIn);
834 RTMemFree(pHstStrmIn);
835 }
836 else
837 *ppHstStrmIn = pHstStrmIn;
838
839 LogFlowFuncLeaveRC(rc);
840 return rc;
841}
842
843/**
844 * Writes VM audio output data from the guest stream into the host stream.
845 * The attached host driver backend then will play out the audio in a
846 * later step then.
847 *
848 * @return IPRT status code.
849 * @return int
850 * @param pThis
851 * @param pGstStrmOut
852 * @param pvBuf
853 * @param cbBuf
854 * @param pcbWritten
855 */
856int drvAudioWrite(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOGSTSTRMOUT pGstStrmOut,
857 const void *pvBuf, uint32_t cbBuf, uint32_t *pcbWritten)
858{
859 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
860 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
861
862 AssertPtrReturn(pGstStrmOut, VERR_INVALID_POINTER);
863 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
864 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
865 /* pcbWritten is optional. */
866
867 if (!pThis->pHostDrvAudio->pfnIsEnabled(pThis->pHostDrvAudio, PDMAUDIODIR_OUT))
868 return VERR_NOT_AVAILABLE;
869
870 PPDMAUDIOHSTSTRMOUT pHstStrmOut = pGstStrmOut->pHstStrmOut;
871 AssertPtrReturn(pHstStrmOut, VERR_INVALID_POINTER);
872
873 AssertMsg(pGstStrmOut->pHstStrmOut->fEnabled,
874 ("Writing to disabled host output stream \"%s\" not possible\n",
875 pHstStrmOut->MixBuf.pszName));
876
877 /*
878 * First, write data from the device emulation into our
879 * guest mixing buffer.
880 */
881 uint32_t cWritten;
882 int rc = AudioMixBufWriteAt(&pGstStrmOut->MixBuf, 0 /* Offset in samples */, pvBuf, cbBuf, &cWritten);
883
884 /*
885 * Second, mix the guest mixing buffer with the host mixing
886 * buffer so that the host backend can play the data lateron.
887 */
888 uint32_t cMixed;
889 if ( RT_SUCCESS(rc)
890 && cWritten)
891 {
892 rc = AudioMixBufMixToParent(&pGstStrmOut->MixBuf, cWritten, &cMixed);
893 }
894 else
895 cMixed = 0;
896
897 if (RT_SUCCESS(rc))
898 {
899 /* Return the number of samples which actually have been mixed
900 * down to the parent, regardless how much samples were written
901 * into the children buffer. */
902 if (pcbWritten)
903 *pcbWritten = AUDIOMIXBUF_S2B(&pGstStrmOut->MixBuf, cMixed);
904 }
905
906 LogFlowFunc(("%s -> %s: Written pvBuf=%p, cbBuf=%zu, cWritten=%RU32 (%RU32 bytes), cMixed=%RU32, rc=%Rrc\n",
907 pGstStrmOut->MixBuf.pszName, pHstStrmOut->MixBuf.pszName, pvBuf, cbBuf, cWritten,
908 AUDIOMIXBUF_S2B(&pGstStrmOut->MixBuf, cWritten), cMixed, rc));
909 return rc;
910}
911
912PPDMAUDIOHSTSTRMOUT drvAudioFindAnyHstOut(PDRVAUDIO pThis, PPDMAUDIOHSTSTRMOUT pHstStrmOut)
913{
914 if (pHstStrmOut)
915 {
916 if (RTListNodeIsLast(&pThis->lstHstStrmOut, &pHstStrmOut->Node))
917 return NULL;
918
919 return RTListNodeGetNext(&pHstStrmOut->Node, PDMAUDIOHSTSTRMOUT, Node);
920 }
921
922 return RTListGetFirst(&pThis->lstHstStrmOut, PDMAUDIOHSTSTRMOUT, Node);
923}
924
925PPDMAUDIOHSTSTRMOUT drvAudioHstFindAnyEnabledOut(PDRVAUDIO pThis, PPDMAUDIOHSTSTRMOUT pHostStrmOut)
926{
927 while ((pHostStrmOut = drvAudioFindAnyHstOut(pThis, pHostStrmOut)))
928 {
929 if (pHostStrmOut->fEnabled)
930 return pHostStrmOut;
931 }
932
933 return NULL;
934}
935
936PPDMAUDIOHSTSTRMOUT drvAudioFindSpecificOut(PDRVAUDIO pThis, PPDMAUDIOHSTSTRMOUT pHstStrmOut,
937 PPDMAUDIOSTREAMCFG pCfg)
938{
939 while ((pHstStrmOut = drvAudioFindAnyHstOut(pThis, pHstStrmOut)))
940 {
941 if (drvAudioPCMPropsAreEqual(&pHstStrmOut->Props, pCfg))
942 return pHstStrmOut;
943 }
944
945 return NULL;
946}
947
948int drvAudioDestroyHstIn(PDRVAUDIO pThis, PPDMAUDIOHSTSTRMIN pHstStrmIn)
949{
950 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
951 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
952
953 LogFlowFunc(("%s\n", pHstStrmIn->MixBuf.pszName));
954
955 int rc;
956 if (!pHstStrmIn->pGstStrmIn) /* No parent anymore? */
957 {
958 rc = pThis->pHostDrvAudio->pfnFiniIn(pThis->pHostDrvAudio, pHstStrmIn);
959 if (RT_SUCCESS(rc))
960 {
961 drvAudioHstInFreeRes(pHstStrmIn);
962
963 /* Remove from driver instance list. */
964 RTListNodeRemove(&pHstStrmIn->Node);
965
966 RTMemFree(pHstStrmIn);
967 pThis->cFreeInputStreams++;
968 }
969 }
970 else
971 {
972 rc = VERR_ACCESS_DENIED;
973 LogFlowFunc(("[%s] Still is being used, rc=%Rrc\n", pHstStrmIn->MixBuf.pszName, rc));
974 }
975
976 return rc;
977}
978
979static int drvAudioDestroyGstIn(PDRVAUDIO pThis, PPDMAUDIOGSTSTRMIN pGstStrmIn)
980{
981 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
982
983 LogFlowFunc(("%s\n", pGstStrmIn->MixBuf.pszName));
984
985 if (pGstStrmIn)
986 {
987 drvAudioGstInFreeRes(pGstStrmIn);
988
989 if (pGstStrmIn->pHstStrmIn)
990 {
991 /* Unlink child. */
992 pGstStrmIn->pHstStrmIn->pGstStrmIn = NULL;
993
994 /* Try destroying the associated host input stream. This could
995 * be skipped if there are other guest input streams with this
996 * host stream. */
997 drvAudioDestroyHstIn(pThis, pGstStrmIn->pHstStrmIn);
998 }
999
1000 RTMemFree(pGstStrmIn);
1001 }
1002
1003 return VINF_SUCCESS;
1004}
1005
1006static DECLCALLBACK(int) drvAudioQueryStatus(PPDMIAUDIOCONNECTOR pInterface,
1007 uint32_t *pcbAvailIn, uint32_t *pcbFreeOut,
1008 uint32_t *pcSamplesLive)
1009{
1010 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1011 /* pcbAvailIn is optional. */
1012 /* pcbFreeOut is optional. */
1013 /* pcSamplesLive is optional. */
1014
1015 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1016
1017 if (!pThis->pHostDrvAudio->pfnIsEnabled(pThis->pHostDrvAudio, PDMAUDIODIR_OUT))
1018 return VERR_NOT_AVAILABLE;
1019
1020 int rc = VINF_SUCCESS;
1021 uint32_t cSamplesLive = 0;
1022
1023 /*
1024 * Playback.
1025 */
1026 uint32_t cbFreeOut = UINT32_MAX;
1027
1028 PPDMAUDIOHSTSTRMOUT pHstStrmOut = NULL;
1029 while ((pHstStrmOut = drvAudioHstFindAnyEnabledOut(pThis, pHstStrmOut)))
1030 {
1031 uint32_t cStreamsLive;
1032 cSamplesLive = drvAudioHstOutSamplesLive(pHstStrmOut, &cStreamsLive);
1033 if (!cStreamsLive)
1034 cSamplesLive = 0;
1035
1036 /* Has this stream marked as disabled but there still were guest streams relying
1037 * on it? Check if this stream now can be closed and do so, if possible. */
1038 if ( pHstStrmOut->fPendingDisable
1039 && !cStreamsLive)
1040 {
1041 /* Stop playing the current (pending) stream. */
1042 int rc2 = pThis->pHostDrvAudio->pfnControlOut(pThis->pHostDrvAudio, pHstStrmOut,
1043 PDMAUDIOSTREAMCMD_DISABLE);
1044 if (RT_SUCCESS(rc2))
1045 {
1046 pHstStrmOut->fEnabled = false;
1047 pHstStrmOut->fPendingDisable = false;
1048
1049 LogFunc(("[%s] Disabling stream\n", pHstStrmOut->MixBuf.pszName));
1050 }
1051 else
1052 LogFunc(("[%s] Backend vetoed against closing output stream, rc=%Rrc\n", pHstStrmOut->MixBuf.pszName, rc2));
1053
1054 continue;
1055 }
1056
1057 LogFlowFunc(("[%s] cSamplesLive=%RU32\n", pHstStrmOut->MixBuf.pszName, cSamplesLive));
1058
1059 /*
1060 * No live samples to play at the moment?
1061 *
1062 * Tell the device emulation for each connected guest stream how many
1063 * bytes are free so that the device emulation can continue writing data to
1064 * these streams.
1065 */
1066 PPDMAUDIOGSTSTRMOUT pGstStrmOut;
1067 uint32_t cbFree2 = UINT32_MAX;
1068 RTListForEach(&pHstStrmOut->lstGstStrmOut, pGstStrmOut, PDMAUDIOGSTSTRMOUT, Node)
1069 {
1070 if (pGstStrmOut->State.fActive)
1071 {
1072 /* Tell the sound device emulation how many samples are free
1073 * so that it can start writing PCM data to us. */
1074 cbFree2 = RT_MIN(cbFree2, AUDIOMIXBUF_S2B_RATIO(&pGstStrmOut->MixBuf,
1075 AudioMixBufFree(&pGstStrmOut->MixBuf)));
1076
1077 LogFlowFunc(("\t[%s] cbFree=%RU32\n", pGstStrmOut->MixBuf.pszName, cbFree2));
1078 }
1079 }
1080
1081 cbFreeOut = RT_MIN(cbFreeOut, cbFree2);
1082 }
1083
1084 /*
1085 * Recording.
1086 */
1087 uint32_t cbAvailIn = 0;
1088
1089 PPDMAUDIOHSTSTRMIN pHstStrmIn = NULL;
1090 while ((pHstStrmIn = drvAudioFindNextEnabledHstIn(pThis, pHstStrmIn)))
1091 {
1092 /* Call the host backend to capture the audio input data. */
1093 uint32_t cSamplesCaptured;
1094 int rc2 = pThis->pHostDrvAudio->pfnCaptureIn(pThis->pHostDrvAudio, pHstStrmIn,
1095 &cSamplesCaptured);
1096 if (RT_FAILURE(rc2))
1097 continue;
1098
1099 PPDMAUDIOGSTSTRMIN pGstStrmIn = pHstStrmIn->pGstStrmIn;
1100 AssertPtrBreak(pGstStrmIn);
1101
1102 if (pGstStrmIn->State.fActive)
1103 {
1104 cbAvailIn = RT_MAX(cbAvailIn, AUDIOMIXBUF_S2B(&pHstStrmIn->MixBuf,
1105 AudioMixBufMixed(&pHstStrmIn->MixBuf)));
1106
1107 LogFlowFunc(("\t[%s] cbFree=%RU32\n", pHstStrmIn->MixBuf.pszName, cbAvailIn));
1108 }
1109 }
1110
1111 if (RT_SUCCESS(rc))
1112 {
1113 if (cbFreeOut == UINT32_MAX)
1114 cbFreeOut = 0;
1115
1116 if (pcbAvailIn)
1117 *pcbAvailIn = cbAvailIn;
1118
1119 if (pcbFreeOut)
1120 *pcbFreeOut = cbFreeOut;
1121
1122 if (pcSamplesLive)
1123 *pcSamplesLive = cSamplesLive;
1124 }
1125
1126 return rc;
1127}
1128
1129static DECLCALLBACK(int) drvAudioPlayOut(PPDMIAUDIOCONNECTOR pInterface, uint32_t *pcSamplesPlayed)
1130{
1131 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1132 /* pcSamplesPlayed is optional. */
1133
1134 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1135
1136 int rc = VINF_SUCCESS;
1137 uint32_t cSamplesPlayedMax = 0;
1138
1139 /*
1140 * Process all enabled host output streams.
1141 */
1142 PPDMAUDIOHSTSTRMOUT pHstStrmOut = NULL;
1143 while ((pHstStrmOut = drvAudioHstFindAnyEnabledOut(pThis, pHstStrmOut)))
1144 {
1145#if 0
1146 uint32_t cStreamsLive;
1147 uint32_t cSamplesLive = drvAudioHstOutSamplesLive(pHstStrmOut, &cStreamsLive);
1148 if (!cStreamsLive)
1149 cSamplesLive = 0;
1150
1151 /* Has this stream marked as disabled but there still were guest streams relying
1152 * on it? Check if this stream now can be closed and do so, if possible. */
1153 if ( pHstStrmOut->fPendingDisable
1154 && !cStreamsLive)
1155 {
1156 /* Stop playing the current (pending) stream. */
1157 int rc2 = pThis->pHostDrvAudio->pfnControlOut(pThis->pHostDrvAudio, pHstStrmOut,
1158 PDMAUDIOSTREAMCMD_DISABLE);
1159 if (RT_SUCCESS(rc2))
1160 {
1161 pHstStrmOut->fEnabled = false;
1162 pHstStrmOut->fPendingDisable = false;
1163
1164 LogFunc(("\t%p: Disabling stream\n", pHstStrmOut));
1165 }
1166 else
1167 LogFunc(("\t%p: Backend vetoed against closing output stream, rc=%Rrc\n",
1168 pHstStrmOut, rc2));
1169
1170 continue;
1171 }
1172#endif
1173
1174 uint32_t cSamplesPlayed = 0;
1175 int rc2 = pThis->pHostDrvAudio->pfnPlayOut(pThis->pHostDrvAudio, pHstStrmOut,
1176 &cSamplesPlayed);
1177 if (RT_SUCCESS(rc2))
1178 cSamplesPlayedMax = RT_MAX(cSamplesPlayed, cSamplesPlayedMax);
1179
1180 LogFlowFunc(("\t[%s] cSamplesPlayed=%RU32, rc=%Rrc\n", pHstStrmOut->MixBuf.pszName, cSamplesPlayed, rc2));
1181
1182 bool fNeedsCleanup = false;
1183
1184 PPDMAUDIOGSTSTRMOUT pGstStrmOut;
1185 RTListForEach(&pHstStrmOut->lstGstStrmOut, pGstStrmOut, PDMAUDIOGSTSTRMOUT, Node)
1186 {
1187 if ( !pGstStrmOut->State.fActive
1188 && pGstStrmOut->State.fEmpty)
1189 continue;
1190
1191 if (AudioMixBufIsEmpty(&pGstStrmOut->MixBuf))
1192 {
1193 pGstStrmOut->State.fEmpty = true;
1194 fNeedsCleanup |= !pGstStrmOut->State.fActive;
1195 }
1196 }
1197
1198 if (fNeedsCleanup)
1199 {
1200 RTListForEach(&pHstStrmOut->lstGstStrmOut, pGstStrmOut, PDMAUDIOGSTSTRMOUT, Node)
1201 {
1202 if (!pGstStrmOut->State.fActive)
1203 drvAudioDestroyGstOut(pThis, pGstStrmOut);
1204 }
1205 }
1206 }
1207
1208 if (RT_SUCCESS(rc))
1209 {
1210 if (pcSamplesPlayed)
1211 *pcSamplesPlayed = cSamplesPlayedMax;
1212 }
1213
1214 return rc;
1215}
1216
1217static int drvAudioHostInit(PCFGMNODE pCfgHandle, PDRVAUDIO pThis)
1218{
1219 /* pCfgHandle is optional. */
1220 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1221
1222 NOREF(pCfgHandle);
1223
1224 LogFlowFuncEnter();
1225
1226 int rc = pThis->pHostDrvAudio->pfnInit(pThis->pHostDrvAudio);
1227 if (RT_FAILURE(rc))
1228 {
1229 LogFlowFunc(("Initialization of lower driver failed with rc=%Rrc\n", rc));
1230 return rc;
1231 }
1232
1233 uint32_t cMaxHstStrmsOut = pThis->BackendCfg.cMaxHstStrmsOut;
1234 uint32_t cbHstStrmsOut = pThis->BackendCfg.cbStreamOut;
1235
1236 if (cbHstStrmsOut)
1237 {
1238 pThis->cFreeOutputStreams = 1; /** @todo Make this configurable. */
1239 if (pThis->cFreeOutputStreams > cMaxHstStrmsOut)
1240 {
1241 LogRel(("Audio: Warning: %RU32 output streams were requested, host driver only supports %RU32\n",
1242 pThis->cFreeOutputStreams, cMaxHstStrmsOut));
1243 pThis->cFreeOutputStreams = cMaxHstStrmsOut;
1244 }
1245 }
1246 else
1247 pThis->cFreeOutputStreams = 0;
1248
1249 uint32_t cMaxHstStrmsIn = pThis->BackendCfg.cMaxHstStrmsIn;
1250 uint32_t cbHstStrmIn = pThis->BackendCfg.cbStreamIn;
1251
1252 if (cbHstStrmIn)
1253 {
1254 /*
1255 * Note:
1256 * - Our AC'97 emulation has two inputs, line (P.IN) and microphone (P.MIC).
1257 ** @todo Document HDA.
1258 */
1259 pThis->cFreeInputStreams = 2; /** @todo Make this configurable. */
1260 if (pThis->cFreeInputStreams > cMaxHstStrmsIn)
1261 {
1262 LogRel(("Audio: Warning: %RU32 input streams were requested, host driver only supports %RU32\n",
1263 pThis->cFreeInputStreams, cMaxHstStrmsIn));
1264 pThis->cFreeInputStreams = cMaxHstStrmsIn;
1265 }
1266 }
1267 else
1268 pThis->cFreeInputStreams = 0;
1269
1270 LogFlowFunc(("cMaxHstStrmsOut=%RU32 (cb=%RU32), cMaxHstStrmsIn=%RU32 (cb=%RU32)\n",
1271 cMaxHstStrmsOut, cbHstStrmsOut, cMaxHstStrmsIn, cbHstStrmIn));
1272
1273 LogFlowFunc(("cFreeInputStreams=%RU8, cFreeOutputStreams=%RU8\n",
1274 pThis->cFreeInputStreams, pThis->cFreeOutputStreams));
1275
1276 LogFlowFuncLeave();
1277 return VINF_SUCCESS;
1278}
1279
1280static void drvAudioStateHandler(PPDMDRVINS pDrvIns, PDMAUDIOSTREAMCMD enmCmd)
1281{
1282 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
1283
1284 LogFlowFunc(("enmCmd=%ld\n", enmCmd));
1285
1286 if (!pThis->pHostDrvAudio)
1287 return;
1288
1289 PPDMAUDIOHSTSTRMOUT pHstStrmOut = NULL;
1290 while ((pHstStrmOut = drvAudioHstFindAnyEnabledOut(pThis, pHstStrmOut)))
1291 pThis->pHostDrvAudio->pfnControlOut(pThis->pHostDrvAudio, pHstStrmOut, enmCmd);
1292
1293 PPDMAUDIOHSTSTRMIN pHstStrmIn = NULL;
1294 while ((pHstStrmIn = drvAudioFindNextEnabledHstIn(pThis, pHstStrmIn)))
1295 pThis->pHostDrvAudio->pfnControlIn(pThis->pHostDrvAudio, pHstStrmIn, enmCmd);
1296}
1297
1298static struct audio_option audio_options[] =
1299{
1300 /* DAC */
1301 {"DACFixedSettings", AUD_OPT_BOOL, &conf.fixed_out.enabled,
1302 "Use fixed settings for host DAC", NULL, 0},
1303
1304 {"DACFixedFreq", AUD_OPT_INT, &conf.fixed_out.settings.uHz,
1305 "Frequency for fixed host DAC", NULL, 0},
1306
1307 {"DACFixedFmt", AUD_OPT_FMT, &conf.fixed_out.settings.enmFormat,
1308 "Format for fixed host DAC", NULL, 0},
1309
1310 {"DACFixedChannels", AUD_OPT_INT, &conf.fixed_out.settings.cChannels,
1311 "Number of channels for fixed DAC (1 - mono, 2 - stereo)", NULL, 0},
1312
1313 {"DACVoices", AUD_OPT_INT, &conf.fixed_out.cStreams, /** @todo Rename! */
1314 "Number of streams for DAC", NULL, 0},
1315
1316 /* ADC */
1317 {"ADCFixedSettings", AUD_OPT_BOOL, &conf.fixed_in.enabled,
1318 "Use fixed settings for host ADC", NULL, 0},
1319
1320 {"ADCFixedFreq", AUD_OPT_INT, &conf.fixed_in.settings.uHz,
1321 "Frequency for fixed host ADC", NULL, 0},
1322
1323 {"ADCFixedFmt", AUD_OPT_FMT, &conf.fixed_in.settings.enmFormat,
1324 "Format for fixed host ADC", NULL, 0},
1325
1326 {"ADCFixedChannels", AUD_OPT_INT, &conf.fixed_in.settings.cChannels,
1327 "Number of channels for fixed ADC (1 - mono, 2 - stereo)", NULL, 0},
1328
1329 {"ADCVoices", AUD_OPT_INT, &conf.fixed_in.cStreams, /** @todo Rename! */
1330 "Number of streams for ADC", NULL, 0},
1331
1332 /* Misc */
1333 {"TimerFreq", AUD_OPT_INT, &conf.period.hz,
1334 "Timer frequency in Hz (0 - use lowest possible)", NULL, 0},
1335
1336 {"PLIVE", AUD_OPT_BOOL, &conf.plive,
1337 "(undocumented)", NULL, 0}, /** @todo What is this? */
1338
1339 NULL
1340};
1341
1342static DECLCALLBACK(int) drvAudioInit(PCFGMNODE pCfgHandle, PPDMDRVINS pDrvIns)
1343{
1344 AssertPtrReturn(pCfgHandle, VERR_INVALID_POINTER);
1345 AssertPtrReturn(pDrvIns, VERR_INVALID_POINTER);
1346
1347 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
1348 LogFlowFunc(("pDrvAudio=%p, pDrvIns=%p\n", pThis, pDrvIns));
1349
1350 RTListInit(&pThis->lstHstStrmIn);
1351 RTListInit(&pThis->lstHstStrmOut);
1352
1353 int rc = VINF_SUCCESS;
1354
1355 /* Get the configuration data from the selected backend (if available). */
1356 AssertPtr(pThis->pHostDrvAudio);
1357 if (RT_LIKELY(pThis->pHostDrvAudio->pfnGetConf))
1358 rc = pThis->pHostDrvAudio->pfnGetConf(pThis->pHostDrvAudio, &pThis->BackendCfg);
1359
1360 if (RT_SUCCESS(rc))
1361 {
1362 rc = drvAudioProcessOptions(pCfgHandle, "AUDIO", audio_options);
1363 /** @todo Check for invalid options? */
1364
1365 pThis->cFreeOutputStreams = conf.fixed_out.cStreams;
1366 pThis->cFreeInputStreams = conf.fixed_in.cStreams;
1367
1368 if (!pThis->cFreeOutputStreams)
1369 pThis->cFreeOutputStreams = 1;
1370
1371 if (!pThis->cFreeInputStreams)
1372 pThis->cFreeInputStreams = 1;
1373 }
1374
1375 /*
1376 * If everything went well, initialize the lower driver.
1377 */
1378 if (RT_SUCCESS(rc))
1379 rc = drvAudioHostInit(pCfgHandle, pThis);
1380
1381 LogFlowFuncLeaveRC(rc);
1382 return rc;
1383}
1384
1385static DECLCALLBACK(int) drvAudioInitNull(PPDMIAUDIOCONNECTOR pInterface)
1386{
1387 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1388 NOREF(pThis);
1389
1390 LogRel(("Audio: Using NULL driver; no sound will be audible\n"));
1391
1392 /* Nothing to do here yet. */
1393 return VINF_SUCCESS;
1394}
1395
1396static DECLCALLBACK(int) drvAudioRead(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOGSTSTRMIN pGstStrmIn,
1397 void *pvBuf, uint32_t cbBuf, uint32_t *pcbRead)
1398{
1399 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1400 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1401
1402 AssertPtrReturn(pGstStrmIn, VERR_INVALID_POINTER);
1403 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
1404 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
1405 /* pcbWritten is optional. */
1406
1407 if (!pThis->pHostDrvAudio->pfnIsEnabled(pThis->pHostDrvAudio, PDMAUDIODIR_IN))
1408 {
1409 if (pcbRead)
1410 *pcbRead = 0;
1411 return VINF_SUCCESS;
1412 }
1413
1414 PPDMAUDIOHSTSTRMIN pHstStrmIn = pGstStrmIn->pHstStrmIn;
1415 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
1416
1417 AssertMsg(pGstStrmIn->pHstStrmIn->fEnabled,
1418 ("Reading from disabled host input stream \"%s\" not possible\n", pGstStrmIn->MixBuf.pszName));
1419
1420 /*
1421 * Read from the parent buffer (that is, the guest buffer) which
1422 * should have the audio data in the format the guest needs.
1423 */
1424 uint32_t cRead;
1425 int rc = AudioMixBufReadCirc(&pGstStrmIn->MixBuf,
1426 pvBuf, cbBuf, &cRead);
1427 if (RT_SUCCESS(rc))
1428 {
1429 AudioMixBufFinish(&pGstStrmIn->MixBuf, cRead);
1430
1431 if (pcbRead)
1432 *pcbRead = AUDIOMIXBUF_S2B(&pGstStrmIn->MixBuf, cRead);
1433 }
1434
1435 LogFlowFunc(("cRead=%RU32 (%RU32 bytes), rc=%Rrc\n",
1436 cRead, AUDIOMIXBUF_S2B(&pGstStrmIn->MixBuf, cRead), rc));
1437 return rc;
1438}
1439
1440static DECLCALLBACK(int) drvAudioEnableOut(PPDMIAUDIOCONNECTOR pInterface,
1441 PPDMAUDIOGSTSTRMOUT pGstStrmOut, bool fEnable)
1442{
1443 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1444 /* pGstStrmOut is optional. */
1445
1446 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1447
1448 if (pGstStrmOut)
1449 {
1450 PPDMAUDIOHSTSTRMOUT pHstStrmOut = pGstStrmOut->pHstStrmOut;
1451 AssertPtr(pHstStrmOut);
1452
1453 LogFlowFunc(("%s: fEnable=%RTbool\n", pGstStrmOut->MixBuf.pszName, fEnable));
1454
1455 if (pGstStrmOut->State.fActive != fEnable)
1456 {
1457 if (fEnable)
1458 {
1459 pHstStrmOut->fPendingDisable = false;
1460 if (!pHstStrmOut->fEnabled)
1461 {
1462 pHstStrmOut->fEnabled = true;
1463 pThis->pHostDrvAudio->pfnControlOut(pThis->pHostDrvAudio, pHstStrmOut,
1464 PDMAUDIOSTREAMCMD_ENABLE);
1465 /** @todo Check rc. */
1466 }
1467 }
1468 else
1469 {
1470 if (pHstStrmOut->fEnabled)
1471 {
1472 uint32_t cGstStrmsActive = 0;
1473
1474 PPDMAUDIOGSTSTRMOUT pIter;
1475 RTListForEach(&pHstStrmOut->lstGstStrmOut, pIter, PDMAUDIOGSTSTRMOUT, Node)
1476 {
1477 if (pIter->State.fActive)
1478 cGstStrmsActive++;
1479 }
1480
1481 pHstStrmOut->fPendingDisable = cGstStrmsActive == 1;
1482 }
1483 }
1484
1485 pGstStrmOut->State.fActive = fEnable;
1486 }
1487 }
1488
1489 return VINF_SUCCESS;
1490}
1491
1492static DECLCALLBACK(int) drvAudioEnableIn(PPDMIAUDIOCONNECTOR pInterface,
1493 PPDMAUDIOGSTSTRMIN pGstStrmIn, bool fEnable)
1494{
1495 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1496 /* pGstStrmIn is optional. */
1497
1498 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1499
1500 if (pGstStrmIn)
1501 {
1502 PPDMAUDIOHSTSTRMIN pHstStrmIn = pGstStrmIn->pHstStrmIn;
1503 AssertPtr(pHstStrmIn);
1504
1505 LogFlowFunc(("%s: fEnable=%RTbool\n", pGstStrmIn->MixBuf.pszName, fEnable));
1506
1507 if (pGstStrmIn->State.fActive != fEnable)
1508 {
1509 if (fEnable)
1510 {
1511 if (!pHstStrmIn->fEnabled)
1512 {
1513 int rc2 = pThis->pHostDrvAudio->pfnControlIn(pThis->pHostDrvAudio, pHstStrmIn,
1514 PDMAUDIOSTREAMCMD_ENABLE);
1515 if (RT_LIKELY(RT_SUCCESS(rc2)))
1516 pHstStrmIn->fEnabled = true;
1517 else
1518 LogFlowFunc(("Error opening host input stream in backend, rc=%Rrc\n", rc2));
1519 }
1520 }
1521 else
1522 {
1523 if (pHstStrmIn->fEnabled)
1524 {
1525 int rc2 = pThis->pHostDrvAudio->pfnControlIn(pThis->pHostDrvAudio, pHstStrmIn,
1526 PDMAUDIOSTREAMCMD_DISABLE);
1527 if (RT_LIKELY(RT_SUCCESS(rc2))) /* Did the backend veto? */
1528 pHstStrmIn->fEnabled = false;
1529 else
1530 LogFlowFunc(("Backend vetoed closing input stream, rc=%Rrc\n", rc2));
1531 }
1532 }
1533
1534 pGstStrmIn->State.fActive = fEnable;
1535 }
1536 }
1537
1538 return VINF_SUCCESS;
1539}
1540
1541static DECLCALLBACK(bool) drvAudioIsInputOK(PPDMIAUDIOCONNECTOR pInterface,
1542 PPDMAUDIOGSTSTRMIN pGstStrmIn)
1543{
1544 return (pGstStrmIn != NULL);
1545}
1546
1547static DECLCALLBACK(bool) drvAudioIsOutputOK(PPDMIAUDIOCONNECTOR pInterface,
1548 PPDMAUDIOGSTSTRMOUT pGstStrmOut)
1549{
1550 return (pGstStrmOut != NULL);
1551}
1552
1553static DECLCALLBACK(int) drvAudioOpenIn(PPDMIAUDIOCONNECTOR pInterface, const char *pszName,
1554 PDMAUDIORECSOURCE enmRecSource, PPDMAUDIOSTREAMCFG pCfg,
1555 PPDMAUDIOGSTSTRMIN *ppGstStrmIn)
1556{
1557 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1558 AssertPtrReturn(ppGstStrmIn, VERR_INVALID_POINTER);
1559 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
1560 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1561 AssertPtrReturn(ppGstStrmIn, VERR_INVALID_POINTER);
1562
1563 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1564
1565 LogFlowFunc(("pszName=%s, pCfg=%p\n", pszName, pCfg));
1566
1567 if (!drvAudioStreamCfgIsValid(pCfg))
1568 {
1569 LogFunc(("Input stream configuration is not valid, bailing out\n"));
1570 return VERR_INVALID_PARAMETER;
1571 }
1572
1573 PPDMAUDIOGSTSTRMIN pGstStrmIn = *ppGstStrmIn;
1574 if ( pGstStrmIn
1575 && drvAudioPCMPropsAreEqual(&pGstStrmIn->Props, pCfg))
1576 {
1577 LogFunc(("[%s] Exists and matches required configuration, skipping creation\n",
1578 pGstStrmIn->MixBuf.pszName));
1579 return VWRN_ALREADY_EXISTS;
1580 }
1581
1582 if ( !conf.fixed_in.enabled
1583 && pGstStrmIn)
1584 {
1585 drvAudioDestroyGstIn(pThis, pGstStrmIn);
1586 pGstStrmIn = NULL;
1587 }
1588
1589 int rc;
1590
1591 if (pGstStrmIn)
1592 {
1593 PPDMAUDIOHSTSTRMIN pHstStrmIn = pGstStrmIn->pHstStrmIn;
1594 AssertPtr(pHstStrmIn);
1595
1596 drvAudioGstInFreeRes(pGstStrmIn);
1597
1598 char *pszTemp;
1599 if (RTStrAPrintf(&pszTemp, "%s (Guest)", pszName) <= 0)
1600 {
1601 RTMemFree(pGstStrmIn);
1602 return VERR_NO_MEMORY;
1603 }
1604
1605 rc = drvAudioGstInInit(pGstStrmIn, pHstStrmIn, pszName, pCfg);
1606
1607 RTStrFree(pszTemp);
1608 }
1609 else
1610 rc = drvAudioCreateStreamPairIn(pThis, pszName, enmRecSource, pCfg, &pGstStrmIn);
1611
1612 if (pGstStrmIn)
1613 *ppGstStrmIn = pGstStrmIn;
1614
1615 LogFlowFuncLeaveRC(rc);
1616 return rc;
1617}
1618
1619DECLCALLBACK(int) drvAudioOpenOut(PPDMIAUDIOCONNECTOR pInterface, const char *pszName,
1620 PPDMAUDIOSTREAMCFG pCfg, PPDMAUDIOGSTSTRMOUT *ppGstStrmOut)
1621{
1622 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1623 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
1624 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1625 AssertPtrReturn(ppGstStrmOut, VERR_INVALID_POINTER);
1626
1627 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1628
1629 LogFlowFunc(("pszName=%s, pCfg=%p\n", pszName, pCfg));
1630
1631 if (!drvAudioStreamCfgIsValid(pCfg))
1632 {
1633 LogFunc(("Output stream configuration is not valid, bailing out\n"));
1634 return VERR_INVALID_PARAMETER;
1635 }
1636
1637 PPDMAUDIOGSTSTRMOUT pGstStrmOut = *ppGstStrmOut;
1638 if ( pGstStrmOut
1639 && drvAudioPCMPropsAreEqual(&pGstStrmOut->Props, pCfg))
1640 {
1641 LogFunc(("[%s] Exists and matches required configuration, skipping creation\n",
1642 pGstStrmOut->MixBuf.pszName));
1643 return VWRN_ALREADY_EXISTS;
1644 }
1645
1646#if 0
1647 /* Any live samples that need to be updated after
1648 * we set the new parameters? */
1649 PPDMAUDIOGSTSTRMOUT pOldGstStrmOut = NULL;
1650 uint32_t cLiveSamples = 0;
1651
1652 if ( conf.plive
1653 && pGstStrmOut
1654 && ( !pGstStrmOut->State.fActive
1655 && !pGstStrmOut->State.fEmpty))
1656 {
1657 cLiveSamples = pGstStrmOut->cTotalSamplesWritten;
1658 if (cLiveSamples)
1659 {
1660 pOldGstStrmOut = pGstStrmOut;
1661 pGstStrmOut = NULL;
1662 }
1663 }
1664#endif
1665
1666 if ( pGstStrmOut
1667 && !conf.fixed_out.enabled)
1668 {
1669 drvAudioDestroyGstOut(pThis, pGstStrmOut);
1670 pGstStrmOut = NULL;
1671 }
1672
1673 int rc;
1674 if (pGstStrmOut)
1675 {
1676 PPDMAUDIOHSTSTRMOUT pHstStrmOut = pGstStrmOut->pHstStrmOut;
1677 AssertPtr(pHstStrmOut);
1678
1679 drvAudioGstOutFreeRes(pGstStrmOut);
1680
1681 rc = drvAudioGstOutInit(pGstStrmOut, pHstStrmOut, pszName, pCfg);
1682 }
1683 else
1684 {
1685 rc = drvAudioCreateStreamPairOut(pThis, pszName, pCfg, &pGstStrmOut);
1686 if (RT_FAILURE(rc))
1687 LogFunc(("Failed to create output stream \"%s\", rc=%Rrc\n", pszName, rc));
1688 }
1689
1690 if (RT_SUCCESS(rc))
1691 {
1692 AssertPtr(pGstStrmOut);
1693 *ppGstStrmOut = pGstStrmOut;
1694#if 0
1695 /* Update remaining live samples with new rate. */
1696 if (cLiveSamples)
1697 {
1698 AssertPtr(pOldGstStrmOut);
1699
1700 uint32_t cSamplesMixed =
1701 (cLiveSamples << pOldGstStrmOut->Props.cShift)
1702 * pOldGstStrmOut->Props.cbPerSec
1703 / (*ppGstStrmOut)->Props.cbPerSec;
1704
1705 pGstStrmOut->cTotalSamplesWritten += cSamplesMixed;
1706 }
1707#endif
1708 }
1709
1710 LogFlowFuncLeaveRC(rc);
1711 return rc;
1712}
1713
1714static DECLCALLBACK(bool) drvAudioIsActiveIn(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOGSTSTRMIN pGstStrmIn)
1715{
1716 return pGstStrmIn ? pGstStrmIn->State.fActive : false;
1717}
1718
1719static DECLCALLBACK(bool) drvAudioIsActiveOut(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOGSTSTRMOUT pGstStrmOut)
1720{
1721 return pGstStrmOut ? pGstStrmOut->State.fActive : false;
1722}
1723
1724static DECLCALLBACK(void) drvAudioCloseIn(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOGSTSTRMIN pGstStrmIn)
1725{
1726 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1727 if (pGstStrmIn)
1728 drvAudioDestroyGstIn(pThis, pGstStrmIn);
1729}
1730
1731DECLCALLBACK(void) drvAudioCloseOut(PPDMIAUDIOCONNECTOR pInterface, PPDMAUDIOGSTSTRMOUT pGstStrmOut)
1732{
1733 PDRVAUDIO pThis = PDMIAUDIOCONNECTOR_2_DRVAUDIO(pInterface);
1734 if (pGstStrmOut)
1735 drvAudioDestroyGstOut(pThis, pGstStrmOut);
1736}
1737
1738/********************************************************************/
1739
1740/**
1741 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1742 */
1743static DECLCALLBACK(void *) drvAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1744{
1745 LogFlowFunc(("pInterface=%p, pszIID=%s\n", pInterface, pszIID));
1746
1747 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
1748 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
1749
1750 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
1751 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIAUDIOCONNECTOR, &pThis->IAudioConnector);
1752
1753 return NULL;
1754}
1755
1756/**
1757 * Power Off notification.
1758 *
1759 * @param pDrvIns The driver instance data.
1760 */
1761static DECLCALLBACK(void) drvAudioPowerOff(PPDMDRVINS pDrvIns)
1762{
1763 LogFlowFuncEnter();
1764 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
1765
1766 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
1767
1768 if (!pThis->pHostDrvAudio)
1769 return;
1770
1771 /* Tear down all host output streams. */
1772 PPDMAUDIOHSTSTRMOUT pHstStrmOut = NULL;
1773 while ((pHstStrmOut = drvAudioFindAnyHstOut(pThis, pHstStrmOut)))
1774 {
1775 pThis->pHostDrvAudio->pfnControlOut(pThis->pHostDrvAudio, pHstStrmOut, PDMAUDIOSTREAMCMD_DISABLE);
1776 pThis->pHostDrvAudio->pfnFiniOut(pThis->pHostDrvAudio, pHstStrmOut);
1777 }
1778
1779 /* Tear down all host input streams. */
1780 PPDMAUDIOHSTSTRMIN pHstStrmIn = NULL;
1781 while ((pHstStrmIn = drvAudioFindNextHstIn(pThis, pHstStrmIn)))
1782 {
1783 pThis->pHostDrvAudio->pfnControlIn(pThis->pHostDrvAudio, pHstStrmIn, PDMAUDIOSTREAMCMD_DISABLE);
1784 pThis->pHostDrvAudio->pfnFiniIn(pThis->pHostDrvAudio, pHstStrmIn);
1785 }
1786
1787 if (pThis->pHostDrvAudio->pfnShutdown)
1788 pThis->pHostDrvAudio->pfnShutdown(pThis->pHostDrvAudio);
1789
1790 LogFlowFuncLeave();
1791}
1792
1793/**
1794 * Constructs an audio driver instance.
1795 *
1796 * @copydoc FNPDMDRVCONSTRUCT
1797 */
1798static DECLCALLBACK(int) drvAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfgHandle, uint32_t fFlags)
1799{
1800 LogFlowFunc(("pDrvIns=%#p, pCfgHandle=%#p, fFlags=%x\n", pDrvIns, pCfgHandle, fFlags));
1801
1802 PDRVAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIO);
1803 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
1804
1805 /*
1806 * Init the static parts.
1807 */
1808 pThis->pDrvIns = pDrvIns;
1809 /* IBase. */
1810 pDrvIns->IBase.pfnQueryInterface = drvAudioQueryInterface;
1811 /* IAudio. */
1812 pThis->IAudioConnector.pfnQueryStatus = drvAudioQueryStatus;
1813 pThis->IAudioConnector.pfnRead = drvAudioRead;
1814 pThis->IAudioConnector.pfnWrite = drvAudioWrite;
1815 pThis->IAudioConnector.pfnIsInputOK = drvAudioIsInputOK;
1816 pThis->IAudioConnector.pfnIsOutputOK = drvAudioIsOutputOK;
1817 pThis->IAudioConnector.pfnInitNull = drvAudioInitNull;
1818 pThis->IAudioConnector.pfnEnableOut = drvAudioEnableOut;
1819 pThis->IAudioConnector.pfnEnableIn = drvAudioEnableIn;
1820 pThis->IAudioConnector.pfnCloseIn = drvAudioCloseIn;
1821 pThis->IAudioConnector.pfnCloseOut = drvAudioCloseOut;
1822 pThis->IAudioConnector.pfnOpenIn = drvAudioOpenIn;
1823 pThis->IAudioConnector.pfnOpenOut = drvAudioOpenOut;
1824 pThis->IAudioConnector.pfnPlayOut = drvAudioPlayOut;
1825 pThis->IAudioConnector.pfnIsActiveIn = drvAudioIsActiveIn;
1826 pThis->IAudioConnector.pfnIsActiveOut = drvAudioIsActiveOut;
1827
1828 /*
1829 * Attach driver below and query its connector interface.
1830 */
1831 PPDMIBASE pDownBase;
1832 int rc = PDMDrvHlpAttach(pDrvIns, fFlags, &pDownBase);
1833 if (RT_FAILURE(rc))
1834 {
1835 LogRel(("Audio: Failed to attach to driver %p below (flags=0x%x), rc=%Rrc\n",
1836 pDrvIns, fFlags, rc));
1837 return rc;
1838 }
1839
1840 pThis->pHostDrvAudio = PDMIBASE_QUERY_INTERFACE(pDownBase, PDMIHOSTAUDIO);
1841 if (!pThis->pHostDrvAudio)
1842 {
1843 LogRel(("Audio: Failed to query interface for underlying host driver\n"));
1844 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_MISSING_INTERFACE_BELOW,
1845 N_("Host audio backend missing or invalid"));
1846 }
1847
1848#ifdef DEBUG_andy
1849 CFGMR3Dump(pCfgHandle);
1850#endif
1851
1852 rc = drvAudioInit(pCfgHandle, pDrvIns);
1853 if (RT_SUCCESS(rc))
1854 {
1855 pThis->fTerminate = false;
1856 pThis->pDrvIns = pDrvIns;
1857 }
1858
1859 LogFlowFuncLeaveRC(rc);
1860 return rc;
1861}
1862
1863/**
1864 * Suspend notification.
1865 *
1866 * @param pDrvIns The driver instance data.
1867 */
1868static DECLCALLBACK(void) drvAudioSuspend(PPDMDRVINS pDrvIns)
1869{
1870 drvAudioStateHandler(pDrvIns, PDMAUDIOSTREAMCMD_DISABLE);
1871}
1872
1873/**
1874 * Resume notification.
1875 *
1876 * @param pDrvIns The driver instance data.
1877 */
1878static DECLCALLBACK(void) drvAudioResume(PPDMDRVINS pDrvIns)
1879{
1880 drvAudioStateHandler(pDrvIns, PDMAUDIOSTREAMCMD_ENABLE);
1881}
1882
1883/**
1884 * Audio driver registration record.
1885 */
1886const PDMDRVREG g_DrvAUDIO =
1887{
1888 /* u32Version */
1889 PDM_DRVREG_VERSION,
1890 /* szName */
1891 "AUDIO",
1892 /* szRCMod */
1893 "",
1894 /* szR0Mod */
1895 "",
1896 /* pszDescription */
1897 "Audio connector driver",
1898 /* fFlags */
1899 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1900 /* fClass. */
1901 PDM_DRVREG_CLASS_AUDIO,
1902 /* cMaxInstances */
1903 2,
1904 /* cbInstance */
1905 sizeof(DRVAUDIO),
1906 /* pfnConstruct */
1907 drvAudioConstruct,
1908 /* pfnDestruct */
1909 NULL,
1910 /* pfnRelocate */
1911 NULL,
1912 /* pfnIOCtl */
1913 NULL,
1914 /* pfnPowerOn */
1915 NULL,
1916 /* pfnReset */
1917 NULL,
1918 /* pfnSuspend */
1919 drvAudioSuspend,
1920 /* pfnResume */
1921 drvAudioResume,
1922 /* pfnAttach */
1923 NULL,
1924 /* pfnDetach */
1925 NULL,
1926 /* pfnPowerOff */
1927 drvAudioPowerOff,
1928 /* pfnSoftReset */
1929 NULL,
1930 /* u32EndVersion */
1931 PDM_DRVREG_VERSION
1932};
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