diff options
author | Daniel Borkmann <dborkman@redhat.com> | 2013-10-30 06:50:52 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2013-11-03 23:04:57 -0500 |
commit | e6d8b64b34aa8a9fe39609bc2db8a243b0331ceb (patch) | |
tree | de3bbb3dc4e08fd17fcde46aedb239d274e45d6b /include/net/sctp | |
parent | 2817a336d4d533fb8b68719723cd60ea7dd7c09e (diff) |
net: sctp: fix and consolidate SCTP checksumming code
This fixes an outstanding bug found through IPVS, where SCTP packets
with skb->data_len > 0 (non-linearized) and empty frag_list, but data
accumulated in frags[] member, are forwarded with incorrect checksum
letting SCTP initial handshake fail on some systems. Linearizing each
SCTP skb in IPVS to prevent that would not be a good solution as
this leads to an additional and unnecessary performance penalty on
the load-balancer itself for no good reason (as we actually only want
to update the checksum, and can do that in a different/better way
presented here).
The actual problem is elsewhere, namely, that SCTP's checksumming
in sctp_compute_cksum() does not take frags[] into account like
skb_checksum() does. So while we are fixing this up, we better reuse
the existing code that we have anyway in __skb_checksum() and use it
for walking through the data doing checksumming. This will not only
fix this issue, but also consolidates some SCTP code with core
sk_buff code, bringing it closer together and removing respectively
avoiding reimplementation of skb_checksum() for no good reason.
As crc32c() can use hardware implementation within the crypto layer,
we leave that intact (it wraps around / falls back to e.g. slice-by-8
algorithm in __crc32c_le() otherwise); plus use the __crc32c_le_combine()
combinator for crc32c blocks.
Also, we remove all other SCTP checksumming code, so that we only
have to use sctp_compute_cksum() from now on; for doing that, we need
to transform SCTP checkumming in output path slightly, and can leave
the rest intact.
Signed-off-by: Daniel Borkmann <dborkman@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'include/net/sctp')
-rw-r--r-- | include/net/sctp/checksum.h | 56 |
1 files changed, 19 insertions, 37 deletions
diff --git a/include/net/sctp/checksum.h b/include/net/sctp/checksum.h index 259924d63ba6..6bd44fe94c26 100644 --- a/include/net/sctp/checksum.h +++ b/include/net/sctp/checksum.h | |||
@@ -42,56 +42,38 @@ | |||
42 | #include <linux/types.h> | 42 | #include <linux/types.h> |
43 | #include <net/sctp/sctp.h> | 43 | #include <net/sctp/sctp.h> |
44 | #include <linux/crc32c.h> | 44 | #include <linux/crc32c.h> |
45 | #include <linux/crc32.h> | ||
45 | 46 | ||
46 | static inline __u32 sctp_crc32c(__u32 crc, u8 *buffer, u16 length) | 47 | static inline __wsum sctp_csum_update(const void *buff, int len, __wsum sum) |
47 | { | 48 | { |
48 | return crc32c(crc, buffer, length); | 49 | /* This uses the crypto implementation of crc32c, which is either |
49 | } | 50 | * implemented w/ hardware support or resolves to __crc32c_le(). |
50 | |||
51 | static inline __u32 sctp_start_cksum(__u8 *buffer, __u16 length) | ||
52 | { | ||
53 | __u32 crc = ~(__u32)0; | ||
54 | __u8 zero[sizeof(__u32)] = {0}; | ||
55 | |||
56 | /* Optimize this routine to be SCTP specific, knowing how | ||
57 | * to skip the checksum field of the SCTP header. | ||
58 | */ | 51 | */ |
59 | 52 | return crc32c(sum, buff, len); | |
60 | /* Calculate CRC up to the checksum. */ | ||
61 | crc = sctp_crc32c(crc, buffer, sizeof(struct sctphdr) - sizeof(__u32)); | ||
62 | |||
63 | /* Skip checksum field of the header. */ | ||
64 | crc = sctp_crc32c(crc, zero, sizeof(__u32)); | ||
65 | |||
66 | /* Calculate the rest of the CRC. */ | ||
67 | crc = sctp_crc32c(crc, &buffer[sizeof(struct sctphdr)], | ||
68 | length - sizeof(struct sctphdr)); | ||
69 | return crc; | ||
70 | } | ||
71 | |||
72 | static inline __u32 sctp_update_cksum(__u8 *buffer, __u16 length, __u32 crc32) | ||
73 | { | ||
74 | return sctp_crc32c(crc32, buffer, length); | ||
75 | } | 53 | } |
76 | 54 | ||
77 | static inline __le32 sctp_end_cksum(__u32 crc32) | 55 | static inline __wsum sctp_csum_combine(__wsum csum, __wsum csum2, |
56 | int offset, int len) | ||
78 | { | 57 | { |
79 | return cpu_to_le32(~crc32); | 58 | return __crc32c_le_combine(csum, csum2, len); |
80 | } | 59 | } |
81 | 60 | ||
82 | /* Calculate the CRC32C checksum of an SCTP packet. */ | ||
83 | static inline __le32 sctp_compute_cksum(const struct sk_buff *skb, | 61 | static inline __le32 sctp_compute_cksum(const struct sk_buff *skb, |
84 | unsigned int offset) | 62 | unsigned int offset) |
85 | { | 63 | { |
86 | const struct sk_buff *iter; | 64 | struct sctphdr *sh = sctp_hdr(skb); |
65 | __le32 ret, old = sh->checksum; | ||
66 | const struct skb_checksum_ops ops = { | ||
67 | .update = sctp_csum_update, | ||
68 | .combine = sctp_csum_combine, | ||
69 | }; | ||
87 | 70 | ||
88 | __u32 crc32 = sctp_start_cksum(skb->data + offset, | 71 | sh->checksum = 0; |
89 | skb_headlen(skb) - offset); | 72 | ret = cpu_to_le32(~__skb_checksum(skb, offset, skb->len - offset, |
90 | skb_walk_frags(skb, iter) | 73 | ~(__u32)0, &ops)); |
91 | crc32 = sctp_update_cksum((__u8 *) iter->data, | 74 | sh->checksum = old; |
92 | skb_headlen(iter), crc32); | ||
93 | 75 | ||
94 | return sctp_end_cksum(crc32); | 76 | return ret; |
95 | } | 77 | } |
96 | 78 | ||
97 | #endif /* __sctp_checksum_h__ */ | 79 | #endif /* __sctp_checksum_h__ */ |