VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/testcase/tstAudioMixBuffer.cpp@ 71045

Last change on this file since 71045 was 70878, checked in by vboxsync, 7 years ago

Audio: Renamed AudioMixBufReadCirc[Ex] -> AudioMixBufAcquireReadBlock[Ex] and added AudioMixBufReleaseReadBlock() -- this adds the ability to read less than initially announced, if required.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 27.4 KB
Line 
1/* $Id: tstAudioMixBuffer.cpp 70878 2018-02-06 10:48:43Z vboxsync $ */
2/** @file
3 * Audio testcase - Mixing buffer.
4 */
5
6/*
7 * Copyright (C) 2014-2018 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#include <iprt/err.h>
23#include <iprt/initterm.h>
24#include <iprt/mem.h>
25#include <iprt/rand.h>
26#include <iprt/stream.h>
27#include <iprt/string.h>
28#include <iprt/test.h>
29
30
31#include "../AudioMixBuffer.h"
32#include "../DrvAudio.h"
33
34
35/*********************************************************************************************************************************
36* Structures and Typedefs *
37*********************************************************************************************************************************/
38
39static int tstSingle(RTTEST hTest)
40{
41 RTTestSubF(hTest, "Single buffer");
42
43 /* 44100Hz, 2 Channels, S16 */
44 PDMAUDIOPCMPROPS config =
45 {
46 16, /* Bits */
47 true, /* Signed */
48 2, /* Channels */
49 44100, /* Hz */
50 PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(16 /* Bits */, 2 /* Channels */), /* Shift */
51 false /* Swap Endian */
52 };
53
54 RTTESTI_CHECK(DrvAudioHlpPCMPropsAreValid(&config));
55
56 uint32_t cBufSize = _1K;
57
58 /*
59 * General stuff.
60 */
61 PDMAUDIOMIXBUF mb;
62 RTTESTI_CHECK_RC_OK(AudioMixBufInit(&mb, "Single", &config, cBufSize));
63 RTTESTI_CHECK(AudioMixBufSize(&mb) == cBufSize);
64 RTTESTI_CHECK(AUDIOMIXBUF_B2F(&mb, AudioMixBufSizeBytes(&mb)) == cBufSize);
65 RTTESTI_CHECK(AUDIOMIXBUF_F2B(&mb, AudioMixBufSize(&mb)) == AudioMixBufSizeBytes(&mb));
66 RTTESTI_CHECK(AudioMixBufFree(&mb) == cBufSize);
67 RTTESTI_CHECK(AUDIOMIXBUF_F2B(&mb, AudioMixBufFree(&mb)) == AudioMixBufFreeBytes(&mb));
68
69 /*
70 * Absolute writes.
71 */
72 uint32_t cFramesRead = 0, cFramesWritten = 0, cFramesWrittenAbs = 0;
73 int8_t aFrames8 [2] = { 0x12, 0x34 };
74 int16_t aFrames16[2] = { 0xAA, 0xBB };
75 int32_t aFrames32[2] = { 0xCC, 0xDD };
76
77 RTTESTI_CHECK_RC_OK(AudioMixBufWriteAt(&mb, 0 /* Offset */, &aFrames8, sizeof(aFrames8), &cFramesWritten));
78 RTTESTI_CHECK(cFramesWritten == 0 /* Frames */);
79 RTTESTI_CHECK(AudioMixBufUsed(&mb) == 0);
80
81 RTTESTI_CHECK_RC_OK(AudioMixBufWriteAt(&mb, 0 /* Offset */, &aFrames16, sizeof(aFrames16), &cFramesWritten));
82 RTTESTI_CHECK(cFramesWritten == 1 /* Frames */);
83 RTTESTI_CHECK(AudioMixBufUsed(&mb) == 1);
84
85 RTTESTI_CHECK_RC_OK(AudioMixBufWriteAt(&mb, 2 /* Offset */, &aFrames32, sizeof(aFrames32), &cFramesWritten));
86 RTTESTI_CHECK(cFramesWritten == 2 /* Frames */);
87 RTTESTI_CHECK(AudioMixBufUsed(&mb) == 2);
88
89 /* Beyond buffer. */
90 RTTESTI_CHECK_RC(AudioMixBufWriteAt(&mb, AudioMixBufSize(&mb) + 1, &aFrames16, sizeof(aFrames16),
91 &cFramesWritten), VERR_BUFFER_OVERFLOW);
92
93 /* Offset wrap-around: When writing as much (or more) frames the mixing buffer can hold. */
94 uint32_t cbSamples = cBufSize * sizeof(int16_t) * 2 /* Channels */;
95 RTTESTI_CHECK(cbSamples);
96 uint16_t *paSamples = (uint16_t *)RTMemAlloc(cbSamples);
97 RTTESTI_CHECK(paSamples);
98 RTTESTI_CHECK_RC_OK(AudioMixBufWriteAt(&mb, 0 /* Offset */, paSamples, cbSamples, &cFramesWritten));
99 RTTESTI_CHECK(cFramesWritten == cBufSize /* Frames */);
100 RTTESTI_CHECK(AudioMixBufUsed(&mb) == cBufSize);
101 RTTESTI_CHECK(AudioMixBufReadPos(&mb) == 0);
102 RTTESTI_CHECK(AudioMixBufWritePos(&mb) == 0);
103 RTMemFree(paSamples);
104 cbSamples = 0;
105
106 /*
107 * Circular writes.
108 */
109 AudioMixBufReset(&mb);
110
111 RTTESTI_CHECK_RC_OK(AudioMixBufWriteAt(&mb, 2 /* Offset */, &aFrames32, sizeof(aFrames32), &cFramesWritten));
112 RTTESTI_CHECK(cFramesWritten == 2 /* Frames */);
113 RTTESTI_CHECK(AudioMixBufUsed(&mb) == 2);
114
115 cFramesWrittenAbs = AudioMixBufUsed(&mb);
116
117 uint32_t cToWrite = AudioMixBufSize(&mb) - cFramesWrittenAbs - 1; /* -1 as padding plus -2 frames for above. */
118 for (uint32_t i = 0; i < cToWrite; i++)
119 {
120 RTTESTI_CHECK_RC_OK(AudioMixBufWriteCirc(&mb, &aFrames16, sizeof(aFrames16), &cFramesWritten));
121 RTTESTI_CHECK(cFramesWritten == 1);
122 }
123 RTTESTI_CHECK(!AudioMixBufIsEmpty(&mb));
124 RTTESTI_CHECK(AudioMixBufFree(&mb) == 1);
125 RTTESTI_CHECK(AudioMixBufFreeBytes(&mb) == AUDIOMIXBUF_F2B(&mb, 1U));
126 RTTESTI_CHECK(AudioMixBufUsed(&mb) == cToWrite + cFramesWrittenAbs /* + last absolute write */);
127
128 RTTESTI_CHECK_RC_OK(AudioMixBufWriteCirc(&mb, &aFrames16, sizeof(aFrames16), &cFramesWritten));
129 RTTESTI_CHECK(cFramesWritten == 1);
130 RTTESTI_CHECK(AudioMixBufFree(&mb) == 0);
131 RTTESTI_CHECK(AudioMixBufFreeBytes(&mb) == AUDIOMIXBUF_F2B(&mb, 0U));
132 RTTESTI_CHECK(AudioMixBufUsed(&mb) == cBufSize);
133
134 /* Circular reads. */
135 uint32_t cToRead = AudioMixBufSize(&mb) - cFramesWrittenAbs - 1;
136 for (uint32_t i = 0; i < cToRead; i++)
137 {
138 RTTESTI_CHECK_RC_OK(AudioMixBufAcquireReadBlock(&mb, &aFrames16, sizeof(aFrames16), &cFramesRead));
139 RTTESTI_CHECK(cFramesRead == 1);
140 AudioMixBufReleaseReadBlock(&mb, cFramesRead);
141 AudioMixBufFinish(&mb, cFramesRead);
142 }
143 RTTESTI_CHECK(!AudioMixBufIsEmpty(&mb));
144 RTTESTI_CHECK(AudioMixBufFree(&mb) == AudioMixBufSize(&mb) - cFramesWrittenAbs - 1);
145 RTTESTI_CHECK(AudioMixBufFreeBytes(&mb) == AUDIOMIXBUF_F2B(&mb, cBufSize - cFramesWrittenAbs - 1));
146 RTTESTI_CHECK(AudioMixBufUsed(&mb) == cBufSize - cToRead);
147
148 RTTESTI_CHECK_RC_OK(AudioMixBufAcquireReadBlock(&mb, &aFrames16, sizeof(aFrames16), &cFramesRead));
149 RTTESTI_CHECK(cFramesRead == 1);
150 AudioMixBufReleaseReadBlock(&mb, cFramesRead);
151 AudioMixBufFinish(&mb, cFramesRead);
152 RTTESTI_CHECK(AudioMixBufFree(&mb) == cBufSize - cFramesWrittenAbs);
153 RTTESTI_CHECK(AudioMixBufFreeBytes(&mb) == AUDIOMIXBUF_F2B(&mb, cBufSize - cFramesWrittenAbs));
154 RTTESTI_CHECK(AudioMixBufUsed(&mb) == cFramesWrittenAbs);
155
156 AudioMixBufDestroy(&mb);
157
158 return RTTestSubErrorCount(hTest) ? VERR_GENERAL_FAILURE : VINF_SUCCESS;
159}
160
161static int tstParentChild(RTTEST hTest)
162{
163 uint32_t cParentBufSize = RTRandU32Ex(_1K /* Min */, _16K /* Max */); /* Enough room for random sizes */
164
165 /* 44100Hz, 2 Channels, S16 */
166 PDMAUDIOPCMPROPS cfg_p =
167 {
168 16, /* Bits */
169 true, /* Signed */
170 2, /* Channels */
171 44100, /* Hz */
172 PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(16 /* Bits */, 2 /* Channels */), /* Shift */
173 false /* Swap Endian */
174 };
175
176 RTTESTI_CHECK(DrvAudioHlpPCMPropsAreValid(&cfg_p));
177
178 PDMAUDIOMIXBUF parent;
179 RTTESTI_CHECK_RC_OK(AudioMixBufInit(&parent, "Parent", &cfg_p, cParentBufSize));
180
181 /* 22050Hz, 2 Channels, S16 */
182 PDMAUDIOPCMPROPS cfg_c1 = /* Upmixing to parent */
183 {
184 16, /* Bits */
185 true, /* Signed */
186 2, /* Channels */
187 22050, /* Hz */
188 PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(16 /* Bits */, 2 /* Channels */), /* Shift */
189 false /* Swap Endian */
190 };
191
192 RTTESTI_CHECK(DrvAudioHlpPCMPropsAreValid(&cfg_c1));
193
194 uint32_t cFrames = 16;
195 uint32_t cChildBufSize = RTRandU32Ex(cFrames /* Min */, 64 /* Max */);
196
197 PDMAUDIOMIXBUF child1;
198 RTTESTI_CHECK_RC_OK(AudioMixBufInit(&child1, "Child1", &cfg_c1, cChildBufSize));
199 RTTESTI_CHECK_RC_OK(AudioMixBufLinkTo(&child1, &parent));
200
201 /* 48000Hz, 2 Channels, S16 */
202 PDMAUDIOPCMPROPS cfg_c2 = /* Downmixing to parent */
203 {
204 16, /* Bits */
205 true, /* Signed */
206 2, /* Channels */
207 48000, /* Hz */
208 PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(16 /* Bits */, 2 /* Channels */), /* Shift */
209 false /* Swap Endian */
210 };
211
212 RTTESTI_CHECK(DrvAudioHlpPCMPropsAreValid(&cfg_c2));
213
214 PDMAUDIOMIXBUF child2;
215 RTTESTI_CHECK_RC_OK(AudioMixBufInit(&child2, "Child2", &cfg_c2, cChildBufSize));
216 RTTESTI_CHECK_RC_OK(AudioMixBufLinkTo(&child2, &parent));
217
218 /*
219 * Writing + mixing from child/children -> parent, sequential.
220 */
221 uint32_t cbBuf = _1K;
222 char pvBuf[_1K];
223 int16_t aFrames16[32] = { 0xAA, 0xBB };
224 uint32_t cFramesRead, cFramesWritten, cFramesMixed;
225
226 uint32_t cFramesChild1 = cFrames;
227 uint32_t cFramesChild2 = cFrames;
228
229 uint32_t t = RTRandU32() % 32;
230
231 RTTestPrintf(hTest, RTTESTLVL_DEBUG,
232 "cParentBufSize=%RU32, cChildBufSize=%RU32, %RU32 frames -> %RU32 iterations total\n",
233 cParentBufSize, cChildBufSize, cFrames, t);
234
235 /*
236 * Using AudioMixBufWriteAt for writing to children.
237 */
238 RTTestSubF(hTest, "2 Children -> Parent (AudioMixBufWriteAt)");
239
240 uint32_t cChildrenSamplesMixedTotal = 0;
241
242 for (uint32_t i = 0; i < t; i++)
243 {
244 RTTestPrintf(hTest, RTTESTLVL_DEBUG, "i=%RU32\n", i);
245
246 uint32_t cChild1Writes = RTRandU32() % 8;
247
248 for (uint32_t c1 = 0; c1 < cChild1Writes; c1++)
249 {
250 /* Child 1. */
251 RTTESTI_CHECK_RC_OK_BREAK(AudioMixBufWriteAt(&child1, 0, &aFrames16, sizeof(aFrames16), &cFramesWritten));
252 RTTESTI_CHECK_MSG_BREAK(cFramesWritten == cFramesChild1, ("Child1: Expected %RU32 written frames, got %RU32\n", cFramesChild1, cFramesWritten));
253 RTTESTI_CHECK_RC_OK_BREAK(AudioMixBufMixToParent(&child1, cFramesWritten, &cFramesMixed));
254
255 cChildrenSamplesMixedTotal += cFramesMixed;
256
257 RTTESTI_CHECK_MSG_BREAK(cFramesWritten == cFramesMixed, ("Child1: Expected %RU32 mixed frames, got %RU32\n", cFramesWritten, cFramesMixed));
258 RTTESTI_CHECK_MSG_BREAK(AudioMixBufUsed(&child1) == 0, ("Child1: Expected %RU32 used frames, got %RU32\n", 0, AudioMixBufUsed(&child1)));
259 }
260
261 uint32_t cChild2Writes = RTRandU32() % 8;
262
263 for (uint32_t c2 = 0; c2 < cChild2Writes; c2++)
264 {
265 /* Child 2. */
266 RTTESTI_CHECK_RC_OK_BREAK(AudioMixBufWriteAt(&child2, 0, &aFrames16, sizeof(aFrames16), &cFramesWritten));
267 RTTESTI_CHECK_MSG_BREAK(cFramesWritten == cFramesChild2, ("Child2: Expected %RU32 written frames, got %RU32\n", cFramesChild2, cFramesWritten));
268 RTTESTI_CHECK_RC_OK_BREAK(AudioMixBufMixToParent(&child2, cFramesWritten, &cFramesMixed));
269
270 cChildrenSamplesMixedTotal += cFramesMixed;
271
272 RTTESTI_CHECK_MSG_BREAK(cFramesWritten == cFramesMixed, ("Child2: Expected %RU32 mixed frames, got %RU32\n", cFramesWritten, cFramesMixed));
273 RTTESTI_CHECK_MSG_BREAK(AudioMixBufUsed(&child2) == 0, ("Child2: Expected %RU32 used frames, got %RU32\n", 0, AudioMixBufUsed(&child2)));
274 }
275
276 /*
277 * Read out all frames from the parent buffer and also mark the just-read frames as finished
278 * so that both connected children buffers can keep track of their stuff.
279 */
280 uint32_t cParentSamples = AudioMixBufUsed(&parent);
281 while (cParentSamples)
282 {
283 RTTESTI_CHECK_RC_OK_BREAK(AudioMixBufAcquireReadBlock(&parent, pvBuf, cbBuf, &cFramesRead));
284 if (!cFramesRead)
285 break;
286
287 AudioMixBufReleaseReadBlock(&parent, cFramesRead);
288 AudioMixBufFinish(&parent, cFramesRead);
289
290 RTTESTI_CHECK(cParentSamples >= cFramesRead);
291 cParentSamples -= cFramesRead;
292 }
293
294 RTTESTI_CHECK(cParentSamples == 0);
295 }
296
297 RTTESTI_CHECK(AudioMixBufUsed(&parent) == 0);
298 RTTESTI_CHECK(AudioMixBufLive(&child1) == 0);
299 RTTESTI_CHECK(AudioMixBufLive(&child2) == 0);
300
301 AudioMixBufDestroy(&parent);
302 AudioMixBufDestroy(&child1);
303 AudioMixBufDestroy(&child2);
304
305 return RTTestSubErrorCount(hTest) ? VERR_GENERAL_FAILURE : VINF_SUCCESS;
306}
307
308/* Test 8-bit sample conversion (8-bit -> internal -> 8-bit). */
309static int tstConversion8(RTTEST hTest)
310{
311 unsigned i;
312 uint32_t cBufSize = 256;
313
314 RTTestSubF(hTest, "Sample conversion (U8)");
315
316 /* 44100Hz, 1 Channel, U8 */
317 PDMAUDIOPCMPROPS cfg_p =
318 {
319 8, /* Bits */
320 false, /* Signed */
321 1, /* Channels */
322 44100, /* Hz */
323 PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(8 /* Bits */, 1 /* Channels */), /* Shift */
324 false /* Swap Endian */
325 };
326
327 RTTESTI_CHECK(DrvAudioHlpPCMPropsAreValid(&cfg_p));
328
329 PDMAUDIOMIXBUF parent;
330 RTTESTI_CHECK_RC_OK(AudioMixBufInit(&parent, "Parent", &cfg_p, cBufSize));
331
332 /* Child uses half the sample rate; that ensures the mixing engine can't
333 * take shortcuts and performs conversion. Because conversion to double
334 * the sample rate effectively inserts one additional sample between every
335 * two source frames, N source frames will be converted to N * 2 - 1
336 * frames. However, the last source sample will be saved for later
337 * interpolation and not immediately output.
338 */
339
340 /* 22050Hz, 1 Channel, U8 */
341 PDMAUDIOPCMPROPS cfg_c = /* Upmixing to parent */
342 {
343 8, /* Bits */
344 false, /* Signed */
345 1, /* Channels */
346 22050, /* Hz */
347 PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(8 /* Bits */, 1 /* Channels */), /* Shift */
348 false /* Swap Endian */
349 };
350
351 RTTESTI_CHECK(DrvAudioHlpPCMPropsAreValid(&cfg_c));
352
353 PDMAUDIOMIXBUF child;
354 RTTESTI_CHECK_RC_OK(AudioMixBufInit(&child, "Child", &cfg_c, cBufSize));
355 RTTESTI_CHECK_RC_OK(AudioMixBufLinkTo(&child, &parent));
356
357 /* 8-bit unsigned frames. Often used with SB16 device. */
358 uint8_t aFrames8U[16] = { 0xAA, 0xBB, 0, 1, 43, 125, 126, 127,
359 128, 129, 130, 131, 132, UINT8_MAX - 1, UINT8_MAX, 0 };
360
361 /*
362 * Writing + mixing from child -> parent, sequential.
363 */
364 uint32_t cbBuf = 256;
365 char achBuf[256];
366 uint32_t cFramesRead, cFramesWritten, cFramesMixed;
367
368 uint32_t cFramesChild = 16;
369 uint32_t cFramesParent = cFramesChild * 2 - 2;
370 uint32_t cFramesTotalRead = 0;
371
372 /**** 8-bit unsigned samples ****/
373 RTTestPrintf(hTest, RTTESTLVL_DEBUG, "Conversion test %uHz %uch 8-bit\n", cfg_c.uHz, cfg_c.cChannels);
374 RTTESTI_CHECK_RC_OK(AudioMixBufWriteCirc(&child, &aFrames8U, sizeof(aFrames8U), &cFramesWritten));
375 RTTESTI_CHECK_MSG(cFramesWritten == cFramesChild, ("Child: Expected %RU32 written frames, got %RU32\n", cFramesChild, cFramesWritten));
376 RTTESTI_CHECK_RC_OK(AudioMixBufMixToParent(&child, cFramesWritten, &cFramesMixed));
377 uint32_t cFrames = AudioMixBufUsed(&parent);
378 RTTESTI_CHECK_MSG(AudioMixBufLive(&child) == cFrames, ("Child: Expected %RU32 mixed frames, got %RU32\n", AudioMixBufLive(&child), cFrames));
379
380 RTTESTI_CHECK(AudioMixBufUsed(&parent) == AudioMixBufLive(&child));
381
382 for (;;)
383 {
384 RTTESTI_CHECK_RC_OK_BREAK(AudioMixBufAcquireReadBlock(&parent, achBuf, cbBuf, &cFramesRead));
385 if (!cFramesRead)
386 break;
387 cFramesTotalRead += cFramesRead;
388 AudioMixBufReleaseReadBlock(&parent, cFramesRead);
389 AudioMixBufFinish(&parent, cFramesRead);
390 }
391
392 RTTESTI_CHECK_MSG(cFramesTotalRead == cFramesParent, ("Parent: Expected %RU32 mixed frames, got %RU32\n", cFramesParent, cFramesTotalRead));
393
394 /* Check that the frames came out unharmed. Every other sample is interpolated and we ignore it. */
395 /* NB: This also checks that the default volume setting is 0dB attenuation. */
396 uint8_t *pSrc8 = &aFrames8U[0];
397 uint8_t *pDst8 = (uint8_t *)achBuf;
398
399 for (i = 0; i < cFramesChild - 1; ++i)
400 {
401 RTTESTI_CHECK_MSG(*pSrc8 == *pDst8, ("index %u: Dst=%d, Src=%d\n", i, *pDst8, *pSrc8));
402 pSrc8 += 1;
403 pDst8 += 2;
404 }
405
406 RTTESTI_CHECK(AudioMixBufUsed(&parent) == 0);
407 RTTESTI_CHECK(AudioMixBufLive(&child) == 0);
408
409 AudioMixBufDestroy(&parent);
410 AudioMixBufDestroy(&child);
411
412 return RTTestSubErrorCount(hTest) ? VERR_GENERAL_FAILURE : VINF_SUCCESS;
413}
414
415/* Test 16-bit sample conversion (16-bit -> internal -> 16-bit). */
416static int tstConversion16(RTTEST hTest)
417{
418 unsigned i;
419 uint32_t cBufSize = 256;
420
421 RTTestSubF(hTest, "Sample conversion (S16)");
422
423 /* 44100Hz, 1 Channel, S16 */
424 PDMAUDIOPCMPROPS cfg_p =
425 {
426 16, /* Bits */
427 true, /* Signed */
428 1, /* Channels */
429 44100, /* Hz */
430 PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(16 /* Bits */, 1 /* Channels */), /* Shift */
431 false /* Swap Endian */
432 };
433
434 RTTESTI_CHECK(DrvAudioHlpPCMPropsAreValid(&cfg_p));
435
436 PDMAUDIOMIXBUF parent;
437 RTTESTI_CHECK_RC_OK(AudioMixBufInit(&parent, "Parent", &cfg_p, cBufSize));
438
439 /* 22050Hz, 1 Channel, S16 */
440 PDMAUDIOPCMPROPS cfg_c = /* Upmixing to parent */
441 {
442 16, /* Bits */
443 true, /* Signed */
444 1, /* Channels */
445 22050, /* Hz */
446 PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(16 /* Bits */, 1 /* Channels */), /* Shift */
447 false /* Swap Endian */
448 };
449
450 RTTESTI_CHECK(DrvAudioHlpPCMPropsAreValid(&cfg_c));
451
452 PDMAUDIOMIXBUF child;
453 RTTESTI_CHECK_RC_OK(AudioMixBufInit(&child, "Child", &cfg_c, cBufSize));
454 RTTESTI_CHECK_RC_OK(AudioMixBufLinkTo(&child, &parent));
455
456 /* 16-bit signed. More or less exclusively used as output, and usually as input, too. */
457 int16_t aFrames16S[16] = { 0xAA, 0xBB, INT16_MIN, INT16_MIN + 1, INT16_MIN / 2, -3, -2, -1,
458 0, 1, 2, 3, INT16_MAX / 2, INT16_MAX - 1, INT16_MAX, 0 };
459
460 /*
461 * Writing + mixing from child -> parent, sequential.
462 */
463 uint32_t cbBuf = 256;
464 char achBuf[256];
465 uint32_t cFramesRead, cFramesWritten, cFramesMixed;
466
467 uint32_t cFramesChild = 16;
468 uint32_t cFramesParent = cFramesChild * 2 - 2;
469 uint32_t cFramesTotalRead = 0;
470
471 /**** 16-bit signed samples ****/
472 RTTestPrintf(hTest, RTTESTLVL_DEBUG, "Conversion test %uHz %uch 16-bit\n", cfg_c.uHz, cfg_c.cChannels);
473 RTTESTI_CHECK_RC_OK(AudioMixBufWriteCirc(&child, &aFrames16S, sizeof(aFrames16S), &cFramesWritten));
474 RTTESTI_CHECK_MSG(cFramesWritten == cFramesChild, ("Child: Expected %RU32 written frames, got %RU32\n", cFramesChild, cFramesWritten));
475 RTTESTI_CHECK_RC_OK(AudioMixBufMixToParent(&child, cFramesWritten, &cFramesMixed));
476 uint32_t cFrames = AudioMixBufUsed(&parent);
477 RTTESTI_CHECK_MSG(AudioMixBufLive(&child) == cFrames, ("Child: Expected %RU32 mixed frames, got %RU32\n", AudioMixBufLive(&child), cFrames));
478
479 RTTESTI_CHECK(AudioMixBufUsed(&parent) == AudioMixBufLive(&child));
480
481 for (;;)
482 {
483 RTTESTI_CHECK_RC_OK_BREAK(AudioMixBufAcquireReadBlock(&parent, achBuf, cbBuf, &cFramesRead));
484 if (!cFramesRead)
485 break;
486 cFramesTotalRead += cFramesRead;
487 AudioMixBufReleaseReadBlock(&parent, cFramesRead);
488 AudioMixBufFinish(&parent, cFramesRead);
489 }
490 RTTESTI_CHECK_MSG(cFramesTotalRead == cFramesParent, ("Parent: Expected %RU32 mixed frames, got %RU32\n", cFramesParent, cFramesTotalRead));
491
492 /* Check that the frames came out unharmed. Every other sample is interpolated and we ignore it. */
493 /* NB: This also checks that the default volume setting is 0dB attenuation. */
494 int16_t *pSrc16 = &aFrames16S[0];
495 int16_t *pDst16 = (int16_t *)achBuf;
496
497 for (i = 0; i < cFramesChild - 1; ++i)
498 {
499 RTTESTI_CHECK_MSG(*pSrc16 == *pDst16, ("index %u: Dst=%d, Src=%d\n", i, *pDst16, *pSrc16));
500 pSrc16 += 1;
501 pDst16 += 2;
502 }
503
504 RTTESTI_CHECK(AudioMixBufUsed(&parent) == 0);
505 RTTESTI_CHECK(AudioMixBufLive(&child) == 0);
506
507 AudioMixBufDestroy(&parent);
508 AudioMixBufDestroy(&child);
509
510 return RTTestSubErrorCount(hTest) ? VERR_GENERAL_FAILURE : VINF_SUCCESS;
511}
512
513/* Test volume control. */
514static int tstVolume(RTTEST hTest)
515{
516 unsigned i;
517 uint32_t cBufSize = 256;
518
519 RTTestSubF(hTest, "Volume control");
520
521 /* Same for parent/child. */
522 /* 44100Hz, 2 Channels, S16 */
523 PDMAUDIOPCMPROPS cfg =
524 {
525 16, /* Bits */
526 true, /* Signed */
527 2, /* Channels */
528 44100, /* Hz */
529 PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(16 /* Bits */, 2 /* Channels */), /* Shift */
530 false /* Swap Endian */
531 };
532
533 RTTESTI_CHECK(DrvAudioHlpPCMPropsAreValid(&cfg));
534
535 PDMAUDIOVOLUME vol = { false, 0, 0 }; /* Not muted. */
536 PDMAUDIOMIXBUF parent;
537 RTTESTI_CHECK_RC_OK(AudioMixBufInit(&parent, "Parent", &cfg, cBufSize));
538
539 PDMAUDIOMIXBUF child;
540 RTTESTI_CHECK_RC_OK(AudioMixBufInit(&child, "Child", &cfg, cBufSize));
541 RTTESTI_CHECK_RC_OK(AudioMixBufLinkTo(&child, &parent));
542
543 /* A few 16-bit signed samples. */
544 int16_t aFrames16S[16] = { INT16_MIN, INT16_MIN + 1, -128, -64, -4, -1, 0, 1,
545 2, 255, 256, INT16_MAX / 2, INT16_MAX - 2, INT16_MAX - 1, INT16_MAX, 0 };
546
547 /*
548 * Writing + mixing from child -> parent.
549 */
550 uint32_t cbBuf = 256;
551 char achBuf[256];
552 uint32_t cFramesRead, cFramesWritten, cFramesMixed;
553
554 uint32_t cFramesChild = 8;
555 uint32_t cFramesParent = cFramesChild;
556 uint32_t cFramesTotalRead;
557 int16_t *pSrc16;
558 int16_t *pDst16;
559
560 /**** Volume control test ****/
561 RTTestPrintf(hTest, RTTESTLVL_DEBUG, "Volume control test %uHz %uch \n", cfg.uHz, cfg.cChannels);
562
563 /* 1) Full volume/0dB attenuation (255). */
564 vol.uLeft = vol.uRight = 255;
565 AudioMixBufSetVolume(&child, &vol);
566
567 RTTESTI_CHECK_RC_OK(AudioMixBufWriteCirc(&child, &aFrames16S, sizeof(aFrames16S), &cFramesWritten));
568 RTTESTI_CHECK_MSG(cFramesWritten == cFramesChild, ("Child: Expected %RU32 written frames, got %RU32\n", cFramesChild, cFramesWritten));
569 RTTESTI_CHECK_RC_OK(AudioMixBufMixToParent(&child, cFramesWritten, &cFramesMixed));
570
571 cFramesTotalRead = 0;
572 for (;;)
573 {
574 RTTESTI_CHECK_RC_OK_BREAK(AudioMixBufAcquireReadBlock(&parent, achBuf, cbBuf, &cFramesRead));
575 if (!cFramesRead)
576 break;
577 cFramesTotalRead += cFramesRead;
578 AudioMixBufReleaseReadBlock(&parent, cFramesRead);
579 AudioMixBufFinish(&parent, cFramesRead);
580 }
581 RTTESTI_CHECK_MSG(cFramesTotalRead == cFramesParent, ("Parent: Expected %RU32 mixed frames, got %RU32\n", cFramesParent, cFramesTotalRead));
582
583 /* Check that at 0dB the frames came out unharmed. */
584 pSrc16 = &aFrames16S[0];
585 pDst16 = (int16_t *)achBuf;
586
587 for (i = 0; i < cFramesParent * 2 /* stereo */; ++i)
588 {
589 RTTESTI_CHECK_MSG(*pSrc16 == *pDst16, ("index %u: Dst=%d, Src=%d\n", i, *pDst16, *pSrc16));
590 ++pSrc16;
591 ++pDst16;
592 }
593 AudioMixBufReset(&child);
594
595 /* 2) Half volume/-6dB attenuation (16 steps down). */
596 vol.uLeft = vol.uRight = 255 - 16;
597 AudioMixBufSetVolume(&child, &vol);
598
599 RTTESTI_CHECK_RC_OK(AudioMixBufWriteCirc(&child, &aFrames16S, sizeof(aFrames16S), &cFramesWritten));
600 RTTESTI_CHECK_MSG(cFramesWritten == cFramesChild, ("Child: Expected %RU32 written frames, got %RU32\n", cFramesChild, cFramesWritten));
601 RTTESTI_CHECK_RC_OK(AudioMixBufMixToParent(&child, cFramesWritten, &cFramesMixed));
602
603 cFramesTotalRead = 0;
604 for (;;)
605 {
606 RTTESTI_CHECK_RC_OK_BREAK(AudioMixBufAcquireReadBlock(&parent, achBuf, cbBuf, &cFramesRead));
607 if (!cFramesRead)
608 break;
609 cFramesTotalRead += cFramesRead;
610 AudioMixBufReleaseReadBlock(&parent, cFramesRead);
611 AudioMixBufFinish(&parent, cFramesRead);
612 }
613 RTTESTI_CHECK_MSG(cFramesTotalRead == cFramesParent, ("Parent: Expected %RU32 mixed frames, got %RU32\n", cFramesParent, cFramesTotalRead));
614
615 /* Check that at -6dB the sample values are halved. */
616 pSrc16 = &aFrames16S[0];
617 pDst16 = (int16_t *)achBuf;
618
619 for (i = 0; i < cFramesParent * 2 /* stereo */; ++i)
620 {
621 /* Watch out! For negative values, x >> 1 is not the same as x / 2. */
622 RTTESTI_CHECK_MSG(*pSrc16 >> 1 == *pDst16, ("index %u: Dst=%d, Src=%d\n", i, *pDst16, *pSrc16));
623 ++pSrc16;
624 ++pDst16;
625 }
626
627 AudioMixBufDestroy(&parent);
628 AudioMixBufDestroy(&child);
629
630 return RTTestSubErrorCount(hTest) ? VERR_GENERAL_FAILURE : VINF_SUCCESS;
631}
632
633int main(int argc, char **argv)
634{
635 RTR3InitExe(argc, &argv, 0);
636
637 /*
638 * Initialize IPRT and create the test.
639 */
640 RTTEST hTest;
641 int rc = RTTestInitAndCreate("tstAudioMixBuffer", &hTest);
642 if (rc)
643 return rc;
644 RTTestBanner(hTest);
645
646 rc = tstSingle(hTest);
647 if (RT_SUCCESS(rc))
648 rc = tstParentChild(hTest);
649 if (RT_SUCCESS(rc))
650 rc = tstConversion8(hTest);
651 if (RT_SUCCESS(rc))
652 rc = tstConversion16(hTest);
653 if (RT_SUCCESS(rc))
654 rc = tstVolume(hTest);
655
656 /*
657 * Summary
658 */
659 return RTTestSummaryAndDestroy(hTest);
660}
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