diff options
author | Zaheer Abbas Merali <zaheerabbas@merali.org> | 2008-09-23 17:34:44 +0000 |
---|---|---|
committer | Zaheer Abbas Merali <zaheerabbas@merali.org> | 2008-09-23 17:34:44 +0000 |
commit | 4c23ebf7216be71fb19811d407fb5f1ce86da964 (patch) | |
tree | 35eb1a4baa9c929fbfcae598f3d9989ac17b1c36 | |
parent | 52fd692bcfae0e29dc821d22fcad2533d390e1a9 (diff) | |
download | gst-plugins-bad-4c23ebf7216be71fb19811d407fb5f1ce86da964.tar.gz gst-plugins-bad-4c23ebf7216be71fb19811d407fb5f1ce86da964.tar.bz2 gst-plugins-bad-4c23ebf7216be71fb19811d407fb5f1ce86da964.zip |
Patch from: Josep Torra
Original commit message from CVS:
Patch from: Josep Torra
* gst/mpegdemux/gstmpegtsdemux.c:
* gst/mpegdemux/gstmpegtsdemux.h:
Use a preallocated buffer per stream for PES packets sent on src pads.
Adaptively adjust buffer size appropriately.
-rw-r--r-- | ChangeLog | 9 | ||||
-rw-r--r-- | gst/mpegdemux/gstmpegtsdemux.c | 99 | ||||
-rw-r--r-- | gst/mpegdemux/gstmpegtsdemux.h | 7 |
3 files changed, 105 insertions, 10 deletions
@@ -1,3 +1,12 @@ +2008-09-23 Zaheer Abbas Merali <zaheerabbas at merali dot org> + + Patch from: Josep Torra + + * gst/mpegdemux/gstmpegtsdemux.c: + * gst/mpegdemux/gstmpegtsdemux.h: + Use a preallocated buffer per stream for PES packets sent on src pads. + Adaptively adjust buffer size appropriately. + 2008-09-17 Tim-Philipp Müller <tim.muller at collabora co uk> * ext/neon/gstneonhttpsrc.c: (gst_neonhttp_src_start), diff --git a/gst/mpegdemux/gstmpegtsdemux.c b/gst/mpegdemux/gstmpegtsdemux.c index 0ba97f9d..a420d646 100644 --- a/gst/mpegdemux/gstmpegtsdemux.c +++ b/gst/mpegdemux/gstmpegtsdemux.c @@ -1951,6 +1951,76 @@ gst_fluts_demux_is_PMT (GstFluTSDemux * demux, guint16 PID) return FALSE; } +static FORCE_INLINE GstFlowReturn +gst_fluts_stream_pes_buffer_flush (GstFluTSStream * stream) +{ + GstFlowReturn ret; + + g_return_val_if_fail (stream->pes_buffer, GST_FLOW_OK); + + GST_BUFFER_SIZE (stream->pes_buffer) = stream->pes_buffer_used; + ret = gst_pes_filter_push (&stream->filter, stream->pes_buffer); + stream->pes_buffer = NULL; + return ret; +} + +static FORCE_INLINE GstFlowReturn +gst_fluts_stream_pes_buffer_push (GstFluTSStream * stream, + const guint8 * in_data, guint in_size) +{ + GstFlowReturn ret = GST_FLOW_OK; + guint8 *out_data; + + if (G_UNLIKELY (stream->pes_buffer + && stream->pes_buffer_used + in_size > stream->pes_buffer_size)) { + GST_DEBUG ("stream with PID 0x%04x have PES buffer full at %u bytes." + " Flushing and growing the buffer", + stream->PID, stream->pes_buffer_size); + stream->pes_buffer_overflow = TRUE; + if (stream->pes_buffer_size < (FLUTS_MAX_PES_BUFFER_SIZE >> 1)) + stream->pes_buffer_size <<= 1; + + ret = gst_fluts_stream_pes_buffer_flush (stream); + } + + if (G_UNLIKELY (!stream->pes_buffer)) { + /* set initial size of PES buffer */ + if (G_UNLIKELY (stream->pes_buffer_size == 0)) + stream->pes_buffer_size = FLUTS_MIN_PES_BUFFER_SIZE; + + stream->pes_buffer = gst_buffer_new_and_alloc (stream->pes_buffer_size); + stream->pes_buffer_used = 0; + } + out_data = GST_BUFFER_DATA (stream->pes_buffer) + stream->pes_buffer_used; +#ifdef USE_LIBOIL + oil_memcpy (out_data, in_data, in_size); +#else + memcpy (out_data, in_data, in_size); +#endif + stream->pes_buffer_used += in_size; + + return ret; +} + +static FORCE_INLINE GstFlowReturn +gst_fluts_demux_pes_buffer_flush (GstFluTSDemux * demux) +{ + gint i; + GstFlowReturn ret = GST_FLOW_OK; + + 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; + } + } + +done: + return ret; +} + /* * transport_packet(){ * sync_byte 8 bslbf == 0x47 @@ -2132,23 +2202,29 @@ gst_fluts_demux_parse_stream (GstFluTSDemux * demux, GstFluTSStream * stream, break; case PID_TYPE_ELEMENTARY: { - GstBuffer *es_buf; - if (payload_unit_start_indicator) { - GST_DEBUG_OBJECT (demux, "new PES start for PID 0x%04x", PID); + GST_DEBUG_OBJECT (demux, "new PES start for PID 0x%04x, used %u" + "bytes of %u bytes in the PES buffer", + PID, stream->pes_buffer_used, stream->pes_buffer_size); + /* Resize the buffer to half if no overflow detected and + * had been used less than half of it */ + if (stream->pes_buffer_overflow == FALSE + && stream->pes_buffer_used < (stream->pes_buffer_size >> 1)) { + stream->pes_buffer_size >>= 1; + GST_DEBUG_OBJECT (demux, "PES buffer size reduced to %u bytes", + stream->pes_buffer_size); + } + stream->pes_buffer_overflow = FALSE; + + /* Flush buffered PES data */ + gst_fluts_stream_pes_buffer_flush (stream); gst_pes_filter_drain (&stream->filter); } GST_LOG_OBJECT (demux, "Elementary packet of size %u for PID 0x%04x", datalen, PID); if (datalen > 0) { - es_buf = gst_buffer_new_and_alloc (datalen); -#ifdef USE_LIBOIL - oil_memcpy (GST_BUFFER_DATA (es_buf), data, datalen); -#else - memcpy (GST_BUFFER_DATA (es_buf), data, datalen); -#endif - ret = gst_pes_filter_push (&stream->filter, es_buf); + ret = gst_fluts_stream_pes_buffer_push (stream, data, datalen); break; } else { GST_WARNING_OBJECT (demux, "overflow of datalen: %u so skipping", @@ -2238,6 +2314,9 @@ gst_fluts_demux_sink_event (GstPad * pad, GstEvent * event) res = gst_fluts_demux_send_event (demux, event); break; case GST_EVENT_EOS: + /* Flush buffered PES data */ + gst_fluts_demux_pes_buffer_flush (demux); + /* Send the EOS event on each stream */ if (!(res = gst_fluts_demux_send_event (demux, event))) { /* we have no streams */ GST_ELEMENT_ERROR (demux, STREAM, TYPE_NOT_FOUND, diff --git a/gst/mpegdemux/gstmpegtsdemux.h b/gst/mpegdemux/gstmpegtsdemux.h index b35fb031..d26aedcd 100644 --- a/gst/mpegdemux/gstmpegtsdemux.h +++ b/gst/mpegdemux/gstmpegtsdemux.h @@ -57,6 +57,9 @@ G_BEGIN_DECLS #define HAVE_LATENCY #endif +#define FLUTS_MIN_PES_BUFFER_SIZE 4 * 1024 +#define FLUTS_MAX_PES_BUFFER_SIZE 256 * 1024 + #define FLUTS_MAX_PID 0x1fff #define FLUTS_NORMAL_TS_PACKETSIZE 188 #define FLUTS_M2TS_TS_PACKETSIZE 192 @@ -157,6 +160,10 @@ struct _GstFluTSStream { /* for PES streams */ guint8 id; guint8 stream_type; + GstBuffer * pes_buffer; + guint32 pes_buffer_size; + guint32 pes_buffer_used; + gboolean pes_buffer_overflow; GstPESFilter filter; GstPad * pad; GstFlowReturn last_ret; |