From 634880b5e3f6dd96e277e3b163946e77979570af Mon Sep 17 00:00:00 2001 From: Connor McLaughlin Date: Tue, 12 May 2020 02:15:16 +1000 Subject: [PATCH] SPU: Clamp before applying main volume not after Fixes popping in Monkey Magic. --- src/core/spu.cpp | 13 +++++++------ src/core/spu.h | 4 ++-- 2 files changed, 9 insertions(+), 8 deletions(-) diff --git a/src/core/spu.cpp b/src/core/spu.cpp index 40613e2a4..b70192f97 100644 --- a/src/core/spu.cpp +++ b/src/core/spu.cpp @@ -749,23 +749,24 @@ void SPU::Execute(TickCount ticks) // Compute reverb. s32 reverb_out_left, reverb_out_right; - ProcessReverb(Clamp16(reverb_in_left), Clamp16(reverb_in_right), &reverb_out_left, &reverb_out_right); + ProcessReverb(static_cast(Clamp16(reverb_in_left)), static_cast(Clamp16(reverb_in_right)), + &reverb_out_left, &reverb_out_right); // Mix in reverb. left_sum += reverb_out_left; right_sum += reverb_out_right; - // Apply main volume before clamping. - *(output_frame++) = Clamp16(ApplyVolume(left_sum, m_main_volume_left.current_level)); - *(output_frame++) = Clamp16(ApplyVolume(right_sum, m_main_volume_right.current_level)); + // Apply main volume after clamping. A maximum volume should not overflow here because both are 16-bit values. + *(output_frame++) = static_cast(ApplyVolume(Clamp16(left_sum), m_main_volume_left.current_level)); + *(output_frame++) = static_cast(ApplyVolume(Clamp16(right_sum), m_main_volume_right.current_level)); m_main_volume_left.Tick(); m_main_volume_right.Tick(); // Write to capture buffers. WriteToCaptureBuffer(0, cd_audio_left); WriteToCaptureBuffer(1, cd_audio_right); - WriteToCaptureBuffer(2, Clamp16(m_voices[1].last_volume)); - WriteToCaptureBuffer(3, Clamp16(m_voices[3].last_volume)); + WriteToCaptureBuffer(2, static_cast(Clamp16(m_voices[1].last_volume))); + WriteToCaptureBuffer(3, static_cast(Clamp16(m_voices[3].last_volume))); IncrementCaptureBufferPosition(); } diff --git a/src/core/spu.h b/src/core/spu.h index 96016b15b..fc0f73218 100644 --- a/src/core/spu.h +++ b/src/core/spu.h @@ -331,9 +331,9 @@ private: }; }; - static constexpr s16 Clamp16(s32 value) + static constexpr s32 Clamp16(s32 value) { - return (value < -0x8000) ? -0x8000 : (value > 0x7FFF) ? 0x7FFF : static_cast(value); + return (value < -0x8000) ? -0x8000 : (value > 0x7FFF) ? 0x7FFF : value; } static constexpr s32 ApplyVolume(s32 sample, s16 volume) { return (sample * s32(volume)) >> 15; }