/* PCSX2 - PS2 Emulator for PCs * Copyright (C) 2002-2022 PCSX2 Dev Team * * PCSX2 is free software: you can redistribute it and/or modify it under the terms * of the GNU Lesser General Public License as published by the Free Software Found- * ation, either version 3 of the License, or (at your option) any later version. * * PCSX2 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 PCSX2. * If not, see . */ #include "PrecompiledHeader.h" #include #include #include #include "common/Assertions.h" #include "common/Console.h" #include "common/Exceptions.h" #include "common/SettingsWrapper.h" #include "common/StringUtil.h" #include "pcsx2/CDVD/CDVD.h" #include "pcsx2/Counters.h" #include "pcsx2/Frontend/InputManager.h" #include "pcsx2/Frontend/ImGuiManager.h" #include "pcsx2/GS.h" #include "pcsx2/GS/GS.h" #include "pcsx2/GSDumpReplayer.h" #include "pcsx2/HostDisplay.h" #include "pcsx2/HostSettings.h" #include "pcsx2/PAD/Host/PAD.h" #include "pcsx2/PerformanceMetrics.h" #include "pcsx2/VMManager.h" #include "DisplayWidget.h" #include "EmuThread.h" #include "MainWindow.h" #include "QtHost.h" #include "QtUtils.h" EmuThread* g_emu_thread = nullptr; static std::unique_ptr s_host_display; EmuThread::EmuThread(QThread* ui_thread) : QThread() , m_ui_thread(ui_thread) { } EmuThread::~EmuThread() = default; bool EmuThread::isOnEmuThread() const { return QThread::currentThread() == this; } void EmuThread::start() { pxAssertRel(!g_emu_thread, "Emu thread does not exist"); g_emu_thread = new EmuThread(QThread::currentThread()); g_emu_thread->QThread::start(); g_emu_thread->m_started_semaphore.acquire(); g_emu_thread->moveToThread(g_emu_thread); } void EmuThread::stop() { pxAssertRel(g_emu_thread, "Emu thread exists"); pxAssertRel(!g_emu_thread->isOnEmuThread(), "Not called on the emu thread"); QMetaObject::invokeMethod(g_emu_thread, &EmuThread::stopInThread, Qt::QueuedConnection); while (g_emu_thread->isRunning()) QApplication::processEvents(QEventLoop::ExcludeUserInputEvents, 1); } void EmuThread::stopInThread() { if (VMManager::HasValidVM()) destroyVM(); m_event_loop->quit(); m_shutdown_flag.store(true); } void EmuThread::startVM(std::shared_ptr boot_params) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "startVM", Qt::QueuedConnection, Q_ARG(std::shared_ptr, boot_params)); return; } pxAssertRel(!VMManager::HasValidVM(), "VM is shut down"); loadOurSettings(); emit onVMStarting(); // create the display, this may take a while... m_is_fullscreen = boot_params->fullscreen.value_or(Host::GetBaseBoolSettingValue("UI", "StartFullscreen", false)); m_is_rendering_to_main = shouldRenderToMain(); m_is_surfaceless = false; m_save_state_on_shutdown = false; if (!VMManager::Initialize(*boot_params)) return; if (!Host::GetBoolSettingValue("UI", "StartPaused", false)) { // This will come back and call OnVMResumed(). VMManager::SetState(VMState::Running); } else { // When starting paused, redraw the window, so there's at least something there. redrawDisplayWindow(); Host::OnVMPaused(); } m_event_loop->quit(); } void EmuThread::resetVM() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::resetVM, Qt::QueuedConnection); return; } VMManager::Reset(); } void EmuThread::setVMPaused(bool paused) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "setVMPaused", Qt::QueuedConnection, Q_ARG(bool, paused)); return; } VMManager::SetPaused(paused); } void EmuThread::shutdownVM(bool save_state /* = true */) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "shutdownVM", Qt::QueuedConnection, Q_ARG(bool, save_state)); return; } const VMState state = VMManager::GetState(); if (state == VMState::Paused) m_event_loop->quit(); else if (state != VMState::Running) return; m_save_state_on_shutdown = save_state; VMManager::SetState(VMState::Stopping); } void EmuThread::loadState(const QString& filename) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "loadState", Qt::QueuedConnection, Q_ARG(const QString&, filename)); return; } if (!VMManager::HasValidVM()) return; VMManager::LoadState(filename.toUtf8().constData()); } void EmuThread::loadStateFromSlot(qint32 slot) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "loadStateFromSlot", Qt::QueuedConnection, Q_ARG(qint32, slot)); return; } if (!VMManager::HasValidVM()) return; VMManager::LoadStateFromSlot(slot); } void EmuThread::saveState(const QString& filename) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "saveState", Qt::QueuedConnection, Q_ARG(const QString&, filename)); return; } if (!VMManager::HasValidVM()) return; if (!VMManager::SaveState(filename.toUtf8().constData())) { // this one is usually the result of a user-chosen path, so we can display a message box safely here Console.Error("Failed to save state"); } } void EmuThread::saveStateToSlot(qint32 slot) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "saveStateToSlot", Qt::QueuedConnection, Q_ARG(qint32, slot)); return; } if (!VMManager::HasValidVM()) return; VMManager::SaveStateToSlot(slot); } void EmuThread::run() { PerformanceMetrics::SetCPUThread(Threading::ThreadHandle::GetForCallingThread()); m_event_loop = new QEventLoop(); m_started_semaphore.release(); // neither of these should ever fail. if (!VMManager::Internal::InitializeGlobals() || !VMManager::Internal::InitializeMemory()) pxFailRel("Failed to allocate memory map"); // we need input sources ready for binding reloadInputSources(); createBackgroundControllerPollTimer(); startBackgroundControllerPollTimer(); connectSignals(); while (!m_shutdown_flag.load()) { if (!VMManager::HasValidVM()) { m_event_loop->exec(); continue; } executeVM(); } stopBackgroundControllerPollTimer(); destroyBackgroundControllerPollTimer(); InputManager::CloseSources(); VMManager::WaitForSaveStateFlush(); VMManager::Internal::ReleaseMemory(); VMManager::Internal::ReleaseGlobals(); PerformanceMetrics::SetCPUThread(Threading::ThreadHandle()); moveToThread(m_ui_thread); deleteLater(); } void EmuThread::destroyVM() { m_last_speed = 0.0f; m_last_game_fps = 0.0f; m_last_video_fps = 0.0f; m_last_internal_width = 0; m_last_internal_height = 0; m_was_paused_by_focus_loss = false; VMManager::Shutdown(m_save_state_on_shutdown); } void EmuThread::executeVM() { for (;;) { switch (VMManager::GetState()) { case VMState::Initializing: pxFailRel("Shouldn't be in the starting state state"); continue; case VMState::Paused: m_event_loop->exec(); continue; case VMState::Running: m_event_loop->processEvents(QEventLoop::AllEvents); VMManager::Execute(); continue; case VMState::Stopping: destroyVM(); m_event_loop->processEvents(QEventLoop::AllEvents); return; default: continue; } } } void EmuThread::createBackgroundControllerPollTimer() { pxAssert(!m_background_controller_polling_timer); m_background_controller_polling_timer = new QTimer(this); m_background_controller_polling_timer->setSingleShot(false); m_background_controller_polling_timer->setTimerType(Qt::CoarseTimer); connect(m_background_controller_polling_timer, &QTimer::timeout, this, &EmuThread::doBackgroundControllerPoll); } void EmuThread::destroyBackgroundControllerPollTimer() { delete m_background_controller_polling_timer; m_background_controller_polling_timer = nullptr; } void EmuThread::startBackgroundControllerPollTimer() { if (m_background_controller_polling_timer->isActive()) return; m_background_controller_polling_timer->start(BACKGROUND_CONTROLLER_POLLING_INTERVAL); } void EmuThread::stopBackgroundControllerPollTimer() { if (!m_background_controller_polling_timer->isActive()) return; m_background_controller_polling_timer->stop(); } void EmuThread::doBackgroundControllerPoll() { InputManager::PollSources(); } void EmuThread::toggleFullscreen() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::toggleFullscreen, Qt::QueuedConnection); return; } setFullscreen(!m_is_fullscreen); } void EmuThread::setFullscreen(bool fullscreen) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "setFullscreen", Qt::QueuedConnection, Q_ARG(bool, fullscreen)); return; } if (!VMManager::HasValidVM() || m_is_fullscreen == fullscreen) return; // This will call back to us on the MTGS thread. m_is_fullscreen = fullscreen; GetMTGS().UpdateDisplayWindow(); GetMTGS().WaitGS(); // If we're using exclusive fullscreen, the refresh rate may have changed. UpdateVSyncRate(); } void EmuThread::setSurfaceless(bool surfaceless) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "setSurfaceless", Qt::QueuedConnection, Q_ARG(bool, surfaceless)); return; } if (!VMManager::HasValidVM() || m_is_surfaceless == surfaceless) return; // This will call back to us on the MTGS thread. m_is_surfaceless = surfaceless; GetMTGS().UpdateDisplayWindow(); GetMTGS().WaitGS(); } void EmuThread::applySettings() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::applySettings, Qt::QueuedConnection); return; } checkForSettingChanges(); VMManager::ApplySettings(); } void EmuThread::reloadGameSettings() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::reloadGameSettings, Qt::QueuedConnection); return; } // this will skip applying settings when they're not active if (VMManager::ReloadGameSettings()) { // none of these settings below are per-game.. for now. but in case they are in the future. checkForSettingChanges(); } } void EmuThread::updateEmuFolders() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::updateEmuFolders, Qt::QueuedConnection); return; } Host::Internal::UpdateEmuFolders(); } void EmuThread::loadOurSettings() { m_verbose_status = Host::GetBaseBoolSettingValue("UI", "VerboseStatusBar", false); m_pause_on_focus_loss = Host::GetBaseBoolSettingValue("UI", "PauseOnFocusLoss", false); } void EmuThread::connectSignals() { connect(qApp, &QGuiApplication::applicationStateChanged, this, &EmuThread::onApplicationStateChanged); } void EmuThread::checkForSettingChanges() { QMetaObject::invokeMethod(g_main_window, &MainWindow::checkForSettingChanges, Qt::QueuedConnection); if (VMManager::HasValidVM()) { const bool render_to_main = shouldRenderToMain(); if (!m_is_fullscreen && m_is_rendering_to_main != render_to_main) { m_is_rendering_to_main = render_to_main; GetMTGS().UpdateDisplayWindow(); GetMTGS().WaitGS(); } } const bool last_verbose_status = m_verbose_status; loadOurSettings(); if (m_verbose_status != last_verbose_status) updatePerformanceMetrics(true); } bool EmuThread::shouldRenderToMain() const { return !Host::GetBaseBoolSettingValue("UI", "RenderToSeparateWindow", false) && !QtHost::InNoGUIMode(); } void EmuThread::toggleSoftwareRendering() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::toggleSoftwareRendering, Qt::QueuedConnection); return; } if (!VMManager::HasValidVM()) return; GetMTGS().ToggleSoftwareRendering(); } void EmuThread::switchRenderer(GSRendererType renderer) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "switchRenderer", Qt::QueuedConnection, Q_ARG(GSRendererType, renderer)); return; } if (!VMManager::HasValidVM()) return; GetMTGS().SwitchRenderer(renderer); } void EmuThread::changeDisc(CDVD_SourceType source, const QString& path) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "changeDisc", Qt::QueuedConnection, Q_ARG(CDVD_SourceType, source), Q_ARG(const QString&, path)); return; } if (!VMManager::HasValidVM()) return; VMManager::ChangeDisc(source, path.toStdString()); } void EmuThread::reloadPatches() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::reloadPatches, Qt::QueuedConnection); return; } if (!VMManager::HasValidVM()) return; VMManager::ReloadPatches(true, true); } void EmuThread::reloadInputSources() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::reloadInputSources, Qt::QueuedConnection); return; } std::unique_lock lock = Host::GetSettingsLock(); SettingsInterface* si = Host::GetSettingsInterface(); SettingsInterface* bindings_si = Host::GetSettingsInterfaceForBindings(); InputManager::ReloadSources(*si, lock); // skip loading bindings if we're not running, since it'll get done on startup anyway if (VMManager::HasValidVM()) InputManager::ReloadBindings(*si, *bindings_si); } void EmuThread::reloadInputBindings() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::reloadInputBindings, Qt::QueuedConnection); return; } // skip loading bindings if we're not running, since it'll get done on startup anyway if (!VMManager::HasValidVM()) return; auto lock = Host::GetSettingsLock(); SettingsInterface* si = Host::GetSettingsInterface(); SettingsInterface* bindings_si = Host::GetSettingsInterfaceForBindings(); InputManager::ReloadBindings(*si, *bindings_si); } void EmuThread::requestDisplaySize(float scale) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "requestDisplaySize", Qt::QueuedConnection, Q_ARG(float, scale)); return; } if (!VMManager::HasValidVM()) return; VMManager::RequestDisplaySize(scale); } void EmuThread::enumerateInputDevices() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::enumerateInputDevices, Qt::QueuedConnection); return; } const std::vector> devs(InputManager::EnumerateDevices()); QList> qdevs; qdevs.reserve(devs.size()); for (const std::pair& dev : devs) qdevs.emplace_back(QString::fromStdString(dev.first), QString::fromStdString(dev.second)); onInputDevicesEnumerated(qdevs); } void EmuThread::enumerateVibrationMotors() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::enumerateVibrationMotors, Qt::QueuedConnection); return; } const std::vector motors(InputManager::EnumerateMotors()); QList qmotors; qmotors.reserve(motors.size()); for (InputBindingKey key : motors) qmotors.push_back(key); onVibrationMotorsEnumerated(qmotors); } void EmuThread::connectDisplaySignals(DisplayWidget* widget) { widget->disconnect(this); connect(widget, &DisplayWidget::windowResizedEvent, this, &EmuThread::onDisplayWindowResized); connect(widget, &DisplayWidget::windowRestoredEvent, this, &EmuThread::redrawDisplayWindow); } void EmuThread::onDisplayWindowResized(int width, int height, float scale) { if (!s_host_display) return; GetMTGS().ResizeDisplayWindow(width, height, scale); } void EmuThread::onApplicationStateChanged(Qt::ApplicationState state) { // NOTE: This is executed on the emu thread, not UI thread. if (!m_pause_on_focus_loss || !VMManager::HasValidVM()) return; const bool focus_loss = (state != Qt::ApplicationActive); if (focus_loss) { if (!m_was_paused_by_focus_loss && VMManager::GetState() == VMState::Running) { m_was_paused_by_focus_loss = true; VMManager::SetPaused(true); } } else { if (m_was_paused_by_focus_loss) { m_was_paused_by_focus_loss = false; if (VMManager::GetState() == VMState::Paused) VMManager::SetPaused(false); } } } void EmuThread::redrawDisplayWindow() { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, &EmuThread::redrawDisplayWindow, Qt::QueuedConnection); return; } // If we're running, we're going to re-present anyway. if (!VMManager::HasValidVM() || VMManager::GetState() == VMState::Running) return; GetMTGS().RunOnGSThread([]() { GetMTGS().PresentCurrentFrame(); }); } void EmuThread::runOnCPUThread(const std::function& func) { func(); } void EmuThread::queueSnapshot(quint32 gsdump_frames) { if (!isOnEmuThread()) { QMetaObject::invokeMethod(this, "queueSnapshot", Qt::QueuedConnection, Q_ARG(quint32, gsdump_frames)); return; } if (!VMManager::HasValidVM()) return; GetMTGS().RunOnGSThread([gsdump_frames]() { GSQueueSnapshot(std::string(), gsdump_frames); }); } void EmuThread::updateDisplay() { pxAssertRel(!isOnEmuThread(), "Not on emu thread"); // finished with the display for now HostDisplay* display = Host::GetHostDisplay(); display->DoneRenderContextCurrent(); // but we should get it back after this call onUpdateDisplayRequested(m_is_fullscreen, !m_is_fullscreen && m_is_rendering_to_main, m_is_surfaceless); if (!display->MakeRenderContextCurrent()) { pxFailRel("Failed to recreate context after updating"); return; } } HostDisplay* EmuThread::acquireHostDisplay(HostDisplay::RenderAPI api) { s_host_display = HostDisplay::CreateDisplayForAPI(api); if (!s_host_display) return nullptr; DisplayWidget* widget = emit onCreateDisplayRequested(m_is_fullscreen, m_is_rendering_to_main); if (!widget) { s_host_display.reset(); return nullptr; } connectDisplaySignals(widget); if (!s_host_display->MakeRenderContextCurrent()) { Console.Error("Failed to make render context current"); releaseHostDisplay(); return nullptr; } if (!s_host_display->InitializeRenderDevice(EmuFolders::Cache, false) || !ImGuiManager::Initialize()) { Console.Error("Failed to initialize device/imgui"); releaseHostDisplay(); return nullptr; } Console.WriteLn(Color_StrongGreen, "%s Graphics Driver Info:", HostDisplay::RenderAPIToString(s_host_display->GetRenderAPI())); Console.Indent().WriteLn(s_host_display->GetDriverInfo()); return s_host_display.get(); } void EmuThread::releaseHostDisplay() { ImGuiManager::Shutdown(); s_host_display.reset(); emit onDestroyDisplayRequested(); } HostDisplay* Host::GetHostDisplay() { return s_host_display.get(); } HostDisplay* Host::AcquireHostDisplay(HostDisplay::RenderAPI api) { return g_emu_thread->acquireHostDisplay(api); } void Host::ReleaseHostDisplay() { g_emu_thread->releaseHostDisplay(); } bool Host::BeginPresentFrame(bool frame_skip) { if (!s_host_display->BeginPresent(frame_skip)) { // if we're skipping a frame, we need to reset imgui's state, since // we won't be calling EndPresentFrame(). ImGuiManager::NewFrame(); return false; } return true; } void Host::EndPresentFrame() { if (GSDumpReplayer::IsReplayingDump()) GSDumpReplayer::RenderUI(); ImGuiManager::RenderOSD(); s_host_display->EndPresent(); ImGuiManager::NewFrame(); } void Host::ResizeHostDisplay(u32 new_window_width, u32 new_window_height, float new_window_scale) { s_host_display->ResizeRenderWindow(new_window_width, new_window_height, new_window_scale); ImGuiManager::WindowResized(); // if we're paused, re-present the current frame at the new window size. if (VMManager::GetState() == VMState::Paused) GetMTGS().PresentCurrentFrame(); } void Host::RequestResizeHostDisplay(s32 width, s32 height) { g_emu_thread->onResizeDisplayRequested(width, height); } void Host::UpdateHostDisplay() { g_emu_thread->updateDisplay(); ImGuiManager::WindowResized(); // if we're paused, re-present the current frame at the new window size. if (VMManager::GetState() == VMState::Paused) GetMTGS().PresentCurrentFrame(); } void Host::OnVMStarting() { g_emu_thread->stopBackgroundControllerPollTimer(); emit g_emu_thread->onVMStarting(); } void Host::OnVMStarted() { emit g_emu_thread->onVMStarted(); } void Host::OnVMDestroyed() { emit g_emu_thread->onVMStopped(); g_emu_thread->startBackgroundControllerPollTimer(); } void Host::OnVMPaused() { g_emu_thread->startBackgroundControllerPollTimer(); emit g_emu_thread->onVMPaused(); } void Host::OnVMResumed() { // exit the event loop when we eventually return g_emu_thread->getEventLoop()->quit(); g_emu_thread->stopBackgroundControllerPollTimer(); // if we were surfaceless (view->game list, system->unpause), get our display widget back if (g_emu_thread->isSurfaceless()) g_emu_thread->setSurfaceless(false); emit g_emu_thread->onVMResumed(); } void Host::OnGameChanged(const std::string& disc_path, const std::string& game_serial, const std::string& game_name, u32 game_crc) { emit g_emu_thread->onGameChanged(QString::fromStdString(disc_path), QString::fromStdString(game_serial), QString::fromStdString(game_name), game_crc); } void EmuThread::updatePerformanceMetrics(bool force) { if (m_verbose_status && VMManager::HasValidVM()) { std::string gs_stat_str; GSgetTitleStats(gs_stat_str); QString gs_stat; if (THREAD_VU1) { gs_stat = QStringLiteral("%1 | EE: %2% | VU: %3% | GS: %4%") .arg(gs_stat_str.c_str()) .arg(PerformanceMetrics::GetCPUThreadUsage(), 0, 'f', 0) .arg(PerformanceMetrics::GetVUThreadUsage(), 0, 'f', 0) .arg(PerformanceMetrics::GetGSThreadUsage(), 0, 'f', 0); } else { gs_stat = QStringLiteral("%1 | EE: %2% | GS: %3%") .arg(gs_stat_str.c_str()) .arg(PerformanceMetrics::GetCPUThreadUsage(), 0, 'f', 0) .arg(PerformanceMetrics::GetGSThreadUsage(), 0, 'f', 0); } QMetaObject::invokeMethod(g_main_window->getStatusVerboseWidget(), "setText", Qt::QueuedConnection, Q_ARG(const QString&, gs_stat)); } const GSRendererType renderer = GSConfig.Renderer; const float speed = std::round(PerformanceMetrics::GetSpeed()); const float gfps = std::round(PerformanceMetrics::GetInternalFPS()); const float vfps = std::round(PerformanceMetrics::GetFPS()); int iwidth, iheight; GSgetInternalResolution(&iwidth, &iheight); if (iwidth != m_last_internal_width || iheight != m_last_internal_height || speed != m_last_speed || gfps != m_last_game_fps || vfps != m_last_video_fps || renderer != m_last_renderer || force) { if (iwidth == 0 && iheight == 0) { // if we don't have width/height yet, we're not going to have fps either. // and we'll probably be <100% due to compiling. so just leave it blank for now. QString blank; QMetaObject::invokeMethod(g_main_window->getStatusRendererWidget(), "setText", Qt::QueuedConnection, Q_ARG(const QString&, blank)); QMetaObject::invokeMethod(g_main_window->getStatusResolutionWidget(), "setText", Qt::QueuedConnection, Q_ARG(const QString&, blank)); QMetaObject::invokeMethod(g_main_window->getStatusFPSWidget(), "setText", Qt::QueuedConnection, Q_ARG(const QString&, blank)); QMetaObject::invokeMethod(g_main_window->getStatusVPSWidget(), "setText", Qt::QueuedConnection, Q_ARG(const QString&, blank)); return; } else { if (renderer != m_last_renderer || force) { QMetaObject::invokeMethod(g_main_window->getStatusRendererWidget(), "setText", Qt::QueuedConnection, Q_ARG(const QString&, QString::fromUtf8(Pcsx2Config::GSOptions::GetRendererName(renderer)))); m_last_renderer = renderer; } if (iwidth != m_last_internal_width || iheight != m_last_internal_height || force) { QMetaObject::invokeMethod(g_main_window->getStatusResolutionWidget(), "setText", Qt::QueuedConnection, Q_ARG(const QString&, tr("%1x%2") .arg(iwidth) .arg(iheight))); m_last_internal_width = iwidth; m_last_internal_height = iheight; } if (gfps != m_last_game_fps || force) { QMetaObject::invokeMethod(g_main_window->getStatusFPSWidget(), "setText", Qt::QueuedConnection, Q_ARG(const QString&, tr("Game: %1 FPS") .arg(gfps, 0, 'f', 0))); m_last_game_fps = gfps; } if (speed != m_last_speed || vfps != m_last_video_fps || force) { QMetaObject::invokeMethod(g_main_window->getStatusVPSWidget(), "setText", Qt::QueuedConnection, Q_ARG(const QString&, tr("Video: %1 FPS (%2%)") .arg(vfps, 0, 'f', 0) .arg(speed, 0, 'f', 0))); m_last_speed = speed; m_last_video_fps = vfps; } } } } void Host::OnPerformanceMetricsUpdated() { g_emu_thread->updatePerformanceMetrics(false); } void Host::OnSaveStateLoading(const std::string_view& filename) { emit g_emu_thread->onSaveStateLoading(QtUtils::StringViewToQString(filename)); } void Host::OnSaveStateLoaded(const std::string_view& filename, bool was_successful) { emit g_emu_thread->onSaveStateLoaded(QtUtils::StringViewToQString(filename), was_successful); } void Host::OnSaveStateSaved(const std::string_view& filename) { emit g_emu_thread->onSaveStateSaved(QtUtils::StringViewToQString(filename)); } void Host::PumpMessagesOnCPUThread() { g_emu_thread->getEventLoop()->processEvents(QEventLoop::AllEvents); } void Host::RunOnCPUThread(std::function function, bool block /* = false */) { if (g_emu_thread->isOnEmuThread()) { // probably shouldn't ever happen, but just in case.. function(); return; } QMetaObject::invokeMethod(g_emu_thread, "runOnCPUThread", block ? Qt::BlockingQueuedConnection : Qt::QueuedConnection, Q_ARG(const std::function&, std::move(function))); } void Host::RefreshGameListAsync(bool invalidate_cache) { QMetaObject::invokeMethod(g_main_window, "refreshGameList", Qt::QueuedConnection, Q_ARG(bool, invalidate_cache)); } void Host::CancelGameListRefresh() { QMetaObject::invokeMethod(g_main_window, "cancelGameListRefresh", Qt::BlockingQueuedConnection); } void Host::RequestExit(bool save_state_if_running) { if (VMManager::HasValidVM()) g_emu_thread->shutdownVM(save_state_if_running); QMetaObject::invokeMethod(g_main_window, "requestExit", Qt::QueuedConnection); } void Host::RequestVMShutdown(bool allow_confirm, bool allow_save_state) { if (!VMManager::HasValidVM()) return; // Run it on the host thread, that way we get the confirm prompt (if enabled). QMetaObject::invokeMethod(g_main_window, "requestShutdown", Qt::QueuedConnection, Q_ARG(bool, allow_confirm), Q_ARG(bool, allow_save_state), Q_ARG(bool, false)); } bool Host::IsFullscreen() { return g_emu_thread->isFullscreen(); } void Host::SetFullscreen(bool enabled) { g_emu_thread->setFullscreen(enabled); } alignas(16) static SysMtgsThread s_mtgs_thread; SysMtgsThread& GetMTGS() { return s_mtgs_thread; } // ------------------------------------------------------------------------ // Hotkeys // ------------------------------------------------------------------------ BEGIN_HOTKEY_LIST(g_host_hotkeys) END_HOTKEY_LIST()