Bug 483240 - VRR below 30-ish FPS turns off completely
Summary: VRR below 30-ish FPS turns off completely
Status: CONFIRMED
Alias: None
Product: kwin
Classification: Plasma
Component: platform-drm (show other bugs)
Version: 6.0.1
Platform: Arch Linux Linux
: NOR normal
Target Milestone: ---
Assignee: KWin default assignee
URL:
Keywords: qt6
Depends on:
Blocks:
 
Reported: 2024-03-11 14:11 UTC by fililip
Modified: 2024-05-13 16:29 UTC (History)
4 users (show)

See Also:
Latest Commit:
Version Fixed In:
Sentry Crash Report:


Attachments

Note You need to log in before you can comment on or make changes to this bug.
Description fililip 2024-03-11 14:11:11 UTC
SUMMARY
Adaptive sync set to "Automatic" or "Always" does not work with content refreshing below 30 FPS (or close). This makes 24 FPS video content less pleasant to watch (than on Plasma 5), as LFC probably doesn't get a chance to kick in.

This does not happen on Plasma 5.

STEPS TO REPRODUCE
Open up mpv with a 23.98 FPS video playing fullscreen or vkcube/vrrtest limited to 30 FPS.

OBSERVED RESULT
Video juddering/monitor OSD saying the screen actually refreshes at 165Hz (my refresh rate).

EXPECTED RESULT
LFC kicks in properly no matter what the refresh rate is.

SOFTWARE/OS VERSIONS
Operating System: Arch Linux 
KDE Plasma Version: 6.0.1
KDE Frameworks Version: 6.0.0
Qt Version: 6.6.2
Kernel Version: 6.7.8-arch1-1 (64-bit)
Graphics Platform: Wayland

ADDITIONAL INFORMATION
I assume this is due to one MR/issue I remember seeing on the Plasma GitLab one day about keeping a certain minimum refresh rate for content, which was introduced to eliminate screen flickering with some panels. My panel has never manifested such issues, I've never ever seen flickering no matter how weird the content refresh rate was and how often it was changing. Is there a way to override this behavior with some kind of envvar?
Comment 1 fililip 2024-03-11 14:21:15 UTC
I should probably add that using the hardware cursor works fine (I was using the software one), but I'd personally prefer to have the option to retain even a low software cursor refresh rate anyway, or at least not have this behavior at all when I'm not moving the mouse.
Comment 2 Zamundaaa 2024-03-11 14:23:39 UTC
This is unfortunately intentional for now; it's indeed to avoid brightness flicker with many displays at such low refresh rates, and to work around driver bugs and API limitations with the hardware cursor.
This limitation will be removed again (or at least changed to a lower minimum refresh rate) once LFC is implemented in KWin instead of the driver.

> Is there a way to override this behavior with some kind of envvar?
No, this is not adjustable.
Comment 3 fililip 2024-03-11 14:25:02 UTC
Ok, thanks for the quick response! I appreciate it. I'll stick to the HW cursor for now then.
Comment 4 fililip 2024-03-19 22:45:30 UTC
One note I'll add: after resuming from suspend, even when using the HW cursor, VRR still breaks below 30 FPS and requires a restart of KWin. Is this just random behavior or a bug?
Comment 5 fililip 2024-04-05 10:41:47 UTC
I managed to add an option to override this behavior with an environment variable, LFC finally works as it did before.

diff --git a/src/core/renderloop.cpp b/src/core/renderloop.cpp
index 5e3a74c..de7b7d3 100644
--- a/src/core/renderloop.cpp
+++ b/src/core/renderloop.cpp
@@ -201,10 +201,12 @@ void RenderLoop::scheduleRepaint(Item *item)
     if (d->pendingRepaint) {
         return;
     }
+    bool isEnvVarSet = false;
+    const bool forceKernelLFC = qEnvironmentVariableIntValue("KWIN_FORCE_KERNEL_LFC", &isEnvVarSet) != 0 && isEnvVarSet;
     const bool vrr = d->presentationMode == PresentationMode::AdaptiveSync || d->presentationMode == PresentationMode::AdaptiveAsync;
     if (vrr && workspace()->activeWindow() && d->output) {
         Window *const activeWindow = workspace()->activeWindow();
-        if (activeWindow->isOnOutput(d->output) && activeWindow->surfaceItem() && item != activeWindow->surfaceItem() && activeWindow->surfaceItem()->frameTimeEstimation() <= std::chrono::nanoseconds(1'000'000'000) / 30) {
+        if (activeWindow->isOnOutput(d->output) && activeWindow->surfaceItem() && item != activeWindow->surfaceItem() && !forceKernelLFC && activeWindow->surfaceItem()->frameTimeEstimation() <= std::chrono::nanoseconds(1'000'000'000) / 30) {
             return;
         }
     }
Comment 6 fililip 2024-05-13 16:29:57 UTC
Update: this is a problem only once https://bugs.kde.org/show_bug.cgi?id=485425 happens. Kernel LFC works fine until then.
(This is why it's not that obviously noticeable, since I don't run such Wine/X11 programs every session.)

In fact, the patch I sent above makes matters even worse - VRR is completely disabled if the situation above occurs.
This means that the bug #485425 is related to this one. (Is it possible to mark it as such?)