/******************************************************************** KWin - the KDE window manager This file is part of the KDE project. Copyright (C) 2007 Rivo Laks 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 . *********************************************************************/ #include "compositingprefs.h" #include "kwinglobals.h" #include #include #include #include namespace KWin { CompositingPrefs::CompositingPrefs() : mXgl( false ) , mRecommendCompositing( false ) , mEnableVSync( true ) , mEnableDirectRendering( true ) , mStrictBinding( true ) { } CompositingPrefs::~CompositingPrefs() { } bool CompositingPrefs::recommendCompositing() const { return mRecommendCompositing; } bool CompositingPrefs::compositingPossible() { #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 kDebug( 1212 ) << "No OpenGL or XRender/XFixes support"; return false; #else return false; #endif } QString CompositingPrefs::compositingNotPossibleReason() { #ifdef KWIN_HAVE_COMPOSITING 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 } // This function checks selected compositing setup and returns false if it should not // be used even if explicitly configured (unless checks are overridden). // More checks like broken XRender setups etc. should be added here. bool CompositingPrefs::validateSetup( CompositingType compositingType ) const { switch( compositingType ) { case NoCompositing: return false; case OpenGLCompositing: if( mDriver == "software" ) { kDebug( 1212 ) << "Software GL renderer detected, forcing compositing off."; return false; } return true; // allow case XRenderCompositing: return true; // xrender - always allow? } abort(); } void CompositingPrefs::detect() { if( !compositingPossible()) { return; } #ifdef KWIN_HAVE_OPENGL_COMPOSITING if( !Extensions::glxAvailable()) { kDebug( 1212 ) << "No GLX available"; 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 } bool CompositingPrefs::initGLXContext() { #ifdef KWIN_HAVE_OPENGL_COMPOSITING mGLContext = NULL; KXErrorHandler handler; // Most of this code has been taken from glxinfo.c QVector 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 } void CompositingPrefs::deleteGLXContext() { #ifdef KWIN_HAVE_OPENGL_COMPOSITING if( mGLContext == NULL ) return; glXDestroyContext( display(), mGLContext ); XDestroyWindow( display(), mGLWindow ); #endif } void CompositingPrefs::detectDriverAndVersion() { #ifdef KWIN_HAVE_OPENGL_COMPOSITING mGLVendor = QString((const char*)glGetString( GL_VENDOR )); mGLRenderer = QString((const char*)glGetString( GL_RENDERER )); mGLVersion = QString((const char*)glGetString( GL_VERSION )); mXgl = detectXgl(); kDebug( 1212 ) << "GL vendor is" << mGLVendor; kDebug( 1212 ) << "GL renderer is" << mGLRenderer; kDebug( 1212 ) << "GL version is" << mGLVersion; kDebug( 1212 ) << "XGL:" << ( mXgl ? "yes" : "no" ); if( mGLRenderer.startsWith( "Mesa DRI Intel" ) || mGLRenderer.startsWith( "Mesa DRI Mobile Intel" )) // krazy:exclude=strings { mDriver = "intel"; QStringList words = mGLRenderer.split(' '); mVersion = Version( words[ words.count() - 2 ] ); } else if( mGLVendor == "NVIDIA Corporation" ) { mDriver = "nvidia"; QStringList words = mGLVersion.split(' '); mVersion = Version( words[ words.count() - 1 ] ); } else if( mGLVendor == "ATI Technologies Inc." ) { mDriver = "fglrx"; // Ati changed the version string. // The GL version is either in the first or second part QStringList versionParts = mGLVersion.split(' '); if( versionParts.first().count(".") == 2 || versionParts.count() == 1 ) mVersion = Version( versionParts.first() ); else { // version in second part is encapsulated in () mVersion = Version( versionParts[1].mid( 1, versionParts[1].length() -2 ) ); } } else if( mGLRenderer.startsWith( "Mesa DRI R" )) // krazy:exclude=strings { mDriver = "radeon"; mVersion = Version( mGLRenderer.split(' ')[ 3 ] ); // Check that the version string is changed, and try the fifth element if it does if (!mVersion.startsWith("20")) mVersion = Version( mGLRenderer.split(' ')[ 5 ] ); } else if( mGLRenderer == "Software Rasterizer" ) { mDriver = "software"; QStringList words = mGLVersion.split(' '); mVersion = Version( words[ words.count() - 1 ] ); } else { mDriver = "unknown"; } kDebug( 1212 ) << "Detected driver" << mDriver << ", version" << mVersion.join("."); #endif } void CompositingPrefs::parseMesaVersion( const QString &version, int *major, int *minor ) { *major = 0; *minor = 0; const QStringList tokens = version.split( ' ' ); int token = 0; while( token < tokens.count() && !tokens.at( token ).endsWith( "Mesa" ) ) token++; if( token < tokens.count() - 1 ) { const QStringList version = tokens.at( token + 1 ).split( '.' ); if( version.count() >= 2 ) { *major = version[ 0 ].toInt(); int end = 0; while( end < version[ 1 ].length() && version[ 1 ][ end ].isDigit() ) end++; *minor = version[ 1 ].left( end ).toInt(); } } } // See http://techbase.kde.org/Projects/KWin/HW for a list of some cards that are known to work. void CompositingPrefs::applyDriverSpecificOptions() { if( mXgl ) { kDebug( 1212 ) << "xgl, enabling"; mRecommendCompositing = true; mStrictBinding = false; } else if( mDriver == "intel" ) { kDebug( 1212 ) << "intel driver, disabling vsync, enabling direct"; mEnableVSync = false; mEnableDirectRendering = true; if( mVersion >= Version( "20061017" )) { if( mGLRenderer.contains( "Intel(R) 9" )) { // Enable compositing by default on 900-series cards kDebug( 1212 ) << "intel >= 20061017 and 900-series card, enabling compositing"; mRecommendCompositing = true; } if( mGLRenderer.contains( "Mesa DRI Intel(R) G" )) { // e.g. G43 chipset kDebug( 1212 ) << "intel >= 20061017 and Gxx-series card, enabling compositing"; mRecommendCompositing = true; } } } else if( mDriver == "nvidia" ) { mStrictBinding = false; if( mVersion >= Version( "173.14.12" )) { kDebug( 1212 ) << "nvidia >= 173.14.12, enabling compositing"; mRecommendCompositing = true; } } else if( mDriver == "radeon" ) { if( mGLRenderer.startsWith( "Mesa DRI R200" ) && mVersion >= Version( "20060602" )) // krazy:exclude=strings { kDebug( 1212 ) << "radeon r200 >= 20060602, enabling compositing"; mRecommendCompositing = true; } if( mGLRenderer.startsWith( "Mesa DRI R300" ) && mVersion >= Version( "20090101" )) // krazy:exclude=strings { kDebug( 1212 ) << "radeon r300 >= 20090101, enabling compositing"; mRecommendCompositing = true; } if( mGLRenderer.startsWith( "Mesa DRI R600" ) ) { // Enable compositing with Mesa 7.7 or later int major, minor; parseMesaVersion( mGLVersion, &major, &minor ); if( major > 7 || ( major == 7 && minor >= 7 ) ) { kDebug( 1212 ) << "Radeon R600/R700, Mesa 7.7 or better. Enabling compositing."; mRecommendCompositing = true; } } } else if( mDriver == "fglrx" ) { // radeon r200 only ? if( mVersion >= Version( "2.1.7412" )) { kDebug( 1212 ) << "fglrx >= 2.1.7412, enabling compositing"; mRecommendCompositing = true; } } } bool CompositingPrefs::detectXgl() { // Xgl apparently uses only this specific X version return VendorRelease(display()) == 70000001; } CompositingPrefs::Version::Version( const QString& str ) : QStringList() { QRegExp numrx( "(\\d+)|(\\D+)" ); int pos = 0; while(( pos = numrx.indexIn( str, pos )) != -1 ) { pos += numrx.matchedLength(); QString part = numrx.cap(); if( part == "." ) continue; append( part ); } } int CompositingPrefs::Version::compare( const Version& v ) const { for( int i = 0; i < qMin( count(), v.count() ); i++ ) { if( at( i )[ 0 ].isDigit() ) { // This part of version string is numeric - compare numbers int num = at( i ).toInt(); int vnum = v.at( i ).toInt(); if( num > vnum ) return 1; else if( num < vnum ) return -1; } else { // This part is string if( at( i ) > v.at( i )) return 1; else if( at( i ) < v.at( i )) return -1; } } if( count() > v.count() ) return 1; else if( count() < v.count() ) return -1; else return 0; } } // namespace