mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-07 07:55:41 +00:00
f341619a4d
That would call into gst_info_describe_stream(), which takes the same mutex a second time and then deadlocks. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/552>
570 lines
14 KiB
C
570 lines
14 KiB
C
/* GStreamer
|
|
*
|
|
* Copyright (C) 2015 Centricular Ltd
|
|
* @author: Edward Hervey <edward@centricular.com>
|
|
* @author: Jan Schmidt <jan@centricular.com>
|
|
*
|
|
* gststreams.c: GstStream and GstStreamCollection object and methods
|
|
*
|
|
* 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 St, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*
|
|
* MT safe.
|
|
*/
|
|
|
|
/**
|
|
* SECTION:gststreams
|
|
* @title: GstStreams
|
|
* @short_description: Base class for stream objects
|
|
*
|
|
* A #GstStream is a high-level object defining a stream of data which is, or
|
|
* can be, present in a #GstPipeline.
|
|
*
|
|
* It is defined by a unique identifier, a "Stream ID". A #GstStream does not
|
|
* automatically imply the stream is present within a pipeline or element.
|
|
*
|
|
* Any element that can introduce new streams in a pipeline should create the
|
|
* appropriate #GstStream object, and can convey that object via the
|
|
* %GST_EVENT_STREAM_START event and/or the #GstStreamCollection.
|
|
*
|
|
* Elements that do not modify the nature of the stream can add extra information
|
|
* on it (such as enrich the #GstCaps, or #GstTagList). This is typically done
|
|
* by parsing elements.
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
|
|
#include "gst_private.h"
|
|
|
|
#include "gstenumtypes.h"
|
|
#include "gstevent.h"
|
|
#include "gststreams.h"
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (streams_debug);
|
|
#define GST_CAT_DEFAULT streams_debug
|
|
|
|
struct _GstStreamPrivate
|
|
{
|
|
GstStreamFlags flags;
|
|
GstStreamType type;
|
|
GstTagList *tags;
|
|
GstCaps *caps;
|
|
};
|
|
|
|
/* stream signals and properties */
|
|
enum
|
|
{
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_STREAM_ID,
|
|
PROP_STREAM_FLAGS,
|
|
PROP_STREAM_TYPE,
|
|
PROP_TAGS,
|
|
PROP_CAPS,
|
|
PROP_LAST
|
|
};
|
|
|
|
static GParamSpec *gst_stream_pspecs[PROP_LAST] = { 0 };
|
|
|
|
#if 0
|
|
static guint gst_stream_signals[LAST_SIGNAL] = { 0 };
|
|
#endif
|
|
|
|
static void gst_stream_finalize (GObject * object);
|
|
|
|
static void gst_stream_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec);
|
|
static void gst_stream_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec);
|
|
|
|
#define _do_init \
|
|
{ \
|
|
GST_DEBUG_CATEGORY_INIT (streams_debug, "streams", GST_DEBUG_BOLD, \
|
|
"debugging info for the stream and stream collection objects"); \
|
|
\
|
|
}
|
|
|
|
#define gst_stream_parent_class parent_class
|
|
G_DEFINE_TYPE_WITH_CODE (GstStream, gst_stream, GST_TYPE_OBJECT,
|
|
G_ADD_PRIVATE (GstStream) _do_init);
|
|
|
|
static void
|
|
gst_stream_class_init (GstStreamClass * klass)
|
|
{
|
|
GObjectClass *gobject_class;
|
|
|
|
gobject_class = (GObjectClass *) klass;
|
|
|
|
gobject_class->set_property = gst_stream_set_property;
|
|
gobject_class->get_property = gst_stream_get_property;
|
|
|
|
/**
|
|
* GstStream:stream-id:
|
|
*
|
|
* The unique identifier of the #GstStream. Can only be set at construction
|
|
* time.
|
|
*/
|
|
g_object_class_install_property (gobject_class, PROP_STREAM_ID,
|
|
g_param_spec_string ("stream-id", "Stream ID",
|
|
"The stream ID of the stream",
|
|
NULL,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* GstStream:flags:
|
|
*
|
|
* The #GstStreamFlags of the #GstStream. Can only be set at construction time.
|
|
**/
|
|
gst_stream_pspecs[PROP_STREAM_FLAGS] =
|
|
g_param_spec_flags ("stream-flags", "Stream Flags", "The stream flags",
|
|
GST_TYPE_STREAM_FLAGS, GST_STREAM_FLAG_NONE,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
|
|
g_object_class_install_property (gobject_class, PROP_STREAM_FLAGS,
|
|
gst_stream_pspecs[PROP_STREAM_FLAGS]);
|
|
|
|
/**
|
|
* GstStream:stream-type:
|
|
*
|
|
* The #GstStreamType of the #GstStream. Can only be set at construction time.
|
|
**/
|
|
gst_stream_pspecs[PROP_STREAM_TYPE] =
|
|
g_param_spec_flags ("stream-type", "Stream Type", "The type of stream",
|
|
GST_TYPE_STREAM_TYPE, GST_STREAM_TYPE_UNKNOWN,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
|
|
g_object_class_install_property (gobject_class, PROP_STREAM_TYPE,
|
|
gst_stream_pspecs[PROP_STREAM_TYPE]);
|
|
|
|
/**
|
|
* GstStream:caps:
|
|
*
|
|
* The #GstCaps of the #GstStream.
|
|
**/
|
|
gst_stream_pspecs[PROP_CAPS] =
|
|
g_param_spec_boxed ("caps", "Caps", "The caps of the stream",
|
|
GST_TYPE_CAPS, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
|
|
g_object_class_install_property (gobject_class, PROP_CAPS,
|
|
gst_stream_pspecs[PROP_CAPS]);
|
|
|
|
/**
|
|
* GstStream:tags:
|
|
*
|
|
* The #GstTagList of the #GstStream.
|
|
**/
|
|
gst_stream_pspecs[PROP_TAGS] =
|
|
g_param_spec_boxed ("tags", "Tags", "The tags of the stream",
|
|
GST_TYPE_TAG_LIST, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
|
|
g_object_class_install_property (gobject_class, PROP_TAGS,
|
|
gst_stream_pspecs[PROP_TAGS]);
|
|
|
|
gobject_class->finalize = gst_stream_finalize;
|
|
}
|
|
|
|
static void
|
|
gst_stream_init (GstStream * stream)
|
|
{
|
|
stream->priv = gst_stream_get_instance_private (stream);
|
|
stream->priv->type = GST_STREAM_TYPE_UNKNOWN;
|
|
}
|
|
|
|
static void
|
|
gst_stream_finalize (GObject * object)
|
|
{
|
|
GstStream *stream = GST_STREAM_CAST (object);
|
|
|
|
gst_mini_object_replace ((GstMiniObject **) & stream->priv->tags,
|
|
(GstMiniObject *) NULL);
|
|
gst_caps_replace (&stream->priv->caps, NULL);
|
|
g_free ((gchar *) stream->stream_id);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
/**
|
|
* gst_stream_new:
|
|
* @stream_id: (allow-none): the id for the new stream. If %NULL,
|
|
* a new one will be automatically generated
|
|
* @caps: (allow-none) (transfer none): the #GstCaps of the stream
|
|
* @type: the #GstStreamType of the stream
|
|
* @flags: the #GstStreamFlags of the stream
|
|
*
|
|
* Create a new #GstStream for the given @stream_id, @caps, @type
|
|
* and @flags
|
|
*
|
|
* Returns: (transfer full): The new #GstStream
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
GstStream *
|
|
gst_stream_new (const gchar * stream_id, GstCaps * caps, GstStreamType type,
|
|
GstStreamFlags flags)
|
|
{
|
|
GstStream *stream;
|
|
|
|
stream = g_object_new (GST_TYPE_STREAM, "stream-id", stream_id, "caps", caps,
|
|
"stream-type", type, "stream-flags", flags, NULL);
|
|
|
|
/* Clear floating flag */
|
|
gst_object_ref_sink (stream);
|
|
|
|
return stream;
|
|
}
|
|
|
|
static void
|
|
gst_stream_set_stream_id (GstStream * stream, const gchar * stream_id)
|
|
{
|
|
g_return_if_fail (GST_IS_STREAM (stream));
|
|
|
|
GST_OBJECT_LOCK (stream);
|
|
g_assert (stream->stream_id == NULL);
|
|
if (stream_id) {
|
|
stream->stream_id = g_strdup (stream_id);
|
|
} else {
|
|
/* Create a random stream_id if NULL */
|
|
stream->stream_id =
|
|
g_strdup_printf ("%08x%08x%08x%08x", g_random_int (), g_random_int (),
|
|
g_random_int (), g_random_int ());
|
|
}
|
|
|
|
GST_OBJECT_UNLOCK (stream);
|
|
|
|
if (!stream_id)
|
|
GST_FIXME_OBJECT (stream, "Created random stream-id, consider "
|
|
"implementing a deterministic way of creating a stream-id");
|
|
}
|
|
|
|
/**
|
|
* gst_stream_get_stream_id:
|
|
* @stream: a #GstStream
|
|
*
|
|
* Returns the stream ID of @stream.
|
|
*
|
|
* Returns: (transfer none) (nullable): the stream ID of @stream. Only valid
|
|
* during the lifetime of @stream.
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
const gchar *
|
|
gst_stream_get_stream_id (GstStream * stream)
|
|
{
|
|
g_return_val_if_fail (GST_IS_STREAM (stream), NULL);
|
|
|
|
return stream->stream_id;
|
|
}
|
|
|
|
/**
|
|
* gst_stream_set_stream_flags:
|
|
* @stream: a #GstStream
|
|
* @flags: the flags to set on @stream
|
|
*
|
|
* Set the @flags for the @stream.
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
void
|
|
gst_stream_set_stream_flags (GstStream * stream, GstStreamFlags flags)
|
|
{
|
|
g_return_if_fail (GST_IS_STREAM (stream));
|
|
|
|
GST_OBJECT_LOCK (stream);
|
|
stream->priv->flags = flags;
|
|
GST_OBJECT_UNLOCK (stream);
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (stream),
|
|
gst_stream_pspecs[PROP_STREAM_FLAGS]);
|
|
}
|
|
|
|
/**
|
|
* gst_stream_get_stream_flags:
|
|
* @stream: a #GstStream
|
|
*
|
|
* Retrieve the current stream flags for @stream
|
|
*
|
|
* Returns: The #GstStreamFlags for @stream
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
GstStreamFlags
|
|
gst_stream_get_stream_flags (GstStream * stream)
|
|
{
|
|
GstStreamFlags res;
|
|
|
|
g_return_val_if_fail (GST_IS_STREAM (stream), GST_STREAM_FLAG_NONE);
|
|
|
|
GST_OBJECT_LOCK (stream);
|
|
res = stream->priv->flags;
|
|
GST_OBJECT_UNLOCK (stream);
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* gst_stream_set_stream_type:
|
|
* @stream: a #GstStream
|
|
* @stream_type: the type to set on @stream
|
|
*
|
|
* Set the stream type of @stream
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
void
|
|
gst_stream_set_stream_type (GstStream * stream, GstStreamType stream_type)
|
|
{
|
|
g_return_if_fail (GST_IS_STREAM (stream));
|
|
|
|
GST_OBJECT_LOCK (stream);
|
|
stream->priv->type = stream_type;
|
|
GST_OBJECT_UNLOCK (stream);
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (stream),
|
|
gst_stream_pspecs[PROP_STREAM_TYPE]);
|
|
}
|
|
|
|
/**
|
|
* gst_stream_get_stream_type:
|
|
* @stream: a #GstStream
|
|
*
|
|
* Retrieve the stream type for @stream
|
|
*
|
|
* Returns: The #GstStreamType for @stream
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
GstStreamType
|
|
gst_stream_get_stream_type (GstStream * stream)
|
|
{
|
|
GstStreamType res;
|
|
|
|
g_return_val_if_fail (GST_IS_STREAM (stream), GST_STREAM_TYPE_UNKNOWN);
|
|
|
|
GST_OBJECT_LOCK (stream);
|
|
res = stream->priv->type;
|
|
GST_OBJECT_UNLOCK (stream);
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* gst_stream_set_tags:
|
|
* @stream: a #GstStream
|
|
* @tags: (transfer none) (allow-none): a #GstTagList
|
|
*
|
|
* Set the tags for the #GstStream
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
void
|
|
gst_stream_set_tags (GstStream * stream, GstTagList * tags)
|
|
{
|
|
gboolean notify = FALSE;
|
|
|
|
g_return_if_fail (GST_IS_STREAM (stream));
|
|
|
|
GST_OBJECT_LOCK (stream);
|
|
if (stream->priv->tags == NULL || tags == NULL
|
|
|| !gst_tag_list_is_equal (stream->priv->tags, tags)) {
|
|
gst_mini_object_replace ((GstMiniObject **) & stream->priv->tags,
|
|
(GstMiniObject *) tags);
|
|
notify = TRUE;
|
|
}
|
|
GST_OBJECT_UNLOCK (stream);
|
|
|
|
if (notify)
|
|
g_object_notify_by_pspec (G_OBJECT (stream), gst_stream_pspecs[PROP_TAGS]);
|
|
}
|
|
|
|
/**
|
|
* gst_stream_get_tags:
|
|
* @stream: a #GstStream
|
|
*
|
|
* Retrieve the tags for @stream, if any
|
|
*
|
|
* Returns: (transfer full) (nullable): The #GstTagList for @stream
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
GstTagList *
|
|
gst_stream_get_tags (GstStream * stream)
|
|
{
|
|
GstTagList *res = NULL;
|
|
|
|
g_return_val_if_fail (GST_IS_STREAM (stream), NULL);
|
|
|
|
GST_OBJECT_LOCK (stream);
|
|
if (stream->priv->tags)
|
|
res = gst_tag_list_ref (stream->priv->tags);
|
|
GST_OBJECT_UNLOCK (stream);
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* gst_stream_set_caps:
|
|
* @stream: a #GstStream
|
|
* @caps: (transfer none) (allow-none): a #GstCaps
|
|
*
|
|
* Set the caps for the #GstStream
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
void
|
|
gst_stream_set_caps (GstStream * stream, GstCaps * caps)
|
|
{
|
|
gboolean notify = FALSE;
|
|
|
|
g_return_if_fail (GST_IS_STREAM (stream));
|
|
|
|
GST_OBJECT_LOCK (stream);
|
|
if (stream->priv->caps == NULL || (caps
|
|
&& !gst_caps_is_equal (stream->priv->caps, caps))) {
|
|
gst_caps_replace (&stream->priv->caps, caps);
|
|
notify = TRUE;
|
|
}
|
|
GST_OBJECT_UNLOCK (stream);
|
|
|
|
if (notify)
|
|
g_object_notify_by_pspec (G_OBJECT (stream), gst_stream_pspecs[PROP_CAPS]);
|
|
}
|
|
|
|
|
|
/**
|
|
* gst_stream_get_caps:
|
|
* @stream: a #GstStream
|
|
*
|
|
* Retrieve the caps for @stream, if any
|
|
*
|
|
* Returns: (transfer full) (nullable): The #GstCaps for @stream
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
GstCaps *
|
|
gst_stream_get_caps (GstStream * stream)
|
|
{
|
|
GstCaps *res = NULL;
|
|
|
|
g_return_val_if_fail (GST_IS_STREAM (stream), NULL);
|
|
|
|
GST_OBJECT_LOCK (stream);
|
|
if (stream->priv->caps)
|
|
res = gst_caps_ref (stream->priv->caps);
|
|
GST_OBJECT_UNLOCK (stream);
|
|
|
|
return res;
|
|
}
|
|
|
|
static void
|
|
gst_stream_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstStream *stream;
|
|
|
|
stream = GST_STREAM_CAST (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_STREAM_ID:
|
|
gst_stream_set_stream_id (stream, g_value_get_string (value));
|
|
break;
|
|
case PROP_STREAM_FLAGS:
|
|
GST_OBJECT_LOCK (stream);
|
|
stream->priv->flags = g_value_get_flags (value);
|
|
GST_OBJECT_UNLOCK (stream);
|
|
break;
|
|
case PROP_STREAM_TYPE:
|
|
GST_OBJECT_LOCK (stream);
|
|
stream->priv->type = g_value_get_flags (value);
|
|
GST_OBJECT_UNLOCK (stream);
|
|
break;
|
|
case PROP_TAGS:
|
|
GST_OBJECT_LOCK (stream);
|
|
gst_mini_object_replace ((GstMiniObject **) & stream->priv->tags,
|
|
(GstMiniObject *) g_value_get_boxed (value));
|
|
GST_OBJECT_UNLOCK (stream);
|
|
break;
|
|
case PROP_CAPS:
|
|
GST_OBJECT_LOCK (stream);
|
|
gst_mini_object_replace ((GstMiniObject **) & stream->priv->caps,
|
|
(GstMiniObject *) g_value_get_boxed (value));
|
|
GST_OBJECT_UNLOCK (stream);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_stream_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstStream *stream;
|
|
|
|
stream = GST_STREAM_CAST (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_STREAM_ID:
|
|
g_value_set_string (value, gst_stream_get_stream_id (stream));
|
|
break;
|
|
case PROP_STREAM_FLAGS:
|
|
g_value_set_flags (value, gst_stream_get_stream_flags (stream));
|
|
break;
|
|
case PROP_STREAM_TYPE:
|
|
g_value_set_flags (value, gst_stream_get_stream_type (stream));
|
|
break;
|
|
case PROP_TAGS:
|
|
g_value_take_boxed (value, gst_stream_get_tags (stream));
|
|
break;
|
|
case PROP_CAPS:
|
|
g_value_take_boxed (value, gst_stream_get_caps (stream));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gst_stream_type_get_name:
|
|
* @stype: a #GstStreamType
|
|
*
|
|
* Get a descriptive string for a given #GstStreamType
|
|
*
|
|
* Returns: (nullable): A string describing the stream type
|
|
*
|
|
* Since: 1.10
|
|
*/
|
|
const gchar *
|
|
gst_stream_type_get_name (GstStreamType stype)
|
|
{
|
|
/* FIXME : Make this more flexible */
|
|
switch (stype) {
|
|
case GST_STREAM_TYPE_UNKNOWN:
|
|
return "unknown";
|
|
case GST_STREAM_TYPE_AUDIO:
|
|
return "audio";
|
|
case GST_STREAM_TYPE_VIDEO:
|
|
return "video";
|
|
case GST_STREAM_TYPE_CONTAINER:
|
|
return "container";
|
|
case GST_STREAM_TYPE_TEXT:
|
|
return "text";
|
|
default:
|
|
return NULL;
|
|
}
|
|
|
|
return NULL;
|
|
}
|