VirtualBox

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

Last change on this file since 62978 was 62977, checked in by vboxsync, 8 years ago

enums defaults to int, so use '%d' and not '%ld' to prinft them!

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