VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostDSound.cpp@ 62877

Last change on this file since 62877 was 62463, checked in by vboxsync, 9 years ago

Devices: scm

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 75.9 KB
Line 
1/* $Id: DrvHostDSound.cpp 62463 2016-07-22 16:32:54Z vboxsync $ */
2/** @file
3 * Windows host backend driver using DirectSound.
4 */
5
6/*
7 * Copyright (C) 2006-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#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
20#include <VBox/log.h>
21#include <dsound.h>
22
23#include <iprt/alloc.h>
24#include <iprt/uuid.h>
25
26#include "AudioMixBuffer.h"
27#include "DrvAudio.h"
28#include "VBoxDD.h"
29
30/*
31 * IDirectSound* interface uses HRESULT status codes and the driver callbacks use
32 * the IPRT status codes. To minimize HRESULT->IPRT conversion most internal functions
33 * in the driver return HRESULT and conversion is done in the driver callbacks.
34 *
35 * Naming convention:
36 * 'dsound*' functions return IPRT status code;
37 * 'directSound*' - return HRESULT.
38 */
39
40/*
41 * Optional release logging, which a user can turn on with the
42 * 'VBoxManage debugvm' command.
43 * Debug logging still uses the common Log* macros from IPRT.
44 * Messages which always should go to the release log use LogRel.
45 */
46/* General code behavior. */
47#define DSLOG(a) do { LogRel2(a); } while(0)
48/* Something which produce a lot of logging during playback/recording. */
49#define DSLOGF(a) do { LogRel3(a); } while(0)
50/* Important messages like errors. Limited in the default release log to avoid log flood. */
51#define DSLOGREL(a) \
52 do { \
53 static int8_t scLogged = 0; \
54 if (scLogged < 8) { \
55 ++scLogged; \
56 LogRel(a); \
57 } \
58 else { \
59 DSLOG(a); \
60 } \
61 } while (0)
62
63/* Dynamically load dsound.dll. */
64typedef HRESULT WINAPI FNDIRECTSOUNDENUMERATEW(LPDSENUMCALLBACKW pDSEnumCallback, LPVOID pContext);
65typedef FNDIRECTSOUNDENUMERATEW *PFNDIRECTSOUNDENUMERATEW;
66typedef HRESULT WINAPI FNDIRECTSOUNDCAPTUREENUMERATEW(LPDSENUMCALLBACKW pDSEnumCallback, LPVOID pContext);
67typedef FNDIRECTSOUNDCAPTUREENUMERATEW *PFNDIRECTSOUNDCAPTUREENUMERATEW;
68
69#ifdef VBOX_WITH_AUDIO_CALLBACKS
70# define VBOX_DSOUND_MAX_EVENTS 3
71
72typedef enum DSOUNDEVENT
73{
74 DSOUNDEVENT_NOTIFY = 0,
75 DSOUNDEVENT_INPUT,
76 DSOUNDEVENT_OUTPUT,
77 } DSOUNDEVENT;
78#endif /* VBOX_WITH_AUDIO_CALLBACKS */
79
80typedef struct DSOUNDHOSTCFG
81{
82 DWORD cbBufferIn;
83 DWORD cbBufferOut;
84 RTUUID uuidPlay;
85 LPCGUID pGuidPlay;
86 RTUUID uuidCapture;
87 LPCGUID pGuidCapture;
88} DSOUNDHOSTCFG, *PDSOUNDHOSTCFG;
89
90typedef struct DSOUNDSTREAMOUT
91{
92 /** Associated host output stream.
93 * Note: Always must come first! */
94 PDMAUDIOSTREAM Stream;
95 LPDIRECTSOUND8 pDS; /** @todo Move this out of this structure! Not required per-stream (e.g. for multi-channel). */
96 LPDIRECTSOUNDBUFFER8 pDSB;
97 DWORD cbPlayWritePos;
98 DWORD csPlaybackBufferSize;
99 bool fEnabled;
100 bool fRestartPlayback;
101 PDMAUDIOSTREAMCFG streamCfg;
102} DSOUNDSTREAMOUT, *PDSOUNDSTREAMOUT;
103
104typedef struct DSOUNDSTREAMIN
105{
106 /** Associated host input stream.
107 * Note: Always must come first! */
108 PDMAUDIOSTREAM Stream;
109 LPDIRECTSOUNDCAPTURE8 pDSC;
110 LPDIRECTSOUNDCAPTUREBUFFER8 pDSCB;
111 DWORD csCaptureReadPos;
112 DWORD csCaptureBufferSize;
113 HRESULT hrLastCapture;
114 PDMAUDIORECSOURCE enmRecSource;
115 bool fEnabled;
116 PDMAUDIOSTREAMCFG streamCfg;
117} DSOUNDSTREAMIN, *PDSOUNDSTREAMIN;
118
119typedef struct DRVHOSTDSOUND
120{
121 /** Pointer to the driver instance structure. */
122 PPDMDRVINS pDrvIns;
123 /** Our audio host audio interface. */
124 PDMIHOSTAUDIO IHostAudio;
125 /** List of found host input devices. */
126 RTLISTANCHOR lstDevInput;
127 /** List of found host output devices. */
128 RTLISTANCHOR lstDevOutput;
129 /** DirectSound configuration options. */
130 DSOUNDHOSTCFG cfg;
131 /** Whether this backend supports any audio input. */
132 bool fEnabledIn;
133 /** Whether this backend supports any audio output. */
134 bool fEnabledOut;
135#ifdef VBOX_WITH_AUDIO_CALLBACKS
136 /** Pointer to the audio connector interface of the driver/device above us. */
137 PPDMIAUDIOCONNECTOR pUpIAudioConnector;
138 /** Stopped indicator. */
139 bool fStopped;
140 /** Shutdown indicator. */
141 bool fShutdown;
142 /** Notification thread. */
143 RTTHREAD Thread;
144 /** Array of events to wait for in notification thread. */
145 HANDLE aEvents[VBOX_DSOUND_MAX_EVENTS];
146 /** Number of events to wait for in notification thread.
147 * Must not exceed VBOX_DSOUND_MAX_EVENTS. */
148 uint8_t cEvents;
149 /** Pointer to the input stream. */
150 PDSOUNDSTREAMIN pDSStrmIn;
151 /** Pointer to the output stream. */
152 PDSOUNDSTREAMOUT pDSStrmOut;
153#endif
154} DRVHOSTDSOUND, *PDRVHOSTDSOUND;
155
156/** No flags specified. */
157#define DSOUNDENUMCBFLAGS_NONE 0
158/** (Release) log found devices. */
159#define DSOUNDENUMCBFLAGS_LOG RT_BIT(0)
160
161/**
162 * Callback context for enumeration callbacks
163 */
164typedef struct DSOUNDENUMCBCTX
165{
166 /** Pointer to host backend driver. */
167 PDRVHOSTDSOUND pDrv;
168 /** Enumeration flags. */
169 uint32_t fFlags;
170 /** Number of found input devices. */
171 uint8_t cDevIn;
172 /** Number of found output devices. */
173 uint8_t cDevOut;
174} DSOUNDENUMCBCTX, *PDSOUNDENUMCBCTX;
175
176typedef struct DSOUNDDEV
177{
178 RTLISTNODE Node;
179 char *pszName;
180 GUID Guid;
181} DSOUNDDEV, *PDSOUNDDEV;
182
183
184/*********************************************************************************************************************************
185* Defines *
186*********************************************************************************************************************************/
187
188/** Maximum number of attempts to restore the sound buffer before giving up. */
189#define DRV_DSOUND_RESTORE_ATTEMPTS_MAX 3
190
191/** Makes DRVHOSTDSOUND out of PDMIHOSTAUDIO. */
192#define PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface) \
193 ( (PDRVHOSTDSOUND)((uintptr_t)pInterface - RT_OFFSETOF(DRVHOSTDSOUND, IHostAudio)) )
194
195
196/*********************************************************************************************************************************
197* Prototypes *
198*********************************************************************************************************************************/
199
200static HRESULT directSoundPlayRestore(PDRVHOSTDSOUND pThis, LPDIRECTSOUNDBUFFER8 pDSB);
201
202static void dsoundDeviceRemove(PDSOUNDDEV pDev);
203static int dsoundDevicesEnumerate(PDRVHOSTDSOUND pThis, PPDMAUDIOBACKENDCFG pCfg);
204#ifdef VBOX_WITH_AUDIO_CALLBACKS
205static int dsoundNotifyThread(PDRVHOSTDSOUND pThis, bool fShutdown);
206#endif
207
208static DWORD dsoundRingDistance(DWORD offEnd, DWORD offBegin, DWORD cSize)
209{
210 return offEnd >= offBegin ? offEnd - offBegin : cSize - offBegin + offEnd;
211}
212
213static int dsoundWaveFmtFromCfg(PPDMAUDIOSTREAMCFG pCfg, PWAVEFORMATEX pFmt)
214{
215 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
216 AssertPtrReturn(pFmt, VERR_INVALID_POINTER);
217
218 RT_BZERO(pFmt, sizeof(WAVEFORMATEX));
219
220 pFmt->wFormatTag = WAVE_FORMAT_PCM;
221 pFmt->nChannels = pCfg->cChannels;
222 pFmt->nSamplesPerSec = pCfg->uHz;
223 pFmt->nAvgBytesPerSec = pCfg->uHz << (pCfg->cChannels == 2 ? 1: 0);
224 pFmt->nBlockAlign = 1 << (pCfg->cChannels == 2 ? 1: 0);
225 pFmt->cbSize = 0; /* No extra data specified. */
226
227 switch (pCfg->enmFormat)
228 {
229 case PDMAUDIOFMT_S8:
230 case PDMAUDIOFMT_U8:
231 pFmt->wBitsPerSample = 8;
232 break;
233
234 case PDMAUDIOFMT_S16:
235 case PDMAUDIOFMT_U16:
236 pFmt->wBitsPerSample = 16;
237 pFmt->nAvgBytesPerSec <<= 1;
238 pFmt->nBlockAlign <<= 1;
239 break;
240
241 case PDMAUDIOFMT_S32:
242 case PDMAUDIOFMT_U32:
243 pFmt->wBitsPerSample = 32;
244 pFmt->nAvgBytesPerSec <<= 2;
245 pFmt->nBlockAlign <<= 2;
246 break;
247
248 default:
249 AssertMsgFailed(("Wave format %ld not supported\n", pCfg->enmFormat));
250 return VERR_NOT_SUPPORTED;
251 }
252
253 return VINF_SUCCESS;
254}
255
256static int dsoundGetPosOut(PDRVHOSTDSOUND pThis,
257 PDSOUNDSTREAMOUT pDSoundStream, DWORD *pdwBuffer, DWORD *pdwFree, DWORD *pdwPlayPos)
258{
259 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
260 AssertPtrReturn(pDSoundStream, VERR_INVALID_POINTER);
261 /* pdwBuffer is optional. */
262 /* pdwFree is optional. */
263 /* pdwPlayPos is optional. */
264
265 LPDIRECTSOUNDBUFFER8 pDSB = pDSoundStream->pDSB;
266 if (!pDSB)
267 return VERR_INVALID_POINTER;
268
269 DWORD cbBuffer = AUDIOMIXBUF_S2B(&pDSoundStream->Stream.MixBuf, pDSoundStream->csPlaybackBufferSize);
270
271 /* Get the current play position which is used for calculating the free space in the buffer. */
272 DWORD cbPlayPos;
273
274 HRESULT hr;
275 for (unsigned i = 0; i < DRV_DSOUND_RESTORE_ATTEMPTS_MAX; i++)
276 {
277 hr = IDirectSoundBuffer8_GetCurrentPosition(pDSB, &cbPlayPos, NULL);
278 if ( SUCCEEDED(hr)
279 || hr != DSERR_BUFFERLOST) /** @todo: MSDN doesn't state this error for GetCurrentPosition(). */
280 {
281 break;
282 }
283 else
284 {
285 LogFlowFunc(("Getting playing position failed due to lost buffer, restoring ...\n"));
286 directSoundPlayRestore(pThis, pDSB);
287 }
288 }
289
290 int rc = VINF_SUCCESS;
291
292 if (FAILED(hr))
293 {
294 if (hr != DSERR_BUFFERLOST) /* Avoid log flooding if the error is still there. */
295 DSLOGREL(("DSound: Getting current playback position failed with %Rhrc\n", hr));
296 LogFlowFunc(("Failed with %Rhrc\n", hr));
297
298 rc = VERR_NOT_AVAILABLE;
299 }
300 else
301 {
302 if (pdwBuffer)
303 *pdwBuffer = cbBuffer;
304
305 if (pdwFree)
306 *pdwFree = cbBuffer - dsoundRingDistance(pDSoundStream->cbPlayWritePos, cbPlayPos, cbBuffer);
307
308 if (pdwPlayPos)
309 *pdwPlayPos = cbPlayPos;
310 }
311
312 if (RT_FAILURE(rc))
313 LogFlowFuncLeaveRC(rc);
314 return rc;
315}
316
317static char *dsoundGUIDToUtf8StrA(LPCGUID lpGUID)
318{
319 if (lpGUID)
320 {
321 LPOLESTR lpOLEStr;
322 HRESULT hr = StringFromCLSID(*lpGUID, &lpOLEStr);
323 if (SUCCEEDED(hr))
324 {
325 char *pszGUID;
326 int rc = RTUtf16ToUtf8(lpOLEStr, &pszGUID);
327 CoTaskMemFree(lpOLEStr);
328
329 return RT_SUCCESS(rc) ? pszGUID : NULL;
330 }
331 }
332
333 return RTStrDup("{Default device}");
334}
335
336/**
337 * Clears the list of the host's playback + capturing devices.
338 *
339 * @param pThis Host audio driver instance.
340 */
341static void dsoundDevicesClear(PDRVHOSTDSOUND pThis)
342{
343 AssertPtrReturnVoid(pThis);
344
345 PDSOUNDDEV pDev;
346 while (!RTListIsEmpty(&pThis->lstDevInput))
347 {
348 pDev = RTListGetFirst(&pThis->lstDevInput, DSOUNDDEV, Node);
349 dsoundDeviceRemove(pDev);
350 }
351
352 while (!RTListIsEmpty(&pThis->lstDevOutput))
353 {
354 pDev = RTListGetFirst(&pThis->lstDevOutput, DSOUNDDEV, Node);
355 dsoundDeviceRemove(pDev);
356 }
357}
358
359static HRESULT directSoundPlayRestore(PDRVHOSTDSOUND pThis, LPDIRECTSOUNDBUFFER8 pDSB)
360{
361 HRESULT hr = IDirectSoundBuffer8_Restore(pDSB);
362 if (FAILED(hr))
363 DSLOGREL(("DSound: Restoring playback buffer failed with %Rhrc\n", hr));
364 return hr;
365}
366
367static HRESULT directSoundPlayUnlock(PDRVHOSTDSOUND pThis, LPDIRECTSOUNDBUFFER8 pDSB,
368 LPVOID pv1, LPVOID pv2,
369 DWORD cb1, DWORD cb2)
370{
371 HRESULT hr = IDirectSoundBuffer8_Unlock(pDSB, pv1, cb1, pv2, cb2);
372 if (FAILED(hr))
373 DSLOGREL(("DSound: Unlocking playback buffer failed with %Rhrc\n", hr));
374 return hr;
375}
376
377static HRESULT directSoundCaptureUnlock(LPDIRECTSOUNDCAPTUREBUFFER8 pDSCB,
378 LPVOID pv1, LPVOID pv2,
379 DWORD cb1, DWORD cb2)
380{
381 HRESULT hr = IDirectSoundCaptureBuffer8_Unlock(pDSCB, pv1, cb1, pv2, cb2);
382 if (FAILED(hr))
383 DSLOGREL(("DSound: Unlocking capture buffer failed with %Rhrc\n", hr));
384 return hr;
385}
386
387static HRESULT directSoundPlayLock(PDRVHOSTDSOUND pThis,
388 LPDIRECTSOUNDBUFFER8 pDSB, PDMPCMPROPS *pProps,
389 DWORD dwOffset, DWORD dwBytes,
390 LPVOID *ppv1, LPVOID *ppv2,
391 DWORD *pcb1, DWORD *pcb2,
392 DWORD dwFlags)
393{
394 LPVOID pv1 = NULL;
395 LPVOID pv2 = NULL;
396 DWORD cb1 = 0;
397 DWORD cb2 = 0;
398
399 HRESULT hr;
400 for (unsigned i = 0; i < DRV_DSOUND_RESTORE_ATTEMPTS_MAX; i++)
401 {
402 hr = IDirectSoundBuffer8_Lock(pDSB, dwOffset, dwBytes, &pv1, &cb1, &pv2, &cb2, dwFlags);
403 if ( SUCCEEDED(hr)
404 || hr != DSERR_BUFFERLOST)
405 break;
406 else
407 {
408 LogFlowFunc(("Locking failed due to lost buffer, restoring ...\n"));
409 directSoundPlayRestore(pThis, pDSB);
410 }
411 }
412
413 if (FAILED(hr))
414 {
415 DSLOGREL(("DSound: Locking playback buffer failed with %Rhrc\n", hr));
416 return hr;
417 }
418
419 if ( (pv1 && (cb1 & pProps->uAlign))
420 || (pv2 && (cb2 & pProps->uAlign)))
421 {
422 DSLOGREL(("DSound: Locking playback buffer returned misaligned buffer: cb1=%RI32, cb2=%RI32 (alignment: %RU32)\n",
423 cb1, cb2, pProps->uAlign));
424 directSoundPlayUnlock(pThis, pDSB, pv1, pv2, cb1, cb2);
425 return E_FAIL;
426 }
427
428 *ppv1 = pv1;
429 *ppv2 = pv2;
430 *pcb1 = cb1;
431 *pcb2 = cb2;
432
433 return S_OK;
434}
435
436static HRESULT directSoundCaptureLock(LPDIRECTSOUNDCAPTUREBUFFER8 pDSCB, PPDMPCMPROPS pProps,
437 DWORD dwOffset, DWORD dwBytes,
438 LPVOID *ppv1, LPVOID *ppv2,
439 DWORD *pcb1, DWORD *pcb2,
440 DWORD dwFlags)
441{
442 LPVOID pv1 = NULL;
443 LPVOID pv2 = NULL;
444 DWORD cb1 = 0;
445 DWORD cb2 = 0;
446
447 HRESULT hr = IDirectSoundCaptureBuffer8_Lock(pDSCB, dwOffset, dwBytes,
448 &pv1, &cb1, &pv2, &cb2, dwFlags);
449 if (FAILED(hr))
450 {
451 DSLOGREL(("DSound: Locking capture buffer failed with %Rhrc\n", hr));
452 return hr;
453 }
454
455 if ( (pv1 && (cb1 & pProps->uAlign))
456 || (pv2 && (cb2 & pProps->uAlign)))
457 {
458 DSLOGREL(("DSound: Locking capture buffer returned misaligned buffer: cb1=%RI32, cb2=%RI32 (alignment: %RU32)\n",
459 cb1, cb2, pProps->uAlign));
460 directSoundCaptureUnlock(pDSCB, pv1, pv2, cb1, cb2);
461 return E_FAIL;
462 }
463
464 *ppv1 = pv1;
465 *ppv2 = pv2;
466 *pcb1 = cb1;
467 *pcb2 = cb2;
468
469 return S_OK;
470}
471
472
473/*
474 * DirectSound playback
475 */
476
477static void directSoundPlayInterfaceRelease(PDSOUNDSTREAMOUT pDSoundStream)
478{
479 if (pDSoundStream->pDS)
480 {
481 IDirectSound8_Release(pDSoundStream->pDS);
482 pDSoundStream->pDS = NULL;
483 }
484}
485
486static HRESULT directSoundPlayInterfaceCreate(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMOUT pDSoundStream)
487{
488 if (pDSoundStream->pDS != NULL)
489 {
490 DSLOG(("DSound: DirectSound instance already exists\n"));
491 return S_OK;
492 }
493
494 HRESULT hr = CoCreateInstance(CLSID_DirectSound8, NULL, CLSCTX_ALL,
495 IID_IDirectSound8, (void **)&pDSoundStream->pDS);
496 if (FAILED(hr))
497 {
498 DSLOGREL(("DSound: Creating playback instance failed with %Rhrc\n", hr));
499 }
500 else
501 {
502 hr = IDirectSound8_Initialize(pDSoundStream->pDS, pThis->cfg.pGuidPlay);
503 if (SUCCEEDED(hr))
504 {
505 HWND hWnd = GetDesktopWindow();
506 hr = IDirectSound8_SetCooperativeLevel(pDSoundStream->pDS, hWnd, DSSCL_PRIORITY);
507 if (FAILED(hr))
508 DSLOGREL(("DSound: Setting cooperative level for window %p failed with %Rhrc\n", hWnd, hr));
509 }
510
511 if (FAILED(hr))
512 {
513 if (hr == DSERR_NODRIVER) /* Usually means that no playback devices are attached. */
514 DSLOGREL(("DSound: DirectSound playback is currently unavailable\n"));
515 else
516 DSLOGREL(("DSound: DirectSound playback initialization failed with %Rhrc\n", hr));
517
518 directSoundPlayInterfaceRelease(pDSoundStream);
519 }
520 }
521
522 return hr;
523}
524
525static HRESULT directSoundPlayClose(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMOUT pDSoundStream)
526{
527 AssertPtrReturn(pThis, E_POINTER);
528 AssertPtrReturn(pDSoundStream, E_POINTER);
529
530 DSLOG(("DSound: Closing playback stream %p, buffer %p\n", pDSoundStream, pDSoundStream->pDSB));
531
532 HRESULT hr = S_OK;
533
534 if (pDSoundStream->pDSB)
535 {
536 hr = IDirectSoundBuffer8_Stop(pDSoundStream->pDSB);
537 if (SUCCEEDED(hr))
538 {
539#ifdef VBOX_WITH_AUDIO_CALLBACKS
540 if (pThis->aEvents[DSOUNDEVENT_OUTPUT] != NULL)
541 {
542 CloseHandle(pThis->aEvents[DSOUNDEVENT_OUTPUT]);
543 pThis->aEvents[DSOUNDEVENT_OUTPUT] = NULL;
544
545 if (pThis->cEvents)
546 pThis->cEvents--;
547
548 pThis->pDSStream = NULL;
549 }
550
551 int rc2 = dsoundNotifyThread(pThis, false /* fShutdown */);
552 AssertRC(rc2);
553#endif
554 IDirectSoundBuffer8_Release(pDSoundStream->pDSB);
555 pDSoundStream->pDSB = NULL;
556 }
557 else
558 DSLOGREL(("DSound: Stop playback stream %p when closing %Rhrc\n", pDSoundStream, hr));
559 }
560
561 if (SUCCEEDED(hr))
562 directSoundPlayInterfaceRelease(pDSoundStream);
563
564 return hr;
565}
566
567static HRESULT directSoundPlayOpen(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMOUT pDSoundStream)
568{
569 AssertPtrReturn(pThis, E_POINTER);
570 AssertPtrReturn(pDSoundStream, E_POINTER);
571
572 DSLOG(("DSound: pDSoundStream=%p, cbBufferOut=%ld, uHz=%RU32, cChannels=%RU8, cBits=%RU8, fSigned=%RTbool\n",
573 pDSoundStream,
574 pThis->cfg.cbBufferOut,
575 pDSoundStream->Stream.Props.uHz,
576 pDSoundStream->Stream.Props.cChannels,
577 pDSoundStream->Stream.Props.cBits,
578 pDSoundStream->Stream.Props.fSigned));
579
580 if (pDSoundStream->pDSB != NULL)
581 {
582 /* Should not happen but be forgiving. */
583 DSLOGREL(("DSound: Playback buffer already exists\n"));
584 directSoundPlayClose(pThis, pDSoundStream);
585 }
586
587 WAVEFORMATEX wfx;
588 int rc = dsoundWaveFmtFromCfg(&pDSoundStream->streamCfg, &wfx);
589 if (RT_FAILURE(rc))
590 return E_INVALIDARG;
591
592 HRESULT hr = directSoundPlayInterfaceCreate(pThis, pDSoundStream);
593 if (FAILED(hr))
594 return hr;
595
596 do /* To use breaks. */
597 {
598 LPDIRECTSOUNDBUFFER pDSB = NULL;
599
600 DSBUFFERDESC bd;
601 RT_ZERO(bd);
602 bd.dwSize = sizeof(bd);
603 bd.lpwfxFormat = &wfx;
604
605 /*
606 * As we reuse our (secondary) buffer for playing out data as it comes in,
607 * we're using this buffer as a so-called static buffer.
608 *
609 * However, as we do not want to use memory on the sound device directly
610 * (as most modern audio hardware on the host doesn't have this anyway),
611 * we're *not* going to use DSBCAPS_STATIC for that.
612 *
613 * Instead we're specifying DSBCAPS_LOCSOFTWARE, as this fits the bill
614 * of copying own buffer data (from AudioMixBuf) to our secondary's Direct Sound buffer.
615 */
616 bd.dwFlags = DSBCAPS_GLOBALFOCUS | DSBCAPS_GETCURRENTPOSITION2 | DSBCAPS_LOCSOFTWARE;
617#ifdef VBOX_WITH_AUDIO_CALLBACKS
618 bd.dwFlags |= DSBCAPS_CTRLPOSITIONNOTIFY;
619#endif
620 bd.dwBufferBytes = pThis->cfg.cbBufferOut;
621
622 hr = IDirectSound8_CreateSoundBuffer(pDSoundStream->pDS, &bd, &pDSB, NULL);
623 if (FAILED(hr))
624 {
625 DSLOGREL(("DSound: Creating playback sound buffer failed with %Rhrc\n", hr));
626 break;
627 }
628
629 /* "Upgrade" to IDirectSoundBuffer8 interface. */
630 hr = IDirectSoundBuffer_QueryInterface(pDSB, IID_IDirectSoundBuffer8, (LPVOID *)&pDSoundStream->pDSB);
631 IDirectSoundBuffer_Release(pDSB);
632 if (FAILED(hr))
633 {
634 DSLOGREL(("DSound: Querying playback sound buffer interface failed with %Rhrc\n", hr));
635 break;
636 }
637
638 /*
639 * Query the actual parameters.
640 */
641 hr = IDirectSoundBuffer8_GetFormat(pDSoundStream->pDSB, &wfx, sizeof(wfx), NULL);
642 if (FAILED(hr))
643 {
644 DSLOGREL(("DSound: Getting playback format failed with %Rhrc\n", hr));
645 break;
646 }
647
648 DSBCAPS bc;
649 RT_ZERO(bc);
650 bc.dwSize = sizeof(bc);
651 hr = IDirectSoundBuffer8_GetCaps(pDSoundStream->pDSB, &bc);
652 if (FAILED(hr))
653 {
654 DSLOGREL(("DSound: Getting playback capabilities failed with %Rhrc\n", hr));
655 break;
656 }
657
658 DSLOG(("DSound: Playback format:\n"
659 " dwBufferBytes = %RI32\n"
660 " dwFlags = 0x%x\n"
661 " wFormatTag = %RI16\n"
662 " nChannels = %RI16\n"
663 " nSamplesPerSec = %RU32\n"
664 " nAvgBytesPerSec = %RU32\n"
665 " nBlockAlign = %RI16\n"
666 " wBitsPerSample = %RI16\n"
667 " cbSize = %RI16\n",
668 bc.dwBufferBytes,
669 bc.dwFlags,
670 wfx.wFormatTag,
671 wfx.nChannels,
672 wfx.nSamplesPerSec,
673 wfx.nAvgBytesPerSec,
674 wfx.nBlockAlign,
675 wfx.wBitsPerSample,
676 wfx.cbSize));
677
678 if (bc.dwBufferBytes & pDSoundStream->Stream.Props.uAlign)
679 DSLOGREL(("DSound: Playback capabilities returned misaligned buffer: size %RU32, alignment %RU32\n",
680 bc.dwBufferBytes, pDSoundStream->Stream.Props.uAlign + 1));
681
682 if (bc.dwBufferBytes != pThis->cfg.cbBufferOut)
683 DSLOGREL(("DSound: Playback buffer size mismatched: DirectSound %RU32, requested %RU32 bytes\n",
684 bc.dwBufferBytes, pThis->cfg.cbBufferOut));
685
686 /*
687 * Initial state.
688 * dsoundPlayStart initializes part of it to make sure that Stop/Start continues with a correct
689 * playback buffer position.
690 */
691 pDSoundStream->csPlaybackBufferSize = bc.dwBufferBytes >> pDSoundStream->Stream.Props.cShift;
692 DSLOG(("DSound: csPlaybackBufferSize=%RU32\n", pDSoundStream->csPlaybackBufferSize));
693
694#ifdef VBOX_WITH_AUDIO_CALLBACKS
695 /*
696 * Install notification.
697 */
698 pThis->aEvents[DSOUNDEVENT_OUTPUT] = CreateEvent(NULL /* Security attribute */,
699 FALSE /* bManualReset */, FALSE /* bInitialState */,
700 NULL /* lpName */);
701 if (pThis->aEvents[DSOUNDEVENT_OUTPUT] == NULL)
702 {
703 hr = HRESULT_FROM_WIN32(GetLastError());
704 DSLOGREL(("DSound: CreateEvent for output failed with %Rhrc\n", hr));
705 break;
706 }
707
708 LPDIRECTSOUNDNOTIFY8 pNotify;
709 hr = IDirectSoundNotify_QueryInterface(pDSoundStream->pDSB, IID_IDirectSoundNotify8, (LPVOID *)&pNotify);
710 if (SUCCEEDED(hr))
711 {
712 DSBPOSITIONNOTIFY dsBufPosNotify;
713 RT_ZERO(dsBufPosNotify);
714 dsBufPosNotify.dwOffset = DSBPN_OFFSETSTOP;
715 dsBufPosNotify.hEventNotify = pThis->aEvents[DSOUNDEVENT_OUTPUT];
716
717 hr = IDirectSoundNotify_SetNotificationPositions(pNotify, 1 /* Count */, &dsBufPosNotify);
718 if (FAILED(hr))
719 DSLOGREL(("DSound: Setting playback position notification failed with %Rhrc\n", hr));
720
721 IDirectSoundNotify_Release(pNotify);
722 }
723 else
724 DSLOGREL(("DSound: Querying interface for position notification failed with %Rhrc\n", hr));
725
726 if (FAILED(hr))
727 break;
728
729 pThis->pDSStreamOut = pDSoundStream;
730
731 Assert(pThis->cEvents < VBOX_DSOUND_MAX_EVENTS);
732 pThis->cEvents++;
733
734 /* Let the thread know. */
735 dsoundNotifyThread(pThis, false /* fShutdown */);
736
737 /* Trigger the just installed output notification. */
738 hr = IDirectSoundBuffer8_Play(pDSoundStream->pDSB, 0, 0, 0);
739
740#endif /* VBOX_WITH_AUDIO_CALLBACKS */
741
742 } while (0);
743
744 if (FAILED(hr))
745 directSoundPlayClose(pThis, pDSoundStream);
746
747 return hr;
748}
749
750static void dsoundPlayClearSamples(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMOUT pDSoundStream)
751{
752 AssertPtrReturnVoid(pDSoundStream);
753
754 PPDMAUDIOSTREAM pStream = &pDSoundStream->Stream;
755
756 LPVOID pv1, pv2;
757 DWORD cb1, cb2;
758 HRESULT hr = directSoundPlayLock(pThis, pDSoundStream->pDSB, &pDSoundStream->Stream.Props,
759 0 /* dwOffset */, AUDIOMIXBUF_S2B(&pStream->MixBuf, pDSoundStream->csPlaybackBufferSize),
760 &pv1, &pv2, &cb1, &cb2, DSBLOCK_ENTIREBUFFER);
761 if (SUCCEEDED(hr))
762 {
763 DWORD len1 = AUDIOMIXBUF_B2S(&pStream->MixBuf, cb1);
764 DWORD len2 = AUDIOMIXBUF_B2S(&pStream->MixBuf, cb2);
765
766 if (pv1 && len1)
767 DrvAudioHlpClearBuf(&pDSoundStream->Stream.Props, pv1, cb1, len1);
768
769 if (pv2 && len2)
770 DrvAudioHlpClearBuf(&pDSoundStream->Stream.Props, pv2, cb2, len2);
771
772 directSoundPlayUnlock(pThis, pDSoundStream->pDSB, pv1, pv2, cb1, cb2);
773 }
774}
775
776static HRESULT directSoundPlayGetStatus(PDRVHOSTDSOUND pThis, LPDIRECTSOUNDBUFFER8 pDSB, DWORD *pdwStatus)
777{
778 AssertPtrReturn(pThis, E_POINTER);
779 AssertPtrReturn(pDSB, E_POINTER);
780 /* pdwStatus is optional. */
781
782 DWORD dwStatus = 0;
783
784 HRESULT hr;
785 for (unsigned i = 0; i < DRV_DSOUND_RESTORE_ATTEMPTS_MAX; i++)
786 {
787 hr = IDirectSoundBuffer8_GetStatus(pDSB, &dwStatus);
788 if ( hr == DSERR_BUFFERLOST
789 || ( SUCCEEDED(hr)
790 && (dwStatus & DSBSTATUS_BUFFERLOST)))
791 {
792 LogFlowFunc(("Getting status failed due to lost buffer, restoring ...\n"));
793 directSoundPlayRestore(pThis, pDSB);
794 }
795 else
796 break;
797 }
798
799 if (SUCCEEDED(hr))
800 {
801 if (pdwStatus)
802 *pdwStatus = dwStatus;
803 }
804 else
805 DSLOGREL(("DSound: Retrieving playback status failed with %Rhrc\n", hr));
806
807 return hr;
808}
809
810static HRESULT directSoundPlayStop(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMOUT pDSoundStream)
811{
812 AssertPtrReturn(pThis, E_POINTER);
813 AssertPtrReturn(pDSoundStream, E_POINTER);
814
815 HRESULT hr;
816
817 if (pDSoundStream->pDSB != NULL)
818 {
819 DSLOG(("DSound: Stopping playback\n"));
820
821 HRESULT hr2 = IDirectSoundBuffer8_Stop(pDSoundStream->pDSB);
822 if (FAILED(hr2))
823 {
824 hr2 = directSoundPlayRestore(pThis, pDSoundStream->pDSB);
825 if (FAILED(hr2))
826 hr2 = IDirectSoundBuffer8_Stop(pDSoundStream->pDSB);
827 }
828
829 if (FAILED(hr2))
830 DSLOG(("DSound: Stopping playback failed with %Rhrc\n", hr2));
831
832 hr = S_OK; /* Always report success here. */
833 }
834 else
835 hr = E_UNEXPECTED;
836
837 if (SUCCEEDED(hr))
838 {
839 dsoundPlayClearSamples(pThis, pDSoundStream);
840 pDSoundStream->fEnabled = false;
841 }
842 else
843 DSLOGREL(("DSound: Stopping playback failed with %Rhrc\n", hr));
844
845 return hr;
846}
847
848static HRESULT directSoundPlayStart(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMOUT pDSoundStream)
849{
850 AssertPtrReturn(pThis, E_POINTER);
851 AssertPtrReturn(pDSoundStream, E_POINTER);
852
853 HRESULT hr;
854 if (pDSoundStream->pDSB != NULL)
855 {
856 DWORD dwStatus;
857 hr = directSoundPlayGetStatus(pThis, pDSoundStream->pDSB, &dwStatus);
858 if (SUCCEEDED(hr))
859 {
860 if (dwStatus & DSBSTATUS_PLAYING)
861 {
862 DSLOG(("DSound: Already playing\n"));
863 }
864 else
865 {
866 dsoundPlayClearSamples(pThis, pDSoundStream);
867
868 pDSoundStream->fRestartPlayback = true;
869 pDSoundStream->fEnabled = true;
870
871 DSLOG(("DSound: Playback started\n"));
872
873 /*
874 * The actual IDirectSoundBuffer8_Play call will be made in drvHostDSoundPlay,
875 * because it is necessary to put some samples into the buffer first.
876 */
877 }
878 }
879 }
880 else
881 hr = E_UNEXPECTED;
882
883 if (FAILED(hr))
884 DSLOGREL(("DSound: Starting playback failed with %Rhrc\n", hr));
885
886 return hr;
887}
888
889/*
890 * DirectSoundCapture
891 */
892
893static LPCGUID dsoundCaptureSelectDevice(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMIN pDSoundStream)
894{
895 AssertPtrReturn(pThis, NULL);
896 AssertPtrReturn(pDSoundStream, NULL);
897
898 int rc = VINF_SUCCESS;
899
900 LPCGUID pGUID = pThis->cfg.pGuidCapture;
901 if (!pGUID)
902 {
903 PDSOUNDDEV pDev = NULL;
904
905 switch (pDSoundStream->enmRecSource)
906 {
907 case PDMAUDIORECSOURCE_LINE:
908 /*
909 * At the moment we're only supporting line-in in the HDA emulation,
910 * and line-in + mic-in in the AC'97 emulation both are expected
911 * to use the host's mic-in as well.
912 *
913 * So the fall through here is intentional for now.
914 */
915 case PDMAUDIORECSOURCE_MIC:
916 {
917 RTListForEach(&pThis->lstDevInput, pDev, DSOUNDDEV, Node)
918 {
919 if (RTStrIStr(pDev->pszName, "Mic")) /** @todo what is with non en_us windows versions? */
920 break;
921 }
922
923 if (RTListNodeIsDummy(&pThis->lstDevInput, pDev, DSOUNDDEV, Node))
924 pDev = NULL; /* Found nothing. */
925
926 break;
927 }
928
929 default:
930 AssertFailedStmt(rc = VERR_NOT_SUPPORTED);
931 break;
932 }
933
934 if ( RT_SUCCESS(rc)
935 && pDev)
936 {
937 DSLOG(("DSound: Guest source '%s' is using host recording device '%s'\n",
938 DrvAudioHlpRecSrcToStr(pDSoundStream->enmRecSource), pDev->pszName));
939
940 pGUID = &pDev->Guid;
941 }
942 }
943
944 if (RT_FAILURE(rc))
945 {
946 LogRel(("DSound: Selecting recording device failed with %Rrc\n", rc));
947 return NULL;
948 }
949
950 char *pszGUID = dsoundGUIDToUtf8StrA(pGUID);
951
952 /* This always has to be in the release log. */
953 LogRel(("DSound: Guest source '%s' is using host recording device with GUID '%s'\n",
954 DrvAudioHlpRecSrcToStr(pDSoundStream->enmRecSource), pszGUID ? pszGUID: "{?}"));
955
956 if (pszGUID)
957 {
958 RTStrFree(pszGUID);
959 pszGUID = NULL;
960 }
961
962 return pGUID;
963}
964
965static void directSoundCaptureInterfaceRelease(PDSOUNDSTREAMIN pDSoundStream)
966{
967 if (pDSoundStream->pDSC)
968 {
969 LogFlowFuncEnter();
970 IDirectSoundCapture_Release(pDSoundStream->pDSC);
971 pDSoundStream->pDSC = NULL;
972 }
973}
974
975static HRESULT directSoundCaptureInterfaceCreate(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMIN pDSoundStream)
976{
977 if (pDSoundStream->pDSC != NULL)
978 {
979 DSLOG(("DSound: DirectSoundCapture instance already exists\n"));
980 return S_OK;
981 }
982
983 HRESULT hr = CoCreateInstance(CLSID_DirectSoundCapture8, NULL, CLSCTX_ALL,
984 IID_IDirectSoundCapture8, (void **)&pDSoundStream->pDSC);
985 if (FAILED(hr))
986 {
987 DSLOGREL(("DSound: Creating capture instance failed with %Rhrc\n", hr));
988 }
989 else
990 {
991 LPCGUID pGUID = dsoundCaptureSelectDevice(pThis, pDSoundStream);
992 hr = IDirectSoundCapture_Initialize(pDSoundStream->pDSC, pGUID);
993 if (FAILED(hr))
994 {
995 if (hr == DSERR_NODRIVER) /* Usually means that no capture devices are attached. */
996 DSLOGREL(("DSound: Capture device currently is unavailable\n"));
997 else
998 DSLOGREL(("DSound: Initializing capturing device failed with %Rhrc\n", hr));
999
1000 directSoundCaptureInterfaceRelease(pDSoundStream);
1001 }
1002 }
1003
1004 LogFlowFunc(("Returning %Rhrc\n", hr));
1005 return hr;
1006}
1007
1008static HRESULT directSoundCaptureClose(PDSOUNDSTREAMIN pDSoundStream)
1009{
1010 AssertPtrReturn(pDSoundStream, E_POINTER);
1011
1012 DSLOG(("DSound: pDSoundStream=%p, pDSCB=%p\n", pDSoundStream, pDSoundStream->pDSCB));
1013
1014 HRESULT hr = S_OK;
1015
1016 if (pDSoundStream->pDSCB)
1017 {
1018 hr = IDirectSoundCaptureBuffer_Stop(pDSoundStream->pDSCB);
1019 if (SUCCEEDED(hr))
1020 {
1021 IDirectSoundCaptureBuffer8_Release(pDSoundStream->pDSCB);
1022 pDSoundStream->pDSCB = NULL;
1023 }
1024 else
1025 DSLOGREL(("DSound: Stopping capture buffer failed with %Rhrc\n", hr));
1026 }
1027
1028 if (SUCCEEDED(hr))
1029 directSoundCaptureInterfaceRelease(pDSoundStream);
1030
1031 LogFlowFunc(("Returning %Rhrc\n", hr));
1032 return hr;
1033}
1034
1035static HRESULT directSoundCaptureOpen(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMIN pDSoundStream)
1036{
1037 AssertPtrReturn(pThis, E_POINTER);
1038 AssertPtrReturn(pDSoundStream, E_POINTER);
1039
1040 DSLOG(("DSound: pDSoundStream=%p, cbBufferIn=%ld, uHz=%RU32, cChannels=%RU8, cBits=%RU8, fSigned=%RTbool\n",
1041 pDSoundStream,
1042 pThis->cfg.cbBufferIn,
1043 pDSoundStream->Stream.Props.uHz,
1044 pDSoundStream->Stream.Props.cChannels,
1045 pDSoundStream->Stream.Props.cBits,
1046 pDSoundStream->Stream.Props.fSigned));
1047
1048 if (pDSoundStream->pDSCB != NULL)
1049 {
1050 /* Should not happen but be forgiving. */
1051 DSLOGREL(("DSound: DirectSoundCaptureBuffer already exists\n"));
1052 directSoundCaptureClose(pDSoundStream);
1053 }
1054
1055 WAVEFORMATEX wfx;
1056 int rc = dsoundWaveFmtFromCfg(&pDSoundStream->streamCfg, &wfx);
1057 if (RT_FAILURE(rc))
1058 return E_INVALIDARG;
1059
1060 HRESULT hr = directSoundCaptureInterfaceCreate(pThis, pDSoundStream);
1061 if (FAILED(hr))
1062 return hr;
1063
1064 do /* To use breaks. */
1065 {
1066 LPDIRECTSOUNDCAPTUREBUFFER pDSCB = NULL;
1067 DSCBUFFERDESC bd;
1068 RT_ZERO(bd);
1069 bd.dwSize = sizeof(bd);
1070 bd.lpwfxFormat = &wfx;
1071 bd.dwBufferBytes = pThis->cfg.cbBufferIn;
1072 hr = IDirectSoundCapture_CreateCaptureBuffer(pDSoundStream->pDSC,
1073 &bd, &pDSCB, NULL);
1074 if (FAILED(hr))
1075 {
1076 if (hr == E_ACCESSDENIED)
1077 {
1078 DSLOGREL(("DSound: Capturing input from host not possible, access denied\n"));
1079 }
1080 else
1081 DSLOGREL(("DSound: Creating capture buffer failed with %Rhrc\n", hr));
1082 break;
1083 }
1084
1085 hr = IDirectSoundCaptureBuffer_QueryInterface(pDSCB, IID_IDirectSoundCaptureBuffer8, (void **)&pDSoundStream->pDSCB);
1086 IDirectSoundCaptureBuffer_Release(pDSCB);
1087 if (FAILED(hr))
1088 {
1089 DSLOGREL(("DSound: Querying interface for capture buffer failed with %Rhrc\n", hr));
1090 break;
1091 }
1092
1093 /*
1094 * Query the actual parameters.
1095 */
1096 DWORD cbReadPos = 0;
1097 hr = IDirectSoundCaptureBuffer8_GetCurrentPosition(pDSoundStream->pDSCB, NULL, &cbReadPos);
1098 if (FAILED(hr))
1099 {
1100 cbReadPos = 0;
1101 DSLOGREL(("DSound: Getting capture position failed with %Rhrc\n", hr));
1102 }
1103
1104 RT_ZERO(wfx);
1105 hr = IDirectSoundCaptureBuffer8_GetFormat(pDSoundStream->pDSCB, &wfx, sizeof(wfx), NULL);
1106 if (FAILED(hr))
1107 {
1108 DSLOGREL(("DSound: Getting capture format failed with %Rhrc\n", hr));
1109 break;
1110 }
1111
1112 DSCBCAPS bc;
1113 RT_ZERO(bc);
1114 bc.dwSize = sizeof(bc);
1115 hr = IDirectSoundCaptureBuffer8_GetCaps(pDSoundStream->pDSCB, &bc);
1116 if (FAILED(hr))
1117 {
1118 DSLOGREL(("Getting capture capabilities failed with %Rhrc\n", hr));
1119 break;
1120 }
1121
1122 DSLOG(("DSound: Capture format:\n"
1123 " dwBufferBytes = %RI32\n"
1124 " dwFlags = 0x%x\n"
1125 " wFormatTag = %RI16\n"
1126 " nChannels = %RI16\n"
1127 " nSamplesPerSec = %RU32\n"
1128 " nAvgBytesPerSec = %RU32\n"
1129 " nBlockAlign = %RI16\n"
1130 " wBitsPerSample = %RI16\n"
1131 " cbSize = %RI16\n",
1132 bc.dwBufferBytes,
1133 bc.dwFlags,
1134 wfx.wFormatTag,
1135 wfx.nChannels,
1136 wfx.nSamplesPerSec,
1137 wfx.nAvgBytesPerSec,
1138 wfx.nBlockAlign,
1139 wfx.wBitsPerSample,
1140 wfx.cbSize));
1141
1142 if (bc.dwBufferBytes & pDSoundStream->Stream.Props.uAlign)
1143 DSLOGREL(("DSound: Capture GetCaps returned misaligned buffer: size %RU32, alignment %RU32\n",
1144 bc.dwBufferBytes, pDSoundStream->Stream.Props.uAlign + 1));
1145
1146 if (bc.dwBufferBytes != pThis->cfg.cbBufferIn)
1147 DSLOGREL(("DSound: Capture buffer size mismatched: DirectSound %RU32, requested %RU32 bytes\n",
1148 bc.dwBufferBytes, pThis->cfg.cbBufferIn));
1149
1150 /* Initial state: reading at the initial capture position, no error. */
1151 pDSoundStream->csCaptureReadPos = cbReadPos >> pDSoundStream->Stream.Props.cShift;
1152 pDSoundStream->csCaptureBufferSize = bc.dwBufferBytes >> pDSoundStream->Stream.Props.cShift;
1153 pDSoundStream->hrLastCapture = S_OK;
1154
1155 DSLOG(("DSound: csCaptureReadPos=%RU32, csCaptureBufferSize=%RU32\n",
1156 pDSoundStream->csCaptureReadPos, pDSoundStream->csCaptureBufferSize));
1157
1158 } while (0);
1159
1160 if (FAILED(hr))
1161 directSoundCaptureClose(pDSoundStream);
1162
1163 LogFlowFunc(("Returning %Rhrc\n", hr));
1164 return hr;
1165}
1166
1167static HRESULT directSoundCaptureStop(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMIN pDSoundStream)
1168{
1169 AssertPtrReturn(pThis , E_POINTER);
1170 AssertPtrReturn(pDSoundStream, E_POINTER);
1171
1172 NOREF(pThis);
1173
1174 HRESULT hr;
1175
1176 if (pDSoundStream->pDSCB)
1177 {
1178 DSLOG(("DSound: Stopping capture\n"));
1179
1180 hr = IDirectSoundCaptureBuffer_Stop(pDSoundStream->pDSCB);
1181 if (FAILED(hr))
1182 DSLOGREL(("DSound: Stopping capture buffer failed with %Rhrc\n", hr));
1183 }
1184 else
1185 hr = E_UNEXPECTED;
1186
1187 if (SUCCEEDED(hr))
1188 pDSoundStream->fEnabled = false;
1189
1190 LogFlowFunc(("Returning %Rhrc\n", hr));
1191 return hr;
1192}
1193
1194static HRESULT directSoundCaptureStart(PDRVHOSTDSOUND pThis, PDSOUNDSTREAMIN pDSoundStream)
1195{
1196 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1197 AssertPtrReturn(pDSoundStream, VERR_INVALID_POINTER);
1198
1199 HRESULT hr;
1200 if (pDSoundStream->pDSCB != NULL)
1201 {
1202 DWORD dwStatus;
1203 hr = IDirectSoundCaptureBuffer8_GetStatus(pDSoundStream->pDSCB, &dwStatus);
1204 if (FAILED(hr))
1205 {
1206 DSLOGREL(("DSound: Retrieving capture status failed with %Rhrc\n", hr));
1207 }
1208 else
1209 {
1210 if (dwStatus & DSCBSTATUS_CAPTURING)
1211 {
1212 DSLOG(("DSound: Already capturing\n"));
1213 }
1214 else
1215 {
1216 DWORD fFlags = 0;
1217#ifndef VBOX_WITH_AUDIO_CALLBACKS
1218 fFlags |= DSCBSTART_LOOPING;
1219#endif
1220 DSLOG(("DSound: Starting to capture\n"));
1221 hr = IDirectSoundCaptureBuffer8_Start(pDSoundStream->pDSCB, fFlags);
1222 if (FAILED(hr))
1223 DSLOGREL(("DSound: Starting to capture failed with %Rhrc\n", hr));
1224 }
1225 }
1226 }
1227 else
1228 hr = E_UNEXPECTED;
1229
1230 if (SUCCEEDED(hr))
1231 pDSoundStream->fEnabled = true;
1232
1233 LogFlowFunc(("Returning %Rhrc\n", hr));
1234 return hr;
1235}
1236
1237static int dsoundDevAdd(PRTLISTANCHOR pList, LPGUID lpGUID,
1238 LPCWSTR lpwstrDescription, PDSOUNDDEV *ppDev)
1239{
1240 AssertPtrReturn(pList, VERR_INVALID_POINTER);
1241 AssertPtrReturn(lpGUID, VERR_INVALID_POINTER);
1242 AssertPtrReturn(lpwstrDescription, VERR_INVALID_POINTER);
1243
1244 PDSOUNDDEV pDev = (PDSOUNDDEV)RTMemAlloc(sizeof(DSOUNDDEV));
1245 if (!pDev)
1246 return VERR_NO_MEMORY;
1247
1248 int rc = RTUtf16ToUtf8(lpwstrDescription, &pDev->pszName);
1249 if (RT_SUCCESS(rc))
1250 memcpy(&pDev->Guid, lpGUID, sizeof(GUID));
1251
1252 if (RT_SUCCESS(rc))
1253 RTListAppend(pList, &pDev->Node);
1254
1255 if (ppDev)
1256 *ppDev = pDev;
1257
1258 return rc;
1259}
1260
1261static void dsoundDeviceRemove(PDSOUNDDEV pDev)
1262{
1263 if (pDev)
1264 {
1265 RTStrFree(pDev->pszName);
1266 pDev->pszName = NULL;
1267
1268 RTListNodeRemove(&pDev->Node);
1269
1270 RTMemFree(pDev);
1271 }
1272}
1273
1274static void dsoundLogDevice(const char *pszType, LPGUID lpGUID, LPCWSTR lpwstrDescription, LPCWSTR lpwstrModule)
1275{
1276 char *pszGUID = dsoundGUIDToUtf8StrA(lpGUID);
1277 /* This always has to be in the release log. */
1278 LogRel(("DSound: %s: GUID: %s [%ls] (Module: %ls)\n",
1279 pszType, pszGUID? pszGUID: "{?}", lpwstrDescription, lpwstrModule));
1280 RTStrFree(pszGUID);
1281}
1282
1283static BOOL CALLBACK dsoundDevicesEnumCbPlayback(LPGUID lpGUID, LPCWSTR lpwstrDescription,
1284 LPCWSTR lpwstrModule, LPVOID lpContext)
1285{
1286 PDSOUNDENUMCBCTX pCtx = (PDSOUNDENUMCBCTX)lpContext;
1287 AssertPtrReturn(pCtx, FALSE);
1288 AssertPtrReturn(pCtx->pDrv, FALSE);
1289
1290 if (!lpGUID)
1291 return TRUE;
1292
1293 AssertPtrReturn(lpwstrDescription, FALSE);
1294 /* Do not care about lpwstrModule. */
1295
1296 if (pCtx->fFlags & DSOUNDENUMCBFLAGS_LOG)
1297 dsoundLogDevice("Output", lpGUID, lpwstrDescription, lpwstrModule);
1298
1299 int rc = dsoundDevAdd(&pCtx->pDrv->lstDevOutput,
1300 lpGUID, lpwstrDescription, NULL /* ppDev */);
1301 if (RT_FAILURE(rc))
1302 return FALSE; /* Abort enumeration. */
1303
1304 pCtx->cDevOut++;
1305
1306 return TRUE;
1307}
1308
1309static BOOL CALLBACK dsoundDevicesEnumCbCapture(LPGUID lpGUID, LPCWSTR lpwstrDescription,
1310 LPCWSTR lpwstrModule, LPVOID lpContext)
1311{
1312 PDSOUNDENUMCBCTX pCtx = (PDSOUNDENUMCBCTX)lpContext;
1313 AssertPtrReturn(pCtx, FALSE);
1314 AssertPtrReturn(pCtx->pDrv, FALSE);
1315
1316 if (!lpGUID)
1317 return TRUE;
1318
1319 if (pCtx->fFlags & DSOUNDENUMCBFLAGS_LOG)
1320 dsoundLogDevice("Input", lpGUID, lpwstrDescription, lpwstrModule);
1321
1322 int rc = dsoundDevAdd(&pCtx->pDrv->lstDevInput,
1323 lpGUID, lpwstrDescription, NULL /* ppDev */);
1324 if (RT_FAILURE(rc))
1325 return FALSE; /* Abort enumeration. */
1326
1327 pCtx->cDevIn++;
1328
1329 return TRUE;
1330}
1331
1332/**
1333 * Does a (Re-)enumeration of the host's playback + capturing devices.
1334 *
1335 * @return IPRT status code.
1336 * @param pThis Host audio driver instance.
1337 * @param pEnmCtx Enumeration context to use.
1338 * @param fEnum Enumeration flags.
1339 */
1340static int dsoundDevicesEnumerate(PDRVHOSTDSOUND pThis, PDSOUNDENUMCBCTX pEnmCtx, uint32_t fEnum)
1341{
1342 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1343 AssertPtrReturn(pEnmCtx, VERR_INVALID_POINTER);
1344
1345 dsoundDevicesClear(pThis);
1346
1347 RTLDRMOD hDSound = NULL;
1348 int rc = RTLdrLoadSystem("dsound.dll", true /*fNoUnload*/, &hDSound);
1349 if (RT_SUCCESS(rc))
1350 {
1351 PFNDIRECTSOUNDENUMERATEW pfnDirectSoundEnumerateW = NULL;
1352 PFNDIRECTSOUNDCAPTUREENUMERATEW pfnDirectSoundCaptureEnumerateW = NULL;
1353
1354 rc = RTLdrGetSymbol(hDSound, "DirectSoundEnumerateW", (void**)&pfnDirectSoundEnumerateW);
1355 if (RT_SUCCESS(rc))
1356 rc = RTLdrGetSymbol(hDSound, "DirectSoundCaptureEnumerateW", (void**)&pfnDirectSoundCaptureEnumerateW);
1357
1358 if (RT_SUCCESS(rc))
1359 {
1360 HRESULT hr = pfnDirectSoundEnumerateW(&dsoundDevicesEnumCbPlayback, pEnmCtx);
1361 if (FAILED(hr))
1362 LogRel2(("DSound: Error enumerating host playback devices: %Rhrc\n", hr));
1363
1364 hr = pfnDirectSoundCaptureEnumerateW(&dsoundDevicesEnumCbCapture, pEnmCtx);
1365 if (FAILED(hr))
1366 LogRel2(("DSound: Error enumerating host capturing devices: %Rhrc\n", hr));
1367
1368 if (fEnum & DSOUNDENUMCBFLAGS_LOG)
1369 {
1370 LogRel2(("DSound: Found %RU8 host playback devices\n", pEnmCtx->cDevOut));
1371 LogRel2(("DSound: Found %RU8 host capturing devices\n", pEnmCtx->cDevIn));
1372 }
1373 }
1374
1375 RTLdrClose(hDSound);
1376 }
1377 else
1378 {
1379 /* No dsound.dll on this system. */
1380 LogRel2(("DSound: Could not load dsound.dll: %Rrc\n", rc));
1381 }
1382
1383 return rc;
1384}
1385
1386/**
1387 * Updates this host driver's internal status, according to the global, overall input/output
1388 * state and all connected (native) audio streams.
1389 *
1390 * @param pThis Host audio driver instance.
1391 * @param pCfg Where to store the backend configuration. Optional.
1392 * @param fEnum Enumeration flags.
1393 */
1394void dsoundUpdateStatusInternalEx(PDRVHOSTDSOUND pThis, PPDMAUDIOBACKENDCFG pCfg, uint32_t fEnum)
1395{
1396 AssertPtrReturnVoid(pThis);
1397 /* pCfg is optional. */
1398
1399 PDMAUDIOBACKENDCFG Cfg;
1400 RT_ZERO(Cfg);
1401
1402 Cfg.cbStreamOut = sizeof(DSOUNDSTREAMOUT);
1403 Cfg.cbStreamIn = sizeof(DSOUNDSTREAMIN);
1404
1405 DSOUNDENUMCBCTX cbCtx = { pThis, fEnum, 0, 0 };
1406
1407 int rc = dsoundDevicesEnumerate(pThis, &cbCtx, fEnum);
1408 if (RT_SUCCESS(rc))
1409 {
1410#ifdef VBOX_WITH_AUDIO_CALLBACKS
1411 if ( pThis->fEnabledOut != RT_BOOL(cbCtx.cDevOut)
1412 || pThis->fEnabledIn != RT_BOOL(cbCtx.cDevIn))
1413 {
1414 /** @todo Use a registered callback to the audio connector (e.g "OnConfigurationChanged") to
1415 * let the connector know that something has changed within the host backend. */
1416 }
1417#else
1418 pThis->fEnabledOut = RT_BOOL(cbCtx.cDevOut);
1419 pThis->fEnabledIn = RT_BOOL(cbCtx.cDevIn);
1420#endif
1421
1422 Cfg.cSources = cbCtx.cDevIn;
1423 Cfg.cSinks = cbCtx.cDevOut;
1424 Cfg.cMaxStreamsIn = UINT32_MAX;
1425 Cfg.cMaxStreamsOut = UINT32_MAX;
1426
1427 if (pCfg)
1428 memcpy(pCfg, &Cfg, sizeof(PDMAUDIOBACKENDCFG));
1429 }
1430
1431 LogFlowFuncLeaveRC(rc);
1432}
1433
1434void dsoundUpdateStatusInternal(PDRVHOSTDSOUND pThis)
1435{
1436 dsoundUpdateStatusInternalEx(pThis, NULL /* pCfg */, 0 /* fEnum */);
1437}
1438
1439static int dsoundCreateStreamOut(PPDMIHOSTAUDIO pInterface,
1440 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
1441{
1442 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1443 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1444 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1445 /* pcSamples is optional. */
1446
1447 LogFlowFunc(("pStream=%p, pCfg=%p\n", pStream, pCfg));
1448
1449 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
1450 PDSOUNDSTREAMOUT pDSoundStream = (PDSOUNDSTREAMOUT)pStream;
1451
1452 pDSoundStream->streamCfg = *pCfg;
1453 pDSoundStream->streamCfg.enmEndianness = PDMAUDIOHOSTENDIANNESS;
1454
1455 int rc = DrvAudioHlpStreamCfgToProps(&pDSoundStream->streamCfg, &pDSoundStream->Stream.Props);
1456 if (RT_SUCCESS(rc))
1457 {
1458 pDSoundStream->pDS = NULL;
1459 pDSoundStream->pDSB = NULL;
1460 pDSoundStream->cbPlayWritePos = 0;
1461 pDSoundStream->fRestartPlayback = true;
1462 pDSoundStream->csPlaybackBufferSize = 0;
1463
1464 if (pcSamples)
1465 *pcSamples = pThis->cfg.cbBufferOut >> pStream->Props.cShift;
1466
1467 /* Try to open playback in case the device is already there. */
1468 directSoundPlayOpen(pThis, pDSoundStream);
1469 }
1470 else
1471 {
1472 RT_ZERO(pDSoundStream->streamCfg);
1473 }
1474
1475 LogFlowFuncLeaveRC(rc);
1476 return rc;
1477}
1478
1479static int dsoundControlStreamOut(PPDMIHOSTAUDIO pInterface,
1480 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
1481{
1482 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1483 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1484
1485 LogFlowFunc(("pStream=%p, cmd=%d\n", pStream, enmStreamCmd));
1486
1487 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
1488 PDSOUNDSTREAMOUT pDSoundStream = (PDSOUNDSTREAMOUT)pStream;
1489
1490 int rc = VINF_SUCCESS;
1491
1492 HRESULT hr;
1493 switch (enmStreamCmd)
1494 {
1495 case PDMAUDIOSTREAMCMD_ENABLE:
1496 case PDMAUDIOSTREAMCMD_RESUME:
1497 {
1498 DSLOG(("DSound: Playback PDMAUDIOSTREAMCMD_ENABLE\n"));
1499 /* Try to start playback. If it fails, then reopen and try again. */
1500 hr = directSoundPlayStart(pThis, pDSoundStream);
1501 if (FAILED(hr))
1502 {
1503 hr = directSoundPlayClose(pThis, pDSoundStream);
1504 if (SUCCEEDED(hr))
1505 hr = directSoundPlayOpen(pThis, pDSoundStream);
1506 if (SUCCEEDED(hr))
1507 hr = directSoundPlayStart(pThis, pDSoundStream);
1508 }
1509
1510 if (FAILED(hr))
1511 rc = VERR_NOT_SUPPORTED;
1512 break;
1513 }
1514
1515 case PDMAUDIOSTREAMCMD_DISABLE:
1516 case PDMAUDIOSTREAMCMD_PAUSE:
1517 {
1518 DSLOG(("DSound: Playback PDMAUDIOSTREAMCMD_DISABLE\n"));
1519 hr = directSoundPlayStop(pThis, pDSoundStream);
1520 if (FAILED(hr))
1521 rc = VERR_NOT_SUPPORTED;
1522 break;
1523 }
1524
1525 default:
1526 {
1527 AssertMsgFailed(("Invalid command: %ld\n", enmStreamCmd));
1528 rc = VERR_INVALID_PARAMETER;
1529 break;
1530 }
1531 }
1532
1533 LogFlowFuncLeaveRC(rc);
1534 return rc;
1535}
1536
1537static DECLCALLBACK(int) drvHostDSoundStreamPlay(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
1538 uint32_t *pcSamplesPlayed)
1539{
1540 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1541 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1542 /* pcSamplesPlayed is optional. */
1543
1544 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
1545 PDSOUNDSTREAMOUT pDSoundStream = (PDSOUNDSTREAMOUT)pStream;
1546
1547 int rc = VINF_SUCCESS;
1548 uint32_t cReadTotal = 0;
1549
1550#ifdef DEBUG_andy
1551 LogFlowFuncEnter();
1552#endif
1553
1554 do /* to use 'break' */
1555 {
1556 DWORD cbBuffer, cbFree, cbPlayPos;
1557 rc = dsoundGetPosOut(pThis, pDSoundStream, &cbBuffer, &cbFree, &cbPlayPos);
1558 if (RT_FAILURE(rc))
1559 break;
1560
1561 /*
1562 * Check for full buffer, do not allow the cbPlayWritePos to catch cbPlayPos during playback,
1563 * i.e. always leave a free space for 1 audio sample.
1564 */
1565 const DWORD cbSample = AUDIOMIXBUF_S2B(&pStream->MixBuf, 1);
1566 if (cbFree <= cbSample)
1567 break;
1568 cbFree -= cbSample;
1569
1570 uint32_t cLive = AudioMixBufLive(&pStream->MixBuf);
1571 uint32_t cbLive = AUDIOMIXBUF_S2B(&pStream->MixBuf, cLive);
1572
1573 /* Do not write more than available space in the DirectSound playback buffer. */
1574 cbLive = RT_MIN(cbFree, cbLive);
1575
1576 cbLive &= ~pStream->Props.uAlign;
1577 if (cbLive == 0 || cbLive > cbBuffer)
1578 {
1579 DSLOG(("DSound: cbLive=%RU32, cbBuffer=%ld, cbPlayWritePos=%ld, cbPlayPos=%ld\n",
1580 cbLive, cbBuffer, pDSoundStream->cbPlayWritePos, cbPlayPos));
1581 break;
1582 }
1583
1584 LPDIRECTSOUNDBUFFER8 pDSB = pDSoundStream->pDSB;
1585 AssertPtr(pDSB);
1586
1587 LPVOID pv1, pv2;
1588 DWORD cb1, cb2;
1589 HRESULT hr = directSoundPlayLock(pThis, pDSB, &pStream->Props, pDSoundStream->cbPlayWritePos, cbLive,
1590 &pv1, &pv2, &cb1, &cb2, 0 /* dwFlags */);
1591 if (FAILED(hr))
1592 {
1593 rc = VERR_ACCESS_DENIED;
1594 break;
1595 }
1596
1597 DWORD len1 = AUDIOMIXBUF_B2S(&pStream->MixBuf, cb1);
1598 DWORD len2 = AUDIOMIXBUF_B2S(&pStream->MixBuf, cb2);
1599
1600 uint32_t cRead = 0;
1601
1602 if (pv1 && cb1)
1603 {
1604 rc = AudioMixBufReadCirc(&pStream->MixBuf, pv1, cb1, &cRead);
1605 if (RT_SUCCESS(rc))
1606 cReadTotal += cRead;
1607 }
1608
1609 if ( RT_SUCCESS(rc)
1610 && cReadTotal == len1
1611 && pv2 && cb2)
1612 {
1613 rc = AudioMixBufReadCirc(&pStream->MixBuf, pv2, cb2, &cRead);
1614 if (RT_SUCCESS(rc))
1615 cReadTotal += cRead;
1616 }
1617
1618 directSoundPlayUnlock(pThis, pDSB, pv1, pv2, cb1, cb2);
1619
1620 pDSoundStream->cbPlayWritePos =
1621 (pDSoundStream->cbPlayWritePos + AUDIOMIXBUF_S2B(&pStream->MixBuf, cReadTotal)) % cbBuffer;
1622
1623 DSLOGF(("DSound: %RU32 (%RU32 samples) out of %RU32%s, buffer write pos %ld, rc=%Rrc\n",
1624 AUDIOMIXBUF_S2B(&pStream->MixBuf, cReadTotal), cReadTotal, cbLive,
1625 cbLive != AUDIOMIXBUF_S2B(&pStream->MixBuf, cReadTotal) ? " !!!": "",
1626 pDSoundStream->cbPlayWritePos, rc));
1627
1628 if (cReadTotal)
1629 {
1630 AudioMixBufFinish(&pStream->MixBuf, cReadTotal);
1631 rc = VINF_SUCCESS; /* Played something. */
1632 }
1633
1634 if (RT_FAILURE(rc))
1635 break;
1636
1637 if (pDSoundStream->fRestartPlayback)
1638 {
1639 /*
1640 * The playback has been just started.
1641 * Some samples of the new sound have been copied to the buffer
1642 * and it can start playing.
1643 */
1644 pDSoundStream->fRestartPlayback = false;
1645
1646 DWORD fFlags = 0;
1647#ifndef VBOX_WITH_AUDIO_CALLBACKS
1648 fFlags |= DSCBSTART_LOOPING;
1649#endif
1650 for (unsigned i = 0; i < DRV_DSOUND_RESTORE_ATTEMPTS_MAX; i++)
1651 {
1652 hr = IDirectSoundBuffer8_Play(pDSoundStream->pDSB, 0, 0, fFlags);
1653 if ( SUCCEEDED(hr)
1654 || hr != DSERR_BUFFERLOST)
1655 break;
1656 else
1657 {
1658 LogFlowFunc(("Restarting playback failed due to lost buffer, restoring ...\n"));
1659 directSoundPlayRestore(pThis, pDSoundStream->pDSB);
1660 }
1661 }
1662
1663 if (FAILED(hr))
1664 {
1665 DSLOGREL(("DSound: Starting playback failed with %Rhrc\n", hr));
1666 rc = VERR_NOT_SUPPORTED;
1667 break;
1668 }
1669 }
1670
1671 } while (0);
1672
1673 if (RT_FAILURE(rc))
1674 {
1675 dsoundUpdateStatusInternal(pThis);
1676 }
1677 else
1678 {
1679 if (pcSamplesPlayed)
1680 *pcSamplesPlayed = cReadTotal;
1681 }
1682
1683 LogFlowFuncLeaveRC(rc);
1684 return rc;
1685}
1686
1687static int dsoundDestroyStreamOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1688{
1689 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
1690 PDSOUNDSTREAMOUT pDSoundStream = (PDSOUNDSTREAMOUT)pStream;
1691
1692 directSoundPlayClose(pThis, pDSoundStream);
1693
1694 pDSoundStream->cbPlayWritePos = 0;
1695 pDSoundStream->fRestartPlayback = true;
1696 pDSoundStream->csPlaybackBufferSize = 0;
1697
1698 RT_ZERO(pDSoundStream->streamCfg);
1699
1700 return VINF_SUCCESS;
1701}
1702
1703static int dsoundCreateStreamIn(PPDMIHOSTAUDIO pInterface,
1704 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
1705{
1706 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1707 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1708 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1709
1710 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
1711 PDSOUNDSTREAMIN pDSoundStream = (PDSOUNDSTREAMIN)pStream;
1712
1713 LogFlowFunc(("pStream=%p, pCfg=%p, enmRecSource=%ld\n", pStream, pCfg, pCfg->DestSource.Source));
1714
1715 memcpy(&pDSoundStream->streamCfg, pCfg, sizeof(PDMAUDIOSTREAMCFG));
1716
1717 /** @todo caller should already init Props? */
1718 int rc = DrvAudioHlpStreamCfgToProps(&pDSoundStream->streamCfg, &pStream->Props);
1719 if (RT_SUCCESS(rc))
1720 {
1721 /* Init the stream structure and save relevant information to it. */
1722 pDSoundStream->csCaptureReadPos = 0;
1723 pDSoundStream->csCaptureBufferSize = 0;
1724 pDSoundStream->pDSC = NULL;
1725 pDSoundStream->pDSCB = NULL;
1726 pDSoundStream->enmRecSource = pCfg->DestSource.Source;
1727 pDSoundStream->hrLastCapture = S_OK;
1728
1729 if (pcSamples)
1730 *pcSamples = pThis->cfg.cbBufferIn >> pStream->Props.cShift;
1731
1732 /* Try to open capture in case the device is already there. */
1733 directSoundCaptureOpen(pThis, pDSoundStream);
1734 }
1735 else
1736 {
1737 RT_ZERO(pDSoundStream->streamCfg);
1738 }
1739
1740 LogFlowFuncLeaveRC(rc);
1741 return rc;
1742}
1743
1744static int dsoundControlStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
1745 PDMAUDIOSTREAMCMD enmStreamCmd)
1746{
1747 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1748 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1749
1750 LogFlowFunc(("pStream=%p, enmStreamCmd=%ld\n", pStream, enmStreamCmd));
1751
1752 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
1753 PDSOUNDSTREAMIN pDSoundStream = (PDSOUNDSTREAMIN)pStream;
1754
1755 int rc = VINF_SUCCESS;
1756
1757 HRESULT hr;
1758 switch (enmStreamCmd)
1759 {
1760 case PDMAUDIOSTREAMCMD_ENABLE:
1761 case PDMAUDIOSTREAMCMD_RESUME:
1762 {
1763 /* Try to start capture. If it fails, then reopen and try again. */
1764 hr = directSoundCaptureStart(pThis, pDSoundStream);
1765 if (FAILED(hr))
1766 {
1767 hr = directSoundCaptureClose(pDSoundStream);
1768 if (SUCCEEDED(hr))
1769 {
1770 hr = directSoundCaptureOpen(pThis, pDSoundStream);
1771 if (SUCCEEDED(hr))
1772 hr = directSoundCaptureStart(pThis, pDSoundStream);
1773 }
1774 }
1775
1776 if (FAILED(hr))
1777 rc = VERR_NOT_SUPPORTED;
1778 break;
1779 }
1780
1781 case PDMAUDIOSTREAMCMD_DISABLE:
1782 case PDMAUDIOSTREAMCMD_PAUSE:
1783 {
1784 hr = directSoundCaptureStop(pThis, pDSoundStream);
1785 if (FAILED(hr))
1786 rc = VERR_NOT_SUPPORTED;
1787 break;
1788 }
1789
1790 default:
1791 {
1792 AssertMsgFailed(("Invalid command %ld\n", enmStreamCmd));
1793 rc = VERR_INVALID_PARAMETER;
1794 break;
1795 }
1796 }
1797
1798 return rc;
1799}
1800
1801static DECLCALLBACK(int) drvHostDSoundStreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
1802 uint32_t *pcSamplesCaptured)
1803{
1804 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
1805
1806 PDSOUNDSTREAMIN pDSoundStream = (PDSOUNDSTREAMIN)pStream;
1807 LPDIRECTSOUNDCAPTUREBUFFER8 pDSCB = pDSoundStream->pDSCB;
1808
1809 int rc = VINF_SUCCESS;
1810
1811 uint32_t cCaptured = 0;
1812
1813 do
1814 {
1815 if (pDSCB == NULL)
1816 {
1817 rc = VERR_NOT_AVAILABLE;
1818 break;
1819 }
1820
1821 /* Get DirectSound capture position in bytes. */
1822 DWORD cbReadPos;
1823 HRESULT hr = IDirectSoundCaptureBuffer_GetCurrentPosition(pDSCB, NULL, &cbReadPos);
1824 if (FAILED(hr))
1825 {
1826 if (hr != pDSoundStream->hrLastCapture)
1827 {
1828 DSLOGREL(("DSound: Getting capture position failed with %Rhrc\n", hr));
1829 pDSoundStream->hrLastCapture = hr;
1830 }
1831
1832 rc = VERR_NOT_AVAILABLE;
1833 break;
1834 }
1835
1836 pDSoundStream->hrLastCapture = hr;
1837
1838 if (cbReadPos & pStream->Props.uAlign)
1839 DSLOGF(("DSound: Misaligned capture read position %ld (alignment: %RU32)\n", cbReadPos, pStream->Props.uAlign));
1840
1841 /* Capture position in samples. */
1842 DWORD csReadPos = cbReadPos >> pStream->Props.cShift;
1843
1844 /* Number of samples available in the DirectSound capture buffer. */
1845 DWORD csCaptured = dsoundRingDistance(csReadPos, pDSoundStream->csCaptureReadPos, pDSoundStream->csCaptureBufferSize);
1846 if (csCaptured == 0)
1847 break;
1848
1849 /* Get number of free samples in the mix buffer and check that is has free space */
1850 uint32_t csMixFree = AudioMixBufFree(&pStream->MixBuf);
1851 if (csMixFree == 0)
1852 {
1853 DSLOGF(("DSound: Capture buffer full\n"));
1854 break;
1855 }
1856
1857 DSLOGF(("DSound: Capture csMixFree=%RU32, csReadPos=%ld, csCaptureReadPos=%ld, csCaptured=%ld\n",
1858 csMixFree, csReadPos, pDSoundStream->csCaptureReadPos, csCaptured));
1859
1860 /* No need to fetch more samples than mix buffer can receive. */
1861 csCaptured = RT_MIN(csCaptured, csMixFree);
1862
1863 /* Lock relevant range in the DirectSound capture buffer. */
1864 LPVOID pv1, pv2;
1865 DWORD cb1, cb2;
1866 hr = directSoundCaptureLock(pDSCB, &pStream->Props,
1867 AUDIOMIXBUF_S2B(&pStream->MixBuf, pDSoundStream->csCaptureReadPos), /* dwOffset */
1868 AUDIOMIXBUF_S2B(&pStream->MixBuf, csCaptured), /* dwBytes */
1869 &pv1, &pv2, &cb1, &cb2,
1870 0 /* dwFlags */);
1871 if (FAILED(hr))
1872 {
1873 rc = VERR_ACCESS_DENIED;
1874 break;
1875 }
1876
1877 DWORD len1 = AUDIOMIXBUF_B2S(&pStream->MixBuf, cb1);
1878 DWORD len2 = AUDIOMIXBUF_B2S(&pStream->MixBuf, cb2);
1879
1880 uint32_t csWrittenTotal = 0;
1881 uint32_t csWritten;
1882 if (pv1 && len1)
1883 {
1884 rc = AudioMixBufWriteCirc(&pStream->MixBuf, pv1, cb1, &csWritten);
1885 if (RT_SUCCESS(rc))
1886 csWrittenTotal += csWritten;
1887 }
1888
1889 if ( RT_SUCCESS(rc)
1890 && csWrittenTotal == len1
1891 && pv2 && len2)
1892 {
1893 rc = AudioMixBufWriteCirc(&pStream->MixBuf, pv2, cb2, &csWritten);
1894 if (RT_SUCCESS(rc))
1895 csWrittenTotal += csWritten;
1896 }
1897
1898 directSoundCaptureUnlock(pDSCB, pv1, pv2, cb1, cb2);
1899
1900 if (csWrittenTotal) /* Captured something? */
1901 rc = AudioMixBufMixToParent(&pStream->MixBuf, csWrittenTotal, &cCaptured);
1902
1903 if (RT_SUCCESS(rc))
1904 {
1905 pDSoundStream->csCaptureReadPos = (pDSoundStream->csCaptureReadPos + cCaptured) % pDSoundStream->csCaptureBufferSize;
1906 DSLOGF(("DSound: Capture %ld (%ld+%ld), processed %RU32/%RU32\n",
1907 csCaptured, len1, len2, cCaptured, csWrittenTotal));
1908 }
1909
1910 } while (0);
1911
1912 if (RT_FAILURE(rc))
1913 {
1914 dsoundUpdateStatusInternal(pThis);
1915 }
1916 else
1917 {
1918 if (pcSamplesCaptured)
1919 *pcSamplesCaptured = cCaptured;
1920 }
1921
1922 LogFlowFuncLeaveRC(rc);
1923 return rc;
1924}
1925
1926static int dsoundDestroyStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1927{
1928 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
1929 PDSOUNDSTREAMIN pDSoundStream = (PDSOUNDSTREAMIN)pStream;
1930
1931 directSoundCaptureClose(pDSoundStream);
1932
1933 pDSoundStream->csCaptureReadPos = 0;
1934 pDSoundStream->csCaptureBufferSize = 0;
1935 RT_ZERO(pDSoundStream->streamCfg);
1936
1937 return VINF_SUCCESS;
1938}
1939
1940static DECLCALLBACK(int) drvHostDSoundGetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pCfg)
1941{
1942 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1943 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1944
1945 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
1946
1947 dsoundUpdateStatusInternalEx(pThis, pCfg, 0 /* fEnum */);
1948
1949 return VINF_SUCCESS;
1950}
1951
1952#ifdef VBOX_WITH_AUDIO_CALLBACKS
1953static int dsoundNotifyThread(PDRVHOSTDSOUND pThis, bool fShutdown)
1954{
1955 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1956
1957 if (fShutdown)
1958 {
1959 LogFlowFunc(("Shutting down thread ...\n"));
1960 pThis->fShutdown = fShutdown;
1961 }
1962
1963 /* Set the notification event so that the thread is being notified. */
1964 BOOL fRc = SetEvent(pThis->aEvents[DSOUNDEVENT_NOTIFY]);
1965 Assert(fRc);
1966
1967 return VINF_SUCCESS;
1968}
1969
1970static DECLCALLBACK(int) dsoundNotificationThread(RTTHREAD hThreadSelf, void *pvUser)
1971{
1972 PDRVHOSTDSOUND pThis = (PDRVHOSTDSOUND)pvUser;
1973 AssertPtr(pThis);
1974
1975 LogFlowFuncEnter();
1976
1977 /* Let caller know that we're done initializing, regardless of the result. */
1978 int rc = RTThreadUserSignal(hThreadSelf);
1979 AssertRC(rc);
1980
1981 do
1982 {
1983 HANDLE aEvents[VBOX_DSOUND_MAX_EVENTS];
1984 DWORD cEvents = 0;
1985 for (uint8_t i = 0; i < VBOX_DSOUND_MAX_EVENTS; i++)
1986 {
1987 if (pThis->aEvents[i])
1988 aEvents[cEvents++] = pThis->aEvents[i];
1989 }
1990 Assert(cEvents);
1991
1992 LogFlowFunc(("Waiting: cEvents=%ld\n", cEvents));
1993
1994 DWORD dwObj = WaitForMultipleObjects(cEvents, aEvents, FALSE /* bWaitAll */, INFINITE);
1995 switch (dwObj)
1996 {
1997 case WAIT_FAILED:
1998 {
1999 rc = VERR_CANCELLED;
2000 break;
2001 }
2002
2003 case WAIT_TIMEOUT:
2004 {
2005 rc = VERR_TIMEOUT;
2006 break;
2007 }
2008
2009 default:
2010 {
2011 dwObj = WAIT_OBJECT_0 + cEvents - 1;
2012 if (aEvents[dwObj] == pThis->aEvents[DSOUNDEVENT_NOTIFY])
2013 {
2014 LogFlowFunc(("Notify\n"));
2015 }
2016 else if (aEvents[dwObj] == pThis->aEvents[DSOUNDEVENT_INPUT])
2017 {
2018
2019 }
2020 else if (aEvents[dwObj] == pThis->aEvents[DSOUNDEVENT_OUTPUT])
2021 {
2022 DWORD cbBuffer, cbFree, cbPlayPos;
2023 rc = dsoundGetPosOut(pThis->pDSStream, &cbBuffer, &cbFree, &cbPlayPos);
2024 if ( RT_SUCCESS(rc)
2025 && cbFree)
2026 {
2027 PDMAUDIOCALLBACKDATAOUT Out;
2028 Out.cbInFree = cbFree;
2029 Out.cbOutWritten = 0;
2030
2031 while (!Out.cbOutWritten)
2032 {
2033 rc = pThis->pUpIAudioConnector->pfnCallback(pThis->pUpIAudioConnector,
2034 PDMAUDIOCALLBACKTYPE_OUTPUT, &Out, sizeof(Out));
2035 if (RT_FAILURE(rc))
2036 break;
2037 RTThreadSleep(100);
2038 }
2039
2040 LogFlowFunc(("Output: cbBuffer=%ld, cbFree=%ld, cbPlayPos=%ld, cbWritten=%RU32, rc=%Rrc\n",
2041 cbBuffer, cbFree, cbPlayPos, Out.cbOutWritten, rc));
2042 }
2043 }
2044 break;
2045 }
2046 }
2047
2048 if (pThis->fShutdown)
2049 break;
2050
2051 } while (RT_SUCCESS(rc));
2052
2053 pThis->fStopped = true;
2054
2055 LogFlowFunc(("Exited with fShutdown=%RTbool, rc=%Rrc\n", pThis->fShutdown, rc));
2056 return rc;
2057}
2058#endif /* VBOX_WITH_AUDIO_CALLBACKS */
2059
2060static DECLCALLBACK(void) drvHostDSoundShutdown(PPDMIHOSTAUDIO pInterface)
2061{
2062 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
2063
2064 LogFlowFuncEnter();
2065
2066#ifdef VBOX_WITH_AUDIO_CALLBACKS
2067 int rc = dsoundNotifyThread(pThis, true /* fShutdown */);
2068 AssertRC(rc);
2069
2070 int rcThread;
2071 rc = RTThreadWait(pThis->Thread, 15 * 1000 /* 15s timeout */, &rcThread);
2072 LogFlowFunc(("rc=%Rrc, rcThread=%Rrc\n", rc, rcThread));
2073
2074 Assert(pThis->fStopped);
2075
2076 if (pThis->aEvents[DSOUNDEVENT_NOTIFY])
2077 {
2078 CloseHandle(pThis->aEvents[DSOUNDEVENT_NOTIFY]);
2079 pThis->aEvents[DSOUNDEVENT_NOTIFY] = NULL;
2080}
2081#endif
2082
2083 LogFlowFuncLeave();
2084}
2085
2086static DECLCALLBACK(int) drvHostDSoundInit(PPDMIHOSTAUDIO pInterface)
2087{
2088 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
2089
2090 LogFlowFuncEnter();
2091
2092 int rc;
2093
2094 /* Verify that IDirectSound is available. */
2095 LPDIRECTSOUND pDirectSound = NULL;
2096 HRESULT hr = CoCreateInstance(CLSID_DirectSound, NULL, CLSCTX_ALL,
2097 IID_IDirectSound, (void **)&pDirectSound);
2098 if (SUCCEEDED(hr))
2099 {
2100 IDirectSound_Release(pDirectSound);
2101
2102#ifdef VBOX_WITH_AUDIO_CALLBACKS
2103 /* Create notification event. */
2104 pThis->aEvents[DSOUNDEVENT_NOTIFY] = CreateEvent(NULL /* Security attribute */,
2105 FALSE /* bManualReset */, FALSE /* bInitialState */,
2106 NULL /* lpName */);
2107 Assert(pThis->aEvents[DSOUNDEVENT_NOTIFY] != NULL);
2108
2109 /* Start notification thread. */
2110 rc = RTThreadCreate(&pThis->Thread, dsoundNotificationThread,
2111 pThis /*pvUser*/, 0 /*cbStack*/,
2112 RTTHREADTYPE_DEFAULT, RTTHREADFLAGS_WAITABLE, "dsoundNtfy");
2113 if (RT_SUCCESS(rc))
2114 {
2115 /* Wait for the thread to initialize. */
2116 rc = RTThreadUserWait(pThis->Thread, RT_MS_1MIN);
2117 if (RT_FAILURE(rc))
2118 DSLOGREL(("DSound: Waiting for thread to initialize failed with rc=%Rrc\n", rc));
2119 }
2120 else
2121 DSLOGREL(("DSound: Creating thread failed with rc=%Rrc\n", rc));
2122#else
2123 rc = VINF_SUCCESS;
2124#endif
2125
2126 dsoundUpdateStatusInternalEx(pThis, NULL /* pCfg */, DSOUNDENUMCBFLAGS_LOG /* fEnum */);
2127 }
2128 else
2129 {
2130 DSLOGREL(("DSound: DirectSound not available: %Rhrc\n", hr));
2131 rc = VERR_NOT_SUPPORTED;
2132 }
2133
2134 LogFlowFuncLeaveRC(rc);
2135 return rc;
2136}
2137
2138static DECLCALLBACK(void *) drvHostDSoundQueryInterface(PPDMIBASE pInterface, const char *pszIID)
2139{
2140 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
2141 PDRVHOSTDSOUND pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTDSOUND);
2142
2143 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
2144 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
2145 return NULL;
2146}
2147
2148static LPCGUID dsoundConfigQueryGUID(PCFGMNODE pCfg, const char *pszName, RTUUID *pUuid)
2149{
2150 LPCGUID pGuid = NULL;
2151
2152 char *pszGuid = NULL;
2153 int rc = CFGMR3QueryStringAlloc(pCfg, pszName, &pszGuid);
2154 if (RT_SUCCESS(rc))
2155 {
2156 rc = RTUuidFromStr(pUuid, pszGuid);
2157 if (RT_SUCCESS(rc))
2158 pGuid = (LPCGUID)&pUuid;
2159 else
2160 DSLOGREL(("DSound: Error parsing device GUID for device '%s': %Rrc\n", pszName, rc));
2161
2162 RTStrFree(pszGuid);
2163 }
2164
2165 return pGuid;
2166}
2167
2168static void dsoundConfigInit(PDRVHOSTDSOUND pThis, PCFGMNODE pCfg)
2169{
2170 unsigned int uBufsizeOut, uBufsizeIn;
2171
2172 CFGMR3QueryUIntDef(pCfg, "BufsizeOut", &uBufsizeOut, _16K);
2173 CFGMR3QueryUIntDef(pCfg, "BufsizeIn", &uBufsizeIn, _16K);
2174 pThis->cfg.cbBufferOut = uBufsizeOut;
2175 pThis->cfg.cbBufferIn = uBufsizeIn;
2176
2177 pThis->cfg.pGuidPlay = dsoundConfigQueryGUID(pCfg, "DeviceGuidOut", &pThis->cfg.uuidPlay);
2178 pThis->cfg.pGuidCapture = dsoundConfigQueryGUID(pCfg, "DeviceGuidIn", &pThis->cfg.uuidCapture);
2179
2180 DSLOG(("DSound: BufsizeOut %u, BufsizeIn %u, DeviceGuidOut {%RTuuid}, DeviceGuidIn {%RTuuid}\n",
2181 pThis->cfg.cbBufferOut,
2182 pThis->cfg.cbBufferIn,
2183 &pThis->cfg.uuidPlay,
2184 &pThis->cfg.uuidCapture));
2185}
2186
2187/*
2188 * PDMIHOSTAUDIO
2189 */
2190
2191static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvHostDSoundGetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
2192{
2193 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
2194
2195 return PDMAUDIOBACKENDSTS_RUNNING;
2196}
2197
2198static DECLCALLBACK(int) drvHostDSoundStreamCreate(PPDMIHOSTAUDIO pInterface,
2199 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
2200{
2201 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2202 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2203 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
2204
2205 int rc;
2206 if (pCfg->enmDir == PDMAUDIODIR_IN)
2207 rc = dsoundCreateStreamIn(pInterface, pStream, pCfg, pcSamples);
2208 else
2209 rc = dsoundCreateStreamOut(pInterface, pStream, pCfg, pcSamples);
2210
2211 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
2212 return rc;
2213}
2214
2215static DECLCALLBACK(int) drvHostDSoundStreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
2216{
2217 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2218 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2219
2220 int rc;
2221 if (pStream->enmDir == PDMAUDIODIR_IN)
2222 rc = dsoundDestroyStreamIn(pInterface, pStream);
2223 else
2224 rc = dsoundDestroyStreamOut(pInterface, pStream);
2225
2226 return rc;
2227}
2228
2229static DECLCALLBACK(int) drvHostDSoundStreamControl(PPDMIHOSTAUDIO pInterface,
2230 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
2231{
2232 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2233 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2234
2235 Assert(pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
2236
2237 int rc;
2238 if (pStream->enmDir == PDMAUDIODIR_IN)
2239 rc = dsoundControlStreamIn(pInterface, pStream, enmStreamCmd);
2240 else
2241 rc = dsoundControlStreamOut(pInterface, pStream, enmStreamCmd);
2242
2243 return rc;
2244}
2245
2246static DECLCALLBACK(PDMAUDIOSTRMSTS) drvHostDSoundStreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
2247{
2248 AssertPtrReturn(pInterface, PDMAUDIOSTRMSTS_FLAG_NONE);
2249 AssertPtrReturn(pStream, PDMAUDIOSTRMSTS_FLAG_NONE);
2250
2251 PDRVHOSTDSOUND pThis = PDMIHOSTAUDIO_2_DRVHOSTDSOUND(pInterface);
2252
2253 PDMAUDIOSTRMSTS strmSts = PDMAUDIOSTRMSTS_FLAG_INITIALIZED;
2254 if (pStream->enmDir == PDMAUDIODIR_IN)
2255 {
2256 PDSOUNDSTREAMIN pDSoundStream = (PDSOUNDSTREAMIN)pStream;
2257 if (pDSoundStream->fEnabled)
2258 strmSts |= PDMAUDIOSTRMSTS_FLAG_ENABLED | PDMAUDIOSTRMSTS_FLAG_DATA_READABLE;
2259 }
2260 else
2261 {
2262 PDSOUNDSTREAMOUT pDSoundStream = (PDSOUNDSTREAMOUT)pStream;
2263 if (pDSoundStream->fEnabled)
2264 {
2265 strmSts |= PDMAUDIOSTRMSTS_FLAG_ENABLED;
2266
2267 DWORD cbFree;
2268 int rc = dsoundGetPosOut(pThis, pDSoundStream, NULL /* cbBuffer */, &cbFree, NULL /* cbPlayPos */);
2269 if ( RT_SUCCESS(rc)
2270 && cbFree)
2271 {
2272 LogFlowFunc(("cbFree=%ld\n", cbFree));
2273 strmSts |= PDMAUDIOSTRMSTS_FLAG_DATA_WRITABLE;
2274 }
2275 }
2276 }
2277
2278 return strmSts;
2279}
2280
2281static DECLCALLBACK(int) drvHostDSoundStreamIterate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
2282{
2283 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
2284 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2285
2286 LogFlowFuncEnter();
2287
2288 /* Nothing to do here for DSound. */
2289 return VINF_SUCCESS;
2290}
2291
2292static DECLCALLBACK(void) drvHostDSoundDestruct(PPDMDRVINS pDrvIns)
2293{
2294 PDRVHOSTDSOUND pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTDSOUND);
2295 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
2296 LogFlowFuncEnter();
2297
2298 if (pThis->pDrvIns)
2299 CoUninitialize();
2300
2301 LogFlowFuncLeave();
2302}
2303
2304/**
2305 * Construct a DirectSound Audio driver instance.
2306 *
2307 * @copydoc FNPDMDRVCONSTRUCT
2308 */
2309static DECLCALLBACK(int) drvHostDSoundConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
2310{
2311 PDRVHOSTDSOUND pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTDSOUND);
2312
2313 LogRel(("Audio: Initializing DirectSound audio driver\n"));
2314
2315 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
2316
2317 HRESULT hr = CoInitializeEx(NULL, COINIT_MULTITHREADED);
2318 if (FAILED(hr))
2319 {
2320 DSLOGREL(("DSound: CoInitializeEx failed with %Rhrc\n", hr));
2321 return VERR_NOT_SUPPORTED;
2322 }
2323
2324 /*
2325 * Init basic data members and interfaces.
2326 */
2327 pThis->pDrvIns = pDrvIns;
2328 /* IBase */
2329 pDrvIns->IBase.pfnQueryInterface = drvHostDSoundQueryInterface;
2330 /* IHostAudio */
2331 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvHostDSound);
2332
2333#ifdef VBOX_WITH_AUDIO_CALLBACKS
2334 /*
2335 * Get the IAudioConnector interface of the above driver/device.
2336 */
2337 pThis->pUpIAudioConnector = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIAUDIOCONNECTOR);
2338 if (!pThis->pUpIAudioConnector)
2339 {
2340 AssertMsgFailed(("Configuration error: No audio connector interface above!\n"));
2341 return VERR_PDM_MISSING_INTERFACE_ABOVE;
2342 }
2343#endif
2344
2345 /*
2346 * Init the static parts.
2347 */
2348 RTListInit(&pThis->lstDevInput);
2349 RTListInit(&pThis->lstDevOutput);
2350
2351 pThis->fEnabledIn = false;
2352 pThis->fEnabledOut = false;
2353#ifdef VBOX_WITH_AUDIO_CALLBACKS
2354 pThis->fStopped = false;
2355 pThis->fShutdown = false;
2356
2357 RT_ZERO(pThis->aEvents);
2358 pThis->cEvents = 0;
2359#endif
2360
2361 /*
2362 * Initialize configuration values.
2363 */
2364 dsoundConfigInit(pThis, pCfg);
2365
2366 return VINF_SUCCESS;
2367}
2368
2369/**
2370 * PDM driver registration.
2371 */
2372const PDMDRVREG g_DrvHostDSound =
2373{
2374 /* u32Version */
2375 PDM_DRVREG_VERSION,
2376 /* szName */
2377 "DSoundAudio",
2378 /* szRCMod */
2379 "",
2380 /* szR0Mod */
2381 "",
2382 /* pszDescription */
2383 "DirectSound Audio host driver",
2384 /* fFlags */
2385 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
2386 /* fClass. */
2387 PDM_DRVREG_CLASS_AUDIO,
2388 /* cMaxInstances */
2389 ~0U,
2390 /* cbInstance */
2391 sizeof(DRVHOSTDSOUND),
2392 /* pfnConstruct */
2393 drvHostDSoundConstruct,
2394 /* pfnDestruct */
2395 drvHostDSoundDestruct,
2396 /* pfnRelocate */
2397 NULL,
2398 /* pfnIOCtl */
2399 NULL,
2400 /* pfnPowerOn */
2401 NULL,
2402 /* pfnReset */
2403 NULL,
2404 /* pfnSuspend */
2405 NULL,
2406 /* pfnResume */
2407 NULL,
2408 /* pfnAttach */
2409 NULL,
2410 /* pfnDetach */
2411 NULL,
2412 /* pfnPowerOff */
2413 NULL,
2414 /* pfnSoftReset */
2415 NULL,
2416 /* u32EndVersion */
2417 PDM_DRVREG_VERSION
2418};
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