From a04982bc84f0328b5641ec08acd4f0e94a480ba2 Mon Sep 17 00:00:00 2001 From: jdpurcell Date: Sat, 24 Jan 2015 21:52:27 +0000 Subject: [PATCH] SoundOutputProvider: Small improvements to resampling (round instead of truncate and noise shaping) - probably inaudible. --- BizHawk.Client.EmuHawk/SoundOutputProvider.cs | 21 +++++++++++++------ 1 file changed, 15 insertions(+), 6 deletions(-) diff --git a/BizHawk.Client.EmuHawk/SoundOutputProvider.cs b/BizHawk.Client.EmuHawk/SoundOutputProvider.cs index 18a184bf03..f9328d7bbf 100644 --- a/BizHawk.Client.EmuHawk/SoundOutputProvider.cs +++ b/BizHawk.Client.EmuHawk/SoundOutputProvider.cs @@ -163,8 +163,8 @@ namespace BizHawk.Client.EmuHawk } } - double newCountTarget = count * scaleFactor; - int newCount = (int)Math.Round(newCountTarget + _resampleLengthRoundingError); + double newCountExact = (count * scaleFactor) + _resampleLengthRoundingError; + int newCount = (int)Math.Round(newCountExact); // Due to small inaccuracies and rounding errors, it's pointless to resample by // just a sample or two because those may be fluctuations that will average out // over time. So instead of immediately resampling to cover small differences, we @@ -179,7 +179,7 @@ namespace BizHawk.Client.EmuHawk // time so we need to keep track of it. With NTSC @ 59.94 FPS, for example, if we // were to always round to 736 samples per frame ignoring the rounding error, we // would drift by ~22 milliseconds per minute. - _resampleLengthRoundingError += newCountTarget - count; + _resampleLengthRoundingError = newCountExact - count; AddSamplesToBuffer(samples, count); } @@ -236,6 +236,7 @@ namespace BizHawk.Client.EmuHawk return output; } + double roundingError = 0.0; for (int iOutput = 0; iOutput < outputCount; iOutput++) { double iInput = ((double)iOutput / (outputCount - 1)) * (inputCount - 1); @@ -249,9 +250,17 @@ namespace BizHawk.Client.EmuHawk for (int iChannel = 0; iChannel < ChannelCount; iChannel++) { - output[iOutput * ChannelCount + iChannel] = (short) - (input[iInput0 * ChannelCount + iChannel] * input0Weight + - input[iInput1 * ChannelCount + iChannel] * input1Weight); + double valueExact = + input[iInput0 * ChannelCount + iChannel] * input0Weight + + input[iInput1 * ChannelCount + iChannel] * input1Weight + + roundingError; + + if (valueExact < -32768.0) valueExact = -32768.0; + if (valueExact > 32767.0) valueExact = 32767.0; + + short value = (short)((int)(valueExact + 32768.5) - 32768); + output[iOutput * ChannelCount + iChannel] = value; + roundingError = valueExact - value; } }