mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-25 16:48:11 +00:00
waylandsink: split window-related code out to a new GstWlWindow class
GstWlWindow also has API ready to support subsurfaces.
This commit is contained in:
parent
05f0842572
commit
ae6aebd9d5
5 changed files with 249 additions and 92 deletions
|
@ -1,7 +1,7 @@
|
||||||
plugin_LTLIBRARIES = libgstwaylandsink.la
|
plugin_LTLIBRARIES = libgstwaylandsink.la
|
||||||
|
|
||||||
libgstwaylandsink_la_SOURCES = gstwaylandsink.c waylandpool.c wldisplay.c \
|
libgstwaylandsink_la_SOURCES = gstwaylandsink.c waylandpool.c \
|
||||||
wlvideoformat.c
|
wldisplay.c wlwindow.c wlvideoformat.c
|
||||||
libgstwaylandsink_la_CFLAGS = $(GST_CFLAGS) $(GST_PLUGINS_BASE_CFLAGS) \
|
libgstwaylandsink_la_CFLAGS = $(GST_CFLAGS) $(GST_PLUGINS_BASE_CFLAGS) \
|
||||||
$(WAYLAND_CFLAGS)
|
$(WAYLAND_CFLAGS)
|
||||||
libgstwaylandsink_la_LIBADD = $(GST_PLUGINS_BASE_LIBS) \
|
libgstwaylandsink_la_LIBADD = $(GST_PLUGINS_BASE_LIBS) \
|
||||||
|
@ -10,4 +10,5 @@ libgstwaylandsink_la_LIBADD = $(GST_PLUGINS_BASE_LIBS) \
|
||||||
libgstwaylandsink_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)
|
libgstwaylandsink_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)
|
||||||
libgstwaylandsink_la_LIBTOOLFLAGS = $(GST_PLUGIN_LIBTOOLFLAGS)
|
libgstwaylandsink_la_LIBTOOLFLAGS = $(GST_PLUGIN_LIBTOOLFLAGS)
|
||||||
|
|
||||||
noinst_HEADERS = gstwaylandsink.h waylandpool.h wldisplay.h wlvideoformat.h
|
noinst_HEADERS = gstwaylandsink.h waylandpool.h \
|
||||||
|
wldisplay.h wlwindow.h wlvideoformat.h
|
||||||
|
|
|
@ -94,8 +94,6 @@ static gboolean gst_wayland_sink_render (GstBaseSink * bsink,
|
||||||
|
|
||||||
static void frame_redraw_callback (void *data,
|
static void frame_redraw_callback (void *data,
|
||||||
struct wl_callback *callback, uint32_t time);
|
struct wl_callback *callback, uint32_t time);
|
||||||
static void create_window (GstWaylandSink * sink, GstWlDisplay * display,
|
|
||||||
int width, int height);
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_wayland_sink_class_init (GstWaylandSinkClass * klass)
|
gst_wayland_sink_class_init (GstWaylandSinkClass * klass)
|
||||||
|
@ -176,18 +174,6 @@ gst_wayland_sink_set_property (GObject * object,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
destroy_window (struct window *window)
|
|
||||||
{
|
|
||||||
if (window->shell_surface)
|
|
||||||
wl_shell_surface_destroy (window->shell_surface);
|
|
||||||
|
|
||||||
if (window->surface)
|
|
||||||
wl_surface_destroy (window->surface);
|
|
||||||
|
|
||||||
free (window);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_wayland_sink_finalize (GObject * object)
|
gst_wayland_sink_finalize (GObject * object)
|
||||||
{
|
{
|
||||||
|
@ -196,7 +182,7 @@ gst_wayland_sink_finalize (GObject * object)
|
||||||
GST_DEBUG_OBJECT (sink, "Finalizing the sink..");
|
GST_DEBUG_OBJECT (sink, "Finalizing the sink..");
|
||||||
|
|
||||||
if (sink->window)
|
if (sink->window)
|
||||||
destroy_window (sink->window);
|
g_object_unref (sink->window);
|
||||||
if (sink->display)
|
if (sink->display)
|
||||||
g_object_unref (sink->display);
|
g_object_unref (sink->display);
|
||||||
|
|
||||||
|
@ -289,62 +275,6 @@ config_failed:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
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
|
|
||||||
create_window (GstWaylandSink * sink, GstWlDisplay * display, int width,
|
|
||||||
int height)
|
|
||||||
{
|
|
||||||
struct window *window;
|
|
||||||
|
|
||||||
if (sink->window)
|
|
||||||
return;
|
|
||||||
|
|
||||||
g_mutex_lock (&sink->wayland_lock);
|
|
||||||
|
|
||||||
window = malloc (sizeof *window);
|
|
||||||
window->width = width;
|
|
||||||
window->height = height;
|
|
||||||
|
|
||||||
window->surface = wl_compositor_create_surface (display->compositor);
|
|
||||||
|
|
||||||
window->shell_surface = wl_shell_get_shell_surface (display->shell,
|
|
||||||
window->surface);
|
|
||||||
|
|
||||||
g_return_if_fail (window->shell_surface);
|
|
||||||
|
|
||||||
wl_shell_surface_add_listener (window->shell_surface,
|
|
||||||
&shell_surface_listener, window);
|
|
||||||
|
|
||||||
wl_shell_surface_set_toplevel (window->shell_surface);
|
|
||||||
|
|
||||||
sink->window = window;
|
|
||||||
|
|
||||||
g_mutex_unlock (&sink->wayland_lock);
|
|
||||||
}
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_wayland_sink_start (GstBaseSink * bsink)
|
gst_wayland_sink_start (GstBaseSink * bsink)
|
||||||
{
|
{
|
||||||
|
@ -471,14 +401,13 @@ gst_wayland_sink_render (GstBaseSink * bsink, GstBuffer * buffer)
|
||||||
GstBuffer *to_render;
|
GstBuffer *to_render;
|
||||||
GstWlMeta *meta;
|
GstWlMeta *meta;
|
||||||
GstFlowReturn ret;
|
GstFlowReturn ret;
|
||||||
struct window *window;
|
struct wl_surface *surface;
|
||||||
struct wl_callback *callback;
|
struct wl_callback *callback;
|
||||||
|
|
||||||
GST_LOG_OBJECT (sink, "render buffer %p", buffer);
|
GST_LOG_OBJECT (sink, "render buffer %p", buffer);
|
||||||
if (!sink->window)
|
if (!sink->window)
|
||||||
create_window (sink, sink->display, sink->video_width, sink->video_height);
|
sink->window = gst_wl_window_new_toplevel (sink->display, sink->video_width,
|
||||||
|
sink->video_height);
|
||||||
window = sink->window;
|
|
||||||
|
|
||||||
meta = gst_buffer_get_wl_meta (buffer);
|
meta = gst_buffer_get_wl_meta (buffer);
|
||||||
|
|
||||||
|
@ -513,11 +442,13 @@ gst_wayland_sink_render (GstBaseSink * bsink, GstBuffer * buffer)
|
||||||
|
|
||||||
gst_video_sink_center_rect (src, dst, &res, FALSE);
|
gst_video_sink_center_rect (src, dst, &res, FALSE);
|
||||||
|
|
||||||
wl_surface_attach (sink->window->surface, meta->wbuffer, 0, 0);
|
surface = gst_wl_window_get_wl_surface (sink->window);
|
||||||
wl_surface_damage (sink->window->surface, 0, 0, res.w, res.h);
|
|
||||||
callback = wl_surface_frame (window->surface);
|
wl_surface_attach (surface, meta->wbuffer, 0, 0);
|
||||||
wl_callback_add_listener (callback, &frame_callback_listener, window);
|
wl_surface_damage (surface, 0, 0, res.w, res.h);
|
||||||
wl_surface_commit (window->surface);
|
callback = wl_surface_frame (surface);
|
||||||
|
wl_callback_add_listener (callback, &frame_callback_listener, NULL);
|
||||||
|
wl_surface_commit (surface);
|
||||||
wl_display_flush (sink->display->display);
|
wl_display_flush (sink->display->display);
|
||||||
|
|
||||||
if (buffer != to_render)
|
if (buffer != to_render)
|
||||||
|
|
|
@ -43,6 +43,7 @@
|
||||||
#include <wayland-client.h>
|
#include <wayland-client.h>
|
||||||
|
|
||||||
#include "wldisplay.h"
|
#include "wldisplay.h"
|
||||||
|
#include "wlwindow.h"
|
||||||
|
|
||||||
G_BEGIN_DECLS
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
@ -59,13 +60,6 @@ G_BEGIN_DECLS
|
||||||
#define GST_WAYLAND_SINK_GET_CLASS(inst) \
|
#define GST_WAYLAND_SINK_GET_CLASS(inst) \
|
||||||
(G_TYPE_INSTANCE_GET_CLASS ((inst), GST_TYPE_WAYLAND_SINK, GstWaylandSinkClass))
|
(G_TYPE_INSTANCE_GET_CLASS ((inst), GST_TYPE_WAYLAND_SINK, GstWaylandSinkClass))
|
||||||
|
|
||||||
struct window
|
|
||||||
{
|
|
||||||
int width, height;
|
|
||||||
struct wl_surface *surface;
|
|
||||||
struct wl_shell_surface *shell_surface;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef struct _GstWaylandSink GstWaylandSink;
|
typedef struct _GstWaylandSink GstWaylandSink;
|
||||||
typedef struct _GstWaylandSinkClass GstWaylandSinkClass;
|
typedef struct _GstWaylandSinkClass GstWaylandSinkClass;
|
||||||
|
|
||||||
|
@ -78,8 +72,7 @@ struct _GstWaylandSink
|
||||||
GstVideoSink parent;
|
GstVideoSink parent;
|
||||||
|
|
||||||
GstWlDisplay *display;
|
GstWlDisplay *display;
|
||||||
struct window *window;
|
GstWlWindow *window;
|
||||||
|
|
||||||
GstBufferPool *pool;
|
GstBufferPool *pool;
|
||||||
|
|
||||||
GMutex wayland_lock;
|
GMutex wayland_lock;
|
||||||
|
|
167
ext/wayland/wlwindow.c
Normal file
167
ext/wayland/wlwindow.c
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
/* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "wlwindow.h"
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self->own_surface) {
|
||||||
|
wl_surface_destroy (self->surface);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_clear_object (&self->display);
|
||||||
|
|
||||||
|
G_OBJECT_CLASS (gst_wl_window_parent_class)->finalize (gobject);
|
||||||
|
}
|
||||||
|
|
||||||
|
GstWlWindow *
|
||||||
|
gst_wl_window_new_toplevel (GstWlDisplay * display, gint width, gint height)
|
||||||
|
{
|
||||||
|
GstWlWindow *window;
|
||||||
|
|
||||||
|
window = g_object_new (GST_TYPE_WL_WINDOW, NULL);
|
||||||
|
window->display = g_object_ref (display);
|
||||||
|
window->width = width;
|
||||||
|
window->height = height;
|
||||||
|
|
||||||
|
window->surface = wl_compositor_create_surface (display->compositor);
|
||||||
|
window->own_surface = TRUE;
|
||||||
|
|
||||||
|
window->shell_surface = wl_shell_get_shell_surface (display->shell,
|
||||||
|
window->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;
|
||||||
|
}
|
||||||
|
|
||||||
|
return window;
|
||||||
|
}
|
||||||
|
|
||||||
|
GstWlWindow *
|
||||||
|
gst_wl_window_new_from_surface (GstWlDisplay * display,
|
||||||
|
struct wl_surface * surface, gint width, gint height)
|
||||||
|
{
|
||||||
|
GstWlWindow *window;
|
||||||
|
|
||||||
|
g_return_val_if_fail (surface != NULL, NULL);
|
||||||
|
|
||||||
|
window = g_object_new (GST_TYPE_WL_WINDOW, NULL);
|
||||||
|
window->display = g_object_ref (display);
|
||||||
|
window->width = width;
|
||||||
|
window->height = height;
|
||||||
|
|
||||||
|
window->surface = surface;
|
||||||
|
window->own_surface = FALSE;
|
||||||
|
|
||||||
|
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->surface;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
gst_wl_window_get_size (GstWlWindow * window, gint * w, gint * h)
|
||||||
|
{
|
||||||
|
g_return_if_fail (window != NULL);
|
||||||
|
|
||||||
|
if (w)
|
||||||
|
*w = window->width;
|
||||||
|
|
||||||
|
if (h)
|
||||||
|
*h = window->height;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
gst_wl_window_set_size (GstWlWindow * window, gint w, gint h)
|
||||||
|
{
|
||||||
|
g_return_if_fail (window != NULL);
|
||||||
|
|
||||||
|
window->width = w;
|
||||||
|
window->height = h;
|
||||||
|
}
|
65
ext/wayland/wlwindow.h
Normal file
65
ext/wayland/wlwindow.h
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
/* GStreamer Wayland video sink
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __GST_WL_WINDOW_H__
|
||||||
|
#define __GST_WL_WINDOW_H__
|
||||||
|
|
||||||
|
#include "wldisplay.h"
|
||||||
|
|
||||||
|
#define GST_TYPE_WL_WINDOW (gst_wl_window_get_type ())
|
||||||
|
#define GST_WL_WINDOW(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GST_TYPE_WL_WINDOW, GstWlWindow))
|
||||||
|
#define GST_IS_WL_WINDOW(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GST_TYPE_WL_WINDOW))
|
||||||
|
#define GST_WL_WINDOW_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GST_TYPE_WL_WINDOW, GstWlWindowClass))
|
||||||
|
#define GST_IS_WL_WINDOW_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GST_TYPE_WL_WINDOW))
|
||||||
|
#define GST_WL_WINDOW_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GST_TYPE_WL_WINDOW, GstWlWindowClass))
|
||||||
|
|
||||||
|
typedef struct _GstWlWindow GstWlWindow;
|
||||||
|
typedef struct _GstWlWindowClass GstWlWindowClass;
|
||||||
|
|
||||||
|
struct _GstWlWindow
|
||||||
|
{
|
||||||
|
GObject parent_instance;
|
||||||
|
|
||||||
|
GstWlDisplay *display;
|
||||||
|
struct wl_surface *surface;
|
||||||
|
struct wl_shell_surface *shell_surface;
|
||||||
|
gint width, height;
|
||||||
|
gboolean own_surface;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct _GstWlWindowClass
|
||||||
|
{
|
||||||
|
GObjectClass parent_class;
|
||||||
|
};
|
||||||
|
|
||||||
|
GType gst_wl_window_get_type (void);
|
||||||
|
|
||||||
|
GstWlWindow *gst_wl_window_new_toplevel (GstWlDisplay * display,
|
||||||
|
gint width, gint height);
|
||||||
|
GstWlWindow *gst_wl_window_new_from_surface (GstWlDisplay * display,
|
||||||
|
struct wl_surface * surface, gint width, gint height);
|
||||||
|
|
||||||
|
GstWlDisplay *gst_wl_window_get_display (GstWlWindow * window);
|
||||||
|
struct wl_surface *gst_wl_window_get_wl_surface (GstWlWindow * window);
|
||||||
|
|
||||||
|
void gst_wl_window_get_size (GstWlWindow * window, gint * w, gint * h);
|
||||||
|
void gst_wl_window_set_size (GstWlWindow * window, gint w, gint h);
|
||||||
|
|
||||||
|
#endif /* __GST_WL_WINDOW_H__ */
|
Loading…
Reference in a new issue