From b299609a9b938ad29694d4346f2e6f5d1290bddb Mon Sep 17 00:00:00 2001 From: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com> Date: Wed, 6 Nov 2024 19:26:57 +0200 Subject: [PATCH] More HLE DSP work --- include/audio/hle_core.hpp | 85 +++++++++++++++++++------------------ src/core/audio/hle_core.cpp | 23 +++++++--- 2 files changed, 60 insertions(+), 48 deletions(-) diff --git a/include/audio/hle_core.hpp b/include/audio/hle_core.hpp index d05e9808..2fccd55d 100644 --- a/include/audio/hle_core.hpp +++ b/include/audio/hle_core.hpp @@ -16,6 +16,46 @@ namespace Audio { using SampleFormat = HLE::SourceConfiguration::Configuration::Format; using SourceType = HLE::SourceConfiguration::Configuration::MonoOrStereo; + class DSPMixer { + public: + template + using Sample = std::array; + + template + using Frame = std::array, 160>; + + template + using MonoFrame = Frame; + + template + using StereoFrame = Frame; + + template + using QuadFrame = Frame; + + // Internally the DSP uses four channels when mixing. + // Neatly, QuadFrame means that every sample is a uint32x4 value, which is particularly nice for SIMD mixing + using IntermediateMix = QuadFrame; + + private: + using ChannelFormat = HLE::DspConfiguration::OutputFormat; + // The audio from each DSP voice is converted to quadraphonic and then fed into 3 intermediate mixing stages + // Two of these intermediate mixers (second and third) are used for effects, including custom effects done on the CPU + static constexpr usize mixerStageCount = 3; + + public: + ChannelFormat channelFormat = ChannelFormat::Stereo; + std::array volumes; + std::array enableAuxStages; + + void reset() { + channelFormat = ChannelFormat::Stereo; + + volumes.fill(0.0); + enableAuxStages.fill(false); + } + }; + struct DSPSource { // Audio buffer information // https://www.3dbrew.org/wiki/DSP_Memory_Region @@ -49,6 +89,7 @@ namespace Audio { using SampleBuffer = std::deque>; using BufferQueue = std::priority_queue; + DSPMixer::StereoFrame currentFrame; BufferQueue buffers; SampleFormat sampleFormat = SampleFormat::ADPCM; @@ -98,46 +139,6 @@ namespace Audio { DSPSource() { reset(); } }; - class DSPMixer { - public: - template - using Sample = std::array; - - template - using Frame = std::array, 160>; - - template - using MonoFrame = Frame; - - template - using StereoFrame = Frame; - - template - using QuadFrame = Frame; - - // Internally the DSP uses four channels when mixing. - // Neatly, QuadFrame means that every sample is a uint32x4 value, which is particularly nice for SIMD mixing - using IntermediateMix = QuadFrame; - - private: - using ChannelFormat = HLE::DspConfiguration::OutputFormat; - // The audio from each DSP voice is converted to quadraphonic and then fed into 3 intermediate mixing stages - // Two of these intermediate mixers (second and third) are used for effects, including custom effects done on the CPU - static constexpr usize mixerStageCount = 3; - - public: - ChannelFormat channelFormat = ChannelFormat::Stereo; - std::array volumes; - std::array enableAuxStages; - - void reset() { - channelFormat = ChannelFormat::Stereo; - - volumes.fill(0.0); - enableAuxStages.fill(false); - } - }; - class HLE_DSP : public DSPCore { // The audio frame types are public in case we want to use them for unit tests public: @@ -159,7 +160,7 @@ namespace Audio { using Source = Audio::DSPSource; using SampleBuffer = Source::SampleBuffer; using IntermediateMix = DSPMixer::IntermediateMix; - + private: enum class DSPState : u32 { Off, @@ -226,7 +227,7 @@ namespace Audio { void outputFrame(); // Perform the final mix, mixing the quadraphonic samples from all voices into the output audio frame void performMix(Audio::HLE::SharedMemory& readRegion, Audio::HLE::SharedMemory& writeRegion); - + // Decode an entire buffer worth of audio void decodeBuffer(DSPSource& source); diff --git a/src/core/audio/hle_core.cpp b/src/core/audio/hle_core.cpp index 9870adef..4bc548dc 100644 --- a/src/core/audio/hle_core.cpp +++ b/src/core/audio/hle_core.cpp @@ -262,10 +262,10 @@ namespace Audio { for (usize sampleIndex = 0; sampleIndex < Audio::samplesInFrame; sampleIndex++) { // Mono samples are in the format: (l, r) // When converting to quad, gain0 and gain2 are applied to the left sample, gain1 and gain3 to the right one - intermediateMix[sampleIndex][0] += s32(source.currentSamples[sampleIndex][0] * gains[0]); - intermediateMix[sampleIndex][1] += s32(source.currentSamples[sampleIndex][1] * gains[1]); - intermediateMix[sampleIndex][2] += s32(source.currentSamples[sampleIndex][0] * gains[2]); - intermediateMix[sampleIndex][3] += s32(source.currentSamples[sampleIndex][1] * gains[3]); + intermediateMix[sampleIndex][0] += s32(source.currentFrame[sampleIndex][0] * gains[0]); + intermediateMix[sampleIndex][1] += s32(source.currentFrame[sampleIndex][1] * gains[1]); + intermediateMix[sampleIndex][2] += s32(source.currentFrame[sampleIndex][0] * gains[2]); + intermediateMix[sampleIndex][3] += s32(source.currentFrame[sampleIndex][1] * gains[3]); } } } @@ -467,6 +467,9 @@ namespace Audio { } void HLE_DSP::generateFrame(DSPSource& source) { + // Zero out all output samples at first. TODO: Don't zero out the entire frame initially, rather only zero-out the "unwritten" samples when the frame is done being processed. + source.currentFrame = {}; + if (source.currentSamples.empty()) { // There's no audio left to play, turn the voice off if (source.buffers.empty()) { @@ -480,7 +483,7 @@ namespace Audio { decodeBuffer(source); } else { - uint maxSampleCount = uint(float(Audio::samplesInFrame) * source.rateMultiplier); + uint maxSampleCount = uint(float(Audio::samplesInFrame) * 1.0); uint outputCount = 0; while (outputCount < maxSampleCount) { @@ -494,8 +497,14 @@ namespace Audio { const uint sampleCount = std::min(maxSampleCount - outputCount, source.currentSamples.size()); - // samples.insert(samples.end(), source.currentSamples.begin(), source.currentSamples.begin() + sampleCount); + // Copy samples to current frame buffer + // TODO: Implement linear/polyphase interpolation + std::copy( + source.currentSamples.begin(), std::next(source.currentSamples.begin(), sampleCount), source.currentFrame.begin() + outputCount + ); + // Remove samples from sample buffer source.currentSamples.erase(source.currentSamples.begin(), std::next(source.currentSamples.begin(), sampleCount)); + // Advance sample position source.samplePosition += sampleCount; outputCount += sampleCount; } @@ -718,5 +727,7 @@ namespace Audio { buffers = {}; currentSamples.clear(); + + gains.fill({}); } } // namespace Audio