#ifndef __LABWC_BUFFER_H
#define __LABWC_BUFFER_H
+#include <cairo.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 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 */
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);
}
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;
};
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;
}
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);