mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-07 07:55:41 +00:00
mpegdemux: Add sparse stream filling.
First stab at sending new-segment events to effect sparse stream updates.
This commit is contained in:
parent
e1d778e559
commit
f7eefea47c
2 changed files with 97 additions and 0 deletions
|
@ -57,6 +57,9 @@
|
||||||
#define SCAN_SCR_SZ 12
|
#define SCAN_SCR_SZ 12
|
||||||
#define SCAN_PTS_SZ 80
|
#define SCAN_PTS_SZ 80
|
||||||
|
|
||||||
|
#define SEGMENT_THRESHOLD (300*GST_MSECOND)
|
||||||
|
#define VIDEO_SEGMENT_THRESHOLD (500*GST_MSECOND)
|
||||||
|
|
||||||
typedef enum
|
typedef enum
|
||||||
{
|
{
|
||||||
SCAN_SCR,
|
SCAN_SCR,
|
||||||
|
@ -217,6 +220,10 @@ static inline gboolean gst_flups_demux_scan_forward_ts (GstFluPSDemux * demux,
|
||||||
static inline gboolean gst_flups_demux_scan_backward_ts (GstFluPSDemux * demux,
|
static inline gboolean gst_flups_demux_scan_backward_ts (GstFluPSDemux * demux,
|
||||||
guint64 * pos, SCAN_MODE mode, guint64 * rts);
|
guint64 * pos, SCAN_MODE mode, guint64 * rts);
|
||||||
|
|
||||||
|
static void gst_flups_demux_send_segment_updates (GstFluPSDemux * demux,
|
||||||
|
GstClockTime new_time);
|
||||||
|
static void gst_flups_demux_clear_times (GstFluPSDemux * demux);
|
||||||
|
|
||||||
static GstElementClass *parent_class = NULL;
|
static GstElementClass *parent_class = NULL;
|
||||||
|
|
||||||
/*static guint gst_flups_demux_signals[LAST_SIGNAL] = { 0 };*/
|
/*static guint gst_flups_demux_signals[LAST_SIGNAL] = { 0 };*/
|
||||||
|
@ -352,6 +359,7 @@ gst_flups_demux_create_stream (GstFluPSDemux * demux, gint id, gint stream_type)
|
||||||
gchar *name;
|
gchar *name;
|
||||||
GstFluPSDemuxClass *klass = GST_FLUPS_DEMUX_GET_CLASS (demux);
|
GstFluPSDemuxClass *klass = GST_FLUPS_DEMUX_GET_CLASS (demux);
|
||||||
GstCaps *caps;
|
GstCaps *caps;
|
||||||
|
GstClockTime threshold = SEGMENT_THRESHOLD;
|
||||||
|
|
||||||
name = NULL;
|
name = NULL;
|
||||||
template = NULL;
|
template = NULL;
|
||||||
|
@ -380,6 +388,7 @@ gst_flups_demux_create_stream (GstFluPSDemux * demux, gint id, gint stream_type)
|
||||||
caps = gst_caps_new_simple ("video/mpeg",
|
caps = gst_caps_new_simple ("video/mpeg",
|
||||||
"mpegversion", G_TYPE_INT, mpeg_version,
|
"mpegversion", G_TYPE_INT, mpeg_version,
|
||||||
"systemstream", G_TYPE_BOOLEAN, FALSE, NULL);
|
"systemstream", G_TYPE_BOOLEAN, FALSE, NULL);
|
||||||
|
threshold = VIDEO_SEGMENT_THRESHOLD;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ST_AUDIO_MPEG1:
|
case ST_AUDIO_MPEG1:
|
||||||
|
@ -399,6 +408,7 @@ gst_flups_demux_create_stream (GstFluPSDemux * demux, gint id, gint stream_type)
|
||||||
template = klass->video_template;
|
template = klass->video_template;
|
||||||
name = g_strdup_printf ("video_%02x", id);
|
name = g_strdup_printf ("video_%02x", id);
|
||||||
caps = gst_caps_new_simple ("video/x-h264", NULL);
|
caps = gst_caps_new_simple ("video/x-h264", NULL);
|
||||||
|
threshold = VIDEO_SEGMENT_THRESHOLD;
|
||||||
break;
|
break;
|
||||||
case ST_PS_AUDIO_AC3:
|
case ST_PS_AUDIO_AC3:
|
||||||
template = klass->audio_template;
|
template = klass->audio_template;
|
||||||
|
@ -439,6 +449,7 @@ gst_flups_demux_create_stream (GstFluPSDemux * demux, gint id, gint stream_type)
|
||||||
stream->notlinked = FALSE;
|
stream->notlinked = FALSE;
|
||||||
stream->type = stream_type;
|
stream->type = stream_type;
|
||||||
stream->pad = gst_pad_new_from_template (template, name);
|
stream->pad = gst_pad_new_from_template (template, name);
|
||||||
|
stream->segment_thresh = threshold;
|
||||||
gst_pad_set_event_function (stream->pad,
|
gst_pad_set_event_function (stream->pad,
|
||||||
GST_DEBUG_FUNCPTR (gst_flups_demux_src_event));
|
GST_DEBUG_FUNCPTR (gst_flups_demux_src_event));
|
||||||
gst_pad_set_query_function (stream->pad,
|
gst_pad_set_query_function (stream->pad,
|
||||||
|
@ -574,6 +585,20 @@ gst_flups_demux_send_data (GstFluPSDemux * demux, GstFluPSStream * stream,
|
||||||
GST_TIME_ARGS (demux->src_segment.last_stop),
|
GST_TIME_ARGS (demux->src_segment.last_stop),
|
||||||
GST_TIME_ARGS (MPEGTIME_TO_GSTTIME (demux->current_scr)));
|
GST_TIME_ARGS (MPEGTIME_TO_GSTTIME (demux->current_scr)));
|
||||||
|
|
||||||
|
if (demux->src_segment.last_stop != GST_CLOCK_TIME_NONE) {
|
||||||
|
GstClockTime new_time = demux->base_time + demux->src_segment.last_stop;
|
||||||
|
|
||||||
|
if (stream->last_ts == GST_CLOCK_TIME_NONE || stream->last_ts < new_time) {
|
||||||
|
#if 0
|
||||||
|
g_print ("last_ts update on pad %s to time %" GST_TIME_FORMAT "\n",
|
||||||
|
GST_PAD_NAME (stream->pad), GST_TIME_ARGS (cur_scr_time));
|
||||||
|
#endif
|
||||||
|
stream->last_ts = new_time;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_flups_demux_send_segment_updates (demux, new_time);
|
||||||
|
}
|
||||||
|
|
||||||
/* Set the buffer discont flag, and clear discont state on the stream */
|
/* Set the buffer discont flag, and clear discont state on the stream */
|
||||||
if (stream->discont) {
|
if (stream->discont) {
|
||||||
GST_DEBUG_OBJECT (demux, "marking discont buffer");
|
GST_DEBUG_OBJECT (demux, "marking discont buffer");
|
||||||
|
@ -746,11 +771,75 @@ gst_flups_demux_flush (GstFluPSDemux * demux)
|
||||||
gst_adapter_clear (demux->adapter);
|
gst_adapter_clear (demux->adapter);
|
||||||
gst_adapter_clear (demux->rev_adapter);
|
gst_adapter_clear (demux->rev_adapter);
|
||||||
gst_pes_filter_drain (&demux->filter);
|
gst_pes_filter_drain (&demux->filter);
|
||||||
|
gst_flups_demux_clear_times (demux);
|
||||||
demux->adapter_offset = G_MAXUINT64;
|
demux->adapter_offset = G_MAXUINT64;
|
||||||
demux->current_scr = G_MAXUINT64;
|
demux->current_scr = G_MAXUINT64;
|
||||||
demux->bytes_since_scr = 0;
|
demux->bytes_since_scr = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_flups_demux_clear_times (GstFluPSDemux * demux)
|
||||||
|
{
|
||||||
|
gint id;
|
||||||
|
|
||||||
|
/* Clear the last ts for all streams */
|
||||||
|
for (id = 0; id < GST_FLUPS_DEMUX_MAX_STREAMS; id++) {
|
||||||
|
GstFluPSStream *stream = demux->streams[id];
|
||||||
|
|
||||||
|
if (stream) {
|
||||||
|
stream->last_seg_start = stream->last_ts = GST_CLOCK_TIME_NONE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_flups_demux_send_segment_updates (GstFluPSDemux * demux,
|
||||||
|
GstClockTime new_time)
|
||||||
|
{
|
||||||
|
/* Advance all lagging streams by sending a segment update */
|
||||||
|
gint id;
|
||||||
|
GstEvent *event = NULL;
|
||||||
|
|
||||||
|
/* FIXME: Handle reverse playback */
|
||||||
|
|
||||||
|
if (new_time > demux->src_segment.stop)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (id = 0; id < GST_FLUPS_DEMUX_MAX_STREAMS; id++) {
|
||||||
|
GstFluPSStream *stream = demux->streams[id];
|
||||||
|
|
||||||
|
if (stream) {
|
||||||
|
if (stream->last_ts == GST_CLOCK_TIME_NONE ||
|
||||||
|
stream->last_ts < demux->src_segment.start)
|
||||||
|
stream->last_ts = demux->src_segment.start;
|
||||||
|
if (stream->last_ts + stream->segment_thresh < new_time) {
|
||||||
|
#if 0
|
||||||
|
g_print ("Segment update to pad %s time %" GST_TIME_FORMAT " stop now %"
|
||||||
|
GST_TIME_FORMAT "\n", GST_PAD_NAME (stream->pad),
|
||||||
|
GST_TIME_ARGS (new_time), GST_TIME_ARGS (demux->src_segment.stop));
|
||||||
|
#endif
|
||||||
|
GST_DEBUG_OBJECT (demux,
|
||||||
|
"Segment update to pad %s time %" GST_TIME_FORMAT,
|
||||||
|
GST_PAD_NAME (stream->pad), GST_TIME_ARGS (new_time));
|
||||||
|
if (event == NULL) {
|
||||||
|
event = gst_event_new_new_segment_full (TRUE,
|
||||||
|
demux->src_segment.rate, demux->src_segment.applied_rate,
|
||||||
|
GST_FORMAT_TIME, new_time,
|
||||||
|
demux->src_segment.stop,
|
||||||
|
demux->src_segment.time + (new_time - demux->src_segment.start));
|
||||||
|
}
|
||||||
|
gst_event_ref (event);
|
||||||
|
gst_pad_push_event (stream->pad, event);
|
||||||
|
stream->last_seg_start = stream->last_ts = new_time;
|
||||||
|
stream->need_segment = FALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event)
|
||||||
|
gst_event_unref (event);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_flups_demux_close_segment (GstFluPSDemux * demux)
|
gst_flups_demux_close_segment (GstFluPSDemux * demux)
|
||||||
{
|
{
|
||||||
|
@ -762,6 +851,10 @@ gst_flups_demux_close_segment (GstFluPSDemux * demux)
|
||||||
&demux->src_segment);
|
&demux->src_segment);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* FIXME: Need to send a different segment-close to each pad where the
|
||||||
|
* last_seg_start != clock_time_none, as that indicates a sparse-stream
|
||||||
|
* event was sent there */
|
||||||
|
|
||||||
/* Close the current segment for a linear playback */
|
/* Close the current segment for a linear playback */
|
||||||
if (demux->src_segment.rate >= 0) {
|
if (demux->src_segment.rate >= 0) {
|
||||||
/* for forward playback, we played from start to last_stop */
|
/* for forward playback, we played from start to last_stop */
|
||||||
|
|
|
@ -86,6 +86,10 @@ struct _GstFluPSStream
|
||||||
gint type;
|
gint type;
|
||||||
gint size_bound;
|
gint size_bound;
|
||||||
|
|
||||||
|
GstClockTime segment_thresh;
|
||||||
|
GstClockTime last_seg_start;
|
||||||
|
GstClockTime last_ts;
|
||||||
|
|
||||||
gboolean discont;
|
gboolean discont;
|
||||||
gboolean notlinked;
|
gboolean notlinked;
|
||||||
gboolean need_segment;
|
gboolean need_segment;
|
||||||
|
|
Loading…
Reference in a new issue