mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-05 18:16:23 +03:00
0023172: Added workaround for BUGs in Intel OpenGL drivers
Check GL context already bound before wglMakeCurrent() call.
This commit is contained in:
parent
90dc2e5b07
commit
86fa64d971
@ -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;
|
||||
}
|
||||
|
@ -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();
|
||||
|
@ -20,6 +20,8 @@
|
||||
/************************************************************************/
|
||||
|
||||
#include <OpenGl_GraphicDriver.hxx>
|
||||
#include <OpenGl_Context.hxx>
|
||||
#include <OpenGl_CView.hxx>
|
||||
#include <OSD_Localizer.hxx>
|
||||
|
||||
#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)
|
||||
|
@ -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();
|
||||
}
|
||||
|
||||
// =======================================================================
|
||||
|
Loading…
x
Reference in New Issue
Block a user