VirtualBox

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

Last change on this file since 64700 was 64652, checked in by vboxsync, 8 years ago

Audio: Reverted part of r111751 to fix unit test failure.

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