struct lab_data_buffer {
struct wlr_buffer base;
- cairo_t *cairo;
- void *data;
+ cairo_surface_t *surface; /* optional */
+ cairo_t *cairo; /* optional */
+ void *data; /* owned by surface if surface != NULL */
uint32_t format;
size_t stride;
- bool free_on_destroy;
- uint32_t unscaled_width;
- uint32_t unscaled_height;
+ /*
+ * The logical size of the surface in layout pixels.
+ * The raw pixel data may be larger or smaller.
+ */
+ uint32_t logical_width;
+ uint32_t logical_height;
};
-/* Create a buffer which creates a new cairo CAIRO_FORMAT_ARGB32 surface */
-struct lab_data_buffer *buffer_create_cairo(uint32_t width, uint32_t height,
- float scale, bool free_on_destroy);
+/*
+ * Create a buffer which holds (and takes ownership of) an existing
+ * CAIRO_FORMAT_ARGB32 image surface.
+ *
+ * The logical size is set to the surface size in pixels, ignoring
+ * device scale. No cairo context is created.
+ */
+struct lab_data_buffer *buffer_adopt_cairo_surface(cairo_surface_t *surface);
-/* Create a buffer which wraps a given DRM_FORMAT_ARGB8888 pointer */
-struct lab_data_buffer *buffer_create_wrap(void *pixel_data, uint32_t width,
- uint32_t height, uint32_t stride, bool free_on_destroy);
+/*
+ * Create a buffer which holds a new CAIRO_FORMAT_ARGB32 image surface.
+ * Additionally create a cairo context for drawing to the surface.
+ */
+struct lab_data_buffer *buffer_create_cairo(uint32_t logical_width,
+ uint32_t logical_height, float scale);
+
+/*
+ * Create a buffer which holds (and takes ownership of) raw pixel data
+ * in pre-multiplied ARGB32 format.
+ *
+ * The logical size is set to the width and height of the pixel data.
+ * No cairo surface or context is created.
+ */
+struct lab_data_buffer *buffer_create_from_data(void *pixel_data, uint32_t width,
+ uint32_t height, uint32_t stride);
#endif /* LABWC_BUFFER_H */
data_buffer_destroy(struct wlr_buffer *wlr_buffer)
{
struct lab_data_buffer *buffer = data_buffer_from_buffer(wlr_buffer);
- if (!buffer->free_on_destroy) {
- free(buffer);
- return;
- }
if (buffer->cairo) {
- cairo_surface_t *surf = cairo_get_target(buffer->cairo);
cairo_destroy(buffer->cairo);
- cairo_surface_destroy(surf);
+ }
+ if (buffer->surface) {
+ /* this also frees buffer->data */
+ cairo_surface_destroy(buffer->surface);
} else if (buffer->data) {
free(buffer->data);
buffer->data = NULL;
};
struct lab_data_buffer *
-buffer_create_cairo(uint32_t width, uint32_t height, float scale,
- bool free_on_destroy)
+buffer_adopt_cairo_surface(cairo_surface_t *surface)
{
- struct lab_data_buffer *buffer = znew(*buffer);
- buffer->unscaled_width = width;
- buffer->unscaled_height = height;
- width *= scale;
- height *= scale;
+ assert(cairo_surface_get_type(surface) == CAIRO_SURFACE_TYPE_IMAGE);
+ assert(cairo_image_surface_get_format(surface) == CAIRO_FORMAT_ARGB32);
- /* Allocate the buffer with the scaled size */
+ int width = cairo_image_surface_get_width(surface);
+ int height = cairo_image_surface_get_height(surface);
+
+ struct lab_data_buffer *buffer = znew(*buffer);
wlr_buffer_init(&buffer->base, &data_buffer_impl, width, height);
- cairo_surface_t *surf =
- cairo_image_surface_create(CAIRO_FORMAT_ARGB32, width, height);
+
+ buffer->surface = surface;
+ buffer->data = cairo_image_surface_get_data(buffer->surface);
+ buffer->format = DRM_FORMAT_ARGB8888;
+ buffer->stride = cairo_image_surface_get_stride(buffer->surface);
+ buffer->logical_width = width;
+ buffer->logical_height = height;
+
+ return buffer;
+}
+
+struct lab_data_buffer *
+buffer_create_cairo(uint32_t logical_width, uint32_t logical_height, float scale)
+{
+ /* Create an image surface with the scaled size */
+ cairo_surface_t *surface =
+ cairo_image_surface_create(CAIRO_FORMAT_ARGB32,
+ lroundf(logical_width * scale),
+ lroundf(logical_height * scale));
/**
* Tell cairo about the device scale so we can keep drawing in unscaled
*
* For a more complete explanation see PR #389
*/
- cairo_surface_set_device_scale(surf, scale, scale);
+ cairo_surface_set_device_scale(surface, scale, scale);
- buffer->cairo = cairo_create(surf);
- buffer->data = cairo_image_surface_get_data(surf);
- buffer->format = DRM_FORMAT_ARGB8888;
- buffer->stride = cairo_image_surface_get_stride(surf);
- buffer->free_on_destroy = free_on_destroy;
+ /*
+ * Adopt the image surface into a buffer, set the correct
+ * logical size, and create a cairo context for drawing
+ */
+ struct lab_data_buffer *buffer = buffer_adopt_cairo_surface(surface);
+ buffer->logical_width = logical_width;
+ buffer->logical_height = logical_height;
+ buffer->cairo = cairo_create(surface);
- if (!buffer->data) {
- cairo_destroy(buffer->cairo);
- cairo_surface_destroy(surf);
- free(buffer);
- buffer = NULL;
- }
return buffer;
}
struct lab_data_buffer *
-buffer_create_wrap(void *pixel_data, uint32_t width, uint32_t height,
- uint32_t stride, bool free_on_destroy)
+buffer_create_from_data(void *pixel_data, uint32_t width, uint32_t height,
+ uint32_t stride)
{
struct lab_data_buffer *buffer = znew(*buffer);
wlr_buffer_init(&buffer->base, &data_buffer_impl, width, height);
- buffer->unscaled_width = width;
- buffer->unscaled_height = height;
+ buffer->logical_width = width;
+ buffer->logical_height = height;
buffer->data = pixel_data;
buffer->format = DRM_FORMAT_ARGB8888;
buffer->stride = stride;
- buffer->free_on_destroy = free_on_destroy;
return buffer;
}
}
*buffer = buffer_create_cairo(text_extents.width + arrow_extents.width,
- text_extents.height, scale, true);
+ text_extents.height, scale);
if (!*buffer) {
wlr_log(WLR_ERROR, "Failed to create font buffer");
return;
}
/* Handle DRM_FORMAT_ARGB8888 buffers */
- int w = buffer->unscaled_width;
- int h = buffer->unscaled_height;
+ int w = buffer->logical_width;
+ int h = buffer->logical_height;
cairo_surface_t *surface =
cairo_image_surface_create(CAIRO_FORMAT_ARGB32, w, h);
if (!surface) {
wlr_log(WLR_ERROR, "font_buffer_create() failed");
}
- self->width = buffer ? buffer->unscaled_width : 0;
- self->height = buffer ? buffer->unscaled_height : 0;
+ self->width = buffer ? buffer->logical_width : 0;
+ self->height = buffer ? buffer->logical_height : 0;
return buffer;
}
/* Ensure the buffer doesn't get deleted behind our back */
wlr_buffer_lock(&buffer->base);
}
- self->width = buffer ? buffer->unscaled_width : 0;
- self->height = buffer ? buffer->unscaled_height : 0;
+ self->width = buffer ? buffer->logical_width : 0;
+ self->height = buffer ? buffer->logical_height : 0;
/* Create or reuse cache entry */
if (wl_list_length(&self->cache) < LAB_SCALED_BUFFER_MAX_CACHE) {
}
cairo_surface_flush(image);
- double w = cairo_image_surface_get_width(image);
- double h = cairo_image_surface_get_height(image);
- *buffer = buffer_create_cairo((int)w, (int)h, 1.0, true);
+ if (cairo_image_surface_get_format(image) == CAIRO_FORMAT_ARGB32) {
+ /* we can wrap ARGB32 image surfaces directly */
+ *buffer = buffer_adopt_cairo_surface(image);
+ return;
+ }
+
+ /* convert to ARGB32 by painting to a new surface */
+ uint32_t w = cairo_image_surface_get_width(image);
+ uint32_t h = cairo_image_surface_get_height(image);
+ *buffer = buffer_create_cairo(w, h, 1);
cairo_t *cairo = (*buffer)->cairo;
cairo_set_source_surface(cairo, image, 0, 0);
- cairo_paint_with_alpha(cairo, 1.0);
+ cairo_paint(cairo);
+ cairo_surface_flush((*buffer)->surface);
+ /* destroy original surface */
+ cairo_surface_destroy(image);
}
return;
}
- cairo_surface_t *image = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, size, size);
- cairo_t *cr = cairo_create(image);
+ *buffer = buffer_create_cairo(size, size, 1.0);
+ cairo_surface_t *image = (*buffer)->surface;
+ cairo_t *cr = (*buffer)->cairo;
rsvg_handle_render_document(svg, cr, &viewport, &err);
if (err) {
}
cairo_surface_flush(image);
- double w = cairo_image_surface_get_width(image);
- double h = cairo_image_surface_get_height(image);
- *buffer = buffer_create_cairo((int)w, (int)h, 1.0, /* free_on_destroy */ true);
- cairo_t *cairo = (*buffer)->cairo;
- cairo_set_source_surface(cairo, image, 0, 0);
- cairo_paint_with_alpha(cairo, 1.0);
+ g_object_unref(svg);
+ return;
error:
- cairo_destroy(cr);
- cairo_surface_destroy(image);
+ wlr_buffer_drop(&(*buffer)->base);
+ *buffer = NULL;
g_object_unref(svg);
}
}
color = argb32(rgba);
pixmap = parse_xbm_builtin(bitmap, 6);
- *buffer = buffer_create_wrap(pixmap.data, pixmap.width, pixmap.height,
- pixmap.width * 4, /* free_on_destroy */ true);
+ *buffer = buffer_create_from_data(pixmap.data, pixmap.width, pixmap.height,
+ pixmap.width * 4);
}
void
return;
}
- /* Create buffer with free_on_destroy being true */
+ /* Create buffer */
if (pixmap.data) {
- *buffer = buffer_create_wrap(pixmap.data, pixmap.width,
- pixmap.height, pixmap.width * 4, true);
+ *buffer = buffer_create_from_data(pixmap.data, pixmap.width,
+ pixmap.height, pixmap.width * 4);
}
}
free(colors);
free(name_buf);
- return buffer_create_wrap(data, w, h, 4 * w, true);
+ return buffer_create_from_data(data, w, h, 4 * w);
out:
g_hash_table_destroy(color_hash);
if (output->osd_buffer) {
wlr_buffer_drop(&output->osd_buffer->base);
}
- output->osd_buffer = buffer_create_cairo(w, h, scale, true);
+ output->osd_buffer = buffer_create_cairo(w, h, scale);
if (!output->osd_buffer) {
wlr_log(WLR_ERROR, "Failed to allocate cairo buffer for the window switcher");
return;
* is different). The buffers are square so width == height.
*/
int corner_size = active
- ? theme->shadow_corner_top_active->unscaled_height
- : theme->shadow_corner_top_inactive->unscaled_height;
+ ? theme->shadow_corner_top_active->logical_height
+ : theme->shadow_corner_top_inactive->logical_height;
wl_list_for_each(part, &subtree->parts, link) {
set_shadow_part_geometry(part, width, height,
int buffer_width = (double)width * scale;
int buffer_height = (double)height * scale;
struct lab_data_buffer *buffer = buffer_create_cairo(
- buffer_width, buffer_height, 1.0, true);
+ buffer_width, buffer_height, 1.0);
cairo_t *cairo = buffer->cairo;
cairo_set_source_surface(cairo, icon.surface,
struct lab_data_buffer *buffer;
/* TODO: scale */
- buffer = buffer_create_cairo(w, h, 1, /*free_on_destroy*/ true);
+ buffer = buffer_create_cairo(w, h, 1);
cairo_t *cairo = buffer->cairo;
cairo_surface_t *surf = cairo_get_target(cairo);
*/
if (visible_active_size > 0) {
theme->shadow_edge_active = buffer_create_cairo(
- visible_active_size, 1, 1.0, true);
+ visible_active_size, 1, 1.0);
theme->shadow_corner_top_active = buffer_create_cairo(
- total_active_size, total_active_size, 1.0, true);
+ total_active_size, total_active_size, 1.0);
theme->shadow_corner_bottom_active = buffer_create_cairo(
- total_active_size, total_active_size, 1.0, true);
+ total_active_size, total_active_size, 1.0);
if (!theme->shadow_corner_top_active
|| !theme->shadow_corner_bottom_active
|| !theme->shadow_edge_active) {
}
if (visible_inactive_size > 0) {
theme->shadow_edge_inactive = buffer_create_cairo(
- visible_inactive_size, 1, 1.0, true);
+ visible_inactive_size, 1, 1.0);
theme->shadow_corner_top_inactive = buffer_create_cairo(
- total_inactive_size, total_inactive_size, 1.0, true);
+ total_inactive_size, total_inactive_size, 1.0);
theme->shadow_corner_bottom_inactive = buffer_create_cairo(
- total_inactive_size, total_inactive_size, 1.0, true);
+ total_inactive_size, total_inactive_size, 1.0);
if (!theme->shadow_corner_top_inactive
|| !theme->shadow_corner_bottom_inactive
|| !theme->shadow_edge_inactive) {
continue;
}
struct lab_data_buffer *buffer = buffer_create_cairo(width, height,
- output->wlr_output->scale, true);
+ output->wlr_output->scale);
if (!buffer) {
wlr_log(WLR_ERROR, "Failed to allocate buffer for workspace OSD");
continue;
wlr_scene_node_set_position(&output->workspace_osd->node, lx, ly);
wlr_scene_buffer_set_buffer(output->workspace_osd, &buffer->base);
wlr_scene_buffer_set_dest_size(output->workspace_osd,
- buffer->unscaled_width, buffer->unscaled_height);
+ buffer->logical_width, buffer->logical_height);
/* And finally drop the buffer so it will get destroyed on OSD hide */
wlr_buffer_drop(&buffer->base);