Add error handling logic to viewer's GLX context creation

This may prevent unrecoverable crashes when attempting to create a
compatibility profile with Intel drivers.

Bug: skia:
Change-Id: I24c1bfe01fed40026c9b467144f454846d6b5b34
Reviewed-on: https://skia-review.googlesource.com/151760
Commit-Queue: Brian Osman <brianosman@google.com>
Commit-Queue: Jim Van Verth <jvanverth@google.com>
Auto-Submit: Brian Osman <brianosman@google.com>
Reviewed-by: Jim Van Verth <jvanverth@google.com>
This commit is contained in:
Brian Osman 2018-09-05 12:40:53 -04:00 committed by Skia Commit-Bot
parent 32ade4cedf
commit 2e52460ff9

View File

@ -17,6 +17,12 @@ using sk_app::GLWindowContext;
namespace { namespace {
static bool gCtxErrorOccurred = false;
static int ctxErrorHandler(Display *dpy, XErrorEvent *ev) {
gCtxErrorOccurred = true;
return 0;
}
class GLWindowContext_xlib : public GLWindowContext { class GLWindowContext_xlib : public GLWindowContext {
public: public:
GLWindowContext_xlib(const XlibWindowInfo&, const DisplayParams&); GLWindowContext_xlib(const XlibWindowInfo&, const DisplayParams&);
@ -65,6 +71,9 @@ sk_sp<const GrGLInterface> GLWindowContext_xlib::onInitializeContext() {
CreateContextAttribsFn* createContextAttribs = (CreateContextAttribsFn*)glXGetProcAddressARB( CreateContextAttribsFn* createContextAttribs = (CreateContextAttribsFn*)glXGetProcAddressARB(
(const GLubyte*)"glXCreateContextAttribsARB"); (const GLubyte*)"glXCreateContextAttribsARB");
if (createContextAttribs && fFBConfig) { if (createContextAttribs && fFBConfig) {
// Install Xlib error handler that will set gCtxErrorOccurred
int (*oldHandler)(Display*, XErrorEvent*) = XSetErrorHandler(&ctxErrorHandler);
// Specifying 3.2 allows an arbitrarily high context version (so long as no 3.2 features // Specifying 3.2 allows an arbitrarily high context version (so long as no 3.2 features
// have been removed). // have been removed).
for (int minor = 2; minor >= 0 && !fGLContext; --minor) { for (int minor = 2; minor >= 0 && !fGLContext; --minor) {
@ -72,12 +81,18 @@ sk_sp<const GrGLInterface> GLWindowContext_xlib::onInitializeContext() {
// requires a core profile. Edit this code to use RenderDoc. // requires a core profile. Edit this code to use RenderDoc.
for (int profile : {GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB, for (int profile : {GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB,
GLX_CONTEXT_CORE_PROFILE_BIT_ARB}) { GLX_CONTEXT_CORE_PROFILE_BIT_ARB}) {
gCtxErrorOccurred = false;
int attribs[] = { int attribs[] = {
GLX_CONTEXT_MAJOR_VERSION_ARB, 3, GLX_CONTEXT_MINOR_VERSION_ARB, minor, GLX_CONTEXT_MAJOR_VERSION_ARB, 3, GLX_CONTEXT_MINOR_VERSION_ARB, minor,
GLX_CONTEXT_PROFILE_MASK_ARB, profile, GLX_CONTEXT_PROFILE_MASK_ARB, profile,
0 0
}; };
fGLContext = createContextAttribs(fDisplay, *fFBConfig, nullptr, True, attribs); fGLContext = createContextAttribs(fDisplay, *fFBConfig, nullptr, True, attribs);
// Sync to ensure any errors generated are processed.
XSync(fDisplay, False);
if (gCtxErrorOccurred) { continue; }
if (fGLContext && profile == GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB && if (fGLContext && profile == GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB &&
glXMakeCurrent(fDisplay, fWindow, fGLContext)) { glXMakeCurrent(fDisplay, fWindow, fGLContext)) {
current = true; current = true;
@ -97,6 +112,8 @@ sk_sp<const GrGLInterface> GLWindowContext_xlib::onInitializeContext() {
} }
} }
} }
// Restore the original error handler
XSetErrorHandler(oldHandler);
} }
if (!fGLContext) { if (!fGLContext) {
fGLContext = glXCreateContext(fDisplay, fVisualInfo, nullptr, GL_TRUE); fGLContext = glXCreateContext(fDisplay, fVisualInfo, nullptr, GL_TRUE);