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;
|
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
|
// function : MakeCurrent
|
||||||
// purpose :
|
// purpose :
|
||||||
@ -109,17 +134,44 @@ OpenGl_Context::~OpenGl_Context()
|
|||||||
Standard_Boolean OpenGl_Context::MakeCurrent()
|
Standard_Boolean OpenGl_Context::MakeCurrent()
|
||||||
{
|
{
|
||||||
#if (defined(_WIN32) || defined(__WIN32__))
|
#if (defined(_WIN32) || defined(__WIN32__))
|
||||||
if (myWindowDC == NULL || myGContext == NULL ||
|
if (myWindowDC == NULL || myGContext == NULL)
|
||||||
!wglMakeCurrent ((HDC )myWindowDC, (HGLRC )myGContext))
|
|
||||||
{
|
{
|
||||||
//GLenum anErrCode = glGetError();
|
Standard_ProgramError_Raise_if (myIsInitialized, "OpenGl_Context::Init() should be called before!");
|
||||||
//const GLubyte* anErrorString = gluErrorString (anErrCode);
|
return Standard_False;
|
||||||
//std::cerr << "wglMakeCurrent() failed: " << anErrCode << " " << anErrorString << "\n";
|
}
|
||||||
|
|
||||||
|
// 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;
|
return Standard_False;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
if (myDisplay == NULL || myWindow == 0 || myGContext == 0 ||
|
if (myDisplay == NULL || myWindow == 0 || myGContext == 0)
|
||||||
!glXMakeCurrent ((Display* )myDisplay, (GLXDrawable )myWindow, (GLXContext )myGContext))
|
{
|
||||||
|
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
|
// if there is no current context it might be impossible to use glGetError() correctly
|
||||||
//std::cerr << "glXMakeCurrent() failed!\n";
|
//std::cerr << "glXMakeCurrent() failed!\n";
|
||||||
@ -251,7 +303,7 @@ Standard_Boolean OpenGl_Context::Init (const Aspect_Drawable theWindow,
|
|||||||
#else
|
#else
|
||||||
myDisplay = theDisplay;
|
myDisplay = theDisplay;
|
||||||
#endif
|
#endif
|
||||||
if (myGContext == NULL)
|
if (myGContext == NULL || !MakeCurrent())
|
||||||
{
|
{
|
||||||
return Standard_False;
|
return Standard_False;
|
||||||
}
|
}
|
||||||
|
@ -121,6 +121,11 @@ public:
|
|||||||
//! Clean up errors stack for this GL context (glGetError() in loop).
|
//! Clean up errors stack for this GL context (glGetError() in loop).
|
||||||
Standard_EXPORT void ResetErrors();
|
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.
|
//! Activates current context.
|
||||||
//! Class should be initialized with appropriate info.
|
//! Class should be initialized with appropriate info.
|
||||||
Standard_EXPORT Standard_Boolean MakeCurrent();
|
Standard_EXPORT Standard_Boolean MakeCurrent();
|
||||||
|
@ -20,6 +20,8 @@
|
|||||||
/************************************************************************/
|
/************************************************************************/
|
||||||
|
|
||||||
#include <OpenGl_GraphicDriver.hxx>
|
#include <OpenGl_GraphicDriver.hxx>
|
||||||
|
#include <OpenGl_Context.hxx>
|
||||||
|
#include <OpenGl_CView.hxx>
|
||||||
#include <OSD_Localizer.hxx>
|
#include <OSD_Localizer.hxx>
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
@ -49,6 +51,14 @@ Standard_Boolean OpenGl_GraphicDriver::Export (const Standard_CString theFileNam
|
|||||||
const Standard_Address /*theProgressObject*/)
|
const Standard_Address /*theProgressObject*/)
|
||||||
{
|
{
|
||||||
#ifdef HAVE_GL2PS
|
#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 aFormat = -1;
|
||||||
Standard_Integer aSortType = Graphic3d_ST_BSP_Tree;
|
Standard_Integer aSortType = Graphic3d_ST_BSP_Tree;
|
||||||
switch (theFormat)
|
switch (theFormat)
|
||||||
|
@ -400,8 +400,12 @@ OpenGl_Window::OpenGl_Window (const Handle(OpenGl_Display)& theDisplay,
|
|||||||
myWindow = aParent;
|
myWindow = aParent;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if (defined(_WIN32) || defined(__WIN32__))
|
||||||
|
myGlContext->Init (myWindow, myWindowDC, myGContext);
|
||||||
|
#else
|
||||||
|
myGlContext->Init (myWindow, myDisplay->GetDisplay(), myGContext);
|
||||||
|
#endif
|
||||||
Init();
|
Init();
|
||||||
myGlContext->Init();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
@ -463,28 +467,7 @@ OpenGl_Window::~OpenGl_Window()
|
|||||||
// =======================================================================
|
// =======================================================================
|
||||||
Standard_Boolean OpenGl_Window::Activate()
|
Standard_Boolean OpenGl_Window::Activate()
|
||||||
{
|
{
|
||||||
DISPLAY* aDisp = (DISPLAY* )myDisplay->GetDisplay();
|
return myGlContext->MakeCurrent();
|
||||||
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;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
|
Loading…
x
Reference in New Issue
Block a user