mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-11 09:55:36 +00:00
gst/playback/gstdecodebin.c: Remove unused properties, and add queues between demuxers and decoders so that a lot mor...
Original commit message from CVS: * gst/playback/gstdecodebin.c: (gst_decode_bin_class_init), (gst_decode_bin_init), (close_pad_link), (try_to_link_1): Remove unused properties, and add queues between demuxers and decoders so that a lot more files can preroll properly.
This commit is contained in:
parent
1a4ad8377d
commit
56c6518c55
2 changed files with 58 additions and 87 deletions
|
@ -1,3 +1,10 @@
|
||||||
|
2005-11-27 Edward Hervey <edward@fluendo.com>
|
||||||
|
|
||||||
|
* gst/playback/gstdecodebin.c: (gst_decode_bin_class_init),
|
||||||
|
(gst_decode_bin_init), (close_pad_link), (try_to_link_1):
|
||||||
|
Remove unused properties, and add queues between demuxers and decoders
|
||||||
|
so that a lot more files can preroll properly.
|
||||||
|
|
||||||
2005-11-27 Thomas Vander Stichele <thomas at apestaart dot org>
|
2005-11-27 Thomas Vander Stichele <thomas at apestaart dot org>
|
||||||
|
|
||||||
* gst-libs/gst/net/Makefile.am:
|
* gst-libs/gst/net/Makefile.am:
|
||||||
|
|
|
@ -58,15 +58,12 @@ struct _GstDecodeBin
|
||||||
GstElement *typefind; /* this holds the typefind object */
|
GstElement *typefind; /* this holds the typefind object */
|
||||||
GstElement *fakesink;
|
GstElement *fakesink;
|
||||||
|
|
||||||
gboolean threaded; /* indicating threaded execution is desired */
|
|
||||||
GList *dynamics; /* list of dynamic connections */
|
GList *dynamics; /* list of dynamic connections */
|
||||||
|
|
||||||
GList *factories; /* factories we can use for selecting elements */
|
GList *factories; /* factories we can use for selecting elements */
|
||||||
gint numpads;
|
gint numpads;
|
||||||
gint numwaiting;
|
gint numwaiting;
|
||||||
|
|
||||||
GList *elements; /* elements we added in autoplugging */
|
|
||||||
|
|
||||||
guint have_type_id; /* signal id for the typefind element */
|
guint have_type_id; /* signal id for the typefind element */
|
||||||
|
|
||||||
gboolean shutting_down; /* stop pluggin if we're shutting down */
|
gboolean shutting_down; /* stop pluggin if we're shutting down */
|
||||||
|
@ -84,15 +81,6 @@ struct _GstDecodeBinClass
|
||||||
void (*unknown_type) (GstElement * element, GstPad * pad, GstCaps * caps);
|
void (*unknown_type) (GstElement * element, GstPad * pad, GstCaps * caps);
|
||||||
};
|
};
|
||||||
|
|
||||||
#define DEFAULT_THREADED FALSE
|
|
||||||
|
|
||||||
/* props */
|
|
||||||
enum
|
|
||||||
{
|
|
||||||
ARG_0,
|
|
||||||
ARG_THREADED,
|
|
||||||
};
|
|
||||||
|
|
||||||
/* signals */
|
/* signals */
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
|
@ -119,18 +107,14 @@ static void gst_decode_bin_class_init (GstDecodeBinClass * klass);
|
||||||
static void gst_decode_bin_init (GstDecodeBin * decode_bin);
|
static void gst_decode_bin_init (GstDecodeBin * decode_bin);
|
||||||
static void gst_decode_bin_dispose (GObject * object);
|
static void gst_decode_bin_dispose (GObject * object);
|
||||||
|
|
||||||
static void gst_decode_bin_set_property (GObject * object, guint prop_id,
|
|
||||||
const GValue * value, GParamSpec * spec);
|
|
||||||
static void gst_decode_bin_get_property (GObject * object, guint prop_id,
|
|
||||||
GValue * value, GParamSpec * spec);
|
|
||||||
static GstStateChangeReturn gst_decode_bin_change_state (GstElement * element,
|
static GstStateChangeReturn gst_decode_bin_change_state (GstElement * element,
|
||||||
GstStateChange transition);
|
GstStateChange transition);
|
||||||
|
|
||||||
static void free_dynamics (GstDecodeBin * decode_bin);
|
static void free_dynamics (GstDecodeBin * decode_bin);
|
||||||
static void type_found (GstElement * typefind, guint probability,
|
static void type_found (GstElement * typefind, guint probability,
|
||||||
GstCaps * caps, GstDecodeBin * decode_bin);
|
GstCaps * caps, GstDecodeBin * decode_bin);
|
||||||
static GstElement *try_to_link_1 (GstDecodeBin * decode_bin, GstPad * pad,
|
static GstElement *try_to_link_1 (GstDecodeBin * decode_bin,
|
||||||
GList * factories);
|
GstElement * origelement, GstPad * pad, GList * factories);
|
||||||
static void close_link (GstElement * element, GstDecodeBin * decode_bin);
|
static void close_link (GstElement * element, GstDecodeBin * decode_bin);
|
||||||
static void close_pad_link (GstElement * element, GstPad * pad,
|
static void close_pad_link (GstElement * element, GstPad * pad,
|
||||||
GstCaps * caps, GstDecodeBin * decode_bin, gboolean more);
|
GstCaps * caps, GstDecodeBin * decode_bin, gboolean more);
|
||||||
|
@ -190,13 +174,6 @@ gst_decode_bin_class_init (GstDecodeBinClass * klass)
|
||||||
|
|
||||||
parent_class = g_type_class_ref (gst_bin_get_type ());
|
parent_class = g_type_class_ref (gst_bin_get_type ());
|
||||||
|
|
||||||
gobject_klass->set_property = gst_decode_bin_set_property;
|
|
||||||
gobject_klass->get_property = gst_decode_bin_get_property;
|
|
||||||
|
|
||||||
g_object_class_install_property (gobject_klass, ARG_THREADED,
|
|
||||||
g_param_spec_boolean ("threaded", "Threaded", "Use threads",
|
|
||||||
DEFAULT_THREADED, G_PARAM_READWRITE));
|
|
||||||
|
|
||||||
gst_decode_bin_signals[SIGNAL_NEW_DECODED_PAD] =
|
gst_decode_bin_signals[SIGNAL_NEW_DECODED_PAD] =
|
||||||
g_signal_new ("new-decoded-pad", G_TYPE_FROM_CLASS (klass),
|
g_signal_new ("new-decoded-pad", G_TYPE_FROM_CLASS (klass),
|
||||||
G_SIGNAL_RUN_LAST,
|
G_SIGNAL_RUN_LAST,
|
||||||
|
@ -351,7 +328,6 @@ gst_decode_bin_init (GstDecodeBin * decode_bin)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
decode_bin->threaded = DEFAULT_THREADED;
|
|
||||||
decode_bin->dynamics = NULL;
|
decode_bin->dynamics = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -597,7 +573,7 @@ close_pad_link (GstElement * element, GstPad * pad, GstCaps * caps,
|
||||||
/* no compatible elements, we cannot go on */
|
/* no compatible elements, we cannot go on */
|
||||||
goto unknown_type;
|
goto unknown_type;
|
||||||
|
|
||||||
try_to_link_1 (decode_bin, pad, to_try);
|
try_to_link_1 (decode_bin, element, pad, to_try);
|
||||||
/* can free the list again now */
|
/* can free the list again now */
|
||||||
g_list_free (to_try);
|
g_list_free (to_try);
|
||||||
}
|
}
|
||||||
|
@ -630,17 +606,29 @@ many_types:
|
||||||
* pad.
|
* pad.
|
||||||
*/
|
*/
|
||||||
static GstElement *
|
static GstElement *
|
||||||
try_to_link_1 (GstDecodeBin * decode_bin, GstPad * pad, GList * factories)
|
try_to_link_1 (GstDecodeBin * decode_bin, GstElement * srcelement, GstPad * pad,
|
||||||
|
GList * factories)
|
||||||
{
|
{
|
||||||
GList *walk;
|
GList *walk;
|
||||||
|
GstElementFactory *srcfactory = NULL;
|
||||||
GstElement *result = NULL;
|
GstElement *result = NULL;
|
||||||
|
gboolean isdemux = FALSE;
|
||||||
|
const gchar *klass;
|
||||||
|
|
||||||
|
/* Check if the parent of the src pad is a demuxer */
|
||||||
|
srcfactory = gst_element_get_factory (srcelement);
|
||||||
|
klass = gst_element_factory_get_klass (srcfactory);
|
||||||
|
isdemux = !!(strstr (klass, "Demux"));
|
||||||
|
|
||||||
/* loop over the factories */
|
/* loop over the factories */
|
||||||
for (walk = factories; walk; walk = g_list_next (walk)) {
|
for (walk = factories; walk; walk = g_list_next (walk)) {
|
||||||
GstElementFactory *factory = GST_ELEMENT_FACTORY (walk->data);
|
GstElementFactory *factory = GST_ELEMENT_FACTORY (walk->data);
|
||||||
GstElement *element;
|
GstElement *element;
|
||||||
|
GstElement *queue = NULL;
|
||||||
GstPadLinkReturn ret;
|
GstPadLinkReturn ret;
|
||||||
GstPad *sinkpad;
|
GstPad *sinkpad;
|
||||||
|
GstPad *usedsrcpad = pad;
|
||||||
|
GstPad *queuesinkpad = NULL, *queuesrcpad = NULL;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (decode_bin, "trying to link %s",
|
GST_DEBUG_OBJECT (decode_bin, "trying to link %s",
|
||||||
gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (factory)));
|
gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (factory)));
|
||||||
|
@ -669,42 +657,53 @@ try_to_link_1 (GstDecodeBin * decode_bin, GstPad * pad, GList * factories)
|
||||||
/* set to ready first so it is ready */
|
/* set to ready first so it is ready */
|
||||||
gst_element_set_state (element, GST_STATE_READY);
|
gst_element_set_state (element, GST_STATE_READY);
|
||||||
|
|
||||||
/* keep our own list of elements */
|
if (isdemux) {
|
||||||
decode_bin->elements = g_list_prepend (decode_bin->elements, element);
|
/* Insert a queue between demuxer and decoder */
|
||||||
|
GST_DEBUG_OBJECT (decode_bin,
|
||||||
|
"Element %s is a demuxer, inserting a queue",
|
||||||
|
GST_OBJECT_NAME (srcelement));
|
||||||
|
|
||||||
if ((ret = gst_pad_link (pad, sinkpad)) != GST_PAD_LINK_OK) {
|
queue = gst_element_factory_make ("queue", NULL);
|
||||||
|
g_object_set (G_OBJECT (queue), (const gchar *) "max-size-buffers", 0,
|
||||||
|
NULL);
|
||||||
|
g_object_set (G_OBJECT (queue), (const gchar *) "max-size-time", 0LL,
|
||||||
|
NULL);
|
||||||
|
gst_bin_add (GST_BIN (decode_bin), queue);
|
||||||
|
gst_element_set_state (queue, GST_STATE_READY);
|
||||||
|
queuesinkpad = gst_element_get_pad (queue, "sink");
|
||||||
|
usedsrcpad = queuesrcpad = gst_element_get_pad (queue, "src");
|
||||||
|
|
||||||
|
gst_pad_link (pad, queuesinkpad);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((ret = gst_pad_link (usedsrcpad, sinkpad)) != GST_PAD_LINK_OK) {
|
||||||
GST_DEBUG_OBJECT (decode_bin, "link failed on pad %s:%s, reason %d",
|
GST_DEBUG_OBJECT (decode_bin, "link failed on pad %s:%s, reason %d",
|
||||||
GST_DEBUG_PAD_NAME (pad), ret);
|
GST_DEBUG_PAD_NAME (pad), ret);
|
||||||
/* get rid of the sinkpad */
|
/* get rid of the sinkpad */
|
||||||
gst_object_unref (sinkpad);
|
gst_object_unref (sinkpad);
|
||||||
/* this element did not work, remove it again and continue trying
|
/* this element did not work, remove it again and continue trying
|
||||||
* other elements, the element will be disposed. */
|
* other elements, the element will be disposed. */
|
||||||
|
if (isdemux)
|
||||||
|
gst_element_set_state (queue, GST_STATE_NULL);
|
||||||
gst_element_set_state (element, GST_STATE_NULL);
|
gst_element_set_state (element, GST_STATE_NULL);
|
||||||
|
if (isdemux) {
|
||||||
|
gst_pad_unlink (pad, queuesrcpad);
|
||||||
|
gst_object_unref (queuesrcpad);
|
||||||
|
gst_object_unref (queuesinkpad);
|
||||||
|
gst_bin_remove (GST_BIN (decode_bin), queue);
|
||||||
|
}
|
||||||
gst_bin_remove (GST_BIN (decode_bin), element);
|
gst_bin_remove (GST_BIN (decode_bin), element);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
const gchar *klass;
|
|
||||||
GstElementFactory *factory;
|
|
||||||
guint sig;
|
guint sig;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (decode_bin, "linked on pad %s:%s",
|
GST_DEBUG_OBJECT (decode_bin, "linked on pad %s:%s",
|
||||||
GST_DEBUG_PAD_NAME (pad));
|
GST_DEBUG_PAD_NAME (usedsrcpad));
|
||||||
|
|
||||||
/* The link worked, now figure out what it was that we connected */
|
/* The link worked, now figure out what it was that we connected */
|
||||||
factory = gst_element_get_factory (element);
|
|
||||||
klass = gst_element_factory_get_klass (factory);
|
|
||||||
|
|
||||||
/* check if we can use threads */
|
|
||||||
if (decode_bin->threaded) {
|
|
||||||
if (strstr (klass, "Demux") != NULL) {
|
|
||||||
/* FIXME, do something with threads here. Not sure that it
|
|
||||||
* really matters here but in general it is better to preroll
|
|
||||||
* on encoded data from the muxer than on raw encoded streams
|
|
||||||
* because that would consume less memory. */
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* make sure we catch unlink signals */
|
/* make sure we catch unlink signals */
|
||||||
sig = g_signal_connect (G_OBJECT (pad), "unlinked",
|
sig = g_signal_connect (G_OBJECT (usedsrcpad), "unlinked",
|
||||||
G_CALLBACK (unlinked), decode_bin);
|
G_CALLBACK (unlinked), decode_bin);
|
||||||
|
|
||||||
/* keep a ref to the signal id so that we can disconnect the signal callback */
|
/* keep a ref to the signal id so that we can disconnect the signal callback */
|
||||||
|
@ -720,6 +719,11 @@ try_to_link_1 (GstDecodeBin * decode_bin, GstPad * pad, GList * factories)
|
||||||
|
|
||||||
/* get rid of the sinkpad now */
|
/* get rid of the sinkpad now */
|
||||||
gst_object_unref (sinkpad);
|
gst_object_unref (sinkpad);
|
||||||
|
if (isdemux) {
|
||||||
|
gst_element_set_state (queue, GST_STATE_PAUSED);
|
||||||
|
gst_object_unref (queuesrcpad);
|
||||||
|
gst_object_unref (queuesinkpad);
|
||||||
|
}
|
||||||
|
|
||||||
/* and exit */
|
/* and exit */
|
||||||
goto done;
|
goto done;
|
||||||
|
@ -1118,46 +1122,6 @@ shutting_down:
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
gst_decode_bin_set_property (GObject * object, guint prop_id,
|
|
||||||
const GValue * value, GParamSpec * pspec)
|
|
||||||
{
|
|
||||||
GstDecodeBin *decode_bin;
|
|
||||||
|
|
||||||
g_return_if_fail (GST_IS_DECODE_BIN (object));
|
|
||||||
|
|
||||||
decode_bin = GST_DECODE_BIN (object);
|
|
||||||
|
|
||||||
switch (prop_id) {
|
|
||||||
case ARG_THREADED:
|
|
||||||
decode_bin->threaded = g_value_get_boolean (value);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
gst_decode_bin_get_property (GObject * object, guint prop_id, GValue * value,
|
|
||||||
GParamSpec * pspec)
|
|
||||||
{
|
|
||||||
GstDecodeBin *decode_bin;
|
|
||||||
|
|
||||||
g_return_if_fail (GST_IS_DECODE_BIN (object));
|
|
||||||
|
|
||||||
decode_bin = GST_DECODE_BIN (object);
|
|
||||||
|
|
||||||
switch (prop_id) {
|
|
||||||
case ARG_THREADED:
|
|
||||||
g_value_set_boolean (value, decode_bin->threaded);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static GstStateChangeReturn
|
static GstStateChangeReturn
|
||||||
gst_decode_bin_change_state (GstElement * element, GstStateChange transition)
|
gst_decode_bin_change_state (GstElement * element, GstStateChange transition)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in a new issue