aboutgitcodebugslistschat
path: root/checksum.c
diff options
context:
space:
mode:
authorDavid Gibson <david@gibson.dropbear.id.au>2022-10-19 11:43:47 +1100
committerStefano Brivio <sbrivio@redhat.com>2022-10-19 03:34:32 +0200
commitbd4be308fc92ed5d75ea45364a9f837d5c0fb0b2 (patch)
treeeaaaedcbf8e01177e072bb0dd168897a0cbb56ff /checksum.c
parent6905ac75ec6643fbc93963ad9ded4903e8eeebf0 (diff)
downloadpasst-bd4be308fc92ed5d75ea45364a9f837d5c0fb0b2.tar
passt-bd4be308fc92ed5d75ea45364a9f837d5c0fb0b2.tar.gz
passt-bd4be308fc92ed5d75ea45364a9f837d5c0fb0b2.tar.bz2
passt-bd4be308fc92ed5d75ea45364a9f837d5c0fb0b2.tar.lz
passt-bd4be308fc92ed5d75ea45364a9f837d5c0fb0b2.tar.xz
passt-bd4be308fc92ed5d75ea45364a9f837d5c0fb0b2.tar.zst
passt-bd4be308fc92ed5d75ea45364a9f837d5c0fb0b2.zip
Add csum_udp4() helper for calculating UDP over IPv4 checksums
At least two places in passt fill in UDP over IPv4 checksums, although since UDP checksums are optional with IPv4 that just amounts to storing a 0 (in tap_ip_send()) or leaving a 0 from an earlier initialization (in dhcp()). For consistency, add a helper for this "calculation". Just for the heck of it, add the option (compile time disabled for now) to calculate real UDP checksums. Signed-off-by: David Gibson <david@gibson.dropbear.id.au> Signed-off-by: Stefano Brivio <sbrivio@redhat.com>
Diffstat (limited to 'checksum.c')
-rw-r--r--checksum.c33
1 files changed, 33 insertions, 0 deletions
diff --git a/checksum.c b/checksum.c
index 175381d..cf6fc31 100644
--- a/checksum.c
+++ b/checksum.c
@@ -56,6 +56,12 @@
#include <linux/icmp.h>
#include <linux/icmpv6.h>
+/* Checksums are optional for UDP over IPv4, so we usually just set
+ * them to 0. Change this to 1 to calculate real UDP over IPv4
+ * checksums
+ */
+#define UDP4_REAL_CHECKSUMS 0
+
/**
* sum_16b() - Calculate sum of 16-bit words
* @buf: Input buffer
@@ -110,6 +116,33 @@ uint16_t csum_unaligned(const void *buf, size_t len, uint32_t init)
}
/**
+ * csum_udp4() - Calculate and set checksum for a UDP over IPv4 packet
+ * @udp4hr: UDP header, initialised apart from checksum
+ * @saddr: IPv4 source address
+ * @daddr: IPv4 destination address
+ * @payload: ICMPv4 packet payload
+ * @len: Length of @payload (not including UDP)
+ */
+void csum_udp4(struct udphdr *udp4hr, in_addr_t saddr, in_addr_t daddr,
+ const void *payload, size_t len)
+{
+ /* UDP checksums are optional, so don't bother */
+ udp4hr->check = 0;
+
+ if (UDP4_REAL_CHECKSUMS) {
+ /* UNTESTED: if we did want real UDPv4 checksums, this
+ * is roughly what we'd need */
+ uint32_t psum = csum_fold(htonl(saddr))
+ + csum_fold(htonl(daddr))
+ + htons(len + sizeof(*udp4hr))
+ + htons(IPPROTO_UDP);
+ /* Add in partial checksum for the UDP header alone */
+ psum += sum_16b(udp4hr, sizeof(*udp4hr));
+ udp4hr->check = csum_unaligned(payload, len, psum);
+ }
+}
+
+/**
* csum_icmp4() - Calculate and set checksum for an ICMP packet
* @icmp4hr: ICMP header, initialised apart from checksum
* @payload: ICMP packet payload