From f0f190bfa5b7ebf2dbb5cc62e20e3b80038382b4 Mon Sep 17 00:00:00 2001 From: Tony Crisci Date: Sun, 13 Aug 2017 17:54:27 -0400 Subject: implement buffer scale Keep track of the scale of the buffer set by surface.set_buffer_scale. Calculate the height and width of the texture from the buffer depending on the scale when it is attached. Use buffer damage to determine the damage of the buffer when flushing damage. Convert surface damage to buffer damage and union to the surface buffer_damage prior to flushing damage. --- include/wlr/render.h | 1 + 1 file changed, 1 insertion(+) (limited to 'include/wlr') diff --git a/include/wlr/render.h b/include/wlr/render.h index 96b61ddf..90081b7a 100644 --- a/include/wlr/render.h +++ b/include/wlr/render.h @@ -61,6 +61,7 @@ struct wlr_texture { bool valid; uint32_t format; int width, height; + int height_from_buffer, width_from_buffer; struct wl_signal destroy_signal; struct wl_resource *resource; }; -- cgit v1.2.3 From 72a33b736fdf2a03d45010182b16532df0753690 Mon Sep 17 00:00:00 2001 From: Tony Crisci Date: Mon, 14 Aug 2017 13:28:59 -0400 Subject: implement texture get buffer size --- include/wlr/render/interface.h | 4 ++++ render/gles2/texture.c | 13 +++++++++++++ render/wlr_texture.c | 5 +++++ types/wlr_surface.c | 14 +++++--------- 4 files changed, 27 insertions(+), 9 deletions(-) (limited to 'include/wlr') diff --git a/include/wlr/render/interface.h b/include/wlr/render/interface.h index 59ece8b1..f98c3bc2 100644 --- a/include/wlr/render/interface.h +++ b/include/wlr/render/interface.h @@ -46,6 +46,8 @@ struct wlr_texture_impl { struct wl_resource *drm_buf); void (*get_matrix)(struct wlr_texture *state, float (*matrix)[16], const float (*projection)[16], int x, int y); + void (*get_buffer_size)(struct wlr_texture *texture, + struct wl_resource *resource, int *width, int *height); void (*bind)(struct wlr_texture *texture); void (*destroy)(struct wlr_texture *texture); }; @@ -53,5 +55,7 @@ struct wlr_texture_impl { void wlr_texture_init(struct wlr_texture *texture, struct wlr_texture_impl *impl); void wlr_texture_bind(struct wlr_texture *texture); +void wlr_texture_get_buffer_size(struct wlr_texture *texture, + struct wl_resource *resource, int *width, int *height); #endif diff --git a/render/gles2/texture.c b/render/gles2/texture.c index 1e80a8d6..1f4ae22c 100644 --- a/render/gles2/texture.c +++ b/render/gles2/texture.c @@ -218,6 +218,18 @@ static void gles2_texture_get_matrix(struct wlr_texture *_texture, wlr_matrix_mul(projection, matrix, matrix); } +static void gles2_texture_get_buffer_size(struct wlr_texture *texture, struct + wl_resource *resource, int *width, int *height) { + struct wl_shm_buffer *buffer = wl_shm_buffer_get(resource); + if (!buffer) { + wlr_log(L_ERROR, "getting buffer size is only implemented for shm buffers"); + return; + } + + *width = wl_shm_buffer_get_width(buffer); + *height = wl_shm_buffer_get_height(buffer); +} + static void gles2_texture_bind(struct wlr_texture *_texture) { struct wlr_gles2_texture *texture = (struct wlr_gles2_texture *)_texture; GL_CALL(glBindTexture(GL_TEXTURE_2D, texture->tex_id)); @@ -247,6 +259,7 @@ static struct wlr_texture_impl wlr_texture_impl = { .update_shm = gles2_texture_update_shm, .upload_drm = gles2_texture_upload_drm, .get_matrix = gles2_texture_get_matrix, + .get_buffer_size = gles2_texture_get_buffer_size, .bind = gles2_texture_bind, .destroy = gles2_texture_destroy, }; diff --git a/render/wlr_texture.c b/render/wlr_texture.c index f98284a1..9faea820 100644 --- a/render/wlr_texture.c +++ b/render/wlr_texture.c @@ -52,3 +52,8 @@ void wlr_texture_get_matrix(struct wlr_texture *texture, float (*matrix)[16], const float (*projection)[16], int x, int y) { texture->impl->get_matrix(texture, matrix, projection, x, y); } + +void wlr_texture_get_buffer_size(struct wlr_texture *texture, struct wl_resource + *resource, int *width, int *height) { + texture->impl->get_buffer_size(texture, resource, width, height); +} diff --git a/types/wlr_surface.c b/types/wlr_surface.c index e862d00e..0861e2e8 100644 --- a/types/wlr_surface.c +++ b/types/wlr_surface.c @@ -126,6 +126,9 @@ static void surface_commit(struct wl_client *client, surface->current.buffer = surface->pending.buffer; } if ((surface->pending.invalid & WLR_SURFACE_INVALID_SURFACE_DAMAGE)) { + int width, height; + wlr_texture_get_buffer_size(surface->texture, surface->current.buffer, &width, &height); + pixman_region32_union(&surface->current.surface_damage, &surface->current.surface_damage, &surface->pending.surface_damage); @@ -140,16 +143,9 @@ static void surface_commit(struct wl_client *client, pixman_region32_union(&surface->current.buffer_damage, &surface->current.buffer_damage, &buffer_damage); - struct wl_shm_buffer *buffer = wl_shm_buffer_get(surface->current.buffer); pixman_region32_intersect_rect(&surface->current.buffer_damage, - &surface->current.buffer_damage, 0, 0, - wl_shm_buffer_get_width(buffer), - wl_shm_buffer_get_height(buffer)); - - // TODO: Surface sizing is complicated - //pixman_region32_intersect_rect(&surface->current.surface_damage, - // &surface->current.surface_damage, - // 0, 0, surface->width, surface->height); + &surface->current.buffer_damage, 0, 0, width, height); + pixman_region32_clear(&surface->pending.surface_damage); pixman_region32_clear(&surface->pending.buffer_damage); } -- cgit v1.2.3 From e46ec57b43054705a6ff649ba358c5d14cccbca5 Mon Sep 17 00:00:00 2001 From: Tony Crisci Date: Mon, 14 Aug 2017 13:54:57 -0400 Subject: implement surface sizing --- examples/compositor/main.c | 4 ++-- include/wlr/types/wlr_surface.h | 2 ++ types/wlr_surface.c | 14 ++++++++++++++ 3 files changed, 18 insertions(+), 2 deletions(-) (limited to 'include/wlr') diff --git a/examples/compositor/main.c b/examples/compositor/main.c index 933e27c7..079f0243 100644 --- a/examples/compositor/main.c +++ b/examples/compositor/main.c @@ -45,8 +45,8 @@ void handle_output_frame(struct output_state *output, struct timespec *ts) { struct wlr_surface *surface = wl_resource_get_user_data(_res); wlr_surface_flush_damage(surface); if (surface->texture->valid) { - wlr_texture_get_matrix(surface->texture, &matrix, - &wlr_output->transform_matrix, 200, 200); + wlr_surface_get_matrix(surface, &matrix, + &wlr_output->transform_matrix, 200, 200); wlr_render_with_matrix(sample->renderer, surface->texture, &matrix); struct wlr_frame_callback *cb, *cnext; diff --git a/include/wlr/types/wlr_surface.h b/include/wlr/types/wlr_surface.h index 60d8b2f6..d3104741 100644 --- a/include/wlr/types/wlr_surface.h +++ b/include/wlr/types/wlr_surface.h @@ -53,5 +53,7 @@ struct wlr_renderer; struct wlr_surface *wlr_surface_create(struct wl_resource *res, struct wlr_renderer *renderer); void wlr_surface_flush_damage(struct wlr_surface *surface); +void wlr_surface_get_matrix(struct wlr_surface *surface, float (*matrix)[16], + const float (*projection)[16], int x, int y); #endif diff --git a/types/wlr_surface.c b/types/wlr_surface.c index b1f273af..12220807 100644 --- a/types/wlr_surface.c +++ b/types/wlr_surface.c @@ -5,6 +5,7 @@ #include #include #include +#include static void surface_destroy(struct wl_client *client, struct wl_resource *resource) { wl_resource_destroy(resource); @@ -268,3 +269,16 @@ struct wlr_surface *wlr_surface_create(struct wl_resource *res, surface, destroy_surface); return surface; } + +void wlr_surface_get_matrix(struct wlr_surface *surface, + float (*matrix)[16], const float (*projection)[16], int x, int y) { + int width = surface->texture->width / surface->current.scale; + int height = surface->texture->height / surface->current.scale; + float world[16]; + wlr_matrix_identity(matrix); + wlr_matrix_translate(&world, x, y, 0); + wlr_matrix_mul(matrix, &world, matrix); + wlr_matrix_scale(&world, width, height, 1); + wlr_matrix_mul(matrix, &world, matrix); + wlr_matrix_mul(projection, matrix, matrix); +} -- cgit v1.2.3 From 84c6dbd58d0dc862488c9c6364f9f78276a893f0 Mon Sep 17 00:00:00 2001 From: Tony Crisci Date: Mon, 14 Aug 2017 15:32:20 -0400 Subject: remove unused dimension from buffer calculations --- include/wlr/render.h | 1 - types/wlr_surface.c | 3 --- 2 files changed, 4 deletions(-) (limited to 'include/wlr') diff --git a/include/wlr/render.h b/include/wlr/render.h index 90081b7a..96b61ddf 100644 --- a/include/wlr/render.h +++ b/include/wlr/render.h @@ -61,7 +61,6 @@ struct wlr_texture { bool valid; uint32_t format; int width, height; - int height_from_buffer, width_from_buffer; struct wl_signal destroy_signal; struct wl_resource *resource; }; diff --git a/types/wlr_surface.c b/types/wlr_surface.c index 12220807..2345bfd1 100644 --- a/types/wlr_surface.c +++ b/types/wlr_surface.c @@ -15,11 +15,8 @@ static void surface_attach(struct wl_client *client, struct wl_resource *resource, struct wl_resource *buffer, int32_t sx, int32_t sy) { struct wlr_surface *surface = wl_resource_get_user_data(resource); - int scale = surface->current.scale; surface->pending.invalid |= WLR_SURFACE_INVALID_BUFFER; surface->pending.buffer = buffer; - surface->texture->height_from_buffer = surface->texture->height / scale; - surface->texture->width_from_buffer = surface->texture->width / scale; } static void surface_damage(struct wl_client *client, -- cgit v1.2.3 From 5b1ce84350d78fa1bf7dc8ff421ae7c919542346 Mon Sep 17 00:00:00 2001 From: Tony Crisci Date: Mon, 14 Aug 2017 16:30:53 -0400 Subject: implement buffer transform --- include/wlr/types/wlr_surface.h | 2 +- types/wlr_surface.c | 107 ++++++++++++++++++++++++++++++++++++---- 2 files changed, 98 insertions(+), 11 deletions(-) (limited to 'include/wlr') diff --git a/include/wlr/types/wlr_surface.h b/include/wlr/types/wlr_surface.h index d3104741..6dbee113 100644 --- a/include/wlr/types/wlr_surface.h +++ b/include/wlr/types/wlr_surface.h @@ -23,7 +23,7 @@ struct wlr_surface_state { int32_t sx, sy; pixman_region32_t surface_damage, buffer_damage; pixman_region32_t opaque, input; - uint32_t transform; + enum wl_output_transform transform; int32_t scale; }; diff --git a/types/wlr_surface.c b/types/wlr_surface.c index 2345bfd1..1872ac3b 100644 --- a/types/wlr_surface.c +++ b/types/wlr_surface.c @@ -91,11 +91,33 @@ static void surface_set_input_region(struct wl_client *client, } } +static void wlr_surface_get_effective_size(struct wlr_surface *surface, + int swidth, int sheight, int *width, int *height) { + int scale = surface->current.scale; + enum wl_output_transform transform = surface->current.transform; + int _width = swidth / scale; + int _height = sheight / scale; + + if (transform == WL_OUTPUT_TRANSFORM_90 || + transform == WL_OUTPUT_TRANSFORM_270 || + transform == WL_OUTPUT_TRANSFORM_FLIPPED_90 || + transform == WL_OUTPUT_TRANSFORM_FLIPPED_270) { + int tmp = _width; + _width = _height; + _height = tmp; + } + + *width = _width; + *height = _height; +} + static void wlr_surface_to_buffer_region(struct wlr_surface *surface, - pixman_region32_t *surface_region, pixman_region32_t *buffer_region) { + pixman_region32_t *surface_region, pixman_region32_t *buffer_region, + int width, int height) { pixman_box32_t *src_rects, *dest_rects; int nrects, i; int scale = surface->current.scale; + enum wl_output_transform transform = surface->current.transform; src_rects = pixman_region32_rectangles(surface_region, &nrects); dest_rects = malloc(nrects * sizeof(*dest_rects)); @@ -104,10 +126,66 @@ static void wlr_surface_to_buffer_region(struct wlr_surface *surface, } for (i = 0; i < nrects; i++) { - dest_rects[i].x1 = src_rects[i].x1 * scale; - dest_rects[i].y1 = src_rects[i].y1 * scale; - dest_rects[i].x2 = src_rects[i].x2 * scale; - dest_rects[i].y2 = src_rects[i].y2 * scale; + switch (transform) { + default: + case WL_OUTPUT_TRANSFORM_NORMAL: + dest_rects[i].x1 = src_rects[i].x1; + dest_rects[i].y1 = src_rects[i].y1; + dest_rects[i].x2 = src_rects[i].x2; + dest_rects[i].y2 = src_rects[i].y2; + break; + case WL_OUTPUT_TRANSFORM_90: + dest_rects[i].x1 = height - src_rects[i].y2; + dest_rects[i].y1 = src_rects[i].x1; + dest_rects[i].x2 = height - src_rects[i].y1; + dest_rects[i].y2 = src_rects[i].x2; + break; + case WL_OUTPUT_TRANSFORM_180: + dest_rects[i].x1 = width - src_rects[i].x2; + dest_rects[i].y1 = height - src_rects[i].y2; + dest_rects[i].x2 = width - src_rects[i].x1; + dest_rects[i].y2 = height - src_rects[i].y1; + break; + case WL_OUTPUT_TRANSFORM_270: + dest_rects[i].x1 = src_rects[i].y1; + dest_rects[i].y1 = width - src_rects[i].x2; + dest_rects[i].x2 = src_rects[i].y2; + dest_rects[i].y2 = width - src_rects[i].x1; + break; + case WL_OUTPUT_TRANSFORM_FLIPPED: + dest_rects[i].x1 = width - src_rects[i].x2; + dest_rects[i].y1 = src_rects[i].y1; + dest_rects[i].x2 = width - src_rects[i].x1; + dest_rects[i].y2 = src_rects[i].y2; + break; + case WL_OUTPUT_TRANSFORM_FLIPPED_90: + dest_rects[i].x1 = height - src_rects[i].y2; + dest_rects[i].y1 = width - src_rects[i].x2; + dest_rects[i].x2 = height - src_rects[i].y1; + dest_rects[i].y2 = width - src_rects[i].x1; + break; + case WL_OUTPUT_TRANSFORM_FLIPPED_180: + dest_rects[i].x1 = src_rects[i].x1; + dest_rects[i].y1 = height - src_rects[i].y2; + dest_rects[i].x2 = src_rects[i].x2; + dest_rects[i].y2 = height - src_rects[i].y1; + break; + case WL_OUTPUT_TRANSFORM_FLIPPED_270: + dest_rects[i].x1 = src_rects[i].y1; + dest_rects[i].y1 = src_rects[i].x1; + dest_rects[i].x2 = src_rects[i].y2; + dest_rects[i].y2 = src_rects[i].x2; + break; + } + } + + if (scale != 1) { + for (i = 0; i < nrects; i++) { + dest_rects[i].x1 *= scale; + dest_rects[i].x2 *= scale; + dest_rects[i].y1 *= scale; + dest_rects[i].y2 *= scale; + } } pixman_region32_fini(buffer_region); @@ -119,20 +197,26 @@ static void surface_commit(struct wl_client *client, struct wl_resource *resource) { struct wlr_surface *surface = wl_resource_get_user_data(resource); surface->current.scale = surface->pending.scale; + surface->current.transform = surface->pending.transform; if ((surface->pending.invalid & WLR_SURFACE_INVALID_BUFFER)) { surface->current.buffer = surface->pending.buffer; } if ((surface->pending.invalid & WLR_SURFACE_INVALID_SURFACE_DAMAGE)) { int width, height; - wlr_texture_get_buffer_size(surface->texture, surface->current.buffer, &width, &height); + wlr_texture_get_buffer_size(surface->texture, surface->current.buffer, + &width, &height); + + int surface_width, surface_height; + wlr_surface_get_effective_size(surface, width, height, &surface_width, + &surface_height); pixman_region32_union(&surface->current.surface_damage, &surface->current.surface_damage, &surface->pending.surface_damage); pixman_region32_intersect_rect(&surface->current.surface_damage, - &surface->current.surface_damage, 0, 0, width / - surface->current.scale, height / surface->current.scale); + &surface->current.surface_damage, 0, 0, surface_width, + surface_height); pixman_region32_union(&surface->current.buffer_damage, &surface->current.buffer_damage, @@ -141,7 +225,7 @@ static void surface_commit(struct wl_client *client, pixman_region32_t buffer_damage; pixman_region32_init(&buffer_damage); wlr_surface_to_buffer_region(surface, &surface->current.surface_damage, - &buffer_damage); + &buffer_damage, surface_width, surface_height); pixman_region32_union(&surface->current.buffer_damage, &surface->current.buffer_damage, &buffer_damage); @@ -203,7 +287,8 @@ release: static void surface_set_buffer_transform(struct wl_client *client, struct wl_resource *resource, int transform) { - wlr_log(L_DEBUG, "TODO: surface surface buffer transform"); + struct wlr_surface *surface = wl_resource_get_user_data(resource); + surface->pending.transform = transform; } static void surface_set_buffer_scale(struct wl_client *client, @@ -260,6 +345,8 @@ struct wlr_surface *wlr_surface_create(struct wl_resource *res, surface->resource = res; surface->current.scale = 1; surface->pending.scale = 1; + surface->current.transform = WL_OUTPUT_TRANSFORM_NORMAL; + surface->pending.transform = WL_OUTPUT_TRANSFORM_NORMAL; wl_signal_init(&surface->signals.commit); wl_list_init(&surface->frame_callback_list); wl_resource_set_implementation(res, &surface_interface, -- cgit v1.2.3 From d5f98dbf61e3627f46c499a2ecac0e581241299b Mon Sep 17 00:00:00 2001 From: Tony Crisci Date: Tue, 15 Aug 2017 08:17:17 -0400 Subject: refactor getting surface size into a function --- include/wlr/types/wlr_surface.h | 2 ++ types/wlr_surface.c | 32 +++++++++++++++----------------- 2 files changed, 17 insertions(+), 17 deletions(-) (limited to 'include/wlr') diff --git a/include/wlr/types/wlr_surface.h b/include/wlr/types/wlr_surface.h index 6dbee113..811097ad 100644 --- a/include/wlr/types/wlr_surface.h +++ b/include/wlr/types/wlr_surface.h @@ -25,6 +25,8 @@ struct wlr_surface_state { pixman_region32_t opaque, input; enum wl_output_transform transform; int32_t scale; + int width, height; + int buffer_width, buffer_height; }; struct wlr_surface { diff --git a/types/wlr_surface.c b/types/wlr_surface.c index 98a5ac27..77acff02 100644 --- a/types/wlr_surface.c +++ b/types/wlr_surface.c @@ -91,12 +91,15 @@ static void surface_set_input_region(struct wl_client *client, } } -static void wlr_surface_get_effective_size(struct wlr_surface *surface, - int swidth, int sheight, int *width, int *height) { +static void wlr_surface_update_size(struct wlr_surface *surface) { int scale = surface->current.scale; enum wl_output_transform transform = surface->current.transform; - int _width = swidth / scale; - int _height = sheight / scale; + + wlr_texture_get_buffer_size(surface->texture, surface->current.buffer, + &surface->current.buffer_width, &surface->current.buffer_height); + + int _width = surface->current.buffer_width / scale; + int _height = surface->current.buffer_height / scale; if (transform == WL_OUTPUT_TRANSFORM_90 || transform == WL_OUTPUT_TRANSFORM_270 || @@ -107,8 +110,8 @@ static void wlr_surface_get_effective_size(struct wlr_surface *surface, _height = tmp; } - *width = _width; - *height = _height; + surface->current.width = _width; + surface->current.height = _height; } static void wlr_surface_to_buffer_region(struct wlr_surface *surface, @@ -203,20 +206,14 @@ static void surface_commit(struct wl_client *client, surface->current.buffer = surface->pending.buffer; } if ((surface->pending.invalid & WLR_SURFACE_INVALID_SURFACE_DAMAGE)) { - int width, height; - wlr_texture_get_buffer_size(surface->texture, surface->current.buffer, - &width, &height); - - int surface_width, surface_height; - wlr_surface_get_effective_size(surface, width, height, &surface_width, - &surface_height); + wlr_surface_update_size(surface); pixman_region32_union(&surface->current.surface_damage, &surface->current.surface_damage, &surface->pending.surface_damage); pixman_region32_intersect_rect(&surface->current.surface_damage, - &surface->current.surface_damage, 0, 0, surface_width, - surface_height); + &surface->current.surface_damage, 0, 0, surface->current.width, + surface->current.height); pixman_region32_union(&surface->current.buffer_damage, &surface->current.buffer_damage, @@ -225,12 +222,13 @@ static void surface_commit(struct wl_client *client, pixman_region32_t buffer_damage; pixman_region32_init(&buffer_damage); wlr_surface_to_buffer_region(surface, &surface->current.surface_damage, - &buffer_damage, surface_width, surface_height); + &buffer_damage, surface->current.width, surface->current.height); pixman_region32_union(&surface->current.buffer_damage, &surface->current.buffer_damage, &buffer_damage); pixman_region32_intersect_rect(&surface->current.buffer_damage, - &surface->current.buffer_damage, 0, 0, width, height); + &surface->current.buffer_damage, 0, 0, + surface->current.buffer_width, surface->current.buffer_height); pixman_region32_clear(&surface->pending.surface_damage); pixman_region32_clear(&surface->pending.buffer_damage); -- cgit v1.2.3