playback: Remove gstdecodebin.c, which is nowaday unused anyway
authorSebastian Dröge <sebastian.droege@collabora.co.uk>
Fri, 6 Apr 2012 08:54:04 +0000 (10:54 +0200)
committerSebastian Dröge <sebastian.droege@collabora.co.uk>
Fri, 6 Apr 2012 08:54:42 +0000 (10:54 +0200)
gst/playback/gstdecodebin.c [deleted file]

diff --git a/gst/playback/gstdecodebin.c b/gst/playback/gstdecodebin.c
deleted file mode 100644 (file)
index 7ccc7f5..0000000
+++ /dev/null
@@ -1,2075 +0,0 @@
-/* GStreamer
- * Copyright (C) <2004> Wim Taymans <wim.taymans@gmail.com>
- * Copyright (C) 2011 Hewlett-Packard Development Company, L.P.
- *
- * 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., 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.
- */
-
-/**
- * SECTION:element-decodebin
- *
- * #GstBin that auto-magically constructs a decoding pipeline using available
- * decoders and demuxers via auto-plugging.
- *
- * When using decodebin in your application, connect a signal handler to
- * #GstDecodeBin::new-decoded-pad and connect your sinks from within the
- * callback function.
- *
- * <note>
- * This element is deprecated and no longer supported. You should use the
- * #uridecodebin or #decodebin2 element instead (or, even better: #playbin2).
- * </note>
- *
- * Deprecated: use uridecodebin or decodebin2 instead.
- */
-
-#ifdef HAVE_CONFIG_H
-#include "config.h"
-#endif
-
-#include <gst/gst-i18n-plugin.h>
-
-#include <string.h>
-#include <gst/gst.h>
-#include <gst/pbutils/pbutils.h>
-#include "gst/glib-compat-private.h"
-
-#include "gstplay-marshal.h"
-
-/* generic templates */
-static GstStaticPadTemplate decoder_bin_sink_template =
-GST_STATIC_PAD_TEMPLATE ("sink",
-    GST_PAD_SINK,
-    GST_PAD_ALWAYS,
-    GST_STATIC_CAPS_ANY);
-
-static GstStaticPadTemplate decoder_bin_src_template =
-GST_STATIC_PAD_TEMPLATE ("src_%u",
-    GST_PAD_SRC,
-    GST_PAD_SOMETIMES,
-    GST_STATIC_CAPS_ANY);
-
-GST_DEBUG_CATEGORY_STATIC (gst_decode_bin_debug);
-#define GST_CAT_DEFAULT gst_decode_bin_debug
-
-#define GST_TYPE_DECODE_BIN             (gst_decode_bin_get_type())
-#define GST_DECODE_BIN(obj)             (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_DECODE_BIN,GstDecodeBin))
-#define GST_DECODE_BIN_CLASS(klass)     (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_DECODE_BIN,GstDecodeBinClass))
-#define GST_IS_DECODE_BIN(obj)          (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_DECODE_BIN))
-#define GST_IS_DECODE_BIN_CLASS(klass)  (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_DECODE_BIN))
-
-typedef struct _GstDecodeBin GstDecodeBin;
-typedef struct _GstDecodeBinClass GstDecodeBinClass;
-
-/**
- * GstDecodeBin:
- *
- * Auto-plugging decoder element structure
- */
-struct _GstDecodeBin
-{
-  GstBin bin;                   /* we extend GstBin */
-
-  GstElement *typefind;         /* this holds the typefind object */
-  GstElement *fakesink;
-
-  GList *dynamics;              /* list of dynamic connections */
-
-  GList *queues;                /* list of demuxer-decoder queues */
-
-  GList *probes;                /* list of PadProbeData */
-
-  GList *factories;             /* factories we can use for selecting elements */
-  guint numpads;
-  gint numwaiting;
-
-  gboolean have_type;
-  guint have_type_id;           /* signal id for the typefind element */
-
-  gboolean shutting_down;       /* stop pluggin if we're shutting down */
-
-  GType queue_type;             /* store the GType of queues, to aid in recognising them */
-
-  GMutex *cb_mutex;             /* Mutex for multi-threaded callbacks, such as removing the fakesink */
-};
-
-struct _GstDecodeBinClass
-{
-  GstBinClass parent_class;
-
-  /* signal we fire when a new pad has been decoded into raw audio/video */
-  void (*new_decoded_pad) (GstElement * element, GstPad * pad, gboolean last);
-  /* signal we fire when a pad has been removed */
-  void (*removed_decoded_pad) (GstElement * element, GstPad * pad);
-  /* signal fired when we found a pad that we cannot decode */
-  void (*unknown_type) (GstElement * element, GstPad * pad, GstCaps * caps);
-};
-
-/* signals */
-enum
-{
-  SIGNAL_NEW_DECODED_PAD,
-  SIGNAL_REMOVED_DECODED_PAD,
-  SIGNAL_UNKNOWN_TYPE,
-  SIGNAL_REDIRECT,
-  LAST_SIGNAL
-};
-
-/* Properties */
-enum
-{
-  PROP_0,
-  PROP_SINK_CAPS,
-};
-
-
-typedef struct
-{
-  GstPad *pad;
-  gulong sigid;
-  gboolean done;
-} PadProbeData;
-
-/* this structure is created for all dynamic pads that could get created
- * at runtime */
-typedef struct
-{
-  GstDecodeBin *decode_bin;     /* pointer to ourself */
-
-  GstElement *element;          /* the element sending the signal */
-  gint np_sig_id;               /* signal id of new_pad */
-  gint nmp_sig_id;              /* signal id of no_more_pads */
-
-  GstPad *pad;                  /* the pad sending the signal */
-  gint caps_sig_id;             /* signal id of caps */
-}
-GstDynamic;
-
-static void gst_decode_bin_class_init (GstDecodeBinClass * klass);
-static void gst_decode_bin_init (GstDecodeBin * decode_bin);
-static void gst_decode_bin_set_property (GObject * object, guint prop_id,
-    const GValue * value, GParamSpec * pspec);
-static void gst_decode_bin_get_property (GObject * object, guint prop_id,
-    GValue * value, GParamSpec * pspec);
-static void gst_decode_bin_dispose (GObject * object);
-static void gst_decode_bin_finalize (GObject * object);
-
-static GstStateChangeReturn gst_decode_bin_change_state (GstElement * element,
-    GstStateChange transition);
-
-static gboolean add_fakesink (GstDecodeBin * decode_bin);
-static void remove_fakesink (GstDecodeBin * decode_bin);
-
-static void dynamic_free (GstDynamic * dyn);
-static void free_dynamics (GstDecodeBin * decode_bin);
-static void type_found (GstElement * typefind, guint probability,
-    GstCaps * caps, GstDecodeBin * decode_bin);
-static GstElement *try_to_link_1 (GstDecodeBin * decode_bin,
-    GstElement * origelement, GstPad * pad, GList * factories);
-static void close_link (GstElement * element, GstDecodeBin * decode_bin);
-static void close_pad_link (GstElement * element, GstPad * pad,
-    GstCaps * caps, GstDecodeBin * decode_bin, gboolean more);
-static void unlinked (GstPad * pad, GstPad * peerpad,
-    GstDecodeBin * decode_bin);
-static void new_pad (GstElement * element, GstPad * pad, GstDynamic * dynamic);
-static void no_more_pads (GstElement * element, GstDynamic * dynamic);
-static void new_caps (GstPad * pad, GParamSpec * unused, GstDynamic * dynamic);
-
-static void queue_filled_cb (GstElement * queue, GstDecodeBin * decode_bin);
-static void queue_underrun_cb (GstElement * queue, GstDecodeBin * decode_bin);
-
-static gboolean is_demuxer_element (GstElement * srcelement);
-
-static GstElementClass *parent_class;
-static guint gst_decode_bin_signals[LAST_SIGNAL] = { 0 };
-
-
-static GType
-gst_decode_bin_get_type (void)
-{
-  static GType gst_decode_bin_type = 0;
-
-  if (!gst_decode_bin_type) {
-    static const GTypeInfo gst_decode_bin_info = {
-      sizeof (GstDecodeBinClass),
-      NULL,
-      NULL,
-      (GClassInitFunc) gst_decode_bin_class_init,
-      NULL,
-      NULL,
-      sizeof (GstDecodeBin),
-      0,
-      (GInstanceInitFunc) gst_decode_bin_init,
-      NULL
-    };
-
-    gst_decode_bin_type =
-        g_type_register_static (GST_TYPE_BIN, "GstDecodeBin",
-        &gst_decode_bin_info, 0);
-  }
-
-  return gst_decode_bin_type;
-}
-
-static void
-gst_decode_bin_class_init (GstDecodeBinClass * klass)
-{
-  GObjectClass *gobject_klass;
-  GstElementClass *gstelement_klass;
-
-  gobject_klass = (GObjectClass *) klass;
-  gstelement_klass = (GstElementClass *) klass;
-
-  parent_class = g_type_class_peek_parent (klass);
-
-  gobject_klass->set_property = gst_decode_bin_set_property;
-  gobject_klass->get_property = gst_decode_bin_get_property;
-  gobject_klass->dispose = gst_decode_bin_dispose;
-  gobject_klass->finalize = gst_decode_bin_finalize;
-
-  /**
-   * GstDecodeBin::new-decoded-pad:
-   * @bin: The decodebin
-   * @pad: The newly created pad
-   * @islast: #TRUE if this is the last pad to be added. Deprecated.
-   *
-   * This signal gets emitted as soon as a new pad of the same type as one of
-   * the valid 'raw' types is added.
-   */
-  gst_decode_bin_signals[SIGNAL_NEW_DECODED_PAD] =
-      g_signal_new ("new-decoded-pad", G_TYPE_FROM_CLASS (klass),
-      G_SIGNAL_RUN_LAST,
-      G_STRUCT_OFFSET (GstDecodeBinClass, new_decoded_pad), NULL, NULL,
-      gst_play_marshal_VOID__OBJECT_BOOLEAN, G_TYPE_NONE, 2, GST_TYPE_PAD,
-      G_TYPE_BOOLEAN);
-  /**
-   * GstDecodeBin::removed-decoded-pad:
-   * @bin: The decodebin
-   * @pad: The pad that was removed
-   *
-   * This signal is emitted when a 'final' caps pad has been removed.
-   */
-  gst_decode_bin_signals[SIGNAL_REMOVED_DECODED_PAD] =
-      g_signal_new ("removed-decoded-pad", G_TYPE_FROM_CLASS (klass),
-      G_SIGNAL_RUN_LAST,
-      G_STRUCT_OFFSET (GstDecodeBinClass, removed_decoded_pad), NULL, NULL,
-      gst_marshal_VOID__OBJECT, G_TYPE_NONE, 1, GST_TYPE_PAD);
-  /**
-   * GstDecodeBin::unknown-type:
-   * @bin: The decodebin
-   * @pad: The new pad containing caps that cannot be resolved to a 'final'
-   *       stream type.
-   * @caps: The #GstCaps of the pad that cannot be resolved.
-   *
-   * This signal is emitted when a pad for which there is no further possible
-   * decoding is added to the decodebin.
-   */
-  gst_decode_bin_signals[SIGNAL_UNKNOWN_TYPE] =
-      g_signal_new ("unknown-type", G_TYPE_FROM_CLASS (klass),
-      G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GstDecodeBinClass, unknown_type),
-      NULL, NULL, gst_marshal_VOID__OBJECT_BOXED, G_TYPE_NONE, 2,
-      GST_TYPE_PAD, GST_TYPE_CAPS);
-
-  g_object_class_install_property (gobject_klass, PROP_SINK_CAPS,
-      g_param_spec_boxed ("sink-caps", "Sink Caps",
-          "The caps of the input data. (NULL = use typefind element)",
-          GST_TYPE_CAPS, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
-
-  gst_element_class_add_pad_template (gstelement_klass,
-      gst_static_pad_template_get (&decoder_bin_sink_template));
-  gst_element_class_add_pad_template (gstelement_klass,
-      gst_static_pad_template_get (&decoder_bin_src_template));
-
-  gst_element_class_set_details_simple (gstelement_klass,
-      "Decoder Bin", "Generic/Bin/Decoder",
-      "Autoplug and decode to raw media",
-      "Wim Taymans <wim.taymans@gmail.com>");
-
-  gstelement_klass->change_state =
-      GST_DEBUG_FUNCPTR (gst_decode_bin_change_state);
-}
-
-/* check if the bin is dynamic.
- *
- * If there are no outstanding dynamic connections, the bin is
- * considered to be non-dynamic.
- */
-static gboolean
-gst_decode_bin_is_dynamic (GstDecodeBin * decode_bin)
-{
-  return decode_bin->dynamics != NULL;
-}
-
-/* the filter function for selecting the elements we can use in
- * autoplugging */
-static gboolean
-gst_decode_bin_factory_filter (GstPluginFeature * feature,
-    GstDecodeBin * decode_bin)
-{
-  guint rank;
-  const gchar *klass;
-
-  /* we only care about element factories */
-  if (!GST_IS_ELEMENT_FACTORY (feature))
-    return FALSE;
-
-  klass = gst_element_factory_get_klass (GST_ELEMENT_FACTORY (feature));
-  /* only demuxers, decoders and parsers can play */
-  if (strstr (klass, "Demux") == NULL &&
-      strstr (klass, "Decoder") == NULL && strstr (klass, "Parse") == NULL &&
-      strstr (klass, "Depayloader") == NULL) {
-    return FALSE;
-  }
-
-  /* only select elements with autoplugging rank */
-  rank = gst_plugin_feature_get_rank (feature);
-  if (rank < GST_RANK_MARGINAL)
-    return FALSE;
-
-  return TRUE;
-}
-
-/* function used to sort element features */
-static gint
-compare_ranks (GstPluginFeature * f1, GstPluginFeature * f2)
-{
-  gint diff;
-  const gchar *rname1, *rname2;
-
-  diff = gst_plugin_feature_get_rank (f2) - gst_plugin_feature_get_rank (f1);
-  if (diff != 0)
-    return diff;
-
-  rname1 = gst_plugin_feature_get_name (f1);
-  rname2 = gst_plugin_feature_get_name (f2);
-
-  diff = strcmp (rname2, rname1);
-
-  return diff;
-}
-
-static void
-print_feature (GstPluginFeature * feature)
-{
-  const gchar *rname;
-
-  rname = gst_plugin_feature_get_name (feature);
-
-  GST_DEBUG ("%s", rname);
-}
-
-static void
-gst_decode_bin_init (GstDecodeBin * decode_bin)
-{
-  GList *factories;
-
-  decode_bin->cb_mutex = g_mutex_new ();
-
-  /* first filter out the interesting element factories */
-  factories = gst_default_registry_feature_filter (
-      (GstPluginFeatureFilter) gst_decode_bin_factory_filter,
-      FALSE, decode_bin);
-
-  /* sort them according to their ranks */
-  decode_bin->factories = g_list_sort (factories, (GCompareFunc) compare_ranks);
-  /* do some debugging */
-  g_list_foreach (decode_bin->factories, (GFunc) print_feature, NULL);
-
-  /* we create the typefind element only once */
-  decode_bin->typefind = gst_element_factory_make ("typefind", "typefind");
-  if (!decode_bin->typefind) {
-    g_warning ("can't find typefind element, decodebin will not work");
-  } else {
-    GstPad *pad, *gpad;
-    GstPadTemplate *pad_tmpl;
-
-    /* add the typefind element */
-    if (!gst_bin_add (GST_BIN (decode_bin), decode_bin->typefind)) {
-      g_warning ("Could not add typefind element, decodebin will not work");
-      gst_object_unref (decode_bin->typefind);
-      decode_bin->typefind = NULL;
-    }
-
-    /* get the sinkpad */
-    pad = gst_element_get_static_pad (decode_bin->typefind, "sink");
-
-    /* get the pad template */
-    pad_tmpl = gst_static_pad_template_get (&decoder_bin_sink_template);
-
-    /* ghost the sink pad to ourself */
-    gpad = gst_ghost_pad_new_from_template ("sink", pad, pad_tmpl);
-    gst_pad_set_active (gpad, TRUE);
-    gst_element_add_pad (GST_ELEMENT (decode_bin), gpad);
-
-    gst_object_unref (pad_tmpl);
-    gst_object_unref (pad);
-
-    /* connect a signal to find out when the typefind element found
-     * a type */
-    decode_bin->have_type_id =
-        g_signal_connect (G_OBJECT (decode_bin->typefind), "have_type",
-        G_CALLBACK (type_found), decode_bin);
-  }
-  add_fakesink (decode_bin);
-
-  decode_bin->dynamics = NULL;
-  decode_bin->queues = NULL;
-  decode_bin->probes = NULL;
-}
-
-static void
-gst_decode_bin_dispose (GObject * object)
-{
-  GstDecodeBin *decode_bin;
-
-  decode_bin = GST_DECODE_BIN (object);
-
-  if (decode_bin->factories)
-    gst_plugin_feature_list_free (decode_bin->factories);
-  decode_bin->factories = NULL;
-
-  G_OBJECT_CLASS (parent_class)->dispose (object);
-
-  /* our parent dispose might trigger new signals when pads are unlinked
-   * etc. clean up the mess here. */
-  /* FIXME do proper cleanup when going to NULL */
-  free_dynamics (decode_bin);
-}
-
-static void
-gst_decode_bin_set_sink_caps (GstDecodeBin * dbin, GstCaps * caps)
-{
-  GST_DEBUG_OBJECT (dbin, "Setting new caps: %" GST_PTR_FORMAT, caps);
-
-  g_object_set (dbin->typefind, "force-caps", caps, NULL);
-}
-
-static GstCaps *
-gst_decode_bin_get_sink_caps (GstDecodeBin * dbin)
-{
-  GstCaps *caps;
-
-  GST_DEBUG_OBJECT (dbin, "Getting currently set caps");
-
-  g_object_get (dbin->typefind, "force-caps", &caps, NULL);
-
-  return caps;
-}
-
-static void
-gst_decode_bin_set_property (GObject * object, guint prop_id,
-    const GValue * value, GParamSpec * pspec)
-{
-  GstDecodeBin *dbin;
-
-  dbin = GST_DECODE_BIN (object);
-
-  switch (prop_id) {
-    case PROP_SINK_CAPS:
-      gst_decode_bin_set_sink_caps (dbin, g_value_get_boxed (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 *dbin;
-
-  dbin = GST_DECODE_BIN (object);
-  switch (prop_id) {
-    case PROP_SINK_CAPS:
-      g_value_take_boxed (value, gst_decode_bin_get_sink_caps (dbin));
-      break;
-    default:
-      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
-      break;
-  }
-}
-
-static void
-gst_decode_bin_finalize (GObject * object)
-{
-  GstDecodeBin *decode_bin = GST_DECODE_BIN (object);
-
-  g_mutex_free (decode_bin->cb_mutex);
-
-  G_OBJECT_CLASS (parent_class)->finalize (object);
-}
-
-struct DynFind
-{
-  GstElement *elem;
-  GstPad *pad;
-};
-
-static gint
-find_dynamic (GstDynamic * dyn, struct DynFind *info)
-{
-  if (dyn->element == info->elem && dyn->pad == info->pad)
-    return 0;
-  return 1;
-}
-
-/* Add either an element (for dynamic pads/pad-added watching) or a
- * pad (for delayed caps/notify::caps watching) to the dynamic list,
- * taking care to ignore repeat entries so we don't end up handling a
- * pad twice, for example */
-static void
-dynamic_add (GstElement * element, GstPad * pad, GstDecodeBin * decode_bin)
-{
-  GstDynamic *dyn;
-  struct DynFind find_info;
-  GList *found;
-
-  g_return_if_fail (element != NULL);
-
-  /* do a search that this entry doesn't already exist */
-  find_info.elem = element;
-  find_info.pad = pad;
-  found = g_list_find_custom (decode_bin->dynamics, &find_info,
-      (GCompareFunc) find_dynamic);
-  if (found != NULL)
-    goto exit;
-
-  /* take refs */
-  dyn = g_new0 (GstDynamic, 1);
-  dyn->element = gst_object_ref (element);
-  dyn->decode_bin = gst_object_ref (decode_bin);
-  if (pad) {
-    dyn->pad = gst_object_ref (pad);
-    GST_DEBUG_OBJECT (decode_bin, "dynamic create for pad %" GST_PTR_FORMAT,
-        pad);
-    dyn->caps_sig_id = g_signal_connect (G_OBJECT (pad), "notify::caps",
-        G_CALLBACK (new_caps), dyn);
-  } else {
-    GST_DEBUG_OBJECT (decode_bin, "dynamic create for element %"
-        GST_PTR_FORMAT, element);
-    dyn->np_sig_id = g_signal_connect (G_OBJECT (element), "pad-added",
-        G_CALLBACK (new_pad), dyn);
-    dyn->nmp_sig_id = g_signal_connect (G_OBJECT (element), "no-more-pads",
-        G_CALLBACK (no_more_pads), dyn);
-  }
-
-  /* and add this element to the dynamic elements */
-  decode_bin->dynamics = g_list_prepend (decode_bin->dynamics, dyn);
-
-  return;
-exit:
-  if (element) {
-    GST_DEBUG_OBJECT (decode_bin, "Dynamic element already added: %"
-        GST_PTR_FORMAT, element);
-  } else {
-    GST_DEBUG_OBJECT (decode_bin, "Dynamic pad already added: %"
-        GST_PTR_FORMAT, pad);
-  }
-}
-
-static void
-dynamic_free (GstDynamic * dyn)
-{
-  GST_DEBUG_OBJECT (dyn->decode_bin, "dynamic free");
-
-  /* disconnect signals */
-  if (dyn->np_sig_id)
-    g_signal_handler_disconnect (G_OBJECT (dyn->element), dyn->np_sig_id);
-  if (dyn->nmp_sig_id)
-    g_signal_handler_disconnect (G_OBJECT (dyn->element), dyn->nmp_sig_id);
-  if (dyn->caps_sig_id)
-    g_signal_handler_disconnect (G_OBJECT (dyn->pad), dyn->caps_sig_id);
-
-  if (dyn->pad)
-    gst_object_unref (dyn->pad);
-  dyn->pad = NULL;
-  if (dyn->element)
-    gst_object_unref (dyn->element);
-  dyn->element = NULL;
-
-  gst_object_unref (dyn->decode_bin);
-  dyn->decode_bin = NULL;
-
-  g_free (dyn);
-}
-
-static void
-free_dynamics (GstDecodeBin * decode_bin)
-{
-  GList *dyns;
-
-  for (dyns = decode_bin->dynamics; dyns; dyns = g_list_next (dyns)) {
-    GstDynamic *dynamic = (GstDynamic *) dyns->data;
-
-    dynamic_free (dynamic);
-  }
-  g_list_free (decode_bin->dynamics);
-  decode_bin->dynamics = NULL;
-}
-
-/* this function runs through the element factories and returns a list
- * of all elements that are able to sink the given caps
- */
-static GList *
-find_compatibles (GstDecodeBin * decode_bin, const GstCaps * caps)
-{
-  GList *factories;
-  GList *to_try = NULL;
-
-  /* loop over all the factories */
-  for (factories = decode_bin->factories; factories;
-      factories = g_list_next (factories)) {
-    GstElementFactory *factory = GST_ELEMENT_FACTORY (factories->data);
-    const GList *templates;
-    GList *walk;
-
-    /* get the templates from the element factory */
-    templates = gst_element_factory_get_static_pad_templates (factory);
-    for (walk = (GList *) templates; walk; walk = g_list_next (walk)) {
-      GstStaticPadTemplate *templ = walk->data;
-
-      /* we only care about the sink templates */
-      if (templ->direction == GST_PAD_SINK) {
-        gboolean can_intersect;
-        GstCaps *tmpl_caps;
-
-        /* try to intersect the caps with the caps of the template */
-        tmpl_caps = gst_static_caps_get (&templ->static_caps);
-
-        can_intersect = gst_caps_can_intersect (caps, tmpl_caps);
-        gst_caps_unref (tmpl_caps);
-
-        /* check if the intersection is empty */
-        if (can_intersect) {
-          /* non empty intersection, we can use this element */
-          to_try = g_list_prepend (to_try, factory);
-          break;
-        }
-      }
-    }
-  }
-  to_try = g_list_reverse (to_try);
-
-  return to_try;
-}
-
-static gboolean
-mimetype_is_raw (const gchar * mimetype)
-{
-  return g_str_has_prefix (mimetype, "video/x-raw") ||
-      g_str_has_prefix (mimetype, "audio/x-raw") ||
-      g_str_has_prefix (mimetype, "text/plain") ||
-      g_str_has_prefix (mimetype, "text/x-pango-markup");
-}
-
-static void
-free_pad_probes (GstDecodeBin * decode_bin)
-{
-  GList *tmp;
-
-  /* Remove pad probes */
-  for (tmp = decode_bin->probes; tmp; tmp = g_list_next (tmp)) {
-    PadProbeData *data = (PadProbeData *) tmp->data;
-
-    gst_pad_remove_data_probe (data->pad, data->sigid);
-    g_free (data);
-  }
-  g_list_free (decode_bin->probes);
-  decode_bin->probes = NULL;
-}
-
-/* used when we need to remove a probe because the decoder we plugged failed
- * to activate */
-static void
-free_pad_probe_for_element (GstDecodeBin * decode_bin, GstElement * element)
-{
-  GList *l;
-
-  for (l = decode_bin->probes; l != NULL; l = g_list_next (l)) {
-    PadProbeData *data = (PadProbeData *) l->data;
-
-    if (GST_ELEMENT_CAST (GST_PAD_PARENT (data->pad)) == element) {
-      gst_pad_remove_data_probe (data->pad, data->sigid);
-      decode_bin->probes = g_list_delete_link (decode_bin->probes, l);
-      g_free (data);
-      return;
-    }
-  }
-}
-
-static gboolean
-add_fakesink (GstDecodeBin * decode_bin)
-{
-  if (decode_bin->fakesink != NULL)
-    return TRUE;
-
-  g_mutex_lock (decode_bin->cb_mutex);
-
-  decode_bin->fakesink = gst_element_factory_make ("fakesink", "fakesink");
-  if (!decode_bin->fakesink)
-    goto no_fakesink;
-
-  /* hacky, remove sink flag, we don't want our decodebin to become a sink
-   * just because we add a fakesink element to make us ASYNC */
-  GST_OBJECT_FLAG_UNSET (decode_bin->fakesink, GST_ELEMENT_IS_SINK);
-
-  /* takes ownership */
-  if (!gst_bin_add (GST_BIN (decode_bin), decode_bin->fakesink)) {
-    g_warning ("Could not add fakesink element, decodebin will not work");
-    gst_object_unref (decode_bin->fakesink);
-    decode_bin->fakesink = NULL;
-  }
-  g_mutex_unlock (decode_bin->cb_mutex);
-  return TRUE;
-
-  /* ERRORS */
-no_fakesink:
-  {
-    g_warning ("can't find fakesink element, decodebin will not work");
-    g_mutex_unlock (decode_bin->cb_mutex);
-    return FALSE;
-  }
-}
-
-static void
-remove_fakesink (GstDecodeBin * decode_bin)
-{
-  gboolean removed_fakesink = FALSE;
-
-  if (decode_bin->fakesink == NULL)
-    return;
-
-  g_mutex_lock (decode_bin->cb_mutex);
-  if (decode_bin->fakesink) {
-    GST_DEBUG_OBJECT (decode_bin, "Removing fakesink and marking state dirty");
-
-    /* Lock the state to prevent it from changing state to non-NULL
-     * before it's removed */
-    gst_element_set_locked_state (decode_bin->fakesink, TRUE);
-    /* setting the state to NULL is never async */
-    gst_element_set_state (decode_bin->fakesink, GST_STATE_NULL);
-    gst_bin_remove (GST_BIN (decode_bin), decode_bin->fakesink);
-    decode_bin->fakesink = NULL;
-
-    removed_fakesink = TRUE;
-  }
-  g_mutex_unlock (decode_bin->cb_mutex);
-
-  if (removed_fakesink) {
-    free_pad_probes (decode_bin);
-  }
-}
-
-/* this should be implemented with _pad_block() */
-static gboolean
-pad_probe (GstPad * pad, GstMiniObject * data, GstDecodeBin * decode_bin)
-{
-  GList *tmp;
-  gboolean alldone = TRUE;
-
-  for (tmp = decode_bin->probes; tmp; tmp = g_list_next (tmp)) {
-    PadProbeData *pdata = (PadProbeData *) tmp->data;
-
-    if (pdata->pad == pad) {
-      if (GST_IS_BUFFER (data)) {
-        if (!pdata->done)
-          decode_bin->numwaiting--;
-        pdata->done = TRUE;
-      } else if (GST_IS_EVENT (data) &&
-          ((GST_EVENT_TYPE (data) == GST_EVENT_EOS) ||
-              (GST_EVENT_TYPE (data) == GST_EVENT_TAG) ||
-              (GST_EVENT_TYPE (data) == GST_EVENT_FLUSH_START))) {
-        /* FIXME, what about NEWSEGMENT? really, use _pad_block()... */
-        if (!pdata->done)
-          decode_bin->numwaiting--;
-        pdata->done = TRUE;
-      }
-    }
-
-    if (!(pdata->done)) {
-      GST_LOG_OBJECT (decode_bin, "Pad probe on pad %" GST_PTR_FORMAT
-          " but pad %" GST_PTR_FORMAT " still needs data.", pad, pdata->pad);
-      alldone = FALSE;
-    }
-  }
-  if (alldone)
-    remove_fakesink (decode_bin);
-  return TRUE;
-}
-
-/* FIXME: this should be somehow merged with the queue code in
- * try_to_link_1() to reduce code duplication */
-static GstPad *
-add_raw_queue (GstDecodeBin * decode_bin, GstPad * pad)
-{
-  GstElement *queue = NULL;
-  GstPad *queuesinkpad = NULL, *queuesrcpad = NULL;
-
-  queue = gst_element_factory_make ("queue", NULL);
-  decode_bin->queue_type = G_OBJECT_TYPE (queue);
-
-  g_object_set (G_OBJECT (queue), "max-size-buffers", 0, NULL);
-  g_object_set (G_OBJECT (queue), "max-size-time", G_GINT64_CONSTANT (0), NULL);
-  g_object_set (G_OBJECT (queue), "max-size-bytes", 8192, NULL);
-  gst_bin_add (GST_BIN (decode_bin), queue);
-  gst_element_set_state (queue, GST_STATE_READY);
-  queuesinkpad = gst_element_get_static_pad (queue, "sink");
-  queuesrcpad = gst_element_get_static_pad (queue, "src");
-
-  if (gst_pad_link (pad, queuesinkpad) != GST_PAD_LINK_OK) {
-    GST_WARNING_OBJECT (decode_bin,
-        "Linking queue failed, trying without queue");
-    gst_element_set_state (queue, GST_STATE_NULL);
-    gst_object_unref (queuesrcpad);
-    gst_object_unref (queuesinkpad);
-    gst_bin_remove (GST_BIN (decode_bin), queue);
-    return gst_object_ref (pad);
-  }
-
-  decode_bin->queues = g_list_append (decode_bin->queues, queue);
-  g_signal_connect (G_OBJECT (queue),
-      "overrun", G_CALLBACK (queue_filled_cb), decode_bin);
-  g_signal_connect (G_OBJECT (queue),
-      "underrun", G_CALLBACK (queue_underrun_cb), decode_bin);
-
-  gst_element_set_state (queue, GST_STATE_PAUSED);
-  gst_object_unref (queuesinkpad);
-
-  return queuesrcpad;
-}
-
-/* given a pad and a caps from an element, find the list of elements
- * that could connect to the pad
- *
- * If the pad has a raw format, this function will create a ghostpad
- * for the pad onto the decodebin.
- *
- * If no compatible elements could be found, this function will signal
- * the unknown_type signal.
- */
-static void
-close_pad_link (GstElement * element, GstPad * pad, GstCaps * caps,
-    GstDecodeBin * decode_bin, gboolean more)
-{
-  GstStructure *structure;
-  const gchar *mimetype;
-  gchar *padname;
-  gint diff;
-
-  padname = gst_pad_get_name (pad);
-  diff = strncmp (padname, "current_", 8);
-  g_free (padname);
-
-  /* hack.. ignore current pads */
-  if (!diff)
-    return;
-
-  /* the caps is empty, this means the pad has no type, we can only
-   * decide to fire the unknown_type signal. */
-  if (caps == NULL || gst_caps_is_empty (caps))
-    goto unknown_type;
-
-  /* the caps is any, this means the pad can be anything and
-   * we don't know yet */
-  if (gst_caps_is_any (caps))
-    goto dont_know_yet;
-
-  GST_LOG_OBJECT (element, "trying to close %" GST_PTR_FORMAT, caps);
-
-  /* FIXME, iterate over more structures? I guess it is possible that
-   * this pad has some encoded and some raw pads. This code will fail
-   * then if the first structure is not the raw type... */
-  structure = gst_caps_get_structure (caps, 0);
-  mimetype = gst_structure_get_name (structure);
-
-  /* first see if this is raw. If the type is raw, we can
-   * create a ghostpad for this pad. It's possible that the caps are not
-   * fixed. */
-  if (mimetype_is_raw (mimetype)) {
-    GstPadTemplate *tmpl;
-    gchar *padname;
-    GstPad *ghost;
-    PadProbeData *data;
-
-    /* If we're at a demuxer element but have raw data already
-     * we have to add a queue here. For non-raw data this is done
-     * in try_to_link_1() */
-    if (is_demuxer_element (element)) {
-      GST_DEBUG_OBJECT (decode_bin,
-          "Element %s is a demuxer, inserting a queue",
-          GST_OBJECT_NAME (element));
-
-      pad = add_raw_queue (decode_bin, pad);
-    }
-
-    /* make a unique name for this new pad */
-    padname = g_strdup_printf ("src_%u", decode_bin->numpads);
-    decode_bin->numpads++;
-
-    /* make it a ghostpad */
-    tmpl = gst_static_pad_template_get (&decoder_bin_src_template);
-    ghost = gst_ghost_pad_new_from_template (padname, pad, tmpl);
-    gst_object_unref (tmpl);
-
-    gst_pad_set_active (ghost, TRUE);
-    gst_element_add_pad (GST_ELEMENT (decode_bin), ghost);
-
-    data = g_new0 (PadProbeData, 1);
-    data->pad = pad;
-    data->done = FALSE;
-
-    /* FIXME, use _pad_block() */
-    data->sigid = gst_pad_add_data_probe (pad, G_CALLBACK (pad_probe),
-        decode_bin);
-    decode_bin->numwaiting++;
-
-    decode_bin->probes = g_list_append (decode_bin->probes, data);
-
-    GST_LOG_OBJECT (element, "closed pad %s", padname);
-
-    /* our own signal with an extra flag that this is the only pad */
-    GST_DEBUG_OBJECT (decode_bin, "emitting new-decoded-pad");
-    g_signal_emit (G_OBJECT (decode_bin),
-        gst_decode_bin_signals[SIGNAL_NEW_DECODED_PAD], 0, ghost, !more);
-    GST_DEBUG_OBJECT (decode_bin, "emitted new-decoded-pad");
-
-    g_free (padname);
-
-    /* If we're at a demuxer element pad was set to a queue's
-     * srcpad and must be unref'd here */
-    if (is_demuxer_element (element))
-      gst_object_unref (pad);
-  } else {
-    GList *to_try;
-
-    /* if the caps has many types, we need to delay */
-    if (!gst_caps_is_fixed (caps))
-      goto many_types;
-
-    /* continue plugging, first find all compatible elements */
-    to_try = find_compatibles (decode_bin, caps);
-    if (to_try == NULL)
-      /* no compatible elements, we cannot go on */
-      goto unknown_type;
-
-    if (try_to_link_1 (decode_bin, element, pad, to_try) == NULL) {
-      g_list_free (to_try);
-      GST_LOG_OBJECT (pad, "none of the allegedly available elements usable");
-      goto unknown_type;
-    }
-
-    /* can free the list again now */
-    g_list_free (to_try);
-  }
-  return;
-
-  /* ERRORS */
-unknown_type:
-  {
-    GST_LOG_OBJECT (pad, "unknown type found, fire signal");
-    g_signal_emit (G_OBJECT (decode_bin),
-        gst_decode_bin_signals[SIGNAL_UNKNOWN_TYPE], 0, pad, caps);
-
-    gst_element_post_message (GST_ELEMENT_CAST (decode_bin),
-        gst_missing_decoder_message_new (GST_ELEMENT_CAST (decode_bin), caps));
-
-    if (element == decode_bin->typefind) {
-      gchar *desc;
-
-      desc = gst_pb_utils_get_decoder_description (caps);
-      GST_ELEMENT_ERROR (decode_bin, STREAM, CODEC_NOT_FOUND,
-          (_("A %s plugin is required to play this stream, but not installed."),
-              desc),
-          ("No decoder to handle media type '%s'",
-              gst_structure_get_name (gst_caps_get_structure (caps, 0))));
-      g_free (desc);
-    }
-
-    return;
-  }
-dont_know_yet:
-  {
-    GST_LOG_OBJECT (pad, "type is not known yet");
-    goto setup_caps_delay;
-  }
-many_types:
-  {
-    GST_LOG_OBJECT (pad, "many possible types");
-    goto setup_caps_delay;
-  }
-setup_caps_delay:
-  {
-    GST_LOG_OBJECT (pad, "setting up a delayed link");
-    dynamic_add (element, pad, decode_bin);
-    return;
-  }
-}
-
-/* Decide whether an element is a demuxer based on the
- * klass and number/type of src pad templates it has */
-static gboolean
-is_demuxer_element (GstElement * srcelement)
-{
-  GstElementFactory *srcfactory;
-  GstElementClass *elemclass;
-  GList *walk;
-  const gchar *klass;
-  gint potential_src_pads = 0;
-
-  srcfactory = gst_element_get_factory (srcelement);
-  klass = gst_element_factory_get_klass (srcfactory);
-
-  /* Can't be a demuxer unless it has Demux in the klass name */
-  if (klass == NULL || !strstr (klass, "Demux"))
-    return FALSE;
-
-  /* Walk the src pad templates and count how many the element
-   * might produce */
-  elemclass = GST_ELEMENT_GET_CLASS (srcelement);
-
-  walk = gst_element_class_get_pad_template_list (elemclass);
-  while (walk != NULL) {
-    GstPadTemplate *templ;
-
-    templ = (GstPadTemplate *) walk->data;
-    if (GST_PAD_TEMPLATE_DIRECTION (templ) == GST_PAD_SRC) {
-      switch (GST_PAD_TEMPLATE_PRESENCE (templ)) {
-        case GST_PAD_ALWAYS:
-        case GST_PAD_SOMETIMES:
-          if (strstr (GST_PAD_TEMPLATE_NAME_TEMPLATE (templ), "%"))
-            potential_src_pads += 2;    /* Might make multiple pads */
-          else
-            potential_src_pads += 1;
-          break;
-        case GST_PAD_REQUEST:
-          potential_src_pads += 2;
-          break;
-      }
-    }
-    walk = g_list_next (walk);
-  }
-
-  if (potential_src_pads < 2)
-    return FALSE;
-
-  return TRUE;
-}
-
-/*
- * given a list of element factories, try to link one of the factories
- * to the given pad.
- *
- * The function returns the element that was successfully linked to the
- * pad.
- */
-static GstElement *
-try_to_link_1 (GstDecodeBin * decode_bin, GstElement * srcelement, GstPad * pad,
-    GList * factories)
-{
-  GList *walk;
-  GstElement *result = NULL;
-  gboolean isdemux = FALSE;
-  GstPad *queuesinkpad = NULL, *queuesrcpad = NULL;
-  GstElement *queue = NULL;
-  GstPad *usedsrcpad = pad;
-
-  /* Check if the parent of the src pad is a demuxer */
-  isdemux = is_demuxer_element (srcelement);
-
-  if (isdemux && factories != NULL) {
-    GstPadLinkReturn dqlink;
-
-    /* Insert a queue between demuxer and decoder */
-    GST_DEBUG_OBJECT (decode_bin,
-        "Element %s is a demuxer, inserting a queue",
-        GST_OBJECT_NAME (srcelement));
-    queue = gst_element_factory_make ("queue", NULL);
-    decode_bin->queue_type = G_OBJECT_TYPE (queue);
-
-    g_object_set (G_OBJECT (queue), "max-size-buffers", 0, NULL);
-    g_object_set (G_OBJECT (queue), "max-size-time", G_GINT64_CONSTANT (0),
-        NULL);
-    g_object_set (G_OBJECT (queue), "max-size-bytes", 8192, NULL);
-    gst_bin_add (GST_BIN (decode_bin), queue);
-    gst_element_set_state (queue, GST_STATE_READY);
-    queuesinkpad = gst_element_get_static_pad (queue, "sink");
-    usedsrcpad = queuesrcpad = gst_element_get_static_pad (queue, "src");
-
-    dqlink = gst_pad_link (pad, queuesinkpad);
-    g_return_val_if_fail (dqlink == GST_PAD_LINK_OK, NULL);
-  }
-
-  /* loop over the factories */
-  for (walk = factories; walk; walk = g_list_next (walk)) {
-    GstElementFactory *factory = GST_ELEMENT_FACTORY (walk->data);
-    GstElementFactory *src_factory;
-    GstElement *element;
-    GstPadLinkReturn ret;
-    GstPad *sinkpad;
-
-    GST_DEBUG_OBJECT (decode_bin, "trying to link %s to %s",
-        gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (factory)),
-        GST_OBJECT_NAME (srcelement));
-
-    /* don't plug the same parser twice, but allow multiple
-     * instances of other elements (e.g. id3demux) in a row */
-    src_factory = gst_element_get_factory (srcelement);
-    if (src_factory == factory
-        && gst_element_factory_list_is_type (factory,
-            GST_ELEMENT_FACTORY_TYPE_PARSER)) {
-      GST_DEBUG_OBJECT (decode_bin,
-          "not inserting parser element %s twice in a row, skipping",
-          GST_PLUGIN_FEATURE_NAME (factory));
-      continue;
-    }
-
-    /* make an element from the factory first */
-    if ((element = gst_element_factory_create (factory, NULL)) == NULL) {
-      /* hmm, strange. Like with all things in life, let's move on.. */
-      GST_WARNING_OBJECT (decode_bin, "could not create an element from %s",
-          gst_plugin_feature_get_name (GST_PLUGIN_FEATURE (factory)));
-      continue;
-    }
-
-    /* try to link the given pad to a sinkpad */
-    /* FIXME, find the sinkpad by looping over the pads instead of
-     * looking it up by name */
-    if ((sinkpad = gst_element_get_static_pad (element, "sink")) == NULL) {
-      /* if no pad is found we can't do anything */
-      GST_WARNING_OBJECT (decode_bin, "could not find sinkpad in element");
-      continue;
-    }
-
-    /* now add the element to the bin first */
-    GST_DEBUG_OBJECT (decode_bin, "adding %s", GST_OBJECT_NAME (element));
-    gst_bin_add (GST_BIN (decode_bin), element);
-
-    /* set to READY first so it is ready, duh. */
-    if (gst_element_set_state (element,
-            GST_STATE_READY) == GST_STATE_CHANGE_FAILURE) {
-      GST_WARNING_OBJECT (decode_bin, "Couldn't set %s to READY",
-          GST_ELEMENT_NAME (element));
-      /* get rid of the sinkpad */
-      gst_object_unref (sinkpad);
-      /* this element did not work, remove it again and continue trying
-       * other elements, the element will be disposed. */
-      /* FIXME: shouldn't we do this before adding it to the bin so that no
-       * error messages get through to the app? (tpm) */
-      gst_bin_remove (GST_BIN (decode_bin), element);
-      continue;
-    }
-
-    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_PAD_NAME (pad), ret);
-      /* get rid of the sinkpad */
-      gst_object_unref (sinkpad);
-      /* this element did not work, remove it again and continue trying
-       * other elements, the element will be disposed. */
-      gst_element_set_state (element, GST_STATE_NULL);
-      gst_bin_remove (GST_BIN (decode_bin), element);
-    } else {
-      GST_DEBUG_OBJECT (decode_bin, "linked on pad %s:%s",
-          GST_DEBUG_PAD_NAME (usedsrcpad));
-
-      /* configure the queue some more */
-      if (queue != NULL) {
-        decode_bin->queues = g_list_append (decode_bin->queues, queue);
-        g_signal_connect (G_OBJECT (queue),
-            "overrun", G_CALLBACK (queue_filled_cb), decode_bin);
-        g_signal_connect (G_OBJECT (queue),
-            "underrun", G_CALLBACK (queue_underrun_cb), decode_bin);
-      }
-
-      /* The link worked, now figure out what it was that we connected */
-
-      /* make sure we catch unlink signals */
-      g_signal_connect (G_OBJECT (pad), "unlinked",
-          G_CALLBACK (unlinked), decode_bin);
-
-      /* now that we added the element we can try to continue autoplugging
-       * on it until we have a raw type */
-      close_link (element, decode_bin);
-
-      /* change the state of the element to that of the parent */
-      if ((gst_element_set_state (element,
-                  GST_STATE_PAUSED)) == GST_STATE_CHANGE_FAILURE) {
-        GST_WARNING_OBJECT (decode_bin, "Couldn't set %s to PAUSED",
-            GST_ELEMENT_NAME (element));
-        /* close_link -> close_pad_link -> might have set up a pad probe */
-        free_pad_probe_for_element (decode_bin, element);
-        gst_element_set_state (element, GST_STATE_NULL);
-        gst_bin_remove (GST_BIN (decode_bin), element);
-        continue;
-      }
-
-      result = element;
-
-      /* get rid of the sinkpad now */
-      gst_object_unref (sinkpad);
-
-      /* Set the queue to paused and set the pointer to NULL so we don't
-       * remove it below */
-      if (queue != NULL) {
-        gst_element_set_state (queue, GST_STATE_PAUSED);
-        queue = NULL;
-        gst_object_unref (queuesrcpad);
-        gst_object_unref (queuesinkpad);
-      }
-
-      /* and exit */
-      goto done;
-    }
-  }
-done:
-  if (queue != NULL) {
-    /* We didn't successfully connect to the queue */
-    gst_pad_unlink (pad, queuesinkpad);
-    gst_element_set_state (queue, GST_STATE_NULL);
-    gst_object_unref (queuesrcpad);
-    gst_object_unref (queuesinkpad);
-    gst_bin_remove (GST_BIN (decode_bin), queue);
-  }
-  return result;
-}
-
-static GstPad *
-get_our_ghost_pad (GstDecodeBin * decode_bin, GstPad * pad)
-{
-  GstIterator *pad_it = NULL;
-  GstPad *db_pad = NULL;
-  gboolean done = FALSE;
-
-  if (pad == NULL || !GST_PAD_IS_SRC (pad)) {
-    GST_DEBUG_OBJECT (decode_bin, "pad NULL or not SRC pad");
-    return NULL;
-  }
-
-  /* our ghostpads are the sourcepads */
-  pad_it = gst_element_iterate_src_pads (GST_ELEMENT (decode_bin));
-  while (!done) {
-    db_pad = NULL;
-    switch (gst_iterator_next (pad_it, (gpointer) & db_pad)) {
-      case GST_ITERATOR_OK:
-        GST_DEBUG_OBJECT (decode_bin, "looking at pad %s:%s",
-            GST_DEBUG_PAD_NAME (db_pad));
-        if (GST_IS_GHOST_PAD (db_pad)) {
-          GstPad *target_pad = NULL;
-
-          target_pad = gst_ghost_pad_get_target (GST_GHOST_PAD (db_pad));
-          done = (target_pad == pad);
-          if (target_pad)
-            gst_object_unref (target_pad);
-
-          if (done) {
-            /* Found our ghost pad */
-            GST_DEBUG_OBJECT (decode_bin, "found ghostpad %s:%s for pad %s:%s",
-                GST_DEBUG_PAD_NAME (db_pad), GST_DEBUG_PAD_NAME (pad));
-            break;
-          }
-        }
-        /* Not the right one */
-        gst_object_unref (db_pad);
-        break;
-      case GST_ITERATOR_RESYNC:
-        gst_iterator_resync (pad_it);
-        break;
-      case GST_ITERATOR_ERROR:
-        done = TRUE;
-        break;
-      case GST_ITERATOR_DONE:
-        done = TRUE;
-        break;
-    }
-  }
-  gst_iterator_free (pad_it);
-
-  return db_pad;
-}
-
-/* remove all downstream elements starting from the given pad.
- * Also make sure to remove the ghostpad we created for the raw
- * decoded stream.
- */
-static void
-remove_element_chain (GstDecodeBin * decode_bin, GstPad * pad)
-{
-  GstIterator *iter;
-  gboolean done = FALSE;
-  gpointer item;
-  GstElement *elem = GST_ELEMENT (GST_OBJECT_PARENT (pad));
-
-  while (GST_OBJECT_PARENT (elem) &&
-      GST_OBJECT_PARENT (elem) != GST_OBJECT (decode_bin))
-    elem = GST_ELEMENT (GST_OBJECT_PARENT (elem));
-
-  if (G_OBJECT_TYPE (elem) == decode_bin->queue_type) {
-    GST_DEBUG_OBJECT (decode_bin,
-        "Encountered demuxer output queue while removing element chain");
-    decode_bin->queues = g_list_remove (decode_bin->queues, elem);
-  }
-
-  GST_DEBUG_OBJECT (decode_bin, "%s:%s", GST_DEBUG_PAD_NAME (pad));
-  iter = gst_pad_iterate_internal_links (pad);
-  if (!iter)
-    goto no_iter;
-
-  /* remove all elements linked to this pad up to the ghostpad
-   * that we created for this stream */
-  while (!done) {
-    switch (gst_iterator_next (iter, &item)) {
-      case GST_ITERATOR_OK:{
-        GstPad *pad;
-        GstPad *ghostpad;
-        GstPad *peer;
-
-        pad = GST_PAD (item);
-        GST_DEBUG_OBJECT (decode_bin, "inspecting internal pad %s:%s",
-            GST_DEBUG_PAD_NAME (pad));
-
-        ghostpad = get_our_ghost_pad (decode_bin, pad);
-        if (ghostpad) {
-          GST_DEBUG_OBJECT (decode_bin, "found our ghost pad %s:%s for %s:%s",
-              GST_DEBUG_PAD_NAME (ghostpad), GST_DEBUG_PAD_NAME (pad));
-
-          g_signal_emit (G_OBJECT (decode_bin),
-              gst_decode_bin_signals[SIGNAL_REMOVED_DECODED_PAD], 0, ghostpad);
-
-          gst_element_remove_pad (GST_ELEMENT (decode_bin), ghostpad);
-          gst_object_unref (ghostpad);
-          continue;
-        } else {
-          GST_DEBUG_OBJECT (decode_bin, "not one of our ghostpads");
-        }
-
-        peer = gst_pad_get_peer (pad);
-        if (peer) {
-          GstObject *parent = gst_pad_get_parent (peer);
-
-          GST_DEBUG_OBJECT (decode_bin,
-              "internal pad %s:%s linked to pad %s:%s",
-              GST_DEBUG_PAD_NAME (pad), GST_DEBUG_PAD_NAME (peer));
-
-          if (parent) {
-            GstObject *grandparent = gst_object_get_parent (parent);
-
-            if (grandparent != NULL) {
-              if (GST_ELEMENT (grandparent) != GST_ELEMENT (decode_bin)) {
-                GST_DEBUG_OBJECT (decode_bin, "dead end pad %s:%s parent %s",
-                    GST_DEBUG_PAD_NAME (peer), GST_OBJECT_NAME (grandparent));
-              } else {
-                GST_DEBUG_OBJECT (decode_bin,
-                    "recursing element %s on pad %s:%s",
-                    GST_ELEMENT_NAME (elem), GST_DEBUG_PAD_NAME (pad));
-                remove_element_chain (decode_bin, peer);
-              }
-              gst_object_unref (grandparent);
-            }
-            gst_object_unref (parent);
-          }
-          gst_object_unref (peer);
-        }
-        gst_object_unref (item);
-      }
-        break;
-      case GST_ITERATOR_RESYNC:
-        gst_iterator_resync (iter);
-        break;
-      case GST_ITERATOR_ERROR:
-        GST_ERROR_OBJECT (pad, "Could not iterate over internally linked pads");
-        done = TRUE;
-        break;
-      case GST_ITERATOR_DONE:
-        done = TRUE;
-        break;
-    }
-  }
-  GST_DEBUG_OBJECT (decode_bin, "removing %s", GST_ELEMENT_NAME (elem));
-
-  gst_iterator_free (iter);
-
-no_iter:
-  gst_element_set_state (elem, GST_STATE_NULL);
-  gst_bin_remove (GST_BIN (decode_bin), elem);
-}
-
-/* there are @bytes bytes in @queue, enlarge it
- *
- * Returns: new max number of bytes in @queue
- */
-static guint
-queue_enlarge (GstElement * queue, guint bytes, GstDecodeBin * decode_bin)
-{
-  /* Increase the queue size by 1Mbyte if it is over 1Mb, else double its current limit
-   */
-  if (bytes > 1024 * 1024)
-    bytes += 1024 * 1024;
-  else
-    bytes *= 2;
-
-  GST_DEBUG_OBJECT (decode_bin,
-      "increasing queue %s max-size-bytes to %d", GST_ELEMENT_NAME (queue),
-      bytes);
-  g_object_set (G_OBJECT (queue), "max-size-bytes", bytes, NULL);
-
-  return bytes;
-}
-
-/* this callback is called when our queues fills up or are empty
- * We then check the status of all other queues to make sure we
- * never have an empty and full queue at the same time since that
- * would block dataflow. In the case of a filled queue, we make
- * it larger.
- */
-static void
-queue_underrun_cb (GstElement * queue, GstDecodeBin * decode_bin)
-{
-  /* FIXME: we don't really do anything here for now. Ideally we should
-   * see if some of the queues are filled and increase their values
-   * in that case.
-   * Note: be very careful with thread safety here as this underrun
-   * signal is done from the streaming thread of queue srcpad which
-   * is different from the pad_added (where we add the queue to the
-   * list) and the overrun signals that are signalled from the
-   * demuxer thread.
-   */
-  GST_DEBUG_OBJECT (decode_bin, "got underrun");
-}
-
-/* Make sure we don't have a full queue and empty queue situation */
-static void
-queue_filled_cb (GstElement * queue, GstDecodeBin * decode_bin)
-{
-  GList *tmp;
-  gboolean increase = FALSE;
-  guint bytes;
-
-  /* get current byte level from the queue that is filled */
-  g_object_get (G_OBJECT (queue), "current-level-bytes", &bytes, NULL);
-  GST_DEBUG_OBJECT (decode_bin, "One of the queues is full at %d bytes", bytes);
-
-  /* we do not buffer more than 20Mb */
-  if (bytes > (20 * 1024 * 1024))
-    goto too_large;
-
-  /* check all other queue to see if one is empty, in that case
-   * we need to enlarge @queue */
-  for (tmp = decode_bin->queues; tmp; tmp = g_list_next (tmp)) {
-    GstElement *aqueue = GST_ELEMENT (tmp->data);
-    guint levelbytes = 0;
-
-    if (aqueue != queue) {
-      g_object_get (G_OBJECT (aqueue), "current-level-bytes", &levelbytes,
-          NULL);
-      if (levelbytes == 0) {
-        /* yup, found an empty queue, we can stop the search and
-         * need to enlarge the queue */
-        increase = TRUE;
-        break;
-      }
-    }
-  }
-
-  if (increase) {
-    /* enlarge @queue */
-    queue_enlarge (queue, bytes, decode_bin);
-  } else {
-    GST_DEBUG_OBJECT (decode_bin,
-        "Queue is full but other queues are not empty, not doing anything");
-  }
-  return;
-
-  /* errors */
-too_large:
-  {
-    GST_WARNING_OBJECT (decode_bin,
-        "Queue is bigger than 20Mbytes, something else is going wrong");
-    return;
-  }
-}
-
-/* This function will be called when a dynamic pad is created on an element.
- * We try to continue autoplugging on this new pad. */
-static void
-new_pad (GstElement * element, GstPad * pad, GstDynamic * dynamic)
-{
-  GstDecodeBin *decode_bin = dynamic->decode_bin;
-  GstCaps *caps;
-  gboolean more;
-
-  GST_OBJECT_LOCK (decode_bin);
-  if (decode_bin->shutting_down)
-    goto shutting_down1;
-  GST_OBJECT_UNLOCK (decode_bin);
-
-  GST_STATE_LOCK (decode_bin);
-  if (decode_bin->shutting_down)
-    goto shutting_down2;
-
-  /* see if any more pending dynamic connections exist */
-  more = gst_decode_bin_is_dynamic (decode_bin);
-
-  caps = gst_pad_get_caps (pad);
-  close_pad_link (element, pad, caps, decode_bin, more);
-  if (caps)
-    gst_caps_unref (caps);
-  GST_STATE_UNLOCK (decode_bin);
-
-  return;
-
-shutting_down1:
-  {
-    GST_DEBUG_OBJECT (decode_bin, "we are shutting down");
-    GST_OBJECT_UNLOCK (decode_bin);
-    return;
-  }
-shutting_down2:
-  {
-    GST_DEBUG_OBJECT (decode_bin, "we are shutting down");
-    GST_STATE_UNLOCK (decode_bin);
-    return;
-  }
-}
-
-static void
-dynamic_remove (GstDynamic * dynamic)
-{
-  GstDecodeBin *decode_bin = dynamic->decode_bin;
-
-  /* remove the dynamic from the list of dynamics */
-  decode_bin->dynamics = g_list_remove (decode_bin->dynamics, dynamic);
-  dynamic_free (dynamic);
-
-  /* if we have no more dynamic elements, we have no chance of creating
-   * more pads, so we fire the no_more_pads signal */
-  if (decode_bin->dynamics == NULL) {
-    if (decode_bin->numwaiting == 0) {
-      GST_DEBUG_OBJECT (decode_bin,
-          "no more dynamic elements, removing fakesink");
-      remove_fakesink (decode_bin);
-    }
-    GST_DEBUG_OBJECT (decode_bin,
-        "no more dynamic elements, signaling no_more_pads");
-    gst_element_no_more_pads (GST_ELEMENT (decode_bin));
-  } else {
-    GST_DEBUG_OBJECT (decode_bin, "we have more dynamic elements");
-  }
-}
-
-/* this signal is fired when an element signals the no_more_pads signal.
- * This means that the element will not generate more dynamic pads and
- * we can remove the element from the list of dynamic elements. When we
- * have no more dynamic elements in the pipeline, we can fire a no_more_pads
- * signal ourselves. */
-static void
-no_more_pads (GstElement * element, GstDynamic * dynamic)
-{
-  GST_DEBUG_OBJECT (dynamic->decode_bin, "no more pads on element %s",
-      GST_ELEMENT_NAME (element));
-
-  dynamic_remove (dynamic);
-}
-
-static void
-new_caps (GstPad * pad, GParamSpec * unused, GstDynamic * dynamic)
-{
-  GST_DEBUG_OBJECT (dynamic->decode_bin, "delayed link triggered");
-
-  new_pad (dynamic->element, pad, dynamic);
-
-  /* assume it worked and remove the dynamic */
-  dynamic_remove (dynamic);
-
-  return;
-}
-
-static gboolean
-is_our_kid (GstElement * e, GstDecodeBin * decode_bin)
-{
-  gboolean ret;
-  GstElement *parent;
-
-  parent = (GstElement *) gst_object_get_parent ((GstObject *) e);
-  ret = (parent == (GstElement *) decode_bin);
-
-  if (parent)
-    gst_object_unref ((GstObject *) parent);
-
-  return ret;
-}
-
-static gboolean
-elem_is_dynamic (GstElement * element, GstDecodeBin * decode_bin)
-{
-  GList *pads;
-
-  /* loop over all the padtemplates */
-  for (pads = GST_ELEMENT_GET_CLASS (element)->padtemplates; pads;
-      pads = g_list_next (pads)) {
-    GstPadTemplate *templ = GST_PAD_TEMPLATE (pads->data);
-    const gchar *templ_name;
-
-    /* we are only interested in source pads */
-    if (GST_PAD_TEMPLATE_DIRECTION (templ) != GST_PAD_SRC)
-      continue;
-
-    templ_name = GST_PAD_TEMPLATE_NAME_TEMPLATE (templ);
-    GST_DEBUG_OBJECT (decode_bin, "got a source pad template %s", templ_name);
-
-    /* figure out what kind of pad this is */
-    switch (GST_PAD_TEMPLATE_PRESENCE (templ)) {
-      case GST_PAD_SOMETIMES:
-      {
-        /* try to get the pad to see if it is already created or
-         * not */
-        GstPad *pad = gst_element_get_static_pad (element, templ_name);
-
-        if (pad) {
-          GST_DEBUG_OBJECT (decode_bin, "got the pad for sometimes template %s",
-              templ_name);
-          gst_object_unref (pad);
-        } else {
-          GST_DEBUG_OBJECT (decode_bin,
-              "did not get the sometimes pad of template %s", templ_name);
-          /* we have an element that will create dynamic pads */
-          return TRUE;
-        }
-        break;
-      }
-      default:
-        /* Don't care about ALWAYS or REQUEST pads */
-        break;
-    }
-  }
-  return FALSE;
-}
-
-/* This function will be called when a pad is disconnected for some reason */
-static void
-unlinked (GstPad * pad, GstPad * peerpad, GstDecodeBin * decode_bin)
-{
-  GstElement *element, *peer;
-
-  /* inactivate pad */
-  gst_pad_set_active (pad, GST_ACTIVATE_NONE);
-
-  peer = gst_pad_get_parent_element (peerpad);
-
-  if (!is_our_kid (peer, decode_bin))
-    goto exit;
-
-  GST_DEBUG_OBJECT (decode_bin, "pad %s:%s removal while alive - chained?",
-      GST_DEBUG_PAD_NAME (pad));
-
-  /* remove all elements linked to the peerpad */
-  remove_element_chain (decode_bin, peerpad);
-
-  /* Re-add as a dynamic element if needed, via close_link */
-  element = gst_pad_get_parent_element (pad);
-  if (element) {
-    if (elem_is_dynamic (element, decode_bin))
-      dynamic_add (element, NULL, decode_bin);
-
-    gst_object_unref (element);
-  }
-
-exit:
-  gst_object_unref (peer);
-}
-
-/* this function inspects the given element and tries to connect something
- * on the srcpads. If there are dynamic pads, it sets up a signal handler to
- * continue autoplugging when they become available */
-static void
-close_link (GstElement * element, GstDecodeBin * decode_bin)
-{
-  GList *pads;
-  gboolean dynamic = FALSE;
-  GList *to_connect = NULL;
-  gboolean more;
-
-  GST_DEBUG_OBJECT (decode_bin, "closing links with element %s",
-      GST_ELEMENT_NAME (element));
-
-  /* loop over all the padtemplates */
-  for (pads = GST_ELEMENT_GET_CLASS (element)->padtemplates; pads;
-      pads = g_list_next (pads)) {
-    GstPadTemplate *templ = GST_PAD_TEMPLATE (pads->data);
-    const gchar *templ_name;
-
-    /* we are only interested in source pads */
-    if (GST_PAD_TEMPLATE_DIRECTION (templ) != GST_PAD_SRC)
-      continue;
-
-    templ_name = GST_PAD_TEMPLATE_NAME_TEMPLATE (templ);
-    GST_DEBUG_OBJECT (decode_bin, "got a source pad template %s", templ_name);
-
-    /* figure out what kind of pad this is */
-    switch (GST_PAD_TEMPLATE_PRESENCE (templ)) {
-      case GST_PAD_ALWAYS:
-      {
-        /* get the pad that we need to autoplug */
-        GstPad *pad = gst_element_get_static_pad (element, templ_name);
-
-        if (pad) {
-          GST_DEBUG_OBJECT (decode_bin, "got the pad for always template %s",
-              templ_name);
-          /* here is the pad, we need to autoplug it */
-          to_connect = g_list_prepend (to_connect, pad);
-        } else {
-          /* strange, pad is marked as always but it's not
-           * there. Fix the element */
-          GST_WARNING_OBJECT (decode_bin,
-              "could not get the pad for always template %s", templ_name);
-        }
-        break;
-      }
-      case GST_PAD_SOMETIMES:
-      {
-        /* try to get the pad to see if it is already created or
-         * not */
-        GstPad *pad = gst_element_get_static_pad (element, templ_name);
-
-        if (pad) {
-          GST_DEBUG_OBJECT (decode_bin, "got the pad for sometimes template %s",
-              templ_name);
-          /* the pad is created, we need to autoplug it */
-          to_connect = g_list_prepend (to_connect, pad);
-        } else {
-          GST_DEBUG_OBJECT (decode_bin,
-              "did not get the sometimes pad of template %s", templ_name);
-          /* we have an element that will create dynamic pads */
-          dynamic = TRUE;
-        }
-        break;
-      }
-      case GST_PAD_REQUEST:
-        /* ignore request pads */
-        GST_DEBUG_OBJECT (decode_bin, "ignoring request padtemplate %s",
-            templ_name);
-        break;
-    }
-  }
-  if (dynamic) {
-    GST_DEBUG_OBJECT (decode_bin, "got a dynamic element here");
-    /* ok, this element has dynamic pads, set up the signal handlers to be
-     * notified of them */
-
-    dynamic_add (element, NULL, decode_bin);
-  }
-
-  /* Check if this is an element with more than 1 pad. If this element
-   * has more than 1 pad, we need to be careful not to signal the
-   * no_more_pads signal after connecting the first pad. */
-  more = g_list_length (to_connect) > 1;
-
-  /* now loop over all the pads we need to connect */
-  for (pads = to_connect; pads; pads = g_list_next (pads)) {
-    GstPad *pad = GST_PAD_CAST (pads->data);
-    GstCaps *caps;
-
-    /* we have more pads if we have more than 1 pad to connect or
-     * dynamics. If we have only 1 pad and no dynamics, more will be
-     * set to FALSE and the no-more-pads signal will be fired. Note
-     * that this can change after the close_pad_link call. */
-    more |= gst_decode_bin_is_dynamic (decode_bin);
-
-    GST_DEBUG_OBJECT (decode_bin, "closing pad link for %s",
-        GST_OBJECT_NAME (pad));
-
-    /* continue autoplugging on the pads */
-    caps = gst_pad_get_caps (pad);
-    close_pad_link (element, pad, caps, decode_bin, more);
-    if (caps)
-      gst_caps_unref (caps);
-
-    gst_object_unref (pad);
-  }
-  g_list_free (to_connect);
-}
-
-/* this is the signal handler for the typefind element have_type signal.
- * It tries to continue autoplugging on the typefind src pad */
-static void
-type_found (GstElement * typefind, guint probability, GstCaps * caps,
-    GstDecodeBin * decode_bin)
-{
-  gboolean dynamic;
-  GstPad *pad;
-
-  GST_DEBUG_OBJECT (decode_bin, "typefind found caps %" GST_PTR_FORMAT, caps);
-
-  GST_OBJECT_LOCK (decode_bin);
-  if (decode_bin->shutting_down)
-    goto shutting_down;
-  GST_OBJECT_UNLOCK (decode_bin);
-
-  GST_STATE_LOCK (decode_bin);
-  if (decode_bin->shutting_down)
-    goto exit;
-
-  /* don't need the typefind anymore if we already found a type, we're not going
-   * to be able to do anything with it anyway except for generating errors */
-  if (decode_bin->have_type)
-    goto exit;
-
-  decode_bin->have_type = TRUE;
-
-  /* special-case text/plain: we only want to accept it as a raw type if it
-   * comes from a subtitle parser element or a demuxer, but not if it is the
-   * type of the entire stream, in which case we just want to error out */
-  if (typefind == decode_bin->typefind &&
-      gst_structure_has_name (gst_caps_get_structure (caps, 0), "text/plain")) {
-    gst_element_no_more_pads (GST_ELEMENT (decode_bin));
-    /* we can't handle this type of stream */
-    GST_ELEMENT_ERROR (decode_bin, STREAM, WRONG_TYPE,
-        (_("This appears to be a text file")),
-        ("decodebin cannot decode plain text files"));
-    goto exit;
-  }
-
-  /* autoplug the new pad with the caps that the signal gave us. */
-  pad = gst_element_get_static_pad (typefind, "src");
-  close_pad_link (typefind, pad, caps, decode_bin, FALSE);
-  gst_object_unref (pad);
-
-  dynamic = gst_decode_bin_is_dynamic (decode_bin);
-  if (dynamic == FALSE) {
-    GST_DEBUG_OBJECT (decode_bin, "we have no dynamic elements anymore");
-    /* if we have no dynamic elements, we know that no new pads
-     * will be created and we can signal out no_more_pads signal */
-    gst_element_no_more_pads (GST_ELEMENT (decode_bin));
-  } else {
-    /* more dynamic elements exist that could create new pads */
-    GST_DEBUG_OBJECT (decode_bin, "we have more dynamic elements");
-  }
-
-exit:
-  GST_STATE_UNLOCK (decode_bin);
-  return;
-
-shutting_down:
-  {
-    GST_DEBUG_OBJECT (decode_bin, "we are shutting down");
-    GST_OBJECT_UNLOCK (decode_bin);
-    return;
-  }
-}
-
-static void
-disconnect_unlinked_signals (GstDecodeBin * decode_bin, GstElement * element)
-{
-  GstIterator *pad_it = NULL;
-  gboolean done = FALSE;
-
-  pad_it = gst_element_iterate_src_pads (element);
-  while (!done) {
-    GstPad *pad = NULL;
-
-    switch (gst_iterator_next (pad_it, (gpointer) & pad)) {
-      case GST_ITERATOR_OK:
-        g_signal_handlers_disconnect_by_func (pad, (gpointer) unlinked,
-            decode_bin);
-        gst_object_unref (pad);
-        break;
-      case GST_ITERATOR_RESYNC:
-        gst_iterator_resync (pad_it);
-        break;
-      default:
-        done = TRUE;
-        break;
-    }
-  }
-  gst_iterator_free (pad_it);
-}
-
-
-static void
-cleanup_decodebin (GstDecodeBin * decode_bin)
-{
-  GstIterator *elem_it = NULL, *gpad_it = NULL;
-  GstPad *typefind_pad = NULL;
-  gboolean done = FALSE;
-
-  g_return_if_fail (GST_IS_DECODE_BIN (decode_bin));
-
-  GST_DEBUG_OBJECT (decode_bin, "cleaning up decodebin");
-
-  typefind_pad = gst_element_get_static_pad (decode_bin->typefind, "src");
-  if (GST_IS_PAD (typefind_pad)) {
-    g_signal_handlers_block_by_func (typefind_pad, (gpointer) unlinked,
-        decode_bin);
-  }
-
-  elem_it = gst_bin_iterate_elements (GST_BIN (decode_bin));
-  while (!done) {
-    GstElement *element = NULL;
-
-    switch (gst_iterator_next (elem_it, (gpointer) & element)) {
-      case GST_ITERATOR_OK:
-        if (element != decode_bin->typefind && element != decode_bin->fakesink) {
-          GST_DEBUG_OBJECT (element, "removing autoplugged element");
-          disconnect_unlinked_signals (decode_bin, element);
-          gst_element_set_state (element, GST_STATE_NULL);
-          gst_bin_remove (GST_BIN (decode_bin), element);
-        }
-        gst_object_unref (element);
-        break;
-      case GST_ITERATOR_RESYNC:
-        gst_iterator_resync (elem_it);
-        break;
-      case GST_ITERATOR_ERROR:
-        done = TRUE;
-        break;
-      case GST_ITERATOR_DONE:
-        done = TRUE;
-        break;
-    }
-  }
-  gst_iterator_free (elem_it);
-
-  done = FALSE;
-  gpad_it = gst_element_iterate_pads (GST_ELEMENT (decode_bin));
-  while (!done) {
-    GstPad *pad = NULL;
-
-    switch (gst_iterator_next (gpad_it, (gpointer) & pad)) {
-      case GST_ITERATOR_OK:
-        GST_DEBUG_OBJECT (pad, "inspecting pad %s:%s",
-            GST_DEBUG_PAD_NAME (pad));
-        if (GST_IS_GHOST_PAD (pad) && GST_PAD_IS_SRC (pad)) {
-          GST_DEBUG_OBJECT (pad, "removing ghost pad");
-          gst_element_remove_pad (GST_ELEMENT (decode_bin), pad);
-        }
-        gst_object_unref (pad);
-        break;
-      case GST_ITERATOR_RESYNC:
-        gst_iterator_resync (gpad_it);
-        break;
-      case GST_ITERATOR_ERROR:
-        done = TRUE;
-        break;
-      case GST_ITERATOR_DONE:
-        done = TRUE;
-        break;
-    }
-  }
-  gst_iterator_free (gpad_it);
-
-  if (GST_IS_PAD (typefind_pad)) {
-    g_signal_handlers_unblock_by_func (typefind_pad, (gpointer) unlinked,
-        decode_bin);
-    g_signal_handlers_disconnect_by_func (typefind_pad, (gpointer) unlinked,
-        decode_bin);
-    gst_object_unref (typefind_pad);
-  }
-}
-
-static GstStateChangeReturn
-gst_decode_bin_change_state (GstElement * element, GstStateChange transition)
-{
-  GstStateChangeReturn ret;
-  GstDecodeBin *decode_bin;
-
-  decode_bin = GST_DECODE_BIN (element);
-
-  switch (transition) {
-    case GST_STATE_CHANGE_NULL_TO_READY:
-      decode_bin->numpads = 0;
-      decode_bin->numwaiting = 0;
-      decode_bin->dynamics = NULL;
-      if (decode_bin->typefind == NULL)
-        goto missing_typefind;
-      break;
-    case GST_STATE_CHANGE_READY_TO_PAUSED:
-      GST_OBJECT_LOCK (decode_bin);
-      decode_bin->shutting_down = FALSE;
-      decode_bin->have_type = FALSE;
-      GST_OBJECT_UNLOCK (decode_bin);
-
-      if (!add_fakesink (decode_bin))
-        goto missing_fakesink;
-      break;
-    case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
-    case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
-      break;
-    case GST_STATE_CHANGE_PAUSED_TO_READY:
-      GST_OBJECT_LOCK (decode_bin);
-      decode_bin->shutting_down = TRUE;
-      GST_OBJECT_UNLOCK (decode_bin);
-      break;
-    default:
-      break;
-  }
-
-  ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
-  if (ret == GST_STATE_CHANGE_FAILURE)
-    return ret;
-
-  switch (transition) {
-    case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
-      break;
-    case GST_STATE_CHANGE_PAUSED_TO_READY:
-    case GST_STATE_CHANGE_READY_TO_NULL:
-      free_dynamics (decode_bin);
-      free_pad_probes (decode_bin);
-      cleanup_decodebin (decode_bin);
-      break;
-    default:
-      break;
-  }
-
-  return ret;
-/* ERRORS */
-missing_typefind:
-  {
-    gst_element_post_message (element,
-        gst_missing_element_message_new (element, "typefind"));
-    GST_ELEMENT_ERROR (element, CORE, MISSING_PLUGIN, (NULL), ("no typefind!"));
-    return GST_STATE_CHANGE_FAILURE;
-  }
-missing_fakesink:
-  {
-    gst_element_post_message (element,
-        gst_missing_element_message_new (element, "fakesink"));
-    GST_ELEMENT_ERROR (element, CORE, MISSING_PLUGIN, (NULL), ("no fakesink!"));
-    return GST_STATE_CHANGE_FAILURE;
-  }
-}
-
-static gboolean
-plugin_init (GstPlugin * plugin)
-{
-  GST_DEBUG_CATEGORY_INIT (gst_decode_bin_debug, "decodebin", 0, "decoder bin");
-
-#ifdef ENABLE_NLS
-  GST_DEBUG ("binding text domain %s to locale dir %s", GETTEXT_PACKAGE,
-      LOCALEDIR);
-  bindtextdomain (GETTEXT_PACKAGE, LOCALEDIR);
-  bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
-#endif /* ENABLE_NLS */
-
-  return gst_element_register (plugin, "decodebin", GST_RANK_NONE,
-      GST_TYPE_DECODE_BIN);
-}
-
-GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
-    GST_VERSION_MINOR,
-    decodebin,
-    "decoder bin", plugin_init, VERSION, GST_LICENSE, GST_PACKAGE_NAME,
-    GST_PACKAGE_ORIGIN)