mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-26 09:08:14 +00:00
rtpjitterbuffer: Option to disable rtx-delay-reorder
When disabled we can save some iterations over timers. There is probably an argument for rtx-delay-reorder to exist, but for normal operations, handling jitter (reordering) is something a jitterbuffer should do, and this variable feels like functionality that is not "in-sync" with what the jitterbuffer is trying to achieve. Example: You have 50ms jitter on your network, and are receiving audio packets with 10ms durations. An audio packet should not be considered late until its rtx-timeout has expired (and hence a rtx-event is sent), but with rtx-delay-reorder, events will be sent pretty much all the time due to the jitter on the network. Point being: The jitterbuffer should adapt its size to the measured network jitter, and then rtx-delay-reorder needs to adapt as well, or simply get out of the way and let the other (better) rtx-mechanisms do their job. Also change find_timer to only use seqnum as an argument, since there will only ever be one timer per seqnum at any given time. In the one case where the type matters, the caller simply checks the type. https://bugzilla.gnome.org/show_bug.cgi?id=769768
This commit is contained in:
parent
0c7e3a860c
commit
1436fc01e9
1 changed files with 32 additions and 30 deletions
|
@ -614,13 +614,15 @@ gst_rtp_jitter_buffer_class_init (GstRtpJitterBufferClass * klass)
|
||||||
* this much packet reordering.
|
* this much packet reordering.
|
||||||
*
|
*
|
||||||
* When -1 is used, the value will be estimated based on observed packet
|
* When -1 is used, the value will be estimated based on observed packet
|
||||||
* reordering.
|
* reordering. When 0 is used packet reordering alone will not cause a
|
||||||
|
* retransmission event (Since 1.10).
|
||||||
*
|
*
|
||||||
* Since: 1.2
|
* Since: 1.2
|
||||||
*/
|
*/
|
||||||
g_object_class_install_property (gobject_class, PROP_RTX_DELAY_REORDER,
|
g_object_class_install_property (gobject_class, PROP_RTX_DELAY_REORDER,
|
||||||
g_param_spec_int ("rtx-delay-reorder", "RTX Delay Reorder",
|
g_param_spec_int ("rtx-delay-reorder", "RTX Delay Reorder",
|
||||||
"Sending retransmission event when this much reordering (-1 automatic)",
|
"Sending retransmission event when this much reordering "
|
||||||
|
"(0 disable, -1 automatic)",
|
||||||
-1, G_MAXINT, DEFAULT_RTX_DELAY_REORDER,
|
-1, G_MAXINT, DEFAULT_RTX_DELAY_REORDER,
|
||||||
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
||||||
/**
|
/**
|
||||||
|
@ -1896,7 +1898,7 @@ apply_offset (GstRtpJitterBuffer * jitterbuffer, GstClockTime timestamp)
|
||||||
}
|
}
|
||||||
|
|
||||||
static TimerData *
|
static TimerData *
|
||||||
find_timer (GstRtpJitterBuffer * jitterbuffer, TimerType type, guint16 seqnum)
|
find_timer (GstRtpJitterBuffer * jitterbuffer, guint16 seqnum)
|
||||||
{
|
{
|
||||||
GstRtpJitterBufferPrivate *priv = jitterbuffer->priv;
|
GstRtpJitterBufferPrivate *priv = jitterbuffer->priv;
|
||||||
TimerData *timer = NULL;
|
TimerData *timer = NULL;
|
||||||
|
@ -1905,7 +1907,7 @@ find_timer (GstRtpJitterBuffer * jitterbuffer, TimerType type, guint16 seqnum)
|
||||||
len = priv->timers->len;
|
len = priv->timers->len;
|
||||||
for (i = 0; i < len; i++) {
|
for (i = 0; i < len; i++) {
|
||||||
TimerData *test = &g_array_index (priv->timers, TimerData, i);
|
TimerData *test = &g_array_index (priv->timers, TimerData, i);
|
||||||
if (test->seqnum == seqnum && test->type == type) {
|
if (test->seqnum == seqnum) {
|
||||||
timer = test;
|
timer = test;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -2042,7 +2044,7 @@ set_timer (GstRtpJitterBuffer * jitterbuffer, TimerType type,
|
||||||
TimerData *timer;
|
TimerData *timer;
|
||||||
|
|
||||||
/* find the seqnum timer */
|
/* find the seqnum timer */
|
||||||
timer = find_timer (jitterbuffer, type, seqnum);
|
timer = find_timer (jitterbuffer, seqnum);
|
||||||
if (timer == NULL) {
|
if (timer == NULL) {
|
||||||
timer = add_timer (jitterbuffer, type, seqnum, 0, timeout, 0, -1);
|
timer = add_timer (jitterbuffer, type, seqnum, 0, timeout, 0, -1);
|
||||||
} else {
|
} else {
|
||||||
|
@ -2138,10 +2140,10 @@ update_timers (GstRtpJitterBuffer * jitterbuffer, guint16 seqnum,
|
||||||
{
|
{
|
||||||
GstRtpJitterBufferPrivate *priv = jitterbuffer->priv;
|
GstRtpJitterBufferPrivate *priv = jitterbuffer->priv;
|
||||||
TimerData *timer = NULL;
|
TimerData *timer = NULL;
|
||||||
gint i, len;
|
|
||||||
|
|
||||||
/* go through all timers and unschedule the ones with a large gap, also find
|
/* go through all timers and unschedule the ones with a large gap */
|
||||||
* the timer for the seqnum */
|
if (priv->do_retransmission && priv->rtx_delay_reorder > 0) {
|
||||||
|
gint i, len;
|
||||||
len = priv->timers->len;
|
len = priv->timers->len;
|
||||||
for (i = 0; i < len; i++) {
|
for (i = 0; i < len; i++) {
|
||||||
TimerData *test = &g_array_index (priv->timers, TimerData, i);
|
TimerData *test = &g_array_index (priv->timers, TimerData, i);
|
||||||
|
@ -2149,17 +2151,13 @@ update_timers (GstRtpJitterBuffer * jitterbuffer, guint16 seqnum,
|
||||||
|
|
||||||
gap = gst_rtp_buffer_compare_seqnum (test->seqnum, seqnum);
|
gap = gst_rtp_buffer_compare_seqnum (test->seqnum, seqnum);
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (jitterbuffer, "%d, %d, #%d<->#%d gap %d", i,
|
GST_DEBUG_OBJECT (jitterbuffer, "%d, #%d<->#%d gap %d",
|
||||||
test->type, test->seqnum, seqnum, gap);
|
test->type, test->seqnum, seqnum, gap);
|
||||||
|
|
||||||
if (gap == 0) {
|
if (gap == 0) {
|
||||||
GST_DEBUG ("found timer for current seqnum");
|
GST_DEBUG ("found timer for current seqnum");
|
||||||
/* the timer for the current seqnum */
|
/* the timer for the current seqnum */
|
||||||
timer = test;
|
timer = test;
|
||||||
/* when no retransmission, we can stop now, we only need to find the
|
|
||||||
* timer for the current seqnum */
|
|
||||||
if (!priv->do_retransmission)
|
|
||||||
break;
|
|
||||||
} else if (gap > priv->rtx_delay_reorder) {
|
} else if (gap > priv->rtx_delay_reorder) {
|
||||||
/* max gap, we exceeded the max reorder distance and we don't expect the
|
/* max gap, we exceeded the max reorder distance and we don't expect the
|
||||||
* missing packet to be this reordered */
|
* missing packet to be this reordered */
|
||||||
|
@ -2167,6 +2165,10 @@ update_timers (GstRtpJitterBuffer * jitterbuffer, guint16 seqnum,
|
||||||
reschedule_timer (jitterbuffer, test, test->seqnum, -1, 0, FALSE);
|
reschedule_timer (jitterbuffer, test, test->seqnum, -1, 0, FALSE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
/* find the timer for the seqnum */
|
||||||
|
timer = find_timer (jitterbuffer, seqnum);
|
||||||
|
}
|
||||||
|
|
||||||
do_next_seqnum = do_next_seqnum && priv->packet_spacing > 0
|
do_next_seqnum = do_next_seqnum && priv->packet_spacing > 0
|
||||||
&& priv->do_retransmission && priv->rtx_next_seqnum;
|
&& priv->do_retransmission && priv->rtx_next_seqnum;
|
||||||
|
@ -2345,11 +2347,11 @@ calculate_expected (GstRtpJitterBuffer * jitterbuffer, guint32 expected,
|
||||||
expected_dts = priv->last_in_dts + duration;
|
expected_dts = priv->last_in_dts + duration;
|
||||||
|
|
||||||
if (priv->do_retransmission) {
|
if (priv->do_retransmission) {
|
||||||
TimerData *timer;
|
TimerData *timer = find_timer (jitterbuffer, expected);
|
||||||
|
|
||||||
type = TIMER_TYPE_EXPECTED;
|
type = TIMER_TYPE_EXPECTED;
|
||||||
/* if we had a timer for the first missing packet, update it. */
|
/* if we had a timer for the first missing packet, update it. */
|
||||||
if ((timer = find_timer (jitterbuffer, type, expected))) {
|
if (timer && timer->type == TIMER_TYPE_EXPECTED) {
|
||||||
GstClockTime timeout = timer->timeout;
|
GstClockTime timeout = timer->timeout;
|
||||||
|
|
||||||
timer->duration = duration;
|
timer->duration = duration;
|
||||||
|
|
Loading…
Reference in a new issue