From 35f7ed7e83491337b51a2f8ac2efaa764aa96515 Mon Sep 17 00:00:00 2001 From: Adrian Perez de Castro Date: Wed, 13 Mar 2024 16:36:15 +0200 Subject: [PATCH 1/6] doc: Document EGL exported images --- include/wpe/exported-image-egl.h | 42 +++++++++++++++++++++++++++++--- 1 file changed, 39 insertions(+), 3 deletions(-) diff --git a/include/wpe/exported-image-egl.h b/include/wpe/exported-image-egl.h index 9e71150..02a4aec 100644 --- a/include/wpe/exported-image-egl.h +++ b/include/wpe/exported-image-egl.h @@ -30,6 +30,18 @@ #ifndef __exported_image_egl_h__ #define __exported_image_egl_h__ +/** + * SECTION:egl_exported_image + * @short_description: EGL exported images. + * @include wpe/fdo-egl.h + * + * Represents an EGL exported image with some associated attributes. + * + * An `wpe_fdo_egl_exported_image` represents an `EGLImageKHR` object, + * which may be retrieved using wpe_fdo_egl_exported_image_get_egl_image(), + * and provides additional information about it. + */ + #include #ifdef __cplusplus @@ -40,14 +52,38 @@ typedef void* EGLImageKHR; struct wpe_fdo_egl_exported_image; +/** + * wpe_fdo_egl_exported_image_get_width: + * @image: (transfer none): An exported EGL image. + * + * Gets the width of an exported @image. + * + * Returns: Image width. + */ uint32_t -wpe_fdo_egl_exported_image_get_width(struct wpe_fdo_egl_exported_image*); +wpe_fdo_egl_exported_image_get_width(struct wpe_fdo_egl_exported_image *image); +/** + * wpe_fdo_egl_exported_image_get_height: + * @image: (transfer none): An exported EGL image. + * + * Gets the height of an exported @image. + * + * Returns: Image height. + */ uint32_t -wpe_fdo_egl_exported_image_get_height(struct wpe_fdo_egl_exported_image*); +wpe_fdo_egl_exported_image_get_height(struct wpe_fdo_egl_exported_image *image); +/** + * wpe_fdo_egl_exported_image_get_egl_image: + * @image: (transfer none): An exported EGL image. + * + * Gets the `EGLImage` for en exported @image. + * + * Returns: (transfer none): An `EGLImage` handle. + */ EGLImageKHR -wpe_fdo_egl_exported_image_get_egl_image(struct wpe_fdo_egl_exported_image*); +wpe_fdo_egl_exported_image_get_egl_image(struct wpe_fdo_egl_exported_image *image); #ifdef __cplusplus } From d37fc38eaaee3c4be826345d37ef6327841061e5 Mon Sep 17 00:00:00 2001 From: Lauro Moura Date: Sun, 2 Apr 2023 22:29:55 -0300 Subject: [PATCH 2/6] ws: Save and expose damage buffer information The wl_surface.damage_buffer request can be used by clients to inform that a given region (in buffer coordinates) is damaged. This request is used by eglSwapBuffersWithDamage[EXT|KHR], with one call for each rect. For more info: https://wayland-book.com/surfaces-in-depth/damaging-surfaces.html This commit stores this information and exposes it to exported EGL image clients. It also bumps the nested compositor interface to version 4, which provides the damage_buffer support. --- include/wpe/exported-image-egl.h | 3 ++ src/exported-image-egl.cpp | 7 +++++ src/ws-client.cpp | 2 +- src/ws.cpp | 50 ++++++++++++++++++++++++++++++-- src/ws.h | 5 ++++ 5 files changed, 64 insertions(+), 3 deletions(-) diff --git a/include/wpe/exported-image-egl.h b/include/wpe/exported-image-egl.h index 02a4aec..2069de8 100644 --- a/include/wpe/exported-image-egl.h +++ b/include/wpe/exported-image-egl.h @@ -85,6 +85,9 @@ wpe_fdo_egl_exported_image_get_height(struct wpe_fdo_egl_exported_image *image); EGLImageKHR wpe_fdo_egl_exported_image_get_egl_image(struct wpe_fdo_egl_exported_image *image); +uint32_t +wpe_fdo_egl_exported_image_get_damage_regions(struct wpe_fdo_egl_exported_image*, const int32_t**); + #ifdef __cplusplus } #endif diff --git a/src/exported-image-egl.cpp b/src/exported-image-egl.cpp index 651dd5a..b7bb018 100644 --- a/src/exported-image-egl.cpp +++ b/src/exported-image-egl.cpp @@ -52,4 +52,11 @@ wpe_fdo_egl_exported_image_get_egl_image(struct wpe_fdo_egl_exported_image* imag return image->eglImage; } +__attribute__((visibility("default"))) +uint32_t +wpe_fdo_egl_exported_image_get_damage_regions(struct wpe_fdo_egl_exported_image* image, const int32_t** target) +{ + return WS::Instance::singleton().exportDamageRegions(image->bufferResource, target); +} + } diff --git a/src/ws-client.cpp b/src/ws-client.cpp index ee1d875..8b888e5 100644 --- a/src/ws-client.cpp +++ b/src/ws-client.cpp @@ -260,7 +260,7 @@ const struct wl_registry_listener BaseTarget::s_registryListener = { auto& target = *reinterpret_cast(data); if (!std::strcmp(interface, "wl_compositor")) - target.m_wl.compositor = static_cast(wl_registry_bind(registry, name, &wl_compositor_interface, 1)); + target.m_wl.compositor = static_cast(wl_registry_bind(registry, name, &wl_compositor_interface, 4)); if (!std::strcmp(interface, "wpe_bridge")) target.m_wl.wpeBridge = static_cast(wl_registry_bind(registry, name, &wpe_bridge_interface, 1)); if (!std::strcmp(interface, "wpe_dmabuf_pool_manager")) diff --git a/src/ws.cpp b/src/ws.cpp index ef10e6a..8a11396 100644 --- a/src/ws.cpp +++ b/src/ws.cpp @@ -96,6 +96,7 @@ static const struct wl_surface_interface s_surfaceInterface = { [](struct wl_client*, struct wl_resource* surfaceResource, struct wl_resource* bufferResource, int32_t, int32_t) { auto& surface = *static_cast(wl_resource_get_user_data(surfaceResource)); + WS::Instance::singleton().clearPendingBufferDamage(surface.bufferResource); Instance::singleton().impl().surfaceAttach(surface, bufferResource); }, // damage @@ -128,7 +129,9 @@ static const struct wl_surface_interface s_surfaceInterface = { { auto& surface = *static_cast(wl_resource_get_user_data(surfaceResource)); surface.commit(); + auto* bufferResource = surface.bufferResource; WS::Instance::singleton().impl().surfaceCommit(surface); + WS::Instance::singleton().clearPendingBufferDamage(bufferResource); }, // set_buffer_transform [](struct wl_client*, struct wl_resource*, int32_t) { }, @@ -136,7 +139,10 @@ static const struct wl_surface_interface s_surfaceInterface = { [](struct wl_client*, struct wl_resource*, int32_t) { }, #if (WAYLAND_VERSION_MAJOR > 1) || (WAYLAND_VERSION_MAJOR == 1 && WAYLAND_VERSION_MINOR >= 10) // damage_buffer - [](struct wl_client*, struct wl_resource*, int32_t, int32_t, int32_t, int32_t) { }, + [](struct wl_client*, struct wl_resource* surfaceResource, int32_t x, int32_t y, int32_t width, int32_t height) { + auto& surface = *static_cast(wl_resource_get_user_data(surfaceResource)); + WS::Instance::singleton().addBufferDamageRegion(surface.bufferResource, x, y, width, height); + }, #endif }; @@ -416,7 +422,7 @@ Instance::Instance(std::unique_ptr&& impl) { m_impl->setInstance(*this); - m_compositor = wl_global_create(m_display, &wl_compositor_interface, 3, this, + m_compositor = wl_global_create(m_display, &wl_compositor_interface, 4, this, [](struct wl_client* client, void*, uint32_t version, uint32_t id) { struct wl_resource* resource = wl_resource_create(client, &wl_compositor_interface, version, id); @@ -673,4 +679,44 @@ bool Instance::dispatchFrameCallbacks(uint32_t bridgeId) return it->second->dispatchFrameCallbacks(); } +void Instance::addBufferDamageRegion(struct wl_resource *surfaceResource, int32_t x, int32_t y, int32_t width, int32_t height) +{ + if (!surfaceResource) + return; + + auto key = surfaceResource; + std::array region = {x, y, width, height}; + auto it = m_damageRegions.find(key); + + if (it == m_damageRegions.end()) { + std::vector> vec = {}; + m_damageRegions.insert(std::make_pair(key, vec)); + it = m_damageRegions.find(key); + } + + auto& vec = it->second; + vec.push_back(region); +} + +void Instance::clearPendingBufferDamage(struct wl_resource* bufferResource) +{ + m_damageRegions.erase(bufferResource); +} + +uint32_t Instance::exportDamageRegions(struct wl_resource* bufferResource, const int32_t** target) +{ + if (!bufferResource) { + *target = nullptr; + return 0; + } + + auto it = m_damageRegions.find(bufferResource); + if (it == m_damageRegions.cend() || it->second.empty() || !it->second.data()) { + *target = nullptr; + return 0; + } + *target = reinterpret_cast(it->second.data()); + return it->second.size(); +} + } // namespace WS diff --git a/src/ws.h b/src/ws.h index 29fe645..b133787 100644 --- a/src/ws.h +++ b/src/ws.h @@ -30,6 +30,7 @@ #include #include #include +#include #include struct linux_dmabuf_buffer; @@ -152,6 +153,9 @@ class Instance { void registerSurface(uint32_t, Surface*); void unregisterSurface(Surface*); + void addBufferDamageRegion(struct wl_resource*, int32_t x, int32_t y, int32_t width, int32_t height); + void clearPendingBufferDamage(struct wl_resource*); + uint32_t exportDamageRegions(struct wl_resource*, const int32_t** target); void registerViewBackend(uint32_t, APIClient&); void unregisterViewBackend(uint32_t); bool dispatchFrameCallbacks(uint32_t); @@ -188,6 +192,7 @@ class Instance { struct wl_global* m_wpeBridge { nullptr }; struct wl_global* m_wpeDmabufPoolManager { nullptr }; GSource* m_source { nullptr }; + std::unordered_map>> m_damageRegions; // (bridgeId -> Surface) std::unordered_map m_viewBackendMap; From 721ffd398d6a98b16b232e3367bb31dc8d79f3f3 Mon Sep 17 00:00:00 2001 From: Adrian Perez de Castro Date: Thu, 7 Mar 2024 16:48:52 +0200 Subject: [PATCH 3/6] Unconditionally use wl_surface_interface::damage_buffer Use the wl_surface_interface::damage_buffer member unconditionally, checking that the definition is available from Meson at configuration time. --- meson.build | 20 ++++++++++++++------ src/ws.cpp | 7 ++++--- 2 files changed, 18 insertions(+), 9 deletions(-) diff --git a/meson.build b/meson.build index fb06475..3545a80 100644 --- a/meson.build +++ b/meson.build @@ -255,14 +255,22 @@ add_project_arguments( ) wpe_dep = dependency('wpe-1.0', version: '>=1.6.0', fallback: ['libwpe', 'libwpe_dep']) +wl_server_dep = dependency('wayland-server') + +assert(cxx.has_header_symbol('wayland-server.h', 'WL_SURFACE_DAMAGE_BUFFER_SINCE_VERSION', dependencies: wl_server_dep), + 'Including does not define WL_SURFACE_DAMAGE_BUFFER_SINCE_VERSION') +assert(cxx.has_member('struct wl_surface_interface', 'damage_buffer', + prefix: '#include ', + dependencies: wl_server_dep), + 'struct member wl_surface_interface::damage_buffer is not defined') deps = [ - dependency('epoxy'), - dependency('gio-2.0'), - dependency('gobject-2.0'), - dependency('wayland-client'), - dependency('wayland-server'), - wpe_dep, + dependency('epoxy'), + dependency('gio-2.0'), + dependency('gobject-2.0'), + dependency('wayland-client'), + wl_server_dep, + wpe_dep, ] # Will be set to the library dependency which provides the wl_egl_* functions. diff --git a/src/ws.cpp b/src/ws.cpp index 8a11396..8885f22 100644 --- a/src/ws.cpp +++ b/src/ws.cpp @@ -137,15 +137,16 @@ static const struct wl_surface_interface s_surfaceInterface = { [](struct wl_client*, struct wl_resource*, int32_t) { }, // set_buffer_scale [](struct wl_client*, struct wl_resource*, int32_t) { }, -#if (WAYLAND_VERSION_MAJOR > 1) || (WAYLAND_VERSION_MAJOR == 1 && WAYLAND_VERSION_MINOR >= 10) // damage_buffer [](struct wl_client*, struct wl_resource* surfaceResource, int32_t x, int32_t y, int32_t width, int32_t height) { auto& surface = *static_cast(wl_resource_get_user_data(surfaceResource)); WS::Instance::singleton().addBufferDamageRegion(surface.bufferResource, x, y, width, height); }, -#endif }; +G_STATIC_ASSERT(WL_SURFACE_DAMAGE_BUFFER_SINCE_VERSION >= WL_SURFACE_SET_BUFFER_TRANSFORM_SINCE_VERSION); +G_STATIC_ASSERT(WL_SURFACE_DAMAGE_BUFFER_SINCE_VERSION >= WL_SURFACE_SET_BUFFER_SCALE_SINCE_VERSION); + static const struct wl_compositor_interface s_compositorInterface = { // create_surface [](struct wl_client* client, struct wl_resource* resource, uint32_t id) @@ -422,7 +423,7 @@ Instance::Instance(std::unique_ptr&& impl) { m_impl->setInstance(*this); - m_compositor = wl_global_create(m_display, &wl_compositor_interface, 4, this, + m_compositor = wl_global_create(m_display, &wl_compositor_interface, WL_SURFACE_DAMAGE_BUFFER_SINCE_VERSION, this, [](struct wl_client* client, void*, uint32_t version, uint32_t id) { struct wl_resource* resource = wl_resource_create(client, &wl_compositor_interface, version, id); From 62281cbd0b2b358017963d6510a461f670dcf9e9 Mon Sep 17 00:00:00 2001 From: Adrian Perez de Castro Date: Sat, 16 Mar 2024 23:17:59 +0200 Subject: [PATCH 4/6] ws: Warn on stub wl_surface_interface methods --- src/ws.cpp | 20 +++++++++++++++----- 1 file changed, 15 insertions(+), 5 deletions(-) diff --git a/src/ws.cpp b/src/ws.cpp index 8885f22..0102ff1 100644 --- a/src/ws.cpp +++ b/src/ws.cpp @@ -100,7 +100,9 @@ static const struct wl_surface_interface s_surfaceInterface = { Instance::singleton().impl().surfaceAttach(surface, bufferResource); }, // damage - [](struct wl_client*, struct wl_resource*, int32_t, int32_t, int32_t, int32_t) { }, + [](struct wl_client*, struct wl_resource*, int32_t, int32_t, int32_t, int32_t) { + g_warning_once("%s:%u: wl_surface_interface::set_buffer_transform is a no-op", __FILE__, __LINE__); + }, // frame [](struct wl_client* client, struct wl_resource* surfaceResource, uint32_t callback) { @@ -121,9 +123,13 @@ static const struct wl_surface_interface s_surfaceInterface = { surface.addFrameCallback(callbackResource); }, // set_opaque_region - [](struct wl_client*, struct wl_resource*, struct wl_resource*) { }, + [](struct wl_client*, struct wl_resource*, struct wl_resource*) { + g_warning_once("%s:%u: wl_surface_interface::set_buffer_transform is a no-op", __FILE__, __LINE__); + }, // set_input_region - [](struct wl_client*, struct wl_resource*, struct wl_resource*) { }, + [](struct wl_client*, struct wl_resource*, struct wl_resource*) { + g_warning_once("%s:%u: wl_surface_interface::set_buffer_transform is a no-op", __FILE__, __LINE__); + }, // commit [](struct wl_client*, struct wl_resource* surfaceResource) { @@ -134,9 +140,13 @@ static const struct wl_surface_interface s_surfaceInterface = { WS::Instance::singleton().clearPendingBufferDamage(bufferResource); }, // set_buffer_transform - [](struct wl_client*, struct wl_resource*, int32_t) { }, + [](struct wl_client*, struct wl_resource*, int32_t) { + g_warning_once("%s:%u: wl_surface_interface::set_buffer_transform is a no-op", __FILE__, __LINE__); + }, // set_buffer_scale - [](struct wl_client*, struct wl_resource*, int32_t) { }, + [](struct wl_client*, struct wl_resource*, int32_t) { + g_warning_once("%s:%u: wl_surface_interface::set_buffer_scale is a no-op", __FILE__, __LINE__); + }, // damage_buffer [](struct wl_client*, struct wl_resource* surfaceResource, int32_t x, int32_t y, int32_t width, int32_t height) { auto& surface = *static_cast(wl_resource_get_user_data(surfaceResource)); From 11b4e6dd9a9f3f54b4a93078580434edcd42679c Mon Sep 17 00:00:00 2001 From: Adrian Perez de Castro Date: Sat, 16 Mar 2024 23:20:40 +0200 Subject: [PATCH 5/6] ws: Do not leak damage list on surface destruction Clear the list of damage rectangles when a surface is destroyed, to avoid leaking them and slowly filling the damage map with unneeded, stale entries. --- src/ws.cpp | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/src/ws.cpp b/src/ws.cpp index 0102ff1..c839ffb 100644 --- a/src/ws.cpp +++ b/src/ws.cpp @@ -91,7 +91,10 @@ GSourceFuncs ServerSource::s_sourceFuncs = { static const struct wl_surface_interface s_surfaceInterface = { // destroy - [](struct wl_client*, struct wl_resource*) { }, + [](struct wl_client*, struct wl_resource *surfaceResource) { + auto& surface = *static_cast(wl_resource_get_user_data(surfaceResource)); + Instance::singleton().clearPendingBufferDamage(surface.bufferResource); + }, // attach [](struct wl_client*, struct wl_resource* surfaceResource, struct wl_resource* bufferResource, int32_t, int32_t) { From f236debc842c92d26925c1dcbaaa5dabd9023115 Mon Sep 17 00:00:00 2001 From: Adrian Perez de Castro Date: Sat, 16 Mar 2024 23:22:50 +0200 Subject: [PATCH 6/6] Use a helper wpe_fdo_rect type for damage regions Instead of a series of groups of four integer values, group each damage entry in a wpe_fdo_rect instance, which defines a rectangle explicitly documented to use the origin of coordinates at the top-left corner of exported images. This approach is more readable, uses the same amount of memory, allows defining the wpe_fdo_egl_exported_image_get_damage_regions() function in a more idiomatic way (returning pointer to rect array, and receiving a pointer where to store the number of entries), simplifies the code that returns the damage rectangle list, and also the code that stores a damage rectangle (also improved by using emplace+move). While at it, document wpe_fdo_egl_exported_image_get_damage_regions() and the new wpe_fdo_rect type. --- include/wpe/exported-image-egl.h | 61 ++++++++++++++++++++++++++++++-- src/exported-image-egl.cpp | 15 ++++++-- src/ws.cpp | 36 ++++++++----------- src/ws.h | 7 ++-- 4 files changed, 89 insertions(+), 30 deletions(-) diff --git a/include/wpe/exported-image-egl.h b/include/wpe/exported-image-egl.h index 2069de8..7324644 100644 --- a/include/wpe/exported-image-egl.h +++ b/include/wpe/exported-image-egl.h @@ -52,6 +52,22 @@ typedef void* EGLImageKHR; struct wpe_fdo_egl_exported_image; +/** + * wpe_fdo_rect: + * @x: Horizontal position. + * @y: Vertical position. + * @width: Rectange width. + * @height: Rectangle height. + * + * Describes the position and size of a rectangle. + * + * This is used to report damaged regions for exported images. + * The origin of coordinates is at the top-left corner of the image. + */ +struct wpe_fdo_rect { + uint32_t x, y, width, height; +}; + /** * wpe_fdo_egl_exported_image_get_width: * @image: (transfer none): An exported EGL image. @@ -85,8 +101,49 @@ wpe_fdo_egl_exported_image_get_height(struct wpe_fdo_egl_exported_image *image); EGLImageKHR wpe_fdo_egl_exported_image_get_egl_image(struct wpe_fdo_egl_exported_image *image); -uint32_t -wpe_fdo_egl_exported_image_get_damage_regions(struct wpe_fdo_egl_exported_image*, const int32_t**); +/** + * wpe_fdo_egl_exported_image_get_damage_regions: + * @image: (transfer none): An exported EGL image. + * @n_rectangles: (transfer none) (out) (not nullable): Location where to store the number of rectangles. + * + * Gets rectangles describing the damage regions for an @image. + * + * Images may indicate which parts have changed since the previous exported + * image. If the set of damaged regions cannot be determined, then `NULL` is + * returned and @n_regions set to zero. This may be the case if: + * + * - The image contains many changes compared to the previous exported image + * and it is considered that it is not worth it using damage regions. + * - Some component in the graphics stack is not able to handle damage + * regions. + * + * When the set of regions is empty, the whole image **must** be considered + * to be changed. + * + * **Important:** Damage regions are only meaningful when an image is used + * to be displayed right after the one exported immediately before it! In + * particular, if an exported image is kept around to be displayed later, + * and not at the moment when it has been exported, damage information *must + * not* be used. + * + * Note that the layout of the `wpe_fdo_rect` type is designed in such a way + * that the result from this function may be passed directly to EGL for + * swapping buffers: + * + * ```c + * struct wpe_fdo_egl_exported_image *image = get_image(); + * + * uint32_t n_regions; + * const struct wpe_fdo_rect *regions = + * wpe_fdo_egl_exported_image_get_damage_regions(image, &n_regions); + * + * eglSwapBuffersWithDamageEXT(eglDisplay, eglSurface, (EGLint*) regions, n_regions); + * ``` + * + * Returns: (transfer none) (nullable): Pointer to an array of regions. + */ +const struct wpe_fdo_rect* +wpe_fdo_egl_exported_image_get_damage_regions(struct wpe_fdo_egl_exported_image*, uint32_t *n_rectangles); #ifdef __cplusplus } diff --git a/src/exported-image-egl.cpp b/src/exported-image-egl.cpp index b7bb018..bdafab5 100644 --- a/src/exported-image-egl.cpp +++ b/src/exported-image-egl.cpp @@ -53,10 +53,19 @@ wpe_fdo_egl_exported_image_get_egl_image(struct wpe_fdo_egl_exported_image* imag } __attribute__((visibility("default"))) -uint32_t -wpe_fdo_egl_exported_image_get_damage_regions(struct wpe_fdo_egl_exported_image* image, const int32_t** target) +const struct wpe_fdo_rect* +wpe_fdo_egl_exported_image_get_damage_regions(struct wpe_fdo_egl_exported_image *image, uint32_t *n_rectangles) { - return WS::Instance::singleton().exportDamageRegions(image->bufferResource, target); + g_return_val_if_fail(n_rectangles, nullptr); + + const auto* regions = WS::Instance::singleton().getDamageRegions(image->bufferResource); + if (regions && regions->size()) { + *n_rectangles = regions->size(); + return regions->data(); + } else { + *n_rectangles = 0; + return nullptr; + } } } diff --git a/src/ws.cpp b/src/ws.cpp index c839ffb..74942c4 100644 --- a/src/ws.cpp +++ b/src/ws.cpp @@ -693,23 +693,19 @@ bool Instance::dispatchFrameCallbacks(uint32_t bridgeId) return it->second->dispatchFrameCallbacks(); } -void Instance::addBufferDamageRegion(struct wl_resource *surfaceResource, int32_t x, int32_t y, int32_t width, int32_t height) +void Instance::addBufferDamageRegion(struct wl_resource *surfaceResource, uint32_t x, uint32_t y, uint32_t width, uint32_t height) { if (!surfaceResource) return; - auto key = surfaceResource; - std::array region = {x, y, width, height}; - auto it = m_damageRegions.find(key); - + wpe_fdo_rect rect {x, y, width, height}; + auto it = m_damageRegions.find(surfaceResource); if (it == m_damageRegions.end()) { - std::vector> vec = {}; - m_damageRegions.insert(std::make_pair(key, vec)); - it = m_damageRegions.find(key); + std::vector regions = { std::move(rect) }; + m_damageRegions.emplace(surfaceResource, std::move(regions)); + } else { + it->second.emplace_back(std::move(rect)); } - - auto& vec = it->second; - vec.push_back(region); } void Instance::clearPendingBufferDamage(struct wl_resource* bufferResource) @@ -717,20 +713,16 @@ void Instance::clearPendingBufferDamage(struct wl_resource* bufferResource) m_damageRegions.erase(bufferResource); } -uint32_t Instance::exportDamageRegions(struct wl_resource* bufferResource, const int32_t** target) +const std::vector* Instance::getDamageRegions(struct wl_resource* bufferResource) { - if (!bufferResource) { - *target = nullptr; - return 0; - } + if (!bufferResource) + return nullptr; auto it = m_damageRegions.find(bufferResource); - if (it == m_damageRegions.cend() || it->second.empty() || !it->second.data()) { - *target = nullptr; - return 0; - } - *target = reinterpret_cast(it->second.data()); - return it->second.size(); + if (it == m_damageRegions.cend()) + return nullptr; + + return &it->second; } } // namespace WS diff --git a/src/ws.h b/src/ws.h index b133787..f454955 100644 --- a/src/ws.h +++ b/src/ws.h @@ -25,6 +25,7 @@ #pragma once +#include "../include/wpe/exported-image-egl.h" #include "ws-types.h" #include #include @@ -153,9 +154,9 @@ class Instance { void registerSurface(uint32_t, Surface*); void unregisterSurface(Surface*); - void addBufferDamageRegion(struct wl_resource*, int32_t x, int32_t y, int32_t width, int32_t height); + void addBufferDamageRegion(struct wl_resource*, uint32_t x, uint32_t y, uint32_t width, uint32_t height); void clearPendingBufferDamage(struct wl_resource*); - uint32_t exportDamageRegions(struct wl_resource*, const int32_t** target); + const std::vector* getDamageRegions(struct wl_resource* bufferResource); void registerViewBackend(uint32_t, APIClient&); void unregisterViewBackend(uint32_t); bool dispatchFrameCallbacks(uint32_t); @@ -192,7 +193,7 @@ class Instance { struct wl_global* m_wpeBridge { nullptr }; struct wl_global* m_wpeDmabufPoolManager { nullptr }; GSource* m_source { nullptr }; - std::unordered_map>> m_damageRegions; + std::unordered_map> m_damageRegions; // (bridgeId -> Surface) std::unordered_map m_viewBackendMap;