Compare commits
8 Commits
7b3db145df
...
71700ffba4
Author | SHA1 | Date |
---|---|---|
SuperSamus | 71700ffba4 | |
JMC47 | 1ba8541da9 | |
Sam Belliveau | ac7d77d9e5 | |
mitaclaw | 3d0d03b871 | |
mitaclaw | 5f3a8ff0de | |
mitaclaw | be0b13da97 | |
mitaclaw | 4fde0f2868 | |
mitaclaw | 0352f24a8e |
|
@ -100,6 +100,12 @@ enum class BooleanSetting(
|
|||
"OverclockEnable",
|
||||
false
|
||||
),
|
||||
MAIN_VI_OVERCLOCK_ENABLE(
|
||||
Settings.FILE_DOLPHIN,
|
||||
Settings.SECTION_INI_CORE,
|
||||
"VIOverclockEnable",
|
||||
false
|
||||
),
|
||||
MAIN_RAM_OVERRIDE_ENABLE(
|
||||
Settings.FILE_DOLPHIN,
|
||||
Settings.SECTION_INI_CORE,
|
||||
|
|
|
@ -11,6 +11,7 @@ enum class FloatSetting(
|
|||
// These entries have the same names and order as in C++, just for consistency.
|
||||
MAIN_EMULATION_SPEED(Settings.FILE_DOLPHIN, Settings.SECTION_INI_CORE, "EmulationSpeed", 1.0f),
|
||||
MAIN_OVERCLOCK(Settings.FILE_DOLPHIN, Settings.SECTION_INI_CORE, "Overclock", 1.0f),
|
||||
MAIN_VI_OVERCLOCK(Settings.FILE_DOLPHIN, Settings.SECTION_INI_CORE, "VIOverclock", 1.0f),
|
||||
GFX_CC_GAME_GAMMA(Settings.FILE_GFX, Settings.SECTION_GFX_COLOR_CORRECTION, "GameGamma", 2.35f);
|
||||
|
||||
override val isOverridden: Boolean
|
||||
|
|
|
@ -1010,6 +1010,27 @@ class SettingsFragmentPresenter(
|
|||
false
|
||||
)
|
||||
)
|
||||
sl.add(
|
||||
SwitchSetting(
|
||||
context,
|
||||
BooleanSetting.MAIN_VI_OVERCLOCK_ENABLE,
|
||||
R.string.vi_overclock_enable,
|
||||
R.string.vi_overclock_enable_description
|
||||
)
|
||||
)
|
||||
sl.add(
|
||||
PercentSliderSetting(
|
||||
context,
|
||||
FloatSetting.MAIN_VI_OVERCLOCK,
|
||||
R.string.vi_overclock_title,
|
||||
R.string.vi_overclock_title_description,
|
||||
0f,
|
||||
400f,
|
||||
"%",
|
||||
1f,
|
||||
false
|
||||
)
|
||||
)
|
||||
|
||||
val mem1Size = ScaledIntSetting(1024 * 1024, IntSetting.MAIN_MEM1_SIZE)
|
||||
val mem2Size = ScaledIntSetting(1024 * 1024, IntSetting.MAIN_MEM2_SIZE)
|
||||
|
|
|
@ -370,7 +370,7 @@
|
|||
<string name="enable_cpu_cache_description">Enables emulation of the CPU write-back cache. Enabling will have a significant impact on performance. This should be left disabled unless absolutely needed.</string>
|
||||
<string name="clock_override">Clock Override</string>
|
||||
<string name="overclock_enable">Override Emulated CPU Clock Speed</string>
|
||||
<string name="overclock_enable_description">Higher values can make variable-framerate games run at a higher framerate, requiring a powerful device. Lower values make games run at a lower framerate, increasing emulation speed, but reducing the emulated console\'s performance.</string>
|
||||
<string name="overclock_enable_description">Higher values can make games run at a higher framerate if they lagged on original hardware, requiring a powerful device. Lower values improve performance by causing lag, but will slow down the game (without affecting audio) if its physics ignore the framerate.</string>
|
||||
<string name="overclock_title">Emulated CPU Clock Speed</string>
|
||||
<string name="overclock_title_description">Adjusts the emulated CPU\'s clock rate if \"Override Emulated CPU Clock Speed\" is enabled.</string>
|
||||
<string name="memory_override">Memory Override</string>
|
||||
|
@ -379,6 +379,10 @@
|
|||
<string name="main_mem1_size">MEM1 Size</string>
|
||||
<string name="main_mem2_size">MEM2 Size</string>
|
||||
<string name="gpu_options">GPU Options</string>
|
||||
<string name="vi_overclock_enable">Override VBI Clock Speed</string>
|
||||
<string name="vi_overclock_enable_description">Makes the game run at a different framerate without impacting audio, making the emulation less demanding when lowered, or improving smoothness when increased. Because many games\' physics ignore the framerate, this may affect game speed.</string>
|
||||
<string name="vi_overclock_title">Emulated VBI Clock Speed</string>
|
||||
<string name="vi_overclock_title_description">Adjusts the emulated VBI clock rate if \"Override VBI Clock Speed\" is enabled.\nAlso adjusts the emulated CPU\'s clock rate, to keep it relatively the same.</string>
|
||||
<string name="synchronize_gpu_thread">Synchronize GPU Thread</string>
|
||||
<string name="synchronize_gpu_thread_description">Synchronizing the GPU thread reduces the risk of games crashing or becoming unstable with dual core enabled, but can also reduce the performance gain of dual core. If unsure, select \"On Idle Skipping\". Selecting \"Never\" is risky and not recommended!</string>
|
||||
<string name="custom_rtc_options">Custom RTC Options</string>
|
||||
|
|
|
@ -62,7 +62,7 @@ bool IsPathAndroidContent(std::string_view uri)
|
|||
std::string OpenModeToAndroid(std::string mode)
|
||||
{
|
||||
// The 'b' specifier is not supported by Android. Since we're on POSIX, it's fine to just skip it.
|
||||
mode.erase(std::remove(mode.begin(), mode.end(), 'b'));
|
||||
std::erase(mode, 'b');
|
||||
|
||||
if (mode == "r")
|
||||
return "r";
|
||||
|
|
|
@ -98,7 +98,8 @@ std::vector<std::string> DoFileSearch(const std::vector<std::string>& directorie
|
|||
// not because std::filesystem returns duplicates). Also note that this pathname-based uniqueness
|
||||
// isn't as thorough as std::filesystem::equivalent.
|
||||
std::ranges::sort(result);
|
||||
result.erase(std::unique(result.begin(), result.end()), result.end());
|
||||
const auto unique_result = std::ranges::unique(result);
|
||||
result.erase(unique_result.begin(), unique_result.end());
|
||||
|
||||
// Dolphin expects to be able to use "/" (DIR_SEP) everywhere.
|
||||
// std::filesystem uses the OS separator.
|
||||
|
|
|
@ -208,6 +208,8 @@ const Info<bool> MAIN_DISABLE_ICACHE{{System::Main, "Core", "DisableICache"}, fa
|
|||
const Info<float> MAIN_EMULATION_SPEED{{System::Main, "Core", "EmulationSpeed"}, 1.0f};
|
||||
const Info<float> MAIN_OVERCLOCK{{System::Main, "Core", "Overclock"}, 1.0f};
|
||||
const Info<bool> MAIN_OVERCLOCK_ENABLE{{System::Main, "Core", "OverclockEnable"}, false};
|
||||
const Info<float> MAIN_VI_OVERCLOCK{{System::Main, "Core", "VIOverclock"}, 1.0f};
|
||||
const Info<bool> MAIN_VI_OVERCLOCK_ENABLE{{System::Main, "Core", "VIOverclockEnable"}, false};
|
||||
const Info<bool> MAIN_RAM_OVERRIDE_ENABLE{{System::Main, "Core", "RAMOverrideEnable"}, false};
|
||||
const Info<u32> MAIN_MEM1_SIZE{{System::Main, "Core", "MEM1Size"}, Memory::MEM1_SIZE_RETAIL};
|
||||
const Info<u32> MAIN_MEM2_SIZE{{System::Main, "Core", "MEM2Size"}, Memory::MEM2_SIZE_RETAIL};
|
||||
|
|
|
@ -125,6 +125,8 @@ extern const Info<bool> MAIN_DISABLE_ICACHE;
|
|||
extern const Info<float> MAIN_EMULATION_SPEED;
|
||||
extern const Info<float> MAIN_OVERCLOCK;
|
||||
extern const Info<bool> MAIN_OVERCLOCK_ENABLE;
|
||||
extern const Info<float> MAIN_VI_OVERCLOCK;
|
||||
extern const Info<bool> MAIN_VI_OVERCLOCK_ENABLE;
|
||||
extern const Info<bool> MAIN_RAM_OVERRIDE_ENABLE;
|
||||
extern const Info<u32> MAIN_MEM1_SIZE;
|
||||
extern const Info<u32> MAIN_MEM2_SIZE;
|
||||
|
|
|
@ -42,6 +42,8 @@ public:
|
|||
layer->Set(Config::MAIN_DSP_HLE, m_settings.dsp_hle);
|
||||
layer->Set(Config::MAIN_OVERCLOCK_ENABLE, m_settings.oc_enable);
|
||||
layer->Set(Config::MAIN_OVERCLOCK, m_settings.oc_factor);
|
||||
layer->Set(Config::MAIN_VI_OVERCLOCK_ENABLE, m_settings.vi_oc_enable);
|
||||
layer->Set(Config::MAIN_VI_OVERCLOCK, m_settings.vi_oc_factor);
|
||||
for (ExpansionInterface::Slot slot : ExpansionInterface::SLOTS)
|
||||
layer->Set(Config::GetInfoForEXIDevice(slot), m_settings.exi_device[slot]);
|
||||
layer->Set(Config::MAIN_MEMORY_CARD_SIZE, m_settings.memcard_size_override);
|
||||
|
|
|
@ -118,7 +118,9 @@ void CoreTimingManager::Shutdown()
|
|||
void CoreTimingManager::RefreshConfig()
|
||||
{
|
||||
m_config_oc_factor =
|
||||
Config::Get(Config::MAIN_OVERCLOCK_ENABLE) ? Config::Get(Config::MAIN_OVERCLOCK) : 1.0f;
|
||||
(Config::Get(Config::MAIN_OVERCLOCK_ENABLE) ? Config::Get(Config::MAIN_OVERCLOCK) : 1.0f) *
|
||||
(Config::Get(Config::MAIN_VI_OVERCLOCK_ENABLE) ? Config::Get(Config::MAIN_VI_OVERCLOCK) :
|
||||
1.0f);
|
||||
m_config_oc_inv_factor = 1.0f / m_config_oc_factor;
|
||||
m_config_sync_on_skip_idle = Config::Get(Config::MAIN_SYNC_ON_SKIP_IDLE);
|
||||
|
||||
|
@ -422,6 +424,11 @@ bool CoreTimingManager::GetVISkip() const
|
|||
return m_throttle_disable_vi_int && g_ActiveConfig.bVISkip && !Core::WantsDeterminism();
|
||||
}
|
||||
|
||||
float CoreTimingManager::GetOverclock() const
|
||||
{
|
||||
return m_config_oc_factor;
|
||||
}
|
||||
|
||||
bool CoreTimingManager::UseSyncOnSkipIdle() const
|
||||
{
|
||||
return m_config_sync_on_skip_idle;
|
||||
|
|
|
@ -162,6 +162,7 @@ public:
|
|||
|
||||
TimePoint GetCPUTimePoint(s64 cyclesLate) const; // Used by Dolphin Analytics
|
||||
bool GetVISkip() const; // Used By VideoInterface
|
||||
float GetOverclock() const;
|
||||
|
||||
bool UseSyncOnSkipIdle() const;
|
||||
|
||||
|
|
|
@ -365,6 +365,8 @@ void DolphinAnalytics::MakePerGameBuilder()
|
|||
builder.AddData("cfg-audio-backend", Config::Get(Config::MAIN_AUDIO_BACKEND));
|
||||
builder.AddData("cfg-oc-enable", Config::Get(Config::MAIN_OVERCLOCK_ENABLE));
|
||||
builder.AddData("cfg-oc-factor", Config::Get(Config::MAIN_OVERCLOCK));
|
||||
builder.AddData("cfg-vi-oc-enable", Config::Get(Config::MAIN_VI_OVERCLOCK_ENABLE));
|
||||
builder.AddData("cfg-vi-oc-factor", Config::Get(Config::MAIN_VI_OVERCLOCK));
|
||||
builder.AddData("cfg-render-to-main", Config::Get(Config::MAIN_RENDER_TO_MAIN));
|
||||
if (g_video_backend)
|
||||
{
|
||||
|
|
|
@ -140,7 +140,7 @@ void SystemTimersManager::VICallback(Core::System& system, u64 userdata, s64 cyc
|
|||
auto& core_timing = system.GetCoreTiming();
|
||||
auto& vi = system.GetVideoInterface();
|
||||
vi.Update(core_timing.GetTicks() - cycles_late);
|
||||
core_timing.ScheduleEvent(vi.GetTicksPerHalfLine() - cycles_late,
|
||||
core_timing.ScheduleEvent(vi.GetTicksPerViCallback() - cycles_late,
|
||||
system.GetSystemTimers().m_event_type_vi);
|
||||
}
|
||||
|
||||
|
@ -296,7 +296,7 @@ void SystemTimersManager::Init()
|
|||
|
||||
core_timing.ScheduleEvent(0, m_event_type_perf_tracker);
|
||||
core_timing.ScheduleEvent(0, m_event_type_gpu_sleeper);
|
||||
core_timing.ScheduleEvent(vi.GetTicksPerHalfLine(), m_event_type_vi);
|
||||
core_timing.ScheduleEvent(vi.GetTicksPerViCallback(), m_event_type_vi);
|
||||
core_timing.ScheduleEvent(0, m_event_type_dsp);
|
||||
|
||||
const int audio_dma_callback_period = GetAudioDMACallbackPeriod(
|
||||
|
|
|
@ -13,8 +13,10 @@
|
|||
#include "Common/Config/Config.h"
|
||||
#include "Common/Logging/Log.h"
|
||||
|
||||
#include "VideoCommon/OnScreenDisplay.h"
|
||||
#include "VideoCommon/PerformanceMetrics.h"
|
||||
|
||||
#include "Core/AchievementManager.h"
|
||||
#include "Core/Config/GraphicsSettings.h"
|
||||
#include "Core/Config/MainSettings.h"
|
||||
#include "Core/Config/SYSCONFSettings.h"
|
||||
|
@ -41,7 +43,10 @@ VideoInterfaceManager::VideoInterfaceManager(Core::System& system) : m_system(sy
|
|||
{
|
||||
}
|
||||
|
||||
VideoInterfaceManager::~VideoInterfaceManager() = default;
|
||||
VideoInterfaceManager::~VideoInterfaceManager()
|
||||
{
|
||||
Config::RemoveConfigChangedCallback(m_config_changed_callback_id);
|
||||
}
|
||||
|
||||
static constexpr std::array<u32, 2> CLOCK_FREQUENCIES{{
|
||||
27000000,
|
||||
|
@ -173,6 +178,21 @@ void VideoInterfaceManager::Preset(bool _bNTSC)
|
|||
void VideoInterfaceManager::Init()
|
||||
{
|
||||
Preset(true);
|
||||
|
||||
m_config_changed_callback_id = Config::AddConfigChangedCallback([this] { RefreshConfig(); });
|
||||
RefreshConfig();
|
||||
}
|
||||
|
||||
void VideoInterfaceManager::RefreshConfig()
|
||||
{
|
||||
m_config_vi_oc_factor =
|
||||
Config::Get(Config::MAIN_VI_OVERCLOCK_ENABLE) ? Config::Get(Config::MAIN_VI_OVERCLOCK) : 1.0f;
|
||||
if (AchievementManager::GetInstance().IsHardcoreModeActive() && m_config_vi_oc_factor < 1.0f)
|
||||
{
|
||||
Config::SetCurrent(Config::MAIN_VI_OVERCLOCK, 1.0f);
|
||||
m_config_vi_oc_factor = 1.0f;
|
||||
OSD::AddMessage("Minimum VBI overclock is 100% in Hardcore Mode");
|
||||
}
|
||||
}
|
||||
|
||||
void VideoInterfaceManager::RegisterMMIO(MMIO::Mapping* mmio, u32 base)
|
||||
|
@ -731,6 +751,11 @@ u32 VideoInterfaceManager::GetTicksPerField() const
|
|||
return GetTicksPerEvenField();
|
||||
}
|
||||
|
||||
u32 VideoInterfaceManager::GetTicksPerViCallback() const
|
||||
{
|
||||
return static_cast<u32>(GetTicksPerHalfLine() / m_config_vi_oc_factor);
|
||||
}
|
||||
|
||||
void VideoInterfaceManager::LogField(FieldType field, u32 xfb_address) const
|
||||
{
|
||||
static constexpr std::array<const char*, 2> field_type_names{{"Odd", "Even"}};
|
||||
|
|
|
@ -7,6 +7,7 @@
|
|||
#include <memory>
|
||||
|
||||
#include "Common/CommonTypes.h"
|
||||
#include "Common/Config/Config.h"
|
||||
|
||||
enum class FieldType;
|
||||
class PointerWrap;
|
||||
|
@ -387,6 +388,7 @@ public:
|
|||
u32 GetTicksPerSample() const;
|
||||
u32 GetTicksPerHalfLine() const;
|
||||
u32 GetTicksPerField() const;
|
||||
u32 GetTicksPerViCallback() const;
|
||||
|
||||
// Get the aspect ratio of VI's active area (rarely matching pure 4:3).
|
||||
// This function only deals with standard aspect ratios. For widescreen aspect ratios, multiply
|
||||
|
@ -407,6 +409,8 @@ private:
|
|||
void BeginField(FieldType field, u64 ticks);
|
||||
void EndField(FieldType field, u64 ticks);
|
||||
|
||||
void RefreshConfig();
|
||||
|
||||
// Registers listed in order:
|
||||
UVIVerticalTimingRegister m_vertical_timing_register;
|
||||
UVIDisplayControlRegister m_display_control_register;
|
||||
|
@ -447,6 +451,9 @@ private:
|
|||
u32 m_even_field_last_hl = 0; // index last halfline of the even field
|
||||
u32 m_odd_field_last_hl = 0; // index last halfline of the odd field
|
||||
|
||||
float m_config_vi_oc_factor = 0.0f;
|
||||
|
||||
Config::ConfigChangedCallbackID m_config_changed_callback_id;
|
||||
Core::System& m_system;
|
||||
};
|
||||
} // namespace VideoInterface
|
||||
|
|
|
@ -68,9 +68,7 @@ IPCReply GetCPUSpeed(Core::System& system, const IOCtlVRequest& request)
|
|||
return IPCReply(IPC_EINVAL);
|
||||
}
|
||||
|
||||
const bool overclock_enabled = Config::Get(Config::MAIN_OVERCLOCK_ENABLE);
|
||||
const float oc = overclock_enabled ? Config::Get(Config::MAIN_OVERCLOCK) : 1.0f;
|
||||
|
||||
const bool oc = system.GetCoreTiming().GetOverclock();
|
||||
const u32 core_clock = u32(float(system.GetSystemTimers().GetTicksPerSecond()) * oc);
|
||||
|
||||
auto& memory = system.GetMemory();
|
||||
|
|
|
@ -1085,10 +1085,10 @@ void WiiSockMan::UpdatePollCommands()
|
|||
std::vector<int> original_order(pfds.size());
|
||||
std::iota(original_order.begin(), original_order.end(), 0);
|
||||
// Select indices with valid fds
|
||||
auto mid = std::partition(original_order.begin(), original_order.end(), [&](auto i) {
|
||||
const auto partition_result = std::ranges::partition(original_order, [&](auto i) {
|
||||
return GetHostSocket(memory.Read_U32(pcmd.buffer_out + 0xc * i)) >= 0;
|
||||
});
|
||||
const auto n_valid = std::distance(original_order.begin(), mid);
|
||||
const auto n_valid = std::distance(original_order.begin(), partition_result.begin());
|
||||
|
||||
// Move all the valid pollfds to the front of the vector
|
||||
for (auto i = 0; i < n_valid; ++i)
|
||||
|
|
|
@ -1081,11 +1081,11 @@ void MovieManager::LoadInput(const std::string& movie_path)
|
|||
std::vector<u8> movInput(m_current_byte);
|
||||
t_record.ReadArray(movInput.data(), movInput.size());
|
||||
|
||||
const auto result = std::mismatch(movInput.begin(), movInput.end(), m_temp_input.begin());
|
||||
const auto mismatch_result = std::ranges::mismatch(movInput, m_temp_input);
|
||||
|
||||
if (result.first != movInput.end())
|
||||
if (mismatch_result.in1 != movInput.end())
|
||||
{
|
||||
const ptrdiff_t mismatch_index = std::distance(movInput.begin(), result.first);
|
||||
const ptrdiff_t mismatch_index = std::distance(movInput.begin(), mismatch_result.in1);
|
||||
|
||||
// this is a "you did something wrong" alert for the user's benefit.
|
||||
// we'll try to say what's going on in excruciating detail, otherwise the user might not
|
||||
|
|
|
@ -47,6 +47,8 @@ struct NetSettings
|
|||
bool allow_sd_writes = false;
|
||||
bool oc_enable = false;
|
||||
float oc_factor = 0;
|
||||
bool vi_oc_enable = false;
|
||||
float vi_oc_factor = 0;
|
||||
Common::EnumMap<ExpansionInterface::EXIDeviceType, ExpansionInterface::MAX_SLOT> exi_device{};
|
||||
int memcard_size_override = -1;
|
||||
|
||||
|
|
|
@ -1371,6 +1371,8 @@ bool NetPlayServer::SetupNetSettings()
|
|||
settings.allow_sd_writes = Config::Get(Config::MAIN_ALLOW_SD_WRITES);
|
||||
settings.oc_enable = Config::Get(Config::MAIN_OVERCLOCK_ENABLE);
|
||||
settings.oc_factor = Config::Get(Config::MAIN_OVERCLOCK);
|
||||
settings.vi_oc_enable = Config::Get(Config::MAIN_VI_OVERCLOCK_ENABLE);
|
||||
settings.vi_oc_factor = Config::Get(Config::MAIN_VI_OVERCLOCK);
|
||||
|
||||
for (ExpansionInterface::Slot slot : ExpansionInterface::SLOTS)
|
||||
{
|
||||
|
|
|
@ -240,11 +240,9 @@ bool NANDImporter::ExtractCertificates()
|
|||
|
||||
for (const PEMCertificate& certificate : certificates)
|
||||
{
|
||||
const auto search_result =
|
||||
std::search(content_bytes.begin(), content_bytes.end(), certificate.search_bytes.begin(),
|
||||
certificate.search_bytes.end());
|
||||
const auto search_result = std::ranges::search(content_bytes, certificate.search_bytes);
|
||||
|
||||
if (search_result == content_bytes.end())
|
||||
if (search_result.empty())
|
||||
{
|
||||
ERROR_LOG_FMT(DISCIO, "ExtractCertificates: Could not find offset for certficate '{}'",
|
||||
certificate.filename);
|
||||
|
@ -252,7 +250,8 @@ bool NANDImporter::ExtractCertificates()
|
|||
}
|
||||
|
||||
const std::string pem_file_path = m_nand_root + std::string(certificate.filename);
|
||||
const ptrdiff_t certificate_offset = std::distance(content_bytes.begin(), search_result);
|
||||
const ptrdiff_t certificate_offset =
|
||||
std::distance(content_bytes.begin(), search_result.begin());
|
||||
constexpr int min_offset = 2;
|
||||
if (certificate_offset < min_offset)
|
||||
{
|
||||
|
|
|
@ -309,11 +309,9 @@ void Settings::RemovePath(const QString& qpath)
|
|||
std::string path = qpath.toStdString();
|
||||
std::vector<std::string> paths = Config::GetIsoPaths();
|
||||
|
||||
auto new_end = std::remove(paths.begin(), paths.end(), path);
|
||||
if (new_end == paths.end())
|
||||
if (std::erase(paths, path) == 0)
|
||||
return;
|
||||
|
||||
paths.erase(new_end, paths.end());
|
||||
Config::SetIsoPaths(paths);
|
||||
emit PathRemoved(qpath);
|
||||
}
|
||||
|
|
|
@ -20,6 +20,7 @@
|
|||
#include "Core/ConfigManager.h"
|
||||
#include "Core/Core.h"
|
||||
#include "Core/HW/SystemTimers.h"
|
||||
#include "Core/HW/VideoInterface.h"
|
||||
#include "Core/PowerPC/PowerPC.h"
|
||||
#include "Core/System.h"
|
||||
|
||||
|
@ -107,17 +108,49 @@ void AdvancedPane::CreateLayout()
|
|||
m_cpu_clock_override_slider_label = new QLabel();
|
||||
cpu_clock_override_slider_layout->addWidget(m_cpu_clock_override_slider_label);
|
||||
|
||||
auto* cpu_clock_override_description =
|
||||
new QLabel(tr("Adjusts the emulated CPU's clock rate.\n\n"
|
||||
"Higher values may make variable-framerate games run at a higher framerate, "
|
||||
"at the expense of performance. Lower values may activate a game's "
|
||||
"internal frameskip, potentially improving performance.\n\n"
|
||||
"WARNING: Changing this from the default (100%) can and will "
|
||||
"break games and cause glitches. Do so at your own risk. "
|
||||
"Please do not report bugs that occur with a non-default clock."));
|
||||
auto* cpu_clock_override_description = new QLabel(tr(
|
||||
"Adjusts the emulated CPU's clock rate.\n\n"
|
||||
"Higher values can make games run at a higher framerate if they lagged on original hardware, "
|
||||
"at the expense of performance. Lower values improve performance by causing lag, "
|
||||
"but will slow down the game (without affecting audio) if its physics ignore the "
|
||||
"framerate.\n\n"
|
||||
"WARNING: Changing this from the default (100%) can and will "
|
||||
"break games and cause glitches. Do so at your own risk. "
|
||||
"Please do not report bugs that occur with a non-default clock."));
|
||||
cpu_clock_override_description->setWordWrap(true);
|
||||
clock_override_layout->addWidget(cpu_clock_override_description);
|
||||
|
||||
auto* vi_clock_override = new QGroupBox(tr("VBI Clock Override"));
|
||||
auto* vi_clock_override_layout = new QVBoxLayout();
|
||||
vi_clock_override->setLayout(vi_clock_override_layout);
|
||||
main_layout->addWidget(vi_clock_override);
|
||||
|
||||
m_vi_clock_override_checkbox = new QCheckBox(tr("Enable VBI Overclocking"));
|
||||
vi_clock_override_layout->addWidget(m_vi_clock_override_checkbox);
|
||||
|
||||
auto* vi_clock_override_slider_layout = new QHBoxLayout();
|
||||
vi_clock_override_slider_layout->setContentsMargins(0, 0, 0, 0);
|
||||
vi_clock_override_layout->addLayout(vi_clock_override_slider_layout);
|
||||
|
||||
m_vi_clock_override_slider = new QSlider(Qt::Horizontal);
|
||||
m_vi_clock_override_slider->setRange(1, 400);
|
||||
vi_clock_override_slider_layout->addWidget(m_vi_clock_override_slider);
|
||||
|
||||
m_vi_clock_override_slider_label = new QLabel();
|
||||
vi_clock_override_slider_layout->addWidget(m_vi_clock_override_slider_label);
|
||||
|
||||
auto* vi_clock_override_description = new QLabel(
|
||||
tr("Adjusts the emulated VBI clock rate. Also adjusts the emulated CPU's clock rate, to "
|
||||
"keep it relatively the same.\n\n"
|
||||
"Makes the game run at a different framerate without impacting audio, making the "
|
||||
"emulation less demanding when lowered, or improving smoothness when increased. Because "
|
||||
"many games' physics ignore the framerate, this may affect game speed.\n\n"
|
||||
"WARNING: Changing this from the default (100%) can and will "
|
||||
"break games and cause glitches. Do so at your own risk. "
|
||||
"Please do not report bugs that occur with a non-default clock."));
|
||||
vi_clock_override_description->setWordWrap(true);
|
||||
vi_clock_override_layout->addWidget(vi_clock_override_description);
|
||||
|
||||
auto* ram_override = new QGroupBox(tr("Memory Override"));
|
||||
auto* ram_override_layout = new QVBoxLayout();
|
||||
ram_override->setLayout(ram_override_layout);
|
||||
|
@ -203,6 +236,19 @@ void AdvancedPane::ConnectLayout()
|
|||
Update();
|
||||
});
|
||||
|
||||
m_vi_clock_override_checkbox->setChecked(Config::Get(Config::MAIN_VI_OVERCLOCK_ENABLE));
|
||||
connect(m_vi_clock_override_checkbox, &QCheckBox::toggled, [this](bool enable_clock_override) {
|
||||
Config::SetBaseOrCurrent(Config::MAIN_VI_OVERCLOCK_ENABLE, enable_clock_override);
|
||||
Update();
|
||||
});
|
||||
|
||||
connect(m_vi_clock_override_slider, &QSlider::valueChanged, [this](int oc_factor) {
|
||||
const float factor = m_vi_clock_override_slider->value() / 100.f;
|
||||
Config::SetBaseOrCurrent(Config::MAIN_VI_OVERCLOCK, factor);
|
||||
Update();
|
||||
});
|
||||
|
||||
m_ram_override_checkbox->setChecked(Config::Get(Config::MAIN_RAM_OVERRIDE_ENABLE));
|
||||
connect(m_ram_override_checkbox, &QCheckBox::toggled, [this](bool enable_ram_override) {
|
||||
Config::SetBaseOrCurrent(Config::MAIN_RAM_OVERRIDE_ENABLE, enable_ram_override);
|
||||
Update();
|
||||
|
@ -236,6 +282,7 @@ void AdvancedPane::Update()
|
|||
{
|
||||
const bool is_uninitialized = Core::IsUninitialized(Core::System::GetInstance());
|
||||
const bool enable_cpu_clock_override_widgets = Config::Get(Config::MAIN_OVERCLOCK_ENABLE);
|
||||
const bool enable_vi_clock_override_widgets = Config::Get(Config::MAIN_VI_OVERCLOCK_ENABLE);
|
||||
const bool enable_ram_override_widgets = Config::Get(Config::MAIN_RAM_OVERRIDE_ENABLE);
|
||||
const bool enable_custom_rtc_widgets =
|
||||
Config::Get(Config::MAIN_CUSTOM_RTC_ENABLE) && is_uninitialized;
|
||||
|
@ -278,6 +325,31 @@ void AdvancedPane::Update()
|
|||
return tr("%1% (%2 MHz)").arg(QString::number(percent), QString::number(clock));
|
||||
}());
|
||||
|
||||
QFont vi_bf = font();
|
||||
vi_bf.setBold(Config::GetActiveLayerForConfig(Config::MAIN_VI_OVERCLOCK_ENABLE) !=
|
||||
Config::LayerType::Base);
|
||||
m_vi_clock_override_checkbox->setFont(vi_bf);
|
||||
m_vi_clock_override_checkbox->setChecked(enable_vi_clock_override_widgets);
|
||||
|
||||
m_vi_clock_override_slider->setEnabled(enable_vi_clock_override_widgets);
|
||||
m_vi_clock_override_slider_label->setEnabled(enable_vi_clock_override_widgets);
|
||||
|
||||
{
|
||||
const QSignalBlocker blocker(m_vi_clock_override_slider);
|
||||
m_vi_clock_override_slider->setValue(
|
||||
static_cast<int>(std::round(Config::Get(Config::MAIN_VI_OVERCLOCK) * 100.f)));
|
||||
}
|
||||
|
||||
m_vi_clock_override_slider_label->setText([] {
|
||||
int percent = static_cast<int>(std::round(Config::Get(Config::MAIN_VI_OVERCLOCK) * 100.f));
|
||||
float core_vps =
|
||||
static_cast<float>(Core::System::GetInstance().GetVideoInterface().GetTargetRefreshRate());
|
||||
if (core_vps == 0.0f)
|
||||
core_vps = 59.94f;
|
||||
float vps = static_cast<float>(core_vps * Config::Get(Config::MAIN_VI_OVERCLOCK));
|
||||
return tr("%1% (%2 VPS)").arg(QString::number(percent), QString::number(vps, 'f', 2));
|
||||
}());
|
||||
|
||||
m_ram_override_checkbox->setEnabled(is_uninitialized);
|
||||
SignalBlocking(m_ram_override_checkbox)->setChecked(enable_ram_override_widgets);
|
||||
|
||||
|
|
|
@ -40,6 +40,11 @@ private:
|
|||
QLabel* m_cpu_clock_override_slider_label;
|
||||
QLabel* m_cpu_clock_override_description;
|
||||
|
||||
QCheckBox* m_vi_clock_override_checkbox;
|
||||
QSlider* m_vi_clock_override_slider;
|
||||
QLabel* m_vi_clock_override_slider_label;
|
||||
QLabel* m_vi_clock_override_description;
|
||||
|
||||
QCheckBox* m_custom_rtc_checkbox;
|
||||
QDateTimeEdit* m_custom_rtc_datetime;
|
||||
|
||||
|
|
|
@ -129,7 +129,8 @@ BuildExpression(const std::vector<ciface::Core::DeviceContainer::InputDetection>
|
|||
|
||||
// Remove duplicates
|
||||
std::ranges::sort(alternations);
|
||||
alternations.erase(std::unique(alternations.begin(), alternations.end()), alternations.end());
|
||||
const auto unique_result = std::ranges::unique(alternations);
|
||||
alternations.erase(unique_result.begin(), unique_result.end());
|
||||
|
||||
return fmt::to_string(fmt::join(alternations, "|"));
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue