mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-26 09:08:14 +00:00
6d8133e41e
We would use a frame callback from the surface to indicate that last buffer is rendered, but when we destroy the surface and that callback is not back yet, it may cause the wayland event queue crash. Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-bad/-/merge_requests/1729>
575 lines
17 KiB
C
575 lines
17 KiB
C
/* GStreamer Wayland video sink
|
|
*
|
|
* Copyright (C) 2011 Intel Corporation
|
|
* Copyright (C) 2011 Sreerenj Balachandran <sreerenj.balachandran@intel.com>
|
|
* Copyright (C) 2014 Collabora Ltd.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the Free
|
|
* Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301 USA.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include "wlwindow.h"
|
|
#include "wlshmallocator.h"
|
|
#include "wlbuffer.h"
|
|
|
|
GST_DEBUG_CATEGORY_EXTERN (gstwayland_debug);
|
|
#define GST_CAT_DEFAULT gstwayland_debug
|
|
|
|
enum
|
|
{
|
|
CLOSED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static guint signals[LAST_SIGNAL] = { 0 };
|
|
|
|
G_DEFINE_TYPE (GstWlWindow, gst_wl_window, G_TYPE_OBJECT);
|
|
|
|
static void gst_wl_window_finalize (GObject * gobject);
|
|
|
|
static void
|
|
handle_xdg_toplevel_close (void *data, struct xdg_toplevel *xdg_toplevel)
|
|
{
|
|
GstWlWindow *window = data;
|
|
|
|
GST_DEBUG ("XDG toplevel got a \"close\" event.");
|
|
g_signal_emit (window, signals[CLOSED], 0);
|
|
}
|
|
|
|
static void
|
|
handle_xdg_toplevel_configure (void *data, struct xdg_toplevel *xdg_toplevel,
|
|
int32_t width, int32_t height, struct wl_array *states)
|
|
{
|
|
GstWlWindow *window = data;
|
|
const uint32_t *state;
|
|
|
|
GST_DEBUG ("XDG toplevel got a \"configure\" event, [ %d, %d ].",
|
|
width, height);
|
|
|
|
wl_array_for_each (state, states) {
|
|
switch (*state) {
|
|
case XDG_TOPLEVEL_STATE_FULLSCREEN:
|
|
case XDG_TOPLEVEL_STATE_MAXIMIZED:
|
|
case XDG_TOPLEVEL_STATE_RESIZING:
|
|
case XDG_TOPLEVEL_STATE_ACTIVATED:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (width <= 0 || height <= 0)
|
|
return;
|
|
|
|
gst_wl_window_set_render_rectangle (window, 0, 0, width, height);
|
|
}
|
|
|
|
static const struct xdg_toplevel_listener xdg_toplevel_listener = {
|
|
handle_xdg_toplevel_configure,
|
|
handle_xdg_toplevel_close,
|
|
};
|
|
|
|
static void
|
|
handle_xdg_surface_configure (void *data, struct xdg_surface *xdg_surface,
|
|
uint32_t serial)
|
|
{
|
|
GstWlWindow *window = data;
|
|
xdg_surface_ack_configure (xdg_surface, serial);
|
|
|
|
g_mutex_lock (&window->configure_mutex);
|
|
window->configured = TRUE;
|
|
g_cond_signal (&window->configure_cond);
|
|
g_mutex_unlock (&window->configure_mutex);
|
|
}
|
|
|
|
static const struct xdg_surface_listener xdg_surface_listener = {
|
|
handle_xdg_surface_configure,
|
|
};
|
|
|
|
static void
|
|
handle_ping (void *data, struct wl_shell_surface *wl_shell_surface,
|
|
uint32_t serial)
|
|
{
|
|
wl_shell_surface_pong (wl_shell_surface, serial);
|
|
}
|
|
|
|
static void
|
|
handle_configure (void *data, struct wl_shell_surface *wl_shell_surface,
|
|
uint32_t edges, int32_t width, int32_t height)
|
|
{
|
|
GstWlWindow *window = data;
|
|
|
|
GST_DEBUG ("Windows configure: edges %x, width = %i, height %i", edges,
|
|
width, height);
|
|
|
|
if (width == 0 || height == 0)
|
|
return;
|
|
|
|
gst_wl_window_set_render_rectangle (window, 0, 0, width, height);
|
|
}
|
|
|
|
static void
|
|
handle_popup_done (void *data, struct wl_shell_surface *wl_shell_surface)
|
|
{
|
|
GST_DEBUG ("Window popup done.");
|
|
}
|
|
|
|
static const struct wl_shell_surface_listener wl_shell_surface_listener = {
|
|
handle_ping,
|
|
handle_configure,
|
|
handle_popup_done
|
|
};
|
|
|
|
static void
|
|
gst_wl_window_class_init (GstWlWindowClass * klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
gobject_class->finalize = gst_wl_window_finalize;
|
|
|
|
signals[CLOSED] = g_signal_new ("closed", G_TYPE_FROM_CLASS (gobject_class),
|
|
G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 0);
|
|
}
|
|
|
|
static void
|
|
gst_wl_window_init (GstWlWindow * self)
|
|
{
|
|
self->configured = TRUE;
|
|
g_cond_init (&self->configure_cond);
|
|
g_mutex_init (&self->configure_mutex);
|
|
}
|
|
|
|
static void
|
|
gst_wl_window_finalize (GObject * gobject)
|
|
{
|
|
GstWlWindow *self = GST_WL_WINDOW (gobject);
|
|
|
|
if (self->callback)
|
|
wl_callback_destroy (self->callback);
|
|
|
|
if (self->wl_shell_surface)
|
|
wl_shell_surface_destroy (self->wl_shell_surface);
|
|
|
|
if (self->xdg_toplevel)
|
|
xdg_toplevel_destroy (self->xdg_toplevel);
|
|
if (self->xdg_surface)
|
|
xdg_surface_destroy (self->xdg_surface);
|
|
|
|
if (self->video_viewport)
|
|
wp_viewport_destroy (self->video_viewport);
|
|
|
|
wl_proxy_wrapper_destroy (self->video_surface_wrapper);
|
|
wl_subsurface_destroy (self->video_subsurface);
|
|
wl_surface_destroy (self->video_surface);
|
|
|
|
if (self->area_subsurface)
|
|
wl_subsurface_destroy (self->area_subsurface);
|
|
|
|
if (self->area_viewport)
|
|
wp_viewport_destroy (self->area_viewport);
|
|
|
|
wl_proxy_wrapper_destroy (self->area_surface_wrapper);
|
|
wl_surface_destroy (self->area_surface);
|
|
|
|
g_clear_object (&self->display);
|
|
|
|
G_OBJECT_CLASS (gst_wl_window_parent_class)->finalize (gobject);
|
|
}
|
|
|
|
static GstWlWindow *
|
|
gst_wl_window_new_internal (GstWlDisplay * display, GMutex * render_lock)
|
|
{
|
|
GstWlWindow *window;
|
|
struct wl_region *region;
|
|
|
|
window = g_object_new (GST_TYPE_WL_WINDOW, NULL);
|
|
window->display = g_object_ref (display);
|
|
window->render_lock = render_lock;
|
|
g_cond_init (&window->configure_cond);
|
|
|
|
window->callback = NULL;
|
|
window->area_surface = wl_compositor_create_surface (display->compositor);
|
|
window->video_surface = wl_compositor_create_surface (display->compositor);
|
|
|
|
window->area_surface_wrapper = wl_proxy_create_wrapper (window->area_surface);
|
|
window->video_surface_wrapper =
|
|
wl_proxy_create_wrapper (window->video_surface);
|
|
|
|
wl_proxy_set_queue ((struct wl_proxy *) window->area_surface_wrapper,
|
|
display->queue);
|
|
wl_proxy_set_queue ((struct wl_proxy *) window->video_surface_wrapper,
|
|
display->queue);
|
|
|
|
/* embed video_surface in area_surface */
|
|
window->video_subsurface =
|
|
wl_subcompositor_get_subsurface (display->subcompositor,
|
|
window->video_surface, window->area_surface);
|
|
wl_subsurface_set_desync (window->video_subsurface);
|
|
|
|
if (display->viewporter) {
|
|
window->area_viewport = wp_viewporter_get_viewport (display->viewporter,
|
|
window->area_surface);
|
|
window->video_viewport = wp_viewporter_get_viewport (display->viewporter,
|
|
window->video_surface);
|
|
}
|
|
|
|
/* do not accept input */
|
|
region = wl_compositor_create_region (display->compositor);
|
|
wl_surface_set_input_region (window->area_surface, region);
|
|
wl_region_destroy (region);
|
|
|
|
region = wl_compositor_create_region (display->compositor);
|
|
wl_surface_set_input_region (window->video_surface, region);
|
|
wl_region_destroy (region);
|
|
|
|
return window;
|
|
}
|
|
|
|
void
|
|
gst_wl_window_ensure_fullscreen (GstWlWindow * window, gboolean fullscreen)
|
|
{
|
|
if (!window)
|
|
return;
|
|
|
|
if (window->display->xdg_wm_base) {
|
|
if (fullscreen)
|
|
xdg_toplevel_set_fullscreen (window->xdg_toplevel, NULL);
|
|
else
|
|
xdg_toplevel_unset_fullscreen (window->xdg_toplevel);
|
|
} else {
|
|
if (fullscreen)
|
|
wl_shell_surface_set_fullscreen (window->wl_shell_surface,
|
|
WL_SHELL_SURFACE_FULLSCREEN_METHOD_SCALE, 0, NULL);
|
|
else
|
|
wl_shell_surface_set_toplevel (window->wl_shell_surface);
|
|
}
|
|
}
|
|
|
|
GstWlWindow *
|
|
gst_wl_window_new_toplevel (GstWlDisplay * display, const GstVideoInfo * info,
|
|
gboolean fullscreen, GMutex * render_lock)
|
|
{
|
|
GstWlWindow *window;
|
|
|
|
window = gst_wl_window_new_internal (display, render_lock);
|
|
|
|
/* Check which protocol we will use (in order of preference) */
|
|
if (display->xdg_wm_base) {
|
|
gint64 timeout;
|
|
|
|
/* First create the XDG surface */
|
|
window->xdg_surface = xdg_wm_base_get_xdg_surface (display->xdg_wm_base,
|
|
window->area_surface);
|
|
if (!window->xdg_surface) {
|
|
GST_ERROR ("Unable to get xdg_surface");
|
|
goto error;
|
|
}
|
|
xdg_surface_add_listener (window->xdg_surface, &xdg_surface_listener,
|
|
window);
|
|
|
|
/* Then the toplevel */
|
|
window->xdg_toplevel = xdg_surface_get_toplevel (window->xdg_surface);
|
|
if (!window->xdg_toplevel) {
|
|
GST_ERROR ("Unable to get xdg_toplevel");
|
|
goto error;
|
|
}
|
|
xdg_toplevel_add_listener (window->xdg_toplevel,
|
|
&xdg_toplevel_listener, window);
|
|
|
|
gst_wl_window_ensure_fullscreen (window, fullscreen);
|
|
|
|
/* Finally, commit the xdg_surface state as toplevel */
|
|
window->configured = FALSE;
|
|
wl_surface_commit (window->video_surface);
|
|
wl_display_flush (display->display);
|
|
|
|
g_mutex_lock (&window->configure_mutex);
|
|
timeout = g_get_monotonic_time () + 100 * G_TIME_SPAN_MILLISECOND;
|
|
while (!window->configured) {
|
|
if (!g_cond_wait_until (&window->configure_cond, &window->configure_mutex,
|
|
timeout)) {
|
|
GST_WARNING ("The compositor did not send configure event.");
|
|
break;
|
|
}
|
|
}
|
|
g_mutex_unlock (&window->configure_mutex);
|
|
} else if (display->wl_shell) {
|
|
/* go toplevel */
|
|
window->wl_shell_surface = wl_shell_get_shell_surface (display->wl_shell,
|
|
window->area_surface);
|
|
if (!window->wl_shell_surface) {
|
|
GST_ERROR ("Unable to get wl_shell_surface");
|
|
goto error;
|
|
}
|
|
|
|
wl_shell_surface_add_listener (window->wl_shell_surface,
|
|
&wl_shell_surface_listener, window);
|
|
gst_wl_window_ensure_fullscreen (window, fullscreen);
|
|
} else if (display->fullscreen_shell) {
|
|
zwp_fullscreen_shell_v1_present_surface (display->fullscreen_shell,
|
|
window->area_surface, ZWP_FULLSCREEN_SHELL_V1_PRESENT_METHOD_ZOOM,
|
|
NULL);
|
|
} else {
|
|
GST_ERROR ("Unable to use either wl_shell, xdg_wm_base or "
|
|
"zwp_fullscreen_shell.");
|
|
goto error;
|
|
}
|
|
|
|
/* render_rectangle is already set via toplevel_configure in
|
|
* xdg_shell fullscreen mode */
|
|
if (!(display->xdg_wm_base && fullscreen)) {
|
|
/* set the initial size to be the same as the reported video size */
|
|
gint width =
|
|
gst_util_uint64_scale_int_round (info->width, info->par_n, info->par_d);
|
|
gst_wl_window_set_render_rectangle (window, 0, 0, width, info->height);
|
|
}
|
|
|
|
return window;
|
|
|
|
error:
|
|
g_object_unref (window);
|
|
return NULL;
|
|
}
|
|
|
|
GstWlWindow *
|
|
gst_wl_window_new_in_surface (GstWlDisplay * display,
|
|
struct wl_surface * parent, GMutex * render_lock)
|
|
{
|
|
GstWlWindow *window;
|
|
window = gst_wl_window_new_internal (display, render_lock);
|
|
|
|
/* embed in parent */
|
|
window->area_subsurface =
|
|
wl_subcompositor_get_subsurface (display->subcompositor,
|
|
window->area_surface, parent);
|
|
wl_subsurface_set_desync (window->area_subsurface);
|
|
|
|
wl_surface_commit (parent);
|
|
|
|
return window;
|
|
}
|
|
|
|
GstWlDisplay *
|
|
gst_wl_window_get_display (GstWlWindow * window)
|
|
{
|
|
g_return_val_if_fail (window != NULL, NULL);
|
|
|
|
return g_object_ref (window->display);
|
|
}
|
|
|
|
struct wl_surface *
|
|
gst_wl_window_get_wl_surface (GstWlWindow * window)
|
|
{
|
|
g_return_val_if_fail (window != NULL, NULL);
|
|
|
|
return window->video_surface_wrapper;
|
|
}
|
|
|
|
gboolean
|
|
gst_wl_window_is_toplevel (GstWlWindow * window)
|
|
{
|
|
g_return_val_if_fail (window != NULL, FALSE);
|
|
|
|
if (window->display->xdg_wm_base)
|
|
return (window->xdg_toplevel != NULL);
|
|
else
|
|
return (window->wl_shell_surface != NULL);
|
|
}
|
|
|
|
static void
|
|
gst_wl_window_resize_video_surface (GstWlWindow * window, gboolean commit)
|
|
{
|
|
GstVideoRectangle src = { 0, };
|
|
GstVideoRectangle dst = { 0, };
|
|
GstVideoRectangle res;
|
|
|
|
/* center the video_subsurface inside area_subsurface */
|
|
src.w = window->video_width;
|
|
src.h = window->video_height;
|
|
dst.w = window->render_rectangle.w;
|
|
dst.h = window->render_rectangle.h;
|
|
|
|
if (window->video_viewport) {
|
|
gst_video_sink_center_rect (src, dst, &res, TRUE);
|
|
wp_viewport_set_destination (window->video_viewport, res.w, res.h);
|
|
} else {
|
|
gst_video_sink_center_rect (src, dst, &res, FALSE);
|
|
}
|
|
|
|
wl_subsurface_set_position (window->video_subsurface, res.x, res.y);
|
|
|
|
if (commit) {
|
|
wl_surface_damage (window->video_surface_wrapper, 0, 0, res.w, res.h);
|
|
wl_surface_commit (window->video_surface_wrapper);
|
|
}
|
|
|
|
if (gst_wl_window_is_toplevel (window)) {
|
|
struct wl_region *region;
|
|
|
|
region = wl_compositor_create_region (window->display->compositor);
|
|
wl_region_add (region, 0, 0, window->render_rectangle.w,
|
|
window->render_rectangle.h);
|
|
wl_surface_set_input_region (window->area_surface, region);
|
|
wl_region_destroy (region);
|
|
}
|
|
|
|
/* this is saved for use in wl_surface_damage */
|
|
window->video_rectangle = res;
|
|
}
|
|
|
|
static void
|
|
gst_wl_window_set_opaque (GstWlWindow * window, const GstVideoInfo * info)
|
|
{
|
|
struct wl_region *region;
|
|
|
|
/* Set area opaque */
|
|
region = wl_compositor_create_region (window->display->compositor);
|
|
wl_region_add (region, 0, 0, window->render_rectangle.w,
|
|
window->render_rectangle.h);
|
|
wl_surface_set_opaque_region (window->area_surface, region);
|
|
wl_region_destroy (region);
|
|
|
|
if (!GST_VIDEO_INFO_HAS_ALPHA (info)) {
|
|
/* Set video opaque */
|
|
region = wl_compositor_create_region (window->display->compositor);
|
|
wl_region_add (region, 0, 0, window->render_rectangle.w,
|
|
window->render_rectangle.h);
|
|
wl_surface_set_opaque_region (window->video_surface, region);
|
|
wl_region_destroy (region);
|
|
}
|
|
}
|
|
|
|
void
|
|
gst_wl_window_render (GstWlWindow * window, GstWlBuffer * buffer,
|
|
const GstVideoInfo * info)
|
|
{
|
|
if (G_UNLIKELY (info)) {
|
|
window->video_width =
|
|
gst_util_uint64_scale_int_round (info->width, info->par_n, info->par_d);
|
|
window->video_height = info->height;
|
|
|
|
wl_subsurface_set_sync (window->video_subsurface);
|
|
gst_wl_window_resize_video_surface (window, FALSE);
|
|
gst_wl_window_set_opaque (window, info);
|
|
}
|
|
|
|
if (G_LIKELY (buffer)) {
|
|
gst_wl_buffer_attach (buffer, window->video_surface_wrapper);
|
|
wl_surface_damage (window->video_surface_wrapper, 0, 0,
|
|
window->video_rectangle.w, window->video_rectangle.h);
|
|
wl_surface_commit (window->video_surface_wrapper);
|
|
} else {
|
|
/* clear both video and parent surfaces */
|
|
wl_surface_attach (window->video_surface_wrapper, NULL, 0, 0);
|
|
wl_surface_commit (window->video_surface_wrapper);
|
|
wl_surface_attach (window->area_surface_wrapper, NULL, 0, 0);
|
|
wl_surface_commit (window->area_surface_wrapper);
|
|
}
|
|
|
|
if (G_UNLIKELY (info)) {
|
|
/* commit also the parent (area_surface) in order to change
|
|
* the position of the video_subsurface */
|
|
wl_surface_damage (window->area_surface_wrapper, 0, 0,
|
|
window->render_rectangle.w, window->render_rectangle.h);
|
|
wl_surface_commit (window->area_surface_wrapper);
|
|
wl_subsurface_set_desync (window->video_subsurface);
|
|
}
|
|
|
|
wl_display_flush (window->display->display);
|
|
}
|
|
|
|
/* Update the buffer used to draw black borders. When we have viewporter
|
|
* support, this is a scaled up 1x1 image, and without we need an black image
|
|
* the size of the rendering areay. */
|
|
static void
|
|
gst_wl_window_update_borders (GstWlWindow * window)
|
|
{
|
|
GstVideoFormat format;
|
|
GstVideoInfo info;
|
|
gint width, height;
|
|
GstBuffer *buf;
|
|
struct wl_buffer *wlbuf;
|
|
GstWlBuffer *gwlbuf;
|
|
GstAllocator *alloc;
|
|
|
|
if (window->no_border_update)
|
|
return;
|
|
|
|
if (window->display->viewporter) {
|
|
width = height = 1;
|
|
window->no_border_update = TRUE;
|
|
} else {
|
|
width = window->render_rectangle.w;
|
|
height = window->render_rectangle.h;
|
|
}
|
|
|
|
/* we want WL_SHM_FORMAT_XRGB8888 */
|
|
format = GST_VIDEO_FORMAT_BGRx;
|
|
|
|
/* draw the area_subsurface */
|
|
gst_video_info_set_format (&info, format, width, height);
|
|
|
|
alloc = gst_wl_shm_allocator_get ();
|
|
|
|
buf = gst_buffer_new_allocate (alloc, info.size, NULL);
|
|
gst_buffer_memset (buf, 0, 0, info.size);
|
|
wlbuf =
|
|
gst_wl_shm_memory_construct_wl_buffer (gst_buffer_peek_memory (buf, 0),
|
|
window->display, &info);
|
|
gwlbuf = gst_buffer_add_wl_buffer (buf, wlbuf, window->display);
|
|
gst_wl_buffer_attach (gwlbuf, window->area_surface_wrapper);
|
|
|
|
/* at this point, the GstWlBuffer keeps the buffer
|
|
* alive and will free it on wl_buffer::release */
|
|
gst_buffer_unref (buf);
|
|
g_object_unref (alloc);
|
|
}
|
|
|
|
void
|
|
gst_wl_window_set_render_rectangle (GstWlWindow * window, gint x, gint y,
|
|
gint w, gint h)
|
|
{
|
|
g_return_if_fail (window != NULL);
|
|
|
|
window->render_rectangle.x = x;
|
|
window->render_rectangle.y = y;
|
|
window->render_rectangle.w = w;
|
|
window->render_rectangle.h = h;
|
|
|
|
/* position the area inside the parent - needs a parent commit to apply */
|
|
if (window->area_subsurface)
|
|
wl_subsurface_set_position (window->area_subsurface, x, y);
|
|
|
|
/* change the size of the area */
|
|
if (window->area_viewport)
|
|
wp_viewport_set_destination (window->area_viewport, w, h);
|
|
|
|
gst_wl_window_update_borders (window);
|
|
|
|
if (!window->configured)
|
|
return;
|
|
|
|
if (window->video_width != 0) {
|
|
wl_subsurface_set_sync (window->video_subsurface);
|
|
gst_wl_window_resize_video_surface (window, TRUE);
|
|
}
|
|
|
|
wl_surface_damage (window->area_surface_wrapper, 0, 0, w, h);
|
|
wl_surface_commit (window->area_surface_wrapper);
|
|
|
|
if (window->video_width != 0)
|
|
wl_subsurface_set_desync (window->video_subsurface);
|
|
}
|