cursor: Split out the structure for the actual textures / buffers out

This will allow us to have a MetaCursorReference 'subclass' that's
lazily loaded. We currently always load all the images.

The long-term plan is to have a subclass for each "backend" and only
have CoglTexture as a common denominator. For the nested X11 backend,
we use XDefineCursor on our stage window. For the Wayland backend, we
would use set_cursor on our stage surface. For the native backend, we
would use the GBM code that's there right now.

The CoglTexture is there to be a "shared fallback" between all devices,
and also for the get_sprite API.

The odd man out is the X11 compositor case. For that, we need to move
the responsibility of setting the final cursor image out of
MetaCursorTracker, and simply have it be about tracking the used sprite
image and pointer position.
This commit is contained in:
Jasper St. Pierre 2014-03-31 17:01:14 -04:00
parent f4e299ca46
commit 78dbf8cb56
3 changed files with 53 additions and 44 deletions

View File

@ -27,12 +27,16 @@
#include <cogl/cogl.h> #include <cogl/cogl.h>
#include <gbm.h> #include <gbm.h>
struct _MetaCursorReference { typedef struct {
int ref_count;
CoglTexture2D *texture; CoglTexture2D *texture;
struct gbm_bo *bo; struct gbm_bo *bo;
int hot_x, hot_y; int hot_x, hot_y;
} MetaCursorImage;
struct _MetaCursorReference {
int ref_count;
MetaCursorImage image;
}; };
CoglTexture *meta_cursor_reference_get_cogl_texture (MetaCursorReference *cursor, CoglTexture *meta_cursor_reference_get_cogl_texture (MetaCursorReference *cursor,

View File

@ -253,9 +253,9 @@ meta_cursor_reference_take_texture (CoglTexture2D *texture,
self = g_slice_new0 (MetaCursorReference); self = g_slice_new0 (MetaCursorReference);
self->ref_count = 1; self->ref_count = 1;
self->texture = texture; self->image.texture = texture;
self->hot_x = hot_x; self->image.hot_x = hot_x;
self->hot_y = hot_y; self->image.hot_y = hot_y;
return self; return self;
} }

View File

@ -46,13 +46,18 @@ meta_cursor_reference_ref (MetaCursorReference *self)
return self; return self;
} }
static void
meta_cursor_image_free (MetaCursorImage *image)
{
cogl_object_unref (image->texture);
if (image->bo)
gbm_bo_destroy (image->bo);
}
static void static void
meta_cursor_reference_free (MetaCursorReference *self) meta_cursor_reference_free (MetaCursorReference *self)
{ {
cogl_object_unref (self->texture); meta_cursor_image_free (&self->image);
if (self->bo)
gbm_bo_destroy (self->bo);
g_slice_free (MetaCursorReference, self); g_slice_free (MetaCursorReference, self);
} }
@ -209,12 +214,12 @@ meta_cursor_reference_from_xcursor_image (MetaCursorTracker *tracker,
self = g_slice_new0 (MetaCursorReference); self = g_slice_new0 (MetaCursorReference);
self->ref_count = 1; self->ref_count = 1;
self->hot_x = image->xhot; self->image.hot_x = image->xhot;
self->hot_y = image->yhot; self->image.hot_y = image->yhot;
clutter_backend = clutter_get_default_backend (); clutter_backend = clutter_get_default_backend ();
cogl_context = clutter_backend_get_cogl_context (clutter_backend); cogl_context = clutter_backend_get_cogl_context (clutter_backend);
self->texture = cogl_texture_2d_new_from_data (cogl_context, self->image.texture = cogl_texture_2d_new_from_data (cogl_context,
width, height, width, height,
cogl_format, cogl_format,
rowstride, rowstride,
@ -235,14 +240,14 @@ meta_cursor_reference_from_xcursor_image (MetaCursorTracker *tracker,
uint32_t buf[64 * 64]; uint32_t buf[64 * 64];
int i; int i;
self->bo = gbm_bo_create (tracker->gbm, 64, 64, self->image.bo = gbm_bo_create (tracker->gbm, 64, 64,
gbm_format, GBM_BO_USE_CURSOR_64X64 | GBM_BO_USE_WRITE); gbm_format, GBM_BO_USE_CURSOR_64X64 | GBM_BO_USE_WRITE);
memset (buf, 0, sizeof(buf)); memset (buf, 0, sizeof(buf));
for (i = 0; i < height; i++) for (i = 0; i < height; i++)
memcpy (buf + i * 64, image->pixels + i * width, width * 4); memcpy (buf + i * 64, image->pixels + i * width, width * 4);
gbm_bo_write (self->bo, buf, 64 * 64 * 4); gbm_bo_write (self->image.bo, buf, 64 * 64 * 4);
} }
else else
meta_warning ("HW cursor for format %d not supported\n", gbm_format); meta_warning ("HW cursor for format %d not supported\n", gbm_format);
@ -287,8 +292,8 @@ meta_cursor_reference_from_buffer (MetaCursorTracker *tracker,
self = g_slice_new0 (MetaCursorReference); self = g_slice_new0 (MetaCursorReference);
self->ref_count = 1; self->ref_count = 1;
self->hot_x = hot_x; self->image.hot_x = hot_x;
self->hot_y = hot_y; self->image.hot_y = hot_y;
backend = clutter_get_default_backend (); backend = clutter_get_default_backend ();
cogl_context = clutter_backend_get_cogl_context (backend); cogl_context = clutter_backend_get_cogl_context (backend);
@ -327,7 +332,7 @@ meta_cursor_reference_from_buffer (MetaCursorTracker *tracker,
gbm_format = GBM_FORMAT_ARGB8888; gbm_format = GBM_FORMAT_ARGB8888;
} }
self->texture = cogl_texture_2d_new_from_data (cogl_context, self->image.texture = cogl_texture_2d_new_from_data (cogl_context,
width, height, width, height,
cogl_format, cogl_format,
rowstride, rowstride,
@ -349,7 +354,7 @@ meta_cursor_reference_from_buffer (MetaCursorTracker *tracker,
uint8_t buf[4 * 64 * 64]; uint8_t buf[4 * 64 * 64];
int i; int i;
self->bo = gbm_bo_create (tracker->gbm, 64, 64, self->image.bo = gbm_bo_create (tracker->gbm, 64, 64,
gbm_format, GBM_BO_USE_CURSOR_64X64 | GBM_BO_USE_WRITE); gbm_format, GBM_BO_USE_CURSOR_64X64 | GBM_BO_USE_WRITE);
data = wl_shm_buffer_get_data (shm_buffer); data = wl_shm_buffer_get_data (shm_buffer);
@ -357,7 +362,7 @@ meta_cursor_reference_from_buffer (MetaCursorTracker *tracker,
for (i = 0; i < height; i++) for (i = 0; i < height; i++)
memcpy (buf + i * 4 * 64, data + i * rowstride, 4 * width); memcpy (buf + i * 4 * 64, data + i * rowstride, 4 * width);
gbm_bo_write (self->bo, buf, 64 * 64 * 4); gbm_bo_write (self->image.bo, buf, 64 * 64 * 4);
} }
else else
meta_warning ("HW cursor for format %d not supported\n", gbm_format); meta_warning ("HW cursor for format %d not supported\n", gbm_format);
@ -367,9 +372,9 @@ meta_cursor_reference_from_buffer (MetaCursorTracker *tracker,
{ {
int width, height; int width, height;
self->texture = cogl_wayland_texture_2d_new_from_buffer (cogl_context, buffer, NULL); self->image.texture = cogl_wayland_texture_2d_new_from_buffer (cogl_context, buffer, NULL);
width = cogl_texture_get_width (COGL_TEXTURE (self->texture)); width = cogl_texture_get_width (COGL_TEXTURE (self->image.texture));
height = cogl_texture_get_height (COGL_TEXTURE (self->texture)); height = cogl_texture_get_height (COGL_TEXTURE (self->image.texture));
/* HW cursors must be 64x64, but 64x64 is huge, and no cursor theme actually uses /* HW cursors must be 64x64, but 64x64 is huge, and no cursor theme actually uses
that, so themed cursors must be padded with transparent pixels to fill the that, so themed cursors must be padded with transparent pixels to fill the
@ -385,9 +390,9 @@ meta_cursor_reference_from_buffer (MetaCursorTracker *tracker,
if (tracker->gbm) if (tracker->gbm)
{ {
self->bo = gbm_bo_import (tracker->gbm, GBM_BO_IMPORT_WL_BUFFER, self->image.bo = gbm_bo_import (tracker->gbm, GBM_BO_IMPORT_WL_BUFFER,
buffer, GBM_BO_USE_CURSOR_64X64); buffer, GBM_BO_USE_CURSOR_64X64);
if (!self->bo) if (!self->image.bo)
meta_warning ("Importing HW cursor from wl_buffer failed\n"); meta_warning ("Importing HW cursor from wl_buffer failed\n");
} }
} }
@ -401,10 +406,10 @@ meta_cursor_reference_get_cogl_texture (MetaCursorReference *cursor,
int *hot_y) int *hot_y)
{ {
if (hot_x) if (hot_x)
*hot_x = cursor->hot_x; *hot_x = cursor->image.hot_x;
if (hot_y) if (hot_y)
*hot_y = cursor->hot_y; *hot_y = cursor->image.hot_y;
return COGL_TEXTURE (cursor->texture); return COGL_TEXTURE (cursor->image.texture);
} }
struct gbm_bo * struct gbm_bo *
@ -413,8 +418,8 @@ meta_cursor_reference_get_gbm_bo (MetaCursorReference *cursor,
int *hot_y) int *hot_y)
{ {
if (hot_x) if (hot_x)
*hot_x = cursor->hot_x; *hot_x = cursor->image.hot_x;
if (hot_y) if (hot_y)
*hot_y = cursor->hot_y; *hot_y = cursor->image.hot_y;
return cursor->bo; return cursor->image.bo;
} }