mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-11-28 06:34:12 +08:00
tcp: simplify tcp_mark_skb_lost
This patch consolidates and simplifes the loss marking logic used by a few loss detections (RACK, RFC6675, NewReno). Previously each detection uses a subset of several intertwined subroutines. This unncessary complexity has led to bugs (and fixes of bug fixes). tcp_mark_skb_lost now is the single one routine to mark a packet loss when a loss detection caller deems an skb ist lost: 1. rewind tp->retransmit_hint_skb if skb has lower sequence or all lost ones have been retransmitted. 2. book-keeping: adjust flags and counts depending on if skb was retransmitted or not. Signed-off-by: Yuchung Cheng <ycheng@google.com> Signed-off-by: Neal Cardwell <ncardwell@google.com> Signed-off-by: Eric Dumazet <edumazet@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
fd2146741c
commit
686989700c
@ -1006,7 +1006,11 @@ static void tcp_check_sack_reordering(struct sock *sk, const u32 low_seq,
|
|||||||
ts ? LINUX_MIB_TCPTSREORDER : LINUX_MIB_TCPSACKREORDER);
|
ts ? LINUX_MIB_TCPTSREORDER : LINUX_MIB_TCPSACKREORDER);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This must be called before lost_out is incremented */
|
/* This must be called before lost_out or retrans_out are updated
|
||||||
|
* on a new loss, because we want to know if all skbs previously
|
||||||
|
* known to be lost have already been retransmitted, indicating
|
||||||
|
* that this newly lost skb is our next skb to retransmit.
|
||||||
|
*/
|
||||||
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 && tp->retrans_out >= tp->lost_out) ||
|
if ((!tp->retransmit_skb_hint && tp->retrans_out >= tp->lost_out) ||
|
||||||
@ -1018,34 +1022,27 @@ static void tcp_verify_retransmit_hint(struct tcp_sock *tp, struct sk_buff *skb)
|
|||||||
|
|
||||||
void tcp_mark_skb_lost(struct sock *sk, struct sk_buff *skb)
|
void tcp_mark_skb_lost(struct sock *sk, struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
|
__u8 sacked = TCP_SKB_CB(skb)->sacked;
|
||||||
struct tcp_sock *tp = tcp_sk(sk);
|
struct tcp_sock *tp = tcp_sk(sk);
|
||||||
|
|
||||||
tcp_skb_mark_lost_uncond_verify(tp, skb);
|
if (sacked & TCPCB_SACKED_ACKED)
|
||||||
if (TCP_SKB_CB(skb)->sacked & TCPCB_SACKED_RETRANS) {
|
return;
|
||||||
/* Account for retransmits that are lost again */
|
|
||||||
TCP_SKB_CB(skb)->sacked &= ~TCPCB_SACKED_RETRANS;
|
tcp_verify_retransmit_hint(tp, skb);
|
||||||
tp->retrans_out -= tcp_skb_pcount(skb);
|
if (sacked & TCPCB_LOST) {
|
||||||
NET_ADD_STATS(sock_net(sk), LINUX_MIB_TCPLOSTRETRANSMIT,
|
if (sacked & TCPCB_SACKED_RETRANS) {
|
||||||
tcp_skb_pcount(skb));
|
/* Account for retransmits that are lost again */
|
||||||
|
TCP_SKB_CB(skb)->sacked &= ~TCPCB_SACKED_RETRANS;
|
||||||
|
tp->retrans_out -= tcp_skb_pcount(skb);
|
||||||
|
NET_ADD_STATS(sock_net(sk), LINUX_MIB_TCPLOSTRETRANSMIT,
|
||||||
|
tcp_skb_pcount(skb));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
tp->lost_out += tcp_skb_pcount(skb);
|
||||||
|
TCP_SKB_CB(skb)->sacked |= TCPCB_LOST;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Sum the number of packets on the wire we have marked as lost.
|
|
||||||
* There are two cases we care about here:
|
|
||||||
* a) Packet hasn't been marked lost (nor retransmitted),
|
|
||||||
* and this is the first loss.
|
|
||||||
* b) Packet has been marked both lost and retransmitted,
|
|
||||||
* and this means we think it was lost again.
|
|
||||||
*/
|
|
||||||
static void tcp_sum_lost(struct tcp_sock *tp, struct sk_buff *skb)
|
|
||||||
{
|
|
||||||
__u8 sacked = TCP_SKB_CB(skb)->sacked;
|
|
||||||
|
|
||||||
if (!(sacked & TCPCB_LOST) ||
|
|
||||||
((sacked & TCPCB_LOST) && (sacked & TCPCB_SACKED_RETRANS)))
|
|
||||||
tp->lost += tcp_skb_pcount(skb);
|
|
||||||
}
|
|
||||||
|
|
||||||
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)
|
||||||
{
|
{
|
||||||
if (!(TCP_SKB_CB(skb)->sacked & (TCPCB_LOST|TCPCB_SACKED_ACKED))) {
|
if (!(TCP_SKB_CB(skb)->sacked & (TCPCB_LOST|TCPCB_SACKED_ACKED))) {
|
||||||
@ -1057,17 +1054,6 @@ 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);
|
|
||||||
|
|
||||||
tcp_sum_lost(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;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Updates the delivered and delivered_ce counts */
|
/* Updates the delivered and delivered_ce counts */
|
||||||
static void tcp_count_delivered(struct tcp_sock *tp, u32 delivered,
|
static void tcp_count_delivered(struct tcp_sock *tp, u32 delivered,
|
||||||
bool ece_ack)
|
bool ece_ack)
|
||||||
@ -2688,8 +2674,7 @@ void tcp_simple_retransmit(struct sock *sk)
|
|||||||
unsigned int mss = tcp_current_mss(sk);
|
unsigned int mss = tcp_current_mss(sk);
|
||||||
|
|
||||||
skb_rbtree_walk(skb, &sk->tcp_rtx_queue) {
|
skb_rbtree_walk(skb, &sk->tcp_rtx_queue) {
|
||||||
if (tcp_skb_seglen(skb) > mss &&
|
if (tcp_skb_seglen(skb) > mss)
|
||||||
!(TCP_SKB_CB(skb)->sacked & TCPCB_SACKED_ACKED))
|
|
||||||
tcp_mark_skb_lost(sk, skb);
|
tcp_mark_skb_lost(sk, skb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user