From a08bfd9891992710a43db882c01a7131ae732fd8 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 3 Jun 2020 19:46:22 +0100 Subject: [PATCH 01/37] User Contexts API with Win32 implementation. --- include/GLFW/glfw3.h | 27 ++++++++++++++++++++ src/context.c | 23 +++++++++++++++++ src/internal.h | 6 +++++ src/wgl_context.c | 56 +++++++++++++++++++++++++++++++++++++++++ src/wgl_context.h | 5 ++++ tests/CMakeLists.txt | 3 ++- tests/usercontext.c | 60 ++++++++++++++++++++++++++++++++++++++++++++ 7 files changed, 179 insertions(+), 1 deletion(-) create mode 100644 tests/usercontext.c diff --git a/include/GLFW/glfw3.h b/include/GLFW/glfw3.h index 6db5b579..afc6ca0c 100644 --- a/include/GLFW/glfw3.h +++ b/include/GLFW/glfw3.h @@ -1276,6 +1276,18 @@ typedef struct GLFWmonitor GLFWmonitor; */ typedef struct GLFWwindow GLFWwindow; +/*! @brief Opaque user OpenGL context object. + * + * Opaque user OpenGL context object. + * + * @see @ref user_gl_context + * + * @since Added in version 3.3. + * + * @ingroup window + */ +typedef struct GLFWusercontext GLFWusercontext; + /*! @brief Opaque cursor object. * * Opaque cursor object. @@ -5772,6 +5784,21 @@ GLFWAPI int glfwExtensionSupported(const char* extension); */ GLFWAPI GLFWglproc glfwGetProcAddress(const char* procname); +/*! @brief Create a new OpenGL user context for a window +* +*/ +GLFWAPI GLFWusercontext* glfwCreateUserContext(GLFWwindow* window); + +/*! @brief Delete an OpenGL user context +* +*/ +GLFWAPI void glfwDestroyUserContext(GLFWusercontext* context); + +/*! @brief Make an OpenGL user context +* +*/ +GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* context); + /*! @brief Returns whether the Vulkan loader and an ICD have been found. * * This function returns whether the Vulkan loader and any minimally functional diff --git a/src/context.c b/src/context.c index 48311e5f..c6a808cb 100644 --- a/src/context.c +++ b/src/context.c @@ -758,3 +758,26 @@ GLFWAPI GLFWglproc glfwGetProcAddress(const char* procname) return window->context.getProcAddress(procname); } +GLFWAPI GLFWusercontext* glfwCreateUserContext(GLFWwindow* handle) +{ + _GLFWusercontext* context; + _GLFWwindow* window = (_GLFWwindow*)handle; + + context = _glfwPlatformCreateUserContext(window); + + return (GLFWusercontext*)context; +} + +GLFWAPI void glfwDestroyUserContext(GLFWusercontext* handle) +{ + _GLFWusercontext* context = (_GLFWusercontext*)handle; + + _glfwPlatformDestroyUserContext(context); +} + +GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* handle) +{ + _GLFWusercontext* context = (_GLFWusercontext*)handle; + + _glfwPlatformMakeUserContextCurrent(context); +} diff --git a/src/internal.h b/src/internal.h index 6d7587c8..a16c8a8c 100644 --- a/src/internal.h +++ b/src/internal.h @@ -75,6 +75,7 @@ typedef struct _GLFWmapping _GLFWmapping; typedef struct _GLFWjoystick _GLFWjoystick; typedef struct _GLFWtls _GLFWtls; typedef struct _GLFWmutex _GLFWmutex; +typedef struct _GLFWusercontext _GLFWusercontext; typedef void (* _GLFWmakecontextcurrentfun)(_GLFWwindow*); typedef void (* _GLFWswapbuffersfun)(_GLFWwindow*); @@ -681,6 +682,11 @@ void _glfwPlatformWaitEvents(void); void _glfwPlatformWaitEventsTimeout(double timeout); void _glfwPlatformPostEmptyEvent(void); +_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window); +void _glfwPlatformDestroyUserContext(_GLFWusercontext* context); +void _glfwPlatformMakeUserContextCurrent(_GLFWusercontext* context); + + void _glfwPlatformGetRequiredInstanceExtensions(char** extensions); int _glfwPlatformGetPhysicalDevicePresentationSupport(VkInstance instance, VkPhysicalDevice device, diff --git a/src/wgl_context.c b/src/wgl_context.c index 4f9a6ffe..16bd167c 100644 --- a/src/wgl_context.c +++ b/src/wgl_context.c @@ -776,6 +776,62 @@ GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, #undef setAttrib +////////////////////////////////////////////////////////////////////////// +////// GLFW platform API ////// +////////////////////////////////////////////////////////////////////////// + +_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) +{ + _GLFWusercontext* context; + context = calloc(1, sizeof(_GLFWusercontext)); + + context->handle = wglCreateContext(window->context.wgl.dc); + context->window = window; + if (!context->handle) + { + _glfwInputErrorWin32(GLFW_VERSION_UNAVAILABLE, + "WGL: Failed to create user OpenGL context"); + free(context); + return GLFW_FALSE; + } + + if (!wglShareLists(window->context.wgl.handle,context->handle)) + { + _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, + "WGL: Failed to enable sharing with window OpenGL context and user context"); + free(context); + return GLFW_FALSE; + } + return context; +} + +void _glfwPlatformDestroyUserContext(_GLFWusercontext* context) +{ + wglDeleteContext(context->handle); + free(context); +} + +void _glfwPlatformMakeUserContextCurrent(_GLFWusercontext* context) +{ + if(context) + { + if(!wglMakeCurrent(context->window->context.wgl.dc,context->handle)) + { + _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, + "WGL: Failed to set current user context"); + } + } + else + { + if (!wglMakeCurrent(NULL, NULL)) + { + _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, + "WGL: Failed to clear current context"); + } + } +} + + ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// ////////////////////////////////////////////////////////////////////////// diff --git a/src/wgl_context.h b/src/wgl_context.h index 2cf7e4e5..6cb6d050 100644 --- a/src/wgl_context.h +++ b/src/wgl_context.h @@ -151,6 +151,11 @@ typedef struct _GLFWlibraryWGL } _GLFWlibraryWGL; +typedef struct _GLFWusercontext +{ + _GLFWwindow* window; + HGLRC handle; +} _GLFWusercontext; GLFWbool _glfwInitWGL(void); void _glfwTerminateWGL(void); diff --git a/tests/CMakeLists.txt b/tests/CMakeLists.txt index e36ed56c..d79e5b87 100644 --- a/tests/CMakeLists.txt +++ b/tests/CMakeLists.txt @@ -28,6 +28,7 @@ add_executable(iconify iconify.c ${GETOPT} ${GLAD_GL}) add_executable(monitors monitors.c ${GETOPT} ${GLAD_GL}) add_executable(reopen reopen.c ${GLAD_GL}) add_executable(cursor cursor.c ${GLAD_GL}) +add_executable(usercontext usercontext.c ${GLAD_GL}) add_executable(empty WIN32 MACOSX_BUNDLE empty.c ${TINYCTHREAD} ${GLAD_GL}) add_executable(gamma WIN32 MACOSX_BUNDLE gamma.c ${GLAD_GL}) @@ -52,7 +53,7 @@ endif() set(GUI_ONLY_BINARIES empty gamma icon inputlag joysticks opacity tearing threads timeout title triangle-vulkan windows) set(CONSOLE_BINARIES clipboard events msaa glfwinfo iconify monitors reopen - cursor) + cursor usercontext) set_target_properties(${GUI_ONLY_BINARIES} ${CONSOLE_BINARIES} PROPERTIES C_STANDARD 99 diff --git a/tests/usercontext.c b/tests/usercontext.c new file mode 100644 index 00000000..58916430 --- /dev/null +++ b/tests/usercontext.c @@ -0,0 +1,60 @@ +#include +#include + +int main(void) +{ + GLFWwindow* window; + GLFWusercontext* usercontext; + + /* Initialize the library */ + if (!glfwInit()) + return -1; + + /* Create a windowed mode window and its OpenGL context */ + window = glfwCreateWindow(640, 480, "User Context", NULL, NULL); + if (!window) + { + glfwTerminate(); + return -1; + } + + /* Make the window's context current */ + glfwMakeContextCurrent(window); + gladLoadGL(glfwGetProcAddress); + + /* make a new context */ + usercontext = glfwCreateUserContext(window); + if (!usercontext) + { + glfwTerminate(); + return -1; + } + + /* set the user context current */ + glfwMakeContextCurrent(NULL); + glfwMakeUserContextCurrent(usercontext); + + /* set the window context current */ + glfwMakeUserContextCurrent(NULL); + glfwMakeContextCurrent(window); + + glClearColor( 0.4f, 0.3f, 0.4f, 0.0f ); + + + /* Loop until the user closes the window */ + while (!glfwWindowShouldClose(window)) + { + /* Render here */ + glClear(GL_COLOR_BUFFER_BIT); + + /* Swap front and back buffers */ + glfwSwapBuffers(window); + + /* Poll for and process events */ + glfwPollEvents(); + } + + glfwDestroyUserContext(usercontext); + glfwTerminate(); + return 0; +} \ No newline at end of file From 0ae4eb4d26e4fe857ace987bee310334859689e9 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Sun, 12 Jul 2020 12:21:18 +0100 Subject: [PATCH 02/37] Improved User Context Win32 implementation _glfwCreateContextWGL and _glfwPlatformCreateUserContext now use new function _glfwCreateContextForDCWGL. --- src/wgl_context.c | 112 +++++++++++++++++++++++++--------------------- src/wgl_context.h | 2 + 2 files changed, 64 insertions(+), 50 deletions(-) diff --git a/src/wgl_context.c b/src/wgl_context.c index 16bd167c..5a57aed7 100644 --- a/src/wgl_context.c +++ b/src/wgl_context.c @@ -538,46 +538,17 @@ void _glfwTerminateWGL(void) attribs[index++] = v; \ } -// Create the OpenGL or OpenGL ES context +// Create the OpenGL or OpenGL ES context for the given HDC // -GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, - const _GLFWctxconfig* ctxconfig, - const _GLFWfbconfig* fbconfig) +GLFWbool _glfwCreateContextForDCWGL(HDC dc, const _GLFWctxconfig* ctxconfig, HGLRC* context) { int attribs[40]; - int pixelFormat; - PIXELFORMATDESCRIPTOR pfd; HGLRC share = NULL; + *context = NULL; if (ctxconfig->share) share = ctxconfig->share->context.wgl.handle; - window->context.wgl.dc = GetDC(window->win32.handle); - if (!window->context.wgl.dc) - { - _glfwInputError(GLFW_PLATFORM_ERROR, - "WGL: Failed to retrieve DC for window"); - return GLFW_FALSE; - } - - pixelFormat = choosePixelFormat(window, ctxconfig, fbconfig); - if (!pixelFormat) - return GLFW_FALSE; - - if (!DescribePixelFormat(window->context.wgl.dc, - pixelFormat, sizeof(pfd), &pfd)) - { - _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, - "WGL: Failed to retrieve PFD for selected pixel format"); - return GLFW_FALSE; - } - - if (!SetPixelFormat(window->context.wgl.dc, pixelFormat, &pfd)) - { - _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, - "WGL: Failed to set selected pixel format"); - return GLFW_FALSE; - } if (ctxconfig->client == GLFW_OPENGL_API) { @@ -692,9 +663,9 @@ GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, setAttrib(0, 0); - window->context.wgl.handle = - wglCreateContextAttribsARB(window->context.wgl.dc, share, attribs); - if (!window->context.wgl.handle) + *context = + wglCreateContextAttribsARB(dc, share, attribs); + if (!(*context)) { const DWORD error = GetLastError(); @@ -744,8 +715,8 @@ GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, } else { - window->context.wgl.handle = wglCreateContext(window->context.wgl.dc); - if (!window->context.wgl.handle) + *context = wglCreateContext(dc); + if (!(*context) ) { _glfwInputErrorWin32(GLFW_VERSION_UNAVAILABLE, "WGL: Failed to create OpenGL context"); @@ -754,7 +725,7 @@ GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, if (share) { - if (!wglShareLists(share, window->context.wgl.handle)) + if (!wglShareLists(share, *context)) { _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, "WGL: Failed to enable sharing with specified OpenGL context"); @@ -763,6 +734,50 @@ GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, } } + return GLFW_TRUE; +} + +// Create the OpenGL or OpenGL ES context +// +GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, + const _GLFWctxconfig* ctxconfig, + const _GLFWfbconfig* fbconfig) +{ + int pixelFormat; + PIXELFORMATDESCRIPTOR pfd; + + window->context.wgl.dc = GetDC(window->win32.handle); + if (!window->context.wgl.dc) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "WGL: Failed to retrieve DC for window"); + return GLFW_FALSE; + } + + pixelFormat = choosePixelFormat(window, ctxconfig, fbconfig); + if (!pixelFormat) + return GLFW_FALSE; + + if (!DescribePixelFormat(window->context.wgl.dc, + pixelFormat, sizeof(pfd), &pfd)) + { + _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, + "WGL: Failed to retrieve PFD for selected pixel format"); + return GLFW_FALSE; + } + + if (!SetPixelFormat(window->context.wgl.dc, pixelFormat, &pfd)) + { + _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, + "WGL: Failed to set selected pixel format"); + return GLFW_FALSE; + } + + if(!_glfwCreateContextForDCWGL( window->context.wgl.dc, ctxconfig, &window->context.wgl.handle )) + { + return GLFW_FALSE; + } + window->context.makeCurrent = makeContextCurrentWGL; window->context.swapBuffers = swapBuffersWGL; window->context.swapInterval = swapIntervalWGL; @@ -783,25 +798,22 @@ GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, _GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) { _GLFWusercontext* context; + _GLFWctxconfig ctxconfig; + context = calloc(1, sizeof(_GLFWusercontext)); - - context->handle = wglCreateContext(window->context.wgl.dc); context->window = window; - if (!context->handle) - { - _glfwInputErrorWin32(GLFW_VERSION_UNAVAILABLE, - "WGL: Failed to create user OpenGL context"); - free(context); - return GLFW_FALSE; - } - if (!wglShareLists(window->context.wgl.handle,context->handle)) + ctxconfig = _glfw.hints.context; + ctxconfig.share = window; + + if (!_glfwCreateContextForDCWGL(window->context.wgl.dc, &ctxconfig, &context->handle)) { _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, - "WGL: Failed to enable sharing with window OpenGL context and user context"); + "WGL: Failed to create user OpenGL context"); free(context); - return GLFW_FALSE; + return NULL; } + return context; } diff --git a/src/wgl_context.h b/src/wgl_context.h index 6cb6d050..abf97559 100644 --- a/src/wgl_context.h +++ b/src/wgl_context.h @@ -151,6 +151,8 @@ typedef struct _GLFWlibraryWGL } _GLFWlibraryWGL; +// WGL-specific user context data +// typedef struct _GLFWusercontext { _GLFWwindow* window; From 998036654c23a5b1388be66463d271a5ec5b25b8 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Sun, 12 Jul 2020 20:28:10 +0100 Subject: [PATCH 03/37] GLX user context implementation --- src/glx_context.c | 108 +++++++++++++++++++++++++++++++++++++--------- src/glx_context.h | 9 ++++ 2 files changed, 96 insertions(+), 21 deletions(-) diff --git a/src/glx_context.c b/src/glx_context.c index 7ccd1374..043c42ba 100644 --- a/src/glx_context.c +++ b/src/glx_context.c @@ -440,26 +440,19 @@ void _glfwTerminateGLX(void) attribs[index++] = v; \ } -// Create the OpenGL or OpenGL ES context + +// Create the OpenGL or OpenGL ES context for the window fbConfig // -GLFWbool _glfwCreateContextGLX(_GLFWwindow* window, - const _GLFWctxconfig* ctxconfig, - const _GLFWfbconfig* fbconfig) +GLFWbool _glfwCreateContextForFBGLX(_GLFWwindow* window, + const _GLFWctxconfig* ctxconfig, + GLXContext* context) { int attribs[40]; - GLXFBConfig native = NULL; GLXContext share = NULL; if (ctxconfig->share) share = ctxconfig->share->context.glx.handle; - if (!chooseGLXFBConfig(fbconfig, &native)) - { - _glfwInputError(GLFW_FORMAT_UNAVAILABLE, - "GLX: Failed to find a suitable GLXFBConfig"); - return GLFW_FALSE; - } - if (ctxconfig->client == GLFW_OPENGL_ES_API) { if (!_glfw.glx.ARB_create_context || @@ -574,9 +567,9 @@ GLFWbool _glfwCreateContextGLX(_GLFWwindow* window, setAttrib(None, None); - window->context.glx.handle = + *context = _glfw.glx.CreateContextAttribsARB(_glfw.x11.display, - native, + window->context.glx.fbconfig, share, True, attribs); @@ -585,34 +578,56 @@ GLFWbool _glfwCreateContextGLX(_GLFWwindow* window, // implementation of GLX_ARB_create_context_profile that fail // default 1.0 context creation with a GLXBadProfileARB error in // violation of the extension spec - if (!window->context.glx.handle) + if (!(*context)) { if (_glfw.x11.errorCode == _glfw.glx.errorBase + GLXBadProfileARB && ctxconfig->client == GLFW_OPENGL_API && ctxconfig->profile == GLFW_OPENGL_ANY_PROFILE && ctxconfig->forward == GLFW_FALSE) { - window->context.glx.handle = - createLegacyContextGLX(window, native, share); + *context = + createLegacyContextGLX(window, window->context.glx.fbconfig, share); } } } else { - window->context.glx.handle = - createLegacyContextGLX(window, native, share); + *context = + createLegacyContextGLX(window, window->context.glx.fbconfig, share); } _glfwReleaseErrorHandlerX11(); - if (!window->context.glx.handle) + if (!(*context)) { _glfwInputErrorX11(GLFW_VERSION_UNAVAILABLE, "GLX: Failed to create context"); return GLFW_FALSE; } + return GLFW_TRUE; +} + +// Create the OpenGL or OpenGL ES context +// +GLFWbool _glfwCreateContextGLX(_GLFWwindow* window, + const _GLFWctxconfig* ctxconfig, + const _GLFWfbconfig* fbconfig) +{ + + if (!chooseGLXFBConfig(fbconfig, &window->context.glx.fbconfig)) + { + _glfwInputError(GLFW_FORMAT_UNAVAILABLE, + "GLX: Failed to find a suitable GLXFBConfig"); + return GLFW_FALSE; + } + + if(!_glfwCreateContextForFBGLX(window,ctxconfig,&window->context.glx.handle)) + { + return GLFW_FALSE; + } + window->context.glx.window = - glXCreateWindow(_glfw.x11.display, native, window->x11.handle, NULL); + glXCreateWindow(_glfw.x11.display, window->context.glx.fbconfig, window->x11.handle, NULL); if (!window->context.glx.window) { _glfwInputError(GLFW_PLATFORM_ERROR, "GLX: Failed to create window"); @@ -663,6 +678,57 @@ GLFWbool _glfwChooseVisualGLX(const _GLFWwndconfig* wndconfig, return GLFW_TRUE; } +////////////////////////////////////////////////////////////////////////// +////// GLFW platform API ////// +////////////////////////////////////////////////////////////////////////// + +_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) +{ + _GLFWusercontext* context; + _GLFWctxconfig ctxconfig; + + context = calloc(1, sizeof(_GLFWusercontext)); + context->window = window; + + ctxconfig = _glfw.hints.context; + ctxconfig.share = window; + + if(!_glfwCreateContextForFBGLX(window,&ctxconfig,&context->handle)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "GLX: Failed to create user OpenGL context"); + free(context); + return NULL; + } + + return context; +} + +void _glfwPlatformDestroyUserContext(_GLFWusercontext* context) +{ + glXDestroyContext(_glfw.x11.display, context->handle); + free(context); +} + +void _glfwPlatformMakeUserContextCurrent(_GLFWusercontext* context) +{ + if(context) + { + if(!glXMakeCurrent(_glfw.x11.display, context->window->context.glx.window,context->handle)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "GLX: Failed to set current user context"); + } + } + else + { + if (!glXMakeCurrent(_glfw.x11.display, None, NULL)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "GLX: Failed to clear current context"); + } + } +} ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// diff --git a/src/glx_context.h b/src/glx_context.h index 94f07e2e..6826922a 100644 --- a/src/glx_context.h +++ b/src/glx_context.h @@ -115,6 +115,7 @@ typedef GLXContext (*PFNGLXCREATECONTEXTATTRIBSARBPROC)(Display*,GLXFBConfig,GLX // typedef struct _GLFWcontextGLX { + GLXFBConfig fbconfig; GLXContext handle; GLXWindow window; @@ -168,6 +169,14 @@ typedef struct _GLFWlibraryGLX } _GLFWlibraryGLX; +// GLX-specific user context data +// +typedef struct _GLFWusercontext +{ + _GLFWwindow* window; + GLXContext handle; +} _GLFWusercontext; + GLFWbool _glfwInitGLX(void); void _glfwTerminateGLX(void); GLFWbool _glfwCreateContextGLX(_GLFWwindow* window, From 5e94092263caec509951698f4ffb8effa4bada4c Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Mon, 13 Jul 2020 18:06:48 +0100 Subject: [PATCH 04/37] Refactor user context implementation to use the standard GLFW platform / context approach --- src/context.c | 8 +++--- src/egl_context.c | 7 +++++ src/egl_context.h | 10 +++++++ src/glx_context.c | 68 ++++++++++++++++++++++---------------------- src/glx_context.h | 16 ++++++----- src/internal.h | 22 ++++++++++++-- src/osmesa_context.c | 5 ++++ src/osmesa_context.h | 9 +++++- src/wgl_context.c | 68 +++++++++++++++++++++----------------------- src/wgl_context.h | 9 ++++-- src/win32_window.c | 17 +++++++++++ src/x11_window.c | 18 ++++++++++++ 12 files changed, 171 insertions(+), 86 deletions(-) diff --git a/src/context.c b/src/context.c index c6a808cb..2ec29c25 100644 --- a/src/context.c +++ b/src/context.c @@ -771,13 +771,13 @@ GLFWAPI GLFWusercontext* glfwCreateUserContext(GLFWwindow* handle) GLFWAPI void glfwDestroyUserContext(GLFWusercontext* handle) { _GLFWusercontext* context = (_GLFWusercontext*)handle; - - _glfwPlatformDestroyUserContext(context); + if (context) + context->destroy(context); } GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* handle) { _GLFWusercontext* context = (_GLFWusercontext*)handle; - - _glfwPlatformMakeUserContextCurrent(context); + if (context) + context->makeCurrent(context); } diff --git a/src/egl_context.c b/src/egl_context.c index 533ed8e7..b059c4d9 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -802,6 +802,13 @@ GLFWbool _glfwChooseVisualEGL(const _GLFWwndconfig* wndconfig, } #endif // _GLFW_X11 +_GLFWusercontext* _glfwCreateUserContextEGL(_GLFWwindow* window) +{ + // TODO + return NULL; +} + + ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// diff --git a/src/egl_context.h b/src/egl_context.h index 9de424c9..1dcab099 100644 --- a/src/egl_context.h +++ b/src/egl_context.h @@ -164,6 +164,14 @@ typedef struct _GLFWcontextEGL } _GLFWcontextEGL; +// EGL-specific per user context data +// +typedef struct _GLFWusercontextEGL +{ + EGLContext handle; +} _GLFWusercontextEGL; + + // EGL-specific global data // typedef struct _GLFWlibraryEGL @@ -218,6 +226,8 @@ void _glfwTerminateEGL(void); GLFWbool _glfwCreateContextEGL(_GLFWwindow* window, const _GLFWctxconfig* ctxconfig, const _GLFWfbconfig* fbconfig); +_GLFWusercontext* _glfwCreateUserContextEGL(_GLFWwindow* window); + #if defined(_GLFW_X11) GLFWbool _glfwChooseVisualEGL(const _GLFWwndconfig* wndconfig, const _GLFWctxconfig* ctxconfig, diff --git a/src/glx_context.c b/src/glx_context.c index 043c42ba..e58584ee 100644 --- a/src/glx_context.c +++ b/src/glx_context.c @@ -678,43 +678,11 @@ GLFWbool _glfwChooseVisualGLX(const _GLFWwndconfig* wndconfig, return GLFW_TRUE; } -////////////////////////////////////////////////////////////////////////// -////// GLFW platform API ////// -////////////////////////////////////////////////////////////////////////// - -_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) -{ - _GLFWusercontext* context; - _GLFWctxconfig ctxconfig; - - context = calloc(1, sizeof(_GLFWusercontext)); - context->window = window; - - ctxconfig = _glfw.hints.context; - ctxconfig.share = window; - - if(!_glfwCreateContextForFBGLX(window,&ctxconfig,&context->handle)) - { - _glfwInputError(GLFW_PLATFORM_ERROR, - "GLX: Failed to create user OpenGL context"); - free(context); - return NULL; - } - - return context; -} - -void _glfwPlatformDestroyUserContext(_GLFWusercontext* context) -{ - glXDestroyContext(_glfw.x11.display, context->handle); - free(context); -} - -void _glfwPlatformMakeUserContextCurrent(_GLFWusercontext* context) +static void _glfwMakeUserContextCurrentGLX(_GLFWusercontext* context) { if(context) { - if(!glXMakeCurrent(_glfw.x11.display, context->window->context.glx.window,context->handle)) + if(!glXMakeCurrent(_glfw.x11.display, context->window->context.glx.window,context->glx.handle)) { _glfwInputError(GLFW_PLATFORM_ERROR, "GLX: Failed to set current user context"); @@ -730,6 +698,38 @@ void _glfwPlatformMakeUserContextCurrent(_GLFWusercontext* context) } } +static void _glfwDestroyUserContextGLX(_GLFWusercontext* context) +{ + glXDestroyContext(_glfw.x11.display, context->glx.handle); + free(context); +} + +_GLFWusercontext* _glfwCreateUserContextGLX(_GLFWwindow* window) +{ + _GLFWusercontext* context; + _GLFWctxconfig ctxconfig; + + context = calloc(1, sizeof(_GLFWusercontext)); + context->window = window; + + ctxconfig = _glfw.hints.context; + ctxconfig.share = window; + + if(!_glfwCreateContextForFBGLX(window,&ctxconfig,&context->glx.handle)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "GLX: Failed to create user OpenGL context"); + free(context); + return NULL; + } + + context->makeCurrent = _glfwMakeUserContextCurrentGLX; + context->destroy = _glfwDestroyUserContextGLX; + + + return context; +} + ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// ////////////////////////////////////////////////////////////////////////// diff --git a/src/glx_context.h b/src/glx_context.h index 6826922a..4046d887 100644 --- a/src/glx_context.h +++ b/src/glx_context.h @@ -109,6 +109,7 @@ typedef GLXContext (*PFNGLXCREATECONTEXTATTRIBSARBPROC)(Display*,GLXFBConfig,GLX #define _GLFW_PLATFORM_CONTEXT_STATE _GLFWcontextGLX glx #define _GLFW_PLATFORM_LIBRARY_CONTEXT_STATE _GLFWlibraryGLX glx +#define _GLFW_PLATFORM_USER_CONTEXT_STATE _GLFWusercontextGLX glx // GLX-specific per-context data @@ -121,6 +122,13 @@ typedef struct _GLFWcontextGLX } _GLFWcontextGLX; +// GLX-specific per user context data +// +typedef struct _GLFWusercontextGLX +{ + GLXContext handle; +} _GLFWusercontextGLX; + // GLX-specific global data // typedef struct _GLFWlibraryGLX @@ -169,13 +177,6 @@ typedef struct _GLFWlibraryGLX } _GLFWlibraryGLX; -// GLX-specific user context data -// -typedef struct _GLFWusercontext -{ - _GLFWwindow* window; - GLXContext handle; -} _GLFWusercontext; GLFWbool _glfwInitGLX(void); void _glfwTerminateGLX(void); @@ -187,4 +188,5 @@ GLFWbool _glfwChooseVisualGLX(const _GLFWwndconfig* wndconfig, const _GLFWctxconfig* ctxconfig, const _GLFWfbconfig* fbconfig, Visual** visual, int* depth); +_GLFWusercontext* _glfwCreateUserContextGLX(_GLFWwindow* window); diff --git a/src/internal.h b/src/internal.h index 04cc4935..bf99478e 100644 --- a/src/internal.h +++ b/src/internal.h @@ -83,6 +83,8 @@ typedef void (* _GLFWswapintervalfun)(int); typedef int (* _GLFWextensionsupportedfun)(const char*); typedef GLFWglproc (* _GLFWgetprocaddressfun)(const char*); typedef void (* _GLFWdestroycontextfun)(_GLFWwindow*); +typedef void (* _GLFWmakeusercontextcurrentfun)(_GLFWusercontext* context); +typedef void (* _GLFWdestroyusercontextfun)(_GLFWusercontext* context); #define GL_VERSION 0x1f02 #define GL_NONE 0 @@ -369,6 +371,24 @@ struct _GLFWcontext _GLFWcontextOSMesa osmesa; }; + +// User Context structure +// +struct _GLFWusercontext +{ + _GLFWwindow* window; + + // This is defined in the context API's context.h + _GLFW_PLATFORM_USER_CONTEXT_STATE; + // This is defined in egl_context.h + _GLFWusercontextEGL egl; + // This is defined in osmesa_context.h + _GLFWusercontextOSMesa osmesa; + + _GLFWmakeusercontextcurrentfun makeCurrent; + _GLFWdestroyusercontextfun destroy; +}; + // Window and context structure // struct _GLFWwindow @@ -687,8 +707,6 @@ void _glfwPlatformWaitEventsTimeout(double timeout); void _glfwPlatformPostEmptyEvent(void); _GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window); -void _glfwPlatformDestroyUserContext(_GLFWusercontext* context); -void _glfwPlatformMakeUserContextCurrent(_GLFWusercontext* context); EGLenum _glfwPlatformGetEGLPlatform(EGLint** attribs); EGLNativeDisplayType _glfwPlatformGetEGLNativeDisplay(void); diff --git a/src/osmesa_context.c b/src/osmesa_context.c index 70e8675b..07baabed 100644 --- a/src/osmesa_context.c +++ b/src/osmesa_context.c @@ -287,6 +287,11 @@ GLFWbool _glfwCreateContextOSMesa(_GLFWwindow* window, #undef setAttrib +_GLFWusercontext* _glfwCreateUserContextOSMesa(_GLFWwindow* window) +{ + // TODO + return NULL; +} ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// diff --git a/src/osmesa_context.h b/src/osmesa_context.h index ce1f1a29..7a20c1f8 100644 --- a/src/osmesa_context.h +++ b/src/osmesa_context.h @@ -65,6 +65,13 @@ typedef struct _GLFWcontextOSMesa } _GLFWcontextOSMesa; +// OSMesa-specific per user context data +// +typedef struct _GLFWusercontextOSMesa +{ + OSMesaContext handle; +} _GLFWusercontextOSMesa; + // OSMesa-specific global data // typedef struct _GLFWlibraryOSMesa @@ -87,4 +94,4 @@ void _glfwTerminateOSMesa(void); GLFWbool _glfwCreateContextOSMesa(_GLFWwindow* window, const _GLFWctxconfig* ctxconfig, const _GLFWfbconfig* fbconfig); - +_GLFWusercontext* _glfwCreateUserContextOSMesa(_GLFWwindow* window); diff --git a/src/wgl_context.c b/src/wgl_context.c index 5a57aed7..d6d668ad 100644 --- a/src/wgl_context.c +++ b/src/wgl_context.c @@ -790,44 +790,11 @@ GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, #undef setAttrib - -////////////////////////////////////////////////////////////////////////// -////// GLFW platform API ////// -////////////////////////////////////////////////////////////////////////// - -_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) -{ - _GLFWusercontext* context; - _GLFWctxconfig ctxconfig; - - context = calloc(1, sizeof(_GLFWusercontext)); - context->window = window; - - ctxconfig = _glfw.hints.context; - ctxconfig.share = window; - - if (!_glfwCreateContextForDCWGL(window->context.wgl.dc, &ctxconfig, &context->handle)) - { - _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, - "WGL: Failed to create user OpenGL context"); - free(context); - return NULL; - } - - return context; -} - -void _glfwPlatformDestroyUserContext(_GLFWusercontext* context) -{ - wglDeleteContext(context->handle); - free(context); -} - -void _glfwPlatformMakeUserContextCurrent(_GLFWusercontext* context) +static void _glfwMakeUserContextCurrentWGL(_GLFWusercontext* context) { if(context) { - if(!wglMakeCurrent(context->window->context.wgl.dc,context->handle)) + if(!wglMakeCurrent(context->window->context.wgl.dc,context->wgl.handle)) { _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, "WGL: Failed to set current user context"); @@ -843,6 +810,37 @@ void _glfwPlatformMakeUserContextCurrent(_GLFWusercontext* context) } } +static void _glfwDestroyUserContextWGL(_GLFWusercontext* context) +{ + wglDeleteContext(context->wgl.handle); + free(context); +} + +_GLFWusercontext* _glfwCreateUserContextWGL(_GLFWwindow* window) +{ + _GLFWusercontext* context; + _GLFWctxconfig ctxconfig; + + context = calloc(1, sizeof(_GLFWusercontext)); + context->window = window; + + ctxconfig = _glfw.hints.context; + ctxconfig.share = window; + + if (!_glfwCreateContextForDCWGL(window->context.wgl.dc, &ctxconfig, &context->wgl.handle)) + { + _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, + "WGL: Failed to create user OpenGL context"); + free(context); + return NULL; + } + + context->makeCurrent = _glfwMakeUserContextCurrentWGL; + context->destroy = _glfwDestroyUserContextWGL; + + return context; +} + ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// diff --git a/src/wgl_context.h b/src/wgl_context.h index abf97559..d04cd661 100644 --- a/src/wgl_context.h +++ b/src/wgl_context.h @@ -106,6 +106,7 @@ typedef BOOL (WINAPI * PFN_wglShareLists)(HGLRC,HGLRC); #define _GLFW_PLATFORM_CONTEXT_STATE _GLFWcontextWGL wgl #define _GLFW_PLATFORM_LIBRARY_CONTEXT_STATE _GLFWlibraryWGL wgl +#define _GLFW_PLATFORM_USER_CONTEXT_STATE _GLFWusercontextWGL wgl // WGL-specific per-context data @@ -153,15 +154,17 @@ typedef struct _GLFWlibraryWGL // WGL-specific user context data // -typedef struct _GLFWusercontext +typedef struct _GLFWusercontextWGL { - _GLFWwindow* window; HGLRC handle; -} _GLFWusercontext; +} _GLFWusercontextWGL; GLFWbool _glfwInitWGL(void); void _glfwTerminateWGL(void); GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, const _GLFWctxconfig* ctxconfig, const _GLFWfbconfig* fbconfig); +_GLFWusercontext* _glfwCreateUserContextWGL(_GLFWwindow* window); + + diff --git a/src/win32_window.c b/src/win32_window.c index 9dc52bab..fe75b765 100644 --- a/src/win32_window.c +++ b/src/win32_window.c @@ -2321,6 +2321,23 @@ VkResult _glfwPlatformCreateWindowSurface(VkInstance instance, return err; } +_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) +{ + if (window->context.wgl.handle) + { + return _glfwCreateUserContextWGL(window); + } + else if (window->context.egl.handle) + { + return _glfwCreateUserContextEGL(window); + } + else if (window->context.osmesa.handle) + { + return _glfwCreateUserContextOSMesa(window); + } + + return GLFW_FALSE; +} ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// diff --git a/src/x11_window.c b/src/x11_window.c index f88a45c4..8704126e 100644 --- a/src/x11_window.c +++ b/src/x11_window.c @@ -3219,6 +3219,24 @@ VkResult _glfwPlatformCreateWindowSurface(VkInstance instance, } } +_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) +{ + if (window->context.glx.handle) + { + return _glfwCreateUserContextGLX(window); + } + else if (window->context.egl.handle) + { + return _glfwCreateUserContextEGL(window); + } + else if (window->context.osmesa.handle) + { + return _glfwCreateUserContextOSMesa(window); + } + + return GLFW_FALSE; +} + ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// From 51f11929f376a217d4307f0827e02f84d002f041 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Mon, 13 Jul 2020 18:48:10 +0100 Subject: [PATCH 05/37] User context Wayland and EGL implementations --- src/egl_context.c | 115 +++++++++++++++++++++++++++++++++++++--------- src/wgl_context.h | 14 +++--- src/wl_platform.h | 1 + src/wl_window.c | 13 ++++++ 4 files changed, 114 insertions(+), 29 deletions(-) diff --git a/src/egl_context.c b/src/egl_context.c index b059c4d9..2ea33da4 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -500,17 +500,15 @@ void _glfwTerminateEGL(void) attribs[index++] = v; \ } -// Create the OpenGL or OpenGL ES context +// Create the OpenGL or OpenGL ES context for the window eglConfig // -GLFWbool _glfwCreateContextEGL(_GLFWwindow* window, - const _GLFWctxconfig* ctxconfig, - const _GLFWfbconfig* fbconfig) +GLFWbool _glfwCreateContextForConfigEGL(_GLFWwindow* window, + const _GLFWctxconfig* ctxconfig, + EGLContext* context) { EGLint attribs[40]; - EGLConfig config; - EGLContext share = NULL; - EGLNativeWindowType native; int index = 0; + EGLContext share = NULL; if (!_glfw.egl.display) { @@ -521,13 +519,6 @@ GLFWbool _glfwCreateContextEGL(_GLFWwindow* window, if (ctxconfig->share) share = ctxconfig->share->context.egl.handle; - if (!chooseEGLConfig(ctxconfig, fbconfig, &config)) - { - _glfwInputError(GLFW_FORMAT_UNAVAILABLE, - "EGL: Failed to find a suitable EGLConfig"); - return GLFW_FALSE; - } - if (ctxconfig->client == GLFW_OPENGL_ES_API) { if (!eglBindAPI(EGL_OPENGL_ES_API)) @@ -624,7 +615,7 @@ GLFWbool _glfwCreateContextEGL(_GLFWwindow* window, setAttrib(EGL_NONE, EGL_NONE); window->context.egl.handle = eglCreateContext(_glfw.egl.display, - config, share, attribs); + window->context.egl.config, share, attribs); if (window->context.egl.handle == EGL_NO_CONTEXT) { @@ -634,9 +625,32 @@ GLFWbool _glfwCreateContextEGL(_GLFWwindow* window, return GLFW_FALSE; } - // Set up attributes for surface creation - index = 0; + return GLFW_TRUE; +} +// Create the OpenGL or OpenGL ES context +// +GLFWbool _glfwCreateContextEGL(_GLFWwindow* window, + const _GLFWctxconfig* ctxconfig, + const _GLFWfbconfig* fbconfig) +{ + EGLNativeWindowType native; + EGLint attribs[40]; + int index = 0; + + if (!chooseEGLConfig(ctxconfig, fbconfig, &window->context.egl.config)) + { + _glfwInputError(GLFW_FORMAT_UNAVAILABLE, + "EGL: Failed to find a suitable EGLConfig"); + return GLFW_FALSE; + } + + if (!_glfwCreateContextForConfigEGL(window,ctxconfig,&window->context.egl.handle)) + { + return GLFW_FALSE; + } + + // Set up attributes for surface creation if (fbconfig->sRGB) { if (_glfw.egl.KHR_gl_colorspace) @@ -651,12 +665,12 @@ GLFWbool _glfwCreateContextEGL(_GLFWwindow* window, if (_glfw.egl.platform && _glfw.egl.platform != EGL_PLATFORM_ANGLE_ANGLE) { window->context.egl.surface = - eglCreatePlatformWindowSurfaceEXT(_glfw.egl.display, config, native, attribs); + eglCreatePlatformWindowSurfaceEXT(_glfw.egl.display, window->context.egl.config, native, attribs); } else { window->context.egl.surface = - eglCreateWindowSurface(_glfw.egl.display, config, native, attribs); + eglCreateWindowSurface(_glfw.egl.display, window->context.egl.config, native, attribs); } if (window->context.egl.surface == EGL_NO_SURFACE) @@ -667,7 +681,6 @@ GLFWbool _glfwCreateContextEGL(_GLFWwindow* window, return GLFW_FALSE; } - window->context.egl.config = config; // Load the appropriate client library if (!_glfw.egl.KHR_get_all_proc_addresses) @@ -802,10 +815,68 @@ GLFWbool _glfwChooseVisualEGL(const _GLFWwndconfig* wndconfig, } #endif // _GLFW_X11 +static void _glfwMakeUserContextCurrentEGL(_GLFWusercontext* context) +{ + if(context) + { + if (!eglMakeCurrent(_glfw.egl.display, + context->window->context.egl.surface, + context->window->context.egl.surface, + context->egl.handle)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "EGL: Failed to make context current: %s", + getEGLErrorString(eglGetError())); + return; + } + } + else + { + if (!eglMakeCurrent(_glfw.egl.display, + EGL_NO_SURFACE, + EGL_NO_SURFACE, + EGL_NO_CONTEXT)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "EGL: Failed to clear current context: %s", + getEGLErrorString(eglGetError())); + return; + } + } +} + +static void _glfwDestroyUserContextEGL(_GLFWusercontext* context) +{ + if (context->egl.handle) + { + eglDestroyContext(_glfw.egl.display, context->egl.handle); + } + free(context); +} + _GLFWusercontext* _glfwCreateUserContextEGL(_GLFWwindow* window) { - // TODO - return NULL; + _GLFWusercontext* context; + _GLFWctxconfig ctxconfig; + + context = calloc(1, sizeof(_GLFWusercontext)); + context->window = window; + + ctxconfig = _glfw.hints.context; + ctxconfig.share = window; + + if(!_glfwCreateContextForConfigEGL(window,&ctxconfig,&context->egl.handle)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "EGL: Failed to create user OpenGL context"); + free(context); + return NULL; + } + + context->makeCurrent = _glfwMakeUserContextCurrentEGL; + context->destroy = _glfwDestroyUserContextEGL; + + return context; } diff --git a/src/wgl_context.h b/src/wgl_context.h index d04cd661..1268065f 100644 --- a/src/wgl_context.h +++ b/src/wgl_context.h @@ -119,6 +119,13 @@ typedef struct _GLFWcontextWGL } _GLFWcontextWGL; +// WGL-specific user context data +// +typedef struct _GLFWusercontextWGL +{ + HGLRC handle; +} _GLFWusercontextWGL; + // WGL-specific global data // typedef struct _GLFWlibraryWGL @@ -152,13 +159,6 @@ typedef struct _GLFWlibraryWGL } _GLFWlibraryWGL; -// WGL-specific user context data -// -typedef struct _GLFWusercontextWGL -{ - HGLRC handle; -} _GLFWusercontextWGL; - GLFWbool _glfwInitWGL(void); void _glfwTerminateWGL(void); GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, diff --git a/src/wl_platform.h b/src/wl_platform.h index 966155fd..a744588c 100644 --- a/src/wl_platform.h +++ b/src/wl_platform.h @@ -72,6 +72,7 @@ typedef VkBool32 (APIENTRY *PFN_vkGetPhysicalDeviceWaylandPresentationSupportKHR #define _GLFW_PLATFORM_CONTEXT_STATE struct { int dummyContext; } #define _GLFW_PLATFORM_LIBRARY_CONTEXT_STATE struct { int dummyLibraryContext; } +#define _GLFW_PLATFORM_USER_CONTEXT_STATE struct { int dummyUserContext; } struct wl_cursor_image { uint32_t width; diff --git a/src/wl_window.c b/src/wl_window.c index d1dad065..a9cefc97 100644 --- a/src/wl_window.c +++ b/src/wl_window.c @@ -1765,6 +1765,19 @@ VkResult _glfwPlatformCreateWindowSurface(VkInstance instance, return err; } +_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) +{ + if (window->context.egl.handle) + { + return _glfwCreateUserContextEGL(window); + } + else if (window->context.osmesa.handle) + { + return _glfwCreateUserContextOSMesa(window); + } + + return GLFW_FALSE; +} ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// From 931ba89aad873df74b6bcb63010b6e20c62c4f13 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 14 Jul 2020 11:41:42 +0100 Subject: [PATCH 06/37] User context null platform and OSMESA implementation. --- src/glx_context.c | 1 - src/null_platform.h | 1 + src/null_window.c | 9 ++++++ src/osmesa_context.c | 77 ++++++++++++++++++++++++++++++++++++++------ 4 files changed, 78 insertions(+), 10 deletions(-) diff --git a/src/glx_context.c b/src/glx_context.c index e58584ee..19ccf7f9 100644 --- a/src/glx_context.c +++ b/src/glx_context.c @@ -726,7 +726,6 @@ _GLFWusercontext* _glfwCreateUserContextGLX(_GLFWwindow* window) context->makeCurrent = _glfwMakeUserContextCurrentGLX; context->destroy = _glfwDestroyUserContextGLX; - return context; } diff --git a/src/null_platform.h b/src/null_platform.h index 49436dcc..d476f2fa 100644 --- a/src/null_platform.h +++ b/src/null_platform.h @@ -34,6 +34,7 @@ #define _GLFW_PLATFORM_CONTEXT_STATE struct { int dummyContext; } #define _GLFW_PLATFORM_CURSOR_STATE struct { int dummyCursor; } #define _GLFW_PLATFORM_LIBRARY_CONTEXT_STATE struct { int dummyLibraryContext; } +#define _GLFW_PLATFORM_USER_CONTEXT_STATE struct { int dummyUserContext; } #include "posix_time.h" #include "posix_thread.h" diff --git a/src/null_window.c b/src/null_window.c index ba85571b..1f760e84 100644 --- a/src/null_window.c +++ b/src/null_window.c @@ -649,3 +649,12 @@ VkResult _glfwPlatformCreateWindowSurface(VkInstance instance, return VK_ERROR_EXTENSION_NOT_PRESENT; } +_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) +{ + if (window->context.osmesa.handle) + { + return _glfwCreateUserContextOSMesa(window); + } + + return GLFW_FALSE; +} diff --git a/src/osmesa_context.c b/src/osmesa_context.c index 07baabed..69ae4e2e 100644 --- a/src/osmesa_context.c +++ b/src/osmesa_context.c @@ -195,9 +195,9 @@ void _glfwTerminateOSMesa(void) attribs[index++] = v; \ } -GLFWbool _glfwCreateContextOSMesa(_GLFWwindow* window, - const _GLFWctxconfig* ctxconfig, - const _GLFWfbconfig* fbconfig) +GLFWbool _glfwCreateContextForConfigOSMesa(const _GLFWctxconfig* ctxconfig, + const _GLFWfbconfig* fbconfig, + OSMesaContext* context ) { OSMesaContext share = NULL; const int accumBits = fbconfig->accumRedBits + @@ -248,7 +248,7 @@ GLFWbool _glfwCreateContextOSMesa(_GLFWwindow* window, setAttrib(0, 0); - window->context.osmesa.handle = + *context = OSMesaCreateContextAttribs(attribs, share); } else @@ -260,7 +260,7 @@ GLFWbool _glfwCreateContextOSMesa(_GLFWwindow* window, return GLFW_FALSE; } - window->context.osmesa.handle = + *context = OSMesaCreateContextExt(OSMESA_RGBA, fbconfig->depthBits, fbconfig->stencilBits, @@ -268,13 +268,27 @@ GLFWbool _glfwCreateContextOSMesa(_GLFWwindow* window, share); } - if (window->context.osmesa.handle == NULL) + if (*context == NULL) { _glfwInputError(GLFW_VERSION_UNAVAILABLE, "OSMesa: Failed to create context"); return GLFW_FALSE; } + return GLFW_TRUE; +} + +#undef setAttrib + +GLFWbool _glfwCreateContextOSMesa(_GLFWwindow* window, + const _GLFWctxconfig* ctxconfig, + const _GLFWfbconfig* fbconfig) +{ + if(!_glfwCreateContextForConfigOSMesa(ctxconfig,fbconfig,&window->context.osmesa.handle)) + { + return GLFW_FALSE; + } + window->context.makeCurrent = makeContextCurrentOSMesa; window->context.swapBuffers = swapBuffersOSMesa; window->context.swapInterval = swapIntervalOSMesa; @@ -285,12 +299,57 @@ GLFWbool _glfwCreateContextOSMesa(_GLFWwindow* window, return GLFW_TRUE; } -#undef setAttrib +static void _glfwMakeUserContextCurrentOSMesa(_GLFWusercontext* context) +{ + if(context) + { + if (!OSMesaMakeCurrent(context->osmesa.handle, + context->window->context.osmesa.buffer, + GL_UNSIGNED_BYTE, + context->window->context.osmesa.width, context->window->context.osmesa.height)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "OSMesa: Failed to make context current"); + return; + } + } +} + +static void _glfwDestroyUserContextOSMesa(_GLFWusercontext* context) +{ + if (context->osmesa.handle) + { + OSMesaDestroyContext(context->osmesa.handle); + } + free(context); +} _GLFWusercontext* _glfwCreateUserContextOSMesa(_GLFWwindow* window) { - // TODO - return NULL; + _GLFWusercontext* context; + _GLFWctxconfig ctxconfig; + _GLFWfbconfig fbconfig; + + context = calloc(1, sizeof(_GLFWusercontext)); + context->window = window; + + ctxconfig = _glfw.hints.context; + ctxconfig.share = window; + + fbconfig = _glfw.hints.framebuffer; + + if(!_glfwCreateContextForConfigOSMesa(&ctxconfig,&fbconfig,&context->osmesa.handle)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "OSMesa: Failed to create user OpenGL context"); + free(context); + return NULL; + } + + context->makeCurrent = _glfwMakeUserContextCurrentOSMesa; + context->destroy = _glfwDestroyUserContextOSMesa; + + return context; } ////////////////////////////////////////////////////////////////////////// From dd854e47ba52ba5692b12da6e2e2848b2ce31308 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 14 Jul 2020 12:38:18 +0100 Subject: [PATCH 07/37] Formatting --- src/egl_context.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/egl_context.c b/src/egl_context.c index 2ea33da4..eb8d6a74 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -503,8 +503,8 @@ void _glfwTerminateEGL(void) // Create the OpenGL or OpenGL ES context for the window eglConfig // GLFWbool _glfwCreateContextForConfigEGL(_GLFWwindow* window, - const _GLFWctxconfig* ctxconfig, - EGLContext* context) + const _GLFWctxconfig* ctxconfig, + EGLContext* context) { EGLint attribs[40]; int index = 0; From f2b86a25b37962dd9e44c7265f8a674b547d03ed Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 14 Jul 2020 19:10:31 +0100 Subject: [PATCH 08/37] Cocoa and NSGL Implementation --- src/cocoa_window.m | 19 +++++++++++++++++++ src/nsgl_context.h | 10 ++++++++++ src/nsgl_context.m | 47 ++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 76 insertions(+) diff --git a/src/cocoa_window.m b/src/cocoa_window.m index e34fb876..e7d048a9 100644 --- a/src/cocoa_window.m +++ b/src/cocoa_window.m @@ -1885,6 +1885,25 @@ VkResult _glfwPlatformCreateWindowSurface(VkInstance instance, } +_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) +{ + if (window->context.nsgl.object) + { + return _glfwCreateUserContextNSGL(window); + } + else if (window->context.egl.handle) + { + return _glfwCreateUserContextEGL(window); + } + else if (window->context.osmesa.handle) + { + return _glfwCreateUserContextOSMesa(window); + } + + return GLFW_FALSE; +} + + ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// ////////////////////////////////////////////////////////////////////////// diff --git a/src/nsgl_context.h b/src/nsgl_context.h index 9c31436c..598f642b 100644 --- a/src/nsgl_context.h +++ b/src/nsgl_context.h @@ -34,6 +34,7 @@ #define _GLFW_PLATFORM_CONTEXT_STATE _GLFWcontextNSGL nsgl #define _GLFW_PLATFORM_LIBRARY_CONTEXT_STATE _GLFWlibraryNSGL nsgl +#define _GLFW_PLATFORM_USER_CONTEXT_STATE _GLFWusercontextNSGL nsgl #include @@ -47,6 +48,14 @@ typedef struct _GLFWcontextNSGL } _GLFWcontextNSGL; +// NSGL-specific per user context data +// +typedef struct _GLFWusercontextNSGL +{ + id object; + +} _GLFWusercontextNSGL; + // NSGL-specific global data // typedef struct _GLFWlibraryNSGL @@ -63,4 +72,5 @@ GLFWbool _glfwCreateContextNSGL(_GLFWwindow* window, const _GLFWctxconfig* ctxconfig, const _GLFWfbconfig* fbconfig); void _glfwDestroyContextNSGL(_GLFWwindow* window); +_GLFWusercontext* _glfwCreateUserContextNSGL(_GLFWwindow* window); diff --git a/src/nsgl_context.m b/src/nsgl_context.m index 3f73f7a6..29430060 100644 --- a/src/nsgl_context.m +++ b/src/nsgl_context.m @@ -349,6 +349,53 @@ GLFWbool _glfwCreateContextNSGL(_GLFWwindow* window, } +static void _glfwMakeUserContextCurrentNSGL(_GLFWusercontext* context) +{ + @autoreleasepool { + + if (context) + [context->nsgl.object makeCurrentContext]; + else + [NSOpenGLContext clearCurrentContext]; + + } // autoreleasepool +} + +static void _glfwDestroyUserContextNSGL(_GLFWusercontext* context) +{ + @autoreleasepool { + + if (context->nsgl.object) + [context->nsgl.object release]; + + } // autoreleasepool + free(context); +} + +_GLFWusercontext* _glfwCreateUserContextNSGL(_GLFWwindow* window) +{ + _GLFWusercontext* context; + + context = calloc(1, sizeof(_GLFWusercontext)); + context->window = window; + + context->nsgl.object = + [[NSOpenGLContext alloc] initWithFormat:window->context.nsgl.pixelFormat + shareContext:window->context.nsgl.object]; + if (window->context.nsgl.object == nil) + { + _glfwInputError(GLFW_VERSION_UNAVAILABLE, + "NSGL: Failed to create OpenGL user context"); + free(context); + return NULL; + } + + context->makeCurrent = _glfwMakeUserContextCurrentNSGL; + context->destroy = _glfwDestroyUserContextNSGL; + + return context; +} + ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// ////////////////////////////////////////////////////////////////////////// From 4caca8b20c71879211f99e418388d5bca9d95875 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 15 Jul 2020 11:18:58 +0100 Subject: [PATCH 09/37] glfwMakeUserContextCurrent now calls glfwMakeContextCurrent(NULL) to ensure TLS set. --- src/context.c | 3 +++ tests/usercontext.c | 2 -- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/src/context.c b/src/context.c index 2ec29c25..fa1a4d78 100644 --- a/src/context.c +++ b/src/context.c @@ -778,6 +778,9 @@ GLFWAPI void glfwDestroyUserContext(GLFWusercontext* handle) GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* handle) { _GLFWusercontext* context = (_GLFWusercontext*)handle; + + glfwMakeContextCurrent(NULL); + if (context) context->makeCurrent(context); } diff --git a/tests/usercontext.c b/tests/usercontext.c index 58916430..a3182929 100644 --- a/tests/usercontext.c +++ b/tests/usercontext.c @@ -31,11 +31,9 @@ int main(void) } /* set the user context current */ - glfwMakeContextCurrent(NULL); glfwMakeUserContextCurrent(usercontext); /* set the window context current */ - glfwMakeUserContextCurrent(NULL); glfwMakeContextCurrent(window); glClearColor( 0.4f, 0.3f, 0.4f, 0.0f ); From a1a1b77150bc9417aa632692ecab106cad885543 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 15 Jul 2020 11:19:18 +0100 Subject: [PATCH 10/37] Added _GLFW_REQUIRE* to user context functions. --- src/context.c | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/src/context.c b/src/context.c index fa1a4d78..5bab2d01 100644 --- a/src/context.c +++ b/src/context.c @@ -763,6 +763,8 @@ GLFWAPI GLFWusercontext* glfwCreateUserContext(GLFWwindow* handle) _GLFWusercontext* context; _GLFWwindow* window = (_GLFWwindow*)handle; + _GLFW_REQUIRE_INIT_OR_RETURN(NULL); + context = _glfwPlatformCreateUserContext(window); return (GLFWusercontext*)context; @@ -771,6 +773,9 @@ GLFWAPI GLFWusercontext* glfwCreateUserContext(GLFWwindow* handle) GLFWAPI void glfwDestroyUserContext(GLFWusercontext* handle) { _GLFWusercontext* context = (_GLFWusercontext*)handle; + + _GLFW_REQUIRE_INIT(); + if (context) context->destroy(context); } @@ -779,6 +784,8 @@ GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* handle) { _GLFWusercontext* context = (_GLFWusercontext*)handle; + _GLFW_REQUIRE_INIT(); + glfwMakeContextCurrent(NULL); if (context) From 6539d101f32c6d75856882032ce61fd3ad2eb35b Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 15 Jul 2020 13:19:14 +0100 Subject: [PATCH 11/37] Added TLS for current user context and simplified code --- src/context.c | 11 +++++++++++ src/egl_context.c | 34 ++++++++++------------------------ src/glx_context.c | 20 ++++++-------------- src/init.c | 4 +++- src/internal.h | 1 + src/nsgl_context.m | 6 ++---- src/osmesa_context.c | 19 +++++++++---------- src/wgl_context.c | 20 ++++++-------------- 8 files changed, 48 insertions(+), 67 deletions(-) diff --git a/src/context.c b/src/context.c index 5bab2d01..6e558d55 100644 --- a/src/context.c +++ b/src/context.c @@ -617,6 +617,8 @@ GLFWAPI void glfwMakeContextCurrent(GLFWwindow* handle) _GLFW_REQUIRE_INIT(); + _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); + if (window && window->context.client == GLFW_NO_API) { _glfwInputError(GLFW_NO_WINDOW_CONTEXT, @@ -773,11 +775,17 @@ GLFWAPI GLFWusercontext* glfwCreateUserContext(GLFWwindow* handle) GLFWAPI void glfwDestroyUserContext(GLFWusercontext* handle) { _GLFWusercontext* context = (_GLFWusercontext*)handle; + _GLFWusercontext* prev = _glfwPlatformGetTls(&_glfw.usercontextSlot); _GLFW_REQUIRE_INIT(); if (context) + { + if(prev==context) + _glfwPlatformSetTls(&_glfw.usercontextSlot,NULL); + context->destroy(context); + } } GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* handle) @@ -786,6 +794,9 @@ GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* handle) _GLFW_REQUIRE_INIT(); + // Call glfwMakeContextCurrent(NULL) to both clear context TLS and set + // context to NULL if required by platform & context, and this + // handles case of calling glfwMakeUserContextCurrent(NULL) glfwMakeContextCurrent(NULL); if (context) diff --git a/src/egl_context.c b/src/egl_context.c index eb8d6a74..43615794 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -817,32 +817,18 @@ GLFWbool _glfwChooseVisualEGL(const _GLFWwndconfig* wndconfig, static void _glfwMakeUserContextCurrentEGL(_GLFWusercontext* context) { - if(context) + if (!eglMakeCurrent(_glfw.egl.display, + context->window->context.egl.surface, + context->window->context.egl.surface, + context->egl.handle)) { - if (!eglMakeCurrent(_glfw.egl.display, - context->window->context.egl.surface, - context->window->context.egl.surface, - context->egl.handle)) - { - _glfwInputError(GLFW_PLATFORM_ERROR, - "EGL: Failed to make context current: %s", - getEGLErrorString(eglGetError())); - return; - } - } - else - { - if (!eglMakeCurrent(_glfw.egl.display, - EGL_NO_SURFACE, - EGL_NO_SURFACE, - EGL_NO_CONTEXT)) - { - _glfwInputError(GLFW_PLATFORM_ERROR, - "EGL: Failed to clear current context: %s", - getEGLErrorString(eglGetError())); - return; - } + _glfwInputError(GLFW_PLATFORM_ERROR, + "EGL: Failed to make user context current: %s", + getEGLErrorString(eglGetError())); + _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); + return; } + _glfwPlatformSetTls(&_glfw.usercontextSlot, context); } static void _glfwDestroyUserContextEGL(_GLFWusercontext* context) diff --git a/src/glx_context.c b/src/glx_context.c index 19ccf7f9..957be797 100644 --- a/src/glx_context.c +++ b/src/glx_context.c @@ -680,22 +680,14 @@ GLFWbool _glfwChooseVisualGLX(const _GLFWwndconfig* wndconfig, static void _glfwMakeUserContextCurrentGLX(_GLFWusercontext* context) { - if(context) + if(!glXMakeCurrent(_glfw.x11.display, context->window->context.glx.window,context->glx.handle)) { - if(!glXMakeCurrent(_glfw.x11.display, context->window->context.glx.window,context->glx.handle)) - { - _glfwInputError(GLFW_PLATFORM_ERROR, - "GLX: Failed to set current user context"); - } - } - else - { - if (!glXMakeCurrent(_glfw.x11.display, None, NULL)) - { - _glfwInputError(GLFW_PLATFORM_ERROR, - "GLX: Failed to clear current context"); - } + _glfwInputError(GLFW_PLATFORM_ERROR, + "GLX: Failed to make user context current"); + _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); + return; } + _glfwPlatformSetTls(&_glfw.usercontextSlot, context); } static void _glfwDestroyUserContextGLX(_GLFWusercontext* context) diff --git a/src/init.c b/src/init.c index f9ce336a..5df22d0e 100644 --- a/src/init.c +++ b/src/init.c @@ -102,6 +102,7 @@ static void terminate(void) free(error); } + _glfwPlatformDestroyTls(&_glfw.usercontextSlot); _glfwPlatformDestroyTls(&_glfw.contextSlot); _glfwPlatformDestroyTls(&_glfw.errorSlot); _glfwPlatformDestroyMutex(&_glfw.errorLock); @@ -244,7 +245,8 @@ GLFWAPI int glfwInit(void) if (!_glfwPlatformCreateMutex(&_glfw.errorLock) || !_glfwPlatformCreateTls(&_glfw.errorSlot) || - !_glfwPlatformCreateTls(&_glfw.contextSlot)) + !_glfwPlatformCreateTls(&_glfw.contextSlot) || + !_glfwPlatformCreateTls(&_glfw.usercontextSlot)) { terminate(); return GLFW_FALSE; diff --git a/src/internal.h b/src/internal.h index bf99478e..ab6c9471 100644 --- a/src/internal.h +++ b/src/internal.h @@ -564,6 +564,7 @@ struct _GLFWlibrary _GLFWtls errorSlot; _GLFWtls contextSlot; + _GLFWtls usercontextSlot; _GLFWmutex errorLock; struct { diff --git a/src/nsgl_context.m b/src/nsgl_context.m index 29430060..866cb4bd 100644 --- a/src/nsgl_context.m +++ b/src/nsgl_context.m @@ -353,10 +353,9 @@ static void _glfwMakeUserContextCurrentNSGL(_GLFWusercontext* context) { @autoreleasepool { - if (context) [context->nsgl.object makeCurrentContext]; - else - [NSOpenGLContext clearCurrentContext]; + + _glfwPlatformSetTls(&_glfw.usercontextSlot, context); } // autoreleasepool } @@ -365,7 +364,6 @@ static void _glfwDestroyUserContextNSGL(_GLFWusercontext* context) { @autoreleasepool { - if (context->nsgl.object) [context->nsgl.object release]; } // autoreleasepool diff --git a/src/osmesa_context.c b/src/osmesa_context.c index 69ae4e2e..95b4017f 100644 --- a/src/osmesa_context.c +++ b/src/osmesa_context.c @@ -301,18 +301,17 @@ GLFWbool _glfwCreateContextOSMesa(_GLFWwindow* window, static void _glfwMakeUserContextCurrentOSMesa(_GLFWusercontext* context) { - if(context) + if (!OSMesaMakeCurrent(context->osmesa.handle, + context->window->context.osmesa.buffer, + GL_UNSIGNED_BYTE, + context->window->context.osmesa.width, context->window->context.osmesa.height)) { - if (!OSMesaMakeCurrent(context->osmesa.handle, - context->window->context.osmesa.buffer, - GL_UNSIGNED_BYTE, - context->window->context.osmesa.width, context->window->context.osmesa.height)) - { - _glfwInputError(GLFW_PLATFORM_ERROR, - "OSMesa: Failed to make context current"); - return; - } + _glfwInputError(GLFW_PLATFORM_ERROR, + "OSMesa: Failed to make user context current"); + _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); + return; } + _glfwPlatformSetTls(&_glfw.usercontextSlot, context); } static void _glfwDestroyUserContextOSMesa(_GLFWusercontext* context) diff --git a/src/wgl_context.c b/src/wgl_context.c index d6d668ad..d0702b66 100644 --- a/src/wgl_context.c +++ b/src/wgl_context.c @@ -792,22 +792,14 @@ GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, static void _glfwMakeUserContextCurrentWGL(_GLFWusercontext* context) { - if(context) + if(!wglMakeCurrent(context->window->context.wgl.dc,context->wgl.handle)) { - if(!wglMakeCurrent(context->window->context.wgl.dc,context->wgl.handle)) - { - _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, - "WGL: Failed to set current user context"); - } - } - else - { - if (!wglMakeCurrent(NULL, NULL)) - { - _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, - "WGL: Failed to clear current context"); - } + _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, + "WGL: Failed to make user context current"); + _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); + return; } + _glfwPlatformSetTls(&_glfw.usercontextSlot, context); } static void _glfwDestroyUserContextWGL(_GLFWusercontext* context) From f2806aa9abd9d15be35c2e8552733773f13f9b73 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 15 Jul 2020 13:19:33 +0100 Subject: [PATCH 12/37] Added glfwGetCurrentUserContext --- include/GLFW/glfw3.h | 6 ++++++ src/context.c | 6 ++++++ 2 files changed, 12 insertions(+) diff --git a/include/GLFW/glfw3.h b/include/GLFW/glfw3.h index d23bbac0..ece72da8 100644 --- a/include/GLFW/glfw3.h +++ b/include/GLFW/glfw3.h @@ -5827,6 +5827,12 @@ GLFWAPI void glfwDestroyUserContext(GLFWusercontext* context); */ GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* context); +/*! @brief Get the current OpenGL user context +* +*/ +GLFWAPI GLFWusercontext* glfwGetCurrentUserContext(void); + + /*! @brief Returns whether the Vulkan loader and an ICD have been found. * * This function returns whether the Vulkan loader and any minimally functional diff --git a/src/context.c b/src/context.c index 6e558d55..8db76c84 100644 --- a/src/context.c +++ b/src/context.c @@ -802,3 +802,9 @@ GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* handle) if (context) context->makeCurrent(context); } + +GLFWAPI GLFWusercontext* glfwGetCurrentUserContext(void) +{ + _GLFW_REQUIRE_INIT_OR_RETURN(NULL); + return _glfwPlatformGetTls(&_glfw.usercontextSlot); +} \ No newline at end of file From d01ad1a78d392985bec21a5706de15f8970ab75c Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 15 Jul 2020 13:19:47 +0100 Subject: [PATCH 13/37] Improved usercontext.c test --- tests/usercontext.c | 36 ++++++++++++++++++++++++++++++++++++ 1 file changed, 36 insertions(+) diff --git a/tests/usercontext.c b/tests/usercontext.c index a3182929..6bb3bd73 100644 --- a/tests/usercontext.c +++ b/tests/usercontext.c @@ -1,11 +1,19 @@ #include #include +#include + +static void error_callback(int error, const char* description) +{ + fprintf(stderr, "Error: %s\n", description); +} int main(void) { GLFWwindow* window; GLFWusercontext* usercontext; + glfwSetErrorCallback(error_callback); + /* Initialize the library */ if (!glfwInit()) return -1; @@ -26,16 +34,44 @@ int main(void) usercontext = glfwCreateUserContext(window); if (!usercontext) { + fprintf(stderr, "Failed to create user context\n"); glfwTerminate(); return -1; } + /* set the user context current */ glfwMakeUserContextCurrent(usercontext); + if (glfwGetCurrentContext()!=NULL) + { + fprintf(stderr, "Current glfw window context not NULL after glfwMakeUserContextCurrent\n"); + glfwTerminate(); + return -1; + } + if (glfwGetCurrentUserContext()!=usercontext) + { + fprintf(stderr, "Current user context not correct after glfwMakeUserContextCurrent\n"); + glfwTerminate(); + return -1; + } + /* set the window context current */ glfwMakeContextCurrent(window); + if ( glfwGetCurrentUserContext() != NULL ) + { + fprintf(stderr, "Current user context not NULL after glfwMakeContextCurrent\n"); + glfwTerminate(); + return -1; + } + if ( glfwGetCurrentContext() != window ) + { + fprintf(stderr, "Current glfw window context not correct after glfwMakeContextCurrent\n"); + glfwTerminate(); + return -1; + } + glClearColor( 0.4f, 0.3f, 0.4f, 0.0f ); From 3de9ed645350bd5bc2dc4c9d762021c7980e278f Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 15 Jul 2020 13:21:02 +0100 Subject: [PATCH 14/37] Fixed EGL glfwMakeUserContextCurrent - requires different surface --- src/egl_context.c | 44 +++++++++++++++++++++++++++++++++++++++----- src/egl_context.h | 14 ++++++++++++++ 2 files changed, 53 insertions(+), 5 deletions(-) diff --git a/src/egl_context.c b/src/egl_context.c index 43615794..7b5bb0f3 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -372,6 +372,8 @@ GLFWbool _glfwInitEGL(void) _glfw_dlsym(_glfw.egl.handle, "eglQueryString"); _glfw.egl.GetProcAddress = (PFN_eglGetProcAddress) _glfw_dlsym(_glfw.egl.handle, "eglGetProcAddress"); + _glfw.egl.CreatePbufferSurface = (PFN_eglCreatePbufferSurface) + _glfw_dlsym(_glfw.egl.handle, "eglCreatePbufferSurface"); if (!_glfw.egl.GetConfigAttrib || !_glfw.egl.GetConfigs || @@ -833,10 +835,10 @@ static void _glfwMakeUserContextCurrentEGL(_GLFWusercontext* context) static void _glfwDestroyUserContextEGL(_GLFWusercontext* context) { - if (context->egl.handle) - { - eglDestroyContext(_glfw.egl.display, context->egl.handle); - } + if (context->egl.surface!=EGL_NO_SURFACE) + eglDestroySurface(_glfw.egl.display,context->egl.surface); + + eglDestroyContext(_glfw.egl.display, context->egl.handle); free(context); } @@ -844,6 +846,22 @@ _GLFWusercontext* _glfwCreateUserContextEGL(_GLFWwindow* window) { _GLFWusercontext* context; _GLFWctxconfig ctxconfig; + const EGLint auxConfigAttribs[] = + { + EGL_SURFACE_TYPE, EGL_PBUFFER_BIT, + EGL_BLUE_SIZE, 8, EGL_GREEN_SIZE, 8, EGL_RED_SIZE, 8, EGL_ALPHA_SIZE, 8, + EGL_DEPTH_SIZE, 0, EGL_STENCIL_SIZE, 0, + EGL_NONE + }; + EGLint dummySurfaceAttribs[] = + { + EGL_WIDTH, 1, EGL_HEIGHT, 1, + EGL_TEXTURE_TARGET, EGL_NO_TEXTURE, + EGL_TEXTURE_FORMAT, EGL_NO_TEXTURE, + EGL_NONE + }; + EGLint dummySurfaceNumConfigs; + EGLConfig dummySurfaceConfig; context = calloc(1, sizeof(_GLFWusercontext)); context->window = window; @@ -851,13 +869,29 @@ _GLFWusercontext* _glfwCreateUserContextEGL(_GLFWwindow* window) ctxconfig = _glfw.hints.context; ctxconfig.share = window; - if(!_glfwCreateContextForConfigEGL(window,&ctxconfig,&context->egl.handle)) + if (!_glfwCreateContextForConfigEGL(window,&ctxconfig,&context->egl.handle)) { _glfwInputError(GLFW_PLATFORM_ERROR, "EGL: Failed to create user OpenGL context"); free(context); return NULL; } + if (glfwExtensionSupported("EGL_KHR_surfaceless_context")) + context->egl.surface = EGL_NO_SURFACE; + else + { + // create dummy surface + eglChooseConfig(_glfw.egl.display, auxConfigAttribs, &dummySurfaceConfig, 1, &dummySurfaceNumConfigs); + context->egl.surface = eglCreatePbufferSurface(_glfw.egl.display, dummySurfaceConfig, dummySurfaceAttribs); + if (!context->egl.surface) + { + eglDestroyContext(_glfw.egl.display, context->egl.handle); + _glfwInputError(GLFW_PLATFORM_ERROR, + "EGL: Failed to create surface for user context and EGL_KHR_surfaceless_context not supported"); + free(context); + return NULL; + } + } context->makeCurrent = _glfwMakeUserContextCurrentEGL; context->destroy = _glfwDestroyUserContextEGL; diff --git a/src/egl_context.h b/src/egl_context.h index 1dcab099..62801740 100644 --- a/src/egl_context.h +++ b/src/egl_context.h @@ -71,6 +71,12 @@ #define EGL_NO_DISPLAY ((EGLDisplay) 0) #define EGL_NO_CONTEXT ((EGLContext) 0) #define EGL_DEFAULT_DISPLAY ((EGLNativeDisplayType) 0) +#define EGL_PBUFFER_BIT 0x0001 +#define EGL_HEIGHT 0x3056 +#define EGL_WIDTH 0x3057 +#define EGL_NO_TEXTURE 0x305C +#define EGL_TEXTURE_FORMAT 0x3080 +#define EGL_TEXTURE_TARGET 0x3081 #define EGL_CONTEXT_OPENGL_FORWARD_COMPATIBLE_BIT_KHR 0x00000002 #define EGL_CONTEXT_OPENGL_CORE_PROFILE_BIT_KHR 0x00000001 @@ -130,6 +136,8 @@ typedef EGLBoolean (EGLAPIENTRY * PFN_eglSwapBuffers)(EGLDisplay,EGLSurface); typedef EGLBoolean (EGLAPIENTRY * PFN_eglSwapInterval)(EGLDisplay,EGLint); typedef const char* (EGLAPIENTRY * PFN_eglQueryString)(EGLDisplay,EGLint); typedef GLFWglproc (EGLAPIENTRY * PFN_eglGetProcAddress)(const char*); +typedef EGLSurface (EGLAPIENTRY * PFN_eglCreatePbufferSurface)(EGLDisplay,EGLConfig,const EGLint*); +typedef EGLBoolean (EGLAPIENTRY * PFN_eglChooseConfig)(EGLDisplay,EGLint const*,EGLConfig*,EGLint,EGLint*); #define eglGetConfigAttrib _glfw.egl.GetConfigAttrib #define eglGetConfigs _glfw.egl.GetConfigs #define eglGetDisplay _glfw.egl.GetDisplay @@ -146,6 +154,8 @@ typedef GLFWglproc (EGLAPIENTRY * PFN_eglGetProcAddress)(const char*); #define eglSwapInterval _glfw.egl.SwapInterval #define eglQueryString _glfw.egl.QueryString #define eglGetProcAddress _glfw.egl.GetProcAddress +#define eglCreatePbufferSurface _glfw.egl.CreatePbufferSurface +#define eglChooseConfig _glfw.egl.ChooseConfig typedef EGLDisplay (EGLAPIENTRY * PFNEGLGETPLATFORMDISPLAYEXTPROC)(EGLenum,void*,const EGLint*); typedef EGLSurface (EGLAPIENTRY * PFNEGLCREATEPLATFORMWINDOWSURFACEEXTPROC)(EGLDisplay,EGLConfig,void*,const EGLint*); @@ -169,6 +179,8 @@ typedef struct _GLFWcontextEGL typedef struct _GLFWusercontextEGL { EGLContext handle; + EGLSurface surface; + } _GLFWusercontextEGL; @@ -214,6 +226,8 @@ typedef struct _GLFWlibraryEGL PFN_eglSwapInterval SwapInterval; PFN_eglQueryString QueryString; PFN_eglGetProcAddress GetProcAddress; + PFN_eglCreatePbufferSurface CreatePbufferSurface; + PFN_eglChooseConfig ChooseConfig; PFNEGLGETPLATFORMDISPLAYEXTPROC GetPlatformDisplayEXT; PFNEGLCREATEPLATFORMWINDOWSURFACEEXTPROC CreatePlatformWindowSurfaceEXT; From 5f52f2a7f85de2c42f34981a524f93d0fb9d4256 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 15 Jul 2020 13:23:38 +0100 Subject: [PATCH 15/37] Fix for ELG wrong surface passed --- src/egl_context.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/egl_context.c b/src/egl_context.c index 7b5bb0f3..5eb8eb47 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -820,8 +820,8 @@ GLFWbool _glfwChooseVisualEGL(const _GLFWwndconfig* wndconfig, static void _glfwMakeUserContextCurrentEGL(_GLFWusercontext* context) { if (!eglMakeCurrent(_glfw.egl.display, - context->window->context.egl.surface, - context->window->context.egl.surface, + context->egl.surface, + context->egl.surface, context->egl.handle)) { _glfwInputError(GLFW_PLATFORM_ERROR, From e55a552eeef98d8a578d1ddfb4d5cf0a21073145 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 15 Jul 2020 14:31:05 +0100 Subject: [PATCH 16/37] User context EGL fixes. --- src/egl_context.c | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/src/egl_context.c b/src/egl_context.c index 5eb8eb47..bef5baa9 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -374,6 +374,8 @@ GLFWbool _glfwInitEGL(void) _glfw_dlsym(_glfw.egl.handle, "eglGetProcAddress"); _glfw.egl.CreatePbufferSurface = (PFN_eglCreatePbufferSurface) _glfw_dlsym(_glfw.egl.handle, "eglCreatePbufferSurface"); + _glfw.egl.ChooseConfig = (PFN_eglChooseConfig) + _glfw_dlsym(_glfw.egl.handle, "eglChooseConfig"); if (!_glfw.egl.GetConfigAttrib || !_glfw.egl.GetConfigs || @@ -390,7 +392,9 @@ GLFWbool _glfwInitEGL(void) !_glfw.egl.SwapBuffers || !_glfw.egl.SwapInterval || !_glfw.egl.QueryString || - !_glfw.egl.GetProcAddress) + !_glfw.egl.GetProcAddress || + !_glfw.egl.CreatePbufferSurface|| + !_glfw.egl.ChooseConfig) { _glfwInputError(GLFW_PLATFORM_ERROR, "EGL: Failed to load required entry points"); @@ -616,10 +620,9 @@ GLFWbool _glfwCreateContextForConfigEGL(_GLFWwindow* window, setAttrib(EGL_NONE, EGL_NONE); - window->context.egl.handle = eglCreateContext(_glfw.egl.display, - window->context.egl.config, share, attribs); + *context = eglCreateContext(_glfw.egl.display, window->context.egl.config, share, attribs); - if (window->context.egl.handle == EGL_NO_CONTEXT) + if (*context == EGL_NO_CONTEXT) { _glfwInputError(GLFW_VERSION_UNAVAILABLE, "EGL: Failed to create context: %s", From 2f2e7f041a9aac47118cfeb77d28c64022ac634a Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 15 Jul 2020 19:40:00 +0100 Subject: [PATCH 17/37] Simplified fallback pbuffer surface path. --- src/egl_context.c | 24 ++++++++++++++---------- 1 file changed, 14 insertions(+), 10 deletions(-) diff --git a/src/egl_context.c b/src/egl_context.c index bef5baa9..9797d63c 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -849,18 +849,15 @@ _GLFWusercontext* _glfwCreateUserContextEGL(_GLFWwindow* window) { _GLFWusercontext* context; _GLFWctxconfig ctxconfig; - const EGLint auxConfigAttribs[] = + EGLint dummyConfigAttribs[] = { EGL_SURFACE_TYPE, EGL_PBUFFER_BIT, - EGL_BLUE_SIZE, 8, EGL_GREEN_SIZE, 8, EGL_RED_SIZE, 8, EGL_ALPHA_SIZE, 8, - EGL_DEPTH_SIZE, 0, EGL_STENCIL_SIZE, 0, + EGL_RED_SIZE, 1, EGL_GREEN_SIZE, 1, EGL_BLUE_SIZE, 1, EGL_NONE }; EGLint dummySurfaceAttribs[] = { EGL_WIDTH, 1, EGL_HEIGHT, 1, - EGL_TEXTURE_TARGET, EGL_NO_TEXTURE, - EGL_TEXTURE_FORMAT, EGL_NO_TEXTURE, EGL_NONE }; EGLint dummySurfaceNumConfigs; @@ -883,14 +880,21 @@ _GLFWusercontext* _glfwCreateUserContextEGL(_GLFWwindow* window) context->egl.surface = EGL_NO_SURFACE; else { - // create dummy surface - eglChooseConfig(_glfw.egl.display, auxConfigAttribs, &dummySurfaceConfig, 1, &dummySurfaceNumConfigs); - context->egl.surface = eglCreatePbufferSurface(_glfw.egl.display, dummySurfaceConfig, dummySurfaceAttribs); - if (!context->egl.surface) + eglChooseConfig(_glfw.egl.display, dummyConfigAttribs, &dummySurfaceConfig, 1, &dummySurfaceNumConfigs); + if( !dummySurfaceNumConfigs) { eglDestroyContext(_glfw.egl.display, context->egl.handle); _glfwInputError(GLFW_PLATFORM_ERROR, - "EGL: Failed to create surface for user context and EGL_KHR_surfaceless_context not supported"); + "EGL: Failed to find surface config for user context: %s", getEGLErrorString(eglGetError())); + free(context); + return NULL; + } + context->egl.surface = eglCreatePbufferSurface(_glfw.egl.display, dummySurfaceConfig, dummySurfaceAttribs); + if (context->egl.surface == EGL_NO_SURFACE) + { + eglDestroyContext(_glfw.egl.display, context->egl.handle); + _glfwInputError(GLFW_PLATFORM_ERROR, + "EGL: Failed to create surface for user context: %s for %s", getEGLErrorString(eglGetError()), eglQueryString(_glfw.egl.display,0x3054)); free(context); return NULL; } From 87ddca831acb9985f33ad68d0260ac8c6c9dcd67 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Wed, 15 Jul 2020 19:49:35 +0100 Subject: [PATCH 18/37] Removed unused EGL defines. --- src/egl_context.h | 3 --- 1 file changed, 3 deletions(-) diff --git a/src/egl_context.h b/src/egl_context.h index 62801740..5a422d8c 100644 --- a/src/egl_context.h +++ b/src/egl_context.h @@ -74,9 +74,6 @@ #define EGL_PBUFFER_BIT 0x0001 #define EGL_HEIGHT 0x3056 #define EGL_WIDTH 0x3057 -#define EGL_NO_TEXTURE 0x305C -#define EGL_TEXTURE_FORMAT 0x3080 -#define EGL_TEXTURE_TARGET 0x3081 #define EGL_CONTEXT_OPENGL_FORWARD_COMPATIBLE_BIT_KHR 0x00000002 #define EGL_CONTEXT_OPENGL_CORE_PROFILE_BIT_KHR 0x00000001 From 1cb8ab8dc6ec0eda054a0bb18fdbe9626784dc0f Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Thu, 16 Jul 2020 11:59:11 +0100 Subject: [PATCH 19/37] Make user context non current when destroyed --- src/context.c | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/context.c b/src/context.c index 8db76c84..d12ba991 100644 --- a/src/context.c +++ b/src/context.c @@ -775,14 +775,14 @@ GLFWAPI GLFWusercontext* glfwCreateUserContext(GLFWwindow* handle) GLFWAPI void glfwDestroyUserContext(GLFWusercontext* handle) { _GLFWusercontext* context = (_GLFWusercontext*)handle; - _GLFWusercontext* prev = _glfwPlatformGetTls(&_glfw.usercontextSlot); + _GLFWusercontext* current = _glfwPlatformGetTls(&_glfw.usercontextSlot); _GLFW_REQUIRE_INIT(); if (context) { - if(prev==context) - _glfwPlatformSetTls(&_glfw.usercontextSlot,NULL); + if(current==context) + glfwMakeContextCurrent(NULL); context->destroy(context); } From cd68bac78d87d3627f6bd3f3d84d5c68bdfb3a1a Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Thu, 16 Jul 2020 11:59:23 +0100 Subject: [PATCH 20/37] Initial user context documentation --- include/GLFW/glfw3.h | 149 ++++++++++++++++++++++++++++++++++++++----- 1 file changed, 133 insertions(+), 16 deletions(-) diff --git a/include/GLFW/glfw3.h b/include/GLFW/glfw3.h index ece72da8..e8626932 100644 --- a/include/GLFW/glfw3.h +++ b/include/GLFW/glfw3.h @@ -1304,13 +1304,13 @@ typedef struct GLFWmonitor GLFWmonitor; */ typedef struct GLFWwindow GLFWwindow; -/*! @brief Opaque user OpenGL context object. +/*! @brief Opaque user OpenGL & OpenGL ES context object. * - * Opaque user OpenGL context object. + * Opaque user OpenGL OpenGL ES context object. * - * @see @ref user_gl_context + * @see @ref usercontext * - * @since Added in version 3.3. + * @since Added in version 3.4. * * @ingroup window */ @@ -5601,6 +5601,9 @@ GLFWAPI uint64_t glfwGetTimerFrequency(void); * a single thread at a time and each thread can have only a single current * context at a time. * + * Making a context of a window current on a given thread will detach + * any user context which is current on that thread and visa versa. + * * When moving a context between threads, you must make it non-current on the * old thread before making it current on the new one. * @@ -5624,6 +5627,9 @@ GLFWAPI uint64_t glfwGetTimerFrequency(void); * * @sa @ref context_current * @sa @ref glfwGetCurrentContext + * @sa @ref usercontext_current + * @sa @ref glfwMakeUserContextCurrent + * @sa @ref glfwGetCurrentUserContext * * @since Added in version 3.0. * @@ -5812,24 +5818,135 @@ GLFWAPI int glfwExtensionSupported(const char* extension); */ GLFWAPI GLFWglproc glfwGetProcAddress(const char* procname); -/*! @brief Create a new OpenGL user context for a window -* -*/ +/*! @brief Create a new OpenGL or OpenGL ES user context for a window + * + * This function creates a new OpenGL or OpenGL ES user context for a + * window, which can be used to call OpenGL or OpenGL ES functions on + * another thread. For a valid user context the window must be created + * with a [GLFW_CLIENT_API](@ref GLFW_CLIENT_API_hint) other than + * `GLFW_NO_API`. + * + * User context creation uses the window context and framebuffer related + * hints to ensure a valid context is created for that window, these hints + * should be the same at the time of user context creation as when the + * window was created. + * + * Contexts share resources with the window context and with any other + * user context created for that window. + * + * @param[in] window The Window for which the user context is to be + * created. + * @return The handle of the user context created, or `NULL` if an + * [error](@ref error_handling) occurred. + * + * @thread_safety This function must only be called from the main thread. + * + * @sa @ref usercontext_creation + * @sa @ref glfwDestroyUserContext + * @sa @ref window_creation + * @sa @ref glfwCreateWindow + * @sa @ref glfwDestroyWindow + * + * @since Added in version 3.4. + * + * @ingroup context + */ GLFWAPI GLFWusercontext* glfwCreateUserContext(GLFWwindow* window); -/*! @brief Delete an OpenGL user context -* -*/ +/*! @brief Destroys the specified user context + * + * This function destroys the specified user context. + * User contexts should be destroyed before destroying the + * window they were made with. + * + * If the user context is current on the main thread, it is + * detached before being destroyed. + * + * @param[in] context The user context to destroy. + * + * @errors Possible errors include @ref GLFW_NOT_INITIALIZED and @ref + * GLFW_PLATFORM_ERROR. + * + * @note The user context must not be current on any other + * thread when this function is called. + * + * @reentrancy This function must not be called from a callback. + * + * @thread_safety This function must only be called from the main thread. + * + * @sa @ref usercontext_creation + * @sa @ref glfwCreateUserContext + * @sa @ref window_creation + * @sa @ref glfwCreateWindow + * @sa @ref glfwDestroyWindow + * + * @since Added in version 3.4. + * + * @ingroup context + */ GLFWAPI void glfwDestroyUserContext(GLFWusercontext* context); -/*! @brief Make an OpenGL user context -* -*/ +/*! @brief Makes the user context current for the calling thread. + * + * This function makes the OpenGL or OpenGL ES context of the specified user + * context current on the calling thread. A context must only be made current on + * a single thread at a time and each thread can have only a single current + * context at a time. + * + * Making a user context current on a given thread will detach the context of + * any window which is current on that thread and visa versa. + * + * When moving a context between threads, you must make it non-current on the + * old thread before making it current on the new one. + * + * By default, making a context non-current implicitly forces a pipeline flush. + * On machines that support `GL_KHR_context_flush_control`, you can control + * whether a context performs this flush by setting the + * [GLFW_CONTEXT_RELEASE_BEHAVIOR](@ref GLFW_CONTEXT_RELEASE_BEHAVIOR_hint) + * hint. + * + * @param[in] context The user context to make current, or `NULL` to + * detach the current context. + * + * @errors Possible errors include @ref GLFW_NOT_INITIALIZED, + * and @ref GLFW_PLATFORM_ERROR. + * + * @thread_safety This function may be called from any thread. + * + * @sa @ref usercontext_current + * @sa @ref glfwGetCurrentUserContext + * @sa @ref context_current + * @sa @ref glfwMakeContextCurrent + * @sa @ref glfwGetCurrentContext + * + * @since Added in version 3.4. + * + * @ingroup context + */ GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* context); -/*! @brief Get the current OpenGL user context -* -*/ +/*! @brief Returns the current OpenGL or OpenGL ES user context + * + * This function returns the user context which is current + * on the calling thread. + * + * @return The user context current, or `NULL` if no user context + * is current. + * + * @errors Possible errors include @ref GLFW_NOT_INITIALIZED. + * + * @thread_safety This function may be called from any thread. + * + * @sa @ref usercontext_current + * @sa @ref glfwMakeUserContextCurrent + * @sa @ref context_current + * @sa @ref glfwMakeContextCurrent + * @sa @ref glfwGetCurrentContext + * + * @since Added in version 3.4. + * + * @ingroup context + */ GLFWAPI GLFWusercontext* glfwGetCurrentUserContext(void); From 1d647668afc8eb4aa6bfa668180a1de26a8e2146 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Thu, 16 Jul 2020 14:02:32 +0100 Subject: [PATCH 21/37] EGL user context code simplification --- src/egl_context.c | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/egl_context.c b/src/egl_context.c index 9797d63c..7b3e1b88 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -508,7 +508,7 @@ void _glfwTerminateEGL(void) // Create the OpenGL or OpenGL ES context for the window eglConfig // -GLFWbool _glfwCreateContextForConfigEGL(_GLFWwindow* window, +GLFWbool _glfwCreateContextForConfigEGL(EGLConfig eglConfig, const _GLFWctxconfig* ctxconfig, EGLContext* context) { @@ -620,7 +620,7 @@ GLFWbool _glfwCreateContextForConfigEGL(_GLFWwindow* window, setAttrib(EGL_NONE, EGL_NONE); - *context = eglCreateContext(_glfw.egl.display, window->context.egl.config, share, attribs); + *context = eglCreateContext(_glfw.egl.display, eglConfig, share, attribs); if (*context == EGL_NO_CONTEXT) { @@ -650,7 +650,7 @@ GLFWbool _glfwCreateContextEGL(_GLFWwindow* window, return GLFW_FALSE; } - if (!_glfwCreateContextForConfigEGL(window,ctxconfig,&window->context.egl.handle)) + if (!_glfwCreateContextForConfigEGL(window->context.egl.config,ctxconfig,&window->context.egl.handle)) { return GLFW_FALSE; } @@ -869,7 +869,7 @@ _GLFWusercontext* _glfwCreateUserContextEGL(_GLFWwindow* window) ctxconfig = _glfw.hints.context; ctxconfig.share = window; - if (!_glfwCreateContextForConfigEGL(window,&ctxconfig,&context->egl.handle)) + if (!_glfwCreateContextForConfigEGL(window->context.egl.config,&ctxconfig,&context->egl.handle)) { _glfwInputError(GLFW_PLATFORM_ERROR, "EGL: Failed to create user OpenGL context"); From 375fcdeadb6ae1e870777505b1925e34caff5899 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Thu, 16 Jul 2020 15:20:25 +0100 Subject: [PATCH 22/37] Added GLFW_NO_API check to glfwCreateUserContext and error documentation. --- include/GLFW/glfw3.h | 5 +++++ src/context.c | 15 +++++++++++++++ 2 files changed, 20 insertions(+) diff --git a/include/GLFW/glfw3.h b/include/GLFW/glfw3.h index e8626932..a25f4144 100644 --- a/include/GLFW/glfw3.h +++ b/include/GLFW/glfw3.h @@ -5834,6 +5834,11 @@ GLFWAPI GLFWglproc glfwGetProcAddress(const char* procname); * Contexts share resources with the window context and with any other * user context created for that window. * + * @errors Possible errors include @ref GLFW_NOT_INITIALIZED, + * @ref GLFW_INVALID_VALUE the window parameter is `NULL`, + * @ref GLFW_NO_WINDOW_CONTEXT if the window has no OpenGL or + * OpenGL US context, and @ref GLFW_PLATFORM_ERROR. + * * @param[in] window The Window for which the user context is to be * created. * @return The handle of the user context created, or `NULL` if an diff --git a/src/context.c b/src/context.c index d12ba991..a3057e92 100644 --- a/src/context.c +++ b/src/context.c @@ -767,6 +767,21 @@ GLFWAPI GLFWusercontext* glfwCreateUserContext(GLFWwindow* handle) _GLFW_REQUIRE_INIT_OR_RETURN(NULL); + if (!window) + { + _glfwInputError(GLFW_INVALID_VALUE, + "Cannot create a user context without a valid window handle"); + return NULL; + } + + if (window->context.client == GLFW_NO_API) + { + _glfwInputError(GLFW_NO_WINDOW_CONTEXT, + "Cannot create a user context for a window that has no OpenGL or OpenGL ES context"); + return NULL; + } + + context = _glfwPlatformCreateUserContext(window); return (GLFWusercontext*)context; From e91b1820dd441dc064a3fb1774da7b807a396400 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 23 Mar 2021 12:37:54 +0000 Subject: [PATCH 23/37] Modified clear colour of usercontext test to remove transparency --- tests/usercontext.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/usercontext.c b/tests/usercontext.c index 6bb3bd73..1420df32 100644 --- a/tests/usercontext.c +++ b/tests/usercontext.c @@ -72,7 +72,7 @@ int main(void) return -1; } - glClearColor( 0.4f, 0.3f, 0.4f, 0.0f ); + glClearColor( 0.4f, 0.3f, 0.4f, 1.0f ); /* Loop until the user closes the window */ From 350aebf20a6600551ddcbd4673d3a8fa2c8fb0b1 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 23 Mar 2021 12:53:17 +0000 Subject: [PATCH 24/37] User contexts: added change log entry in README.md, listing all new symbols --- README.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/README.md b/README.md index 26e88828..5849b762 100644 --- a/README.md +++ b/README.md @@ -116,6 +116,9 @@ information on what to include when reporting a bug. ## Changelog + - Added OpenGL and OpenGL ES user contexts for multiple window contexts via + `GLFWusercontext`, `glfwCreateUserContext`, `glfwDestroyUserContext`, + `glfwMakeUserContextCurrent`, `glfwGetCurrentUserContext` (#1687) - Added `GLFW_RESIZE_NWSE_CURSOR`, `GLFW_RESIZE_NESW_CURSOR`, `GLFW_RESIZE_ALL_CURSOR` and `GLFW_NOT_ALLOWED_CURSOR` cursor shapes (#427) - Added `GLFW_RESIZE_EW_CURSOR` alias for `GLFW_HRESIZE_CURSOR` (#427) From 78749928888802703dcdc063df68bfc0a91757d0 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 23 Mar 2021 13:09:17 +0000 Subject: [PATCH 25/37] User contexts: added news entry --- docs/news.dox | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/docs/news.dox b/docs/news.dox index 19df848a..064230fc 100644 --- a/docs/news.dox +++ b/docs/news.dox @@ -9,6 +9,15 @@ @subsection features_34 New features in version 3.4 +@subsubsection user_context_34 Multiple window contexts + +GLFW now provides the ability to create multiple OpenGL and OpenGL ES +contexts for a given window. Called user contexts, a [GLFWusercontext](@ref usercontext) +can be created using @ref glfwCreateUserContext, +destroyed using @ref glfwDestroyUserContext, and managed with +@ref glfwMakeUserContextCurrent and @ref glfwGetCurrentUserContext. +For more information see the [user context](@ref usercontext) documentation. + @subsubsection standard_cursors_34 More standard cursors GLFW now provides the standard cursor shapes @ref GLFW_RESIZE_NWSE_CURSOR and From 1ba83ede448058c1da58966a87ff730e6a6182fe Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 23 Mar 2021 14:40:54 +0000 Subject: [PATCH 26/37] User contexts: added context documentation and updated references --- docs/context.dox | 42 ++++++++++++++++++++++++++++++++++++++++++ docs/news.dox | 6 +++--- include/GLFW/glfw3.h | 12 ++++++++---- 3 files changed, 53 insertions(+), 7 deletions(-) diff --git a/docs/context.dox b/docs/context.dox index dd952122..eefe3a67 100644 --- a/docs/context.dox +++ b/docs/context.dox @@ -92,6 +92,28 @@ You can disable context creation by setting the without contexts must not be passed to @ref glfwMakeContextCurrent or @ref glfwSwapBuffers. +@subsection context_user User contexts for multi context windows + +GLFW supports multiple OpenGL or OpenGL ES contexts per window. Providing +a window with an existing OpenGL or OpenGL ES context has been created further +user contexts can be created using @ref glfwCreateUserContext with the same +API sharing the window context objects. + +@code +GLFWusercontext* usercontext = glfwCreateUserContext(window); + +/* make the user context current */ +glfwMakeUserContextCurrent(usercontext); + +/* make the window context current */ +glfwMakeContextCurrent(window); + +/* destroy the user context */ +glfwDestroyUserContext(usercontext); + +@endcode + +User contexts See also the test program `usercontext`. @section context_current Current context @@ -122,6 +144,26 @@ error. - @ref glfwExtensionSupported - @ref glfwGetProcAddress +@subsection context_current_user Current user context + +When using [user contexts](@ref context_user) the user context can be +made current using @ref glfwMakeUserContextCurrent. + +@code +glfwMakeUserContextCurrent(usercontext); +@endcode + +This makes the any window context non-current on the calling thread, such that +a call to @ref glfwGetCurrentContext will return `NULL`. + +The current user context is returned by @ref glfwGetCurrentUserContext. + +@code +GLFWusercontext* usercontext = glfwGetCurrentUserContext(); +@endcode + +This will return the current user context or `NULL` if either the main window context +or no context is current. @section context_swap Buffer swapping diff --git a/docs/news.dox b/docs/news.dox index 064230fc..a5e02535 100644 --- a/docs/news.dox +++ b/docs/news.dox @@ -9,14 +9,14 @@ @subsection features_34 New features in version 3.4 -@subsubsection user_context_34 Multiple window contexts +@subsubsection context_user_34 Multiple window contexts GLFW now provides the ability to create multiple OpenGL and OpenGL ES -contexts for a given window. Called user contexts, a [GLFWusercontext](@ref usercontext) +contexts for a given window. Called user contexts, a [GLFWusercontext](@ref context_user) can be created using @ref glfwCreateUserContext, destroyed using @ref glfwDestroyUserContext, and managed with @ref glfwMakeUserContextCurrent and @ref glfwGetCurrentUserContext. -For more information see the [user context](@ref usercontext) documentation. +For more information see the [user context](@ref context_user) documentation. @subsubsection standard_cursors_34 More standard cursors diff --git a/include/GLFW/glfw3.h b/include/GLFW/glfw3.h index 6540cf38..e3723380 100644 --- a/include/GLFW/glfw3.h +++ b/include/GLFW/glfw3.h @@ -1315,7 +1315,7 @@ typedef struct GLFWwindow GLFWwindow; * * Opaque user OpenGL OpenGL ES context object. * - * @see @ref usercontext + * @see @ref context_user * * @since Added in version 3.4. * @@ -5635,7 +5635,7 @@ GLFWAPI uint64_t glfwGetTimerFrequency(void); * * @sa @ref context_current * @sa @ref glfwGetCurrentContext - * @sa @ref usercontext_current + * @sa @ref context_current_user * @sa @ref glfwMakeUserContextCurrent * @sa @ref glfwGetCurrentUserContext * @@ -5854,6 +5854,7 @@ GLFWAPI GLFWglproc glfwGetProcAddress(const char* procname); * * @thread_safety This function must only be called from the main thread. * + * @sa @ref context_user * @sa @ref usercontext_creation * @sa @ref glfwDestroyUserContext * @sa @ref window_creation @@ -5887,6 +5888,7 @@ GLFWAPI GLFWusercontext* glfwCreateUserContext(GLFWwindow* window); * * @thread_safety This function must only be called from the main thread. * + * @sa @ref context_user * @sa @ref usercontext_creation * @sa @ref glfwCreateUserContext * @sa @ref window_creation @@ -5926,7 +5928,8 @@ GLFWAPI void glfwDestroyUserContext(GLFWusercontext* context); * * @thread_safety This function may be called from any thread. * - * @sa @ref usercontext_current + * @sa @ref context_user + * @sa @ref context_current_user * @sa @ref glfwGetCurrentUserContext * @sa @ref context_current * @sa @ref glfwMakeContextCurrent @@ -5950,7 +5953,8 @@ GLFWAPI void glfwMakeUserContextCurrent(GLFWusercontext* context); * * @thread_safety This function may be called from any thread. * - * @sa @ref usercontext_current + * @sa @ref context_user + * @sa @ref context_current_user * @sa @ref glfwMakeUserContextCurrent * @sa @ref context_current * @sa @ref glfwMakeContextCurrent From c01aa3fec42fb52e63e93d85e640f581e98ce27c Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 23 Mar 2021 15:56:56 +0000 Subject: [PATCH 27/37] Removed trailing white space --- include/GLFW/glfw3.h | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/include/GLFW/glfw3.h b/include/GLFW/glfw3.h index e3723380..7ea00509 100644 --- a/include/GLFW/glfw3.h +++ b/include/GLFW/glfw3.h @@ -5833,12 +5833,12 @@ GLFWAPI GLFWglproc glfwGetProcAddress(const char* procname); * another thread. For a valid user context the window must be created * with a [GLFW_CLIENT_API](@ref GLFW_CLIENT_API_hint) other than * `GLFW_NO_API`. - * + * * User context creation uses the window context and framebuffer related * hints to ensure a valid context is created for that window, these hints * should be the same at the time of user context creation as when the * window was created. - * + * * Contexts share resources with the window context and with any other * user context created for that window. * From 7b6f25c360e2e4d3fa88698769979de151093ed1 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 23 Mar 2021 16:11:07 +0000 Subject: [PATCH 28/37] Removed more trailing white space --- src/egl_context.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/egl_context.c b/src/egl_context.c index 7b3e1b88..6359733c 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -840,7 +840,7 @@ static void _glfwDestroyUserContextEGL(_GLFWusercontext* context) { if (context->egl.surface!=EGL_NO_SURFACE) eglDestroySurface(_glfw.egl.display,context->egl.surface); - + eglDestroyContext(_glfw.egl.display, context->egl.handle); free(context); } From b0d196c43caeec092d7f41a77e19e92b8d9af8a5 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 23 Mar 2021 18:45:29 +0000 Subject: [PATCH 29/37] Formatting consistency fixes --- src/egl_context.c | 2 +- tests/usercontext.c | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/src/egl_context.c b/src/egl_context.c index 6359733c..cd155827 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -881,7 +881,7 @@ _GLFWusercontext* _glfwCreateUserContextEGL(_GLFWwindow* window) else { eglChooseConfig(_glfw.egl.display, dummyConfigAttribs, &dummySurfaceConfig, 1, &dummySurfaceNumConfigs); - if( !dummySurfaceNumConfigs) + if (!dummySurfaceNumConfigs) { eglDestroyContext(_glfw.egl.display, context->egl.handle); _glfwInputError(GLFW_PLATFORM_ERROR, diff --git a/tests/usercontext.c b/tests/usercontext.c index 1420df32..c9c1db1a 100644 --- a/tests/usercontext.c +++ b/tests/usercontext.c @@ -43,13 +43,13 @@ int main(void) /* set the user context current */ glfwMakeUserContextCurrent(usercontext); - if (glfwGetCurrentContext()!=NULL) + if (glfwGetCurrentContext() != NULL) { fprintf(stderr, "Current glfw window context not NULL after glfwMakeUserContextCurrent\n"); glfwTerminate(); return -1; } - if (glfwGetCurrentUserContext()!=usercontext) + if (glfwGetCurrentUserContext() != usercontext) { fprintf(stderr, "Current user context not correct after glfwMakeUserContextCurrent\n"); glfwTerminate(); @@ -59,13 +59,13 @@ int main(void) /* set the window context current */ glfwMakeContextCurrent(window); - if ( glfwGetCurrentUserContext() != NULL ) + if (glfwGetCurrentUserContext() != NULL) { fprintf(stderr, "Current user context not NULL after glfwMakeContextCurrent\n"); glfwTerminate(); return -1; } - if ( glfwGetCurrentContext() != window ) + if (glfwGetCurrentContext() != window) { fprintf(stderr, "Current glfw window context not correct after glfwMakeContextCurrent\n"); glfwTerminate(); From b88a8c93649881ad651d227b51aedd89aa8a359b Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Sat, 4 Sep 2021 12:41:39 +0100 Subject: [PATCH 30/37] Fixed usercontext glad use after merge --- tests/usercontext.c | 2 ++ 1 file changed, 2 insertions(+) diff --git a/tests/usercontext.c b/tests/usercontext.c index c9c1db1a..22c39a47 100644 --- a/tests/usercontext.c +++ b/tests/usercontext.c @@ -1,4 +1,6 @@ +#define GLAD_GL_IMPLEMENTATION #include +#define GLFW_INCLUDE_NONE #include #include From 9dfa1649ff1000432e2c082c91ee07e2dc233305 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Thu, 31 Mar 2022 16:59:01 +0100 Subject: [PATCH 31/37] UserContext: Null platform fixes --- src/null_init.c | 1 + src/null_platform.h | 2 ++ src/null_window.c | 2 +- 3 files changed, 4 insertions(+), 1 deletion(-) diff --git a/src/null_init.c b/src/null_init.c index de4b28f3..f914bb75 100644 --- a/src/null_init.c +++ b/src/null_init.c @@ -106,6 +106,7 @@ GLFWbool _glfwConnectNull(int platformID, _GLFWplatform* platform) _glfwWaitEventsNull, _glfwWaitEventsTimeoutNull, _glfwPostEmptyEventNull, + _glfwCreateUserContextNull, _glfwGetEGLPlatformNull, _glfwGetEGLNativeDisplayNull, _glfwGetEGLNativeWindowNull, diff --git a/src/null_platform.h b/src/null_platform.h index 4c25b112..f7a94c0f 100644 --- a/src/null_platform.h +++ b/src/null_platform.h @@ -147,3 +147,5 @@ VkResult _glfwCreateWindowSurfaceNull(VkInstance instance, _GLFWwindow* window, void _glfwPollMonitorsNull(void); +_GLFWusercontext* _glfwCreateUserContextNull(_GLFWwindow* window); + diff --git a/src/null_window.c b/src/null_window.c index c7370e61..cb54fa68 100644 --- a/src/null_window.c +++ b/src/null_window.c @@ -691,7 +691,7 @@ VkResult _glfwCreateWindowSurfaceNull(VkInstance instance, return VK_ERROR_EXTENSION_NOT_PRESENT; } -_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) +_GLFWusercontext* _glfwCreateUserContextNull(_GLFWwindow* window) { if (window->context.osmesa.handle) { From 3b003dd5f03dad665783620bb650c01676d99963 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Thu, 31 Mar 2022 19:21:42 +0100 Subject: [PATCH 32/37] UserContext: error return for CreateUserContext should be NULL --- src/cocoa_window.m | 2 +- src/null_window.c | 2 +- src/wl_window.c | 2 +- src/x11_window.c | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/cocoa_window.m b/src/cocoa_window.m index d85fac1e..521f8179 100644 --- a/src/cocoa_window.m +++ b/src/cocoa_window.m @@ -1928,7 +1928,7 @@ _GLFWusercontext* _glfwCreateUserContextCocoa(_GLFWwindow* window) return _glfwCreateUserContextOSMesa(window); } - return GLFW_FALSE; + return NULL; } diff --git a/src/null_window.c b/src/null_window.c index cb54fa68..270c8b2d 100644 --- a/src/null_window.c +++ b/src/null_window.c @@ -698,5 +698,5 @@ _GLFWusercontext* _glfwCreateUserContextNull(_GLFWwindow* window) return _glfwCreateUserContextOSMesa(window); } - return GLFW_FALSE; + return NULL; } diff --git a/src/wl_window.c b/src/wl_window.c index dc393bac..165a1bd4 100644 --- a/src/wl_window.c +++ b/src/wl_window.c @@ -1871,7 +1871,7 @@ _GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) return _glfwCreateUserContextOSMesa(window); } - return GLFW_FALSE; + return NULL; } ////////////////////////////////////////////////////////////////////////// diff --git a/src/x11_window.c b/src/x11_window.c index 1506125a..78e2cb83 100644 --- a/src/x11_window.c +++ b/src/x11_window.c @@ -3262,7 +3262,7 @@ _GLFWusercontext* _glfwCreateUserContextX11(_GLFWwindow* window) return _glfwCreateUserContextOSMesa(window); } - return GLFW_FALSE; + return NULL; } From 23d31b805a681213f3b4d1d37c26e2304d6dc771 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Thu, 31 Mar 2022 19:27:11 +0100 Subject: [PATCH 33/37] UserContext: fixed x11 platform --- src/platform.h | 2 +- src/x11_platform.h | 3 ++- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/src/platform.h b/src/platform.h index c68e987d..9f8e3e00 100644 --- a/src/platform.h +++ b/src/platform.h @@ -183,4 +183,4 @@ #define GLFW_PLATFORM_USER_CONTEXT_STATE \ GLFW_WGL_USER_CONTEXT_STATE \ GLFW_NSGL_USER_CONTEXT_STATE \ - GLFW_GLX_USER_CONTEXT_STATE \ No newline at end of file + GLFW_GLX_USER_CONTEXT_STATE diff --git a/src/x11_platform.h b/src/x11_platform.h index eaa11ada..95eeed05 100644 --- a/src/x11_platform.h +++ b/src/x11_platform.h @@ -470,6 +470,7 @@ typedef struct _GLFWcontextGLX { GLXContext handle; GLXWindow window; + GLXFBConfig fbconfig; } _GLFWcontextGLX; // GLX-specific global data @@ -1009,4 +1010,4 @@ GLFWbool _glfwChooseVisualGLX(const _GLFWwndconfig* wndconfig, Visual** visual, int* depth); _GLFWusercontext* _glfwCreateUserContextX11(_GLFWwindow* window); -_GLFWusercontext* _glfwCreateUserContextWGL(_GLFWwindow* window); +_GLFWusercontext* _glfwCreateUserContextGLX(_GLFWwindow* window); From dcf9a51b78791f831b05787adc494cd5930a3b1d Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Thu, 31 Mar 2022 19:53:09 +0100 Subject: [PATCH 34/37] UserContext: fixed wayland platform --- src/wl_init.c | 1 + src/wl_platform.h | 1 + src/wl_window.c | 2 +- 3 files changed, 3 insertions(+), 1 deletion(-) diff --git a/src/wl_init.c b/src/wl_init.c index f02c6320..16570829 100644 --- a/src/wl_init.c +++ b/src/wl_init.c @@ -1102,6 +1102,7 @@ GLFWbool _glfwConnectWayland(int platformID, _GLFWplatform* platform) _glfwWaitEventsWayland, _glfwWaitEventsTimeoutWayland, _glfwPostEmptyEventWayland, + _glfwCreateUserContextWayland, _glfwGetEGLPlatformWayland, _glfwGetEGLNativeDisplayWayland, _glfwGetEGLNativeWindowWayland, diff --git a/src/wl_platform.h b/src/wl_platform.h index ba405714..2b84a10c 100644 --- a/src/wl_platform.h +++ b/src/wl_platform.h @@ -498,3 +498,4 @@ void _glfwSetGammaRampWayland(_GLFWmonitor* monitor, const GLFWgammaramp* ramp); void _glfwAddOutputWayland(uint32_t name, uint32_t version); GLFWbool _glfwInputTextWayland(_GLFWwindow* window, uint32_t scancode); +_GLFWusercontext* _glfwCreateUserContextWayland(_GLFWwindow* window); diff --git a/src/wl_window.c b/src/wl_window.c index 165a1bd4..f3f5648a 100644 --- a/src/wl_window.c +++ b/src/wl_window.c @@ -1860,7 +1860,7 @@ VkResult _glfwCreateWindowSurfaceWayland(VkInstance instance, return err; } -_GLFWusercontext* _glfwPlatformCreateUserContext(_GLFWwindow* window) +_GLFWusercontext* _glfwCreateUserContextWayland(_GLFWwindow* window) { if (window->context.egl.handle) { From 7de1c5c10a6340ff46dc774bccf9e8091c6662b8 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Thu, 9 Jun 2022 13:55:33 +0100 Subject: [PATCH 35/37] Minor formating fix --- src/cocoa_init.m | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/cocoa_init.m b/src/cocoa_init.m index 46892515..67931159 100644 --- a/src/cocoa_init.m +++ b/src/cocoa_init.m @@ -550,7 +550,7 @@ GLFWbool _glfwConnectCocoa(int platformID, _GLFWplatform* platform) _glfwWaitEventsCocoa, _glfwWaitEventsTimeoutCocoa, _glfwPostEmptyEventCocoa, - _glfwCreateUserContextCocoa, + _glfwCreateUserContextCocoa, _glfwGetEGLPlatformCocoa, _glfwGetEGLNativeDisplayCocoa, _glfwGetEGLNativeWindowCocoa, From 58aad6c13665fb44fa1cf21902c10c8d6153a135 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Mon, 10 Feb 2025 18:36:01 +0000 Subject: [PATCH 36/37] Fix for not being able to clear the user context --- src/context.c | 9 ++++++++- src/egl_context.c | 34 +++++++++++++++++++++++++--------- src/glx_context.c | 22 +++++++++++++++++----- src/nsgl_context.m | 7 +++++-- src/osmesa_context.c | 19 +++++++++++-------- src/wgl_context.c | 22 +++++++++++++++++----- 6 files changed, 83 insertions(+), 30 deletions(-) diff --git a/src/context.c b/src/context.c index b4675267..234465cb 100644 --- a/src/context.c +++ b/src/context.c @@ -619,8 +619,9 @@ GLFWAPI void glfwMakeContextCurrent(GLFWwindow* handle) _GLFWwindow* window = (_GLFWwindow*) handle; _GLFWwindow* previous; + _GLFWusercontext* previousUserContext; - _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); + previousUserContext = _glfwPlatformGetTls(&_glfw.usercontextSlot); previous = _glfwPlatformGetTls(&_glfw.contextSlot); if (window && window->context.client == GLFW_NO_API) @@ -630,6 +631,12 @@ GLFWAPI void glfwMakeContextCurrent(GLFWwindow* handle) return; } + if (previousUserContext) + { + assert(previous==NULL); + previousUserContext->makeCurrent(NULL); + } + if (previous) { if (!window || window->context.source != previous->context.source) diff --git a/src/egl_context.c b/src/egl_context.c index a82c3271..36aee66e 100644 --- a/src/egl_context.c +++ b/src/egl_context.c @@ -919,16 +919,32 @@ GLFWbool _glfwChooseVisualEGL(const _GLFWwndconfig* wndconfig, static void _glfwMakeUserContextCurrentEGL(_GLFWusercontext* context) { - if (!eglMakeCurrent(_glfw.egl.display, - context->egl.surface, - context->egl.surface, - context->egl.handle)) + if (context) { - _glfwInputError(GLFW_PLATFORM_ERROR, - "EGL: Failed to make user context current: %s", - getEGLErrorString(eglGetError())); - _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); - return; + if (!eglMakeCurrent(_glfw.egl.display, + context->egl.surface, + context->egl.surface, + context->egl.handle)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "EGL: Failed to make user context current: %s", + getEGLErrorString(eglGetError())); + _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); + return; + } + } + else + { + if (!eglMakeCurrent(_glfw.egl.display, + EGL_NO_SURFACE, + EGL_NO_SURFACE, + EGL_NO_CONTEXT)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "EGL: Failed to clear current user context: %s", + getEGLErrorString(eglGetError())); + return; + } } _glfwPlatformSetTls(&_glfw.usercontextSlot, context); } diff --git a/src/glx_context.c b/src/glx_context.c index e70ebfcc..ec8a033a 100644 --- a/src/glx_context.c +++ b/src/glx_context.c @@ -687,12 +687,24 @@ GLFWbool _glfwChooseVisualGLX(const _GLFWwndconfig* wndconfig, static void _glfwMakeUserContextCurrentGLX(_GLFWusercontext* context) { - if(!glXMakeCurrent(_glfw.x11.display, context->window->context.glx.window,context->glx.handle)) + if (context) + { + if(!glXMakeCurrent(_glfw.x11.display, context->window->context.glx.window,context->glx.handle)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "GLX: Failed to make user context current"); + _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); + return; + } + } + else { - _glfwInputError(GLFW_PLATFORM_ERROR, - "GLX: Failed to make user context current"); - _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); - return; + if (!glXMakeCurrent(_glfw.x11.display, None, NULL)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "GLX: Failed to clear current user context"); + return; + } } _glfwPlatformSetTls(&_glfw.usercontextSlot, context); } diff --git a/src/nsgl_context.m b/src/nsgl_context.m index 74fbbcd9..dc68d96a 100644 --- a/src/nsgl_context.m +++ b/src/nsgl_context.m @@ -357,9 +357,12 @@ static void _glfwMakeUserContextCurrentNSGL(_GLFWusercontext* context) { @autoreleasepool { + if (context) [context->nsgl.object makeCurrentContext]; + else + [NSOpenGLContext clearCurrentContext]; - _glfwPlatformSetTls(&_glfw.usercontextSlot, context); + _glfwPlatformSetTls(&_glfw.usercontextSlot, context); } // autoreleasepool } @@ -368,7 +371,7 @@ static void _glfwDestroyUserContextNSGL(_GLFWusercontext* context) { @autoreleasepool { - [context->nsgl.object release]; + [context->nsgl.object release]; } // autoreleasepool free(context); diff --git a/src/osmesa_context.c b/src/osmesa_context.c index fcd978c5..f482d0fd 100644 --- a/src/osmesa_context.c +++ b/src/osmesa_context.c @@ -300,15 +300,18 @@ GLFWbool _glfwCreateContextOSMesa(_GLFWwindow* window, static void _glfwMakeUserContextCurrentOSMesa(_GLFWusercontext* context) { - if (!OSMesaMakeCurrent(context->osmesa.handle, - context->window->context.osmesa.buffer, - GL_UNSIGNED_BYTE, - context->window->context.osmesa.width, context->window->context.osmesa.height)) + if (context) { - _glfwInputError(GLFW_PLATFORM_ERROR, - "OSMesa: Failed to make user context current"); - _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); - return; + if (!OSMesaMakeCurrent(context->osmesa.handle, + context->window->context.osmesa.buffer, + GL_UNSIGNED_BYTE, + context->window->context.osmesa.width, context->window->context.osmesa.height)) + { + _glfwInputError(GLFW_PLATFORM_ERROR, + "OSMesa: Failed to make user context current"); + _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); + return; + } } _glfwPlatformSetTls(&_glfw.usercontextSlot, context); } diff --git a/src/wgl_context.c b/src/wgl_context.c index de2ca0a5..057c6f5a 100644 --- a/src/wgl_context.c +++ b/src/wgl_context.c @@ -787,13 +787,25 @@ GLFWbool _glfwCreateContextWGL(_GLFWwindow* window, static void _glfwMakeUserContextCurrentWGL(_GLFWusercontext* context) { - if(!wglMakeCurrent(context->window->context.wgl.dc,context->wgl.handle)) + if (context) { - _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, - "WGL: Failed to make user context current"); - _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); - return; + if (!wglMakeCurrent(context->window->context.wgl.dc,context->wgl.handle)) + { + _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, + "WGL: Failed to make user context current"); + _glfwPlatformSetTls(&_glfw.usercontextSlot, NULL); + return; + } } + else + { + if (!wglMakeCurrent(NULL, NULL)) + { + _glfwInputErrorWin32(GLFW_PLATFORM_ERROR, + "WGL: Failed to clear current user context"); + } + } + _glfwPlatformSetTls(&_glfw.usercontextSlot, context); } From e2f9340a8cac6b20bff0bad3f9e447b9a41cf076 Mon Sep 17 00:00:00 2001 From: Doug Binks Date: Tue, 11 Feb 2025 16:17:45 +0000 Subject: [PATCH 37/37] Improved usercontext test with threading --- tests/CMakeLists.txt | 2 +- tests/usercontext.c | 110 +++++++++++++++++++++++++++++++++++++++++-- 2 files changed, 108 insertions(+), 4 deletions(-) diff --git a/tests/CMakeLists.txt b/tests/CMakeLists.txt index ad73a034..de9e4d16 100644 --- a/tests/CMakeLists.txt +++ b/tests/CMakeLists.txt @@ -28,7 +28,7 @@ add_executable(iconify iconify.c ${GETOPT} ${GLAD_GL}) add_executable(monitors monitors.c ${GETOPT} ${GLAD_GL}) add_executable(reopen reopen.c ${GLAD_GL}) add_executable(cursor cursor.c ${GLAD_GL}) -add_executable(usercontext usercontext.c ${GLAD_GL}) +add_executable(usercontext usercontext.c ${TINYCTHREAD} ${GLAD_GL}) add_executable(empty WIN32 MACOSX_BUNDLE empty.c ${TINYCTHREAD} ${GLAD_GL}) add_executable(gamma WIN32 MACOSX_BUNDLE gamma.c ${GLAD_GL}) diff --git a/tests/usercontext.c b/tests/usercontext.c index 22c39a47..4a819540 100644 --- a/tests/usercontext.c +++ b/tests/usercontext.c @@ -1,3 +1,35 @@ +//======================================================================== +// User context test +// Copyright (c) Camilla Löwy +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would +// be appreciated but is not required. +// +// 2. Altered source versions must be plainly marked as such, and must not +// be misrepresented as being the original software. +// +// 3. This notice may not be removed or altered from any source +// distribution. +// +//======================================================================== +// +// This test is intended to verify whether the OpenGL user context part of +// the GLFW API is able to be used from multiple threads +// +//======================================================================== + +#include "tinycthread.h" + #define GLAD_GL_IMPLEMENTATION #include #define GLFW_INCLUDE_NONE @@ -9,10 +41,44 @@ static void error_callback(int error, const char* description) fprintf(stderr, "Error: %s\n", description); } +static int thread_main(void* data) +{ + GLFWusercontext* usercontext = (GLFWusercontext*)data; + + /* set the user context current */ + glfwMakeUserContextCurrent(usercontext); + + if (glfwGetCurrentContext() != NULL) + { + fprintf(stderr, "Current glfw window context not NULL after glfwMakeUserContextCurrent\n"); + glfwTerminate(); + return -1; + } + if (glfwGetCurrentUserContext() != usercontext) + { + fprintf(stderr, "Current user context not correct after glfwMakeUserContextCurrent\n"); + glfwTerminate(); + return -1; + } + + /* set the user context to NULL */ + glfwMakeUserContextCurrent(NULL); + if (glfwGetCurrentUserContext() != NULL) + { + fprintf(stderr, "Current user context not NULL after glfwMakeContextCurrent\n"); + glfwTerminate(); + return -1; + } + + return 0; +} + int main(void) { GLFWwindow* window; GLFWusercontext* usercontext; + thrd_t thread_id; + int result, count; glfwSetErrorCallback(error_callback); @@ -41,7 +107,6 @@ int main(void) return -1; } - /* set the user context current */ glfwMakeUserContextCurrent(usercontext); @@ -76,9 +141,19 @@ int main(void) glClearColor( 0.4f, 0.3f, 0.4f, 1.0f ); + // Launch a thread which should create and use the usercontext + if (thrd_create(&thread_id, thread_main, usercontext ) != + thrd_success) + { + fprintf(stderr, "Failed to create secondary thread\n"); - /* Loop until the user closes the window */ - while (!glfwWindowShouldClose(window)) + glfwTerminate(); + exit(EXIT_FAILURE); + } + + /* Loop 60 times or until the user closes the window */ + count = 0; + while (!glfwWindowShouldClose(window) && count++ < 60) { /* Render here */ glClear(GL_COLOR_BUFFER_BIT); @@ -90,6 +165,35 @@ int main(void) glfwPollEvents(); } + thrd_join(thread_id, &result); + + /* One more test now the thread has joined */ + + /* set the user context current */ + glfwMakeUserContextCurrent(usercontext); + + if (glfwGetCurrentContext() != NULL) + { + fprintf(stderr, "Current glfw window context not NULL after glfwMakeUserContextCurrent\n"); + glfwTerminate(); + return -1; + } + if (glfwGetCurrentUserContext() != usercontext) + { + fprintf(stderr, "Current user context not correct after glfwMakeUserContextCurrent\n"); + glfwTerminate(); + return -1; + } + + /* set the user context to NULL */ + glfwMakeUserContextCurrent(NULL); + if (glfwGetCurrentUserContext() != NULL) + { + fprintf(stderr, "Current user context not NULL after glfwMakeContextCurrent\n"); + glfwTerminate(); + return -1; + } + glfwDestroyUserContext(usercontext); glfwTerminate(); return 0;