scene: add damage tracking support

This commit is contained in:
Simon Ser 2021-08-17 15:12:47 +02:00 committed by Simon Zeni
parent 9195b77e14
commit 44f0f7a0a7
2 changed files with 226 additions and 21 deletions

View file

@ -68,6 +68,7 @@ struct wlr_scene_surface {
// private state // private state
struct wl_listener surface_destroy; struct wl_listener surface_destroy;
struct wl_listener surface_commit;
}; };
/** A scene-graph node displaying a solid-colored rectangle */ /** A scene-graph node displaying a solid-colored rectangle */
@ -84,6 +85,8 @@ struct wlr_scene_output {
struct wlr_scene *scene; struct wlr_scene *scene;
struct wlr_addon addon; struct wlr_addon addon;
struct wlr_output_damage *damage;
int x, y; int x, y;
}; };

View file

@ -4,8 +4,10 @@
#include <wlr/backend.h> #include <wlr/backend.h>
#include <wlr/render/wlr_renderer.h> #include <wlr/render/wlr_renderer.h>
#include <wlr/types/wlr_matrix.h> #include <wlr/types/wlr_matrix.h>
#include <wlr/types/wlr_output_damage.h>
#include <wlr/types/wlr_scene.h> #include <wlr/types/wlr_scene.h>
#include <wlr/types/wlr_surface.h> #include <wlr/types/wlr_surface.h>
#include <wlr/util/region.h>
#include "util/signal.h" #include "util/signal.h"
static struct wlr_scene *scene_root_from_node(struct wlr_scene_node *node) { static struct wlr_scene *scene_root_from_node(struct wlr_scene_node *node) {
@ -61,11 +63,14 @@ static void scene_node_finish(struct wlr_scene_node *node) {
scene_node_state_finish(&node->state); scene_node_state_finish(&node->state);
} }
static void scene_node_damage_whole(struct wlr_scene_node *node);
void wlr_scene_node_destroy(struct wlr_scene_node *node) { void wlr_scene_node_destroy(struct wlr_scene_node *node) {
if (node == NULL) { if (node == NULL) {
return; return;
} }
scene_node_damage_whole(node);
scene_node_finish(node); scene_node_finish(node);
switch (node->type) { switch (node->type) {
@ -108,6 +113,53 @@ static void scene_surface_handle_surface_destroy(struct wl_listener *listener,
wlr_scene_node_destroy(&scene_surface->node); wlr_scene_node_destroy(&scene_surface->node);
} }
static struct wlr_scene *scene_node_get_root(struct wlr_scene_node *node) {
while (node->parent != NULL) {
node = node->parent;
}
return scene_root_from_node(node);
}
static void scene_surface_handle_surface_commit(struct wl_listener *listener,
void *data) {
struct wlr_scene_surface *scene_surface =
wl_container_of(listener, scene_surface, surface_commit);
struct wlr_surface *surface = scene_surface->surface;
if (!pixman_region32_not_empty(&surface->buffer_damage)) {
return;
}
int lx, ly;
if (!wlr_scene_node_coords(&scene_surface->node, &lx, &ly)) {
return;
}
struct wlr_scene *scene = scene_node_get_root(&scene_surface->node);
struct wlr_scene_output *scene_output;
wl_list_for_each(scene_output, &scene->outputs, link) {
struct wlr_output *output = scene_output->output;
pixman_region32_t damage;
pixman_region32_init(&damage);
wlr_surface_get_effective_damage(surface, &damage);
pixman_region32_translate(&damage,
lx - scene_output->x, ly - scene_output->y);
wlr_region_scale(&damage, &damage, output->scale);
if (ceil(output->scale) > surface->current.scale) {
// When scaling up a surface it'll become blurry, so we need to
// expand the damage region.
wlr_region_expand(&damage, &damage,
ceil(output->scale) - surface->current.scale);
}
wlr_output_damage_add(scene_output->damage, &damage);
pixman_region32_fini(&damage);
}
}
struct wlr_scene_surface *wlr_scene_surface_create(struct wlr_scene_node *parent, struct wlr_scene_surface *wlr_scene_surface_create(struct wlr_scene_node *parent,
struct wlr_surface *surface) { struct wlr_surface *surface) {
struct wlr_scene_surface *scene_surface = struct wlr_scene_surface *scene_surface =
@ -122,6 +174,11 @@ struct wlr_scene_surface *wlr_scene_surface_create(struct wlr_scene_node *parent
scene_surface->surface_destroy.notify = scene_surface_handle_surface_destroy; scene_surface->surface_destroy.notify = scene_surface_handle_surface_destroy;
wl_signal_add(&surface->events.destroy, &scene_surface->surface_destroy); wl_signal_add(&surface->events.destroy, &scene_surface->surface_destroy);
scene_surface->surface_commit.notify = scene_surface_handle_surface_commit;
wl_signal_add(&surface->events.commit, &scene_surface->surface_commit);
scene_node_damage_whole(&scene_surface->node);
return scene_surface; return scene_surface;
} }
@ -138,41 +195,150 @@ struct wlr_scene_rect *wlr_scene_rect_create(struct wlr_scene_node *parent,
scene_rect->height = height; scene_rect->height = height;
memcpy(scene_rect->color, color, sizeof(scene_rect->color)); memcpy(scene_rect->color, color, sizeof(scene_rect->color));
scene_node_damage_whole(&scene_rect->node);
return scene_rect; return scene_rect;
} }
void wlr_scene_rect_set_size(struct wlr_scene_rect *rect, int width, int height) { void wlr_scene_rect_set_size(struct wlr_scene_rect *rect, int width, int height) {
if (rect->width == width && rect->height == height) {
return;
}
scene_node_damage_whole(&rect->node);
rect->width = width; rect->width = width;
rect->height = height; rect->height = height;
scene_node_damage_whole(&rect->node);
} }
void wlr_scene_rect_set_color(struct wlr_scene_rect *rect, const float color[static 4]) { void wlr_scene_rect_set_color(struct wlr_scene_rect *rect, const float color[static 4]) {
if (memcmp(rect->color, color, sizeof(rect->color)) == 0) {
return;
}
memcpy(rect->color, color, sizeof(rect->color)); memcpy(rect->color, color, sizeof(rect->color));
scene_node_damage_whole(&rect->node);
}
static int scale_length(int length, int offset, float scale) {
return round((offset + length) * scale) - round(offset * scale);
}
static void scale_box(struct wlr_box *box, float scale) {
box->width = scale_length(box->width, box->x, scale);
box->height = scale_length(box->height, box->y, scale);
box->x = round(box->x * scale);
box->y = round(box->y * scale);
}
static void _scene_node_damage_whole(struct wlr_scene_node *node,
struct wlr_scene *scene, int lx, int ly) {
if (!node->state.enabled) {
return;
}
struct wlr_scene_node *child;
wl_list_for_each(child, &node->state.children, state.link) {
_scene_node_damage_whole(child, scene,
lx + child->state.x, ly + child->state.y);
}
int width, height;
switch (node->type) {
case WLR_SCENE_NODE_ROOT:
return;
case WLR_SCENE_NODE_SURFACE:;
struct wlr_scene_surface *scene_surface =
wlr_scene_surface_from_node(node);
width = scene_surface->surface->current.width;
height = scene_surface->surface->current.height;
break;
case WLR_SCENE_NODE_RECT:;
struct wlr_scene_rect *scene_rect = scene_rect_from_node(node);
width = scene_rect->width;
height = scene_rect->height;
break;
}
struct wlr_scene_output *scene_output;
wl_list_for_each(scene_output, &scene->outputs, link) {
struct wlr_box box = {
.x = lx - scene_output->x,
.y = ly - scene_output->y,
.width = width,
.height = height,
};
scale_box(&box, scene_output->output->scale);
wlr_output_damage_add_box(scene_output->damage, &box);
}
}
static void scene_node_damage_whole(struct wlr_scene_node *node) {
struct wlr_scene *scene = scene_node_get_root(node);
if (wl_list_empty(&scene->outputs)) {
return;
}
int lx, ly;
if (!wlr_scene_node_coords(node, &lx, &ly)) {
return;
}
_scene_node_damage_whole(node, scene, lx, ly);
} }
void wlr_scene_node_set_enabled(struct wlr_scene_node *node, bool enabled) { void wlr_scene_node_set_enabled(struct wlr_scene_node *node, bool enabled) {
if (node->state.enabled == enabled) {
return;
}
// One of these damage_whole() calls will short-circuit and be a no-op
scene_node_damage_whole(node);
node->state.enabled = enabled; node->state.enabled = enabled;
scene_node_damage_whole(node);
} }
void wlr_scene_node_set_position(struct wlr_scene_node *node, int x, int y) { void wlr_scene_node_set_position(struct wlr_scene_node *node, int x, int y) {
if (node->state.x == x && node->state.y == y) {
return;
}
scene_node_damage_whole(node);
node->state.x = x; node->state.x = x;
node->state.y = y; node->state.y = y;
scene_node_damage_whole(node);
} }
void wlr_scene_node_place_above(struct wlr_scene_node *node, void wlr_scene_node_place_above(struct wlr_scene_node *node,
struct wlr_scene_node *sibling) { struct wlr_scene_node *sibling) {
assert(node->parent == sibling->parent); assert(node->parent == sibling->parent);
if (node->state.link.prev == &sibling->state.link) {
return;
}
wl_list_remove(&node->state.link); wl_list_remove(&node->state.link);
wl_list_insert(&sibling->state.link, &node->state.link); wl_list_insert(&sibling->state.link, &node->state.link);
scene_node_damage_whole(node);
scene_node_damage_whole(sibling);
} }
void wlr_scene_node_place_below(struct wlr_scene_node *node, void wlr_scene_node_place_below(struct wlr_scene_node *node,
struct wlr_scene_node *sibling) { struct wlr_scene_node *sibling) {
assert(node->parent == sibling->parent); assert(node->parent == sibling->parent);
if (node->state.link.next == &sibling->state.link) {
return;
}
wl_list_remove(&node->state.link); wl_list_remove(&node->state.link);
wl_list_insert(sibling->state.link.prev, &node->state.link); wl_list_insert(sibling->state.link.prev, &node->state.link);
scene_node_damage_whole(node);
scene_node_damage_whole(sibling);
} }
void wlr_scene_node_reparent(struct wlr_scene_node *node, void wlr_scene_node_reparent(struct wlr_scene_node *node,
@ -182,15 +348,20 @@ void wlr_scene_node_reparent(struct wlr_scene_node *node,
if (node->parent == new_parent) { if (node->parent == new_parent) {
return; return;
} }
/* Ensure that a node cannot become its own ancestor */ /* Ensure that a node cannot become its own ancestor */
for (struct wlr_scene_node *ancestor = new_parent; ancestor != NULL; for (struct wlr_scene_node *ancestor = new_parent; ancestor != NULL;
ancestor = ancestor->parent) { ancestor = ancestor->parent) {
assert(ancestor != node); assert(ancestor != node);
} }
scene_node_damage_whole(node);
wl_list_remove(&node->state.link); wl_list_remove(&node->state.link);
node->parent = new_parent; node->parent = new_parent;
wl_list_insert(new_parent->state.children.prev, &node->state.link); wl_list_insert(new_parent->state.children.prev, &node->state.link);
scene_node_damage_whole(node);
} }
bool wlr_scene_node_coords(struct wlr_scene_node *node, bool wlr_scene_node_coords(struct wlr_scene_node *node,
@ -286,17 +457,6 @@ struct wlr_scene_node *wlr_scene_node_at(struct wlr_scene_node *node,
return NULL; return NULL;
} }
static int scale_length(int length, int offset, float scale) {
return round((offset + length) * scale) - round(offset * scale);
}
static void scale_box(struct wlr_box *box, float scale) {
box->width = scale_length(box->width, box->x, scale);
box->height = scale_length(box->height, box->y, scale);
box->x = round(box->x * scale);
box->y = round(box->y * scale);
}
static void scissor_output(struct wlr_output *output, pixman_box32_t *rect) { static void scissor_output(struct wlr_output *output, pixman_box32_t *rect) {
struct wlr_renderer *renderer = wlr_backend_get_renderer(output->backend); struct wlr_renderer *renderer = wlr_backend_get_renderer(output->backend);
assert(renderer); assert(renderer);
@ -474,11 +634,19 @@ struct wlr_scene_output *wlr_scene_output_create(struct wlr_scene *scene,
return NULL; return NULL;
} }
scene_output->damage = wlr_output_damage_create(output);
if (scene_output->damage == NULL) {
free(scene_output);
return NULL;
}
scene_output->output = output; scene_output->output = output;
scene_output->scene = scene; scene_output->scene = scene;
wlr_addon_init(&scene_output->addon, &output->addons, scene, &output_addon_impl); wlr_addon_init(&scene_output->addon, &output->addons, scene, &output_addon_impl);
wl_list_insert(&scene->outputs, &scene_output->link); wl_list_insert(&scene->outputs, &scene_output->link);
wlr_output_damage_add_whole(scene_output->damage);
return scene_output; return scene_output;
} }
@ -490,30 +658,64 @@ void wlr_scene_output_destroy(struct wlr_scene_output *scene_output) {
void wlr_scene_output_set_position(struct wlr_scene_output *scene_output, void wlr_scene_output_set_position(struct wlr_scene_output *scene_output,
int lx, int ly) { int lx, int ly) {
if (scene_output->x == lx && scene_output->y == ly) {
return;
}
scene_output->x = lx; scene_output->x = lx;
scene_output->y = ly; scene_output->y = ly;
wlr_output_damage_add_whole(scene_output->damage);
} }
bool wlr_scene_output_commit(struct wlr_scene_output *scene_output) { bool wlr_scene_output_commit(struct wlr_scene_output *scene_output) {
struct wlr_output *output = scene_output->output; struct wlr_output *output = scene_output->output;
if (!wlr_output_attach_render(output, NULL)) {
return false;
}
struct wlr_renderer *renderer = wlr_backend_get_renderer(output->backend); struct wlr_renderer *renderer = wlr_backend_get_renderer(output->backend);
assert(renderer != NULL); assert(renderer != NULL);
int width, height; bool needs_frame;
wlr_output_effective_resolution(output, &width, &height); pixman_region32_t damage;
wlr_renderer_begin(renderer, width, height); pixman_region32_init(&damage);
wlr_renderer_clear(renderer, (float[4]){ 0.0, 0.0, 0.0, 0.0 }); if (!wlr_output_damage_attach_render(scene_output->damage,
&needs_frame, &damage)) {
pixman_region32_fini(&damage);
return false;
}
if (!needs_frame) {
pixman_region32_fini(&damage);
wlr_output_rollback(output);
return true;
}
wlr_renderer_begin(renderer, output->width, output->height);
int nrects;
pixman_box32_t *rects = pixman_region32_rectangles(&damage, &nrects);
for (int i = 0; i < nrects; ++i) {
scissor_output(output, &rects[i]);
wlr_renderer_clear(renderer, (float[4]){ 0.0, 0.0, 0.0, 1.0 });
}
wlr_scene_render_output(scene_output->scene, output, wlr_scene_render_output(scene_output->scene, output,
scene_output->x, scene_output->y, NULL); scene_output->x, scene_output->y, &damage);
wlr_output_render_software_cursors(output, NULL); wlr_output_render_software_cursors(output, &damage);
wlr_renderer_end(renderer); wlr_renderer_end(renderer);
pixman_region32_fini(&damage);
int tr_width, tr_height;
wlr_output_transformed_resolution(output, &tr_width, &tr_height);
enum wl_output_transform transform =
wlr_output_transform_invert(output->transform);
pixman_region32_t frame_damage;
pixman_region32_init(&frame_damage);
wlr_region_transform(&frame_damage, &scene_output->damage->current,
transform, tr_width, tr_height);
wlr_output_set_damage(output, &frame_damage);
pixman_region32_fini(&frame_damage);
return wlr_output_commit(output); return wlr_output_commit(output);
} }