mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-15 03:45:38 +00:00
c51fe83d42
The intention of this code is to find the center rectangle relative to (0,0), since subsurface coordinates are relative to the parent surface. The old code used to work but was wrong and broken by http://cgit.freedesktop.org/gstreamer/gst-plugins-base/commit/gst-libs/gst/video/gstvideosink.c?id=ff57f6913456ec1991e55517cf1f239e80eeddef
336 lines
9.6 KiB
C
336 lines
9.6 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
|
|
|
|
G_DEFINE_TYPE (GstWlWindow, gst_wl_window, G_TYPE_OBJECT);
|
|
|
|
static void gst_wl_window_finalize (GObject * gobject);
|
|
|
|
static void
|
|
handle_ping (void *data, struct wl_shell_surface *shell_surface,
|
|
uint32_t serial)
|
|
{
|
|
wl_shell_surface_pong (shell_surface, serial);
|
|
}
|
|
|
|
static void
|
|
handle_configure (void *data, struct wl_shell_surface *shell_surface,
|
|
uint32_t edges, int32_t width, int32_t height)
|
|
{
|
|
}
|
|
|
|
static void
|
|
handle_popup_done (void *data, struct wl_shell_surface *shell_surface)
|
|
{
|
|
}
|
|
|
|
static const struct wl_shell_surface_listener 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;
|
|
}
|
|
|
|
static void
|
|
gst_wl_window_init (GstWlWindow * self)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gst_wl_window_finalize (GObject * gobject)
|
|
{
|
|
GstWlWindow *self = GST_WL_WINDOW (gobject);
|
|
|
|
if (self->shell_surface) {
|
|
wl_shell_surface_destroy (self->shell_surface);
|
|
}
|
|
|
|
wl_viewport_destroy (self->video_viewport);
|
|
wl_subsurface_destroy (self->video_subsurface);
|
|
wl_surface_destroy (self->video_surface);
|
|
|
|
if (self->area_subsurface) {
|
|
wl_subsurface_destroy (self->area_subsurface);
|
|
}
|
|
wl_viewport_destroy (self->area_viewport);
|
|
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)
|
|
{
|
|
GstWlWindow *window;
|
|
GstVideoInfo info;
|
|
GstBuffer *buf;
|
|
GstMapInfo mapinfo;
|
|
struct wl_buffer *wlbuf;
|
|
GstWlBuffer *gwlbuf;
|
|
struct wl_region *region;
|
|
|
|
window = g_object_new (GST_TYPE_WL_WINDOW, NULL);
|
|
window->display = g_object_ref (display);
|
|
|
|
window->area_surface = wl_compositor_create_surface (display->compositor);
|
|
window->video_surface = wl_compositor_create_surface (display->compositor);
|
|
|
|
wl_proxy_set_queue ((struct wl_proxy *) window->area_surface, display->queue);
|
|
wl_proxy_set_queue ((struct wl_proxy *) window->video_surface,
|
|
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);
|
|
|
|
window->area_viewport = wl_scaler_get_viewport (display->scaler,
|
|
window->area_surface);
|
|
window->video_viewport = wl_scaler_get_viewport (display->scaler,
|
|
window->video_surface);
|
|
|
|
/* draw the area_subsurface */
|
|
gst_video_info_set_format (&info,
|
|
/* we want WL_SHM_FORMAT_XRGB8888 */
|
|
#if G_BYTE_ORDER == G_BIG_ENDIAN
|
|
GST_VIDEO_FORMAT_xRGB,
|
|
#else
|
|
GST_VIDEO_FORMAT_BGRx,
|
|
#endif
|
|
1, 1);
|
|
|
|
buf = gst_buffer_new_allocate (gst_wl_shm_allocator_get (), info.size, NULL);
|
|
gst_buffer_map (buf, &mapinfo, GST_MAP_WRITE);
|
|
*((guint32 *) mapinfo.data) = 0; /* paint it black */
|
|
gst_buffer_unmap (buf, &mapinfo);
|
|
wlbuf =
|
|
gst_wl_shm_memory_construct_wl_buffer (gst_buffer_peek_memory (buf, 0),
|
|
display, &info);
|
|
gwlbuf = gst_buffer_add_wl_buffer (buf, wlbuf, display);
|
|
gst_wl_buffer_attach (gwlbuf, window->area_surface);
|
|
|
|
/* at this point, the GstWlBuffer keeps the buffer
|
|
* alive and will free it on wl_buffer::release */
|
|
gst_buffer_unref (buf);
|
|
|
|
/* 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;
|
|
}
|
|
|
|
GstWlWindow *
|
|
gst_wl_window_new_toplevel (GstWlDisplay * display, const GstVideoInfo * info)
|
|
{
|
|
GstWlWindow *window;
|
|
gint width;
|
|
|
|
window = gst_wl_window_new_internal (display);
|
|
|
|
/* go toplevel */
|
|
window->shell_surface = wl_shell_get_shell_surface (display->shell,
|
|
window->area_surface);
|
|
|
|
if (window->shell_surface) {
|
|
wl_shell_surface_add_listener (window->shell_surface,
|
|
&shell_surface_listener, window);
|
|
wl_shell_surface_set_toplevel (window->shell_surface);
|
|
} else {
|
|
GST_ERROR ("Unable to get wl_shell_surface");
|
|
|
|
g_object_unref (window);
|
|
return NULL;
|
|
}
|
|
|
|
/* set the initial size to be the same as the reported video size */
|
|
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;
|
|
}
|
|
|
|
GstWlWindow *
|
|
gst_wl_window_new_in_surface (GstWlDisplay * display,
|
|
struct wl_surface * parent)
|
|
{
|
|
GstWlWindow *window;
|
|
window = gst_wl_window_new_internal (display);
|
|
|
|
/* embed in parent */
|
|
window->area_subsurface =
|
|
wl_subcompositor_get_subsurface (display->subcompositor,
|
|
window->area_surface, parent);
|
|
wl_subsurface_set_desync (window->area_subsurface);
|
|
|
|
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;
|
|
}
|
|
|
|
gboolean
|
|
gst_wl_window_is_toplevel (GstWlWindow * window)
|
|
{
|
|
g_return_val_if_fail (window != NULL, FALSE);
|
|
|
|
return (window->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;
|
|
gst_video_sink_center_rect (src, dst, &res, TRUE);
|
|
|
|
wl_subsurface_set_position (window->video_subsurface, res.x, res.y);
|
|
wl_viewport_set_destination (window->video_viewport, res.w, res.h);
|
|
|
|
if (commit) {
|
|
wl_surface_damage (window->video_surface, 0, 0, res.w, res.h);
|
|
wl_surface_commit (window->video_surface);
|
|
}
|
|
|
|
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->surface_width = res.w;
|
|
window->surface_height = res.h;
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
if (G_LIKELY (buffer))
|
|
gst_wl_buffer_attach (buffer, window->video_surface);
|
|
else
|
|
wl_surface_attach (window->video_surface, NULL, 0, 0);
|
|
|
|
wl_surface_damage (window->video_surface, 0, 0, window->surface_width,
|
|
window->surface_height);
|
|
wl_surface_commit (window->video_surface);
|
|
|
|
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, 0, 0, window->render_rectangle.w,
|
|
window->render_rectangle.h);
|
|
wl_surface_commit (window->area_surface);
|
|
wl_subsurface_set_desync (window->video_subsurface);
|
|
}
|
|
|
|
wl_display_flush (window->display->display);
|
|
}
|
|
|
|
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 */
|
|
wl_viewport_set_destination (window->area_viewport, w, h);
|
|
|
|
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, 0, 0, w, h);
|
|
wl_surface_commit (window->area_surface);
|
|
|
|
if (window->video_width != 0)
|
|
wl_subsurface_set_desync (window->video_subsurface);
|
|
}
|