VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/AudioMixer.cpp@ 71191

Last change on this file since 71191 was 70640, checked in by vboxsync, 7 years ago

Audio: Added VERR_AUDIO_BACKEND_NOT_ATTACHED and support for it in PDMIAUDIOCONNECTOR::pfnGetStatus() and AudioMixerSinkCreateStream().

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 55.8 KB
Line 
1/* $Id: AudioMixer.cpp 70640 2018-01-19 12:14:12Z vboxsync $ */
2/** @file
3 * Audio mixing routines for multiplexing audio sources in device emulations.
4 *
5 * This mixer acts as a layer between the audio connector interface and
6 * the actual device emulation, providing mechanisms for audio sources (input)
7 * and audio sinks (output).
8 *
9 * Think of this mixer as kind of a high(er) level interface for the audio
10 * connector interface, abstracting common tasks such as creating and managing
11 * various audio sources and sinks. This mixer class is purely optional and can
12 * be left out when implementing a new device emulation, using only the audi
13 * connector interface instead. For example, the SB16 emulation does not use
14 * this mixer and does all its stream management on its own.
15 *
16 * As audio driver instances are handled as LUNs on the device level, this
17 * audio mixer then can take care of e.g. mixing various inputs/outputs to/from
18 * a specific source/sink.
19 *
20 * How and which audio streams are connected to sinks/sources depends on how
21 * the audio mixer has been set up.
22 *
23 * A sink can connect multiple output streams together, whereas a source
24 * does this with input streams. Each sink / source consists of one or more
25 * so-called mixer streams, which then in turn have pointers to the actual
26 * PDM audio input/output streams.
27 */
28
29/*
30 * Copyright (C) 2014-2017 Oracle Corporation
31 *
32 * This file is part of VirtualBox Open Source Edition (OSE), as
33 * available from http://www.virtualbox.org. This file is free software;
34 * you can redistribute it and/or modify it under the terms of the GNU
35 * General Public License (GPL) as published by the Free Software
36 * Foundation, in version 2 as it comes in the "COPYING" file of the
37 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
38 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
39 */
40
41
42/*********************************************************************************************************************************
43* Header Files *
44*********************************************************************************************************************************/
45#define LOG_GROUP LOG_GROUP_AUDIO_MIXER
46#include <VBox/log.h>
47#include "AudioMixer.h"
48#include "AudioMixBuffer.h"
49#include "DrvAudio.h"
50
51#include <VBox/vmm/pdm.h>
52#include <VBox/err.h>
53#include <VBox/vmm/mm.h>
54#include <VBox/vmm/pdmaudioifs.h>
55
56#include <iprt/alloc.h>
57#include <iprt/asm-math.h>
58#include <iprt/assert.h>
59#include <iprt/string.h>
60
61
62/*********************************************************************************************************************************
63* Internal Functions *
64*********************************************************************************************************************************/
65static int audioMixerRemoveSinkInternal(PAUDIOMIXER pMixer, PAUDMIXSINK pSink);
66
67static void audioMixerSinkDestroyInternal(PAUDMIXSINK pSink);
68static int audioMixerSinkUpdateVolume(PAUDMIXSINK pSink, const PPDMAUDIOVOLUME pVolMaster);
69static void audioMixerSinkRemoveAllStreamsInternal(PAUDMIXSINK pSink);
70static int audioMixerSinkRemoveStreamInternal(PAUDMIXSINK pSink, PAUDMIXSTREAM pStream);
71static void audioMixerSinkReset(PAUDMIXSINK pSink);
72static int audioMixerSinkUpdateInternal(PAUDMIXSINK pSink);
73
74int audioMixerStreamCtlInternal(PAUDMIXSTREAM pMixStream, PDMAUDIOSTREAMCMD enmCmd, uint32_t fCtl);
75static void audioMixerStreamDestroyInternal(PAUDMIXSTREAM pStream);
76
77
78#ifdef LOG_ENABLED
79/**
80 * Converts a mixer sink status to a string.
81 *
82 * @returns Stringified mixer sink flags. Must be free'd with RTStrFree().
83 * "NONE" if no flags set.
84 * @param fFlags Mixer sink flags to convert.
85 */
86static char *dbgAudioMixerSinkStatusToStr(AUDMIXSINKSTS fStatus)
87{
88#define APPEND_FLAG_TO_STR(_aFlag) \
89 if (fStatus & AUDMIXSINK_STS_##_aFlag) \
90 { \
91 if (pszFlags) \
92 { \
93 rc2 = RTStrAAppend(&pszFlags, " "); \
94 if (RT_FAILURE(rc2)) \
95 break; \
96 } \
97 \
98 rc2 = RTStrAAppend(&pszFlags, #_aFlag); \
99 if (RT_FAILURE(rc2)) \
100 break; \
101 } \
102
103 char *pszFlags = NULL;
104 int rc2 = VINF_SUCCESS;
105
106 do
107 {
108 APPEND_FLAG_TO_STR(NONE);
109 APPEND_FLAG_TO_STR(RUNNING);
110 APPEND_FLAG_TO_STR(PENDING_DISABLE);
111 APPEND_FLAG_TO_STR(DIRTY);
112
113 } while (0);
114
115 if ( RT_FAILURE(rc2)
116 && pszFlags)
117 {
118 RTStrFree(pszFlags);
119 pszFlags = NULL;
120 }
121
122#undef APPEND_FLAG_TO_STR
123
124 return pszFlags;
125}
126#endif /* DEBUG */
127
128/**
129 * Creates an audio sink and attaches it to the given mixer.
130 *
131 * @returns IPRT status code.
132 * @param pMixer Mixer to attach created sink to.
133 * @param pszName Name of the sink to create.
134 * @param enmDir Direction of the sink to create.
135 * @param ppSink Pointer which returns the created sink on success.
136 */
137int AudioMixerCreateSink(PAUDIOMIXER pMixer, const char *pszName, AUDMIXSINKDIR enmDir, PAUDMIXSINK *ppSink)
138{
139 AssertPtrReturn(pMixer, VERR_INVALID_POINTER);
140 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
141 /* ppSink is optional. */
142
143 int rc = RTCritSectEnter(&pMixer->CritSect);
144 if (RT_FAILURE(rc))
145 return rc;
146
147 PAUDMIXSINK pSink = (PAUDMIXSINK)RTMemAllocZ(sizeof(AUDMIXSINK));
148 if (pSink)
149 {
150 pSink->pszName = RTStrDup(pszName);
151 if (!pSink->pszName)
152 rc = VERR_NO_MEMORY;
153
154 if (RT_SUCCESS(rc))
155 rc = RTCritSectInit(&pSink->CritSect);
156
157 if (RT_SUCCESS(rc))
158 {
159 pSink->pParent = pMixer;
160 pSink->enmDir = enmDir;
161 RTListInit(&pSink->lstStreams);
162
163 /* Set initial volume to max. */
164 pSink->Volume.fMuted = false;
165 pSink->Volume.uLeft = PDMAUDIO_VOLUME_MAX;
166 pSink->Volume.uRight = PDMAUDIO_VOLUME_MAX;
167
168 /* Ditto for the combined volume. */
169 pSink->VolumeCombined.fMuted = false;
170 pSink->VolumeCombined.uLeft = PDMAUDIO_VOLUME_MAX;
171 pSink->VolumeCombined.uRight = PDMAUDIO_VOLUME_MAX;
172
173 RTListAppend(&pMixer->lstSinks, &pSink->Node);
174 pMixer->cSinks++;
175
176 LogFlowFunc(("pMixer=%p, pSink=%p, cSinks=%RU8\n",
177 pMixer, pSink, pMixer->cSinks));
178
179 if (ppSink)
180 *ppSink = pSink;
181 }
182
183 if (RT_FAILURE(rc))
184 {
185 RTCritSectDelete(&pSink->CritSect);
186
187 if (pSink)
188 {
189 RTMemFree(pSink);
190 pSink = NULL;
191 }
192 }
193 }
194 else
195 rc = VERR_NO_MEMORY;
196
197 int rc2 = RTCritSectLeave(&pMixer->CritSect);
198 AssertRC(rc2);
199
200 return rc;
201}
202
203/**
204 * Creates an audio mixer.
205 *
206 * @returns IPRT status code.
207 * @param pszName Name of the audio mixer.
208 * @param fFlags Creation flags. Not used at the moment and must be 0.
209 * @param ppMixer Pointer which returns the created mixer object.
210 */
211int AudioMixerCreate(const char *pszName, uint32_t fFlags, PAUDIOMIXER *ppMixer)
212{
213 RT_NOREF(fFlags);
214 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
215 /** @todo Add fFlags validation. */
216 AssertPtrReturn(ppMixer, VERR_INVALID_POINTER);
217
218 int rc = VINF_SUCCESS;
219
220 PAUDIOMIXER pMixer = (PAUDIOMIXER)RTMemAllocZ(sizeof(AUDIOMIXER));
221 if (pMixer)
222 {
223 pMixer->pszName = RTStrDup(pszName);
224 if (!pMixer->pszName)
225 rc = VERR_NO_MEMORY;
226
227 if (RT_SUCCESS(rc))
228 rc = RTCritSectInit(&pMixer->CritSect);
229
230 if (RT_SUCCESS(rc))
231 {
232 pMixer->cSinks = 0;
233 RTListInit(&pMixer->lstSinks);
234
235 /* Set master volume to the max. */
236 pMixer->VolMaster.fMuted = false;
237 pMixer->VolMaster.uLeft = PDMAUDIO_VOLUME_MAX;
238 pMixer->VolMaster.uRight = PDMAUDIO_VOLUME_MAX;
239
240 LogFlowFunc(("Created mixer '%s'\n", pMixer->pszName));
241
242 *ppMixer = pMixer;
243 }
244 else
245 RTMemFree(pMixer);
246 }
247 else
248 rc = VERR_NO_MEMORY;
249
250 LogFlowFuncLeaveRC(rc);
251 return rc;
252}
253
254/**
255 * Helper function for the internal debugger to print the mixer's current
256 * state, along with the attached sinks.
257 *
258 * @param pMixer Mixer to print debug output for.
259 * @param pHlp Debug info helper to use.
260 * @param pszArgs Optional arguments. Not being used at the moment.
261 */
262void AudioMixerDebug(PAUDIOMIXER pMixer, PCDBGFINFOHLP pHlp, const char *pszArgs)
263{
264 RT_NOREF(pszArgs);
265 PAUDMIXSINK pSink;
266 unsigned iSink = 0;
267
268 int rc2 = RTCritSectEnter(&pMixer->CritSect);
269 if (RT_FAILURE(rc2))
270 return;
271
272 pHlp->pfnPrintf(pHlp, "[Master] %s: lVol=%u, rVol=%u, fMuted=%RTbool\n", pMixer->pszName,
273 pMixer->VolMaster.uLeft, pMixer->VolMaster.uRight, pMixer->VolMaster.fMuted);
274
275 RTListForEach(&pMixer->lstSinks, pSink, AUDMIXSINK, Node)
276 {
277 pHlp->pfnPrintf(pHlp, "[Sink %u] %s: lVol=%u, rVol=%u, fMuted=%RTbool\n", iSink, pSink->pszName,
278 pSink->Volume.uLeft, pSink->Volume.uRight, pSink->Volume.fMuted);
279 ++iSink;
280 }
281
282 rc2 = RTCritSectLeave(&pMixer->CritSect);
283 AssertRC(rc2);
284}
285
286/**
287 * Destroys an audio mixer.
288 *
289 * @param pMixer Audio mixer to destroy.
290 */
291void AudioMixerDestroy(PAUDIOMIXER pMixer)
292{
293 if (!pMixer)
294 return;
295
296 int rc2 = RTCritSectEnter(&pMixer->CritSect);
297 AssertRC(rc2);
298
299 LogFlowFunc(("Destroying %s ...\n", pMixer->pszName));
300
301 PAUDMIXSINK pSink, pSinkNext;
302 RTListForEachSafe(&pMixer->lstSinks, pSink, pSinkNext, AUDMIXSINK, Node)
303 {
304 /* Save a pointer to the sink to remove, as pSink
305 * will not be valid anymore after calling audioMixerRemoveSinkInternal(). */
306 PAUDMIXSINK pSinkToRemove = pSink;
307
308 audioMixerRemoveSinkInternal(pMixer, pSinkToRemove);
309 audioMixerSinkDestroyInternal(pSinkToRemove);
310 }
311
312 pMixer->cSinks = 0;
313
314 if (pMixer->pszName)
315 {
316 RTStrFree(pMixer->pszName);
317 pMixer->pszName = NULL;
318 }
319
320 rc2 = RTCritSectLeave(&pMixer->CritSect);
321 AssertRC(rc2);
322
323 RTCritSectDelete(&pMixer->CritSect);
324
325 RTMemFree(pMixer);
326 pMixer = NULL;
327}
328
329/**
330 * Invalidates all internal data, internal version.
331 *
332 * @returns IPRT status code.
333 * @param pMixer Mixer to invalidate data for.
334 */
335int audioMixerInvalidateInternal(PAUDIOMIXER pMixer)
336{
337 AssertPtrReturn(pMixer, VERR_INVALID_POINTER);
338
339 LogFlowFunc(("[%s]\n", pMixer->pszName));
340
341 /* Propagate new master volume to all connected sinks. */
342 PAUDMIXSINK pSink;
343 RTListForEach(&pMixer->lstSinks, pSink, AUDMIXSINK, Node)
344 {
345 int rc2 = audioMixerSinkUpdateVolume(pSink, &pMixer->VolMaster);
346 AssertRC(rc2);
347 }
348
349 return VINF_SUCCESS;
350}
351
352/**
353 * Invalidates all internal data.
354 *
355 * @returns IPRT status code.
356 * @param pMixer Mixer to invalidate data for.
357 */
358void AudioMixerInvalidate(PAUDIOMIXER pMixer)
359{
360 AssertPtrReturnVoid(pMixer);
361
362 int rc2 = RTCritSectEnter(&pMixer->CritSect);
363 AssertRC(rc2);
364
365 LogFlowFunc(("[%s]\n", pMixer->pszName));
366
367 rc2 = audioMixerInvalidateInternal(pMixer);
368 AssertRC(rc2);
369
370 rc2 = RTCritSectLeave(&pMixer->CritSect);
371 AssertRC(rc2);
372}
373
374/**
375 * Removes a formerly attached audio sink for an audio mixer, internal version.
376 *
377 * @returns IPRT status code.
378 * @param pMixer Mixer to remove sink from.
379 * @param pSink Sink to remove.
380 */
381static int audioMixerRemoveSinkInternal(PAUDIOMIXER pMixer, PAUDMIXSINK pSink)
382{
383 AssertPtrReturn(pMixer, VERR_INVALID_POINTER);
384 if (!pSink)
385 return VERR_NOT_FOUND;
386
387 AssertMsgReturn(pSink->pParent == pMixer, ("%s: Is not part of mixer '%s'\n",
388 pSink->pszName, pMixer->pszName), VERR_NOT_FOUND);
389
390 LogFlowFunc(("[%s] pSink=%s, cSinks=%RU8\n",
391 pMixer->pszName, pSink->pszName, pMixer->cSinks));
392
393 /* Remove sink from mixer. */
394 RTListNodeRemove(&pSink->Node);
395 Assert(pMixer->cSinks);
396
397 /* Set mixer to NULL so that we know we're not part of any mixer anymore. */
398 pSink->pParent = NULL;
399
400 return VINF_SUCCESS;
401}
402
403/**
404 * Removes a formerly attached audio sink for an audio mixer.
405 *
406 * @returns IPRT status code.
407 * @param pMixer Mixer to remove sink from.
408 * @param pSink Sink to remove.
409 */
410void AudioMixerRemoveSink(PAUDIOMIXER pMixer, PAUDMIXSINK pSink)
411{
412 int rc2 = RTCritSectEnter(&pMixer->CritSect);
413 AssertRC(rc2);
414
415 audioMixerSinkRemoveAllStreamsInternal(pSink);
416 audioMixerRemoveSinkInternal(pMixer, pSink);
417
418 rc2 = RTCritSectLeave(&pMixer->CritSect);
419}
420
421/**
422 * Sets the mixer's master volume.
423 *
424 * @returns IPRT status code.
425 * @param pMixer Mixer to set master volume for.
426 * @param pVol Volume to set.
427 */
428int AudioMixerSetMasterVolume(PAUDIOMIXER pMixer, PPDMAUDIOVOLUME pVol)
429{
430 AssertPtrReturn(pMixer, VERR_INVALID_POINTER);
431 AssertPtrReturn(pVol, VERR_INVALID_POINTER);
432
433 int rc = RTCritSectEnter(&pMixer->CritSect);
434 if (RT_FAILURE(rc))
435 return rc;
436
437 memcpy(&pMixer->VolMaster, pVol, sizeof(PDMAUDIOVOLUME));
438
439 LogFlowFunc(("[%s] lVol=%RU32, rVol=%RU32 => fMuted=%RTbool, lVol=%RU32, rVol=%RU32\n",
440 pMixer->pszName, pVol->uLeft, pVol->uRight,
441 pMixer->VolMaster.fMuted, pMixer->VolMaster.uLeft, pMixer->VolMaster.uRight));
442
443 rc = audioMixerInvalidateInternal(pMixer);
444
445 int rc2 = RTCritSectLeave(&pMixer->CritSect);
446 AssertRC(rc2);
447
448 return rc;
449}
450
451/*********************************************************************************************************************************
452 * Mixer Sink implementation.
453 ********************************************************************************************************************************/
454
455/**
456 * Adds an audio stream to a specific audio sink.
457 *
458 * @returns IPRT status code.
459 * @param pSink Sink to add audio stream to.
460 * @param pStream Stream to add.
461 */
462int AudioMixerSinkAddStream(PAUDMIXSINK pSink, PAUDMIXSTREAM pStream)
463{
464 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
465 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
466
467 int rc = RTCritSectEnter(&pSink->CritSect);
468 if (RT_FAILURE(rc))
469 return rc;
470
471 if (pSink->cStreams == UINT8_MAX) /* 255 streams per sink max. */
472 {
473 int rc2 = RTCritSectLeave(&pSink->CritSect);
474 AssertRC(rc2);
475
476 return VERR_NO_MORE_HANDLES;
477 }
478
479 LogFlowFuncEnter();
480
481#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
482 /* Make sure only compatible streams are added. */
483 if (pStream->enmDir == PDMAUDIODIR_IN)
484 {
485 if (DrvAudioHlpPCMPropsAreEqual(&pSink->PCMProps, &pStream->InOut.pIn->Props))
486 {
487#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
488 /* Chain: Stream (Child) -> Sink (Child) -> Guest (Parent). */
489 PPDMAUDIOMIXBUF pHstIn = &pStream->InOut.pIn->pHstStrmIn->MixBuf;
490 PPDMAUDIOMIXBUF pGstIn = &pStream->InOut.pIn->MixBuf;
491
492 /* Unlink any former parent from host input. */
493 AudioMixBufUnlink(pHstIn);
494
495 /* Link host input to this sink as a parent. */
496 rc = AudioMixBufLinkTo(pHstIn, &pSink->MixBuf);
497 AssertRC(rc);
498
499 /* Unlink any former parent from this sink. */
500 AudioMixBufUnlink(&pSink->MixBuf);
501
502 /* Link guest input to this sink as a parent. */
503 rc = AudioMixBufLinkTo(&pSink->MixBuf, pGstIn);
504 AssertRC(rc);
505# ifdef DEBUG
506 AudioMixBufDbgPrintChain(&pStream->InOut.pIn->MixBuf);
507# endif
508#endif /* VBOX_AUDIO_MIXER_WITH_MIXBUF */
509 }
510 else
511 AssertFailedStmt(rc = VERR_WRONG_TYPE);
512 }
513 else if (pStream->enmDir == PDMAUDIODIR_OUT)
514 {
515 if (DrvAudioHlpPCMPropsAreEqual(&pSink->PCMProps, &pStream->InOut.pOut->Props))
516 {
517#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
518 /* Chain: Guest (Child) -> Sink (Child) -> Stream (Parent). */
519 rc = AudioMixBufLinkTo(&pStream->InOut.pOut->pHstStrmOut->MixBuf, &pSink->MixBuf);
520# ifdef DEBUG
521 AudioMixBufDbgPrintChain(&pSink->MixBuf);
522# endif
523#endif /* VBOX_AUDIO_MIXER_WITH_MIXBUF */
524 }
525 else
526 AssertFailedStmt(rc = VERR_WRONG_TYPE);
527 }
528 else
529 AssertFailedStmt(rc = VERR_NOT_IMPLEMENTED);
530#else
531 rc = VINF_SUCCESS;
532#endif
533
534 if (RT_SUCCESS(rc))
535 {
536 /** @todo Check if stream already is assigned to (another) sink. */
537
538 /* If the sink is running and not in pending disable mode,
539 * make sure that the added stream also is enabled. */
540 if ( (pSink->fStatus & AUDMIXSINK_STS_RUNNING)
541 && !(pSink->fStatus & AUDMIXSINK_STS_PENDING_DISABLE))
542 {
543 rc = audioMixerStreamCtlInternal(pStream, PDMAUDIOSTREAMCMD_ENABLE, AUDMIXSTRMCTL_FLAG_NONE);
544 }
545
546 if (RT_SUCCESS(rc))
547 {
548 /* Apply the sink's combined volume to the stream. */
549 rc = pStream->pConn->pfnStreamSetVolume(pStream->pConn, pStream->pStream, &pSink->VolumeCombined);
550 AssertRC(rc);
551 }
552
553 if (RT_SUCCESS(rc))
554 {
555 /* Save pointer to sink the stream is attached to. */
556 pStream->pSink = pSink;
557
558 /* Append stream to sink's list. */
559 RTListAppend(&pSink->lstStreams, &pStream->Node);
560 pSink->cStreams++;
561 }
562 }
563
564 LogFlowFunc(("[%s] cStreams=%RU8, rc=%Rrc\n", pSink->pszName, pSink->cStreams, rc));
565
566 int rc2 = RTCritSectLeave(&pSink->CritSect);
567 AssertRC(rc2);
568
569 return rc;
570}
571
572/**
573 * Creates an audio mixer stream.
574 *
575 * @returns IPRT status code.
576 * @param pSink Sink to use for creating the stream.
577 * @param pConn Audio connector interface to use.
578 * @param pCfg Audio stream configuration to use.
579 * @param fFlags Stream creation flags. Currently unused, set to 0.
580 * @param ppStream Pointer which receives the newly created audio stream.
581 */
582int AudioMixerSinkCreateStream(PAUDMIXSINK pSink,
583 PPDMIAUDIOCONNECTOR pConn, PPDMAUDIOSTREAMCFG pCfg, uint32_t fFlags, PAUDMIXSTREAM *ppStream)
584{
585 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
586 AssertPtrReturn(pConn, VERR_INVALID_POINTER);
587 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
588 /** @todo Validate fFlags. */
589 /* ppStream is optional. */
590
591 if (pConn->pfnGetStatus(pConn, PDMAUDIODIR_ANY) == PDMAUDIOBACKENDSTS_NOT_ATTACHED)
592 return VERR_AUDIO_BACKEND_NOT_ATTACHED;
593
594 PAUDMIXSTREAM pMixStream = (PAUDMIXSTREAM)RTMemAllocZ(sizeof(AUDMIXSTREAM));
595 if (!pMixStream)
596 return VERR_NO_MEMORY;
597
598 pMixStream->pszName = RTStrDup(pCfg->szName);
599 if (!pMixStream->pszName)
600 {
601 RTMemFree(pMixStream);
602 return VERR_NO_MEMORY;
603 }
604
605 int rc = RTCritSectEnter(&pSink->CritSect);
606 if (RT_FAILURE(rc))
607 return rc;
608
609 LogFlowFunc(("[%s] fFlags=0x%x (enmDir=%ld, %RU8 bits, %RU8 channels, %RU32Hz)\n",
610 pSink->pszName, fFlags, pCfg->enmDir, pCfg->Props.cBits, pCfg->Props.cChannels, pCfg->Props.uHz));
611
612 /*
613 * Initialize the host-side configuration for the stream to be created.
614 * Always use the sink's PCM audio format as the host side when creating a stream for it.
615 */
616 PDMAUDIOSTREAMCFG CfgHost;
617 rc = DrvAudioHlpPCMPropsToStreamCfg(&pSink->PCMProps, &CfgHost);
618 AssertRCReturn(rc, rc);
619
620 /* Apply the sink's direction for the configuration to use to
621 * create the stream. */
622 if (pSink->enmDir == AUDMIXSINKDIR_INPUT)
623 {
624 CfgHost.DestSource.Source = pCfg->DestSource.Source;
625 CfgHost.enmDir = PDMAUDIODIR_IN;
626 CfgHost.enmLayout = pCfg->enmLayout;
627 }
628 else
629 {
630 CfgHost.DestSource.Dest = pCfg->DestSource.Dest;
631 CfgHost.enmDir = PDMAUDIODIR_OUT;
632 CfgHost.enmLayout = pCfg->enmLayout;
633 }
634
635 RTStrPrintf(CfgHost.szName, sizeof(CfgHost.szName), "%s", pCfg->szName);
636
637 rc = RTCritSectInit(&pMixStream->CritSect);
638 if (RT_SUCCESS(rc))
639 {
640 PPDMAUDIOSTREAM pStream;
641 rc = pConn->pfnStreamCreate(pConn, &CfgHost, pCfg, &pStream);
642 if (RT_SUCCESS(rc))
643 {
644 /* Save the audio stream pointer to this mixing stream. */
645 pMixStream->pStream = pStream;
646
647 /* Increase the stream's reference count to let others know
648 * we're reyling on it to be around now. */
649 pConn->pfnStreamRetain(pConn, pStream);
650 }
651 }
652
653 if (RT_SUCCESS(rc))
654 {
655 pMixStream->fFlags = fFlags;
656 pMixStream->pConn = pConn;
657
658 if (ppStream)
659 *ppStream = pMixStream;
660 }
661 else if (pMixStream)
662 {
663 int rc2 = RTCritSectDelete(&pMixStream->CritSect);
664 AssertRC(rc2);
665
666 if (pMixStream->pszName)
667 {
668 RTStrFree(pMixStream->pszName);
669 pMixStream->pszName = NULL;
670 }
671
672 RTMemFree(pMixStream);
673 pMixStream = NULL;
674 }
675
676 int rc2 = RTCritSectLeave(&pSink->CritSect);
677 AssertRC(rc2);
678
679 return rc;
680}
681
682/**
683 * Static helper function to translate a sink command
684 * to a PDM audio stream command.
685 *
686 * @returns PDM audio stream command, or PDMAUDIOSTREAMCMD_UNKNOWN if not found.
687 * @param enmCmd Mixer sink command to translate.
688 */
689static PDMAUDIOSTREAMCMD audioMixerSinkToStreamCmd(AUDMIXSINKCMD enmCmd)
690{
691 switch (enmCmd)
692 {
693 case AUDMIXSINKCMD_ENABLE: return PDMAUDIOSTREAMCMD_ENABLE;
694 case AUDMIXSINKCMD_DISABLE: return PDMAUDIOSTREAMCMD_DISABLE;
695 case AUDMIXSINKCMD_PAUSE: return PDMAUDIOSTREAMCMD_PAUSE;
696 case AUDMIXSINKCMD_RESUME: return PDMAUDIOSTREAMCMD_RESUME;
697 default: break;
698 }
699
700 AssertMsgFailed(("Unsupported sink command %d\n", enmCmd));
701 return PDMAUDIOSTREAMCMD_UNKNOWN;
702}
703
704/**
705 * Controls a mixer sink.
706 *
707 * @returns IPRT status code.
708 * @param pSink Mixer sink to control.
709 * @param enmSinkCmd Sink command to set.
710 */
711int AudioMixerSinkCtl(PAUDMIXSINK pSink, AUDMIXSINKCMD enmSinkCmd)
712{
713 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
714
715 PDMAUDIOSTREAMCMD enmCmdStream = audioMixerSinkToStreamCmd(enmSinkCmd);
716 if (enmCmdStream == PDMAUDIOSTREAMCMD_UNKNOWN)
717 return VERR_NOT_SUPPORTED;
718
719 int rc = RTCritSectEnter(&pSink->CritSect);
720 if (RT_FAILURE(rc))
721 return rc;
722
723 PAUDMIXSTREAM pStream;
724 RTListForEach(&pSink->lstStreams, pStream, AUDMIXSTREAM, Node)
725 {
726 int rc2 = audioMixerStreamCtlInternal(pStream, enmCmdStream, AUDMIXSTRMCTL_FLAG_NONE);
727 if (RT_SUCCESS(rc))
728 rc = rc2;
729 /* Keep going. Flag? */
730 }
731
732 if (enmSinkCmd == AUDMIXSINKCMD_ENABLE)
733 {
734 /* Make sure to clear any other former flags again by assigning AUDMIXSINK_STS_RUNNING directly. */
735 pSink->fStatus = AUDMIXSINK_STS_RUNNING;
736 }
737 else if (enmSinkCmd == AUDMIXSINKCMD_DISABLE)
738 {
739 if (pSink->fStatus & AUDMIXSINK_STS_RUNNING)
740 {
741 /* Set the sink in a pending disable state first.
742 * The final status (disabled) will be set in the sink's iteration. */
743 pSink->fStatus |= AUDMIXSINK_STS_PENDING_DISABLE;
744 }
745 }
746
747#ifdef LOG_ENABLED
748 char *pszStatus = dbgAudioMixerSinkStatusToStr(pSink->fStatus);
749 LogFlowFunc(("[%s] enmCmd=%d, fStatus=%s, rc=%Rrc\n", pSink->pszName, enmSinkCmd, pszStatus, rc));
750 RTStrFree(pszStatus);
751#endif
752
753 int rc2 = RTCritSectLeave(&pSink->CritSect);
754 AssertRC(rc2);
755
756 return rc;
757}
758
759/**
760 * Destroys a mixer sink and removes it from the attached mixer (if any).
761 *
762 * @param pSink Mixer sink to destroy.
763 */
764void AudioMixerSinkDestroy(PAUDMIXSINK pSink)
765{
766 if (!pSink)
767 return;
768
769 int rc2 = RTCritSectEnter(&pSink->CritSect);
770 AssertRC(rc2);
771
772 if (pSink->pParent)
773 {
774 /* Save mixer pointer, as after audioMixerRemoveSinkInternal() the
775 * pointer will be gone from the stream. */
776 PAUDIOMIXER pMixer = pSink->pParent;
777 AssertPtr(pMixer);
778
779 audioMixerRemoveSinkInternal(pMixer, pSink);
780
781 Assert(pMixer->cSinks);
782 pMixer->cSinks--;
783 }
784
785 rc2 = RTCritSectLeave(&pSink->CritSect);
786 AssertRC(rc2);
787
788 audioMixerSinkDestroyInternal(pSink);
789}
790
791/**
792 * Destroys a mixer sink.
793 *
794 * @param pSink Mixer sink to destroy.
795 */
796static void audioMixerSinkDestroyInternal(PAUDMIXSINK pSink)
797{
798 AssertPtrReturnVoid(pSink);
799
800 LogFunc(("%s\n", pSink->pszName));
801
802 PAUDMIXSTREAM pStream, pStreamNext;
803 RTListForEachSafe(&pSink->lstStreams, pStream, pStreamNext, AUDMIXSTREAM, Node)
804 {
805 /* Save a pointer to the stream to remove, as pStream
806 * will not be valid anymore after calling audioMixerSinkRemoveStreamInternal(). */
807 PAUDMIXSTREAM pStreamToRemove = pStream;
808
809 audioMixerSinkRemoveStreamInternal(pSink, pStreamToRemove);
810 audioMixerStreamDestroyInternal(pStreamToRemove);
811 }
812
813 if (pSink->pszName)
814 {
815 RTStrFree(pSink->pszName);
816 pSink->pszName = NULL;
817 }
818
819 RTCritSectDelete(&pSink->CritSect);
820
821 RTMemFree(pSink);
822 pSink = NULL;
823}
824
825/**
826 * Returns the amount of bytes ready to be read from a sink since the last call
827 * to AudioMixerSinkUpdate().
828 *
829 * @returns Amount of bytes ready to be read from the sink.
830 * @param pSink Sink to return number of available bytes for.
831 */
832uint32_t AudioMixerSinkGetReadable(PAUDMIXSINK pSink)
833{
834 AssertPtrReturn(pSink, 0);
835
836 AssertMsg(pSink->enmDir == AUDMIXSINKDIR_INPUT, ("%s: Can't read from a non-input sink\n", pSink->pszName));
837
838 int rc = RTCritSectEnter(&pSink->CritSect);
839 if (RT_FAILURE(rc))
840 return 0;
841
842 uint32_t cbReadable = 0;
843
844#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
845# error "Implement me!"
846#else
847 /* The hosts sets the pace --
848 * so we try to find the maximum of readable data of all connected streams to this sink. */
849 PAUDMIXSTREAM pMixStream;
850 RTListForEach(&pSink->lstStreams, pMixStream, AUDMIXSTREAM, Node)
851 {
852 if (!(pMixStream->pConn->pfnStreamGetStatus(pMixStream->pConn, pMixStream->pStream) & PDMAUDIOSTREAMSTS_FLAG_ENABLED))
853 {
854 Log3Func(("[%s] Stream '%s' disabled, skipping ...\n", pSink->pszName, pMixStream->pszName));
855 continue;
856 }
857
858 cbReadable = RT_MAX(cbReadable,
859 pMixStream->pConn->pfnStreamGetReadable(pMixStream->pConn, pMixStream->pStream));
860
861 break; /** @todo For now we only support recording by the first stream added. */
862 }
863#endif
864
865 Log3Func(("[%s] cbReadable=%RU32\n", pSink->pszName, cbReadable));
866
867 int rc2 = RTCritSectLeave(&pSink->CritSect);
868 AssertRC(rc2);
869
870 return cbReadable;
871}
872
873/**
874 * Returns the amount of bytes ready to be written to a sink since the last call
875 * to AudioMixerSinkUpdate().
876 *
877 * @returns Amount of bytes ready to be written to the sink.
878 * @param pSink Sink to return number of available bytes for.
879 */
880uint32_t AudioMixerSinkGetWritable(PAUDMIXSINK pSink)
881{
882 AssertPtrReturn(pSink, 0);
883
884 AssertMsg(pSink->enmDir == AUDMIXSINKDIR_OUTPUT, ("%s: Can't write to a non-output sink\n", pSink->pszName));
885
886 int rc = RTCritSectEnter(&pSink->CritSect);
887 if (RT_FAILURE(rc))
888 return 0;
889
890 uint32_t cbWritable = UINT32_MAX;
891
892 if ( (pSink->fStatus & AUDMIXSINK_STS_RUNNING)
893 && !(pSink->fStatus & AUDMIXSINK_STS_PENDING_DISABLE))
894 {
895#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
896# error "Implement me!"
897#else
898 /* The hosts sets the pace --
899 * so we try to find the minimum of writable data to all connected streams to this sink. */
900 PAUDMIXSTREAM pMixStream;
901 RTListForEach(&pSink->lstStreams, pMixStream, AUDMIXSTREAM, Node)
902 {
903 const uint32_t cbWritableStream = pMixStream->pConn->pfnStreamGetWritable(pMixStream->pConn, pMixStream->pStream);
904
905 if (cbWritableStream < cbWritable)
906 cbWritable = cbWritableStream;
907 }
908#endif
909 }
910
911 if (cbWritable == UINT32_MAX)
912 cbWritable = 0;
913
914 Log3Func(("[%s] cbWritable=%RU32\n", pSink->pszName, cbWritable));
915
916 int rc2 = RTCritSectLeave(&pSink->CritSect);
917 AssertRC(rc2);
918
919 return cbWritable;
920}
921
922/**
923 * Returns the sink's mixing direction.
924 *
925 * @returns Mixing direction.
926 * @param pSink Sink to return direction for.
927 */
928AUDMIXSINKDIR AudioMixerSinkGetDir(PAUDMIXSINK pSink)
929{
930 AssertPtrReturn(pSink, AUDMIXSINKDIR_UNKNOWN);
931
932 int rc = RTCritSectEnter(&pSink->CritSect);
933 if (RT_FAILURE(rc))
934 return AUDMIXSINKDIR_UNKNOWN;
935
936 AUDMIXSINKDIR enmDir = pSink->enmDir;
937
938 int rc2 = RTCritSectLeave(&pSink->CritSect);
939 AssertRC(rc2);
940
941 return enmDir;
942}
943
944/**
945 * Returns a specific mixer stream from a sink, based on its index.
946 *
947 * @returns Mixer stream if found, or NULL if not found.
948 * @param pSink Sink to retrieve mixer stream from.
949 * @param uIndex Index of the mixer stream to return.
950 */
951PAUDMIXSTREAM AudioMixerSinkGetStream(PAUDMIXSINK pSink, uint8_t uIndex)
952{
953 AssertPtrReturn(pSink, NULL);
954
955 int rc = RTCritSectEnter(&pSink->CritSect);
956 if (RT_FAILURE(rc))
957 return NULL;
958
959 AssertMsgReturn(uIndex < pSink->cStreams,
960 ("Index %RU8 exceeds stream count (%RU8)", uIndex, pSink->cStreams), NULL);
961
962 /* Slow lookup, d'oh. */
963 PAUDMIXSTREAM pStream = RTListGetFirst(&pSink->lstStreams, AUDMIXSTREAM, Node);
964 while (uIndex)
965 {
966 pStream = RTListGetNext(&pSink->lstStreams, pStream, AUDMIXSTREAM, Node);
967 uIndex--;
968 }
969
970 /** @todo Do we need to raise the stream's reference count here? */
971
972 int rc2 = RTCritSectLeave(&pSink->CritSect);
973 AssertRC(rc2);
974
975 AssertPtr(pStream);
976 return pStream;
977}
978
979/**
980 * Returns the current status of a mixer sink.
981 *
982 * @returns The sink's current status.
983 * @param pSink Mixer sink to return status for.
984 */
985AUDMIXSINKSTS AudioMixerSinkGetStatus(PAUDMIXSINK pSink)
986{
987 if (!pSink)
988 return AUDMIXSINK_STS_NONE;
989
990 int rc2 = RTCritSectEnter(&pSink->CritSect);
991 if (RT_FAILURE(rc2))
992 return AUDMIXSINK_STS_NONE;
993
994 /* If the dirty flag is set, there is unprocessed data in the sink. */
995 AUDMIXSINKSTS stsSink = pSink->fStatus;
996
997 rc2 = RTCritSectLeave(&pSink->CritSect);
998 AssertRC(rc2);
999
1000 return stsSink;
1001}
1002
1003/**
1004 * Returns the number of attached mixer streams to a mixer sink.
1005 *
1006 * @returns The number of attached mixer streams.
1007 * @param pSink Mixer sink to return number for.
1008 */
1009uint8_t AudioMixerSinkGetStreamCount(PAUDMIXSINK pSink)
1010{
1011 if (!pSink)
1012 return 0;
1013
1014 int rc2 = RTCritSectEnter(&pSink->CritSect);
1015 if (RT_FAILURE(rc2))
1016 return 0;
1017
1018 uint8_t cStreams = pSink->cStreams;
1019
1020 rc2 = RTCritSectLeave(&pSink->CritSect);
1021 AssertRC(rc2);
1022
1023 return cStreams;
1024}
1025
1026/**
1027 * Returns whether the sink is in an active state or not.
1028 * Note: The pending disable state also counts as active.
1029 *
1030 * @returns True if active, false if not.
1031 * @param pSink Sink to return active state for.
1032 */
1033bool AudioMixerSinkIsActive(PAUDMIXSINK pSink)
1034{
1035 if (!pSink)
1036 return false;
1037
1038 int rc2 = RTCritSectEnter(&pSink->CritSect);
1039 if (RT_FAILURE(rc2))
1040 return false;
1041
1042 bool fIsActive = pSink->fStatus & AUDMIXSINK_STS_RUNNING;
1043 /* Note: AUDMIXSINK_STS_PENDING_DISABLE implies AUDMIXSINK_STS_RUNNING. */
1044
1045 Log3Func(("[%s] fActive=%RTbool\n", pSink->pszName, fIsActive));
1046
1047 rc2 = RTCritSectLeave(&pSink->CritSect);
1048 AssertRC(rc2);
1049
1050 return fIsActive;
1051}
1052
1053/**
1054 * Reads audio data from a mixer sink.
1055 *
1056 * @returns IPRT status code.
1057 * @param pSink Mixer sink to read data from.
1058 * @param enmOp Mixer operation to use for reading the data.
1059 * @param pvBuf Buffer where to store the read data.
1060 * @param cbBuf Buffer size (in bytes) where to store the data.
1061 * @param pcbRead Number of bytes read. Optional.
1062 */
1063int AudioMixerSinkRead(PAUDMIXSINK pSink, AUDMIXOP enmOp, void *pvBuf, uint32_t cbBuf, uint32_t *pcbRead)
1064{
1065 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1066 RT_NOREF(enmOp);
1067 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
1068 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
1069 /* pcbRead is optional. */
1070
1071 /** @todo Handle mixing operation enmOp! */
1072
1073 int rc = RTCritSectEnter(&pSink->CritSect);
1074 if (RT_FAILURE(rc))
1075 return rc;
1076
1077 AssertMsg(pSink->enmDir == AUDMIXSINKDIR_INPUT,
1078 ("Can't read from a sink which is not an input sink\n"));
1079
1080#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
1081# error "Implement me!"
1082#else
1083 uint8_t *pvMixBuf = (uint8_t *)RTMemAlloc(cbBuf);
1084 if (!pvMixBuf)
1085 {
1086 int rc2 = RTCritSectLeave(&pSink->CritSect);
1087 AssertRC(rc2);
1088
1089 return VERR_NO_MEMORY;
1090 }
1091#endif
1092
1093 uint32_t cbRead = 0;
1094
1095 /* Flag indicating whether this sink is in a 'clean' state,
1096 * e.g. there is no more data to read from. */
1097 bool fClean = true;
1098
1099 PAUDMIXSTREAM pMixStream;
1100 RTListForEach(&pSink->lstStreams, pMixStream, AUDMIXSTREAM, Node)
1101 {
1102 if (!(pMixStream->pConn->pfnStreamGetStatus(pMixStream->pConn, pMixStream->pStream) & PDMAUDIOSTREAMSTS_FLAG_ENABLED))
1103 {
1104 Log3Func(("[%s] Stream '%s' disabled, skipping ...\n", pSink->pszName, pMixStream->pszName));
1105 continue;
1106 }
1107
1108 uint32_t cbTotalRead = 0;
1109 uint32_t cbToRead = cbBuf;
1110
1111 int rc2 = VINF_SUCCESS;
1112
1113 while (cbToRead)
1114 {
1115 uint32_t cbReadStrm;
1116 AssertPtr(pMixStream->pConn);
1117#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
1118# error "Implement me!"
1119#else
1120 rc2 = pMixStream->pConn->pfnStreamRead(pMixStream->pConn, pMixStream->pStream,
1121 (uint8_t *)pvMixBuf + cbTotalRead, cbToRead, &cbReadStrm);
1122#endif
1123 if (RT_FAILURE(rc2))
1124 LogFunc(("[%s] Failed reading from stream '%s': %Rrc\n", pSink->pszName, pMixStream->pszName, rc2));
1125
1126 Log3Func(("[%s] Stream '%s': Read %RU32 bytes\n", pSink->pszName, pMixStream->pszName, cbReadStrm));
1127
1128 if ( RT_FAILURE(rc2)
1129 || !cbReadStrm)
1130 break;
1131
1132 /** @todo Right now we only handle one stream (the last one added in fact). */
1133
1134 AssertBreakStmt(cbReadStrm <= cbToRead, rc = VERR_BUFFER_OVERFLOW);
1135 cbToRead -= cbReadStrm;
1136 cbTotalRead += cbReadStrm;
1137 }
1138
1139 if (RT_FAILURE(rc2))
1140 continue;
1141
1142 cbRead = RT_MAX(cbRead, cbTotalRead);
1143
1144 uint32_t cbReadable = pMixStream->pConn->pfnStreamGetReadable(pMixStream->pConn, pMixStream->pStream);
1145
1146 /* Still some data available? Then sink is not clean (yet). */
1147 if (cbReadable)
1148 fClean = false;
1149 }
1150
1151 if (RT_SUCCESS(rc))
1152 {
1153 if (fClean)
1154 pSink->fStatus &= ~AUDMIXSINK_STS_DIRTY;
1155
1156#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
1157# error "Implement me!"
1158#else
1159 if (cbRead)
1160 memcpy(pvBuf, pvMixBuf, cbRead);
1161#endif
1162 if (pcbRead)
1163 *pcbRead = cbRead;
1164 }
1165
1166#ifndef VBOX_AUDIO_MIXER_WITH_MIXBUF
1167 RTMemFree(pvMixBuf);
1168#endif
1169
1170
1171#ifdef LOG_ENABLED
1172 char *pszStatus = dbgAudioMixerSinkStatusToStr(pSink->fStatus);
1173 Log2Func(("[%s] cbRead=%RU32, fClean=%RTbool, fStatus=%s, rc=%Rrc\n", pSink->pszName, cbRead, fClean, pszStatus, rc));
1174 RTStrFree(pszStatus);
1175#endif
1176
1177 int rc2 = RTCritSectLeave(&pSink->CritSect);
1178 AssertRC(rc2);
1179
1180 return rc;
1181}
1182
1183/**
1184 * Removes a mixer stream from a mixer sink, internal version.
1185 *
1186 * @returns IPRT status code.
1187 * @param pSink Sink to remove mixer stream from.
1188 * @param pStream Stream to remove.
1189 */
1190static int audioMixerSinkRemoveStreamInternal(PAUDMIXSINK pSink, PAUDMIXSTREAM pStream)
1191{
1192 AssertPtrReturn(pSink, VERR_INVALID_PARAMETER);
1193 if ( !pStream
1194 || !pStream->pSink) /* Not part of a sink anymore? */
1195 {
1196 return VERR_NOT_FOUND;
1197 }
1198
1199 AssertMsgReturn(pStream->pSink == pSink, ("Stream '%s' is not part of sink '%s'\n",
1200 pStream->pszName, pSink->pszName), VERR_NOT_FOUND);
1201
1202 LogFlowFunc(("[%s] (Stream = %s), cStreams=%RU8\n",
1203 pSink->pszName, pStream->pStream->szName, pSink->cStreams));
1204
1205#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
1206 /* Unlink mixing buffer. */
1207 AudioMixBufUnlink(&pStream->pStream->MixBuf);
1208#endif
1209
1210 /* Remove stream from sink. */
1211 RTListNodeRemove(&pStream->Node);
1212
1213 /* Set sink to NULL so that we know we're not part of any sink anymore. */
1214 pStream->pSink = NULL;
1215
1216 return VINF_SUCCESS;
1217}
1218
1219/**
1220 * Removes a mixer stream from a mixer sink.
1221 *
1222 * @param pSink Sink to remove mixer stream from.
1223 * @param pStream Stream to remove.
1224 */
1225void AudioMixerSinkRemoveStream(PAUDMIXSINK pSink, PAUDMIXSTREAM pStream)
1226{
1227 int rc2 = RTCritSectEnter(&pSink->CritSect);
1228 AssertRC(rc2);
1229
1230 rc2 = audioMixerSinkRemoveStreamInternal(pSink, pStream);
1231 if (RT_SUCCESS(rc2))
1232 {
1233 Assert(pSink->cStreams);
1234 pSink->cStreams--;
1235 }
1236
1237 rc2 = RTCritSectLeave(&pSink->CritSect);
1238 AssertRC(rc2);
1239}
1240
1241/**
1242 * Removes all attached streams from a given sink.
1243 *
1244 * @param pSink Sink to remove attached streams from.
1245 */
1246static void audioMixerSinkRemoveAllStreamsInternal(PAUDMIXSINK pSink)
1247{
1248 if (!pSink)
1249 return;
1250
1251 LogFunc(("%s\n", pSink->pszName));
1252
1253 PAUDMIXSTREAM pStream, pStreamNext;
1254 RTListForEachSafe(&pSink->lstStreams, pStream, pStreamNext, AUDMIXSTREAM, Node)
1255 audioMixerSinkRemoveStreamInternal(pSink, pStream);
1256}
1257
1258/**
1259 * Resets the sink's state.
1260 *
1261 * @param pSink Sink to reset.
1262 */
1263static void audioMixerSinkReset(PAUDMIXSINK pSink)
1264{
1265 if (!pSink)
1266 return;
1267
1268 LogFunc(("[%s]\n", pSink->pszName));
1269
1270 if (pSink->enmDir == AUDMIXSINKDIR_INPUT)
1271 {
1272#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
1273 AudioMixBufReset(&pSink->MixBuf);
1274#else
1275 pSink->In.cbReadable = 0;
1276#endif
1277 }
1278 else if (pSink->enmDir == AUDMIXSINKDIR_OUTPUT)
1279 {
1280#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
1281 AudioMixBufReset(&pSink->MixBuf);
1282#else
1283 pSink->Out.cbWritable = 0;
1284#endif
1285 }
1286
1287 /* Update last updated timestamp. */
1288 pSink->tsLastUpdatedMS = RTTimeMilliTS();
1289
1290 /* Reset status. */
1291 pSink->fStatus = AUDMIXSINK_STS_NONE;
1292}
1293
1294/**
1295 * Removes all attached streams from a given sink.
1296 *
1297 * @param pSink Sink to remove attached streams from.
1298 */
1299void AudioMixerSinkRemoveAllStreams(PAUDMIXSINK pSink)
1300{
1301 if (!pSink)
1302 return;
1303
1304 int rc2 = RTCritSectEnter(&pSink->CritSect);
1305 AssertRC(rc2);
1306
1307 audioMixerSinkRemoveAllStreamsInternal(pSink);
1308
1309 pSink->cStreams = 0;
1310
1311 rc2 = RTCritSectLeave(&pSink->CritSect);
1312 AssertRC(rc2);
1313}
1314
1315/**
1316 * Resets a sink. This will immediately stop all processing.
1317 *
1318 * @param pSink Sink to reset.
1319 */
1320void AudioMixerSinkReset(PAUDMIXSINK pSink)
1321{
1322 if (!pSink)
1323 return;
1324
1325 int rc2 = RTCritSectEnter(&pSink->CritSect);
1326 AssertRC(rc2);
1327
1328 LogFlowFunc(("[%s]\n", pSink->pszName));
1329
1330 audioMixerSinkReset(pSink);
1331
1332 rc2 = RTCritSectLeave(&pSink->CritSect);
1333 AssertRC(rc2);
1334}
1335
1336/**
1337 * Returns the audio format of a mixer sink.
1338 *
1339 * @param pSink Sink to retrieve audio format for.
1340 * @param pPCMProps Where to the returned audio format.
1341 */
1342void AudioMixerSinkGetFormat(PAUDMIXSINK pSink, PPDMAUDIOPCMPROPS pPCMProps)
1343{
1344 AssertPtrReturnVoid(pSink);
1345 AssertPtrReturnVoid(pPCMProps);
1346
1347 int rc2 = RTCritSectEnter(&pSink->CritSect);
1348 if (RT_FAILURE(rc2))
1349 return;
1350
1351 memcpy(pPCMProps, &pSink->PCMProps, sizeof(PDMAUDIOPCMPROPS));
1352
1353 rc2 = RTCritSectLeave(&pSink->CritSect);
1354 AssertRC(rc2);
1355}
1356
1357/**
1358 * Sets the audio format of a mixer sink.
1359 *
1360 * @returns IPRT status code.
1361 * @param pSink Sink to set audio format for.
1362 * @param pPCMProps Audio format (PCM properties) to set.
1363 */
1364int AudioMixerSinkSetFormat(PAUDMIXSINK pSink, PPDMAUDIOPCMPROPS pPCMProps)
1365{
1366 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1367 AssertPtrReturn(pPCMProps, VERR_INVALID_POINTER);
1368
1369 int rc = RTCritSectEnter(&pSink->CritSect);
1370 if (RT_FAILURE(rc))
1371 return rc;
1372
1373 if (DrvAudioHlpPCMPropsAreEqual(&pSink->PCMProps, pPCMProps)) /* Bail out early if PCM properties are equal. */
1374 {
1375 rc = RTCritSectLeave(&pSink->CritSect);
1376 AssertRC(rc);
1377
1378 return rc;
1379 }
1380
1381 if (pSink->PCMProps.uHz)
1382 LogFlowFunc(("[%s] Old format: %RU8 bit, %RU8 channels, %RU32Hz\n",
1383 pSink->pszName, pSink->PCMProps.cBits, pSink->PCMProps.cChannels, pSink->PCMProps.uHz));
1384
1385 memcpy(&pSink->PCMProps, pPCMProps, sizeof(PDMAUDIOPCMPROPS));
1386
1387 LogFlowFunc(("[%s] New format %RU8 bit, %RU8 channels, %RU32Hz\n",
1388 pSink->pszName, pSink->PCMProps.cBits, pSink->PCMProps.cChannels, pSink->PCMProps.uHz));
1389
1390#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
1391 /* Also update the sink's mixing buffer format. */
1392 AudioMixBufDestroy(&pSink->MixBuf);
1393 rc = AudioMixBufInit(&pSink->MixBuf, pSink->pszName, pPCMProps, _4K /** @todo Make configurable? */);
1394 if (RT_SUCCESS(rc))
1395 {
1396 PAUDMIXSTREAM pStream;
1397 RTListForEach(&pSink->lstStreams, pStream, AUDMIXSTREAM, Node)
1398 {
1399 /** @todo Invalidate mix buffers! */
1400 }
1401 }
1402#endif /* VBOX_AUDIO_MIXER_WITH_MIXBUF */
1403
1404 int rc2 = RTCritSectLeave(&pSink->CritSect);
1405 AssertRC(rc2);
1406
1407 LogFlowFuncLeaveRC(rc);
1408 return rc;
1409}
1410
1411/**
1412 * Set the volume of an individual sink.
1413 *
1414 * @returns IPRT status code.
1415 * @param pSink Sink to set volume for.
1416 * @param pVol Volume to set.
1417 */
1418int AudioMixerSinkSetVolume(PAUDMIXSINK pSink, PPDMAUDIOVOLUME pVol)
1419{
1420 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1421 AssertPtrReturn(pVol, VERR_INVALID_POINTER);
1422
1423 int rc = RTCritSectEnter(&pSink->CritSect);
1424 if (RT_FAILURE(rc))
1425 return rc;
1426
1427 memcpy(&pSink->Volume, pVol, sizeof(PDMAUDIOVOLUME));
1428
1429 LogFlowFunc(("[%s] fMuted=%RTbool, lVol=%RU32, rVol=%RU32\n",
1430 pSink->pszName, pSink->Volume.fMuted, pSink->Volume.uLeft, pSink->Volume.uRight));
1431
1432 AssertPtr(pSink->pParent);
1433 rc = audioMixerSinkUpdateVolume(pSink, &pSink->pParent->VolMaster);
1434
1435 int rc2 = RTCritSectLeave(&pSink->CritSect);
1436 AssertRC(rc2);
1437
1438 return rc;
1439}
1440
1441/**
1442 * Updates a mixer sink, internal version.
1443 *
1444 * @returns IPRT status code.
1445 * @param pSink Mixer sink to update.
1446 */
1447static int audioMixerSinkUpdateInternal(PAUDMIXSINK pSink)
1448{
1449 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1450
1451 int rc = VINF_SUCCESS;
1452
1453#ifdef LOG_ENABLED
1454 char *pszStatus = dbgAudioMixerSinkStatusToStr(pSink->fStatus);
1455 Log3Func(("[%s] fStatus=%s\n", pSink->pszName, pszStatus));
1456 RTStrFree(pszStatus);
1457#endif
1458
1459 /* Sink disabled? Take a shortcut. */
1460 if (!(pSink->fStatus & AUDMIXSINK_STS_RUNNING))
1461 return rc;
1462
1463 /* Number of detected disabled streams of this sink. */
1464 uint8_t cStreamsDisabled = 0;
1465
1466 PAUDMIXSTREAM pMixStream, pMixStreamNext;
1467 RTListForEachSafe(&pSink->lstStreams, pMixStream, pMixStreamNext, AUDMIXSTREAM, Node)
1468 {
1469 PPDMAUDIOSTREAM pStream = pMixStream->pStream;
1470 AssertPtr(pStream);
1471
1472 PPDMIAUDIOCONNECTOR pConn = pMixStream->pConn;
1473 AssertPtr(pConn);
1474
1475 uint32_t cfProc = 0;
1476
1477 int rc2 = pConn->pfnStreamIterate(pConn, pStream);
1478 if (RT_SUCCESS(rc2))
1479 {
1480 if (pSink->enmDir == AUDMIXSINKDIR_INPUT)
1481 {
1482 rc = pConn->pfnStreamCapture(pConn, pStream, &cfProc);
1483 if (RT_FAILURE(rc2))
1484 {
1485 LogFunc(("%s: Failed capturing stream '%s', rc=%Rrc\n", pSink->pszName, pStream->szName, rc2));
1486 if (RT_SUCCESS(rc))
1487 rc = rc2;
1488 continue;
1489 }
1490
1491 if (cfProc)
1492 pSink->fStatus |= AUDMIXSINK_STS_DIRTY;
1493 }
1494 else if (pSink->enmDir == AUDMIXSINKDIR_OUTPUT)
1495 {
1496 rc2 = pConn->pfnStreamPlay(pConn, pStream, &cfProc);
1497 if (RT_FAILURE(rc2))
1498 {
1499 LogFunc(("%s: Failed playing stream '%s', rc=%Rrc\n", pSink->pszName, pStream->szName, rc2));
1500 if (RT_SUCCESS(rc))
1501 rc = rc2;
1502 continue;
1503 }
1504 }
1505 else
1506 {
1507 AssertFailedStmt(rc = VERR_NOT_IMPLEMENTED);
1508 continue;
1509 }
1510
1511 rc2 = pConn->pfnStreamIterate(pConn, pStream);
1512 if (RT_FAILURE(rc2))
1513 {
1514 LogFunc(("%s: Failed re-iterating stream '%s', rc=%Rrc\n", pSink->pszName, pStream->szName, rc2));
1515 if (RT_SUCCESS(rc))
1516 rc = rc2;
1517 continue;
1518 }
1519
1520 PDMAUDIOSTREAMSTS strmSts = pConn->pfnStreamGetStatus(pConn, pStream);
1521
1522 /* Is the stream not enabled and also is not in a pending disable state anymore? */
1523 if ( !(strmSts & PDMAUDIOSTREAMSTS_FLAG_ENABLED)
1524 && !(strmSts & PDMAUDIOSTREAMSTS_FLAG_PENDING_DISABLE))
1525 {
1526 cStreamsDisabled++;
1527 }
1528 }
1529
1530 Log3Func(("\t%s: cPlayed/cCaptured=%RU32, rc2=%Rrc\n", pStream->szName, cfProc, rc2));
1531 }
1532
1533 Log3Func(("[%s] fPendingDisable=%RTbool, %RU8/%RU8 streams disabled\n",
1534 pSink->pszName, RT_BOOL(pSink->fStatus & AUDMIXSINK_STS_PENDING_DISABLE), cStreamsDisabled, pSink->cStreams));
1535
1536 /* Update last updated timestamp. */
1537 pSink->tsLastUpdatedMS = RTTimeMilliTS();
1538
1539 /* All streams disabled and the sink is in pending disable mode? */
1540 if ( cStreamsDisabled == pSink->cStreams
1541 && (pSink->fStatus & AUDMIXSINK_STS_PENDING_DISABLE))
1542 {
1543 audioMixerSinkReset(pSink);
1544 }
1545
1546 Log3Func(("[%s] cbReadable=%RU32, cbWritable=%RU32, rc=%Rrc\n",
1547 pSink->pszName, pSink->In.cbReadable, pSink->Out.cbWritable, rc));
1548
1549 return rc;
1550}
1551
1552/**
1553 * Updates (invalidates) a mixer sink.
1554 *
1555 * @returns IPRT status code.
1556 * @param pSink Mixer sink to update.
1557 */
1558int AudioMixerSinkUpdate(PAUDMIXSINK pSink)
1559{
1560 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1561
1562 int rc = RTCritSectEnter(&pSink->CritSect);
1563 if (RT_FAILURE(rc))
1564 return rc;
1565
1566 rc = audioMixerSinkUpdateInternal(pSink);
1567
1568 int rc2 = RTCritSectLeave(&pSink->CritSect);
1569 AssertRC(rc2);
1570
1571 return rc;
1572}
1573
1574/**
1575 * Updates the (master) volume of a mixer sink.
1576 *
1577 * @returns IPRT status code.
1578 * @param pSink Mixer sink to update volume for.
1579 * @param pVolMaster Master volume to set.
1580 */
1581static int audioMixerSinkUpdateVolume(PAUDMIXSINK pSink, const PPDMAUDIOVOLUME pVolMaster)
1582{
1583 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1584 AssertPtrReturn(pVolMaster, VERR_INVALID_POINTER);
1585
1586 LogFlowFunc(("[%s] Master fMuted=%RTbool, lVol=%RU32, rVol=%RU32\n",
1587 pSink->pszName, pVolMaster->fMuted, pVolMaster->uLeft, pVolMaster->uRight));
1588 LogFlowFunc(("[%s] fMuted=%RTbool, lVol=%RU32, rVol=%RU32 ",
1589 pSink->pszName, pSink->Volume.fMuted, pSink->Volume.uLeft, pSink->Volume.uRight));
1590
1591 /** @todo Very crude implementation for now -- needs more work! */
1592
1593 pSink->VolumeCombined.fMuted = pVolMaster->fMuted || pSink->Volume.fMuted;
1594
1595 pSink->VolumeCombined.uLeft = ( (pSink->Volume.uLeft ? pSink->Volume.uLeft : 1)
1596 * (pVolMaster->uLeft ? pVolMaster->uLeft : 1)) / PDMAUDIO_VOLUME_MAX;
1597
1598 pSink->VolumeCombined.uRight = ( (pSink->Volume.uRight ? pSink->Volume.uRight : 1)
1599 * (pVolMaster->uRight ? pVolMaster->uRight : 1)) / PDMAUDIO_VOLUME_MAX;
1600
1601 LogFlow(("-> fMuted=%RTbool, lVol=%RU32, rVol=%RU32\n",
1602 pSink->VolumeCombined.fMuted, pSink->VolumeCombined.uLeft, pSink->VolumeCombined.uRight));
1603
1604 /* Propagate new sink volume to all streams in the sink. */
1605 PAUDMIXSTREAM pMixStream;
1606 RTListForEach(&pSink->lstStreams, pMixStream, AUDMIXSTREAM, Node)
1607 {
1608 int rc2 = pMixStream->pConn->pfnStreamSetVolume(pMixStream->pConn, pMixStream->pStream, &pSink->VolumeCombined);
1609 AssertRC(rc2);
1610 }
1611
1612 return VINF_SUCCESS;
1613}
1614
1615/**
1616 * Writes data to a mixer sink.
1617 *
1618 * @returns IPRT status code.
1619 * @param pSink Sink to write data to.
1620 * @param enmOp Mixer operation to use when writing data to the sink.
1621 * @param pvBuf Buffer containing the audio data to write.
1622 * @param cbBuf Size (in bytes) of the buffer containing the audio data.
1623 * @param pcbWritten Number of bytes written. Optional.
1624 */
1625int AudioMixerSinkWrite(PAUDMIXSINK pSink, AUDMIXOP enmOp, const void *pvBuf, uint32_t cbBuf, uint32_t *pcbWritten)
1626{
1627 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1628 RT_NOREF(enmOp);
1629 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
1630 AssertReturn (cbBuf, VERR_INVALID_PARAMETER);
1631 /* pcbWritten is optional. */
1632
1633 int rc = RTCritSectEnter(&pSink->CritSect);
1634 if (RT_FAILURE(rc))
1635 return rc;
1636
1637 AssertMsg(pSink->fStatus & AUDMIXSINK_STS_RUNNING,
1638 ("%s: Can't write to a sink which is not running (anymore) (status 0x%x)\n", pSink->pszName, pSink->fStatus));
1639 AssertMsg(pSink->enmDir == AUDMIXSINKDIR_OUTPUT,
1640 ("%s: Can't write to a sink which is not an output sink\n", pSink->pszName));
1641
1642 Log3Func(("[%s] enmOp=%d, cbBuf=%RU32\n", pSink->pszName, enmOp, cbBuf));
1643
1644 uint32_t cbWritten = UINT32_MAX;
1645
1646 PAUDMIXSTREAM pMixStream;
1647 RTListForEach(&pSink->lstStreams, pMixStream, AUDMIXSTREAM, Node)
1648 {
1649 uint32_t cbProcessed = 0;
1650 int rc2 = pMixStream->pConn->pfnStreamWrite(pMixStream->pConn, pMixStream->pStream, pvBuf, cbBuf, &cbProcessed);
1651 if (RT_FAILURE(rc2))
1652 LogFunc(("[%s] Failed writing to stream '%s': %Rrc\n", pSink->pszName, pMixStream->pszName, rc2));
1653
1654 Log3Func(("[%s] Written %RU32 to '%s'\n", pSink->pszName, cbProcessed, pMixStream->pszName));
1655
1656 if (cbProcessed)
1657 {
1658 /* Set dirty bit. */
1659 pSink->fStatus |= AUDMIXSINK_STS_DIRTY;
1660
1661 /*
1662 * Return the minimum bytes processed by all connected streams.
1663 * The host sets the pace, so all backends have to behave accordingly.
1664 */
1665 if (cbWritten > cbProcessed)
1666 cbWritten = cbProcessed;
1667 }
1668 }
1669
1670 if (cbWritten == UINT32_MAX)
1671 cbWritten = 0;
1672
1673 Log3Func(("[%s] cbWritten=%RU32\n", pSink->pszName, cbWritten));
1674
1675 if (pcbWritten)
1676 *pcbWritten = cbWritten;
1677
1678 int rc2 = RTCritSectLeave(&pSink->CritSect);
1679 AssertRC(rc2);
1680
1681 return rc;
1682}
1683
1684/*********************************************************************************************************************************
1685 * Mixer Stream implementation.
1686 ********************************************************************************************************************************/
1687
1688/**
1689 * Controls a mixer stream, internal version.
1690 *
1691 * @returns IPRT status code.
1692 * @param pMixStream Mixer stream to control.
1693 * @param enmCmd Mixer stream command to use.
1694 * @param fCtl Additional control flags. Pass 0.
1695 */
1696int audioMixerStreamCtlInternal(PAUDMIXSTREAM pMixStream, PDMAUDIOSTREAMCMD enmCmd, uint32_t fCtl)
1697{
1698 AssertPtr(pMixStream->pConn);
1699 AssertPtr(pMixStream->pStream);
1700
1701 RT_NOREF(fCtl);
1702
1703 int rc = pMixStream->pConn->pfnStreamControl(pMixStream->pConn, pMixStream->pStream, enmCmd);
1704
1705 LogFlowFunc(("[%s] enmCmd=%ld, rc=%Rrc\n", pMixStream->pszName, enmCmd, rc));
1706
1707 return rc;
1708}
1709
1710/**
1711 * Controls a mixer stream.
1712 *
1713 * @returns IPRT status code.
1714 * @param pMixStream Mixer stream to control.
1715 * @param enmCmd Mixer stream command to use.
1716 * @param fCtl Additional control flags. Pass 0.
1717 */
1718int AudioMixerStreamCtl(PAUDMIXSTREAM pMixStream, PDMAUDIOSTREAMCMD enmCmd, uint32_t fCtl)
1719{
1720 RT_NOREF(fCtl);
1721 AssertPtrReturn(pMixStream, VERR_INVALID_POINTER);
1722 /** @todo Validate fCtl. */
1723
1724 int rc = RTCritSectEnter(&pMixStream->CritSect);
1725 if (RT_FAILURE(rc))
1726 return rc;
1727
1728 rc = audioMixerStreamCtlInternal(pMixStream, enmCmd, fCtl);
1729
1730 int rc2 = RTCritSectLeave(&pMixStream->CritSect);
1731 if (RT_SUCCESS(rc))
1732 rc = rc2;
1733
1734 return rc;
1735}
1736
1737/**
1738 * Destroys a mixer stream, internal version.
1739 *
1740 * @param pMixStream Mixer stream to destroy.
1741 */
1742static void audioMixerStreamDestroyInternal(PAUDMIXSTREAM pMixStream)
1743{
1744 AssertPtrReturnVoid(pMixStream);
1745
1746 LogFunc(("%s\n", pMixStream->pszName));
1747
1748 if (pMixStream->pConn) /* Stream has a connector interface present? */
1749 {
1750 if (pMixStream->pStream)
1751 {
1752 pMixStream->pConn->pfnStreamRelease(pMixStream->pConn, pMixStream->pStream);
1753 pMixStream->pConn->pfnStreamDestroy(pMixStream->pConn, pMixStream->pStream);
1754
1755 pMixStream->pStream = NULL;
1756 }
1757
1758 pMixStream->pConn = NULL;
1759 }
1760
1761 if (pMixStream->pszName)
1762 {
1763 RTStrFree(pMixStream->pszName);
1764 pMixStream->pszName = NULL;
1765 }
1766
1767 int rc2 = RTCritSectDelete(&pMixStream->CritSect);
1768 AssertRC(rc2);
1769
1770 RTMemFree(pMixStream);
1771 pMixStream = NULL;
1772}
1773
1774/**
1775 * Destroys a mixer stream.
1776 *
1777 * @param pMixStream Mixer stream to destroy.
1778 */
1779void AudioMixerStreamDestroy(PAUDMIXSTREAM pMixStream)
1780{
1781 if (!pMixStream)
1782 return;
1783
1784 int rc2 = RTCritSectEnter(&pMixStream->CritSect);
1785 AssertRC(rc2);
1786
1787 LogFunc(("%s\n", pMixStream->pszName));
1788
1789 if (pMixStream->pSink) /* Is the stream part of a sink? */
1790 {
1791 /* Save sink pointer, as after audioMixerSinkRemoveStreamInternal() the
1792 * pointer will be gone from the stream. */
1793 PAUDMIXSINK pSink = pMixStream->pSink;
1794
1795 rc2 = audioMixerSinkRemoveStreamInternal(pSink, pMixStream);
1796 if (RT_SUCCESS(rc2))
1797 {
1798 Assert(pSink->cStreams);
1799 pSink->cStreams--;
1800 }
1801 }
1802 else
1803 rc2 = VINF_SUCCESS;
1804
1805 int rc3 = RTCritSectLeave(&pMixStream->CritSect);
1806 AssertRC(rc3);
1807
1808 if (RT_SUCCESS(rc2))
1809 {
1810 audioMixerStreamDestroyInternal(pMixStream);
1811 pMixStream = NULL;
1812 }
1813
1814 LogFlowFunc(("Returning %Rrc\n", rc2));
1815}
1816
1817/**
1818 * Returns whether a mixer stream currently is active (playing/recording) or not.
1819 *
1820 * @returns @c true if playing/recording, @c false if not.
1821 * @param pMixStream Mixer stream to return status for.
1822 */
1823bool AudioMixerStreamIsActive(PAUDMIXSTREAM pMixStream)
1824{
1825 int rc2 = RTCritSectEnter(&pMixStream->CritSect);
1826 if (RT_FAILURE(rc2))
1827 return false;
1828
1829 AssertPtr(pMixStream->pConn);
1830 AssertPtr(pMixStream->pStream);
1831
1832 bool fIsActive;
1833
1834 if ( pMixStream->pConn
1835 && pMixStream->pStream
1836 && RT_BOOL(pMixStream->pConn->pfnStreamGetStatus(pMixStream->pConn, pMixStream->pStream) & PDMAUDIOSTREAMSTS_FLAG_ENABLED))
1837 {
1838 fIsActive = true;
1839 }
1840 else
1841 fIsActive = false;
1842
1843 rc2 = RTCritSectLeave(&pMixStream->CritSect);
1844 AssertRC(rc2);
1845
1846 return fIsActive;
1847}
1848
1849/**
1850 * Returns whether a mixer stream is valid (e.g. initialized and in a working state) or not.
1851 *
1852 * @returns @c true if valid, @c false if not.
1853 * @param pMixStream Mixer stream to return status for.
1854 */
1855bool AudioMixerStreamIsValid(PAUDMIXSTREAM pMixStream)
1856{
1857 if (!pMixStream)
1858 return false;
1859
1860 int rc2 = RTCritSectEnter(&pMixStream->CritSect);
1861 if (RT_FAILURE(rc2))
1862 return false;
1863
1864 bool fIsValid;
1865
1866 if ( pMixStream->pConn
1867 && pMixStream->pStream
1868 && RT_BOOL(pMixStream->pConn->pfnStreamGetStatus(pMixStream->pConn, pMixStream->pStream) & PDMAUDIOSTREAMSTS_FLAG_INITIALIZED))
1869 {
1870 fIsValid = true;
1871 }
1872 else
1873 fIsValid = false;
1874
1875 rc2 = RTCritSectLeave(&pMixStream->CritSect);
1876 AssertRC(rc2);
1877
1878 return fIsValid;
1879}
1880
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