VirtualBox

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

Last change on this file since 69304 was 69119, checked in by vboxsync, 7 years ago

Audio: More cleanups (missing keywords, incorrect #endif docs, stuff)

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