aboutsummaryrefslogtreecommitdiffstats
path: root/net/ipv4/tcp_input.c
diff options
context:
space:
mode:
Diffstat (limited to 'net/ipv4/tcp_input.c')
-rw-r--r--net/ipv4/tcp_input.c55
1 files changed, 29 insertions, 26 deletions
diff --git a/net/ipv4/tcp_input.c b/net/ipv4/tcp_input.c
index 41fa5df9abbc..84e69e02fe20 100644
--- a/net/ipv4/tcp_input.c
+++ b/net/ipv4/tcp_input.c
@@ -4548,15 +4548,13 @@ static bool tcp_try_coalesce(struct sock *sk,
4548 int i, delta, len = from->len; 4548 int i, delta, len = from->len;
4549 4549
4550 *fragstolen = false; 4550 *fragstolen = false;
4551
4551 if (tcp_hdr(from)->fin || skb_cloned(to)) 4552 if (tcp_hdr(from)->fin || skb_cloned(to))
4552 return false; 4553 return false;
4554
4553 if (len <= skb_tailroom(to)) { 4555 if (len <= skb_tailroom(to)) {
4554 BUG_ON(skb_copy_bits(from, 0, skb_put(to, len), len)); 4556 BUG_ON(skb_copy_bits(from, 0, skb_put(to, len), len));
4555merge: 4557 goto merge;
4556 NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPRCVCOALESCE);
4557 TCP_SKB_CB(to)->end_seq = TCP_SKB_CB(from)->end_seq;
4558 TCP_SKB_CB(to)->ack_seq = TCP_SKB_CB(from)->ack_seq;
4559 return true;
4560 } 4558 }
4561 4559
4562 if (skb_has_frag_list(to) || skb_has_frag_list(from)) 4560 if (skb_has_frag_list(to) || skb_has_frag_list(from))
@@ -4581,7 +4579,6 @@ merge:
4581 skb_fill_page_desc(to, skb_shinfo(to)->nr_frags, 4579 skb_fill_page_desc(to, skb_shinfo(to)->nr_frags,
4582 page, offset, skb_headlen(from)); 4580 page, offset, skb_headlen(from));
4583 *fragstolen = true; 4581 *fragstolen = true;
4584 goto copyfrags;
4585 } else { 4582 } else {
4586 if (skb_shinfo(to)->nr_frags + 4583 if (skb_shinfo(to)->nr_frags +
4587 skb_shinfo(from)->nr_frags > MAX_SKB_FRAGS) 4584 skb_shinfo(from)->nr_frags > MAX_SKB_FRAGS)
@@ -4589,27 +4586,33 @@ merge:
4589 4586
4590 delta = from->truesize - 4587 delta = from->truesize -
4591 SKB_TRUESIZE(skb_end_pointer(from) - from->head); 4588 SKB_TRUESIZE(skb_end_pointer(from) - from->head);
4592copyfrags:
4593 WARN_ON_ONCE(delta < len);
4594 memcpy(skb_shinfo(to)->frags + skb_shinfo(to)->nr_frags,
4595 skb_shinfo(from)->frags,
4596 skb_shinfo(from)->nr_frags * sizeof(skb_frag_t));
4597 skb_shinfo(to)->nr_frags += skb_shinfo(from)->nr_frags;
4598
4599 if (skb_cloned(from))
4600 for (i = 0; i < skb_shinfo(from)->nr_frags; i++)
4601 skb_frag_ref(from, i);
4602 else
4603 skb_shinfo(from)->nr_frags = 0;
4604
4605 to->truesize += delta;
4606 atomic_add(delta, &sk->sk_rmem_alloc);
4607 sk_mem_charge(sk, delta);
4608 to->len += len;
4609 to->data_len += len;
4610 goto merge;
4611 } 4589 }
4612 return false; 4590
4591 WARN_ON_ONCE(delta < len);
4592
4593 memcpy(skb_shinfo(to)->frags + skb_shinfo(to)->nr_frags,
4594 skb_shinfo(from)->frags,
4595 skb_shinfo(from)->nr_frags * sizeof(skb_frag_t));
4596 skb_shinfo(to)->nr_frags += skb_shinfo(from)->nr_frags;
4597
4598 if (!skb_cloned(from))
4599 skb_shinfo(from)->nr_frags = 0;
4600
4601 /* if the skb is cloned this does nothing since we set nr_frags to 0 */
4602 for (i = 0; i < skb_shinfo(from)->nr_frags; i++)
4603 skb_frag_ref(from, i);
4604
4605 to->truesize += delta;
4606 atomic_add(delta, &sk->sk_rmem_alloc);
4607 sk_mem_charge(sk, delta);
4608 to->len += len;
4609 to->data_len += len;
4610
4611merge:
4612 NET_INC_STATS_BH(sock_net(sk), LINUX_MIB_TCPRCVCOALESCE);
4613 TCP_SKB_CB(to)->end_seq = TCP_SKB_CB(from)->end_seq;
4614 TCP_SKB_CB(to)->ack_seq = TCP_SKB_CB(from)->ack_seq;
4615 return true;
4613} 4616}
4614 4617
4615static void kfree_skb_partial(struct sk_buff *skb, bool head_stolen) 4618static void kfree_skb_partial(struct sk_buff *skb, bool head_stolen)