VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostAudioDebug.cpp@ 88574

Last change on this file since 88574 was 88534, checked in by vboxsync, 4 years ago

Audio: Merged the cbStreamOut and cbStreamIn fields in PDMAUDIOBACKENDCFG (into cbStream). Added a fFlags member to PDMAUDIOBACKENDCFG, currently must-be-zero. bugref:9890

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 16.9 KB
Line 
1/* $Id: DrvHostAudioDebug.cpp 88534 2021-04-15 12:16:56Z vboxsync $ */
2/** @file
3 * Host audio driver - Debug - For dumping and injecting audio data from/to the device emulation.
4 */
5
6/*
7 * Copyright (C) 2016-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
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
23#include <VBox/log.h>
24#include <VBox/vmm/pdmaudioifs.h>
25#include <VBox/vmm/pdmaudioinline.h>
26
27#include <iprt/rand.h>
28#include <iprt/uuid.h> /* For PDMIBASE_2_PDMDRV. */
29
30#define _USE_MATH_DEFINES
31#include <math.h> /* sin, M_PI */
32
33#include "AudioHlp.h"
34#include "VBoxDD.h"
35
36
37/*********************************************************************************************************************************
38* Structures and Typedefs *
39*********************************************************************************************************************************/
40/**
41 * Debug host audio stream.
42 */
43typedef struct DEBUGAUDIOSTREAM
44{
45 /** The stream's acquired configuration. */
46 PDMAUDIOSTREAMCFG Cfg;
47 /** Audio file to dump output to or read input from. */
48 PAUDIOHLPFILE pFile;
49 union
50 {
51 struct
52 {
53 /** Current sample index for generate the sine wave. */
54 uint64_t uSample;
55 /** The fixed portion of the sin() input. */
56 double rdFixed;
57 /** Timestamp of last captured samples. */
58 uint64_t tsLastCaptured;
59 /** Frequency (in Hz) of the sine wave to generate. */
60 double rdFreqHz;
61 } In;
62 };
63} DEBUGAUDIOSTREAM;
64/** Pointer to a debug host audio stream. */
65typedef DEBUGAUDIOSTREAM *PDEBUGAUDIOSTREAM;
66
67/**
68 * Debug audio driver instance data.
69 * @implements PDMIAUDIOCONNECTOR
70 */
71typedef struct DRVHOSTDEBUGAUDIO
72{
73 /** Pointer to the driver instance structure. */
74 PPDMDRVINS pDrvIns;
75 /** Pointer to host audio interface. */
76 PDMIHOSTAUDIO IHostAudio;
77} DRVHOSTDEBUGAUDIO;
78/** Pointer to a debug host audio driver. */
79typedef DRVHOSTDEBUGAUDIO *PDRVHOSTDEBUGAUDIO;
80
81
82/*********************************************************************************************************************************
83* Global Variables *
84*********************************************************************************************************************************/
85/** Frequency selection for input streams. */
86static const double s_ardInputFreqsHz[] =
87{
88 349.2282 /*F4*/,
89 440.0000 /*A4*/,
90 523.2511 /*C5*/,
91 698.4565 /*F5*/,
92 880.0000 /*A5*/,
93 1046.502 /*C6*/,
94 1174.659 /*D6*/,
95 1396.913 /*F6*/,
96 1760.0000 /*A6*/
97};
98
99
100/**
101 * @interface_method_impl{PDMIHOSTAUDIO,pfnGetConfig}
102 */
103static DECLCALLBACK(int) drvHostDebugAudioHA_GetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pBackendCfg)
104{
105 RT_NOREF(pInterface);
106 AssertPtrReturn(pBackendCfg, VERR_INVALID_POINTER);
107
108 /*
109 * Fill in the config structure.
110 */
111 RTStrCopy(pBackendCfg->szName, sizeof(pBackendCfg->szName), "DebugAudio");
112 pBackendCfg->cbStream = sizeof(DEBUGAUDIOSTREAM);
113 pBackendCfg->fFlags = 0;
114 pBackendCfg->cMaxStreamsOut = 1; /* Output; writing to a file. */
115 pBackendCfg->cMaxStreamsIn = 1; /* Input; generates a sine wave. */
116
117 return VINF_SUCCESS;
118}
119
120
121/**
122 * @interface_method_impl{PDMIHOSTAUDIO,pfnGetStatus}
123 */
124static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvHostDebugAudioHA_GetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
125{
126 RT_NOREF(pInterface, enmDir);
127 return PDMAUDIOBACKENDSTS_RUNNING;
128}
129
130
131/**
132 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamCreate}
133 */
134static DECLCALLBACK(int) drvHostDebugAudioHA_StreamCreate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream,
135 PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
136{
137 PDRVHOSTDEBUGAUDIO pDrv = RT_FROM_MEMBER(pInterface, DRVHOSTDEBUGAUDIO, IHostAudio);
138 PDEBUGAUDIOSTREAM pStreamDbg = (PDEBUGAUDIOSTREAM)pStream;
139 AssertPtrReturn(pStreamDbg, VERR_INVALID_POINTER);
140 AssertPtrReturn(pCfgReq, VERR_INVALID_POINTER);
141 AssertPtrReturn(pCfgAcq, VERR_INVALID_POINTER);
142
143 PDMAudioStrmCfgCopy(&pStreamDbg->Cfg, pCfgAcq);
144
145 if (pCfgReq->enmDir == PDMAUDIODIR_IN)
146 {
147 /* Pick a frequency from our selection, so that every time a recording starts
148 we'll hopfully generate a different note. */
149 pStreamDbg->In.rdFreqHz = s_ardInputFreqsHz[RTRandU32Ex(0, RT_ELEMENTS(s_ardInputFreqsHz) - 1)];
150 pStreamDbg->In.rdFixed = 2.0 * M_PI * pStreamDbg->In.rdFreqHz / PDMAudioPropsHz(&pStreamDbg->Cfg.Props);
151 pStreamDbg->In.uSample = 0;
152 }
153
154 int rc = AudioHlpFileCreateAndOpenEx(&pStreamDbg->pFile, AUDIOHLPFILETYPE_WAV, NULL /*use temp dir*/,
155 pCfgReq->enmDir == PDMAUDIODIR_IN ? "DebugAudioIn" : "DebugAudioOut",
156 pDrv->pDrvIns->iInstance, AUDIOHLPFILENAME_FLAGS_NONE, AUDIOHLPFILE_FLAGS_NONE,
157 &pCfgReq->Props, RTFILE_O_WRITE | RTFILE_O_DENY_WRITE | RTFILE_O_CREATE_REPLACE);
158 if (RT_FAILURE(rc))
159 LogRel(("DebugAudio: Failed to creating debug file for %s stream '%s' in the temp directory: %Rrc\n",
160 pCfgReq->enmDir == PDMAUDIODIR_IN ? "input" : "output", pCfgReq->szName, rc));
161
162 return rc;
163}
164
165
166/**
167 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamDestroy}
168 */
169static DECLCALLBACK(int) drvHostDebugAudioHA_StreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
170{
171 RT_NOREF(pInterface);
172 PDEBUGAUDIOSTREAM pStreamDbg = (PDEBUGAUDIOSTREAM)pStream;
173 AssertPtrReturn(pStreamDbg, VERR_INVALID_POINTER);
174
175 if (pStreamDbg->pFile)
176 {
177 AudioHlpFileDestroy(pStreamDbg->pFile);
178 pStreamDbg->pFile = NULL;
179 }
180
181 return VINF_SUCCESS;
182}
183
184
185/**
186 * @ interface_method_impl{PDMIHOSTAUDIO,pfnStreamEnable}
187 */
188static DECLCALLBACK(int) drvHostDebugAudioHA_StreamControlStub(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
189{
190 RT_NOREF(pInterface, pStream);
191 return VINF_SUCCESS;
192}
193
194
195/**
196 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamControl}
197 */
198static DECLCALLBACK(int) drvHostDebugAudioHA_StreamControl(PPDMIHOSTAUDIO pInterface,
199 PPDMAUDIOBACKENDSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
200{
201 /** @todo r=bird: I'd like to get rid of this pfnStreamControl method,
202 * replacing it with individual StreamXxxx methods. That would save us
203 * potentally huge switches and more easily see which drivers implement
204 * which operations (grep for pfnStreamXxxx). */
205 switch (enmStreamCmd)
206 {
207 case PDMAUDIOSTREAMCMD_ENABLE:
208 return drvHostDebugAudioHA_StreamControlStub(pInterface, pStream);
209 case PDMAUDIOSTREAMCMD_DISABLE:
210 return drvHostDebugAudioHA_StreamControlStub(pInterface, pStream);
211 case PDMAUDIOSTREAMCMD_PAUSE:
212 return drvHostDebugAudioHA_StreamControlStub(pInterface, pStream);
213 case PDMAUDIOSTREAMCMD_RESUME:
214 return drvHostDebugAudioHA_StreamControlStub(pInterface, pStream);
215 case PDMAUDIOSTREAMCMD_DRAIN:
216 return drvHostDebugAudioHA_StreamControlStub(pInterface, pStream);
217
218 case PDMAUDIOSTREAMCMD_END:
219 case PDMAUDIOSTREAMCMD_32BIT_HACK:
220 case PDMAUDIOSTREAMCMD_INVALID:
221 /* no default*/
222 break;
223 }
224 return VERR_NOT_SUPPORTED;
225}
226
227
228/**
229 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetReadable}
230 */
231static DECLCALLBACK(uint32_t) drvHostDebugAudioHA_StreamGetReadable(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
232{
233 RT_NOREF(pInterface);
234 PDEBUGAUDIOSTREAM pStreamDbg = (PDEBUGAUDIOSTREAM)pStream;
235
236 return PDMAudioPropsMilliToBytes(&pStreamDbg->Cfg.Props, 10 /*ms*/);
237}
238
239
240/**
241 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetWritable}
242 */
243static DECLCALLBACK(uint32_t) drvHostDebugAudioHA_StreamGetWritable(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
244{
245 RT_NOREF(pInterface, pStream);
246 return UINT32_MAX;
247}
248
249
250/**
251 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetPending}
252 */
253static DECLCALLBACK(uint32_t) drvHostDebugAudioHA_StreamGetPending(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
254{
255 RT_NOREF(pInterface, pStream);
256 return 0;
257}
258
259
260/**
261 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetWritable}
262 */
263static DECLCALLBACK(PDMAUDIOSTREAMSTS) drvHostDebugAudioHA_StreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
264{
265 RT_NOREF(pInterface, pStream);
266 return PDMAUDIOSTREAMSTS_FLAGS_INITIALIZED | PDMAUDIOSTREAMSTS_FLAGS_ENABLED;
267}
268
269
270/**
271 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamPlay}
272 */
273static DECLCALLBACK(int) drvHostDebugAudioHA_StreamPlay(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream,
274 const void *pvBuf, uint32_t cbBuf, uint32_t *pcbWritten)
275{
276 RT_NOREF(pInterface);
277 PDEBUGAUDIOSTREAM pStreamDbg = (PDEBUGAUDIOSTREAM)pStream;
278
279 int rc = AudioHlpFileWrite(pStreamDbg->pFile, pvBuf, cbBuf, 0 /* fFlags */);
280 if (RT_SUCCESS(rc))
281 *pcbWritten = cbBuf;
282 else
283 LogRelMax(32, ("DebugAudio: Writing output failed with %Rrc\n", rc));
284 return rc;
285}
286
287
288/**
289 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamCapture}
290 */
291static DECLCALLBACK(int) drvHostDebugAudioHA_StreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream,
292 void *pvBuf, uint32_t cbBuf, uint32_t *pcbRead)
293{
294 RT_NOREF(pInterface);
295 PDEBUGAUDIOSTREAM pStreamDbg = (PDEBUGAUDIOSTREAM)pStream;
296/** @todo rate limit this? */
297
298 /*
299 * Clear the buffer first so we don't need to thing about additional channels.
300 */
301 uint32_t cFrames = PDMAudioPropsBytesToFrames(&pStreamDbg->Cfg.Props, cbBuf);
302 PDMAudioPropsClearBuffer(&pStreamDbg->Cfg.Props, pvBuf, cbBuf, cFrames);
303
304 /*
305 * Generate the select sin wave in the first channel:
306 */
307 uint32_t const cbFrame = PDMAudioPropsFrameSize(&pStreamDbg->Cfg.Props);
308 double const rdFixed = pStreamDbg->In.rdFixed;
309 uint64_t iSrcFrame = pStreamDbg->In.uSample;
310 switch (PDMAudioPropsSampleSize(&pStreamDbg->Cfg.Props))
311 {
312 case 1:
313 /* untested */
314 if (PDMAudioPropsIsSigned(&pStreamDbg->Cfg.Props))
315 {
316 int8_t *piSample = (int8_t *)pvBuf;
317 while (cFrames-- > 0)
318 {
319 *piSample = 126 /*Amplitude*/ * sin(rdFixed * iSrcFrame);
320 iSrcFrame++;
321 piSample += cbFrame;
322 }
323 }
324 else
325 {
326 /* untested */
327 uint16_t *pbSample = (uint16_t *)pvBuf;
328 while (cFrames-- > 0)
329 {
330 *pbSample = 126 /*Amplitude*/ * sin(rdFixed * iSrcFrame) + 0x80;
331 iSrcFrame++;
332 pbSample += cbFrame;
333 }
334 }
335 break;
336
337 case 2:
338 if (PDMAudioPropsIsSigned(&pStreamDbg->Cfg.Props))
339 {
340 int16_t *piSample = (int16_t *)pvBuf;
341 while (cFrames-- > 0)
342 {
343 *piSample = 32760 /*Amplitude*/ * sin(rdFixed * iSrcFrame);
344 iSrcFrame++;
345 piSample = (int16_t *)((uint8_t *)piSample + cbFrame);
346 }
347 }
348 else
349 {
350 /* untested */
351 uint16_t *puSample = (uint16_t *)pvBuf;
352 while (cFrames-- > 0)
353 {
354 *puSample = 32760 /*Amplitude*/ * sin(rdFixed * iSrcFrame) + 0x8000;
355 iSrcFrame++;
356 puSample = (uint16_t *)((uint8_t *)puSample + cbFrame);
357 }
358 }
359 break;
360
361 case 4:
362 /* untested */
363 if (PDMAudioPropsIsSigned(&pStreamDbg->Cfg.Props))
364 {
365 int32_t *piSample = (int32_t *)pvBuf;
366 while (cFrames-- > 0)
367 {
368 *piSample = (32760 << 16) /*Amplitude*/ * sin(rdFixed * iSrcFrame);
369 iSrcFrame++;
370 piSample = (int32_t *)((uint8_t *)piSample + cbFrame);
371 }
372 }
373 else
374 {
375 uint32_t *puSample = (uint32_t *)pvBuf;
376 while (cFrames-- > 0)
377 {
378 *puSample = (32760 << 16) /*Amplitude*/ * sin(rdFixed * iSrcFrame) + UINT32_C(0x80000000);
379 iSrcFrame++;
380 puSample = (uint32_t *)((uint8_t *)puSample + cbFrame);
381 }
382 }
383 break;
384
385 default:
386 AssertFailed();
387 }
388 pStreamDbg->In.uSample = iSrcFrame;
389
390 /*
391 * Write it.
392 */
393 int rc = AudioHlpFileWrite(pStreamDbg->pFile, pvBuf, cbBuf, 0 /* fFlags */);
394 if (RT_SUCCESS(rc))
395 *pcbRead = cbBuf;
396 else
397 LogRelMax(32, ("DebugAudio: Writing input failed with %Rrc\n", rc));
398 return rc;
399}
400
401
402/**
403 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
404 */
405static DECLCALLBACK(void *) drvHostDebugAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
406{
407 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
408 PDRVHOSTDEBUGAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTDEBUGAUDIO);
409
410 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
411 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
412 return NULL;
413}
414
415
416/**
417 * Constructs a Null audio driver instance.
418 *
419 * @copydoc FNPDMDRVCONSTRUCT
420 */
421static DECLCALLBACK(int) drvHostDebugAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
422{
423 RT_NOREF(pCfg, fFlags);
424 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
425 PDRVHOSTDEBUGAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTDEBUGAUDIO);
426 LogRel(("Audio: Initializing DEBUG driver\n"));
427
428 /*
429 * Init the static parts.
430 */
431 pThis->pDrvIns = pDrvIns;
432 /* IBase */
433 pDrvIns->IBase.pfnQueryInterface = drvHostDebugAudioQueryInterface;
434 /* IHostAudio */
435 pThis->IHostAudio.pfnGetConfig = drvHostDebugAudioHA_GetConfig;
436 pThis->IHostAudio.pfnGetDevices = NULL;
437 pThis->IHostAudio.pfnGetStatus = drvHostDebugAudioHA_GetStatus;
438 pThis->IHostAudio.pfnStreamCreate = drvHostDebugAudioHA_StreamCreate;
439 pThis->IHostAudio.pfnStreamDestroy = drvHostDebugAudioHA_StreamDestroy;
440 pThis->IHostAudio.pfnStreamControl = drvHostDebugAudioHA_StreamControl;
441 pThis->IHostAudio.pfnStreamGetReadable = drvHostDebugAudioHA_StreamGetReadable;
442 pThis->IHostAudio.pfnStreamGetWritable = drvHostDebugAudioHA_StreamGetWritable;
443 pThis->IHostAudio.pfnStreamGetPending = drvHostDebugAudioHA_StreamGetPending;
444 pThis->IHostAudio.pfnStreamGetStatus = drvHostDebugAudioHA_StreamGetStatus;
445 pThis->IHostAudio.pfnStreamPlay = drvHostDebugAudioHA_StreamPlay;
446 pThis->IHostAudio.pfnStreamCapture = drvHostDebugAudioHA_StreamCapture;
447
448#ifdef VBOX_AUDIO_DEBUG_DUMP_PCM_DATA
449 RTFileDelete(VBOX_AUDIO_DEBUG_DUMP_PCM_DATA_PATH "AudioDebugOutput.pcm");
450#endif
451
452 return VINF_SUCCESS;
453}
454
455/**
456 * Char driver registration record.
457 */
458const PDMDRVREG g_DrvHostDebugAudio =
459{
460 /* u32Version */
461 PDM_DRVREG_VERSION,
462 /* szName */
463 "DebugAudio",
464 /* szRCMod */
465 "",
466 /* szR0Mod */
467 "",
468 /* pszDescription */
469 "Debug audio host driver",
470 /* fFlags */
471 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
472 /* fClass. */
473 PDM_DRVREG_CLASS_AUDIO,
474 /* cMaxInstances */
475 ~0U,
476 /* cbInstance */
477 sizeof(DRVHOSTDEBUGAUDIO),
478 /* pfnConstruct */
479 drvHostDebugAudioConstruct,
480 /* pfnDestruct */
481 NULL,
482 /* pfnRelocate */
483 NULL,
484 /* pfnIOCtl */
485 NULL,
486 /* pfnPowerOn */
487 NULL,
488 /* pfnReset */
489 NULL,
490 /* pfnSuspend */
491 NULL,
492 /* pfnResume */
493 NULL,
494 /* pfnAttach */
495 NULL,
496 /* pfnDetach */
497 NULL,
498 /* pfnPowerOff */
499 NULL,
500 /* pfnSoftReset */
501 NULL,
502 /* u32EndVersion */
503 PDM_DRVREG_VERSION
504};
505
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