aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--net/ipv4/tcp_input.c25
1 files changed, 13 insertions, 12 deletions
diff --git a/net/ipv4/tcp_input.c b/net/ipv4/tcp_input.c
index 366f63ae4d5c..7bac1fac065f 100644
--- a/net/ipv4/tcp_input.c
+++ b/net/ipv4/tcp_input.c
@@ -105,6 +105,7 @@ int sysctl_tcp_abc __read_mostly;
105#define FLAG_SND_UNA_ADVANCED 0x400 /* Snd_una was changed (!= FLAG_DATA_ACKED) */ 105#define FLAG_SND_UNA_ADVANCED 0x400 /* Snd_una was changed (!= FLAG_DATA_ACKED) */
106#define FLAG_DSACKING_ACK 0x800 /* SACK blocks contained D-SACK info */ 106#define FLAG_DSACKING_ACK 0x800 /* SACK blocks contained D-SACK info */
107#define FLAG_NONHEAD_RETRANS_ACKED 0x1000 /* Non-head rexmitted data was ACKed */ 107#define FLAG_NONHEAD_RETRANS_ACKED 0x1000 /* Non-head rexmitted data was ACKed */
108#define FLAG_SACK_RENEGING 0x2000 /* snd_una advanced to a sacked seq */
108 109
109#define FLAG_ACKED (FLAG_DATA_ACKED|FLAG_SYN_ACKED) 110#define FLAG_ACKED (FLAG_DATA_ACKED|FLAG_SYN_ACKED)
110#define FLAG_NOT_DUP (FLAG_DATA|FLAG_WIN_UPDATE|FLAG_ACKED) 111#define FLAG_NOT_DUP (FLAG_DATA|FLAG_WIN_UPDATE|FLAG_ACKED)
@@ -1918,18 +1919,15 @@ void tcp_enter_loss(struct sock *sk, int how)
1918 tp->frto_counter = 0; 1919 tp->frto_counter = 0;
1919} 1920}
1920 1921
1921static int tcp_check_sack_reneging(struct sock *sk) 1922/* If ACK arrived pointing to a remembered SACK, it means that our
1923 * remembered SACKs do not reflect real state of receiver i.e.
1924 * receiver _host_ is heavily congested (or buggy).
1925 *
1926 * Do processing similar to RTO timeout.
1927 */
1928static int tcp_check_sack_reneging(struct sock *sk, int flag)
1922{ 1929{
1923 struct sk_buff *skb; 1930 if (flag & FLAG_SACK_RENEGING) {
1924
1925 /* If ACK arrived pointing to a remembered SACK,
1926 * it means that our remembered SACKs do not reflect
1927 * real state of receiver i.e.
1928 * receiver _host_ is heavily congested (or buggy).
1929 * Do processing similar to RTO timeout.
1930 */
1931 if ((skb = tcp_write_queue_head(sk)) != NULL &&
1932 (TCP_SKB_CB(skb)->sacked & TCPCB_SACKED_ACKED)) {
1933 struct inet_connection_sock *icsk = inet_csk(sk); 1931 struct inet_connection_sock *icsk = inet_csk(sk);
1934 NET_INC_STATS_BH(LINUX_MIB_TCPSACKRENEGING); 1932 NET_INC_STATS_BH(LINUX_MIB_TCPSACKRENEGING);
1935 1933
@@ -2515,7 +2513,7 @@ tcp_fastretrans_alert(struct sock *sk, int pkts_acked, int flag)
2515 tp->prior_ssthresh = 0; 2513 tp->prior_ssthresh = 0;
2516 2514
2517 /* B. In all the states check for reneging SACKs. */ 2515 /* B. In all the states check for reneging SACKs. */
2518 if (tp->sacked_out && tcp_check_sack_reneging(sk)) 2516 if (tcp_check_sack_reneging(sk, flag))
2519 return; 2517 return;
2520 2518
2521 /* C. Process data loss notification, provided it is valid. */ 2519 /* C. Process data loss notification, provided it is valid. */
@@ -2852,6 +2850,9 @@ static int tcp_clean_rtx_queue(struct sock *sk, int prior_fackets)
2852 tcp_clear_all_retrans_hints(tp); 2850 tcp_clear_all_retrans_hints(tp);
2853 } 2851 }
2854 2852
2853 if (skb && (TCP_SKB_CB(skb)->sacked & TCPCB_SACKED_ACKED))
2854 flag |= FLAG_SACK_RENEGING;
2855
2855 if (flag & FLAG_ACKED) { 2856 if (flag & FLAG_ACKED) {
2856 const struct tcp_congestion_ops *ca_ops 2857 const struct tcp_congestion_ops *ca_ops
2857 = inet_csk(sk)->icsk_ca_ops; 2858 = inet_csk(sk)->icsk_ca_ops;