/* SPDX-License-Identifier: GPL-2.0-only */

#include <stdint.h>
#include <ip_checksum.h>

unsigned long compute_ip_checksum(const void *addr, unsigned long length)
{
	const uint8_t *ptr;
	volatile union {
		uint8_t  byte[2];
		uint16_t word;
	} value;
	unsigned long sum;
	unsigned long i;
	/* In the most straight forward way possible,
	 * compute an ip style checksum.
	 */
	sum = 0;
	ptr = addr;
	for (i = 0; i < length; i++) {
		unsigned long v;
		v = ptr[i];
		if (i & 1)
			v <<= 8;
		/* Add the new value */
		sum += v;
		/* Wrap around the carry */
		if (sum > 0xFFFF)
			sum = (sum + (sum >> 16)) & 0xFFFF;
	}
	value.byte[0] = sum & 0xff;
	value.byte[1] = (sum >> 8) & 0xff;
	return (~value.word) & 0xFFFF;
}

unsigned long add_ip_checksums(unsigned long offset, unsigned long sum,
	unsigned long new)
{
	unsigned long checksum;
	sum = ~sum & 0xFFFF;
	new = ~new & 0xFFFF;
	if (offset & 1) {
		/* byte swap the sum if it came from an odd offset
		 * since the computation is endian independent this
		 * works.
		 */
		new = ((new >> 8) & 0xff) | ((new << 8) & 0xff00);
	}
	checksum = sum + new;
	if (checksum > 0xFFFF)
		checksum -= 0xFFFF;
	return (~checksum) & 0xFFFF;
}