Commit 37e355f9 authored by Yaogong Wang's avatar Yaogong Wang Committed by Khalid Elmously

tcp: use an RB tree for ooo receive queue

BugLink: https://bugs.launchpad.net/bugs/1801893

[ Upstream commit 9f5afeae ]

Over the years, TCP BDP has increased by several orders of magnitude,
and some people are considering to reach the 2 Gbytes limit.

Even with current window scale limit of 14, ~1 Gbytes maps to ~740,000
MSS.

In presence of packet losses (or reorders), TCP stores incoming packets
into an out of order queue, and number of skbs sitting there waiting for
the missing packets to be received can be in the 10^5 range.

Most packets are appended to the tail of this queue, and when
packets can finally be transferred to receive queue, we scan the queue
from its head.

However, in presence of heavy losses, we might have to find an arbitrary
point in this queue, involving a linear scan for every incoming packet,
throwing away cpu caches.

This patch converts it to a RB tree, to get bounded latencies.

Yaogong wrote a preliminary patch about 2 years ago.
Eric did the rebase, added ofo_last_skb cache, polishing and tests.

Tested with network dropping between 1 and 10 % packets, with good
success (about 30 % increase of throughput in stress tests)

Next step would be to also use an RB tree for the write queue at sender
side ;)
Signed-off-by: default avatarYaogong Wang <wygivan@google.com>
Signed-off-by: default avatarEric Dumazet <edumazet@google.com>
Cc: Yuchung Cheng <ycheng@google.com>
Cc: Neal Cardwell <ncardwell@google.com>
Cc: Ilpo Järvinen <ilpo.jarvinen@helsinki.fi>
Acked-By: default avatarIlpo Järvinen <ilpo.jarvinen@helsinki.fi>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
Signed-off-by: default avatarMao Wenan <maowenan@huawei.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: default avatarJuerg Haefliger <juergh@canonical.com>
Signed-off-by: default avatarKhalid Elmously <khalid.elmously@canonical.com>
parent 356dd413
...@@ -2275,6 +2275,8 @@ static inline void __skb_queue_purge(struct sk_buff_head *list) ...@@ -2275,6 +2275,8 @@ static inline void __skb_queue_purge(struct sk_buff_head *list)
kfree_skb(skb); kfree_skb(skb);
} }
void skb_rbtree_purge(struct rb_root *root);
void *netdev_alloc_frag(unsigned int fragsz); void *netdev_alloc_frag(unsigned int fragsz);
struct sk_buff *__netdev_alloc_skb(struct net_device *dev, unsigned int length, struct sk_buff *__netdev_alloc_skb(struct net_device *dev, unsigned int length,
...@@ -2812,6 +2814,12 @@ static inline int pskb_trim_rcsum(struct sk_buff *skb, unsigned int len) ...@@ -2812,6 +2814,12 @@ static inline int pskb_trim_rcsum(struct sk_buff *skb, unsigned int len)
return __pskb_trim(skb, len); return __pskb_trim(skb, len);
} }
#define rb_to_skb(rb) rb_entry_safe(rb, struct sk_buff, rbnode)
#define skb_rb_first(root) rb_to_skb(rb_first(root))
#define skb_rb_last(root) rb_to_skb(rb_last(root))
#define skb_rb_next(skb) rb_to_skb(rb_next(&(skb)->rbnode))
#define skb_rb_prev(skb) rb_to_skb(rb_prev(&(skb)->rbnode))
#define skb_queue_walk(queue, skb) \ #define skb_queue_walk(queue, skb) \
for (skb = (queue)->next; \ for (skb = (queue)->next; \
skb != (struct sk_buff *)(queue); \ skb != (struct sk_buff *)(queue); \
......
...@@ -279,10 +279,9 @@ struct tcp_sock { ...@@ -279,10 +279,9 @@ struct tcp_sock {
struct sk_buff* lost_skb_hint; struct sk_buff* lost_skb_hint;
struct sk_buff *retransmit_skb_hint; struct sk_buff *retransmit_skb_hint;
/* OOO segments go in this list. Note that socket lock must be held, /* OOO segments go in this rbtree. Socket lock must be held. */
* as we do not use sk_buff_head lock. struct rb_root out_of_order_queue;
*/ struct sk_buff *ooo_last_skb; /* cache rb_last(out_of_order_queue) */
struct sk_buff_head out_of_order_queue;
/* SACKs data, these 2 need to be together (see tcp_options_write) */ /* SACKs data, these 2 need to be together (see tcp_options_write) */
struct tcp_sack_block duplicate_sack[1]; /* D-SACK block */ struct tcp_sack_block duplicate_sack[1]; /* D-SACK block */
......
...@@ -649,7 +649,7 @@ static inline void tcp_fast_path_check(struct sock *sk) ...@@ -649,7 +649,7 @@ static inline void tcp_fast_path_check(struct sock *sk)
{ {
struct tcp_sock *tp = tcp_sk(sk); struct tcp_sock *tp = tcp_sk(sk);
if (skb_queue_empty(&tp->out_of_order_queue) && if (RB_EMPTY_ROOT(&tp->out_of_order_queue) &&
tp->rcv_wnd && tp->rcv_wnd &&
atomic_read(&sk->sk_rmem_alloc) < sk->sk_rcvbuf && atomic_read(&sk->sk_rmem_alloc) < sk->sk_rcvbuf &&
!tp->urg_data) !tp->urg_data)
......
...@@ -2448,6 +2448,25 @@ void skb_queue_purge(struct sk_buff_head *list) ...@@ -2448,6 +2448,25 @@ void skb_queue_purge(struct sk_buff_head *list)
} }
EXPORT_SYMBOL(skb_queue_purge); EXPORT_SYMBOL(skb_queue_purge);
/**
* skb_rbtree_purge - empty a skb rbtree
* @root: root of the rbtree to empty
*
* Delete all buffers on an &sk_buff rbtree. Each buffer is removed from
* the list and one reference dropped. This function does not take
* any lock. Synchronization should be handled by the caller (e.g., TCP
* out-of-order queue is protected by the socket lock).
*/
void skb_rbtree_purge(struct rb_root *root)
{
struct sk_buff *skb, *next;
rbtree_postorder_for_each_entry_safe(skb, next, root, rbnode)
kfree_skb(skb);
*root = RB_ROOT;
}
/** /**
* skb_queue_head - queue a buffer at the list head * skb_queue_head - queue a buffer at the list head
* @list: list to use * @list: list to use
......
...@@ -382,7 +382,7 @@ void tcp_init_sock(struct sock *sk) ...@@ -382,7 +382,7 @@ void tcp_init_sock(struct sock *sk)
struct inet_connection_sock *icsk = inet_csk(sk); struct inet_connection_sock *icsk = inet_csk(sk);
struct tcp_sock *tp = tcp_sk(sk); struct tcp_sock *tp = tcp_sk(sk);
__skb_queue_head_init(&tp->out_of_order_queue); tp->out_of_order_queue = RB_ROOT;
tcp_init_xmit_timers(sk); tcp_init_xmit_timers(sk);
tcp_prequeue_init(tp); tcp_prequeue_init(tp);
INIT_LIST_HEAD(&tp->tsq_node); INIT_LIST_HEAD(&tp->tsq_node);
...@@ -2240,7 +2240,7 @@ int tcp_disconnect(struct sock *sk, int flags) ...@@ -2240,7 +2240,7 @@ int tcp_disconnect(struct sock *sk, int flags)
tcp_clear_xmit_timers(sk); tcp_clear_xmit_timers(sk);
__skb_queue_purge(&sk->sk_receive_queue); __skb_queue_purge(&sk->sk_receive_queue);
tcp_write_queue_purge(sk); tcp_write_queue_purge(sk);
__skb_queue_purge(&tp->out_of_order_queue); skb_rbtree_purge(&tp->out_of_order_queue);
inet->inet_dport = 0; inet->inet_dport = 0;
......
...@@ -4070,7 +4070,7 @@ static void tcp_fin(struct sock *sk) ...@@ -4070,7 +4070,7 @@ static void tcp_fin(struct sock *sk)
/* It _is_ possible, that we have something out-of-order _after_ FIN. /* It _is_ possible, that we have something out-of-order _after_ FIN.
* Probably, we should reset in this case. For now drop them. * Probably, we should reset in this case. For now drop them.
*/ */
__skb_queue_purge(&tp->out_of_order_queue); skb_rbtree_purge(&tp->out_of_order_queue);
if (tcp_is_sack(tp)) if (tcp_is_sack(tp))
tcp_sack_reset(&tp->rx_opt); tcp_sack_reset(&tp->rx_opt);
sk_mem_reclaim(sk); sk_mem_reclaim(sk);
...@@ -4230,7 +4230,7 @@ static void tcp_sack_remove(struct tcp_sock *tp) ...@@ -4230,7 +4230,7 @@ static void tcp_sack_remove(struct tcp_sock *tp)
int this_sack; int this_sack;
/* Empty ofo queue, hence, all the SACKs are eaten. Clear. */ /* Empty ofo queue, hence, all the SACKs are eaten. Clear. */
if (skb_queue_empty(&tp->out_of_order_queue)) { if (RB_EMPTY_ROOT(&tp->out_of_order_queue)) {
tp->rx_opt.num_sacks = 0; tp->rx_opt.num_sacks = 0;
return; return;
} }
...@@ -4306,10 +4306,13 @@ static void tcp_ofo_queue(struct sock *sk) ...@@ -4306,10 +4306,13 @@ static void tcp_ofo_queue(struct sock *sk)
{ {
struct tcp_sock *tp = tcp_sk(sk); struct tcp_sock *tp = tcp_sk(sk);
__u32 dsack_high = tp->rcv_nxt; __u32 dsack_high = tp->rcv_nxt;
bool fin, fragstolen, eaten;
struct sk_buff *skb, *tail; struct sk_buff *skb, *tail;
bool fragstolen, eaten; struct rb_node *p;
while ((skb = skb_peek(&tp->out_of_order_queue)) != NULL) { p = rb_first(&tp->out_of_order_queue);
while (p) {
skb = rb_entry(p, struct sk_buff, rbnode);
if (after(TCP_SKB_CB(skb)->seq, tp->rcv_nxt)) if (after(TCP_SKB_CB(skb)->seq, tp->rcv_nxt))
break; break;
...@@ -4319,9 +4322,10 @@ static void tcp_ofo_queue(struct sock *sk) ...@@ -4319,9 +4322,10 @@ static void tcp_ofo_queue(struct sock *sk)
dsack_high = TCP_SKB_CB(skb)->end_seq; dsack_high = TCP_SKB_CB(skb)->end_seq;
tcp_dsack_extend(sk, TCP_SKB_CB(skb)->seq, dsack); tcp_dsack_extend(sk, TCP_SKB_CB(skb)->seq, dsack);
} }
p = rb_next(p);
rb_erase(&skb->rbnode, &tp->out_of_order_queue);
__skb_unlink(skb, &tp->out_of_order_queue); if (unlikely(!after(TCP_SKB_CB(skb)->end_seq, tp->rcv_nxt))) {
if (!after(TCP_SKB_CB(skb)->end_seq, tp->rcv_nxt)) {
SOCK_DEBUG(sk, "ofo packet was already received\n"); SOCK_DEBUG(sk, "ofo packet was already received\n");
tcp_drop(sk, skb); tcp_drop(sk, skb);
continue; continue;
...@@ -4333,12 +4337,19 @@ static void tcp_ofo_queue(struct sock *sk) ...@@ -4333,12 +4337,19 @@ static void tcp_ofo_queue(struct sock *sk)
tail = skb_peek_tail(&sk->sk_receive_queue); tail = skb_peek_tail(&sk->sk_receive_queue);
eaten = tail && tcp_try_coalesce(sk, tail, skb, &fragstolen); eaten = tail && tcp_try_coalesce(sk, tail, skb, &fragstolen);
tcp_rcv_nxt_update(tp, TCP_SKB_CB(skb)->end_seq); tcp_rcv_nxt_update(tp, TCP_SKB_CB(skb)->end_seq);
fin = TCP_SKB_CB(skb)->tcp_flags & TCPHDR_FIN;
if (!eaten) if (!eaten)
__skb_queue_tail(&sk->sk_receive_queue, skb); __skb_queue_tail(&sk->sk_receive_queue, skb);
if (TCP_SKB_CB(skb)->tcp_flags & TCPHDR_FIN) else
tcp_fin(sk);
if (eaten)
kfree_skb_partial(skb, fragstolen); kfree_skb_partial(skb, fragstolen);
if (unlikely(fin)) {
tcp_fin(sk);
/* tcp_fin() purges tp->out_of_order_queue,
* so we must end this loop right now.
*/
break;
}
} }
} }
...@@ -4368,8 +4379,10 @@ static int tcp_try_rmem_schedule(struct sock *sk, struct sk_buff *skb, ...@@ -4368,8 +4379,10 @@ static int tcp_try_rmem_schedule(struct sock *sk, struct sk_buff *skb,
static void tcp_data_queue_ofo(struct sock *sk, struct sk_buff *skb) static void tcp_data_queue_ofo(struct sock *sk, struct sk_buff *skb)
{ {
struct tcp_sock *tp = tcp_sk(sk); struct tcp_sock *tp = tcp_sk(sk);
struct rb_node **p, *q, *parent;
struct sk_buff *skb1; struct sk_buff *skb1;
u32 seq, end_seq; u32 seq, end_seq;
bool fragstolen;
tcp_ecn_check_ce(sk, skb); tcp_ecn_check_ce(sk, skb);
...@@ -4384,89 +4397,86 @@ static void tcp_data_queue_ofo(struct sock *sk, struct sk_buff *skb) ...@@ -4384,89 +4397,86 @@ static void tcp_data_queue_ofo(struct sock *sk, struct sk_buff *skb)
inet_csk_schedule_ack(sk); inet_csk_schedule_ack(sk);
NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPOFOQUEUE); NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPOFOQUEUE);
seq = TCP_SKB_CB(skb)->seq;
end_seq = TCP_SKB_CB(skb)->end_seq;
SOCK_DEBUG(sk, "out of order segment: rcv_next %X seq %X - %X\n", SOCK_DEBUG(sk, "out of order segment: rcv_next %X seq %X - %X\n",
tp->rcv_nxt, TCP_SKB_CB(skb)->seq, TCP_SKB_CB(skb)->end_seq); tp->rcv_nxt, seq, end_seq);
skb1 = skb_peek_tail(&tp->out_of_order_queue); p = &tp->out_of_order_queue.rb_node;
if (!skb1) { if (RB_EMPTY_ROOT(&tp->out_of_order_queue)) {
/* Initial out of order segment, build 1 SACK. */ /* Initial out of order segment, build 1 SACK. */
if (tcp_is_sack(tp)) { if (tcp_is_sack(tp)) {
tp->rx_opt.num_sacks = 1; tp->rx_opt.num_sacks = 1;
tp->selective_acks[0].start_seq = TCP_SKB_CB(skb)->seq; tp->selective_acks[0].start_seq = seq;
tp->selective_acks[0].end_seq = tp->selective_acks[0].end_seq = end_seq;
TCP_SKB_CB(skb)->end_seq;
} }
__skb_queue_head(&tp->out_of_order_queue, skb); rb_link_node(&skb->rbnode, NULL, p);
rb_insert_color(&skb->rbnode, &tp->out_of_order_queue);
tp->ooo_last_skb = skb;
goto end; goto end;
} }
seq = TCP_SKB_CB(skb)->seq; /* In the typical case, we are adding an skb to the end of the list.
end_seq = TCP_SKB_CB(skb)->end_seq; * Use of ooo_last_skb avoids the O(Log(N)) rbtree lookup.
*/
if (seq == TCP_SKB_CB(skb1)->end_seq) { if (tcp_try_coalesce(sk, tp->ooo_last_skb, skb, &fragstolen)) {
bool fragstolen; coalesce_done:
if (!tcp_try_coalesce(sk, skb1, skb, &fragstolen)) {
__skb_queue_after(&tp->out_of_order_queue, skb1, skb);
} else {
tcp_grow_window(sk, skb); tcp_grow_window(sk, skb);
kfree_skb_partial(skb, fragstolen); kfree_skb_partial(skb, fragstolen);
skb = NULL; skb = NULL;
}
if (!tp->rx_opt.num_sacks ||
tp->selective_acks[0].end_seq != seq)
goto add_sack; goto add_sack;
/* Common case: data arrive in order after hole. */
tp->selective_acks[0].end_seq = end_seq;
goto end;
} }
/* Find place to insert this segment. */ /* Find place to insert this segment. Handle overlaps on the way. */
while (1) { parent = NULL;
if (!after(TCP_SKB_CB(skb1)->seq, seq)) while (*p) {
break; parent = *p;
if (skb_queue_is_first(&tp->out_of_order_queue, skb1)) { skb1 = rb_entry(parent, struct sk_buff, rbnode);
skb1 = NULL; if (before(seq, TCP_SKB_CB(skb1)->seq)) {
break; p = &parent->rb_left;
} continue;
skb1 = skb_queue_prev(&tp->out_of_order_queue, skb1);
} }
/* Do skb overlap to previous one? */ if (before(seq, TCP_SKB_CB(skb1)->end_seq)) {
if (skb1 && before(seq, TCP_SKB_CB(skb1)->end_seq)) {
if (!after(end_seq, TCP_SKB_CB(skb1)->end_seq)) { if (!after(end_seq, TCP_SKB_CB(skb1)->end_seq)) {
/* All the bits are present. Drop. */ /* All the bits are present. Drop. */
NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPOFOMERGE); NET_INC_STATS(sock_net(sk),
tcp_drop(sk, skb); LINUX_MIB_TCPOFOMERGE);
__kfree_skb(skb);
skb = NULL; skb = NULL;
tcp_dsack_set(sk, seq, end_seq); tcp_dsack_set(sk, seq, end_seq);
goto add_sack; goto add_sack;
} }
if (after(seq, TCP_SKB_CB(skb1)->seq)) { if (after(seq, TCP_SKB_CB(skb1)->seq)) {
/* Partial overlap. */ /* Partial overlap. */
tcp_dsack_set(sk, seq, tcp_dsack_set(sk, seq, TCP_SKB_CB(skb1)->end_seq);
TCP_SKB_CB(skb1)->end_seq);
} else { } else {
if (skb_queue_is_first(&tp->out_of_order_queue, /* skb's seq == skb1's seq and skb covers skb1.
skb1)) * Replace skb1 with skb.
skb1 = NULL; */
else rb_replace_node(&skb1->rbnode, &skb->rbnode,
skb1 = skb_queue_prev( &tp->out_of_order_queue);
&tp->out_of_order_queue, tcp_dsack_extend(sk,
skb1); TCP_SKB_CB(skb1)->seq,
TCP_SKB_CB(skb1)->end_seq);
NET_INC_STATS(sock_net(sk),
LINUX_MIB_TCPOFOMERGE);
__kfree_skb(skb1);
goto add_sack;
} }
} else if (tcp_try_coalesce(sk, skb1, skb, &fragstolen)) {
goto coalesce_done;
}
p = &parent->rb_right;
} }
if (!skb1)
__skb_queue_head(&tp->out_of_order_queue, skb);
else
__skb_queue_after(&tp->out_of_order_queue, skb1, skb);
/* And clean segments covered by new one as whole. */ /* Insert segment into RB tree. */
while (!skb_queue_is_last(&tp->out_of_order_queue, skb)) { rb_link_node(&skb->rbnode, parent, p);
skb1 = skb_queue_next(&tp->out_of_order_queue, skb); rb_insert_color(&skb->rbnode, &tp->out_of_order_queue);
/* Remove other segments covered by skb. */
while ((q = rb_next(&skb->rbnode)) != NULL) {
skb1 = rb_entry(q, struct sk_buff, rbnode);
if (!after(end_seq, TCP_SKB_CB(skb1)->seq)) if (!after(end_seq, TCP_SKB_CB(skb1)->seq))
break; break;
if (before(end_seq, TCP_SKB_CB(skb1)->end_seq)) { if (before(end_seq, TCP_SKB_CB(skb1)->end_seq)) {
...@@ -4474,12 +4484,15 @@ static void tcp_data_queue_ofo(struct sock *sk, struct sk_buff *skb) ...@@ -4474,12 +4484,15 @@ static void tcp_data_queue_ofo(struct sock *sk, struct sk_buff *skb)
end_seq); end_seq);
break; break;
} }
__skb_unlink(skb1, &tp->out_of_order_queue); rb_erase(&skb1->rbnode, &tp->out_of_order_queue);
tcp_dsack_extend(sk, TCP_SKB_CB(skb1)->seq, tcp_dsack_extend(sk, TCP_SKB_CB(skb1)->seq,
TCP_SKB_CB(skb1)->end_seq); TCP_SKB_CB(skb1)->end_seq);
NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPOFOMERGE); NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPOFOMERGE);
tcp_drop(sk, skb1); tcp_drop(sk, skb1);
} }
/* If there is no skb after us, we are the last_skb ! */
if (!q)
tp->ooo_last_skb = skb;
add_sack: add_sack:
if (tcp_is_sack(tp)) if (tcp_is_sack(tp))
...@@ -4618,13 +4631,13 @@ static void tcp_data_queue(struct sock *sk, struct sk_buff *skb) ...@@ -4618,13 +4631,13 @@ static void tcp_data_queue(struct sock *sk, struct sk_buff *skb)
if (TCP_SKB_CB(skb)->tcp_flags & TCPHDR_FIN) if (TCP_SKB_CB(skb)->tcp_flags & TCPHDR_FIN)
tcp_fin(sk); tcp_fin(sk);
if (!skb_queue_empty(&tp->out_of_order_queue)) { if (!RB_EMPTY_ROOT(&tp->out_of_order_queue)) {
tcp_ofo_queue(sk); tcp_ofo_queue(sk);
/* RFC2581. 4.2. SHOULD send immediate ACK, when /* RFC2581. 4.2. SHOULD send immediate ACK, when
* gap in queue is filled. * gap in queue is filled.
*/ */
if (skb_queue_empty(&tp->out_of_order_queue)) if (RB_EMPTY_ROOT(&tp->out_of_order_queue))
inet_csk(sk)->icsk_ack.pingpong = 0; inet_csk(sk)->icsk_ack.pingpong = 0;
} }
...@@ -4676,48 +4689,76 @@ static void tcp_data_queue(struct sock *sk, struct sk_buff *skb) ...@@ -4676,48 +4689,76 @@ static void tcp_data_queue(struct sock *sk, struct sk_buff *skb)
tcp_data_queue_ofo(sk, skb); tcp_data_queue_ofo(sk, skb);
} }
static struct sk_buff *tcp_collapse_one(struct sock *sk, struct sk_buff *skb, static struct sk_buff *tcp_skb_next(struct sk_buff *skb, struct sk_buff_head *list)
struct sk_buff_head *list)
{ {
struct sk_buff *next = NULL; if (list)
return !skb_queue_is_last(list, skb) ? skb->next : NULL;
return rb_entry_safe(rb_next(&skb->rbnode), struct sk_buff, rbnode);
}
if (!skb_queue_is_last(list, skb)) static struct sk_buff *tcp_collapse_one(struct sock *sk, struct sk_buff *skb,
next = skb_queue_next(list, skb); struct sk_buff_head *list,
struct rb_root *root)
{
struct sk_buff *next = tcp_skb_next(skb, list);
if (list)
__skb_unlink(skb, list); __skb_unlink(skb, list);
else
rb_erase(&skb->rbnode, root);
__kfree_skb(skb); __kfree_skb(skb);
NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPRCVCOLLAPSED); NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPRCVCOLLAPSED);
return next; return next;
} }
/* Insert skb into rb tree, ordered by TCP_SKB_CB(skb)->seq */
static void tcp_rbtree_insert(struct rb_root *root, struct sk_buff *skb)
{
struct rb_node **p = &root->rb_node;
struct rb_node *parent = NULL;
struct sk_buff *skb1;
while (*p) {
parent = *p;
skb1 = rb_entry(parent, struct sk_buff, rbnode);
if (before(TCP_SKB_CB(skb)->seq, TCP_SKB_CB(skb1)->seq))
p = &parent->rb_left;
else
p = &parent->rb_right;
}
rb_link_node(&skb->rbnode, parent, p);
rb_insert_color(&skb->rbnode, root);
}
/* Collapse contiguous sequence of skbs head..tail with /* Collapse contiguous sequence of skbs head..tail with
* sequence numbers start..end. * sequence numbers start..end.
* *
* If tail is NULL, this means until the end of the list. * If tail is NULL, this means until the end of the queue.
* *
* Segments with FIN/SYN are not collapsed (only because this * Segments with FIN/SYN are not collapsed (only because this
* simplifies code) * simplifies code)
*/ */
static void static void
tcp_collapse(struct sock *sk, struct sk_buff_head *list, tcp_collapse(struct sock *sk, struct sk_buff_head *list, struct rb_root *root,
struct sk_buff *head, struct sk_buff *tail, struct sk_buff *head, struct sk_buff *tail, u32 start, u32 end)
u32 start, u32 end)
{ {
struct sk_buff *skb, *n; struct sk_buff *skb = head, *n;
struct sk_buff_head tmp;
bool end_of_skbs; bool end_of_skbs;
/* First, check that queue is collapsible and find /* First, check that queue is collapsible and find
* the point where collapsing can be useful. */ * the point where collapsing can be useful.
skb = head; */
restart: restart:
end_of_skbs = true; for (end_of_skbs = true; skb != NULL && skb != tail; skb = n) {
skb_queue_walk_from_safe(list, skb, n) { n = tcp_skb_next(skb, list);
if (skb == tail)
break;
/* No new bits? It is possible on ofo queue. */ /* No new bits? It is possible on ofo queue. */
if (!before(start, TCP_SKB_CB(skb)->end_seq)) { if (!before(start, TCP_SKB_CB(skb)->end_seq)) {
skb = tcp_collapse_one(sk, skb, list); skb = tcp_collapse_one(sk, skb, list, root);
if (!skb) if (!skb)
break; break;
goto restart; goto restart;
...@@ -4735,14 +4776,11 @@ tcp_collapse(struct sock *sk, struct sk_buff_head *list, ...@@ -4735,14 +4776,11 @@ tcp_collapse(struct sock *sk, struct sk_buff_head *list,
break; break;
} }
if (!skb_queue_is_last(list, skb)) { if (n && n != tail &&
struct sk_buff *next = skb_queue_next(list, skb); TCP_SKB_CB(skb)->end_seq != TCP_SKB_CB(n)->seq) {
if (next != tail &&
TCP_SKB_CB(skb)->end_seq != TCP_SKB_CB(next)->seq) {
end_of_skbs = false; end_of_skbs = false;
break; break;
} }
}
/* Decided to skip this, advance start seq. */ /* Decided to skip this, advance start seq. */
start = TCP_SKB_CB(skb)->end_seq; start = TCP_SKB_CB(skb)->end_seq;
...@@ -4751,17 +4789,22 @@ tcp_collapse(struct sock *sk, struct sk_buff_head *list, ...@@ -4751,17 +4789,22 @@ tcp_collapse(struct sock *sk, struct sk_buff_head *list,
(TCP_SKB_CB(skb)->tcp_flags & (TCPHDR_SYN | TCPHDR_FIN))) (TCP_SKB_CB(skb)->tcp_flags & (TCPHDR_SYN | TCPHDR_FIN)))
return; return;
__skb_queue_head_init(&tmp);
while (before(start, end)) { while (before(start, end)) {
int copy = min_t(int, SKB_MAX_ORDER(0, 0), end - start); int copy = min_t(int, SKB_MAX_ORDER(0, 0), end - start);
struct sk_buff *nskb; struct sk_buff *nskb;
nskb = alloc_skb(copy, GFP_ATOMIC); nskb = alloc_skb(copy, GFP_ATOMIC);
if (!nskb) if (!nskb)
return; break;
memcpy(nskb->cb, skb->cb, sizeof(skb->cb)); memcpy(nskb->cb, skb->cb, sizeof(skb->cb));
TCP_SKB_CB(nskb)->seq = TCP_SKB_CB(nskb)->end_seq = start; TCP_SKB_CB(nskb)->seq = TCP_SKB_CB(nskb)->end_seq = start;
if (list)
__skb_queue_before(list, skb, nskb); __skb_queue_before(list, skb, nskb);
else
__skb_queue_tail(&tmp, nskb); /* defer rbtree insertion */
skb_set_owner_r(nskb, sk); skb_set_owner_r(nskb, sk);
/* Copy data, releasing collapsed skbs. */ /* Copy data, releasing collapsed skbs. */
...@@ -4779,14 +4822,17 @@ tcp_collapse(struct sock *sk, struct sk_buff_head *list, ...@@ -4779,14 +4822,17 @@ tcp_collapse(struct sock *sk, struct sk_buff_head *list,
start += size; start += size;
} }
if (!before(start, TCP_SKB_CB(skb)->end_seq)) { if (!before(start, TCP_SKB_CB(skb)->end_seq)) {
skb = tcp_collapse_one(sk, skb, list); skb = tcp_collapse_one(sk, skb, list, root);
if (!skb || if (!skb ||
skb == tail || skb == tail ||
(TCP_SKB_CB(skb)->tcp_flags & (TCPHDR_SYN | TCPHDR_FIN))) (TCP_SKB_CB(skb)->tcp_flags & (TCPHDR_SYN | TCPHDR_FIN)))
return; goto end;
} }
} }
} }
end:
skb_queue_walk_safe(&tmp, skb, n)
tcp_rbtree_insert(root, skb);
} }
/* Collapse ofo queue. Algorithm: select contiguous sequence of skbs /* Collapse ofo queue. Algorithm: select contiguous sequence of skbs
...@@ -4796,34 +4842,39 @@ static void tcp_collapse_ofo_queue(struct sock *sk) ...@@ -4796,34 +4842,39 @@ static void tcp_collapse_ofo_queue(struct sock *sk)
{ {
struct tcp_sock *tp = tcp_sk(sk); struct tcp_sock *tp = tcp_sk(sk);
u32 range_truesize, sum_tiny = 0; u32 range_truesize, sum_tiny = 0;
struct sk_buff *skb = skb_peek(&tp->out_of_order_queue); struct sk_buff *skb, *head;
struct sk_buff *head; struct rb_node *p;
u32 start, end; u32 start, end;
if (!skb) p = rb_first(&tp->out_of_order_queue);
skb = rb_entry_safe(p, struct sk_buff, rbnode);
new_range:
if (!skb) {
p = rb_last(&tp->out_of_order_queue);
/* Note: This is possible p is NULL here. We do not
* use rb_entry_safe(), as ooo_last_skb is valid only
* if rbtree is not empty.
*/
tp->ooo_last_skb = rb_entry(p, struct sk_buff, rbnode);
return; return;
}
start = TCP_SKB_CB(skb)->seq; start = TCP_SKB_CB(skb)->seq;
end = TCP_SKB_CB(skb)->end_seq; end = TCP_SKB_CB(skb)->end_seq;
range_truesize = skb->truesize; range_truesize = skb->truesize;
head = skb;
for (;;) {
struct sk_buff *next = NULL;
if (!skb_queue_is_last(&tp->out_of_order_queue, skb)) for (head = skb;;) {
next = skb_queue_next(&tp->out_of_order_queue, skb); skb = tcp_skb_next(skb, NULL);
skb = next;
/* Segment is terminated when we see gap or when /* Range is terminated when we see a gap or when
* we are at the end of all the queue. */ * we are at the queue end.
*/
if (!skb || if (!skb ||
after(TCP_SKB_CB(skb)->seq, end) || after(TCP_SKB_CB(skb)->seq, end) ||
before(TCP_SKB_CB(skb)->end_seq, start)) { before(TCP_SKB_CB(skb)->end_seq, start)) {
/* Do not attempt collapsing tiny skbs */ /* Do not attempt collapsing tiny skbs */
if (range_truesize != head->truesize || if (range_truesize != head->truesize ||
end - start >= SKB_WITH_OVERHEAD(SK_MEM_QUANTUM)) { end - start >= SKB_WITH_OVERHEAD(SK_MEM_QUANTUM)) {
tcp_collapse(sk, &tp->out_of_order_queue, tcp_collapse(sk, NULL, &tp->out_of_order_queue,
head, skb, start, end); head, skb, start, end);
} else { } else {
sum_tiny += range_truesize; sum_tiny += range_truesize;
...@@ -4831,21 +4882,15 @@ static void tcp_collapse_ofo_queue(struct sock *sk) ...@@ -4831,21 +4882,15 @@ static void tcp_collapse_ofo_queue(struct sock *sk)
return; return;
} }
head = skb; goto new_range;
if (!skb) }
break;
/* Start new segment */
start = TCP_SKB_CB(skb)->seq;
end = TCP_SKB_CB(skb)->end_seq;
range_truesize = skb->truesize;
} else {
range_truesize += skb->truesize; range_truesize += skb->truesize;
if (before(TCP_SKB_CB(skb)->seq, start)) if (unlikely(before(TCP_SKB_CB(skb)->seq, start)))
start = TCP_SKB_CB(skb)->seq; start = TCP_SKB_CB(skb)->seq;
if (after(TCP_SKB_CB(skb)->end_seq, end)) if (after(TCP_SKB_CB(skb)->end_seq, end))
end = TCP_SKB_CB(skb)->end_seq; end = TCP_SKB_CB(skb)->end_seq;
} }
}
} }
/* /*
...@@ -4855,11 +4900,26 @@ static void tcp_collapse_ofo_queue(struct sock *sk) ...@@ -4855,11 +4900,26 @@ static void tcp_collapse_ofo_queue(struct sock *sk)
static bool tcp_prune_ofo_queue(struct sock *sk) static bool tcp_prune_ofo_queue(struct sock *sk)
{ {
struct tcp_sock *tp = tcp_sk(sk); struct tcp_sock *tp = tcp_sk(sk);
bool res = false; struct rb_node *node, *prev;
if (RB_EMPTY_ROOT(&tp->out_of_order_queue))
return false;
if (!skb_queue_empty(&tp->out_of_order_queue)) {
NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_OFOPRUNED); NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_OFOPRUNED);
__skb_queue_purge(&tp->out_of_order_queue);
node = &tp->ooo_last_skb->rbnode;
do {
prev = rb_prev(node);
rb_erase(node, &tp->out_of_order_queue);
__kfree_skb(rb_to_skb(node));
sk_mem_reclaim(sk);
if (atomic_read(&sk->sk_rmem_alloc) <= sk->sk_rcvbuf &&
!tcp_under_memory_pressure(sk))
break;
node = prev;
} while (node);
tp->ooo_last_skb = rb_entry(prev, struct sk_buff, rbnode);
/* Reset SACK state. A conforming SACK implementation will /* Reset SACK state. A conforming SACK implementation will
* do the same at a timeout based retransmit. When a connection * do the same at a timeout based retransmit. When a connection
...@@ -4868,10 +4928,8 @@ static bool tcp_prune_ofo_queue(struct sock *sk) ...@@ -4868,10 +4928,8 @@ static bool tcp_prune_ofo_queue(struct sock *sk)
*/ */
if (tp->rx_opt.sack_ok) if (tp->rx_opt.sack_ok)
tcp_sack_reset(&tp->rx_opt); tcp_sack_reset(&tp->rx_opt);
sk_mem_reclaim(sk);
res = true; return true;
}
return res;
} }
/* Reduce allocated memory if we can, trying to get /* Reduce allocated memory if we can, trying to get
...@@ -4899,7 +4957,7 @@ static int tcp_prune_queue(struct sock *sk) ...@@ -4899,7 +4957,7 @@ static int tcp_prune_queue(struct sock *sk)
tcp_collapse_ofo_queue(sk); tcp_collapse_ofo_queue(sk);
if (!skb_queue_empty(&sk->sk_receive_queue)) if (!skb_queue_empty(&sk->sk_receive_queue))
tcp_collapse(sk, &sk->sk_receive_queue, tcp_collapse(sk, &sk->sk_receive_queue, NULL,
skb_peek(&sk->sk_receive_queue), skb_peek(&sk->sk_receive_queue),
NULL, NULL,
tp->copied_seq, tp->rcv_nxt); tp->copied_seq, tp->rcv_nxt);
...@@ -5004,7 +5062,7 @@ static void __tcp_ack_snd_check(struct sock *sk, int ofo_possible) ...@@ -5004,7 +5062,7 @@ static void __tcp_ack_snd_check(struct sock *sk, int ofo_possible)
/* We ACK each frame or... */ /* We ACK each frame or... */
tcp_in_quickack_mode(sk) || tcp_in_quickack_mode(sk) ||
/* We have out of order data. */ /* We have out of order data. */
(ofo_possible && skb_peek(&tp->out_of_order_queue))) { (ofo_possible && !RB_EMPTY_ROOT(&tp->out_of_order_queue))) {
/* Then ack it now */ /* Then ack it now */
tcp_send_ack(sk); tcp_send_ack(sk);
} else { } else {
......
...@@ -1830,7 +1830,7 @@ void tcp_v4_destroy_sock(struct sock *sk) ...@@ -1830,7 +1830,7 @@ void tcp_v4_destroy_sock(struct sock *sk)
tcp_write_queue_purge(sk); tcp_write_queue_purge(sk);
/* Cleans up our, hopefully empty, out_of_order_queue. */ /* Cleans up our, hopefully empty, out_of_order_queue. */
__skb_queue_purge(&tp->out_of_order_queue); skb_rbtree_purge(&tp->out_of_order_queue);
#ifdef CONFIG_TCP_MD5SIG #ifdef CONFIG_TCP_MD5SIG
/* Clean up the MD5 key list, if any */ /* Clean up the MD5 key list, if any */
......
...@@ -496,7 +496,6 @@ struct sock *tcp_create_openreq_child(const struct sock *sk, ...@@ -496,7 +496,6 @@ struct sock *tcp_create_openreq_child(const struct sock *sk,
newtp->snd_cwnd_cnt = 0; newtp->snd_cwnd_cnt = 0;
tcp_init_xmit_timers(newsk); tcp_init_xmit_timers(newsk);
__skb_queue_head_init(&newtp->out_of_order_queue);
newtp->write_seq = newtp->pushed_seq = treq->snt_isn + 1; newtp->write_seq = newtp->pushed_seq = treq->snt_isn + 1;
newtp->rx_opt.saw_tstamp = 0; newtp->rx_opt.saw_tstamp = 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment