Blame plugins/elements/gstconcat.c

Packit Service 963350
/* GStreamer concat element
Packit Service 963350
 *
Packit Service 963350
 *  Copyright (c) 2014 Sebastian Dröge <sebastian@centricular.com>
Packit Service 963350
 *
Packit Service 963350
 * This library is free software; you can redistribute it and/or
Packit Service 963350
 * modify it under the terms of the GNU Library General Public
Packit Service 963350
 * License as published by the Free Software Foundation; either
Packit Service 963350
 * version 2 of the License, or (at your option) any later version.
Packit Service 963350
 *
Packit Service 963350
 * This library is distributed in the hope that it will be useful,
Packit Service 963350
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 963350
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 963350
 * Library General Public License for more details.
Packit Service 963350
 *
Packit Service 963350
 * You should have received a copy of the GNU Library General Public
Packit Service 963350
 * License along with this library; if not, write to the
Packit Service 963350
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit Service 963350
 * Boston, MA 02110-1301, USA.
Packit Service 963350
 *
Packit Service 963350
 */
Packit Service 963350
/**
Packit Service 963350
 * SECTION:element-concat
Packit Service 963350
 * @title: concat
Packit Service 963350
 * @see_also: #GstFunnel
Packit Service 963350
 *
Packit Service 963350
 * Concatenates streams together to one continous stream.
Packit Service 963350
 *
Packit Service 963350
 * All streams but the current one are blocked until the current one
Packit Service 963350
 * finished with %GST_EVENT_EOS. Then the next stream is enabled, while
Packit Service 963350
 * keeping the running time continous for %GST_FORMAT_TIME segments or
Packit Service 963350
 * keeping the segment continous for %GST_FORMAT_BYTES segments.
Packit Service 963350
 *
Packit Service 963350
 * Streams are switched in the order in which the sinkpads were requested.
Packit Service 963350
 *
Packit Service 963350
 * By default, the stream segment's base values are adjusted to ensure
Packit Service 963350
 * the segment transitions between streams are continuous. In some cases,
Packit Service 963350
 * it may be desirable to turn off these adjustments (for example, because
Packit Service 963350
 * another downstream element like a streamsynchronizer adjusts the base
Packit Service 963350
 * values on its own). The adjust-base property can be used for this purpose.
Packit Service 963350
 *
Packit Service 963350
 * ## Example launch line
Packit Service 963350
 * |[
Packit Service 963350
 * gst-launch-1.0 concat name=c ! xvimagesink  videotestsrc num-buffers=100 ! c.   videotestsrc num-buffers=100 pattern=ball ! c.
Packit Service 963350
 * ]| Plays two video streams one after another.
Packit Service 963350
 *
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
#ifdef HAVE_CONFIG_H
Packit Service 963350
#include "config.h"
Packit Service 963350
#endif
Packit Service 963350
Packit Service 963350
#include "gstconcat.h"
Packit Service 963350
Packit Service 963350
GST_DEBUG_CATEGORY_STATIC (gst_concat_debug);
Packit Service 963350
#define GST_CAT_DEFAULT gst_concat_debug
Packit Service 963350
Packit Service 963350
G_GNUC_INTERNAL GType gst_concat_pad_get_type (void);
Packit Service 963350
Packit Service 963350
#define GST_TYPE_CONCAT_PAD (gst_concat_pad_get_type())
Packit Service 963350
#define GST_CONCAT_PAD(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GST_TYPE_CONCAT_PAD, GstConcatPad))
Packit Service 963350
#define GST_CONCAT_PAD_CAST(obj) ((GstConcatPad *)(obj))
Packit Service 963350
#define GST_CONCAT_PAD_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GST_TYPE_CONCAT_PAD, GstConcatPadClass))
Packit Service 963350
#define GST_IS_CONCAT_PAD(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GST_TYPE_CONCAT_PAD))
Packit Service 963350
#define GST_IS_CONCAT_PAD_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GST_TYPE_CONCAT_PAD))
Packit Service 963350
Packit Service 963350
typedef struct _GstConcatPad GstConcatPad;
Packit Service 963350
typedef struct _GstConcatPadClass GstConcatPadClass;
Packit Service 963350
Packit Service 963350
struct _GstConcatPad
Packit Service 963350
{
Packit Service 963350
  GstPad parent;
Packit Service 963350
Packit Service 963350
  GstSegment segment;
Packit Service 963350
Packit Service 963350
  /* Protected by the concat lock */
Packit Service 963350
  gboolean flushing;
Packit Service 963350
};
Packit Service 963350
Packit Service 963350
struct _GstConcatPadClass
Packit Service 963350
{
Packit Service 963350
  GstPadClass parent;
Packit Service 963350
};
Packit Service 963350
Packit Service 963350
G_DEFINE_TYPE (GstConcatPad, gst_concat_pad, GST_TYPE_PAD);
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_concat_pad_class_init (GstConcatPadClass * klass)
Packit Service 963350
{
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_concat_pad_init (GstConcatPad * self)
Packit Service 963350
{
Packit Service 963350
  gst_segment_init (&self->segment, GST_FORMAT_UNDEFINED);
Packit Service 963350
  self->flushing = FALSE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static GstStaticPadTemplate sink_template = GST_STATIC_PAD_TEMPLATE ("sink_%u",
Packit Service 963350
    GST_PAD_SINK,
Packit Service 963350
    GST_PAD_REQUEST,
Packit Service 963350
    GST_STATIC_CAPS_ANY);
Packit Service 963350
Packit Service 963350
static GstStaticPadTemplate src_template = GST_STATIC_PAD_TEMPLATE ("src",
Packit Service 963350
    GST_PAD_SRC,
Packit Service 963350
    GST_PAD_ALWAYS,
Packit Service 963350
    GST_STATIC_CAPS_ANY);
Packit Service 963350
Packit Service 963350
enum
Packit Service 963350
{
Packit Service 963350
  PROP_0,
Packit Service 963350
  PROP_ACTIVE_PAD,
Packit Service 963350
  PROP_ADJUST_BASE
Packit Service 963350
};
Packit Service 963350
Packit Service 963350
#define DEFAULT_ADJUST_BASE TRUE
Packit Service 963350
Packit Service 963350
#define _do_init \
Packit Service 963350
  GST_DEBUG_CATEGORY_INIT (gst_concat_debug, "concat", 0, "concat element");
Packit Service 963350
#define gst_concat_parent_class parent_class
Packit Service 963350
G_DEFINE_TYPE_WITH_CODE (GstConcat, gst_concat, GST_TYPE_ELEMENT, _do_init);
Packit Service 963350
Packit Service 963350
static void gst_concat_dispose (GObject * object);
Packit Service 963350
static void gst_concat_finalize (GObject * object);
Packit Service 963350
static void gst_concat_get_property (GObject * object,
Packit Service 963350
    guint prop_id, GValue * value, GParamSpec * pspec);
Packit Service 963350
static void gst_concat_set_property (GObject * object,
Packit Service 963350
    guint prop_id, const GValue * value, GParamSpec * pspec);
Packit Service 963350
Packit Service 963350
static GstStateChangeReturn gst_concat_change_state (GstElement * element,
Packit Service 963350
    GstStateChange transition);
Packit Service 963350
static GstPad *gst_concat_request_new_pad (GstElement * element,
Packit Service 963350
    GstPadTemplate * templ, const gchar * name, const GstCaps * caps);
Packit Service 963350
static void gst_concat_release_pad (GstElement * element, GstPad * pad);
Packit Service 963350
Packit Service 963350
static GstFlowReturn gst_concat_sink_chain (GstPad * pad, GstObject * parent,
Packit Service 963350
    GstBuffer * buffer);
Packit Service 963350
static gboolean gst_concat_sink_event (GstPad * pad, GstObject * parent,
Packit Service 963350
    GstEvent * event);
Packit Service 963350
static gboolean gst_concat_sink_query (GstPad * pad, GstObject * parent,
Packit Service 963350
    GstQuery * query);
Packit Service 963350
Packit Service 963350
static gboolean gst_concat_src_event (GstPad * pad, GstObject * parent,
Packit Service 963350
    GstEvent * event);
Packit Service 963350
static gboolean gst_concat_src_query (GstPad * pad, GstObject * parent,
Packit Service 963350
    GstQuery * query);
Packit Service 963350
Packit Service 963350
static gboolean gst_concat_switch_pad (GstConcat * self);
Packit Service 963350
Packit Service 963350
static void gst_concat_notify_active_pad (GstConcat * self);
Packit Service 963350
Packit Service 963350
static GParamSpec *pspec_active_pad = NULL;
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_concat_class_init (GstConcatClass * klass)
Packit Service 963350
{
Packit Service 963350
  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
Packit Service 963350
  GstElementClass *gstelement_class = GST_ELEMENT_CLASS (klass);
Packit Service 963350
Packit Service 963350
  gobject_class->dispose = GST_DEBUG_FUNCPTR (gst_concat_dispose);
Packit Service 963350
  gobject_class->finalize = GST_DEBUG_FUNCPTR (gst_concat_finalize);
Packit Service 963350
Packit Service 963350
  gobject_class->get_property = gst_concat_get_property;
Packit Service 963350
  gobject_class->set_property = gst_concat_set_property;
Packit Service 963350
Packit Service 963350
  pspec_active_pad = g_param_spec_object ("active-pad", "Active pad",
Packit Service 963350
      "Currently active src pad", GST_TYPE_PAD, G_PARAM_READABLE |
Packit Service 963350
      G_PARAM_STATIC_STRINGS);
Packit Service 963350
  g_object_class_install_property (gobject_class, PROP_ACTIVE_PAD,
Packit Service 963350
      pspec_active_pad);
Packit Service 963350
  g_object_class_install_property (gobject_class, PROP_ADJUST_BASE,
Packit Service 963350
      g_param_spec_boolean ("adjust-base", "Adjust segment base",
Packit Service 963350
          "Adjust the base value of segments to ensure they are adjacent",
Packit Service 963350
          DEFAULT_ADJUST_BASE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
Packit Service 963350
Packit Service 963350
  gst_element_class_set_static_metadata (gstelement_class,
Packit Service 963350
      "Concat", "Generic", "Concatenate multiple streams",
Packit Service 963350
      "Sebastian Dröge <sebastian@centricular.com>");
Packit Service 963350
Packit Service 963350
  gst_element_class_add_static_pad_template (gstelement_class, &sink_template);
Packit Service 963350
  gst_element_class_add_static_pad_template (gstelement_class, &src_template);
Packit Service 963350
Packit Service 963350
  gstelement_class->request_new_pad =
Packit Service 963350
      GST_DEBUG_FUNCPTR (gst_concat_request_new_pad);
Packit Service 963350
  gstelement_class->release_pad = GST_DEBUG_FUNCPTR (gst_concat_release_pad);
Packit Service 963350
  gstelement_class->change_state = GST_DEBUG_FUNCPTR (gst_concat_change_state);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_concat_init (GstConcat * self)
Packit Service 963350
{
Packit Service 963350
  g_mutex_init (&self->lock);
Packit Service 963350
  g_cond_init (&self->cond);
Packit Service 963350
Packit Service 963350
  self->srcpad = gst_pad_new_from_static_template (&src_template, "src");
Packit Service 963350
  gst_pad_set_event_function (self->srcpad,
Packit Service 963350
      GST_DEBUG_FUNCPTR (gst_concat_src_event));
Packit Service 963350
  gst_pad_set_query_function (self->srcpad,
Packit Service 963350
      GST_DEBUG_FUNCPTR (gst_concat_src_query));
Packit Service 963350
  gst_pad_use_fixed_caps (self->srcpad);
Packit Service 963350
Packit Service 963350
  gst_element_add_pad (GST_ELEMENT (self), self->srcpad);
Packit Service 963350
Packit Service 963350
  self->adjust_base = DEFAULT_ADJUST_BASE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_concat_dispose (GObject * object)
Packit Service 963350
{
Packit Service 963350
  GstConcat *self = GST_CONCAT (object);
Packit Service 963350
  GList *item;
Packit Service 963350
Packit Service 963350
  gst_object_replace ((GstObject **) & self->current_sinkpad, NULL);
Packit Service 963350
Packit Service 963350
restart:
Packit Service 963350
  for (item = GST_ELEMENT_PADS (object); item; item = g_list_next (item)) {
Packit Service 963350
    GstPad *pad = GST_PAD (item->data);
Packit Service 963350
Packit Service 963350
    if (GST_PAD_IS_SINK (pad)) {
Packit Service 963350
      gst_element_release_request_pad (GST_ELEMENT (object), pad);
Packit Service 963350
      goto restart;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  G_OBJECT_CLASS (parent_class)->dispose (object);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_concat_finalize (GObject * object)
Packit Service 963350
{
Packit Service 963350
  GstConcat *self = GST_CONCAT (object);
Packit Service 963350
Packit Service 963350
  g_mutex_clear (&self->lock);
Packit Service 963350
  g_cond_clear (&self->cond);
Packit Service 963350
Packit Service 963350
  G_OBJECT_CLASS (parent_class)->finalize (object);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_concat_get_property (GObject * object, guint prop_id, GValue * value,
Packit Service 963350
    GParamSpec * pspec)
Packit Service 963350
{
Packit Service 963350
  GstConcat *self = GST_CONCAT (object);
Packit Service 963350
Packit Service 963350
  switch (prop_id) {
Packit Service 963350
    case PROP_ACTIVE_PAD:{
Packit Service 963350
      g_mutex_lock (&self->lock);
Packit Service 963350
      g_value_set_object (value, self->current_sinkpad);
Packit Service 963350
      g_mutex_unlock (&self->lock);
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    case PROP_ADJUST_BASE:{
Packit Service 963350
      g_mutex_lock (&self->lock);
Packit Service 963350
      g_value_set_boolean (value, self->adjust_base);
Packit Service 963350
      g_mutex_unlock (&self->lock);
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    default:
Packit Service 963350
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit Service 963350
      break;
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_concat_set_property (GObject * object, guint prop_id, const GValue * value,
Packit Service 963350
    GParamSpec * pspec)
Packit Service 963350
{
Packit Service 963350
  GstConcat *self = GST_CONCAT (object);
Packit Service 963350
Packit Service 963350
  switch (prop_id) {
Packit Service 963350
    case PROP_ADJUST_BASE:{
Packit Service 963350
      g_mutex_lock (&self->lock);
Packit Service 963350
      self->adjust_base = g_value_get_boolean (value);
Packit Service 963350
      g_mutex_unlock (&self->lock);
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    default:
Packit Service 963350
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit Service 963350
      break;
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static GstPad *
Packit Service 963350
gst_concat_request_new_pad (GstElement * element, GstPadTemplate * templ,
Packit Service 963350
    const gchar * name, const GstCaps * caps)
Packit Service 963350
{
Packit Service 963350
  GstConcat *self = GST_CONCAT (element);
Packit Service 963350
  GstPad *sinkpad;
Packit Service 963350
  gchar *pad_name;
Packit Service 963350
  gboolean do_notify = FALSE;
Packit Service 963350
Packit Service 963350
  GST_DEBUG_OBJECT (element, "requesting pad");
Packit Service 963350
Packit Service 963350
  g_mutex_lock (&self->lock);
Packit Service 963350
  pad_name = g_strdup_printf ("sink_%u", self->pad_count);
Packit Service 963350
  self->pad_count++;
Packit Service 963350
  g_mutex_unlock (&self->lock);
Packit Service 963350
Packit Service 963350
  sinkpad = GST_PAD_CAST (g_object_new (GST_TYPE_CONCAT_PAD,
Packit Service 963350
          "name", pad_name, "direction", templ->direction, "template", templ,
Packit Service 963350
          NULL));
Packit Service 963350
  g_free (pad_name);
Packit Service 963350
Packit Service 963350
  gst_pad_set_chain_function (sinkpad,
Packit Service 963350
      GST_DEBUG_FUNCPTR (gst_concat_sink_chain));
Packit Service 963350
  gst_pad_set_event_function (sinkpad,
Packit Service 963350
      GST_DEBUG_FUNCPTR (gst_concat_sink_event));
Packit Service 963350
  gst_pad_set_query_function (sinkpad,
Packit Service 963350
      GST_DEBUG_FUNCPTR (gst_concat_sink_query));
Packit Service 963350
  GST_OBJECT_FLAG_SET (sinkpad, GST_PAD_FLAG_PROXY_CAPS);
Packit Service 963350
  GST_OBJECT_FLAG_SET (sinkpad, GST_PAD_FLAG_PROXY_ALLOCATION);
Packit Service 963350
Packit Service 963350
  gst_pad_set_active (sinkpad, TRUE);
Packit Service 963350
Packit Service 963350
  g_mutex_lock (&self->lock);
Packit Service 963350
  self->sinkpads = g_list_prepend (self->sinkpads, gst_object_ref (sinkpad));
Packit Service 963350
  if (!self->current_sinkpad) {
Packit Service 963350
    do_notify = TRUE;
Packit Service 963350
    self->current_sinkpad = gst_object_ref (sinkpad);
Packit Service 963350
  }
Packit Service 963350
  g_mutex_unlock (&self->lock);
Packit Service 963350
Packit Service 963350
  gst_element_add_pad (element, sinkpad);
Packit Service 963350
Packit Service 963350
  if (do_notify)
Packit Service 963350
    gst_concat_notify_active_pad (self);
Packit Service 963350
Packit Service 963350
  return sinkpad;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_concat_release_pad (GstElement * element, GstPad * pad)
Packit Service 963350
{
Packit Service 963350
  GstConcat *self = GST_CONCAT (element);
Packit Service 963350
  GstConcatPad *spad = GST_CONCAT_PAD_CAST (pad);
Packit Service 963350
  GList *l;
Packit Service 963350
  gboolean current_pad_removed = FALSE;
Packit Service 963350
  gboolean eos = FALSE;
Packit Service 963350
  gboolean do_notify = FALSE;
Packit Service 963350
Packit Service 963350
  GST_DEBUG_OBJECT (self, "releasing pad");
Packit Service 963350
Packit Service 963350
  g_mutex_lock (&self->lock);
Packit Service 963350
  spad->flushing = TRUE;
Packit Service 963350
  g_cond_broadcast (&self->cond);
Packit Service 963350
  g_mutex_unlock (&self->lock);
Packit Service 963350
Packit Service 963350
  gst_pad_set_active (pad, FALSE);
Packit Service 963350
Packit Service 963350
  /* Now the pad is definitely not running anymore */
Packit Service 963350
Packit Service 963350
  g_mutex_lock (&self->lock);
Packit Service 963350
  if (self->current_sinkpad == GST_PAD_CAST (spad)) {
Packit Service 963350
    eos = !gst_concat_switch_pad (self);
Packit Service 963350
    current_pad_removed = TRUE;
Packit Service 963350
    do_notify = TRUE;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  for (l = self->sinkpads; l; l = l->next) {
Packit Service 963350
    if ((gpointer) spad == l->data) {
Packit Service 963350
      gst_object_unref (spad);
Packit Service 963350
      self->sinkpads = g_list_delete_link (self->sinkpads, l);
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
  g_mutex_unlock (&self->lock);
Packit Service 963350
Packit Service 963350
  gst_element_remove_pad (GST_ELEMENT_CAST (self), pad);
Packit Service 963350
Packit Service 963350
  if (do_notify)
Packit Service 963350
    gst_concat_notify_active_pad (self);
Packit Service 963350
Packit Service 963350
  if (GST_STATE (self) > GST_STATE_READY) {
Packit Service 963350
    if (current_pad_removed && !eos)
Packit Service 963350
      gst_element_post_message (GST_ELEMENT_CAST (self),
Packit Service 963350
          gst_message_new_duration_changed (GST_OBJECT_CAST (self)));
Packit Service 963350
Packit Service 963350
    /* FIXME: Sending EOS from application thread */
Packit Service 963350
    if (eos)
Packit Service 963350
      gst_pad_push_event (self->srcpad, gst_event_new_eos ());
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/* Returns FALSE if flushing
Packit Service 963350
 * Must be called from the pad's streaming thread
Packit Service 963350
 */
Packit Service 963350
static gboolean
Packit Service 963350
gst_concat_pad_wait (GstConcatPad * spad, GstConcat * self)
Packit Service 963350
{
Packit Service 963350
  g_mutex_lock (&self->lock);
Packit Service 963350
  if (spad->flushing) {
Packit Service 963350
    g_mutex_unlock (&self->lock);
Packit Service 963350
    GST_DEBUG_OBJECT (spad, "Flushing");
Packit Service 963350
    return FALSE;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  while (spad != GST_CONCAT_PAD_CAST (self->current_sinkpad)) {
Packit Service 963350
    GST_TRACE_OBJECT (spad, "Not the current sinkpad - waiting");
Packit Service 963350
    if (self->current_sinkpad == NULL && g_list_length (self->sinkpads) == 1) {
Packit Service 963350
      GST_LOG_OBJECT (spad, "Sole pad waiting, switching");
Packit Service 963350
      /* If we are the only sinkpad, take active pad ownership */
Packit Service 963350
      self->current_sinkpad = gst_object_ref (self->sinkpads->data);
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    g_cond_wait (&self->cond, &self->lock);
Packit Service 963350
    if (spad->flushing) {
Packit Service 963350
      g_mutex_unlock (&self->lock);
Packit Service 963350
      GST_DEBUG_OBJECT (spad, "Flushing");
Packit Service 963350
      return FALSE;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
  /* This pad can only become not the current sinkpad from
Packit Service 963350
   * a) This streaming thread (we hold the stream lock)
Packit Service 963350
   * b) Releasing the pad (takes the stream lock, see above)
Packit Service 963350
   *
Packit Service 963350
   * Unlocking here is thus safe and we can safely push
Packit Service 963350
   * serialized data to our srcpad
Packit Service 963350
   */
Packit Service 963350
  GST_DEBUG_OBJECT (spad, "Now the current sinkpad");
Packit Service 963350
  g_mutex_unlock (&self->lock);
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static GstFlowReturn
Packit Service 963350
gst_concat_sink_chain (GstPad * pad, GstObject * parent, GstBuffer * buffer)
Packit Service 963350
{
Packit Service 963350
  GstFlowReturn ret;
Packit Service 963350
  GstConcat *self = GST_CONCAT (parent);
Packit Service 963350
  GstConcatPad *spad = GST_CONCAT_PAD (pad);
Packit Service 963350
Packit Service 963350
  GST_LOG_OBJECT (pad, "received buffer %p", buffer);
Packit Service 963350
Packit Service 963350
  if (!gst_concat_pad_wait (spad, self))
Packit Service 963350
    return GST_FLOW_FLUSHING;
Packit Service 963350
Packit Service 963350
  if (self->last_stop == GST_CLOCK_TIME_NONE)
Packit Service 963350
    self->last_stop = spad->segment.start;
Packit Service 963350
Packit Service 963350
  if (self->format == GST_FORMAT_TIME) {
Packit Service 963350
    GstClockTime start_time = GST_BUFFER_TIMESTAMP (buffer);
Packit Service 963350
    GstClockTime end_time = GST_CLOCK_TIME_NONE;
Packit Service 963350
Packit Service 963350
    if (start_time != GST_CLOCK_TIME_NONE)
Packit Service 963350
      end_time = start_time;
Packit Service 963350
    if (GST_BUFFER_DURATION_IS_VALID (buffer))
Packit Service 963350
      end_time += GST_BUFFER_DURATION (buffer);
Packit Service 963350
Packit Service 963350
    if (end_time != GST_CLOCK_TIME_NONE && end_time > self->last_stop)
Packit Service 963350
      self->last_stop = end_time;
Packit Service 963350
  } else {
Packit Service 963350
    self->last_stop += gst_buffer_get_size (buffer);
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  ret = gst_pad_push (self->srcpad, buffer);
Packit Service 963350
Packit Service 963350
  GST_LOG_OBJECT (pad, "handled buffer %s", gst_flow_get_name (ret));
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/* Returns FALSE if no further pad, must be called with concat lock */
Packit Service 963350
static gboolean
Packit Service 963350
gst_concat_switch_pad (GstConcat * self)
Packit Service 963350
{
Packit Service 963350
  GList *l;
Packit Service 963350
  gboolean next;
Packit Service 963350
  GstSegment segment;
Packit Service 963350
  gint64 last_stop;
Packit Service 963350
Packit Service 963350
  segment = GST_CONCAT_PAD (self->current_sinkpad)->segment;
Packit Service 963350
Packit Service 963350
  last_stop = self->last_stop;
Packit Service 963350
  if (last_stop == GST_CLOCK_TIME_NONE)
Packit Service 963350
    last_stop = segment.stop;
Packit Service 963350
  if (last_stop == GST_CLOCK_TIME_NONE)
Packit Service 963350
    last_stop = segment.start;
Packit Service 963350
  g_assert (last_stop != GST_CLOCK_TIME_NONE);
Packit Service 963350
Packit Service 963350
  if (last_stop > segment.stop)
Packit Service 963350
    last_stop = segment.stop;
Packit Service 963350
Packit Service 963350
  if (segment.format == GST_FORMAT_TIME)
Packit Service 963350
    last_stop =
Packit Service 963350
        gst_segment_to_running_time (&segment, segment.format, last_stop);
Packit Service 963350
  else
Packit Service 963350
    last_stop += segment.start;
Packit Service 963350
Packit Service 963350
  self->current_start_offset += last_stop;
Packit Service 963350
Packit Service 963350
  for (l = self->sinkpads; l; l = l->next) {
Packit Service 963350
    if ((gpointer) self->current_sinkpad == l->data) {
Packit Service 963350
      l = l->prev;
Packit Service 963350
      GST_DEBUG_OBJECT (self,
Packit Service 963350
          "Switching from pad %" GST_PTR_FORMAT " to %" GST_PTR_FORMAT,
Packit Service 963350
          self->current_sinkpad, l ? l->data : NULL);
Packit Service 963350
      gst_object_unref (self->current_sinkpad);
Packit Service 963350
      self->current_sinkpad = l ? gst_object_ref (l->data) : NULL;
Packit Service 963350
      g_cond_broadcast (&self->cond);
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  next = self->current_sinkpad != NULL;
Packit Service 963350
Packit Service 963350
  self->last_stop = GST_CLOCK_TIME_NONE;
Packit Service 963350
Packit Service 963350
  return next;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_concat_notify_active_pad (GstConcat * self)
Packit Service 963350
{
Packit Service 963350
  g_object_notify_by_pspec ((GObject *) self, pspec_active_pad);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static gboolean
Packit Service 963350
gst_concat_sink_event (GstPad * pad, GstObject * parent, GstEvent * event)
Packit Service 963350
{
Packit Service 963350
  GstConcat *self = GST_CONCAT (parent);
Packit Service 963350
  GstConcatPad *spad = GST_CONCAT_PAD_CAST (pad);
Packit Service 963350
  gboolean ret = TRUE;
Packit Service 963350
Packit Service 963350
  GST_LOG_OBJECT (pad, "received event %" GST_PTR_FORMAT, event);
Packit Service 963350
Packit Service 963350
  switch (GST_EVENT_TYPE (event)) {
Packit Service 963350
    case GST_EVENT_STREAM_START:{
Packit Service 963350
      if (!gst_concat_pad_wait (spad, self)) {
Packit Service 963350
        ret = FALSE;
Packit Service 963350
        gst_event_unref (event);
Packit Service 963350
      } else {
Packit Service 963350
        ret = gst_pad_event_default (pad, parent, event);
Packit Service 963350
      }
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    case GST_EVENT_SEGMENT:{
Packit Service 963350
      gboolean adjust_base;
Packit Service 963350
Packit Service 963350
      /* Drop segment event, we create our own one */
Packit Service 963350
      gst_event_copy_segment (event, &spad->segment);
Packit Service 963350
      gst_event_unref (event);
Packit Service 963350
Packit Service 963350
      g_mutex_lock (&self->lock);
Packit Service 963350
      adjust_base = self->adjust_base;
Packit Service 963350
      if (self->format == GST_FORMAT_UNDEFINED) {
Packit Service 963350
        if (spad->segment.format != GST_FORMAT_TIME
Packit Service 963350
            && spad->segment.format != GST_FORMAT_BYTES) {
Packit Service 963350
          g_mutex_unlock (&self->lock);
Packit Service 963350
          GST_ELEMENT_ERROR (self, CORE, FAILED, (NULL),
Packit Service 963350
              ("Can only operate in TIME or BYTES format"));
Packit Service 963350
          ret = FALSE;
Packit Service 963350
          break;
Packit Service 963350
        }
Packit Service 963350
        self->format = spad->segment.format;
Packit Service 963350
        GST_DEBUG_OBJECT (self, "Operating in %s format",
Packit Service 963350
            gst_format_get_name (self->format));
Packit Service 963350
        g_mutex_unlock (&self->lock);
Packit Service 963350
      } else if (self->format != spad->segment.format) {
Packit Service 963350
        g_mutex_unlock (&self->lock);
Packit Service 963350
        GST_ELEMENT_ERROR (self, CORE, FAILED, (NULL),
Packit Service 963350
            ("Operating in %s format but new pad has %s",
Packit Service 963350
                gst_format_get_name (self->format),
Packit Service 963350
                gst_format_get_name (spad->segment.format)));
Packit Service 963350
        ret = FALSE;
Packit Service 963350
      } else {
Packit Service 963350
        g_mutex_unlock (&self->lock);
Packit Service 963350
      }
Packit Service 963350
Packit Service 963350
      if (!gst_concat_pad_wait (spad, self)) {
Packit Service 963350
        ret = FALSE;
Packit Service 963350
      } else {
Packit Service 963350
        GstSegment segment = spad->segment;
Packit Service 963350
Packit Service 963350
        if (adjust_base) {
Packit Service 963350
          /* We know no duration */
Packit Service 963350
          segment.duration = -1;
Packit Service 963350
Packit Service 963350
          /* Update segment values to be continous with last stream */
Packit Service 963350
          if (self->format == GST_FORMAT_TIME) {
Packit Service 963350
            segment.base += self->current_start_offset;
Packit Service 963350
          } else {
Packit Service 963350
            /* Shift start/stop byte position */
Packit Service 963350
            segment.start += self->current_start_offset;
Packit Service 963350
            if (segment.stop != -1)
Packit Service 963350
              segment.stop += self->current_start_offset;
Packit Service 963350
          }
Packit Service 963350
        }
Packit Service 963350
Packit Service 963350
        gst_pad_push_event (self->srcpad, gst_event_new_segment (&segment));
Packit Service 963350
      }
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    case GST_EVENT_EOS:{
Packit Service 963350
      gst_event_unref (event);
Packit Service 963350
Packit Service 963350
      if (!gst_concat_pad_wait (spad, self)) {
Packit Service 963350
        ret = FALSE;
Packit Service 963350
      } else {
Packit Service 963350
        gboolean next;
Packit Service 963350
Packit Service 963350
        g_mutex_lock (&self->lock);
Packit Service 963350
        next = gst_concat_switch_pad (self);
Packit Service 963350
        g_mutex_unlock (&self->lock);
Packit Service 963350
        ret = TRUE;
Packit Service 963350
Packit Service 963350
        gst_concat_notify_active_pad (self);
Packit Service 963350
Packit Service 963350
        if (!next) {
Packit Service 963350
          gst_pad_push_event (self->srcpad, gst_event_new_eos ());
Packit Service 963350
        } else {
Packit Service 963350
          gst_element_post_message (GST_ELEMENT_CAST (self),
Packit Service 963350
              gst_message_new_duration_changed (GST_OBJECT_CAST (self)));
Packit Service 963350
        }
Packit Service 963350
      }
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    case GST_EVENT_FLUSH_START:{
Packit Service 963350
      gboolean forward;
Packit Service 963350
Packit Service 963350
      g_mutex_lock (&self->lock);
Packit Service 963350
      spad->flushing = TRUE;
Packit Service 963350
      g_cond_broadcast (&self->cond);
Packit Service 963350
      forward = (self->current_sinkpad == GST_PAD_CAST (spad));
Packit Service 963350
      if (!forward && g_list_length (self->sinkpads) == 1)
Packit Service 963350
        forward = TRUE;
Packit Service 963350
      g_mutex_unlock (&self->lock);
Packit Service 963350
Packit Service 963350
      if (forward)
Packit Service 963350
        ret = gst_pad_event_default (pad, parent, event);
Packit Service 963350
      else
Packit Service 963350
        gst_event_unref (event);
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    case GST_EVENT_FLUSH_STOP:{
Packit Service 963350
      gboolean forward;
Packit Service 963350
Packit Service 963350
      gst_segment_init (&spad->segment, GST_FORMAT_UNDEFINED);
Packit Service 963350
      spad->flushing = FALSE;
Packit Service 963350
Packit Service 963350
      g_mutex_lock (&self->lock);
Packit Service 963350
      forward = (self->current_sinkpad == GST_PAD_CAST (spad));
Packit Service 963350
      if (!forward && g_list_length (self->sinkpads) == 1)
Packit Service 963350
        forward = TRUE;
Packit Service 963350
      g_mutex_unlock (&self->lock);
Packit Service 963350
Packit Service 963350
      if (forward) {
Packit Service 963350
        gboolean reset_time;
Packit Service 963350
Packit Service 963350
        gst_event_parse_flush_stop (event, &reset_time);
Packit Service 963350
        if (reset_time) {
Packit Service 963350
          GST_DEBUG_OBJECT (self,
Packit Service 963350
              "resetting start offset to 0 after flushing with reset_time = TRUE");
Packit Service 963350
          self->current_start_offset = 0;
Packit Service 963350
        }
Packit Service 963350
        ret = gst_pad_event_default (pad, parent, event);
Packit Service 963350
      } else {
Packit Service 963350
        gst_event_unref (event);
Packit Service 963350
      }
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    default:{
Packit Service 963350
      /* Wait for other serialized events before forwarding */
Packit Service 963350
      if (GST_EVENT_IS_SERIALIZED (event) && !gst_concat_pad_wait (spad, self)) {
Packit Service 963350
        gst_event_unref (event);
Packit Service 963350
        ret = FALSE;
Packit Service 963350
      } else {
Packit Service 963350
        ret = gst_pad_event_default (pad, parent, event);
Packit Service 963350
      }
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static gboolean
Packit Service 963350
gst_concat_sink_query (GstPad * pad, GstObject * parent, GstQuery * query)
Packit Service 963350
{
Packit Service 963350
  GstConcat *self = GST_CONCAT (parent);
Packit Service 963350
  GstConcatPad *spad = GST_CONCAT_PAD_CAST (pad);
Packit Service 963350
  gboolean ret = TRUE;
Packit Service 963350
Packit Service 963350
  GST_LOG_OBJECT (pad, "received query %" GST_PTR_FORMAT, query);
Packit Service 963350
Packit Service 963350
  switch (GST_QUERY_TYPE (query)) {
Packit Service 963350
    default:
Packit Service 963350
      /* Wait for other serialized queries before forwarding */
Packit Service 963350
      if (GST_QUERY_IS_SERIALIZED (query) && !gst_concat_pad_wait (spad, self)) {
Packit Service 963350
        ret = FALSE;
Packit Service 963350
      } else {
Packit Service 963350
        ret = gst_pad_query_default (pad, parent, query);
Packit Service 963350
      }
Packit Service 963350
      break;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static gboolean
Packit Service 963350
gst_concat_src_event (GstPad * pad, GstObject * parent, GstEvent * event)
Packit Service 963350
{
Packit Service 963350
  GstConcat *self = GST_CONCAT (parent);
Packit Service 963350
  gboolean ret = TRUE;
Packit Service 963350
Packit Service 963350
  GST_LOG_OBJECT (pad, "received event %" GST_PTR_FORMAT, event);
Packit Service 963350
Packit Service 963350
  switch (GST_EVENT_TYPE (event)) {
Packit Service 963350
    case GST_EVENT_SEEK:{
Packit Service 963350
      GstPad *sinkpad = NULL;
Packit Service 963350
Packit Service 963350
      g_mutex_lock (&self->lock);
Packit Service 963350
      if ((sinkpad = self->current_sinkpad))
Packit Service 963350
        gst_object_ref (sinkpad);
Packit Service 963350
      /* If no current active sinkpad but only one sinkpad, try reactivating that pad */
Packit Service 963350
      if (sinkpad == NULL && g_list_length (self->sinkpads) == 1) {
Packit Service 963350
        sinkpad = gst_object_ref (self->sinkpads->data);
Packit Service 963350
      }
Packit Service 963350
      g_mutex_unlock (&self->lock);
Packit Service 963350
      if (sinkpad) {
Packit Service 963350
        ret = gst_pad_push_event (sinkpad, event);
Packit Service 963350
        gst_object_unref (sinkpad);
Packit Service 963350
      } else {
Packit Service 963350
        gst_event_unref (event);
Packit Service 963350
        ret = FALSE;
Packit Service 963350
      }
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    case GST_EVENT_QOS:{
Packit Service 963350
      GstQOSType type;
Packit Service 963350
      GstClockTimeDiff diff;
Packit Service 963350
      GstClockTime timestamp;
Packit Service 963350
      gdouble proportion;
Packit Service 963350
      GstPad *sinkpad = NULL;
Packit Service 963350
Packit Service 963350
      g_mutex_lock (&self->lock);
Packit Service 963350
      if ((sinkpad = self->current_sinkpad))
Packit Service 963350
        gst_object_ref (sinkpad);
Packit Service 963350
      g_mutex_unlock (&self->lock);
Packit Service 963350
Packit Service 963350
      if (sinkpad) {
Packit Service 963350
        gst_event_parse_qos (event, &type, &proportion, &diff, &timestamp);
Packit Service 963350
        gst_event_unref (event);
Packit Service 963350
Packit Service 963350
        if (timestamp != GST_CLOCK_TIME_NONE
Packit Service 963350
            && timestamp > self->current_start_offset) {
Packit Service 963350
          timestamp -= self->current_start_offset;
Packit Service 963350
          event = gst_event_new_qos (type, proportion, diff, timestamp);
Packit Service 963350
          ret = gst_pad_push_event (self->current_sinkpad, event);
Packit Service 963350
        } else {
Packit Service 963350
          ret = FALSE;
Packit Service 963350
        }
Packit Service 963350
        gst_object_unref (sinkpad);
Packit Service 963350
      } else {
Packit Service 963350
        gst_event_unref (event);
Packit Service 963350
        ret = FALSE;
Packit Service 963350
      }
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    case GST_EVENT_FLUSH_STOP:{
Packit Service 963350
      gboolean reset_time;
Packit Service 963350
Packit Service 963350
      gst_event_parse_flush_stop (event, &reset_time);
Packit Service 963350
      if (reset_time) {
Packit Service 963350
        GST_DEBUG_OBJECT (self,
Packit Service 963350
            "resetting start offset to 0 after flushing with reset_time = TRUE");
Packit Service 963350
        self->current_start_offset = 0;
Packit Service 963350
      }
Packit Service 963350
Packit Service 963350
      ret = gst_pad_event_default (pad, parent, event);
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    default:
Packit Service 963350
      ret = gst_pad_event_default (pad, parent, event);
Packit Service 963350
      break;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static gboolean
Packit Service 963350
gst_concat_src_query (GstPad * pad, GstObject * parent, GstQuery * query)
Packit Service 963350
{
Packit Service 963350
  gboolean ret = TRUE;
Packit Service 963350
Packit Service 963350
  GST_LOG_OBJECT (pad, "received query %" GST_PTR_FORMAT, query);
Packit Service 963350
Packit Service 963350
  switch (GST_QUERY_TYPE (query)) {
Packit Service 963350
    default:
Packit Service 963350
      ret = gst_pad_query_default (pad, parent, query);
Packit Service 963350
      break;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
reset_pad (const GValue * data, gpointer user_data)
Packit Service 963350
{
Packit Service 963350
  GstPad *pad = g_value_get_object (data);
Packit Service 963350
  GstConcatPad *spad = GST_CONCAT_PAD_CAST (pad);
Packit Service 963350
Packit Service 963350
  gst_segment_init (&spad->segment, GST_FORMAT_UNDEFINED);
Packit Service 963350
  spad->flushing = FALSE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
unblock_pad (const GValue * data, gpointer user_data)
Packit Service 963350
{
Packit Service 963350
  GstPad *pad = g_value_get_object (data);
Packit Service 963350
  GstConcatPad *spad = GST_CONCAT_PAD_CAST (pad);
Packit Service 963350
Packit Service 963350
  spad->flushing = TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static GstStateChangeReturn
Packit Service 963350
gst_concat_change_state (GstElement * element, GstStateChange transition)
Packit Service 963350
{
Packit Service 963350
  GstConcat *self = GST_CONCAT (element);
Packit Service 963350
  GstStateChangeReturn ret;
Packit Service 963350
Packit Service 963350
  switch (transition) {
Packit Service 963350
    case GST_STATE_CHANGE_READY_TO_PAUSED:{
Packit Service 963350
      GstIterator *iter = gst_element_iterate_sink_pads (element);
Packit Service 963350
      GstIteratorResult res;
Packit Service 963350
Packit Service 963350
      self->format = GST_FORMAT_UNDEFINED;
Packit Service 963350
      self->current_start_offset = 0;
Packit Service 963350
      self->last_stop = GST_CLOCK_TIME_NONE;
Packit Service 963350
Packit Service 963350
      while ((res =
Packit Service 963350
              gst_iterator_foreach (iter, reset_pad,
Packit Service 963350
                  NULL)) == GST_ITERATOR_RESYNC)
Packit Service 963350
        gst_iterator_resync (iter);
Packit Service 963350
      gst_iterator_free (iter);
Packit Service 963350
Packit Service 963350
      if (res == GST_ITERATOR_ERROR)
Packit Service 963350
        return GST_STATE_CHANGE_FAILURE;
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    case GST_STATE_CHANGE_PAUSED_TO_READY:{
Packit Service 963350
      GstIterator *iter = gst_element_iterate_sink_pads (element);
Packit Service 963350
      GstIteratorResult res;
Packit Service 963350
Packit Service 963350
      g_mutex_lock (&self->lock);
Packit Service 963350
      while ((res =
Packit Service 963350
              gst_iterator_foreach (iter, unblock_pad,
Packit Service 963350
                  NULL)) == GST_ITERATOR_RESYNC)
Packit Service 963350
        gst_iterator_resync (iter);
Packit Service 963350
      gst_iterator_free (iter);
Packit Service 963350
      g_cond_broadcast (&self->cond);
Packit Service 963350
      g_mutex_unlock (&self->lock);
Packit Service 963350
Packit Service 963350
      if (res == GST_ITERATOR_ERROR)
Packit Service 963350
        return GST_STATE_CHANGE_FAILURE;
Packit Service 963350
Packit Service 963350
      break;
Packit Service 963350
    }
Packit Service 963350
    default:
Packit Service 963350
      break;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}