blob: 8af77829705db9f5b3fefc45c2027d4358e5b54c [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Dirk Behme8b3cc062010-01-03 08:33:58 +01002/*
3 * Generic network code. Moved from net.c
4 *
5 * Copyright 1994 - 2000 Neil Russell.
6 * Copyright 2000 Roland Borde
7 * Copyright 2000 Paolo Scaffardi
8 * Copyright 2000-2002 Wolfgang Denk, wd@denx.de
9 * Copyright 2009 Dirk Behme, dirk.behme@googlemail.com
Dirk Behme8b3cc062010-01-03 08:33:58 +010010 */
11
12#include <common.h>
13
Joe Hershberger5874dec2015-04-08 01:41:01 -050014struct in_addr string_to_ip(const char *s)
Dirk Behme8b3cc062010-01-03 08:33:58 +010015{
Joe Hershberger5874dec2015-04-08 01:41:01 -050016 struct in_addr addr;
Dirk Behme8b3cc062010-01-03 08:33:58 +010017 char *e;
18 int i;
19
Joe Hershberger5874dec2015-04-08 01:41:01 -050020 addr.s_addr = 0;
Dirk Behme8b3cc062010-01-03 08:33:58 +010021 if (s == NULL)
Joe Hershberger5874dec2015-04-08 01:41:01 -050022 return addr;
Dirk Behme8b3cc062010-01-03 08:33:58 +010023
Joe Hershberger5874dec2015-04-08 01:41:01 -050024 for (addr.s_addr = 0, i = 0; i < 4; ++i) {
Dirk Behme8b3cc062010-01-03 08:33:58 +010025 ulong val = s ? simple_strtoul(s, &e, 10) : 0;
Chris Packhame917c362017-01-04 13:36:25 +130026 if (val > 255) {
27 addr.s_addr = 0;
28 return addr;
29 }
Chris Packham06562482017-01-04 13:36:26 +130030 if (i != 3 && *e != '.') {
31 addr.s_addr = 0;
32 return addr;
33 }
Joe Hershberger5874dec2015-04-08 01:41:01 -050034 addr.s_addr <<= 8;
35 addr.s_addr |= (val & 0xFF);
Dirk Behme8b3cc062010-01-03 08:33:58 +010036 if (s) {
37 s = (*e) ? e+1 : e;
38 }
39 }
40
Joe Hershberger5874dec2015-04-08 01:41:01 -050041 addr.s_addr = htonl(addr.s_addr);
42 return addr;
Dirk Behme8b3cc062010-01-03 08:33:58 +010043}
Joe Hershberger8e7545e2019-09-13 19:21:16 -050044
45void string_to_enetaddr(const char *addr, uint8_t *enetaddr)
46{
47 char *end;
48 int i;
49
50 if (!enetaddr)
51 return;
52
53 for (i = 0; i < 6; ++i) {
54 enetaddr[i] = addr ? simple_strtoul(addr, &end, 16) : 0;
55 if (addr)
56 addr = (*end) ? end + 1 : end;
57 }
58}
Simon Glassd70d8d82019-12-06 21:41:39 -070059
60uint compute_ip_checksum(const void *vptr, uint nbytes)
61{
62 int sum, oddbyte;
63 const unsigned short *ptr = vptr;
64
65 sum = 0;
66 while (nbytes > 1) {
67 sum += *ptr++;
68 nbytes -= 2;
69 }
70 if (nbytes == 1) {
71 oddbyte = 0;
72 ((u8 *)&oddbyte)[0] = *(u8 *)ptr;
73 ((u8 *)&oddbyte)[1] = 0;
74 sum += oddbyte;
75 }
76 sum = (sum >> 16) + (sum & 0xffff);
77 sum += (sum >> 16);
78 sum = ~sum & 0xffff;
79
80 return sum;
81}
82
83uint add_ip_checksums(uint offset, uint sum, uint new)
84{
85 ulong checksum;
86
87 sum = ~sum & 0xffff;
88 new = ~new & 0xffff;
89 if (offset & 1) {
90 /*
91 * byte-swap the sum if it came from an odd offset; since the
92 * computation is endian-independent this works.
93 */
94 new = ((new >> 8) & 0xff) | ((new << 8) & 0xff00);
95 }
96 checksum = sum + new;
97 if (checksum > 0xffff)
98 checksum -= 0xffff;
99
100 return (~checksum) & 0xffff;
101}
102
103int ip_checksum_ok(const void *addr, uint nbytes)
104{
105 return !(compute_ip_checksum(addr, nbytes) & 0xfffe);
106}