summaryrefslogtreecommitdiffstats
path: root/gst
diff options
context:
space:
mode:
Diffstat (limited to 'gst')
-rw-r--r--gst/mpegdemux/gstmpegtsdemux.c95
-rw-r--r--gst/mpegdemux/gstmpegtsdemux.h1
2 files changed, 75 insertions, 21 deletions
diff --git a/gst/mpegdemux/gstmpegtsdemux.c b/gst/mpegdemux/gstmpegtsdemux.c
index 6ac05031..1aa2f387 100644
--- a/gst/mpegdemux/gstmpegtsdemux.c
+++ b/gst/mpegdemux/gstmpegtsdemux.c
@@ -1046,7 +1046,7 @@ gst_fluts_demux_get_stream_for_PID (GstFluTSDemux * demux, guint16 PID)
stream->PAT.version_number = -1;
stream->PMT_pid = FLUTS_MAX_PID + 1;
stream->flags |= FLUTS_STREAM_FLAG_STREAM_TYPE_UNKNOWN;
-
+ stream->pes_buffer_in_sync = FALSE;
switch (PID) {
/* check for fixed mapping */
case PID_PROGRAM_ASSOCIATION_TABLE:
@@ -1959,6 +1959,8 @@ gst_fluts_stream_pes_buffer_flush (GstFluTSStream * stream)
if (stream->pes_buffer) {
GST_BUFFER_SIZE (stream->pes_buffer) = stream->pes_buffer_used;
ret = gst_pes_filter_push (&stream->filter, stream->pes_buffer);
+ if (ret == GST_FLOW_LOST_SYNC)
+ stream->pes_buffer_in_sync = FALSE;
stream->pes_buffer = NULL;
}
return ret;
@@ -1981,6 +1983,8 @@ gst_fluts_stream_pes_buffer_push (GstFluTSStream * stream,
stream->pes_buffer_size <<= 1;
ret = gst_fluts_stream_pes_buffer_flush (stream);
+ if (ret == GST_FLOW_LOST_SYNC)
+ goto done;
}
if (G_UNLIKELY (!stream->pes_buffer)) {
@@ -1998,7 +2002,7 @@ gst_fluts_stream_pes_buffer_push (GstFluTSStream * stream,
memcpy (out_data, in_data, in_size);
#endif
stream->pes_buffer_used += in_size;
-
+done:
return ret;
}
@@ -2011,13 +2015,32 @@ gst_fluts_demux_pes_buffer_flush (GstFluTSDemux * demux)
for (i = 0; i < FLUTS_MAX_PID + 1; i++) {
GstFluTSStream *stream = demux->streams[i];
if (stream && stream->pad) {
- ret = gst_fluts_stream_pes_buffer_flush (stream);
- if (G_UNLIKELY (ret == GST_FLOW_OK))
- goto done;
+ gst_fluts_stream_pes_buffer_flush (stream);
+ stream->pes_buffer_in_sync = FALSE;
}
}
+ return ret;
+}
-done:
+static FORCE_INLINE GstFlowReturn
+gst_fluts_demux_push_fragment (GstFluTSStream * stream,
+ const guint8 * in_data, guint in_size)
+{
+ GstFlowReturn ret;
+ GstBuffer *es_buf = gst_buffer_new_and_alloc (in_size);
+#ifdef USE_LIBOIL
+ oil_memcpy (GST_BUFFER_DATA (es_buf), in_data, in_size);
+#else
+ memcpy (GST_BUFFER_DATA (es_buf), in_data, in_size);
+#endif
+ ret = gst_pes_filter_push (&stream->filter, es_buf);
+
+ /* If PES filter return ok then PES fragment buffering
+ * can be enabled */
+ if (ret == GST_FLOW_OK)
+ stream->pes_buffer_in_sync = TRUE;
+ else if (ret == GST_FLOW_LOST_SYNC)
+ stream->pes_buffer_in_sync = FALSE;
return ret;
}
@@ -2207,7 +2230,7 @@ gst_fluts_demux_parse_stream (GstFluTSDemux * demux, GstFluTSStream * stream,
"bytes of %u bytes in the PES buffer",
PID, stream->pes_buffer_used, stream->pes_buffer_size);
/* Flush buffered PES data */
- ret = gst_fluts_stream_pes_buffer_flush (stream);
+ gst_fluts_stream_pes_buffer_flush (stream);
gst_pes_filter_drain (&stream->filter);
/* Resize the buffer to half if no overflow detected and
* had been used less than half of it */
@@ -2219,15 +2242,33 @@ gst_fluts_demux_parse_stream (GstFluTSDemux * demux, GstFluTSStream * stream,
GST_DEBUG_OBJECT (demux, "PES buffer size reduced to %u bytes",
stream->pes_buffer_size);
}
- if (ret == GST_FLOW_LOST_SYNC)
- goto done;
+ /* mark the stream not in sync to give a chance on PES filter to
+ * detect lost sync */
+ stream->pes_buffer_in_sync = FALSE;
stream->pes_buffer_overflow = FALSE;
}
GST_LOG_OBJECT (demux, "Elementary packet of size %u for PID 0x%04x",
datalen, PID);
if (datalen > 0) {
- ret = gst_fluts_stream_pes_buffer_push (stream, data, datalen);
+ if (!stream->pes_buffer_in_sync) {
+ /* Push the first fragment to PES filter to have a chance to
+ * detect GST_FLOW_LOST_SYNC.
+ */
+ GST_LOG_OBJECT (demux, "fragment directly pushed to PES filter");
+ ret = gst_fluts_demux_push_fragment (stream, data, datalen);
+ } else {
+ /* Otherwhise we buffer the PES fragment */
+ ret = gst_fluts_stream_pes_buffer_push (stream, data, datalen);
+ /* If sync is lost here is due a pes_buffer_flush and we can try
+ * to resync in the PES filter with the current fragment
+ */
+ if (ret == GST_FLOW_LOST_SYNC) {
+ GST_LOG_OBJECT (demux, "resync, fragment pushed to PES filter");
+ ret = gst_fluts_demux_push_fragment (stream, data, datalen);
+ }
+ }
+
break;
} else {
GST_WARNING_OBJECT (demux, "overflow of datalen: %u so skipping",
@@ -2428,15 +2469,20 @@ gst_fluts_demux_sync_scan (GstFluTSDemux * demux, const guint8 * in_data,
while (ptr_data <= end_scan && sync_count < LENGHT_SYNC_LUT) {
/* if sync code is found try to store it in the LUT */
if (G_LIKELY (IS_MPEGTS_SYNC (ptr_data))) {
- demux->sync_lut[sync_count] = ptr_data;
- sync_count++;
/* skip paketsize bytes and try find next */
- ptr_data += demux->packetsize;
+ guint8 *next_sync = ptr_data + demux->packetsize;
+ if (next_sync < end_scan) {
+ demux->sync_lut[sync_count] = ptr_data;
+ sync_count++;
+ ptr_data += demux->packetsize;
+ } else
+ goto done;
+
} else {
ptr_data++;
}
}
-
+done:
*flush = ptr_data - in_data;
return sync_count;
@@ -2453,9 +2499,11 @@ gst_fluts_demux_chain (GstPad * pad, GstBuffer * buffer)
gint i;
guint sync_count;
- if (GST_BUFFER_IS_DISCONT (buffer))
+ if (GST_BUFFER_IS_DISCONT (buffer)) {
+ /* Flush buffered PES data */
+ gst_fluts_demux_pes_buffer_flush (demux);
gst_adapter_clear (demux->adapter);
-
+ }
/* first push the new buffer into the adapter */
gst_adapter_push (demux->adapter, buffer);
@@ -2473,19 +2521,24 @@ gst_fluts_demux_chain (GstPad * pad, GstBuffer * buffer)
/* process all packets */
for (i = 0; i < sync_count; i++) {
ret = gst_fluts_demux_parse_transport_packet (demux, demux->sync_lut[i]);
+ if (G_UNLIKELY (ret == GST_FLOW_LOST_SYNC)) {
+ flush = demux->sync_lut[i] - data + 1;
+ ret = GST_FLOW_OK;
+ continue;
+ }
if (G_UNLIKELY (ret != GST_FLOW_OK)) {
- if (ret == GST_FLOW_LOST_SYNC) {
- ret = GST_FLOW_OK;
- continue;
- }
flush = demux->sync_lut[i] - data + demux->packetsize;
+ flush = MIN (avail, flush);
goto done;
}
}
done:
/* flush processed data */
- gst_adapter_flush (demux->adapter, flush);
+ if (flush) {
+ GST_DEBUG_OBJECT (demux, "flushing %d/%d", flush, avail);
+ gst_adapter_flush (demux->adapter, flush);
+ }
gst_object_unref (demux);
diff --git a/gst/mpegdemux/gstmpegtsdemux.h b/gst/mpegdemux/gstmpegtsdemux.h
index d26aedcd..4878290a 100644
--- a/gst/mpegdemux/gstmpegtsdemux.h
+++ b/gst/mpegdemux/gstmpegtsdemux.h
@@ -164,6 +164,7 @@ struct _GstFluTSStream {
guint32 pes_buffer_size;
guint32 pes_buffer_used;
gboolean pes_buffer_overflow;
+ gboolean pes_buffer_in_sync;
GstPESFilter filter;
GstPad * pad;
GstFlowReturn last_ret;