VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostNullAudio.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: 13.2 KB
Line 
1/* $Id: DrvHostNullAudio.cpp 63711 2016-09-05 12:04:01Z vboxsync $ */
2/** @file
3 * NULL audio driver -- also acts as a fallback if no
4 * other backend is available.
5 */
6
7/*
8 * Copyright (C) 2006-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 * This code is based on: noaudio.c QEMU based code.
20 *
21 * QEMU Timer based audio emulation
22 *
23 * Copyright (c) 2004-2005 Vassili Karpov (malc)
24 *
25 * Permission is hereby granted, free of charge, to any person obtaining a copy
26 * of this software and associated documentation files (the "Software"), to deal
27 * in the Software without restriction, including without limitation the rights
28 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
29 * copies of the Software, and to permit persons to whom the Software is
30 * furnished to do so, subject to the following conditions:
31 *
32 * The above copyright notice and this permission notice shall be included in
33 * all copies or substantial portions of the Software.
34 *
35 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
36 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
37 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
38 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
39 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
40 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
41 * THE SOFTWARE.
42 */
43
44
45/*********************************************************************************************************************************
46* Header Files *
47*********************************************************************************************************************************/
48#include <iprt/mem.h>
49#include <iprt/uuid.h> /* For PDMIBASE_2_PDMDRV. */
50
51#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
52#include <VBox/log.h>
53#include <VBox/vmm/pdmaudioifs.h>
54
55#include "DrvAudio.h"
56#include "AudioMixBuffer.h"
57#include "VBoxDD.h"
58
59
60/*********************************************************************************************************************************
61* Structures and Typedefs *
62*********************************************************************************************************************************/
63typedef struct NULLAUDIOSTREAMOUT
64{
65 /** @note Always must come first! */
66 PDMAUDIOSTREAM Stream;
67 /** The PCM properties of this stream. */
68 PDMAUDIOPCMPROPS Props;
69 uint64_t u64TicksLast;
70 uint64_t cMaxSamplesInPlayBuffer;
71 uint8_t *pbPlayBuffer;
72} NULLAUDIOSTREAMOUT;
73typedef NULLAUDIOSTREAMOUT *PNULLAUDIOSTREAMOUT;
74
75typedef struct NULLAUDIOSTREAMIN
76{
77 /** @note Always must come first! */
78 PDMAUDIOSTREAM Stream;
79 /** The PCM properties of this stream. */
80 PDMAUDIOPCMPROPS Props;
81} NULLAUDIOSTREAMIN;
82typedef NULLAUDIOSTREAMIN *PNULLAUDIOSTREAMIN;
83
84/**
85 * NULL audio driver instance data.
86 * @implements PDMIAUDIOCONNECTOR
87 */
88typedef struct DRVHOSTNULLAUDIO
89{
90 /** Pointer to the driver instance structure. */
91 PPDMDRVINS pDrvIns;
92 /** Pointer to host audio interface. */
93 PDMIHOSTAUDIO IHostAudio;
94} DRVHOSTNULLAUDIO, *PDRVHOSTNULLAUDIO;
95
96
97
98/**
99 * @interface_method_impl{PDMIHOSTAUDIO,pfnGetConfig}
100 */
101static DECLCALLBACK(int) drvHostNullAudioGetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pBackendCfg)
102{
103 NOREF(pInterface);
104 AssertPtrReturn(pBackendCfg, VERR_INVALID_POINTER);
105
106 pBackendCfg->cbStreamOut = sizeof(NULLAUDIOSTREAMOUT);
107 pBackendCfg->cbStreamIn = sizeof(NULLAUDIOSTREAMIN);
108
109 pBackendCfg->cMaxStreamsOut = 1; /* Output */
110 pBackendCfg->cMaxStreamsIn = 2; /* Line input + microphone input. */
111
112 return VINF_SUCCESS;
113}
114
115
116/**
117 * @interface_method_impl{PDMIHOSTAUDIO,pfnInit}
118 */
119static DECLCALLBACK(int) drvHostNullAudioInit(PPDMIHOSTAUDIO pInterface)
120{
121 NOREF(pInterface);
122
123 LogFlowFuncLeaveRC(VINF_SUCCESS);
124 return VINF_SUCCESS;
125}
126
127
128/**
129 * @interface_method_impl{PDMIHOSTAUDIO,pfnShutdown}
130 */
131static DECLCALLBACK(void) drvHostNullAudioShutdown(PPDMIHOSTAUDIO pInterface)
132{
133 RT_NOREF(pInterface);
134}
135
136
137/**
138 * @interface_method_impl{PDMIHOSTAUDIO,pfnGetStatus}
139 */
140static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvHostNullAudioGetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
141{
142 RT_NOREF(enmDir);
143 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
144
145 return PDMAUDIOBACKENDSTS_RUNNING;
146}
147
148
149/**
150 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamPlay}
151 */
152static DECLCALLBACK(int) drvHostNullAudioStreamPlay(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream, const void *pvBuf, uint32_t cbBuf, uint32_t *pcbWritten)
153{
154 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
155 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
156
157 RT_NOREF2(pvBuf, cbBuf);
158
159 PDRVHOSTNULLAUDIO pDrv = RT_FROM_MEMBER(pInterface, DRVHOSTNULLAUDIO, IHostAudio);
160 PNULLAUDIOSTREAMOUT pNullStream = RT_FROM_MEMBER(pStream, NULLAUDIOSTREAMOUT, Stream);
161
162 /* Consume as many samples as would be played at the current frequency since last call. */
163 uint32_t cLive = AudioMixBufLive(&pStream->MixBuf);
164
165 uint64_t u64TicksNow = PDMDrvHlpTMGetVirtualTime(pDrv->pDrvIns);
166 uint64_t u64TicksElapsed = u64TicksNow - pNullStream->u64TicksLast;
167 uint64_t u64TicksFreq = PDMDrvHlpTMGetVirtualFreq(pDrv->pDrvIns);
168
169 /* Remember when samples were consumed. */
170 pNullStream->u64TicksLast = u64TicksNow;
171
172 /*
173 * Minimize the rounding error by adding 0.5: samples = int((u64TicksElapsed * samplesFreq) / u64TicksFreq + 0.5).
174 * If rounding is not taken into account then the playback rate will be consistently lower that expected.
175 */
176 uint64_t cSamplesPlayed = (2 * u64TicksElapsed * pNullStream->Props.uHz + u64TicksFreq) / u64TicksFreq / 2;
177
178 /* Don't play more than available. */
179 if (cSamplesPlayed > cLive)
180 cSamplesPlayed = cLive;
181
182 cSamplesPlayed = RT_MIN(cSamplesPlayed, pNullStream->cMaxSamplesInPlayBuffer);
183
184 uint32_t cSamplesToRead = 0;
185 AudioMixBufReadCirc(&pStream->MixBuf, pNullStream->pbPlayBuffer,
186 AUDIOMIXBUF_S2B(&pStream->MixBuf, cSamplesPlayed), &cSamplesToRead);
187 AudioMixBufFinish(&pStream->MixBuf, cSamplesToRead);
188
189 if (pcbWritten)
190 *pcbWritten = cSamplesToRead;
191
192 return VINF_SUCCESS;
193}
194
195
196/**
197 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamCapture}
198 */
199static DECLCALLBACK(int) drvHostNullAudioStreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream, void *pvBuf, uint32_t cbBuf, uint32_t *pcbRead)
200{
201 RT_NOREF4(pInterface, pStream, pvBuf, cbBuf);
202
203 /* Never capture anything. */
204 if (pcbRead)
205 *pcbRead = 0;
206
207 return VINF_SUCCESS;
208}
209
210
211static int nullCreateStreamIn(PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
212{
213 PNULLAUDIOSTREAMIN pNullStream = RT_FROM_MEMBER(pStream, NULLAUDIOSTREAMIN, Stream);
214
215 /* Just adopt the wanted stream configuration. */
216 int rc = DrvAudioHlpStreamCfgToProps(pCfgReq, &pNullStream->Props);
217 if (RT_SUCCESS(rc))
218 {
219 if (pCfgAcq)
220 pCfgAcq->cSampleBufferSize = _1K;
221 }
222
223 LogFlowFuncLeaveRC(rc);
224 return rc;
225}
226
227
228static int nullCreateStreamOut(PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
229{
230 PNULLAUDIOSTREAMOUT pNullStream = RT_FROM_MEMBER(pStream, NULLAUDIOSTREAMOUT, Stream);
231
232 /* Just adopt the wanted stream configuration. */
233 int rc = DrvAudioHlpStreamCfgToProps(pCfgReq, &pNullStream->Props);
234 if (RT_SUCCESS(rc))
235 {
236 pNullStream->u64TicksLast = 0;
237 pNullStream->cMaxSamplesInPlayBuffer = _1K;
238
239 pNullStream->pbPlayBuffer = (uint8_t *)RTMemAlloc(_1K << pNullStream->Props.cShift);
240 if (pNullStream->pbPlayBuffer)
241 {
242 if (pCfgAcq)
243 pCfgAcq->cSampleBufferSize = pNullStream->cMaxSamplesInPlayBuffer;
244 }
245 else
246 rc = VERR_NO_MEMORY;
247 }
248
249 LogFlowFuncLeaveRC(rc);
250 return rc;
251}
252
253
254/**
255 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamCreate}
256 */
257static DECLCALLBACK(int) drvHostNullAudioStreamCreate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
258{
259 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
260 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
261 AssertPtrReturn(pCfgReq, VERR_INVALID_POINTER);
262
263 int rc;
264 if (pCfgReq->enmDir == PDMAUDIODIR_IN)
265 rc = nullCreateStreamIn( pStream, pCfgReq, pCfgAcq);
266 else
267 rc = nullCreateStreamOut(pStream, pCfgReq, pCfgAcq);
268
269 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
270 return rc;
271}
272
273
274static int nullDestroyStreamIn(void)
275{
276 LogFlowFuncLeaveRC(VINF_SUCCESS);
277 return VINF_SUCCESS;
278}
279
280
281static int nullDestroyStreamOut(PPDMAUDIOSTREAM pStream)
282{
283 PNULLAUDIOSTREAMOUT pNullStream = RT_FROM_MEMBER(pStream, NULLAUDIOSTREAMOUT, Stream);
284 if ( pNullStream
285 && pNullStream->pbPlayBuffer)
286 {
287 RTMemFree(pNullStream->pbPlayBuffer);
288 pNullStream->pbPlayBuffer = NULL;
289 }
290
291 LogFlowFuncLeaveRC(VINF_SUCCESS);
292 return VINF_SUCCESS;
293}
294
295
296/**
297 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamDestroy}
298 */
299static DECLCALLBACK(int) drvHostNullAudioStreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
300{
301 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
302 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
303
304 int rc;
305 if (pStream->enmDir == PDMAUDIODIR_IN)
306 rc = nullDestroyStreamIn();
307 else
308 rc = nullDestroyStreamOut(pStream);
309
310 return rc;
311}
312
313
314/**
315 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamControl}
316 */
317static DECLCALLBACK(int) drvHostNullAudioStreamControl(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
318{
319 RT_NOREF(enmStreamCmd);
320 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
321 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
322
323 Assert(pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
324
325 return VINF_SUCCESS;
326}
327
328
329/**
330 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetStatus}
331 */
332static DECLCALLBACK(PDMAUDIOSTRMSTS) drvHostNullAudioStreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
333{
334 RT_NOREF(pInterface, pStream);
335 return PDMAUDIOSTRMSTS_FLAG_INITIALIZED | PDMAUDIOSTRMSTS_FLAG_ENABLED
336 | PDMAUDIOSTRMSTS_FLAG_DATA_READABLE | PDMAUDIOSTRMSTS_FLAG_DATA_WRITABLE;
337}
338
339
340/**
341 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamIterate}
342 */
343static DECLCALLBACK(int) drvHostNullAudioStreamIterate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
344{
345 NOREF(pInterface);
346 NOREF(pStream);
347
348 return VINF_SUCCESS;
349}
350
351
352/**
353 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
354 */
355static DECLCALLBACK(void *) drvHostNullAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
356{
357 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
358 PDRVHOSTNULLAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTNULLAUDIO);
359
360 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
361 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
362 return NULL;
363}
364
365
366/**
367 * Constructs a Null audio driver instance.
368 *
369 * @copydoc FNPDMDRVCONSTRUCT
370 */
371static DECLCALLBACK(int) drvHostNullAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
372{
373 RT_NOREF(pCfg, fFlags);
374 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
375 AssertPtrReturn(pDrvIns, VERR_INVALID_POINTER);
376 /* pCfg is optional. */
377
378 PDRVHOSTNULLAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTNULLAUDIO);
379 LogRel(("Audio: Initializing NULL driver\n"));
380
381 /*
382 * Init the static parts.
383 */
384 pThis->pDrvIns = pDrvIns;
385 /* IBase */
386 pDrvIns->IBase.pfnQueryInterface = drvHostNullAudioQueryInterface;
387 /* IHostAudio */
388 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvHostNullAudio);
389
390 return VINF_SUCCESS;
391}
392
393
394/**
395 * Char driver registration record.
396 */
397const PDMDRVREG g_DrvHostNullAudio =
398{
399 /* u32Version */
400 PDM_DRVREG_VERSION,
401 /* szName */
402 "NullAudio",
403 /* szRCMod */
404 "",
405 /* szR0Mod */
406 "",
407 /* pszDescription */
408 "NULL audio host driver",
409 /* fFlags */
410 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
411 /* fClass. */
412 PDM_DRVREG_CLASS_AUDIO,
413 /* cMaxInstances */
414 ~0U,
415 /* cbInstance */
416 sizeof(DRVHOSTNULLAUDIO),
417 /* pfnConstruct */
418 drvHostNullAudioConstruct,
419 /* pfnDestruct */
420 NULL,
421 /* pfnRelocate */
422 NULL,
423 /* pfnIOCtl */
424 NULL,
425 /* pfnPowerOn */
426 NULL,
427 /* pfnReset */
428 NULL,
429 /* pfnSuspend */
430 NULL,
431 /* pfnResume */
432 NULL,
433 /* pfnAttach */
434 NULL,
435 /* pfnDetach */
436 NULL,
437 /* pfnPowerOff */
438 NULL,
439 /* pfnSoftReset */
440 NULL,
441 /* u32EndVersion */
442 PDM_DRVREG_VERSION
443};
444
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