mirror of
https://github.com/hyprwm/xdg-desktop-portal-hyprland.git
synced 2025-01-26 07:49:49 +01:00
screencast: destroy session if output is removed
We teardown all existing screencast_instances using the removed output by destroying the frame, removing all timers and then marking the instance as ready for teardown so we can destroy it after the last connected session is closed. Any wlr_screencopy_frame_v1 has to be destroyed before the connected output can be removed. Otherwise wlr_screencopy_frame_v1_destroy segfaults the whole program. To ensure this we will make all frame callbacks safe to be triggered for a previous destroyed frame.
This commit is contained in:
parent
ab0a07142d
commit
b5491df0c0
5 changed files with 56 additions and 0 deletions
|
@ -4,5 +4,6 @@
|
|||
#include "screencast_common.h"
|
||||
|
||||
void xdpw_screencast_instance_destroy(struct xdpw_screencast_instance *cast);
|
||||
void xdpw_screencast_instance_teardown(struct xdpw_screencast_instance *cast);
|
||||
|
||||
#endif
|
||||
|
|
|
@ -161,6 +161,7 @@ struct xdpw_screencast_instance {
|
|||
bool with_cursor;
|
||||
int err;
|
||||
bool quit;
|
||||
bool teardown;
|
||||
enum buffer_type buffer_type;
|
||||
|
||||
// fps limit
|
||||
|
|
|
@ -71,6 +71,9 @@ void xdpw_session_destroy(struct xdpw_session *sess) {
|
|||
if (cast->frame_state == XDPW_FRAME_STATE_NONE) {
|
||||
logprint(TRACE, "xdpw: screencast instance not streaming, destroy it");
|
||||
xdpw_screencast_instance_destroy(cast);
|
||||
} else if (cast->teardown) {
|
||||
logprint(TRACE, "xdpw: screencast instance marked for teardown, destroy it");
|
||||
xdpw_screencast_instance_destroy(cast);
|
||||
} else {
|
||||
logprint(TRACE, "xdpw: screencast instance still streaming, set quit flag");
|
||||
cast->quit = true;
|
||||
|
|
|
@ -71,6 +71,7 @@ void xdpw_screencast_instance_init(struct xdpw_screencast_context *ctx,
|
|||
cast->refcount = 1;
|
||||
cast->node_id = SPA_ID_INVALID;
|
||||
cast->avoid_dmabufs = false;
|
||||
cast->teardown = false;
|
||||
wl_list_init(&cast->buffer_list);
|
||||
logprint(INFO, "xdpw: screencast instance %p has %d references", cast, cast->refcount);
|
||||
wl_list_insert(&ctx->screencast_instances, &cast->link);
|
||||
|
@ -97,6 +98,15 @@ void xdpw_screencast_instance_destroy(struct xdpw_screencast_instance *cast) {
|
|||
free(cast);
|
||||
}
|
||||
|
||||
void xdpw_screencast_instance_teardown(struct xdpw_screencast_instance *cast) {
|
||||
struct xdpw_session *sess, *tmp;
|
||||
wl_list_for_each_safe(sess, tmp, &cast->ctx->state->xdpw_sessions, link) {
|
||||
if (sess->screencast_instance == cast) {
|
||||
xdpw_session_destroy(sess);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
bool setup_outputs(struct xdpw_screencast_context *ctx, struct xdpw_session *sess, bool with_cursor) {
|
||||
|
||||
struct xdpw_wlr_output *output, *tmp_o;
|
||||
|
|
|
@ -24,6 +24,9 @@
|
|||
#include "fps_limit.h"
|
||||
|
||||
void wlr_frame_free(struct xdpw_screencast_instance *cast) {
|
||||
if (!cast->wlr_frame) {
|
||||
return;
|
||||
}
|
||||
zwlr_screencopy_frame_v1_destroy(cast->wlr_frame);
|
||||
cast->wlr_frame = NULL;
|
||||
logprint(TRACE, "wlroots: frame destroyed");
|
||||
|
@ -89,6 +92,9 @@ static void wlr_frame_buffer_done(void *data,
|
|||
static void wlr_frame_buffer(void *data, struct zwlr_screencopy_frame_v1 *frame,
|
||||
uint32_t format, uint32_t width, uint32_t height, uint32_t stride) {
|
||||
struct xdpw_screencast_instance *cast = data;
|
||||
if (!frame) {
|
||||
return;
|
||||
}
|
||||
|
||||
logprint(TRACE, "wlroots: buffer event handler");
|
||||
cast->wlr_frame = frame;
|
||||
|
@ -108,6 +114,9 @@ static void wlr_frame_linux_dmabuf(void *data,
|
|||
struct zwlr_screencopy_frame_v1 *frame,
|
||||
uint32_t format, uint32_t width, uint32_t height) {
|
||||
struct xdpw_screencast_instance *cast = data;
|
||||
if (!frame) {
|
||||
return;
|
||||
}
|
||||
|
||||
logprint(TRACE, "wlroots: linux_dmabuf event handler");
|
||||
|
||||
|
@ -119,6 +128,9 @@ static void wlr_frame_linux_dmabuf(void *data,
|
|||
static void wlr_frame_buffer_done(void *data,
|
||||
struct zwlr_screencopy_frame_v1 *frame) {
|
||||
struct xdpw_screencast_instance *cast = data;
|
||||
if (!frame) {
|
||||
return;
|
||||
}
|
||||
|
||||
logprint(TRACE, "wlroots: buffer_done event handler");
|
||||
|
||||
|
@ -171,6 +183,9 @@ static void wlr_frame_buffer_done(void *data,
|
|||
static void wlr_frame_flags(void *data, struct zwlr_screencopy_frame_v1 *frame,
|
||||
uint32_t flags) {
|
||||
struct xdpw_screencast_instance *cast = data;
|
||||
if (!frame) {
|
||||
return;
|
||||
}
|
||||
|
||||
logprint(TRACE, "wlroots: flags event handler");
|
||||
cast->current_frame.y_invert = flags & ZWLR_SCREENCOPY_FRAME_V1_FLAGS_Y_INVERT;
|
||||
|
@ -179,6 +194,9 @@ static void wlr_frame_flags(void *data, struct zwlr_screencopy_frame_v1 *frame,
|
|||
static void wlr_frame_damage(void *data, struct zwlr_screencopy_frame_v1 *frame,
|
||||
uint32_t x, uint32_t y, uint32_t width, uint32_t height) {
|
||||
struct xdpw_screencast_instance *cast = data;
|
||||
if (!frame) {
|
||||
return;
|
||||
}
|
||||
|
||||
logprint(TRACE, "wlroots: damage event handler");
|
||||
|
||||
|
@ -191,6 +209,9 @@ static void wlr_frame_damage(void *data, struct zwlr_screencopy_frame_v1 *frame,
|
|||
static void wlr_frame_ready(void *data, struct zwlr_screencopy_frame_v1 *frame,
|
||||
uint32_t tv_sec_hi, uint32_t tv_sec_lo, uint32_t tv_nsec) {
|
||||
struct xdpw_screencast_instance *cast = data;
|
||||
if (!frame) {
|
||||
return;
|
||||
}
|
||||
|
||||
logprint(TRACE, "wlroots: ready event handler");
|
||||
|
||||
|
@ -205,6 +226,9 @@ static void wlr_frame_ready(void *data, struct zwlr_screencopy_frame_v1 *frame,
|
|||
static void wlr_frame_failed(void *data,
|
||||
struct zwlr_screencopy_frame_v1 *frame) {
|
||||
struct xdpw_screencast_instance *cast = data;
|
||||
if (!frame) {
|
||||
return;
|
||||
}
|
||||
|
||||
logprint(TRACE, "wlroots: failed event handler");
|
||||
|
||||
|
@ -773,6 +797,23 @@ static void wlr_registry_handle_remove(void *data, struct wl_registry *reg,
|
|||
struct xdpw_screencast_context *ctx = data;
|
||||
struct xdpw_wlr_output *output = xdpw_wlr_output_find(ctx, NULL, id);
|
||||
if (output) {
|
||||
logprint(DEBUG, "wlroots: output removed (%s)", output->name);
|
||||
struct xdpw_screencast_instance *cast, *tmp;
|
||||
wl_list_for_each_safe(cast, tmp, &ctx->screencast_instances, link) {
|
||||
if (cast->target_output == output) {
|
||||
// screencopy might be in process for this instance
|
||||
wlr_frame_free(cast);
|
||||
// instance might be waiting for wakeup by the frame limiter
|
||||
struct xdpw_timer *timer, *ttmp;
|
||||
wl_list_for_each_safe(timer, ttmp, &cast->ctx->state->timers, link) {
|
||||
if (timer->user_data == cast) {
|
||||
xdpw_destroy_timer(timer);
|
||||
}
|
||||
}
|
||||
cast->teardown = true;
|
||||
xdpw_screencast_instance_teardown(cast);
|
||||
}
|
||||
}
|
||||
wlr_remove_output(output);
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue