diff options
author | Simon Ser <contact@emersion.fr> | 2020-12-04 16:41:16 +0100 |
---|---|---|
committer | Simon Ser <contact@emersion.fr> | 2021-06-02 11:08:52 +0200 |
commit | 2b0a1aeed5246fe41e6268c333f1e00b874fab02 (patch) | |
tree | 423b9b89f425efe0ed7feb407de196d6e833189e /backend | |
parent | 01e0f51fadaec17fde401da795b958b721922344 (diff) |
output: take a wlr_buffer in set_cursor
Instead of passing a wlr_texture to the backend, directly pass a
wlr_buffer. Use get_cursor_size and get_cursor_formats to create
a wlr_buffer that can be used as a cursor.
We don't want to pass a wlr_texture because we want to remove as
many rendering bits from the backend as possible.
Diffstat (limited to 'backend')
-rw-r--r-- | backend/drm/drm.c | 115 | ||||
-rw-r--r-- | backend/drm/renderer.c | 6 | ||||
-rw-r--r-- | backend/wayland/output.c | 81 | ||||
-rw-r--r-- | backend/x11/output.c | 105 |
4 files changed, 67 insertions, 240 deletions
diff --git a/backend/drm/drm.c b/backend/drm/drm.c index c9d3b54a..9efff464 100644 --- a/backend/drm/drm.c +++ b/backend/drm/drm.c @@ -25,7 +25,9 @@ #include "backend/drm/iface.h" #include "backend/drm/util.h" #include "render/pixel_format.h" +#include "render/drm_format_set.h" #include "render/swapchain.h" +#include "render/wlr_renderer.h" #include "types/wlr_buffer.h" #include "util/signal.h" @@ -329,7 +331,7 @@ static bool drm_connector_attach_render(struct wlr_output *output, static void drm_plane_set_committed(struct wlr_drm_plane *plane) { drm_fb_move(&plane->queued_fb, &plane->pending_fb); - if (plane->queued_fb) { + if (plane->queued_fb && plane->surf.swapchain) { wlr_swapchain_set_buffer_submitted(plane->surf.swapchain, plane->queued_fb->wlr_buf); } @@ -834,9 +836,7 @@ struct wlr_output_mode *wlr_drm_connector_add_mode(struct wlr_output *output, } static bool drm_connector_set_cursor(struct wlr_output *output, - struct wlr_texture *texture, float scale, - enum wl_output_transform transform, - int32_t hotspot_x, int32_t hotspot_y, bool update_texture) { + struct wlr_buffer *buffer, int hotspot_x, int hotspot_y) { struct wlr_drm_connector *conn = get_drm_connector_from_output(output); struct wlr_drm_backend *drm = conn->backend; struct wlr_drm_crtc *crtc = conn->crtc; @@ -850,90 +850,59 @@ static bool drm_connector_set_cursor(struct wlr_output *output, return false; } - if (!plane->surf.swapchain) { - int ret; - uint64_t w, h; - ret = drmGetCap(drm->fd, DRM_CAP_CURSOR_WIDTH, &w); - w = ret ? 64 : w; - ret = drmGetCap(drm->fd, DRM_CAP_CURSOR_HEIGHT, &h); - h = ret ? 64 : h; - - if (!drm_plane_init_surface(plane, drm, w, h, true)) { - wlr_drm_conn_log(conn, WLR_ERROR, "Cannot allocate cursor resources"); - return false; - } - } - - struct wlr_box hotspot = { .x = hotspot_x, .y = hotspot_y }; - wlr_box_transform(&hotspot, &hotspot, - wlr_output_transform_invert(output->transform), - plane->surf.width, plane->surf.height); - - if (plane->cursor_hotspot_x != hotspot.x || - plane->cursor_hotspot_y != hotspot.y) { + if (plane->cursor_hotspot_x != hotspot_x || + plane->cursor_hotspot_y != hotspot_y) { // Update cursor hotspot - conn->cursor_x -= hotspot.x - plane->cursor_hotspot_x; - conn->cursor_y -= hotspot.y - plane->cursor_hotspot_y; - plane->cursor_hotspot_x = hotspot.x; - plane->cursor_hotspot_y = hotspot.y; + conn->cursor_x -= hotspot_x - plane->cursor_hotspot_x; + conn->cursor_y -= hotspot_y - plane->cursor_hotspot_y; + plane->cursor_hotspot_x = hotspot_x; + plane->cursor_hotspot_y = hotspot_y; wlr_output_update_needs_frame(output); } - if (!update_texture) { - // Don't update cursor image - return true; - } - plane->cursor_enabled = false; - if (texture != NULL) { - int width = texture->width * output->scale / scale; - int height = texture->height * output->scale / scale; - - if (width > (int)plane->surf.width || height > (int)plane->surf.height) { - wlr_drm_conn_log(conn, WLR_ERROR, "Cursor too large (max %dx%d)", - (int)plane->surf.width, (int)plane->surf.height); - return false; - } - - if (!drm_surface_make_current(&plane->surf, NULL)) { + if (buffer != NULL) { + if ((uint64_t)buffer->width != drm->cursor_width || + (uint64_t)buffer->height != drm->cursor_height) { + wlr_drm_conn_log(conn, WLR_DEBUG, "Cursor buffer size mismatch"); return false; } - struct wlr_renderer *rend = plane->surf.renderer->wlr_rend; - - struct wlr_box cursor_box = { .width = width, .height = height }; + struct wlr_buffer *local_buf; + if (drm->parent) { + struct wlr_drm_format *format = + drm_plane_pick_render_format(plane, &drm->renderer); + if (format == NULL) { + wlr_log(WLR_ERROR, "Failed to pick cursor plane format"); + return false; + } - float output_matrix[9]; - wlr_matrix_identity(output_matrix); - if (output->transform != WL_OUTPUT_TRANSFORM_NORMAL) { - struct wlr_box tr_size = { - .width = plane->surf.width, - .height = plane->surf.height, - }; - wlr_box_transform(&tr_size, &tr_size, output->transform, 0, 0); + bool ok = init_drm_surface(&plane->mgpu_surf, &drm->renderer, + buffer->width, buffer->height, format); + free(format); + if (!ok) { + return false; + } - wlr_matrix_translate(output_matrix, plane->surf.width / 2.0, - plane->surf.height / 2.0); - wlr_matrix_transform(output_matrix, output->transform); - wlr_matrix_translate(output_matrix, - tr_size.width / 2.0, - - tr_size.height / 2.0); + local_buf = drm_surface_blit(&plane->mgpu_surf, buffer); + if (local_buf == NULL) { + return false; + } + } else { + local_buf = wlr_buffer_lock(buffer); } - float matrix[9]; - wlr_matrix_project_box(matrix, &cursor_box, transform, 0, - output_matrix); - - wlr_renderer_begin(rend, plane->surf.width, plane->surf.height); - wlr_renderer_clear(rend, (float[]){ 0.0, 0.0, 0.0, 0.0 }); - wlr_render_texture_with_matrix(rend, texture, matrix, 1.0); - wlr_renderer_end(rend); - - if (!drm_plane_lock_surface(plane, drm)) { + bool ok = drm_fb_import(&plane->pending_fb, drm, local_buf, + &plane->formats); + wlr_buffer_unlock(local_buf); + if (!ok) { return false; } plane->cursor_enabled = true; + plane->cursor_width = buffer->width; + plane->cursor_height = buffer->height; } wlr_output_update_needs_frame(output); @@ -977,8 +946,8 @@ bool drm_connector_is_cursor_visible(struct wlr_drm_connector *conn) { return plane->cursor_enabled && conn->cursor_x < conn->output.width && conn->cursor_y < conn->output.height && - conn->cursor_x + (int)plane->surf.width >= 0 && - conn->cursor_y + (int)plane->surf.height >= 0; + conn->cursor_x + plane->cursor_width >= 0 && + conn->cursor_y + plane->cursor_height >= 0; } static void dealloc_crtc(struct wlr_drm_connector *conn); diff --git a/backend/drm/renderer.c b/backend/drm/renderer.c index 7d6c155a..e1e35e63 100644 --- a/backend/drm/renderer.c +++ b/backend/drm/renderer.c @@ -62,7 +62,7 @@ void finish_drm_renderer(struct wlr_drm_renderer *renderer) { gbm_device_destroy(renderer->gbm); } -static bool init_drm_surface(struct wlr_drm_surface *surf, +bool init_drm_surface(struct wlr_drm_surface *surf, struct wlr_drm_renderer *renderer, uint32_t width, uint32_t height, const struct wlr_drm_format *drm_format) { if (surf->width == width && surf->height == height) { @@ -126,7 +126,7 @@ void drm_surface_unset_current(struct wlr_drm_surface *surf) { surf->back_buffer = NULL; } -static struct wlr_buffer *drm_surface_blit(struct wlr_drm_surface *surf, +struct wlr_buffer *drm_surface_blit(struct wlr_drm_surface *surf, struct wlr_buffer *buffer) { struct wlr_renderer *renderer = surf->renderer->wlr_rend; @@ -191,7 +191,7 @@ static struct wlr_drm_format *create_linear_format(uint32_t format) { return fmt; } -static struct wlr_drm_format *drm_plane_pick_render_format( +struct wlr_drm_format *drm_plane_pick_render_format( struct wlr_drm_plane *plane, struct wlr_drm_renderer *renderer) { const struct wlr_drm_format_set *render_formats = wlr_renderer_get_render_formats(renderer->wlr_rend); diff --git a/backend/wayland/output.c b/backend/wayland/output.c index cdf78b2a..62e9b5af 100644 --- a/backend/wayland/output.c +++ b/backend/wayland/output.c @@ -406,28 +406,12 @@ static void output_rollback_render(struct wlr_output *wlr_output) { } static bool output_set_cursor(struct wlr_output *wlr_output, - struct wlr_texture *texture, float scale, - enum wl_output_transform transform, - int32_t hotspot_x, int32_t hotspot_y, bool update_texture) { + struct wlr_buffer *wlr_buffer, int hotspot_x, int hotspot_y) { struct wlr_wl_output *output = get_wl_output_from_output(wlr_output); struct wlr_wl_backend *backend = output->backend; - struct wlr_renderer *renderer = wlr_backend_get_renderer(&backend->backend); - struct wlr_allocator *allocator = backend_get_allocator(&backend->backend); - - struct wlr_box hotspot = { .x = hotspot_x, .y = hotspot_y }; - wlr_box_transform(&hotspot, &hotspot, - wlr_output_transform_invert(wlr_output->transform), - output->cursor.width, output->cursor.height); - - // TODO: use output->wlr_output.transform to transform pixels and hotpot - output->cursor.hotspot_x = hotspot.x; - output->cursor.hotspot_y = hotspot.y; - if (!update_texture) { - // Update hotspot without changing cursor image - update_wl_output_cursor(output); - return true; - } + output->cursor.hotspot_x = hotspot_x; + output->cursor.hotspot_y = hotspot_y; if (output->cursor.surface == NULL) { output->cursor.surface = @@ -435,58 +419,7 @@ static bool output_set_cursor(struct wlr_output *wlr_output, } struct wl_surface *surface = output->cursor.surface; - if (texture != NULL) { - int width = texture->width * wlr_output->scale / scale; - int height = texture->height * wlr_output->scale / scale; - - if (output->cursor.swapchain == NULL || - output->cursor.swapchain->width != width || - output->cursor.swapchain->height != height) { - wlr_swapchain_destroy(output->cursor.swapchain); - output->cursor.swapchain = wlr_swapchain_create(allocator, - width, height, output->backend->format); - if (output->cursor.swapchain == NULL) { - return false; - } - } - - struct wlr_buffer *wlr_buffer = - wlr_swapchain_acquire(output->cursor.swapchain, NULL); - if (wlr_buffer == NULL) { - return false; - } - - if (!wlr_renderer_bind_buffer(renderer, wlr_buffer)) { - return false; - } - - struct wlr_box cursor_box = { - .width = width, - .height = height, - }; - - float output_matrix[9]; - wlr_matrix_identity(output_matrix); - if (wlr_output->transform != WL_OUTPUT_TRANSFORM_NORMAL) { - struct wlr_box tr_size = { .width = width, .height = height }; - wlr_box_transform(&tr_size, &tr_size, wlr_output->transform, 0, 0); - - wlr_matrix_translate(output_matrix, width / 2.0, height / 2.0); - wlr_matrix_transform(output_matrix, wlr_output->transform); - wlr_matrix_translate(output_matrix, - - tr_size.width / 2.0, - tr_size.height / 2.0); - } - - float matrix[9]; - wlr_matrix_project_box(matrix, &cursor_box, transform, 0, output_matrix); - - wlr_renderer_begin(renderer, width, height); - wlr_renderer_clear(renderer, (float[]){ 0.0, 0.0, 0.0, 0.0 }); - wlr_render_texture_with_matrix(renderer, texture, matrix, 1.0); - wlr_renderer_end(renderer); - - wlr_renderer_bind_buffer(renderer, NULL); - + if (wlr_buffer != NULL) { struct wlr_wl_buffer *buffer = get_or_create_wl_buffer(output->backend, wlr_buffer); if (buffer == NULL) { @@ -496,11 +429,6 @@ static bool output_set_cursor(struct wlr_output *wlr_output, wl_surface_attach(surface, buffer->wl_buffer, 0, 0); wl_surface_damage_buffer(surface, 0, 0, INT32_MAX, INT32_MAX); wl_surface_commit(surface); - - wlr_buffer_unlock(wlr_buffer); - - output->cursor.width = width; - output->cursor.height = height; } else { wl_surface_attach(surface, NULL, 0, 0); wl_surface_commit(surface); @@ -530,7 +458,6 @@ static void output_destroy(struct wlr_output *wlr_output) { wl_list_remove(&output->link); - wlr_swapchain_destroy(output->cursor.swapchain); if (output->cursor.surface) { wl_surface_destroy(output->cursor.surface); } diff --git a/backend/x11/output.c b/backend/x11/output.c index bba938d2..56161dee 100644 --- a/backend/x11/output.c +++ b/backend/x11/output.c @@ -420,83 +420,38 @@ static void update_x11_output_cursor(struct wlr_x11_output *output, } static bool output_cursor_to_picture(struct wlr_x11_output *output, - struct wlr_texture *texture, enum wl_output_transform transform, - int width, int height) { + struct wlr_buffer *buffer) { struct wlr_x11_backend *x11 = output->x11; - struct wlr_allocator *allocator = backend_get_allocator(&x11->backend); struct wlr_renderer *renderer = wlr_backend_get_renderer(&x11->backend); - int depth = 32; - int stride = width * 4; if (output->cursor.pic != XCB_NONE) { xcb_render_free_picture(x11->xcb, output->cursor.pic); } output->cursor.pic = XCB_NONE; - if (texture == NULL) { + if (buffer == NULL) { return true; } - if (output->cursor.swapchain == NULL || - output->cursor.swapchain->width != width || - output->cursor.swapchain->height != height) { - wlr_swapchain_destroy(output->cursor.swapchain); - output->cursor.swapchain = wlr_swapchain_create(allocator, - width, height, x11->drm_format); - if (output->cursor.swapchain == NULL) { - return false; - } - } - - struct wlr_buffer *wlr_buffer = - wlr_swapchain_acquire(output->cursor.swapchain, NULL); - if (wlr_buffer == NULL) { - return false; - } + int depth = 32; + int stride = buffer->width * 4; - if (!wlr_renderer_bind_buffer(renderer, wlr_buffer)) { + if (!wlr_renderer_bind_buffer(renderer, buffer)) { return false; } - uint8_t *data = malloc(width * height * 4); + uint8_t *data = malloc(buffer->height * stride); if (data == NULL) { return false; } - struct wlr_box cursor_box = { - .width = width, - .height = height, - }; - - float output_matrix[9]; - wlr_matrix_identity(output_matrix); - if (output->wlr_output.transform != WL_OUTPUT_TRANSFORM_NORMAL) { - struct wlr_box tr_size = { .width = width, .height = height }; - wlr_box_transform(&tr_size, &tr_size, output->wlr_output.transform, 0, 0); - - wlr_matrix_translate(output_matrix, width / 2.0, height / 2.0); - wlr_matrix_transform(output_matrix, output->wlr_output.transform); - wlr_matrix_translate(output_matrix, - - tr_size.width / 2.0, - tr_size.height / 2.0); - } - - float matrix[9]; - wlr_matrix_project_box(matrix, &cursor_box, transform, 0, output_matrix); - - wlr_renderer_begin(renderer, width, height); - wlr_renderer_clear(renderer, (float[]){ 0.0, 0.0, 0.0, 0.0 }); - wlr_render_texture_with_matrix(renderer, texture, matrix, 1.0); - wlr_renderer_end(renderer); - bool result = wlr_renderer_read_pixels( renderer, DRM_FORMAT_ARGB8888, NULL, - width * 4, width, height, 0, 0, 0, 0, + stride, buffer->width, buffer->height, 0, 0, 0, 0, data); wlr_renderer_bind_buffer(renderer, NULL); - wlr_buffer_unlock(wlr_buffer); - if (!result) { free(data); return false; @@ -504,7 +459,7 @@ static bool output_cursor_to_picture(struct wlr_x11_output *output, xcb_pixmap_t pix = xcb_generate_id(x11->xcb); xcb_create_pixmap(x11->xcb, depth, pix, output->win, - width, height); + buffer->width, buffer->height); output->cursor.pic = xcb_generate_id(x11->xcb); xcb_render_create_picture(x11->xcb, output->cursor.pic, @@ -514,9 +469,8 @@ static bool output_cursor_to_picture(struct wlr_x11_output *output, xcb_create_gc(x11->xcb, gc, pix, 0, NULL); xcb_put_image(x11->xcb, XCB_IMAGE_FORMAT_Z_PIXMAP, - pix, gc, width, height, 0, 0, 0, depth, - stride * height * sizeof(uint8_t), - data); + pix, gc, buffer->width, buffer->height, 0, 0, 0, depth, + stride * buffer->height * sizeof(uint8_t), data); free(data); xcb_free_gc(x11->xcb, gc); xcb_free_pixmap(x11->xcb, pix); @@ -525,55 +479,32 @@ static bool output_cursor_to_picture(struct wlr_x11_output *output, } static bool output_set_cursor(struct wlr_output *wlr_output, - struct wlr_texture *texture, float scale, - enum wl_output_transform transform, - int32_t hotspot_x, int32_t hotspot_y, bool update_texture) { + struct wlr_buffer *buffer, int32_t hotspot_x, int32_t hotspot_y) { struct wlr_x11_output *output = get_x11_output_from_output(wlr_output); struct wlr_x11_backend *x11 = output->x11; - int width = 0, height = 0; if (x11->argb32 == XCB_NONE) { return false; } - if (texture != NULL) { - width = texture->width * wlr_output->scale / scale; - height = texture->height * wlr_output->scale / scale; - + if (buffer != NULL) { if (hotspot_x < 0) { hotspot_x = 0; } - if ((uint32_t)hotspot_x > texture->width) { - hotspot_x = texture->width; + if (hotspot_x > buffer->width) { + hotspot_x = buffer->width; } if (hotspot_y < 0) { hotspot_y = 0; } - if ((uint32_t)hotspot_y > texture->height) { - hotspot_y = texture->height; - } - } - - struct wlr_box hotspot = { .x = hotspot_x, .y = hotspot_y }; - wlr_box_transform(&hotspot, &hotspot, - wlr_output_transform_invert(wlr_output->transform), - width, height); - - if (!update_texture) { - // This means we previously had a failure of some sort. - if (texture != NULL && output->cursor.pic == XCB_NONE) { - return false; + if (hotspot_y > buffer->height) { + hotspot_y = buffer->height; } - - // Update hotspot without changing cursor image - update_x11_output_cursor(output, hotspot.x, hotspot.y); - return true; } - bool success = output_cursor_to_picture(output, texture, transform, - width, height); + bool success = output_cursor_to_picture(output, buffer); - update_x11_output_cursor(output, hotspot.x, hotspot.y); + update_x11_output_cursor(output, hotspot_x, hotspot_y); return success; } |