|
@@ -979,17 +979,17 @@ static void tcp_update_reordering(struct sock *sk, const int metric,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-/* RFC: This is from the original, I doubt that this is necessary at all:
|
|
|
|
- * clear xmit_retrans hint if seq of this skb is beyond hint. How could we
|
|
|
|
- * retransmitted past LOST markings in the first place? I'm not fully sure
|
|
|
|
- * about undo and end of connection cases, which can cause R without L?
|
|
|
|
- */
|
|
|
|
|
|
+/* This must be called before lost_out is incremented */
|
|
static void tcp_verify_retransmit_hint(struct tcp_sock *tp, struct sk_buff *skb)
|
|
static void tcp_verify_retransmit_hint(struct tcp_sock *tp, struct sk_buff *skb)
|
|
{
|
|
{
|
|
- if ((tp->retransmit_skb_hint != NULL) &&
|
|
|
|
|
|
+ if ((tp->retransmit_skb_hint == NULL) ||
|
|
before(TCP_SKB_CB(skb)->seq,
|
|
before(TCP_SKB_CB(skb)->seq,
|
|
TCP_SKB_CB(tp->retransmit_skb_hint)->seq))
|
|
TCP_SKB_CB(tp->retransmit_skb_hint)->seq))
|
|
- tp->retransmit_skb_hint = NULL;
|
|
|
|
|
|
+ tp->retransmit_skb_hint = skb;
|
|
|
|
+
|
|
|
|
+ if (!tp->lost_out ||
|
|
|
|
+ after(TCP_SKB_CB(skb)->end_seq, tp->retransmit_high))
|
|
|
|
+ tp->retransmit_high = TCP_SKB_CB(skb)->end_seq;
|
|
}
|
|
}
|
|
|
|
|
|
static void tcp_skb_mark_lost(struct tcp_sock *tp, struct sk_buff *skb)
|
|
static void tcp_skb_mark_lost(struct tcp_sock *tp, struct sk_buff *skb)
|
|
@@ -1002,6 +1002,16 @@ static void tcp_skb_mark_lost(struct tcp_sock *tp, struct sk_buff *skb)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+void tcp_skb_mark_lost_uncond_verify(struct tcp_sock *tp, struct sk_buff *skb)
|
|
|
|
+{
|
|
|
|
+ tcp_verify_retransmit_hint(tp, skb);
|
|
|
|
+
|
|
|
|
+ if (!(TCP_SKB_CB(skb)->sacked & (TCPCB_LOST|TCPCB_SACKED_ACKED))) {
|
|
|
|
+ tp->lost_out += tcp_skb_pcount(skb);
|
|
|
|
+ TCP_SKB_CB(skb)->sacked |= TCPCB_LOST;
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
/* This procedure tags the retransmission queue when SACKs arrive.
|
|
/* This procedure tags the retransmission queue when SACKs arrive.
|
|
*
|
|
*
|
|
* We have three tag bits: SACKED(S), RETRANS(R) and LOST(L).
|
|
* We have three tag bits: SACKED(S), RETRANS(R) and LOST(L).
|
|
@@ -1178,13 +1188,7 @@ static void tcp_mark_lost_retrans(struct sock *sk)
|
|
TCP_SKB_CB(skb)->sacked &= ~TCPCB_SACKED_RETRANS;
|
|
TCP_SKB_CB(skb)->sacked &= ~TCPCB_SACKED_RETRANS;
|
|
tp->retrans_out -= tcp_skb_pcount(skb);
|
|
tp->retrans_out -= tcp_skb_pcount(skb);
|
|
|
|
|
|
- /* clear lost hint */
|
|
|
|
- tp->retransmit_skb_hint = NULL;
|
|
|
|
-
|
|
|
|
- if (!(TCP_SKB_CB(skb)->sacked & (TCPCB_LOST|TCPCB_SACKED_ACKED))) {
|
|
|
|
- tp->lost_out += tcp_skb_pcount(skb);
|
|
|
|
- TCP_SKB_CB(skb)->sacked |= TCPCB_LOST;
|
|
|
|
- }
|
|
|
|
|
|
+ tcp_skb_mark_lost_uncond_verify(tp, skb);
|
|
NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPLOSTRETRANSMIT);
|
|
NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPLOSTRETRANSMIT);
|
|
} else {
|
|
} else {
|
|
if (before(ack_seq, new_low_seq))
|
|
if (before(ack_seq, new_low_seq))
|
|
@@ -1890,6 +1894,7 @@ static void tcp_enter_frto_loss(struct sock *sk, int allowed_segments, int flag)
|
|
if (!(TCP_SKB_CB(skb)->sacked & TCPCB_SACKED_ACKED)) {
|
|
if (!(TCP_SKB_CB(skb)->sacked & TCPCB_SACKED_ACKED)) {
|
|
TCP_SKB_CB(skb)->sacked |= TCPCB_LOST;
|
|
TCP_SKB_CB(skb)->sacked |= TCPCB_LOST;
|
|
tp->lost_out += tcp_skb_pcount(skb);
|
|
tp->lost_out += tcp_skb_pcount(skb);
|
|
|
|
+ tp->retransmit_high = TCP_SKB_CB(skb)->end_seq;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
tcp_verify_left_out(tp);
|
|
tcp_verify_left_out(tp);
|
|
@@ -1974,6 +1979,7 @@ void tcp_enter_loss(struct sock *sk, int how)
|
|
TCP_SKB_CB(skb)->sacked &= ~TCPCB_SACKED_ACKED;
|
|
TCP_SKB_CB(skb)->sacked &= ~TCPCB_SACKED_ACKED;
|
|
TCP_SKB_CB(skb)->sacked |= TCPCB_LOST;
|
|
TCP_SKB_CB(skb)->sacked |= TCPCB_LOST;
|
|
tp->lost_out += tcp_skb_pcount(skb);
|
|
tp->lost_out += tcp_skb_pcount(skb);
|
|
|
|
+ tp->retransmit_high = TCP_SKB_CB(skb)->end_seq;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
tcp_verify_left_out(tp);
|
|
tcp_verify_left_out(tp);
|