From ba909e4fc7823ab3d131d574d22b30bc17c16cee Mon Sep 17 00:00:00 2001 From: Johan Malm Date: Sun, 13 Feb 2022 11:47:03 +0000 Subject: [PATCH] Refactor buffer implementation to take a cairo_t Remove buffer_drop() and just destroy it. --- include/buffer.h | 26 +++++------------------ src/buffer.c | 55 +++++++++++++++++++++--------------------------- src/osd.c | 10 ++------- 3 files changed, 31 insertions(+), 60 deletions(-) diff --git a/include/buffer.h b/include/buffer.h index 8c141f8a..dd841803 100644 --- a/include/buffer.h +++ b/include/buffer.h @@ -26,35 +26,19 @@ #ifndef __LABWC_BUFFER_H #define __LABWC_BUFFER_H +#include #include "labwc.h" -/** - * A read-only buffer that holds a data pointer. - * - * This is suitable for passing raw pixel data to a function that accepts a - * wlr_buffer. - */ struct lab_data_buffer { struct wlr_buffer base; - const void *data; + cairo_t *cairo; + void *data; uint32_t format; size_t stride; - - void *saved_data; }; -/** - * Wraps a read-only data pointer into a wlr_buffer. The data pointer may be - * accessed until readonly_data_buffer_drop() is called. - */ -struct lab_data_buffer *buffer_create(uint32_t format, size_t stride, - uint32_t width, uint32_t height, const void *data); - -/** - * Drops ownership of the buffer (see wlr_buffer_drop() for more details) and - * perform a copy of the data pointer if a consumer still has the buffer locked. - */ -bool buffer_drop(struct lab_data_buffer *buffer); +struct lab_data_buffer *buffer_create(cairo_t *cairo); +void buffer_destroy(struct lab_data_buffer *buffer); #endif /* __LABWC_BUFFER_H */ diff --git a/src/buffer.c b/src/buffer.c index 0188ee01..640a2189 100644 --- a/src/buffer.c +++ b/src/buffer.c @@ -42,7 +42,7 @@ static void data_buffer_destroy(struct wlr_buffer *wlr_buffer) { struct lab_data_buffer *buffer = data_buffer_from_buffer(wlr_buffer); - free(buffer->saved_data); + free(buffer->data); free(buffer); } @@ -50,13 +50,9 @@ static bool data_buffer_begin_data_ptr_access(struct wlr_buffer *wlr_buffer, uint32_t flags, void **data, uint32_t *format, size_t *stride) { - struct lab_data_buffer *buffer = data_buffer_from_buffer(wlr_buffer); - if (buffer->data == NULL) { - return false; - } - if (flags & WLR_BUFFER_DATA_PTR_ACCESS_WRITE) { - return false; - } + struct lab_data_buffer *buffer = + wl_container_of(wlr_buffer, buffer, base); + assert(buffer->data); *data = (void *)buffer->data; *format = buffer->format; *stride = buffer->stride; @@ -76,38 +72,35 @@ static const struct wlr_buffer_impl data_buffer_impl = { }; struct lab_data_buffer * -buffer_create(uint32_t format, size_t stride, uint32_t width, uint32_t height, - const void *data) +buffer_create(cairo_t *cairo) { struct lab_data_buffer *buffer = calloc(1, sizeof(*buffer)); - if (buffer == NULL) { + if (!buffer) { return NULL; } + cairo_surface_t *surf = cairo_get_target(cairo); + int width = cairo_image_surface_get_width(surf); + int height = cairo_image_surface_get_height(surf); wlr_buffer_init(&buffer->base, &data_buffer_impl, width, height); - buffer->data = data; - buffer->format = format; - buffer->stride = stride; + + buffer->cairo = cairo; + buffer->data = cairo_image_surface_get_data(surf); + buffer->format = DRM_FORMAT_ARGB8888; + buffer->stride = cairo_image_surface_get_stride(surf); + + if (!buffer->data) { + cairo_destroy(cairo); + free(buffer); + } return buffer; } - -bool -buffer_drop(struct lab_data_buffer *buffer) +void +buffer_destroy(struct lab_data_buffer *buffer) { - bool ok = true; - - if (buffer->base.n_locks > 0) { - size_t size = buffer->stride * buffer->base.height; - buffer->saved_data = malloc(size); - if (!buffer->saved_data) { - wlr_log_errno(WLR_ERROR, "allocation failed"); - ok = false; - buffer->data = NULL; - } else { - memcpy(buffer->saved_data, buffer->data, size); - buffer->data = buffer->saved_data; - } + if (!buffer) { + return; } + cairo_destroy(buffer->cairo); wlr_buffer_drop(&buffer->base); - return ok; } diff --git a/src/osd.c b/src/osd.c index f790cd16..9f7beea1 100644 --- a/src/osd.c +++ b/src/osd.c @@ -197,14 +197,8 @@ osd_update(struct server *server) g_object_unref(layout); cairo_surface_flush(surf); - unsigned char *data = cairo_image_surface_get_data(surf); - if (output->osd_buffer) { - buffer_drop(output->osd_buffer); - } - output->osd_buffer = buffer_create(DRM_FORMAT_ARGB8888, - cairo_image_surface_get_stride(surf), w, h, data); - - cairo_surface_destroy(surf); + buffer_destroy(output->osd_buffer); + output->osd_buffer = buffer_create(cairo); struct wlr_scene_buffer *scene_buffer = wlr_scene_buffer_create( &server->osd_tree->node, &output->osd_buffer->base);