kwin/plugins/platforms/x11/standalone/omlsynccontrolvsyncmonitor.cpp
Vlad Zahorodnii ffc013292f platforms/x11: Don't monitor vblanks on the root window
This makes scrolling in Firefox, which also monitors vblanks on the root
window, laggy.
2021-02-01 18:49:16 +00:00

161 lines
4.6 KiB
C++

/*
SPDX-FileCopyrightText: 2020 Vlad Zahorodnii <vlad.zahorodnii@kde.org>
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "omlsynccontrolvsyncmonitor.h"
#include "glxconvenience.h"
#include "logging.h"
#include <QX11Info>
namespace KWin
{
OMLSyncControlVsyncMonitor *OMLSyncControlVsyncMonitor::create(QObject *parent)
{
const char *extensions = glXQueryExtensionsString(QX11Info::display(),
QX11Info::appScreen());
if (!strstr(extensions, "GLX_OML_sync_control")) {
return nullptr; // GLX_OML_sync_control is unsupported.
}
OMLSyncControlVsyncMonitor *monitor = new OMLSyncControlVsyncMonitor(parent);
if (monitor->isValid()) {
return monitor;
}
delete monitor;
return nullptr;
}
OMLSyncControlVsyncMonitorHelper::OMLSyncControlVsyncMonitorHelper(QObject *parent)
: QObject(parent)
{
// Establish a new X11 connection to avoid locking up the main X11 connection.
m_display = XOpenDisplay(DisplayString(QX11Info::display()));
if (!m_display) {
qCDebug(KWIN_X11STANDALONE) << "Failed to establish vsync monitor X11 connection";
return;
}
Window rootWindow = DefaultRootWindow(m_display);
const int attribs[] = {
GLX_RENDER_TYPE, GLX_RGBA_BIT,
GLX_DRAWABLE_TYPE, GLX_WINDOW_BIT,
0
};
GLXFBConfig config = chooseGlxFbConfig(m_display, attribs);
if (!config) {
qCDebug(KWIN_X11STANDALONE) << "Couldn't find any suitable FBConfig for vsync monitor";
return;
}
XVisualInfo *visualInfo = glXGetVisualFromFBConfig(m_display, config);
if (!visualInfo) {
return;
}
Visual *visual = visualInfo->visual;
const int depth = visualInfo->depth;
XFree(visualInfo);
Colormap colormap = XCreateColormap(m_display, rootWindow, visual, AllocNone);
XSetWindowAttributes attributes;
attributes.colormap = colormap;
m_dummyWindow = XCreateWindow(m_display, rootWindow, 0, 0, 1, 1, 0, depth,
InputOutput, visual, CWColormap, &attributes);
XFreeColormap(m_display, colormap);
if (!m_dummyWindow) {
qCDebug(KWIN_X11STANDALONE) << "Failed to create a dummy window for vsync monitor";
return;
}
m_drawable = glXCreateWindow(m_display, config, m_dummyWindow, nullptr);
if (!m_drawable) {
qCDebug(KWIN_X11STANDALONE) << "Failed to create GLXWindow for dummy window";
return;
}
m_localContext = glXCreateNewContext(m_display, config, GLX_RGBA_TYPE, 0, true);
if (!m_localContext) {
qCDebug(KWIN_X11STANDALONE) << "Failed to create opengl context for vsync monitor";
return;
}
}
OMLSyncControlVsyncMonitorHelper::~OMLSyncControlVsyncMonitorHelper()
{
if (m_localContext) {
glXDestroyContext(m_display, m_localContext);
}
if (m_drawable) {
glXDestroyWindow(m_display, m_drawable);
}
if (m_dummyWindow) {
XDestroyWindow(m_display, m_dummyWindow);
}
if (m_display) {
XCloseDisplay(m_display);
}
}
bool OMLSyncControlVsyncMonitorHelper::isValid() const
{
return m_display && m_localContext && m_drawable;
}
void OMLSyncControlVsyncMonitorHelper::poll()
{
if (!glXMakeCurrent(m_display, m_drawable, m_localContext)) {
qCDebug(KWIN_X11STANDALONE) << "Failed to make vsync monitor OpenGL context current";
return;
}
int64_t ust, msc, sbc;
glXGetSyncValuesOML(m_display, m_drawable, &ust, &msc, &sbc);
glXWaitForMscOML(m_display, m_drawable, 0, 2, (msc + 1) % 2, &ust, &msc, &sbc);
emit vblankOccurred(std::chrono::microseconds(ust));
}
OMLSyncControlVsyncMonitor::OMLSyncControlVsyncMonitor(QObject *parent)
: VsyncMonitor(parent)
, m_thread(new QThread)
, m_helper(new OMLSyncControlVsyncMonitorHelper)
{
m_helper->moveToThread(m_thread);
connect(m_helper, &OMLSyncControlVsyncMonitorHelper::errorOccurred,
this, &OMLSyncControlVsyncMonitor::errorOccurred);
connect(m_helper, &OMLSyncControlVsyncMonitorHelper::vblankOccurred,
this, &OMLSyncControlVsyncMonitor::vblankOccurred);
m_thread->setObjectName(QStringLiteral("vsync event monitor"));
m_thread->start();
}
OMLSyncControlVsyncMonitor::~OMLSyncControlVsyncMonitor()
{
m_thread->quit();
m_thread->wait();
delete m_helper;
delete m_thread;
}
bool OMLSyncControlVsyncMonitor::isValid() const
{
return m_helper->isValid();
}
void OMLSyncControlVsyncMonitor::arm()
{
QMetaObject::invokeMethod(m_helper, &OMLSyncControlVsyncMonitorHelper::poll);
}
} // namespace KWin