Refactor buffer implementation to take a cairo_t

Remove buffer_drop() and just destroy it.
This commit is contained in:
Johan Malm 2022-02-13 11:47:03 +00:00
parent ebabc066ba
commit ba909e4fc7
3 changed files with 31 additions and 60 deletions

View file

@ -26,35 +26,19 @@
#ifndef __LABWC_BUFFER_H #ifndef __LABWC_BUFFER_H
#define __LABWC_BUFFER_H #define __LABWC_BUFFER_H
#include <cairo.h>
#include "labwc.h" #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 lab_data_buffer {
struct wlr_buffer base; struct wlr_buffer base;
const void *data; cairo_t *cairo;
void *data;
uint32_t format; uint32_t format;
size_t stride; size_t stride;
void *saved_data;
}; };
/** struct lab_data_buffer *buffer_create(cairo_t *cairo);
* Wraps a read-only data pointer into a wlr_buffer. The data pointer may be void buffer_destroy(struct lab_data_buffer *buffer);
* 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);
#endif /* __LABWC_BUFFER_H */ #endif /* __LABWC_BUFFER_H */

View file

@ -42,7 +42,7 @@ static void
data_buffer_destroy(struct wlr_buffer *wlr_buffer) data_buffer_destroy(struct wlr_buffer *wlr_buffer)
{ {
struct lab_data_buffer *buffer = data_buffer_from_buffer(wlr_buffer); struct lab_data_buffer *buffer = data_buffer_from_buffer(wlr_buffer);
free(buffer->saved_data); free(buffer->data);
free(buffer); free(buffer);
} }
@ -50,13 +50,9 @@ static bool
data_buffer_begin_data_ptr_access(struct wlr_buffer *wlr_buffer, uint32_t flags, data_buffer_begin_data_ptr_access(struct wlr_buffer *wlr_buffer, uint32_t flags,
void **data, uint32_t *format, size_t *stride) void **data, uint32_t *format, size_t *stride)
{ {
struct lab_data_buffer *buffer = data_buffer_from_buffer(wlr_buffer); struct lab_data_buffer *buffer =
if (buffer->data == NULL) { wl_container_of(wlr_buffer, buffer, base);
return false; assert(buffer->data);
}
if (flags & WLR_BUFFER_DATA_PTR_ACCESS_WRITE) {
return false;
}
*data = (void *)buffer->data; *data = (void *)buffer->data;
*format = buffer->format; *format = buffer->format;
*stride = buffer->stride; *stride = buffer->stride;
@ -76,38 +72,35 @@ static const struct wlr_buffer_impl data_buffer_impl = {
}; };
struct lab_data_buffer * struct lab_data_buffer *
buffer_create(uint32_t format, size_t stride, uint32_t width, uint32_t height, buffer_create(cairo_t *cairo)
const void *data)
{ {
struct lab_data_buffer *buffer = calloc(1, sizeof(*buffer)); struct lab_data_buffer *buffer = calloc(1, sizeof(*buffer));
if (buffer == NULL) { if (!buffer) {
return NULL; 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); wlr_buffer_init(&buffer->base, &data_buffer_impl, width, height);
buffer->data = data;
buffer->format = format; buffer->cairo = cairo;
buffer->stride = stride; 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; return buffer;
} }
void
bool buffer_destroy(struct lab_data_buffer *buffer)
buffer_drop(struct lab_data_buffer *buffer)
{ {
bool ok = true; if (!buffer) {
return;
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;
}
} }
cairo_destroy(buffer->cairo);
wlr_buffer_drop(&buffer->base); wlr_buffer_drop(&buffer->base);
return ok;
} }

View file

@ -197,14 +197,8 @@ osd_update(struct server *server)
g_object_unref(layout); g_object_unref(layout);
cairo_surface_flush(surf); cairo_surface_flush(surf);
unsigned char *data = cairo_image_surface_get_data(surf); buffer_destroy(output->osd_buffer);
if (output->osd_buffer) { output->osd_buffer = buffer_create(cairo);
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);
struct wlr_scene_buffer *scene_buffer = wlr_scene_buffer_create( struct wlr_scene_buffer *scene_buffer = wlr_scene_buffer_create(
&server->osd_tree->node, &output->osd_buffer->base); &server->osd_tree->node, &output->osd_buffer->base);