1
0
mirror of https://github.com/doitsujin/dxvk.git synced 2025-01-19 05:52:11 +01:00

[dxvk] Don't disable frame rate limiter if vsync is enabled

This only works if we know the actual refresh rate of the display.
However, in a wine virtual desktop or with proton's fshack, this
is often not the case, so we'd see a 60 Hz mode on a high-refresh
rate display and never actually enable the limiter.
This commit is contained in:
Philip Rebohle 2022-08-17 01:49:42 +02:00
parent 04bc1bac73
commit 61025c0079
No known key found for this signature in database
GPG Key ID: C8CC613427A31C99
7 changed files with 4 additions and 53 deletions

View File

@ -19,9 +19,10 @@
# d3d9.maxFrameLatency = 0 # d3d9.maxFrameLatency = 0
# Enables a frame rate limiter, unless the game is already # Enables frame rate limiter. The main purpose of this is to work around
# limited to the same refresh rate by vertical synchronization. # bugs in games that have physics or other simulation tied to their frame
# # rate, but do not provide their own limiter.
#
# Supported values : Any non-negative integer # Supported values : Any non-negative integer
# dxgi.maxFrameRate = 0 # dxgi.maxFrameRate = 0

View File

@ -252,9 +252,6 @@ namespace dxvk {
DXGI_RATIONAL rate = pDisplayMode->RefreshRate; DXGI_RATIONAL rate = pDisplayMode->RefreshRate;
m_displayRefreshRate = double(rate.Numerator) / double(rate.Denominator); m_displayRefreshRate = double(rate.Numerator) / double(rate.Denominator);
} }
if (m_presenter != nullptr)
m_presenter->setFrameRateLimiterRefreshRate(m_displayRefreshRate);
} }
@ -407,7 +404,6 @@ namespace dxvk {
presenterDesc); presenterDesc);
m_presenter->setFrameRateLimit(m_parent->GetOptions()->maxFrameRate); m_presenter->setFrameRateLimit(m_parent->GetOptions()->maxFrameRate);
m_presenter->setFrameRateLimiterRefreshRate(m_displayRefreshRate);
CreateRenderTargetViews(); CreateRenderTargetViews();
} }

View File

@ -798,7 +798,6 @@ namespace dxvk {
presenterDesc); presenterDesc);
m_presenter->setFrameRateLimit(m_parent->GetOptions()->maxFrameRate); m_presenter->setFrameRateLimit(m_parent->GetOptions()->maxFrameRate);
m_presenter->setFrameRateLimiterRefreshRate(m_displayRefreshRate);
CreateRenderTargetViews(); CreateRenderTargetViews();
} }
@ -1026,9 +1025,6 @@ namespace dxvk {
void D3D9SwapChainEx::NotifyDisplayRefreshRate( void D3D9SwapChainEx::NotifyDisplayRefreshRate(
double RefreshRate) { double RefreshRate) {
m_displayRefreshRate = RefreshRate; m_displayRefreshRate = RefreshRate;
if (m_presenter != nullptr)
m_presenter->setFrameRateLimiterRefreshRate(RefreshRate);
} }

View File

@ -44,28 +44,12 @@ namespace dxvk {
} }
void FpsLimiter::setDisplayRefreshRate(double refreshRate) {
std::lock_guard<dxvk::mutex> lock(m_mutex);
m_refreshInterval = refreshRate > 0.0
? TimerDuration(int64_t(double(TimerDuration::period::den) / refreshRate))
: TimerDuration::zero();
}
void FpsLimiter::delay(bool vsyncEnabled) { void FpsLimiter::delay(bool vsyncEnabled) {
std::lock_guard<dxvk::mutex> lock(m_mutex); std::lock_guard<dxvk::mutex> lock(m_mutex);
if (!isEnabled()) if (!isEnabled())
return; return;
// If the swap chain is known to have vsync enabled and the
// refresh rate is similar to the target frame rate, disable
// the limiter so it does not screw up frame times
if (vsyncEnabled && !m_envOverride
&& m_refreshInterval * 100 > m_targetInterval * 97)
return;
auto t0 = m_lastFrame; auto t0 = m_lastFrame;
auto t1 = dxvk::high_resolution_clock::now(); auto t1 = dxvk::high_resolution_clock::now();

View File

@ -28,16 +28,6 @@ namespace dxvk {
*/ */
void setTargetFrameRate(double frameRate); void setTargetFrameRate(double frameRate);
/**
* \brief Sets display refresh rate
*
* This information is used to decide whether or not
* the limiter should be active in the first place in
* case vertical synchronization is enabled.
* \param [in] refreshRate Current refresh rate
*/
void setDisplayRefreshRate(double refreshRate);
/** /**
* \brief Stalls calling thread as necessary * \brief Stalls calling thread as necessary
* *
@ -75,7 +65,6 @@ namespace dxvk {
dxvk::mutex m_mutex; dxvk::mutex m_mutex;
TimerDuration m_targetInterval = TimerDuration::zero(); TimerDuration m_targetInterval = TimerDuration::zero();
TimerDuration m_refreshInterval = TimerDuration::zero();
TimerDuration m_deviation = TimerDuration::zero(); TimerDuration m_deviation = TimerDuration::zero();
TimePoint m_lastFrame; TimePoint m_lastFrame;

View File

@ -232,11 +232,6 @@ namespace dxvk::vk {
} }
void Presenter::setFrameRateLimiterRefreshRate(double refreshRate) {
m_fpsLimiter.setDisplayRefreshRate(refreshRate);
}
VkResult Presenter::getSupportedFormats(std::vector<VkSurfaceFormatKHR>& formats, const PresenterDesc& desc) { VkResult Presenter::getSupportedFormats(std::vector<VkSurfaceFormatKHR>& formats, const PresenterDesc& desc) {
uint32_t numFormats = 0; uint32_t numFormats = 0;

View File

@ -161,16 +161,6 @@ namespace dxvk::vk {
*/ */
void setFrameRateLimit(double frameRate); void setFrameRateLimit(double frameRate);
/**
* \brief Notifies frame rate limiter about the display refresh rate
*
* Used to dynamically disable the frame rate limiter in case
* vertical synchronization is used and the target frame rate
* roughly equals the display's refresh rate.
* \param [in] refresnRate Current refresh rate
*/
void setFrameRateLimiterRefreshRate(double refreshRate);
/** /**
* \brief Checks whether a Vulkan swap chain exists * \brief Checks whether a Vulkan swap chain exists
* *