VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/utils/audio/vkatDriverStack.cpp@ 90721

Last change on this file since 90721 was 89575, checked in by vboxsync, 4 years ago

Audio/ValKit: More code for completely self-contained (self) testing. bugref:10008

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 54.4 KB
Line 
1/* $Id: vkatDriverStack.cpp 89575 2021-06-09 09:16:59Z vboxsync $ */
2/** @file
3 * Validation Kit Audio Test (VKAT) - Driver stack code.
4 */
5
6/*
7 * Copyright (C) 2021 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/errcore.h>
32#include <iprt/message.h>
33#include <iprt/stream.h>
34#include <iprt/string.h>
35#include <iprt/uuid.h>
36#include <iprt/test.h>
37
38
39/**
40 * Internal driver instance data
41 * @note This must be put here as it's needed before pdmdrv.h is included.
42 */
43typedef struct PDMDRVINSINT
44{
45 /** The stack the drive belongs to. */
46 struct AUDIOTESTDRVSTACK *pStack;
47} PDMDRVINSINT;
48#define PDMDRVINSINT_DECLARED
49
50#include "vkatInternal.h"
51#include "VBoxDD.h"
52
53
54
55/*********************************************************************************************************************************
56* Fake PDM Driver Handling. *
57*********************************************************************************************************************************/
58
59/** @name Driver Fakes/Stubs
60 *
61 * @note The VMM functions defined here will turn into driver helpers before
62 * long, as the drivers aren't supposed to import directly from the VMM in
63 * the future.
64 *
65 * @{ */
66
67VMMR3DECL(PCFGMNODE) CFGMR3GetChild(PCFGMNODE pNode, const char *pszPath)
68{
69 RT_NOREF(pNode, pszPath);
70 return NULL;
71}
72
73
74VMMR3DECL(int) CFGMR3QueryString(PCFGMNODE pNode, const char *pszName, char *pszString, size_t cchString)
75{
76 if (pNode != NULL)
77 {
78 PCPDMDRVREG pDrvReg = (PCPDMDRVREG)pNode;
79 if (g_uVerbosity > 2)
80 RTPrintf("debug: CFGMR3QueryString([%s], %s, %p, %#x)\n", pDrvReg->szName, pszName, pszString, cchString);
81
82 if ( ( strcmp(pDrvReg->szName, "PulseAudio") == 0
83 || strcmp(pDrvReg->szName, "HostAudioWas") == 0)
84 && strcmp(pszName, "VmName") == 0)
85 return RTStrCopy(pszString, cchString, "vkat");
86
87 if ( strcmp(pDrvReg->szName, "HostAudioWas") == 0
88 && strcmp(pszName, "VmUuid") == 0)
89 return RTStrCopy(pszString, cchString, "794c9192-d045-4f28-91ed-46253ac9998e");
90 }
91 else if (g_uVerbosity > 2)
92 RTPrintf("debug: CFGMR3QueryString(%p, %s, %p, %#x)\n", pNode, pszName, pszString, cchString);
93
94 return VERR_CFGM_VALUE_NOT_FOUND;
95}
96
97
98VMMR3DECL(int) CFGMR3QueryStringAlloc(PCFGMNODE pNode, const char *pszName, char **ppszString)
99{
100 char szStr[128];
101 int rc = CFGMR3QueryString(pNode, pszName, szStr, sizeof(szStr));
102 if (RT_SUCCESS(rc))
103 *ppszString = RTStrDup(szStr);
104
105 return rc;
106}
107
108
109VMMR3DECL(void) MMR3HeapFree(void *pv)
110{
111 /* counterpart to CFGMR3QueryStringAlloc */
112 RTStrFree((char *)pv);
113}
114
115
116VMMR3DECL(int) CFGMR3QueryStringDef(PCFGMNODE pNode, const char *pszName, char *pszString, size_t cchString, const char *pszDef)
117{
118 PCPDMDRVREG pDrvReg = (PCPDMDRVREG)pNode;
119 if (RT_VALID_PTR(pDrvReg))
120 {
121 const char *pszRet = pszDef;
122 if ( g_pszDrvAudioDebug
123 && strcmp(pDrvReg->szName, "AUDIO") == 0
124 && strcmp(pszName, "DebugPathOut") == 0)
125 pszRet = g_pszDrvAudioDebug;
126
127 int rc = RTStrCopy(pszString, cchString, pszRet);
128
129 if (g_uVerbosity > 2)
130 RTPrintf("debug: CFGMR3QueryStringDef([%s], %s, %p, %#x, %s) -> '%s' + %Rrc\n",
131 pDrvReg->szName, pszName, pszString, cchString, pszDef, pszRet, rc);
132 return rc;
133 }
134
135 if (g_uVerbosity > 2)
136 RTPrintf("debug: CFGMR3QueryStringDef(%p, %s, %p, %#x, %s)\n", pNode, pszName, pszString, cchString, pszDef);
137 return RTStrCopy(pszString, cchString, pszDef);
138}
139
140
141VMMR3DECL(int) CFGMR3QueryBoolDef(PCFGMNODE pNode, const char *pszName, bool *pf, bool fDef)
142{
143 PCPDMDRVREG pDrvReg = (PCPDMDRVREG)pNode;
144 if (RT_VALID_PTR(pDrvReg))
145 {
146 *pf = fDef;
147 if ( strcmp(pDrvReg->szName, "AUDIO") == 0
148 && strcmp(pszName, "DebugEnabled") == 0)
149 *pf = g_fDrvAudioDebug;
150
151 if (g_uVerbosity > 2)
152 RTPrintf("debug: CFGMR3QueryBoolDef([%s], %s, %p, %RTbool) -> %RTbool\n", pDrvReg->szName, pszName, pf, fDef, *pf);
153 return VINF_SUCCESS;
154 }
155 *pf = fDef;
156 return VINF_SUCCESS;
157}
158
159
160VMMR3DECL(int) CFGMR3QueryU8(PCFGMNODE pNode, const char *pszName, uint8_t *pu8)
161{
162 RT_NOREF(pNode, pszName, pu8);
163 return VERR_CFGM_VALUE_NOT_FOUND;
164}
165
166
167VMMR3DECL(int) CFGMR3QueryU32(PCFGMNODE pNode, const char *pszName, uint32_t *pu32)
168{
169 RT_NOREF(pNode, pszName, pu32);
170 return VERR_CFGM_VALUE_NOT_FOUND;
171}
172
173
174VMMR3DECL(int) CFGMR3ValidateConfig(PCFGMNODE pNode, const char *pszNode,
175 const char *pszValidValues, const char *pszValidNodes,
176 const char *pszWho, uint32_t uInstance)
177{
178 RT_NOREF(pNode, pszNode, pszValidValues, pszValidNodes, pszWho, uInstance);
179 return VINF_SUCCESS;
180}
181
182/** @} */
183
184/** @name Driver Helper Fakes
185 * @{ */
186
187static DECLCALLBACK(int) audioTestDrvHlp_Attach(PPDMDRVINS pDrvIns, uint32_t fFlags, PPDMIBASE *ppBaseInterface)
188{
189 /* DrvAudio must be allowed to attach the backend driver (paranoid
190 backend drivers may call us to check that nothing is attached). */
191 if (strcmp(pDrvIns->pReg->szName, "AUDIO") == 0)
192 {
193 PAUDIOTESTDRVSTACK pDrvStack = pDrvIns->Internal.s.pStack;
194 AssertReturn(pDrvStack->pDrvBackendIns == NULL, VERR_PDM_DRIVER_ALREADY_ATTACHED);
195
196 if (g_uVerbosity > 1)
197 RTMsgInfo("Attaching backend '%s' to DrvAudio...\n", pDrvStack->pDrvReg->szName);
198 int rc = audioTestDrvConstruct(pDrvStack, pDrvStack->pDrvReg, pDrvIns, &pDrvStack->pDrvBackendIns);
199 if (RT_SUCCESS(rc))
200 {
201 if (ppBaseInterface)
202 *ppBaseInterface = &pDrvStack->pDrvBackendIns->IBase;
203 }
204 else
205 RTMsgError("Failed to attach backend: %Rrc", rc);
206 return rc;
207 }
208 RT_NOREF(fFlags);
209 return VERR_PDM_NO_ATTACHED_DRIVER;
210}
211
212
213static DECLCALLBACK(void) audioTestDrvHlp_STAMRegister(PPDMDRVINS pDrvIns, void *pvSample, STAMTYPE enmType, const char *pszName,
214 STAMUNIT enmUnit, const char *pszDesc)
215{
216 RT_NOREF(pDrvIns, pvSample, enmType, pszName, enmUnit, pszDesc);
217}
218
219
220static DECLCALLBACK(void) audioTestDrvHlp_STAMRegisterF(PPDMDRVINS pDrvIns, void *pvSample, STAMTYPE enmType,
221 STAMVISIBILITY enmVisibility, STAMUNIT enmUnit, const char *pszDesc,
222 const char *pszName, ...)
223{
224 RT_NOREF(pDrvIns, pvSample, enmType, enmVisibility, enmUnit, pszDesc, pszName);
225}
226
227
228static DECLCALLBACK(void) audioTestDrvHlp_STAMRegisterV(PPDMDRVINS pDrvIns, void *pvSample, STAMTYPE enmType,
229 STAMVISIBILITY enmVisibility, STAMUNIT enmUnit, const char *pszDesc,
230 const char *pszName, va_list args)
231{
232 RT_NOREF(pDrvIns, pvSample, enmType, enmVisibility, enmUnit, pszDesc, pszName, args);
233}
234
235
236static DECLCALLBACK(int) audioTestDrvHlp_STAMDeregister(PPDMDRVINS pDrvIns, void *pvSample)
237{
238 RT_NOREF(pDrvIns, pvSample);
239 return VINF_SUCCESS;
240}
241
242
243static DECLCALLBACK(int) audioTestDrvHlp_STAMDeregisterByPrefix(PPDMDRVINS pDrvIns, const char *pszPrefix)
244{
245 RT_NOREF(pDrvIns, pszPrefix);
246 return VINF_SUCCESS;
247}
248
249/**
250 * Get the driver helpers.
251 */
252static const PDMDRVHLPR3 *audioTestFakeGetDrvHlp(void)
253{
254 /*
255 * Note! No initializer for s_DrvHlp (also why it's not a file global).
256 * We do not want to have to update this code every time PDMDRVHLPR3
257 * grows new entries or are otherwise modified. Only when the
258 * entries used by the audio driver changes do we want to change
259 * our code.
260 */
261 static PDMDRVHLPR3 s_DrvHlp;
262 if (s_DrvHlp.u32Version != PDM_DRVHLPR3_VERSION)
263 {
264 s_DrvHlp.u32Version = PDM_DRVHLPR3_VERSION;
265 s_DrvHlp.u32TheEnd = PDM_DRVHLPR3_VERSION;
266 s_DrvHlp.pfnAttach = audioTestDrvHlp_Attach;
267 s_DrvHlp.pfnSTAMRegister = audioTestDrvHlp_STAMRegister;
268 s_DrvHlp.pfnSTAMRegisterF = audioTestDrvHlp_STAMRegisterF;
269 s_DrvHlp.pfnSTAMRegisterV = audioTestDrvHlp_STAMRegisterV;
270 s_DrvHlp.pfnSTAMDeregister = audioTestDrvHlp_STAMDeregister;
271 s_DrvHlp.pfnSTAMDeregisterByPrefix = audioTestDrvHlp_STAMDeregisterByPrefix;
272 }
273 return &s_DrvHlp;
274}
275
276/** @} */
277
278
279/**
280 * Implementation of PDMIBASE::pfnQueryInterface for a fake device above
281 * DrvAudio.
282 */
283static DECLCALLBACK(void *) audioTestFakeDeviceIBaseQueryInterface(PPDMIBASE pInterface, const char *pszIID)
284{
285 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, pInterface);
286 RTMsgWarning("audioTestFakeDeviceIBaseQueryInterface: Unknown interface: %s\n", pszIID);
287 return NULL;
288}
289
290/** IBase interface for a fake device above DrvAudio. */
291static PDMIBASE g_AudioTestFakeDeviceIBase = { audioTestFakeDeviceIBaseQueryInterface };
292
293
294static DECLCALLBACK(int) audioTestIHostAudioPort_DoOnWorkerThread(PPDMIHOSTAUDIOPORT pInterface, PPDMAUDIOBACKENDSTREAM pStream,
295 uintptr_t uUser, void *pvUser)
296{
297 RT_NOREF(pInterface, pStream, uUser, pvUser);
298 RTMsgWarning("audioTestIHostAudioPort_DoOnWorkerThread was called\n");
299 return VERR_NOT_IMPLEMENTED;
300}
301
302
303DECLCALLBACK(void) audioTestIHostAudioPort_NotifyDeviceChanged(PPDMIHOSTAUDIOPORT pInterface, PDMAUDIODIR enmDir, void *pvUser)
304{
305 RT_NOREF(pInterface, enmDir, pvUser);
306 RTMsgWarning("audioTestIHostAudioPort_NotifyDeviceChanged was called\n");
307}
308
309
310static DECLCALLBACK(void) audioTestIHostAudioPort_StreamNotifyPreparingDeviceSwitch(PPDMIHOSTAUDIOPORT pInterface,
311 PPDMAUDIOBACKENDSTREAM pStream)
312{
313 RT_NOREF(pInterface, pStream);
314 RTMsgWarning("audioTestIHostAudioPort_StreamNotifyPreparingDeviceSwitch was called\n");
315}
316
317
318static DECLCALLBACK(void) audioTestIHostAudioPort_StreamNotifyDeviceChanged(PPDMIHOSTAUDIOPORT pInterface,
319 PPDMAUDIOBACKENDSTREAM pStream, bool fReInit)
320{
321 RT_NOREF(pInterface, pStream, fReInit);
322 RTMsgWarning("audioTestIHostAudioPort_StreamNotifyDeviceChanged was called\n");
323}
324
325
326static DECLCALLBACK(void) audioTestIHostAudioPort_NotifyDevicesChanged(PPDMIHOSTAUDIOPORT pInterface)
327{
328 RT_NOREF(pInterface);
329 RTMsgWarning("audioTestIHostAudioPort_NotifyDevicesChanged was called\n");
330}
331
332
333static PDMIHOSTAUDIOPORT g_AudioTestIHostAudioPort =
334{
335 audioTestIHostAudioPort_DoOnWorkerThread,
336 audioTestIHostAudioPort_NotifyDeviceChanged,
337 audioTestIHostAudioPort_StreamNotifyPreparingDeviceSwitch,
338 audioTestIHostAudioPort_StreamNotifyDeviceChanged,
339 audioTestIHostAudioPort_NotifyDevicesChanged,
340};
341
342
343/**
344 * Implementation of PDMIBASE::pfnQueryInterface for a fake DrvAudio above a
345 * backend.
346 */
347static DECLCALLBACK(void *) audioTestFakeDrvAudioIBaseQueryInterface(PPDMIBASE pInterface, const char *pszIID)
348{
349 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, pInterface);
350 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIOPORT, &g_AudioTestIHostAudioPort);
351 RTMsgWarning("audioTestFakeDrvAudioIBaseQueryInterface: Unknown interface: %s\n", pszIID);
352 return NULL;
353}
354
355
356/** IBase interface for a fake DrvAudio above a lonesome backend. */
357static PDMIBASE g_AudioTestFakeDrvAudioIBase = { audioTestFakeDrvAudioIBaseQueryInterface };
358
359
360
361/**
362 * Constructs a PDM audio driver instance.
363 *
364 * @returns VBox status code.
365 * @param pDrvStack The stack this is associated with.
366 * @param pDrvReg PDM driver registration record to use for construction.
367 * @param pParentDrvIns The parent driver (if any).
368 * @param ppDrvIns Where to return the driver instance structure.
369 */
370int audioTestDrvConstruct(PAUDIOTESTDRVSTACK pDrvStack, PCPDMDRVREG pDrvReg, PPDMDRVINS pParentDrvIns,
371 PPPDMDRVINS ppDrvIns)
372{
373 /* The destruct function must have valid data to work with. */
374 *ppDrvIns = NULL;
375
376 /*
377 * Check registration structure validation (doesn't need to be too
378 * thorough, PDM check it in detail on every VM startup).
379 */
380 AssertPtrReturn(pDrvReg, VERR_INVALID_POINTER);
381 RTMsgInfo("Initializing backend '%s' ...\n", pDrvReg->szName);
382 AssertPtrReturn(pDrvReg->pfnConstruct, VERR_INVALID_PARAMETER);
383
384 /*
385 * Create the instance data structure.
386 */
387 PPDMDRVINS pDrvIns = (PPDMDRVINS)RTMemAllocZVar(RT_UOFFSETOF_DYN(PDMDRVINS, achInstanceData[pDrvReg->cbInstance]));
388 RTTEST_CHECK_RET(g_hTest, pDrvIns, VERR_NO_MEMORY);
389
390 pDrvIns->u32Version = PDM_DRVINS_VERSION;
391 pDrvIns->iInstance = 0;
392 pDrvIns->pHlpR3 = audioTestFakeGetDrvHlp();
393 pDrvIns->pvInstanceDataR3 = &pDrvIns->achInstanceData[0];
394 pDrvIns->pReg = pDrvReg;
395 pDrvIns->pCfg = (PCFGMNODE)pDrvReg;
396 pDrvIns->Internal.s.pStack = pDrvStack;
397 pDrvIns->pUpBase = NULL;
398 pDrvIns->pDownBase = NULL;
399 if (pParentDrvIns)
400 {
401 Assert(pParentDrvIns->pDownBase == NULL);
402 pParentDrvIns->pDownBase = &pDrvIns->IBase;
403 pDrvIns->pUpBase = &pParentDrvIns->IBase;
404 }
405 else if (strcmp(pDrvReg->szName, "AUDIO") == 0)
406 pDrvIns->pUpBase = &g_AudioTestFakeDeviceIBase;
407 else
408 pDrvIns->pUpBase = &g_AudioTestFakeDrvAudioIBase;
409
410 /*
411 * Invoke the constructor.
412 */
413 int rc = pDrvReg->pfnConstruct(pDrvIns, pDrvIns->pCfg, 0 /*fFlags*/);
414 if (RT_SUCCESS(rc))
415 {
416 *ppDrvIns = pDrvIns;
417 return VINF_SUCCESS;
418 }
419
420 RTTestFailed(g_hTest, "Failed to construct audio driver '%s': %Rrc", pDrvReg->szName, rc);
421 if (pDrvReg->pfnDestruct)
422 pDrvReg->pfnDestruct(pDrvIns);
423 RTMemFree(pDrvIns);
424 return rc;
425}
426
427
428/**
429 * Destructs a PDM audio driver instance.
430 *
431 * @param pDrvIns Driver instance to destruct.
432 */
433static void audioTestDrvDestruct(PPDMDRVINS pDrvIns)
434{
435 if (pDrvIns)
436 {
437 Assert(pDrvIns->u32Version == PDM_DRVINS_VERSION);
438
439 if (pDrvIns->pReg->pfnDestruct)
440 pDrvIns->pReg->pfnDestruct(pDrvIns);
441
442 pDrvIns->u32Version = 0;
443 pDrvIns->pReg = NULL;
444 RTMemFree(pDrvIns);
445 }
446}
447
448
449/**
450 * Sends the PDM driver a power off notification.
451 *
452 * @param pDrvIns Driver instance to notify.
453 */
454static void audioTestDrvNotifyPowerOff(PPDMDRVINS pDrvIns)
455{
456 if (pDrvIns)
457 {
458 Assert(pDrvIns->u32Version == PDM_DRVINS_VERSION);
459 if (pDrvIns->pReg->pfnPowerOff)
460 pDrvIns->pReg->pfnPowerOff(pDrvIns);
461 }
462}
463
464
465/**
466 * Deletes a driver stack.
467 *
468 * This will power off and destroy the drivers.
469 */
470void audioTestDriverStackDelete(PAUDIOTESTDRVSTACK pDrvStack)
471{
472 /*
473 * Do power off notifications (top to bottom).
474 */
475 audioTestDrvNotifyPowerOff(pDrvStack->pDrvAudioIns);
476 audioTestDrvNotifyPowerOff(pDrvStack->pDrvBackendIns);
477
478 /*
479 * Drivers are destroyed from bottom to top (closest to the device).
480 */
481 audioTestDrvDestruct(pDrvStack->pDrvBackendIns);
482 pDrvStack->pDrvBackendIns = NULL;
483 pDrvStack->pIHostAudio = NULL;
484
485 audioTestDrvDestruct(pDrvStack->pDrvAudioIns);
486 pDrvStack->pDrvAudioIns = NULL;
487 pDrvStack->pIAudioConnector = NULL;
488}
489
490
491/**
492 * Initializes a driver stack, extended version.
493 *
494 * @returns VBox status code.
495 * @param pDrvStack The driver stack to initialize.
496 * @param pDrvReg The backend driver to use.
497 * @param fEnabledIn Whether input is enabled or not on creation time.
498 * @param fEnabledOut Whether output is enabled or not on creation time.
499 * @param fWithDrvAudio Whether to include DrvAudio in the stack or not.
500 */
501int audioTestDriverStackInitEx(PAUDIOTESTDRVSTACK pDrvStack, PCPDMDRVREG pDrvReg, bool fEnabledIn, bool fEnabledOut, bool fWithDrvAudio)
502{
503 int rc;
504
505 RT_ZERO(*pDrvStack);
506 pDrvStack->pDrvReg = pDrvReg;
507
508 if (!fWithDrvAudio)
509 rc = audioTestDrvConstruct(pDrvStack, pDrvReg, NULL /*pParentDrvIns*/, &pDrvStack->pDrvBackendIns);
510 else
511 {
512 rc = audioTestDrvConstruct(pDrvStack, &g_DrvAUDIO, NULL /*pParentDrvIns*/, &pDrvStack->pDrvAudioIns);
513 if (RT_SUCCESS(rc))
514 {
515 Assert(pDrvStack->pDrvAudioIns);
516 PPDMIBASE const pIBase = &pDrvStack->pDrvAudioIns->IBase;
517 pDrvStack->pIAudioConnector = (PPDMIAUDIOCONNECTOR)pIBase->pfnQueryInterface(pIBase, PDMIAUDIOCONNECTOR_IID);
518 if (pDrvStack->pIAudioConnector)
519 {
520 /* Both input and output is disabled by default. */
521 if (fEnabledIn)
522 rc = pDrvStack->pIAudioConnector->pfnEnable(pDrvStack->pIAudioConnector, PDMAUDIODIR_IN, true);
523
524 if (RT_SUCCESS(rc))
525 {
526 if (fEnabledOut)
527 rc = pDrvStack->pIAudioConnector->pfnEnable(pDrvStack->pIAudioConnector, PDMAUDIODIR_OUT, true);
528 }
529
530 if (RT_FAILURE(rc))
531 {
532 RTTestFailed(g_hTest, "Failed to enabled input and output: %Rrc", rc);
533 audioTestDriverStackDelete(pDrvStack);
534 }
535 }
536 else
537 {
538 RTTestFailed(g_hTest, "Failed to query PDMIAUDIOCONNECTOR");
539 audioTestDriverStackDelete(pDrvStack);
540 rc = VERR_PDM_MISSING_INTERFACE;
541 }
542 }
543 }
544
545 /*
546 * Get the IHostAudio interface and check that the host driver is working.
547 */
548 if (RT_SUCCESS(rc))
549 {
550 PPDMIBASE const pIBase = &pDrvStack->pDrvBackendIns->IBase;
551 pDrvStack->pIHostAudio = (PPDMIHOSTAUDIO)pIBase->pfnQueryInterface(pIBase, PDMIHOSTAUDIO_IID);
552 if (pDrvStack->pIHostAudio)
553 {
554 PDMAUDIOBACKENDSTS enmStatus = pDrvStack->pIHostAudio->pfnGetStatus(pDrvStack->pIHostAudio, PDMAUDIODIR_OUT);
555 if (enmStatus == PDMAUDIOBACKENDSTS_RUNNING)
556 return VINF_SUCCESS;
557
558 RTTestFailed(g_hTest, "Expected backend status RUNNING, got %d instead", enmStatus);
559 }
560 else
561 RTTestFailed(g_hTest, "Failed to query PDMIHOSTAUDIO for '%s'", pDrvReg->szName);
562 audioTestDriverStackDelete(pDrvStack);
563 }
564
565 return rc;
566}
567
568
569/**
570 * Initializes a driver stack.
571 *
572 * @returns VBox status code.
573 * @param pDrvStack The driver stack to initialize.
574 * @param pDrvReg The backend driver to use.
575 * @param fEnabledIn Whether input is enabled or not on creation time.
576 * @param fEnabledOut Whether output is enabled or not on creation time.
577 * @param fWithDrvAudio Whether to include DrvAudio in the stack or not.
578 */
579int audioTestDriverStackInit(PAUDIOTESTDRVSTACK pDrvStack, PCPDMDRVREG pDrvReg, bool fWithDrvAudio)
580{
581 return audioTestDriverStackInitEx(pDrvStack, pDrvReg, true /* fEnabledIn */, true /* fEnabledOut */, fWithDrvAudio);
582}
583
584
585/**
586 * Wrapper around PDMIHOSTAUDIO::pfnSetDevice.
587 */
588int audioTestDriverStackSetDevice(PAUDIOTESTDRVSTACK pDrvStack, PDMAUDIODIR enmDir, const char *pszDevId)
589{
590 int rc;
591 if ( pDrvStack->pIHostAudio
592 && pDrvStack->pIHostAudio->pfnSetDevice)
593 rc = pDrvStack->pIHostAudio->pfnSetDevice(pDrvStack->pIHostAudio, enmDir, pszDevId);
594 else if (!pszDevId || *pszDevId)
595 rc = VINF_SUCCESS;
596 else
597 rc = VERR_INVALID_FUNCTION;
598 return rc;
599}
600
601
602/**
603 * Common stream creation code.
604 *
605 * @returns VBox status code.
606 * @param pDrvStack The audio driver stack to create it via.
607 * @param pCfgReq The requested config.
608 * @param ppStream Where to return the stream pointer on success.
609 * @param pCfgAcq Where to return the actual (well, not necessarily when
610 * using DrvAudio, but probably the same) stream config on
611 * success (not used as input).
612 */
613static int audioTestDriverStackStreamCreate(PAUDIOTESTDRVSTACK pDrvStack, PCPDMAUDIOSTREAMCFG pCfgReq,
614 PPDMAUDIOSTREAM *ppStream, PPDMAUDIOSTREAMCFG pCfgAcq)
615{
616 char szTmp[PDMAUDIOSTRMCFGTOSTRING_MAX + 16];
617 int rc;
618 *ppStream = NULL;
619
620 if (pDrvStack->pIAudioConnector)
621 {
622 /*
623 * DrvAudio does most of the work here.
624 */
625 rc = pDrvStack->pIAudioConnector->pfnStreamCreate(pDrvStack->pIAudioConnector, 0 /*fFlags*/, pCfgReq, ppStream);
626 if (RT_SUCCESS(rc))
627 {
628 *pCfgAcq = (*ppStream)->Cfg;
629 RTMsgInfo("Created backend stream: %s\n", PDMAudioStrmCfgToString(pCfgReq, szTmp, sizeof(szTmp)));
630 return rc;
631 }
632 RTTestFailed(g_hTest, "pfnStreamCreate failed: %Rrc", rc);
633 }
634 else
635 {
636 /*
637 * Get the config so we can see how big the PDMAUDIOBACKENDSTREAM
638 * structure actually is for this backend.
639 */
640 PDMAUDIOBACKENDCFG BackendCfg;
641 rc = pDrvStack->pIHostAudio->pfnGetConfig(pDrvStack->pIHostAudio, &BackendCfg);
642 if (RT_SUCCESS(rc))
643 {
644 if (BackendCfg.cbStream >= sizeof(PDMAUDIOBACKENDSTREAM))
645 {
646 /*
647 * Allocate and initialize the stream.
648 */
649 uint32_t const cbStream = sizeof(AUDIOTESTDRVSTACKSTREAM) - sizeof(PDMAUDIOBACKENDSTREAM) + BackendCfg.cbStream;
650 PAUDIOTESTDRVSTACKSTREAM pStreamAt = (PAUDIOTESTDRVSTACKSTREAM)RTMemAllocZVar(cbStream);
651 if (pStreamAt)
652 {
653 pStreamAt->Core.uMagic = PDMAUDIOSTREAM_MAGIC;
654 pStreamAt->Core.Cfg = *pCfgReq;
655 pStreamAt->Core.cbBackend = cbStream;
656
657 pStreamAt->Backend.uMagic = PDMAUDIOBACKENDSTREAM_MAGIC;
658 pStreamAt->Backend.pStream = &pStreamAt->Core;
659
660 /*
661 * Call the backend to create the stream.
662 */
663 rc = pDrvStack->pIHostAudio->pfnStreamCreate(pDrvStack->pIHostAudio, &pStreamAt->Backend,
664 pCfgReq, &pStreamAt->Core.Cfg);
665 if (RT_SUCCESS(rc))
666 {
667 if (g_uVerbosity > 1)
668 RTMsgInfo("Created backend stream: %s\n",
669 PDMAudioStrmCfgToString(&pStreamAt->Core.Cfg, szTmp, sizeof(szTmp)));
670
671 /* Return if stream is ready: */
672 if (rc == VINF_SUCCESS)
673 {
674 *ppStream = &pStreamAt->Core;
675 *pCfgAcq = pStreamAt->Core.Cfg;
676 return VINF_SUCCESS;
677 }
678 if (rc == VINF_AUDIO_STREAM_ASYNC_INIT_NEEDED)
679 {
680 /*
681 * Do async init right here and now.
682 */
683 rc = pDrvStack->pIHostAudio->pfnStreamInitAsync(pDrvStack->pIHostAudio, &pStreamAt->Backend,
684 false /*fDestroyed*/);
685 if (RT_SUCCESS(rc))
686 {
687 *ppStream = &pStreamAt->Core;
688 *pCfgAcq = pStreamAt->Core.Cfg;
689 return VINF_SUCCESS;
690 }
691
692 RTTestFailed(g_hTest, "pfnStreamInitAsync failed: %Rrc\n", rc);
693 }
694 else
695 {
696 RTTestFailed(g_hTest, "pfnStreamCreate returned unexpected info status: %Rrc", rc);
697 rc = VERR_IPE_UNEXPECTED_INFO_STATUS;
698 }
699 pDrvStack->pIHostAudio->pfnStreamDestroy(pDrvStack->pIHostAudio, &pStreamAt->Backend, true /*fImmediate*/);
700 }
701 else
702 RTTestFailed(g_hTest, "pfnStreamCreate failed: %Rrc\n", rc);
703 }
704 else
705 {
706 RTTestFailed(g_hTest, "Out of memory!\n");
707 rc = VERR_NO_MEMORY;
708 }
709 RTMemFree(pStreamAt);
710 }
711 else
712 {
713 RTTestFailed(g_hTest, "cbStream=%#x is too small, min %#zx!\n", BackendCfg.cbStream, sizeof(PDMAUDIOBACKENDSTREAM));
714 rc = VERR_OUT_OF_RANGE;
715 }
716 }
717 else
718 RTTestFailed(g_hTest, "pfnGetConfig failed: %Rrc\n", rc);
719 }
720 return rc;
721}
722
723
724/**
725 * Creates an output stream.
726 *
727 * @returns VBox status code.
728 * @param pDrvStack The audio driver stack to create it via.
729 * @param pProps The audio properties to use.
730 * @param cMsBufferSize The buffer size in milliseconds.
731 * @param cMsPreBuffer The pre-buffering amount in milliseconds.
732 * @param cMsSchedulingHint The scheduling hint in milliseconds.
733 * @param ppStream Where to return the stream pointer on success.
734 * @param pCfgAcq Where to return the actual (well, not
735 * necessarily when using DrvAudio, but probably
736 * the same) stream config on success (not used as
737 * input).
738 */
739int audioTestDriverStackStreamCreateOutput(PAUDIOTESTDRVSTACK pDrvStack, PCPDMAUDIOPCMPROPS pProps,
740 uint32_t cMsBufferSize, uint32_t cMsPreBuffer, uint32_t cMsSchedulingHint,
741 PPDMAUDIOSTREAM *ppStream, PPDMAUDIOSTREAMCFG pCfgAcq)
742{
743 /*
744 * Calculate the stream config.
745 */
746 PDMAUDIOSTREAMCFG CfgReq;
747 int rc = PDMAudioStrmCfgInitWithProps(&CfgReq, pProps);
748 AssertRC(rc);
749 CfgReq.enmDir = PDMAUDIODIR_OUT;
750 CfgReq.enmPath = PDMAUDIOPATH_OUT_FRONT;
751 CfgReq.Device.cMsSchedulingHint = cMsSchedulingHint == UINT32_MAX || cMsSchedulingHint == 0
752 ? 10 : cMsSchedulingHint;
753 if (pDrvStack->pIAudioConnector && (cMsBufferSize == UINT32_MAX || cMsBufferSize == 0))
754 CfgReq.Backend.cFramesBufferSize = 0; /* DrvAudio picks the default */
755 else
756 CfgReq.Backend.cFramesBufferSize = PDMAudioPropsMilliToFrames(pProps,
757 cMsBufferSize == UINT32_MAX || cMsBufferSize == 0
758 ? 300 : cMsBufferSize);
759 if (cMsPreBuffer == UINT32_MAX)
760 CfgReq.Backend.cFramesPreBuffering = pDrvStack->pIAudioConnector ? UINT32_MAX /*DrvAudo picks the default */
761 : CfgReq.Backend.cFramesBufferSize * 2 / 3;
762 else
763 CfgReq.Backend.cFramesPreBuffering = PDMAudioPropsMilliToFrames(pProps, cMsPreBuffer);
764 if ( CfgReq.Backend.cFramesPreBuffering >= CfgReq.Backend.cFramesBufferSize + 16
765 && !pDrvStack->pIAudioConnector /*DrvAudio deals with it*/ )
766 {
767 RTMsgWarning("Cannot pre-buffer %#x frames with only %#x frames of buffer!",
768 CfgReq.Backend.cFramesPreBuffering, CfgReq.Backend.cFramesBufferSize);
769 CfgReq.Backend.cFramesPreBuffering = CfgReq.Backend.cFramesBufferSize > 16
770 ? CfgReq.Backend.cFramesBufferSize - 16 : 0;
771 }
772
773 static uint32_t s_idxStream = 0;
774 uint32_t const idxStream = s_idxStream++;
775 RTStrPrintf(CfgReq.szName, sizeof(CfgReq.szName), "out-%u", idxStream);
776
777 /*
778 * Call common code to do the actual work.
779 */
780 return audioTestDriverStackStreamCreate(pDrvStack, &CfgReq, ppStream, pCfgAcq);
781}
782
783
784/**
785 * Creates an input stream.
786 *
787 * @returns VBox status code.
788 * @param pDrvStack The audio driver stack to create it via.
789 * @param pProps The audio properties to use.
790 * @param cMsBufferSize The buffer size in milliseconds.
791 * @param cMsPreBuffer The pre-buffering amount in milliseconds.
792 * @param cMsSchedulingHint The scheduling hint in milliseconds.
793 * @param ppStream Where to return the stream pointer on success.
794 * @param pCfgAcq Where to return the actual (well, not
795 * necessarily when using DrvAudio, but probably
796 * the same) stream config on success (not used as
797 * input).
798 */
799int audioTestDriverStackStreamCreateInput(PAUDIOTESTDRVSTACK pDrvStack, PCPDMAUDIOPCMPROPS pProps,
800 uint32_t cMsBufferSize, uint32_t cMsPreBuffer, uint32_t cMsSchedulingHint,
801 PPDMAUDIOSTREAM *ppStream, PPDMAUDIOSTREAMCFG pCfgAcq)
802{
803 /*
804 * Calculate the stream config.
805 */
806 PDMAUDIOSTREAMCFG CfgReq;
807 int rc = PDMAudioStrmCfgInitWithProps(&CfgReq, pProps);
808 AssertRC(rc);
809 CfgReq.enmDir = PDMAUDIODIR_IN;
810 CfgReq.enmPath = PDMAUDIOPATH_IN_LINE;
811 CfgReq.Device.cMsSchedulingHint = cMsSchedulingHint == UINT32_MAX || cMsSchedulingHint == 0
812 ? 10 : cMsSchedulingHint;
813 if (pDrvStack->pIAudioConnector && (cMsBufferSize == UINT32_MAX || cMsBufferSize == 0))
814 CfgReq.Backend.cFramesBufferSize = 0; /* DrvAudio picks the default */
815 else
816 CfgReq.Backend.cFramesBufferSize = PDMAudioPropsMilliToFrames(pProps,
817 cMsBufferSize == UINT32_MAX || cMsBufferSize == 0
818 ? 300 : cMsBufferSize);
819 if (cMsPreBuffer == UINT32_MAX)
820 CfgReq.Backend.cFramesPreBuffering = pDrvStack->pIAudioConnector ? UINT32_MAX /*DrvAudio picks the default */
821 : CfgReq.Backend.cFramesBufferSize / 2;
822 else
823 CfgReq.Backend.cFramesPreBuffering = PDMAudioPropsMilliToFrames(pProps, cMsPreBuffer);
824 if ( CfgReq.Backend.cFramesPreBuffering >= CfgReq.Backend.cFramesBufferSize + 16 /** @todo way to little */
825 && !pDrvStack->pIAudioConnector /*DrvAudio deals with it*/ )
826 {
827 RTMsgWarning("Cannot pre-buffer %#x frames with only %#x frames of buffer!",
828 CfgReq.Backend.cFramesPreBuffering, CfgReq.Backend.cFramesBufferSize);
829 CfgReq.Backend.cFramesPreBuffering = CfgReq.Backend.cFramesBufferSize > 16
830 ? CfgReq.Backend.cFramesBufferSize - 16 : 0;
831 }
832
833 static uint32_t s_idxStream = 0;
834 uint32_t const idxStream = s_idxStream++;
835 RTStrPrintf(CfgReq.szName, sizeof(CfgReq.szName), "in-%u", idxStream);
836
837 /*
838 * Call common code to do the actual work.
839 */
840 return audioTestDriverStackStreamCreate(pDrvStack, &CfgReq, ppStream, pCfgAcq);
841}
842
843
844/**
845 * Destroys a stream.
846 */
847void audioTestDriverStackStreamDestroy(PAUDIOTESTDRVSTACK pDrvStack, PPDMAUDIOSTREAM pStream)
848{
849 if (pStream)
850 {
851 if (pDrvStack->pIAudioConnector)
852 {
853 int rc = pDrvStack->pIAudioConnector->pfnStreamDestroy(pDrvStack->pIAudioConnector, pStream, true /*fImmediate*/);
854 if (RT_FAILURE(rc))
855 RTTestFailed(g_hTest, "pfnStreamDestroy failed: %Rrc", rc);
856 }
857 else
858 {
859 PAUDIOTESTDRVSTACKSTREAM pStreamAt = (PAUDIOTESTDRVSTACKSTREAM)pStream;
860 int rc = pDrvStack->pIHostAudio->pfnStreamDestroy(pDrvStack->pIHostAudio, &pStreamAt->Backend, true /*fImmediate*/);
861 if (RT_SUCCESS(rc))
862 {
863 pStreamAt->Core.uMagic = ~PDMAUDIOSTREAM_MAGIC;
864 pStreamAt->Backend.uMagic = ~PDMAUDIOBACKENDSTREAM_MAGIC;
865 RTMemFree(pStreamAt);
866 }
867 else
868 RTTestFailed(g_hTest, "PDMIHOSTAUDIO::pfnStreamDestroy failed: %Rrc", rc);
869 }
870 }
871}
872
873
874/**
875 * Enables a stream.
876 */
877int audioTestDriverStackStreamEnable(PAUDIOTESTDRVSTACK pDrvStack, PPDMAUDIOSTREAM pStream)
878{
879 int rc;
880 if (pDrvStack->pIAudioConnector)
881 {
882 rc = pDrvStack->pIAudioConnector->pfnStreamControl(pDrvStack->pIAudioConnector, pStream, PDMAUDIOSTREAMCMD_ENABLE);
883 if (RT_FAILURE(rc))
884 RTTestFailed(g_hTest, "pfnStreamControl/ENABLE failed: %Rrc", rc);
885 }
886 else
887 {
888 PAUDIOTESTDRVSTACKSTREAM pStreamAt = (PAUDIOTESTDRVSTACKSTREAM)pStream;
889 rc = pDrvStack->pIHostAudio->pfnStreamEnable(pDrvStack->pIHostAudio, &pStreamAt->Backend);
890 if (RT_FAILURE(rc))
891 RTTestFailed(g_hTest, "PDMIHOSTAUDIO::pfnStreamEnable failed: %Rrc", rc);
892 }
893 return rc;
894}
895
896
897/**
898 * Disables a stream.
899 */
900int AudioTestDriverStackStreamDisable(PAUDIOTESTDRVSTACK pDrvStack, PPDMAUDIOSTREAM pStream)
901{
902 int rc;
903 if (pDrvStack->pIAudioConnector)
904 {
905 rc = pDrvStack->pIAudioConnector->pfnStreamControl(pDrvStack->pIAudioConnector, pStream, PDMAUDIOSTREAMCMD_DISABLE);
906 if (RT_FAILURE(rc))
907 RTTestFailed(g_hTest, "pfnStreamControl/DISABLE failed: %Rrc", rc);
908 }
909 else
910 {
911 PAUDIOTESTDRVSTACKSTREAM pStreamAt = (PAUDIOTESTDRVSTACKSTREAM)pStream;
912 rc = pDrvStack->pIHostAudio->pfnStreamDisable(pDrvStack->pIHostAudio, &pStreamAt->Backend);
913 if (RT_FAILURE(rc))
914 RTTestFailed(g_hTest, "PDMIHOSTAUDIO::pfnStreamDisable failed: %Rrc", rc);
915 }
916 return rc;
917}
918
919
920/**
921 * Drains an output stream.
922 */
923int audioTestDriverStackStreamDrain(PAUDIOTESTDRVSTACK pDrvStack, PPDMAUDIOSTREAM pStream, bool fSync)
924{
925 int rc;
926 if (pDrvStack->pIAudioConnector)
927 {
928 /*
929 * Issue the drain request.
930 */
931 rc = pDrvStack->pIAudioConnector->pfnStreamControl(pDrvStack->pIAudioConnector, pStream, PDMAUDIOSTREAMCMD_DRAIN);
932 if (RT_SUCCESS(rc) && fSync)
933 {
934 /*
935 * This is a synchronous drain, so wait for the driver to change state to inactive.
936 */
937 PDMAUDIOSTREAMSTATE enmState;
938 while ( (enmState = pDrvStack->pIAudioConnector->pfnStreamGetState(pDrvStack->pIAudioConnector, pStream))
939 >= PDMAUDIOSTREAMSTATE_ENABLED)
940 {
941 RTThreadSleep(2);
942 rc = pDrvStack->pIAudioConnector->pfnStreamIterate(pDrvStack->pIAudioConnector, pStream);
943 if (RT_FAILURE(rc))
944 {
945 RTTestFailed(g_hTest, "pfnStreamIterate/DRAIN failed: %Rrc", rc);
946 break;
947 }
948 }
949 if (enmState != PDMAUDIOSTREAMSTATE_INACTIVE)
950 {
951 RTTestFailed(g_hTest, "Stream state not INACTIVE after draining: %s", PDMAudioStreamStateGetName(enmState));
952 rc = VERR_AUDIO_STREAM_NOT_READY;
953 }
954 }
955 else if (RT_FAILURE(rc))
956 RTTestFailed(g_hTest, "pfnStreamControl/ENABLE failed: %Rrc", rc);
957 }
958 else
959 {
960 /*
961 * Issue the drain request.
962 */
963 PAUDIOTESTDRVSTACKSTREAM pStreamAt = (PAUDIOTESTDRVSTACKSTREAM)pStream;
964 rc = pDrvStack->pIHostAudio->pfnStreamDrain(pDrvStack->pIHostAudio, &pStreamAt->Backend);
965 if (RT_SUCCESS(rc) && fSync)
966 {
967 /*
968 * This is a synchronous drain, so wait for the driver to change state to inactive.
969 */
970 PDMHOSTAUDIOSTREAMSTATE enmHostState;
971 while ( (enmHostState = pDrvStack->pIHostAudio->pfnStreamGetState(pDrvStack->pIHostAudio, &pStreamAt->Backend))
972 == PDMHOSTAUDIOSTREAMSTATE_DRAINING)
973 {
974 RTThreadSleep(2);
975 uint32_t cbWritten = UINT32_MAX;
976 rc = pDrvStack->pIHostAudio->pfnStreamPlay(pDrvStack->pIHostAudio, &pStreamAt->Backend,
977 NULL /*pvBuf*/, 0 /*cbBuf*/, &cbWritten);
978 if (RT_FAILURE(rc))
979 {
980 RTTestFailed(g_hTest, "pfnStreamPlay/DRAIN failed: %Rrc", rc);
981 break;
982 }
983 if (cbWritten != 0)
984 {
985 RTTestFailed(g_hTest, "pfnStreamPlay/DRAIN did not set cbWritten to zero: %#x", cbWritten);
986 rc = VERR_MISSING;
987 break;
988 }
989 }
990 if (enmHostState != PDMHOSTAUDIOSTREAMSTATE_OKAY)
991 {
992 RTTestFailed(g_hTest, "Stream state not OKAY after draining: %s", PDMHostAudioStreamStateGetName(enmHostState));
993 rc = VERR_AUDIO_STREAM_NOT_READY;
994 }
995 }
996 else if (RT_FAILURE(rc))
997 RTTestFailed(g_hTest, "PDMIHOSTAUDIO::pfnStreamControl/ENABLE failed: %Rrc", rc);
998 }
999 return rc;
1000}
1001
1002
1003/**
1004 * Checks if the stream is okay.
1005 * @returns true if okay, false if not.
1006 */
1007bool audioTestDriverStackStreamIsOkay(PAUDIOTESTDRVSTACK pDrvStack, PPDMAUDIOSTREAM pStream)
1008{
1009 /*
1010 * Get the stream status and check if it means is okay or not.
1011 */
1012 bool fRc = false;
1013 if (pDrvStack->pIAudioConnector)
1014 {
1015 PDMAUDIOSTREAMSTATE enmState = pDrvStack->pIAudioConnector->pfnStreamGetState(pDrvStack->pIAudioConnector, pStream);
1016 switch (enmState)
1017 {
1018 case PDMAUDIOSTREAMSTATE_NOT_WORKING:
1019 case PDMAUDIOSTREAMSTATE_NEED_REINIT:
1020 break;
1021 case PDMAUDIOSTREAMSTATE_INACTIVE:
1022 case PDMAUDIOSTREAMSTATE_ENABLED:
1023 case PDMAUDIOSTREAMSTATE_ENABLED_READABLE:
1024 case PDMAUDIOSTREAMSTATE_ENABLED_WRITABLE:
1025 fRc = true;
1026 break;
1027 /* no default */
1028 case PDMAUDIOSTREAMSTATE_INVALID:
1029 case PDMAUDIOSTREAMSTATE_END:
1030 case PDMAUDIOSTREAMSTATE_32BIT_HACK:
1031 break;
1032 }
1033 }
1034 else
1035 {
1036 PAUDIOTESTDRVSTACKSTREAM pStreamAt = (PAUDIOTESTDRVSTACKSTREAM)pStream;
1037 PDMHOSTAUDIOSTREAMSTATE enmHostState = pDrvStack->pIHostAudio->pfnStreamGetState(pDrvStack->pIHostAudio,
1038 &pStreamAt->Backend);
1039 switch (enmHostState)
1040 {
1041 case PDMHOSTAUDIOSTREAMSTATE_INITIALIZING:
1042 case PDMHOSTAUDIOSTREAMSTATE_NOT_WORKING:
1043 break;
1044 case PDMHOSTAUDIOSTREAMSTATE_OKAY:
1045 case PDMHOSTAUDIOSTREAMSTATE_DRAINING:
1046 case PDMHOSTAUDIOSTREAMSTATE_INACTIVE:
1047 fRc = true;
1048 break;
1049 /* no default */
1050 case PDMHOSTAUDIOSTREAMSTATE_INVALID:
1051 case PDMHOSTAUDIOSTREAMSTATE_END:
1052 case PDMHOSTAUDIOSTREAMSTATE_32BIT_HACK:
1053 break;
1054 }
1055 }
1056 return fRc;
1057}
1058
1059
1060/**
1061 * Gets the number of bytes it's currently possible to write to the stream.
1062 */
1063uint32_t audioTestDriverStackStreamGetWritable(PAUDIOTESTDRVSTACK pDrvStack, PPDMAUDIOSTREAM pStream)
1064{
1065 uint32_t cbWritable;
1066 if (pDrvStack->pIAudioConnector)
1067 cbWritable = pDrvStack->pIAudioConnector->pfnStreamGetWritable(pDrvStack->pIAudioConnector, pStream);
1068 else
1069 {
1070 PAUDIOTESTDRVSTACKSTREAM pStreamAt = (PAUDIOTESTDRVSTACKSTREAM)pStream;
1071 cbWritable = pDrvStack->pIHostAudio->pfnStreamGetWritable(pDrvStack->pIHostAudio, &pStreamAt->Backend);
1072 }
1073 return cbWritable;
1074}
1075
1076
1077/**
1078 * Tries to play the @a cbBuf bytes of samples in @a pvBuf.
1079 */
1080int audioTestDriverStackStreamPlay(PAUDIOTESTDRVSTACK pDrvStack, PPDMAUDIOSTREAM pStream,
1081 void const *pvBuf, uint32_t cbBuf, uint32_t *pcbPlayed)
1082{
1083 int rc;
1084 if (pDrvStack->pIAudioConnector)
1085 {
1086 rc = pDrvStack->pIAudioConnector->pfnStreamPlay(pDrvStack->pIAudioConnector, pStream, pvBuf, cbBuf, pcbPlayed);
1087 if (RT_FAILURE(rc))
1088 RTTestFailed(g_hTest, "pfnStreamPlay(,,,%#x,) failed: %Rrc", cbBuf, rc);
1089 }
1090 else
1091 {
1092 PAUDIOTESTDRVSTACKSTREAM pStreamAt = (PAUDIOTESTDRVSTACKSTREAM)pStream;
1093 rc = pDrvStack->pIHostAudio->pfnStreamPlay(pDrvStack->pIHostAudio, &pStreamAt->Backend, pvBuf, cbBuf, pcbPlayed);
1094 if (RT_FAILURE(rc))
1095 RTTestFailed(g_hTest, "PDMIHOSTAUDIO::pfnStreamPlay(,,,%#x,) failed: %Rrc", cbBuf, rc);
1096 }
1097 return rc;
1098}
1099
1100
1101/**
1102 * Gets the number of bytes it's currently possible to write to the stream.
1103 */
1104uint32_t audioTestDriverStackStreamGetReadable(PAUDIOTESTDRVSTACK pDrvStack, PPDMAUDIOSTREAM pStream)
1105{
1106 uint32_t cbReadable;
1107 if (pDrvStack->pIAudioConnector)
1108 cbReadable = pDrvStack->pIAudioConnector->pfnStreamGetReadable(pDrvStack->pIAudioConnector, pStream);
1109 else
1110 {
1111 PAUDIOTESTDRVSTACKSTREAM pStreamAt = (PAUDIOTESTDRVSTACKSTREAM)pStream;
1112 cbReadable = pDrvStack->pIHostAudio->pfnStreamGetReadable(pDrvStack->pIHostAudio, &pStreamAt->Backend);
1113 }
1114 return cbReadable;
1115}
1116
1117
1118/**
1119 * Tries to capture @a cbBuf bytes of samples in @a pvBuf.
1120 */
1121int audioTestDriverStackStreamCapture(PAUDIOTESTDRVSTACK pDrvStack, PPDMAUDIOSTREAM pStream,
1122 void *pvBuf, uint32_t cbBuf, uint32_t *pcbCaptured)
1123{
1124 int rc;
1125 if (pDrvStack->pIAudioConnector)
1126 {
1127 rc = pDrvStack->pIAudioConnector->pfnStreamCapture(pDrvStack->pIAudioConnector, pStream, pvBuf, cbBuf, pcbCaptured);
1128 if (RT_FAILURE(rc))
1129 RTTestFailed(g_hTest, "pfnStreamCapture(,,,%#x,) failed: %Rrc", cbBuf, rc);
1130 }
1131 else
1132 {
1133 PAUDIOTESTDRVSTACKSTREAM pStreamAt = (PAUDIOTESTDRVSTACKSTREAM)pStream;
1134 rc = pDrvStack->pIHostAudio->pfnStreamCapture(pDrvStack->pIHostAudio, &pStreamAt->Backend, pvBuf, cbBuf, pcbCaptured);
1135 if (RT_FAILURE(rc))
1136 RTTestFailed(g_hTest, "PDMIHOSTAUDIO::pfnStreamCapture(,,,%#x,) failed: %Rrc", cbBuf, rc);
1137 }
1138 return rc;
1139}
1140
1141
1142/*********************************************************************************************************************************
1143* Mixed streams *
1144*********************************************************************************************************************************/
1145
1146/**
1147 * Initializing mixing for a stream.
1148 *
1149 * This can be used as a do-nothing wrapper for the stack.
1150 *
1151 * @returns VBox status code.
1152 * @param pMix The mixing state.
1153 * @param pStream The stream to mix to/from.
1154 * @param pProps The mixer properties. Pass NULL for no mixing, just
1155 * wrap the driver stack functionality.
1156 * @param cMsBuffer The buffer size.
1157 */
1158int AudioTestMixStreamInit(PAUDIOTESTDRVMIXSTREAM pMix, PAUDIOTESTDRVSTACK pDrvStack, PPDMAUDIOSTREAM pStream,
1159 PCPDMAUDIOPCMPROPS pProps, uint32_t cMsBuffer)
1160{
1161 RT_ZERO(*pMix);
1162
1163 AssertReturn(pDrvStack, VERR_INVALID_PARAMETER);
1164 AssertReturn(pStream, VERR_INVALID_PARAMETER);
1165
1166 pMix->pDrvStack = pDrvStack;
1167 pMix->pStream = pStream;
1168 if (!pProps)
1169 {
1170 pMix->pProps = &pStream->Cfg.Props;
1171 return VINF_SUCCESS;
1172 }
1173
1174 /*
1175 * Okay, we're doing mixing so we need to set up the mixer buffer
1176 * and associated states.
1177 */
1178 pMix->fDoMixing = true;
1179 int rc = AudioMixBufInit(&pMix->MixBuf, "mixer", pProps, PDMAudioPropsMilliToFrames(pProps, cMsBuffer));
1180 if (RT_SUCCESS(rc))
1181 {
1182 pMix->pProps = &pMix->MixBuf.Props;
1183
1184 if (pStream->Cfg.enmDir == PDMAUDIODIR_IN)
1185 {
1186 rc = AudioMixBufInitPeekState(&pMix->MixBuf, &pMix->PeekState, &pMix->MixBuf.Props);
1187 if (RT_SUCCESS(rc))
1188 {
1189 rc = AudioMixBufInitWriteState(&pMix->MixBuf, &pMix->WriteState, &pStream->Cfg.Props);
1190 if (RT_SUCCESS(rc))
1191 return rc;
1192 }
1193 }
1194 else if (pStream->Cfg.enmDir == PDMAUDIODIR_OUT)
1195 {
1196 rc = AudioMixBufInitWriteState(&pMix->MixBuf, &pMix->WriteState, &pMix->MixBuf.Props);
1197 if (RT_SUCCESS(rc))
1198 {
1199 rc = AudioMixBufInitPeekState(&pMix->MixBuf, &pMix->PeekState, &pStream->Cfg.Props);
1200 if (RT_SUCCESS(rc))
1201 return rc;
1202 }
1203 }
1204 else
1205 {
1206 RTTestFailed(g_hTest, "Bogus stream direction!");
1207 rc = VERR_INVALID_STATE;
1208 }
1209 }
1210 else
1211 RTTestFailed(g_hTest, "AudioMixBufInit failed: %Rrc", rc);
1212 RT_ZERO(*pMix);
1213 return rc;
1214}
1215
1216
1217/**
1218 * Terminate mixing (leaves the stream untouched).
1219 *
1220 * @param pMix The mixing state.
1221 */
1222void AudioTestMixStreamTerm(PAUDIOTESTDRVMIXSTREAM pMix)
1223{
1224 if (pMix->fDoMixing)
1225 {
1226 AudioMixBufTerm(&pMix->MixBuf);
1227 pMix->pStream = NULL;
1228 }
1229 RT_ZERO(*pMix);
1230}
1231
1232
1233/**
1234 * Worker that transports data between the mixer buffer and the drivers.
1235 *
1236 * @returns VBox status code.
1237 * @param pMix The mixer stream setup to do transfers for.
1238 */
1239static int audioTestMixStreamTransfer(PAUDIOTESTDRVMIXSTREAM pMix)
1240{
1241 uint8_t abBuf[16384];
1242 if (pMix->pStream->Cfg.enmDir == PDMAUDIODIR_IN)
1243 {
1244 /*
1245 * Try fill up the mixer buffer as much as possible.
1246 *
1247 * Slight fun part is that we have to calculate conversion
1248 * ratio and be rather pessimistic about it.
1249 */
1250 uint32_t const cbBuf = PDMAudioPropsFloorBytesToFrame(&pMix->pStream->Cfg.Props, sizeof(abBuf));
1251 for (;;)
1252 {
1253 /*
1254 * Figure out how much we can move in this iteration.
1255 */
1256 uint32_t cDstFrames = AudioMixBufFree(&pMix->MixBuf);
1257 if (!cDstFrames)
1258 break;
1259
1260 uint32_t cbReadable = audioTestDriverStackStreamGetReadable(pMix->pDrvStack, pMix->pStream);
1261 if (!cbReadable)
1262 break;
1263
1264 uint32_t cbToRead;
1265 if (PDMAudioPropsHz(&pMix->pStream->Cfg.Props) == PDMAudioPropsHz(&pMix->MixBuf.Props))
1266 cbToRead = PDMAudioPropsFramesToBytes(&pMix->pStream->Cfg.Props, cDstFrames);
1267 else
1268 cbToRead = PDMAudioPropsFramesToBytes(&pMix->pStream->Cfg.Props,
1269 (uint64_t)cDstFrames * PDMAudioPropsHz(&pMix->pStream->Cfg.Props)
1270 / PDMAudioPropsHz(&pMix->MixBuf.Props));
1271 cbToRead = RT_MIN(cbToRead, RT_MIN(cbReadable, cbBuf));
1272 if (!cbToRead)
1273 break;
1274
1275 /*
1276 * Get the data.
1277 */
1278 uint32_t cbCaptured = 0;
1279 int rc = audioTestDriverStackStreamCapture(pMix->pDrvStack, pMix->pStream, abBuf, cbToRead, &cbCaptured);
1280 if (RT_FAILURE(rc))
1281 return rc;
1282 Assert(cbCaptured == cbToRead);
1283 AssertBreak(cbCaptured > 0);
1284
1285 /*
1286 * Feed it to the mixer.
1287 */
1288 uint32_t cDstFramesWritten = 0;
1289 if ((abBuf[0] >> 4) & 1) /* some cheap random */
1290 AudioMixBufWrite(&pMix->MixBuf, &pMix->WriteState, abBuf, cbCaptured,
1291 0 /*offDstFrame*/, cDstFrames, &cDstFramesWritten);
1292 else
1293 {
1294 AudioMixBufSilence(&pMix->MixBuf, &pMix->WriteState, 0 /*offFrame*/, cDstFrames);
1295 AudioMixBufBlend(&pMix->MixBuf, &pMix->WriteState, abBuf, cbCaptured,
1296 0 /*offDstFrame*/, cDstFrames, &cDstFramesWritten);
1297 }
1298 AudioMixBufCommit(&pMix->MixBuf, cDstFramesWritten);
1299 }
1300 }
1301 else
1302 {
1303 /*
1304 * The goal here is to empty the mixer buffer by transfering all
1305 * the data to the drivers.
1306 */
1307 uint32_t const cbBuf = PDMAudioPropsFloorBytesToFrame(&pMix->MixBuf.Props, sizeof(abBuf));
1308 for (;;)
1309 {
1310 uint32_t cFrames = AudioMixBufUsed(&pMix->MixBuf);
1311 if (!cFrames)
1312 break;
1313
1314 uint32_t cbWritable = audioTestDriverStackStreamGetWritable(pMix->pDrvStack, pMix->pStream);
1315 if (!cbWritable)
1316 break;
1317
1318 uint32_t cSrcFramesPeeked;
1319 uint32_t cbDstPeeked;
1320 AudioMixBufPeek(&pMix->MixBuf, 0 /*offSrcFrame*/, cFrames, &cSrcFramesPeeked,
1321 &pMix->PeekState, abBuf, RT_MIN(cbBuf, cbWritable), &cbDstPeeked);
1322 AudioMixBufAdvance(&pMix->MixBuf, cSrcFramesPeeked);
1323
1324 if (!cbDstPeeked)
1325 break;
1326
1327 uint32_t offBuf = 0;
1328 while (offBuf < cbDstPeeked)
1329 {
1330 uint32_t cbPlayed = 0;
1331 int rc = audioTestDriverStackStreamPlay(pMix->pDrvStack, pMix->pStream,
1332 &abBuf[offBuf], cbDstPeeked - offBuf, &cbPlayed);
1333 if (RT_FAILURE(rc))
1334 return rc;
1335 if (!cbPlayed)
1336 RTThreadSleep(1);
1337 offBuf += cbPlayed;
1338 }
1339 }
1340 }
1341 return VINF_SUCCESS;
1342}
1343
1344
1345/**
1346 * Same as audioTestDriverStackStreamEnable.
1347 */
1348int AudioTestMixStreamEnable(PAUDIOTESTDRVMIXSTREAM pMix)
1349{
1350 return audioTestDriverStackStreamEnable(pMix->pDrvStack, pMix->pStream);
1351}
1352
1353
1354/**
1355 * Same as audioTestDriverStackStreamDrain.
1356 */
1357int AudioTestMixStreamDrain(PAUDIOTESTDRVMIXSTREAM pMix, bool fSync)
1358{
1359 /*
1360 * If we're mixing, we must first make sure the buffer is empty.
1361 */
1362 if (pMix->fDoMixing)
1363 {
1364 audioTestMixStreamTransfer(pMix);
1365 while (AudioMixBufUsed(&pMix->MixBuf) > 0)
1366 {
1367 RTThreadSleep(1);
1368 audioTestMixStreamTransfer(pMix);
1369 }
1370 }
1371
1372 /*
1373 * Then we do the regular workt.
1374 */
1375 return audioTestDriverStackStreamDrain(pMix->pDrvStack, pMix->pStream, fSync);
1376}
1377
1378/**
1379 * Same as audioTestDriverStackStreamDisable.
1380 */
1381int AudioTestMixStreamDisable(PAUDIOTESTDRVMIXSTREAM pMix)
1382{
1383 return AudioTestDriverStackStreamDisable(pMix->pDrvStack, pMix->pStream);
1384}
1385
1386
1387/**
1388 * Same as audioTestDriverStackStreamIsOkay.
1389 */
1390bool AudioTestMixStreamIsOkay(PAUDIOTESTDRVMIXSTREAM pMix)
1391{
1392 return audioTestDriverStackStreamIsOkay(pMix->pDrvStack, pMix->pStream);
1393}
1394
1395
1396/**
1397 * Same as audioTestDriverStackStreamGetWritable
1398 */
1399uint32_t AudioTestMixStreamGetWritable(PAUDIOTESTDRVMIXSTREAM pMix)
1400{
1401 if (!pMix->fDoMixing)
1402 return audioTestDriverStackStreamGetWritable(pMix->pDrvStack, pMix->pStream);
1403 uint32_t cbRet = AudioMixBufFreeBytes(&pMix->MixBuf);
1404 if (!cbRet)
1405 {
1406 audioTestMixStreamTransfer(pMix);
1407 cbRet = AudioMixBufFreeBytes(&pMix->MixBuf);
1408 }
1409 return cbRet;
1410}
1411
1412
1413
1414
1415/**
1416 * Same as audioTestDriverStackStreamPlay.
1417 */
1418int AudioTestMixStreamPlay(PAUDIOTESTDRVMIXSTREAM pMix, void const *pvBuf, uint32_t cbBuf, uint32_t *pcbPlayed)
1419{
1420 if (!pMix->fDoMixing)
1421 return audioTestDriverStackStreamPlay(pMix->pDrvStack, pMix->pStream, pvBuf, cbBuf, pcbPlayed);
1422
1423 *pcbPlayed = 0;
1424
1425 int rc = audioTestMixStreamTransfer(pMix);
1426 if (RT_FAILURE(rc))
1427 return rc;
1428
1429 uint32_t const cbFrame = PDMAudioPropsFrameSize(&pMix->MixBuf.Props);
1430 while (cbBuf >= cbFrame)
1431 {
1432 uint32_t const cFrames = AudioMixBufFree(&pMix->MixBuf);
1433 if (!cFrames)
1434 break;
1435 uint32_t cbToWrite = PDMAudioPropsFramesToBytes(&pMix->MixBuf.Props, cFrames);
1436 cbToWrite = RT_MIN(cbToWrite, cbBuf);
1437 cbToWrite = PDMAudioPropsFloorBytesToFrame(&pMix->MixBuf.Props, cbToWrite);
1438
1439 uint32_t cFramesWritten = 0;
1440 AudioMixBufWrite(&pMix->MixBuf, &pMix->WriteState, pvBuf, cbToWrite, 0 /*offDstFrame*/, cFrames, &cFramesWritten);
1441 Assert(cFramesWritten == PDMAudioPropsBytesToFrames(&pMix->MixBuf.Props, cbToWrite));
1442 AudioMixBufCommit(&pMix->MixBuf, cFramesWritten);
1443
1444 *pcbPlayed += cbToWrite;
1445 cbBuf -= cbToWrite;
1446 pvBuf = (uint8_t const *)pvBuf + cbToWrite;
1447
1448 rc = audioTestMixStreamTransfer(pMix);
1449 if (RT_FAILURE(rc))
1450 return *pcbPlayed ? VINF_SUCCESS : rc;
1451 }
1452
1453 return VINF_SUCCESS;
1454}
1455
1456
1457/**
1458 * Same as audioTestDriverStackStreamGetReadable
1459 */
1460uint32_t AudioTestMixStreamGetReadable(PAUDIOTESTDRVMIXSTREAM pMix)
1461{
1462 if (!pMix->fDoMixing)
1463 return audioTestDriverStackStreamGetReadable(pMix->pDrvStack, pMix->pStream);
1464
1465 audioTestMixStreamTransfer(pMix);
1466 uint32_t cbRet = AudioMixBufUsedBytes(&pMix->MixBuf);
1467 return cbRet;
1468}
1469
1470
1471
1472
1473/**
1474 * Same as audioTestDriverStackStreamCapture.
1475 */
1476int AudioTestMixStreamCapture(PAUDIOTESTDRVMIXSTREAM pMix, void *pvBuf, uint32_t cbBuf, uint32_t *pcbCaptured)
1477{
1478 if (!pMix->fDoMixing)
1479 return audioTestDriverStackStreamCapture(pMix->pDrvStack, pMix->pStream, pvBuf, cbBuf, pcbCaptured);
1480
1481 *pcbCaptured = 0;
1482
1483 int rc = audioTestMixStreamTransfer(pMix);
1484 if (RT_FAILURE(rc))
1485 return rc;
1486
1487 uint32_t const cbFrame = PDMAudioPropsFrameSize(&pMix->MixBuf.Props);
1488 while (cbBuf >= cbFrame)
1489 {
1490 uint32_t const cFrames = AudioMixBufUsed(&pMix->MixBuf);
1491 if (!cFrames)
1492 break;
1493 uint32_t cbToRead = PDMAudioPropsFramesToBytes(&pMix->MixBuf.Props, cFrames);
1494 cbToRead = RT_MIN(cbToRead, cbBuf);
1495 cbToRead = PDMAudioPropsFloorBytesToFrame(&pMix->MixBuf.Props, cbToRead);
1496
1497 uint32_t cFramesPeeked = 0;
1498 uint32_t cbPeeked = 0;
1499 AudioMixBufPeek(&pMix->MixBuf, 0 /*offSrcFrame*/, cFrames, &cFramesPeeked, &pMix->PeekState, pvBuf, cbToRead, &cbPeeked);
1500 Assert(cFramesPeeked == PDMAudioPropsBytesToFrames(&pMix->MixBuf.Props, cbPeeked));
1501 AudioMixBufAdvance(&pMix->MixBuf, cFramesPeeked);
1502
1503 *pcbCaptured += cbToRead;
1504 cbBuf -= cbToRead;
1505 pvBuf = (uint8_t *)pvBuf + cbToRead;
1506
1507 rc = audioTestMixStreamTransfer(pMix);
1508 if (RT_FAILURE(rc))
1509 return *pcbCaptured ? VINF_SUCCESS : rc;
1510 }
1511
1512 return VINF_SUCCESS;
1513}
1514
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