diff options
author | Eric Dumazet <edumazet@google.com> | 2015-12-30 08:51:12 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2016-01-04 17:23:36 -0500 |
commit | 197c949e7798fbf28cfadc69d9ca0c2abbf93191 (patch) | |
tree | b11734b5208854fbbbdcbcdb02a994e03c87d0c9 /net/ipv6 | |
parent | 815bc580fe4ebffc35cc837a775f5400f5a607c4 (diff) |
udp: properly support MSG_PEEK with truncated buffers
Backport of this upstream commit into stable kernels :
89c22d8c3b27 ("net: Fix skb csum races when peeking")
exposed a bug in udp stack vs MSG_PEEK support, when user provides
a buffer smaller than skb payload.
In this case,
skb_copy_and_csum_datagram_iovec(skb, sizeof(struct udphdr),
msg->msg_iov);
returns -EFAULT.
This bug does not happen in upstream kernels since Al Viro did a great
job to replace this into :
skb_copy_and_csum_datagram_msg(skb, sizeof(struct udphdr), msg);
This variant is safe vs short buffers.
For the time being, instead reverting Herbert Xu patch and add back
skb->ip_summed invalid changes, simply store the result of
udp_lib_checksum_complete() so that we avoid computing the checksum a
second time, and avoid the problematic
skb_copy_and_csum_datagram_iovec() call.
This patch can be applied on recent kernels as it avoids a double
checksumming, then backported to stable kernels as a bug fix.
Signed-off-by: Eric Dumazet <edumazet@google.com>
Acked-by: Herbert Xu <herbert@gondor.apana.org.au>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv6')
-rw-r--r-- | net/ipv6/udp.c | 6 |
1 files changed, 4 insertions, 2 deletions
diff --git a/net/ipv6/udp.c b/net/ipv6/udp.c index 9da3287a3923..00775ee27d86 100644 --- a/net/ipv6/udp.c +++ b/net/ipv6/udp.c | |||
@@ -402,6 +402,7 @@ int udpv6_recvmsg(struct sock *sk, struct msghdr *msg, size_t len, | |||
402 | int peeked, off = 0; | 402 | int peeked, off = 0; |
403 | int err; | 403 | int err; |
404 | int is_udplite = IS_UDPLITE(sk); | 404 | int is_udplite = IS_UDPLITE(sk); |
405 | bool checksum_valid = false; | ||
405 | int is_udp4; | 406 | int is_udp4; |
406 | bool slow; | 407 | bool slow; |
407 | 408 | ||
@@ -433,11 +434,12 @@ try_again: | |||
433 | */ | 434 | */ |
434 | 435 | ||
435 | if (copied < ulen || UDP_SKB_CB(skb)->partial_cov) { | 436 | if (copied < ulen || UDP_SKB_CB(skb)->partial_cov) { |
436 | if (udp_lib_checksum_complete(skb)) | 437 | checksum_valid = !udp_lib_checksum_complete(skb); |
438 | if (!checksum_valid) | ||
437 | goto csum_copy_err; | 439 | goto csum_copy_err; |
438 | } | 440 | } |
439 | 441 | ||
440 | if (skb_csum_unnecessary(skb)) | 442 | if (checksum_valid || skb_csum_unnecessary(skb)) |
441 | err = skb_copy_datagram_msg(skb, sizeof(struct udphdr), | 443 | err = skb_copy_datagram_msg(skb, sizeof(struct udphdr), |
442 | msg, copied); | 444 | msg, copied); |
443 | else { | 445 | else { |