]> Git Repo - u-boot.git/blobdiff - lib/net_utils.c
list: use list_count_nodes() to count list entries
[u-boot.git] / lib / net_utils.c
index cfae842752414ebb7a24d3ed5229245a114e02da..c70fef0d991bcbd497b77a675b0ab4158fdbdf36 100644 (file)
@@ -1,3 +1,4 @@
+// SPDX-License-Identifier: GPL-2.0+
 /*
  * Generic network code. Moved from net.c
  *
@@ -6,11 +7,11 @@
  * Copyright 2000 Paolo Scaffardi
  * Copyright 2000-2002 Wolfgang Denk, [email protected]
  * Copyright 2009 Dirk Behme, [email protected]
- *
- * SPDX-License-Identifier:    GPL-2.0+
  */
 
-#include <common.h>
+#include <net.h>
+#include <net6.h>
+#include <vsprintf.h>
 
 struct in_addr string_to_ip(const char *s)
 {
@@ -23,7 +24,15 @@ struct in_addr string_to_ip(const char *s)
                return addr;
 
        for (addr.s_addr = 0, i = 0; i < 4; ++i) {
-               ulong val = s ? simple_strtoul(s, &e, 10) : 0;
+               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) {
@@ -34,3 +43,174 @@ struct in_addr string_to_ip(const char *s)
        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);
+}
This page took 0.031076 seconds and 4 git commands to generate.