Skip to content

Commit 1c2709c

Browse files
nealcardwellkuba-moo
authored andcommitted
tcp: fix excessive TLP and RACK timeouts from HZ rounding
We discovered from packet traces of slow loss recovery on kernels with the default HZ=250 setting (and min_rtt < 1ms) that after reordering, when receiving a SACKed sequence range, the RACK reordering timer was firing after about 16ms rather than the desired value of roughly min_rtt/4 + 2ms. The problem is largely due to the RACK reorder timer calculation adding in TCP_TIMEOUT_MIN, which is 2 jiffies. On kernels with HZ=250, this is 2*4ms = 8ms. The TLP timer calculation has the exact same issue. This commit fixes the TLP transmit timer and RACK reordering timer floor calculation to more closely match the intended 2ms floor even on kernels with HZ=250. It does this by adding in a new TCP_TIMEOUT_MIN_US floor of 2000 us and then converting to jiffies, instead of the current approach of converting to jiffies and then adding th TCP_TIMEOUT_MIN value of 2 jiffies. Our testing has verified that on kernels with HZ=1000, as expected, this does not produce significant changes in behavior, but on kernels with the default HZ=250 the latency improvement can be large. For example, our tests show that for HZ=250 kernels at low RTTs this fix roughly halves the latency for the RACK reorder timer: instead of mostly firing at 16ms it mostly fires at 8ms. Suggested-by: Eric Dumazet <[email protected]> Signed-off-by: Neal Cardwell <[email protected]> Signed-off-by: Yuchung Cheng <[email protected]> Fixes: bb4d991 ("tcp: adjust tail loss probe timeout") Reviewed-by: Eric Dumazet <[email protected]> Link: https://lore.kernel.org/r/[email protected] Signed-off-by: Jakub Kicinski <[email protected]>
1 parent 95535e3 commit 1c2709c

File tree

3 files changed

+9
-5
lines changed

3 files changed

+9
-5
lines changed

include/net/tcp.h

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -141,6 +141,9 @@ void tcp_time_wait(struct sock *sk, int state, int timeo);
141141
#define TCP_RTO_MAX ((unsigned)(120*HZ))
142142
#define TCP_RTO_MIN ((unsigned)(HZ/5))
143143
#define TCP_TIMEOUT_MIN (2U) /* Min timeout for TCP timers in jiffies */
144+
145+
#define TCP_TIMEOUT_MIN_US (2*USEC_PER_MSEC) /* Min TCP timeout in microsecs */
146+
144147
#define TCP_TIMEOUT_INIT ((unsigned)(1*HZ)) /* RFC6298 2.1 initial RTO value */
145148
#define TCP_TIMEOUT_FALLBACK ((unsigned)(3*HZ)) /* RFC 1122 initial RTO value, now
146149
* used as a fallback RTO for the

net/ipv4/tcp_output.c

Lines changed: 5 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -2788,7 +2788,7 @@ bool tcp_schedule_loss_probe(struct sock *sk, bool advancing_rto)
27882788
{
27892789
struct inet_connection_sock *icsk = inet_csk(sk);
27902790
struct tcp_sock *tp = tcp_sk(sk);
2791-
u32 timeout, rto_delta_us;
2791+
u32 timeout, timeout_us, rto_delta_us;
27922792
int early_retrans;
27932793

27942794
/* Don't do any loss probe on a Fast Open connection before 3WHS
@@ -2812,11 +2812,12 @@ bool tcp_schedule_loss_probe(struct sock *sk, bool advancing_rto)
28122812
* sample is available then probe after TCP_TIMEOUT_INIT.
28132813
*/
28142814
if (tp->srtt_us) {
2815-
timeout = usecs_to_jiffies(tp->srtt_us >> 2);
2815+
timeout_us = tp->srtt_us >> 2;
28162816
if (tp->packets_out == 1)
2817-
timeout += TCP_RTO_MIN;
2817+
timeout_us += tcp_rto_min_us(sk);
28182818
else
2819-
timeout += TCP_TIMEOUT_MIN;
2819+
timeout_us += TCP_TIMEOUT_MIN_US;
2820+
timeout = usecs_to_jiffies(timeout_us);
28202821
} else {
28212822
timeout = TCP_TIMEOUT_INIT;
28222823
}

net/ipv4/tcp_recovery.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@ bool tcp_rack_mark_lost(struct sock *sk)
104104
tp->rack.advanced = 0;
105105
tcp_rack_detect_loss(sk, &timeout);
106106
if (timeout) {
107-
timeout = usecs_to_jiffies(timeout) + TCP_TIMEOUT_MIN;
107+
timeout = usecs_to_jiffies(timeout + TCP_TIMEOUT_MIN_US);
108108
inet_csk_reset_xmit_timer(sk, ICSK_TIME_REO_TIMEOUT,
109109
timeout, inet_csk(sk)->icsk_rto);
110110
}

0 commit comments

Comments
 (0)