aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--net/ipv4/tcp_input.c31
1 files changed, 18 insertions, 13 deletions
diff --git a/net/ipv4/tcp_input.c b/net/ipv4/tcp_input.c
index 01e8004911b4..f50d8433f042 100644
--- a/net/ipv4/tcp_input.c
+++ b/net/ipv4/tcp_input.c
@@ -1169,10 +1169,11 @@ static void tcp_mark_lost_retrans(struct sock *sk)
1169 tp->lost_retrans_low = new_low_seq; 1169 tp->lost_retrans_low = new_low_seq;
1170} 1170}
1171 1171
1172static int tcp_check_dsack(struct tcp_sock *tp, struct sk_buff *ack_skb, 1172static int tcp_check_dsack(struct sock *sk, struct sk_buff *ack_skb,
1173 struct tcp_sack_block_wire *sp, int num_sacks, 1173 struct tcp_sack_block_wire *sp, int num_sacks,
1174 u32 prior_snd_una) 1174 u32 prior_snd_una)
1175{ 1175{
1176 struct tcp_sock *tp = tcp_sk(sk);
1176 u32 start_seq_0 = get_unaligned_be32(&sp[0].start_seq); 1177 u32 start_seq_0 = get_unaligned_be32(&sp[0].start_seq);
1177 u32 end_seq_0 = get_unaligned_be32(&sp[0].end_seq); 1178 u32 end_seq_0 = get_unaligned_be32(&sp[0].end_seq);
1178 int dup_sack = 0; 1179 int dup_sack = 0;
@@ -1434,7 +1435,7 @@ tcp_sacktag_write_queue(struct sock *sk, struct sk_buff *ack_skb,
1434 tcp_highest_sack_reset(sk); 1435 tcp_highest_sack_reset(sk);
1435 } 1436 }
1436 1437
1437 found_dup_sack = tcp_check_dsack(tp, ack_skb, sp_wire, 1438 found_dup_sack = tcp_check_dsack(sk, ack_skb, sp_wire,
1438 num_sacks, prior_snd_una); 1439 num_sacks, prior_snd_una);
1439 if (found_dup_sack) 1440 if (found_dup_sack)
1440 flag |= FLAG_DSACKING_ACK; 1441 flag |= FLAG_DSACKING_ACK;
@@ -3711,8 +3712,10 @@ static inline int tcp_sack_extend(struct tcp_sack_block *sp, u32 seq,
3711 return 0; 3712 return 0;
3712} 3713}
3713 3714
3714static void tcp_dsack_set(struct tcp_sock *tp, u32 seq, u32 end_seq) 3715static void tcp_dsack_set(struct sock *sk, u32 seq, u32 end_seq)
3715{ 3716{
3717 struct tcp_sock *tp = tcp_sk(sk);
3718
3716 if (tcp_is_sack(tp) && sysctl_tcp_dsack) { 3719 if (tcp_is_sack(tp) && sysctl_tcp_dsack) {
3717 int mib_idx; 3720 int mib_idx;
3718 3721
@@ -3731,10 +3734,12 @@ static void tcp_dsack_set(struct tcp_sock *tp, u32 seq, u32 end_seq)
3731 } 3734 }
3732} 3735}
3733 3736
3734static void tcp_dsack_extend(struct tcp_sock *tp, u32 seq, u32 end_seq) 3737static void tcp_dsack_extend(struct sock *sk, u32 seq, u32 end_seq)
3735{ 3738{
3739 struct tcp_sock *tp = tcp_sk(sk);
3740
3736 if (!tp->rx_opt.dsack) 3741 if (!tp->rx_opt.dsack)
3737 tcp_dsack_set(tp, seq, end_seq); 3742 tcp_dsack_set(sk, seq, end_seq);
3738 else 3743 else
3739 tcp_sack_extend(tp->duplicate_sack, seq, end_seq); 3744 tcp_sack_extend(tp->duplicate_sack, seq, end_seq);
3740} 3745}
@@ -3753,7 +3758,7 @@ static void tcp_send_dupack(struct sock *sk, struct sk_buff *skb)
3753 3758
3754 if (after(TCP_SKB_CB(skb)->end_seq, tp->rcv_nxt)) 3759 if (after(TCP_SKB_CB(skb)->end_seq, tp->rcv_nxt))
3755 end_seq = tp->rcv_nxt; 3760 end_seq = tp->rcv_nxt;
3756 tcp_dsack_set(tp, TCP_SKB_CB(skb)->seq, end_seq); 3761 tcp_dsack_set(sk, TCP_SKB_CB(skb)->seq, end_seq);
3757 } 3762 }
3758 } 3763 }
3759 3764
@@ -3906,7 +3911,7 @@ static void tcp_ofo_queue(struct sock *sk)
3906 __u32 dsack = dsack_high; 3911 __u32 dsack = dsack_high;
3907 if (before(TCP_SKB_CB(skb)->end_seq, dsack_high)) 3912 if (before(TCP_SKB_CB(skb)->end_seq, dsack_high))
3908 dsack_high = TCP_SKB_CB(skb)->end_seq; 3913 dsack_high = TCP_SKB_CB(skb)->end_seq;
3909 tcp_dsack_extend(tp, TCP_SKB_CB(skb)->seq, dsack); 3914 tcp_dsack_extend(sk, TCP_SKB_CB(skb)->seq, dsack);
3910 } 3915 }
3911 3916
3912 if (!after(TCP_SKB_CB(skb)->end_seq, tp->rcv_nxt)) { 3917 if (!after(TCP_SKB_CB(skb)->end_seq, tp->rcv_nxt)) {
@@ -4035,7 +4040,7 @@ queue_and_out:
4035 if (!after(TCP_SKB_CB(skb)->end_seq, tp->rcv_nxt)) { 4040 if (!after(TCP_SKB_CB(skb)->end_seq, tp->rcv_nxt)) {
4036 /* A retransmit, 2nd most common case. Force an immediate ack. */ 4041 /* A retransmit, 2nd most common case. Force an immediate ack. */
4037 NET_INC_STATS_BH(LINUX_MIB_DELAYEDACKLOST); 4042 NET_INC_STATS_BH(LINUX_MIB_DELAYEDACKLOST);
4038 tcp_dsack_set(tp, TCP_SKB_CB(skb)->seq, TCP_SKB_CB(skb)->end_seq); 4043 tcp_dsack_set(sk, TCP_SKB_CB(skb)->seq, TCP_SKB_CB(skb)->end_seq);
4039 4044
4040out_of_window: 4045out_of_window:
4041 tcp_enter_quickack_mode(sk); 4046 tcp_enter_quickack_mode(sk);
@@ -4057,7 +4062,7 @@ drop:
4057 tp->rcv_nxt, TCP_SKB_CB(skb)->seq, 4062 tp->rcv_nxt, TCP_SKB_CB(skb)->seq,
4058 TCP_SKB_CB(skb)->end_seq); 4063 TCP_SKB_CB(skb)->end_seq);
4059 4064
4060 tcp_dsack_set(tp, TCP_SKB_CB(skb)->seq, tp->rcv_nxt); 4065 tcp_dsack_set(sk, TCP_SKB_CB(skb)->seq, tp->rcv_nxt);
4061 4066
4062 /* If window is closed, drop tail of packet. But after 4067 /* If window is closed, drop tail of packet. But after
4063 * remembering D-SACK for its head made in previous line. 4068 * remembering D-SACK for its head made in previous line.
@@ -4122,12 +4127,12 @@ drop:
4122 if (!after(end_seq, TCP_SKB_CB(skb1)->end_seq)) { 4127 if (!after(end_seq, TCP_SKB_CB(skb1)->end_seq)) {
4123 /* All the bits are present. Drop. */ 4128 /* All the bits are present. Drop. */
4124 __kfree_skb(skb); 4129 __kfree_skb(skb);
4125 tcp_dsack_set(tp, seq, end_seq); 4130 tcp_dsack_set(sk, seq, end_seq);
4126 goto add_sack; 4131 goto add_sack;
4127 } 4132 }
4128 if (after(seq, TCP_SKB_CB(skb1)->seq)) { 4133 if (after(seq, TCP_SKB_CB(skb1)->seq)) {
4129 /* Partial overlap. */ 4134 /* Partial overlap. */
4130 tcp_dsack_set(tp, seq, 4135 tcp_dsack_set(sk, seq,
4131 TCP_SKB_CB(skb1)->end_seq); 4136 TCP_SKB_CB(skb1)->end_seq);
4132 } else { 4137 } else {
4133 skb1 = skb1->prev; 4138 skb1 = skb1->prev;
@@ -4140,12 +4145,12 @@ drop:
4140 (struct sk_buff *)&tp->out_of_order_queue && 4145 (struct sk_buff *)&tp->out_of_order_queue &&
4141 after(end_seq, TCP_SKB_CB(skb1)->seq)) { 4146 after(end_seq, TCP_SKB_CB(skb1)->seq)) {
4142 if (before(end_seq, TCP_SKB_CB(skb1)->end_seq)) { 4147 if (before(end_seq, TCP_SKB_CB(skb1)->end_seq)) {
4143 tcp_dsack_extend(tp, TCP_SKB_CB(skb1)->seq, 4148 tcp_dsack_extend(sk, TCP_SKB_CB(skb1)->seq,
4144 end_seq); 4149 end_seq);
4145 break; 4150 break;
4146 } 4151 }
4147 __skb_unlink(skb1, &tp->out_of_order_queue); 4152 __skb_unlink(skb1, &tp->out_of_order_queue);
4148 tcp_dsack_extend(tp, TCP_SKB_CB(skb1)->seq, 4153 tcp_dsack_extend(sk, TCP_SKB_CB(skb1)->seq,
4149 TCP_SKB_CB(skb1)->end_seq); 4154 TCP_SKB_CB(skb1)->end_seq);
4150 __kfree_skb(skb1); 4155 __kfree_skb(skb1);
4151 } 4156 }