VirtualBox

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

Last change on this file since 57200 was 56648, checked in by vboxsync, 10 years ago

Audio: Remove DEV_AUDIO logging group and split it up into per device and driver groups for finer grained logging

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