2014-12-30 23:58:34 +00:00
|
|
|
/* GStreamer aggregator base class
|
2014-05-22 17:44:37 +00:00
|
|
|
* Copyright (C) 2014 Mathieu Duponchelle <mathieu.duponchelle@opencreed.com>
|
|
|
|
* Copyright (C) 2014 Thibault Saunier <tsaunier@gnome.org>
|
|
|
|
*
|
|
|
|
* gstaggregator.c:
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the
|
|
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
|
|
* Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* SECTION: gstaggregator
|
2017-03-08 18:01:13 +00:00
|
|
|
* @title: GstAggregator
|
2014-05-22 17:44:37 +00:00
|
|
|
* @short_description: manages a set of pads with the purpose of
|
|
|
|
* aggregating their buffers.
|
|
|
|
* @see_also: gstcollectpads for historical reasons.
|
|
|
|
*
|
|
|
|
* Manages a set of pads with the purpose of aggregating their buffers.
|
|
|
|
* Control is given to the subclass when all pads have data.
|
2017-03-08 18:01:13 +00:00
|
|
|
*
|
|
|
|
* * Base class for mixers and muxers. Subclasses should at least implement
|
2015-01-01 15:46:00 +00:00
|
|
|
* the #GstAggregatorClass.aggregate() virtual method.
|
2017-03-08 18:01:13 +00:00
|
|
|
*
|
2017-09-17 17:18:56 +00:00
|
|
|
* * When data is queued on all pads, the aggregate vmethod is called.
|
2017-03-08 18:01:13 +00:00
|
|
|
*
|
|
|
|
* * One can peek at the data on any given GstAggregatorPad with the
|
2014-05-22 17:44:37 +00:00
|
|
|
* gst_aggregator_pad_get_buffer () method, and take ownership of it
|
|
|
|
* with the gst_aggregator_pad_steal_buffer () method. When a buffer
|
|
|
|
* has been taken with steal_buffer (), a new buffer can be queued
|
|
|
|
* on that pad.
|
2017-03-08 18:01:13 +00:00
|
|
|
*
|
|
|
|
* * If the subclass wishes to push a buffer downstream in its aggregate
|
2014-05-22 17:44:37 +00:00
|
|
|
* implementation, it should do so through the
|
|
|
|
* gst_aggregator_finish_buffer () method. This method will take care
|
|
|
|
* of sending and ordering mandatory events such as stream start, caps
|
|
|
|
* and segment.
|
2017-03-08 18:01:13 +00:00
|
|
|
*
|
|
|
|
* * Same goes for EOS events, which should not be pushed directly by the
|
2014-05-22 17:44:37 +00:00
|
|
|
* subclass, it should instead return GST_FLOW_EOS in its aggregate
|
|
|
|
* implementation.
|
2017-03-08 18:01:13 +00:00
|
|
|
*
|
|
|
|
* * Note that the aggregator logic regarding gap event handling is to turn
|
2015-03-29 20:53:23 +00:00
|
|
|
* these into gap buffers with matching PTS and duration. It will also
|
|
|
|
* flag these buffers with GST_BUFFER_FLAG_GAP and GST_BUFFER_FLAG_DROPPABLE
|
|
|
|
* to ease their identification and subsequent processing.
|
2017-03-08 18:01:13 +00:00
|
|
|
*
|
2014-05-22 17:44:37 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <string.h> /* strlen */
|
|
|
|
|
|
|
|
#include "gstaggregator.h"
|
|
|
|
|
2015-06-15 16:30:20 +00:00
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
GST_AGGREGATOR_START_TIME_SELECTION_ZERO,
|
|
|
|
GST_AGGREGATOR_START_TIME_SELECTION_FIRST,
|
|
|
|
GST_AGGREGATOR_START_TIME_SELECTION_SET
|
|
|
|
} GstAggregatorStartTimeSelection;
|
|
|
|
|
|
|
|
static GType
|
|
|
|
gst_aggregator_start_time_selection_get_type (void)
|
|
|
|
{
|
|
|
|
static GType gtype = 0;
|
|
|
|
|
|
|
|
if (gtype == 0) {
|
|
|
|
static const GEnumValue values[] = {
|
|
|
|
{GST_AGGREGATOR_START_TIME_SELECTION_ZERO,
|
|
|
|
"Start at 0 running time (default)", "zero"},
|
|
|
|
{GST_AGGREGATOR_START_TIME_SELECTION_FIRST,
|
|
|
|
"Start at first observed input running time", "first"},
|
|
|
|
{GST_AGGREGATOR_START_TIME_SELECTION_SET,
|
|
|
|
"Set start time with start-time property", "set"},
|
|
|
|
{0, NULL, NULL}
|
|
|
|
};
|
|
|
|
|
|
|
|
gtype = g_enum_register_static ("GstAggregatorStartTimeSelection", values);
|
|
|
|
}
|
|
|
|
return gtype;
|
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
/* Might become API */
|
|
|
|
static void gst_aggregator_merge_tags (GstAggregator * aggregator,
|
|
|
|
const GstTagList * tags, GstTagMergeMode mode);
|
2014-12-05 07:19:54 +00:00
|
|
|
static void gst_aggregator_set_latency_property (GstAggregator * agg,
|
|
|
|
gint64 latency);
|
|
|
|
static gint64 gst_aggregator_get_latency_property (GstAggregator * agg);
|
2014-10-06 07:23:03 +00:00
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-02-18 20:53:53 +00:00
|
|
|
/* Locking order, locks in this element must always be taken in this order
|
|
|
|
*
|
|
|
|
* standard sink pad stream lock -> GST_PAD_STREAM_LOCK (aggpad)
|
|
|
|
* Aggregator pad flush lock -> PAD_FLUSH_LOCK(aggpad)
|
|
|
|
* standard src pad stream lock -> GST_PAD_STREAM_LOCK (srcpad)
|
|
|
|
* Aggregator src lock -> SRC_LOCK(agg) w/ SRC_WAIT/BROADCAST
|
|
|
|
* standard element object lock -> GST_OBJECT_LOCK(agg)
|
|
|
|
* Aggregator pad lock -> PAD_LOCK (aggpad) w/ PAD_WAIT/BROADCAST_EVENT(aggpad)
|
|
|
|
* standard src pad object lock -> GST_OBJECT_LOCK(srcpad)
|
|
|
|
* standard sink pad object lock -> GST_OBJECT_LOCK(aggpad)
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
static GstClockTime gst_aggregator_get_latency_unlocked (GstAggregator * self);
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
GST_DEBUG_CATEGORY_STATIC (aggregator_debug);
|
|
|
|
#define GST_CAT_DEFAULT aggregator_debug
|
|
|
|
|
|
|
|
/* GstAggregatorPad definitions */
|
2015-02-18 20:04:04 +00:00
|
|
|
#define PAD_LOCK(pad) G_STMT_START { \
|
|
|
|
GST_TRACE_OBJECT (pad, "Taking PAD lock from thread %p", \
|
2014-05-22 17:44:37 +00:00
|
|
|
g_thread_self()); \
|
2015-02-18 20:04:04 +00:00
|
|
|
g_mutex_lock(&pad->priv->lock); \
|
|
|
|
GST_TRACE_OBJECT (pad, "Took PAD lock from thread %p", \
|
2014-05-22 17:44:37 +00:00
|
|
|
g_thread_self()); \
|
|
|
|
} G_STMT_END
|
|
|
|
|
2015-02-18 20:04:04 +00:00
|
|
|
#define PAD_UNLOCK(pad) G_STMT_START { \
|
|
|
|
GST_TRACE_OBJECT (pad, "Releasing PAD lock from thread %p", \
|
|
|
|
g_thread_self()); \
|
|
|
|
g_mutex_unlock(&pad->priv->lock); \
|
|
|
|
GST_TRACE_OBJECT (pad, "Release PAD lock from thread %p", \
|
2014-05-22 17:44:37 +00:00
|
|
|
g_thread_self()); \
|
|
|
|
} G_STMT_END
|
|
|
|
|
|
|
|
|
|
|
|
#define PAD_WAIT_EVENT(pad) G_STMT_START { \
|
2015-04-02 01:45:01 +00:00
|
|
|
GST_LOG_OBJECT (pad, "Waiting for buffer to be consumed thread %p", \
|
2014-05-22 17:44:37 +00:00
|
|
|
g_thread_self()); \
|
2015-01-01 14:03:02 +00:00
|
|
|
g_cond_wait(&(((GstAggregatorPad* )pad)->priv->event_cond), \
|
2015-02-18 20:04:04 +00:00
|
|
|
(&((GstAggregatorPad*)pad)->priv->lock)); \
|
2015-04-02 01:45:01 +00:00
|
|
|
GST_LOG_OBJECT (pad, "DONE Waiting for buffer to be consumed on thread %p", \
|
2014-05-22 17:44:37 +00:00
|
|
|
g_thread_self()); \
|
|
|
|
} G_STMT_END
|
|
|
|
|
2015-01-01 14:03:02 +00:00
|
|
|
#define PAD_BROADCAST_EVENT(pad) G_STMT_START { \
|
2015-04-02 01:45:01 +00:00
|
|
|
GST_LOG_OBJECT (pad, "Signaling buffer consumed from thread %p", \
|
2015-01-01 14:03:02 +00:00
|
|
|
g_thread_self()); \
|
|
|
|
g_cond_broadcast(&(((GstAggregatorPad* )pad)->priv->event_cond)); \
|
|
|
|
} G_STMT_END
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2014-10-03 10:34:15 +00:00
|
|
|
|
2015-02-18 20:06:01 +00:00
|
|
|
#define PAD_FLUSH_LOCK(pad) G_STMT_START { \
|
2015-01-01 14:03:02 +00:00
|
|
|
GST_TRACE_OBJECT (pad, "Taking lock from thread %p", \
|
2014-08-02 16:25:01 +00:00
|
|
|
g_thread_self()); \
|
2015-02-18 20:06:01 +00:00
|
|
|
g_mutex_lock(&pad->priv->flush_lock); \
|
2015-01-01 14:03:02 +00:00
|
|
|
GST_TRACE_OBJECT (pad, "Took lock from thread %p", \
|
2014-08-02 16:25:01 +00:00
|
|
|
g_thread_self()); \
|
|
|
|
} G_STMT_END
|
|
|
|
|
2015-02-18 20:06:01 +00:00
|
|
|
#define PAD_FLUSH_UNLOCK(pad) G_STMT_START { \
|
2015-01-01 14:03:02 +00:00
|
|
|
GST_TRACE_OBJECT (pad, "Releasing lock from thread %p", \
|
2014-08-02 16:25:01 +00:00
|
|
|
g_thread_self()); \
|
2015-02-18 20:06:01 +00:00
|
|
|
g_mutex_unlock(&pad->priv->flush_lock); \
|
2015-01-01 14:03:02 +00:00
|
|
|
GST_TRACE_OBJECT (pad, "Release lock from thread %p", \
|
2014-08-02 16:25:01 +00:00
|
|
|
g_thread_self()); \
|
|
|
|
} G_STMT_END
|
|
|
|
|
2015-02-18 20:11:14 +00:00
|
|
|
#define SRC_LOCK(self) G_STMT_START { \
|
|
|
|
GST_TRACE_OBJECT (self, "Taking src lock from thread %p", \
|
|
|
|
g_thread_self()); \
|
|
|
|
g_mutex_lock(&self->priv->src_lock); \
|
|
|
|
GST_TRACE_OBJECT (self, "Took src lock from thread %p", \
|
|
|
|
g_thread_self()); \
|
2014-12-05 07:19:54 +00:00
|
|
|
} G_STMT_END
|
|
|
|
|
2015-02-18 20:11:14 +00:00
|
|
|
#define SRC_UNLOCK(self) G_STMT_START { \
|
|
|
|
GST_TRACE_OBJECT (self, "Releasing src lock from thread %p", \
|
|
|
|
g_thread_self()); \
|
|
|
|
g_mutex_unlock(&self->priv->src_lock); \
|
|
|
|
GST_TRACE_OBJECT (self, "Released src lock from thread %p", \
|
|
|
|
g_thread_self()); \
|
2014-12-05 07:19:54 +00:00
|
|
|
} G_STMT_END
|
|
|
|
|
2015-02-18 20:11:14 +00:00
|
|
|
#define SRC_WAIT(self) G_STMT_START { \
|
|
|
|
GST_LOG_OBJECT (self, "Waiting for src on thread %p", \
|
|
|
|
g_thread_self()); \
|
|
|
|
g_cond_wait(&(self->priv->src_cond), &(self->priv->src_lock)); \
|
|
|
|
GST_LOG_OBJECT (self, "DONE Waiting for src on thread %p", \
|
|
|
|
g_thread_self()); \
|
2014-12-05 07:19:54 +00:00
|
|
|
} G_STMT_END
|
|
|
|
|
2015-02-18 20:11:14 +00:00
|
|
|
#define SRC_BROADCAST(self) G_STMT_START { \
|
|
|
|
GST_LOG_OBJECT (self, "Signaling src from thread %p", \
|
|
|
|
g_thread_self()); \
|
|
|
|
if (self->priv->aggregate_id) \
|
|
|
|
gst_clock_id_unschedule (self->priv->aggregate_id); \
|
|
|
|
g_cond_broadcast(&(self->priv->src_cond)); \
|
2014-12-26 22:49:52 +00:00
|
|
|
} G_STMT_END
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
struct _GstAggregatorPadPrivate
|
|
|
|
{
|
2015-01-26 16:06:29 +00:00
|
|
|
/* Following fields are protected by the PAD_LOCK */
|
2015-04-02 02:10:11 +00:00
|
|
|
GstFlowReturn flow_return;
|
2014-05-22 17:44:37 +00:00
|
|
|
gboolean pending_flush_start;
|
|
|
|
gboolean pending_flush_stop;
|
|
|
|
gboolean pending_eos;
|
|
|
|
|
2015-08-31 13:12:40 +00:00
|
|
|
gboolean first_buffer;
|
|
|
|
|
2017-09-17 18:39:12 +00:00
|
|
|
GQueue data; /* buffers, events and queries */
|
2016-07-06 20:39:17 +00:00
|
|
|
GstBuffer *clipped_buffer;
|
2015-09-17 23:42:34 +00:00
|
|
|
guint num_buffers;
|
2015-03-07 00:50:08 +00:00
|
|
|
GstClockTime head_position;
|
|
|
|
GstClockTime tail_position;
|
2017-09-17 19:24:54 +00:00
|
|
|
GstClockTime head_time; /* running time */
|
2015-03-07 00:50:08 +00:00
|
|
|
GstClockTime tail_time;
|
2017-09-17 19:24:54 +00:00
|
|
|
GstClockTime time_level; /* how much head is ahead of tail */
|
2016-07-06 20:39:17 +00:00
|
|
|
GstSegment head_segment; /* segment before the queue */
|
2015-03-07 00:50:08 +00:00
|
|
|
|
2017-05-21 12:34:13 +00:00
|
|
|
gboolean negotiated;
|
|
|
|
|
2015-01-26 10:25:54 +00:00
|
|
|
gboolean eos;
|
|
|
|
|
2015-01-26 10:29:08 +00:00
|
|
|
GMutex lock;
|
2014-05-22 17:44:37 +00:00
|
|
|
GCond event_cond;
|
2015-02-18 20:06:01 +00:00
|
|
|
/* This lock prevents a flush start processing happening while
|
|
|
|
* the chain function is also happening.
|
|
|
|
*/
|
|
|
|
GMutex flush_lock;
|
2014-05-22 17:44:37 +00:00
|
|
|
};
|
|
|
|
|
2017-05-10 00:05:55 +00:00
|
|
|
/* Must be called with PAD_LOCK held */
|
|
|
|
static void
|
|
|
|
gst_aggregator_pad_reset_unlocked (GstAggregatorPad * aggpad)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2015-04-02 01:38:11 +00:00
|
|
|
aggpad->priv->pending_eos = FALSE;
|
2015-01-26 10:25:54 +00:00
|
|
|
aggpad->priv->eos = FALSE;
|
2015-04-02 02:10:11 +00:00
|
|
|
aggpad->priv->flow_return = GST_FLOW_OK;
|
2015-03-07 00:50:08 +00:00
|
|
|
GST_OBJECT_LOCK (aggpad);
|
|
|
|
gst_segment_init (&aggpad->segment, GST_FORMAT_UNDEFINED);
|
2016-07-06 20:39:17 +00:00
|
|
|
gst_segment_init (&aggpad->priv->head_segment, GST_FORMAT_UNDEFINED);
|
2015-03-07 00:50:08 +00:00
|
|
|
GST_OBJECT_UNLOCK (aggpad);
|
|
|
|
aggpad->priv->head_position = GST_CLOCK_TIME_NONE;
|
|
|
|
aggpad->priv->tail_position = GST_CLOCK_TIME_NONE;
|
|
|
|
aggpad->priv->head_time = GST_CLOCK_TIME_NONE;
|
|
|
|
aggpad->priv->tail_time = GST_CLOCK_TIME_NONE;
|
|
|
|
aggpad->priv->time_level = 0;
|
2017-05-10 00:06:29 +00:00
|
|
|
aggpad->priv->first_buffer = TRUE;
|
2017-05-10 00:05:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gst_aggregator_pad_flush (GstAggregatorPad * aggpad, GstAggregator * agg)
|
|
|
|
{
|
|
|
|
GstAggregatorPadClass *klass = GST_AGGREGATOR_PAD_GET_CLASS (aggpad);
|
|
|
|
|
|
|
|
PAD_LOCK (aggpad);
|
|
|
|
gst_aggregator_pad_reset_unlocked (aggpad);
|
2015-01-14 19:38:09 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
if (klass->flush)
|
|
|
|
return klass->flush (aggpad, agg);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*************************************
|
|
|
|
* GstAggregator implementation *
|
|
|
|
*************************************/
|
|
|
|
static GstElementClass *aggregator_parent_class = NULL;
|
|
|
|
|
2015-01-21 23:41:43 +00:00
|
|
|
/* All members are protected by the object lock unless otherwise noted */
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
struct _GstAggregatorPrivate
|
|
|
|
{
|
2016-03-27 13:11:30 +00:00
|
|
|
gint max_padserial;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
/* Our state is >= PAUSED */
|
2015-02-18 20:11:14 +00:00
|
|
|
gboolean running; /* protected by src_lock */
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2014-07-08 14:48:08 +00:00
|
|
|
gint seqnum;
|
2015-01-21 23:41:43 +00:00
|
|
|
gboolean send_stream_start; /* protected by srcpad stream lock */
|
2014-05-22 17:44:37 +00:00
|
|
|
gboolean send_segment;
|
|
|
|
gboolean flush_seeking;
|
|
|
|
gboolean pending_flush_start;
|
2015-01-21 23:53:20 +00:00
|
|
|
gboolean send_eos; /* protected by srcpad stream lock */
|
|
|
|
|
2015-01-22 00:35:25 +00:00
|
|
|
GstCaps *srccaps; /* protected by the srcpad stream lock */
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
GstTagList *tags;
|
|
|
|
gboolean tags_changed;
|
2014-10-03 10:34:15 +00:00
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
gboolean peer_latency_live; /* protected by src_lock */
|
|
|
|
GstClockTime peer_latency_min; /* protected by src_lock */
|
|
|
|
GstClockTime peer_latency_max; /* protected by src_lock */
|
2015-11-03 19:37:26 +00:00
|
|
|
gboolean has_peer_latency; /* protected by src_lock */
|
2014-12-05 07:19:54 +00:00
|
|
|
|
2015-02-20 02:21:56 +00:00
|
|
|
GstClockTime sub_latency_min; /* protected by src_lock */
|
|
|
|
GstClockTime sub_latency_max; /* protected by src_lock */
|
2014-12-17 18:51:32 +00:00
|
|
|
|
2014-12-05 07:19:54 +00:00
|
|
|
/* aggregate */
|
2015-01-21 23:41:43 +00:00
|
|
|
GstClockID aggregate_id; /* protected by src_lock */
|
2014-12-05 07:19:54 +00:00
|
|
|
GMutex src_lock;
|
|
|
|
GCond src_cond;
|
2014-12-31 18:16:21 +00:00
|
|
|
|
2015-11-03 19:37:26 +00:00
|
|
|
gboolean first_buffer; /* protected by object lock */
|
2015-06-15 16:30:20 +00:00
|
|
|
GstAggregatorStartTimeSelection start_time_selection;
|
|
|
|
GstClockTime start_time;
|
|
|
|
|
2017-05-20 14:58:54 +00:00
|
|
|
/* protected by the object lock */
|
|
|
|
GstQuery *allocation_query;
|
|
|
|
GstAllocator *allocator;
|
|
|
|
GstBufferPool *pool;
|
|
|
|
GstAllocationParams allocation_params;
|
|
|
|
|
2014-12-31 18:16:21 +00:00
|
|
|
/* properties */
|
2015-03-07 00:50:08 +00:00
|
|
|
gint64 latency; /* protected by both src_lock and all pad locks */
|
2014-05-22 17:44:37 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
GstEvent *event;
|
|
|
|
gboolean result;
|
|
|
|
gboolean flush;
|
2015-08-31 13:12:40 +00:00
|
|
|
gboolean only_to_active_pads;
|
2014-07-18 11:58:55 +00:00
|
|
|
|
|
|
|
gboolean one_actually_seeked;
|
2014-05-22 17:44:37 +00:00
|
|
|
} EventData;
|
|
|
|
|
2015-06-15 16:30:20 +00:00
|
|
|
#define DEFAULT_LATENCY 0
|
2015-07-30 00:07:09 +00:00
|
|
|
#define DEFAULT_START_TIME_SELECTION GST_AGGREGATOR_START_TIME_SELECTION_ZERO
|
2015-06-15 16:30:20 +00:00
|
|
|
#define DEFAULT_START_TIME (-1)
|
2014-10-06 07:23:03 +00:00
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
2014-12-05 07:19:54 +00:00
|
|
|
PROP_LATENCY,
|
2015-06-15 16:30:20 +00:00
|
|
|
PROP_START_TIME_SELECTION,
|
|
|
|
PROP_START_TIME,
|
2014-10-06 07:23:03 +00:00
|
|
|
PROP_LAST
|
|
|
|
};
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
static GstFlowReturn gst_aggregator_pad_chain_internal (GstAggregator * self,
|
|
|
|
GstAggregatorPad * aggpad, GstBuffer * buffer, gboolean head);
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
/**
|
|
|
|
* gst_aggregator_iterate_sinkpads:
|
|
|
|
* @self: The #GstAggregator
|
2015-01-01 15:46:00 +00:00
|
|
|
* @func: (scope call): The function to call.
|
|
|
|
* @user_data: (closure): The data to pass to @func.
|
2014-05-22 17:44:37 +00:00
|
|
|
*
|
|
|
|
* Iterate the sinkpads of aggregator to call a function on them.
|
|
|
|
*
|
|
|
|
* This method guarantees that @func will be called only once for each
|
|
|
|
* sink pad.
|
2017-07-13 19:55:55 +00:00
|
|
|
*
|
|
|
|
* Returns: %FALSE if there are no sinkpads or if @func returned %FALSE
|
2014-05-22 17:44:37 +00:00
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
gst_aggregator_iterate_sinkpads (GstAggregator * self,
|
|
|
|
GstAggregatorPadForeachFunc func, gpointer user_data)
|
|
|
|
{
|
|
|
|
gboolean result = FALSE;
|
|
|
|
GstIterator *iter;
|
|
|
|
gboolean done = FALSE;
|
|
|
|
GValue item = { 0, };
|
|
|
|
GList *seen_pads = NULL;
|
|
|
|
|
|
|
|
iter = gst_element_iterate_sink_pads (GST_ELEMENT (self));
|
|
|
|
|
|
|
|
if (!iter)
|
|
|
|
goto no_iter;
|
|
|
|
|
|
|
|
while (!done) {
|
|
|
|
switch (gst_iterator_next (iter, &item)) {
|
|
|
|
case GST_ITERATOR_OK:
|
|
|
|
{
|
2014-12-30 19:22:01 +00:00
|
|
|
GstAggregatorPad *pad;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
pad = g_value_get_object (&item);
|
|
|
|
|
|
|
|
/* if already pushed, skip. FIXME, find something faster to tag pads */
|
|
|
|
if (pad == NULL || g_list_find (seen_pads, pad)) {
|
|
|
|
g_value_reset (&item);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-12-30 19:22:01 +00:00
|
|
|
GST_LOG_OBJECT (pad, "calling function %s on pad",
|
|
|
|
GST_DEBUG_FUNCPTR_NAME (func));
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
result = func (self, pad, user_data);
|
|
|
|
|
|
|
|
done = !result;
|
|
|
|
|
|
|
|
seen_pads = g_list_prepend (seen_pads, pad);
|
|
|
|
|
|
|
|
g_value_reset (&item);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case GST_ITERATOR_RESYNC:
|
|
|
|
gst_iterator_resync (iter);
|
|
|
|
break;
|
|
|
|
case GST_ITERATOR_ERROR:
|
|
|
|
GST_ERROR_OBJECT (self,
|
|
|
|
"Could not iterate over internally linked pads");
|
|
|
|
done = TRUE;
|
|
|
|
break;
|
|
|
|
case GST_ITERATOR_DONE:
|
|
|
|
done = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_value_unset (&item);
|
|
|
|
gst_iterator_free (iter);
|
|
|
|
|
|
|
|
if (seen_pads == NULL) {
|
|
|
|
GST_DEBUG_OBJECT (self, "No pad seen");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_list_free (seen_pads);
|
|
|
|
|
|
|
|
no_iter:
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
static gboolean
|
|
|
|
gst_aggregator_pad_queue_is_empty (GstAggregatorPad * pad)
|
|
|
|
{
|
2017-09-17 18:39:12 +00:00
|
|
|
return (g_queue_peek_tail (&pad->priv->data) == NULL &&
|
2016-07-06 20:39:17 +00:00
|
|
|
pad->priv->clipped_buffer == NULL);
|
2015-03-07 00:50:08 +00:00
|
|
|
}
|
|
|
|
|
2015-01-04 16:57:05 +00:00
|
|
|
static gboolean
|
|
|
|
gst_aggregator_check_pads_ready (GstAggregator * self)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2015-01-04 16:57:05 +00:00
|
|
|
GstAggregatorPad *pad;
|
|
|
|
GList *l, *sinkpads;
|
2016-05-15 13:04:58 +00:00
|
|
|
gboolean have_buffer = TRUE;
|
|
|
|
gboolean have_event = FALSE;
|
2015-01-04 16:57:05 +00:00
|
|
|
|
|
|
|
GST_LOG_OBJECT (self, "checking pads");
|
|
|
|
|
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
|
|
|
|
sinkpads = GST_ELEMENT_CAST (self)->sinkpads;
|
|
|
|
if (sinkpads == NULL)
|
|
|
|
goto no_sinkpads;
|
|
|
|
|
|
|
|
for (l = sinkpads; l != NULL; l = l->next) {
|
|
|
|
pad = l->data;
|
|
|
|
|
2015-01-14 19:38:09 +00:00
|
|
|
PAD_LOCK (pad);
|
2015-06-15 16:30:20 +00:00
|
|
|
|
2016-05-15 13:04:58 +00:00
|
|
|
if (pad->priv->num_buffers == 0) {
|
|
|
|
if (!gst_aggregator_pad_queue_is_empty (pad))
|
|
|
|
have_event = TRUE;
|
2016-04-22 14:15:39 +00:00
|
|
|
if (!pad->priv->eos) {
|
2016-05-15 13:04:58 +00:00
|
|
|
have_buffer = FALSE;
|
2015-06-15 16:30:20 +00:00
|
|
|
|
2016-04-22 14:15:39 +00:00
|
|
|
/* If not live we need data on all pads, so leave the loop */
|
|
|
|
if (!self->priv->peer_latency_live) {
|
|
|
|
PAD_UNLOCK (pad);
|
|
|
|
goto pad_not_ready;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (self->priv->peer_latency_live) {
|
|
|
|
/* In live mode, having a single pad with buffers is enough to
|
|
|
|
* generate a start time from it. In non-live mode all pads need
|
|
|
|
* to have a buffer
|
|
|
|
*/
|
|
|
|
self->priv->first_buffer = FALSE;
|
2015-01-14 19:38:09 +00:00
|
|
|
}
|
|
|
|
|
2016-04-22 14:15:39 +00:00
|
|
|
PAD_UNLOCK (pad);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
2016-05-15 13:04:58 +00:00
|
|
|
if (!have_buffer && !have_event)
|
2016-04-22 14:15:39 +00:00
|
|
|
goto pad_not_ready;
|
|
|
|
|
2016-05-15 13:04:58 +00:00
|
|
|
if (have_buffer)
|
|
|
|
self->priv->first_buffer = FALSE;
|
2015-06-15 16:30:20 +00:00
|
|
|
|
2015-01-04 16:57:05 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
GST_LOG_OBJECT (self, "pads are ready");
|
|
|
|
return TRUE;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-01-04 16:57:05 +00:00
|
|
|
no_sinkpads:
|
|
|
|
{
|
|
|
|
GST_LOG_OBJECT (self, "pads not ready: no sink pads");
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
pad_not_ready:
|
|
|
|
{
|
2016-05-15 13:04:58 +00:00
|
|
|
if (have_event)
|
|
|
|
GST_LOG_OBJECT (pad, "pad not ready to be aggregated yet,"
|
|
|
|
" but waking up for serialized event");
|
|
|
|
else
|
|
|
|
GST_LOG_OBJECT (pad, "pad not ready to be aggregated yet");
|
2015-01-04 16:57:05 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2016-05-15 13:04:58 +00:00
|
|
|
return have_event;
|
2015-01-04 16:57:05 +00:00
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_reset_flow_values (GstAggregator * self)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
self->priv->send_stream_start = TRUE;
|
|
|
|
self->priv->send_segment = TRUE;
|
|
|
|
gst_segment_init (&self->segment, GST_FORMAT_TIME);
|
2015-06-15 16:30:20 +00:00
|
|
|
self->priv->first_buffer = TRUE;
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_push_mandatory_events (GstAggregator * self)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
GstAggregatorPrivate *priv = self->priv;
|
2015-01-22 00:33:18 +00:00
|
|
|
GstEvent *segment = NULL;
|
|
|
|
GstEvent *tags = NULL;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-01-10 02:51:40 +00:00
|
|
|
if (self->priv->send_stream_start) {
|
2014-05-22 17:44:37 +00:00
|
|
|
gchar s_id[32];
|
|
|
|
|
|
|
|
GST_INFO_OBJECT (self, "pushing stream start");
|
|
|
|
/* stream-start (FIXME: create id based on input ids) */
|
|
|
|
g_snprintf (s_id, sizeof (s_id), "agg-%08x", g_random_int ());
|
|
|
|
if (!gst_pad_push_event (self->srcpad, gst_event_new_stream_start (s_id))) {
|
|
|
|
GST_WARNING_OBJECT (self->srcpad, "Sending stream start event failed");
|
|
|
|
}
|
2015-01-10 02:51:40 +00:00
|
|
|
self->priv->send_stream_start = FALSE;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (self->priv->srccaps) {
|
|
|
|
|
|
|
|
GST_INFO_OBJECT (self, "pushing caps: %" GST_PTR_FORMAT,
|
|
|
|
self->priv->srccaps);
|
|
|
|
if (!gst_pad_push_event (self->srcpad,
|
|
|
|
gst_event_new_caps (self->priv->srccaps))) {
|
|
|
|
GST_WARNING_OBJECT (self->srcpad, "Sending caps event failed");
|
|
|
|
}
|
2014-06-26 00:53:16 +00:00
|
|
|
gst_caps_unref (self->priv->srccaps);
|
2014-05-22 17:44:37 +00:00
|
|
|
self->priv->srccaps = NULL;
|
|
|
|
}
|
|
|
|
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
if (self->priv->send_segment && !self->priv->flush_seeking) {
|
2015-01-22 00:33:18 +00:00
|
|
|
segment = gst_event_new_segment (&self->segment);
|
2015-01-21 23:45:36 +00:00
|
|
|
|
|
|
|
if (!self->priv->seqnum)
|
2015-01-22 00:33:18 +00:00
|
|
|
self->priv->seqnum = gst_event_get_seqnum (segment);
|
2015-01-21 23:45:36 +00:00
|
|
|
else
|
2015-01-22 00:33:18 +00:00
|
|
|
gst_event_set_seqnum (segment, self->priv->seqnum);
|
2015-01-21 23:45:36 +00:00
|
|
|
self->priv->send_segment = FALSE;
|
2014-07-08 14:48:08 +00:00
|
|
|
|
2015-01-22 00:33:18 +00:00
|
|
|
GST_DEBUG_OBJECT (self, "pushing segment %" GST_PTR_FORMAT, segment);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
2015-02-19 23:30:35 +00:00
|
|
|
if (priv->tags && priv->tags_changed && !self->priv->flush_seeking) {
|
2015-01-22 00:33:18 +00:00
|
|
|
tags = gst_event_new_tag (gst_tag_list_ref (priv->tags));
|
2014-05-22 17:44:37 +00:00
|
|
|
priv->tags_changed = FALSE;
|
|
|
|
}
|
2015-01-22 00:33:18 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
|
|
|
|
if (segment)
|
|
|
|
gst_pad_push_event (self->srcpad, segment);
|
|
|
|
if (tags)
|
|
|
|
gst_pad_push_event (self->srcpad, tags);
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
2014-08-07 09:54:36 +00:00
|
|
|
/**
|
|
|
|
* gst_aggregator_set_src_caps:
|
|
|
|
* @self: The #GstAggregator
|
2014-08-11 13:38:40 +00:00
|
|
|
* @caps: The #GstCaps to set on the src pad.
|
2014-08-07 09:54:36 +00:00
|
|
|
*
|
|
|
|
* Sets the caps to be used on the src pad.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gst_aggregator_set_src_caps (GstAggregator * self, GstCaps * caps)
|
|
|
|
{
|
2015-01-22 00:35:25 +00:00
|
|
|
GST_PAD_STREAM_LOCK (self->srcpad);
|
2014-08-07 09:54:36 +00:00
|
|
|
gst_caps_replace (&self->priv->srccaps, caps);
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_push_mandatory_events (self);
|
2015-01-22 00:35:25 +00:00
|
|
|
GST_PAD_STREAM_UNLOCK (self->srcpad);
|
2014-08-07 09:54:36 +00:00
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
/**
|
|
|
|
* gst_aggregator_finish_buffer:
|
|
|
|
* @self: The #GstAggregator
|
2015-01-01 15:46:00 +00:00
|
|
|
* @buffer: (transfer full): the #GstBuffer to push.
|
2014-05-22 17:44:37 +00:00
|
|
|
*
|
2015-01-01 15:46:00 +00:00
|
|
|
* This method will push the provided output buffer downstream. If needed,
|
|
|
|
* mandatory events such as stream-start, caps, and segment events will be
|
|
|
|
* sent before pushing the buffer.
|
2014-05-22 17:44:37 +00:00
|
|
|
*/
|
|
|
|
GstFlowReturn
|
|
|
|
gst_aggregator_finish_buffer (GstAggregator * self, GstBuffer * buffer)
|
|
|
|
{
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_push_mandatory_events (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
if (!self->priv->flush_seeking && gst_pad_is_active (self->srcpad)) {
|
2014-05-22 17:44:37 +00:00
|
|
|
GST_TRACE_OBJECT (self, "pushing buffer %" GST_PTR_FORMAT, buffer);
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
return gst_pad_push (self->srcpad, buffer);
|
|
|
|
} else {
|
|
|
|
GST_INFO_OBJECT (self, "Not pushing (active: %i, flushing: %i)",
|
2015-01-21 23:45:36 +00:00
|
|
|
self->priv->flush_seeking, gst_pad_is_active (self->srcpad));
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-06-28 12:32:32 +00:00
|
|
|
gst_buffer_unref (buffer);
|
2014-05-22 17:44:37 +00:00
|
|
|
return GST_FLOW_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_push_eos (GstAggregator * self)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2014-07-08 14:48:08 +00:00
|
|
|
GstEvent *event;
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_push_mandatory_events (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2014-07-08 14:48:08 +00:00
|
|
|
event = gst_event_new_eos ();
|
2015-01-26 10:32:47 +00:00
|
|
|
|
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
self->priv->send_eos = FALSE;
|
2014-07-08 14:48:08 +00:00
|
|
|
gst_event_set_seqnum (event, self->priv->seqnum);
|
2015-01-26 10:32:47 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
|
2014-07-08 14:48:08 +00:00
|
|
|
gst_pad_push_event (self->srcpad, event);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
2014-12-05 07:19:54 +00:00
|
|
|
static GstClockTime
|
|
|
|
gst_aggregator_get_next_time (GstAggregator * self)
|
|
|
|
{
|
|
|
|
GstAggregatorClass *klass = GST_AGGREGATOR_GET_CLASS (self);
|
|
|
|
|
|
|
|
if (klass->get_next_time)
|
|
|
|
return klass->get_next_time (self);
|
|
|
|
|
|
|
|
return GST_CLOCK_TIME_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_wait_and_check (GstAggregator * self, gboolean * timeout)
|
2014-12-05 07:19:54 +00:00
|
|
|
{
|
2015-03-07 02:12:52 +00:00
|
|
|
GstClockTime latency;
|
2014-12-05 07:19:54 +00:00
|
|
|
GstClockTime start;
|
2015-03-07 02:12:52 +00:00
|
|
|
gboolean res;
|
2014-12-05 07:19:54 +00:00
|
|
|
|
2014-12-17 16:54:09 +00:00
|
|
|
*timeout = FALSE;
|
|
|
|
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_LOCK (self);
|
2014-12-26 22:49:52 +00:00
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
latency = gst_aggregator_get_latency_unlocked (self);
|
2014-12-05 07:19:54 +00:00
|
|
|
|
2015-01-04 16:57:05 +00:00
|
|
|
if (gst_aggregator_check_pads_ready (self)) {
|
2014-12-05 07:19:54 +00:00
|
|
|
GST_DEBUG_OBJECT (self, "all pads have data");
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_UNLOCK (self);
|
2014-12-26 22:49:52 +00:00
|
|
|
|
2014-12-05 07:19:54 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2014-12-22 14:00:36 +00:00
|
|
|
/* Before waiting, check if we're actually still running */
|
|
|
|
if (!self->priv->running || !self->priv->send_eos) {
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_UNLOCK (self);
|
2014-12-22 14:00:36 +00:00
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2014-12-05 07:19:54 +00:00
|
|
|
start = gst_aggregator_get_next_time (self);
|
|
|
|
|
2015-06-15 16:30:20 +00:00
|
|
|
/* If we're not live, or if we use the running time
|
|
|
|
* of the first buffer as start time, we wait until
|
|
|
|
* all pads have buffers.
|
|
|
|
* Otherwise (i.e. if we are live!), we wait on the clock
|
|
|
|
* and if a pad does not have a buffer in time we ignore
|
|
|
|
* that pad.
|
|
|
|
*/
|
2016-04-15 20:51:17 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
2015-03-07 02:12:52 +00:00
|
|
|
if (!GST_CLOCK_TIME_IS_VALID (latency) ||
|
|
|
|
!GST_IS_CLOCK (GST_ELEMENT_CLOCK (self)) ||
|
2015-06-15 16:30:20 +00:00
|
|
|
!GST_CLOCK_TIME_IS_VALID (start) ||
|
|
|
|
(self->priv->first_buffer
|
|
|
|
&& self->priv->start_time_selection ==
|
|
|
|
GST_AGGREGATOR_START_TIME_SELECTION_FIRST)) {
|
2014-12-22 14:00:36 +00:00
|
|
|
/* We wake up here when something happened, and below
|
|
|
|
* then check if we're ready now. If we return FALSE,
|
|
|
|
* we will be directly called again.
|
|
|
|
*/
|
2016-04-15 20:51:17 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_WAIT (self);
|
2014-12-05 07:19:54 +00:00
|
|
|
} else {
|
2014-12-16 16:33:01 +00:00
|
|
|
GstClockTime base_time, time;
|
|
|
|
GstClock *clock;
|
2014-12-05 07:19:54 +00:00
|
|
|
GstClockReturn status;
|
2015-01-09 15:43:39 +00:00
|
|
|
GstClockTimeDiff jitter;
|
2014-12-05 07:19:54 +00:00
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (self, "got subclass start time: %" GST_TIME_FORMAT,
|
|
|
|
GST_TIME_ARGS (start));
|
|
|
|
|
2014-12-16 16:33:01 +00:00
|
|
|
base_time = GST_ELEMENT_CAST (self)->base_time;
|
2016-04-15 20:51:17 +00:00
|
|
|
clock = gst_object_ref (GST_ELEMENT_CLOCK (self));
|
2015-02-20 02:21:56 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-12-16 16:33:01 +00:00
|
|
|
|
|
|
|
time = base_time + start;
|
2015-03-07 02:12:52 +00:00
|
|
|
time += latency;
|
2014-12-05 07:19:54 +00:00
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (self, "possibly waiting for clock to reach %"
|
|
|
|
GST_TIME_FORMAT " (base %" GST_TIME_FORMAT " start %" GST_TIME_FORMAT
|
2015-03-07 02:12:52 +00:00
|
|
|
" latency %" GST_TIME_FORMAT " current %" GST_TIME_FORMAT ")",
|
|
|
|
GST_TIME_ARGS (time),
|
2016-04-15 20:51:17 +00:00
|
|
|
GST_TIME_ARGS (base_time),
|
2015-03-07 02:12:52 +00:00
|
|
|
GST_TIME_ARGS (start), GST_TIME_ARGS (latency),
|
2014-12-16 16:33:01 +00:00
|
|
|
GST_TIME_ARGS (gst_clock_get_time (clock)));
|
2014-12-05 07:19:54 +00:00
|
|
|
|
2014-12-16 16:33:01 +00:00
|
|
|
self->priv->aggregate_id = gst_clock_new_single_shot_id (clock, time);
|
|
|
|
gst_object_unref (clock);
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_UNLOCK (self);
|
2014-12-05 07:19:54 +00:00
|
|
|
|
2015-01-09 15:43:39 +00:00
|
|
|
jitter = 0;
|
|
|
|
status = gst_clock_id_wait (self->priv->aggregate_id, &jitter);
|
2014-12-05 07:19:54 +00:00
|
|
|
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_LOCK (self);
|
2014-12-05 07:19:54 +00:00
|
|
|
if (self->priv->aggregate_id) {
|
|
|
|
gst_clock_id_unref (self->priv->aggregate_id);
|
|
|
|
self->priv->aggregate_id = NULL;
|
|
|
|
}
|
|
|
|
|
2015-01-14 18:17:19 +00:00
|
|
|
GST_DEBUG_OBJECT (self,
|
2015-11-05 12:36:48 +00:00
|
|
|
"clock returned %d (jitter: %" GST_STIME_FORMAT ")",
|
|
|
|
status, GST_STIME_ARGS (jitter));
|
2014-12-05 07:19:54 +00:00
|
|
|
|
|
|
|
/* we timed out */
|
|
|
|
if (status == GST_CLOCK_OK || status == GST_CLOCK_EARLY) {
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_UNLOCK (self);
|
2014-12-17 16:54:09 +00:00
|
|
|
*timeout = TRUE;
|
2014-12-05 07:19:54 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-04 16:57:05 +00:00
|
|
|
res = gst_aggregator_check_pads_ready (self);
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_UNLOCK (self);
|
2014-12-26 22:49:52 +00:00
|
|
|
|
|
|
|
return res;
|
2014-12-05 07:19:54 +00:00
|
|
|
}
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
static gboolean
|
2017-09-17 19:25:37 +00:00
|
|
|
do_events_and_queries (GstAggregator * self, GstAggregatorPad * pad,
|
|
|
|
gpointer user_data)
|
2015-03-07 00:50:08 +00:00
|
|
|
{
|
|
|
|
GstEvent *event = NULL;
|
2017-05-22 22:53:57 +00:00
|
|
|
GstQuery *query = NULL;
|
2015-03-07 00:50:08 +00:00
|
|
|
GstAggregatorClass *klass = NULL;
|
|
|
|
gboolean *processed_event = user_data;
|
|
|
|
|
|
|
|
do {
|
|
|
|
event = NULL;
|
2017-09-17 19:25:37 +00:00
|
|
|
query = NULL;
|
2015-03-07 00:50:08 +00:00
|
|
|
|
|
|
|
PAD_LOCK (pad);
|
2016-07-07 20:13:57 +00:00
|
|
|
if (pad->priv->num_buffers == 0 && pad->priv->pending_eos) {
|
2015-03-07 00:50:08 +00:00
|
|
|
pad->priv->pending_eos = FALSE;
|
|
|
|
pad->priv->eos = TRUE;
|
|
|
|
}
|
2016-07-06 20:39:17 +00:00
|
|
|
if (pad->priv->clipped_buffer == NULL &&
|
2017-09-17 18:39:12 +00:00
|
|
|
!GST_IS_BUFFER (g_queue_peek_tail (&pad->priv->data))) {
|
|
|
|
if (GST_IS_EVENT (g_queue_peek_tail (&pad->priv->data)))
|
|
|
|
event = gst_event_ref (g_queue_peek_tail (&pad->priv->data));
|
|
|
|
if (GST_IS_QUERY (g_queue_peek_tail (&pad->priv->data)))
|
|
|
|
query = g_queue_peek_tail (&pad->priv->data);
|
2015-03-07 00:50:08 +00:00
|
|
|
}
|
|
|
|
PAD_UNLOCK (pad);
|
2017-05-22 22:53:57 +00:00
|
|
|
if (event || query) {
|
2017-05-21 12:34:13 +00:00
|
|
|
gboolean ret;
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
if (processed_event)
|
|
|
|
*processed_event = TRUE;
|
|
|
|
if (klass == NULL)
|
|
|
|
klass = GST_AGGREGATOR_GET_CLASS (self);
|
|
|
|
|
2017-05-22 22:53:57 +00:00
|
|
|
if (event) {
|
2017-07-13 19:55:55 +00:00
|
|
|
GST_LOG_OBJECT (pad, "Processing %" GST_PTR_FORMAT, event);
|
2017-05-22 22:53:57 +00:00
|
|
|
gst_event_ref (event);
|
|
|
|
ret = klass->sink_event (self, pad, event);
|
|
|
|
|
|
|
|
PAD_LOCK (pad);
|
|
|
|
if (GST_EVENT_TYPE (event) == GST_EVENT_CAPS)
|
|
|
|
pad->priv->negotiated = ret;
|
2017-09-17 18:39:12 +00:00
|
|
|
if (g_queue_peek_tail (&pad->priv->data) == event)
|
|
|
|
gst_event_unref (g_queue_pop_tail (&pad->priv->data));
|
2017-05-22 22:53:57 +00:00
|
|
|
gst_event_unref (event);
|
2017-09-17 19:25:37 +00:00
|
|
|
} else if (query) {
|
2017-07-13 19:55:55 +00:00
|
|
|
GST_LOG_OBJECT (pad, "Processing %" GST_PTR_FORMAT, query);
|
2017-05-22 22:53:57 +00:00
|
|
|
ret = klass->sink_query (self, pad, query);
|
|
|
|
|
|
|
|
PAD_LOCK (pad);
|
2017-09-17 18:39:12 +00:00
|
|
|
if (g_queue_peek_tail (&pad->priv->data) == query) {
|
2017-05-22 22:53:57 +00:00
|
|
|
GstStructure *s;
|
|
|
|
|
|
|
|
s = gst_query_writable_structure (query);
|
|
|
|
gst_structure_set (s, "gst-aggregator-retval", G_TYPE_BOOLEAN, ret,
|
|
|
|
NULL);
|
2017-09-17 18:39:12 +00:00
|
|
|
g_queue_pop_tail (&pad->priv->data);
|
2017-05-22 22:53:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-21 12:34:13 +00:00
|
|
|
PAD_BROADCAST_EVENT (pad);
|
|
|
|
PAD_UNLOCK (pad);
|
2015-03-07 00:50:08 +00:00
|
|
|
}
|
2017-09-17 19:25:37 +00:00
|
|
|
} while (event || query);
|
2015-03-07 00:50:08 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2015-04-02 02:10:11 +00:00
|
|
|
static void
|
|
|
|
gst_aggregator_pad_set_flushing (GstAggregatorPad * aggpad,
|
2015-03-07 00:50:08 +00:00
|
|
|
GstFlowReturn flow_return, gboolean full)
|
2015-04-02 02:10:11 +00:00
|
|
|
{
|
2015-03-07 00:50:08 +00:00
|
|
|
GList *item;
|
|
|
|
|
2015-04-02 02:10:11 +00:00
|
|
|
PAD_LOCK (aggpad);
|
|
|
|
if (flow_return == GST_FLOW_NOT_LINKED)
|
|
|
|
aggpad->priv->flow_return = MIN (flow_return, aggpad->priv->flow_return);
|
|
|
|
else
|
|
|
|
aggpad->priv->flow_return = flow_return;
|
2015-03-07 00:50:08 +00:00
|
|
|
|
2017-09-17 18:39:12 +00:00
|
|
|
item = g_queue_peek_head_link (&aggpad->priv->data);
|
2015-03-07 00:50:08 +00:00
|
|
|
while (item) {
|
|
|
|
GList *next = item->next;
|
|
|
|
|
|
|
|
/* In partial flush, we do like the pad, we get rid of non-sticky events
|
|
|
|
* and EOS/SEGMENT.
|
|
|
|
*/
|
|
|
|
if (full || GST_IS_BUFFER (item->data) ||
|
|
|
|
GST_EVENT_TYPE (item->data) == GST_EVENT_EOS ||
|
|
|
|
GST_EVENT_TYPE (item->data) == GST_EVENT_SEGMENT ||
|
|
|
|
!GST_EVENT_IS_STICKY (item->data)) {
|
2017-05-22 22:53:57 +00:00
|
|
|
if (!GST_IS_QUERY (item->data))
|
|
|
|
gst_mini_object_unref (item->data);
|
2017-09-17 18:39:12 +00:00
|
|
|
g_queue_delete_link (&aggpad->priv->data, item);
|
2015-03-07 00:50:08 +00:00
|
|
|
}
|
|
|
|
item = next;
|
|
|
|
}
|
2015-09-17 23:42:34 +00:00
|
|
|
aggpad->priv->num_buffers = 0;
|
2016-07-06 20:39:17 +00:00
|
|
|
gst_buffer_replace (&aggpad->priv->clipped_buffer, NULL);
|
2015-03-07 00:50:08 +00:00
|
|
|
|
2015-04-02 02:10:11 +00:00
|
|
|
PAD_BROADCAST_EVENT (aggpad);
|
|
|
|
PAD_UNLOCK (aggpad);
|
|
|
|
}
|
|
|
|
|
2017-05-20 12:24:57 +00:00
|
|
|
static GstFlowReturn
|
|
|
|
gst_aggregator_default_update_src_caps (GstAggregator * agg, GstCaps * caps,
|
|
|
|
GstCaps ** ret)
|
|
|
|
{
|
|
|
|
*ret = gst_caps_ref (caps);
|
|
|
|
|
|
|
|
return GST_FLOW_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GstCaps *
|
|
|
|
gst_aggregator_default_fixate_src_caps (GstAggregator * agg, GstCaps * caps)
|
|
|
|
{
|
|
|
|
caps = gst_caps_fixate (caps);
|
|
|
|
|
|
|
|
return caps;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gst_aggregator_default_negotiated_src_caps (GstAggregator * agg, GstCaps * caps)
|
|
|
|
{
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2017-05-20 14:58:54 +00:00
|
|
|
|
|
|
|
/* takes ownership of the pool, allocator and query */
|
|
|
|
static gboolean
|
|
|
|
gst_aggregator_set_allocation (GstAggregator * self,
|
|
|
|
GstBufferPool * pool, GstAllocator * allocator,
|
|
|
|
GstAllocationParams * params, GstQuery * query)
|
|
|
|
{
|
|
|
|
GstAllocator *oldalloc;
|
|
|
|
GstBufferPool *oldpool;
|
|
|
|
GstQuery *oldquery;
|
|
|
|
|
|
|
|
GST_DEBUG ("storing allocation query");
|
|
|
|
|
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
oldpool = self->priv->pool;
|
|
|
|
self->priv->pool = pool;
|
|
|
|
|
|
|
|
oldalloc = self->priv->allocator;
|
|
|
|
self->priv->allocator = allocator;
|
|
|
|
|
|
|
|
oldquery = self->priv->allocation_query;
|
|
|
|
self->priv->allocation_query = query;
|
|
|
|
|
|
|
|
if (params)
|
|
|
|
self->priv->allocation_params = *params;
|
|
|
|
else
|
|
|
|
gst_allocation_params_init (&self->priv->allocation_params);
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
|
|
|
|
if (oldpool) {
|
|
|
|
GST_DEBUG_OBJECT (self, "deactivating old pool %p", oldpool);
|
|
|
|
gst_buffer_pool_set_active (oldpool, FALSE);
|
|
|
|
gst_object_unref (oldpool);
|
|
|
|
}
|
|
|
|
if (oldalloc) {
|
|
|
|
gst_object_unref (oldalloc);
|
|
|
|
}
|
|
|
|
if (oldquery) {
|
|
|
|
gst_query_unref (oldquery);
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gst_aggregator_decide_allocation (GstAggregator * self, GstQuery * query)
|
|
|
|
{
|
|
|
|
GstAggregatorClass *aggclass = GST_AGGREGATOR_GET_CLASS (self);
|
|
|
|
|
|
|
|
if (aggclass->decide_allocation)
|
|
|
|
if (!aggclass->decide_allocation (self, query))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gst_aggregator_do_allocation (GstAggregator * self, GstCaps * caps)
|
|
|
|
{
|
|
|
|
GstQuery *query;
|
|
|
|
gboolean result = TRUE;
|
|
|
|
GstBufferPool *pool = NULL;
|
|
|
|
GstAllocator *allocator;
|
|
|
|
GstAllocationParams params;
|
|
|
|
|
|
|
|
/* find a pool for the negotiated caps now */
|
|
|
|
GST_DEBUG_OBJECT (self, "doing allocation query");
|
|
|
|
query = gst_query_new_allocation (caps, TRUE);
|
|
|
|
if (!gst_pad_peer_query (self->srcpad, query)) {
|
|
|
|
/* not a problem, just debug a little */
|
|
|
|
GST_DEBUG_OBJECT (self, "peer ALLOCATION query failed");
|
|
|
|
}
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (self, "calling decide_allocation");
|
|
|
|
result = gst_aggregator_decide_allocation (self, query);
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (self, "ALLOCATION (%d) params: %" GST_PTR_FORMAT, result,
|
|
|
|
query);
|
|
|
|
|
|
|
|
if (!result)
|
|
|
|
goto no_decide_allocation;
|
|
|
|
|
|
|
|
/* we got configuration from our peer or the decide_allocation method,
|
|
|
|
* parse them */
|
|
|
|
if (gst_query_get_n_allocation_params (query) > 0) {
|
|
|
|
gst_query_parse_nth_allocation_param (query, 0, &allocator, ¶ms);
|
|
|
|
} else {
|
|
|
|
allocator = NULL;
|
|
|
|
gst_allocation_params_init (¶ms);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gst_query_get_n_allocation_pools (query) > 0)
|
|
|
|
gst_query_parse_nth_allocation_pool (query, 0, &pool, NULL, NULL, NULL);
|
|
|
|
|
|
|
|
/* now store */
|
|
|
|
result =
|
|
|
|
gst_aggregator_set_allocation (self, pool, allocator, ¶ms, query);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
|
|
|
/* Errors */
|
|
|
|
no_decide_allocation:
|
|
|
|
{
|
|
|
|
GST_WARNING_OBJECT (self, "Failed to decide allocation");
|
|
|
|
gst_query_unref (query);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-05-20 12:24:57 +00:00
|
|
|
/* WITH SRC_LOCK held */
|
|
|
|
static GstFlowReturn
|
|
|
|
gst_aggregator_update_src_caps (GstAggregator * self)
|
|
|
|
{
|
|
|
|
GstAggregatorClass *agg_klass = GST_AGGREGATOR_GET_CLASS (self);
|
|
|
|
GstCaps *downstream_caps, *template_caps, *caps = NULL;
|
|
|
|
GstFlowReturn ret = GST_FLOW_OK;
|
|
|
|
|
|
|
|
template_caps = gst_pad_get_pad_template_caps (self->srcpad);
|
|
|
|
downstream_caps = gst_pad_peer_query_caps (self->srcpad, template_caps);
|
|
|
|
|
|
|
|
if (gst_caps_is_empty (downstream_caps)) {
|
|
|
|
GST_INFO_OBJECT (self, "Downstream caps (%"
|
|
|
|
GST_PTR_FORMAT ") not compatible with pad template caps (%"
|
|
|
|
GST_PTR_FORMAT ")", downstream_caps, template_caps);
|
|
|
|
ret = GST_FLOW_NOT_NEGOTIATED;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_assert (agg_klass->update_src_caps);
|
|
|
|
GST_DEBUG_OBJECT (self, "updating caps from %" GST_PTR_FORMAT,
|
|
|
|
downstream_caps);
|
|
|
|
ret = agg_klass->update_src_caps (self, downstream_caps, &caps);
|
|
|
|
if (ret < GST_FLOW_OK) {
|
|
|
|
GST_WARNING_OBJECT (self, "Subclass failed to update provided caps");
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if ((caps == NULL || gst_caps_is_empty (caps)) && ret >= GST_FLOW_OK) {
|
|
|
|
ret = GST_FLOW_NOT_NEGOTIATED;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
GST_DEBUG_OBJECT (self, " to %" GST_PTR_FORMAT, caps);
|
|
|
|
|
|
|
|
#ifdef GST_ENABLE_EXTRA_CHECKS
|
|
|
|
if (!gst_caps_is_subset (caps, template_caps)) {
|
|
|
|
GstCaps *intersection;
|
|
|
|
|
|
|
|
GST_ERROR_OBJECT (self,
|
|
|
|
"update_src_caps returned caps %" GST_PTR_FORMAT
|
|
|
|
" which are not a real subset of the template caps %"
|
|
|
|
GST_PTR_FORMAT, caps, template_caps);
|
|
|
|
g_warning ("%s: update_src_caps returned caps which are not a real "
|
|
|
|
"subset of the filter caps", GST_ELEMENT_NAME (self));
|
|
|
|
|
|
|
|
intersection =
|
|
|
|
gst_caps_intersect_full (template_caps, caps, GST_CAPS_INTERSECT_FIRST);
|
|
|
|
gst_caps_unref (caps);
|
|
|
|
caps = intersection;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (gst_caps_is_any (caps)) {
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!gst_caps_is_fixed (caps)) {
|
|
|
|
g_assert (agg_klass->fixate_src_caps);
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (self, "fixate caps from %" GST_PTR_FORMAT, caps);
|
|
|
|
if (!(caps = agg_klass->fixate_src_caps (self, caps))) {
|
|
|
|
GST_WARNING_OBJECT (self, "Subclass failed to fixate provided caps");
|
|
|
|
ret = GST_FLOW_NOT_NEGOTIATED;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
GST_DEBUG_OBJECT (self, " to %" GST_PTR_FORMAT, caps);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (agg_klass->negotiated_src_caps) {
|
|
|
|
if (!agg_klass->negotiated_src_caps (self, caps)) {
|
|
|
|
GST_WARNING_OBJECT (self, "Subclass failed to accept negotiated caps");
|
|
|
|
ret = GST_FLOW_NOT_NEGOTIATED;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gst_aggregator_set_src_caps (self, caps);
|
|
|
|
|
2017-05-20 14:58:54 +00:00
|
|
|
if (!gst_aggregator_do_allocation (self, caps)) {
|
|
|
|
GST_WARNING_OBJECT (self, "Allocation negotiation failed");
|
|
|
|
ret = GST_FLOW_NOT_NEGOTIATED;
|
|
|
|
}
|
|
|
|
|
2017-05-20 12:24:57 +00:00
|
|
|
done:
|
|
|
|
gst_caps_unref (downstream_caps);
|
|
|
|
gst_caps_unref (template_caps);
|
|
|
|
|
|
|
|
if (caps)
|
|
|
|
gst_caps_unref (caps);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
static void
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_aggregate_func (GstAggregator * self)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
GstAggregatorPrivate *priv = self->priv;
|
|
|
|
GstAggregatorClass *klass = GST_AGGREGATOR_GET_CLASS (self);
|
2014-12-17 16:54:09 +00:00
|
|
|
gboolean timeout = FALSE;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2014-09-17 14:48:02 +00:00
|
|
|
if (self->priv->running == FALSE) {
|
|
|
|
GST_DEBUG_OBJECT (self, "Not running anymore");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
GST_LOG_OBJECT (self, "Checking aggregate");
|
2014-12-05 07:19:54 +00:00
|
|
|
while (priv->send_eos && priv->running) {
|
2017-05-20 12:24:57 +00:00
|
|
|
GstFlowReturn flow_return = GST_FLOW_OK;
|
2015-03-07 00:50:08 +00:00
|
|
|
gboolean processed_event = FALSE;
|
|
|
|
|
2017-09-17 19:25:37 +00:00
|
|
|
gst_aggregator_iterate_sinkpads (self, do_events_and_queries, NULL);
|
2015-01-21 23:53:20 +00:00
|
|
|
|
2014-12-30 23:58:34 +00:00
|
|
|
if (!gst_aggregator_wait_and_check (self, &timeout))
|
2014-12-05 07:19:54 +00:00
|
|
|
continue;
|
|
|
|
|
2017-09-17 19:25:37 +00:00
|
|
|
gst_aggregator_iterate_sinkpads (self, do_events_and_queries,
|
|
|
|
&processed_event);
|
2015-03-07 00:50:08 +00:00
|
|
|
if (processed_event)
|
|
|
|
continue;
|
|
|
|
|
2017-05-20 12:24:57 +00:00
|
|
|
if (gst_pad_check_reconfigure (GST_AGGREGATOR_SRC_PAD (self))) {
|
|
|
|
flow_return = gst_aggregator_update_src_caps (self);
|
|
|
|
if (flow_return != GST_FLOW_OK)
|
|
|
|
gst_pad_mark_reconfigure (GST_AGGREGATOR_SRC_PAD (self));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (timeout || flow_return >= GST_FLOW_OK) {
|
|
|
|
GST_TRACE_OBJECT (self, "Actually aggregating!");
|
|
|
|
flow_return = klass->aggregate (self, timeout);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flow_return == GST_AGGREGATOR_FLOW_NEED_DATA)
|
|
|
|
continue;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
2015-04-02 02:10:11 +00:00
|
|
|
if (flow_return == GST_FLOW_FLUSHING && priv->flush_seeking) {
|
|
|
|
/* We don't want to set the pads to flushing, but we want to
|
|
|
|
* stop the thread, so just break here */
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
break;
|
|
|
|
}
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-05-27 14:59:39 +00:00
|
|
|
if (flow_return == GST_FLOW_EOS || flow_return == GST_FLOW_ERROR) {
|
2015-01-21 23:53:20 +00:00
|
|
|
gst_aggregator_push_eos (self);
|
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-01-21 23:53:20 +00:00
|
|
|
GST_LOG_OBJECT (self, "flow return is %s", gst_flow_get_name (flow_return));
|
|
|
|
|
2015-04-02 02:10:11 +00:00
|
|
|
if (flow_return != GST_FLOW_OK) {
|
|
|
|
GList *item;
|
|
|
|
|
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
for (item = GST_ELEMENT (self)->sinkpads; item; item = item->next) {
|
|
|
|
GstAggregatorPad *aggpad = GST_AGGREGATOR_PAD (item->data);
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
gst_aggregator_pad_set_flushing (aggpad, flow_return, TRUE);
|
2015-04-02 02:10:11 +00:00
|
|
|
}
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
break;
|
2015-04-02 02:10:11 +00:00
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
2015-02-10 09:49:16 +00:00
|
|
|
|
|
|
|
/* Pause the task here, the only ways to get here are:
|
|
|
|
* 1) We're stopping, in which case the task is stopped anyway
|
|
|
|
* 2) We got a flow error above, in which case it might take
|
|
|
|
* some time to forward the flow return upstream and we
|
|
|
|
* would otherwise call the task function over and over
|
|
|
|
* again without doing anything
|
|
|
|
*/
|
|
|
|
gst_pad_pause_task (self->srcpad);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 17:50:17 +00:00
|
|
|
gst_aggregator_start (GstAggregator * self)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2014-12-30 17:50:17 +00:00
|
|
|
GstAggregatorClass *klass;
|
|
|
|
gboolean result;
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
self->priv->send_stream_start = TRUE;
|
|
|
|
self->priv->send_segment = TRUE;
|
|
|
|
self->priv->send_eos = TRUE;
|
|
|
|
self->priv->srccaps = NULL;
|
|
|
|
|
2017-05-20 14:58:54 +00:00
|
|
|
gst_aggregator_set_allocation (self, NULL, NULL, NULL, NULL);
|
|
|
|
|
2014-12-30 17:50:17 +00:00
|
|
|
klass = GST_AGGREGATOR_GET_CLASS (self);
|
|
|
|
|
|
|
|
if (klass->start)
|
|
|
|
result = klass->start (self);
|
|
|
|
else
|
|
|
|
result = TRUE;
|
|
|
|
|
|
|
|
return result;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
_check_pending_flush_stop (GstAggregatorPad * pad)
|
|
|
|
{
|
2015-01-26 16:06:29 +00:00
|
|
|
gboolean res;
|
|
|
|
|
|
|
|
PAD_LOCK (pad);
|
|
|
|
res = (!pad->priv->pending_flush_stop && !pad->priv->pending_flush_start);
|
|
|
|
PAD_UNLOCK (pad);
|
|
|
|
|
|
|
|
return res;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_stop_srcpad_task (GstAggregator * self, GstEvent * flush_start)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
gboolean res = TRUE;
|
|
|
|
|
|
|
|
GST_INFO_OBJECT (self, "%s srcpad task",
|
|
|
|
flush_start ? "Pausing" : "Stopping");
|
|
|
|
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_LOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
self->priv->running = FALSE;
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_BROADCAST (self);
|
|
|
|
SRC_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
if (flush_start) {
|
|
|
|
res = gst_pad_push_event (self->srcpad, flush_start);
|
|
|
|
}
|
|
|
|
|
|
|
|
gst_pad_stop_task (self->srcpad);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_start_srcpad_task (GstAggregator * self)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
GST_INFO_OBJECT (self, "Starting srcpad task");
|
|
|
|
|
2016-05-25 17:38:47 +00:00
|
|
|
self->priv->running = TRUE;
|
|
|
|
gst_pad_start_task (GST_PAD (self->srcpad),
|
|
|
|
(GstTaskFunction) gst_aggregator_aggregate_func, self, NULL);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static GstFlowReturn
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_flush (GstAggregator * self)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
GstFlowReturn ret = GST_FLOW_OK;
|
|
|
|
GstAggregatorPrivate *priv = self->priv;
|
|
|
|
GstAggregatorClass *klass = GST_AGGREGATOR_GET_CLASS (self);
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (self, "Flushing everything");
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
priv->send_segment = TRUE;
|
|
|
|
priv->flush_seeking = FALSE;
|
2015-01-22 00:33:18 +00:00
|
|
|
priv->tags_changed = FALSE;
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
if (klass->flush)
|
|
|
|
ret = klass->flush (self);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-01-21 23:45:36 +00:00
|
|
|
|
|
|
|
/* Called with GstAggregator's object lock held */
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
static gboolean
|
2015-01-21 23:45:36 +00:00
|
|
|
gst_aggregator_all_flush_stop_received_locked (GstAggregator * self)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
GList *tmp;
|
|
|
|
GstAggregatorPad *tmppad;
|
|
|
|
|
|
|
|
for (tmp = GST_ELEMENT (self)->sinkpads; tmp; tmp = tmp->next) {
|
|
|
|
tmppad = (GstAggregatorPad *) tmp->data;
|
|
|
|
|
|
|
|
if (_check_pending_flush_stop (tmppad) == FALSE) {
|
|
|
|
GST_DEBUG_OBJECT (tmppad, "Is not last %i -- %i",
|
|
|
|
tmppad->priv->pending_flush_start, tmppad->priv->pending_flush_stop);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2014-08-02 16:25:01 +00:00
|
|
|
static void
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_flush_start (GstAggregator * self, GstAggregatorPad * aggpad,
|
|
|
|
GstEvent * event)
|
2014-08-02 16:25:01 +00:00
|
|
|
{
|
|
|
|
GstAggregatorPrivate *priv = self->priv;
|
|
|
|
GstAggregatorPadPrivate *padpriv = aggpad->priv;
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
gst_aggregator_pad_set_flushing (aggpad, GST_FLOW_FLUSHING, FALSE);
|
2015-01-26 16:06:29 +00:00
|
|
|
|
2015-02-18 20:06:01 +00:00
|
|
|
PAD_FLUSH_LOCK (aggpad);
|
2015-01-26 16:06:29 +00:00
|
|
|
PAD_LOCK (aggpad);
|
|
|
|
if (padpriv->pending_flush_start) {
|
2014-08-02 16:25:01 +00:00
|
|
|
GST_DEBUG_OBJECT (aggpad, "Expecting FLUSH_STOP now");
|
2015-01-26 16:06:29 +00:00
|
|
|
|
|
|
|
padpriv->pending_flush_start = FALSE;
|
|
|
|
padpriv->pending_flush_stop = TRUE;
|
2014-08-02 16:25:01 +00:00
|
|
|
}
|
2015-01-26 16:06:29 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2014-08-02 16:25:01 +00:00
|
|
|
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
if (priv->flush_seeking) {
|
2014-08-02 16:25:01 +00:00
|
|
|
/* If flush_seeking we forward the first FLUSH_START */
|
2015-01-21 23:45:36 +00:00
|
|
|
if (priv->pending_flush_start) {
|
|
|
|
priv->pending_flush_start = FALSE;
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-08-02 16:25:01 +00:00
|
|
|
|
|
|
|
GST_INFO_OBJECT (self, "Flushing, pausing srcpad task");
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_stop_srcpad_task (self, event);
|
2014-08-02 16:25:01 +00:00
|
|
|
|
|
|
|
GST_INFO_OBJECT (self, "Getting STREAM_LOCK while seeking");
|
|
|
|
GST_PAD_STREAM_LOCK (self->srcpad);
|
|
|
|
GST_LOG_OBJECT (self, "GOT STREAM_LOCK");
|
|
|
|
event = NULL;
|
2014-12-23 10:45:05 +00:00
|
|
|
} else {
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-12-23 10:45:05 +00:00
|
|
|
gst_event_unref (event);
|
2014-08-02 16:25:01 +00:00
|
|
|
}
|
|
|
|
} else {
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-08-02 16:25:01 +00:00
|
|
|
gst_event_unref (event);
|
|
|
|
}
|
2015-02-18 20:06:01 +00:00
|
|
|
PAD_FLUSH_UNLOCK (aggpad);
|
2015-03-07 00:50:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Must be called with the the PAD_LOCK held */
|
|
|
|
static void
|
|
|
|
update_time_level (GstAggregatorPad * aggpad, gboolean head)
|
|
|
|
{
|
|
|
|
if (head) {
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (aggpad->priv->head_position) &&
|
2016-07-06 20:39:17 +00:00
|
|
|
aggpad->priv->head_segment.format == GST_FORMAT_TIME)
|
2015-03-07 00:50:08 +00:00
|
|
|
aggpad->priv->head_time =
|
2016-07-06 20:39:17 +00:00
|
|
|
gst_segment_to_running_time (&aggpad->priv->head_segment,
|
2015-03-07 00:50:08 +00:00
|
|
|
GST_FORMAT_TIME, aggpad->priv->head_position);
|
|
|
|
else
|
|
|
|
aggpad->priv->head_time = GST_CLOCK_TIME_NONE;
|
2017-05-10 00:20:07 +00:00
|
|
|
|
|
|
|
if (!GST_CLOCK_TIME_IS_VALID (aggpad->priv->tail_time))
|
|
|
|
aggpad->priv->tail_time = aggpad->priv->head_time;
|
2015-03-07 00:50:08 +00:00
|
|
|
} else {
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (aggpad->priv->tail_position) &&
|
|
|
|
aggpad->segment.format == GST_FORMAT_TIME)
|
|
|
|
aggpad->priv->tail_time =
|
|
|
|
gst_segment_to_running_time (&aggpad->segment,
|
|
|
|
GST_FORMAT_TIME, aggpad->priv->tail_position);
|
|
|
|
else
|
|
|
|
aggpad->priv->tail_time = aggpad->priv->head_time;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aggpad->priv->head_time == GST_CLOCK_TIME_NONE ||
|
|
|
|
aggpad->priv->tail_time == GST_CLOCK_TIME_NONE) {
|
|
|
|
aggpad->priv->time_level = 0;
|
|
|
|
return;
|
|
|
|
}
|
2014-08-02 16:25:01 +00:00
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
if (aggpad->priv->tail_time > aggpad->priv->head_time)
|
|
|
|
aggpad->priv->time_level = 0;
|
|
|
|
else
|
|
|
|
aggpad->priv->time_level = aggpad->priv->head_time -
|
|
|
|
aggpad->priv->tail_time;
|
2014-08-02 16:25:01 +00:00
|
|
|
}
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
/* GstAggregator vmethods default implementations */
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_default_sink_event (GstAggregator * self,
|
|
|
|
GstAggregatorPad * aggpad, GstEvent * event)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
gboolean res = TRUE;
|
|
|
|
GstPad *pad = GST_PAD (aggpad);
|
|
|
|
GstAggregatorPrivate *priv = self->priv;
|
|
|
|
|
2017-07-30 10:17:57 +00:00
|
|
|
GST_DEBUG_OBJECT (aggpad, "Got event: %" GST_PTR_FORMAT, event);
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
switch (GST_EVENT_TYPE (event)) {
|
|
|
|
case GST_EVENT_FLUSH_START:
|
|
|
|
{
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_flush_start (self, aggpad, event);
|
2014-05-22 17:44:37 +00:00
|
|
|
/* We forward only in one case: right after flush_seeking */
|
2014-08-02 16:25:01 +00:00
|
|
|
event = NULL;
|
2014-05-22 17:44:37 +00:00
|
|
|
goto eat;
|
|
|
|
}
|
|
|
|
case GST_EVENT_FLUSH_STOP:
|
|
|
|
{
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_pad_flush (aggpad, self);
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
if (priv->flush_seeking) {
|
2014-05-22 17:44:37 +00:00
|
|
|
g_atomic_int_set (&aggpad->priv->pending_flush_stop, FALSE);
|
2015-01-21 23:45:36 +00:00
|
|
|
if (gst_aggregator_all_flush_stop_received_locked (self)) {
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
/* That means we received FLUSH_STOP/FLUSH_STOP on
|
|
|
|
* all sinkpads -- Seeking is Done... sending FLUSH_STOP */
|
|
|
|
gst_aggregator_flush (self);
|
|
|
|
gst_pad_push_event (self->srcpad, event);
|
|
|
|
event = NULL;
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_LOCK (self);
|
2015-01-21 23:45:36 +00:00
|
|
|
priv->send_eos = TRUE;
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_BROADCAST (self);
|
|
|
|
SRC_UNLOCK (self);
|
2015-01-21 23:45:36 +00:00
|
|
|
|
|
|
|
GST_INFO_OBJECT (self, "Releasing source pad STREAM_LOCK");
|
|
|
|
GST_PAD_STREAM_UNLOCK (self->srcpad);
|
|
|
|
gst_aggregator_start_srcpad_task (self);
|
|
|
|
} else {
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
2015-01-21 23:45:36 +00:00
|
|
|
} else {
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* We never forward the event */
|
|
|
|
goto eat;
|
|
|
|
}
|
|
|
|
case GST_EVENT_EOS:
|
|
|
|
{
|
|
|
|
/* We still have a buffer, and we don't want the subclass to have to
|
|
|
|
* check for it. Mark pending_eos, eos will be set when steal_buffer is
|
|
|
|
* called
|
|
|
|
*/
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_LOCK (self);
|
2015-01-14 19:35:15 +00:00
|
|
|
PAD_LOCK (aggpad);
|
2016-07-07 20:13:57 +00:00
|
|
|
if (aggpad->priv->num_buffers == 0) {
|
2015-01-26 10:25:54 +00:00
|
|
|
aggpad->priv->eos = TRUE;
|
2014-05-22 17:44:37 +00:00
|
|
|
} else {
|
|
|
|
aggpad->priv->pending_eos = TRUE;
|
|
|
|
}
|
2015-01-14 19:35:15 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_BROADCAST (self);
|
|
|
|
SRC_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
goto eat;
|
|
|
|
}
|
|
|
|
case GST_EVENT_SEGMENT:
|
|
|
|
{
|
2015-03-07 00:50:08 +00:00
|
|
|
PAD_LOCK (aggpad);
|
2015-01-26 10:29:08 +00:00
|
|
|
GST_OBJECT_LOCK (aggpad);
|
2014-05-22 17:44:37 +00:00
|
|
|
gst_event_copy_segment (event, &aggpad->segment);
|
2017-04-13 20:11:55 +00:00
|
|
|
/* We've got a new segment, tail_position is now meaningless
|
|
|
|
* and may interfere with the time_level calculation
|
|
|
|
*/
|
|
|
|
aggpad->priv->tail_position = GST_CLOCK_TIME_NONE;
|
2015-03-07 00:50:08 +00:00
|
|
|
update_time_level (aggpad, FALSE);
|
2015-01-26 10:29:08 +00:00
|
|
|
GST_OBJECT_UNLOCK (aggpad);
|
2015-03-07 00:50:08 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2015-01-26 10:32:47 +00:00
|
|
|
|
|
|
|
GST_OBJECT_LOCK (self);
|
2014-08-05 13:36:30 +00:00
|
|
|
self->priv->seqnum = gst_event_get_seqnum (event);
|
2015-01-26 10:32:47 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
goto eat;
|
|
|
|
}
|
|
|
|
case GST_EVENT_STREAM_START:
|
|
|
|
{
|
|
|
|
goto eat;
|
|
|
|
}
|
2015-02-13 23:45:20 +00:00
|
|
|
case GST_EVENT_GAP:
|
|
|
|
{
|
2015-03-07 00:50:08 +00:00
|
|
|
GstClockTime pts, endpts;
|
2015-03-18 01:13:06 +00:00
|
|
|
GstClockTime duration;
|
|
|
|
GstBuffer *gapbuf;
|
|
|
|
|
|
|
|
gst_event_parse_gap (event, &pts, &duration);
|
|
|
|
gapbuf = gst_buffer_new ();
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
if (GST_CLOCK_TIME_IS_VALID (duration))
|
|
|
|
endpts = pts + duration;
|
|
|
|
else
|
|
|
|
endpts = GST_CLOCK_TIME_NONE;
|
|
|
|
|
|
|
|
GST_OBJECT_LOCK (aggpad);
|
|
|
|
res = gst_segment_clip (&aggpad->segment, GST_FORMAT_TIME, pts, endpts,
|
|
|
|
&pts, &endpts);
|
|
|
|
GST_OBJECT_UNLOCK (aggpad);
|
|
|
|
|
|
|
|
if (!res) {
|
|
|
|
GST_WARNING_OBJECT (self, "GAP event outside segment, dropping");
|
|
|
|
goto eat;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (endpts) && GST_CLOCK_TIME_IS_VALID (pts))
|
|
|
|
duration = endpts - pts;
|
|
|
|
else
|
|
|
|
duration = GST_CLOCK_TIME_NONE;
|
|
|
|
|
2015-03-18 01:13:06 +00:00
|
|
|
GST_BUFFER_PTS (gapbuf) = pts;
|
|
|
|
GST_BUFFER_DURATION (gapbuf) = duration;
|
|
|
|
GST_BUFFER_FLAG_SET (gapbuf, GST_BUFFER_FLAG_GAP);
|
|
|
|
GST_BUFFER_FLAG_SET (gapbuf, GST_BUFFER_FLAG_DROPPABLE);
|
|
|
|
|
2017-07-24 15:38:57 +00:00
|
|
|
/* Remove GAP event so we can replace it with the buffer */
|
2017-09-17 18:39:12 +00:00
|
|
|
if (g_queue_peek_tail (&aggpad->priv->data) == event)
|
|
|
|
gst_event_unref (g_queue_pop_tail (&aggpad->priv->data));
|
2017-07-24 15:38:57 +00:00
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
if (gst_aggregator_pad_chain_internal (self, aggpad, gapbuf, FALSE) !=
|
|
|
|
GST_FLOW_OK) {
|
2015-03-18 01:13:06 +00:00
|
|
|
GST_WARNING_OBJECT (self, "Failed to chain gap buffer");
|
|
|
|
res = FALSE;
|
|
|
|
}
|
|
|
|
|
2015-02-13 23:45:20 +00:00
|
|
|
goto eat;
|
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
case GST_EVENT_TAG:
|
|
|
|
{
|
|
|
|
GstTagList *tags;
|
|
|
|
|
|
|
|
gst_event_parse_tag (event, &tags);
|
|
|
|
|
|
|
|
if (gst_tag_list_get_scope (tags) == GST_TAG_SCOPE_STREAM) {
|
|
|
|
gst_aggregator_merge_tags (self, tags, GST_TAG_MERGE_REPLACE);
|
|
|
|
gst_event_unref (event);
|
|
|
|
event = NULL;
|
|
|
|
goto eat;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (pad, "Forwarding event: %" GST_PTR_FORMAT, event);
|
|
|
|
return gst_pad_event_default (pad, GST_OBJECT (self), event);
|
|
|
|
|
|
|
|
eat:
|
|
|
|
GST_DEBUG_OBJECT (pad, "Eating event: %" GST_PTR_FORMAT, event);
|
|
|
|
if (event)
|
|
|
|
gst_event_unref (event);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2014-12-30 19:22:01 +00:00
|
|
|
static inline gboolean
|
|
|
|
gst_aggregator_stop_pad (GstAggregator * self, GstAggregatorPad * pad,
|
|
|
|
gpointer unused_udata)
|
2014-07-10 11:18:21 +00:00
|
|
|
{
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_pad_flush (pad, self);
|
2014-07-10 11:18:21 +00:00
|
|
|
|
2017-05-21 12:34:13 +00:00
|
|
|
PAD_LOCK (pad);
|
2017-05-22 22:53:23 +00:00
|
|
|
pad->priv->flow_return = GST_FLOW_FLUSHING;
|
2017-05-21 12:34:13 +00:00
|
|
|
pad->priv->negotiated = FALSE;
|
2017-05-22 22:53:23 +00:00
|
|
|
PAD_BROADCAST_EVENT (pad);
|
2017-05-21 12:34:13 +00:00
|
|
|
PAD_UNLOCK (pad);
|
|
|
|
|
2014-07-10 11:18:21 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
static gboolean
|
2014-12-30 17:50:17 +00:00
|
|
|
gst_aggregator_stop (GstAggregator * agg)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2014-12-30 17:50:17 +00:00
|
|
|
GstAggregatorClass *klass;
|
|
|
|
gboolean result;
|
|
|
|
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_reset_flow_values (agg);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2014-12-30 19:22:01 +00:00
|
|
|
gst_aggregator_iterate_sinkpads (agg, gst_aggregator_stop_pad, NULL);
|
2014-07-10 11:18:21 +00:00
|
|
|
|
2014-12-30 17:50:17 +00:00
|
|
|
klass = GST_AGGREGATOR_GET_CLASS (agg);
|
|
|
|
|
|
|
|
if (klass->stop)
|
|
|
|
result = klass->stop (agg);
|
|
|
|
else
|
|
|
|
result = TRUE;
|
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
agg->priv->has_peer_latency = FALSE;
|
|
|
|
agg->priv->peer_latency_live = FALSE;
|
|
|
|
agg->priv->peer_latency_min = agg->priv->peer_latency_max = FALSE;
|
|
|
|
|
2015-01-22 00:33:18 +00:00
|
|
|
if (agg->priv->tags)
|
|
|
|
gst_tag_list_unref (agg->priv->tags);
|
|
|
|
agg->priv->tags = NULL;
|
|
|
|
|
2017-05-20 14:58:54 +00:00
|
|
|
gst_aggregator_set_allocation (agg, NULL, NULL, NULL, NULL);
|
|
|
|
|
2014-12-30 17:50:17 +00:00
|
|
|
return result;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* GstElement vmethods implementations */
|
|
|
|
static GstStateChangeReturn
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_change_state (GstElement * element, GstStateChange transition)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
GstStateChangeReturn ret;
|
|
|
|
GstAggregator *self = GST_AGGREGATOR (element);
|
|
|
|
|
|
|
|
switch (transition) {
|
|
|
|
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
2014-12-30 17:50:17 +00:00
|
|
|
if (!gst_aggregator_start (self))
|
|
|
|
goto error_start;
|
2014-05-22 17:44:37 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((ret =
|
|
|
|
GST_ELEMENT_CLASS (aggregator_parent_class)->change_state (element,
|
|
|
|
transition)) == GST_STATE_CHANGE_FAILURE)
|
|
|
|
goto failure;
|
|
|
|
|
|
|
|
|
|
|
|
switch (transition) {
|
|
|
|
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
2014-12-30 17:50:17 +00:00
|
|
|
if (!gst_aggregator_stop (self)) {
|
|
|
|
/* What to do in this case? Error out? */
|
|
|
|
GST_ERROR_OBJECT (self, "Subclass failed to stop.");
|
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
2014-12-30 17:50:17 +00:00
|
|
|
/* ERRORS */
|
2014-05-22 17:44:37 +00:00
|
|
|
failure:
|
|
|
|
{
|
|
|
|
GST_ERROR_OBJECT (element, "parent failed state change");
|
|
|
|
return ret;
|
|
|
|
}
|
2014-12-30 17:50:17 +00:00
|
|
|
error_start:
|
|
|
|
{
|
|
|
|
GST_ERROR_OBJECT (element, "Subclass failed to start");
|
|
|
|
return GST_STATE_CHANGE_FAILURE;
|
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_release_pad (GstElement * element, GstPad * pad)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2014-12-05 07:19:54 +00:00
|
|
|
GstAggregator *self = GST_AGGREGATOR (element);
|
2014-05-22 17:44:37 +00:00
|
|
|
GstAggregatorPad *aggpad = GST_AGGREGATOR_PAD (pad);
|
|
|
|
|
|
|
|
GST_INFO_OBJECT (pad, "Removing pad");
|
|
|
|
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_LOCK (self);
|
2015-03-07 00:50:08 +00:00
|
|
|
gst_aggregator_pad_set_flushing (aggpad, GST_FLOW_FLUSHING, TRUE);
|
2014-05-22 17:44:37 +00:00
|
|
|
gst_element_remove_pad (element, pad);
|
|
|
|
|
2015-07-28 18:15:43 +00:00
|
|
|
self->priv->has_peer_latency = FALSE;
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_BROADCAST (self);
|
|
|
|
SRC_UNLOCK (self);
|
2014-10-06 07:23:03 +00:00
|
|
|
}
|
|
|
|
|
2015-10-23 12:42:24 +00:00
|
|
|
static GstAggregatorPad *
|
|
|
|
gst_aggregator_default_create_new_pad (GstAggregator * self,
|
2014-05-22 17:44:37 +00:00
|
|
|
GstPadTemplate * templ, const gchar * req_name, const GstCaps * caps)
|
|
|
|
{
|
|
|
|
GstAggregatorPad *agg_pad;
|
2015-10-23 12:42:24 +00:00
|
|
|
GstAggregatorPrivate *priv = self->priv;
|
2015-11-04 03:09:33 +00:00
|
|
|
gint serial = 0;
|
|
|
|
gchar *name = NULL;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2016-11-18 19:41:54 +00:00
|
|
|
if (templ->direction != GST_PAD_SINK)
|
2015-11-04 03:09:33 +00:00
|
|
|
goto not_sink;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2016-11-18 19:44:16 +00:00
|
|
|
if (templ->presence != GST_PAD_REQUEST)
|
|
|
|
goto not_request;
|
|
|
|
|
2015-11-04 03:09:33 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
if (req_name == NULL || strlen (req_name) < 6
|
|
|
|
|| !g_str_has_prefix (req_name, "sink_")) {
|
|
|
|
/* no name given when requesting the pad, use next available int */
|
2016-03-27 13:11:30 +00:00
|
|
|
serial = ++priv->max_padserial;
|
2015-11-04 03:09:33 +00:00
|
|
|
} else {
|
|
|
|
/* parse serial number from requested padname */
|
|
|
|
serial = g_ascii_strtoull (&req_name[5], NULL, 10);
|
2016-03-27 13:11:30 +00:00
|
|
|
if (serial > priv->max_padserial)
|
|
|
|
priv->max_padserial = serial;
|
2015-11-04 03:09:33 +00:00
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2016-03-27 13:11:30 +00:00
|
|
|
name = g_strdup_printf ("sink_%u", serial);
|
2015-11-04 03:09:33 +00:00
|
|
|
agg_pad = g_object_new (GST_AGGREGATOR_GET_CLASS (self)->sinkpads_type,
|
|
|
|
"name", name, "direction", GST_PAD_SINK, "template", templ, NULL);
|
|
|
|
g_free (name);
|
2014-10-06 07:23:03 +00:00
|
|
|
|
2015-11-04 03:09:33 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-11-04 03:09:33 +00:00
|
|
|
return agg_pad;
|
|
|
|
|
|
|
|
/* errors */
|
|
|
|
not_sink:
|
|
|
|
{
|
2016-11-18 19:44:16 +00:00
|
|
|
GST_WARNING_OBJECT (self, "request new pad that is not a SINK pad");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
not_request:
|
|
|
|
{
|
|
|
|
GST_WARNING_OBJECT (self, "request new pad that is not a REQUEST pad");
|
2014-05-22 17:44:37 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2015-10-23 12:42:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static GstPad *
|
|
|
|
gst_aggregator_request_new_pad (GstElement * element,
|
|
|
|
GstPadTemplate * templ, const gchar * req_name, const GstCaps * caps)
|
|
|
|
{
|
|
|
|
GstAggregator *self;
|
|
|
|
GstAggregatorPad *agg_pad;
|
|
|
|
GstAggregatorClass *klass = GST_AGGREGATOR_GET_CLASS (element);
|
2016-05-25 17:38:47 +00:00
|
|
|
GstAggregatorPrivate *priv = GST_AGGREGATOR (element)->priv;
|
2015-10-23 12:42:24 +00:00
|
|
|
|
|
|
|
self = GST_AGGREGATOR (element);
|
|
|
|
|
|
|
|
agg_pad = klass->create_new_pad (self, templ, req_name, caps);
|
|
|
|
if (!agg_pad) {
|
|
|
|
GST_ERROR_OBJECT (element, "Couldn't create new pad");
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (element, "Adding pad %s", GST_PAD_NAME (agg_pad));
|
|
|
|
|
2016-05-25 17:38:47 +00:00
|
|
|
if (priv->running)
|
2014-05-22 17:44:37 +00:00
|
|
|
gst_pad_set_active (GST_PAD (agg_pad), TRUE);
|
|
|
|
|
|
|
|
/* add the pad to the element */
|
|
|
|
gst_element_add_pad (element, GST_PAD (agg_pad));
|
|
|
|
|
|
|
|
return GST_PAD (agg_pad);
|
|
|
|
}
|
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
/* Must be called with SRC_LOCK held */
|
|
|
|
|
2014-10-06 10:46:24 +00:00
|
|
|
static gboolean
|
2015-03-07 02:12:52 +00:00
|
|
|
gst_aggregator_query_latency_unlocked (GstAggregator * self, GstQuery * query)
|
2014-10-06 10:46:24 +00:00
|
|
|
{
|
2015-02-26 18:56:00 +00:00
|
|
|
gboolean query_ret, live;
|
|
|
|
GstClockTime our_latency, min, max;
|
2014-10-06 10:46:24 +00:00
|
|
|
|
2015-02-26 18:56:00 +00:00
|
|
|
query_ret = gst_pad_query_default (self->srcpad, GST_OBJECT (self), query);
|
2014-10-06 10:46:24 +00:00
|
|
|
|
2015-02-19 09:04:28 +00:00
|
|
|
if (!query_ret) {
|
|
|
|
GST_WARNING_OBJECT (self, "Latency query failed");
|
|
|
|
return FALSE;
|
|
|
|
}
|
2014-10-06 10:46:24 +00:00
|
|
|
|
2015-02-26 18:56:00 +00:00
|
|
|
gst_query_parse_latency (query, &live, &min, &max);
|
|
|
|
|
2014-12-31 18:16:21 +00:00
|
|
|
our_latency = self->priv->latency;
|
|
|
|
|
2015-02-26 18:56:00 +00:00
|
|
|
if (G_UNLIKELY (!GST_CLOCK_TIME_IS_VALID (min))) {
|
2015-03-07 02:12:13 +00:00
|
|
|
GST_ERROR_OBJECT (self, "Invalid minimum latency %" GST_TIME_FORMAT
|
|
|
|
". Please file a bug at " PACKAGE_BUGREPORT ".", GST_TIME_ARGS (min));
|
|
|
|
return FALSE;
|
2014-12-16 16:33:01 +00:00
|
|
|
}
|
|
|
|
|
2015-03-07 02:12:13 +00:00
|
|
|
if (min > max && GST_CLOCK_TIME_IS_VALID (max)) {
|
|
|
|
GST_ELEMENT_WARNING (self, CORE, CLOCK, (NULL),
|
|
|
|
("Impossible to configure latency: max %" GST_TIME_FORMAT " < min %"
|
|
|
|
GST_TIME_FORMAT ". Add queues or other buffering elements.",
|
|
|
|
GST_TIME_ARGS (max), GST_TIME_ARGS (min)));
|
|
|
|
return FALSE;
|
2014-12-26 22:51:36 +00:00
|
|
|
}
|
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
self->priv->peer_latency_live = live;
|
|
|
|
self->priv->peer_latency_min = min;
|
|
|
|
self->priv->peer_latency_max = max;
|
|
|
|
self->priv->has_peer_latency = TRUE;
|
2014-10-06 10:46:24 +00:00
|
|
|
|
|
|
|
/* add our own */
|
2015-02-26 18:56:00 +00:00
|
|
|
min += our_latency;
|
|
|
|
min += self->priv->sub_latency_min;
|
2014-12-22 21:19:52 +00:00
|
|
|
if (GST_CLOCK_TIME_IS_VALID (self->priv->sub_latency_max)
|
2015-02-26 18:56:00 +00:00
|
|
|
&& GST_CLOCK_TIME_IS_VALID (max))
|
2015-03-07 00:50:08 +00:00
|
|
|
max += self->priv->sub_latency_max + our_latency;
|
2015-02-11 13:16:21 +00:00
|
|
|
else
|
2015-02-26 18:56:00 +00:00
|
|
|
max = GST_CLOCK_TIME_NONE;
|
2014-12-22 21:19:52 +00:00
|
|
|
|
2015-02-20 02:21:56 +00:00
|
|
|
SRC_BROADCAST (self);
|
2015-01-22 00:44:57 +00:00
|
|
|
|
2014-10-06 10:46:24 +00:00
|
|
|
GST_DEBUG_OBJECT (self, "configured latency live:%s min:%" G_GINT64_FORMAT
|
2015-02-26 18:56:00 +00:00
|
|
|
" max:%" G_GINT64_FORMAT, live ? "true" : "false", min, max);
|
2014-10-06 10:46:24 +00:00
|
|
|
|
2015-02-26 18:56:00 +00:00
|
|
|
gst_query_set_latency (query, live, min, max);
|
2014-10-06 10:46:24 +00:00
|
|
|
|
2015-02-19 09:04:28 +00:00
|
|
|
return query_ret;
|
2014-10-06 10:46:24 +00:00
|
|
|
}
|
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
/*
|
|
|
|
* MUST be called with the src_lock held.
|
|
|
|
*
|
|
|
|
* See gst_aggregator_get_latency() for doc
|
|
|
|
*/
|
|
|
|
static GstClockTime
|
|
|
|
gst_aggregator_get_latency_unlocked (GstAggregator * self)
|
|
|
|
{
|
|
|
|
GstClockTime latency;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GST_IS_AGGREGATOR (self), 0);
|
|
|
|
|
|
|
|
if (!self->priv->has_peer_latency) {
|
|
|
|
GstQuery *query = gst_query_new_latency ();
|
|
|
|
gboolean ret;
|
|
|
|
|
|
|
|
ret = gst_aggregator_query_latency_unlocked (self, query);
|
|
|
|
gst_query_unref (query);
|
|
|
|
if (!ret)
|
|
|
|
return GST_CLOCK_TIME_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!self->priv->has_peer_latency || !self->priv->peer_latency_live)
|
|
|
|
return GST_CLOCK_TIME_NONE;
|
|
|
|
|
|
|
|
/* latency_min is never GST_CLOCK_TIME_NONE by construction */
|
|
|
|
latency = self->priv->peer_latency_min;
|
|
|
|
|
|
|
|
/* add our own */
|
|
|
|
latency += self->priv->latency;
|
|
|
|
latency += self->priv->sub_latency_min;
|
|
|
|
|
|
|
|
return latency;
|
|
|
|
}
|
|
|
|
|
2015-02-19 08:57:09 +00:00
|
|
|
/**
|
2015-03-07 02:12:52 +00:00
|
|
|
* gst_aggregator_get_latency:
|
2015-02-19 08:57:09 +00:00
|
|
|
* @self: a #GstAggregator
|
|
|
|
*
|
2015-03-07 02:12:52 +00:00
|
|
|
* Retrieves the latency values reported by @self in response to the latency
|
|
|
|
* query, or %GST_CLOCK_TIME_NONE if there is not live source connected and the element
|
|
|
|
* will not wait for the clock.
|
2015-02-19 08:57:09 +00:00
|
|
|
*
|
|
|
|
* Typically only called by subclasses.
|
|
|
|
*
|
2015-03-07 02:12:52 +00:00
|
|
|
* Returns: The latency or %GST_CLOCK_TIME_NONE if the element does not sync
|
2015-02-19 08:57:09 +00:00
|
|
|
*/
|
2015-03-07 02:12:52 +00:00
|
|
|
GstClockTime
|
|
|
|
gst_aggregator_get_latency (GstAggregator * self)
|
2015-02-19 08:57:09 +00:00
|
|
|
{
|
2015-03-07 02:12:52 +00:00
|
|
|
GstClockTime ret;
|
2015-02-19 08:57:09 +00:00
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
SRC_LOCK (self);
|
|
|
|
ret = gst_aggregator_get_latency_unlocked (self);
|
|
|
|
SRC_UNLOCK (self);
|
2015-02-19 08:57:09 +00:00
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
return ret;
|
2015-02-19 08:57:09 +00:00
|
|
|
}
|
|
|
|
|
2014-07-08 14:16:55 +00:00
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_send_event (GstElement * element, GstEvent * event)
|
2014-07-08 14:16:55 +00:00
|
|
|
{
|
|
|
|
GstAggregator *self = GST_AGGREGATOR (element);
|
|
|
|
|
|
|
|
GST_STATE_LOCK (element);
|
|
|
|
if (GST_EVENT_TYPE (event) == GST_EVENT_SEEK &&
|
|
|
|
GST_STATE (element) < GST_STATE_PAUSED) {
|
|
|
|
gdouble rate;
|
|
|
|
GstFormat fmt;
|
|
|
|
GstSeekFlags flags;
|
|
|
|
GstSeekType start_type, stop_type;
|
|
|
|
gint64 start, stop;
|
|
|
|
|
|
|
|
gst_event_parse_seek (event, &rate, &fmt, &flags, &start_type,
|
|
|
|
&start, &stop_type, &stop);
|
2015-01-26 10:32:47 +00:00
|
|
|
|
|
|
|
GST_OBJECT_LOCK (self);
|
2014-07-08 14:16:55 +00:00
|
|
|
gst_segment_do_seek (&self->segment, rate, fmt, flags, start_type, start,
|
|
|
|
stop_type, stop, NULL);
|
|
|
|
self->priv->seqnum = gst_event_get_seqnum (event);
|
2015-08-29 03:05:20 +00:00
|
|
|
self->priv->first_buffer = FALSE;
|
2015-01-26 10:32:47 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
|
2014-07-08 14:16:55 +00:00
|
|
|
GST_DEBUG_OBJECT (element, "Storing segment %" GST_PTR_FORMAT, event);
|
|
|
|
}
|
|
|
|
GST_STATE_UNLOCK (element);
|
|
|
|
|
|
|
|
|
|
|
|
return GST_ELEMENT_CLASS (aggregator_parent_class)->send_event (element,
|
|
|
|
event);
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_default_src_query (GstAggregator * self, GstQuery * query)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
gboolean res = TRUE;
|
|
|
|
|
|
|
|
switch (GST_QUERY_TYPE (query)) {
|
|
|
|
case GST_QUERY_SEEKING:
|
|
|
|
{
|
|
|
|
GstFormat format;
|
|
|
|
|
|
|
|
/* don't pass it along as some (file)sink might claim it does
|
|
|
|
* whereas with a collectpads in between that will not likely work */
|
|
|
|
gst_query_parse_seeking (query, &format, NULL, NULL, NULL);
|
|
|
|
gst_query_set_seeking (query, format, FALSE, 0, -1);
|
|
|
|
res = TRUE;
|
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
break;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
2014-10-06 10:46:24 +00:00
|
|
|
case GST_QUERY_LATENCY:
|
2015-03-07 02:12:52 +00:00
|
|
|
SRC_LOCK (self);
|
|
|
|
res = gst_aggregator_query_latency_unlocked (self, query);
|
|
|
|
SRC_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
break;
|
2015-03-07 02:12:52 +00:00
|
|
|
default:
|
|
|
|
return gst_pad_query_default (self->srcpad, GST_OBJECT (self), query);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_event_forward_func (GstPad * pad, gpointer user_data)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2014-12-30 23:58:34 +00:00
|
|
|
EventData *evdata = user_data;
|
2014-05-22 17:44:37 +00:00
|
|
|
gboolean ret = TRUE;
|
|
|
|
GstPad *peer = gst_pad_get_peer (pad);
|
2015-01-26 16:06:29 +00:00
|
|
|
GstAggregatorPad *aggpad = GST_AGGREGATOR_PAD (pad);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
if (peer) {
|
2015-08-31 13:12:40 +00:00
|
|
|
if (evdata->only_to_active_pads && aggpad->priv->first_buffer) {
|
|
|
|
GST_DEBUG_OBJECT (pad, "not sending event to inactive pad");
|
|
|
|
ret = TRUE;
|
|
|
|
} else {
|
|
|
|
ret = gst_pad_send_event (peer, gst_event_ref (evdata->event));
|
|
|
|
GST_DEBUG_OBJECT (pad, "return of event push is %d", ret);
|
|
|
|
gst_object_unref (peer);
|
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ret == FALSE) {
|
2014-07-18 11:58:55 +00:00
|
|
|
if (GST_EVENT_TYPE (evdata->event) == GST_EVENT_SEEK) {
|
|
|
|
GstQuery *seeking = gst_query_new_seeking (GST_FORMAT_TIME);
|
|
|
|
|
2015-09-30 17:05:35 +00:00
|
|
|
GST_DEBUG_OBJECT (pad, "Event %" GST_PTR_FORMAT " failed", evdata->event);
|
|
|
|
|
2014-07-18 11:58:55 +00:00
|
|
|
if (gst_pad_query (peer, seeking)) {
|
|
|
|
gboolean seekable;
|
|
|
|
|
|
|
|
gst_query_parse_seeking (seeking, NULL, &seekable, NULL, NULL);
|
|
|
|
|
|
|
|
if (seekable == FALSE) {
|
|
|
|
GST_INFO_OBJECT (pad,
|
|
|
|
"Source not seekable, We failed but it does not matter!");
|
|
|
|
|
|
|
|
ret = TRUE;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
GST_ERROR_OBJECT (pad, "Query seeking FAILED");
|
|
|
|
}
|
2015-01-10 02:30:36 +00:00
|
|
|
|
|
|
|
gst_query_unref (seeking);
|
2014-07-18 11:58:55 +00:00
|
|
|
}
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
if (evdata->flush) {
|
2015-01-26 16:06:29 +00:00
|
|
|
PAD_LOCK (aggpad);
|
|
|
|
aggpad->priv->pending_flush_start = FALSE;
|
|
|
|
aggpad->priv->pending_flush_stop = FALSE;
|
|
|
|
PAD_UNLOCK (aggpad);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
2014-07-18 11:58:55 +00:00
|
|
|
} else {
|
|
|
|
evdata->one_actually_seeked = TRUE;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
2014-07-18 11:58:55 +00:00
|
|
|
evdata->result &= ret;
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
/* Always send to all pads */
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2014-07-18 11:58:55 +00:00
|
|
|
static EventData
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_forward_event_to_all_sinkpads (GstAggregator * self,
|
2015-08-31 13:12:40 +00:00
|
|
|
GstEvent * event, gboolean flush, gboolean only_to_active_pads)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
EventData evdata;
|
|
|
|
|
|
|
|
evdata.event = event;
|
|
|
|
evdata.result = TRUE;
|
|
|
|
evdata.flush = flush;
|
2014-07-18 11:58:55 +00:00
|
|
|
evdata.one_actually_seeked = FALSE;
|
2015-08-31 13:12:40 +00:00
|
|
|
evdata.only_to_active_pads = only_to_active_pads;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
/* We first need to set all pads as flushing in a first pass
|
|
|
|
* as flush_start flush_stop is sometimes sent synchronously
|
|
|
|
* while we send the seek event */
|
2015-01-04 17:15:37 +00:00
|
|
|
if (flush) {
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
for (l = GST_ELEMENT_CAST (self)->sinkpads; l != NULL; l = l->next) {
|
|
|
|
GstAggregatorPad *pad = l->data;
|
|
|
|
|
2015-01-26 16:06:29 +00:00
|
|
|
PAD_LOCK (pad);
|
2015-01-04 17:15:37 +00:00
|
|
|
pad->priv->pending_flush_start = TRUE;
|
|
|
|
pad->priv->pending_flush_stop = FALSE;
|
2015-01-26 16:06:29 +00:00
|
|
|
PAD_UNLOCK (pad);
|
2015-01-04 17:15:37 +00:00
|
|
|
}
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
}
|
2014-12-30 23:58:34 +00:00
|
|
|
|
|
|
|
gst_pad_forward (self->srcpad, gst_aggregator_event_forward_func, &evdata);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
gst_event_unref (event);
|
|
|
|
|
2014-07-18 11:58:55 +00:00
|
|
|
return evdata;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_do_seek (GstAggregator * self, GstEvent * event)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
gdouble rate;
|
|
|
|
GstFormat fmt;
|
|
|
|
GstSeekFlags flags;
|
|
|
|
GstSeekType start_type, stop_type;
|
|
|
|
gint64 start, stop;
|
|
|
|
gboolean flush;
|
2014-07-18 11:58:55 +00:00
|
|
|
EventData evdata;
|
2014-05-22 17:44:37 +00:00
|
|
|
GstAggregatorPrivate *priv = self->priv;
|
|
|
|
|
|
|
|
gst_event_parse_seek (event, &rate, &fmt, &flags, &start_type,
|
|
|
|
&start, &stop_type, &stop);
|
|
|
|
|
|
|
|
GST_INFO_OBJECT (self, "starting SEEK");
|
|
|
|
|
|
|
|
flush = flags & GST_SEEK_FLAG_FLUSH;
|
|
|
|
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
if (flush) {
|
2015-01-21 23:45:36 +00:00
|
|
|
priv->pending_flush_start = TRUE;
|
|
|
|
priv->flush_seeking = TRUE;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gst_segment_do_seek (&self->segment, rate, fmt, flags, start_type, start,
|
|
|
|
stop_type, stop, NULL);
|
2015-06-15 16:30:20 +00:00
|
|
|
|
|
|
|
/* Seeking sets a position */
|
|
|
|
self->priv->first_buffer = FALSE;
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
/* forward the seek upstream */
|
2015-08-31 13:12:40 +00:00
|
|
|
evdata =
|
|
|
|
gst_aggregator_forward_event_to_all_sinkpads (self, event, flush, FALSE);
|
2014-05-22 17:44:37 +00:00
|
|
|
event = NULL;
|
|
|
|
|
2014-07-18 11:58:55 +00:00
|
|
|
if (!evdata.result || !evdata.one_actually_seeked) {
|
2015-01-21 23:45:36 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
priv->flush_seeking = FALSE;
|
|
|
|
priv->pending_flush_start = FALSE;
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
2014-07-18 11:58:55 +00:00
|
|
|
GST_INFO_OBJECT (self, "seek done, result: %d", evdata.result);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2014-07-18 11:58:55 +00:00
|
|
|
return evdata.result;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_default_src_event (GstAggregator * self, GstEvent * event)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2014-07-18 11:58:55 +00:00
|
|
|
EventData evdata;
|
2014-05-22 17:44:37 +00:00
|
|
|
gboolean res = TRUE;
|
|
|
|
|
|
|
|
switch (GST_EVENT_TYPE (event)) {
|
|
|
|
case GST_EVENT_SEEK:
|
|
|
|
{
|
2014-07-08 14:48:08 +00:00
|
|
|
gst_event_ref (event);
|
2014-12-30 23:58:34 +00:00
|
|
|
res = gst_aggregator_do_seek (self, event);
|
2014-07-08 14:48:08 +00:00
|
|
|
gst_event_unref (event);
|
2014-05-22 17:44:37 +00:00
|
|
|
event = NULL;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
case GST_EVENT_NAVIGATION:
|
|
|
|
{
|
|
|
|
/* navigation is rather pointless. */
|
|
|
|
res = FALSE;
|
|
|
|
gst_event_unref (event);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-31 13:12:40 +00:00
|
|
|
/* Don't forward QOS events to pads that had no active buffer yet. Otherwise
|
|
|
|
* they will receive a QOS event that has earliest_time=0 (because we can't
|
|
|
|
* have negative timestamps), and consider their buffer as too late */
|
|
|
|
evdata =
|
|
|
|
gst_aggregator_forward_event_to_all_sinkpads (self, event, FALSE,
|
|
|
|
GST_EVENT_TYPE (event) == GST_EVENT_QOS);
|
2014-07-18 11:58:55 +00:00
|
|
|
res = evdata.result;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
done:
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_src_pad_event_func (GstPad * pad, GstObject * parent,
|
|
|
|
GstEvent * event)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
GstAggregatorClass *klass = GST_AGGREGATOR_GET_CLASS (parent);
|
|
|
|
|
|
|
|
return klass->src_event (GST_AGGREGATOR (parent), event);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_src_pad_query_func (GstPad * pad, GstObject * parent,
|
|
|
|
GstQuery * query)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
GstAggregatorClass *klass = GST_AGGREGATOR_GET_CLASS (parent);
|
|
|
|
|
|
|
|
return klass->src_query (GST_AGGREGATOR (parent), query);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_src_pad_activate_mode_func (GstPad * pad,
|
2014-05-22 17:44:37 +00:00
|
|
|
GstObject * parent, GstPadMode mode, gboolean active)
|
|
|
|
{
|
|
|
|
GstAggregator *self = GST_AGGREGATOR (parent);
|
|
|
|
GstAggregatorClass *klass = GST_AGGREGATOR_GET_CLASS (parent);
|
|
|
|
|
|
|
|
if (klass->src_activate) {
|
|
|
|
if (klass->src_activate (self, mode, active) == FALSE) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (active == TRUE) {
|
|
|
|
switch (mode) {
|
|
|
|
case GST_PAD_MODE_PUSH:
|
|
|
|
{
|
|
|
|
GST_INFO_OBJECT (pad, "Activating pad!");
|
2016-05-25 17:38:47 +00:00
|
|
|
gst_aggregator_start_srcpad_task (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
GST_ERROR_OBJECT (pad, "Only supported mode is PUSH");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* deactivating */
|
|
|
|
GST_INFO_OBJECT (self, "Deactivating srcpad");
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_stop_srcpad_task (self, FALSE);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_default_sink_query (GstAggregator * self,
|
|
|
|
GstAggregatorPad * aggpad, GstQuery * query)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
GstPad *pad = GST_PAD (aggpad);
|
|
|
|
|
2017-05-21 13:19:17 +00:00
|
|
|
if (GST_QUERY_TYPE (query) == GST_QUERY_ALLOCATION) {
|
|
|
|
GstQuery *decide_query = NULL;
|
|
|
|
GstAggregatorClass *agg_class;
|
|
|
|
gboolean ret;
|
|
|
|
|
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
PAD_LOCK (aggpad);
|
|
|
|
if (G_UNLIKELY (!aggpad->priv->negotiated)) {
|
|
|
|
GST_DEBUG_OBJECT (self,
|
|
|
|
"not negotiated yet, can't answer ALLOCATION query");
|
|
|
|
PAD_UNLOCK (aggpad);
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((decide_query = self->priv->allocation_query))
|
|
|
|
gst_query_ref (decide_query);
|
|
|
|
PAD_UNLOCK (aggpad);
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (self,
|
|
|
|
"calling propose allocation with query %" GST_PTR_FORMAT, decide_query);
|
|
|
|
|
|
|
|
agg_class = GST_AGGREGATOR_GET_CLASS (self);
|
|
|
|
|
|
|
|
/* pass the query to the propose_allocation vmethod if any */
|
|
|
|
if (agg_class->propose_allocation)
|
|
|
|
ret = agg_class->propose_allocation (self, aggpad, decide_query, query);
|
|
|
|
else
|
|
|
|
ret = FALSE;
|
|
|
|
|
|
|
|
if (decide_query)
|
|
|
|
gst_query_unref (decide_query);
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (self, "ALLOCATION ret %d, %" GST_PTR_FORMAT, ret, query);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
return gst_pad_query_default (pad, GST_OBJECT (self), query);
|
|
|
|
}
|
|
|
|
|
2014-06-28 12:31:55 +00:00
|
|
|
static void
|
|
|
|
gst_aggregator_finalize (GObject * object)
|
|
|
|
{
|
|
|
|
GstAggregator *self = (GstAggregator *) object;
|
|
|
|
|
2014-12-05 07:19:54 +00:00
|
|
|
g_mutex_clear (&self->priv->src_lock);
|
|
|
|
g_cond_clear (&self->priv->src_cond);
|
2014-06-28 12:31:55 +00:00
|
|
|
|
|
|
|
G_OBJECT_CLASS (aggregator_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
2014-12-05 07:19:54 +00:00
|
|
|
/*
|
|
|
|
* gst_aggregator_set_latency_property:
|
2014-10-06 07:23:03 +00:00
|
|
|
* @agg: a #GstAggregator
|
2015-06-01 22:50:14 +00:00
|
|
|
* @latency: the new latency value (in nanoseconds).
|
2014-10-06 07:23:03 +00:00
|
|
|
*
|
2014-12-05 07:19:54 +00:00
|
|
|
* Sets the new latency value to @latency. This value is used to limit the
|
2014-10-06 07:23:03 +00:00
|
|
|
* amount of time a pad waits for data to appear before considering the pad
|
|
|
|
* as unresponsive.
|
|
|
|
*/
|
|
|
|
static void
|
2014-12-05 07:19:54 +00:00
|
|
|
gst_aggregator_set_latency_property (GstAggregator * self, gint64 latency)
|
2014-10-06 07:23:03 +00:00
|
|
|
{
|
2014-12-22 14:26:37 +00:00
|
|
|
gboolean changed;
|
|
|
|
|
2014-10-06 10:46:24 +00:00
|
|
|
g_return_if_fail (GST_IS_AGGREGATOR (self));
|
2015-02-06 09:59:27 +00:00
|
|
|
g_return_if_fail (GST_CLOCK_TIME_IS_VALID (latency));
|
2014-10-06 07:23:03 +00:00
|
|
|
|
2015-02-20 02:21:56 +00:00
|
|
|
SRC_LOCK (self);
|
2015-03-07 00:50:08 +00:00
|
|
|
changed = (self->priv->latency != latency);
|
|
|
|
|
|
|
|
if (changed) {
|
|
|
|
GList *item;
|
|
|
|
|
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
/* First lock all the pads */
|
|
|
|
for (item = GST_ELEMENT_CAST (self)->sinkpads; item; item = item->next) {
|
|
|
|
GstAggregatorPad *aggpad = GST_AGGREGATOR_PAD (item->data);
|
|
|
|
PAD_LOCK (aggpad);
|
2015-02-06 09:33:59 +00:00
|
|
|
}
|
2014-10-06 10:46:24 +00:00
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
self->priv->latency = latency;
|
2015-02-20 02:21:56 +00:00
|
|
|
|
|
|
|
SRC_BROADCAST (self);
|
2015-03-07 00:50:08 +00:00
|
|
|
|
|
|
|
/* Now wake up the pads */
|
|
|
|
for (item = GST_ELEMENT_CAST (self)->sinkpads; item; item = item->next) {
|
|
|
|
GstAggregatorPad *aggpad = GST_AGGREGATOR_PAD (item->data);
|
|
|
|
PAD_BROADCAST_EVENT (aggpad);
|
|
|
|
PAD_UNLOCK (aggpad);
|
|
|
|
}
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
}
|
|
|
|
|
2015-02-20 02:21:56 +00:00
|
|
|
SRC_UNLOCK (self);
|
2014-12-22 14:26:37 +00:00
|
|
|
|
|
|
|
if (changed)
|
|
|
|
gst_element_post_message (GST_ELEMENT_CAST (self),
|
|
|
|
gst_message_new_latency (GST_OBJECT_CAST (self)));
|
2014-10-06 07:23:03 +00:00
|
|
|
}
|
|
|
|
|
2014-12-05 07:19:54 +00:00
|
|
|
/*
|
|
|
|
* gst_aggregator_get_latency_property:
|
2014-10-06 07:23:03 +00:00
|
|
|
* @agg: a #GstAggregator
|
|
|
|
*
|
2014-12-05 07:19:54 +00:00
|
|
|
* Gets the latency value. See gst_aggregator_set_latency for
|
2014-10-06 07:23:03 +00:00
|
|
|
* more details.
|
|
|
|
*
|
2015-11-04 03:09:33 +00:00
|
|
|
* Returns: The time in nanoseconds to wait for data to arrive on a sink pad
|
2014-10-06 07:23:03 +00:00
|
|
|
* before a pad is deemed unresponsive. A value of -1 means an
|
|
|
|
* unlimited time.
|
|
|
|
*/
|
|
|
|
static gint64
|
2014-12-05 07:19:54 +00:00
|
|
|
gst_aggregator_get_latency_property (GstAggregator * agg)
|
2014-10-06 07:23:03 +00:00
|
|
|
{
|
|
|
|
gint64 res;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GST_IS_AGGREGATOR (agg), -1);
|
|
|
|
|
|
|
|
GST_OBJECT_LOCK (agg);
|
2014-12-31 18:16:21 +00:00
|
|
|
res = agg->priv->latency;
|
2014-10-06 07:23:03 +00:00
|
|
|
GST_OBJECT_UNLOCK (agg);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_aggregator_set_property (GObject * object, guint prop_id,
|
|
|
|
const GValue * value, GParamSpec * pspec)
|
|
|
|
{
|
|
|
|
GstAggregator *agg = GST_AGGREGATOR (object);
|
|
|
|
|
|
|
|
switch (prop_id) {
|
2014-12-05 07:19:54 +00:00
|
|
|
case PROP_LATENCY:
|
|
|
|
gst_aggregator_set_latency_property (agg, g_value_get_int64 (value));
|
2014-10-06 07:23:03 +00:00
|
|
|
break;
|
2015-06-15 16:30:20 +00:00
|
|
|
case PROP_START_TIME_SELECTION:
|
|
|
|
agg->priv->start_time_selection = g_value_get_enum (value);
|
|
|
|
break;
|
|
|
|
case PROP_START_TIME:
|
|
|
|
agg->priv->start_time = g_value_get_uint64 (value);
|
|
|
|
break;
|
2014-10-06 07:23:03 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_aggregator_get_property (GObject * object, guint prop_id,
|
|
|
|
GValue * value, GParamSpec * pspec)
|
|
|
|
{
|
|
|
|
GstAggregator *agg = GST_AGGREGATOR (object);
|
|
|
|
|
|
|
|
switch (prop_id) {
|
2014-12-05 07:19:54 +00:00
|
|
|
case PROP_LATENCY:
|
|
|
|
g_value_set_int64 (value, gst_aggregator_get_latency_property (agg));
|
2014-10-06 07:23:03 +00:00
|
|
|
break;
|
2015-06-15 16:30:20 +00:00
|
|
|
case PROP_START_TIME_SELECTION:
|
|
|
|
g_value_set_enum (value, agg->priv->start_time_selection);
|
|
|
|
break;
|
|
|
|
case PROP_START_TIME:
|
|
|
|
g_value_set_uint64 (value, agg->priv->start_time);
|
|
|
|
break;
|
2014-10-06 07:23:03 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
/* GObject vmethods implementations */
|
|
|
|
static void
|
|
|
|
gst_aggregator_class_init (GstAggregatorClass * klass)
|
|
|
|
{
|
2014-06-28 12:31:55 +00:00
|
|
|
GObjectClass *gobject_class = (GObjectClass *) klass;
|
2014-05-22 17:44:37 +00:00
|
|
|
GstElementClass *gstelement_class = (GstElementClass *) klass;
|
|
|
|
|
|
|
|
aggregator_parent_class = g_type_class_peek_parent (klass);
|
|
|
|
g_type_class_add_private (klass, sizeof (GstAggregatorPrivate));
|
|
|
|
|
|
|
|
GST_DEBUG_CATEGORY_INIT (aggregator_debug, "aggregator",
|
|
|
|
GST_DEBUG_FG_MAGENTA, "GstAggregator");
|
|
|
|
|
|
|
|
klass->sinkpads_type = GST_TYPE_AGGREGATOR_PAD;
|
|
|
|
|
2014-12-30 23:58:34 +00:00
|
|
|
klass->sink_event = gst_aggregator_default_sink_event;
|
|
|
|
klass->sink_query = gst_aggregator_default_sink_query;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2014-12-30 23:58:34 +00:00
|
|
|
klass->src_event = gst_aggregator_default_src_event;
|
|
|
|
klass->src_query = gst_aggregator_default_src_query;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-10-23 12:42:24 +00:00
|
|
|
klass->create_new_pad = gst_aggregator_default_create_new_pad;
|
2017-05-20 12:24:57 +00:00
|
|
|
klass->update_src_caps = gst_aggregator_default_update_src_caps;
|
|
|
|
klass->fixate_src_caps = gst_aggregator_default_fixate_src_caps;
|
|
|
|
klass->negotiated_src_caps = gst_aggregator_default_negotiated_src_caps;
|
2015-10-23 12:42:24 +00:00
|
|
|
|
2014-12-30 23:58:34 +00:00
|
|
|
gstelement_class->request_new_pad =
|
|
|
|
GST_DEBUG_FUNCPTR (gst_aggregator_request_new_pad);
|
|
|
|
gstelement_class->send_event = GST_DEBUG_FUNCPTR (gst_aggregator_send_event);
|
|
|
|
gstelement_class->release_pad =
|
|
|
|
GST_DEBUG_FUNCPTR (gst_aggregator_release_pad);
|
|
|
|
gstelement_class->change_state =
|
|
|
|
GST_DEBUG_FUNCPTR (gst_aggregator_change_state);
|
2014-06-28 12:31:55 +00:00
|
|
|
|
2014-10-06 07:23:03 +00:00
|
|
|
gobject_class->set_property = gst_aggregator_set_property;
|
|
|
|
gobject_class->get_property = gst_aggregator_get_property;
|
2014-06-28 12:31:55 +00:00
|
|
|
gobject_class->finalize = gst_aggregator_finalize;
|
2014-10-06 07:23:03 +00:00
|
|
|
|
2014-12-05 07:19:54 +00:00
|
|
|
g_object_class_install_property (gobject_class, PROP_LATENCY,
|
|
|
|
g_param_spec_int64 ("latency", "Buffer latency",
|
2014-12-23 08:52:20 +00:00
|
|
|
"Additional latency in live mode to allow upstream "
|
|
|
|
"to take longer to produce buffers for the current "
|
2015-06-01 22:50:14 +00:00
|
|
|
"position (in nanoseconds)", 0,
|
2014-10-20 12:55:08 +00:00
|
|
|
(G_MAXLONG == G_MAXINT64) ? G_MAXINT64 : (G_MAXLONG * GST_SECOND - 1),
|
2014-12-05 07:19:54 +00:00
|
|
|
DEFAULT_LATENCY, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
2015-01-01 14:10:05 +00:00
|
|
|
|
2015-06-15 16:30:20 +00:00
|
|
|
g_object_class_install_property (gobject_class, PROP_START_TIME_SELECTION,
|
|
|
|
g_param_spec_enum ("start-time-selection", "Start Time Selection",
|
|
|
|
"Decides which start time is output",
|
|
|
|
gst_aggregator_start_time_selection_get_type (),
|
|
|
|
DEFAULT_START_TIME_SELECTION,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
|
|
|
|
g_object_class_install_property (gobject_class, PROP_START_TIME,
|
|
|
|
g_param_spec_uint64 ("start-time", "Start Time",
|
|
|
|
"Start time to use if start-time-selection=set", 0,
|
|
|
|
G_MAXUINT64,
|
|
|
|
DEFAULT_START_TIME, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
|
2015-01-01 14:10:05 +00:00
|
|
|
GST_DEBUG_REGISTER_FUNCPTR (gst_aggregator_stop_pad);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_aggregator_init (GstAggregator * self, GstAggregatorClass * klass)
|
|
|
|
{
|
|
|
|
GstPadTemplate *pad_template;
|
|
|
|
GstAggregatorPrivate *priv;
|
|
|
|
|
|
|
|
g_return_if_fail (klass->aggregate != NULL);
|
|
|
|
|
|
|
|
self->priv =
|
|
|
|
G_TYPE_INSTANCE_GET_PRIVATE (self, GST_TYPE_AGGREGATOR,
|
|
|
|
GstAggregatorPrivate);
|
|
|
|
|
|
|
|
priv = self->priv;
|
|
|
|
|
|
|
|
pad_template =
|
|
|
|
gst_element_class_get_pad_template (GST_ELEMENT_CLASS (klass), "src");
|
|
|
|
g_return_if_fail (pad_template != NULL);
|
|
|
|
|
2016-03-27 13:11:30 +00:00
|
|
|
priv->max_padserial = -1;
|
2014-05-22 17:44:37 +00:00
|
|
|
priv->tags_changed = FALSE;
|
2014-10-06 10:46:24 +00:00
|
|
|
|
2015-03-07 02:12:52 +00:00
|
|
|
self->priv->peer_latency_live = FALSE;
|
|
|
|
self->priv->peer_latency_min = self->priv->sub_latency_min = 0;
|
|
|
|
self->priv->peer_latency_max = self->priv->sub_latency_max = 0;
|
|
|
|
self->priv->has_peer_latency = FALSE;
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_reset_flow_values (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
self->srcpad = gst_pad_new_from_template (pad_template, "src");
|
|
|
|
|
|
|
|
gst_pad_set_event_function (self->srcpad,
|
2014-12-30 23:58:34 +00:00
|
|
|
GST_DEBUG_FUNCPTR (gst_aggregator_src_pad_event_func));
|
2014-05-22 17:44:37 +00:00
|
|
|
gst_pad_set_query_function (self->srcpad,
|
2014-12-30 23:58:34 +00:00
|
|
|
GST_DEBUG_FUNCPTR (gst_aggregator_src_pad_query_func));
|
2014-05-22 17:44:37 +00:00
|
|
|
gst_pad_set_activatemode_function (self->srcpad,
|
2014-12-30 23:58:34 +00:00
|
|
|
GST_DEBUG_FUNCPTR (gst_aggregator_src_pad_activate_mode_func));
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
gst_element_add_pad (GST_ELEMENT (self), self->srcpad);
|
2014-06-28 12:31:55 +00:00
|
|
|
|
2014-12-31 18:16:21 +00:00
|
|
|
self->priv->latency = DEFAULT_LATENCY;
|
2015-06-15 16:30:20 +00:00
|
|
|
self->priv->start_time_selection = DEFAULT_START_TIME_SELECTION;
|
|
|
|
self->priv->start_time = DEFAULT_START_TIME;
|
2014-10-06 07:23:03 +00:00
|
|
|
|
2014-12-05 07:19:54 +00:00
|
|
|
g_mutex_init (&self->priv->src_lock);
|
|
|
|
g_cond_init (&self->priv->src_cond);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* we can't use G_DEFINE_ABSTRACT_TYPE because we need the klass in the _init
|
|
|
|
* method to get to the padtemplates */
|
|
|
|
GType
|
|
|
|
gst_aggregator_get_type (void)
|
|
|
|
{
|
|
|
|
static volatile gsize type = 0;
|
|
|
|
|
|
|
|
if (g_once_init_enter (&type)) {
|
|
|
|
GType _type;
|
|
|
|
static const GTypeInfo info = {
|
|
|
|
sizeof (GstAggregatorClass),
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
(GClassInitFunc) gst_aggregator_class_init,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
sizeof (GstAggregator),
|
|
|
|
0,
|
|
|
|
(GInstanceInitFunc) gst_aggregator_init,
|
|
|
|
};
|
|
|
|
|
|
|
|
_type = g_type_register_static (GST_TYPE_ELEMENT,
|
|
|
|
"GstAggregator", &info, G_TYPE_FLAG_ABSTRACT);
|
|
|
|
g_once_init_leave (&type, _type);
|
|
|
|
}
|
|
|
|
return type;
|
|
|
|
}
|
|
|
|
|
2015-09-17 23:42:34 +00:00
|
|
|
/* Must be called with SRC lock and PAD lock held */
|
2015-03-07 00:50:08 +00:00
|
|
|
static gboolean
|
|
|
|
gst_aggregator_pad_has_space (GstAggregator * self, GstAggregatorPad * aggpad)
|
|
|
|
{
|
|
|
|
/* Empty queue always has space */
|
2016-07-06 20:39:17 +00:00
|
|
|
if (aggpad->priv->num_buffers == 0 && aggpad->priv->clipped_buffer == NULL)
|
2015-03-07 00:50:08 +00:00
|
|
|
return TRUE;
|
|
|
|
|
2015-09-17 23:42:34 +00:00
|
|
|
/* We also want at least two buffers, one is being processed and one is ready
|
|
|
|
* for the next iteration when we operate in live mode. */
|
|
|
|
if (self->priv->peer_latency_live && aggpad->priv->num_buffers < 2)
|
|
|
|
return TRUE;
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
/* zero latency, if there is a buffer, it's full */
|
|
|
|
if (self->priv->latency == 0)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* Allow no more buffers than the latency */
|
|
|
|
return (aggpad->priv->time_level <= self->priv->latency);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Must be called with the PAD_LOCK held */
|
|
|
|
static void
|
|
|
|
apply_buffer (GstAggregatorPad * aggpad, GstBuffer * buffer, gboolean head)
|
|
|
|
{
|
|
|
|
GstClockTime timestamp;
|
|
|
|
|
|
|
|
if (GST_BUFFER_DTS_IS_VALID (buffer))
|
|
|
|
timestamp = GST_BUFFER_DTS (buffer);
|
|
|
|
else
|
|
|
|
timestamp = GST_BUFFER_PTS (buffer);
|
|
|
|
|
|
|
|
if (timestamp == GST_CLOCK_TIME_NONE) {
|
|
|
|
if (head)
|
|
|
|
timestamp = aggpad->priv->head_position;
|
|
|
|
else
|
|
|
|
timestamp = aggpad->priv->tail_position;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* add duration */
|
|
|
|
if (GST_BUFFER_DURATION_IS_VALID (buffer))
|
|
|
|
timestamp += GST_BUFFER_DURATION (buffer);
|
|
|
|
|
|
|
|
if (head)
|
|
|
|
aggpad->priv->head_position = timestamp;
|
|
|
|
else
|
|
|
|
aggpad->priv->tail_position = timestamp;
|
|
|
|
|
|
|
|
update_time_level (aggpad, head);
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
static GstFlowReturn
|
2015-03-07 00:50:08 +00:00
|
|
|
gst_aggregator_pad_chain_internal (GstAggregator * self,
|
|
|
|
GstAggregatorPad * aggpad, GstBuffer * buffer, gboolean head)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2015-01-21 23:53:20 +00:00
|
|
|
GstFlowReturn flow_return;
|
2015-03-07 00:50:08 +00:00
|
|
|
GstClockTime buf_pts;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (aggpad, "Start chaining a buffer %" GST_PTR_FORMAT, buffer);
|
|
|
|
|
2015-02-18 20:06:01 +00:00
|
|
|
PAD_FLUSH_LOCK (aggpad);
|
2014-10-06 07:23:03 +00:00
|
|
|
|
2015-01-26 16:06:29 +00:00
|
|
|
PAD_LOCK (aggpad);
|
2015-04-02 02:10:11 +00:00
|
|
|
flow_return = aggpad->priv->flow_return;
|
|
|
|
if (flow_return != GST_FLOW_OK)
|
|
|
|
goto flushing;
|
|
|
|
|
2015-01-26 16:06:29 +00:00
|
|
|
if (aggpad->priv->pending_eos == TRUE)
|
2014-05-22 17:44:37 +00:00
|
|
|
goto eos;
|
|
|
|
|
2015-04-02 01:45:01 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
|
|
|
|
2016-07-06 20:41:44 +00:00
|
|
|
buf_pts = GST_BUFFER_PTS (buffer);
|
2015-03-07 00:50:08 +00:00
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
SRC_LOCK (self);
|
2015-11-03 01:10:35 +00:00
|
|
|
GST_OBJECT_LOCK (self);
|
2015-03-07 00:50:08 +00:00
|
|
|
PAD_LOCK (aggpad);
|
2017-05-20 11:10:53 +00:00
|
|
|
|
|
|
|
if (aggpad->priv->first_buffer) {
|
|
|
|
self->priv->has_peer_latency = FALSE;
|
|
|
|
aggpad->priv->first_buffer = FALSE;
|
|
|
|
}
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
if (gst_aggregator_pad_has_space (self, aggpad)
|
|
|
|
&& aggpad->priv->flow_return == GST_FLOW_OK) {
|
|
|
|
if (head)
|
2017-09-17 18:39:12 +00:00
|
|
|
g_queue_push_head (&aggpad->priv->data, buffer);
|
2015-03-07 00:50:08 +00:00
|
|
|
else
|
2017-09-17 18:39:12 +00:00
|
|
|
g_queue_push_tail (&aggpad->priv->data, buffer);
|
2016-07-06 20:41:44 +00:00
|
|
|
apply_buffer (aggpad, buffer, head);
|
2015-09-17 23:42:34 +00:00
|
|
|
aggpad->priv->num_buffers++;
|
2016-07-06 20:41:44 +00:00
|
|
|
buffer = NULL;
|
2015-03-07 00:50:08 +00:00
|
|
|
SRC_BROADCAST (self);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
flow_return = aggpad->priv->flow_return;
|
|
|
|
if (flow_return != GST_FLOW_OK) {
|
2015-11-03 01:10:35 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2015-03-07 00:50:08 +00:00
|
|
|
SRC_UNLOCK (self);
|
|
|
|
goto flushing;
|
|
|
|
}
|
|
|
|
GST_DEBUG_OBJECT (aggpad, "Waiting for buffer to be consumed");
|
2015-11-03 01:10:35 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2015-03-07 00:50:08 +00:00
|
|
|
SRC_UNLOCK (self);
|
|
|
|
PAD_WAIT_EVENT (aggpad);
|
|
|
|
|
|
|
|
PAD_UNLOCK (aggpad);
|
|
|
|
}
|
2015-04-02 02:10:11 +00:00
|
|
|
|
2015-06-15 16:30:20 +00:00
|
|
|
if (self->priv->first_buffer) {
|
|
|
|
GstClockTime start_time;
|
|
|
|
|
|
|
|
switch (self->priv->start_time_selection) {
|
|
|
|
case GST_AGGREGATOR_START_TIME_SELECTION_ZERO:
|
|
|
|
default:
|
|
|
|
start_time = 0;
|
|
|
|
break;
|
|
|
|
case GST_AGGREGATOR_START_TIME_SELECTION_FIRST:
|
2015-11-03 01:10:35 +00:00
|
|
|
GST_OBJECT_LOCK (aggpad);
|
2016-07-06 20:39:17 +00:00
|
|
|
if (aggpad->priv->head_segment.format == GST_FORMAT_TIME) {
|
2015-03-07 00:50:08 +00:00
|
|
|
start_time = buf_pts;
|
2015-07-30 00:06:11 +00:00
|
|
|
if (start_time != -1) {
|
2016-07-06 20:39:17 +00:00
|
|
|
start_time = MAX (start_time, aggpad->priv->head_segment.start);
|
2015-07-30 00:06:11 +00:00
|
|
|
start_time =
|
2016-07-06 20:39:17 +00:00
|
|
|
gst_segment_to_running_time (&aggpad->priv->head_segment,
|
|
|
|
GST_FORMAT_TIME, start_time);
|
2015-07-30 00:06:11 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
start_time = 0;
|
|
|
|
GST_WARNING_OBJECT (aggpad,
|
|
|
|
"Ignoring request of selecting the first start time "
|
|
|
|
"as the segment is a %s segment instead of a time segment",
|
|
|
|
gst_format_get_name (aggpad->segment.format));
|
2015-06-15 16:30:20 +00:00
|
|
|
}
|
2015-11-03 01:10:35 +00:00
|
|
|
GST_OBJECT_UNLOCK (aggpad);
|
2015-06-15 16:30:20 +00:00
|
|
|
break;
|
|
|
|
case GST_AGGREGATOR_START_TIME_SELECTION_SET:
|
|
|
|
start_time = self->priv->start_time;
|
|
|
|
if (start_time == -1)
|
|
|
|
start_time = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (start_time != -1) {
|
|
|
|
if (self->segment.position == -1)
|
|
|
|
self->segment.position = start_time;
|
|
|
|
else
|
|
|
|
self->segment.position = MIN (start_time, self->segment.position);
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (self, "Selecting start time %" GST_TIME_FORMAT,
|
|
|
|
GST_TIME_ARGS (start_time));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-14 19:35:15 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2015-11-03 01:10:35 +00:00
|
|
|
GST_OBJECT_UNLOCK (self);
|
2015-02-18 20:11:14 +00:00
|
|
|
SRC_UNLOCK (self);
|
2014-10-06 07:23:03 +00:00
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
PAD_FLUSH_UNLOCK (aggpad);
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
GST_DEBUG_OBJECT (aggpad, "Done chaining");
|
|
|
|
|
2015-01-21 23:53:20 +00:00
|
|
|
return flow_return;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
flushing:
|
2015-04-02 01:45:01 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2015-02-18 20:06:01 +00:00
|
|
|
PAD_FLUSH_UNLOCK (aggpad);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-04-02 02:10:11 +00:00
|
|
|
GST_DEBUG_OBJECT (aggpad, "Pad is %s, dropping buffer",
|
|
|
|
gst_flow_get_name (flow_return));
|
2015-03-07 00:50:08 +00:00
|
|
|
if (buffer)
|
|
|
|
gst_buffer_unref (buffer);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-04-02 02:10:11 +00:00
|
|
|
return flow_return;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
eos:
|
2015-01-26 16:06:29 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2015-02-18 20:06:01 +00:00
|
|
|
PAD_FLUSH_UNLOCK (aggpad);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2014-06-28 12:34:05 +00:00
|
|
|
gst_buffer_unref (buffer);
|
2015-03-07 00:50:08 +00:00
|
|
|
GST_DEBUG_OBJECT (aggpad, "We are EOS already...");
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
return GST_FLOW_EOS;
|
|
|
|
}
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
static GstFlowReturn
|
|
|
|
gst_aggregator_pad_chain (GstPad * pad, GstObject * object, GstBuffer * buffer)
|
|
|
|
{
|
|
|
|
return gst_aggregator_pad_chain_internal (GST_AGGREGATOR_CAST (object),
|
|
|
|
GST_AGGREGATOR_PAD_CAST (pad), buffer, TRUE);
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_pad_query_func (GstPad * pad, GstObject * parent,
|
|
|
|
GstQuery * query)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2017-05-22 22:53:57 +00:00
|
|
|
GstAggregator *self = GST_AGGREGATOR (parent);
|
2014-11-19 16:03:41 +00:00
|
|
|
GstAggregatorPad *aggpad = GST_AGGREGATOR_PAD (pad);
|
2014-05-22 17:44:37 +00:00
|
|
|
GstAggregatorClass *klass = GST_AGGREGATOR_GET_CLASS (parent);
|
|
|
|
|
2014-11-19 16:03:41 +00:00
|
|
|
if (GST_QUERY_IS_SERIALIZED (query)) {
|
2017-05-22 22:53:57 +00:00
|
|
|
GstStructure *s;
|
|
|
|
gboolean ret = FALSE;
|
|
|
|
|
|
|
|
SRC_LOCK (self);
|
2015-01-14 19:35:15 +00:00
|
|
|
PAD_LOCK (aggpad);
|
2014-11-19 16:17:06 +00:00
|
|
|
|
2017-05-22 22:53:57 +00:00
|
|
|
if (aggpad->priv->flow_return != GST_FLOW_OK) {
|
|
|
|
SRC_UNLOCK (self);
|
|
|
|
goto flushing;
|
|
|
|
}
|
|
|
|
|
2017-09-17 18:39:12 +00:00
|
|
|
g_queue_push_head (&aggpad->priv->data, query);
|
2017-05-22 22:53:57 +00:00
|
|
|
SRC_BROADCAST (self);
|
|
|
|
SRC_UNLOCK (self);
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
while (!gst_aggregator_pad_queue_is_empty (aggpad)
|
|
|
|
&& aggpad->priv->flow_return == GST_FLOW_OK) {
|
|
|
|
GST_DEBUG_OBJECT (aggpad, "Waiting for buffer to be consumed");
|
2014-11-19 16:03:41 +00:00
|
|
|
PAD_WAIT_EVENT (aggpad);
|
2015-03-07 00:50:08 +00:00
|
|
|
}
|
2014-11-19 16:15:02 +00:00
|
|
|
|
2017-05-22 22:53:57 +00:00
|
|
|
s = gst_query_writable_structure (query);
|
|
|
|
if (gst_structure_get_boolean (s, "gst-aggregator-retval", &ret))
|
|
|
|
gst_structure_remove_field (s, "gst-aggregator-retval");
|
|
|
|
else
|
2017-09-17 18:39:12 +00:00
|
|
|
g_queue_remove (&aggpad->priv->data, query);
|
2017-05-22 22:53:57 +00:00
|
|
|
|
2015-04-02 02:10:11 +00:00
|
|
|
if (aggpad->priv->flow_return != GST_FLOW_OK)
|
2014-11-19 16:15:02 +00:00
|
|
|
goto flushing;
|
2015-04-02 01:45:01 +00:00
|
|
|
|
|
|
|
PAD_UNLOCK (aggpad);
|
2017-05-22 22:53:57 +00:00
|
|
|
|
|
|
|
return ret;
|
2014-11-19 16:03:41 +00:00
|
|
|
}
|
|
|
|
|
2017-05-22 22:53:57 +00:00
|
|
|
return klass->sink_query (self, aggpad, query);
|
2014-11-19 16:15:02 +00:00
|
|
|
|
|
|
|
flushing:
|
2015-04-02 02:10:11 +00:00
|
|
|
GST_DEBUG_OBJECT (aggpad, "Pad is %s, dropping query",
|
|
|
|
gst_flow_get_name (aggpad->priv->flow_return));
|
2015-04-02 01:45:01 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2017-05-22 22:53:57 +00:00
|
|
|
|
2014-11-19 16:15:02 +00:00
|
|
|
return FALSE;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
2016-09-06 19:05:53 +00:00
|
|
|
static GstFlowReturn
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_pad_event_func (GstPad * pad, GstObject * parent,
|
|
|
|
GstEvent * event)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2016-09-06 19:05:53 +00:00
|
|
|
GstFlowReturn ret = GST_FLOW_OK;
|
2015-03-07 00:50:08 +00:00
|
|
|
GstAggregator *self = GST_AGGREGATOR (parent);
|
2014-11-19 16:03:41 +00:00
|
|
|
GstAggregatorPad *aggpad = GST_AGGREGATOR_PAD (pad);
|
2014-05-22 17:44:37 +00:00
|
|
|
GstAggregatorClass *klass = GST_AGGREGATOR_GET_CLASS (parent);
|
|
|
|
|
2014-11-19 16:03:41 +00:00
|
|
|
if (GST_EVENT_IS_SERIALIZED (event) && GST_EVENT_TYPE (event) != GST_EVENT_EOS
|
2015-03-07 00:50:08 +00:00
|
|
|
/* && GST_EVENT_TYPE (event) != GST_EVENT_SEGMENT_DONE */ ) {
|
|
|
|
SRC_LOCK (self);
|
2015-01-14 19:35:15 +00:00
|
|
|
PAD_LOCK (aggpad);
|
2014-11-19 16:17:06 +00:00
|
|
|
|
2015-04-02 02:10:11 +00:00
|
|
|
if (aggpad->priv->flow_return != GST_FLOW_OK
|
2016-09-06 19:05:53 +00:00
|
|
|
&& GST_EVENT_TYPE (event) != GST_EVENT_FLUSH_STOP) {
|
|
|
|
ret = aggpad->priv->flow_return;
|
2014-11-19 16:15:02 +00:00
|
|
|
goto flushing;
|
2016-09-06 19:05:53 +00:00
|
|
|
}
|
2015-04-02 01:45:01 +00:00
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
if (GST_EVENT_TYPE (event) == GST_EVENT_SEGMENT) {
|
|
|
|
GST_OBJECT_LOCK (aggpad);
|
2016-07-06 20:39:17 +00:00
|
|
|
gst_event_copy_segment (event, &aggpad->priv->head_segment);
|
|
|
|
aggpad->priv->head_position = aggpad->priv->head_segment.position;
|
2015-03-07 00:50:08 +00:00
|
|
|
update_time_level (aggpad, TRUE);
|
|
|
|
GST_OBJECT_UNLOCK (aggpad);
|
|
|
|
}
|
|
|
|
|
2016-05-14 13:52:37 +00:00
|
|
|
if (GST_EVENT_TYPE (event) != GST_EVENT_FLUSH_STOP) {
|
2015-03-07 00:50:08 +00:00
|
|
|
GST_DEBUG_OBJECT (aggpad, "Store event in queue: %" GST_PTR_FORMAT,
|
|
|
|
event);
|
2017-09-17 18:39:12 +00:00
|
|
|
g_queue_push_head (&aggpad->priv->data, event);
|
2015-03-07 00:50:08 +00:00
|
|
|
event = NULL;
|
|
|
|
SRC_BROADCAST (self);
|
|
|
|
}
|
2015-04-02 01:45:01 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2015-03-07 00:50:08 +00:00
|
|
|
SRC_UNLOCK (self);
|
2014-11-19 16:03:41 +00:00
|
|
|
}
|
|
|
|
|
2016-09-06 19:05:53 +00:00
|
|
|
if (event) {
|
|
|
|
if (!klass->sink_event (self, aggpad, event)) {
|
|
|
|
/* Copied from GstPad to convert boolean to a GstFlowReturn in
|
|
|
|
* the event handling func */
|
2017-05-21 12:28:00 +00:00
|
|
|
ret = GST_FLOW_ERROR;
|
2016-09-06 19:05:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2014-11-19 16:15:02 +00:00
|
|
|
|
|
|
|
flushing:
|
2015-04-02 02:10:11 +00:00
|
|
|
GST_DEBUG_OBJECT (aggpad, "Pad is %s, dropping event",
|
|
|
|
gst_flow_get_name (aggpad->priv->flow_return));
|
2015-04-02 01:45:01 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2015-03-07 00:50:08 +00:00
|
|
|
SRC_UNLOCK (self);
|
2014-11-19 16:15:02 +00:00
|
|
|
if (GST_EVENT_IS_STICKY (event))
|
|
|
|
gst_pad_store_sticky_event (pad, event);
|
|
|
|
gst_event_unref (event);
|
2016-09-06 19:05:53 +00:00
|
|
|
|
|
|
|
return ret;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2014-12-30 23:58:34 +00:00
|
|
|
gst_aggregator_pad_activate_mode_func (GstPad * pad,
|
2014-05-22 17:44:37 +00:00
|
|
|
GstObject * parent, GstPadMode mode, gboolean active)
|
|
|
|
{
|
2015-03-07 00:50:08 +00:00
|
|
|
GstAggregator *self = GST_AGGREGATOR (parent);
|
2014-05-22 17:44:37 +00:00
|
|
|
GstAggregatorPad *aggpad = GST_AGGREGATOR_PAD (pad);
|
|
|
|
|
|
|
|
if (active == FALSE) {
|
2015-03-07 00:50:08 +00:00
|
|
|
SRC_LOCK (self);
|
|
|
|
gst_aggregator_pad_set_flushing (aggpad, GST_FLOW_FLUSHING, TRUE);
|
|
|
|
SRC_BROADCAST (self);
|
|
|
|
SRC_UNLOCK (self);
|
2014-05-22 17:44:37 +00:00
|
|
|
} else {
|
2015-01-14 19:35:15 +00:00
|
|
|
PAD_LOCK (aggpad);
|
2015-04-02 02:10:11 +00:00
|
|
|
aggpad->priv->flow_return = GST_FLOW_OK;
|
2014-05-22 17:44:37 +00:00
|
|
|
PAD_BROADCAST_EVENT (aggpad);
|
2015-01-14 19:35:15 +00:00
|
|
|
PAD_UNLOCK (aggpad);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************
|
|
|
|
* GstAggregatorPad implementation *
|
|
|
|
************************************/
|
|
|
|
G_DEFINE_TYPE (GstAggregatorPad, gst_aggregator_pad, GST_TYPE_PAD);
|
|
|
|
|
|
|
|
static void
|
2014-12-28 18:26:49 +00:00
|
|
|
gst_aggregator_pad_constructed (GObject * object)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
|
|
|
GstPad *pad = GST_PAD (object);
|
|
|
|
|
|
|
|
gst_pad_set_chain_function (pad,
|
2014-12-30 23:58:34 +00:00
|
|
|
GST_DEBUG_FUNCPTR (gst_aggregator_pad_chain));
|
2016-09-06 19:05:53 +00:00
|
|
|
gst_pad_set_event_full_function_full (pad,
|
2017-05-10 00:13:58 +00:00
|
|
|
GST_DEBUG_FUNCPTR (gst_aggregator_pad_event_func), NULL, NULL);
|
2014-05-22 17:44:37 +00:00
|
|
|
gst_pad_set_query_function (pad,
|
2014-12-30 23:58:34 +00:00
|
|
|
GST_DEBUG_FUNCPTR (gst_aggregator_pad_query_func));
|
2014-05-22 17:44:37 +00:00
|
|
|
gst_pad_set_activatemode_function (pad,
|
2014-12-30 23:58:34 +00:00
|
|
|
GST_DEBUG_FUNCPTR (gst_aggregator_pad_activate_mode_func));
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
2014-06-28 12:31:55 +00:00
|
|
|
static void
|
|
|
|
gst_aggregator_pad_finalize (GObject * object)
|
|
|
|
{
|
|
|
|
GstAggregatorPad *pad = (GstAggregatorPad *) object;
|
|
|
|
|
|
|
|
g_cond_clear (&pad->priv->event_cond);
|
2015-02-18 20:06:01 +00:00
|
|
|
g_mutex_clear (&pad->priv->flush_lock);
|
2015-01-26 10:29:08 +00:00
|
|
|
g_mutex_clear (&pad->priv->lock);
|
2014-06-28 12:31:55 +00:00
|
|
|
|
2014-12-28 18:24:21 +00:00
|
|
|
G_OBJECT_CLASS (gst_aggregator_pad_parent_class)->finalize (object);
|
2014-06-28 12:31:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_aggregator_pad_dispose (GObject * object)
|
|
|
|
{
|
|
|
|
GstAggregatorPad *pad = (GstAggregatorPad *) object;
|
|
|
|
|
2015-03-07 00:50:08 +00:00
|
|
|
gst_aggregator_pad_set_flushing (pad, GST_FLOW_FLUSHING, TRUE);
|
2014-06-28 12:31:55 +00:00
|
|
|
|
2014-12-28 18:24:21 +00:00
|
|
|
G_OBJECT_CLASS (gst_aggregator_pad_parent_class)->dispose (object);
|
2014-06-28 12:31:55 +00:00
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
static void
|
|
|
|
gst_aggregator_pad_class_init (GstAggregatorPadClass * klass)
|
|
|
|
{
|
|
|
|
GObjectClass *gobject_class = (GObjectClass *) klass;
|
|
|
|
|
|
|
|
g_type_class_add_private (klass, sizeof (GstAggregatorPadPrivate));
|
|
|
|
|
2014-12-28 18:26:49 +00:00
|
|
|
gobject_class->constructed = gst_aggregator_pad_constructed;
|
|
|
|
gobject_class->finalize = gst_aggregator_pad_finalize;
|
|
|
|
gobject_class->dispose = gst_aggregator_pad_dispose;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_aggregator_pad_init (GstAggregatorPad * pad)
|
|
|
|
{
|
|
|
|
pad->priv =
|
|
|
|
G_TYPE_INSTANCE_GET_PRIVATE (pad, GST_TYPE_AGGREGATOR_PAD,
|
|
|
|
GstAggregatorPadPrivate);
|
|
|
|
|
2017-09-17 18:39:12 +00:00
|
|
|
g_queue_init (&pad->priv->data);
|
2014-05-22 17:44:37 +00:00
|
|
|
g_cond_init (&pad->priv->event_cond);
|
|
|
|
|
2015-02-18 20:06:01 +00:00
|
|
|
g_mutex_init (&pad->priv->flush_lock);
|
2015-01-26 10:29:08 +00:00
|
|
|
g_mutex_init (&pad->priv->lock);
|
2015-08-31 13:12:40 +00:00
|
|
|
|
2017-05-10 00:05:55 +00:00
|
|
|
gst_aggregator_pad_reset_unlocked (pad);
|
2017-05-21 12:34:13 +00:00
|
|
|
pad->priv->negotiated = FALSE;
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
|
|
|
|
2016-07-06 20:39:17 +00:00
|
|
|
/* Must be called with the PAD_LOCK held */
|
|
|
|
static void
|
|
|
|
gst_aggregator_pad_buffer_consumed (GstAggregatorPad * pad)
|
|
|
|
{
|
|
|
|
pad->priv->num_buffers--;
|
|
|
|
GST_TRACE_OBJECT (pad, "Consuming buffer");
|
|
|
|
if (gst_aggregator_pad_queue_is_empty (pad) && pad->priv->pending_eos) {
|
|
|
|
pad->priv->pending_eos = FALSE;
|
|
|
|
pad->priv->eos = TRUE;
|
|
|
|
}
|
|
|
|
PAD_BROADCAST_EVENT (pad);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Must be called with the PAD_LOCK held */
|
|
|
|
static void
|
|
|
|
gst_aggregator_pad_clip_buffer_unlocked (GstAggregatorPad * pad)
|
|
|
|
{
|
|
|
|
GstAggregator *self = NULL;
|
2017-07-01 18:23:25 +00:00
|
|
|
GstAggregatorClass *aggclass = NULL;
|
2016-07-06 20:39:17 +00:00
|
|
|
GstBuffer *buffer = NULL;
|
|
|
|
|
|
|
|
while (pad->priv->clipped_buffer == NULL &&
|
2017-09-17 18:39:12 +00:00
|
|
|
GST_IS_BUFFER (g_queue_peek_tail (&pad->priv->data))) {
|
|
|
|
buffer = g_queue_pop_tail (&pad->priv->data);
|
2016-07-06 20:39:17 +00:00
|
|
|
|
|
|
|
apply_buffer (pad, buffer, FALSE);
|
|
|
|
|
|
|
|
/* We only take the parent here so that it's not taken if the buffer is
|
|
|
|
* already clipped or if the queue is empty.
|
|
|
|
*/
|
|
|
|
if (self == NULL) {
|
|
|
|
self = GST_AGGREGATOR (gst_pad_get_parent_element (GST_PAD (pad)));
|
|
|
|
if (self == NULL) {
|
|
|
|
gst_buffer_unref (buffer);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
aggclass = GST_AGGREGATOR_GET_CLASS (self);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aggclass->clip) {
|
|
|
|
GST_TRACE_OBJECT (pad, "Clipping: %" GST_PTR_FORMAT, buffer);
|
|
|
|
|
|
|
|
buffer = aggclass->clip (self, pad, buffer);
|
|
|
|
|
|
|
|
if (buffer == NULL) {
|
|
|
|
gst_aggregator_pad_buffer_consumed (pad);
|
|
|
|
GST_TRACE_OBJECT (pad, "Clipping consumed the buffer");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pad->priv->clipped_buffer = buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self)
|
|
|
|
gst_object_unref (self);
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
/**
|
2015-01-26 10:29:08 +00:00
|
|
|
* gst_aggregator_pad_steal_buffer:
|
2014-05-22 17:44:37 +00:00
|
|
|
* @pad: the pad to get buffer from
|
|
|
|
*
|
|
|
|
* Steal the ref to the buffer currently queued in @pad.
|
|
|
|
*
|
|
|
|
* Returns: (transfer full): The buffer in @pad or NULL if no buffer was
|
|
|
|
* queued. You should unref the buffer after usage.
|
|
|
|
*/
|
|
|
|
GstBuffer *
|
2015-01-26 10:29:08 +00:00
|
|
|
gst_aggregator_pad_steal_buffer (GstAggregatorPad * pad)
|
2014-05-22 17:44:37 +00:00
|
|
|
{
|
2016-07-06 20:39:17 +00:00
|
|
|
GstBuffer *buffer;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-01-26 10:29:08 +00:00
|
|
|
PAD_LOCK (pad);
|
2016-07-06 20:39:17 +00:00
|
|
|
|
|
|
|
gst_aggregator_pad_clip_buffer_unlocked (pad);
|
|
|
|
|
|
|
|
buffer = pad->priv->clipped_buffer;
|
2015-03-07 00:50:08 +00:00
|
|
|
|
|
|
|
if (buffer) {
|
2017-09-17 19:30:37 +00:00
|
|
|
pad->priv->clipped_buffer = NULL;
|
2016-07-06 20:39:17 +00:00
|
|
|
gst_aggregator_pad_buffer_consumed (pad);
|
2014-11-19 16:03:33 +00:00
|
|
|
GST_DEBUG_OBJECT (pad, "Consumed: %" GST_PTR_FORMAT, buffer);
|
2014-05-22 17:44:37 +00:00
|
|
|
}
|
2016-07-06 20:39:17 +00:00
|
|
|
|
2015-01-14 19:35:15 +00:00
|
|
|
PAD_UNLOCK (pad);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
2015-02-13 15:49:50 +00:00
|
|
|
/**
|
|
|
|
* gst_aggregator_pad_drop_buffer:
|
|
|
|
* @pad: the pad where to drop any pending buffer
|
|
|
|
*
|
|
|
|
* Drop the buffer currently queued in @pad.
|
|
|
|
*
|
|
|
|
* Returns: TRUE if there was a buffer queued in @pad, or FALSE if not.
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
gst_aggregator_pad_drop_buffer (GstAggregatorPad * pad)
|
|
|
|
{
|
|
|
|
GstBuffer *buf;
|
|
|
|
|
|
|
|
buf = gst_aggregator_pad_steal_buffer (pad);
|
|
|
|
|
|
|
|
if (buf == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
gst_buffer_unref (buf);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
/**
|
|
|
|
* gst_aggregator_pad_get_buffer:
|
|
|
|
* @pad: the pad to get buffer from
|
|
|
|
*
|
|
|
|
* Returns: (transfer full): A reference to the buffer in @pad or
|
|
|
|
* NULL if no buffer was queued. You should unref the buffer after
|
|
|
|
* usage.
|
|
|
|
*/
|
|
|
|
GstBuffer *
|
|
|
|
gst_aggregator_pad_get_buffer (GstAggregatorPad * pad)
|
|
|
|
{
|
2016-07-06 20:39:17 +00:00
|
|
|
GstBuffer *buffer;
|
2014-05-22 17:44:37 +00:00
|
|
|
|
2015-01-14 19:35:15 +00:00
|
|
|
PAD_LOCK (pad);
|
2015-03-07 00:50:08 +00:00
|
|
|
|
2016-07-06 20:39:17 +00:00
|
|
|
gst_aggregator_pad_clip_buffer_unlocked (pad);
|
|
|
|
|
|
|
|
if (pad->priv->clipped_buffer) {
|
|
|
|
buffer = gst_buffer_ref (pad->priv->clipped_buffer);
|
|
|
|
} else {
|
2015-03-07 00:50:08 +00:00
|
|
|
buffer = NULL;
|
2016-07-06 20:39:17 +00:00
|
|
|
}
|
2015-01-14 19:35:15 +00:00
|
|
|
PAD_UNLOCK (pad);
|
2014-05-22 17:44:37 +00:00
|
|
|
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
2015-01-26 10:25:54 +00:00
|
|
|
gboolean
|
|
|
|
gst_aggregator_pad_is_eos (GstAggregatorPad * pad)
|
|
|
|
{
|
|
|
|
gboolean is_eos;
|
|
|
|
|
|
|
|
PAD_LOCK (pad);
|
|
|
|
is_eos = pad->priv->eos;
|
|
|
|
PAD_UNLOCK (pad);
|
|
|
|
|
|
|
|
return is_eos;
|
|
|
|
}
|
|
|
|
|
2014-05-22 17:44:37 +00:00
|
|
|
/**
|
|
|
|
* gst_aggregator_merge_tags:
|
|
|
|
* @self: a #GstAggregator
|
|
|
|
* @tags: a #GstTagList to merge
|
|
|
|
* @mode: the #GstTagMergeMode to use
|
|
|
|
*
|
|
|
|
* Adds tags to so-called pending tags, which will be processed
|
|
|
|
* before pushing out data downstream.
|
|
|
|
*
|
|
|
|
* Note that this is provided for convenience, and the subclass is
|
|
|
|
* not required to use this and can still do tag handling on its own.
|
|
|
|
*
|
|
|
|
* MT safe.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gst_aggregator_merge_tags (GstAggregator * self,
|
|
|
|
const GstTagList * tags, GstTagMergeMode mode)
|
|
|
|
{
|
|
|
|
GstTagList *otags;
|
|
|
|
|
|
|
|
g_return_if_fail (GST_IS_AGGREGATOR (self));
|
|
|
|
g_return_if_fail (tags == NULL || GST_IS_TAG_LIST (tags));
|
|
|
|
|
|
|
|
/* FIXME Check if we can use OBJECT lock here! */
|
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
if (tags)
|
|
|
|
GST_DEBUG_OBJECT (self, "merging tags %" GST_PTR_FORMAT, tags);
|
|
|
|
otags = self->priv->tags;
|
|
|
|
self->priv->tags = gst_tag_list_merge (self->priv->tags, tags, mode);
|
|
|
|
if (otags)
|
|
|
|
gst_tag_list_unref (otags);
|
|
|
|
self->priv->tags_changed = TRUE;
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
}
|
2014-12-17 18:51:32 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* gst_aggregator_set_latency:
|
|
|
|
* @self: a #GstAggregator
|
|
|
|
* @min_latency: minimum latency
|
|
|
|
* @max_latency: maximum latency
|
|
|
|
*
|
|
|
|
* Lets #GstAggregator sub-classes tell the baseclass what their internal
|
|
|
|
* latency is. Will also post a LATENCY message on the bus so the pipeline
|
|
|
|
* can reconfigure its global latency.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gst_aggregator_set_latency (GstAggregator * self,
|
|
|
|
GstClockTime min_latency, GstClockTime max_latency)
|
|
|
|
{
|
2015-02-12 13:32:39 +00:00
|
|
|
gboolean changed = FALSE;
|
|
|
|
|
2014-12-17 18:51:32 +00:00
|
|
|
g_return_if_fail (GST_IS_AGGREGATOR (self));
|
2014-12-27 08:49:43 +00:00
|
|
|
g_return_if_fail (GST_CLOCK_TIME_IS_VALID (min_latency));
|
2014-12-17 18:51:32 +00:00
|
|
|
g_return_if_fail (max_latency >= min_latency);
|
|
|
|
|
2015-02-20 02:21:56 +00:00
|
|
|
SRC_LOCK (self);
|
2015-02-12 13:32:39 +00:00
|
|
|
if (self->priv->sub_latency_min != min_latency) {
|
|
|
|
self->priv->sub_latency_min = min_latency;
|
|
|
|
changed = TRUE;
|
|
|
|
}
|
|
|
|
if (self->priv->sub_latency_max != max_latency) {
|
|
|
|
self->priv->sub_latency_max = max_latency;
|
|
|
|
changed = TRUE;
|
|
|
|
}
|
2015-02-20 02:21:56 +00:00
|
|
|
|
|
|
|
if (changed)
|
|
|
|
SRC_BROADCAST (self);
|
|
|
|
SRC_UNLOCK (self);
|
2014-12-17 18:51:32 +00:00
|
|
|
|
2015-02-12 13:32:39 +00:00
|
|
|
if (changed) {
|
|
|
|
gst_element_post_message (GST_ELEMENT_CAST (self),
|
|
|
|
gst_message_new_latency (GST_OBJECT_CAST (self)));
|
|
|
|
}
|
2014-12-17 18:51:32 +00:00
|
|
|
}
|
2017-05-20 14:58:54 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* gst_aggregator_get_buffer_pool:
|
|
|
|
* @self: a #GstAggregator
|
|
|
|
*
|
|
|
|
* Returns: (transfer full): the instance of the #GstBufferPool used
|
|
|
|
* by @trans; free it after use it
|
|
|
|
*/
|
|
|
|
GstBufferPool *
|
|
|
|
gst_aggregator_get_buffer_pool (GstAggregator * self)
|
|
|
|
{
|
|
|
|
GstBufferPool *pool;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GST_IS_AGGREGATOR (self), NULL);
|
|
|
|
|
|
|
|
GST_OBJECT_LOCK (self);
|
|
|
|
pool = self->priv->pool;
|
|
|
|
if (pool)
|
|
|
|
gst_object_ref (pool);
|
|
|
|
GST_OBJECT_UNLOCK (self);
|
|
|
|
|
|
|
|
return pool;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gst_aggregator_get_allocator:
|
|
|
|
* @self: a #GstAggregator
|
|
|
|
* @allocator: (out) (allow-none) (transfer full): the #GstAllocator
|
|
|
|
* used
|
|
|
|
* @params: (out) (allow-none) (transfer full): the
|
|
|
|
* #GstAllocationParams of @allocator
|
|
|
|
*
|
|
|
|
* Lets #GstAggregator sub-classes get the memory @allocator
|
|
|
|
* acquired by the base class and its @params.
|
|
|
|
*
|
|
|
|
* Unref the @allocator after use it.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gst_aggregator_get_allocator (GstAggregator * self,
|
|
|
|
GstAllocator ** allocator, GstAllocationParams * params)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GST_IS_AGGREGATOR (self));
|
|
|
|
|
|
|
|
if (allocator)
|
|
|
|
*allocator = self->priv->allocator ?
|
|
|
|
gst_object_ref (self->priv->allocator) : NULL;
|
|
|
|
|
|
|
|
if (params)
|
|
|
|
*params = self->priv->allocation_params;
|
|
|
|
}
|