VirtualBox

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

Last change on this file since 62877 was 62605, checked in by vboxsync, 8 years ago

Audio: Documentation, misc. cleanup.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 44.2 KB
Line 
1/* $Id: AudioMixer.cpp 62605 2016-07-27 16:31:50Z vboxsync $ */
2/** @file
3 * VBox audio: Mixing routines, mainly used by the various audio device
4 * emulations to achieve proper multiplexing from/to attached
5 * devices LUNs.
6 *
7 * This mixer acts as a layer between the audio connector interface and
8 * the actual device emulation, providing mechanisms for audio sources (input) and
9 * audio sinks (output).
10 *
11 * As audio driver instances are handled as LUNs on the device level, this
12 * audio mixer then can take care of e.g. mixing various inputs/outputs to/from
13 * a specific source/sink.
14 *
15 * How and which audio streams are connected to sinks/sources depends on how
16 * the audio mixer has been set up.
17 *
18 * A sink can connect multiple output streams together, whereas a source
19 * does this with input streams. Each sink / source consists of one or more
20 * so-called mixer streams, which then in turn have pointers to the actual
21 * PDM audio input/output streams.
22 */
23
24/*
25 * Copyright (C) 2014-2016 Oracle Corporation
26 *
27 * This file is part of VirtualBox Open Source Edition (OSE), as
28 * available from http://www.virtualbox.org. This file is free software;
29 * you can redistribute it and/or modify it under the terms of the GNU
30 * General Public License (GPL) as published by the Free Software
31 * Foundation, in version 2 as it comes in the "COPYING" file of the
32 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
33 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
34 */
35#define LOG_GROUP LOG_GROUP_AUDIO_MIXER
36#include <VBox/log.h>
37#include "AudioMixer.h"
38#include "AudioMixBuffer.h"
39#include "DrvAudio.h"
40
41#include <VBox/vmm/pdm.h>
42#include <VBox/err.h>
43#include <VBox/vmm/mm.h>
44#include <VBox/vmm/pdmaudioifs.h>
45
46#include <iprt/alloc.h>
47#include <iprt/asm-math.h>
48#include <iprt/assert.h>
49#include <iprt/string.h>
50
51static int audioMixerRemoveSinkInternal(PAUDIOMIXER pMixer, PAUDMIXSINK pSink);
52
53static void audioMixerSinkDestroyInternal(PAUDMIXSINK pSink);
54static int audioMixerSinkUpdateVolume(PAUDMIXSINK pSink, const PPDMAUDIOVOLUME pVolMaster);
55static void audioMixerSinkRemoveAllStreamsInternal(PAUDMIXSINK pSink);
56static int audioMixerSinkRemoveStreamInternal(PAUDMIXSINK pSink, PAUDMIXSTREAM pStream);
57static void audioMixerSinkReset(PAUDMIXSINK pSink);
58static int audioMixerSinkUpdateInternal(PAUDMIXSINK pSink);
59
60static void audioMixerStreamDestroyInternal(PAUDMIXSTREAM pStream);
61
62
63/**
64 * Creates an audio sink and attaches it to the given mixer.
65 *
66 * @returns IPRT status code.
67 * @param pMixer Mixer to attach created sink to.
68 * @param pszName Name of the sink to create.
69 * @param enmDir Direction of the sink to create.
70 * @param ppSink Pointer which returns the created sink on success.
71 */
72int AudioMixerCreateSink(PAUDIOMIXER pMixer, const char *pszName, AUDMIXSINKDIR enmDir, PAUDMIXSINK *ppSink)
73{
74 AssertPtrReturn(pMixer, VERR_INVALID_POINTER);
75 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
76 /* ppSink is optional. */
77
78 int rc = VINF_SUCCESS;
79
80 PAUDMIXSINK pSink = (PAUDMIXSINK)RTMemAllocZ(sizeof(AUDMIXSINK));
81 if (pSink)
82 {
83 pSink->pszName = RTStrDup(pszName);
84 if (!pSink->pszName)
85 rc = VERR_NO_MEMORY;
86
87 if (RT_SUCCESS(rc))
88 {
89 pSink->pParent = pMixer;
90 pSink->enmDir = enmDir;
91 RTListInit(&pSink->lstStreams);
92
93 /* Set initial volume to max. */
94 pSink->Volume.fMuted = false;
95 pSink->Volume.uLeft = PDMAUDIO_VOLUME_MAX;
96 pSink->Volume.uRight = PDMAUDIO_VOLUME_MAX;
97
98 /* Ditto for the combined volume. */
99 pSink->VolumeCombined.fMuted = false;
100 pSink->VolumeCombined.uLeft = PDMAUDIO_VOLUME_MAX;
101 pSink->VolumeCombined.uRight = PDMAUDIO_VOLUME_MAX;
102
103 RTListAppend(&pMixer->lstSinks, &pSink->Node);
104 pMixer->cSinks++;
105
106 LogFlowFunc(("pMixer=%p, pSink=%p, cSinks=%RU8\n",
107 pMixer, pSink, pMixer->cSinks));
108
109 if (ppSink)
110 *ppSink = pSink;
111 }
112 else
113 RTMemFree(pSink);
114 }
115 else
116 rc = VERR_NO_MEMORY;
117
118 return rc;
119}
120
121/**
122 * Creates an audio mixer.
123 *
124 * @returns IPRT status code.
125 * @param pszName Name of the audio mixer.
126 * @param fFlags Creation flags. Not used at the moment and must be 0.
127 * @param ppMixer Pointer which returns the created mixer object.
128 */
129int AudioMixerCreate(const char *pszName, uint32_t fFlags, PAUDIOMIXER *ppMixer)
130{
131 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
132 /** @todo Add fFlags validation. */
133 AssertPtrReturn(ppMixer, VERR_INVALID_POINTER);
134
135 int rc = VINF_SUCCESS;
136
137 PAUDIOMIXER pMixer = (PAUDIOMIXER)RTMemAllocZ(sizeof(AUDIOMIXER));
138 if (pMixer)
139 {
140 pMixer->pszName = RTStrDup(pszName);
141 if (!pMixer->pszName)
142 rc = VERR_NO_MEMORY;
143
144 if (RT_SUCCESS(rc))
145 {
146 pMixer->cSinks = 0;
147 RTListInit(&pMixer->lstSinks);
148
149 /* Set master volume to the max. */
150 pMixer->VolMaster.fMuted = false;
151 pMixer->VolMaster.uLeft = PDMAUDIO_VOLUME_MAX;
152 pMixer->VolMaster.uRight = PDMAUDIO_VOLUME_MAX;
153
154 LogFlowFunc(("Created mixer '%s'\n", pMixer->pszName));
155
156 *ppMixer = pMixer;
157 }
158 else
159 RTMemFree(pMixer);
160 }
161 else
162 rc = VERR_NO_MEMORY;
163
164 LogFlowFuncLeaveRC(rc);
165 return rc;
166}
167
168/**
169 * Helper function for the internal debugger to print the mixer's current
170 * state, along with the attached sinks.
171 *
172 * @param pMixer Mixer to print debug output for.
173 * @param pHlp Debug info helper to use.
174 * @param pszArgs Optional arguments. Not being used at the moment.
175 */
176void AudioMixerDebug(PAUDIOMIXER pMixer, PCDBGFINFOHLP pHlp, const char *pszArgs)
177{
178 PAUDMIXSINK pSink;
179 unsigned iSink = 0;
180
181 pHlp->pfnPrintf(pHlp, "[Master] %s: lVol=%u, rVol=%u, fMuted=%RTbool\n", pMixer->pszName,
182 pMixer->VolMaster.uLeft, pMixer->VolMaster.uRight, pMixer->VolMaster.fMuted);
183
184 RTListForEach(&pMixer->lstSinks, pSink, AUDMIXSINK, Node)
185 {
186 pHlp->pfnPrintf(pHlp, "[Sink %u] %s: lVol=%u, rVol=%u, fMuted=%RTbool\n", iSink, pSink->pszName,
187 pSink->Volume.uLeft, pSink->Volume.uRight, pSink->Volume.fMuted);
188 ++iSink;
189 }
190}
191
192/**
193 * Destroys an audio mixer.
194 *
195 * @param pMixer Audio mixer to destroy.
196 */
197void AudioMixerDestroy(PAUDIOMIXER pMixer)
198{
199 if (!pMixer)
200 return;
201
202 LogFlowFunc(("Destroying %s ...\n", pMixer->pszName));
203
204 PAUDMIXSINK pSink, pSinkNext;
205 RTListForEachSafe(&pMixer->lstSinks, pSink, pSinkNext, AUDMIXSINK, Node)
206 {
207 /* Save a pointer to the sink to remove, as pSink
208 * will not be valid anymore after calling audioMixerRemoveSinkInternal(). */
209 PAUDMIXSINK pSinkToRemove = pSink;
210
211 audioMixerRemoveSinkInternal(pMixer, pSinkToRemove);
212 audioMixerSinkDestroyInternal(pSinkToRemove);
213 }
214
215 pMixer->cSinks = 0;
216
217 if (pMixer->pszName)
218 {
219 RTStrFree(pMixer->pszName);
220 pMixer->pszName = NULL;
221 }
222
223 RTMemFree(pMixer);
224 pMixer = NULL;
225}
226
227/**
228 * Invalidates all internal data, internal version.
229 *
230 * @returns IPRT status code.
231 * @param pMixer Mixer to invalidate data for.
232 */
233int audioMixerInvalidateInternal(PAUDIOMIXER pMixer)
234{
235 AssertPtrReturn(pMixer, VERR_INVALID_POINTER);
236
237 LogFlowFunc(("[%s]\n", pMixer->pszName));
238
239 /* Propagate new master volume to all connected sinks. */
240 PAUDMIXSINK pSink;
241 RTListForEach(&pMixer->lstSinks, pSink, AUDMIXSINK, Node)
242 {
243 int rc2 = audioMixerSinkUpdateVolume(pSink, &pMixer->VolMaster);
244 AssertRC(rc2);
245 }
246
247 return VINF_SUCCESS;
248}
249
250/**
251 * Invalidates all internal data.
252 *
253 * @returns IPRT status code.
254 * @param pMixer Mixer to invalidate data for.
255 */
256void AudioMixerInvalidate(PAUDIOMIXER pMixer)
257{
258 AssertPtrReturnVoid(pMixer);
259
260 LogFlowFunc(("[%s]\n", pMixer->pszName));
261
262 int rc2 = audioMixerInvalidateInternal(pMixer);
263 AssertRC(rc2);
264}
265
266/**
267 * Removes a formerly attached audio sink for an audio mixer, internal version.
268 *
269 * @returns IPRT status code.
270 * @param pMixer Mixer to remove sink from.
271 * @param pSink Sink to remove.
272 */
273static int audioMixerRemoveSinkInternal(PAUDIOMIXER pMixer, PAUDMIXSINK pSink)
274{
275 AssertPtrReturn(pMixer, VERR_INVALID_POINTER);
276 if (!pSink)
277 return VERR_NOT_FOUND;
278
279 AssertMsgReturn(pSink->pParent == pMixer, ("Sink '%s' is not part of mixer '%s'\n",
280 pSink->pszName, pMixer->pszName), VERR_NOT_FOUND);
281
282 LogFlowFunc(("[%s]: pSink=%s, cSinks=%RU8\n",
283 pMixer->pszName, pSink->pszName, pMixer->cSinks));
284
285 /* Remove sink from mixer. */
286 RTListNodeRemove(&pSink->Node);
287 Assert(pMixer->cSinks);
288 pMixer->cSinks--;
289
290 /* Set mixer to NULL so that we know we're not part of any mixer anymore. */
291 pSink->pParent = NULL;
292
293 return VINF_SUCCESS;
294}
295
296/**
297 * Removes a formerly attached audio sink for an audio mixer.
298 *
299 * @returns IPRT status code.
300 * @param pMixer Mixer to remove sink from.
301 * @param pSink Sink to remove.
302 */
303
304void AudioMixerRemoveSink(PAUDIOMIXER pMixer, PAUDMIXSINK pSink)
305{
306 audioMixerSinkRemoveAllStreamsInternal(pSink);
307 audioMixerRemoveSinkInternal(pMixer, pSink);
308}
309
310/**
311 * Sets the mixer's master volume.
312 *
313 * @returns IPRT status code.
314 * @param pMixer Mixer to set master volume for.
315 * @param pVol Volume to set.
316 */
317int AudioMixerSetMasterVolume(PAUDIOMIXER pMixer, PPDMAUDIOVOLUME pVol)
318{
319 AssertPtrReturn(pMixer, VERR_INVALID_POINTER);
320 AssertPtrReturn(pVol, VERR_INVALID_POINTER);
321
322 memcpy(&pMixer->VolMaster, pVol, sizeof(PDMAUDIOVOLUME));
323
324 LogFlowFunc(("[%s]: lVol=%RU32, rVol=%RU32 => fMuted=%RTbool, lVol=%RU32, rVol=%RU32\n",
325 pMixer->pszName, pVol->uLeft, pVol->uRight,
326 pMixer->VolMaster.fMuted, pMixer->VolMaster.uLeft, pMixer->VolMaster.uRight));
327
328 return audioMixerInvalidateInternal(pMixer);
329}
330
331/*********************************************************************************************************************************
332 * Mixer Sink implementation.
333 ********************************************************************************************************************************/
334
335/**
336 * Adds an audio stream to a specific audio sink.
337 *
338 * @returns IPRT status code.
339 * @param pSink Sink to add audio stream to.
340 * @param pStream Stream to add.
341 */
342int AudioMixerSinkAddStream(PAUDMIXSINK pSink, PAUDMIXSTREAM pStream)
343{
344 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
345 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
346
347 if (pSink->cStreams == UINT8_MAX) /* 255 streams per sink max. */
348 return VERR_NO_MORE_HANDLES;
349
350 int rc;
351
352 LogFlowFuncEnter();
353
354#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
355 /* Make sure only compatible streams are added. */
356 if (pStream->enmDir == PDMAUDIODIR_IN)
357 {
358 if (DrvAudioHlpPCMPropsAreEqual(&pSink->PCMProps, &pStream->InOut.pIn->Props))
359 {
360#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
361 /* Chain: Stream (Child) -> Sink (Child) -> Guest (Parent). */
362 PPDMAUDIOMIXBUF pHstIn = &pStream->InOut.pIn->pHstStrmIn->MixBuf;
363 PPDMAUDIOMIXBUF pGstIn = &pStream->InOut.pIn->MixBuf;
364
365 /* Unlink any former parent from host input. */
366 AudioMixBufUnlink(pHstIn);
367
368 /* Link host input to this sink as a parent. */
369 rc = AudioMixBufLinkTo(pHstIn, &pSink->MixBuf);
370 AssertRC(rc);
371
372 /* Unlink any former parent from this sink. */
373 AudioMixBufUnlink(&pSink->MixBuf);
374
375 /* Link guest input to this sink as a parent. */
376 rc = AudioMixBufLinkTo(&pSink->MixBuf, pGstIn);
377 AssertRC(rc);
378# ifdef DEBUG
379 AudioMixBufDbgPrintChain(&pStream->InOut.pIn->MixBuf);
380# endif
381#endif /* VBOX_AUDIO_MIXER_WITH_MIXBUF */
382 }
383 else
384 AssertFailedStmt(rc = VERR_WRONG_TYPE);
385 }
386 else if (pStream->enmDir == PDMAUDIODIR_OUT)
387 {
388 if (DrvAudioHlpPCMPropsAreEqual(&pSink->PCMProps, &pStream->InOut.pOut->Props))
389 {
390#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
391 /* Chain: Guest (Child) -> Sink (Child) -> Stream (Parent). */
392 rc = AudioMixBufLinkTo(&pStream->InOut.pOut->pHstStrmOut->MixBuf, &pSink->MixBuf);
393# ifdef DEBUG
394 AudioMixBufDbgPrintChain(&pSink->MixBuf);
395# endif
396#endif /* VBOX_AUDIO_MIXER_WITH_MIXBUF */
397 }
398 else
399 AssertFailedStmt(rc = VERR_WRONG_TYPE);
400 }
401 else
402 AssertFailedStmt(rc = VERR_NOT_IMPLEMENTED);
403#else
404 rc = VINF_SUCCESS;
405#endif
406
407 if (RT_SUCCESS(rc))
408 {
409 /** @todo Check if stream already is assigned to (another) sink. */
410
411 /* Apply the sink's combined volume to the stream. */
412 AssertPtr(pStream->pConn);
413 int rc2 = pStream->pConn->pfnStreamSetVolume(pStream->pConn, pStream->pStream, &pSink->VolumeCombined);
414 AssertRC(rc2);
415
416 /* Save pointer to sink the stream is attached to. */
417 pStream->pSink = pSink;
418
419 /* Append stream to sink's list. */
420 RTListAppend(&pSink->lstStreams, &pStream->Node);
421 pSink->cStreams++;
422 }
423
424 LogFlowFunc(("[%s]: cStreams=%RU8, rc=%Rrc\n", pSink->pszName, pSink->cStreams, rc));
425 return rc;
426}
427
428/**
429 * Creates an audio mixer stream.
430 *
431 * @returns IPRT status code.
432 * @param pSink Sink to use for creating the stream.
433 * @param pConn Audio connector interface to use.
434 * @param pCfg Audio stream configuration to use.
435 * @param fFlags Stream creation flags. Currently unused, set to 0.
436 * @param ppStream Pointer which receives the the newly created audio stream.
437 */
438int AudioMixerSinkCreateStream(PAUDMIXSINK pSink,
439 PPDMIAUDIOCONNECTOR pConn, PPDMAUDIOSTREAMCFG pCfg, uint32_t fFlags, PAUDMIXSTREAM *ppStream)
440{
441 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
442 AssertPtrReturn(pConn, VERR_INVALID_POINTER);
443 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
444 /** @todo Validate fFlags. */
445 /* ppStream is optional. */
446
447 PAUDMIXSTREAM pMixStream = (PAUDMIXSTREAM)RTMemAllocZ(sizeof(AUDMIXSTREAM));
448 if (!pMixStream)
449 return VERR_NO_MEMORY;
450
451 pMixStream->pszName = RTStrDup(pCfg->szName);
452 if (!pMixStream->pszName)
453 {
454 RTMemFree(pMixStream);
455 return VERR_NO_MEMORY;
456 }
457
458 LogFlowFunc(("[%s]: fFlags=0x%x (enmDir=%ld, %s, %RU8 channels, %RU32Hz)\n",
459 pSink->pszName, fFlags, pCfg->enmDir, DrvAudioHlpAudFmtToStr(pCfg->enmFormat), pCfg->cChannels, pCfg->uHz));
460
461 /*
462 * Initialize the host-side configuration for the stream to be created.
463 * Always use the sink's PCM audio format as the host side when creating a stream for it.
464 */
465 PDMAUDIOSTREAMCFG CfgHost;
466 int rc = DrvAudioHlpPCMPropsToStreamCfg(&pSink->PCMProps, &CfgHost);
467 AssertRCReturn(rc, rc);
468
469 /* Apply the sink's direction for the configuration to use to
470 * create the stream. */
471 if (pSink->enmDir == AUDMIXSINKDIR_INPUT)
472 {
473 CfgHost.DestSource.Source = pCfg->DestSource.Source;
474 CfgHost.enmDir = PDMAUDIODIR_IN;
475 }
476 else
477 {
478 CfgHost.DestSource.Dest = pCfg->DestSource.Dest;
479 CfgHost.enmDir = PDMAUDIODIR_OUT;
480 }
481
482 RTStrPrintf(CfgHost.szName, sizeof(CfgHost.szName), "%s", pCfg->szName);
483
484 PPDMAUDIOSTREAM pStream;
485 rc = pConn->pfnStreamCreate(pConn, &CfgHost, pCfg, &pStream);
486 if (RT_SUCCESS(rc))
487 {
488 /* Save the audio stream pointer to this mixing stream. */
489 pMixStream->pStream = pStream;
490
491 /* Increase the stream's reference count to let others know
492 * we're reyling on it to be around now. */
493 pConn->pfnStreamAddRef(pConn, pStream);
494 }
495
496 if (RT_SUCCESS(rc))
497 {
498 pMixStream->fFlags = fFlags;
499 pMixStream->pConn = pConn;
500
501 if (ppStream)
502 *ppStream = pMixStream;
503 }
504 else if (pMixStream)
505 {
506 if (pMixStream->pszName)
507 {
508 RTStrFree(pMixStream->pszName);
509 pMixStream->pszName = NULL;
510 }
511
512 RTMemFree(pMixStream);
513 pMixStream = NULL;
514 }
515
516 return rc;
517}
518
519/**
520 * Static helper function to translate a sink command
521 * to a PDM audio stream command.
522 *
523 * @returns PDM audio stream command, or PDMAUDIOSTREAMCMD_UNKNOWN if not found.
524 * @param enmCmd Mixer sink command to translate.
525 */
526static PDMAUDIOSTREAMCMD audioMixerSinkToStreamCmd(AUDMIXSINKCMD enmCmd)
527{
528 switch (enmCmd)
529 {
530 case AUDMIXSINKCMD_ENABLE: return PDMAUDIOSTREAMCMD_ENABLE;
531 case AUDMIXSINKCMD_DISABLE: return PDMAUDIOSTREAMCMD_DISABLE;
532 case AUDMIXSINKCMD_PAUSE: return PDMAUDIOSTREAMCMD_PAUSE;
533 case AUDMIXSINKCMD_RESUME: return PDMAUDIOSTREAMCMD_RESUME;
534 default: break;
535 }
536
537 AssertMsgFailed(("Unsupported sink command %ld\n", enmCmd));
538 return PDMAUDIOSTREAMCMD_UNKNOWN;
539}
540
541/**
542 * Controls a mixer sink.
543 *
544 * @returns IPRT status code.
545 * @param pSink Mixer sink to control.
546 * @param enmSinkCmd Sink command to set.
547 */
548int AudioMixerSinkCtl(PAUDMIXSINK pSink, AUDMIXSINKCMD enmSinkCmd)
549{
550 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
551
552 PDMAUDIOSTREAMCMD enmCmdStream = audioMixerSinkToStreamCmd(enmSinkCmd);
553 if (enmCmdStream == PDMAUDIOSTREAMCMD_UNKNOWN)
554 return VERR_NOT_SUPPORTED;
555
556 int rc = VINF_SUCCESS;
557
558 PAUDMIXSTREAM pStream;
559 RTListForEach(&pSink->lstStreams, pStream, AUDMIXSTREAM, Node)
560 {
561 int rc2 = AudioMixerStreamCtl(pStream, enmCmdStream, AUDMIXSTRMCTL_FLAG_NONE);
562 if (RT_SUCCESS(rc))
563 rc = rc2;
564 /* Keep going. Flag? */
565 }
566
567 if (enmSinkCmd == AUDMIXSINKCMD_ENABLE)
568 {
569 pSink->fStatus |= AUDMIXSINK_STS_RUNNING;
570 }
571 else if (enmSinkCmd == AUDMIXSINKCMD_DISABLE)
572 {
573 /* Set the sink in a pending disable state first.
574 * The final status (disabled) will be set in the sink's iteration. */
575 pSink->fStatus |= AUDMIXSINK_STS_PENDING_DISABLE;
576 }
577
578 /* Not running anymore? Reset. */
579 if (!(pSink->fStatus & AUDMIXSINK_STS_RUNNING))
580 audioMixerSinkReset(pSink);
581
582 LogFlowFunc(("[%s]: enmCmd=%ld, fStatus=0x%x, rc=%Rrc\n", pSink->pszName, enmSinkCmd, pSink->fStatus, rc));
583 return rc;
584}
585
586/**
587 * Destroys a mixer sink and removes it from the attached mixer (if any).
588 *
589 * @param pSink Mixer sink to destroy.
590 */
591void AudioMixerSinkDestroy(PAUDMIXSINK pSink)
592{
593 if (!pSink)
594 return;
595
596 if (pSink->pParent)
597 {
598 /* Save sink pointer, as after audioMixerRemoveSinkInternal() the
599 * pointer will be gone from the stream. */
600 PAUDIOMIXER pMixer = pSink->pParent;
601
602 audioMixerRemoveSinkInternal(pMixer, pSink);
603
604 Assert(pMixer->cSinks);
605 pMixer->cSinks--;
606 }
607
608 audioMixerSinkDestroyInternal(pSink);
609}
610
611/**
612 * Destroys a mixer sink.
613 *
614 * @param pSink Mixer sink to destroy.
615 */
616static void audioMixerSinkDestroyInternal(PAUDMIXSINK pSink)
617{
618 AssertPtrReturnVoid(pSink);
619
620 LogFunc(("%s\n", pSink->pszName));
621
622 PAUDMIXSTREAM pStream, pStreamNext;
623 RTListForEachSafe(&pSink->lstStreams, pStream, pStreamNext, AUDMIXSTREAM, Node)
624 {
625 /* Save a pointer to the stream to remove, as pStream
626 * will not be valid anymore after calling audioMixerSinkRemoveStreamInternal(). */
627 PAUDMIXSTREAM pStreamToRemove = pStream;
628
629 audioMixerSinkRemoveStreamInternal(pSink, pStreamToRemove);
630 audioMixerStreamDestroyInternal(pStreamToRemove);
631 }
632
633 if (pSink->pszName)
634 {
635 RTStrFree(pSink->pszName);
636 pSink->pszName = NULL;
637 }
638
639 RTMemFree(pSink);
640 pSink = NULL;
641}
642
643/**
644 * Returns the amount of bytes ready to be read from a sink since the last call
645 * to AudioMixerSinkUpdate().
646 *
647 * @returns Amount of bytes ready to be read from the sink.
648 * @param pSink Sink to return number of available samples for.
649 */
650uint32_t AudioMixerSinkGetReadable(PAUDMIXSINK pSink)
651{
652 AssertPtrReturn(pSink, 0);
653
654 AssertMsg(pSink->enmDir == AUDMIXSINKDIR_INPUT, ("Can't read from a non-input sink\n"));
655
656#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
657# error "Implement me!"
658#else
659 Log3Func(("[%s]: cbReadable=%RU32\n", pSink->pszName, pSink->In.cbReadable));
660 return pSink->In.cbReadable;
661#endif
662}
663
664/**
665 * Returns the amount of bytes ready to be written to a sink since the last call
666 * to AudioMixerSinkUpdate().
667 *
668 * @returns Amount of bytes ready to be written to the sink.
669 * @param pSink Sink to return number of available samples for.
670 */
671uint32_t AudioMixerSinkGetWritable(PAUDMIXSINK pSink)
672{
673 AssertPtrReturn(pSink, 0);
674
675 AssertMsg(pSink->enmDir == AUDMIXSINKDIR_OUTPUT, ("Can't write to a non-output sink\n"));
676
677#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
678# error "Implement me!"
679#else
680 Log3Func(("[%s]: cbWritable=%RU32\n", pSink->pszName, pSink->Out.cbWritable));
681 return pSink->Out.cbWritable;
682#endif
683}
684
685/**
686 * Returns the sink's mixing direction.
687 *
688 * @returns Mixing direction.
689 * @param pSink Sink to return direction for.
690 */
691AUDMIXSINKDIR AudioMixerSinkGetDir(PAUDMIXSINK pSink)
692{
693 AssertPtrReturn(pSink, AUDMIXSINKDIR_UNKNOWN);
694 return pSink->enmDir;
695}
696
697/**
698 * Returns a specific mixer stream from a sink, based on its index.
699 *
700 * @returns Mixer stream if found, or NULL if not found.
701 * @param pSink Sink to retrieve mixer stream from.
702 * @param uIndex Index of the mixer stream to return.
703 */
704PAUDMIXSTREAM AudioMixerSinkGetStream(PAUDMIXSINK pSink, uint8_t uIndex)
705{
706 AssertPtrReturn(pSink, NULL);
707 AssertMsgReturn(uIndex < pSink->cStreams,
708 ("Index %RU8 exceeds stream count (%RU8)", uIndex, pSink->cStreams), NULL);
709
710 /* Slow lookup, d'oh. */
711 PAUDMIXSTREAM pStream = RTListGetFirst(&pSink->lstStreams, AUDMIXSTREAM, Node);
712 while (uIndex)
713 {
714 pStream = RTListGetNext(&pSink->lstStreams, pStream, AUDMIXSTREAM, Node);
715 uIndex--;
716 }
717
718 AssertPtr(pStream);
719 return pStream;
720}
721
722/**
723 * Returns the current status of a mixer sink.
724 *
725 * @returns IPRT status code.
726 * @param pSink Mixer sink to return status for.
727 */
728AUDMIXSINKSTS AudioMixerSinkGetStatus(PAUDMIXSINK pSink)
729{
730 if (!pSink)
731 return false;
732
733 LogFlowFunc(("[%s]: fStatus=0x%x\n", pSink->pszName, pSink->fStatus));
734
735 /* If the dirty flag is set, there is unprocessed data in the sink. */
736 return pSink->fStatus;
737}
738
739/**
740 * Returns the number of attached mixer streams to a mixer sink.
741 *
742 * @returns IPRT status code.
743 * @param pSink Mixer sink to return number for.
744 */
745uint8_t AudioMixerSinkGetStreamCount(PAUDMIXSINK pSink)
746{
747 if (!pSink)
748 return 0;
749
750 return pSink->cStreams;
751}
752
753/**
754 * Reads audio data from a mixer sink.
755 *
756 * @returns IPRT status code.
757 * @param pSink Mixer sink to read data from.
758 * @param enmOp Mixer operation to use for reading the data.
759 * @param pvBuf Buffer where to store the read data.
760 * @param cbBuf Buffer size (in bytes) where to store the data.
761 * @param pcbRead Number of bytes read. Optional.
762 */
763int AudioMixerSinkRead(PAUDMIXSINK pSink, AUDMIXOP enmOp, void *pvBuf, uint32_t cbBuf, uint32_t *pcbRead)
764{
765 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
766 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
767 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
768 /* pcbRead is optional. */
769
770 /** @todo Handle mixing operation enmOp! */
771
772 AssertMsg(pSink->enmDir == AUDMIXSINKDIR_INPUT,
773 ("Can't read from a sink which is not an input sink\n"));
774
775#ifndef VBOX_AUDIO_MIXER_WITH_MIXBUF
776 uint8_t *pvMixBuf = (uint8_t *)RTMemAlloc(cbBuf);
777 if (!pvMixBuf)
778 return VERR_NO_MEMORY;
779#endif
780
781 int rc = VERR_NOT_FOUND;
782 uint32_t cbRead = 0;
783
784 /* Flag indicating whether this sink is in a 'clean' state,
785 * e.g. there is no more data to read from. */
786 bool fClean = true;
787
788 PAUDMIXSTREAM pMixStream;
789 RTListForEach(&pSink->lstStreams, pMixStream, AUDMIXSTREAM, Node)
790 {
791 if (!(pMixStream->pConn->pfnStreamGetStatus(pMixStream->pConn, pMixStream->pStream) & PDMAUDIOSTRMSTS_FLAG_ENABLED))
792 {
793 LogFlowFunc(("%s: Stream '%s' Disabled, skipping ...\n", pMixStream->pszName, pMixStream->pStream->szName));
794 continue;
795 }
796
797 uint32_t cbTotalRead = 0;
798 uint32_t cbToRead = cbBuf;
799
800 while (cbToRead)
801 {
802 uint32_t cbReadStrm;
803 AssertPtr(pMixStream->pConn);
804#ifndef VBOX_AUDIO_MIXER_WITH_MIXBUF
805 rc = pMixStream->pConn->pfnStreamRead(pMixStream->pConn, pMixStream->pStream,
806 (uint8_t *)pvMixBuf + cbTotalRead, cbToRead, &cbReadStrm);
807#endif
808 if ( RT_FAILURE(rc)
809 || !cbReadStrm)
810 break;
811
812 /** @todo Right now we only handle one stream (the last one added in fact). */
813
814 AssertBreakStmt(cbReadStrm <= cbToRead, rc = VERR_BUFFER_OVERFLOW);
815 cbToRead -= cbReadStrm;
816 cbTotalRead += cbReadStrm;
817 }
818
819 if (RT_FAILURE(rc))
820 continue;
821
822 cbRead = RT_MAX(cbRead, cbTotalRead);
823
824 PDMAUDIOSTRMSTS strmSts = pMixStream->pConn->pfnStreamGetStatus(pMixStream->pConn, pMixStream->pStream);
825 fClean &= !(strmSts & PDMAUDIOSTRMSTS_FLAG_DATA_READABLE);
826 }
827
828 if (RT_SUCCESS(rc))
829 {
830 if (fClean)
831 pSink->fStatus &= ~AUDMIXSINK_STS_DIRTY;
832
833#ifndef VBOX_AUDIO_MIXER_WITH_MIXBUF
834 if (cbRead)
835 memcpy(pvBuf, pvMixBuf, cbRead);
836#endif
837 if (pcbRead)
838 *pcbRead = cbRead;
839 }
840
841#ifndef VBOX_AUDIO_MIXER_WITH_MIXBUF
842 RTMemFree(pvMixBuf);
843#endif
844
845 Log3Func(("[%s]: cbRead=%RU32, fStatus=0x%x, rc=%Rrc\n", pSink->pszName, cbRead, pSink->fStatus, rc));
846 return rc;
847}
848
849/**
850 * Removes a mixer stream from a mixer sink, internal version.
851 *
852 * @returns IPRT status code.
853 * @param pSink Sink to remove mixer stream from.
854 * @param pStream Stream to remove.
855 */
856static int audioMixerSinkRemoveStreamInternal(PAUDMIXSINK pSink, PAUDMIXSTREAM pStream)
857{
858 AssertPtrReturn(pSink, VERR_INVALID_PARAMETER);
859 if ( !pStream
860 || !pStream->pSink) /* Not part of a sink anymore? */
861 {
862 return VERR_NOT_FOUND;
863 }
864
865 AssertMsgReturn(pStream->pSink == pSink, ("Stream '%s' is not part of sink '%s'\n",
866 pStream->pszName, pSink->pszName), VERR_NOT_FOUND);
867
868 LogFlowFunc(("[%s]: (Stream = %s), cStreams=%RU8\n",
869 pSink->pszName, pStream->pStream->szName, pSink->cStreams));
870
871#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
872 /* Unlink mixing buffer. */
873 AudioMixBufUnlink(&pStream->pStream->MixBuf);
874#endif
875
876 /* Remove stream from sink. */
877 RTListNodeRemove(&pStream->Node);
878
879 /* Set sink to NULL so that we know we're not part of any sink anymore. */
880 pStream->pSink = NULL;
881
882 return VINF_SUCCESS;
883}
884
885/**
886 * Removes a mixer stream from a mixer sink.
887 *
888 * @param pSink Sink to remove mixer stream from.
889 * @param pStream Stream to remove.
890 */
891void AudioMixerSinkRemoveStream(PAUDMIXSINK pSink, PAUDMIXSTREAM pStream)
892{
893 int rc = audioMixerSinkRemoveStreamInternal(pSink, pStream);
894 if (RT_SUCCESS(rc))
895 {
896 Assert(pSink->cStreams);
897 pSink->cStreams--;
898 }
899}
900
901/**
902 * Removes all attached streams from a given sink.
903 *
904 * @param pSink Sink to remove attached streams from.
905 */
906static void audioMixerSinkRemoveAllStreamsInternal(PAUDMIXSINK pSink)
907{
908 if (!pSink)
909 return;
910
911 LogFunc(("%s\n", pSink->pszName));
912
913 PAUDMIXSTREAM pStream, pStreamNext;
914 RTListForEachSafe(&pSink->lstStreams, pStream, pStreamNext, AUDMIXSTREAM, Node)
915 audioMixerSinkRemoveStreamInternal(pSink, pStream);
916}
917
918/**
919 * Resets the sink's state.
920 *
921 * @param pSink Sink to reset.
922 */
923static void audioMixerSinkReset(PAUDMIXSINK pSink)
924{
925 if (!pSink)
926 return;
927
928 LogFunc(("%s\n", pSink->pszName));
929
930 if (pSink->enmDir == AUDMIXSINKDIR_INPUT)
931 {
932#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
933 AudioMixBufReset(&pSink->MixBuf);
934#else
935 pSink->In.cbReadable = 0;
936#endif
937 }
938 else if (pSink->enmDir == AUDMIXSINKDIR_OUTPUT)
939 {
940#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
941 AudioMixBufReset(&pSink->MixBuf);
942#else
943 pSink->Out.cbWritable = 0;
944#endif
945 }
946
947 /* Update last updated timestamp. */
948 pSink->tsLastUpdatedMS = RTTimeMilliTS();
949
950 /* Reset status. */
951 pSink->fStatus = AUDMIXSINK_STS_NONE;
952}
953
954/**
955 * Removes all attached streams from a given sink.
956 *
957 * @param pSink Sink to remove attached streams from.
958 */
959void AudioMixerSinkRemoveAllStreams(PAUDMIXSINK pSink)
960{
961 if (!pSink)
962 return;
963
964 audioMixerSinkRemoveAllStreamsInternal(pSink);
965
966 pSink->cStreams = 0;
967}
968
969/**
970 * Sets the audio format of a mixer sink.
971 *
972 * @returns IPRT status code.
973 * @param pSink Sink to set audio format for.
974 * @param pPCMProps Audio format (PCM properties) to set.
975 */
976int AudioMixerSinkSetFormat(PAUDMIXSINK pSink, PPDMPCMPROPS pPCMProps)
977{
978 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
979 AssertPtrReturn(pPCMProps, VERR_INVALID_POINTER);
980
981 if (DrvAudioHlpPCMPropsAreEqual(&pSink->PCMProps, pPCMProps)) /* Bail out early if PCM properties are equal. */
982 return VINF_SUCCESS;
983
984 if (pSink->PCMProps.uHz)
985 LogFlowFunc(("[%s]: Old format: %RU8 bit, %RU8 channels, %RU32Hz\n",
986 pSink->pszName, pSink->PCMProps.cBits, pSink->PCMProps.cChannels, pSink->PCMProps.uHz));
987
988 memcpy(&pSink->PCMProps, pPCMProps, sizeof(PDMPCMPROPS));
989
990 LogFlowFunc(("[%s]: New format %RU8 bit, %RU8 channels, %RU32Hz\n",
991 pSink->pszName, pSink->PCMProps.cBits, pSink->PCMProps.cChannels, pSink->PCMProps.uHz));
992
993 int rc = VINF_SUCCESS;
994
995#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
996 /* Also update the sink's mixing buffer format. */
997 AudioMixBufDestroy(&pSink->MixBuf);
998 rc = AudioMixBufInit(&pSink->MixBuf, pSink->pszName, pPCMProps, _4K /** @todo Make configurable? */);
999 if (RT_SUCCESS(rc))
1000 {
1001 PAUDMIXSTREAM pStream;
1002 RTListForEach(&pSink->lstStreams, pStream, AUDMIXSTREAM, Node)
1003 {
1004 /** @todo Invalidate mix buffers! */
1005 }
1006 }
1007#endif /* VBOX_AUDIO_MIXER_WITH_MIXBUF */
1008
1009 LogFlowFuncLeaveRC(rc);
1010 return rc;
1011}
1012
1013/**
1014 * Set the volume of an individual sink.
1015 *
1016 * @returns IPRT status code.
1017 * @param pSink Sink to set volume for.
1018 * @param pVol Volume to set.
1019 */
1020int AudioMixerSinkSetVolume(PAUDMIXSINK pSink, PPDMAUDIOVOLUME pVol)
1021{
1022 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1023 AssertPtrReturn(pVol, VERR_INVALID_POINTER);
1024
1025 memcpy(&pSink->Volume, pVol, sizeof(PDMAUDIOVOLUME));
1026
1027 LogFlowFunc(("[%s]: fMuted=%RTbool, lVol=%RU32, rVol=%RU32\n",
1028 pSink->pszName, pSink->Volume.fMuted, pSink->Volume.uLeft, pSink->Volume.uRight));
1029
1030 AssertPtr(pSink->pParent);
1031 return audioMixerSinkUpdateVolume(pSink, &pSink->pParent->VolMaster);
1032}
1033
1034/**
1035 * Updates a mixer sink, internal version.
1036 *
1037 * @returns IPRT status code.
1038 * @param pSink Mixer sink to update.
1039 */
1040static int audioMixerSinkUpdateInternal(PAUDMIXSINK pSink)
1041{
1042 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1043
1044 int rc = VINF_SUCCESS;
1045
1046 Log3Func(("[%s] fStatus=0x%x\n", pSink->pszName, pSink->fStatus));
1047
1048 /* Sink disabled? Take a shortcut. */
1049 if (!(pSink->fStatus & AUDMIXSINK_STS_RUNNING))
1050 return rc;
1051
1052 /* Number of detected disabled streams of this sink. */
1053 uint8_t cStreamsDisabled = 0;
1054
1055 /* Get the time delta and calculate the bytes that need to be processed. */
1056 uint64_t tsDeltaMS = RTTimeMilliTS() - pSink->tsLastUpdatedMS;
1057 uint32_t cbDelta = (pSink->PCMProps.cbBitrate / 1000 /* s to ms */) * tsDeltaMS;
1058
1059 Log3Func(("[%s] Bitrate is %RU32 bytes/s -> %RU64ms / %RU32 bytes elapsed\n",
1060 pSink->pszName, pSink->PCMProps.cbBitrate, tsDeltaMS, cbDelta));
1061
1062 if (pSink->enmDir == AUDMIXSINKDIR_INPUT)
1063 {
1064 pSink->In.cbReadable = cbDelta;
1065 }
1066 else if (pSink->enmDir == AUDMIXSINKDIR_OUTPUT)
1067 {
1068 pSink->Out.cbWritable = cbDelta;
1069 }
1070
1071 uint8_t uCurLUN = 0;
1072
1073 PAUDMIXSTREAM pMixStream, pMixStreamNext;
1074 RTListForEachSafe(&pSink->lstStreams, pMixStream, pMixStreamNext, AUDMIXSTREAM, Node)
1075 {
1076 PPDMAUDIOSTREAM pStream = pMixStream->pStream;
1077 AssertPtr(pStream);
1078
1079 PPDMIAUDIOCONNECTOR pConn = pMixStream->pConn;
1080 AssertPtr(pConn);
1081
1082 uint32_t cPlayed = 0;
1083 uint32_t cCaptured = 0;
1084
1085 int rc2 = pConn->pfnStreamIterate(pConn, pStream);
1086 if (RT_SUCCESS(rc2))
1087 {
1088 if (pSink->enmDir == AUDMIXSINKDIR_INPUT)
1089 {
1090 rc = pConn->pfnStreamCapture(pConn, pMixStream->pStream, &cCaptured);
1091 if (RT_FAILURE(rc2))
1092 {
1093 LogFlowFunc(("%s: Failed capturing stream '%s', rc=%Rrc\n", pSink->pszName, pMixStream->pStream->szName, rc2));
1094 if (RT_SUCCESS(rc))
1095 rc = rc2;
1096 continue;
1097 }
1098
1099 if (cCaptured)
1100 pSink->fStatus |= AUDMIXSINK_STS_DIRTY;
1101 }
1102 else if (pSink->enmDir == AUDMIXSINKDIR_OUTPUT)
1103 {
1104 rc2 = pConn->pfnStreamPlay(pConn, pMixStream->pStream, NULL /* cPlayed */);
1105 if (RT_FAILURE(rc2))
1106 {
1107 LogFlowFunc(("%s: Failed playing stream '%s', rc=%Rrc\n", pSink->pszName, pMixStream->pStream->szName, rc2));
1108 if (RT_SUCCESS(rc))
1109 rc = rc2;
1110 continue;
1111 }
1112 }
1113 else
1114 {
1115 AssertFailedStmt(rc = VERR_NOT_IMPLEMENTED);
1116 continue;
1117 }
1118
1119 rc2 = pConn->pfnStreamIterate(pConn, pStream);
1120 if (RT_FAILURE(rc2))
1121 {
1122 LogFlowFunc(("%s: Failed re-iterating stream '%s', rc=%Rrc\n", pSink->pszName, pMixStream->pStream->szName, rc2));
1123 if (RT_SUCCESS(rc))
1124 rc = rc2;
1125 continue;
1126 }
1127
1128 PDMAUDIOSTRMSTS strmSts = pConn->pfnStreamGetStatus(pConn, pMixStream->pStream);
1129
1130 /* Is the stream not enabled and also is not in a pending disable state anymore? */
1131 if ( !(strmSts & PDMAUDIOSTRMSTS_FLAG_ENABLED)
1132 && !(strmSts & PDMAUDIOSTRMSTS_FLAG_PENDING_DISABLE))
1133 {
1134 cStreamsDisabled++;
1135 }
1136
1137 if (pSink->enmDir == AUDMIXSINKDIR_INPUT)
1138 {
1139#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
1140# error "Implement me!"
1141#else
1142 if (uCurLUN == 0)
1143 {
1144 pSink->In.cbReadable = pConn->pfnStreamGetReadable(pConn, pMixStream->pStream);
1145 Log3Func(("\t%s: cbReadable=%RU32\n", pMixStream->pStream->szName, pSink->In.cbReadable));
1146 uCurLUN++;
1147 }
1148#endif
1149 }
1150 else if (pSink->enmDir == AUDMIXSINKDIR_OUTPUT)
1151 {
1152#ifdef VBOX_AUDIO_MIXER_WITH_MIXBUF
1153# error "Implement me!"
1154#else
1155 if (uCurLUN == 0)
1156 {
1157 pSink->Out.cbWritable = pConn->pfnStreamGetWritable(pConn, pMixStream->pStream);
1158 Log3Func(("\t%s: cbWritable=%RU32\n", pMixStream->pStream->szName, pSink->Out.cbWritable));
1159 uCurLUN++;
1160 }
1161#endif
1162 }
1163 }
1164
1165 Log3Func(("\t%s: cPlayed=%RU32, cCaptured=%RU32\n", pMixStream->pStream->szName, cPlayed, cCaptured));
1166 }
1167
1168 /* All streams disabled and the sink is in pending disable mode? */
1169 if ( cStreamsDisabled == pSink->cStreams
1170 && (pSink->fStatus & AUDMIXSINK_STS_PENDING_DISABLE))
1171 {
1172 audioMixerSinkReset(pSink);
1173 }
1174
1175 /* Update last updated timestamp. */
1176 pSink->tsLastUpdatedMS = RTTimeMilliTS();
1177
1178 Log3Func(("[%s] cbReadable=%RU32, cbWritable=%RU32, rc=%Rrc\n",
1179 pSink->pszName, pSink->In.cbReadable, pSink->Out.cbWritable, rc));
1180
1181 return rc;
1182}
1183
1184/**
1185 * Updates (invalidates) a mixer sink.
1186 *
1187 * @returns IPRT status code.
1188 * @param pSink Mixer sink to update.
1189 */
1190int AudioMixerSinkUpdate(PAUDMIXSINK pSink)
1191{
1192 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1193
1194 /*
1195 * Note: Hz elapsed = cTicksElapsed / cTimerTicks
1196 * Bytes / second = Sample rate (Hz) * Audio channels * Bytes per sample
1197 */
1198// uint32_t cSamples = (int)((pSink->PCMProps.cChannels * cTicksElapsed * pSink->PCMProps.uHz + cTimerTicks) / cTimerTicks / pSink->PCMProps.cChannels);
1199
1200// LogFlowFunc(("[%s]: cTimerTicks=%RU64, cTicksElapsed=%RU64\n", pSink->pszName, cTimerTicks, cTicksElapsed));
1201// LogFlowFunc(("\t%zuHz elapsed, %RU32 samples (%RU32 bytes)\n", cTicksElapsed / cTimerTicks, cSamples, cSamples << 1));
1202
1203 return audioMixerSinkUpdateInternal(pSink);
1204}
1205
1206/**
1207 * Updates the (master) volume of a mixer sink.
1208 *
1209 * @returns IPRT status code.
1210 * @param pSink Mixer sink to update volume for.
1211 * @param pVolMaster Master volume to set.
1212 */
1213static int audioMixerSinkUpdateVolume(PAUDMIXSINK pSink, const PPDMAUDIOVOLUME pVolMaster)
1214{
1215 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1216 AssertPtrReturn(pVolMaster, VERR_INVALID_POINTER);
1217
1218 LogFlowFunc(("[%s]: Master fMuted=%RTbool, lVol=%RU32, rVol=%RU32\n",
1219 pSink->pszName, pVolMaster->fMuted, pVolMaster->uLeft, pVolMaster->uRight));
1220 LogFlowFunc(("[%s]: fMuted=%RTbool, lVol=%RU32, rVol=%RU32 ",
1221 pSink->pszName, pSink->Volume.fMuted, pSink->Volume.uLeft, pSink->Volume.uRight));
1222
1223 /** @todo Very crude implementation for now -- needs more work! */
1224
1225 pSink->VolumeCombined.fMuted = pVolMaster->fMuted || pSink->Volume.fMuted;
1226
1227 pSink->VolumeCombined.uLeft = ( (pSink->Volume.uLeft ? pSink->Volume.uLeft : 1)
1228 * (pVolMaster->uLeft ? pVolMaster->uLeft : 1)) / PDMAUDIO_VOLUME_MAX;
1229
1230 pSink->VolumeCombined.uRight = ( (pSink->Volume.uRight ? pSink->Volume.uRight : 1)
1231 * (pVolMaster->uRight ? pVolMaster->uRight : 1)) / PDMAUDIO_VOLUME_MAX;
1232
1233 LogFlow(("-> fMuted=%RTbool, lVol=%RU32, rVol=%RU32\n",
1234 pSink->VolumeCombined.fMuted, pSink->VolumeCombined.uLeft, pSink->VolumeCombined.uRight));
1235
1236 /* Propagate new sink volume to all streams in the sink. */
1237 PAUDMIXSTREAM pMixStream;
1238 RTListForEach(&pSink->lstStreams, pMixStream, AUDMIXSTREAM, Node)
1239 {
1240 int rc2 = pMixStream->pConn->pfnStreamSetVolume(pMixStream->pConn, pMixStream->pStream, &pSink->VolumeCombined);
1241 AssertRC(rc2);
1242 }
1243
1244 return VINF_SUCCESS;
1245}
1246
1247/**
1248 * Writes data to a mixer sink.
1249 *
1250 * @returns IPRT status code.
1251 * @param pSink Sink to write data to.
1252 * @param enmOp Mixer operation to use when writing data to the sink.
1253 * @param pvBuf Buffer containing the audio data to write.
1254 * @param cbBuf Size (in bytes) of the buffer containing the audio data.
1255 * @param pcbWritten Number of bytes written. Optional.
1256 */
1257int AudioMixerSinkWrite(PAUDMIXSINK pSink, AUDMIXOP enmOp, const void *pvBuf, uint32_t cbBuf, uint32_t *pcbWritten)
1258{
1259 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
1260 /* pcbWritten is optional. */
1261
1262 if (!pvBuf || !cbBuf)
1263 {
1264 if (pcbWritten)
1265 *pcbWritten = 0;
1266 return VINF_SUCCESS;
1267 }
1268
1269 AssertMsg(pSink->enmDir == AUDMIXSINKDIR_OUTPUT,
1270 ("Can't write to a sink which is not an output sink\n"));
1271
1272 LogFlowFunc(("%s: enmOp=%ld, cbBuf=%RU32\n", pSink->pszName, enmOp, cbBuf));
1273
1274 uint32_t cbProcessed;
1275
1276 PAUDMIXSTREAM pMixStream;
1277 RTListForEach(&pSink->lstStreams, pMixStream, AUDMIXSTREAM, Node)
1278 {
1279 if (!(pMixStream->pConn->pfnStreamGetStatus(pMixStream->pConn, pMixStream->pStream) & PDMAUDIOSTRMSTS_FLAG_ENABLED))
1280 {
1281 LogFlowFunc(("%s: Stream '%s' Disabled, skipping ...\n", pMixStream->pszName, pMixStream->pStream->szName));
1282 continue;
1283 }
1284
1285 int rc2 = pMixStream->pConn->pfnStreamWrite(pMixStream->pConn, pMixStream->pStream, pvBuf, cbBuf, &cbProcessed);
1286 if (RT_FAILURE(rc2))
1287 LogFlowFunc(("%s: Failed writing to stream '%s': %Rrc\n", pSink->pszName, pMixStream->pStream->szName, rc2));
1288
1289 if (cbProcessed < cbBuf)
1290 {
1291 LogFlowFunc(("%s: Only written %RU32/%RU32 bytes for stream '%s'\n",
1292 pSink->pszName, cbProcessed, cbBuf, pMixStream->pStream->szName));
1293 }
1294 }
1295
1296 if (cbBuf)
1297 {
1298 /* Set dirty bit. */
1299 pSink->fStatus |= AUDMIXSINK_STS_DIRTY;
1300 }
1301
1302 if (pcbWritten)
1303 *pcbWritten = cbBuf; /* Always report back a complete write for now. */
1304
1305 return VINF_SUCCESS;
1306}
1307
1308/*********************************************************************************************************************************
1309 * Mixer Stream implementation.
1310 ********************************************************************************************************************************/
1311
1312/**
1313 * Controls a mixer stream.
1314 *
1315 * @returns IPRT status code.
1316 * @param pMixStream Mixer stream to control.
1317 * @param enmCmd Mixer stream command to use.
1318 * @param fCtl Additional control flags. Pass 0.
1319 */
1320int AudioMixerStreamCtl(PAUDMIXSTREAM pMixStream, PDMAUDIOSTREAMCMD enmCmd, uint32_t fCtl)
1321{
1322 AssertPtrReturn(pMixStream, VERR_INVALID_POINTER);
1323 /** @todo Validate fCtl. */
1324
1325 LogFlowFunc(("[%s] enmCmd=%ld\n", pMixStream->pszName, enmCmd));
1326
1327 return pMixStream->pConn->pfnStreamControl(pMixStream->pConn, pMixStream->pStream, enmCmd);
1328}
1329
1330/**
1331 * Destroys a mixer stream, internal version.
1332 *
1333 * @param pMixStream Mixer stream to destroy.
1334 */
1335static void audioMixerStreamDestroyInternal(PAUDMIXSTREAM pMixStream)
1336{
1337 AssertPtrReturnVoid(pMixStream);
1338
1339 LogFunc(("%s\n", pMixStream->pszName));
1340
1341 if (pMixStream->pConn) /* Stream has a connector interface present? */
1342 {
1343 if (pMixStream->pStream)
1344 {
1345 pMixStream->pConn->pfnStreamRelease(pMixStream->pConn, pMixStream->pStream);
1346 pMixStream->pConn->pfnStreamDestroy(pMixStream->pConn, pMixStream->pStream);
1347
1348 pMixStream->pStream = NULL;
1349 }
1350
1351 pMixStream->pConn = NULL;
1352 }
1353
1354 if (pMixStream->pszName)
1355 {
1356 RTStrFree(pMixStream->pszName);
1357 pMixStream->pszName = NULL;
1358 }
1359
1360 RTMemFree(pMixStream);
1361 pMixStream = NULL;
1362}
1363
1364/**
1365 * Destroys a mixer stream.
1366 *
1367 * @param pMixStream Mixer stream to destroy.
1368 */
1369void AudioMixerStreamDestroy(PAUDMIXSTREAM pMixStream)
1370{
1371 if (!pMixStream)
1372 return;
1373
1374 LogFunc(("%s\n", pMixStream->pszName));
1375
1376 int rc;
1377
1378 if (pMixStream->pSink) /* Is the stream part of a sink? */
1379 {
1380 /* Save sink pointer, as after audioMixerSinkRemoveStreamInternal() the
1381 * pointer will be gone from the stream. */
1382 PAUDMIXSINK pSink = pMixStream->pSink;
1383
1384 rc = audioMixerSinkRemoveStreamInternal(pSink, pMixStream);
1385 if (RT_SUCCESS(rc))
1386 {
1387 Assert(pSink->cStreams);
1388 pSink->cStreams--;
1389 }
1390 }
1391 else
1392 rc = VINF_SUCCESS;
1393
1394 if (RT_SUCCESS(rc))
1395 audioMixerStreamDestroyInternal(pMixStream);
1396
1397 LogFlowFunc(("Returning %Rrc\n", rc));
1398}
1399
1400/**
1401 * Returns whether a mixer stream currently is active (playing/recording) or not.
1402 *
1403 * @returns @true if playing/recording, @false if not.
1404 * @param pMixStream Mixer stream to return status for.
1405 */
1406bool AudioMixerStreamIsActive(PAUDMIXSTREAM pMixStream)
1407{
1408 bool fIsActive =
1409 pMixStream
1410 && RT_BOOL(pMixStream->pConn->pfnStreamGetStatus(pMixStream->pConn, pMixStream->pStream) & PDMAUDIOSTRMSTS_FLAG_ENABLED);
1411
1412 return fIsActive;
1413}
1414
1415/**
1416 * Returns whether a mixer stream is valid (e.g. initialized and in a working state) or not.
1417 *
1418 * @returns @true if valid, @false if not.
1419 * @param pMixStream Mixer stream to return status for.
1420 */
1421bool AudioMixerStreamIsValid(PAUDMIXSTREAM pMixStream)
1422{
1423 bool fIsValid =
1424 pMixStream
1425 && RT_BOOL(pMixStream->pConn->pfnStreamGetStatus(pMixStream->pConn, pMixStream->pStream) & PDMAUDIOSTRMSTS_FLAG_INITIALIZED);
1426
1427 return fIsValid;
1428}
1429
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