From 8a0d73c1558bab054823a76331eba1cef5bb96dd Mon Sep 17 00:00:00 2001 From: jdpurcell Date: Wed, 28 Jan 2015 04:48:52 +0000 Subject: [PATCH] SoundOutputProvider: Better rewind/fast-forward behavior especially with large buffer sizes. --- BizHawk.Client.EmuHawk/Sound.cs | 19 +++++++++++++------ BizHawk.Client.EmuHawk/SoundOutputProvider.cs | 3 ++- 2 files changed, 15 insertions(+), 7 deletions(-) diff --git a/BizHawk.Client.EmuHawk/Sound.cs b/BizHawk.Client.EmuHawk/Sound.cs index 8412ccd3bc..6807beeba6 100644 --- a/BizHawk.Client.EmuHawk/Sound.cs +++ b/BizHawk.Client.EmuHawk/Sound.cs @@ -124,10 +124,13 @@ namespace BizHawk.Client.EmuHawk _lastWriteTime = 0; _lastWriteCursor = 0; - int minBufferFullnessMs = - Global.Config.SoundBufferSizeMs < 80 ? 35 : - Global.Config.SoundBufferSizeMs < 100 ? 45 : - 55; + // 35 to 65 milliseconds depending on how big the buffer is. This is a trade-off + // between more frequent but less severe glitches (i.e. catching underruns before + // they happen and filling the buffer with silence) or less frequent but more + // severe glitches. At least on my Windows 8 machines, the distance between the + // play and write cursors can be up to 30 milliseconds, so that would be the + // absolute minimum we could use here. + int minBufferFullnessMs = Math.Min(35 + ((Global.Config.SoundBufferSizeMs - 60) / 2), 65); _outputProvider = new SoundOutputProvider(); _outputProvider.MaxSamplesDeficit = BufferSizeSamples - MillisecondsToSamples(minBufferFullnessMs); @@ -229,7 +232,7 @@ namespace BizHawk.Client.EmuHawk { if (LogUnderruns) Console.WriteLine("DirectSound underrun detected!"); detectedUnderrun = true; - _outputProvider.OnUnderrun(); + _outputProvider.OnVolatility(); } } bool isInitializing = _actualWriteOffsetBytes == -1; @@ -307,12 +310,16 @@ namespace BizHawk.Client.EmuHawk while (samplesNeeded < samplesProvided && !Global.DisableSecondaryThrottling) { - Thread.Sleep((samplesProvided - samplesNeeded) / (SampleRate / 1000)); // let audio clock control sleep time + Thread.Sleep((samplesProvided - samplesNeeded) / (SampleRate / 1000)); // Let the audio clock control sleep time samplesNeeded = CalculateSamplesNeeded(); } } else { + if (Global.DisableSecondaryThrottling) // This indicates rewind or fast-forward + { + _outputProvider.OnVolatility(); + } _outputProvider.GetSamples(samplesNeeded, out samples, out samplesProvided); } } diff --git a/BizHawk.Client.EmuHawk/SoundOutputProvider.cs b/BizHawk.Client.EmuHawk/SoundOutputProvider.cs index 9c0a677269..d8e7c2096c 100644 --- a/BizHawk.Client.EmuHawk/SoundOutputProvider.cs +++ b/BizHawk.Client.EmuHawk/SoundOutputProvider.cs @@ -76,7 +76,8 @@ namespace BizHawk.Client.EmuHawk } } - public void OnUnderrun() + // To let us know about buffer underruns, rewinding, fast-forwarding, etc. + public void OnVolatility() { _extraCountHistory.Clear(); _outputCountHistory.Clear();