VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/AudioMixBuffer.cpp@ 63848

Last change on this file since 63848 was 63847, checked in by vboxsync, 8 years ago

Audio/AudioMixBuffer.cpp: Return rc in audioMixBufMixTo().

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 65.8 KB
Line 
1/* $Id: AudioMixBuffer.cpp 63847 2016-09-15 09:45:29Z vboxsync $ */
2/** @file
3 * VBox audio: Audio mixing buffer for converting reading/writing audio
4 * samples.
5 */
6
7/*
8 * Copyright (C) 2014-2016 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18#define LOG_GROUP LOG_GROUP_AUDIO_MIXER_BUFFER
19#include <VBox/log.h>
20
21#if 0
22/*
23 * AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA enables dumping the raw PCM data
24 * to a file on the host. Be sure to adjust AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA_PATH
25 * to your needs before using this!
26 */
27# define AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA
28# ifdef RT_OS_WINDOWS
29# define AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA_PATH "c:\\temp\\"
30# else
31# define AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA_PATH "/tmp/"
32# endif
33/* Warning: Enabling this will generate *huge* logs! */
34//# define AUDIOMIXBUF_DEBUG_MACROS
35#endif
36
37#include <iprt/asm-math.h>
38#include <iprt/assert.h>
39#ifdef AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA
40# include <iprt/file.h>
41#endif
42#include <iprt/mem.h>
43#include <iprt/string.h> /* For RT_BZERO. */
44
45#ifdef VBOX_AUDIO_TESTCASE
46# define LOG_ENABLED
47# include <iprt/stream.h>
48#endif
49#include <VBox/err.h>
50
51#include "AudioMixBuffer.h"
52
53#ifndef VBOX_AUDIO_TESTCASE
54# ifdef DEBUG
55# define AUDMIXBUF_LOG(x) LogFlowFunc(x)
56# else
57# define AUDMIXBUF_LOG(x) do {} while (0)
58# endif
59#else /* VBOX_AUDIO_TESTCASE */
60# define AUDMIXBUF_LOG(x) RTPrintf x
61#endif
62
63#ifdef DEBUG
64DECLINLINE(void) audioMixBufDbgPrintInternal(PPDMAUDIOMIXBUF pMixBuf);
65#endif
66
67/*
68 * Soft Volume Control
69 *
70 * The external code supplies an 8-bit volume (attenuation) value in the
71 * 0 .. 255 range. This represents 0 to -96dB attenuation where an input
72 * value of 0 corresponds to -96dB and 255 corresponds to 0dB (unchanged).
73 *
74 * Each step thus corresponds to 96 / 256 or 0.375dB. Every 6dB (16 steps)
75 * represents doubling the sample value.
76 *
77 * For internal use, the volume control needs to be converted to a 16-bit
78 * (sort of) exponential value between 1 and 65536. This is used with fixed
79 * point arithmetic such that 65536 means 1.0 and 1 means 1/65536.
80 *
81 * For actual volume calculation, 33.31 fixed point is used. Maximum (or
82 * unattenuated) volume is represented as 0x40000000; conveniently, this
83 * value fits into a uint32_t.
84 *
85 * To enable fast processing, the maximum volume must be a power of two
86 * and must not have a sign when converted to int32_t. While 0x80000000
87 * violates these constraints, 0x40000000 does not.
88 */
89
90
91/** Logarithmic/exponential volume conversion table. */
92static uint32_t s_aVolumeConv[256] = {
93 1, 1, 1, 1, 1, 1, 1, 1, /* 7 */
94 1, 2, 2, 2, 2, 2, 2, 2, /* 15 */
95 2, 2, 2, 2, 2, 3, 3, 3, /* 23 */
96 3, 3, 3, 3, 4, 4, 4, 4, /* 31 */
97 4, 4, 5, 5, 5, 5, 5, 6, /* 39 */
98 6, 6, 6, 7, 7, 7, 8, 8, /* 47 */
99 8, 9, 9, 10, 10, 10, 11, 11, /* 55 */
100 12, 12, 13, 13, 14, 15, 15, 16, /* 63 */
101 17, 17, 18, 19, 20, 21, 22, 23, /* 71 */
102 24, 25, 26, 27, 28, 29, 31, 32, /* 79 */
103 33, 35, 36, 38, 40, 41, 43, 45, /* 87 */
104 47, 49, 52, 54, 56, 59, 61, 64, /* 95 */
105 67, 70, 73, 76, 79, 83, 87, 91, /* 103 */
106 95, 99, 103, 108, 112, 117, 123, 128, /* 111 */
107 134, 140, 146, 152, 159, 166, 173, 181, /* 119 */
108 189, 197, 206, 215, 225, 235, 245, 256, /* 127 */
109 267, 279, 292, 304, 318, 332, 347, 362, /* 135 */
110 378, 395, 412, 431, 450, 470, 490, 512, /* 143 */
111 535, 558, 583, 609, 636, 664, 693, 724, /* 151 */
112 756, 790, 825, 861, 899, 939, 981, 1024, /* 159 */
113 1069, 1117, 1166, 1218, 1272, 1328, 1387, 1448, /* 167 */
114 1512, 1579, 1649, 1722, 1798, 1878, 1961, 2048, /* 175 */
115 2139, 2233, 2332, 2435, 2543, 2656, 2774, 2896, /* 183 */
116 3025, 3158, 3298, 3444, 3597, 3756, 3922, 4096, /* 191 */
117 4277, 4467, 4664, 4871, 5087, 5312, 5547, 5793, /* 199 */
118 6049, 6317, 6597, 6889, 7194, 7512, 7845, 8192, /* 207 */
119 8555, 8933, 9329, 9742, 10173, 10624, 11094, 11585, /* 215 */
120 12098, 12634, 13193, 13777, 14387, 15024, 15689, 16384, /* 223 */
121 17109, 17867, 18658, 19484, 20347, 21247, 22188, 23170, /* 231 */
122 24196, 25268, 26386, 27554, 28774, 30048, 31379, 32768, /* 239 */
123 34219, 35734, 37316, 38968, 40693, 42495, 44376, 46341, /* 247 */
124 48393, 50535, 52773, 55109, 57549, 60097, 62757, 65536, /* 255 */
125};
126
127/* Bit shift for fixed point conversion. */
128#define AUDIOMIXBUF_VOL_SHIFT 30
129
130/* Internal representation of 0dB volume (1.0 in fixed point). */
131#define AUDIOMIXBUF_VOL_0DB (1 << AUDIOMIXBUF_VOL_SHIFT)
132
133AssertCompile(AUDIOMIXBUF_VOL_0DB <= 0x40000000); /* Must always hold. */
134AssertCompile(AUDIOMIXBUF_VOL_0DB == 0x40000000); /* For now -- when only attenuation is used. */
135
136#ifdef DEBUG
137static uint64_t s_cSamplesMixedTotal = 0;
138#endif
139
140
141/**
142 * Acquires (reads) a mutable pointer to the mixing buffer's audio samples without
143 * any conversion done.
144 ** @todo Rename to AudioMixBufPeek(Mutable/Raw)?
145 ** @todo Protect the buffer's data?
146 *
147 * @return IPRT status code. VINF_TRY_AGAIN for getting next pointer at beginning (circular).
148 * @param pMixBuf Mixing buffer to acquire audio samples from.
149 * @param cSamplesToRead Number of audio samples to read.
150 * @param ppvSamples Returns a mutable pointer to the buffer's audio sample data.
151 * @param pcSamplesRead Number of audio samples read (acquired).
152 *
153 * @remark This function is not thread safe!
154 */
155int AudioMixBufAcquire(PPDMAUDIOMIXBUF pMixBuf, uint32_t cSamplesToRead,
156 PPDMAUDIOSAMPLE *ppvSamples, uint32_t *pcSamplesRead)
157{
158 AssertPtrReturn(pMixBuf, VERR_INVALID_POINTER);
159 AssertPtrReturn(ppvSamples, VERR_INVALID_POINTER);
160 AssertPtrReturn(pcSamplesRead, VERR_INVALID_POINTER);
161
162 int rc;
163
164 if (!cSamplesToRead)
165 {
166 *pcSamplesRead = 0;
167 return VINF_SUCCESS;
168 }
169
170 uint32_t cSamplesRead;
171 if (pMixBuf->offRead + cSamplesToRead > pMixBuf->cSamples)
172 {
173 cSamplesRead = pMixBuf->cSamples - pMixBuf->offRead;
174 rc = VINF_TRY_AGAIN;
175 }
176 else
177 {
178 cSamplesRead = cSamplesToRead;
179 rc = VINF_SUCCESS;
180 }
181
182 *ppvSamples = &pMixBuf->pSamples[pMixBuf->offRead];
183 AssertPtr(ppvSamples);
184
185 pMixBuf->offRead = (pMixBuf->offRead + cSamplesRead) % pMixBuf->cSamples;
186 Assert(pMixBuf->offRead <= pMixBuf->cSamples);
187 pMixBuf->cUsed -= RT_MIN(cSamplesRead, pMixBuf->cUsed);
188
189 *pcSamplesRead = cSamplesRead;
190
191 return rc;
192}
193
194/**
195 * Clears the entire sample buffer.
196 *
197 * @param pMixBuf Mixing buffer to clear.
198 *
199 */
200void AudioMixBufClear(PPDMAUDIOMIXBUF pMixBuf)
201{
202 AssertPtrReturnVoid(pMixBuf);
203
204 if (pMixBuf->cSamples)
205 RT_BZERO(pMixBuf->pSamples, pMixBuf->cSamples * sizeof(PDMAUDIOSAMPLE));
206}
207
208/**
209 * Clears (zeroes) the buffer by a certain amount of (used) samples and
210 * keeps track to eventually assigned children buffers.
211 *
212 * @param pMixBuf Mixing buffer to clear.
213 * @param cSamplesToClear Number of audio samples to clear.
214 */
215void AudioMixBufFinish(PPDMAUDIOMIXBUF pMixBuf, uint32_t cSamplesToClear)
216{
217 AUDMIXBUF_LOG(("cSamplesToClear=%RU32\n", cSamplesToClear));
218 AUDMIXBUF_LOG(("%s: offRead=%RU32, cUsed=%RU32\n",
219 pMixBuf->pszName, pMixBuf->offRead, pMixBuf->cUsed));
220
221 PPDMAUDIOMIXBUF pIter;
222 RTListForEach(&pMixBuf->lstChildren, pIter, PDMAUDIOMIXBUF, Node)
223 {
224 AUDMIXBUF_LOG(("\t%s: cMixed=%RU32 -> %RU32\n",
225 pIter->pszName, pIter->cMixed, pIter->cMixed - cSamplesToClear));
226
227 pIter->cMixed -= RT_MIN(pIter->cMixed, cSamplesToClear);
228 }
229
230 Assert(cSamplesToClear <= pMixBuf->cSamples);
231
232 uint32_t cClearOff;
233 uint32_t cClearLen;
234
235 /* Clear end of buffer (wrap around). */
236 if (cSamplesToClear > pMixBuf->offRead)
237 {
238 cClearOff = pMixBuf->cSamples - (cSamplesToClear - pMixBuf->offRead);
239 cClearLen = pMixBuf->cSamples - cClearOff;
240
241 AUDMIXBUF_LOG(("Clearing1: %RU32 - %RU32\n", cClearOff, cClearOff + cClearLen));
242
243 RT_BZERO(pMixBuf->pSamples + cClearOff, cClearLen * sizeof(PDMAUDIOSAMPLE));
244
245 Assert(cSamplesToClear >= cClearLen);
246 cSamplesToClear -= cClearLen;
247 }
248
249 /* Clear beginning of buffer. */
250 if ( cSamplesToClear
251 && pMixBuf->offRead)
252 {
253 Assert(pMixBuf->offRead >= cSamplesToClear);
254
255 cClearOff = pMixBuf->offRead - cSamplesToClear;
256 cClearLen = cSamplesToClear;
257
258 AUDMIXBUF_LOG(("Clearing2: %RU32 - %RU32\n", cClearOff, cClearOff + cClearLen));
259
260 RT_BZERO(pMixBuf->pSamples + cClearOff, cClearLen * sizeof(PDMAUDIOSAMPLE));
261 }
262}
263
264/**
265 * Destroys (uninitializes) a mixing buffer.
266 *
267 * @param pMixBuf Mixing buffer to destroy.
268 */
269void AudioMixBufDestroy(PPDMAUDIOMIXBUF pMixBuf)
270{
271 if (!pMixBuf)
272 return;
273
274 AudioMixBufUnlink(pMixBuf);
275
276 if (pMixBuf->pszName)
277 {
278 AUDMIXBUF_LOG(("%s\n", pMixBuf->pszName));
279
280 RTStrFree(pMixBuf->pszName);
281 pMixBuf->pszName = NULL;
282 }
283
284 if (pMixBuf->pRate)
285 {
286 RTMemFree(pMixBuf->pRate);
287 pMixBuf->pRate = NULL;
288 }
289
290 if (pMixBuf->pSamples)
291 {
292 Assert(pMixBuf->cSamples);
293
294 RTMemFree(pMixBuf->pSamples);
295 pMixBuf->pSamples = NULL;
296 }
297
298 pMixBuf->cSamples = 0;
299}
300
301/**
302 * Returns the size (in audio samples) of free audio buffer space.
303 *
304 * @return uint32_t Size (in audio samples) of free audio buffer space.
305 * @param pMixBuf Mixing buffer to return free size for.
306 */
307uint32_t AudioMixBufFree(PPDMAUDIOMIXBUF pMixBuf)
308{
309 AssertPtrReturn(pMixBuf, 0);
310
311 uint32_t cSamples, cSamplesFree;
312 if (pMixBuf->pParent)
313 {
314 /*
315 * As a linked child buffer we want to know how many samples
316 * already have been consumed by the parent.
317 */
318 cSamples = pMixBuf->pParent->cSamples;
319
320 Assert(pMixBuf->cMixed <= cSamples);
321 cSamplesFree = cSamples - pMixBuf->cMixed;
322 }
323 else /* As a parent. */
324 {
325 cSamples = pMixBuf->cSamples;
326 Assert(cSamples >= pMixBuf->cUsed);
327 cSamplesFree = pMixBuf->cSamples - pMixBuf->cUsed;
328 }
329
330 AUDMIXBUF_LOG(("%s: %RU32 of %RU32\n", pMixBuf->pszName, cSamplesFree, cSamples));
331 return cSamplesFree;
332}
333
334/**
335 * Returns the size (in bytes) of free audio buffer space.
336 *
337 * @return uint32_t Size (in bytes) of free audio buffer space.
338 * @param pMixBuf Mixing buffer to return free size for.
339 */
340uint32_t AudioMixBufFreeBytes(PPDMAUDIOMIXBUF pMixBuf)
341{
342 return AUDIOMIXBUF_S2B(pMixBuf, AudioMixBufFree(pMixBuf));
343}
344
345/**
346 * Allocates the internal audio sample buffer.
347 *
348 * @return IPRT status code.
349 * @param pMixBuf Mixing buffer to allocate sample buffer for.
350 * @param cSamples Number of audio samples to allocate.
351 */
352static int audioMixBufAlloc(PPDMAUDIOMIXBUF pMixBuf, uint32_t cSamples)
353{
354 AssertPtrReturn(pMixBuf, VERR_INVALID_POINTER);
355 AssertReturn(cSamples, VERR_INVALID_PARAMETER);
356
357 AUDMIXBUF_LOG(("%s: cSamples=%RU32\n", pMixBuf->pszName, cSamples));
358
359 size_t cbSamples = cSamples * sizeof(PDMAUDIOSAMPLE);
360 pMixBuf->pSamples = (PPDMAUDIOSAMPLE)RTMemAllocZ(cbSamples);
361 if (pMixBuf->pSamples)
362 {
363 pMixBuf->cSamples = cSamples;
364 return VINF_SUCCESS;
365 }
366 return VERR_NO_MEMORY;
367}
368
369#ifdef AUDIOMIXBUF_DEBUG_MACROS
370# define AUDMIXBUF_MACRO_LOG(x) AUDMIXBUF_LOG(x)
371#elif defined(VBOX_AUDIO_TESTCASE_VERBOSE) /* Warning: VBOX_AUDIO_TESTCASE_VERBOSE will generate huge logs! */
372# define AUDMIXBUF_MACRO_LOG(x) RTPrintf x
373#else
374# define AUDMIXBUF_MACRO_LOG(x) do {} while (0)
375#endif
376
377/**
378 * Macro for generating the conversion routines from/to different formats.
379 * Be careful what to pass in/out, as most of the macros are optimized for speed and
380 * thus don't do any bounds checking!
381 *
382 * Note: Currently does not handle any endianness conversion yet!
383 */
384#define AUDMIXBUF_CONVERT(_aName, _aType, _aMin, _aMax, _aSigned, _aShift) \
385 /* Clips a specific output value to a single sample value. */ \
386 DECLCALLBACK(int64_t) audioMixBufClipFrom##_aName(_aType aVal) \
387 { \
388 if (_aSigned) \
389 return ((int64_t) aVal) << (32 - _aShift); \
390 return ((int64_t) aVal - ((_aMax >> 1) + 1)) << (32 - _aShift); \
391 } \
392 \
393 /* Clips a single sample value to a specific output value. */ \
394 DECLCALLBACK(_aType) audioMixBufClipTo##_aName(int64_t iVal) \
395 { \
396 if (iVal >= 0x7fffffff) \
397 return _aMax; \
398 if (iVal < -INT64_C(0x80000000)) \
399 return _aMin; \
400 \
401 if (_aSigned) \
402 return (_aType) (iVal >> (32 - _aShift)); \
403 return ((_aType) ((iVal >> (32 - _aShift)) + ((_aMax >> 1) + 1))); \
404 } \
405 \
406 DECLCALLBACK(uint32_t) audioMixBufConvFrom##_aName##Stereo(PPDMAUDIOSAMPLE paDst, const void *pvSrc, uint32_t cbSrc, \
407 PCPDMAUDMIXBUFCONVOPTS pOpts) \
408 { \
409 _aType const *pSrc = (_aType const *)pvSrc; \
410 uint32_t cSamples = RT_MIN(pOpts->cSamples, cbSrc / sizeof(_aType)); \
411 AUDMIXBUF_MACRO_LOG(("cSamples=%RU32, BpS=%zu, lVol=%RU32, rVol=%RU32\n", \
412 pOpts->cSamples, sizeof(_aType), pOpts->From.Volume.uLeft, pOpts->From.Volume.uRight)); \
413 for (uint32_t i = 0; i < cSamples; i++) \
414 { \
415 paDst->i64LSample = ASMMult2xS32RetS64((int32_t)audioMixBufClipFrom##_aName(*pSrc++), pOpts->From.Volume.uLeft ) >> AUDIOMIXBUF_VOL_SHIFT; \
416 paDst->i64RSample = ASMMult2xS32RetS64((int32_t)audioMixBufClipFrom##_aName(*pSrc++), pOpts->From.Volume.uRight) >> AUDIOMIXBUF_VOL_SHIFT; \
417 paDst++; \
418 } \
419 \
420 return cSamples; \
421 } \
422 \
423 DECLCALLBACK(uint32_t) audioMixBufConvFrom##_aName##Mono(PPDMAUDIOSAMPLE paDst, const void *pvSrc, uint32_t cbSrc, \
424 PCPDMAUDMIXBUFCONVOPTS pOpts) \
425 { \
426 _aType const *pSrc = (_aType const *)pvSrc; \
427 const uint32_t cSamples = RT_MIN(pOpts->cSamples, cbSrc / sizeof(_aType)); \
428 AUDMIXBUF_MACRO_LOG(("cSamples=%RU32, BpS=%zu, lVol=%RU32, rVol=%RU32\n", \
429 cSamples, sizeof(_aType), pOpts->From.Volume.uLeft, pOpts->From.Volume.uRight)); \
430 for (uint32_t i = 0; i < cSamples; i++) \
431 { \
432 paDst->i64LSample = ASMMult2xS32RetS64((int32_t)audioMixBufClipFrom##_aName(*pSrc), pOpts->From.Volume.uLeft) >> AUDIOMIXBUF_VOL_SHIFT; \
433 paDst->i64RSample = ASMMult2xS32RetS64((int32_t)audioMixBufClipFrom##_aName(*pSrc), pOpts->From.Volume.uRight) >> AUDIOMIXBUF_VOL_SHIFT; \
434 pSrc++; \
435 paDst++; \
436 } \
437 \
438 return cSamples; \
439 } \
440 \
441 DECLCALLBACK(void) audioMixBufConvTo##_aName##Stereo(void *pvDst, PCPDMAUDIOSAMPLE paSrc, PCPDMAUDMIXBUFCONVOPTS pOpts) \
442 { \
443 PCPDMAUDIOSAMPLE pSrc = paSrc; \
444 _aType *pDst = (_aType *)pvDst; \
445 _aType l, r; \
446 uint32_t cSamples = pOpts->cSamples; \
447 while (cSamples--) \
448 { \
449 AUDMIXBUF_MACRO_LOG(("%p: l=%RI64, r=%RI64\n", pSrc, pSrc->i64LSample, pSrc->i64RSample)); \
450 l = audioMixBufClipTo##_aName(pSrc->i64LSample); \
451 r = audioMixBufClipTo##_aName(pSrc->i64RSample); \
452 AUDMIXBUF_MACRO_LOG(("\t-> l=%RI16, r=%RI16\n", l, r)); \
453 *pDst++ = l; \
454 *pDst++ = r; \
455 pSrc++; \
456 } \
457 } \
458 \
459 DECLCALLBACK(void) audioMixBufConvTo##_aName##Mono(void *pvDst, PCPDMAUDIOSAMPLE paSrc, PCPDMAUDMIXBUFCONVOPTS pOpts) \
460 { \
461 PCPDMAUDIOSAMPLE pSrc = paSrc; \
462 _aType *pDst = (_aType *)pvDst; \
463 uint32_t cSamples = pOpts->cSamples; \
464 while (cSamples--) \
465 { \
466 *pDst++ = audioMixBufClipTo##_aName((pSrc->i64LSample + pSrc->i64RSample) / 2); \
467 pSrc++; \
468 } \
469 }
470
471/* audioMixBufConvXXXS8: 8 bit, signed. */
472AUDMIXBUF_CONVERT(S8 /* Name */, int8_t, INT8_MIN /* Min */, INT8_MAX /* Max */, true /* fSigned */, 8 /* cShift */)
473/* audioMixBufConvXXXU8: 8 bit, unsigned. */
474AUDMIXBUF_CONVERT(U8 /* Name */, uint8_t, 0 /* Min */, UINT8_MAX /* Max */, false /* fSigned */, 8 /* cShift */)
475/* audioMixBufConvXXXS16: 16 bit, signed. */
476AUDMIXBUF_CONVERT(S16 /* Name */, int16_t, INT16_MIN /* Min */, INT16_MAX /* Max */, true /* fSigned */, 16 /* cShift */)
477/* audioMixBufConvXXXU16: 16 bit, unsigned. */
478AUDMIXBUF_CONVERT(U16 /* Name */, uint16_t, 0 /* Min */, UINT16_MAX /* Max */, false /* fSigned */, 16 /* cShift */)
479/* audioMixBufConvXXXS32: 32 bit, signed. */
480AUDMIXBUF_CONVERT(S32 /* Name */, int32_t, INT32_MIN /* Min */, INT32_MAX /* Max */, true /* fSigned */, 32 /* cShift */)
481/* audioMixBufConvXXXU32: 32 bit, unsigned. */
482AUDMIXBUF_CONVERT(U32 /* Name */, uint32_t, 0 /* Min */, UINT32_MAX /* Max */, false /* fSigned */, 32 /* cShift */)
483
484#undef AUDMIXBUF_CONVERT
485
486#define AUDMIXBUF_MIXOP(_aName, _aOp) \
487 static void audioMixBufOp##_aName(PPDMAUDIOSAMPLE paDst, uint32_t cDstSamples, \
488 PPDMAUDIOSAMPLE paSrc, uint32_t cSrcSamples, \
489 PPDMAUDIOSTRMRATE pRate, \
490 uint32_t *pcDstWritten, uint32_t *pcSrcRead) \
491 { \
492 AUDMIXBUF_MACRO_LOG(("cSrcSamples=%RU32, cDstSamples=%RU32\n", cSrcSamples, cDstSamples)); \
493 AUDMIXBUF_MACRO_LOG(("Rate: srcOffset=%RU32, dstOffset=%RU32, dstInc=%RU32\n", \
494 pRate->srcOffset, \
495 (uint32_t)(pRate->dstOffset >> 32), (uint32_t)(pRate->dstInc >> 32))); \
496 \
497 if (pRate->dstInc == (UINT64_C(1) + UINT32_MAX)) /* No conversion needed? */ \
498 { \
499 uint32_t cSamples = RT_MIN(cSrcSamples, cDstSamples); \
500 AUDMIXBUF_MACRO_LOG(("cSamples=%RU32\n", cSamples)); \
501 for (uint32_t i = 0; i < cSamples; i++) \
502 { \
503 paDst[i].i64LSample _aOp paSrc[i].i64LSample; \
504 paDst[i].i64RSample _aOp paSrc[i].i64RSample; \
505 } \
506 \
507 if (pcDstWritten) \
508 *pcDstWritten = cSamples; \
509 if (pcSrcRead) \
510 *pcSrcRead = cSamples; \
511 return; \
512 } \
513 \
514 PPDMAUDIOSAMPLE paSrcStart = paSrc; \
515 PPDMAUDIOSAMPLE paSrcEnd = paSrc + cSrcSamples; \
516 PPDMAUDIOSAMPLE paDstStart = paDst; \
517 PPDMAUDIOSAMPLE paDstEnd = paDst + cDstSamples; \
518 PDMAUDIOSAMPLE samCur = { 0 }; \
519 PDMAUDIOSAMPLE samOut; \
520 PDMAUDIOSAMPLE samLast = pRate->srcSampleLast; \
521 \
522 while (paDst < paDstEnd) \
523 { \
524 Assert(paSrc <= paSrcEnd); \
525 Assert(paDst <= paDstEnd); \
526 if (paSrc >= paSrcEnd) \
527 break; \
528 \
529 while (pRate->srcOffset <= (pRate->dstOffset >> 32)) \
530 { \
531 Assert(paSrc <= paSrcEnd); \
532 samLast = *paSrc++; \
533 pRate->srcOffset++; \
534 if (paSrc == paSrcEnd) \
535 break; \
536 } \
537 \
538 Assert(paSrc <= paSrcEnd); \
539 if (paSrc == paSrcEnd) \
540 break; \
541 \
542 samCur = *paSrc; \
543 \
544 /* Interpolate. */ \
545 int64_t iDstOffInt = pRate->dstOffset & UINT32_MAX; \
546 \
547 samOut.i64LSample = (samLast.i64LSample * ((int64_t) (INT64_C(1) << 32) - iDstOffInt) + samCur.i64LSample * iDstOffInt) >> 32; \
548 samOut.i64RSample = (samLast.i64RSample * ((int64_t) (INT64_C(1) << 32) - iDstOffInt) + samCur.i64RSample * iDstOffInt) >> 32; \
549 \
550 paDst->i64LSample _aOp samOut.i64LSample; \
551 paDst->i64RSample _aOp samOut.i64RSample; \
552 \
553 AUDMIXBUF_MACRO_LOG(("\tiDstOffInt=%RI64, l=%RI64, r=%RI64 (cur l=%RI64, r=%RI64)\n", \
554 iDstOffInt, \
555 paDst->i64LSample >> 32, paDst->i64RSample >> 32, \
556 samCur.i64LSample >> 32, samCur.i64RSample >> 32)); \
557 \
558 paDst++; \
559 pRate->dstOffset += pRate->dstInc; \
560 \
561 AUDMIXBUF_MACRO_LOG(("\t\tpRate->dstOffset=%RU32\n", pRate->dstOffset >> 32)); \
562 \
563 } \
564 \
565 AUDMIXBUF_MACRO_LOG(("%zu source samples -> %zu dest samples\n", paSrc - paSrcStart, paDst - paDstStart)); \
566 \
567 pRate->srcSampleLast = samLast; \
568 \
569 AUDMIXBUF_MACRO_LOG(("pRate->srcSampleLast l=%RI64, r=%RI64\n", \
570 pRate->srcSampleLast.i64LSample, pRate->srcSampleLast.i64RSample)); \
571 \
572 if (pcDstWritten) \
573 *pcDstWritten = paDst - paDstStart; \
574 if (pcSrcRead) \
575 *pcSrcRead = paSrc - paSrcStart; \
576 }
577
578/* audioMixBufOpAssign: Assigns values from source buffer to destination bufffer, overwriting the destination. */
579AUDMIXBUF_MIXOP(Assign /* Name */, = /* Operation */)
580#if 0 /* unused */
581/* audioMixBufOpBlend: Blends together the values from both, the source and the destination buffer. */
582AUDMIXBUF_MIXOP(Blend /* Name */, += /* Operation */)
583#endif
584
585#undef AUDMIXBUF_MIXOP
586#undef AUDMIXBUF_MACRO_LOG
587
588/** Dummy conversion used when the source is muted. */
589static DECLCALLBACK(uint32_t)
590audioMixBufConvFromSilence(PPDMAUDIOSAMPLE paDst, const void *pvSrc, uint32_t cbSrc, PCPDMAUDMIXBUFCONVOPTS pOpts)
591{
592 RT_NOREF(cbSrc, pvSrc);
593
594 /* Internally zero always corresponds to silence. */
595 RT_BZERO(paDst, pOpts->cSamples * sizeof(paDst[0]));
596 return pOpts->cSamples;
597}
598
599/**
600 * Looks up the matching conversion (macro) routine for converting
601 * audio samples from a source format.
602 *
603 ** @todo Speed up the lookup by binding it to the actual stream state.
604 *
605 * @return PAUDMIXBUF_FN_CONVFROM Function pointer to conversion macro if found, NULL if not supported.
606 * @param enmFmt Audio format to lookup conversion macro for.
607 */
608static PFNPDMAUDIOMIXBUFCONVFROM audioMixBufConvFromLookup(PDMAUDIOMIXBUFFMT enmFmt)
609{
610 if (AUDMIXBUF_FMT_SIGNED(enmFmt))
611 {
612 if (AUDMIXBUF_FMT_CHANNELS(enmFmt) == 2)
613 {
614 switch (AUDMIXBUF_FMT_BITS_PER_SAMPLE(enmFmt))
615 {
616 case 8: return audioMixBufConvFromS8Stereo;
617 case 16: return audioMixBufConvFromS16Stereo;
618 case 32: return audioMixBufConvFromS32Stereo;
619 default: return NULL;
620 }
621 }
622 else
623 {
624 switch (AUDMIXBUF_FMT_BITS_PER_SAMPLE(enmFmt))
625 {
626 case 8: return audioMixBufConvFromS8Mono;
627 case 16: return audioMixBufConvFromS16Mono;
628 case 32: return audioMixBufConvFromS32Mono;
629 default: return NULL;
630 }
631 }
632 }
633 else /* Unsigned */
634 {
635 if (AUDMIXBUF_FMT_CHANNELS(enmFmt) == 2)
636 {
637 switch (AUDMIXBUF_FMT_BITS_PER_SAMPLE(enmFmt))
638 {
639 case 8: return audioMixBufConvFromU8Stereo;
640 case 16: return audioMixBufConvFromU16Stereo;
641 case 32: return audioMixBufConvFromU32Stereo;
642 default: return NULL;
643 }
644 }
645 else
646 {
647 switch (AUDMIXBUF_FMT_BITS_PER_SAMPLE(enmFmt))
648 {
649 case 8: return audioMixBufConvFromU8Mono;
650 case 16: return audioMixBufConvFromU16Mono;
651 case 32: return audioMixBufConvFromU32Mono;
652 default: return NULL;
653 }
654 }
655 }
656 /* not reached */
657}
658
659/**
660 * Looks up the matching conversion (macro) routine for converting
661 * audio samples to a destination format.
662 *
663 ** @todo Speed up the lookup by binding it to the actual stream state.
664 *
665 * @return PAUDMIXBUF_FN_CONVTO Function pointer to conversion macro if found, NULL if not supported.
666 * @param enmFmt Audio format to lookup conversion macro for.
667 */
668static PFNPDMAUDIOMIXBUFCONVTO audioMixBufConvToLookup(PDMAUDIOMIXBUFFMT enmFmt)
669{
670 if (AUDMIXBUF_FMT_SIGNED(enmFmt))
671 {
672 if (AUDMIXBUF_FMT_CHANNELS(enmFmt) == 2)
673 {
674 switch (AUDMIXBUF_FMT_BITS_PER_SAMPLE(enmFmt))
675 {
676 case 8: return audioMixBufConvToS8Stereo;
677 case 16: return audioMixBufConvToS16Stereo;
678 case 32: return audioMixBufConvToS32Stereo;
679 default: return NULL;
680 }
681 }
682 else
683 {
684 switch (AUDMIXBUF_FMT_BITS_PER_SAMPLE(enmFmt))
685 {
686 case 8: return audioMixBufConvToS8Mono;
687 case 16: return audioMixBufConvToS16Mono;
688 case 32: return audioMixBufConvToS32Mono;
689 default: return NULL;
690 }
691 }
692 }
693 else /* Unsigned */
694 {
695 if (AUDMIXBUF_FMT_CHANNELS(enmFmt) == 2)
696 {
697 switch (AUDMIXBUF_FMT_BITS_PER_SAMPLE(enmFmt))
698 {
699 case 8: return audioMixBufConvToU8Stereo;
700 case 16: return audioMixBufConvToU16Stereo;
701 case 32: return audioMixBufConvToU32Stereo;
702 default: return NULL;
703 }
704 }
705 else
706 {
707 switch (AUDMIXBUF_FMT_BITS_PER_SAMPLE(enmFmt))
708 {
709 case 8: return audioMixBufConvToU8Mono;
710 case 16: return audioMixBufConvToU16Mono;
711 case 32: return audioMixBufConvToU32Mono;
712 default: return NULL;
713 }
714 }
715 }
716 /* not reached */
717}
718
719/**
720 * Converts a PDM audio volume to an internal mixing buffer volume.
721 *
722 * @returns IPRT status code.
723 * @param pVolDst Where to store the converted mixing buffer volume.
724 * @param pVolSrc Volume to convert.
725 */
726static int audioMixBufConvVol(PPDMAUDMIXBUFVOL pVolDst, PPDMAUDIOVOLUME pVolSrc)
727{
728 if (!pVolSrc->fMuted) /* Only change/convert the volume value if we're not muted. */
729 {
730 uint8_t uVolL = pVolSrc->uLeft & 0xFF;
731 uint8_t uVolR = pVolSrc->uRight & 0xFF;
732
733 /** @todo Ensure that the input is in the correct range/initialized! */
734 pVolDst->uLeft = s_aVolumeConv[uVolL] * (AUDIOMIXBUF_VOL_0DB >> 16);
735 pVolDst->uRight = s_aVolumeConv[uVolR] * (AUDIOMIXBUF_VOL_0DB >> 16);
736 }
737
738 pVolDst->fMuted = pVolSrc->fMuted;
739
740 return VINF_SUCCESS;
741}
742
743/**
744 * Initializes a mixing buffer.
745 *
746 * @return IPRT status code.
747 * @param pMixBuf Mixing buffer to initialize.
748 * @param pszName Name of mixing buffer for easier identification. Optional.
749 * @param pProps PCM audio properties to use for the mixing buffer.
750 * @param cSamples Maximum number of audio samples the mixing buffer can hold.
751 */
752int AudioMixBufInit(PPDMAUDIOMIXBUF pMixBuf, const char *pszName, PPDMAUDIOPCMPROPS pProps, uint32_t cSamples)
753{
754 AssertPtrReturn(pMixBuf, VERR_INVALID_POINTER);
755 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
756 AssertPtrReturn(pProps, VERR_INVALID_POINTER);
757
758 pMixBuf->pParent = NULL;
759 RTListInit(&pMixBuf->lstChildren);
760
761 pMixBuf->pSamples = NULL;
762 pMixBuf->cSamples = 0;
763
764 pMixBuf->offRead = 0;
765 pMixBuf->offWrite = 0;
766 pMixBuf->cMixed = 0;
767 pMixBuf->cUsed = 0;
768
769 /* Set initial volume to max. */
770 pMixBuf->Volume.fMuted = false;
771 pMixBuf->Volume.uLeft = AUDIOMIXBUF_VOL_0DB;
772 pMixBuf->Volume.uRight = AUDIOMIXBUF_VOL_0DB;
773
774 /* Prevent division by zero.
775 * Do a 1:1 conversion according to AUDIOMIXBUF_S2B_RATIO. */
776 pMixBuf->iFreqRatio = 1 << 20;
777
778 pMixBuf->pRate = NULL;
779
780 pMixBuf->AudioFmt = AUDMIXBUF_AUDIO_FMT_MAKE(pProps->uHz,
781 pProps->cChannels,
782 pProps->cBits,
783 pProps->fSigned);
784
785 pMixBuf->pfnConvFrom = audioMixBufConvFromLookup(pMixBuf->AudioFmt);
786 pMixBuf->pfnConvTo = audioMixBufConvToLookup(pMixBuf->AudioFmt);
787
788 pMixBuf->cShift = pProps->cShift;
789 pMixBuf->pszName = RTStrDup(pszName);
790 if (!pMixBuf->pszName)
791 return VERR_NO_MEMORY;
792
793 AUDMIXBUF_LOG(("%s: uHz=%RU32, cChan=%RU8, cBits=%RU8, fSigned=%RTbool\n",
794 pMixBuf->pszName,
795 AUDMIXBUF_FMT_SAMPLE_FREQ(pMixBuf->AudioFmt),
796 AUDMIXBUF_FMT_CHANNELS(pMixBuf->AudioFmt),
797 AUDMIXBUF_FMT_BITS_PER_SAMPLE(pMixBuf->AudioFmt),
798 RT_BOOL(AUDMIXBUF_FMT_SIGNED(pMixBuf->AudioFmt))));
799
800 return audioMixBufAlloc(pMixBuf, cSamples);
801}
802
803/**
804 * Returns @true if there are any audio samples available for processing,
805 * @false if not.
806 *
807 * @return bool @true if there are any audio samples available for processing, @false if not.
808 * @param pMixBuf Mixing buffer to return value for.
809 */
810bool AudioMixBufIsEmpty(PPDMAUDIOMIXBUF pMixBuf)
811{
812 AssertPtrReturn(pMixBuf, true);
813
814 if (pMixBuf->pParent)
815 return (pMixBuf->cMixed == 0);
816 return (pMixBuf->cUsed == 0);
817}
818
819/**
820 * Links an audio mixing buffer to a parent mixing buffer. A parent mixing
821 * buffer can have multiple children mixing buffers [1:N], whereas a child only can
822 * have one parent mixing buffer [N:1].
823 *
824 * The mixing direction always goes from the child/children buffer(s) to the
825 * parent buffer.
826 *
827 * For guest audio output the host backend owns the parent mixing buffer, the
828 * device emulation owns the child/children.
829 *
830 * The audio format of each mixing buffer can vary; the internal mixing code
831 * then will automatically do the (needed) conversion.
832 *
833 * @return IPRT status code.
834 * @param pMixBuf Mixing buffer to link parent to.
835 * @param pParent Parent mixing buffer to use for linking.
836 *
837 * @remark Circular linking is not allowed.
838 */
839int AudioMixBufLinkTo(PPDMAUDIOMIXBUF pMixBuf, PPDMAUDIOMIXBUF pParent)
840{
841 AssertPtrReturn(pMixBuf, VERR_INVALID_POINTER);
842 AssertPtrReturn(pParent, VERR_INVALID_POINTER);
843
844 AssertMsgReturn(AUDMIXBUF_FMT_SAMPLE_FREQ(pParent->AudioFmt),
845 ("Parent sample frequency (Hz) not set\n"), VERR_INVALID_PARAMETER);
846 AssertMsgReturn(AUDMIXBUF_FMT_SAMPLE_FREQ(pMixBuf->AudioFmt),
847 ("Buffer sample frequency (Hz) not set\n"), VERR_INVALID_PARAMETER);
848 AssertMsgReturn(pMixBuf != pParent,
849 ("Circular linking not allowed\n"), VERR_INVALID_PARAMETER);
850
851 if (pMixBuf->pParent) /* Already linked? */
852 {
853 AUDMIXBUF_LOG(("%s: Already linked to parent '%s'\n",
854 pMixBuf->pszName, pMixBuf->pParent->pszName));
855 return VERR_ACCESS_DENIED;
856 }
857
858 RTListAppend(&pParent->lstChildren, &pMixBuf->Node);
859 pMixBuf->pParent = pParent;
860
861 /* Calculate the frequency ratio. */
862 pMixBuf->iFreqRatio = ((int64_t)AUDMIXBUF_FMT_SAMPLE_FREQ(pParent->AudioFmt) << 32)
863 / AUDMIXBUF_FMT_SAMPLE_FREQ(pMixBuf->AudioFmt);
864
865 if (pMixBuf->iFreqRatio == 0) /* Catch division by zero. */
866 pMixBuf->iFreqRatio = 1 << 20; /* Do a 1:1 conversion instead. */
867
868 int rc = VINF_SUCCESS;
869#if 0
870 uint32_t cSamples = (uint32_t)RT_MIN( ((uint64_t)pParent->cSamples << 32)
871 / pMixBuf->iFreqRatio, _64K /* 64K samples max. */);
872 if (!cSamples)
873 cSamples = pParent->cSamples;
874
875 int rc = VINF_SUCCESS;
876
877 if (cSamples != pMixBuf->cSamples)
878 {
879 AUDMIXBUF_LOG(("%s: Reallocating samples %RU32 -> %RU32\n",
880 pMixBuf->pszName, pMixBuf->cSamples, cSamples));
881
882 uint32_t cbSamples = cSamples * sizeof(PDMAUDIOSAMPLE);
883 Assert(cbSamples);
884 pMixBuf->pSamples = (PPDMAUDIOSAMPLE)RTMemRealloc(pMixBuf->pSamples, cbSamples);
885 if (!pMixBuf->pSamples)
886 rc = VERR_NO_MEMORY;
887
888 if (RT_SUCCESS(rc))
889 {
890 pMixBuf->cSamples = cSamples;
891
892 /* Make sure to zero the reallocated buffer so that it can be
893 * used properly when blending with another buffer later. */
894 RT_BZERO(pMixBuf->pSamples, cbSamples);
895 }
896 }
897#endif
898
899 if (RT_SUCCESS(rc))
900 {
901 if (!pMixBuf->pRate)
902 {
903 /* Create rate conversion. */
904 pMixBuf->pRate = (PPDMAUDIOSTRMRATE)RTMemAllocZ(sizeof(PDMAUDIOSTRMRATE));
905 if (!pMixBuf->pRate)
906 return VERR_NO_MEMORY;
907 }
908 else
909 RT_BZERO(pMixBuf->pRate, sizeof(PDMAUDIOSTRMRATE));
910
911 pMixBuf->pRate->dstInc = ((uint64_t)AUDMIXBUF_FMT_SAMPLE_FREQ(pMixBuf->AudioFmt) << 32)
912 / AUDMIXBUF_FMT_SAMPLE_FREQ(pParent->AudioFmt);
913
914 AUDMIXBUF_LOG(("uThisHz=%RU32, uParentHz=%RU32, iFreqRatio=0x%RX64 (%RI64), uRateInc=0x%RX64 (%RU64), cSamples=%RU32 (%RU32 parent)\n",
915 AUDMIXBUF_FMT_SAMPLE_FREQ(pMixBuf->AudioFmt),
916 AUDMIXBUF_FMT_SAMPLE_FREQ(pParent->AudioFmt),
917 pMixBuf->iFreqRatio, pMixBuf->iFreqRatio,
918 pMixBuf->pRate->dstInc, pMixBuf->pRate->dstInc,
919 pMixBuf->cSamples,
920 pParent->cSamples));
921 AUDMIXBUF_LOG(("%s (%RU32Hz) -> %s (%RU32Hz)\n",
922 pMixBuf->pszName, AUDMIXBUF_FMT_SAMPLE_FREQ(pMixBuf->AudioFmt),
923 pMixBuf->pParent->pszName, AUDMIXBUF_FMT_SAMPLE_FREQ(pParent->AudioFmt)));
924 }
925
926 return rc;
927}
928
929/**
930 * Returns number of available live samples, that is, samples that
931 * have been written into the mixing buffer but not have been processed yet.
932 *
933 * For a parent buffer, this simply returns the currently used number of samples
934 * in the buffer.
935 *
936 * For a child buffer, this returns the number of samples which have been mixed
937 * to the parent and were not processed by the parent yet.
938 *
939 * @return uint32_t Number of live samples available.
940 * @param pMixBuf Mixing buffer to return value for.
941 */
942uint32_t AudioMixBufLive(PPDMAUDIOMIXBUF pMixBuf)
943{
944 AssertPtrReturn(pMixBuf, 0);
945
946#ifdef RT_STRICT
947 uint32_t cSamples;
948#endif
949 uint32_t cAvail;
950 if (pMixBuf->pParent) /* Is this a child buffer? */
951 {
952#ifdef RT_STRICT
953 /* Use the sample count from the parent, as
954 * pMixBuf->cMixed specifies the sample count
955 * in parent samples. */
956 cSamples = pMixBuf->pParent->cSamples;
957#endif
958 cAvail = pMixBuf->cMixed;
959 }
960 else
961 {
962#ifdef RT_STRICT
963 cSamples = pMixBuf->cSamples;
964#endif
965 cAvail = pMixBuf->cUsed;
966 }
967
968 Assert(cAvail <= cSamples);
969 return cAvail;
970}
971
972/**
973 * Mixes audio samples from a source mixing buffer to a destination mixing buffer.
974 *
975 * @return IPRT status code.
976 * @param pDst Destination mixing buffer.
977 * @param pSrc Source mixing buffer.
978 * @param cSrcSamples Number of source audio samples to mix.
979 * @param pcProcessed Number of audio samples successfully mixed.
980 */
981static int audioMixBufMixTo(PPDMAUDIOMIXBUF pDst, PPDMAUDIOMIXBUF pSrc, uint32_t cSrcSamples, uint32_t *pcProcessed)
982{
983 AssertPtrReturn(pDst, VERR_INVALID_POINTER);
984 AssertPtrReturn(pSrc, VERR_INVALID_POINTER);
985 /* pcProcessed is optional. */
986
987 AssertMsgReturn(pDst == pSrc->pParent, ("Source buffer '%s' is not a child of destination '%s'\n",
988 pSrc->pszName, pDst->pszName), VERR_INVALID_PARAMETER);
989 uint32_t cReadTotal = 0;
990 uint32_t cWrittenTotal = 0;
991
992 if (pSrc->cMixed >= pDst->cSamples)
993 {
994 AUDMIXBUF_LOG(("Warning: Destination buffer '%s' full (%RU32 samples max), got %RU32 mixed samples\n",
995 pDst->pszName, pDst->cSamples, pSrc->cMixed));
996 if (pcProcessed)
997 *pcProcessed = 0;
998 return VINF_SUCCESS;
999 }
1000
1001 Assert(pSrc->cUsed >= pDst->cMixed);
1002
1003 uint32_t cSrcAvail = RT_MIN(cSrcSamples, pSrc->cUsed - pDst->cMixed);
1004 uint32_t offSrcRead = pSrc->offRead;
1005 uint32_t cDstMixed = pSrc->cMixed;
1006
1007 Assert(pDst->cUsed <= pDst->cSamples);
1008 uint32_t cDstAvail = pDst->cSamples - pDst->cUsed;
1009 uint32_t offDstWrite = pDst->offWrite;
1010
1011 if ( !cSrcAvail
1012 || !cDstAvail)
1013 {
1014 if (pcProcessed)
1015 *pcProcessed = 0;
1016 return VINF_SUCCESS;
1017 }
1018
1019 AUDMIXBUF_LOG(("cSrcSamples=%RU32, cSrcAvail=%RU32 -> cDstAvail=%RU32\n", cSrcSamples, cSrcAvail, cDstAvail));
1020
1021#ifdef DEBUG
1022 audioMixBufDbgPrintInternal(pDst);
1023#endif
1024
1025 uint32_t cSrcToRead = 0;
1026 uint32_t cSrcRead;
1027
1028 uint32_t cDstToWrite;
1029 uint32_t cDstWritten;
1030
1031 int rc = VINF_SUCCESS;
1032
1033 while ( cSrcAvail
1034 && cDstAvail)
1035 {
1036 cSrcToRead = RT_MIN(cSrcAvail, pSrc->cSamples - offSrcRead);
1037 cDstToWrite = RT_MIN(cDstAvail, pDst->cSamples - offDstWrite);
1038
1039 AUDMIXBUF_LOG(("\tSource: %RU32 samples available, %RU32 @ %RU32 -> reading %RU32\n", cSrcAvail, offSrcRead, pSrc->cSamples, cSrcToRead));
1040 AUDMIXBUF_LOG(("\tDest : %RU32 samples available, %RU32 @ %RU32 -> writing %RU32\n", cDstAvail, offDstWrite, pDst->cSamples, cDstToWrite));
1041
1042 cDstWritten = cSrcRead = 0;
1043
1044 if ( cDstToWrite
1045 && cSrcToRead)
1046 {
1047 Assert(offSrcRead < pSrc->cSamples);
1048 Assert(offSrcRead + cSrcToRead <= pSrc->cSamples);
1049
1050 Assert(offDstWrite < pDst->cSamples);
1051 Assert(offDstWrite + cDstToWrite <= pDst->cSamples);
1052
1053 audioMixBufOpAssign(pDst->pSamples + offDstWrite, cDstToWrite,
1054 pSrc->pSamples + offSrcRead, cSrcToRead,
1055 pSrc->pRate, &cDstWritten, &cSrcRead);
1056 }
1057
1058 cReadTotal += cSrcRead;
1059 cWrittenTotal += cDstWritten;
1060
1061 offSrcRead = (offSrcRead + cSrcRead) % pSrc->cSamples;
1062 offDstWrite = (offDstWrite + cDstWritten) % pDst->cSamples;
1063
1064 cDstMixed += cDstWritten;
1065
1066 Assert(cSrcAvail >= cSrcRead);
1067 cSrcAvail -= cSrcRead;
1068 Assert(cDstAvail >= cDstWritten);
1069 cDstAvail -= cDstWritten;
1070
1071 AUDMIXBUF_LOG(("\t%RU32 read (%RU32 left), %RU32 written (%RU32 left)\n", cSrcRead, cSrcAvail, cDstWritten, cDstAvail));
1072 }
1073
1074 pSrc->offRead = offSrcRead;
1075 Assert(pSrc->cUsed >= cReadTotal);
1076 pSrc->cUsed -= cReadTotal;
1077
1078 /* Note: Always count in parent samples, as the rate can differ! */
1079 pSrc->cMixed = RT_MIN(cDstMixed, pDst->cSamples);
1080
1081 pDst->offWrite = offDstWrite;
1082 Assert(pDst->offWrite <= pDst->cSamples);
1083 Assert((pDst->cUsed + cWrittenTotal) <= pDst->cSamples);
1084 pDst->cUsed += cWrittenTotal;
1085
1086 /* If there are more used samples than fitting in the destination buffer,
1087 * adjust the values accordingly.
1088 *
1089 * This can happen if this routine has been called too often without
1090 * actually processing the destination buffer in between. */
1091 if (pDst->cUsed > pDst->cSamples)
1092 {
1093 LogFlowFunc(("Warning: Destination buffer used %RU32 / %RU32 samples\n", pDst->cUsed, pDst->cSamples));
1094 pDst->offWrite = 0;
1095 pDst->cUsed = pDst->cSamples;
1096
1097 rc = VERR_BUFFER_OVERFLOW;
1098 }
1099 else if (!cSrcToRead && cDstAvail)
1100 {
1101 AUDMIXBUF_LOG(("Warning: Source buffer '%s' ran out of data\n", pSrc->pszName));
1102 rc = VERR_BUFFER_UNDERFLOW;
1103 }
1104 else if (cSrcAvail && !cDstAvail)
1105 {
1106 AUDMIXBUF_LOG(("Warning: Destination buffer '%s' full (%RU32 source samples left)\n", pDst->pszName, cSrcAvail));
1107 rc = VERR_BUFFER_OVERFLOW;
1108 }
1109
1110#ifdef DEBUG
1111 s_cSamplesMixedTotal += cWrittenTotal;
1112 audioMixBufDbgPrintInternal(pDst);
1113#endif
1114
1115 if (pcProcessed)
1116 *pcProcessed = cReadTotal;
1117
1118 AUDMIXBUF_LOG(("cReadTotal=%RU32 (pcProcessed), cWrittenTotal=%RU32, cSrcMixed=%RU32, cDstUsed=%RU32, rc=%Rrc\n",
1119 cReadTotal, cWrittenTotal, pSrc->cMixed, pDst->cUsed, rc));
1120 return rc;
1121}
1122
1123/**
1124 * Mixes audio samples down to the parent mixing buffer.
1125 *
1126 * @return IPRT status code.
1127 * @param pMixBuf Mixing buffer to mix samples down to parent.
1128 * @param cSamples Number of audio samples of specified mixing buffer to to mix
1129 * to its attached parent mixing buffer (if any).
1130 * @param pcProcessed Number of audio samples successfully processed. Optional.
1131 */
1132int AudioMixBufMixToParent(PPDMAUDIOMIXBUF pMixBuf, uint32_t cSamples,
1133 uint32_t *pcProcessed)
1134{
1135 AssertMsgReturn(VALID_PTR(pMixBuf->pParent),
1136 ("Buffer is not linked to a parent buffer\n"),
1137 VERR_INVALID_PARAMETER);
1138
1139 return audioMixBufMixTo(pMixBuf->pParent, pMixBuf, cSamples, pcProcessed);
1140}
1141
1142#ifdef DEBUG
1143
1144/**
1145 * Prints a single mixing buffer.
1146 * Internal helper function for debugging. Do not use directly.
1147 *
1148 * @return IPRT status code.
1149 * @param pMixBuf Mixing buffer to print.
1150 * @param fIsParent Whether this is a parent buffer or not.
1151 * @param uIdtLvl Indention level to use.
1152 */
1153DECL_FORCE_INLINE(void) audioMixBufDbgPrintSingle(PPDMAUDIOMIXBUF pMixBuf, bool fIsParent, uint16_t uIdtLvl)
1154{
1155 AUDMIXBUF_LOG(("%*s[%s] %s: offRead=%RU32, offWrite=%RU32, cMixed=%RU32 -> %RU32/%RU32\n",
1156 uIdtLvl * 4, "", fIsParent ? "PARENT" : "CHILD",
1157 pMixBuf->pszName, pMixBuf->offRead, pMixBuf->offWrite, pMixBuf->cMixed, pMixBuf->cUsed, pMixBuf->cSamples));
1158}
1159
1160/**
1161 * Internal helper function for audioMixBufPrintChain().
1162 * Do not use directly.
1163 *
1164 * @return IPRT status code.
1165 * @param pMixBuf Mixing buffer to print.
1166 * @param uIdtLvl Indention level to use.
1167 * @param pcChildren Pointer to children counter.
1168 */
1169DECL_FORCE_INLINE(void) audioMixBufDbgPrintChainHelper(PPDMAUDIOMIXBUF pMixBuf, uint16_t uIdtLvl, size_t *pcChildren)
1170{
1171 PPDMAUDIOMIXBUF pIter;
1172 RTListForEach(&pMixBuf->lstChildren, pIter, PDMAUDIOMIXBUF, Node)
1173 {
1174 audioMixBufDbgPrintSingle(pIter, false /* ifIsParent */, uIdtLvl + 1);
1175 *pcChildren++;
1176 }
1177}
1178
1179DECL_FORCE_INLINE(void) audioMixBufDbgPrintChainInternal(PPDMAUDIOMIXBUF pMixBuf)
1180{
1181 PPDMAUDIOMIXBUF pParent = pMixBuf->pParent;
1182 while (pParent)
1183 {
1184 if (!pParent->pParent)
1185 break;
1186
1187 pParent = pParent->pParent;
1188 }
1189
1190 if (!pParent)
1191 pParent = pMixBuf;
1192
1193 AUDMIXBUF_LOG(("********************************************\n"));
1194
1195 audioMixBufDbgPrintSingle(pParent, true /* fIsParent */, 0 /* uIdtLvl */);
1196
1197 /* Recursively iterate children. */
1198 size_t cChildren = 0;
1199 audioMixBufDbgPrintChainHelper(pParent, 0 /* uIdtLvl */, &cChildren);
1200
1201 AUDMIXBUF_LOG(("Children: %zu - Total samples mixed: %RU64\n", cChildren, s_cSamplesMixedTotal));
1202 AUDMIXBUF_LOG(("********************************************\n"));
1203}
1204
1205/**
1206 * Prints statistics and status of the full chain of a mixing buffer to the logger,
1207 * starting from the top root mixing buffer.
1208 * For debug versions only.
1209 *
1210 * @return IPRT status code.
1211 * @param pMixBuf Mixing buffer to print.
1212 */
1213void AudioMixBufDbgPrintChain(PPDMAUDIOMIXBUF pMixBuf)
1214{
1215 audioMixBufDbgPrintChainInternal(pMixBuf);
1216}
1217
1218DECL_FORCE_INLINE(void) audioMixBufDbgPrintInternal(PPDMAUDIOMIXBUF pMixBuf)
1219{
1220 PPDMAUDIOMIXBUF pParent = pMixBuf;
1221 if (pMixBuf->pParent)
1222 pParent = pMixBuf->pParent;
1223
1224 AUDMIXBUF_LOG(("***************************************************************************************\n"));
1225
1226 audioMixBufDbgPrintSingle(pMixBuf, pParent == pMixBuf /* fIsParent */, 0 /* iIdtLevel */);
1227
1228 PPDMAUDIOMIXBUF pIter;
1229 RTListForEach(&pParent->lstChildren, pIter, PDMAUDIOMIXBUF, Node)
1230 {
1231 if (pIter == pMixBuf)
1232 continue;
1233 audioMixBufDbgPrintSingle(pIter, false /* fIsParent */, 1 /* iIdtLevel */);
1234 }
1235
1236 AUDMIXBUF_LOG(("***************************************************************************************\n"));
1237}
1238
1239/**
1240 * Prints statistics and status of a mixing buffer to the logger.
1241 * For debug versions only.
1242 *
1243 * @return IPRT status code.
1244 * @param pMixBuf Mixing buffer to print.
1245 */
1246void AudioMixBufDbgPrint(PPDMAUDIOMIXBUF pMixBuf)
1247{
1248 audioMixBufDbgPrintInternal(pMixBuf);
1249}
1250
1251#endif /* DEBUG */
1252
1253/**
1254 * Returns the total number of samples used.
1255 *
1256 * @return uint32_t
1257 * @param pMixBuf
1258 */
1259uint32_t AudioMixBufUsed(PPDMAUDIOMIXBUF pMixBuf)
1260{
1261 AssertPtrReturn(pMixBuf, 0);
1262
1263 AUDMIXBUF_LOG(("%s: cUsed=%RU32\n", pMixBuf->pszName, pMixBuf->cUsed));
1264 return pMixBuf->cUsed;
1265}
1266
1267/**
1268 * Reads audio samples at a specific offset.
1269 *
1270 * @return IPRT status code.
1271 * @param pMixBuf Mixing buffer to read audio samples from.
1272 * @param offSamples Offset (in audio samples) to start reading from.
1273 * @param pvBuf Pointer to buffer to write output to.
1274 * @param cbBuf Size (in bytes) of buffer to write to.
1275 * @param pcbRead Size (in bytes) of data read. Optional.
1276 */
1277int AudioMixBufReadAt(PPDMAUDIOMIXBUF pMixBuf,
1278 uint32_t offSamples,
1279 void *pvBuf, uint32_t cbBuf,
1280 uint32_t *pcbRead)
1281{
1282 return AudioMixBufReadAtEx(pMixBuf, pMixBuf->AudioFmt,
1283 offSamples, pvBuf, cbBuf, pcbRead);
1284}
1285
1286/**
1287 * Reads audio samples at a specific offset.
1288 * If the audio format of the mixing buffer and the requested audio format do
1289 * not match the output will be converted accordingly.
1290 *
1291 * @return IPRT status code.
1292 * @param pMixBuf Mixing buffer to read audio samples from.
1293 * @param enmFmt Audio format to use for output.
1294 * @param offSamples Offset (in audio samples) to start reading from.
1295 * @param pvBuf Pointer to buffer to write output to.
1296 * @param cbBuf Size (in bytes) of buffer to write to.
1297 * @param pcbRead Size (in bytes) of data read. Optional.
1298 */
1299int AudioMixBufReadAtEx(PPDMAUDIOMIXBUF pMixBuf, PDMAUDIOMIXBUFFMT enmFmt,
1300 uint32_t offSamples,
1301 void *pvBuf, uint32_t cbBuf,
1302 uint32_t *pcbRead)
1303{
1304 AssertPtrReturn(pMixBuf, VERR_INVALID_POINTER);
1305 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
1306 /* pcbRead is optional. */
1307
1308 uint32_t cDstSamples = pMixBuf->cSamples;
1309 uint32_t cLive = pMixBuf->cUsed;
1310
1311 uint32_t cDead = cDstSamples - cLive;
1312 uint32_t cToProcess = (uint32_t)AUDIOMIXBUF_S2S_RATIO(pMixBuf, cDead);
1313 cToProcess = RT_MIN(cToProcess, AUDIOMIXBUF_B2S(pMixBuf, cbBuf));
1314
1315 AUDMIXBUF_LOG(("%s: offSamples=%RU32, cLive=%RU32, cDead=%RU32, cToProcess=%RU32\n",
1316 pMixBuf->pszName, offSamples, cLive, cDead, cToProcess));
1317
1318 int rc;
1319 if (cToProcess)
1320 {
1321 PFNPDMAUDIOMIXBUFCONVTO pfnConvTo = NULL;
1322 if (pMixBuf->AudioFmt != enmFmt)
1323 pfnConvTo = audioMixBufConvToLookup(enmFmt);
1324 else
1325 pfnConvTo = pMixBuf->pfnConvTo;
1326
1327 if (pfnConvTo)
1328 {
1329 PDMAUDMIXBUFCONVOPTS convOpts;
1330 RT_ZERO(convOpts);
1331 /* Note: No volume handling/conversion done in the conversion-to macros (yet). */
1332
1333 convOpts.cSamples = cToProcess;
1334
1335 pfnConvTo(pvBuf, pMixBuf->pSamples + offSamples, &convOpts);
1336
1337#ifdef DEBUG
1338 AudioMixBufDbgPrint(pMixBuf);
1339#endif
1340 rc = VINF_SUCCESS;
1341 }
1342 else
1343 {
1344 AssertFailed();
1345 rc = VERR_NOT_SUPPORTED;
1346 }
1347 }
1348 else
1349 rc = VINF_SUCCESS;
1350
1351 if (RT_SUCCESS(rc))
1352 {
1353 if (pcbRead)
1354 *pcbRead = AUDIOMIXBUF_S2B(pMixBuf, cToProcess);
1355 }
1356
1357 AUDMIXBUF_LOG(("cbRead=%RU32, rc=%Rrc\n", AUDIOMIXBUF_S2B(pMixBuf, cToProcess), rc));
1358 return rc;
1359}
1360
1361/**
1362 * Reads audio samples. The audio format of the mixing buffer will be used.
1363 *
1364 * @return IPRT status code.
1365 * @param pMixBuf Mixing buffer to read audio samples from.
1366 * @param pvBuf Pointer to buffer to write output to.
1367 * @param cbBuf Size (in bytes) of buffer to write to.
1368 * @param pcRead Number of audio samples read. Optional.
1369 */
1370int AudioMixBufReadCirc(PPDMAUDIOMIXBUF pMixBuf, void *pvBuf, uint32_t cbBuf, uint32_t *pcRead)
1371{
1372 return AudioMixBufReadCircEx(pMixBuf, pMixBuf->AudioFmt, pvBuf, cbBuf, pcRead);
1373}
1374
1375/**
1376 * Reads audio samples in a specific audio format.
1377 * If the audio format of the mixing buffer and the requested audio format do
1378 * not match the output will be converted accordingly.
1379 *
1380 * @return IPRT status code.
1381 * @param pMixBuf Mixing buffer to read audio samples from.
1382 * @param enmFmt Audio format to use for output.
1383 * @param pvBuf Pointer to buffer to write output to.
1384 * @param cbBuf Size (in bytes) of buffer to write to.
1385 * @param pcRead Number of audio samples read. Optional.
1386 */
1387int AudioMixBufReadCircEx(PPDMAUDIOMIXBUF pMixBuf, PDMAUDIOMIXBUFFMT enmFmt, void *pvBuf, uint32_t cbBuf, uint32_t *pcRead)
1388{
1389 AssertPtrReturn(pMixBuf, VERR_INVALID_POINTER);
1390 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
1391 /* pcbRead is optional. */
1392
1393 if (!cbBuf)
1394 {
1395 if (pcRead)
1396 *pcRead = 0;
1397 return VINF_SUCCESS;
1398 }
1399
1400 uint32_t cToRead = RT_MIN(AUDIOMIXBUF_B2S(pMixBuf, cbBuf), pMixBuf->cUsed);
1401
1402 AUDMIXBUF_LOG(("%s: pvBuf=%p, cbBuf=%zu (%RU32 samples), cToRead=%RU32\n",
1403 pMixBuf->pszName, pvBuf, cbBuf, AUDIOMIXBUF_B2S(pMixBuf, cbBuf), cToRead));
1404
1405 if (!cToRead)
1406 {
1407#ifdef DEBUG
1408 audioMixBufDbgPrintInternal(pMixBuf);
1409#endif
1410 if (pcRead)
1411 *pcRead = 0;
1412 return VINF_SUCCESS;
1413 }
1414
1415 PFNPDMAUDIOMIXBUFCONVTO pfnConvTo = NULL;
1416 if (pMixBuf->AudioFmt != enmFmt)
1417 pfnConvTo = audioMixBufConvToLookup(enmFmt);
1418 else
1419 pfnConvTo = pMixBuf->pfnConvTo;
1420
1421 if (!pfnConvTo) /* Audio format not supported. */
1422 {
1423 AssertFailed();
1424 return VERR_NOT_SUPPORTED;
1425 }
1426
1427 PPDMAUDIOSAMPLE pSamplesSrc1 = pMixBuf->pSamples + pMixBuf->offRead;
1428 uint32_t cLenSrc1 = cToRead;
1429
1430 PPDMAUDIOSAMPLE pSamplesSrc2 = NULL;
1431 uint32_t cLenSrc2 = 0;
1432
1433 /*
1434 * Do we need to wrap around to read all requested data, that is,
1435 * starting at the beginning of our circular buffer? This then will
1436 * be the optional second part to do.
1437 */
1438 if ((pMixBuf->offRead + cToRead) > pMixBuf->cSamples)
1439 {
1440 Assert(pMixBuf->offRead <= pMixBuf->cSamples);
1441 cLenSrc1 = pMixBuf->cSamples - pMixBuf->offRead;
1442
1443 pSamplesSrc2 = pMixBuf->pSamples;
1444 Assert(cToRead >= cLenSrc1);
1445 cLenSrc2 = RT_MIN(cToRead - cLenSrc1, pMixBuf->cSamples);
1446 }
1447
1448 PDMAUDMIXBUFCONVOPTS convOpts;
1449 RT_ZERO(convOpts);
1450 /* Note: No volume handling/conversion done in the conversion-to macros (yet). */
1451
1452 /* Anything to do at all? */
1453 int rc = VINF_SUCCESS;
1454 if (cLenSrc1)
1455 {
1456 AssertPtr(pSamplesSrc1);
1457
1458 convOpts.cSamples = cLenSrc1;
1459
1460 AUDMIXBUF_LOG(("P1: offRead=%RU32, cToRead=%RU32\n", pMixBuf->offRead, cLenSrc1));
1461 pfnConvTo(pvBuf, pSamplesSrc1, &convOpts);
1462 }
1463
1464 /* Second part present? */
1465 if ( RT_LIKELY(RT_SUCCESS(rc))
1466 && cLenSrc2)
1467 {
1468 AssertPtr(pSamplesSrc2);
1469
1470 convOpts.cSamples = cLenSrc2;
1471
1472 AUDMIXBUF_LOG(("P2: cToRead=%RU32, offWrite=%RU32 (%zu bytes)\n", cLenSrc2, cLenSrc1,
1473 AUDIOMIXBUF_S2B(pMixBuf, cLenSrc1)));
1474 pfnConvTo((uint8_t *)pvBuf + AUDIOMIXBUF_S2B(pMixBuf, cLenSrc1), pSamplesSrc2, &convOpts);
1475 }
1476
1477 if (RT_SUCCESS(rc))
1478 {
1479#ifdef AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA
1480 RTFILE fh;
1481 rc = RTFileOpen(&fh, AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA_PATH "mixbuf_readcirc.pcm",
1482 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
1483 if (RT_SUCCESS(rc))
1484 {
1485 RTFileWrite(fh, pvBuf, AUDIOMIXBUF_S2B(pMixBuf, cLenSrc1 + cLenSrc2), NULL);
1486 RTFileClose(fh);
1487 }
1488#endif
1489 pMixBuf->offRead = (pMixBuf->offRead + cToRead) % pMixBuf->cSamples;
1490 Assert(cToRead <= pMixBuf->cUsed);
1491 pMixBuf->cUsed -= RT_MIN(cToRead, pMixBuf->cUsed);
1492
1493 if (pcRead)
1494 *pcRead = cToRead;
1495 }
1496
1497#ifdef DEBUG
1498 audioMixBufDbgPrintInternal(pMixBuf);
1499#endif
1500
1501 AUDMIXBUF_LOG(("cRead=%RU32 (%zu bytes), rc=%Rrc\n", cToRead, AUDIOMIXBUF_S2B(pMixBuf, cToRead), rc));
1502 return rc;
1503}
1504
1505/**
1506 * Resets a mixing buffer.
1507 *
1508 * @param pMixBuf Mixing buffer to reset.
1509 */
1510void AudioMixBufReset(PPDMAUDIOMIXBUF pMixBuf)
1511{
1512 AssertPtrReturnVoid(pMixBuf);
1513
1514 AUDMIXBUF_LOG(("%s\n", pMixBuf->pszName));
1515
1516 pMixBuf->offRead = 0;
1517 pMixBuf->offWrite = 0;
1518 pMixBuf->cMixed = 0;
1519 pMixBuf->cUsed = 0;
1520
1521 AudioMixBufClear(pMixBuf);
1522}
1523
1524/**
1525 * Sets the overall (master) volume.
1526 *
1527 * @param pMixBuf Mixing buffer to set volume for.
1528 * @param pVol Pointer to volume structure to set.
1529 */
1530void AudioMixBufSetVolume(PPDMAUDIOMIXBUF pMixBuf, PPDMAUDIOVOLUME pVol)
1531{
1532 AssertPtrReturnVoid(pMixBuf);
1533 AssertPtrReturnVoid(pVol);
1534
1535 LogFlowFunc(("%s: lVol=%RU8, rVol=%RU8, fMuted=%RTbool\n", pMixBuf->pszName, pVol->uLeft, pVol->uRight, pVol->fMuted));
1536
1537 int rc2 = audioMixBufConvVol(&pMixBuf->Volume /* Dest */, pVol /* Source */);
1538 AssertRC(rc2);
1539}
1540
1541/**
1542 * Returns the maximum amount of audio samples this buffer can hold.
1543 *
1544 * @return uint32_t Size (in audio samples) the mixing buffer can hold.
1545 * @param pMixBuf Mixing buffer to retrieve maximum for.
1546 */
1547uint32_t AudioMixBufSize(PPDMAUDIOMIXBUF pMixBuf)
1548{
1549 AssertPtrReturn(pMixBuf, 0);
1550 return pMixBuf->cSamples;
1551}
1552
1553/**
1554 * Returns the maximum amount of bytes this buffer can hold.
1555 *
1556 * @return uint32_t Size (in bytes) the mixing buffer can hold.
1557 * @param pMixBuf Mixing buffer to retrieve maximum for.
1558 */
1559uint32_t AudioMixBufSizeBytes(PPDMAUDIOMIXBUF pMixBuf)
1560{
1561 AssertPtrReturn(pMixBuf, 0);
1562 return AUDIOMIXBUF_S2B(pMixBuf, pMixBuf->cSamples);
1563}
1564
1565/**
1566 * Unlinks a mixing buffer from its parent, if any.
1567 *
1568 * @return IPRT status code.
1569 * @param pMixBuf Mixing buffer to unlink from parent.
1570 */
1571void AudioMixBufUnlink(PPDMAUDIOMIXBUF pMixBuf)
1572{
1573 if (!pMixBuf || !pMixBuf->pszName)
1574 return;
1575
1576 AUDMIXBUF_LOG(("%s\n", pMixBuf->pszName));
1577
1578 if (pMixBuf->pParent)
1579 {
1580 AUDMIXBUF_LOG(("%s: Unlinking from parent \"%s\"\n",
1581 pMixBuf->pszName, pMixBuf->pParent->pszName));
1582
1583 RTListNodeRemove(&pMixBuf->Node);
1584
1585 /* Make sure to reset the parent mixing buffer each time it gets linked
1586 * to a new child. */
1587 AudioMixBufReset(pMixBuf->pParent);
1588 pMixBuf->pParent = NULL;
1589 }
1590
1591 PPDMAUDIOMIXBUF pChild, pChildNext;
1592 RTListForEachSafe(&pMixBuf->lstChildren, pChild, pChildNext, PDMAUDIOMIXBUF, Node)
1593 {
1594 AUDMIXBUF_LOG(("\tUnlinking \"%s\"\n", pChild->pszName));
1595
1596 AudioMixBufReset(pChild);
1597
1598 Assert(pChild->pParent == pMixBuf);
1599 pChild->pParent = NULL;
1600
1601 RTListNodeRemove(&pChild->Node);
1602 }
1603
1604 Assert(RTListIsEmpty(&pMixBuf->lstChildren));
1605
1606 AudioMixBufReset(pMixBuf);
1607
1608 if (pMixBuf->pRate)
1609 {
1610 pMixBuf->pRate->dstOffset = pMixBuf->pRate->srcOffset = 0;
1611 pMixBuf->pRate->dstInc = 0;
1612 }
1613
1614 pMixBuf->iFreqRatio = 1; /* Prevent division by zero. */
1615}
1616
1617/**
1618 * Writes audio samples at a specific offset.
1619 * The sample format being written must match the format of the mixing buffer.
1620 *
1621 * @return IPRT status code.
1622 * @param pMixBuf Pointer to mixing buffer to write to.
1623 * @param offSamples Offset (in samples) starting to write at.
1624 * @param pvBuf Pointer to audio buffer to be written.
1625 * @param cbBuf Size (in bytes) of audio buffer.
1626 * @param pcWritten Returns number of audio samples written. Optional.
1627 */
1628int AudioMixBufWriteAt(PPDMAUDIOMIXBUF pMixBuf, uint32_t offSamples, const void *pvBuf, uint32_t cbBuf, uint32_t *pcWritten)
1629{
1630 return AudioMixBufWriteAtEx(pMixBuf, pMixBuf->AudioFmt, offSamples, pvBuf, cbBuf, pcWritten);
1631}
1632
1633/**
1634 * Writes audio samples at a specific offset.
1635 *
1636 * The audio sample format to be written can be different from the audio format
1637 * the mixing buffer operates on.
1638 *
1639 * @return IPRT status code.
1640 * @param pMixBuf Pointer to mixing buffer to write to.
1641 * @param enmFmt Audio format supplied in the buffer.
1642 * @param offSamples Offset (in samples) starting to write at.
1643 * @param pvBuf Pointer to audio buffer to be written.
1644 * @param cbBuf Size (in bytes) of audio buffer.
1645 * @param pcWritten Returns number of audio samples written. Optional.
1646 */
1647int AudioMixBufWriteAtEx(PPDMAUDIOMIXBUF pMixBuf, PDMAUDIOMIXBUFFMT enmFmt,
1648 uint32_t offSamples,
1649 const void *pvBuf, uint32_t cbBuf,
1650 uint32_t *pcWritten)
1651{
1652 AssertPtrReturn(pMixBuf, VERR_INVALID_POINTER);
1653 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
1654 /* pcWritten is optional. */
1655
1656 /*
1657 * Adjust cToWrite so we don't overflow our buffers.
1658 */
1659 int rc;
1660 uint32_t cToWrite = AUDIOMIXBUF_B2S(pMixBuf, cbBuf);
1661 if (offSamples <= pMixBuf->cSamples)
1662 {
1663 if (offSamples + cToWrite <= pMixBuf->cSamples)
1664 rc = VINF_SUCCESS;
1665 else
1666 {
1667 rc = VINF_BUFFER_OVERFLOW;
1668 cToWrite = pMixBuf->cSamples - offSamples;
1669 }
1670 }
1671 else
1672 {
1673 rc = VINF_BUFFER_OVERFLOW;
1674 cToWrite = 0;
1675 }
1676
1677#ifdef AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA
1678 /*
1679 * Now that we know how much we'll be converting we can log it.
1680 */
1681 RTFILE hFile;
1682 int rc2 = RTFileOpen(&hFile, AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA_PATH "mixbuf_writeat.pcm",
1683 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
1684 if (RT_SUCCESS(rc2))
1685 {
1686 RTFileWrite(hFile, pvBuf, AUDIOMIXBUF_S2B(pMixBuf, cToWrite), NULL);
1687 RTFileClose(hFile);
1688 }
1689#endif
1690
1691 /*
1692 * Pick the conversion function and do the conversion.
1693 */
1694 PFNPDMAUDIOMIXBUFCONVFROM pfnConvFrom = NULL;
1695 if (!pMixBuf->Volume.fMuted)
1696 {
1697 if (pMixBuf->AudioFmt != enmFmt)
1698 pfnConvFrom = audioMixBufConvFromLookup(enmFmt);
1699 else
1700 pfnConvFrom = pMixBuf->pfnConvFrom;
1701 }
1702 else
1703 pfnConvFrom = &audioMixBufConvFromSilence;
1704
1705 uint32_t cWritten;
1706 if ( pfnConvFrom
1707 && cToWrite)
1708 {
1709 PDMAUDMIXBUFCONVOPTS convOpts;
1710
1711 convOpts.cSamples = cToWrite;
1712 convOpts.From.Volume.fMuted = pMixBuf->Volume.fMuted;
1713 convOpts.From.Volume.uLeft = pMixBuf->Volume.uLeft;
1714 convOpts.From.Volume.uRight = pMixBuf->Volume.uRight;
1715
1716 cWritten = pfnConvFrom(pMixBuf->pSamples + offSamples, pvBuf, AUDIOMIXBUF_S2B(pMixBuf, cToWrite), &convOpts);
1717 }
1718 else
1719 {
1720 cWritten = 0;
1721 if (!pfnConvFrom)
1722 {
1723 AssertFailed();
1724 rc = VERR_NOT_SUPPORTED;
1725 }
1726 }
1727
1728#ifdef DEBUG
1729 audioMixBufDbgPrintInternal(pMixBuf);
1730#endif
1731
1732 AUDMIXBUF_LOG(("%s: offSamples=%RU32, cbBuf=%RU32, cToWrite=%RU32 (%zu bytes), cWritten=%RU32 (%zu bytes), rc=%Rrc\n",
1733 pMixBuf->pszName, offSamples, cbBuf,
1734 cToWrite, AUDIOMIXBUF_S2B(pMixBuf, cToWrite),
1735 cWritten, AUDIOMIXBUF_S2B(pMixBuf, cWritten), rc));
1736
1737 if (RT_SUCCESS(rc) && pcWritten)
1738 *pcWritten = cWritten;
1739
1740 return rc;
1741}
1742
1743/**
1744 * Writes audio samples.
1745 *
1746 * The sample format being written must match the format of the mixing buffer.
1747 *
1748 * @return IPRT status code, or VINF_BUFFER_OVERFLOW if samples which not have
1749 * been processed yet have been overwritten (due to cyclic buffer).
1750 * @param pMixBuf Pointer to mixing buffer to write to.
1751 * @param pvBuf Pointer to audio buffer to be written.
1752 * @param cbBuf Size (in bytes) of audio buffer.
1753 * @param pcWritten Returns number of audio samples written. Optional.
1754 */
1755int AudioMixBufWriteCirc(PPDMAUDIOMIXBUF pMixBuf,
1756 const void *pvBuf, uint32_t cbBuf,
1757 uint32_t *pcWritten)
1758{
1759 return AudioMixBufWriteCircEx(pMixBuf, pMixBuf->AudioFmt, pvBuf, cbBuf, pcWritten);
1760}
1761
1762/**
1763 * Writes audio samples of a specific format.
1764 *
1765 * @return IPRT status code, or VINF_BUFFER_OVERFLOW if samples which not have
1766 * been processed yet have been overwritten (due to cyclic buffer).
1767 * @param pMixBuf Pointer to mixing buffer to write to.
1768 * @param enmFmt Audio format supplied in the buffer.
1769 * @param pvBuf Pointer to audio buffer to be written.
1770 * @param cbBuf Size (in bytes) of audio buffer.
1771 * @param pcWritten Returns number of audio samples written. Optional.
1772 */
1773int AudioMixBufWriteCircEx(PPDMAUDIOMIXBUF pMixBuf, PDMAUDIOMIXBUFFMT enmFmt,
1774 const void *pvBuf, uint32_t cbBuf, uint32_t *pcWritten)
1775{
1776 AssertPtrReturn(pMixBuf, VERR_INVALID_POINTER);
1777 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
1778 /* pcbWritten is optional. */
1779
1780 if (!cbBuf)
1781 {
1782 if (pcWritten)
1783 *pcWritten = 0;
1784 return VINF_SUCCESS;
1785 }
1786
1787 PPDMAUDIOMIXBUF pParent = pMixBuf->pParent;
1788
1789 AUDMIXBUF_LOG(("%s: enmFmt=%d, pvBuf=%p, cbBuf=%RU32 (%RU32 samples)\n",
1790 pMixBuf->pszName, enmFmt, pvBuf, cbBuf, AUDIOMIXBUF_B2S(pMixBuf, cbBuf)));
1791
1792 if ( pParent
1793 && pParent->cSamples < pMixBuf->cMixed)
1794 {
1795 if (pcWritten)
1796 *pcWritten = 0;
1797
1798 AUDMIXBUF_LOG(("%s: Parent buffer '%s' is full\n",
1799 pMixBuf->pszName, pMixBuf->pParent->pszName));
1800
1801 return VINF_BUFFER_OVERFLOW;
1802 }
1803
1804 PFNPDMAUDIOMIXBUFCONVFROM pfnConvFrom = NULL;
1805 if (!pMixBuf->Volume.fMuted)
1806 {
1807 if (pMixBuf->AudioFmt != enmFmt)
1808 pfnConvFrom = audioMixBufConvFromLookup(enmFmt);
1809 else
1810 pfnConvFrom = pMixBuf->pfnConvFrom;
1811 }
1812 else
1813 pfnConvFrom = &audioMixBufConvFromSilence;
1814
1815 if (!pfnConvFrom)
1816 {
1817 AssertFailed();
1818 return VERR_NOT_SUPPORTED;
1819 }
1820
1821 uint32_t cToWrite = AUDIOMIXBUF_B2S(pMixBuf, cbBuf);
1822 AssertMsg(cToWrite, ("cToWrite is 0 (cbBuf=%zu)\n", cbBuf));
1823
1824 PPDMAUDIOSAMPLE pSamplesDst1 = pMixBuf->pSamples + pMixBuf->offWrite;
1825 uint32_t cLenDst1 = cToWrite;
1826
1827 PPDMAUDIOSAMPLE pSamplesDst2 = NULL;
1828 uint32_t cLenDst2 = 0;
1829
1830 uint32_t cOffWrite = pMixBuf->offWrite + cToWrite;
1831
1832 /*
1833 * Do we need to wrap around to write all requested data, that is,
1834 * starting at the beginning of our circular buffer? This then will
1835 * be the optional second part to do.
1836 */
1837 if (cOffWrite >= pMixBuf->cSamples)
1838 {
1839 Assert(pMixBuf->offWrite <= pMixBuf->cSamples);
1840 cLenDst1 = pMixBuf->cSamples - pMixBuf->offWrite;
1841
1842 pSamplesDst2 = pMixBuf->pSamples;
1843 Assert(cToWrite >= cLenDst1);
1844 cLenDst2 = RT_MIN(cToWrite - cLenDst1, pMixBuf->cSamples);
1845
1846 /* Save new read offset. */
1847 cOffWrite = cLenDst2;
1848 }
1849
1850#ifdef AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA
1851 RTFILE fh;
1852 RTFileOpen(&fh, AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA_PATH "mixbuf_writecirc_ex.pcm",
1853 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
1854#endif
1855
1856 uint32_t cWrittenTotal = 0;
1857
1858 PDMAUDMIXBUFCONVOPTS convOpts;
1859 convOpts.From.Volume.fMuted = pMixBuf->Volume.fMuted;
1860 convOpts.From.Volume.uLeft = pMixBuf->Volume.uLeft;
1861 convOpts.From.Volume.uRight = pMixBuf->Volume.uRight;
1862
1863 /* Anything to do at all? */
1864 if (cLenDst1)
1865 {
1866 convOpts.cSamples = cLenDst1;
1867 cWrittenTotal = pfnConvFrom(pSamplesDst1, pvBuf, AUDIOMIXBUF_S2B(pMixBuf, cLenDst1), &convOpts);
1868 Assert(cWrittenTotal == cLenDst1);
1869
1870#ifdef AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA
1871 RTFileWrite(fh, pvBuf, AUDIOMIXBUF_S2B(pMixBuf, cLenDst1), NULL);
1872#endif
1873 }
1874
1875 /* Second part present? */
1876 if (cLenDst2)
1877 {
1878 AssertPtr(pSamplesDst2);
1879
1880 convOpts.cSamples = cLenDst2;
1881 cWrittenTotal += pfnConvFrom(pSamplesDst2,
1882 (uint8_t *)pvBuf + AUDIOMIXBUF_S2B(pMixBuf, cLenDst1),
1883 cbBuf - AUDIOMIXBUF_S2B(pMixBuf, cLenDst1),
1884 &convOpts);
1885 Assert(cWrittenTotal == cLenDst1 + cLenDst2);
1886
1887#ifdef AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA
1888 RTFileWrite(fh, (uint8_t *)pvBuf + AUDIOMIXBUF_S2B(pMixBuf, cLenDst1),
1889 cbBuf - AUDIOMIXBUF_S2B(pMixBuf, cLenDst1), NULL);
1890#endif
1891 }
1892
1893#ifdef AUDIOMIXBUF_DEBUG_DUMP_PCM_DATA
1894 RTFileClose(fh);
1895#endif
1896
1897 pMixBuf->offWrite = (pMixBuf->offWrite + cWrittenTotal) % pMixBuf->cSamples;
1898 pMixBuf->cUsed += cWrittenTotal;
1899
1900 int rc = VINF_SUCCESS;
1901
1902 if (pMixBuf->cUsed > pMixBuf->cSamples)
1903 {
1904 AUDMIXBUF_LOG(("Warning: %RU32 unprocessed samples overwritten\n", pMixBuf->cUsed - pMixBuf->cSamples));
1905 pMixBuf->cUsed = pMixBuf->cSamples;
1906
1907 rc = VINF_BUFFER_OVERFLOW;
1908 }
1909
1910 if (pcWritten)
1911 *pcWritten = cWrittenTotal;
1912
1913#ifdef DEBUG
1914 audioMixBufDbgPrintInternal(pMixBuf);
1915#endif
1916
1917 AUDMIXBUF_LOG(("offWrite=%RU32, cLenDst1=%RU32, cLenDst2=%RU32, cTotal=%RU32 (%zu bytes), rc=%Rrc\n",
1918 pMixBuf->offWrite, cLenDst1, cLenDst2, cLenDst1 + cLenDst2,
1919 AUDIOMIXBUF_S2B(pMixBuf, cLenDst1 + cLenDst2), rc));
1920 return rc;
1921}
1922
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