VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/DrvAudioVRDE.cpp@ 62497

Last change on this file since 62497 was 62379, checked in by vboxsync, 9 years ago

Main: 2nd try: fixes for a few -Wunused -Wconversion

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 21.1 KB
Line 
1/* $Id: DrvAudioVRDE.cpp 62379 2016-07-20 20:11:50Z vboxsync $ */
2/** @file
3 * VRDE audio backend for Main.
4 */
5
6/*
7 * Copyright (C) 2013-2016 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_VRDE_AUDIO
23#include <VBox/log.h>
24#include "DrvAudioVRDE.h"
25#include "ConsoleImpl.h"
26#include "ConsoleVRDPServer.h"
27
28#include "Logging.h"
29
30#include "../../Devices/Audio/DrvAudio.h"
31#include "../../Devices/Audio/AudioMixBuffer.h"
32
33#include <iprt/mem.h>
34#include <iprt/cdefs.h>
35#include <iprt/circbuf.h>
36
37#include <VBox/vmm/pdmaudioifs.h>
38#include <VBox/vmm/pdmdrv.h>
39#include <VBox/RemoteDesktop/VRDE.h>
40#include <VBox/vmm/cfgm.h>
41#include <VBox/err.h>
42
43
44/*********************************************************************************************************************************
45* Structures and Typedefs *
46*********************************************************************************************************************************/
47/**
48 * Audio VRDE driver instance data.
49 */
50typedef struct DRVAUDIOVRDE
51{
52 /** Pointer to audio VRDE object. */
53 AudioVRDE *pAudioVRDE;
54 PPDMDRVINS pDrvIns;
55 /** Pointer to the driver instance structure. */
56 PDMIHOSTAUDIO IHostAudio;
57 /** Pointer to the VRDP's console object. */
58 ConsoleVRDPServer *pConsoleVRDPServer;
59 /** Pointer to the DrvAudio port interface that is above us. */
60 PPDMIAUDIOCONNECTOR pDrvAudio;
61 /** Whether this driver is enabled or not. */
62 bool fEnabled;
63} DRVAUDIOVRDE, *PDRVAUDIOVRDE;
64
65typedef struct VRDESTREAMIN
66{
67 /** Associated host input stream.
68 * Note: Always must come first! */
69 PDMAUDIOSTREAM Stream;
70 /** Number of samples captured asynchronously in the
71 * onVRDEInputXXX callbacks. */
72 uint32_t cSamplesCaptured;
73 /** Critical section. */
74 RTCRITSECT CritSect;
75} VRDESTREAMIN, *PVRDESTREAMIN;
76
77typedef struct VRDESTREAMOUT
78{
79 /** Associated host output stream.
80 * Note: Always must come first! */
81 PDMAUDIOSTREAM Stream;
82 uint64_t old_ticks;
83 uint64_t cSamplesSentPerSec;
84} VRDESTREAMOUT, *PVRDESTREAMOUT;
85
86
87static int vrdeCreateStreamIn(PPDMIHOSTAUDIO pInterface,
88 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
89{
90 PDRVAUDIOVRDE pDrv = RT_FROM_MEMBER(pInterface, DRVAUDIOVRDE, IHostAudio);
91 AssertPtrReturn(pDrv, VERR_INVALID_POINTER);
92
93 PVRDESTREAMIN pVRDEStrmIn = (PVRDESTREAMIN)pStream;
94 AssertPtrReturn(pVRDEStrmIn, VERR_INVALID_POINTER);
95
96 if (pcSamples)
97 *pcSamples = _4K; /** @todo Make this configurable. */
98
99 return DrvAudioHlpStreamCfgToProps(pCfg, &pVRDEStrmIn->Stream.Props);
100}
101
102static int vrdeCreateStreamOut(PPDMIHOSTAUDIO pInterface,
103 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
104{
105 PDRVAUDIOVRDE pDrv = RT_FROM_MEMBER(pInterface, DRVAUDIOVRDE, IHostAudio);
106 AssertPtrReturn(pDrv, VERR_INVALID_POINTER);
107
108 LogFlowFunc(("pStream=%p, pCfg=%p\n", pStream, pCfg));
109
110 PVRDESTREAMOUT pVRDEStrmOut = (PVRDESTREAMOUT)pStream;
111 AssertPtrReturn(pVRDEStrmOut, VERR_INVALID_POINTER);
112
113 if (pcSamples)
114 *pcSamples = _4K; /** @todo Make this configurable. */
115
116 return DrvAudioHlpStreamCfgToProps(pCfg, &pVRDEStrmOut->Stream.Props);
117}
118
119static int vrdeControlStreamOut(PPDMIHOSTAUDIO pInterface,
120 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
121{
122 PDRVAUDIOVRDE pDrv = RT_FROM_MEMBER(pInterface, DRVAUDIOVRDE, IHostAudio);
123 AssertPtrReturn(pDrv, VERR_INVALID_POINTER);
124
125 PVRDESTREAMOUT pVRDEStrmOut = (PVRDESTREAMOUT)pStream;
126 AssertPtrReturn(pVRDEStrmOut, VERR_INVALID_POINTER);
127
128 LogFlowFunc(("enmStreamCmd=%ld\n", enmStreamCmd));
129
130 AudioMixBufReset(&pStream->MixBuf);
131
132 return VINF_SUCCESS;
133}
134
135static int vrdeControlStreamIn(PPDMIHOSTAUDIO pInterface,
136 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
137{
138 PDRVAUDIOVRDE pDrv = RT_FROM_MEMBER(pInterface, DRVAUDIOVRDE, IHostAudio);
139 AssertPtrReturn(pDrv, VERR_INVALID_POINTER);
140
141 PVRDESTREAMIN pVRDEStrmIn = (PVRDESTREAMIN)pStream;
142 AssertPtrReturn(pVRDEStrmIn, VERR_INVALID_POINTER);
143
144 LogFlowFunc(("enmStreamCmd=%ld\n", enmStreamCmd));
145
146 if (!pDrv->pConsoleVRDPServer)
147 return VINF_SUCCESS;
148
149 AudioMixBufReset(&pStream->MixBuf);
150
151 /* Initialize only if not already done. */
152 int rc;
153 if (enmStreamCmd == PDMAUDIOSTREAMCMD_ENABLE)
154 {
155 rc = pDrv->pConsoleVRDPServer->SendAudioInputBegin(NULL, pVRDEStrmIn, AudioMixBufSize(&pStream->MixBuf),
156 pStream->Props.uHz,
157 pStream->Props.cChannels, pStream->Props.cBits);
158 if (rc == VERR_NOT_SUPPORTED)
159 {
160 LogFlowFunc(("No RDP client connected, so no input recording supported\n"));
161 rc = VINF_SUCCESS;
162 }
163 }
164 else if (enmStreamCmd == PDMAUDIOSTREAMCMD_DISABLE)
165 {
166 pDrv->pConsoleVRDPServer->SendAudioInputEnd(NULL /* pvUserCtx */);
167 rc = VINF_SUCCESS;
168 }
169 else
170 rc = VERR_INVALID_PARAMETER;
171
172 return rc;
173}
174
175static DECLCALLBACK(int) drvAudioVRDEInit(PPDMIHOSTAUDIO pInterface)
176{
177 LogFlowFuncEnter();
178
179 return VINF_SUCCESS;
180}
181
182/**
183 * {FIXME - Missing brief description - FIXME}
184 *
185 * Transfers audio input formerly sent by a connected RDP client / VRDE backend
186 * (using the onVRDEInputXXX methods) over to the VRDE host (VM). The audio device
187 * emulation then will read and send the data to the guest.
188 *
189 * @return IPRT status code.
190 * @param pInterface
191 * @param pStream
192 * @param pcSamplesCaptured
193 */
194static DECLCALLBACK(int) drvAudioVRDEStreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
195 uint32_t *pcSamplesCaptured)
196{
197 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
198 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
199 AssertPtrReturn(pcSamplesCaptured, VERR_INVALID_POINTER);
200
201 PDRVAUDIOVRDE pDrv = RT_FROM_MEMBER(pInterface, DRVAUDIOVRDE, IHostAudio);
202 AssertPtrReturn(pDrv, VERR_INVALID_POINTER);
203
204 PVRDESTREAMIN pVRDEStrmIn = (PVRDESTREAMIN)pStream;
205 AssertPtrReturn(pVRDEStrmIn, VERR_INVALID_POINTER);
206
207 /** @todo Use CritSect! */
208
209 int rc;
210
211 uint32_t cProcessed = 0;
212 if (pVRDEStrmIn->cSamplesCaptured)
213 {
214 rc = AudioMixBufMixToParent(&pVRDEStrmIn->Stream.MixBuf, pVRDEStrmIn->cSamplesCaptured,
215 &cProcessed);
216 }
217 else
218 rc = VINF_SUCCESS;
219
220 if (RT_SUCCESS(rc))
221 {
222 *pcSamplesCaptured = cProcessed;
223
224 Assert(pVRDEStrmIn->cSamplesCaptured >= cProcessed);
225 pVRDEStrmIn->cSamplesCaptured -= cProcessed;
226 }
227
228 LogFlowFunc(("cSamplesCaptured=%RU32, cProcessed=%RU32 rc=%Rrc\n", pVRDEStrmIn->cSamplesCaptured, cProcessed, rc));
229 return rc;
230}
231
232/**
233 * Transfers VM audio output to remote client.
234 *
235 * Transfers VM audio output over to the VRDE instance for playing remotely
236 * on the client.
237 *
238 * @return IPRT status code.
239 * @param pInterface
240 * @param pStream
241 * @param pcSamplesPlayed
242 */
243static DECLCALLBACK(int) drvAudioVRDEStreamPlay(PPDMIHOSTAUDIO pInterface,
244 PPDMAUDIOSTREAM pStream, uint32_t *pcSamplesPlayed)
245{
246 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
247 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
248 /* pcSamplesPlayed is optional. */
249
250 PDRVAUDIOVRDE pDrv = RT_FROM_MEMBER(pInterface, DRVAUDIOVRDE, IHostAudio);
251 AssertPtrReturn(pDrv, VERR_INVALID_POINTER);
252
253 PVRDESTREAMOUT pVRDEStrmOut = (PVRDESTREAMOUT)pStream;
254 AssertPtrReturn(pVRDEStrmOut, VERR_INVALID_POINTER);
255
256 uint32_t cLive = AudioMixBufLive(&pStream->MixBuf);
257
258 uint64_t now = PDMDrvHlpTMGetVirtualTime(pDrv->pDrvIns);
259 uint64_t ticks = now - pVRDEStrmOut->old_ticks;
260 uint64_t ticks_per_second = PDMDrvHlpTMGetVirtualFreq(pDrv->pDrvIns);
261
262 /* Minimize the rounding error: samples = int((ticks * freq) / ticks_per_second + 0.5). */
263 uint32_t cSamplesPlayed = (int)((2 * ticks * pStream->Props.uHz + ticks_per_second) / ticks_per_second / 2);
264
265 /* Don't play more than available. */
266 if (cSamplesPlayed > cLive)
267 cSamplesPlayed = cLive;
268
269 /* Remember when samples were consumed. */
270 pVRDEStrmOut->old_ticks = now;
271
272 VRDEAUDIOFORMAT format = VRDE_AUDIO_FMT_MAKE(pStream->Props.uHz,
273 pStream->Props.cChannels,
274 pStream->Props.cBits,
275 pStream->Props.fSigned);
276
277 int cSamplesToSend = cSamplesPlayed;
278
279 LogFlowFunc(("uFreq=%RU32, cChan=%RU8, cBits=%RU8, fSigned=%RTbool, enmFormat=%ld, cSamplesToSend=%RU32\n",
280 pStream->Props.uHz, pStream->Props.cChannels,
281 pStream->Props.cBits, pStream->Props.fSigned,
282 format, cSamplesToSend));
283
284 /*
285 * Call the VRDP server with the data.
286 */
287 uint32_t cReadTotal = 0;
288
289 PPDMAUDIOSAMPLE pSamples;
290 uint32_t cRead;
291 int rc = AudioMixBufAcquire(&pStream->MixBuf, cSamplesToSend,
292 &pSamples, &cRead);
293 if ( RT_SUCCESS(rc)
294 && cRead)
295 {
296 cReadTotal = cRead;
297 pDrv->pConsoleVRDPServer->SendAudioSamples(pSamples, cRead, format);
298
299 if (rc == VINF_TRY_AGAIN)
300 {
301 rc = AudioMixBufAcquire(&pStream->MixBuf, cSamplesToSend - cRead,
302 &pSamples, &cRead);
303 if (RT_SUCCESS(rc))
304 pDrv->pConsoleVRDPServer->SendAudioSamples(pSamples, cRead, format);
305
306 cReadTotal += cRead;
307 }
308 }
309
310 AudioMixBufFinish(&pStream->MixBuf, cSamplesToSend);
311
312 /*
313 * Always report back all samples acquired, regardless of whether the
314 * VRDP server actually did process those.
315 */
316 if (pcSamplesPlayed)
317 *pcSamplesPlayed = cReadTotal;
318
319 LogFlowFunc(("cReadTotal=%RU32, rc=%Rrc\n", cReadTotal, rc));
320 return rc;
321}
322
323static int vrdeDestroyStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
324{
325 PDRVAUDIOVRDE pDrv = RT_FROM_MEMBER(pInterface, DRVAUDIOVRDE, IHostAudio);
326 AssertPtrReturn(pDrv, VERR_INVALID_POINTER);
327
328 if (pDrv->pConsoleVRDPServer)
329 pDrv->pConsoleVRDPServer->SendAudioInputEnd(NULL);
330
331 return VINF_SUCCESS;
332}
333
334static int vrdeDestroyStreamOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
335{
336 PDRVAUDIOVRDE pDrv = RT_FROM_MEMBER(pInterface, DRVAUDIOVRDE, IHostAudio);
337 AssertPtrReturn(pDrv, VERR_INVALID_POINTER);
338
339 return VINF_SUCCESS;
340}
341
342static DECLCALLBACK(int) drvAudioVRDEGetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pCfg)
343{
344 NOREF(pInterface);
345 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
346
347 pCfg->cbStreamOut = sizeof(VRDESTREAMOUT);
348 pCfg->cbStreamIn = sizeof(VRDESTREAMIN);
349 pCfg->cMaxStreamsIn = UINT32_MAX;
350 pCfg->cMaxStreamsOut = UINT32_MAX;
351 pCfg->cSources = 1;
352 pCfg->cSinks = 1;
353
354 return VINF_SUCCESS;
355}
356
357static DECLCALLBACK(void) drvAudioVRDEShutdown(PPDMIHOSTAUDIO pInterface)
358{
359 PDRVAUDIOVRDE pDrv = RT_FROM_MEMBER(pInterface, DRVAUDIOVRDE, IHostAudio);
360 AssertPtrReturnVoid(pDrv);
361
362 if (pDrv->pConsoleVRDPServer)
363 pDrv->pConsoleVRDPServer->SendAudioInputEnd(NULL);
364}
365
366static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvAudioVRDEGetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
367{
368 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
369
370 return PDMAUDIOBACKENDSTS_RUNNING;
371}
372
373static DECLCALLBACK(int) drvAudioVRDEStreamCreate(PPDMIHOSTAUDIO pInterface,
374 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
375{
376 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
377 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
378 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
379
380 int rc;
381 if (pCfg->enmDir == PDMAUDIODIR_IN)
382 rc = vrdeCreateStreamIn(pInterface, pStream, pCfg, pcSamples);
383 else
384 rc = vrdeCreateStreamOut(pInterface, pStream, pCfg, pcSamples);
385
386 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
387 return rc;
388}
389
390static DECLCALLBACK(int) drvAudioVRDEStreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
391{
392 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
393 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
394
395 int rc;
396 if (pStream->enmDir == PDMAUDIODIR_IN)
397 rc = vrdeDestroyStreamIn(pInterface, pStream);
398 else
399 rc = vrdeDestroyStreamOut(pInterface, pStream);
400
401 return rc;
402}
403
404static DECLCALLBACK(int) drvAudioVRDEStreamControl(PPDMIHOSTAUDIO pInterface,
405 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
406{
407 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
408 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
409
410 Assert(pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
411
412 int rc;
413 if (pStream->enmDir == PDMAUDIODIR_IN)
414 rc = vrdeControlStreamIn(pInterface, pStream, enmStreamCmd);
415 else
416 rc = vrdeControlStreamOut(pInterface, pStream, enmStreamCmd);
417
418 return rc;
419}
420
421static DECLCALLBACK(PDMAUDIOSTRMSTS) drvAudioVRDEStreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
422{
423 NOREF(pInterface);
424 NOREF(pStream);
425
426 return ( PDMAUDIOSTRMSTS_FLAG_INITIALIZED | PDMAUDIOSTRMSTS_FLAG_ENABLED
427 | PDMAUDIOSTRMSTS_FLAG_DATA_READABLE | PDMAUDIOSTRMSTS_FLAG_DATA_WRITABLE);
428}
429
430static DECLCALLBACK(int) drvAudioVRDEStreamIterate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
431{
432 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
433 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
434
435 LogFlowFuncEnter();
436
437 /* Nothing to do here for VRDE. */
438 return VINF_SUCCESS;
439}
440
441/**
442 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
443 */
444static DECLCALLBACK(void *) drvAudioVRDEQueryInterface(PPDMIBASE pInterface, const char *pszIID)
445{
446 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
447 PDRVAUDIOVRDE pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIOVRDE);
448
449 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
450 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
451 return NULL;
452}
453
454AudioVRDE::AudioVRDE(Console *pConsole)
455 : mpDrv(NULL),
456 mParent(pConsole)
457{
458}
459
460AudioVRDE::~AudioVRDE(void)
461{
462 if (mpDrv)
463 {
464 mpDrv->pAudioVRDE = NULL;
465 mpDrv = NULL;
466 }
467}
468
469int AudioVRDE::onVRDEControl(bool fEnable, uint32_t uFlags)
470{
471 LogFlowThisFunc(("fEnable=%RTbool, uFlags=0x%x\n", fEnable, uFlags));
472
473 if (mpDrv == NULL)
474 return VERR_INVALID_STATE;
475
476 mpDrv->fEnabled = fEnable;
477
478 return VINF_SUCCESS; /* Never veto. */
479}
480
481/**
482 * Marks the beginning of sending captured audio data from a connected
483 * RDP client.
484 *
485 * @return IPRT status code.
486 * @param pvContext The context; in this case a pointer to a
487 * VRDESTREAMIN structure.
488 * @param pVRDEAudioBegin Pointer to a VRDEAUDIOINBEGIN structure.
489 */
490int AudioVRDE::onVRDEInputBegin(void *pvContext, PVRDEAUDIOINBEGIN pVRDEAudioBegin)
491{
492 AssertPtrReturn(pvContext, VERR_INVALID_POINTER);
493 AssertPtrReturn(pVRDEAudioBegin, VERR_INVALID_POINTER);
494
495 PVRDESTREAMIN pVRDEStrmIn = (PVRDESTREAMIN)pvContext;
496 AssertPtrReturn(pVRDEStrmIn, VERR_INVALID_POINTER);
497
498 VRDEAUDIOFORMAT audioFmt = pVRDEAudioBegin->fmt;
499
500 int iSampleHz = VRDE_AUDIO_FMT_SAMPLE_FREQ(audioFmt); NOREF(iSampleHz);
501 int cChannels = VRDE_AUDIO_FMT_CHANNELS(audioFmt); NOREF(cChannels);
502 int cBits = VRDE_AUDIO_FMT_BITS_PER_SAMPLE(audioFmt); NOREF(cBits);
503 bool fUnsigned = VRDE_AUDIO_FMT_SIGNED(audioFmt); NOREF(fUnsigned);
504
505 LogFlowFunc(("cbSample=%RU32, iSampleHz=%d, cChannels=%d, cBits=%d, fUnsigned=%RTbool\n",
506 VRDE_AUDIO_FMT_BYTES_PER_SAMPLE(audioFmt), iSampleHz, cChannels, cBits, fUnsigned));
507
508 return VINF_SUCCESS;
509}
510
511int AudioVRDE::onVRDEInputData(void *pvContext, const void *pvData, uint32_t cbData)
512{
513 PVRDESTREAMIN pVRDEStrmIn = (PVRDESTREAMIN)pvContext;
514 AssertPtrReturn(pVRDEStrmIn, VERR_INVALID_POINTER);
515
516 PPDMAUDIOSTREAM pStream = &pVRDEStrmIn->Stream;
517 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
518
519 /** @todo Use CritSect! */
520
521 uint32_t cWritten;
522 int rc = AudioMixBufWriteCirc(&pStream->MixBuf, pvData, cbData, &cWritten);
523 if (RT_SUCCESS(rc))
524 pVRDEStrmIn->cSamplesCaptured += cWritten;
525
526 LogFlowFunc(("cbData=%RU32, cWritten=%RU32, cSamplesCaptured=%RU32, rc=%Rrc\n",
527 cbData, cWritten, pVRDEStrmIn->cSamplesCaptured, rc));
528 return rc;
529}
530
531int AudioVRDE::onVRDEInputEnd(void *pvContext)
532{
533 NOREF(pvContext);
534
535 return VINF_SUCCESS;
536}
537
538int AudioVRDE::onVRDEInputIntercept(bool fEnabled)
539{
540 return VINF_SUCCESS; /* Never veto. */
541}
542
543/**
544 * Construct a VRDE audio driver instance.
545 *
546 * @copydoc FNPDMDRVCONSTRUCT
547 */
548/* static */
549DECLCALLBACK(int) AudioVRDE::drvConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
550{
551 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
552 PDRVAUDIOVRDE pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIOVRDE);
553 AssertPtrReturn(pDrvIns, VERR_INVALID_POINTER);
554 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
555
556 LogRel(("Audio: Initializing VRDE driver\n"));
557 LogFlowFunc(("fFlags=0x%x\n", fFlags));
558
559 AssertMsgReturn(PDMDrvHlpNoAttach(pDrvIns) == VERR_PDM_NO_ATTACHED_DRIVER,
560 ("Configuration error: Not possible to attach anything to this driver!\n"),
561 VERR_PDM_DRVINS_NO_ATTACH);
562
563 /*
564 * Init the static parts.
565 */
566 pThis->pDrvIns = pDrvIns;
567 /* IBase */
568 pDrvIns->IBase.pfnQueryInterface = drvAudioVRDEQueryInterface;
569 /* IHostAudio */
570 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvAudioVRDE);
571
572 /* Init defaults. */
573 pThis->fEnabled = false;
574
575 /*
576 * Get the ConsoleVRDPServer object pointer.
577 */
578 void *pvUser;
579 int rc = CFGMR3QueryPtr(pCfg, "ObjectVRDPServer", &pvUser);
580 AssertMsgRCReturn(rc, ("Confguration error: No/bad \"ObjectVRDPServer\" value, rc=%Rrc\n", rc), rc);
581
582 /* CFGM tree saves the pointer to ConsoleVRDPServer in the Object node of AudioVRDE. */
583 pThis->pConsoleVRDPServer = (ConsoleVRDPServer *)pvUser;
584
585 /*
586 * Get the AudioVRDE object pointer.
587 */
588 pvUser = NULL;
589 rc = CFGMR3QueryPtr(pCfg, "Object", &pvUser);
590 AssertMsgRCReturn(rc, ("Confguration error: No/bad \"Object\" value, rc=%Rrc\n", rc), rc);
591
592 pThis->pAudioVRDE = (AudioVRDE *)pvUser;
593 pThis->pAudioVRDE->mpDrv = pThis;
594
595 /*
596 * Get the interface for the above driver (DrvAudio) to make mixer/conversion calls.
597 * Described in CFGM tree.
598 */
599 pThis->pDrvAudio = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIAUDIOCONNECTOR);
600 AssertMsgReturn(pThis->pDrvAudio, ("Configuration error: No upper interface specified!\n"), VERR_PDM_MISSING_INTERFACE_ABOVE);
601
602 return VINF_SUCCESS;
603}
604
605
606/**
607 * @interface_method_impl{PDMDRVREG,pfnDestruct}
608 */
609/* static */
610DECLCALLBACK(void) AudioVRDE::drvDestruct(PPDMDRVINS pDrvIns)
611{
612 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
613 PDRVAUDIOVRDE pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIOVRDE);
614 LogFlowFuncEnter();
615
616 /*
617 * If the AudioVRDE object is still alive, we must clear it's reference to
618 * us since we'll be invalid when we return from this method.
619 */
620 if (pThis->pAudioVRDE)
621 {
622 pThis->pAudioVRDE->mpDrv = NULL;
623 pThis->pAudioVRDE = NULL;
624 }
625}
626
627
628/**
629 * VRDE audio driver registration record.
630 */
631const PDMDRVREG AudioVRDE::DrvReg =
632{
633 PDM_DRVREG_VERSION,
634 /* szName */
635 "AudioVRDE",
636 /* szRCMod */
637 "",
638 /* szR0Mod */
639 "",
640 /* pszDescription */
641 "Audio driver for VRDE backend",
642 /* fFlags */
643 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
644 /* fClass. */
645 PDM_DRVREG_CLASS_AUDIO,
646 /* cMaxInstances */
647 ~0U,
648 /* cbInstance */
649 sizeof(DRVAUDIOVRDE),
650 /* pfnConstruct */
651 AudioVRDE::drvConstruct,
652 /* pfnDestruct */
653 AudioVRDE::drvDestruct,
654 /* pfnRelocate */
655 NULL,
656 /* pfnIOCtl */
657 NULL,
658 /* pfnPowerOn */
659 NULL,
660 /* pfnReset */
661 NULL,
662 /* pfnSuspend */
663 NULL,
664 /* pfnResume */
665 NULL,
666 /* pfnAttach */
667 NULL,
668 /* pfnDetach */
669 NULL,
670 /* pfnPowerOff */
671 NULL,
672 /* pfnSoftReset */
673 NULL,
674 /* u32EndVersion */
675 PDM_DRVREG_VERSION
676};
677
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