diff options
| -rw-r--r-- | include/net/sctp/checksum.h | 56 | ||||
| -rw-r--r-- | net/sctp/output.c | 9 |
2 files changed, 20 insertions, 45 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__ */ |
diff --git a/net/sctp/output.c b/net/sctp/output.c index 319137340d15..e650978daf27 100644 --- a/net/sctp/output.c +++ b/net/sctp/output.c | |||
| @@ -390,7 +390,6 @@ int sctp_packet_transmit(struct sctp_packet *packet) | |||
| 390 | __u8 has_data = 0; | 390 | __u8 has_data = 0; |
| 391 | struct dst_entry *dst = tp->dst; | 391 | struct dst_entry *dst = tp->dst; |
| 392 | unsigned char *auth = NULL; /* pointer to auth in skb data */ | 392 | unsigned char *auth = NULL; /* pointer to auth in skb data */ |
| 393 | __u32 cksum_buf_len = sizeof(struct sctphdr); | ||
| 394 | 393 | ||
| 395 | pr_debug("%s: packet:%p\n", __func__, packet); | 394 | pr_debug("%s: packet:%p\n", __func__, packet); |
| 396 | 395 | ||
| @@ -493,7 +492,6 @@ int sctp_packet_transmit(struct sctp_packet *packet) | |||
| 493 | if (chunk == packet->auth) | 492 | if (chunk == packet->auth) |
| 494 | auth = skb_tail_pointer(nskb); | 493 | auth = skb_tail_pointer(nskb); |
| 495 | 494 | ||
| 496 | cksum_buf_len += chunk->skb->len; | ||
| 497 | memcpy(skb_put(nskb, chunk->skb->len), | 495 | memcpy(skb_put(nskb, chunk->skb->len), |
| 498 | chunk->skb->data, chunk->skb->len); | 496 | chunk->skb->data, chunk->skb->len); |
| 499 | 497 | ||
| @@ -538,12 +536,7 @@ int sctp_packet_transmit(struct sctp_packet *packet) | |||
| 538 | if (!sctp_checksum_disable) { | 536 | if (!sctp_checksum_disable) { |
| 539 | if (!(dst->dev->features & NETIF_F_SCTP_CSUM) || | 537 | if (!(dst->dev->features & NETIF_F_SCTP_CSUM) || |
| 540 | (dst_xfrm(dst) != NULL) || packet->ipfragok) { | 538 | (dst_xfrm(dst) != NULL) || packet->ipfragok) { |
| 541 | __u32 crc32 = sctp_start_cksum((__u8 *)sh, cksum_buf_len); | 539 | sh->checksum = sctp_compute_cksum(nskb, 0); |
| 542 | |||
| 543 | /* 3) Put the resultant value into the checksum field in the | ||
| 544 | * common header, and leave the rest of the bits unchanged. | ||
| 545 | */ | ||
| 546 | sh->checksum = sctp_end_cksum(crc32); | ||
| 547 | } else { | 540 | } else { |
| 548 | /* no need to seed pseudo checksum for SCTP */ | 541 | /* no need to seed pseudo checksum for SCTP */ |
| 549 | nskb->ip_summed = CHECKSUM_PARTIAL; | 542 | nskb->ip_summed = CHECKSUM_PARTIAL; |
