diff --git a/src/audio_core/dsp_interface.h b/src/audio_core/dsp_interface.h index 4126c03ed5..8bc40ff5ed 100644 --- a/src/audio_core/dsp_interface.h +++ b/src/audio_core/dsp_interface.h @@ -11,6 +11,12 @@ #include "common/common_types.h" #include "core/memory.h" +namespace Service { +namespace DSP { +class DSP_DSP; +} // namespace DSP +} // namespace Service + namespace AudioCore { class Sink; @@ -60,6 +66,9 @@ public: /// Returns a reference to the array backing DSP memory virtual std::array& GetDspMemory() = 0; + /// Sets the dsp class that we trigger interrupts for + virtual void SetServiceToInterrupt(std::weak_ptr dsp) = 0; + /// Select the sink to use based on sink id. void SetSink(const std::string& sink_id, const std::string& audio_device); /// Get the current sink diff --git a/src/audio_core/hle/hle.cpp b/src/audio_core/hle/hle.cpp index 703112740c..4e58e3f08d 100644 --- a/src/audio_core/hle/hle.cpp +++ b/src/audio_core/hle/hle.cpp @@ -13,7 +13,9 @@ #include "common/common_types.h" #include "common/logging/log.h" #include "core/core_timing.h" -#include "core/hle/service/dsp/dsp_dsp.h" + +using InterruptType = Service::DSP::DSP_DSP::InterruptType; +using Service::DSP::DSP_DSP; namespace AudioCore { @@ -32,6 +34,8 @@ public: std::array& GetDspMemory(); + void SetServiceToInterrupt(std::weak_ptr dsp); + private: void ResetPipes(); void WriteU16(DspPipe pipe_number, u16 value); @@ -60,6 +64,8 @@ private: DspHle& parent; CoreTiming::EventType* tick_event; + + std::weak_ptr dsp_dsp; }; DspHle::Impl::Impl(DspHle& parent_) : parent(parent_) { @@ -187,6 +193,10 @@ std::array& DspHle::Impl::GetDspMemory() { return dsp_memory.raw_memory; } +void DspHle::Impl::SetServiceToInterrupt(std::weak_ptr dsp) { + dsp_dsp = std::move(dsp); +} + void DspHle::Impl::ResetPipes() { for (auto& data : pipe_data) { data.clear(); @@ -231,7 +241,9 @@ void DspHle::Impl::AudioPipeWriteStructAddresses() { WriteU16(DspPipe::Audio, addr); } // Signal that we have data on this pipe. - Service::DSP::SignalPipeInterrupt(DspPipe::Audio); + if (auto service = dsp_dsp.lock()) { + service->SignalInterrupt(InterruptType::Pipe, DspPipe::Audio); + } } size_t DspHle::Impl::CurrentRegionIndex() const { @@ -307,9 +319,11 @@ bool DspHle::Impl::Tick() { void DspHle::Impl::AudioTickCallback(int cycles_late) { if (Tick()) { // TODO(merry): Signal all the other interrupts as appropriate. - Service::DSP::SignalPipeInterrupt(DspPipe::Audio); - // HACK(merry): Added to prevent regressions. Will remove soon. - Service::DSP::SignalPipeInterrupt(DspPipe::Binary); + if (auto service = dsp_dsp.lock()) { + service->SignalInterrupt(InterruptType::Pipe, DspPipe::Audio); + // HACK(merry): Added to prevent regressions. Will remove soon. + service->SignalInterrupt(InterruptType::Pipe, DspPipe::Binary); + } } // Reschedule recurrent event @@ -339,4 +353,8 @@ std::array& DspHle::GetDspMemory() { return impl->GetDspMemory(); } +void DspHle::SetServiceToInterrupt(std::weak_ptr dsp) { + impl->SetServiceToInterrupt(std::move(dsp)); +} + } // namespace AudioCore diff --git a/src/audio_core/hle/hle.h b/src/audio_core/hle/hle.h index 00ccb6dcf7..e93abf4bcb 100644 --- a/src/audio_core/hle/hle.h +++ b/src/audio_core/hle/hle.h @@ -10,6 +10,7 @@ #include "audio_core/audio_types.h" #include "audio_core/dsp_interface.h" #include "common/common_types.h" +#include "core/hle/service/dsp/dsp_dsp.h" #include "core/memory.h" namespace AudioCore { @@ -27,6 +28,8 @@ public: std::array& GetDspMemory() override; + void SetServiceToInterrupt(std::weak_ptr dsp) override; + private: struct Impl; friend struct Impl; diff --git a/src/core/hle/service/dsp/dsp_dsp.cpp b/src/core/hle/service/dsp/dsp_dsp.cpp index 5d1648a152..47b352702c 100644 --- a/src/core/hle/service/dsp/dsp_dsp.cpp +++ b/src/core/hle/service/dsp/dsp_dsp.cpp @@ -21,8 +21,6 @@ enum class DspPipe; namespace Service { namespace DSP { -static std::weak_ptr dsp_dsp; - void DSP_DSP::RecvData(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x01, 1, 0); const u32 register_number = rp.Pop(); @@ -313,6 +311,10 @@ void DSP_DSP::ForceHeadphoneOut(Kernel::HLERequestContext& ctx) { LOG_DEBUG(Service_DSP, "(STUBBED) called, force={}", force); } +// DSP Interrupts: +// The audio-pipe interrupt occurs every frame tick. Userland programs normally have a thread +// that's waiting for an interrupt event. Immediately after this interrupt event, userland +// normally updates the state in the next region and increments the relevant frame counter by two. void DSP_DSP::SignalInterrupt(InterruptType type, DspPipe pipe) { LOG_DEBUG(Service_DSP, "called, type={}, pipe={}", static_cast(type), static_cast(pipe)); @@ -398,20 +400,10 @@ DSP_DSP::~DSP_DSP() { pipes = {}; } -// DSP Interrupts: -// The audio-pipe interrupt occurs every frame tick. Userland programs normally have a thread -// that's waiting for an interrupt event. Immediately after this interrupt event, userland -// normally updates the state in the next region and increments the relevant frame counter by two. -void SignalPipeInterrupt(DspPipe pipe) { - auto dsp = dsp_dsp.lock(); - ASSERT(dsp != nullptr); - return dsp->SignalInterrupt(InterruptType::Pipe, pipe); -} - void InstallInterfaces(SM::ServiceManager& service_manager) { auto dsp = std::make_shared(); dsp->InstallAsService(service_manager); - dsp_dsp = dsp; + Core::DSP().SetServiceToInterrupt(std::move(dsp)); } } // namespace DSP diff --git a/src/core/hle/service/dsp/dsp_dsp.h b/src/core/hle/service/dsp/dsp_dsp.h index 3b782f9d1e..c7213cce4b 100644 --- a/src/core/hle/service/dsp/dsp_dsp.h +++ b/src/core/hle/service/dsp/dsp_dsp.h @@ -251,12 +251,6 @@ private: std::array, AudioCore::num_dsp_pipe> pipes = {{}}; }; -/** - * Signal a specific DSP related interrupt of type == InterruptType::Pipe, pipe == pipe. - * @param pipe The DSP pipe for which to signal an interrupt for. - */ -void SignalPipeInterrupt(AudioCore::DspPipe pipe); - void InstallInterfaces(SM::ServiceManager& service_manager); } // namespace DSP