mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-19 13:55:41 +00:00
waylandsink: protect access to the display with a new display_lock
Access is protected only for setting/creating/destroying the display handle. set_caps() for example is not protected because it cannot be called before changing state to READY, at which point there will be a display handle available and which cannot change by any thread at that point
This commit is contained in:
parent
c323171dc1
commit
d4d46fa43a
2 changed files with 42 additions and 14 deletions
|
@ -162,6 +162,7 @@ gst_wayland_sink_class_init (GstWaylandSinkClass * klass)
|
||||||
static void
|
static void
|
||||||
gst_wayland_sink_init (GstWaylandSink * sink)
|
gst_wayland_sink_init (GstWaylandSink * sink)
|
||||||
{
|
{
|
||||||
|
g_mutex_init (&sink->display_lock);
|
||||||
g_mutex_init (&sink->render_lock);
|
g_mutex_init (&sink->render_lock);
|
||||||
g_cond_init (&sink->render_cond);
|
g_cond_init (&sink->render_cond);
|
||||||
}
|
}
|
||||||
|
@ -223,12 +224,32 @@ gst_wayland_sink_finalize (GObject * object)
|
||||||
if (sink->display_name)
|
if (sink->display_name)
|
||||||
g_free (sink->display_name);
|
g_free (sink->display_name);
|
||||||
|
|
||||||
|
g_mutex_clear (&sink->display_lock);
|
||||||
g_mutex_clear (&sink->render_lock);
|
g_mutex_clear (&sink->render_lock);
|
||||||
g_cond_clear (&sink->render_cond);
|
g_cond_clear (&sink->render_cond);
|
||||||
|
|
||||||
G_OBJECT_CLASS (parent_class)->finalize (object);
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* must be called with the display_lock */
|
||||||
|
static void
|
||||||
|
gst_wayland_sink_set_display_from_context (GstWaylandSink * sink,
|
||||||
|
GstContext * context)
|
||||||
|
{
|
||||||
|
struct wl_display *display;
|
||||||
|
GError *error = NULL;
|
||||||
|
|
||||||
|
display = gst_wayland_display_handle_context_get_handle (context);
|
||||||
|
sink->display = gst_wl_display_new_existing (display, FALSE, &error);
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
GST_ELEMENT_WARNING (sink, RESOURCE, OPEN_READ_WRITE,
|
||||||
|
("Could not set display handle"),
|
||||||
|
("Failed to use the external wayland display: '%s'", error->message));
|
||||||
|
g_error_free (error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_wayland_sink_find_display (GstWaylandSink * sink)
|
gst_wayland_sink_find_display (GstWaylandSink * sink)
|
||||||
{
|
{
|
||||||
|
@ -238,12 +259,14 @@ gst_wayland_sink_find_display (GstWaylandSink * sink)
|
||||||
GError *error = NULL;
|
GError *error = NULL;
|
||||||
gboolean ret = TRUE;
|
gboolean ret = TRUE;
|
||||||
|
|
||||||
|
g_mutex_lock (&sink->display_lock);
|
||||||
|
|
||||||
if (!sink->display) {
|
if (!sink->display) {
|
||||||
/* first query upstream for the needed display handle */
|
/* first query upstream for the needed display handle */
|
||||||
query = gst_query_new_context (GST_WAYLAND_DISPLAY_HANDLE_CONTEXT_TYPE);
|
query = gst_query_new_context (GST_WAYLAND_DISPLAY_HANDLE_CONTEXT_TYPE);
|
||||||
if (gst_pad_peer_query (GST_VIDEO_SINK_PAD (sink), query)) {
|
if (gst_pad_peer_query (GST_VIDEO_SINK_PAD (sink), query)) {
|
||||||
gst_query_parse_context (query, &context);
|
gst_query_parse_context (query, &context);
|
||||||
gst_wayland_sink_set_context (GST_ELEMENT (sink), context);
|
gst_wayland_sink_set_display_from_context (sink, context);
|
||||||
}
|
}
|
||||||
gst_query_unref (query);
|
gst_query_unref (query);
|
||||||
|
|
||||||
|
@ -251,9 +274,12 @@ gst_wayland_sink_find_display (GstWaylandSink * sink)
|
||||||
/* now ask the application to set the display handle */
|
/* now ask the application to set the display handle */
|
||||||
msg = gst_message_new_need_context (GST_OBJECT_CAST (sink),
|
msg = gst_message_new_need_context (GST_OBJECT_CAST (sink),
|
||||||
GST_WAYLAND_DISPLAY_HANDLE_CONTEXT_TYPE);
|
GST_WAYLAND_DISPLAY_HANDLE_CONTEXT_TYPE);
|
||||||
|
|
||||||
|
g_mutex_unlock (&sink->display_lock);
|
||||||
gst_element_post_message (GST_ELEMENT_CAST (sink), msg);
|
gst_element_post_message (GST_ELEMENT_CAST (sink), msg);
|
||||||
/* at this point we expect gst_wayland_sink_set_context
|
/* at this point we expect gst_wayland_sink_set_context
|
||||||
* to get called and fill sink->display */
|
* to get called and fill sink->display */
|
||||||
|
g_mutex_lock (&sink->display_lock);
|
||||||
|
|
||||||
if (!sink->display) {
|
if (!sink->display) {
|
||||||
/* if the application didn't set a display, let's create it ourselves */
|
/* if the application didn't set a display, let's create it ourselves */
|
||||||
|
@ -275,6 +301,8 @@ gst_wayland_sink_find_display (GstWaylandSink * sink)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
g_mutex_unlock (&sink->display_lock);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -305,6 +333,7 @@ gst_wayland_sink_change_state (GstElement * element, GstStateChange transition)
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case GST_STATE_CHANGE_READY_TO_NULL:
|
case GST_STATE_CHANGE_READY_TO_NULL:
|
||||||
|
g_mutex_lock (&sink->display_lock);
|
||||||
/* We don't need to keep the display around, unless we are embedded
|
/* We don't need to keep the display around, unless we are embedded
|
||||||
* in another window as a subsurface, in which case we should continue
|
* in another window as a subsurface, in which case we should continue
|
||||||
* to respond to expose() and therefore both the window and the display
|
* to respond to expose() and therefore both the window and the display
|
||||||
|
@ -330,6 +359,7 @@ gst_wayland_sink_change_state (GstElement * element, GstStateChange transition)
|
||||||
g_clear_object (&sink->display);
|
g_clear_object (&sink->display);
|
||||||
g_clear_object (&sink->pool);
|
g_clear_object (&sink->pool);
|
||||||
}
|
}
|
||||||
|
g_mutex_unlock (&sink->display_lock);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
|
@ -345,19 +375,12 @@ gst_wayland_sink_set_context (GstElement * element, GstContext * context)
|
||||||
|
|
||||||
if (gst_context_has_context_type (context,
|
if (gst_context_has_context_type (context,
|
||||||
GST_WAYLAND_DISPLAY_HANDLE_CONTEXT_TYPE)) {
|
GST_WAYLAND_DISPLAY_HANDLE_CONTEXT_TYPE)) {
|
||||||
struct wl_display *display;
|
g_mutex_lock (&sink->display_lock);
|
||||||
GError *error = NULL;
|
if (G_LIKELY (!sink->display))
|
||||||
|
gst_wayland_sink_set_display_from_context (sink, context);
|
||||||
g_clear_object (&sink->display);
|
else
|
||||||
display = gst_wayland_display_handle_context_get_handle (context);
|
GST_WARNING_OBJECT (element, "changing display handle is not supported");
|
||||||
sink->display = gst_wl_display_new_existing (display, FALSE, &error);
|
g_mutex_unlock (&sink->display_lock);
|
||||||
|
|
||||||
if (error) {
|
|
||||||
GST_ELEMENT_WARNING (sink, RESOURCE, OPEN_READ_WRITE,
|
|
||||||
("Could not set display handle"),
|
|
||||||
("Failed to use the external wayland display: '%s'", error->message));
|
|
||||||
g_error_free (error);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (GST_ELEMENT_CLASS (parent_class)->set_context)
|
if (GST_ELEMENT_CLASS (parent_class)->set_context)
|
||||||
|
@ -374,6 +397,8 @@ gst_wayland_sink_get_caps (GstBaseSink * bsink, GstCaps * filter)
|
||||||
|
|
||||||
caps = gst_pad_get_pad_template_caps (GST_VIDEO_SINK_PAD (sink));
|
caps = gst_pad_get_pad_template_caps (GST_VIDEO_SINK_PAD (sink));
|
||||||
|
|
||||||
|
g_mutex_lock (&sink->display_lock);
|
||||||
|
|
||||||
if (sink->display) {
|
if (sink->display) {
|
||||||
GValue list = G_VALUE_INIT;
|
GValue list = G_VALUE_INIT;
|
||||||
GValue value = G_VALUE_INIT;
|
GValue value = G_VALUE_INIT;
|
||||||
|
@ -397,6 +422,8 @@ gst_wayland_sink_get_caps (GstBaseSink * bsink, GstCaps * filter)
|
||||||
GST_DEBUG_OBJECT (sink, "display caps: %" GST_PTR_FORMAT, caps);
|
GST_DEBUG_OBJECT (sink, "display caps: %" GST_PTR_FORMAT, caps);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
g_mutex_unlock (&sink->display_lock);
|
||||||
|
|
||||||
if (filter) {
|
if (filter) {
|
||||||
GstCaps *intersection;
|
GstCaps *intersection;
|
||||||
|
|
||||||
|
|
|
@ -52,6 +52,7 @@ struct _GstWaylandSink
|
||||||
{
|
{
|
||||||
GstVideoSink parent;
|
GstVideoSink parent;
|
||||||
|
|
||||||
|
GMutex display_lock;
|
||||||
GstWlDisplay *display;
|
GstWlDisplay *display;
|
||||||
GstWlWindow *window;
|
GstWlWindow *window;
|
||||||
GstBufferPool *pool;
|
GstBufferPool *pool;
|
||||||
|
|
Loading…
Reference in a new issue