diff options
author | Eric Dumazet <eric.dumazet@gmail.com> | 2011-12-04 03:51:08 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2011-12-05 18:30:03 -0500 |
commit | 4fa48bf3c75069d636fc8830743c929a062e80dc (patch) | |
tree | 3fe5e488ee6ac1af1b76497a5e6281106a8b2d2e /net/ipv4 | |
parent | 7d3113042823344dcc175b0ea00a83d0273c98a4 (diff) |
tcp: fix tcp_trim_head()
commit f07d960df3 (tcp: avoid frag allocation for small frames)
breaked assumption in tcp stack that skb is either linear (skb->data_len
== 0), or fully fragged (skb->data_len == skb->len)
tcp_trim_head() made this assumption, we must fix it.
Thanks to Vijay for providing a very detailed explanation.
Reported-by: Vijay Subramanian <subramanian.vijay@gmail.com>
Signed-off-by: Eric Dumazet <eric.dumazet@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv4')
-rw-r--r-- | net/ipv4/tcp_output.c | 13 |
1 files changed, 8 insertions, 5 deletions
diff --git a/net/ipv4/tcp_output.c b/net/ipv4/tcp_output.c index 50788d67bdb7..cf3068038942 100644 --- a/net/ipv4/tcp_output.c +++ b/net/ipv4/tcp_output.c | |||
@@ -1093,6 +1093,13 @@ static void __pskb_trim_head(struct sk_buff *skb, int len) | |||
1093 | { | 1093 | { |
1094 | int i, k, eat; | 1094 | int i, k, eat; |
1095 | 1095 | ||
1096 | eat = min_t(int, len, skb_headlen(skb)); | ||
1097 | if (eat) { | ||
1098 | __skb_pull(skb, eat); | ||
1099 | len -= eat; | ||
1100 | if (!len) | ||
1101 | return; | ||
1102 | } | ||
1096 | eat = len; | 1103 | eat = len; |
1097 | k = 0; | 1104 | k = 0; |
1098 | for (i = 0; i < skb_shinfo(skb)->nr_frags; i++) { | 1105 | for (i = 0; i < skb_shinfo(skb)->nr_frags; i++) { |
@@ -1124,11 +1131,7 @@ int tcp_trim_head(struct sock *sk, struct sk_buff *skb, u32 len) | |||
1124 | if (skb_cloned(skb) && pskb_expand_head(skb, 0, 0, GFP_ATOMIC)) | 1131 | if (skb_cloned(skb) && pskb_expand_head(skb, 0, 0, GFP_ATOMIC)) |
1125 | return -ENOMEM; | 1132 | return -ENOMEM; |
1126 | 1133 | ||
1127 | /* If len == headlen, we avoid __skb_pull to preserve alignment. */ | 1134 | __pskb_trim_head(skb, len); |
1128 | if (unlikely(len < skb_headlen(skb))) | ||
1129 | __skb_pull(skb, len); | ||
1130 | else | ||
1131 | __pskb_trim_head(skb, len - skb_headlen(skb)); | ||
1132 | 1135 | ||
1133 | TCP_SKB_CB(skb)->seq += len; | 1136 | TCP_SKB_CB(skb)->seq += len; |
1134 | skb->ip_summed = CHECKSUM_PARTIAL; | 1137 | skb->ip_summed = CHECKSUM_PARTIAL; |