2019-12-31 06:17:17 +00:00
|
|
|
#include "qthostinterface.h"
|
2020-01-10 03:31:12 +00:00
|
|
|
#include "common/assert.h"
|
2020-01-11 03:53:15 +00:00
|
|
|
#include "common/audio_stream.h"
|
2020-01-10 03:31:12 +00:00
|
|
|
#include "common/byte_stream.h"
|
|
|
|
#include "common/log.h"
|
|
|
|
#include "common/string_util.h"
|
2020-01-02 06:13:03 +00:00
|
|
|
#include "core/controller.h"
|
2019-12-31 06:17:17 +00:00
|
|
|
#include "core/game_list.h"
|
|
|
|
#include "core/gpu.h"
|
|
|
|
#include "core/system.h"
|
2020-02-25 13:40:46 +00:00
|
|
|
#include "frontend-common/sdl_audio_stream.h"
|
2020-02-15 15:14:53 +00:00
|
|
|
#include "frontend-common/sdl_controller_interface.h"
|
2019-12-31 06:17:17 +00:00
|
|
|
#include "qtsettingsinterface.h"
|
2020-01-02 06:13:03 +00:00
|
|
|
#include "qtutils.h"
|
2019-12-31 06:17:17 +00:00
|
|
|
#include <QtCore/QCoreApplication>
|
2020-02-15 15:14:04 +00:00
|
|
|
#include <QtCore/QDateTime>
|
2020-01-02 06:13:03 +00:00
|
|
|
#include <QtCore/QDebug>
|
2020-01-10 03:28:13 +00:00
|
|
|
#include <QtCore/QEventLoop>
|
2020-02-15 15:14:53 +00:00
|
|
|
#include <QtCore/QTimer>
|
2020-02-15 15:14:04 +00:00
|
|
|
#include <QtWidgets/QMenu>
|
2019-12-31 06:17:17 +00:00
|
|
|
#include <QtWidgets/QMessageBox>
|
|
|
|
#include <memory>
|
|
|
|
Log_SetChannel(QtHostInterface);
|
|
|
|
|
2020-01-07 05:54:37 +00:00
|
|
|
#ifdef WIN32
|
|
|
|
#include "d3d11displaywindow.h"
|
|
|
|
#endif
|
|
|
|
|
2019-12-31 06:17:17 +00:00
|
|
|
QtHostInterface::QtHostInterface(QObject* parent)
|
2020-02-28 07:00:09 +00:00
|
|
|
: QObject(parent), CommonHostInterface(),
|
|
|
|
m_qsettings(QString::fromStdString(GetSettingsFileName()), QSettings::IniFormat)
|
2019-12-31 06:17:17 +00:00
|
|
|
{
|
2020-02-28 07:00:09 +00:00
|
|
|
loadSettings();
|
2020-01-24 04:51:07 +00:00
|
|
|
refreshGameList();
|
2019-12-31 06:17:17 +00:00
|
|
|
createThread();
|
|
|
|
}
|
|
|
|
|
|
|
|
QtHostInterface::~QtHostInterface()
|
|
|
|
{
|
2020-01-02 07:45:25 +00:00
|
|
|
Assert(!m_display_window);
|
2019-12-31 06:17:17 +00:00
|
|
|
stopThread();
|
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::ReportError(const char* message)
|
|
|
|
{
|
2020-01-24 04:51:30 +00:00
|
|
|
HostInterface::ReportError(message);
|
|
|
|
|
|
|
|
emit errorReported(QString::fromLocal8Bit(message));
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::ReportMessage(const char* message)
|
|
|
|
{
|
2020-01-24 04:51:30 +00:00
|
|
|
HostInterface::ReportMessage(message);
|
|
|
|
|
|
|
|
emit messageReported(QString::fromLocal8Bit(message));
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
2020-02-26 09:25:57 +00:00
|
|
|
bool QtHostInterface::ConfirmMessage(const char* message)
|
|
|
|
{
|
|
|
|
return messageConfirmed(QString::fromLocal8Bit(message));
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:49 +00:00
|
|
|
QVariant QtHostInterface::getSettingValue(const QString& name, const QVariant& default_value)
|
2020-01-24 04:49:49 +00:00
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> guard(m_qsettings_mutex);
|
2020-02-15 15:14:49 +00:00
|
|
|
return m_qsettings.value(name, default_value);
|
2020-01-24 04:49:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::putSettingValue(const QString& name, const QVariant& value)
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> guard(m_qsettings_mutex);
|
|
|
|
m_qsettings.setValue(name, value);
|
2020-01-01 04:01:58 +00:00
|
|
|
}
|
|
|
|
|
2020-01-24 04:49:49 +00:00
|
|
|
void QtHostInterface::removeSettingValue(const QString& name)
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> guard(m_qsettings_mutex);
|
|
|
|
m_qsettings.remove(name);
|
|
|
|
}
|
|
|
|
|
2020-02-28 07:00:09 +00:00
|
|
|
void QtHostInterface::setDefaultSettings()
|
2020-01-01 04:01:58 +00:00
|
|
|
{
|
2020-02-28 07:00:09 +00:00
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(this, "setDefaultSettings", Qt::QueuedConnection);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::lock_guard<std::mutex> guard(m_qsettings_mutex);
|
2020-01-01 04:01:58 +00:00
|
|
|
QtSettingsInterface si(m_qsettings);
|
2020-02-28 07:00:09 +00:00
|
|
|
UpdateSettings([this, &si]() { m_settings.Load(si); });
|
|
|
|
UpdateInputMap(si);
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::applySettings()
|
|
|
|
{
|
2020-01-24 04:49:49 +00:00
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(this, "applySettings", Qt::QueuedConnection);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-28 07:00:09 +00:00
|
|
|
std::lock_guard<std::mutex> guard(m_qsettings_mutex);
|
|
|
|
QtSettingsInterface si(m_qsettings);
|
|
|
|
UpdateSettings([this, &si]() { m_settings.Load(si); });
|
|
|
|
UpdateInputMap(si);
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
2020-02-28 07:00:09 +00:00
|
|
|
void QtHostInterface::loadSettings()
|
2019-12-31 06:17:17 +00:00
|
|
|
{
|
2020-02-28 07:00:09 +00:00
|
|
|
// no need to lock here because the emu thread doesn't exist yet
|
|
|
|
QtSettingsInterface si(m_qsettings);
|
|
|
|
|
2019-12-31 06:17:17 +00:00
|
|
|
const QSettings::Status settings_status = m_qsettings.status();
|
|
|
|
if (settings_status != QSettings::NoError)
|
|
|
|
{
|
|
|
|
m_qsettings.clear();
|
2020-02-28 07:00:09 +00:00
|
|
|
SetDefaultSettings(si);
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
2020-01-01 04:01:58 +00:00
|
|
|
|
2020-02-28 07:00:09 +00:00
|
|
|
CheckSettings(si);
|
2020-01-24 04:49:49 +00:00
|
|
|
m_settings.Load(si);
|
2020-02-28 07:00:09 +00:00
|
|
|
|
|
|
|
// input map update is done on the emu thread
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
2020-01-08 03:37:43 +00:00
|
|
|
void QtHostInterface::refreshGameList(bool invalidate_cache /* = false */, bool invalidate_database /* = false */)
|
2019-12-31 06:17:17 +00:00
|
|
|
{
|
2020-01-24 04:50:46 +00:00
|
|
|
std::lock_guard<std::mutex> lock(m_qsettings_mutex);
|
2019-12-31 06:17:17 +00:00
|
|
|
QtSettingsInterface si(m_qsettings);
|
2020-01-24 04:51:09 +00:00
|
|
|
m_game_list->SetSearchDirectoriesFromSettings(si);
|
2020-01-08 03:37:43 +00:00
|
|
|
m_game_list->Refresh(invalidate_cache, invalidate_database);
|
2019-12-31 06:17:17 +00:00
|
|
|
emit gameListRefreshed();
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
QtDisplayWindow* QtHostInterface::createDisplayWindow()
|
2019-12-31 06:17:17 +00:00
|
|
|
{
|
2020-02-15 15:14:28 +00:00
|
|
|
Assert(!m_display_window);
|
|
|
|
|
2020-01-07 05:54:37 +00:00
|
|
|
#ifdef WIN32
|
2020-01-07 08:55:36 +00:00
|
|
|
if (m_settings.gpu_renderer == GPURenderer::HardwareOpenGL)
|
|
|
|
m_display_window = new OpenGLDisplayWindow(this, nullptr);
|
|
|
|
else
|
|
|
|
m_display_window = new D3D11DisplayWindow(this, nullptr);
|
2020-01-07 05:54:37 +00:00
|
|
|
#else
|
2020-01-02 07:45:25 +00:00
|
|
|
m_display_window = new OpenGLDisplayWindow(this, nullptr);
|
2020-01-07 05:54:37 +00:00
|
|
|
#endif
|
2020-01-02 09:14:16 +00:00
|
|
|
connect(m_display_window, &QtDisplayWindow::windowResizedEvent, this, &QtHostInterface::onDisplayWindowResized);
|
2020-02-15 15:14:28 +00:00
|
|
|
return m_display_window;
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:30 +00:00
|
|
|
void QtHostInterface::bootSystemFromFile(const QString& filename)
|
2020-01-24 04:49:51 +00:00
|
|
|
{
|
2020-02-15 15:14:28 +00:00
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
2020-02-15 15:14:30 +00:00
|
|
|
QMetaObject::invokeMethod(this, "bootSystemFromFile", Qt::QueuedConnection, Q_ARG(const QString&, filename));
|
2020-02-15 15:14:28 +00:00
|
|
|
return;
|
|
|
|
}
|
2020-01-24 04:49:51 +00:00
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
HostInterface::BootSystemFromFile(filename.toStdString().c_str());
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:49 +00:00
|
|
|
void QtHostInterface::resumeSystemFromState(const QString& filename, bool boot_on_failure)
|
|
|
|
{
|
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(this, "resumeSystemFromState", Qt::QueuedConnection, Q_ARG(const QString&, filename),
|
|
|
|
Q_ARG(bool, boot_on_failure));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (filename.isEmpty())
|
|
|
|
HostInterface::ResumeSystemFromMostRecentState();
|
|
|
|
else
|
|
|
|
HostInterface::ResumeSystemFromState(filename.toStdString().c_str(), boot_on_failure);
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
void QtHostInterface::bootSystemFromBIOS()
|
2019-12-31 06:17:17 +00:00
|
|
|
{
|
2020-02-15 15:14:28 +00:00
|
|
|
if (!isOnWorkerThread())
|
2019-12-31 06:17:17 +00:00
|
|
|
{
|
2020-02-15 15:14:28 +00:00
|
|
|
QMetaObject::invokeMethod(this, "bootSystemFromBIOS", Qt::QueuedConnection);
|
2019-12-31 06:17:17 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
HostInterface::BootSystemFromBIOS();
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
2020-01-02 06:13:03 +00:00
|
|
|
void QtHostInterface::handleKeyEvent(int key, bool pressed)
|
|
|
|
{
|
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
2020-02-28 07:00:09 +00:00
|
|
|
QMetaObject::invokeMethod(this, "handleKeyEvent", Qt::QueuedConnection, Q_ARG(int, key), Q_ARG(bool, pressed));
|
2020-01-02 06:13:03 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-28 07:00:09 +00:00
|
|
|
HandleHostKeyEvent(key, pressed);
|
2020-01-02 06:13:03 +00:00
|
|
|
}
|
|
|
|
|
2020-01-02 09:14:16 +00:00
|
|
|
void QtHostInterface::onDisplayWindowResized(int width, int height)
|
|
|
|
{
|
2020-02-15 15:14:56 +00:00
|
|
|
// this can be null if it was destroyed and the main thread is late catching up
|
|
|
|
if (m_display_window)
|
2020-02-28 07:01:01 +00:00
|
|
|
m_display_window->WindowResized(width, height);
|
2020-01-02 09:14:16 +00:00
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
bool QtHostInterface::AcquireHostDisplay()
|
2020-01-24 04:51:52 +00:00
|
|
|
{
|
2020-02-15 15:14:28 +00:00
|
|
|
DebugAssert(!m_display_window);
|
|
|
|
|
|
|
|
emit createDisplayWindowRequested(m_worker_thread, m_settings.gpu_use_debug_device);
|
|
|
|
if (!m_display_window->hasDeviceContext())
|
2020-01-24 04:51:52 +00:00
|
|
|
{
|
2020-02-15 15:14:28 +00:00
|
|
|
m_display_window = nullptr;
|
|
|
|
emit destroyDisplayWindowRequested();
|
|
|
|
return false;
|
|
|
|
}
|
2020-01-24 04:51:52 +00:00
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
if (!m_display_window->initializeDeviceContext(m_settings.gpu_use_debug_device))
|
|
|
|
{
|
2020-01-24 04:51:52 +00:00
|
|
|
m_display_window->destroyDeviceContext();
|
2020-02-15 15:14:28 +00:00
|
|
|
m_display_window = nullptr;
|
|
|
|
emit destroyDisplayWindowRequested();
|
|
|
|
return false;
|
2020-01-24 04:51:52 +00:00
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
m_display = m_display_window->getHostDisplayInterface();
|
|
|
|
return true;
|
|
|
|
}
|
2020-01-24 04:51:52 +00:00
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
void QtHostInterface::ReleaseHostDisplay()
|
|
|
|
{
|
|
|
|
DebugAssert(m_display_window && m_display == m_display_window->getHostDisplayInterface());
|
|
|
|
m_display = nullptr;
|
|
|
|
m_display_window->destroyDeviceContext();
|
|
|
|
m_display_window = nullptr;
|
|
|
|
emit destroyDisplayWindowRequested();
|
|
|
|
}
|
2020-01-24 04:51:52 +00:00
|
|
|
|
2020-02-28 07:00:09 +00:00
|
|
|
void QtHostInterface::SetFullscreen(bool enabled)
|
2020-02-15 15:14:28 +00:00
|
|
|
{
|
2020-02-28 07:00:09 +00:00
|
|
|
emit setFullscreenRequested(enabled);
|
|
|
|
}
|
2020-01-24 04:51:52 +00:00
|
|
|
|
2020-02-28 07:00:09 +00:00
|
|
|
void QtHostInterface::ToggleFullscreen()
|
|
|
|
{
|
|
|
|
emit toggleFullscreenRequested();
|
|
|
|
}
|
2020-02-15 15:14:35 +00:00
|
|
|
|
2020-02-28 07:00:09 +00:00
|
|
|
std::optional<CommonHostInterface::HostKeyCode> QtHostInterface::GetHostKeyCode(const std::string_view key_code) const
|
|
|
|
{
|
|
|
|
const std::optional<int> code =
|
|
|
|
QtUtils::ParseKeyString(QString::fromUtf8(key_code.data(), static_cast<int>(key_code.length())));
|
|
|
|
if (!code)
|
|
|
|
return std::nullopt;
|
2020-02-25 13:40:46 +00:00
|
|
|
|
2020-02-28 07:00:09 +00:00
|
|
|
return static_cast<s32>(*code);
|
2020-01-24 04:51:52 +00:00
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
void QtHostInterface::OnSystemCreated()
|
|
|
|
{
|
|
|
|
HostInterface::OnSystemCreated();
|
|
|
|
|
|
|
|
wakeThread();
|
2020-02-15 15:14:53 +00:00
|
|
|
destroyBackgroundControllerPollTimer();
|
2020-02-15 15:14:28 +00:00
|
|
|
|
|
|
|
emit emulationStarted();
|
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::OnSystemPaused(bool paused)
|
|
|
|
{
|
|
|
|
HostInterface::OnSystemPaused(paused);
|
|
|
|
|
2020-02-26 09:26:14 +00:00
|
|
|
emit emulationPaused(paused);
|
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
if (!paused)
|
2020-02-26 09:26:14 +00:00
|
|
|
{
|
2020-02-15 15:14:28 +00:00
|
|
|
wakeThread();
|
2020-02-26 09:26:14 +00:00
|
|
|
emit focusDisplayWidgetRequested();
|
|
|
|
}
|
2020-02-15 15:14:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::OnSystemDestroyed()
|
|
|
|
{
|
|
|
|
HostInterface::OnSystemDestroyed();
|
|
|
|
|
2020-02-15 15:14:53 +00:00
|
|
|
if (m_background_controller_polling_enable_count > 0)
|
|
|
|
createBackgroundControllerPollTimer();
|
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
emit emulationStopped();
|
|
|
|
}
|
|
|
|
|
2020-02-09 13:16:25 +00:00
|
|
|
void QtHostInterface::OnSystemPerformanceCountersUpdated()
|
2020-01-24 04:49:47 +00:00
|
|
|
{
|
2020-02-09 13:16:25 +00:00
|
|
|
HostInterface::OnSystemPerformanceCountersUpdated();
|
2020-01-24 04:49:47 +00:00
|
|
|
|
2020-02-09 13:16:25 +00:00
|
|
|
DebugAssert(m_system);
|
|
|
|
emit systemPerformanceCountersUpdated(m_system->GetEmulationSpeed(), m_system->GetFPS(), m_system->GetVPS(),
|
|
|
|
m_system->GetAverageFrameTime(), m_system->GetWorstFrameTime());
|
2020-01-24 04:49:47 +00:00
|
|
|
}
|
|
|
|
|
2020-01-24 04:51:05 +00:00
|
|
|
void QtHostInterface::OnRunningGameChanged()
|
2020-01-24 04:50:46 +00:00
|
|
|
{
|
2020-01-24 04:51:05 +00:00
|
|
|
HostInterface::OnRunningGameChanged();
|
2020-01-24 04:50:46 +00:00
|
|
|
|
2020-01-24 04:51:05 +00:00
|
|
|
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());
|
|
|
|
}
|
2020-01-24 04:50:46 +00:00
|
|
|
}
|
|
|
|
|
2020-02-28 07:00:09 +00:00
|
|
|
void QtHostInterface::OnSystemStateSaved(bool global, s32 slot)
|
|
|
|
{
|
|
|
|
emit stateSaved(QString::fromStdString(m_system->GetRunningCode()), global, slot);
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:44 +00:00
|
|
|
void QtHostInterface::OnControllerTypeChanged(u32 slot)
|
|
|
|
{
|
|
|
|
HostInterface::OnControllerTypeChanged(slot);
|
|
|
|
|
2020-03-02 01:05:12 +00:00
|
|
|
// this assumes the settings mutex is already locked - as it comes from updateSettings().
|
|
|
|
QtSettingsInterface si(m_qsettings);
|
|
|
|
UpdateInputMap(si);
|
2020-02-15 15:14:44 +00:00
|
|
|
}
|
|
|
|
|
2020-01-02 06:13:03 +00:00
|
|
|
void QtHostInterface::updateInputMap()
|
|
|
|
{
|
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
2020-02-28 07:00:09 +00:00
|
|
|
QMetaObject::invokeMethod(this, "updateInputMap", Qt::QueuedConnection);
|
2020-01-02 06:13:03 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:53 +00:00
|
|
|
std::lock_guard<std::mutex> lock(m_qsettings_mutex);
|
2020-02-28 07:00:09 +00:00
|
|
|
QtSettingsInterface si(m_qsettings);
|
|
|
|
UpdateInputMap(si);
|
2020-02-17 15:06:28 +00:00
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:49 +00:00
|
|
|
void QtHostInterface::powerOffSystem()
|
2019-12-31 06:17:17 +00:00
|
|
|
{
|
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
2020-02-15 15:14:49 +00:00
|
|
|
QMetaObject::invokeMethod(this, "powerOffSystem", Qt::QueuedConnection);
|
2019-12-31 06:17:17 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!m_system)
|
|
|
|
return;
|
|
|
|
|
2020-02-15 15:14:49 +00:00
|
|
|
if (m_settings.save_state_on_exit)
|
|
|
|
SaveResumeSaveState();
|
|
|
|
|
2020-01-24 04:50:54 +00:00
|
|
|
DestroySystem();
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:49 +00:00
|
|
|
void QtHostInterface::synchronousPowerOffSystem()
|
|
|
|
{
|
|
|
|
if (!isOnWorkerThread())
|
|
|
|
QMetaObject::invokeMethod(this, "powerOffSystem", Qt::BlockingQueuedConnection);
|
|
|
|
else
|
|
|
|
powerOffSystem();
|
|
|
|
}
|
|
|
|
|
2019-12-31 06:17:17 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2020-02-26 09:26:20 +00:00
|
|
|
if (!m_system)
|
|
|
|
return;
|
|
|
|
|
2019-12-31 06:17:17 +00:00
|
|
|
m_paused = paused;
|
2020-01-07 04:17:41 +00:00
|
|
|
m_audio_stream->PauseOutput(paused);
|
2020-01-10 03:28:13 +00:00
|
|
|
if (!paused)
|
|
|
|
wakeThread();
|
2019-12-31 06:17:17 +00:00
|
|
|
emit emulationPaused(paused);
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:33 +00:00
|
|
|
void QtHostInterface::changeDisc(const QString& new_disc_filename)
|
|
|
|
{
|
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(this, "changeDisc", Qt::QueuedConnection, Q_ARG(const QString&, new_disc_filename));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!m_system)
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_system->InsertMedia(new_disc_filename.toStdString().c_str());
|
|
|
|
}
|
2019-12-31 06:17:17 +00:00
|
|
|
|
2020-02-15 15:14:04 +00:00
|
|
|
void QtHostInterface::populateSaveStateMenus(const char* game_code, QMenu* load_menu, QMenu* save_menu)
|
|
|
|
{
|
|
|
|
const std::vector<SaveStateInfo> available_states(GetAvailableSaveStates(game_code));
|
|
|
|
|
|
|
|
load_menu->clear();
|
|
|
|
if (!available_states.empty())
|
|
|
|
{
|
|
|
|
bool last_global = available_states.front().global;
|
2020-02-15 15:14:49 +00:00
|
|
|
s32 last_slot = available_states.front().slot;
|
2020-02-15 15:14:04 +00:00
|
|
|
for (const SaveStateInfo& ssi : available_states)
|
|
|
|
{
|
|
|
|
const s32 slot = ssi.slot;
|
|
|
|
const bool global = ssi.global;
|
|
|
|
const QDateTime timestamp(QDateTime::fromSecsSinceEpoch(static_cast<qint64>(ssi.timestamp)));
|
2020-02-15 15:14:49 +00:00
|
|
|
const QString timestamp_str(timestamp.toString(Qt::SystemLocaleShortDate));
|
2020-02-15 15:14:04 +00:00
|
|
|
const QString path(QString::fromStdString(ssi.path));
|
|
|
|
|
2020-02-15 15:14:49 +00:00
|
|
|
QString title;
|
|
|
|
if (slot < 0)
|
|
|
|
title = tr("Resume Save (%1)").arg(timestamp_str);
|
|
|
|
else
|
|
|
|
title = tr("%1 Save %2 (%3)").arg(global ? tr("Global") : tr("Game")).arg(slot).arg(timestamp_str);
|
2020-02-15 15:14:04 +00:00
|
|
|
|
2020-02-15 15:14:49 +00:00
|
|
|
if (global != last_global || last_slot < 0)
|
2020-02-15 15:14:04 +00:00
|
|
|
load_menu->addSeparator();
|
2020-02-15 15:14:49 +00:00
|
|
|
|
|
|
|
last_global = global;
|
|
|
|
last_slot = slot;
|
2020-02-15 15:14:04 +00:00
|
|
|
|
|
|
|
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));
|
2020-02-15 15:15:18 +00:00
|
|
|
connect(action, &QAction::triggered, [this, i]() { saveState(false, i); });
|
2020-02-15 15:14:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
save_menu->addSeparator();
|
|
|
|
}
|
|
|
|
|
|
|
|
for (s32 i = 1; i <= GLOBAL_SAVE_STATE_SLOTS; i++)
|
|
|
|
{
|
|
|
|
QAction* action = save_menu->addAction(tr("Global Save %1").arg(i));
|
2020-02-15 15:15:18 +00:00
|
|
|
connect(action, &QAction::triggered, [this, i]() { saveState(true, i); });
|
2020-02-15 15:14:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:30 +00:00
|
|
|
void QtHostInterface::loadState(const QString& filename)
|
2020-02-15 15:14:04 +00:00
|
|
|
{
|
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
2020-02-15 15:14:30 +00:00
|
|
|
QMetaObject::invokeMethod(this, "loadState", Qt::QueuedConnection, Q_ARG(const QString&, filename));
|
2020-02-15 15:14:04 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
LoadState(filename.toStdString().c_str());
|
2020-02-15 15:14:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::loadState(bool global, qint32 slot)
|
|
|
|
{
|
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(this, "loadState", Qt::QueuedConnection, Q_ARG(bool, global), Q_ARG(qint32, slot));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:28 +00:00
|
|
|
LoadState(global, slot);
|
2020-02-15 15:14:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:14:53 +00:00
|
|
|
void QtHostInterface::enableBackgroundControllerPolling()
|
|
|
|
{
|
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(this, "enableBackgroundControllerPolling", Qt::BlockingQueuedConnection);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_background_controller_polling_enable_count++ > 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!m_system)
|
|
|
|
{
|
|
|
|
createBackgroundControllerPollTimer();
|
|
|
|
|
|
|
|
// drain the event queue so we don't get events late
|
|
|
|
g_sdl_controller_interface.PumpSDLEvents();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::disableBackgroundControllerPolling()
|
|
|
|
{
|
|
|
|
if (!isOnWorkerThread())
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(this, "disableBackgroundControllerPolling");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Assert(m_background_controller_polling_enable_count > 0);
|
|
|
|
if (--m_background_controller_polling_enable_count > 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!m_system)
|
|
|
|
destroyBackgroundControllerPollTimer();
|
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::doBackgroundControllerPoll()
|
|
|
|
{
|
|
|
|
g_sdl_controller_interface.PumpSDLEvents();
|
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::createBackgroundControllerPollTimer()
|
|
|
|
{
|
|
|
|
DebugAssert(!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::VeryCoarseTimer);
|
|
|
|
connect(m_background_controller_polling_timer, &QTimer::timeout, this, &QtHostInterface::doBackgroundControllerPoll);
|
|
|
|
m_background_controller_polling_timer->start(BACKGROUND_CONTROLLER_POLLING_INTERVAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::destroyBackgroundControllerPollTimer()
|
|
|
|
{
|
|
|
|
delete m_background_controller_polling_timer;
|
|
|
|
m_background_controller_polling_timer = nullptr;
|
|
|
|
}
|
|
|
|
|
2019-12-31 06:17:17 +00:00
|
|
|
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);
|
2020-01-11 04:21:08 +00:00
|
|
|
m_worker_thread_event_loop->quit();
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void QtHostInterface::threadEntryPoint()
|
|
|
|
{
|
2020-01-10 03:28:13 +00:00
|
|
|
m_worker_thread_event_loop = new QEventLoop();
|
2020-01-11 03:49:52 +00:00
|
|
|
|
2020-02-15 15:14:53 +00:00
|
|
|
// set up controller interface and immediate poll to pick up the controller attached events
|
2020-02-15 15:15:05 +00:00
|
|
|
g_sdl_controller_interface.Initialize(this);
|
2020-02-15 15:14:53 +00:00
|
|
|
g_sdl_controller_interface.PumpSDLEvents();
|
2020-02-28 07:00:09 +00:00
|
|
|
updateInputMap();
|
2020-02-15 15:14:53 +00:00
|
|
|
|
2020-01-11 03:49:52 +00:00
|
|
|
// TODO: Event which flags the thread as ready
|
2019-12-31 06:17:17 +00:00
|
|
|
while (!m_shutdown_flag.load())
|
|
|
|
{
|
2020-01-10 03:28:13 +00:00
|
|
|
if (!m_system || m_paused)
|
2019-12-31 06:17:17 +00:00
|
|
|
{
|
|
|
|
// wait until we have a system before running
|
2020-01-10 03:28:13 +00:00
|
|
|
m_worker_thread_event_loop->exec();
|
2019-12-31 06:17:17 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-02-09 13:16:25 +00:00
|
|
|
m_system->RunFrame();
|
2019-12-31 06:17:17 +00:00
|
|
|
|
2020-02-09 13:16:37 +00:00
|
|
|
m_system->GetGPU()->ResetGraphicsAPIState();
|
2019-12-31 06:17:17 +00:00
|
|
|
|
2020-02-09 13:16:37 +00:00
|
|
|
DrawDebugWindows();
|
|
|
|
DrawOSDMessages();
|
2019-12-31 06:17:17 +00:00
|
|
|
|
2020-02-09 13:16:37 +00:00
|
|
|
m_display->Render();
|
2019-12-31 06:17:17 +00:00
|
|
|
|
2020-02-09 13:16:37 +00:00
|
|
|
m_system->GetGPU()->RestoreGraphicsAPIState();
|
2019-12-31 06:17:17 +00:00
|
|
|
|
2020-02-09 13:16:37 +00:00
|
|
|
if (m_speed_limiter_enabled)
|
|
|
|
m_system->Throttle();
|
2019-12-31 06:17:17 +00:00
|
|
|
|
2020-01-10 03:28:13 +00:00
|
|
|
m_worker_thread_event_loop->processEvents(QEventLoop::AllEvents);
|
2020-02-15 15:14:53 +00:00
|
|
|
g_sdl_controller_interface.PumpSDLEvents();
|
2019-12-31 06:17:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
m_system.reset();
|
2020-01-11 03:49:52 +00:00
|
|
|
m_audio_stream.reset();
|
2020-02-15 15:14:53 +00:00
|
|
|
|
|
|
|
g_sdl_controller_interface.Shutdown();
|
|
|
|
|
2020-01-10 03:28:13 +00:00
|
|
|
delete m_worker_thread_event_loop;
|
|
|
|
m_worker_thread_event_loop = nullptr;
|
2019-12-31 06:17:17 +00:00
|
|
|
|
|
|
|
// move back to UI thread
|
|
|
|
moveToThread(m_original_thread);
|
|
|
|
}
|
|
|
|
|
2020-01-10 03:28:13 +00:00
|
|
|
void QtHostInterface::wakeThread()
|
|
|
|
{
|
|
|
|
if (isOnWorkerThread())
|
|
|
|
m_worker_thread_event_loop->quit();
|
|
|
|
else
|
|
|
|
QMetaObject::invokeMethod(m_worker_thread_event_loop, "quit", Qt::QueuedConnection);
|
|
|
|
}
|
|
|
|
|
2019-12-31 06:17:17 +00:00
|
|
|
QtHostInterface::Thread::Thread(QtHostInterface* parent) : QThread(parent), m_parent(parent) {}
|
|
|
|
|
|
|
|
QtHostInterface::Thread::~Thread() = default;
|
|
|
|
|
|
|
|
void QtHostInterface::Thread::run()
|
|
|
|
{
|
|
|
|
m_parent->threadEntryPoint();
|
|
|
|
}
|