// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin // SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0) #include "consolesettingswidget.h" #include "core/system.h" #include "qtutils.h" #include "settingsdialog.h" #include "settingwidgetbinder.h" #include "util/cd_image.h" #include #include ConsoleSettingsWidget::ConsoleSettingsWidget(SettingsDialog* dialog, QWidget* parent) : QWidget(parent), m_dialog(dialog) { SettingsInterface* sif = dialog->getSettingsInterface(); m_ui.setupUi(this); for (u32 i = 0; i < static_cast(ConsoleRegion::Count); i++) m_ui.region->addItem(QString::fromUtf8(Settings::GetConsoleRegionDisplayName(static_cast(i)))); for (u32 i = 0; i < static_cast(CPUExecutionMode::Count); i++) { m_ui.cpuExecutionMode->addItem( QString::fromUtf8(Settings::GetCPUExecutionModeDisplayName(static_cast(i)))); } static constexpr float TIME_PER_SECTOR_DOUBLE_SPEED = 1000.0f / 150.0f; m_ui.cdromReadaheadSectors->addItem(tr("Disabled (Synchronous)")); for (u32 i = 1; i <= 32; i++) { m_ui.cdromReadaheadSectors->addItem(tr("%1 sectors (%2 KB / %3 ms)") .arg(i) .arg(static_cast(i) * TIME_PER_SECTOR_DOUBLE_SPEED, 0, 'f', 0) .arg(static_cast(i * CDImage::DATA_SECTOR_SIZE) / 1024.0f)); } SettingWidgetBinder::BindWidgetToEnumSetting(sif, m_ui.region, "Console", "Region", &Settings::ParseConsoleRegionName, &Settings::GetConsoleRegionName, Settings::DEFAULT_CONSOLE_REGION); SettingWidgetBinder::BindWidgetToBoolSetting(sif, m_ui.enable8MBRAM, "Console", "Enable8MBRAM", false); SettingWidgetBinder::BindWidgetToEnumSetting(sif, m_ui.cpuExecutionMode, "CPU", "ExecutionMode", &Settings::ParseCPUExecutionMode, &Settings::GetCPUExecutionModeName, Settings::DEFAULT_CPU_EXECUTION_MODE); SettingWidgetBinder::BindWidgetToBoolSetting(sif, m_ui.enableCPUClockSpeedControl, "CPU", "OverclockEnable", false); SettingWidgetBinder::BindWidgetToBoolSetting(sif, m_ui.recompilerICache, "CPU", "RecompilerICache", false); SettingWidgetBinder::BindWidgetToIntSetting(sif, m_ui.cdromReadaheadSectors, "CDROM", "ReadaheadSectors", Settings::DEFAULT_CDROM_READAHEAD_SECTORS); SettingWidgetBinder::BindWidgetToBoolSetting(sif, m_ui.cdromRegionCheck, "CDROM", "RegionCheck", false); SettingWidgetBinder::BindWidgetToBoolSetting(sif, m_ui.cdromLoadImageToRAM, "CDROM", "LoadImageToRAM", false); SettingWidgetBinder::BindWidgetToBoolSetting(sif, m_ui.cdromLoadImagePatches, "CDROM", "LoadImagePatches", false); SettingWidgetBinder::BindWidgetToIntSetting(sif, m_ui.cdromSeekSpeedup, "CDROM", "SeekSpeedup", 1); SettingWidgetBinder::BindWidgetToIntSetting(sif, m_ui.cdromReadSpeedup, "CDROM", "ReadSpeedup", 1, 1); dialog->registerWidgetHelp(m_ui.region, tr("Region"), tr("Auto-Detect"), tr("Determines the emulated hardware type.")); dialog->registerWidgetHelp(m_ui.cpuExecutionMode, tr("Execution Mode"), tr("Recompiler (Fastest)"), tr("Determines how the emulated CPU executes instructions.")); dialog->registerWidgetHelp(m_ui.enableCPUClockSpeedControl, tr("Enable Clock Speed Control (Overclocking/Underclocking)"), tr("Unchecked"), tr("When this option is chosen, the clock speed set below will be used.")); dialog->registerWidgetHelp(m_ui.cpuClockSpeed, tr("Overclocking Percentage"), tr("100%"), tr("Selects the percentage of the normal clock speed the emulated hardware will run at.")); dialog->registerWidgetHelp(m_ui.recompilerICache, tr("Enable Recompiler ICache"), tr("Unchecked"), tr("Simulates stalls in the recompilers when the emulated CPU would have to fetch " "instructions into its cache. Makes games run closer to their console framerate, at a " "small cost to performance. Interpreter mode always simulates the instruction cache.")); dialog->registerWidgetHelp( m_ui.enable8MBRAM, tr("Enable 8MB RAM (Dev Console)"), tr("Unchecked"), tr("Enables an additional 6MB of RAM to obtain a total of 2+6 = 8MB, usually present on dev consoles. Games have " "to use a larger heap size for " "this additional RAM to be usable. Titles which rely on memory mirrors may break, so it should only be used " "with compatible mods.")); dialog->registerWidgetHelp( m_ui.cdromLoadImageToRAM, tr("Preload Image to RAM"), tr("Unchecked"), tr("Loads the game image into RAM. Useful for network paths that may become unreliable during gameplay. In some " "cases also eliminates stutter when games initiate audio track playback.")); dialog->registerWidgetHelp( m_ui.cdromReadSpeedup, tr("CD-ROM Read Speedup"), tr("None (Double Speed)"), tr("Speeds up CD-ROM reads by the specified factor. Only applies to double-speed reads, and is ignored when audio " "is playing. May improve loading speeds in some games, at the cost of breaking others.")); dialog->registerWidgetHelp( m_ui.cdromSeekSpeedup, tr("CD-ROM Seek Speedup"), tr("None (Normal Speed)"), tr("Reduces the simulated time for the CD-ROM sled to move to different areas of the disc. Can improve loading " "times, but crash games which do not expect the CD-ROM to operate faster.")); dialog->registerWidgetHelp(m_ui.cdromReadaheadSectors, tr("Asynchronous Readahead"), tr("8 Sectors"), tr("Reduces hitches in emulation by reading/decompressing CD data asynchronously on a " "worker thread. Higher sector numbers can reduce spikes when streaming FMVs or audio " "on slower storage or when using compression formats such as CHD.")); dialog->registerWidgetHelp(m_ui.cdromRegionCheck, tr("Enable Region Check"), tr("Checked"), tr("Simulates the region check present in original, unmodified consoles.")); dialog->registerWidgetHelp( m_ui.cdromLoadImageToRAM, tr("Preload Image to RAM"), tr("Unchecked"), tr("Loads the game image into RAM. Useful for network paths that may become unreliable during gameplay. In some " "cases also eliminates stutter when games initiate audio track playback.")); dialog->registerWidgetHelp(m_ui.cdromLoadImagePatches, tr("Apply Image Patches"), tr("Unchecked"), tr("Automatically applies patches to disc images when they are present in the same " "directory. Currently only PPF patches are supported with this option.")); m_ui.cpuClockSpeed->setEnabled(m_dialog->getEffectiveBoolValue("CPU", "OverclockEnable", false)); connect(m_ui.enableCPUClockSpeedControl, &QCheckBox::stateChanged, this, &ConsoleSettingsWidget::onEnableCPUClockSpeedControlChecked); connect(m_ui.cpuClockSpeed, &QSlider::valueChanged, this, &ConsoleSettingsWidget::onCPUClockSpeedValueChanged); calculateCPUClockValue(); } ConsoleSettingsWidget::~ConsoleSettingsWidget() = default; void ConsoleSettingsWidget::updateRecompilerICacheEnabled() { const CPUExecutionMode mode = Settings::ParseCPUExecutionMode( m_dialog ->getEffectiveStringValue("CPU", "ExecutionMode", Settings::GetCPUExecutionModeName(Settings::DEFAULT_CPU_EXECUTION_MODE)) .c_str()) .value_or(Settings::DEFAULT_CPU_EXECUTION_MODE); m_ui.recompilerICache->setEnabled(mode != CPUExecutionMode::Interpreter); } void ConsoleSettingsWidget::onEnableCPUClockSpeedControlChecked(int state) { if (state == Qt::Checked && (!m_dialog->isPerGameSettings() || !Host::GetBaseBoolSettingValue("CPU", "OverclockEnable", false)) && !Host::GetBaseBoolSettingValue("UI", "CPUOverclockingWarningShown", false)) { const QString message = tr("Enabling CPU overclocking will break games, cause bugs, reduce performance and can significantly increase " "system requirements.\n\nBy enabling this option you are agreeing to not create any bug reports unless you " "have confirmed the bug also occurs with overclocking disabled.\n\nThis warning will only be shown once."); QMessageBox mb(QMessageBox::Warning, tr("CPU Overclocking Warning"), message, QMessageBox::NoButton, this); const QAbstractButton* const yes_button = mb.addButton(tr("Yes, I will confirm bugs without overclocking before reporting."), QMessageBox::YesRole); mb.addButton(tr("No, take me back to safety."), QMessageBox::NoRole); mb.exec(); if (mb.clickedButton() != yes_button) { QSignalBlocker sb(m_ui.enableCPUClockSpeedControl); if (m_dialog->isPerGameSettings()) { m_ui.enableCPUClockSpeedControl->setCheckState(Qt::PartiallyChecked); m_dialog->removeSettingValue("CPU", "OverclockEnable"); } else { m_ui.enableCPUClockSpeedControl->setCheckState(Qt::Unchecked); m_dialog->setBoolSettingValue("CPU", "OverclockEnable", false); } return; } Host::SetBaseBoolSettingValue("UI", "CPUOverclockingWarningShown", true); Host::CommitBaseSettingChanges(); } m_ui.cpuClockSpeed->setEnabled(m_dialog->getEffectiveBoolValue("CPU", "OverclockEnable", false)); updateCPUClockSpeedLabel(); } void ConsoleSettingsWidget::onCPUClockSpeedValueChanged(int value) { const u32 percent = static_cast(m_ui.cpuClockSpeed->value()); u32 numerator, denominator; Settings::CPUOverclockPercentToFraction(percent, &numerator, &denominator); m_dialog->setIntSettingValue("CPU", "OverclockNumerator", static_cast(numerator)); m_dialog->setIntSettingValue("CPU", "OverclockDenominator", static_cast(denominator)); updateCPUClockSpeedLabel(); } void ConsoleSettingsWidget::updateCPUClockSpeedLabel() { const int percent = m_ui.enableCPUClockSpeedControl->isChecked() ? m_ui.cpuClockSpeed->value() : 100; const double frequency = (static_cast(System::MASTER_CLOCK) * static_cast(percent)) / 100.0; m_ui.cpuClockSpeedLabel->setText(tr("%1% (%2MHz)").arg(percent).arg(frequency / 1000000.0, 0, 'f', 2)); } void ConsoleSettingsWidget::calculateCPUClockValue() { const u32 numerator = static_cast(m_dialog->getEffectiveIntValue("CPU", "OverclockNumerator", 1)); const u32 denominator = static_cast(m_dialog->getEffectiveIntValue("CPU", "OverclockDenominator", 1)); const u32 percent = Settings::CPUOverclockFractionToPercent(numerator, denominator); QSignalBlocker sb(m_ui.cpuClockSpeed); m_ui.cpuClockSpeed->setValue(static_cast(percent)); updateCPUClockSpeedLabel(); }