From 8c296905f4a84409eafd98f5b2f7da62929b2814 Mon Sep 17 00:00:00 2001 From: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com> Date: Fri, 26 Apr 2024 00:54:21 +0300 Subject: [PATCH 1/2] Qt: Add support for dumping DSP firmware --- include/panda_qt/main_window.hpp | 1 + include/services/dsp.hpp | 20 +++++++++++--- src/core/services/dsp.cpp | 25 +++++++++++++++--- src/panda_qt/main_window.cpp | 45 ++++++++++++++++++++++++++++++++ 4 files changed, 84 insertions(+), 7 deletions(-) diff --git a/include/panda_qt/main_window.hpp b/include/panda_qt/main_window.hpp index 208da2c3..7e93bdf6 100644 --- a/include/panda_qt/main_window.hpp +++ b/include/panda_qt/main_window.hpp @@ -106,6 +106,7 @@ class MainWindow : public QMainWindow { void emuThreadMainLoop(); void selectLuaFile(); void selectROM(); + void dumpDspFirmware(); void dumpRomFS(); void openLuaEditor(); void openCheatsEditor(); diff --git a/include/services/dsp.hpp b/include/services/dsp.hpp index bc27377d..5cbd4fd5 100644 --- a/include/services/dsp.hpp +++ b/include/services/dsp.hpp @@ -1,6 +1,9 @@ #pragma once #include +#include #include +#include + #include "audio/dsp_core.hpp" #include "helpers.hpp" #include "logger.hpp" @@ -34,9 +37,10 @@ class DSPService { // Total number of DSP service events registered with registerInterruptEvents size_t totalEventCount; + std::vector loadedComponent; // Service functions - void convertProcessAddressFromDspDram(u32 messagePointer); // Nice function name + void convertProcessAddressFromDspDram(u32 messagePointer); // Nice function name void flushDataCache(u32 messagePointer); void getHeadphoneStatus(u32 messagePointer); void getSemaphoreEventHandle(u32 messagePointer); @@ -51,23 +55,31 @@ class DSPService { void unloadComponent(u32 messagePointer); void writeProcessPipe(u32 messagePointer); -public: + public: DSPService(Memory& mem, Kernel& kernel) : mem(mem), kernel(kernel) {} void reset(); void handleSyncRequest(u32 messagePointer); void setDSPCore(Audio::DSPCore* pointer) { dsp = pointer; } - + // Special callback that's ran when the semaphore event is signalled void onSemaphoreEventSignal() { dsp->setSemaphore(semaphoreMask); } enum class SoundOutputMode : u8 { Mono = 0, Stereo = 1, - Surround = 2 + Surround = 2, + }; + + enum class ComponentDumpResult : u8 { + Success = 0, + NotLoaded, + FileFailure, }; void triggerPipeEvent(int index); void triggerSemaphoreEvent(); void triggerInterrupt0(); void triggerInterrupt1(); + + ComponentDumpResult dumpComponent(const std::filesystem::path& path); }; \ No newline at end of file diff --git a/src/core/services/dsp.cpp b/src/core/services/dsp.cpp index 33c1703d..8c514761 100644 --- a/src/core/services/dsp.cpp +++ b/src/core/services/dsp.cpp @@ -3,6 +3,7 @@ #include "kernel.hpp" #include +#include namespace DSPCommands { enum : u32 { @@ -41,6 +42,8 @@ void DSPService::reset() { for (DSPEvent& e : pipeEvents) { e = std::nullopt; } + + loadedComponent.clear(); } void DSPService::handleSyncRequest(u32 messagePointer) { @@ -80,13 +83,14 @@ void DSPService::loadComponent(u32 messagePointer) { u32 dataMask = mem.read32(messagePointer + 12); u32 buffer = mem.read32(messagePointer + 20); - std::vector data(size); + loadedComponent.resize(size); + for (u32 i = 0; i < size; i++) { - data[i] = mem.read8(buffer + i); + loadedComponent[i] = mem.read8(buffer + i); } log("DSP::LoadComponent (size = %08X, program mask = %X, data mask = %X\n", size, programMask, dataMask); - dsp->loadComponent(data, programMask, dataMask); + dsp->loadComponent(loadedComponent, programMask, dataMask); mem.write32(messagePointer, IPC::responseHeader(0x11, 2, 2)); mem.write32(messagePointer + 4, Result::Success); @@ -262,6 +266,21 @@ void DSPService::invalidateDCache(u32 messagePointer) { mem.write32(messagePointer + 4, Result::Success); } +DSPService::ComponentDumpResult DSPService::dumpComponent(const std::filesystem::path& path) { + if (loadedComponent.empty()) { + return ComponentDumpResult::NotLoaded; + } + + std::ofstream file(path, std::ios::out | std::ios::binary); + if (file.bad()) { + return ComponentDumpResult::FileFailure; + } + + file.write((char*)&loadedComponent[0], loadedComponent.size() * sizeof(u8)); + file.close(); + return ComponentDumpResult::Success; +} + void DSPService::triggerPipeEvent(int index) { if (index < pipeCount && pipeEvents[index].has_value()) { kernel.signalEvent(*pipeEvents[index]); diff --git a/src/panda_qt/main_window.cpp b/src/panda_qt/main_window.cpp index a4fc20f0..da9d2706 100644 --- a/src/panda_qt/main_window.cpp +++ b/src/panda_qt/main_window.cpp @@ -9,6 +9,7 @@ #include "cheats.hpp" #include "input_mappings.hpp" +#include "services/dsp.hpp" MainWindow::MainWindow(QApplication* app, QWidget* parent) : QMainWindow(parent), keyboardMappings(InputMappings::defaultKeyboardMappings()), screen(this) { setWindowTitle("Alber"); @@ -53,9 +54,12 @@ MainWindow::MainWindow(QApplication* app, QWidget* parent) : QMainWindow(parent) auto dumpRomFSAction = toolsMenu->addAction(tr("Dump RomFS")); auto luaEditorAction = toolsMenu->addAction(tr("Open Lua Editor")); auto cheatsEditorAction = toolsMenu->addAction(tr("Open Cheats Editor")); + auto dumpDspFirmware = toolsMenu->addAction(tr("Dump loaded DSP firmware")); + connect(dumpRomFSAction, &QAction::triggered, this, &MainWindow::dumpRomFS); connect(luaEditorAction, &QAction::triggered, this, &MainWindow::openLuaEditor); connect(cheatsEditorAction, &QAction::triggered, this, &MainWindow::openCheatsEditor); + connect(dumpDspFirmware, &QAction::triggered, this, &MainWindow::dumpDspFirmware); auto aboutAction = aboutMenu->addAction(tr("About Panda3DS")); connect(aboutAction, &QAction::triggered, this, &MainWindow::showAboutMenu); @@ -241,6 +245,47 @@ void MainWindow::dumpRomFS() { } } +void MainWindow::dumpDspFirmware() { + auto file = QFileDialog::getSaveFileName(this, tr("Select file"), "", tr("DSP firmware file (*.cdc)")); + + if (file.isEmpty()) { + return; + } + std::filesystem::path path(file.toStdU16String()); + + messageQueueMutex.lock(); + auto res = emu->getServiceManager().getDSP().dumpComponent(path); + messageQueueMutex.unlock(); + + switch (res) { + case DSPService::ComponentDumpResult::Success: break; + case DSPService::ComponentDumpResult::NotLoaded: { + QMessageBox messageBox( + QMessageBox::Icon::Warning, tr("No DSP firmware loaded"), + tr("The currently loaded app has not uploaded a firmware to the DSP") + ); + + QAbstractButton* button = messageBox.addButton(tr("OK"), QMessageBox::ButtonRole::YesRole); + button->setIcon(QIcon(":/docs/img/rsob_icon.png")); + messageBox.exec(); + break; + } + + case DSPService::ComponentDumpResult::FileFailure: { + QMessageBox messageBox( + QMessageBox::Icon::Warning, tr("Failed to open output file"), + tr("The currently loaded DSP firmware could not be written to the selected file. Please make sure you have permission to access this " + "file") + ); + + QAbstractButton* button = messageBox.addButton(tr("OK"), QMessageBox::ButtonRole::YesRole); + button->setIcon(QIcon(":/docs/img/rstarstruck_icon.png")); + messageBox.exec(); + break; + } + } +} + void MainWindow::showAboutMenu() { AboutWindow about(this); about.exec(); From 01875e080a269cf422789a3492b3bd672a331275 Mon Sep 17 00:00:00 2001 From: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com> Date: Fri, 26 Apr 2024 01:21:02 +0300 Subject: [PATCH 2/2] CI: Switch to MacOS 13 --- .github/workflows/Hydra_Build.yml | 2 +- .github/workflows/MacOS_Build.yml | 2 +- .github/workflows/Qt_Build.yml | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/Hydra_Build.yml b/.github/workflows/Hydra_Build.yml index 3387d46d..1664d7d2 100644 --- a/.github/workflows/Hydra_Build.yml +++ b/.github/workflows/Hydra_Build.yml @@ -40,7 +40,7 @@ jobs: MacOS: - runs-on: macos-latest + runs-on: macos-13 steps: - uses: actions/checkout@v2 diff --git a/.github/workflows/MacOS_Build.yml b/.github/workflows/MacOS_Build.yml index b659e3fa..f6fafde9 100644 --- a/.github/workflows/MacOS_Build.yml +++ b/.github/workflows/MacOS_Build.yml @@ -16,7 +16,7 @@ jobs: # well on Windows or Mac. You can convert this to a matrix build if you need # cross-platform coverage. # See: https://docs.github.com/en/free-pro-team@latest/actions/learn-github-actions/managing-complex-workflows#using-a-build-matrix - runs-on: macos-latest + runs-on: macos-13 steps: - uses: actions/checkout@v2 diff --git a/.github/workflows/Qt_Build.yml b/.github/workflows/Qt_Build.yml index 0b3910d7..5e622c54 100644 --- a/.github/workflows/Qt_Build.yml +++ b/.github/workflows/Qt_Build.yml @@ -51,7 +51,7 @@ jobs: path: upload MacOS: - runs-on: macos-latest + runs-on: macos-13 steps: - uses: actions/checkout@v2