aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSimon Ser <contact@emersion.fr>2023-11-30 17:47:11 +0100
committerSimon Ser <contact@emersion.fr>2023-11-30 17:47:11 +0100
commite8b187cc92b9b039dcbb6715665ee8ba5d065c6b (patch)
treee9c125e531d740ed5c60847797fb8f79faece5e1
parent1208ba6c285f19558b1eaf01167ec6b0ceb423a5 (diff)
render/gles2: save/restore context when creating/submitting a render pass
This is useful for e.g. lazily blitting a texture for readback purposes while rendering.
-rw-r--r--include/render/gles2.h5
-rw-r--r--include/wlr/render/gles2.h4
-rw-r--r--render/gles2/pass.c4
-rw-r--r--render/gles2/renderer.c5
4 files changed, 15 insertions, 3 deletions
diff --git a/include/render/gles2.h b/include/render/gles2.h
index 60bf4d17..dffd2059 100644
--- a/include/render/gles2.h
+++ b/include/render/gles2.h
@@ -15,6 +15,8 @@
#include <wlr/util/addon.h>
#include <wlr/util/log.h>
+#include "render/egl.h"
+
// mesa ships old GL headers that don't include this type, so for distros that use headers from
// mesa we need to def it ourselves until they update.
// https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/23144
@@ -132,6 +134,7 @@ struct wlr_gles2_render_pass {
struct wlr_render_pass base;
struct wlr_gles2_buffer *buffer;
float projection_matrix[9];
+ struct wlr_egl_context prev_ctx;
struct wlr_gles2_render_timer *timer;
};
@@ -164,6 +167,6 @@ void push_gles2_debug_(struct wlr_gles2_renderer *renderer,
void pop_gles2_debug(struct wlr_gles2_renderer *renderer);
struct wlr_gles2_render_pass *begin_gles2_buffer_pass(struct wlr_gles2_buffer *buffer,
- struct wlr_gles2_render_timer *timer);
+ struct wlr_egl_context *prev_ctx, struct wlr_gles2_render_timer *timer);
#endif
diff --git a/include/wlr/render/gles2.h b/include/wlr/render/gles2.h
index 66401548..740e2f2d 100644
--- a/include/wlr/render/gles2.h
+++ b/include/wlr/render/gles2.h
@@ -21,6 +21,10 @@ struct wlr_egl;
* the current EGL is global state. The GLES2 renderer operations will save
* and restore any previous EGL context when called. A render pass is seen as
* a single operation.
+ *
+ * The GLES2 renderer doesn't support arbitrarily nested render passes. It
+ * supports a subset only: after a nested render pass is created, any parent
+ * render pass can't be used before the nested render pass is submitted.
*/
struct wlr_renderer *wlr_gles2_renderer_create_with_drm_fd(int drm_fd);
diff --git a/render/gles2/pass.c b/render/gles2/pass.c
index f8a0f3e8..4c9f9e28 100644
--- a/render/gles2/pass.c
+++ b/render/gles2/pass.c
@@ -41,6 +41,7 @@ static bool render_pass_submit(struct wlr_render_pass *wlr_pass) {
glBindFramebuffer(GL_FRAMEBUFFER, 0);
pop_gles2_debug(renderer);
+ wlr_egl_restore_context(&pass->prev_ctx);
wlr_buffer_unlock(pass->buffer->buffer);
free(pass);
@@ -247,7 +248,7 @@ static const char *reset_status_str(GLenum status) {
}
struct wlr_gles2_render_pass *begin_gles2_buffer_pass(struct wlr_gles2_buffer *buffer,
- struct wlr_gles2_render_timer *timer) {
+ struct wlr_egl_context *prev_ctx, struct wlr_gles2_render_timer *timer) {
struct wlr_gles2_renderer *renderer = buffer->renderer;
struct wlr_buffer *wlr_buffer = buffer->buffer;
@@ -274,6 +275,7 @@ struct wlr_gles2_render_pass *begin_gles2_buffer_pass(struct wlr_gles2_buffer *b
wlr_buffer_lock(wlr_buffer);
pass->buffer = buffer;
pass->timer = timer;
+ pass->prev_ctx = *prev_ctx;
matrix_projection(pass->projection_matrix, wlr_buffer->width, wlr_buffer->height,
WL_OUTPUT_TRANSFORM_FLIPPED_180);
diff --git a/render/gles2/renderer.c b/render/gles2/renderer.c
index 327d197e..1b106af5 100644
--- a/render/gles2/renderer.c
+++ b/render/gles2/renderer.c
@@ -433,6 +433,9 @@ static void gles2_destroy(struct wlr_renderer *wlr_renderer) {
static struct wlr_render_pass *gles2_begin_buffer_pass(struct wlr_renderer *wlr_renderer,
struct wlr_buffer *wlr_buffer, const struct wlr_buffer_pass_options *options) {
struct wlr_gles2_renderer *renderer = gles2_get_renderer(wlr_renderer);
+
+ struct wlr_egl_context prev_ctx = {0};
+ wlr_egl_save_context(&prev_ctx);
if (!wlr_egl_make_current(renderer->egl)) {
return NULL;
}
@@ -448,7 +451,7 @@ static struct wlr_render_pass *gles2_begin_buffer_pass(struct wlr_renderer *wlr_
return NULL;
}
- struct wlr_gles2_render_pass *pass = begin_gles2_buffer_pass(buffer, timer);
+ struct wlr_gles2_render_pass *pass = begin_gles2_buffer_pass(buffer, &prev_ctx, timer);
if (!pass) {
return NULL;
}