VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostNullAudio.cpp@ 61648

Last change on this file since 61648 was 61609, checked in by vboxsync, 9 years ago

Audio: Update.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 11.6 KB
Line 
1/* $Id: DrvHostNullAudio.cpp 61609 2016-06-09 10:22:39Z 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#include <iprt/alloc.h>
44#include <iprt/uuid.h> /* For PDMIBASE_2_PDMDRV. */
45
46#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
47#include <VBox/log.h>
48#include <VBox/vmm/pdmaudioifs.h>
49
50#include "DrvAudio.h"
51#include "AudioMixBuffer.h"
52#include "VBoxDD.h"
53
54
55typedef struct NULLAUDIOSTREAMOUT
56{
57 /** Note: Always must come first! */
58 PDMAUDIOSTREAM Stream;
59 uint64_t u64TicksLast;
60 uint64_t csPlayBuffer;
61 uint8_t *pu8PlayBuffer;
62} NULLAUDIOSTREAMOUT, *PNULLAUDIOSTREAMOUT;
63
64typedef struct NULLAUDIOSTREAMIN
65{
66 /** Note: Always must come first! */
67 PDMAUDIOSTREAM Stream;
68} NULLAUDIOSTREAMIN, *PNULLAUDIOSTREAMIN;
69
70/**
71 * NULL audio driver instance data.
72 * @implements PDMIAUDIOCONNECTOR
73 */
74typedef struct DRVHOSTNULLAUDIO
75{
76 /** Pointer to the driver instance structure. */
77 PPDMDRVINS pDrvIns;
78 /** Pointer to host audio interface. */
79 PDMIHOSTAUDIO IHostAudio;
80} DRVHOSTNULLAUDIO, *PDRVHOSTNULLAUDIO;
81
82/*******************************************PDM_AUDIO_DRIVER******************************/
83
84
85static DECLCALLBACK(int) drvHostNullAudioGetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pCfg)
86{
87 NOREF(pInterface);
88 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
89
90 pCfg->cbStreamOut = sizeof(NULLAUDIOSTREAMOUT);
91 pCfg->cbStreamIn = sizeof(NULLAUDIOSTREAMIN);
92
93 /* The NULL backend has exactly one input source and one output sink. */
94 pCfg->cSources = 1;
95 pCfg->cSinks = 1;
96
97 pCfg->cMaxStreamsOut = 1; /* Output */
98 pCfg->cMaxStreamsIn = 2; /* Line input + microphone input. */
99
100 return VINF_SUCCESS;
101}
102
103static DECLCALLBACK(int) drvHostNullAudioInit(PPDMIHOSTAUDIO pInterface)
104{
105 NOREF(pInterface);
106
107 LogFlowFuncLeaveRC(VINF_SUCCESS);
108 return VINF_SUCCESS;
109}
110
111static int nullCreateStreamIn(PPDMIHOSTAUDIO pInterface,
112 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
113{
114 NOREF(pInterface);
115
116 /* Just adopt the wanted stream configuration. */
117 int rc = DrvAudioHlpStreamCfgToProps(pCfg, &pStream->Props);
118 if (RT_SUCCESS(rc))
119 {
120 if (pcSamples)
121 *pcSamples = _1K;
122 }
123
124 LogFlowFuncLeaveRC(rc);
125 return rc;
126}
127
128static int nullCreateStreamOut(PPDMIHOSTAUDIO pInterface,
129 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg,
130 uint32_t *pcSamples)
131{
132 NOREF(pInterface);
133
134 /* Just adopt the wanted stream configuration. */
135 int rc = DrvAudioHlpStreamCfgToProps(pCfg, &pStream->Props);
136 if (RT_SUCCESS(rc))
137 {
138 PNULLAUDIOSTREAMOUT pNullStream = (PNULLAUDIOSTREAMOUT)pStream;
139 pNullStream->u64TicksLast = 0;
140 pNullStream->csPlayBuffer = _1K;
141 pNullStream->pu8PlayBuffer = (uint8_t *)RTMemAlloc(_1K << pStream->Props.cShift);
142 if (pNullStream->pu8PlayBuffer)
143 {
144 if (pcSamples)
145 *pcSamples = pNullStream->csPlayBuffer;
146 }
147 else
148 rc = VERR_NO_MEMORY;
149 }
150
151 LogFlowFuncLeaveRC(rc);
152 return rc;
153}
154
155static DECLCALLBACK(int) drvHostNullAudioStreamPlay(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
156 uint32_t *pcSamplesPlayed)
157{
158 PDRVHOSTNULLAUDIO pDrv = RT_FROM_MEMBER(pInterface, DRVHOSTNULLAUDIO, IHostAudio);
159 PNULLAUDIOSTREAMOUT pNullStream = (PNULLAUDIOSTREAMOUT)pStream;
160
161 /* Consume as many samples as would be played at the current frequency since last call. */
162 uint32_t cLive = AudioMixBufLive(&pStream->MixBuf);
163
164 uint64_t u64TicksNow = PDMDrvHlpTMGetVirtualTime(pDrv->pDrvIns);
165 uint64_t u64TicksElapsed = u64TicksNow - pNullStream->u64TicksLast;
166 uint64_t u64TicksFreq = PDMDrvHlpTMGetVirtualFreq(pDrv->pDrvIns);
167
168 /* Remember when samples were consumed. */
169 pNullStream->u64TicksLast = u64TicksNow;
170
171 /*
172 * Minimize the rounding error by adding 0.5: samples = int((u64TicksElapsed * samplesFreq) / u64TicksFreq + 0.5).
173 * If rounding is not taken into account then the playback rate will be consistently lower that expected.
174 */
175 uint64_t cSamplesPlayed = (2 * u64TicksElapsed * pStream->Props.uHz + u64TicksFreq) / u64TicksFreq / 2;
176
177 /* Don't play more than available. */
178 if (cSamplesPlayed > cLive)
179 cSamplesPlayed = cLive;
180
181 cSamplesPlayed = RT_MIN(cSamplesPlayed, pNullStream->csPlayBuffer);
182
183 uint32_t csRead = 0;
184 AudioMixBufReadCirc(&pStream->MixBuf, pNullStream->pu8PlayBuffer,
185 AUDIOMIXBUF_S2B(&pStream->MixBuf, cSamplesPlayed), &csRead);
186 AudioMixBufFinish(&pStream->MixBuf, csRead);
187
188 if (pcSamplesPlayed)
189 *pcSamplesPlayed = csRead;
190
191 return VINF_SUCCESS;
192}
193
194static DECLCALLBACK(int) drvHostNullAudioStreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
195 uint32_t *pcSamplesCaptured)
196{
197 /* Never capture anything. */
198 if (pcSamplesCaptured)
199 *pcSamplesCaptured = 0;
200
201 return VINF_SUCCESS;
202}
203
204static int nullDestroyStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
205{
206 LogFlowFuncLeaveRC(VINF_SUCCESS);
207 return VINF_SUCCESS;
208}
209
210static int nullDestroyStreamOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
211{
212 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
213
214 PNULLAUDIOSTREAMOUT pNullStream = (PNULLAUDIOSTREAMOUT)pStream;
215 if ( pNullStream
216 && pNullStream->pu8PlayBuffer)
217 {
218 RTMemFree(pNullStream->pu8PlayBuffer);
219 pNullStream->pu8PlayBuffer = NULL;
220 }
221
222 LogFlowFuncLeaveRC(VINF_SUCCESS);
223 return VINF_SUCCESS;
224}
225
226static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvHostNullAudioGetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
227{
228 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
229
230 return PDMAUDIOBACKENDSTS_RUNNING;
231}
232
233static DECLCALLBACK(int) drvHostNullAudioStreamCreate(PPDMIHOSTAUDIO pInterface,
234 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
235{
236 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
237 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
238 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
239
240 int rc;
241 if (pCfg->enmDir == PDMAUDIODIR_IN)
242 rc = nullCreateStreamIn(pInterface, pStream, pCfg, pcSamples);
243 else
244 rc = nullCreateStreamOut(pInterface, pStream, pCfg, pcSamples);
245
246 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
247 return rc;
248}
249
250static DECLCALLBACK(int) drvHostNullAudioStreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
251{
252 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
253 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
254
255 int rc;
256 if (pStream->enmDir == PDMAUDIODIR_IN)
257 rc = nullDestroyStreamIn(pInterface, pStream);
258 else
259 rc = nullDestroyStreamOut(pInterface, pStream);
260
261 return rc;
262}
263
264static DECLCALLBACK(int) drvHostNullAudioStreamControl(PPDMIHOSTAUDIO pInterface,
265 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
266{
267 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
268 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
269
270 Assert(pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
271
272 return VINF_SUCCESS;
273}
274
275static DECLCALLBACK(PDMAUDIOSTRMSTS) drvHostNullAudioStreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
276{
277 NOREF(pInterface);
278 NOREF(pStream);
279
280 return (PDMAUDIOSTRMSTS_FLAG_INITIALIZED | PDMAUDIOSTRMSTS_FLAG_ENABLED);
281}
282
283static DECLCALLBACK(int) drvHostNullAudioStreamIterate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
284{
285 NOREF(pInterface);
286 NOREF(pStream);
287
288 return VINF_SUCCESS;
289}
290
291/**
292 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
293 */
294static DECLCALLBACK(void *) drvHostNullAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
295{
296 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
297 PDRVHOSTNULLAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTNULLAUDIO);
298
299 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
300 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
301 return NULL;
302}
303
304static DECLCALLBACK(void) drvHostNullAudioShutdown(PPDMIHOSTAUDIO pInterface)
305{
306 NOREF(pInterface);
307}
308
309/**
310 * Constructs a Null audio driver instance.
311 *
312 * @copydoc FNPDMDRVCONSTRUCT
313 */
314static DECLCALLBACK(int) drvHostNullAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
315{
316 AssertPtrReturn(pDrvIns, VERR_INVALID_POINTER);
317 /* pCfg is optional. */
318
319 PDRVHOSTNULLAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTNULLAUDIO);
320 LogRel(("Audio: Initializing NULL driver\n"));
321
322 /*
323 * Init the static parts.
324 */
325 pThis->pDrvIns = pDrvIns;
326 /* IBase */
327 pDrvIns->IBase.pfnQueryInterface = drvHostNullAudioQueryInterface;
328 /* IHostAudio */
329 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvHostNullAudio);
330
331 return VINF_SUCCESS;
332}
333
334/**
335 * Char driver registration record.
336 */
337const PDMDRVREG g_DrvHostNullAudio =
338{
339 /* u32Version */
340 PDM_DRVREG_VERSION,
341 /* szName */
342 "NullAudio",
343 /* szRCMod */
344 "",
345 /* szR0Mod */
346 "",
347 /* pszDescription */
348 "NULL audio host driver",
349 /* fFlags */
350 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
351 /* fClass. */
352 PDM_DRVREG_CLASS_AUDIO,
353 /* cMaxInstances */
354 ~0U,
355 /* cbInstance */
356 sizeof(DRVHOSTNULLAUDIO),
357 /* pfnConstruct */
358 drvHostNullAudioConstruct,
359 /* pfnDestruct */
360 NULL,
361 /* pfnRelocate */
362 NULL,
363 /* pfnIOCtl */
364 NULL,
365 /* pfnPowerOn */
366 NULL,
367 /* pfnReset */
368 NULL,
369 /* pfnSuspend */
370 NULL,
371 /* pfnResume */
372 NULL,
373 /* pfnAttach */
374 NULL,
375 /* pfnDetach */
376 NULL,
377 /* pfnPowerOff */
378 NULL,
379 /* pfnSoftReset */
380 NULL,
381 /* u32EndVersion */
382 PDM_DRVREG_VERSION
383};
384
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