gst/rtpmanager/gstrtpjitterbuffer.c: Fix problem with using the output seqnum counter to check for input seqnum disco...

Original commit message from CVS:
* gst/rtpmanager/gstrtpjitterbuffer.c:
(gst_jitter_buffer_sink_parse_caps),
(gst_rtp_jitter_buffer_flush_start),
(gst_rtp_jitter_buffer_flush_stop), (gst_rtp_jitter_buffer_chain),
(gst_rtp_jitter_buffer_loop):
Fix problem with using the output seqnum counter to check for input
seqnum discontinuities.
Improve gap detection and recovery, reset and flush the jitterbuffer on
seqnum restart. Fixes #556520.
* gst/rtpmanager/rtpjitterbuffer.c: (rtp_jitter_buffer_insert):
Fix wrong G_LIKELY.
This commit is contained in:
Wim Taymans 2008-10-16 13:05:37 +00:00
parent 4d87da5b96
commit 9f83f6868c
3 changed files with 61 additions and 28 deletions

View file

@ -1,3 +1,18 @@
2008-10-16 Wim Taymans <wim.taymans@collabora.co.uk>
* gst/rtpmanager/gstrtpjitterbuffer.c:
(gst_jitter_buffer_sink_parse_caps),
(gst_rtp_jitter_buffer_flush_start),
(gst_rtp_jitter_buffer_flush_stop), (gst_rtp_jitter_buffer_chain),
(gst_rtp_jitter_buffer_loop):
Fix problem with using the output seqnum counter to check for input
seqnum discontinuities.
Improve gap detection and recovery, reset and flush the jitterbuffer on
seqnum restart. Fixes #556520.
* gst/rtpmanager/rtpjitterbuffer.c: (rtp_jitter_buffer_insert):
Fix wrong G_LIKELY.
2008-10-16 Jan Schmidt <jan.schmidt@sun.com> 2008-10-16 Jan Schmidt <jan.schmidt@sun.com>
* configure.ac: * configure.ac:

View file

@ -142,10 +142,12 @@ struct _GstRtpJitterBufferPrivate
/* the last seqnum we pushed out */ /* the last seqnum we pushed out */
guint32 last_popped_seqnum; guint32 last_popped_seqnum;
/* the next expected seqnum */ /* the next expected seqnum we push */
guint32 next_seqnum; guint32 next_seqnum;
/* last output time */ /* last output time */
GstClockTime last_out_time; GstClockTime last_out_time;
/* the next expected seqnum we receive */
guint32 next_in_seqnum;
/* state */ /* state */
gboolean eos; gboolean eos;
@ -163,6 +165,7 @@ struct _GstRtpJitterBufferPrivate
/* for sync */ /* for sync */
GstSegment segment; GstSegment segment;
GstClockID clock_id; GstClockID clock_id;
gboolean unscheduled;
/* the latency of the upstream peer, we have to take this into account when /* the latency of the upstream peer, we have to take this into account when
* synchronizing the buffers. */ * synchronizing the buffers. */
GstClockTime peer_latency; GstClockTime peer_latency;
@ -483,14 +486,14 @@ gst_jitter_buffer_sink_parse_caps (GstRtpJitterBuffer * jitterbuffer,
priv->clock_base); priv->clock_base);
/* first expected seqnum, only update when we didn't have a previous base. */ /* first expected seqnum, only update when we didn't have a previous base. */
if (priv->next_seqnum == -1) { if (priv->next_in_seqnum == -1) {
if (gst_structure_get_uint (caps_struct, "seqnum-base", &val)) if (gst_structure_get_uint (caps_struct, "seqnum-base", &val))
priv->next_seqnum = val; priv->next_in_seqnum = val;
else else
priv->next_seqnum = -1; priv->next_in_seqnum = -1;
} }
GST_DEBUG_OBJECT (jitterbuffer, "got seqnum-base %d", priv->next_seqnum); GST_DEBUG_OBJECT (jitterbuffer, "got seqnum-base %d", priv->next_in_seqnum);
return TRUE; return TRUE;
@ -543,8 +546,10 @@ gst_rtp_jitter_buffer_flush_start (GstRtpJitterBuffer * jitterbuffer)
JBUF_SIGNAL (priv); JBUF_SIGNAL (priv);
/* unlock clock, we just unschedule, the entry will be released by the /* unlock clock, we just unschedule, the entry will be released by the
* locking streaming thread. */ * locking streaming thread. */
if (priv->clock_id) if (priv->clock_id) {
gst_clock_id_unschedule (priv->clock_id); gst_clock_id_unschedule (priv->clock_id);
priv->unscheduled = TRUE;
}
JBUF_UNLOCK (priv); JBUF_UNLOCK (priv);
} }
@ -563,6 +568,7 @@ gst_rtp_jitter_buffer_flush_stop (GstRtpJitterBuffer * jitterbuffer)
priv->last_popped_seqnum = -1; priv->last_popped_seqnum = -1;
priv->last_out_time = -1; priv->last_out_time = -1;
priv->next_seqnum = -1; priv->next_seqnum = -1;
priv->next_in_seqnum = -1;
priv->clock_rate = -1; priv->clock_rate = -1;
priv->eos = FALSE; priv->eos = FALSE;
rtp_jitter_buffer_flush (priv->jbuf); rtp_jitter_buffer_flush (priv->jbuf);
@ -881,47 +887,56 @@ gst_rtp_jitter_buffer_chain (GstPad * pad, GstBuffer * buffer)
if (G_UNLIKELY (priv->eos)) if (G_UNLIKELY (priv->eos))
goto have_eos; goto have_eos;
/* let's check if this buffer is too late, we can only accept packets with /* now check against our expected seqnum */
* bigger seqnum than the one we last pushed. */ if (G_LIKELY (priv->next_in_seqnum != -1)) {
if (G_LIKELY (priv->last_popped_seqnum != -1)) {
gint gap; gint gap;
gboolean reset = FALSE; gboolean reset = FALSE;
gap = gst_rtp_buffer_compare_seqnum (priv->last_popped_seqnum, seqnum); gap = gst_rtp_buffer_compare_seqnum (priv->next_in_seqnum, seqnum);
if (G_UNLIKELY (gap != 0)) {
if (G_UNLIKELY (gap <= 0)) { GST_DEBUG_OBJECT (jitterbuffer, "expected #%d, got #%d, gap of %d",
/* priv->last_popped_seqnum >= seqnum, this packet is too late or the priv->next_in_seqnum, seqnum, gap);
/* priv->next_in_seqnum >= seqnum, this packet is too late or the
* sender might have been restarted with different seqnum. */ * sender might have been restarted with different seqnum. */
if (gap < -RTP_MAX_MISORDER) { if (gap < -RTP_MAX_MISORDER) {
GST_DEBUG_OBJECT (jitterbuffer, "reset: buffer too old %d", gap); GST_DEBUG_OBJECT (jitterbuffer, "reset: buffer too old %d", gap);
reset = TRUE; reset = TRUE;
} else {
goto too_late;
} }
} else { /* priv->next_in_seqnum < seqnum, this is a new packet */
/* priv->last_popped_seqnum < seqnum, this is a new packet */ else if (G_UNLIKELY (gap > RTP_MAX_DROPOUT)) {
if (G_UNLIKELY (gap > RTP_MAX_DROPOUT)) {
GST_DEBUG_OBJECT (jitterbuffer, "reset: too many dropped packets %d", GST_DEBUG_OBJECT (jitterbuffer, "reset: too many dropped packets %d",
gap); gap);
reset = TRUE; reset = TRUE;
} else { } else {
GST_DEBUG_OBJECT (jitterbuffer, "dropped packets %d but <= %d", gap, GST_DEBUG_OBJECT (jitterbuffer, "tolerable gap");
RTP_MAX_DROPOUT);
} }
} }
if (G_UNLIKELY (reset)) { if (G_UNLIKELY (reset)) {
priv->last_popped_seqnum = -1; rtp_jitter_buffer_flush (priv->jbuf);
priv->next_seqnum = -1;
rtp_jitter_buffer_reset_skew (priv->jbuf); rtp_jitter_buffer_reset_skew (priv->jbuf);
priv->last_popped_seqnum = -1;
priv->next_seqnum = seqnum;
} }
} }
priv->next_in_seqnum = (seqnum + 1) & 0xffff;
/* let's check if this buffer is too late, we can only accept packets with
* bigger seqnum than the one we last pushed. */
if (G_LIKELY (priv->last_popped_seqnum != -1)) {
gint gap;
gap = gst_rtp_buffer_compare_seqnum (priv->last_popped_seqnum, seqnum);
/* priv->last_popped_seqnum >= seqnum, we're too late. */
if (G_UNLIKELY (gap <= 0))
goto too_late;
}
/* let's drop oldest packet if the queue is already full and drop-on-latency /* let's drop oldest packet if the queue is already full and drop-on-latency
* is set. We can only do this when there actually is a latency. When no * is set. We can only do this when there actually is a latency. When no
* latency is set, we just pump it in the queue and let the other end push it * latency is set, we just pump it in the queue and let the other end push it
* out as fast as possible. */ * out as fast as possible. */
if (priv->latency_ms && priv->drop_on_latency) { if (priv->latency_ms && priv->drop_on_latency) {
latency_ts = latency_ts =
gst_util_uint64_scale_int (priv->latency_ms, priv->clock_rate, 1000); gst_util_uint64_scale_int (priv->latency_ms, priv->clock_rate, 1000);
@ -958,10 +973,11 @@ gst_rtp_jitter_buffer_chain (GstPad * pad, GstBuffer * buffer)
GST_DEBUG_OBJECT (jitterbuffer, GST_DEBUG_OBJECT (jitterbuffer,
"Unscheduling waiting buffer, new tail buffer"); "Unscheduling waiting buffer, new tail buffer");
gst_clock_id_unschedule (priv->clock_id); gst_clock_id_unschedule (priv->clock_id);
priv->unscheduled = TRUE;
} }
GST_DEBUG_OBJECT (jitterbuffer, "Pushed packet #%d, now %d packets", GST_DEBUG_OBJECT (jitterbuffer, "Pushed packet #%d, now %d packets, tail: %d",
seqnum, rtp_jitter_buffer_num_packets (priv->jbuf)); seqnum, rtp_jitter_buffer_num_packets (priv->jbuf), tail);
finished: finished:
JBUF_UNLOCK (priv); JBUF_UNLOCK (priv);
@ -1183,6 +1199,7 @@ again:
/* create an entry for the clock */ /* create an entry for the clock */
id = priv->clock_id = gst_clock_new_single_shot_id (clock, sync_time); id = priv->clock_id = gst_clock_new_single_shot_id (clock, sync_time);
priv->unscheduled = FALSE;
GST_OBJECT_UNLOCK (jitterbuffer); GST_OBJECT_UNLOCK (jitterbuffer);
/* release the lock so that the other end can push stuff or unlock */ /* release the lock so that the other end can push stuff or unlock */
@ -1203,8 +1220,9 @@ again:
goto flushing; goto flushing;
/* if we got unscheduled and we are not flushing, it's because a new tail /* if we got unscheduled and we are not flushing, it's because a new tail
* element became available in the queue. Grab it and try to push or sync. */ * element became available in the queue or we flushed the queue.
if (ret == GST_CLOCK_UNSCHEDULED) { * Grab it and try to push or sync. */
if (ret == GST_CLOCK_UNSCHEDULED || priv->unscheduled) {
GST_DEBUG_OBJECT (jitterbuffer, GST_DEBUG_OBJECT (jitterbuffer,
"Wait got unscheduled, will retry to push with new buffer"); "Wait got unscheduled, will retry to push with new buffer");
goto again; goto again;

View file

@ -385,7 +385,7 @@ rtp_jitter_buffer_insert (RTPJitterBuffer * jbuf, GstBuffer * buf,
/* tail was changed when we did not find a previous packet, we set the return /* tail was changed when we did not find a previous packet, we set the return
* flag when requested. */ * flag when requested. */
if (G_UNLIKELY (tail)) if (G_LIKELY (tail))
*tail = (list == NULL); *tail = (list == NULL);
return TRUE; return TRUE;