#include "qthostinterface.h" #include "common/assert.h" #include "common/audio_stream.h" #include "common/byte_stream.h" #include "common/log.h" #include "common/string_util.h" #include "core/controller.h" #include "core/game_list.h" #include "core/gpu.h" #include "core/system.h" #include "qtsettingsinterface.h" #include "qtutils.h" #include #include #include #include #include #include #include Log_SetChannel(QtHostInterface); #ifdef WIN32 #include "d3d11displaywindow.h" #endif QtHostInterface::QtHostInterface(QObject* parent) : QObject(parent), HostInterface(), m_qsettings(QString::fromStdString(GetSettingsFileName()), QSettings::IniFormat) { checkSettings(); refreshGameList(); doUpdateInputMap(); createThread(); } QtHostInterface::~QtHostInterface() { Assert(!m_display_window); stopThread(); } void QtHostInterface::ReportError(const char* message) { HostInterface::ReportError(message); emit errorReported(QString::fromLocal8Bit(message)); } void QtHostInterface::ReportMessage(const char* message) { HostInterface::ReportMessage(message); emit messageReported(QString::fromLocal8Bit(message)); } void QtHostInterface::setDefaultSettings() { HostInterface::UpdateSettings([this]() { HostInterface::SetDefaultSettings(); }); // default input settings for Qt std::lock_guard guard(m_qsettings_mutex); m_qsettings.setValue(QStringLiteral("Controller1/ButtonUp"), QStringLiteral("Keyboard/W")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonDown"), QStringLiteral("Keyboard/S")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonLeft"), QStringLiteral("Keyboard/A")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonRight"), QStringLiteral("Keyboard/D")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonSelect"), QStringLiteral("Keyboard/Backspace")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonStart"), QStringLiteral("Keyboard/Return")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonTriangle"), QStringLiteral("Keyboard/8")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonCross"), QStringLiteral("Keyboard/2")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonSquare"), QStringLiteral("Keyboard/4")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonCircle"), QStringLiteral("Keyboard/6")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonL1"), QStringLiteral("Keyboard/Q")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonL2"), QStringLiteral("Keyboard/1")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonR1"), QStringLiteral("Keyboard/E")); m_qsettings.setValue(QStringLiteral("Controller1/ButtonR2"), QStringLiteral("Keyboard/3")); updateQSettingsFromCoreSettings(); } QVariant QtHostInterface::getSettingValue(const QString& name) { std::lock_guard guard(m_qsettings_mutex); return m_qsettings.value(name); } void QtHostInterface::putSettingValue(const QString& name, const QVariant& value) { std::lock_guard guard(m_qsettings_mutex); m_qsettings.setValue(name, value); } void QtHostInterface::removeSettingValue(const QString& name) { std::lock_guard guard(m_qsettings_mutex); m_qsettings.remove(name); } void QtHostInterface::updateQSettingsFromCoreSettings() { QtSettingsInterface si(m_qsettings); m_settings.Save(si); } void QtHostInterface::applySettings() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "applySettings", Qt::QueuedConnection); return; } UpdateSettings([this]() { std::lock_guard guard(m_qsettings_mutex); QtSettingsInterface si(m_qsettings); m_settings.Load(si); }); } void QtHostInterface::checkSettings() { const QSettings::Status settings_status = m_qsettings.status(); if (settings_status != QSettings::NoError) m_qsettings.clear(); const QString settings_version_key = QStringLiteral("General/SettingsVersion"); const int expected_version = 1; const QVariant settings_version_var = m_qsettings.value(settings_version_key); bool settings_version_okay; int settings_version = settings_version_var.toInt(&settings_version_okay); if (!settings_version_okay) settings_version = 0; if (settings_version != expected_version) { Log_WarningPrintf("Settings version %d does not match expected version %d, resetting", settings_version, expected_version); m_qsettings.clear(); m_qsettings.setValue(settings_version_key, expected_version); setDefaultSettings(); } // initial setting init - we don't do this locked since the thread hasn't been created yet QtSettingsInterface si(m_qsettings); m_settings.Load(si); } void QtHostInterface::refreshGameList(bool invalidate_cache /* = false */, bool invalidate_database /* = false */) { std::lock_guard lock(m_qsettings_mutex); QtSettingsInterface si(m_qsettings); m_game_list->SetSearchDirectoriesFromSettings(si); m_game_list->Refresh(invalidate_cache, invalidate_database); emit gameListRefreshed(); } QWidget* QtHostInterface::createDisplayWidget(QWidget* parent) { #ifdef WIN32 if (m_settings.gpu_renderer == GPURenderer::HardwareOpenGL) m_display_window = new OpenGLDisplayWindow(this, nullptr); else m_display_window = new D3D11DisplayWindow(this, nullptr); #else m_display_window = new OpenGLDisplayWindow(this, nullptr); #endif connect(m_display_window, &QtDisplayWindow::windowResizedEvent, this, &QtHostInterface::onDisplayWindowResized); m_display.release(); m_display = std::unique_ptr(m_display_window->getHostDisplayInterface()); m_display->SetDisplayLinearFiltering(m_settings.display_linear_filtering); QWidget* widget = QWidget::createWindowContainer(m_display_window, parent); widget->setFocusPolicy(Qt::StrongFocus); return widget; } bool QtHostInterface::createDisplayDeviceContext() { return m_display_window->createDeviceContext(m_worker_thread, m_settings.gpu_use_debug_device); } void QtHostInterface::displayWidgetDestroyed() { m_display.release(); m_display_window = nullptr; } void QtHostInterface::bootSystem(QString initial_filename, QString initial_save_state_filename) { Assert(!isOnWorkerThread()); emit emulationStarting(); if (!createDisplayDeviceContext()) { emit emulationStopped(); return; } QMetaObject::invokeMethod(this, "doBootSystem", Qt::QueuedConnection, Q_ARG(QString, initial_filename), Q_ARG(QString, initial_save_state_filename)); } void QtHostInterface::handleKeyEvent(int key, bool pressed) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "doHandleKeyEvent", Qt::QueuedConnection, Q_ARG(int, key), Q_ARG(bool, pressed)); return; } doHandleKeyEvent(key, pressed); } void QtHostInterface::doHandleKeyEvent(int key, bool pressed) { const auto iter = m_keyboard_input_handlers.find(key); if (iter == m_keyboard_input_handlers.end()) return; iter->second(pressed); } void QtHostInterface::onDisplayWindowResized(int width, int height) { m_display_window->onWindowResized(width, height); } void QtHostInterface::SwitchGPURenderer() { // Due to the GPU class owning textures, we have to shut the system down. std::unique_ptr stream; if (m_system) { stream = ByteStream_CreateGrowableMemoryStream(nullptr, 8 * 1024); if (!m_system->SaveState(stream.get()) || !stream->SeekAbsolute(0)) ReportError("Failed to save state before GPU renderer switch"); DestroySystem(); m_audio_stream->PauseOutput(true); m_display_window->destroyDeviceContext(); } const bool restore_state = static_cast(stream); emit recreateDisplayWidgetRequested(restore_state); Assert(m_display_window != nullptr); if (restore_state) { if (!m_display_window->initializeDeviceContext(m_settings.gpu_use_debug_device)) { emit runningGameChanged(QString(), QString(), QString()); emit emulationStopped(); return; } CreateSystem(); if (!BootSystem(nullptr, nullptr) || !m_system->LoadState(stream.get())) { ReportError("Failed to load state after GPU renderer switch, resetting"); m_system->Reset(); } if (!m_paused) { m_audio_stream->PauseOutput(false); UpdateSpeedLimiterState(); } m_system->ResetPerformanceCounters(); } } void QtHostInterface::OnSystemPerformanceCountersUpdated() { HostInterface::OnSystemPerformanceCountersUpdated(); DebugAssert(m_system); emit systemPerformanceCountersUpdated(m_system->GetEmulationSpeed(), m_system->GetFPS(), m_system->GetVPS(), m_system->GetAverageFrameTime(), m_system->GetWorstFrameTime()); } void QtHostInterface::OnRunningGameChanged() { HostInterface::OnRunningGameChanged(); if (m_system) { emit runningGameChanged(QString::fromStdString(m_system->GetRunningPath()), QString::fromStdString(m_system->GetRunningCode()), QString::fromStdString(m_system->GetRunningTitle())); } else { emit runningGameChanged(QString(), QString(), QString()); } } void QtHostInterface::updateInputMap() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "doUpdateInputMap", Qt::QueuedConnection); return; } doUpdateInputMap(); } void QtHostInterface::doUpdateInputMap() { m_keyboard_input_handlers.clear(); updateControllerInputMap(); updateHotkeyInputMap(); } void QtHostInterface::updateControllerInputMap() { for (u32 controller_index = 0; controller_index < 2; controller_index++) { const ControllerType ctype = m_settings.controller_types[controller_index]; if (ctype == ControllerType::None) continue; const auto button_names = Controller::GetButtonNames(ctype); for (const auto& it : button_names) { const std::string& button_name = it.first; const s32 button_code = it.second; QVariant var = m_qsettings.value( QStringLiteral("Controller%1/Button%2").arg(controller_index + 1).arg(QString::fromStdString(button_name))); if (!var.isValid()) continue; addButtonToInputMap(var.toString(), [this, controller_index, button_code](bool pressed) { if (!m_system) return; Controller* controller = m_system->GetController(controller_index); if (controller) controller->SetButtonState(button_code, pressed); }); } } } std::vector QtHostInterface::getHotkeyList() const { std::vector hotkeys = { {QStringLiteral("FastForward"), QStringLiteral("Toggle Fast Forward"), QStringLiteral("General")}, {QStringLiteral("Fullscreen"), QStringLiteral("Toggle Fullscreen"), QStringLiteral("General")}, {QStringLiteral("Pause"), QStringLiteral("Toggle Pause"), QStringLiteral("General")}, {QStringLiteral("ToggleSoftwareRendering"), QStringLiteral("Toggle Software Rendering"), QStringLiteral("Graphics")}, {QStringLiteral("IncreaseResolutionScale"), QStringLiteral("Increase Resolution Scale"), QStringLiteral("Graphics")}, {QStringLiteral("DecreaseResolutionScale"), QStringLiteral("Decrease Resolution Scale"), QStringLiteral("Graphics")}}; for (u32 global_i = 0; global_i < 2; global_i++) { const bool global = ConvertToBoolUnchecked(global_i); const u32 count = global ? GLOBAL_SAVE_STATE_SLOTS : PER_GAME_SAVE_STATE_SLOTS; for (u32 i = 1; i <= count; i++) { hotkeys.push_back({QStringLiteral("Load%1State%2").arg(global ? "Global" : "Game").arg(i), QStringLiteral("Load %1 State %2").arg(global ? tr("Global") : tr("Game")).arg(i), QStringLiteral("Save States")}); } for (u32 slot = 1; slot <= count; slot++) { hotkeys.push_back({QStringLiteral("Save%1State%2").arg(global ? "Global" : "Game").arg(slot), QStringLiteral("Save %1 State %2").arg(global ? tr("Global") : tr("Game")).arg(slot), QStringLiteral("Save States")}); } } return hotkeys; } void QtHostInterface::updateHotkeyInputMap() { auto hk = [this](const QString& hotkey_name, InputButtonHandler handler) { QVariant var = m_qsettings.value(QStringLiteral("Hotkeys/%1").arg(hotkey_name)); if (!var.isValid()) return; addButtonToInputMap(var.toString(), std::move(handler)); }; hk(QStringLiteral("FastForward"), [this](bool pressed) { m_speed_limiter_temp_disabled = pressed; HostInterface::UpdateSpeedLimiterState(); }); hk(QStringLiteral("Fullscreen"), [this](bool pressed) { if (!pressed) emit toggleFullscreenRequested(); }); hk(QStringLiteral("Pause"), [this](bool pressed) { if (!pressed) pauseSystem(!m_paused); }); hk(QStringLiteral("ToggleSoftwareRendering"), [this](bool pressed) { if (!pressed) ToggleSoftwareRendering(); }); hk(QStringLiteral("IncreaseResolutionScale"), [this](bool pressed) { if (!pressed) ModifyResolutionScale(1); }); hk(QStringLiteral("DecreaseResolutionScale"), [this](bool pressed) { if (!pressed) ModifyResolutionScale(-1); }); for (u32 global_i = 0; global_i < 2; global_i++) { const bool global = ConvertToBoolUnchecked(global_i); const u32 count = global ? GLOBAL_SAVE_STATE_SLOTS : PER_GAME_SAVE_STATE_SLOTS; for (u32 slot = 1; slot <= count; slot++) { hk(QStringLiteral("Load%1State%2").arg(global ? "Global" : "Game").arg(slot), [this, global, slot](bool pressed) { if (!pressed) loadState(global, slot); }); hk(QStringLiteral("Save%1State%2").arg(global ? "Global" : "Game").arg(slot), [this, global, slot](bool pressed) { if (!pressed) saveState(global, slot); }); } } } void QtHostInterface::addButtonToInputMap(const QString& binding, InputButtonHandler handler) { const QString device = binding.section('/', 0, 0); const QString button = binding.section('/', 1, 1); if (device == QStringLiteral("Keyboard")) { std::optional key_id = QtUtils::ParseKeyString(button); if (!key_id.has_value()) { qWarning() << "Unknown keyboard key " << button; return; } m_keyboard_input_handlers.emplace(key_id.value(), std::move(handler)); } else { qWarning() << "Unknown input device: " << device; return; } } void QtHostInterface::powerOffSystem(bool save_resume_state /* = false */, bool block_until_done /* = false */) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "powerOffSystem", block_until_done ? Qt::BlockingQueuedConnection : Qt::QueuedConnection, Q_ARG(bool, save_resume_state), Q_ARG(bool, block_until_done)); return; } if (!m_system) return; if (save_resume_state) Log_InfoPrintf("TODO: Save resume state"); DestroySystem(); m_audio_stream->PauseOutput(true); m_display_window->destroyDeviceContext(); emit runningGameChanged(QString(), QString(), QString()); emit emulationStopped(); } void QtHostInterface::resetSystem() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "resetSystem", Qt::QueuedConnection); return; } if (!m_system) { Log_ErrorPrintf("resetSystem() called without system"); return; } HostInterface::ResetSystem(); } void QtHostInterface::pauseSystem(bool paused) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "pauseSystem", Qt::QueuedConnection, Q_ARG(bool, paused)); return; } m_paused = paused; m_audio_stream->PauseOutput(paused); if (!paused) wakeThread(); emit emulationPaused(paused); } void QtHostInterface::changeDisc(QString new_disc_filename) {} void QtHostInterface::doBootSystem(QString initial_filename, QString initial_save_state_filename) { if (!m_display_window->initializeDeviceContext(m_settings.gpu_use_debug_device)) { emit emulationStopped(); return; } std::string initial_filename_str = initial_filename.toStdString(); std::string initial_save_state_filename_str = initial_save_state_filename.toStdString(); std::lock_guard lock(m_qsettings_mutex); if (!CreateSystem() || !BootSystem(initial_filename_str.empty() ? nullptr : initial_filename_str.c_str(), initial_save_state_filename_str.empty() ? nullptr : initial_save_state_filename_str.c_str())) { DestroySystem(); m_display_window->destroyDeviceContext(); emit emulationStopped(); return; } wakeThread(); m_audio_stream->PauseOutput(false); UpdateSpeedLimiterState(); emit emulationStarted(); } void QtHostInterface::createAudioStream() { switch (m_settings.audio_backend) { case AudioBackend::Default: case AudioBackend::Cubeb: m_audio_stream = AudioStream::CreateCubebAudioStream(); break; case AudioBackend::Null: default: m_audio_stream = AudioStream::CreateNullAudioStream(); break; } if (!m_audio_stream->Reconfigure(AUDIO_SAMPLE_RATE, AUDIO_CHANNELS, AUDIO_BUFFER_SIZE, 4)) { qWarning() << "Failed to configure audio stream, falling back to null output"; // fall back to null output m_audio_stream.reset(); m_audio_stream = AudioStream::CreateNullAudioStream(); m_audio_stream->Reconfigure(AUDIO_SAMPLE_RATE, AUDIO_CHANNELS, AUDIO_BUFFER_SIZE, 4); } } void QtHostInterface::populateSaveStateMenus(const char* game_code, QMenu* load_menu, QMenu* save_menu) { const std::vector available_states(GetAvailableSaveStates(game_code)); load_menu->clear(); if (!available_states.empty()) { bool last_global = available_states.front().global; for (const SaveStateInfo& ssi : available_states) { const s32 slot = ssi.slot; const bool global = ssi.global; const QDateTime timestamp(QDateTime::fromSecsSinceEpoch(static_cast(ssi.timestamp))); const QString path(QString::fromStdString(ssi.path)); QString title = tr("%1 Save %2 (%3)") .arg(global ? tr("Global") : tr("Game")) .arg(slot) .arg(timestamp.toString(Qt::SystemLocaleShortDate)); if (global != last_global) { load_menu->addSeparator(); last_global = global; } QAction* action = load_menu->addAction(title); connect(action, &QAction::triggered, [this, path]() { loadState(path); }); } } save_menu->clear(); if (game_code && std::strlen(game_code) > 0) { for (s32 i = 1; i <= PER_GAME_SAVE_STATE_SLOTS; i++) { QAction* action = save_menu->addAction(tr("Game Save %1").arg(i)); connect(action, &QAction::triggered, [this, i]() { saveState(i, false); }); } save_menu->addSeparator(); } for (s32 i = 1; i <= GLOBAL_SAVE_STATE_SLOTS; i++) { QAction* action = save_menu->addAction(tr("Global Save %1").arg(i)); connect(action, &QAction::triggered, [this, i]() { saveState(i, true); }); } } void QtHostInterface::loadState(QString filename) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "loadState", Qt::QueuedConnection, Q_ARG(QString, filename)); return; } if (m_system) LoadState(filename.toStdString().c_str()); else doBootSystem(QString(), filename); } void QtHostInterface::loadState(bool global, qint32 slot) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "loadState", Qt::QueuedConnection, Q_ARG(bool, global), Q_ARG(qint32, slot)); return; } if (m_system) { LoadState(slot, global); return; } if (!global) { // can't load a non-global system without a game code return; } loadState(QString::fromStdString(GetGlobalSaveStateFileName(slot))); } void QtHostInterface::saveState(bool global, qint32 slot, bool block_until_done /* = false */) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "saveState", block_until_done ? Qt::BlockingQueuedConnection : Qt::QueuedConnection, Q_ARG(bool, global), Q_ARG(qint32, slot), Q_ARG(bool, block_until_done)); return; } if (m_system) SaveState(global, slot); } void QtHostInterface::createThread() { m_original_thread = QThread::currentThread(); m_worker_thread = new Thread(this); m_worker_thread->start(); moveToThread(m_worker_thread); } void QtHostInterface::stopThread() { Assert(!isOnWorkerThread()); QMetaObject::invokeMethod(this, "doStopThread", Qt::QueuedConnection); m_worker_thread->wait(); } void QtHostInterface::doStopThread() { m_shutdown_flag.store(true); m_worker_thread_event_loop->quit(); } void QtHostInterface::threadEntryPoint() { m_worker_thread_event_loop = new QEventLoop(); createAudioStream(); // TODO: Event which flags the thread as ready while (!m_shutdown_flag.load()) { if (!m_system || m_paused) { // wait until we have a system before running m_worker_thread_event_loop->exec(); continue; } m_system->RunFrame(); m_system->GetGPU()->ResetGraphicsAPIState(); DrawDebugWindows(); DrawOSDMessages(); m_display->Render(); m_system->GetGPU()->RestoreGraphicsAPIState(); if (m_speed_limiter_enabled) m_system->Throttle(); m_worker_thread_event_loop->processEvents(QEventLoop::AllEvents); } m_system.reset(); m_audio_stream.reset(); delete m_worker_thread_event_loop; m_worker_thread_event_loop = nullptr; // move back to UI thread moveToThread(m_original_thread); } void QtHostInterface::wakeThread() { if (isOnWorkerThread()) m_worker_thread_event_loop->quit(); else QMetaObject::invokeMethod(m_worker_thread_event_loop, "quit", Qt::QueuedConnection); } QtHostInterface::Thread::Thread(QtHostInterface* parent) : QThread(parent), m_parent(parent) {} QtHostInterface::Thread::~Thread() = default; void QtHostInterface::Thread::run() { m_parent->threadEntryPoint(); }