| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216 |
- // SPDX-License-Identifier: GPL-2.0+
- /*
- * Generic network code. Moved from net.c
- *
- * Copyright 1994 - 2000 Neil Russell.
- * Copyright 2000 Roland Borde
- * Copyright 2000 Paolo Scaffardi
- * Copyright 2000-2002 Wolfgang Denk, wd@denx.de
- * Copyright 2009 Dirk Behme, dirk.behme@googlemail.com
- */
- #include <common.h>
- #include <net.h>
- #include <net6.h>
- struct in_addr string_to_ip(const char *s)
- {
- struct in_addr addr;
- char *e;
- int i;
- addr.s_addr = 0;
- if (s == NULL)
- return addr;
- for (addr.s_addr = 0, i = 0; i < 4; ++i) {
- ulong val = s ? dectoul(s, &e) : 0;
- if (val > 255) {
- addr.s_addr = 0;
- return addr;
- }
- if (i != 3 && *e != '.') {
- addr.s_addr = 0;
- return addr;
- }
- addr.s_addr <<= 8;
- addr.s_addr |= (val & 0xFF);
- if (s) {
- s = (*e) ? e+1 : e;
- }
- }
- addr.s_addr = htonl(addr.s_addr);
- return addr;
- }
- #if IS_ENABLED(CONFIG_IPV6)
- int string_to_ip6(const char *str, size_t len, struct in6_addr *addr)
- {
- int colon_count = 0;
- int found_double_colon = 0;
- int xstart = 0; /* first zero (double colon) */
- int section_num = 7; /* num words the double colon represents */
- int i;
- const char *s = str;
- const char *const e = s + len;
- struct in_addr zero_ip = {.s_addr = 0};
- if (!str)
- return -1;
- /* First pass, verify the syntax and locate the double colon */
- while (s < e) {
- while (s < e && isxdigit((int)*s))
- s++;
- if (*s == '\0')
- break;
- if (*s != ':') {
- if (*s == '.' && section_num >= 2) {
- struct in_addr v4;
- while (s != str && *(s - 1) != ':')
- --s;
- v4 = string_to_ip(s);
- if (memcmp(&zero_ip, &v4,
- sizeof(struct in_addr)) != 0) {
- section_num -= 2;
- break;
- }
- }
- /* This could be a valid address */
- break;
- }
- if (s == str) {
- /* The address begins with a colon */
- if (*++s != ':')
- /* Must start with a double colon or a number */
- goto out_err;
- } else {
- s++;
- if (found_double_colon)
- section_num--;
- else
- xstart++;
- }
- if (*s == ':') {
- if (found_double_colon)
- /* Two double colons are not allowed */
- goto out_err;
- found_double_colon = 1;
- section_num -= xstart;
- s++;
- }
- if (++colon_count == 7)
- /* Found all colons */
- break;
- ++s;
- }
- if (colon_count == 0)
- goto out_err;
- if (*--s == ':')
- section_num++;
- /* Second pass, read the address */
- s = str;
- for (i = 0; i < 8; i++) {
- int val = 0;
- char *end;
- if (found_double_colon &&
- i >= xstart && i < xstart + section_num) {
- addr->s6_addr16[i] = 0;
- continue;
- }
- while (*s == ':')
- s++;
- if (i == 6 && isdigit((int)*s)) {
- struct in_addr v4 = string_to_ip(s);
- if (memcmp(&zero_ip, &v4,
- sizeof(struct in_addr)) != 0) {
- /* Ending with :IPv4-address */
- addr->s6_addr32[3] = v4.s_addr;
- break;
- }
- }
- val = simple_strtoul(s, &end, 16);
- if (end != e && *end != '\0' && *end != ':')
- goto out_err;
- addr->s6_addr16[i] = htons(val);
- s = end;
- }
- return 0;
- out_err:
- return -1;
- }
- #endif
- void string_to_enetaddr(const char *addr, uint8_t *enetaddr)
- {
- char *end;
- int i;
- if (!enetaddr)
- return;
- for (i = 0; i < 6; ++i) {
- enetaddr[i] = addr ? hextoul(addr, &end) : 0;
- if (addr)
- addr = (*end) ? end + 1 : end;
- }
- }
- uint compute_ip_checksum(const void *vptr, uint nbytes)
- {
- int sum, oddbyte;
- const unsigned short *ptr = vptr;
- sum = 0;
- while (nbytes > 1) {
- sum += *ptr++;
- nbytes -= 2;
- }
- if (nbytes == 1) {
- oddbyte = 0;
- ((u8 *)&oddbyte)[0] = *(u8 *)ptr;
- ((u8 *)&oddbyte)[1] = 0;
- sum += oddbyte;
- }
- sum = (sum >> 16) + (sum & 0xffff);
- sum += (sum >> 16);
- sum = ~sum & 0xffff;
- return sum;
- }
- uint add_ip_checksums(uint offset, uint sum, uint new)
- {
- ulong 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;
- }
- int ip_checksum_ok(const void *addr, uint nbytes)
- {
- return !(compute_ip_checksum(addr, nbytes) & 0xfffe);
- }
|