364ce6fbfc
Also add a usable "doesn't work why" info and WARN! the user about clicking the rearm button. Merge "OpenGLIsUnsafe" and "CheckIsSafe" config keys Move the entire checking into CompositingPrefs BUG:250865 FIXED-IN:4.7
396 lines
13 KiB
C++
396 lines
13 KiB
C++
/********************************************************************
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
Copyright (C) 2007 Rivo Laks <rivolaks@hot.ee>
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*********************************************************************/
|
|
|
|
#include "compositingprefs.h"
|
|
|
|
#include "kwinglobals.h"
|
|
#include "kwinglplatform.h"
|
|
|
|
#include <kconfiggroup.h>
|
|
#include <kdebug.h>
|
|
#include <kxerrorhandler.h>
|
|
#include <klocale.h>
|
|
#include <kdeversion.h>
|
|
#include <ksharedconfig.h>
|
|
#include <kstandarddirs.h>
|
|
|
|
#include <qprocess.h>
|
|
|
|
|
|
namespace KWin
|
|
{
|
|
|
|
CompositingPrefs::CompositingPrefs()
|
|
: mRecommendCompositing(false)
|
|
, mEnableVSync(true)
|
|
, mEnableDirectRendering(true)
|
|
, mStrictBinding(true)
|
|
{
|
|
}
|
|
|
|
CompositingPrefs::~CompositingPrefs()
|
|
{
|
|
}
|
|
|
|
bool CompositingPrefs::recommendCompositing() const
|
|
{
|
|
return mRecommendCompositing;
|
|
}
|
|
|
|
bool CompositingPrefs::openGlIsBroken()
|
|
{
|
|
KSharedConfigPtr config = KSharedConfig::openConfig("kwinrc");
|
|
return KConfigGroup(config, "Compositing").readEntry("OpenGLIsUnsafe", false);
|
|
}
|
|
|
|
bool CompositingPrefs::compositingPossible()
|
|
{
|
|
// first off, check whether we figured that we'll crash on detection because of a buggy driver
|
|
KSharedConfigPtr config = KSharedConfig::openConfig("kwinrc");
|
|
KConfigGroup gl_workaround_group(config, "Compositing");
|
|
if (gl_workaround_group.readEntry("Backend", "OpenGL") == "OpenGL" &&
|
|
gl_workaround_group.readEntry("OpenGLIsUnsafe", false))
|
|
return false;
|
|
|
|
#ifdef KWIN_HAVE_COMPOSITING
|
|
Extensions::init();
|
|
if (!Extensions::compositeAvailable()) {
|
|
kDebug(1212) << "No composite extension available";
|
|
return false;
|
|
}
|
|
if (!Extensions::damageAvailable()) {
|
|
kDebug(1212) << "No damage extension available";
|
|
return false;
|
|
}
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
if (Extensions::glxAvailable())
|
|
return true;
|
|
#endif
|
|
#ifdef KWIN_HAVE_XRENDER_COMPOSITING
|
|
if (Extensions::renderAvailable() && Extensions::fixesAvailable())
|
|
return true;
|
|
#endif
|
|
#ifdef KWIN_HAVE_OPENGLES
|
|
return true;
|
|
#endif
|
|
kDebug(1212) << "No OpenGL or XRender/XFixes support";
|
|
return false;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
QString CompositingPrefs::compositingNotPossibleReason()
|
|
{
|
|
#ifdef KWIN_HAVE_COMPOSITING
|
|
// first off, check whether we figured that we'll crash on detection because of a buggy driver
|
|
KSharedConfigPtr config = KSharedConfig::openConfig("kwinrc");
|
|
KConfigGroup gl_workaround_group(config, "Compositing");
|
|
if (gl_workaround_group.readEntry("Backend", "OpenGL") == "OpenGL" &&
|
|
gl_workaround_group.readEntry("OpenGLIsUnsafe", false))
|
|
return i18n("<b>OpenGL compositing (the default) has crashed KWin in the past.</b><br>"
|
|
"This was most likely due to a driver bug."
|
|
"<p>If you think that you have meanwhile upgraded to a stable driver,<br>"
|
|
"you can reset this protection but <b>be aware that this might result in an immediate crash!</b></p>"
|
|
"<p>Alternatively, you might want to use the XRender backend instead.</p>");
|
|
|
|
Extensions::init();
|
|
if (!Extensions::compositeAvailable() || !Extensions::damageAvailable()) {
|
|
return i18n("Required X extensions (XComposite and XDamage) are not available.");
|
|
}
|
|
#if defined( KWIN_HAVE_OPENGL_COMPOSITING ) && !defined( KWIN_HAVE_XRENDER_COMPOSITING )
|
|
if (!Extensions::glxAvailable())
|
|
return i18n("GLX/OpenGL are not available and only OpenGL support is compiled.");
|
|
#elif !defined( KWIN_HAVE_OPENGL_COMPOSITING ) && defined( KWIN_HAVE_XRENDER_COMPOSITING )
|
|
if (!(Extensions::renderAvailable() && Extensions::fixesAvailable()))
|
|
return i18n("XRender/XFixes extensions are not available and only XRender support"
|
|
" is compiled.");
|
|
#else
|
|
if (!(Extensions::glxAvailable()
|
|
|| (Extensions::renderAvailable() && Extensions::fixesAvailable()))) {
|
|
return i18n("GLX/OpenGL and XRender/XFixes are not available.");
|
|
}
|
|
#endif
|
|
return QString();
|
|
#else
|
|
return i18n("Compositing was disabled at compile time.\n"
|
|
"It is likely Xorg development headers were not installed.");
|
|
#endif
|
|
}
|
|
|
|
void CompositingPrefs::detect()
|
|
{
|
|
if (!compositingPossible() || openGlIsBroken()) {
|
|
return;
|
|
}
|
|
|
|
// NOTICE: this is intended to workaround broken GL implementations that successfully segfault
|
|
// on glXQuery :-(
|
|
// we tag GL as unsafe. It *must* be reset before every return, and in case we "unexpectedly"
|
|
// end (aka "segfaulted") we know that we shall not try again
|
|
KSharedConfigPtr config = KSharedConfig::openConfig("kwinrc");
|
|
KConfigGroup gl_workaround_config = KConfigGroup(config, "Compositing");
|
|
gl_workaround_config.writeEntry("OpenGLIsUnsafe", true);
|
|
gl_workaround_config.sync();
|
|
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
#ifdef KWIN_HAVE_OPENGLES
|
|
bool haveContext = false;
|
|
bool canDetect = false;
|
|
EGLDisplay dpy = eglGetCurrentDisplay();
|
|
if (dpy != EGL_NO_DISPLAY) {
|
|
EGLContext ctx = eglGetCurrentContext();
|
|
if (ctx != EGL_NO_CONTEXT) {
|
|
haveContext = true;
|
|
canDetect = true;
|
|
}
|
|
}
|
|
if (!haveContext) {
|
|
canDetect = initEGLContext();
|
|
}
|
|
if (canDetect) {
|
|
detectDriverAndVersion();
|
|
applyDriverSpecificOptions();
|
|
}
|
|
if (!haveContext) {
|
|
deleteEGLContext();
|
|
}
|
|
#else
|
|
// HACK: This is needed for AIGLX
|
|
if (qstrcmp(qgetenv("KWIN_DIRECT_GL"), "1") != 0) {
|
|
// Start an external helper program that initializes GLX and returns
|
|
// 0 if we can use direct rendering, and 1 otherwise.
|
|
// The reason we have to use an external program is that after GLX
|
|
// has been initialized, it's too late to set the LIBGL_ALWAYS_INDIRECT
|
|
// environment variable.
|
|
// Direct rendering is preferred, since not all OpenGL extensions are
|
|
// available with indirect rendering.
|
|
const QString opengl_test = KStandardDirs::findExe("kwin_opengl_test");
|
|
if (QProcess::execute(opengl_test) != 0)
|
|
setenv("LIBGL_ALWAYS_INDIRECT", "1", true);
|
|
}
|
|
if (!Extensions::glxAvailable()) {
|
|
kDebug(1212) << "No GLX available";
|
|
gl_workaround_config.writeEntry("OpenGLIsUnsafe", false);
|
|
gl_workaround_config.sync();
|
|
return;
|
|
}
|
|
int glxmajor, glxminor;
|
|
glXQueryVersion(display(), &glxmajor, &glxminor);
|
|
kDebug(1212) << "glx version is " << glxmajor << "." << glxminor;
|
|
bool hasglx13 = (glxmajor > 1 || (glxmajor == 1 && glxminor >= 3));
|
|
|
|
// remember and later restore active context
|
|
GLXContext oldcontext = glXGetCurrentContext();
|
|
GLXDrawable olddrawable = glXGetCurrentDrawable();
|
|
GLXDrawable oldreaddrawable = None;
|
|
if (hasglx13)
|
|
oldreaddrawable = glXGetCurrentReadDrawable();
|
|
|
|
if (initGLXContext()) {
|
|
detectDriverAndVersion();
|
|
applyDriverSpecificOptions();
|
|
}
|
|
if (hasglx13)
|
|
glXMakeContextCurrent(display(), olddrawable, oldreaddrawable, oldcontext);
|
|
else
|
|
glXMakeCurrent(display(), olddrawable, oldcontext);
|
|
deleteGLXContext();
|
|
#endif
|
|
gl_workaround_config.writeEntry("OpenGLIsUnsafe", false);
|
|
gl_workaround_config.sync();
|
|
#endif
|
|
}
|
|
|
|
bool CompositingPrefs::initGLXContext()
|
|
{
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
#ifndef KWIN_HAVE_OPENGLES
|
|
mGLContext = NULL;
|
|
KXErrorHandler handler;
|
|
// Most of this code has been taken from glxinfo.c
|
|
QVector<int> attribs;
|
|
attribs << GLX_RGBA;
|
|
attribs << GLX_RED_SIZE << 1;
|
|
attribs << GLX_GREEN_SIZE << 1;
|
|
attribs << GLX_BLUE_SIZE << 1;
|
|
attribs << None;
|
|
|
|
XVisualInfo* visinfo = glXChooseVisual(display(), DefaultScreen(display()), attribs.data());
|
|
if (!visinfo) {
|
|
attribs.last() = GLX_DOUBLEBUFFER;
|
|
attribs << None;
|
|
visinfo = glXChooseVisual(display(), DefaultScreen(display()), attribs.data());
|
|
if (!visinfo) {
|
|
kDebug(1212) << "Error: couldn't find RGB GLX visual";
|
|
return false;
|
|
}
|
|
}
|
|
|
|
mGLContext = glXCreateContext(display(), visinfo, NULL, True);
|
|
if (!mGLContext) {
|
|
kDebug(1212) << "glXCreateContext failed";
|
|
XDestroyWindow(display(), mGLWindow);
|
|
return false;
|
|
}
|
|
|
|
XSetWindowAttributes attr;
|
|
attr.background_pixel = 0;
|
|
attr.border_pixel = 0;
|
|
attr.colormap = XCreateColormap(display(), rootWindow(), visinfo->visual, AllocNone);
|
|
attr.event_mask = StructureNotifyMask | ExposureMask;
|
|
unsigned long mask = CWBackPixel | CWBorderPixel | CWColormap | CWEventMask;
|
|
int width = 100, height = 100;
|
|
mGLWindow = XCreateWindow(display(), rootWindow(), 0, 0, width, height,
|
|
0, visinfo->depth, InputOutput,
|
|
visinfo->visual, mask, &attr);
|
|
|
|
return glXMakeCurrent(display(), mGLWindow, mGLContext) && !handler.error(true);
|
|
#else
|
|
return false;
|
|
#endif
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
void CompositingPrefs::deleteGLXContext()
|
|
{
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
#ifndef KWIN_HAVE_OPENGLES
|
|
if (mGLContext == NULL)
|
|
return;
|
|
glXDestroyContext(display(), mGLContext);
|
|
XDestroyWindow(display(), mGLWindow);
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
bool CompositingPrefs::initEGLContext()
|
|
{
|
|
#ifdef KWIN_HAVE_OPENGLES
|
|
mEGLDisplay = eglGetDisplay(display());
|
|
if (mEGLDisplay == EGL_NO_DISPLAY) {
|
|
return false;
|
|
}
|
|
if (eglInitialize(mEGLDisplay, 0, 0) == EGL_FALSE) {
|
|
return false;
|
|
}
|
|
eglBindAPI(EGL_OPENGL_ES_API);
|
|
|
|
const EGLint config_attribs[] = {
|
|
EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
|
|
EGL_RED_SIZE, 1,
|
|
EGL_GREEN_SIZE, 1,
|
|
EGL_BLUE_SIZE, 1,
|
|
EGL_ALPHA_SIZE, 0,
|
|
EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
|
|
EGL_CONFIG_CAVEAT, EGL_NONE,
|
|
EGL_NONE,
|
|
};
|
|
|
|
EGLint count;
|
|
EGLConfig configs[1024];
|
|
eglChooseConfig(mEGLDisplay, config_attribs, configs, 1024, &count);
|
|
|
|
EGLint visualId = XVisualIDFromVisual((Visual*)QX11Info::appVisual());
|
|
|
|
EGLConfig config = configs[0];
|
|
for (int i = 0; i < count; i++) {
|
|
EGLint val;
|
|
eglGetConfigAttrib(mEGLDisplay, configs[i], EGL_NATIVE_VISUAL_ID, &val);
|
|
if (visualId == val) {
|
|
config = configs[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
XSetWindowAttributes attr;
|
|
attr.background_pixel = 0;
|
|
attr.border_pixel = 0;
|
|
attr.colormap = XCreateColormap(display(), rootWindow(), (Visual*)QX11Info::appVisual(), AllocNone);
|
|
attr.event_mask = StructureNotifyMask | ExposureMask;
|
|
unsigned long mask = CWBackPixel | CWBorderPixel | CWColormap | CWEventMask;
|
|
int width = 100, height = 100;
|
|
mGLWindow = XCreateWindow(display(), rootWindow(), 0, 0, width, height,
|
|
0, QX11Info::appDepth(), InputOutput,
|
|
(Visual*)QX11Info::appVisual(), mask, &attr);
|
|
|
|
mEGLSurface = eglCreateWindowSurface(mEGLDisplay, config, mGLWindow, 0);
|
|
|
|
const EGLint context_attribs[] = {
|
|
EGL_CONTEXT_CLIENT_VERSION, 2,
|
|
EGL_NONE
|
|
};
|
|
|
|
mEGLContext = eglCreateContext(mEGLDisplay, config, EGL_NO_CONTEXT, context_attribs);
|
|
if (mEGLContext == EGL_NO_CONTEXT) {
|
|
return false;
|
|
}
|
|
if (eglMakeCurrent(mEGLDisplay, mEGLSurface, mEGLSurface, mEGLContext) == EGL_FALSE) {
|
|
return false;
|
|
}
|
|
EGLint error = eglGetError();
|
|
if (error != EGL_SUCCESS) {
|
|
return false;
|
|
}
|
|
return true;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
void CompositingPrefs::deleteEGLContext()
|
|
{
|
|
#ifdef KWIN_HAVE_OPENGLES
|
|
eglMakeCurrent(mEGLDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
|
|
eglDestroyContext(mEGLDisplay, mEGLContext);
|
|
eglDestroySurface(mEGLDisplay, mEGLSurface);
|
|
eglTerminate(mEGLDisplay);
|
|
eglReleaseThread();
|
|
XDestroyWindow(display(), mGLWindow);
|
|
#endif
|
|
}
|
|
|
|
void CompositingPrefs::detectDriverAndVersion()
|
|
{
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
GLPlatform *gl = GLPlatform::instance();
|
|
gl->detect();
|
|
gl->printResults();
|
|
#endif
|
|
}
|
|
|
|
// See http://techbase.kde.org/Projects/KWin/HW for a list of some cards that are known to work.
|
|
void CompositingPrefs::applyDriverSpecificOptions()
|
|
{
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
// Always recommend
|
|
mRecommendCompositing = true;
|
|
|
|
GLPlatform *gl = GLPlatform::instance();
|
|
mStrictBinding = !gl->supports(LooseBinding);
|
|
if (gl->driver() == Driver_Intel)
|
|
mEnableVSync = false;
|
|
#endif
|
|
}
|
|
|
|
} // namespace
|
|
|