mirror of
https://gitlab.freedesktop.org/wlroots/wlroots.git
synced 2026-02-04 04:06:09 -05:00
xcursor: introduce wlr_xcursor_image_get_buffer()
This makes it so callers no longer need to juggle with raw pixel pointers anymore, and only need a single wlr_buffer-based codepath. Additionally, cursors can be unloaded without risking use-after-free.
This commit is contained in:
parent
98733c91b4
commit
d7f7b68f49
2 changed files with 31 additions and 3 deletions
|
|
@ -56,6 +56,10 @@ struct wlr_xcursor_image {
|
|||
uint32_t hotspot_y; /* hot-spot y (must be inside image) */
|
||||
uint32_t delay; /* animation delay to next frame (ms) */
|
||||
uint8_t *buffer; /* pixel data */
|
||||
|
||||
struct {
|
||||
struct wlr_readonly_data_buffer *readonly_buffer;
|
||||
} WLR_PRIVATE;
|
||||
};
|
||||
|
||||
/**
|
||||
|
|
@ -119,6 +123,11 @@ struct wlr_xcursor *wlr_xcursor_theme_get_cursor(
|
|||
*/
|
||||
int wlr_xcursor_frame(struct wlr_xcursor *cursor, uint32_t time);
|
||||
|
||||
/**
|
||||
* Get a struct wlr_buffer from a cursor image.
|
||||
*/
|
||||
struct wlr_buffer *wlr_xcursor_image_get_buffer(struct wlr_xcursor_image *image);
|
||||
|
||||
/**
|
||||
* Get the name of the resize cursor for the given edges.
|
||||
*/
|
||||
|
|
|
|||
|
|
@ -23,18 +23,21 @@
|
|||
* SOFTWARE.
|
||||
*/
|
||||
|
||||
#include <drm_fourcc.h>
|
||||
#include <inttypes.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <wlr/util/log.h>
|
||||
#include <wlr/xcursor.h>
|
||||
#include "types/wlr_buffer.h"
|
||||
#include "xcursor/xcursor.h"
|
||||
|
||||
#include "xcursor/cursor_data.h"
|
||||
|
||||
static void xcursor_destroy(struct wlr_xcursor *cursor) {
|
||||
for (size_t i = 0; i < cursor->image_count; i++) {
|
||||
readonly_data_buffer_drop(cursor->images[i]->readonly_buffer);
|
||||
free(cursor->images[i]->buffer);
|
||||
free(cursor->images[i]);
|
||||
}
|
||||
|
|
@ -57,16 +60,28 @@ static struct wlr_xcursor_image *xcursor_image_create(uint32_t width, uint32_t h
|
|||
image->hotspot_y = hotspot_y;
|
||||
image->delay = delay;
|
||||
|
||||
size_t size = width * height * sizeof(uint32_t);
|
||||
size_t stride = width * sizeof(uint32_t);
|
||||
size_t size = stride * height;
|
||||
image->buffer = malloc(size);
|
||||
if (image->buffer == NULL) {
|
||||
free(image);
|
||||
return NULL;
|
||||
goto err_image;
|
||||
}
|
||||
|
||||
memcpy(image->buffer, buffer, size);
|
||||
|
||||
image->readonly_buffer = readonly_data_buffer_create(DRM_FORMAT_ARGB8888,
|
||||
stride, width, height, image->buffer);
|
||||
if (image->readonly_buffer == NULL) {
|
||||
goto err_buffer;
|
||||
}
|
||||
|
||||
return image;
|
||||
|
||||
err_buffer:
|
||||
free(image->buffer);
|
||||
err_image:
|
||||
free(image);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static struct wlr_xcursor *xcursor_create_from_data(
|
||||
|
|
@ -330,6 +345,10 @@ int wlr_xcursor_frame(struct wlr_xcursor *_cursor, uint32_t time) {
|
|||
return xcursor_frame_and_duration(_cursor, time, NULL);
|
||||
}
|
||||
|
||||
struct wlr_buffer *wlr_xcursor_image_get_buffer(struct wlr_xcursor_image *image) {
|
||||
return &image->readonly_buffer->base;
|
||||
}
|
||||
|
||||
const char *wlr_xcursor_get_resize_name(enum wlr_edges edges) {
|
||||
if (edges & WLR_EDGE_TOP) {
|
||||
if (edges & WLR_EDGE_RIGHT) {
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue