From 86fa64d9713e950eaa9ea52998b00852f3d4fd38 Mon Sep 17 00:00:00 2001 From: kgv Date: Thu, 24 May 2012 10:03:23 +0400 Subject: [PATCH] 0023172: Added workaround for BUGs in Intel OpenGL drivers Check GL context already bound before wglMakeCurrent() call. --- src/OpenGl/OpenGl_Context.cxx | 68 +++++++++++++++++++--- src/OpenGl/OpenGl_Context.hxx | 5 ++ src/OpenGl/OpenGl_GraphicDriver_Export.cxx | 10 ++++ src/OpenGl/OpenGl_Window.cxx | 29 ++------- 4 files changed, 81 insertions(+), 31 deletions(-) diff --git a/src/OpenGl/OpenGl_Context.cxx b/src/OpenGl/OpenGl_Context.cxx index eb99984efc..d5c01e6263 100644 --- a/src/OpenGl/OpenGl_Context.cxx +++ b/src/OpenGl/OpenGl_Context.cxx @@ -102,6 +102,31 @@ OpenGl_Context::~OpenGl_Context() delete extFBO; } +// ======================================================================= +// function : IsCurrent +// purpose : +// ======================================================================= +Standard_Boolean OpenGl_Context::IsCurrent() const +{ +#if (defined(_WIN32) || defined(__WIN32__)) + if (myWindowDC == NULL || myGContext == NULL) + { + return Standard_False; + } + return (( (HDC )myWindowDC == wglGetCurrentDC()) + && ((HGLRC )myGContext == wglGetCurrentContext())); +#else + if (myDisplay == NULL || myWindow == 0 || myGContext == 0) + { + return Standard_False; + } + + return ( ((Display* )myDisplay == glXGetCurrentDisplay()) + && ((GLXContext )myGContext == glXGetCurrentContext()) + && ((GLXDrawable )myWindow == glXGetCurrentDrawable())); +#endif +} + // ======================================================================= // function : MakeCurrent // purpose : @@ -109,17 +134,44 @@ OpenGl_Context::~OpenGl_Context() Standard_Boolean OpenGl_Context::MakeCurrent() { #if (defined(_WIN32) || defined(__WIN32__)) - if (myWindowDC == NULL || myGContext == NULL || - !wglMakeCurrent ((HDC )myWindowDC, (HGLRC )myGContext)) + if (myWindowDC == NULL || myGContext == NULL) { - //GLenum anErrCode = glGetError(); - //const GLubyte* anErrorString = gluErrorString (anErrCode); - //std::cerr << "wglMakeCurrent() failed: " << anErrCode << " " << anErrorString << "\n"; + Standard_ProgramError_Raise_if (myIsInitialized, "OpenGl_Context::Init() should be called before!"); + return Standard_False; + } + + // technically it should be safe to activate already bound GL context + // however some drivers (Intel etc.) may FAIL doing this for unknown reason + if (IsCurrent()) + { + return Standard_True; + } + else if (wglMakeCurrent ((HDC )myWindowDC, (HGLRC )myGContext) != TRUE) + { + // notice that glGetError() couldn't be used here! + wchar_t* aMsgBuff = NULL; + DWORD anErrorCode = GetLastError(); + FormatMessageW (FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, + NULL, anErrorCode, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (wchar_t* )&aMsgBuff, 0, NULL); + if (aMsgBuff != NULL) + { + std::wcerr << L"OpenGL interface: wglMakeCurrent() failed. " << aMsgBuff << L" (" << int(anErrorCode) << L")\n"; + LocalFree (aMsgBuff); + } + else + { + std::wcerr << L"OpenGL interface: wglMakeCurrent() failed with #" << int(anErrorCode) << L" error code\n"; + } return Standard_False; } #else - if (myDisplay == NULL || myWindow == 0 || myGContext == 0 || - !glXMakeCurrent ((Display* )myDisplay, (GLXDrawable )myWindow, (GLXContext )myGContext)) + if (myDisplay == NULL || myWindow == 0 || myGContext == 0) + { + Standard_ProgramError_Raise_if (myIsInitialized, "OpenGl_Context::Init() should be called before!"); + return Standard_False; + } + + if (!glXMakeCurrent ((Display* )myDisplay, (GLXDrawable )myWindow, (GLXContext )myGContext)) { // if there is no current context it might be impossible to use glGetError() correctly //std::cerr << "glXMakeCurrent() failed!\n"; @@ -251,7 +303,7 @@ Standard_Boolean OpenGl_Context::Init (const Aspect_Drawable theWindow, #else myDisplay = theDisplay; #endif - if (myGContext == NULL) + if (myGContext == NULL || !MakeCurrent()) { return Standard_False; } diff --git a/src/OpenGl/OpenGl_Context.hxx b/src/OpenGl/OpenGl_Context.hxx index 9fa6dd8102..c054baa426 100644 --- a/src/OpenGl/OpenGl_Context.hxx +++ b/src/OpenGl/OpenGl_Context.hxx @@ -121,6 +121,11 @@ public: //! Clean up errors stack for this GL context (glGetError() in loop). Standard_EXPORT void ResetErrors(); + //! This method uses system-dependent API to retrieve information + //! about GL context bound to the current thread. + //! @return true if current thread is bound to this GL context + Standard_EXPORT Standard_Boolean IsCurrent() const; + //! Activates current context. //! Class should be initialized with appropriate info. Standard_EXPORT Standard_Boolean MakeCurrent(); diff --git a/src/OpenGl/OpenGl_GraphicDriver_Export.cxx b/src/OpenGl/OpenGl_GraphicDriver_Export.cxx index fbcfa9887a..75a9b8fab2 100755 --- a/src/OpenGl/OpenGl_GraphicDriver_Export.cxx +++ b/src/OpenGl/OpenGl_GraphicDriver_Export.cxx @@ -20,6 +20,8 @@ /************************************************************************/ #include +#include +#include #include #ifdef HAVE_CONFIG_H @@ -49,6 +51,14 @@ Standard_Boolean OpenGl_GraphicDriver::Export (const Standard_CString theFileNam const Standard_Address /*theProgressObject*/) { #ifdef HAVE_GL2PS + // gl2psBeginPage() will call OpenGL functions + // so we should activate correct GL context before redraw scene call + const OpenGl_CView* aCView = (const OpenGl_CView* )theView.ptrView; + if (aCView == NULL || !aCView->WS->GetGlContext()->MakeCurrent()) + { + return Standard_False; + } + Standard_Integer aFormat = -1; Standard_Integer aSortType = Graphic3d_ST_BSP_Tree; switch (theFormat) diff --git a/src/OpenGl/OpenGl_Window.cxx b/src/OpenGl/OpenGl_Window.cxx index e777b3b9cf..a2507668e1 100644 --- a/src/OpenGl/OpenGl_Window.cxx +++ b/src/OpenGl/OpenGl_Window.cxx @@ -400,8 +400,12 @@ OpenGl_Window::OpenGl_Window (const Handle(OpenGl_Display)& theDisplay, myWindow = aParent; #endif +#if (defined(_WIN32) || defined(__WIN32__)) + myGlContext->Init (myWindow, myWindowDC, myGContext); +#else + myGlContext->Init (myWindow, myDisplay->GetDisplay(), myGContext); +#endif Init(); - myGlContext->Init(); } // ======================================================================= @@ -463,28 +467,7 @@ OpenGl_Window::~OpenGl_Window() // ======================================================================= Standard_Boolean OpenGl_Window::Activate() { - DISPLAY* aDisp = (DISPLAY* )myDisplay->GetDisplay(); - if (aDisp == NULL) - return Standard_False; - -#if (defined(_WIN32) || defined(__WIN32__)) - if (!wglMakeCurrent (myWindowDC, myGContext)) - { - //GLenum errorcode = glGetError(); - //const GLubyte *errorstring = gluErrorString(errorcode); - //printf("wglMakeCurrent failed: %d %s\n", errorcode, errorstring); - return Standard_False; - } -#else - if (!glXMakeCurrent (aDisp, myWindow, myGContext)) - { - // if there is no current context it might be impossible to use glGetError correctly - //printf("glXMakeCurrent failed!\n"); - return Standard_False; - } -#endif - - return Standard_True; + return myGlContext->MakeCurrent(); } // =======================================================================