Merge branch 'wip/otte/for-main' into 'main'

gl: Fix initial EGL context creation on X11

Closes #6964

See merge request GNOME/gtk!7663
This commit is contained in:
Benjamin Otte 2024-08-29 00:36:19 +00:00
commit 328020b241
7 changed files with 47 additions and 28 deletions

View File

@ -2037,8 +2037,11 @@ gdk_gl_context_clear_current (void)
* *
* Does a gdk_gl_context_clear_current() if the current context is attached * Does a gdk_gl_context_clear_current() if the current context is attached
* to @surface, leaves the current context alone otherwise. * to @surface, leaves the current context alone otherwise.
*
* Returns: (nullable) (transfer full): The context that was cleared, so that it can be
* re-made current later
**/ **/
void GdkGLContext *
gdk_gl_context_clear_current_if_surface (GdkSurface *surface) gdk_gl_context_clear_current_if_surface (GdkSurface *surface)
{ {
MaskedContext *current; MaskedContext *current;
@ -2049,11 +2052,20 @@ gdk_gl_context_clear_current_if_surface (GdkSurface *surface)
GdkGLContext *context = unmask_context (current); GdkGLContext *context = unmask_context (current);
if (gdk_gl_context_get_surface (context) != surface) if (gdk_gl_context_get_surface (context) != surface)
return; return NULL;
g_object_ref (context);
if (GDK_GL_CONTEXT_GET_CLASS (context)->clear_current (context)) if (GDK_GL_CONTEXT_GET_CLASS (context)->clear_current (context))
{
g_private_replace (&thread_current_context, NULL); g_private_replace (&thread_current_context, NULL);
return context;
} }
g_object_unref (context);
}
return NULL;
} }
/** /**

View File

@ -120,7 +120,7 @@ gboolean gdk_gl_backend_can_be_used (GdkGLBackend
GError **error); GError **error);
void gdk_gl_backend_use (GdkGLBackend backend_type); void gdk_gl_backend_use (GdkGLBackend backend_type);
void gdk_gl_context_clear_current_if_surface (GdkSurface *surface); GdkGLContext * gdk_gl_context_clear_current_if_surface (GdkSurface *surface) G_GNUC_WARN_UNUSED_RESULT;
GdkGLContext * gdk_gl_context_new (GdkDisplay *display, GdkGLContext * gdk_gl_context_new (GdkDisplay *display,
GdkSurface *surface, GdkSurface *surface,

View File

@ -1136,6 +1136,7 @@ gdk_surface_set_egl_native_window (GdkSurface *self,
{ {
#ifdef HAVE_EGL #ifdef HAVE_EGL
GdkSurfacePrivate *priv = gdk_surface_get_instance_private (self); GdkSurfacePrivate *priv = gdk_surface_get_instance_private (self);
GdkGLContext *current = NULL;
/* This checks that all EGL platforms we support conform to the same struct sizes. /* This checks that all EGL platforms we support conform to the same struct sizes.
* When this ever fails, there will be some fun times happening for whoever tries * When this ever fails, there will be some fun times happening for whoever tries
@ -1146,13 +1147,19 @@ gdk_surface_set_egl_native_window (GdkSurface *self,
{ {
GdkDisplay *display = gdk_surface_get_display (self); GdkDisplay *display = gdk_surface_get_display (self);
current = gdk_gl_context_clear_current_if_surface (self);
eglDestroySurface (gdk_display_get_egl_display (display), priv->egl_surface); eglDestroySurface (gdk_display_get_egl_display (display), priv->egl_surface);
priv->egl_surface = NULL; priv->egl_surface = NULL;
} }
gdk_gl_context_clear_current_if_surface (self);
priv->egl_native_window = native_window; priv->egl_native_window = native_window;
if (current)
{
gdk_gl_context_make_current (current);
g_object_unref (current);
}
} }
gpointer /* EGLSurface */ gpointer /* EGLSurface */
@ -1180,20 +1187,18 @@ gdk_surface_ensure_egl_surface (GdkSurface *self,
depth = priv->egl_surface_depth; depth = priv->egl_surface_depth;
} }
if (priv->egl_surface_depth != depth && if (priv->egl_surface == NULL ||
priv->egl_surface != NULL && (priv->egl_surface != NULL &&
gdk_display_get_egl_config (display, priv->egl_surface_depth) != gdk_display_get_egl_config (display, depth)) gdk_display_get_egl_config (display, priv->egl_surface_depth) != gdk_display_get_egl_config (display, depth)))
{
gdk_gl_context_clear_current_if_surface (self);
eglDestroySurface (gdk_display_get_egl_display (display), priv->egl_surface);
priv->egl_surface = NULL;
}
if (priv->egl_surface == NULL)
{ {
GdkGLContext *cleared;
EGLint attribs[4]; EGLint attribs[4];
int i; int i;
cleared = gdk_gl_context_clear_current_if_surface (self);
if (priv->egl_surface != NULL)
eglDestroySurface (gdk_display_get_egl_display (display), priv->egl_surface);
i = 0; i = 0;
if (depth == GDK_MEMORY_U8_SRGB && display->have_egl_gl_colorspace) if (depth == GDK_MEMORY_U8_SRGB && display->have_egl_gl_colorspace)
{ {
@ -1218,6 +1223,12 @@ gdk_surface_ensure_egl_surface (GdkSurface *self,
NULL); NULL);
} }
priv->egl_surface_depth = depth; priv->egl_surface_depth = depth;
if (cleared)
{
gdk_gl_context_make_current (cleared);
g_object_unref (cleared);
}
} }
return priv->egl_surface_depth; return priv->egl_surface_depth;

View File

@ -53,14 +53,9 @@ gdk_wayland_gl_context_begin_frame (GdkDrawContext *draw_context,
GdkColorState **out_color_state, GdkColorState **out_color_state,
GdkMemoryDepth *out_depth) GdkMemoryDepth *out_depth)
{ {
gboolean created_window; gdk_wayland_surface_ensure_wl_egl_window (gdk_draw_context_get_surface (draw_context));
created_window = gdk_wayland_surface_ensure_wl_egl_window (gdk_draw_context_get_surface (draw_context));
GDK_DRAW_CONTEXT_CLASS (gdk_wayland_gl_context_parent_class)->begin_frame (draw_context, depth, region, out_color_state, out_depth); GDK_DRAW_CONTEXT_CLASS (gdk_wayland_gl_context_parent_class)->begin_frame (draw_context, depth, region, out_color_state, out_depth);
if (created_window)
gdk_gl_context_make_current (GDK_GL_CONTEXT (draw_context));
} }
static void static void

View File

@ -1447,21 +1447,19 @@ _gdk_wayland_surface_offset_next_wl_buffer (GdkSurface *surface,
impl->pending_buffer_offset_y = y; impl->pending_buffer_offset_y = y;
} }
gboolean void
gdk_wayland_surface_ensure_wl_egl_window (GdkSurface *surface) gdk_wayland_surface_ensure_wl_egl_window (GdkSurface *surface)
{ {
GdkWaylandSurface *impl = GDK_WAYLAND_SURFACE (surface); GdkWaylandSurface *impl = GDK_WAYLAND_SURFACE (surface);
int width, height; int width, height;
if (impl->display_server.egl_window != NULL) if (impl->display_server.egl_window != NULL)
return FALSE; return;
get_egl_window_size (surface, &width, &height); get_egl_window_size (surface, &width, &height);
impl->display_server.egl_window = impl->display_server.egl_window =
wl_egl_window_create (impl->display_server.wl_surface, width, height); wl_egl_window_create (impl->display_server.wl_surface, width, height);
gdk_surface_set_egl_native_window (surface, impl->display_server.egl_window); gdk_surface_set_egl_native_window (surface, impl->display_server.egl_window);
return TRUE;
} }
/* }}} */ /* }}} */

View File

@ -26,6 +26,6 @@
G_BEGIN_DECLS G_BEGIN_DECLS
gboolean gdk_wayland_surface_ensure_wl_egl_window (GdkSurface *surface); void gdk_wayland_surface_ensure_wl_egl_window (GdkSurface *surface);
G_END_DECLS G_END_DECLS

View File

@ -84,10 +84,13 @@ gdk_x11_surface_get_glx_drawable (GdkSurface *surface)
void void
gdk_x11_surface_destroy_glx_drawable (GdkX11Surface *self) gdk_x11_surface_destroy_glx_drawable (GdkX11Surface *self)
{ {
GdkGLContext *context;
if (self->glx_drawable == None) if (self->glx_drawable == None)
return; return;
gdk_gl_context_clear_current_if_surface (GDK_SURFACE (self)); context = gdk_gl_context_clear_current_if_surface (GDK_SURFACE (self));
g_clear_object (&context);
glXDestroyWindow (gdk_x11_display_get_xdisplay (gdk_surface_get_display (GDK_SURFACE (self))), glXDestroyWindow (gdk_x11_display_get_xdisplay (gdk_surface_get_display (GDK_SURFACE (self))),
self->glx_drawable); self->glx_drawable);