Changeset 55462 in vbox for trunk/src/VBox/Devices/Audio
- Timestamp:
- Apr 27, 2015 8:01:05 PM (10 years ago)
- Location:
- trunk/src/VBox/Devices/Audio
- Files:
-
- 2 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Devices/Audio/AudioMixBuffer.cpp
r55447 r55462 74 74 * 75 75 * For actual volume calculation, 33.31 fixed point is used. Maximum (or 76 * unattenuated) volume is represented as 0x 80000000; conveniently, this76 * unattenuated) volume is represented as 0x40000000; conveniently, this 77 77 * value fits into a uint32_t. 78 * 79 * To enable fast processing, the maximum volume must be a power of two 80 * and must not have a sign when converted to int32_t. While 0x80000000 81 * violates these constraints, 0x40000000 does not. 78 82 */ 79 83 … … 115 119 }; 116 120 121 /* Bit shift for fixed point conversion. */ 122 #define VOL_SHIFT 30 123 124 /* Internal representation of 0dB volume (1.0 in fixed point). */ 125 #define VOL_0DB (1 << VOL_SHIFT) 126 127 AssertCompile(VOL_0DB <= 0x40000000); /* Must always hold. */ 128 AssertCompile(VOL_0DB == 0x40000000); /* For now -- when only attenuation is used. */ 129 117 130 /** 118 131 * Structure for holding sample conversion parameters for … … 343 356 AUDMIXBUF_MACRO_FN _aType audioMixBufClipTo##_aName(int64_t iVal) \ 344 357 { \ 345 if (iVal >= 0x7f 000000) \358 if (iVal >= 0x7fffffff) \ 346 359 return _aMax; \ 347 else if (iVal < - 2147483648LL) \360 else if (iVal < -INT64_C(0x80000000)) \ 348 361 return _aMin; \ 349 362 \ … … 363 376 { \ 364 377 AUDMIXBUF_MACRO_LOG(("%p: l=%RI16, r=%RI16\n", paDst, *pSrc, *(pSrc + 1))); \ 365 paDst->i64LSample = ASMMult2xS32RetS64((int32_t)audioMixBufClipFrom##_aName(*pSrc++), pOpts->Volume.uLeft ) >> 31; \366 paDst->i64RSample = ASMMult2xS32RetS64((int32_t)audioMixBufClipFrom##_aName(*pSrc++), pOpts->Volume.uRight) >> 31; \378 paDst->i64LSample = ASMMult2xS32RetS64((int32_t)audioMixBufClipFrom##_aName(*pSrc++), pOpts->Volume.uLeft ) >> VOL_SHIFT; \ 379 paDst->i64RSample = ASMMult2xS32RetS64((int32_t)audioMixBufClipFrom##_aName(*pSrc++), pOpts->Volume.uRight) >> VOL_SHIFT; \ 367 380 AUDMIXBUF_MACRO_LOG(("\t-> l=%RI64, r=%RI64\n", paDst->i64LSample, paDst->i64RSample)); \ 368 381 paDst++; \ … … 382 395 { \ 383 396 AUDMIXBUF_MACRO_LOG(("%p: s=%RI16\n", paDst, *pSrc)); \ 384 paDst->i64LSample = ASMMult2xS32RetS64((int32_t)audioMixBufClipFrom##_aName(*pSrc++), pOpts->Volume.uLeft) >> 31; \397 paDst->i64LSample = ASMMult2xS32RetS64((int32_t)audioMixBufClipFrom##_aName(*pSrc++), pOpts->Volume.uLeft) >> VOL_SHIFT; \ 385 398 paDst->i64RSample = paDst->i64LSample; \ 386 399 AUDMIXBUF_MACRO_LOG(("\t-> l=%RI64, r=%RI64\n", paDst->i64LSample, paDst->i64RSample)); \ … … 418 431 while (cSamples--) \ 419 432 { \ 420 *pDst++ = audioMixBufClipTo##_aName( pSrc->i64LSample + pSrc->i64RSample); \433 *pDst++ = audioMixBufClipTo##_aName((pSrc->i64LSample + pSrc->i64RSample) / 2); \ 421 434 pSrc++; \ 422 435 } \ … … 506 519 int64_t iDstOffInt = pRate->dstOffset & UINT32_MAX; \ 507 520 \ 508 samOut.i64LSample = (samLast.i64LSample * ((int64_t) UINT32_MAX- iDstOffInt) + samCur.i64LSample * iDstOffInt) >> 32; \509 samOut.i64RSample = (samLast.i64RSample * ((int64_t) UINT32_MAX- iDstOffInt) + samCur.i64RSample * iDstOffInt) >> 32; \521 samOut.i64LSample = (samLast.i64LSample * ((int64_t) (INT64_C(1) << 32) - iDstOffInt) + samCur.i64LSample * iDstOffInt) >> 32; \ 522 samOut.i64RSample = (samLast.i64RSample * ((int64_t) (INT64_C(1) << 32) - iDstOffInt) + samCur.i64RSample * iDstOffInt) >> 32; \ 510 523 \ 511 524 paDst->i64LSample _aOp samOut.i64LSample; \ … … 709 722 /* Set initial volume to max. */ 710 723 pMixBuf->Volume.fMuted = false; 711 pMixBuf->Volume.uLeft = 0x7FFFFFFF;712 pMixBuf->Volume.uRight = 0x7FFFFFFF;724 pMixBuf->Volume.uLeft = VOL_0DB; 725 pMixBuf->Volume.uRight = VOL_0DB; 713 726 714 727 /* Prevent division by zero. … … 1196 1209 1197 1210 pMixBuf->Volume.fMuted = pVol->fMuted; 1198 pMixBuf->Volume.uLeft = (UINT64_C(0x80000000) * pVol->uLeft) / 255;1199 pMixBuf->Volume.uRight = (UINT64_C(0x80000000) * pVol->uRight) / 255;1200 1211 //@todo: Ensure that the input is in the correct range/initialized! 1201 pMixBuf->Volume.uLeft = aVolumeConv[pVol->uLeft & 0xFF] * UINT 64_C(0x8000);1202 pMixBuf->Volume.uRight = aVolumeConv[pVol->uRight & 0xFF] * UINT 64_C(0x8000);1212 pMixBuf->Volume.uLeft = aVolumeConv[pVol->uLeft & 0xFF] * UINT32_C(VOL_0DB >> 16); 1213 pMixBuf->Volume.uRight = aVolumeConv[pVol->uRight & 0xFF] * UINT32_C(VOL_0DB >> 16); 1203 1214 1204 1215 LogFlowFunc(("\t-> lVol=%#RX32, rVol=%#RX32\n", pMixBuf->Volume.uLeft, pMixBuf->Volume.uRight)); -
trunk/src/VBox/Devices/Audio/testcase/tstAudioMixBuffer.cpp
r55449 r55462 236 236 } 237 237 238 static int tstConversion(RTTEST hTest) 239 { 240 unsigned i; 241 uint32_t cBufSize = 256; 242 PDMPCMPROPS props; 243 244 245 RTTestSubF(hTest, "Sample conversion"); 246 247 PDMAUDIOSTREAMCFG cfg_p = 248 { 249 44100, /* Hz */ 250 1 /* Channels */, 251 AUD_FMT_S16 /* Format */, 252 PDMAUDIOENDIANNESS_LITTLE /* ENDIANNESS */ 253 }; 254 255 int rc = drvAudioStreamCfgToProps(&cfg_p, &props); 256 AssertRC(rc); 257 258 PDMAUDIOMIXBUF parent; 259 RTTESTI_CHECK_RC_OK(audioMixBufInit(&parent, "Parent", &props, cBufSize)); 260 261 /* Child uses half the sample rate; that ensures the mixing engine can't 262 * take shortcuts and performs conversion. Because conversion to double 263 * the sample rate effectively inserts one additional sample between every 264 * two source samples, N source samples will be converted to N * 2 - 1 265 * samples. However, the last source sample will be saved for later 266 * interpolation and not immediately output. 267 */ 268 PDMAUDIOSTREAMCFG cfg_c = /* Upmixing to parent */ 269 { 270 22050, /* Hz */ 271 1 /* Channels */, 272 AUD_FMT_S16 /* Format */, 273 PDMAUDIOENDIANNESS_LITTLE /* ENDIANNESS */ 274 }; 275 276 rc = drvAudioStreamCfgToProps(&cfg_c, &props); 277 AssertRC(rc); 278 279 PDMAUDIOMIXBUF child; 280 RTTESTI_CHECK_RC_OK(audioMixBufInit(&child, "Child", &props, cBufSize)); 281 RTTESTI_CHECK_RC_OK(audioMixBufLinkTo(&child, &parent)); 282 283 /* 284 * Writing + mixing from child -> parent, sequential. 285 */ 286 uint32_t cbBuf = 256; 287 char pvBuf[256]; 288 int16_t samples[16] = { 0xAA, 0xBB, INT16_MIN, INT16_MIN + 1, INT16_MIN / 2, -3, -2, -1, 289 0, 1, 2, 3, INT16_MAX / 2, INT16_MAX - 1, INT16_MAX, 0 }; 290 uint32_t read, written, mixed, temp; 291 292 uint32_t cChildFree = cBufSize; 293 uint32_t cChildMixed = 0; 294 uint32_t cSamplesChild = 16; 295 uint32_t cSamplesParent = cSamplesChild * 2 - 2; 296 uint32_t cSamplesRead = 0; 297 298 RTTestPrintf(hTest, RTTESTLVL_DEBUG, "Conversion test %uHz %uch\n", cfg_c.uHz, cfg_c.cChannels); 299 RTTESTI_CHECK_RC_OK(audioMixBufWriteAt(&child, 0, &samples, sizeof(samples), &written)); 300 RTTESTI_CHECK_MSG(written == cSamplesChild, ("Child: Expected %RU32 written samples, got %RU32\n", cSamplesChild, written)); 301 RTTESTI_CHECK_RC_OK(audioMixBufMixToParent(&child, written, &mixed)); 302 temp = audioMixBufProcessed(&parent); 303 RTTESTI_CHECK_MSG(audioMixBufMixed(&child) == temp, ("Child: Expected %RU32 mixed samples, got %RU32\n", audioMixBufMixed(&child), temp)); 304 305 RTTESTI_CHECK(audioMixBufProcessed(&parent) == audioMixBufMixed(&child)); 306 307 for (;;) 308 { 309 RTTESTI_CHECK_RC_OK_BREAK(audioMixBufReadCirc(&parent, pvBuf, cbBuf, &read)); 310 if (!read) 311 break; 312 cSamplesRead += read; 313 audioMixBufFinish(&parent, read); 314 } 315 RTTESTI_CHECK_MSG(cSamplesRead == cSamplesParent, ("Parent: Expected %RU32 mixed samples, got %RU32\n", cSamplesParent, cSamplesRead)); 316 317 /* Check that the samples came out unharmed. Every other sample is interpolated and we ignore it. */ 318 int16_t *pSrc16 = &samples[0]; 319 int16_t *pDst16 = (int16_t *)pvBuf; 320 321 for (i = 0; i < cSamplesChild; ++i) 322 { 323 RTTESTI_CHECK_MSG(*pSrc16 == *pDst16, ("index %u: Dst=%d, Src=%d\n", i, *pDst16, *pSrc16)); 324 pSrc16 += 1; 325 pDst16 += 2; 326 } 327 328 RTTESTI_CHECK(audioMixBufProcessed(&parent) == 0); 329 RTTESTI_CHECK(audioMixBufMixed(&child) == 0); 330 331 return RTTestSubErrorCount(hTest) ? VERR_GENERAL_FAILURE : VINF_SUCCESS; 332 } 333 238 334 int main(int argc, char **argv) 239 335 { … … 252 348 if (RT_SUCCESS(rc)) 253 349 rc = tstParentChild(hTest); 350 if (RT_SUCCESS(rc)) 351 rc = tstConversion(hTest); 254 352 255 353 /*
Note:
See TracChangeset
for help on using the changeset viewer.