mirror of
https://gitlab.freedesktop.org/wlroots/wlroots.git
synced 2026-04-18 06:47:31 -04:00
wlr_damage_ring: Introduce wlr_damage_ring_damage_for_buffer
This cannot be used along with `wlr_damage_ring_get_buffer_damage` in the same damage ring.
This commit is contained in:
parent
600d995e94
commit
a6676e81ed
2 changed files with 93 additions and 0 deletions
|
|
@ -16,10 +16,15 @@
|
|||
#include <wayland-server-core.h>
|
||||
|
||||
struct wlr_box;
|
||||
struct wlr_buffer;
|
||||
|
||||
struct wlr_damage_ring_entry {
|
||||
pixman_region32_t damage;
|
||||
struct wlr_buffer *buffer;
|
||||
|
||||
struct wl_listener buffer_destroy;
|
||||
|
||||
struct wlr_damage_ring *ring;
|
||||
struct wl_list link; // struct wlr_damage_ring.previous
|
||||
};
|
||||
|
||||
|
|
@ -84,4 +89,14 @@ void wlr_damage_ring_rotate(struct wlr_damage_ring *ring);
|
|||
void wlr_damage_ring_get_buffer_damage(struct wlr_damage_ring *ring,
|
||||
int buffer_age, pixman_region32_t *damage);
|
||||
|
||||
/**
|
||||
* Get the since accumulated damage for this buffer. These buffers should
|
||||
* typically come from a wlr_swapchain. In the context of rendering, the
|
||||
* damage is the region that needs to be redrawn.
|
||||
*
|
||||
* The damage ring is automatically rotated during invocation.
|
||||
*/
|
||||
void wlr_damage_ring_damage_for_buffer(struct wlr_damage_ring *ring,
|
||||
struct wlr_buffer *buffer, pixman_region32_t *damage);
|
||||
|
||||
#endif
|
||||
|
|
|
|||
|
|
@ -2,6 +2,7 @@
|
|||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <pixman.h>
|
||||
#include <wlr/types/wlr_buffer.h>
|
||||
#include <wlr/types/wlr_damage_ring.h>
|
||||
#include <wlr/util/box.h>
|
||||
|
||||
|
|
@ -18,6 +19,10 @@ void wlr_damage_ring_init(struct wlr_damage_ring *ring) {
|
|||
}
|
||||
|
||||
static void entry_destroy(struct wlr_damage_ring_entry *entry) {
|
||||
if (entry->buffer) {
|
||||
wl_list_remove(&entry->buffer_destroy.link);
|
||||
}
|
||||
|
||||
wl_list_remove(&entry->link);
|
||||
pixman_region32_fini(&entry->damage);
|
||||
free(entry);
|
||||
|
|
@ -132,3 +137,76 @@ void wlr_damage_ring_get_buffer_damage(struct wlr_damage_ring *ring,
|
|||
extents->y2 - extents->y1);
|
||||
}
|
||||
}
|
||||
|
||||
static void entry_squash_damage(struct wlr_damage_ring_entry *entry) {
|
||||
pixman_region32_t *prev;
|
||||
if (entry->link.prev == &entry->ring->previous) {
|
||||
// this entry is the first in the list
|
||||
prev = &entry->ring->current;
|
||||
} else {
|
||||
struct wlr_damage_ring_entry *last =
|
||||
wl_container_of(entry->link.prev, last, link);
|
||||
prev = &last->damage;
|
||||
}
|
||||
|
||||
pixman_region32_union(prev, prev, &entry->damage);
|
||||
}
|
||||
|
||||
static void handle_buffer_destroy(struct wl_listener *listener, void *data) {
|
||||
struct wlr_damage_ring_entry *entry =
|
||||
wl_container_of(listener, entry, buffer_destroy);
|
||||
entry_squash_damage(entry);
|
||||
entry_destroy(entry);
|
||||
}
|
||||
|
||||
void wlr_damage_ring_damage_for_buffer(struct wlr_damage_ring *ring,
|
||||
struct wlr_buffer *buffer, pixman_region32_t *damage) {
|
||||
pixman_region32_copy(damage, &ring->current);
|
||||
|
||||
struct wlr_damage_ring_entry *entry;
|
||||
wl_list_for_each(entry, &ring->previous, link) {
|
||||
if (entry->buffer != buffer) {
|
||||
pixman_region32_union(damage, damage, &entry->damage);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check the number of rectangles
|
||||
int n_rects = pixman_region32_n_rects(damage);
|
||||
if (n_rects > WLR_DAMAGE_RING_MAX_RECTS) {
|
||||
pixman_box32_t *extents = pixman_region32_extents(damage);
|
||||
pixman_region32_union_rect(damage, damage,
|
||||
extents->x1, extents->y1,
|
||||
extents->x2 - extents->x1,
|
||||
extents->y2 - extents->y1);
|
||||
}
|
||||
|
||||
// rotate
|
||||
entry_squash_damage(entry);
|
||||
pixman_region32_copy(&entry->damage, &ring->current);
|
||||
pixman_region32_clear(&ring->current);
|
||||
|
||||
wl_list_remove(&entry->link);
|
||||
wl_list_insert(&ring->previous, &entry->link);
|
||||
return;
|
||||
}
|
||||
|
||||
pixman_region32_clear(damage);
|
||||
pixman_region32_union_rect(damage, damage,
|
||||
0, 0, ring->width, ring->height);
|
||||
|
||||
entry = calloc(1, sizeof(*entry));
|
||||
if (!entry) {
|
||||
return;
|
||||
}
|
||||
|
||||
pixman_region32_init(&entry->damage);
|
||||
pixman_region32_copy(&entry->damage, &ring->current);
|
||||
pixman_region32_clear(&ring->current);
|
||||
|
||||
wl_list_insert(&ring->previous, &entry->link);
|
||||
entry->buffer = buffer;
|
||||
entry->ring = ring;
|
||||
|
||||
entry->buffer_destroy.notify = handle_buffer_destroy;
|
||||
wl_signal_add(&buffer->events.destroy, &entry->buffer_destroy);
|
||||
}
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue