mpegpsmux: use DTS in addition to PTS

And refactor choose_best_stream() a little.

videotestsrc pattern=ball ! x264enc ! mpegpsmux ! ...

plays much nicer now.
This commit is contained in:
Tim-Philipp Müller 2013-01-01 11:56:16 +00:00
parent 9e1faac4df
commit d1e9a96a69
2 changed files with 97 additions and 65 deletions

View file

@ -373,6 +373,66 @@ no_stream:
return ret; return ret;
} }
static GstBuffer *
mpegpsmux_queue_buffer_for_stream (MpegPsMux * mux, MpegPsPadData * ps_data)
{
GstCollectData *c_data = (GstCollectData *) ps_data;
GstBuffer *buf;
g_assert (ps_data->queued.buf == NULL);
buf = gst_collect_pads_peek (mux->collect, c_data);
if (buf == NULL)
return NULL;
ps_data->queued.buf = buf;
/* do any raw -> byte-stream format conversions (e.g. for H.264, AAC) */
if (ps_data->prepare_func) {
buf = ps_data->prepare_func (buf, ps_data, mux);
if (buf) { /* Take the prepared buffer instead */
gst_buffer_unref (ps_data->queued.buf);
ps_data->queued.buf = buf;
} else { /* If data preparation returned NULL, use unprepared one */
buf = ps_data->queued.buf;
}
}
ps_data->queued.pts = GST_BUFFER_PTS (buf);
if (GST_CLOCK_TIME_IS_VALID (ps_data->queued.pts)) {
ps_data->queued.pts = gst_segment_to_running_time (&c_data->segment,
GST_FORMAT_TIME, ps_data->queued.pts);
}
ps_data->queued.dts = GST_BUFFER_DTS (buf);
if (GST_CLOCK_TIME_IS_VALID (ps_data->queued.dts)) {
ps_data->queued.dts = gst_segment_to_running_time (&c_data->segment,
GST_FORMAT_TIME, ps_data->queued.dts);
}
if (GST_BUFFER_PTS_IS_VALID (buf) && GST_BUFFER_DTS_IS_VALID (buf)) {
ps_data->queued.ts = MIN (ps_data->queued.dts, ps_data->queued.pts);
} else if (GST_BUFFER_PTS_IS_VALID (buf) && !GST_BUFFER_DTS_IS_VALID (buf)) {
ps_data->queued.ts = ps_data->queued.pts;
} else if (GST_BUFFER_DTS_IS_VALID (buf) && !GST_BUFFER_PTS_IS_VALID (buf)) {
GST_WARNING_OBJECT (c_data->pad, "got DTS without PTS");
ps_data->queued.ts = ps_data->queued.dts;
} else {
ps_data->queued.ts = GST_CLOCK_TIME_NONE;
}
GST_DEBUG_OBJECT (mux, "Queued buffer with ts %" GST_TIME_FORMAT ": "
"uncorrected pts %" GST_TIME_FORMAT " dts %" GST_TIME_FORMAT ", "
"buffer pts %" GST_TIME_FORMAT " dts %" GST_TIME_FORMAT " for PID 0x%04x",
GST_TIME_ARGS (ps_data->queued.ts),
GST_TIME_ARGS (GST_BUFFER_PTS (buf)),
GST_TIME_ARGS (GST_BUFFER_DTS (buf)),
GST_TIME_ARGS (ps_data->queued.pts),
GST_TIME_ARGS (ps_data->queued.dts), ps_data->stream_id);
return buf;
}
static MpegPsPadData * static MpegPsPadData *
mpegpsmux_choose_best_stream (MpegPsMux * mux) mpegpsmux_choose_best_stream (MpegPsMux * mux)
{ {
@ -387,42 +447,14 @@ mpegpsmux_choose_best_stream (MpegPsMux * mux)
MpegPsPadData *ps_data = (MpegPsPadData *) walk->data; MpegPsPadData *ps_data = (MpegPsPadData *) walk->data;
if (ps_data->eos == FALSE) { if (ps_data->eos == FALSE) {
if (ps_data->queued_buf == NULL) { if (ps_data->queued.buf == NULL) {
GstBuffer *buf; GstBuffer *buf;
ps_data->queued_buf = buf = buf = mpegpsmux_queue_buffer_for_stream (mux, ps_data);
gst_collect_pads_peek (mux->collect, c_data); if (buf == NULL) {
ps_data->eos = TRUE;
if (buf != NULL) { continue;
if (ps_data->prepare_func) {
buf = ps_data->prepare_func (buf, ps_data, mux);
if (buf) { /* Take the prepared buffer instead */
gst_buffer_unref (ps_data->queued_buf);
ps_data->queued_buf = buf;
} else { /* If data preparation returned NULL, use unprepared one */
buf = ps_data->queued_buf;
} }
}
if (GST_BUFFER_TIMESTAMP (buf) != GST_CLOCK_TIME_NONE) {
/* Ignore timestamps that go backward for now. FIXME: Handle all
* incoming PTS */
if (ps_data->last_ts == GST_CLOCK_TIME_NONE ||
ps_data->last_ts < GST_BUFFER_TIMESTAMP (buf)) {
ps_data->cur_ts = ps_data->last_ts =
gst_segment_to_running_time (&c_data->segment,
GST_FORMAT_TIME, GST_BUFFER_TIMESTAMP (buf));
} else {
GST_DEBUG_OBJECT (mux, "Ignoring PTS that has gone backward");
}
} else
ps_data->cur_ts = GST_CLOCK_TIME_NONE;
GST_DEBUG_OBJECT (mux, "Pulled buffer with ts %" GST_TIME_FORMAT
" (uncorrected ts %" GST_TIME_FORMAT " %" G_GUINT64_FORMAT
") for PID 0x%04x",
GST_TIME_ARGS (ps_data->cur_ts),
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf)),
GST_BUFFER_TIMESTAMP (buf), ps_data->stream_id);
/* Choose a stream we've never seen a timestamp for to ensure /* Choose a stream we've never seen a timestamp for to ensure
* we push enough buffers from it to reach a timestamp */ * we push enough buffers from it to reach a timestamp */
@ -430,10 +462,6 @@ mpegpsmux_choose_best_stream (MpegPsMux * mux)
best = ps_data; best = ps_data;
c_best = c_data; c_best = c_data;
} }
} else {
ps_data->eos = TRUE;
continue;
}
} }
/* If we don't yet have a best pad, take this one, otherwise take /* If we don't yet have a best pad, take this one, otherwise take
@ -505,22 +533,20 @@ mpegpsmux_collected (GstCollectPads * pads, MpegPsMux * mux)
} }
if (best != NULL) { if (best != NULL) {
/* @*buf : the buffer to be processed */ GstBuffer *buf = best->queued.buf;
GstBuffer *buf = best->queued_buf; gint64 pts, dts;
gint64 pts = -1;
g_return_val_if_fail (buf != NULL, GST_FLOW_ERROR); g_assert (buf != NULL);
GST_DEBUG_OBJECT (mux, GST_LOG_OBJECT (mux,
"Chose stream from pad %" GST_PTR_FORMAT " for output (PID: 0x%04x)", "Chose stream from pad %" GST_PTR_FORMAT " for output (PID: 0x%04x): "
best->collect.pad, best->stream_id); "adjusted pts: %" GST_TIME_FORMAT ", dts: %" GST_TIME_FORMAT,
best->collect.pad, best->stream_id,
GST_TIME_ARGS (best->queued.pts), GST_TIME_ARGS (best->queued.dts));
/* set timestamp */ /* and convert to mpeg time stamps */
if (GST_CLOCK_TIME_IS_VALID (best->cur_ts)) { pts = GSTTIME_TO_MPEGTIME (best->queued.pts);
pts = GSTTIME_TO_MPEGTIME (best->cur_ts); /* @pts: current timestamp */ dts = GSTTIME_TO_MPEGTIME (best->queued.dts);
GST_DEBUG_OBJECT (mux, "Buffer has TS %" GST_TIME_FORMAT " pts %"
G_GINT64_FORMAT, GST_TIME_ARGS (best->cur_ts), pts);
}
/* start of new GOP? */ /* start of new GOP? */
keyunit = !GST_BUFFER_FLAG_IS_SET (buf, GST_BUFFER_FLAG_DELTA_UNIT); keyunit = !GST_BUFFER_FLAG_IS_SET (buf, GST_BUFFER_FLAG_DELTA_UNIT);
@ -532,11 +558,10 @@ mpegpsmux_collected (GstCollectPads * pads, MpegPsMux * mux)
goto done; goto done;
} }
/* FIXME: porting: add DTS */
/* give the buffer to libpsmux for processing */ /* give the buffer to libpsmux for processing */
psmux_stream_add_data (best->stream, buf, pts, -1, keyunit); psmux_stream_add_data (best->stream, buf, pts, dts, keyunit);
best->queued_buf = NULL; best->queued.buf = NULL;
/* write the data from libpsmux to stream */ /* write the data from libpsmux to stream */
while (psmux_stream_bytes_in_buffer (best->stream) > 0) { while (psmux_stream_bytes_in_buffer (best->stream) > 0) {

View file

@ -94,9 +94,16 @@ struct MpegPsPadData {
guint8 stream_id_ext; guint8 stream_id_ext;
PsMuxStream *stream; PsMuxStream *stream;
GstBuffer *queued_buf; /* Currently pulled buffer */ /* Currently pulled buffer */
GstClockTime cur_ts; /* Adjusted TS for the pulled buffer */ struct {
GstClockTime last_ts; /* Most recent valid TS for this stream */ GstBuffer *buf;
GstClockTime ts; /* adjusted TS = MIN (DTS, PTS) for the pulled buffer */
GstClockTime pts; /* adjusted PTS (running time) */
GstClockTime dts; /* adjusted DTS (running time) */
} queued;
/* Most recent valid TS (DTS or PTS) for this stream */
GstClockTime last_ts;
GstBuffer * codec_data; /* Optional codec data available in the caps */ GstBuffer * codec_data; /* Optional codec data available in the caps */
@ -110,10 +117,10 @@ GType mpegpsmux_get_type (void);
#define CLOCK_BASE 9LL #define CLOCK_BASE 9LL
#define CLOCK_FREQ (CLOCK_BASE * 10000) #define CLOCK_FREQ (CLOCK_BASE * 10000)
#define MPEGTIME_TO_GSTTIME(time) (gst_util_uint64_scale ((time), \ #define GSTTIME_TO_MPEGTIME(time) \
GST_MSECOND/10, CLOCK_BASE)) (GST_CLOCK_TIME_IS_VALID(time) ? \
#define GSTTIME_TO_MPEGTIME(time) (gst_util_uint64_scale ((time), \ gst_util_uint64_scale ((time), CLOCK_BASE, GST_MSECOND/10) : \
CLOCK_BASE, GST_MSECOND/10)) -1)
#define NORMAL_TS_PACKET_LENGTH 188 #define NORMAL_TS_PACKET_LENGTH 188
#define M2TS_PACKET_LENGTH 192 #define M2TS_PACKET_LENGTH 192