/* Copyright 2016-2023 melonDS team This file is part of melonDS. melonDS is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. melonDS is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with melonDS. If not, see http://www.gnu.org/licenses/. */ #include #include #include #include #include #include #include #include #include #include "main.h" #include "Input.h" #include "AudioInOut.h" #include "types.h" #include "version.h" #include "FrontendUtil.h" #include "Args.h" #include "NDS.h" #include "NDSCart.h" #include "GBACart.h" #include "GPU.h" #include "SPU.h" #include "Wifi.h" #include "Platform.h" #include "LocalMP.h" #include "Config.h" #include "RTC.h" #include "DSi.h" #include "DSi_I2C.h" #include "GPU3D_Soft.h" #include "GPU3D_OpenGL.h" #include "Savestate.h" #include "ROMManager.h" //#include "ArchiveUtil.h" //#include "CameraManager.h" //#include "CLI.h" // TODO: uniform variable spelling using namespace melonDS; // TEMP extern bool RunningSomething; extern MainWindow* mainWindow; extern int autoScreenSizing; extern int videoRenderer; extern bool videoSettingsDirty; EmuThread::EmuThread(QObject* parent) : QThread(parent) { EmuStatus = emuStatus_Exit; EmuRunning = emuStatus_Paused; EmuPauseStack = EmuPauseStackRunning; RunningSomething = false; connect(this, SIGNAL(windowUpdate()), mainWindow->panel, SLOT(repaint())); connect(this, SIGNAL(windowTitleChange(QString)), mainWindow, SLOT(onTitleUpdate(QString))); connect(this, SIGNAL(windowEmuStart()), mainWindow, SLOT(onEmuStart())); connect(this, SIGNAL(windowEmuStop()), mainWindow, SLOT(onEmuStop())); connect(this, SIGNAL(windowEmuPause()), mainWindow->actPause, SLOT(trigger())); connect(this, SIGNAL(windowEmuReset()), mainWindow->actReset, SLOT(trigger())); connect(this, SIGNAL(windowEmuFrameStep()), mainWindow->actFrameStep, SLOT(trigger())); connect(this, SIGNAL(windowLimitFPSChange()), mainWindow->actLimitFramerate, SLOT(trigger())); connect(this, SIGNAL(screenLayoutChange()), mainWindow->panel, SLOT(onScreenLayoutChanged())); connect(this, SIGNAL(windowFullscreenToggle()), mainWindow, SLOT(onFullscreenToggled())); connect(this, SIGNAL(swapScreensToggle()), mainWindow->actScreenSwap, SLOT(trigger())); connect(this, SIGNAL(screenEmphasisToggle()), mainWindow, SLOT(onScreenEmphasisToggled())); } std::unique_ptr EmuThread::CreateConsole( std::unique_ptr&& ndscart, std::unique_ptr&& gbacart ) noexcept { auto arm7bios = ROMManager::LoadARM7BIOS(); if (!arm7bios) return nullptr; auto arm9bios = ROMManager::LoadARM9BIOS(); if (!arm9bios) return nullptr; auto firmware = ROMManager::LoadFirmware(Config::ConsoleType); if (!firmware) return nullptr; #ifdef JIT_ENABLED JITArgs jitargs { static_cast(Config::JIT_MaxBlockSize), Config::JIT_LiteralOptimisations, Config::JIT_BranchOptimisations, Config::JIT_FastMemory, }; #endif #ifdef GDBSTUB_ENABLED GDBArgs gdbargs { static_cast(Config::GdbPortARM7), static_cast(Config::GdbPortARM9), Config::GdbARM7BreakOnStartup, Config::GdbARM9BreakOnStartup, }; #endif NDSArgs ndsargs { std::move(ndscart), std::move(gbacart), *arm9bios, *arm7bios, std::move(*firmware), #ifdef JIT_ENABLED Config::JIT_Enable ? std::make_optional(jitargs) : std::nullopt, #else std::nullopt, #endif static_cast(Config::AudioBitDepth), static_cast(Config::AudioInterp), #ifdef GDBSTUB_ENABLED Config::GdbEnabled ? std::make_optional(gdbargs) : std::nullopt, #else std::nullopt, #endif }; if (Config::ConsoleType == 1) { auto arm7ibios = ROMManager::LoadDSiARM7BIOS(); if (!arm7ibios) return nullptr; auto arm9ibios = ROMManager::LoadDSiARM9BIOS(); if (!arm9ibios) return nullptr; auto nand = ROMManager::LoadNAND(*arm7ibios); if (!nand) return nullptr; auto sdcard = ROMManager::LoadDSiSDCard(); DSiArgs args { std::move(ndsargs), *arm9ibios, *arm7ibios, std::move(*nand), std::move(sdcard), Config::DSiFullBIOSBoot, }; args.GBAROM = nullptr; return std::make_unique(std::move(args)); } return std::make_unique(std::move(ndsargs)); } bool EmuThread::UpdateConsole(UpdateConsoleNDSArgs&& ndsargs, UpdateConsoleGBAArgs&& gbaargs) noexcept { // Let's get the cart we want to use; // if we wnat to keep the cart, we'll eject it from the existing console first. std::unique_ptr nextndscart; if (std::holds_alternative(ndsargs)) { // If we want to keep the existing cart (if any)... nextndscart = NDS ? NDS->EjectCart() : nullptr; ndsargs = {}; } else if (const auto ptr = std::get_if>(&ndsargs)) { nextndscart = std::move(*ptr); ndsargs = {}; } if (auto* cartsd = dynamic_cast(nextndscart.get())) { // LoadDLDISDCard will return nullopt if the SD card is disabled; // SetSDCard will accept nullopt, which means no SD card cartsd->SetSDCard(ROMManager::GetDLDISDCardArgs()); } std::unique_ptr nextgbacart; if (std::holds_alternative(gbaargs)) { nextgbacart = NDS ? NDS->EjectGBACart() : nullptr; } else if (const auto ptr = std::get_if>(&gbaargs)) { nextgbacart = std::move(*ptr); gbaargs = {}; } if (!NDS || NDS->ConsoleType != Config::ConsoleType) { // If we're switching between DS and DSi mode, or there's no console... // To ensure the destructor is called before a new one is created, // as the presence of global signal handlers still complicates things a bit NDS = nullptr; NDS::Current = nullptr; NDS = CreateConsole(std::move(nextndscart), std::move(nextgbacart)); if (NDS == nullptr) return false; NDS->Reset(); NDS::Current = NDS.get(); return true; } auto arm9bios = ROMManager::LoadARM9BIOS(); if (!arm9bios) return false; auto arm7bios = ROMManager::LoadARM7BIOS(); if (!arm7bios) return false; auto firmware = ROMManager::LoadFirmware(NDS->ConsoleType); if (!firmware) return false; if (NDS->ConsoleType == 1) { // If the console we're updating is a DSi... DSi& dsi = static_cast(*NDS); auto arm9ibios = ROMManager::LoadDSiARM9BIOS(); if (!arm9ibios) return false; auto arm7ibios = ROMManager::LoadDSiARM7BIOS(); if (!arm7ibios) return false; auto nandimage = ROMManager::LoadNAND(*arm7ibios); if (!nandimage) return false; auto dsisdcard = ROMManager::LoadDSiSDCard(); dsi.SetFullBIOSBoot(Config::DSiFullBIOSBoot); dsi.ARM7iBIOS = *arm7ibios; dsi.ARM9iBIOS = *arm9ibios; dsi.SetNAND(std::move(*nandimage)); dsi.SetSDCard(std::move(dsisdcard)); // We're moving the optional, not the card // (inserting std::nullopt here is okay, it means no card) dsi.EjectGBACart(); } if (NDS->ConsoleType == 0) { NDS->SetGBACart(std::move(nextgbacart)); } #ifdef JIT_ENABLED JITArgs jitargs { static_cast(Config::JIT_MaxBlockSize), Config::JIT_LiteralOptimisations, Config::JIT_BranchOptimisations, Config::JIT_FastMemory, }; NDS->SetJITArgs(Config::JIT_Enable ? std::make_optional(jitargs) : std::nullopt); #endif NDS->SetARM7BIOS(*arm7bios); NDS->SetARM9BIOS(*arm9bios); NDS->SetFirmware(std::move(*firmware)); NDS->SetNDSCart(std::move(nextndscart)); NDS->SPU.SetInterpolation(static_cast(Config::AudioInterp)); NDS->SPU.SetDegrade10Bit(static_cast(Config::AudioBitDepth)); NDS::Current = NDS.get(); return true; } void EmuThread::run() { u32 mainScreenPos[3]; Platform::FileHandle* file; UpdateConsole(nullptr, nullptr); // No carts are inserted when melonDS first boots mainScreenPos[0] = 0; mainScreenPos[1] = 0; mainScreenPos[2] = 0; autoScreenSizing = 0; videoSettingsDirty = false; if (mainWindow->hasOGL) { screenGL = static_cast(mainWindow->panel); screenGL->initOpenGL(); videoRenderer = Config::_3DRenderer; } else { screenGL = nullptr; videoRenderer = 0; } if (videoRenderer == 0) { // If we're using the software renderer... NDS->GPU.SetRenderer3D(std::make_unique(Config::Threaded3D != 0)); } else { auto glrenderer = melonDS::GLRenderer::New(); glrenderer->SetRenderSettings(Config::GL_BetterPolygons, Config::GL_ScaleFactor); NDS->GPU.SetRenderer3D(std::move(glrenderer)); } Input::Init(); u32 nframes = 0; double perfCountsSec = 1.0 / SDL_GetPerformanceFrequency(); double lastTime = SDL_GetPerformanceCounter() * perfCountsSec; double frameLimitError = 0.0; double lastMeasureTime = lastTime; u32 winUpdateCount = 0, winUpdateFreq = 1; u8 dsiVolumeLevel = 0x1F; file = Platform::OpenLocalFile("rtc.bin", Platform::FileMode::Read); if (file) { RTC::StateData state; Platform::FileRead(&state, sizeof(state), 1, file); Platform::CloseFile(file); NDS->RTC.SetState(state); } char melontitle[100]; while (EmuRunning != emuStatus_Exit) { Input::Process(); if (Input::HotkeyPressed(HK_FastForwardToggle)) emit windowLimitFPSChange(); if (Input::HotkeyPressed(HK_Pause)) emit windowEmuPause(); if (Input::HotkeyPressed(HK_Reset)) emit windowEmuReset(); if (Input::HotkeyPressed(HK_FrameStep)) emit windowEmuFrameStep(); if (Input::HotkeyPressed(HK_FullscreenToggle)) emit windowFullscreenToggle(); if (Input::HotkeyPressed(HK_SwapScreens)) emit swapScreensToggle(); if (Input::HotkeyPressed(HK_SwapScreenEmphasis)) emit screenEmphasisToggle(); if (EmuRunning == emuStatus_Running || EmuRunning == emuStatus_FrameStep) { EmuStatus = emuStatus_Running; if (EmuRunning == emuStatus_FrameStep) EmuRunning = emuStatus_Paused; if (Input::HotkeyPressed(HK_SolarSensorDecrease)) { int level = NDS->GBACartSlot.SetInput(GBACart::Input_SolarSensorDown, true); if (level != -1) { mainWindow->osdAddMessage(0, "Solar sensor level: %d", level); } } if (Input::HotkeyPressed(HK_SolarSensorIncrease)) { int level = NDS->GBACartSlot.SetInput(GBACart::Input_SolarSensorUp, true); if (level != -1) { mainWindow->osdAddMessage(0, "Solar sensor level: %d", level); } } if (NDS->ConsoleType == 1) { DSi& dsi = static_cast(*NDS); double currentTime = SDL_GetPerformanceCounter() * perfCountsSec; // Handle power button if (Input::HotkeyDown(HK_PowerButton)) { dsi.I2C.GetBPTWL()->SetPowerButtonHeld(currentTime); } else if (Input::HotkeyReleased(HK_PowerButton)) { dsi.I2C.GetBPTWL()->SetPowerButtonReleased(currentTime); } // Handle volume buttons if (Input::HotkeyDown(HK_VolumeUp)) { dsi.I2C.GetBPTWL()->SetVolumeSwitchHeld(DSi_BPTWL::volumeKey_Up); } else if (Input::HotkeyReleased(HK_VolumeUp)) { dsi.I2C.GetBPTWL()->SetVolumeSwitchReleased(DSi_BPTWL::volumeKey_Up); } if (Input::HotkeyDown(HK_VolumeDown)) { dsi.I2C.GetBPTWL()->SetVolumeSwitchHeld(DSi_BPTWL::volumeKey_Down); } else if (Input::HotkeyReleased(HK_VolumeDown)) { dsi.I2C.GetBPTWL()->SetVolumeSwitchReleased(DSi_BPTWL::volumeKey_Down); } dsi.I2C.GetBPTWL()->ProcessVolumeSwitchInput(currentTime); } // update render settings if needed // HACK: // once the fast forward hotkey is released, we need to update vsync // to the old setting again if (videoSettingsDirty || Input::HotkeyReleased(HK_FastForward)) { if (screenGL) { screenGL->setSwapInterval(Config::ScreenVSync ? Config::ScreenVSyncInterval : 0); videoRenderer = Config::_3DRenderer; } #ifdef OGLRENDERER_ENABLED else #endif { videoRenderer = 0; } videoRenderer = screenGL ? Config::_3DRenderer : 0; videoSettingsDirty = false; if (videoRenderer == 0) { // If we're using the software renderer... NDS->GPU.SetRenderer3D(std::make_unique(Config::Threaded3D != 0)); } else { auto glrenderer = melonDS::GLRenderer::New(); glrenderer->SetRenderSettings(Config::GL_BetterPolygons, Config::GL_ScaleFactor); NDS->GPU.SetRenderer3D(std::move(glrenderer)); } } // process input and hotkeys NDS->SetKeyMask(Input::InputMask); if (Input::HotkeyPressed(HK_Lid)) { bool lid = !NDS->IsLidClosed(); NDS->SetLidClosed(lid); mainWindow->osdAddMessage(0, lid ? "Lid closed" : "Lid opened"); } // microphone input AudioInOut::MicProcess(*NDS); // auto screen layout if (Config::ScreenSizing == Frontend::screenSizing_Auto) { mainScreenPos[2] = mainScreenPos[1]; mainScreenPos[1] = mainScreenPos[0]; mainScreenPos[0] = NDS->PowerControl9 >> 15; int guess; if (mainScreenPos[0] == mainScreenPos[2] && mainScreenPos[0] != mainScreenPos[1]) { // constant flickering, likely displaying 3D on both screens // TODO: when both screens are used for 2D only...??? guess = Frontend::screenSizing_Even; } else { if (mainScreenPos[0] == 1) guess = Frontend::screenSizing_EmphTop; else guess = Frontend::screenSizing_EmphBot; } if (guess != autoScreenSizing) { autoScreenSizing = guess; emit screenLayoutChange(); } } // emulate u32 nlines = NDS->RunFrame(); if (ROMManager::NDSSave) ROMManager::NDSSave->CheckFlush(); if (ROMManager::GBASave) ROMManager::GBASave->CheckFlush(); if (ROMManager::FirmwareSave) ROMManager::FirmwareSave->CheckFlush(); if (!screenGL) { FrontBufferLock.lock(); FrontBuffer = NDS->GPU.FrontBuffer; FrontBufferLock.unlock(); } else { FrontBuffer = NDS->GPU.FrontBuffer; screenGL->drawScreenGL(); } #ifdef MELONCAP MelonCap::Update(); #endif // MELONCAP if (EmuRunning == emuStatus_Exit) break; winUpdateCount++; if (winUpdateCount >= winUpdateFreq && !screenGL) { emit windowUpdate(); winUpdateCount = 0; } bool fastforward = Input::HotkeyDown(HK_FastForward); if (fastforward && screenGL && Config::ScreenVSync) { screenGL->setSwapInterval(0); } if (Config::DSiVolumeSync && NDS->ConsoleType == 1) { DSi& dsi = static_cast(*NDS); u8 volumeLevel = dsi.I2C.GetBPTWL()->GetVolumeLevel(); if (volumeLevel != dsiVolumeLevel) { dsiVolumeLevel = volumeLevel; emit syncVolumeLevel(); } Config::AudioVolume = volumeLevel * (256.0 / 31.0); } if (Config::AudioSync && !fastforward) AudioInOut::AudioSync(*this->NDS); double frametimeStep = nlines / (60.0 * 263.0); { bool limitfps = Config::LimitFPS && !fastforward; double practicalFramelimit = limitfps ? frametimeStep : 1.0 / Config::MaxFPS; double curtime = SDL_GetPerformanceCounter() * perfCountsSec; frameLimitError += practicalFramelimit - (curtime - lastTime); if (frameLimitError < -practicalFramelimit) frameLimitError = -practicalFramelimit; if (frameLimitError > practicalFramelimit) frameLimitError = practicalFramelimit; if (round(frameLimitError * 1000.0) > 0.0) { SDL_Delay(round(frameLimitError * 1000.0)); double timeBeforeSleep = curtime; curtime = SDL_GetPerformanceCounter() * perfCountsSec; frameLimitError -= curtime - timeBeforeSleep; } lastTime = curtime; } nframes++; if (nframes >= 30) { double time = SDL_GetPerformanceCounter() * perfCountsSec; double dt = time - lastMeasureTime; lastMeasureTime = time; u32 fps = round(nframes / dt); nframes = 0; float fpstarget = 1.0/frametimeStep; winUpdateFreq = fps / (u32)round(fpstarget); if (winUpdateFreq < 1) winUpdateFreq = 1; int inst = Platform::InstanceID(); if (inst == 0) sprintf(melontitle, "[%d/%.0f] melonDS " MELONDS_VERSION, fps, fpstarget); else sprintf(melontitle, "[%d/%.0f] melonDS (%d)", fps, fpstarget, inst+1); changeWindowTitle(melontitle); } } else { // paused nframes = 0; lastTime = SDL_GetPerformanceCounter() * perfCountsSec; lastMeasureTime = lastTime; emit windowUpdate(); EmuStatus = EmuRunning; int inst = Platform::InstanceID(); if (inst == 0) sprintf(melontitle, "melonDS " MELONDS_VERSION); else sprintf(melontitle, "melonDS (%d)", inst+1); changeWindowTitle(melontitle); SDL_Delay(75); if (screenGL) screenGL->drawScreenGL(); ContextRequestKind contextRequest = ContextRequest; if (contextRequest == contextRequest_InitGL) { screenGL = static_cast(mainWindow->panel); screenGL->initOpenGL(); ContextRequest = contextRequest_None; } else if (contextRequest == contextRequest_DeInitGL) { screenGL->deinitOpenGL(); screenGL = nullptr; ContextRequest = contextRequest_None; } } } file = Platform::OpenLocalFile("rtc.bin", Platform::FileMode::Write); if (file) { RTC::StateData state; NDS->RTC.GetState(state); Platform::FileWrite(&state, sizeof(state), 1, file); Platform::CloseFile(file); } EmuStatus = emuStatus_Exit; NDS::Current = nullptr; // nds is out of scope, so unique_ptr cleans it up for us } void EmuThread::changeWindowTitle(char* title) { emit windowTitleChange(QString(title)); } void EmuThread::emuRun() { EmuRunning = emuStatus_Running; EmuPauseStack = EmuPauseStackRunning; RunningSomething = true; // checkme emit windowEmuStart(); AudioInOut::Enable(); } void EmuThread::initContext() { ContextRequest = contextRequest_InitGL; while (ContextRequest != contextRequest_None); } void EmuThread::deinitContext() { ContextRequest = contextRequest_DeInitGL; while (ContextRequest != contextRequest_None); } void EmuThread::emuPause() { EmuPauseStack++; if (EmuPauseStack > EmuPauseStackPauseThreshold) return; PrevEmuStatus = EmuRunning; EmuRunning = emuStatus_Paused; while (EmuStatus != emuStatus_Paused); AudioInOut::Disable(); } void EmuThread::emuUnpause() { if (EmuPauseStack < EmuPauseStackPauseThreshold) return; EmuPauseStack--; if (EmuPauseStack >= EmuPauseStackPauseThreshold) return; EmuRunning = PrevEmuStatus; AudioInOut::Enable(); } void EmuThread::emuStop() { EmuRunning = emuStatus_Exit; EmuPauseStack = EmuPauseStackRunning; AudioInOut::Disable(); } void EmuThread::emuFrameStep() { if (EmuPauseStack < EmuPauseStackPauseThreshold) emit windowEmuPause(); EmuRunning = emuStatus_FrameStep; } bool EmuThread::emuIsRunning() { return EmuRunning == emuStatus_Running; } bool EmuThread::emuIsActive() { return (RunningSomething == 1); }