VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostDebugAudio.cpp@ 63713

Last change on this file since 63713 was 63711, checked in by vboxsync, 8 years ago

Audio: Implemented support for audio device enumeration handling, audio device information and audio backend notifications. This will enable to let the backends tell the audio subsystem that the host audio configuration has changed and react accordingly to it. For now only the Core Audio backend supports device enumeration. Further this also will get rid of the static initialization on the device emulation side, which, if at VM startup no audio input(s) / output(s) were available, was triggering a warning. The NULL backend therefore does not need to act as a (static) fallback anymore.

Work in progress.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.2 KB
Line 
1/* $Id: DrvHostDebugAudio.cpp 63711 2016-09-05 12:04:01Z vboxsync $ */
2/** @file
3 * Debug audio driver -- host backend for dumping and injecting audio data
4 * from/to the device emulation.
5 */
6
7/*
8 * Copyright (C) 2016 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 * --------------------------------------------------------------------
18 */
19
20#include <iprt/alloc.h>
21#include <iprt/uuid.h> /* For PDMIBASE_2_PDMDRV. */
22
23#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
24#include <VBox/log.h>
25#include <VBox/vmm/pdmaudioifs.h>
26
27#include "DrvAudio.h"
28#include "AudioMixBuffer.h"
29#include "VBoxDD.h"
30
31
32/**
33 * Structure for keeping a debug input/output stream.
34 */
35typedef struct DEBUGAUDIOSTREAM
36{
37 /** Note: Always must come first! */
38 PDMAUDIOSTREAM Stream;
39 /** Audio file to dump output to or read input from. */
40 PDMAUDIOFILE File;
41 union
42 {
43 struct
44 {
45 /** Timestamp of last captured samples. */
46 uint64_t tsLastCaptured;
47 } In;
48 struct
49 {
50 /** Timestamp of last played samples. */
51 uint64_t tsLastPlayed;
52 uint64_t cMaxSamplesInPlayBuffer;
53 uint8_t *pu8PlayBuffer;
54 } Out;
55 };
56
57} DEBUGAUDIOSTREAM, *PDEBUGAUDIOSTREAM;
58
59/**
60 * Debug audio driver instance data.
61 * @implements PDMIAUDIOCONNECTOR
62 */
63typedef struct DRVHOSTDEBUGAUDIO
64{
65 /** Pointer to the driver instance structure. */
66 PPDMDRVINS pDrvIns;
67 /** Pointer to host audio interface. */
68 PDMIHOSTAUDIO IHostAudio;
69} DRVHOSTDEBUGAUDIO, *PDRVHOSTDEBUGAUDIO;
70
71/*******************************************PDM_AUDIO_DRIVER******************************/
72
73
74/**
75 * @interface_method_impl{PDMIHOSTAUDIO,pfnGetConfig}
76 */
77static DECLCALLBACK(int) drvHostDebugAudioGetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pBackendCfg)
78{
79 NOREF(pInterface);
80 AssertPtrReturn(pBackendCfg, VERR_INVALID_POINTER);
81
82 pBackendCfg->cbStreamOut = sizeof(DEBUGAUDIOSTREAM);
83 pBackendCfg->cbStreamIn = sizeof(DEBUGAUDIOSTREAM);
84
85 pBackendCfg->cMaxStreamsOut = 1; /* Output */
86 pBackendCfg->cMaxStreamsIn = 2; /* Line input + microphone input. */
87
88 return VINF_SUCCESS;
89}
90
91
92/**
93 * @interface_method_impl{PDMIHOSTAUDIO,pfnInit}
94 */
95static DECLCALLBACK(int) drvHostDebugAudioInit(PPDMIHOSTAUDIO pInterface)
96{
97 NOREF(pInterface);
98
99 LogFlowFuncLeaveRC(VINF_SUCCESS);
100 return VINF_SUCCESS;
101}
102
103
104/**
105 * @interface_method_impl{PDMIHOSTAUDIO,pfnShutdown}
106 */
107static DECLCALLBACK(void) drvHostDebugAudioShutdown(PPDMIHOSTAUDIO pInterface)
108{
109 NOREF(pInterface);
110}
111
112
113/**
114 * @interface_method_impl{PDMIHOSTAUDIO,pfnGetStatus}
115 */
116static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvHostDebugAudioGetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
117{
118 RT_NOREF(enmDir);
119 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
120
121 return PDMAUDIOBACKENDSTS_RUNNING;
122}
123
124
125static int debugCreateStreamIn(PPDMIHOSTAUDIO pInterface,
126 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
127{
128 RT_NOREF(pInterface, pStream);
129
130 /* Just adopt the wanted stream configuration. */
131 PDMAUDIOPCMPROPS Props;
132 int rc = DrvAudioHlpStreamCfgToProps(pCfgReq, &Props);
133 if (RT_SUCCESS(rc))
134 {
135 if (pCfgAcq)
136 pCfgAcq->cSampleBufferSize = _1K;
137 }
138
139 LogFlowFuncLeaveRC(rc);
140 return rc;
141}
142
143
144static int debugCreateStreamOut(PPDMIHOSTAUDIO pInterface,
145 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
146{
147 NOREF(pInterface);
148
149 PDEBUGAUDIOSTREAM pDbgStream = (PDEBUGAUDIOSTREAM)pStream;
150
151 /* Just adopt the wanted stream configuration. */
152 PDMAUDIOPCMPROPS Props;
153 int rc = DrvAudioHlpStreamCfgToProps(pCfgReq, &Props);
154 if (RT_SUCCESS(rc))
155 {
156 pDbgStream->Out.tsLastPlayed = 0;
157 pDbgStream->Out.cMaxSamplesInPlayBuffer = _1K;
158 pDbgStream->Out.pu8PlayBuffer = (uint8_t *)RTMemAlloc(pDbgStream->Out.cMaxSamplesInPlayBuffer << Props.cShift);
159 if (!pDbgStream->Out.pu8PlayBuffer)
160 rc = VERR_NO_MEMORY;
161 }
162
163 if (RT_SUCCESS(rc))
164 {
165 char szTemp[RTPATH_MAX];
166 rc = RTPathTemp(szTemp, sizeof(szTemp));
167 if (RT_SUCCESS(rc))
168 {
169 char szFile[RTPATH_MAX];
170 rc = DrvAudioHlpGetFileName(szFile, RT_ELEMENTS(szFile), szTemp, NULL, PDMAUDIOFILETYPE_WAV);
171 if (RT_SUCCESS(rc))
172 {
173 LogFlowFunc(("%s\n", szFile));
174 rc = DrvAudioHlpWAVFileOpen(&pDbgStream->File, szFile,
175 RTFILE_O_WRITE | RTFILE_O_DENY_WRITE | RTFILE_O_CREATE_REPLACE,
176 &Props, PDMAUDIOFILEFLAG_NONE);
177 if (RT_FAILURE(rc))
178 LogRel(("DebugAudio: Creating output file '%s' failed with %Rrc\n", szFile, rc));
179 }
180 else
181 LogRel(("DebugAudio: Unable to build file name for temp dir '%s': %Rrc\n", szTemp, rc));
182 }
183 else
184 LogRel(("DebugAudio: Unable to retrieve temp dir: %Rrc\n", rc));
185 }
186
187 if (RT_SUCCESS(rc))
188 {
189 if (pCfgAcq)
190 pCfgAcq->cSampleBufferSize = pDbgStream->Out.cMaxSamplesInPlayBuffer;
191 }
192
193 LogFlowFuncLeaveRC(rc);
194 return rc;
195}
196
197
198/**
199 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamCreate}
200 */
201static DECLCALLBACK(int) drvHostDebugAudioStreamCreate(PPDMIHOSTAUDIO pInterface,
202 PPDMAUDIOSTREAM pStream,
203 PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
204{
205 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
206 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
207 AssertPtrReturn(pCfgReq, VERR_INVALID_POINTER);
208
209 int rc;
210 if (pCfgReq->enmDir == PDMAUDIODIR_IN)
211 rc = debugCreateStreamIn( pInterface, pStream, pCfgReq, pCfgAcq);
212 else
213 rc = debugCreateStreamOut(pInterface, pStream, pCfgReq, pCfgAcq);
214
215 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
216 return rc;
217}
218
219
220/**
221 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamPlay}
222 */
223static DECLCALLBACK(int) drvHostDebugAudioStreamPlay(PPDMIHOSTAUDIO pInterface,
224 PPDMAUDIOSTREAM pStream, const void *pvBuf, uint32_t cbBuf,
225 uint32_t *pcbWritten)
226{
227 RT_NOREF(pvBuf, cbBuf);
228
229 PDRVHOSTDEBUGAUDIO pDrv = RT_FROM_MEMBER(pInterface, DRVHOSTDEBUGAUDIO, IHostAudio);
230 PDEBUGAUDIOSTREAM pDbgStream = (PDEBUGAUDIOSTREAM)pStream;
231
232 /* Consume as many samples as would be played at the current frequency since last call. */
233 /*uint32_t cLive = AudioMixBufLive(&pStream->MixBuf);*/
234
235 uint64_t u64TicksNow = PDMDrvHlpTMGetVirtualTime(pDrv->pDrvIns);
236 // uint64_t u64TicksElapsed = u64TicksNow - pDbgStream->Out.tsLastPlayed;
237 // uint64_t u64TicksFreq = PDMDrvHlpTMGetVirtualFreq(pDrv->pDrvIns);
238
239 /*
240 * Minimize the rounding error by adding 0.5: samples = int((u64TicksElapsed * samplesFreq) / u64TicksFreq + 0.5).
241 * If rounding is not taken into account then the playback rate will be consistently lower that expected.
242 */
243 // uint64_t cSamplesPlayed = (2 * u64TicksElapsed * pStream->Props.uHz + u64TicksFreq) / u64TicksFreq / 2;
244
245 /* Don't play more than available. */
246 /*if (cSamplesPlayed > cLive)
247 cSamplesPlayed = cLive;*/
248
249 uint32_t cSamplesPlayed = 0;
250 uint32_t cSamplesAvail = RT_MIN(AudioMixBufUsed(&pStream->MixBuf), pDbgStream->Out.cMaxSamplesInPlayBuffer);
251 while (cSamplesAvail)
252 {
253 uint32_t cSamplesRead = 0;
254 int rc2 = AudioMixBufReadCirc(&pStream->MixBuf, pDbgStream->Out.pu8PlayBuffer,
255 AUDIOMIXBUF_S2B(&pStream->MixBuf, cSamplesAvail), &cSamplesRead);
256
257 if (RT_FAILURE(rc2))
258 LogRel(("DebugAudio: Reading output failed with %Rrc\n", rc2));
259
260 if (!cSamplesRead)
261 break;
262#if 0
263 RTFILE fh;
264 RTFileOpen(&fh, "/tmp/AudioDebug-Output.pcm",
265 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
266 RTFileWrite(fh, pDbgStream->Out.pu8PlayBuffer, AUDIOMIXBUF_S2B(&pStream->MixBuf, cSamplesRead), NULL);
267 RTFileClose(fh);
268#endif
269 rc2 = DrvAudioHlpWAVFileWrite(&pDbgStream->File,
270 pDbgStream->Out.pu8PlayBuffer, AUDIOMIXBUF_S2B(&pStream->MixBuf, cSamplesRead),
271 0 /* fFlags */);
272 if (RT_FAILURE(rc2))
273 LogRel(("DebugAudio: Writing output failed with %Rrc\n", rc2));
274
275 AudioMixBufFinish(&pStream->MixBuf, cSamplesRead);
276
277 Assert(cSamplesAvail >= cSamplesRead);
278 cSamplesAvail -= cSamplesRead;
279
280 cSamplesPlayed += cSamplesRead;
281 }
282
283 /* Remember when samples were consumed. */
284 pDbgStream->Out.tsLastPlayed = u64TicksNow;
285
286 if (pcbWritten)
287 *pcbWritten = cSamplesPlayed;
288
289 return VINF_SUCCESS;
290}
291
292
293/**
294 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamCapture}
295 */
296static DECLCALLBACK(int) drvHostDebugAudioStreamCapture(PPDMIHOSTAUDIO pInterface,
297 PPDMAUDIOSTREAM pStream, void *pvBuf, uint32_t cbBuf, uint32_t *pcbRead)
298{
299 RT_NOREF(pInterface, pStream, pvBuf, cbBuf);
300
301 /* Never capture anything. */
302 if (pcbRead)
303 *pcbRead = 0;
304
305 return VINF_SUCCESS;
306}
307
308
309static int debugDestroyStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
310{
311 RT_NOREF(pInterface, pStream);
312 LogFlowFuncLeaveRC(VINF_SUCCESS);
313 return VINF_SUCCESS;
314}
315
316
317static int debugDestroyStreamOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
318{
319 RT_NOREF(pInterface);
320 PDEBUGAUDIOSTREAM pDbgStream = (PDEBUGAUDIOSTREAM)pStream;
321 if ( pDbgStream
322 && pDbgStream->Out.pu8PlayBuffer)
323 {
324 RTMemFree(pDbgStream->Out.pu8PlayBuffer);
325 pDbgStream->Out.pu8PlayBuffer = NULL;
326 }
327
328 size_t cbDataSize = DrvAudioHlpWAVFileGetDataSize(&pDbgStream->File);
329
330 int rc = DrvAudioHlpWAVFileClose(&pDbgStream->File);
331 if (RT_SUCCESS(rc))
332 {
333 /* Delete the file again if nothing but the header was written to it. */
334 bool fDeleteEmptyFiles = true; /** @todo Make deletion configurable? */
335
336 if ( !cbDataSize
337 && fDeleteEmptyFiles)
338 {
339 rc = RTFileDelete(pDbgStream->File.szName);
340 }
341 else
342 LogRel(("DebugAudio: Created output file '%s' (%zu bytes)\n", pDbgStream->File.szName, cbDataSize));
343 }
344
345 LogFlowFuncLeaveRC(rc);
346 return rc;
347}
348
349
350static DECLCALLBACK(int) drvHostDebugAudioStreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
351{
352 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
353 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
354
355 int rc;
356 if (pStream->enmDir == PDMAUDIODIR_IN)
357 rc = debugDestroyStreamIn(pInterface, pStream);
358 else
359 rc = debugDestroyStreamOut(pInterface, pStream);
360
361 return rc;
362}
363
364static DECLCALLBACK(int) drvHostDebugAudioStreamControl(PPDMIHOSTAUDIO pInterface,
365 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
366{
367 RT_NOREF(enmStreamCmd);
368 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
369 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
370
371 Assert(pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
372
373 return VINF_SUCCESS;
374}
375
376static DECLCALLBACK(PDMAUDIOSTRMSTS) drvHostDebugAudioStreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
377{
378 NOREF(pInterface);
379 NOREF(pStream);
380
381 return ( PDMAUDIOSTRMSTS_FLAG_INITIALIZED | PDMAUDIOSTRMSTS_FLAG_ENABLED
382 | PDMAUDIOSTRMSTS_FLAG_DATA_READABLE | PDMAUDIOSTRMSTS_FLAG_DATA_WRITABLE);
383}
384
385static DECLCALLBACK(int) drvHostDebugAudioStreamIterate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
386{
387 NOREF(pInterface);
388 NOREF(pStream);
389
390 return VINF_SUCCESS;
391}
392
393
394/**
395 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
396 */
397static DECLCALLBACK(void *) drvHostDebugAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
398{
399 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
400 PDRVHOSTDEBUGAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTDEBUGAUDIO);
401
402 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
403 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
404 return NULL;
405}
406
407
408/**
409 * Constructs a Null audio driver instance.
410 *
411 * @copydoc FNPDMDRVCONSTRUCT
412 */
413static DECLCALLBACK(int) drvHostDebugAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
414{
415 RT_NOREF(pCfg, fFlags);
416 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
417 PDRVHOSTDEBUGAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTDEBUGAUDIO);
418 LogRel(("Audio: Initializing DEBUG driver\n"));
419
420 /*
421 * Init the static parts.
422 */
423 pThis->pDrvIns = pDrvIns;
424 /* IBase */
425 pDrvIns->IBase.pfnQueryInterface = drvHostDebugAudioQueryInterface;
426 /* IHostAudio */
427 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvHostDebugAudio);
428
429 return VINF_SUCCESS;
430}
431
432/**
433 * Char driver registration record.
434 */
435const PDMDRVREG g_DrvHostDebugAudio =
436{
437 /* u32Version */
438 PDM_DRVREG_VERSION,
439 /* szName */
440 "DebugAudio",
441 /* szRCMod */
442 "",
443 /* szR0Mod */
444 "",
445 /* pszDescription */
446 "Debug audio host driver",
447 /* fFlags */
448 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
449 /* fClass. */
450 PDM_DRVREG_CLASS_AUDIO,
451 /* cMaxInstances */
452 ~0U,
453 /* cbInstance */
454 sizeof(DRVHOSTDEBUGAUDIO),
455 /* pfnConstruct */
456 drvHostDebugAudioConstruct,
457 /* pfnDestruct */
458 NULL,
459 /* pfnRelocate */
460 NULL,
461 /* pfnIOCtl */
462 NULL,
463 /* pfnPowerOn */
464 NULL,
465 /* pfnReset */
466 NULL,
467 /* pfnSuspend */
468 NULL,
469 /* pfnResume */
470 NULL,
471 /* pfnAttach */
472 NULL,
473 /* pfnDetach */
474 NULL,
475 /* pfnPowerOff */
476 NULL,
477 /* pfnSoftReset */
478 NULL,
479 /* u32EndVersion */
480 PDM_DRVREG_VERSION
481};
482
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