c/client: implement EGL context begin and end

This commit is contained in:
Simon Zeni 2022-05-26 15:56:54 -04:00 committed by Jakob Bornecrantz
parent f91ada32c2
commit c8b4f7c64c
3 changed files with 54 additions and 54 deletions

View file

@ -72,50 +72,34 @@ typedef EGLBoolean
* *
*/ */
struct old_helper
{
EGLDisplay dpy;
EGLContext ctx;
EGLSurface read, draw;
};
static inline struct old_helper
old_save(void)
{
struct old_helper old = {
.dpy = eglGetCurrentDisplay(),
.ctx = EGL_NO_CONTEXT,
.read = EGL_NO_SURFACE,
.draw = EGL_NO_SURFACE,
};
// Do we have a valid display?
if (old.dpy != EGL_NO_DISPLAY) {
old.ctx = eglGetCurrentContext();
old.read = eglGetCurrentSurface(EGL_READ);
old.draw = eglGetCurrentSurface(EGL_DRAW);
}
return old;
}
static inline void static inline void
old_restore(struct old_helper *old, EGLDisplay current_dpy) save_context(struct client_egl_context *ctx)
{ {
if (old->dpy == EGL_NO_DISPLAY) { ctx->dpy = eglGetCurrentDisplay();
// There were no display, just unbind the context. ctx->ctx = EGL_NO_CONTEXT;
if (eglMakeCurrent(current_dpy, EGL_NO_CONTEXT, EGL_NO_SURFACE, EGL_NO_SURFACE)) { ctx->read = EGL_NO_SURFACE;
return; ctx->draw = EGL_NO_SURFACE;
}
} else { if (ctx->dpy != EGL_NO_DISPLAY) {
if (eglMakeCurrent(old->dpy, old->draw, old->read, old->ctx)) { ctx->ctx = eglGetCurrentContext();
return; ctx->read = eglGetCurrentSurface(EGL_READ);
ctx->draw = eglGetCurrentSurface(EGL_DRAW);
} }
} }
EGL_ERROR("Failed to make old EGL context current! (%p, %p, %p, %p)", old->dpy, old->draw, old->read, old->ctx); static inline bool
restore_context(struct client_egl_context *ctx)
{
/* We're using the current display if we're trying to restore a null context */
EGLDisplay dpy = ctx->dpy == EGL_NO_DISPLAY ? eglGetCurrentDisplay() : ctx->dpy;
if (dpy == EGL_NO_DISPLAY) {
/* If the current display is also null then the call is a no-op */
return true;
} }
return eglMakeCurrent(dpy, ctx->draw, ctx->read, ctx->ctx);
}
/* /*
* *
@ -207,7 +191,7 @@ insert_fence_android_native(struct xrt_compositor *xc, xrt_graphics_sync_handle_
struct client_egl_compositor *ceglc = client_egl_compositor(xc); struct client_egl_compositor *ceglc = client_egl_compositor(xc);
*out_handle = XRT_GRAPHICS_SYNC_HANDLE_INVALID; *out_handle = XRT_GRAPHICS_SYNC_HANDLE_INVALID;
EGLDisplay dpy = ceglc->dpy; EGLDisplay dpy = ceglc->current.dpy;
#ifdef XRT_GRAPHICS_SYNC_HANDLE_IS_FD #ifdef XRT_GRAPHICS_SYNC_HANDLE_IS_FD
@ -239,14 +223,23 @@ insert_fence_android_native(struct xrt_compositor *xc, xrt_graphics_sync_handle_
static xrt_result_t static xrt_result_t
client_egl_context_begin(struct xrt_compositor *xc) client_egl_context_begin(struct xrt_compositor *xc)
{ {
//! @todo Implement struct client_egl_compositor *eglc = client_egl_compositor(xc);
save_context(&eglc->previous);
struct client_egl_context *cur = &eglc->current;
if (!eglMakeCurrent(cur->dpy, EGL_NO_SURFACE, EGL_NO_SURFACE, cur->ctx)) {
return XRT_ERROR_OPENGL;
}
return XRT_SUCCESS; return XRT_SUCCESS;
} }
static void static void
client_egl_context_end(struct xrt_compositor *xc) client_egl_context_end(struct xrt_compositor *xc)
{ {
//! @todo Implement struct client_egl_compositor *eglc = client_egl_compositor(xc);
restore_context(&eglc->previous);
} }
static void static void
@ -286,8 +279,9 @@ xrt_gfx_provider_create_gl_egl(struct xrt_compositor_native *xcn,
// On Android this extension is 'hidden'. // On Android this extension is 'hidden'.
ensure_native_fence_is_loaded(display, get_gl_procaddr); ensure_native_fence_is_loaded(display, get_gl_procaddr);
// Save old display, context and drawables. // Save old EGL display, context and drawables.
struct old_helper old = old_save(); struct client_egl_context old = {0};
save_context(&old);
if (!eglMakeCurrent(display, EGL_NO_SURFACE, EGL_NO_SURFACE, context)) { if (!eglMakeCurrent(display, EGL_NO_SURFACE, EGL_NO_SURFACE, context)) {
EGL_ERROR("eglMakeCurrent: %s\n\tFailed to make EGL context current", egl_error_str(eglGetError())); EGL_ERROR("eglMakeCurrent: %s\n\tFailed to make EGL context current", egl_error_str(eglGetError()));
@ -297,7 +291,7 @@ xrt_gfx_provider_create_gl_egl(struct xrt_compositor_native *xcn,
EGLint egl_client_type; EGLint egl_client_type;
if (!eglQueryContext(display, context, EGL_CONTEXT_CLIENT_TYPE, &egl_client_type)) { if (!eglQueryContext(display, context, EGL_CONTEXT_CLIENT_TYPE, &egl_client_type)) {
old_restore(&old, display); restore_context(&old);
return XRT_ERROR_OPENGL; return XRT_ERROR_OPENGL;
} }
@ -308,7 +302,7 @@ xrt_gfx_provider_create_gl_egl(struct xrt_compositor_native *xcn,
break; break;
#else #else
EGL_ERROR("OpenGL support not including in this runtime build"); EGL_ERROR("OpenGL support not including in this runtime build");
old_restore(&old, display); restore_context(&old);
return XRT_ERROR_OPENGL; return XRT_ERROR_OPENGL;
#endif #endif
@ -318,14 +312,15 @@ xrt_gfx_provider_create_gl_egl(struct xrt_compositor_native *xcn,
break; break;
#else #else
EGL_ERROR("OpenGL|ES support not including in this runtime build"); EGL_ERROR("OpenGL|ES support not including in this runtime build");
old_restore(&old, display); restore_context(&old);
return XRT_ERROR_OPENGL; return XRT_ERROR_OPENGL;
#endif #endif
default: EGL_ERROR("Unsupported EGL client type"); return XRT_ERROR_OPENGL; default: EGL_ERROR("Unsupported EGL client type"); return XRT_ERROR_OPENGL;
} }
struct client_egl_compositor *ceglc = U_TYPED_CALLOC(struct client_egl_compositor); struct client_egl_compositor *ceglc = U_TYPED_CALLOC(struct client_egl_compositor);
ceglc->dpy = display; ceglc->current.dpy = display;
ceglc->current.ctx = context;
client_gl_swapchain_create_func_t sc_create = NULL; client_gl_swapchain_create_func_t sc_create = NULL;
@ -365,7 +360,7 @@ xrt_gfx_provider_create_gl_egl(struct xrt_compositor_native *xcn,
EGL_ERROR( EGL_ERROR(
"Could not find a required extension: need either EGL_EXT_image_dma_buf_import or " "Could not find a required extension: need either EGL_EXT_image_dma_buf_import or "
"GL_EXT_memory_object_fd"); "GL_EXT_memory_object_fd");
old_restore(&old, display); restore_context(&old);
return XRT_ERROR_OPENGL; return XRT_ERROR_OPENGL;
} }
@ -396,12 +391,12 @@ xrt_gfx_provider_create_gl_egl(struct xrt_compositor_native *xcn,
if (!bret) { if (!bret) {
free(ceglc); free(ceglc);
EGL_ERROR("Failed to initialize compositor"); EGL_ERROR("Failed to initialize compositor");
old_restore(&old, display); restore_context(&old);
return XRT_ERROR_OPENGL; return XRT_ERROR_OPENGL;
} }
ceglc->base.base.base.destroy = client_egl_compositor_destroy; ceglc->base.base.base.destroy = client_egl_compositor_destroy;
old_restore(&old, display); restore_context(&old);
*out_xcgl = &ceglc->base.base; *out_xcgl = &ceglc->base.base;
return XRT_SUCCESS; return XRT_SUCCESS;

View file

@ -20,6 +20,12 @@
extern "C" { extern "C" {
#endif #endif
struct client_egl_context
{
EGLDisplay dpy;
EGLContext ctx;
EGLSurface read, draw;
};
/*! /*!
* EGL based compositor, carries the extra needed EGL information needed by the * EGL based compositor, carries the extra needed EGL information needed by the
@ -30,8 +36,7 @@ extern "C" {
struct client_egl_compositor struct client_egl_compositor
{ {
struct client_gl_compositor base; struct client_gl_compositor base;
struct client_egl_context current, previous;
EGLDisplay dpy;
}; };
/*! /*!

View file

@ -200,7 +200,7 @@ client_gl_eglimage_swapchain_create(struct xrt_compositor *xc,
sc->base.base.base.image_count = sc->base.base.base.image_count =
native_xsc->image_count; // Fetch the number of images from the native swapchain. native_xsc->image_count; // Fetch the number of images from the native swapchain.
sc->base.xscn = xscn; sc->base.xscn = xscn;
sc->display = ceglc->dpy; sc->display = ceglc->current.dpy;
struct xrt_swapchain_gl *xscgl = &sc->base.base; struct xrt_swapchain_gl *xscgl = &sc->base.base;