mirror of
https://gitlab.freedesktop.org/wlroots/wlroots.git
synced 2026-04-14 08:22:25 -04:00
Merge branch 'remove-screencopy-v1' into 'master'
types: remove support for wlr-screencopy-unstable-v1 See merge request wlroots/wlroots!5149
This commit is contained in:
commit
d50b4ef6e7
5 changed files with 0 additions and 1023 deletions
|
|
@ -1,67 +0,0 @@
|
||||||
/*
|
|
||||||
* This an unstable interface of wlroots. No guarantees are made regarding the
|
|
||||||
* future consistency of this API.
|
|
||||||
*/
|
|
||||||
#ifndef WLR_USE_UNSTABLE
|
|
||||||
#error "Add -DWLR_USE_UNSTABLE to enable unstable wlroots features"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef WLR_TYPES_WLR_SCREENCOPY_V1_H
|
|
||||||
#define WLR_TYPES_WLR_SCREENCOPY_V1_H
|
|
||||||
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include <wayland-server-core.h>
|
|
||||||
#include <wlr/types/wlr_buffer.h>
|
|
||||||
#include <wlr/util/box.h>
|
|
||||||
|
|
||||||
struct wlr_screencopy_manager_v1 {
|
|
||||||
struct wl_global *global;
|
|
||||||
struct wl_list frames; // wlr_screencopy_frame_v1.link
|
|
||||||
|
|
||||||
struct {
|
|
||||||
struct wl_signal destroy;
|
|
||||||
} events;
|
|
||||||
|
|
||||||
void *data;
|
|
||||||
|
|
||||||
struct {
|
|
||||||
struct wl_listener display_destroy;
|
|
||||||
} WLR_PRIVATE;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct wlr_screencopy_v1_client {
|
|
||||||
int ref;
|
|
||||||
struct wlr_screencopy_manager_v1 *manager;
|
|
||||||
struct wl_list damages;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct wlr_screencopy_frame_v1 {
|
|
||||||
struct wl_resource *resource;
|
|
||||||
struct wlr_screencopy_v1_client *client;
|
|
||||||
struct wl_list link; // wlr_screencopy_manager_v1.frames
|
|
||||||
|
|
||||||
uint32_t shm_format, dmabuf_format; // DRM format codes
|
|
||||||
struct wlr_box box;
|
|
||||||
int shm_stride;
|
|
||||||
|
|
||||||
bool overlay_cursor, cursor_locked;
|
|
||||||
|
|
||||||
bool with_damage;
|
|
||||||
|
|
||||||
enum wlr_buffer_cap buffer_cap;
|
|
||||||
struct wlr_buffer *buffer;
|
|
||||||
|
|
||||||
struct wlr_output *output;
|
|
||||||
|
|
||||||
void *data;
|
|
||||||
|
|
||||||
struct {
|
|
||||||
struct wl_listener output_commit;
|
|
||||||
struct wl_listener output_destroy;
|
|
||||||
} WLR_PRIVATE;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct wlr_screencopy_manager_v1 *wlr_screencopy_manager_v1_create(
|
|
||||||
struct wl_display *display);
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
@ -74,7 +74,6 @@ protocols = {
|
||||||
'wlr-layer-shell-unstable-v1': 'wlr-layer-shell-unstable-v1.xml',
|
'wlr-layer-shell-unstable-v1': 'wlr-layer-shell-unstable-v1.xml',
|
||||||
'wlr-output-management-unstable-v1': 'wlr-output-management-unstable-v1.xml',
|
'wlr-output-management-unstable-v1': 'wlr-output-management-unstable-v1.xml',
|
||||||
'wlr-output-power-management-unstable-v1': 'wlr-output-power-management-unstable-v1.xml',
|
'wlr-output-power-management-unstable-v1': 'wlr-output-power-management-unstable-v1.xml',
|
||||||
'wlr-screencopy-unstable-v1': 'wlr-screencopy-unstable-v1.xml',
|
|
||||||
'wlr-virtual-pointer-unstable-v1': 'wlr-virtual-pointer-unstable-v1.xml',
|
'wlr-virtual-pointer-unstable-v1': 'wlr-virtual-pointer-unstable-v1.xml',
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,232 +0,0 @@
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<protocol name="wlr_screencopy_unstable_v1">
|
|
||||||
<copyright>
|
|
||||||
Copyright © 2018 Simon Ser
|
|
||||||
Copyright © 2019 Andri Yngvason
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a
|
|
||||||
copy of this software and associated documentation files (the "Software"),
|
|
||||||
to deal in the Software without restriction, including without limitation
|
|
||||||
the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
||||||
and/or sell copies of the Software, and to permit persons to whom the
|
|
||||||
Software is furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice (including the next
|
|
||||||
paragraph) shall be included in all copies or substantial portions of the
|
|
||||||
Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
||||||
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
||||||
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
||||||
DEALINGS IN THE SOFTWARE.
|
|
||||||
</copyright>
|
|
||||||
|
|
||||||
<description summary="screen content capturing on client buffers">
|
|
||||||
This protocol allows clients to ask the compositor to copy part of the
|
|
||||||
screen content to a client buffer.
|
|
||||||
|
|
||||||
Warning! The protocol described in this file is experimental and
|
|
||||||
backward incompatible changes may be made. Backward compatible changes
|
|
||||||
may be added together with the corresponding interface version bump.
|
|
||||||
Backward incompatible changes are done by bumping the version number in
|
|
||||||
the protocol and interface names and resetting the interface version.
|
|
||||||
Once the protocol is to be declared stable, the 'z' prefix and the
|
|
||||||
version number in the protocol and interface names are removed and the
|
|
||||||
interface version number is reset.
|
|
||||||
</description>
|
|
||||||
|
|
||||||
<interface name="zwlr_screencopy_manager_v1" version="3">
|
|
||||||
<description summary="manager to inform clients and begin capturing">
|
|
||||||
This object is a manager which offers requests to start capturing from a
|
|
||||||
source.
|
|
||||||
</description>
|
|
||||||
|
|
||||||
<request name="capture_output">
|
|
||||||
<description summary="capture an output">
|
|
||||||
Capture the next frame of an entire output.
|
|
||||||
</description>
|
|
||||||
<arg name="frame" type="new_id" interface="zwlr_screencopy_frame_v1"/>
|
|
||||||
<arg name="overlay_cursor" type="int"
|
|
||||||
summary="composite cursor onto the frame"/>
|
|
||||||
<arg name="output" type="object" interface="wl_output"/>
|
|
||||||
</request>
|
|
||||||
|
|
||||||
<request name="capture_output_region">
|
|
||||||
<description summary="capture an output's region">
|
|
||||||
Capture the next frame of an output's region.
|
|
||||||
|
|
||||||
The region is given in output logical coordinates, see
|
|
||||||
xdg_output.logical_size. The region will be clipped to the output's
|
|
||||||
extents.
|
|
||||||
</description>
|
|
||||||
<arg name="frame" type="new_id" interface="zwlr_screencopy_frame_v1"/>
|
|
||||||
<arg name="overlay_cursor" type="int"
|
|
||||||
summary="composite cursor onto the frame"/>
|
|
||||||
<arg name="output" type="object" interface="wl_output"/>
|
|
||||||
<arg name="x" type="int"/>
|
|
||||||
<arg name="y" type="int"/>
|
|
||||||
<arg name="width" type="int"/>
|
|
||||||
<arg name="height" type="int"/>
|
|
||||||
</request>
|
|
||||||
|
|
||||||
<request name="destroy" type="destructor">
|
|
||||||
<description summary="destroy the manager">
|
|
||||||
All objects created by the manager will still remain valid, until their
|
|
||||||
appropriate destroy request has been called.
|
|
||||||
</description>
|
|
||||||
</request>
|
|
||||||
</interface>
|
|
||||||
|
|
||||||
<interface name="zwlr_screencopy_frame_v1" version="3">
|
|
||||||
<description summary="a frame ready for copy">
|
|
||||||
This object represents a single frame.
|
|
||||||
|
|
||||||
When created, a series of buffer events will be sent, each representing a
|
|
||||||
supported buffer type. The "buffer_done" event is sent afterwards to
|
|
||||||
indicate that all supported buffer types have been enumerated. The client
|
|
||||||
will then be able to send a "copy" request. If the capture is successful,
|
|
||||||
the compositor will send a "flags" followed by a "ready" event.
|
|
||||||
|
|
||||||
For objects version 2 or lower, wl_shm buffers are always supported, ie.
|
|
||||||
the "buffer" event is guaranteed to be sent.
|
|
||||||
|
|
||||||
If the capture failed, the "failed" event is sent. This can happen anytime
|
|
||||||
before the "ready" event.
|
|
||||||
|
|
||||||
Once either a "ready" or a "failed" event is received, the client should
|
|
||||||
destroy the frame.
|
|
||||||
</description>
|
|
||||||
|
|
||||||
<event name="buffer">
|
|
||||||
<description summary="wl_shm buffer information">
|
|
||||||
Provides information about wl_shm buffer parameters that need to be
|
|
||||||
used for this frame. This event is sent once after the frame is created
|
|
||||||
if wl_shm buffers are supported.
|
|
||||||
</description>
|
|
||||||
<arg name="format" type="uint" enum="wl_shm.format" summary="buffer format"/>
|
|
||||||
<arg name="width" type="uint" summary="buffer width"/>
|
|
||||||
<arg name="height" type="uint" summary="buffer height"/>
|
|
||||||
<arg name="stride" type="uint" summary="buffer stride"/>
|
|
||||||
</event>
|
|
||||||
|
|
||||||
<request name="copy">
|
|
||||||
<description summary="copy the frame">
|
|
||||||
Copy the frame to the supplied buffer. The buffer must have a the
|
|
||||||
correct size, see zwlr_screencopy_frame_v1.buffer and
|
|
||||||
zwlr_screencopy_frame_v1.linux_dmabuf. The buffer needs to have a
|
|
||||||
supported format.
|
|
||||||
|
|
||||||
If the frame is successfully copied, a "flags" and a "ready" events are
|
|
||||||
sent. Otherwise, a "failed" event is sent.
|
|
||||||
</description>
|
|
||||||
<arg name="buffer" type="object" interface="wl_buffer"/>
|
|
||||||
</request>
|
|
||||||
|
|
||||||
<enum name="error">
|
|
||||||
<entry name="already_used" value="0"
|
|
||||||
summary="the object has already been used to copy a wl_buffer"/>
|
|
||||||
<entry name="invalid_buffer" value="1"
|
|
||||||
summary="buffer attributes are invalid"/>
|
|
||||||
</enum>
|
|
||||||
|
|
||||||
<enum name="flags" bitfield="true">
|
|
||||||
<entry name="y_invert" value="1" summary="contents are y-inverted"/>
|
|
||||||
</enum>
|
|
||||||
|
|
||||||
<event name="flags">
|
|
||||||
<description summary="frame flags">
|
|
||||||
Provides flags about the frame. This event is sent once before the
|
|
||||||
"ready" event.
|
|
||||||
</description>
|
|
||||||
<arg name="flags" type="uint" enum="flags" summary="frame flags"/>
|
|
||||||
</event>
|
|
||||||
|
|
||||||
<event name="ready">
|
|
||||||
<description summary="indicates frame is available for reading">
|
|
||||||
Called as soon as the frame is copied, indicating it is available
|
|
||||||
for reading. This event includes the time at which presentation happened
|
|
||||||
at.
|
|
||||||
|
|
||||||
The timestamp is expressed as tv_sec_hi, tv_sec_lo, tv_nsec triples,
|
|
||||||
each component being an unsigned 32-bit value. Whole seconds are in
|
|
||||||
tv_sec which is a 64-bit value combined from tv_sec_hi and tv_sec_lo,
|
|
||||||
and the additional fractional part in tv_nsec as nanoseconds. Hence,
|
|
||||||
for valid timestamps tv_nsec must be in [0, 999999999]. The seconds part
|
|
||||||
may have an arbitrary offset at start.
|
|
||||||
|
|
||||||
After receiving this event, the client should destroy the object.
|
|
||||||
</description>
|
|
||||||
<arg name="tv_sec_hi" type="uint"
|
|
||||||
summary="high 32 bits of the seconds part of the timestamp"/>
|
|
||||||
<arg name="tv_sec_lo" type="uint"
|
|
||||||
summary="low 32 bits of the seconds part of the timestamp"/>
|
|
||||||
<arg name="tv_nsec" type="uint"
|
|
||||||
summary="nanoseconds part of the timestamp"/>
|
|
||||||
</event>
|
|
||||||
|
|
||||||
<event name="failed">
|
|
||||||
<description summary="frame copy failed">
|
|
||||||
This event indicates that the attempted frame copy has failed.
|
|
||||||
|
|
||||||
After receiving this event, the client should destroy the object.
|
|
||||||
</description>
|
|
||||||
</event>
|
|
||||||
|
|
||||||
<request name="destroy" type="destructor">
|
|
||||||
<description summary="delete this object, used or not">
|
|
||||||
Destroys the frame. This request can be sent at any time by the client.
|
|
||||||
</description>
|
|
||||||
</request>
|
|
||||||
|
|
||||||
<!-- Version 2 additions -->
|
|
||||||
<request name="copy_with_damage" since="2">
|
|
||||||
<description summary="copy the frame when it's damaged">
|
|
||||||
Same as copy, except it waits until there is damage to copy.
|
|
||||||
</description>
|
|
||||||
<arg name="buffer" type="object" interface="wl_buffer"/>
|
|
||||||
</request>
|
|
||||||
|
|
||||||
<event name="damage" since="2">
|
|
||||||
<description summary="carries the coordinates of the damaged region">
|
|
||||||
This event is sent right before the ready event when copy_with_damage is
|
|
||||||
requested. It may be generated multiple times for each copy_with_damage
|
|
||||||
request.
|
|
||||||
|
|
||||||
The arguments describe a box around an area that has changed since the
|
|
||||||
last copy request that was derived from the current screencopy manager
|
|
||||||
instance.
|
|
||||||
|
|
||||||
The union of all regions received between the call to copy_with_damage
|
|
||||||
and a ready event is the total damage since the prior ready event.
|
|
||||||
</description>
|
|
||||||
<arg name="x" type="uint" summary="damaged x coordinates"/>
|
|
||||||
<arg name="y" type="uint" summary="damaged y coordinates"/>
|
|
||||||
<arg name="width" type="uint" summary="current width"/>
|
|
||||||
<arg name="height" type="uint" summary="current height"/>
|
|
||||||
</event>
|
|
||||||
|
|
||||||
<!-- Version 3 additions -->
|
|
||||||
<event name="linux_dmabuf" since="3">
|
|
||||||
<description summary="linux-dmabuf buffer information">
|
|
||||||
Provides information about linux-dmabuf buffer parameters that need to
|
|
||||||
be used for this frame. This event is sent once after the frame is
|
|
||||||
created if linux-dmabuf buffers are supported.
|
|
||||||
</description>
|
|
||||||
<arg name="format" type="uint" summary="fourcc pixel format"/>
|
|
||||||
<arg name="width" type="uint" summary="buffer width"/>
|
|
||||||
<arg name="height" type="uint" summary="buffer height"/>
|
|
||||||
</event>
|
|
||||||
|
|
||||||
<event name="buffer_done" since="3">
|
|
||||||
<description summary="all buffer types reported">
|
|
||||||
This event is sent once after all buffer events have been sent.
|
|
||||||
|
|
||||||
The client should proceed to create a buffer of one of the supported
|
|
||||||
types, and send a "copy" request.
|
|
||||||
</description>
|
|
||||||
</event>
|
|
||||||
</interface>
|
|
||||||
</protocol>
|
|
||||||
|
|
@ -78,7 +78,6 @@ wlr_files += files(
|
||||||
'wlr_primary_selection.c',
|
'wlr_primary_selection.c',
|
||||||
'wlr_region.c',
|
'wlr_region.c',
|
||||||
'wlr_relative_pointer_v1.c',
|
'wlr_relative_pointer_v1.c',
|
||||||
'wlr_screencopy_v1.c',
|
|
||||||
'wlr_security_context_v1.c',
|
'wlr_security_context_v1.c',
|
||||||
'wlr_server_decoration.c',
|
'wlr_server_decoration.c',
|
||||||
'wlr_session_lock_v1.c',
|
'wlr_session_lock_v1.c',
|
||||||
|
|
|
||||||
|
|
@ -1,722 +0,0 @@
|
||||||
#include <assert.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <drm_fourcc.h>
|
|
||||||
#include <wlr/interfaces/wlr_output.h>
|
|
||||||
#include <wlr/render/allocator.h>
|
|
||||||
#include <wlr/render/swapchain.h>
|
|
||||||
#include <wlr/render/wlr_renderer.h>
|
|
||||||
#include <wlr/types/wlr_screencopy_v1.h>
|
|
||||||
#include <wlr/backend.h>
|
|
||||||
#include <wlr/util/box.h>
|
|
||||||
#include <wlr/util/log.h>
|
|
||||||
#include <wlr/util/transform.h>
|
|
||||||
#include "wlr-screencopy-unstable-v1-protocol.h"
|
|
||||||
#include "render/pixel_format.h"
|
|
||||||
#include "render/wlr_renderer.h"
|
|
||||||
|
|
||||||
#define SCREENCOPY_MANAGER_VERSION 3
|
|
||||||
|
|
||||||
struct screencopy_damage {
|
|
||||||
struct wl_list link;
|
|
||||||
struct wlr_output *output;
|
|
||||||
struct pixman_region32 damage;
|
|
||||||
struct wl_listener output_precommit;
|
|
||||||
struct wl_listener output_destroy;
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct zwlr_screencopy_frame_v1_interface frame_impl;
|
|
||||||
|
|
||||||
static struct screencopy_damage *screencopy_damage_find(
|
|
||||||
struct wlr_screencopy_v1_client *client,
|
|
||||||
struct wlr_output *output) {
|
|
||||||
struct screencopy_damage *damage;
|
|
||||||
|
|
||||||
wl_list_for_each(damage, &client->damages, link) {
|
|
||||||
if (damage->output == output) {
|
|
||||||
return damage;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void screencopy_damage_accumulate(struct screencopy_damage *damage,
|
|
||||||
const struct wlr_output_state *state) {
|
|
||||||
struct pixman_region32 *region = &damage->damage;
|
|
||||||
struct wlr_output *output = damage->output;
|
|
||||||
|
|
||||||
if (state->committed & WLR_OUTPUT_STATE_DAMAGE) {
|
|
||||||
// If the compositor submitted damage, copy it over
|
|
||||||
pixman_region32_union(region, region, &state->damage);
|
|
||||||
pixman_region32_intersect_rect(region, region, 0, 0,
|
|
||||||
output->width, output->height);
|
|
||||||
} else if (state->committed & WLR_OUTPUT_STATE_BUFFER) {
|
|
||||||
// If the compositor did not submit damage but did submit a buffer
|
|
||||||
// damage everything
|
|
||||||
pixman_region32_union_rect(region, region, 0, 0,
|
|
||||||
output->width, output->height);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void screencopy_damage_handle_output_precommit(
|
|
||||||
struct wl_listener *listener, void *data) {
|
|
||||||
struct screencopy_damage *damage =
|
|
||||||
wl_container_of(listener, damage, output_precommit);
|
|
||||||
const struct wlr_output_event_precommit *event = data;
|
|
||||||
screencopy_damage_accumulate(damage, event->state);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void screencopy_damage_destroy(struct screencopy_damage *damage) {
|
|
||||||
wl_list_remove(&damage->output_destroy.link);
|
|
||||||
wl_list_remove(&damage->output_precommit.link);
|
|
||||||
wl_list_remove(&damage->link);
|
|
||||||
pixman_region32_fini(&damage->damage);
|
|
||||||
free(damage);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void screencopy_damage_handle_output_destroy(
|
|
||||||
struct wl_listener *listener, void *data) {
|
|
||||||
struct screencopy_damage *damage =
|
|
||||||
wl_container_of(listener, damage, output_destroy);
|
|
||||||
screencopy_damage_destroy(damage);
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct screencopy_damage *screencopy_damage_create(
|
|
||||||
struct wlr_screencopy_v1_client *client,
|
|
||||||
struct wlr_output *output) {
|
|
||||||
struct screencopy_damage *damage = calloc(1, sizeof(*damage));
|
|
||||||
if (!damage) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
damage->output = output;
|
|
||||||
pixman_region32_init_rect(&damage->damage, 0, 0, output->width,
|
|
||||||
output->height);
|
|
||||||
wl_list_insert(&client->damages, &damage->link);
|
|
||||||
|
|
||||||
wl_signal_add(&output->events.precommit, &damage->output_precommit);
|
|
||||||
damage->output_precommit.notify =
|
|
||||||
screencopy_damage_handle_output_precommit;
|
|
||||||
|
|
||||||
wl_signal_add(&output->events.destroy, &damage->output_destroy);
|
|
||||||
damage->output_destroy.notify = screencopy_damage_handle_output_destroy;
|
|
||||||
|
|
||||||
return damage;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct screencopy_damage *screencopy_damage_get_or_create(
|
|
||||||
struct wlr_screencopy_v1_client *client,
|
|
||||||
struct wlr_output *output) {
|
|
||||||
struct screencopy_damage *damage = screencopy_damage_find(client, output);
|
|
||||||
return damage ? damage : screencopy_damage_create(client, output);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void client_unref(struct wlr_screencopy_v1_client *client) {
|
|
||||||
assert(client->ref > 0);
|
|
||||||
|
|
||||||
if (--client->ref != 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct screencopy_damage *damage, *tmp_damage;
|
|
||||||
wl_list_for_each_safe(damage, tmp_damage, &client->damages, link) {
|
|
||||||
screencopy_damage_destroy(damage);
|
|
||||||
}
|
|
||||||
|
|
||||||
free(client);
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct wlr_screencopy_frame_v1 *frame_from_resource(
|
|
||||||
struct wl_resource *resource) {
|
|
||||||
assert(wl_resource_instance_of(resource,
|
|
||||||
&zwlr_screencopy_frame_v1_interface, &frame_impl));
|
|
||||||
return wl_resource_get_user_data(resource);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void frame_destroy(struct wlr_screencopy_frame_v1 *frame) {
|
|
||||||
if (frame == NULL) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (frame->output != NULL && frame->buffer != NULL) {
|
|
||||||
wlr_output_lock_attach_render(frame->output, false);
|
|
||||||
if (frame->cursor_locked) {
|
|
||||||
wlr_output_lock_software_cursors(frame->output, false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
wl_list_remove(&frame->link);
|
|
||||||
wl_list_remove(&frame->output_commit.link);
|
|
||||||
wl_list_remove(&frame->output_destroy.link);
|
|
||||||
// Make the frame resource inert
|
|
||||||
wl_resource_set_user_data(frame->resource, NULL);
|
|
||||||
wlr_buffer_unlock(frame->buffer);
|
|
||||||
client_unref(frame->client);
|
|
||||||
free(frame);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void frame_send_damage(struct wlr_screencopy_frame_v1 *frame) {
|
|
||||||
if (!frame->with_damage) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct screencopy_damage *damage =
|
|
||||||
screencopy_damage_get_or_create(frame->client, frame->output);
|
|
||||||
if (damage == NULL) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
int n_boxes;
|
|
||||||
const pixman_box32_t *boxes = pixman_region32_rectangles(&damage->damage, &n_boxes);
|
|
||||||
for (int i = 0; i < n_boxes; i++) {
|
|
||||||
const pixman_box32_t *box = &boxes[i];
|
|
||||||
|
|
||||||
int damage_x = box->x1;
|
|
||||||
int damage_y = box->y1;
|
|
||||||
int damage_width = box->x2 - box->x1;
|
|
||||||
int damage_height = box->y2 - box->y1;
|
|
||||||
|
|
||||||
zwlr_screencopy_frame_v1_send_damage(frame->resource,
|
|
||||||
damage_x, damage_y, damage_width, damage_height);
|
|
||||||
}
|
|
||||||
|
|
||||||
pixman_region32_clear(&damage->damage);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void frame_send_ready(struct wlr_screencopy_frame_v1 *frame,
|
|
||||||
struct timespec *when) {
|
|
||||||
time_t tv_sec = when->tv_sec;
|
|
||||||
uint32_t tv_sec_hi = (sizeof(tv_sec) > 4) ? tv_sec >> 32 : 0;
|
|
||||||
uint32_t tv_sec_lo = tv_sec & 0xFFFFFFFF;
|
|
||||||
zwlr_screencopy_frame_v1_send_ready(frame->resource,
|
|
||||||
tv_sec_hi, tv_sec_lo, when->tv_nsec);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool frame_shm_copy(struct wlr_screencopy_frame_v1 *frame,
|
|
||||||
struct wlr_buffer *src_buffer) {
|
|
||||||
struct wlr_output *output = frame->output;
|
|
||||||
struct wlr_renderer *renderer = output->renderer;
|
|
||||||
assert(renderer);
|
|
||||||
|
|
||||||
void *data;
|
|
||||||
uint32_t format;
|
|
||||||
size_t stride;
|
|
||||||
if (!wlr_buffer_begin_data_ptr_access(frame->buffer,
|
|
||||||
WLR_BUFFER_DATA_PTR_ACCESS_WRITE, &data, &format, &stride)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ok = false;
|
|
||||||
|
|
||||||
struct wlr_texture *texture = wlr_texture_from_buffer(renderer, src_buffer);
|
|
||||||
if (!texture) {
|
|
||||||
wlr_log(WLR_DEBUG, "Failed to grab a texture from a buffer during shm screencopy");
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
ok = wlr_texture_read_pixels(texture, &(struct wlr_texture_read_pixels_options) {
|
|
||||||
.data = data,
|
|
||||||
.format = format,
|
|
||||||
.stride = stride,
|
|
||||||
.src_box = frame->box,
|
|
||||||
});
|
|
||||||
|
|
||||||
wlr_texture_destroy(texture);
|
|
||||||
|
|
||||||
out:
|
|
||||||
wlr_buffer_end_data_ptr_access(frame->buffer);
|
|
||||||
|
|
||||||
if (!ok) {
|
|
||||||
wlr_log(WLR_DEBUG, "Failed to copy to destination during shm screencopy");
|
|
||||||
}
|
|
||||||
|
|
||||||
return ok;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool frame_dma_copy(struct wlr_screencopy_frame_v1 *frame,
|
|
||||||
struct wlr_buffer *src_buffer) {
|
|
||||||
struct wlr_buffer *dst_buffer = frame->buffer;
|
|
||||||
struct wlr_output *output = frame->output;
|
|
||||||
struct wlr_renderer *renderer = output->renderer;
|
|
||||||
assert(renderer);
|
|
||||||
|
|
||||||
struct wlr_texture *src_tex =
|
|
||||||
wlr_texture_from_buffer(renderer, src_buffer);
|
|
||||||
if (src_tex == NULL) {
|
|
||||||
wlr_log(WLR_DEBUG, "Failed to grab a texture from a buffer during dma screencopy");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ok = false;
|
|
||||||
|
|
||||||
struct wlr_render_pass *pass =
|
|
||||||
wlr_renderer_begin_buffer_pass(renderer, dst_buffer, NULL);
|
|
||||||
if (!pass) {
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
wlr_render_pass_add_texture(pass, &(struct wlr_render_texture_options) {
|
|
||||||
.texture = src_tex,
|
|
||||||
.blend_mode = WLR_RENDER_BLEND_MODE_NONE,
|
|
||||||
.dst_box = (struct wlr_box){
|
|
||||||
.width = dst_buffer->width,
|
|
||||||
.height = dst_buffer->height,
|
|
||||||
},
|
|
||||||
.src_box = (struct wlr_fbox){
|
|
||||||
.x = frame->box.x,
|
|
||||||
.y = frame->box.y,
|
|
||||||
.width = frame->box.width,
|
|
||||||
.height = frame->box.height,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
ok = wlr_render_pass_submit(pass);
|
|
||||||
|
|
||||||
out:
|
|
||||||
wlr_texture_destroy(src_tex);
|
|
||||||
|
|
||||||
if (!ok) {
|
|
||||||
wlr_log(WLR_DEBUG, "Failed to render to destination during dma screencopy");
|
|
||||||
}
|
|
||||||
|
|
||||||
return ok;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void frame_handle_output_commit(struct wl_listener *listener,
|
|
||||||
void *data) {
|
|
||||||
struct wlr_screencopy_frame_v1 *frame =
|
|
||||||
wl_container_of(listener, frame, output_commit);
|
|
||||||
struct wlr_output_event_commit *event = data;
|
|
||||||
struct wlr_output *output = frame->output;
|
|
||||||
|
|
||||||
if (event->state->committed & WLR_OUTPUT_STATE_ENABLED && !output->enabled) {
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(event->state->committed & WLR_OUTPUT_STATE_BUFFER)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!frame->buffer) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (frame->with_damage) {
|
|
||||||
struct screencopy_damage *damage =
|
|
||||||
screencopy_damage_get_or_create(frame->client, output);
|
|
||||||
if (damage && pixman_region32_empty(&damage->damage)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
wl_list_remove(&frame->output_commit.link);
|
|
||||||
wl_list_init(&frame->output_commit.link);
|
|
||||||
|
|
||||||
struct wlr_buffer *src_buffer = event->state->buffer;
|
|
||||||
if (frame->box.x < 0 || frame->box.y < 0 ||
|
|
||||||
frame->box.x + frame->box.width > src_buffer->width ||
|
|
||||||
frame->box.y + frame->box.height > src_buffer->height) {
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (frame->buffer_cap) {
|
|
||||||
case WLR_BUFFER_CAP_DMABUF:
|
|
||||||
if (!frame_dma_copy(frame, src_buffer)) {
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case WLR_BUFFER_CAP_DATA_PTR:
|
|
||||||
if (!frame_shm_copy(frame, src_buffer)) {
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
abort(); // unreachable
|
|
||||||
}
|
|
||||||
|
|
||||||
zwlr_screencopy_frame_v1_send_flags(frame->resource, 0);
|
|
||||||
frame_send_damage(frame);
|
|
||||||
frame_send_ready(frame, &event->when);
|
|
||||||
frame_destroy(frame);
|
|
||||||
return;
|
|
||||||
|
|
||||||
err:
|
|
||||||
zwlr_screencopy_frame_v1_send_failed(frame->resource);
|
|
||||||
frame_destroy(frame);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void frame_handle_output_destroy(struct wl_listener *listener,
|
|
||||||
void *data) {
|
|
||||||
struct wlr_screencopy_frame_v1 *frame =
|
|
||||||
wl_container_of(listener, frame, output_destroy);
|
|
||||||
zwlr_screencopy_frame_v1_send_failed(frame->resource);
|
|
||||||
frame_destroy(frame);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void frame_handle_copy(struct wl_client *wl_client,
|
|
||||||
struct wl_resource *frame_resource,
|
|
||||||
struct wl_resource *buffer_resource) {
|
|
||||||
struct wlr_screencopy_frame_v1 *frame = frame_from_resource(frame_resource);
|
|
||||||
if (frame == NULL) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct wlr_output *output = frame->output;
|
|
||||||
|
|
||||||
if (!output->enabled) {
|
|
||||||
zwlr_screencopy_frame_v1_send_failed(frame->resource);
|
|
||||||
frame_destroy(frame);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct wlr_buffer *buffer = wlr_buffer_try_from_resource(buffer_resource);
|
|
||||||
if (buffer == NULL) {
|
|
||||||
wl_resource_post_error(frame->resource,
|
|
||||||
ZWLR_SCREENCOPY_FRAME_V1_ERROR_INVALID_BUFFER,
|
|
||||||
"invalid buffer");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (buffer->width != frame->box.width || buffer->height != frame->box.height) {
|
|
||||||
wl_resource_post_error(frame->resource,
|
|
||||||
ZWLR_SCREENCOPY_FRAME_V1_ERROR_INVALID_BUFFER,
|
|
||||||
"invalid buffer dimensions");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (frame->buffer != NULL) {
|
|
||||||
wl_resource_post_error(frame->resource,
|
|
||||||
ZWLR_SCREENCOPY_FRAME_V1_ERROR_ALREADY_USED,
|
|
||||||
"frame already used");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
enum wlr_buffer_cap cap;
|
|
||||||
struct wlr_dmabuf_attributes dmabuf;
|
|
||||||
void *data;
|
|
||||||
uint32_t format;
|
|
||||||
size_t stride;
|
|
||||||
if (wlr_buffer_get_dmabuf(buffer, &dmabuf)) {
|
|
||||||
cap = WLR_BUFFER_CAP_DMABUF;
|
|
||||||
|
|
||||||
if (dmabuf.format != frame->dmabuf_format) {
|
|
||||||
wl_resource_post_error(frame->resource,
|
|
||||||
ZWLR_SCREENCOPY_FRAME_V1_ERROR_INVALID_BUFFER,
|
|
||||||
"invalid buffer format");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
} else if (wlr_buffer_begin_data_ptr_access(buffer,
|
|
||||||
WLR_BUFFER_DATA_PTR_ACCESS_WRITE, &data, &format, &stride)) {
|
|
||||||
wlr_buffer_end_data_ptr_access(buffer);
|
|
||||||
|
|
||||||
cap = WLR_BUFFER_CAP_DATA_PTR;
|
|
||||||
|
|
||||||
if (format != frame->shm_format) {
|
|
||||||
wl_resource_post_error(frame->resource,
|
|
||||||
ZWLR_SCREENCOPY_FRAME_V1_ERROR_INVALID_BUFFER,
|
|
||||||
"invalid buffer format");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (stride != (size_t)frame->shm_stride) {
|
|
||||||
wl_resource_post_error(frame->resource,
|
|
||||||
ZWLR_SCREENCOPY_FRAME_V1_ERROR_INVALID_BUFFER,
|
|
||||||
"invalid buffer stride");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
wl_resource_post_error(frame->resource,
|
|
||||||
ZWLR_SCREENCOPY_FRAME_V1_ERROR_INVALID_BUFFER,
|
|
||||||
"unsupported buffer type");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
frame->buffer = buffer;
|
|
||||||
frame->buffer_cap = cap;
|
|
||||||
|
|
||||||
wl_signal_add(&output->events.commit, &frame->output_commit);
|
|
||||||
frame->output_commit.notify = frame_handle_output_commit;
|
|
||||||
|
|
||||||
// Request a frame because we can't assume that the current front buffer is still usable. It may
|
|
||||||
// have been released already, and we shouldn't lock it here because compositors want to render
|
|
||||||
// into the least damaged buffer.
|
|
||||||
wlr_output_update_needs_frame(output);
|
|
||||||
|
|
||||||
wlr_output_lock_attach_render(output, true);
|
|
||||||
if (frame->overlay_cursor) {
|
|
||||||
wlr_output_lock_software_cursors(output, true);
|
|
||||||
frame->cursor_locked = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void frame_handle_copy_with_damage(struct wl_client *wl_client,
|
|
||||||
struct wl_resource *frame_resource,
|
|
||||||
struct wl_resource *buffer_resource) {
|
|
||||||
struct wlr_screencopy_frame_v1 *frame = frame_from_resource(frame_resource);
|
|
||||||
if (frame == NULL) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
frame->with_damage = true;
|
|
||||||
frame_handle_copy(wl_client, frame_resource, buffer_resource);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void frame_handle_destroy(struct wl_client *wl_client,
|
|
||||||
struct wl_resource *frame_resource) {
|
|
||||||
wl_resource_destroy(frame_resource);
|
|
||||||
}
|
|
||||||
|
|
||||||
static const struct zwlr_screencopy_frame_v1_interface frame_impl = {
|
|
||||||
.copy = frame_handle_copy,
|
|
||||||
.destroy = frame_handle_destroy,
|
|
||||||
.copy_with_damage = frame_handle_copy_with_damage,
|
|
||||||
};
|
|
||||||
|
|
||||||
static void frame_handle_resource_destroy(struct wl_resource *frame_resource) {
|
|
||||||
struct wlr_screencopy_frame_v1 *frame = frame_from_resource(frame_resource);
|
|
||||||
frame_destroy(frame);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static const struct zwlr_screencopy_manager_v1_interface manager_impl;
|
|
||||||
|
|
||||||
static struct wlr_screencopy_v1_client *client_from_resource(
|
|
||||||
struct wl_resource *resource) {
|
|
||||||
assert(wl_resource_instance_of(resource,
|
|
||||||
&zwlr_screencopy_manager_v1_interface, &manager_impl));
|
|
||||||
return wl_resource_get_user_data(resource);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void capture_output(struct wl_client *wl_client,
|
|
||||||
struct wlr_screencopy_v1_client *client, uint32_t version,
|
|
||||||
uint32_t id, int32_t overlay_cursor, struct wlr_output *output,
|
|
||||||
const struct wlr_box *box) {
|
|
||||||
struct wlr_screencopy_frame_v1 *frame = calloc(1, sizeof(*frame));
|
|
||||||
if (frame == NULL) {
|
|
||||||
wl_client_post_no_memory(wl_client);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
frame->output = output;
|
|
||||||
frame->overlay_cursor = !!overlay_cursor;
|
|
||||||
|
|
||||||
frame->resource = wl_resource_create(wl_client,
|
|
||||||
&zwlr_screencopy_frame_v1_interface, version, id);
|
|
||||||
if (frame->resource == NULL) {
|
|
||||||
free(frame);
|
|
||||||
wl_client_post_no_memory(wl_client);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
wl_resource_set_implementation(frame->resource, &frame_impl, frame,
|
|
||||||
frame_handle_resource_destroy);
|
|
||||||
|
|
||||||
if (output == NULL) {
|
|
||||||
wl_resource_set_user_data(frame->resource, NULL);
|
|
||||||
zwlr_screencopy_frame_v1_send_failed(frame->resource);
|
|
||||||
free(frame);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
frame->client = client;
|
|
||||||
client->ref++;
|
|
||||||
|
|
||||||
wl_list_insert(&client->manager->frames, &frame->link);
|
|
||||||
|
|
||||||
wl_list_init(&frame->output_commit.link);
|
|
||||||
|
|
||||||
wl_signal_add(&output->events.destroy, &frame->output_destroy);
|
|
||||||
frame->output_destroy.notify = frame_handle_output_destroy;
|
|
||||||
|
|
||||||
if (output == NULL || !output->enabled) {
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct wlr_renderer *renderer = output->renderer;
|
|
||||||
assert(renderer);
|
|
||||||
|
|
||||||
if (!wlr_output_configure_primary_swapchain(output, NULL, &output->swapchain)) {
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct wlr_buffer *buffer = wlr_swapchain_acquire(output->swapchain);
|
|
||||||
if (buffer == NULL) {
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct wlr_texture *texture = wlr_texture_from_buffer(renderer, buffer);
|
|
||||||
wlr_buffer_unlock(buffer);
|
|
||||||
if (!texture) {
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
frame->shm_format = wlr_texture_preferred_read_format(texture);
|
|
||||||
wlr_texture_destroy(texture);
|
|
||||||
|
|
||||||
if (frame->shm_format == DRM_FORMAT_INVALID) {
|
|
||||||
wlr_log(WLR_ERROR,
|
|
||||||
"Failed to capture output: no read format supported by renderer");
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
const struct wlr_pixel_format_info *shm_info =
|
|
||||||
drm_get_pixel_format_info(frame->shm_format);
|
|
||||||
if (!shm_info) {
|
|
||||||
wlr_log(WLR_ERROR,
|
|
||||||
"Failed to capture output: no pixel format info matching read format");
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (output->allocator &&
|
|
||||||
(output->allocator->buffer_caps & WLR_BUFFER_CAP_DMABUF)) {
|
|
||||||
frame->dmabuf_format = output->render_format;
|
|
||||||
} else {
|
|
||||||
frame->dmabuf_format = DRM_FORMAT_INVALID;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct wlr_box buffer_box = {0};
|
|
||||||
if (box == NULL) {
|
|
||||||
buffer_box.width = output->width;
|
|
||||||
buffer_box.height = output->height;
|
|
||||||
} else {
|
|
||||||
int ow, oh;
|
|
||||||
wlr_output_effective_resolution(output, &ow, &oh);
|
|
||||||
|
|
||||||
buffer_box = *box;
|
|
||||||
|
|
||||||
wlr_box_transform(&buffer_box, &buffer_box,
|
|
||||||
wlr_output_transform_invert(output->transform), ow, oh);
|
|
||||||
buffer_box.x *= output->scale;
|
|
||||||
buffer_box.y *= output->scale;
|
|
||||||
buffer_box.width *= output->scale;
|
|
||||||
buffer_box.height *= output->scale;
|
|
||||||
}
|
|
||||||
|
|
||||||
frame->box = buffer_box;
|
|
||||||
frame->shm_stride = pixel_format_info_min_stride(shm_info, buffer_box.width);
|
|
||||||
|
|
||||||
zwlr_screencopy_frame_v1_send_buffer(frame->resource,
|
|
||||||
convert_drm_format_to_wl_shm(frame->shm_format),
|
|
||||||
buffer_box.width, buffer_box.height, frame->shm_stride);
|
|
||||||
|
|
||||||
if (version >= 3) {
|
|
||||||
if (frame->dmabuf_format != DRM_FORMAT_INVALID) {
|
|
||||||
zwlr_screencopy_frame_v1_send_linux_dmabuf(
|
|
||||||
frame->resource, frame->dmabuf_format,
|
|
||||||
buffer_box.width, buffer_box.height);
|
|
||||||
}
|
|
||||||
|
|
||||||
zwlr_screencopy_frame_v1_send_buffer_done(frame->resource);
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
|
|
||||||
error:
|
|
||||||
zwlr_screencopy_frame_v1_send_failed(frame->resource);
|
|
||||||
frame_destroy(frame);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void manager_handle_capture_output(struct wl_client *wl_client,
|
|
||||||
struct wl_resource *manager_resource, uint32_t id,
|
|
||||||
int32_t overlay_cursor, struct wl_resource *output_resource) {
|
|
||||||
struct wlr_screencopy_v1_client *client =
|
|
||||||
client_from_resource(manager_resource);
|
|
||||||
uint32_t version = wl_resource_get_version(manager_resource);
|
|
||||||
struct wlr_output *output = wlr_output_from_resource(output_resource);
|
|
||||||
|
|
||||||
capture_output(wl_client, client, version, id, overlay_cursor, output,
|
|
||||||
NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void manager_handle_capture_output_region(struct wl_client *wl_client,
|
|
||||||
struct wl_resource *manager_resource, uint32_t id,
|
|
||||||
int32_t overlay_cursor, struct wl_resource *output_resource,
|
|
||||||
int32_t x, int32_t y, int32_t width, int32_t height) {
|
|
||||||
struct wlr_screencopy_v1_client *client =
|
|
||||||
client_from_resource(manager_resource);
|
|
||||||
uint32_t version = wl_resource_get_version(manager_resource);
|
|
||||||
struct wlr_output *output = wlr_output_from_resource(output_resource);
|
|
||||||
|
|
||||||
struct wlr_box box = {
|
|
||||||
.x = x,
|
|
||||||
.y = y,
|
|
||||||
.width = width,
|
|
||||||
.height = height,
|
|
||||||
};
|
|
||||||
capture_output(wl_client, client, version, id, overlay_cursor, output,
|
|
||||||
&box);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void manager_handle_destroy(struct wl_client *wl_client,
|
|
||||||
struct wl_resource *manager_resource) {
|
|
||||||
wl_resource_destroy(manager_resource);
|
|
||||||
}
|
|
||||||
|
|
||||||
static const struct zwlr_screencopy_manager_v1_interface manager_impl = {
|
|
||||||
.capture_output = manager_handle_capture_output,
|
|
||||||
.capture_output_region = manager_handle_capture_output_region,
|
|
||||||
.destroy = manager_handle_destroy,
|
|
||||||
};
|
|
||||||
|
|
||||||
static void manager_handle_resource_destroy(struct wl_resource *resource) {
|
|
||||||
struct wlr_screencopy_v1_client *client =
|
|
||||||
client_from_resource(resource);
|
|
||||||
client_unref(client);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void manager_bind(struct wl_client *wl_client, void *data,
|
|
||||||
uint32_t version, uint32_t id) {
|
|
||||||
struct wlr_screencopy_manager_v1 *manager = data;
|
|
||||||
|
|
||||||
struct wlr_screencopy_v1_client *client = calloc(1, sizeof(*client));
|
|
||||||
if (client == NULL) {
|
|
||||||
goto failure;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct wl_resource *resource = wl_resource_create(wl_client,
|
|
||||||
&zwlr_screencopy_manager_v1_interface, version, id);
|
|
||||||
if (resource == NULL) {
|
|
||||||
goto failure;
|
|
||||||
}
|
|
||||||
|
|
||||||
client->ref = 1;
|
|
||||||
client->manager = manager;
|
|
||||||
wl_list_init(&client->damages);
|
|
||||||
|
|
||||||
wl_resource_set_implementation(resource, &manager_impl, client,
|
|
||||||
manager_handle_resource_destroy);
|
|
||||||
|
|
||||||
return;
|
|
||||||
failure:
|
|
||||||
free(client);
|
|
||||||
wl_client_post_no_memory(wl_client);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void handle_display_destroy(struct wl_listener *listener, void *data) {
|
|
||||||
struct wlr_screencopy_manager_v1 *manager =
|
|
||||||
wl_container_of(listener, manager, display_destroy);
|
|
||||||
wl_signal_emit_mutable(&manager->events.destroy, manager);
|
|
||||||
|
|
||||||
assert(wl_list_empty(&manager->events.destroy.listener_list));
|
|
||||||
|
|
||||||
wl_list_remove(&manager->display_destroy.link);
|
|
||||||
wl_global_destroy(manager->global);
|
|
||||||
free(manager);
|
|
||||||
}
|
|
||||||
|
|
||||||
struct wlr_screencopy_manager_v1 *wlr_screencopy_manager_v1_create(
|
|
||||||
struct wl_display *display) {
|
|
||||||
struct wlr_screencopy_manager_v1 *manager = calloc(1, sizeof(*manager));
|
|
||||||
if (manager == NULL) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
manager->global = wl_global_create(display,
|
|
||||||
&zwlr_screencopy_manager_v1_interface, SCREENCOPY_MANAGER_VERSION,
|
|
||||||
manager, manager_bind);
|
|
||||||
if (manager->global == NULL) {
|
|
||||||
free(manager);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
wl_list_init(&manager->frames);
|
|
||||||
|
|
||||||
wl_signal_init(&manager->events.destroy);
|
|
||||||
|
|
||||||
manager->display_destroy.notify = handle_display_destroy;
|
|
||||||
wl_display_add_destroy_listener(display, &manager->display_destroy);
|
|
||||||
|
|
||||||
return manager;
|
|
||||||
}
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue