]> Git Repo - J-u-boot.git/blame - net/net.c
include: Use CONFIG_XPL_BUILD instead of CONFIG_SPL_BUILD
[J-u-boot.git] / net / net.c
CommitLineData
f739fcd8 1// SPDX-License-Identifier: GPL-2.0
2d966958
WD
2/*
3 * Copied from Linux Monitor (LiMon) - Networking.
4 *
5 * Copyright 1994 - 2000 Neil Russell.
6 * (See License)
7 * Copyright 2000 Roland Borde
8 * Copyright 2000 Paolo Scaffardi
9 * Copyright 2000-2002 Wolfgang Denk, [email protected]
10 */
11
12/*
13 * General Desription:
14 *
15 * The user interface supports commands for BOOTP, RARP, and TFTP.
16 * Also, we support ARP internally. Depending on available data,
17 * these interact as follows:
18 *
19 * BOOTP:
20 *
21 * Prerequisites: - own ethernet address
22 * We want: - own IP address
23 * - TFTP server IP address
24 * - name of bootfile
25 * Next step: ARP
26 *
6de98b60 27 * LINKLOCAL:
d22c338e
JH
28 *
29 * Prerequisites: - own ethernet address
30 * We want: - own IP address
31 * Next step: ARP
32 *
2d966958
WD
33 * RARP:
34 *
35 * Prerequisites: - own ethernet address
36 * We want: - own IP address
37 * - TFTP server IP address
38 * Next step: ARP
39 *
40 * ARP:
41 *
42 * Prerequisites: - own ethernet address
43 * - own IP address
44 * - TFTP server IP address
45 * We want: - TFTP server ethernet address
46 * Next step: TFTP
47 *
48 * DHCP:
49 *
b2f50807
WD
50 * Prerequisites: - own ethernet address
51 * We want: - IP, Netmask, ServerIP, Gateway IP
52 * - bootfilename, lease time
53 * Next step: - TFTP
2d966958
WD
54 *
55 * TFTP:
56 *
57 * Prerequisites: - own ethernet address
58 * - own IP address
59 * - TFTP server IP address
60 * - TFTP server ethernet address
61 * - name of bootfile (if unknown, we use a default name
62 * derived from our own IP address)
63 * We want: - load the boot file
64 * Next step: none
cbd8a35c
WD
65 *
66 * NFS:
67 *
68 * Prerequisites: - own ethernet address
69 * - own IP address
70 * - name of bootfile (if unknown, we use a default name
71 * derived from our own IP address)
72 * We want: - load the boot file
73 * Next step: none
ea287deb 74 *
d8970dae
LF
75 *
76 * WOL:
77 *
78 * Prerequisites: - own ethernet address
79 * We want: - magic packet or timeout
80 * Next step: none
2d966958 81 */
2d966958 82
52f24238 83#include <bootstage.h>
2d966958 84#include <command.h>
24b852a7 85#include <console.h>
c7694dd4 86#include <env.h>
f3998fdc 87#include <env_internal.h>
60304592 88#include <errno.h>
8e8ccfe1 89#include <image.h>
f7ae49fc 90#include <log.h>
2d966958 91#include <net.h>
ffdbf3ba
VM
92#include <net6.h>
93#include <ndisc.h>
443d3191
DM
94#include <net/fastboot_udp.h>
95#include <net/fastboot_tcp.h>
34696958 96#include <net/tftp.h>
09bd3d0b 97#include <net/ncsi.h>
3eaac630
RF
98#if defined(CONFIG_CMD_PCAP)
99#include <net/pcap.h>
100#endif
b43ea1bf 101#include <net/udp.h>
2d8d190c 102#if defined(CONFIG_LED_STATUS)
fc3e2165 103#include <miiphy.h>
4545f4e6 104#include <status_led.h>
fc3e2165 105#endif
4545f4e6
JH
106#include <watchdog.h>
107#include <linux/compiler.h>
f43b2df3 108#include <test/test.h>
a0245818
SE
109#include <net/tcp.h>
110#include <net/wget.h>
4545f4e6
JH
111#include "arp.h"
112#include "bootp.h"
f575ae1f 113#include "cdp.h"
1a32bf41
RG
114#if defined(CONFIG_CMD_DNS)
115#include "dns.h"
116#endif
d22c338e 117#include "link_local.h"
4545f4e6 118#include "nfs.h"
a36b12f9 119#include "ping.h"
4545f4e6 120#include "rarp.h"
d8970dae
LF
121#if defined(CONFIG_CMD_WOL)
122#include "wol.h"
123#endif
a0245818 124#include "dhcpv6.h"
6de98b60 125#include "net_rand.h"
2d966958 126
2d966958
WD
127/** BOOTP EXTENTIONS **/
128
3e38e429 129/* Our subnet mask (0=unknown) */
049a95a7 130struct in_addr net_netmask;
3e38e429 131/* Our gateways IP address */
049a95a7 132struct in_addr net_gateway;
3e38e429 133/* Our DNS IP address */
049a95a7 134struct in_addr net_dns_server;
1fe80d79 135#if defined(CONFIG_BOOTP_DNS2)
3e38e429 136/* Our 2nd DNS IP address */
049a95a7 137struct in_addr net_dns_server2;
3e38e429 138#endif
a0245818
SE
139/* Indicates whether the pxe path prefix / config file was specified in dhcp option */
140char *pxelinux_configfile;
2d966958
WD
141
142/** END OF BOOTP EXTENTIONS **/
143
3e38e429 144/* Our ethernet address */
0adb5b76 145u8 net_ethaddr[6];
3e38e429 146/* Boot server enet address */
0adb5b76 147u8 net_server_ethaddr[6];
3e38e429 148/* Our IP addr (0 = unknown) */
049a95a7 149struct in_addr net_ip;
3e38e429 150/* Server IP addr (0 = unknown) */
049a95a7 151struct in_addr net_server_ip;
3e38e429 152/* Current receive packet */
1203fcce 153uchar *net_rx_packet;
3e38e429 154/* Current rx packet length */
1203fcce 155int net_rx_packet_len;
3e38e429 156/* IP packet ID */
bc0571fc 157static unsigned net_ip_id;
3e38e429 158/* Ethernet bcast address */
0adb5b76
JH
159const u8 net_bcast_ethaddr[6] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
160const u8 net_null_ethaddr[6];
0efe1bcf 161#if defined(CONFIG_API) || defined(CONFIG_EFI_LOADER)
bc0571fc 162void (*push_packet)(void *, int len) = 0;
f85b6071 163#endif
3e38e429 164/* Network loop state */
22f6e99d 165enum net_loop_state net_state;
3e38e429 166/* Tried all network devices */
bc0571fc 167int net_restart_wrap;
3e38e429 168/* Network loop restarted */
bc0571fc 169static int net_restarted;
3e38e429 170/* At least one device configured */
bc0571fc 171static int net_dev_exists;
2d966958 172
6e592385 173/* XXX in both little & big endian machines 0xFFFF == ntohs(-1) */
3e38e429 174/* default is without VLAN */
4fd5055f 175ushort net_our_vlan = 0xFFFF;
3e38e429 176/* ditto */
4fd5055f 177ushort net_native_vlan = 0xFFFF;
a3d991bd 178
3e38e429 179/* Boot File name */
11a69ff8 180char net_boot_file_name[1024];
449312c1
AG
181/* Indicates whether the file name was specified on the command line */
182bool net_boot_file_name_explicit;
1411157d
JH
183/* The actual transferred size of the bootfile (in bytes) */
184u32 net_boot_file_size;
185/* Boot file size in blocks as reported by the DHCP server */
186u32 net_boot_file_expected_size_in_blocks;
2d966958 187
1203fcce 188static uchar net_pkt_buf[(PKTBUFSRX+1) * PKTSIZE_ALIGN + PKTALIGN];
2a504df0
JH
189/* Receive packets */
190uchar *net_rx_packets[PKTBUFSRX];
ece223b5
JH
191/* Current UDP RX packet handler */
192static rxhand_f *udp_packet_handler;
193/* Current ARP RX packet handler */
194static rxhand_f *arp_packet_handler;
39bccd21 195#ifdef CONFIG_CMD_TFTPPUT
ece223b5
JH
196/* Current ICMP rx handler */
197static rxhand_icmp_f *packet_icmp_handler;
39bccd21 198#endif
3e38e429 199/* Current timeout handler */
bc0571fc 200static thand_f *time_handler;
3e38e429 201/* Time base value */
bc0571fc 202static ulong time_start;
3e38e429 203/* Current timeout value */
bc0571fc 204static ulong time_delta;
3e38e429 205/* THE transmit packet */
1203fcce 206uchar *net_tx_packet;
2d966958 207
e4bf0c5c 208static int net_check_prereq(enum proto_t protocol);
2d966958 209
bc0571fc 210static int net_try_count;
67b96e87 211
b63056d6
JL
212int __maybe_unused net_busy_flag;
213
73a8b27c
WD
214/**********************************************************************/
215
fd305633
JH
216static int on_ipaddr(const char *name, const char *value, enum env_op op,
217 int flags)
218{
219 if (flags & H_PROGRAMMATIC)
220 return 0;
221
222 net_ip = string_to_ip(value);
223
224 return 0;
225}
226U_BOOT_ENV_CALLBACK(ipaddr, on_ipaddr);
227
228static int on_gatewayip(const char *name, const char *value, enum env_op op,
229 int flags)
230{
231 if (flags & H_PROGRAMMATIC)
232 return 0;
233
234 net_gateway = string_to_ip(value);
235
236 return 0;
237}
238U_BOOT_ENV_CALLBACK(gatewayip, on_gatewayip);
239
240static int on_netmask(const char *name, const char *value, enum env_op op,
241 int flags)
242{
243 if (flags & H_PROGRAMMATIC)
244 return 0;
245
246 net_netmask = string_to_ip(value);
247
248 return 0;
249}
250U_BOOT_ENV_CALLBACK(netmask, on_netmask);
251
252static int on_serverip(const char *name, const char *value, enum env_op op,
253 int flags)
254{
255 if (flags & H_PROGRAMMATIC)
256 return 0;
257
258 net_server_ip = string_to_ip(value);
259
260 return 0;
261}
262U_BOOT_ENV_CALLBACK(serverip, on_serverip);
263
264static int on_nvlan(const char *name, const char *value, enum env_op op,
265 int flags)
266{
267 if (flags & H_PROGRAMMATIC)
268 return 0;
269
270 net_native_vlan = string_to_vlan(value);
271
272 return 0;
273}
274U_BOOT_ENV_CALLBACK(nvlan, on_nvlan);
275
276static int on_vlan(const char *name, const char *value, enum env_op op,
277 int flags)
278{
279 if (flags & H_PROGRAMMATIC)
280 return 0;
281
282 net_our_vlan = string_to_vlan(value);
283
284 return 0;
285}
286U_BOOT_ENV_CALLBACK(vlan, on_vlan);
287
288#if defined(CONFIG_CMD_DNS)
289static int on_dnsip(const char *name, const char *value, enum env_op op,
290 int flags)
291{
292 if (flags & H_PROGRAMMATIC)
293 return 0;
294
295 net_dns_server = string_to_ip(value);
296
297 return 0;
298}
299U_BOOT_ENV_CALLBACK(dnsip, on_dnsip);
300#endif
301
e4a3d57d
SG
302/*
303 * Check if autoload is enabled. If so, use either NFS or TFTP to download
304 * the boot file.
305 */
306void net_auto_load(void)
307{
a6ab4b54 308#if defined(CONFIG_CMD_NFS) && !defined(CONFIG_SPL_BUILD)
00caae6d 309 const char *s = env_get("autoload");
e4a3d57d 310
ec8a252c 311 if (s != NULL && strcmp(s, "NFS") == 0) {
3855cad6
JH
312 if (net_check_prereq(NFS)) {
313/* We aren't expecting to get a serverip, so just accept the assigned IP */
43407539
SG
314 if (IS_ENABLED(CONFIG_BOOTP_SERVERIP)) {
315 net_set_state(NETLOOP_SUCCESS);
316 } else {
317 printf("Cannot autoload with NFS\n");
318 net_set_state(NETLOOP_FAIL);
319 }
3855cad6
JH
320 return;
321 }
ec8a252c
JH
322 /*
323 * Use NFS to load the bootfile.
324 */
68c76a3a 325 nfs_start();
ec8a252c
JH
326 return;
327 }
e4a3d57d 328#endif
bfebc8c9 329 if (env_get_yesno("autoload") == 0) {
ec8a252c
JH
330 /*
331 * Just use BOOTP/RARP to configure system;
332 * Do not use TFTP to load the bootfile.
333 */
334 net_set_state(NETLOOP_SUCCESS);
335 return;
e4a3d57d 336 }
278e9ac8
JF
337 if (IS_ENABLED(CONFIG_CMD_TFTPBOOT)) {
338 if (net_check_prereq(TFTPGET)) {
339 /*
340 * We aren't expecting to get a serverip, so just
341 * accept the assigned IP
342 */
343 if (IS_ENABLED(CONFIG_BOOTP_SERVERIP)) {
344 net_set_state(NETLOOP_SUCCESS);
345 } else {
346 printf("Cannot autoload with TFTPGET\n");
347 net_set_state(NETLOOP_FAIL);
348 }
349 return;
43407539 350 }
278e9ac8 351 tftp_start(TFTPGET);
3855cad6 352 }
e4a3d57d
SG
353}
354
c3f0278e 355static int net_init_loop(void)
2f70c49e 356{
6de98b60
EM
357 static bool first_call = true;
358
ffdbf3ba 359 if (eth_get_dev()) {
0adb5b76 360 memcpy(net_ethaddr, eth_get_ethaddr(), 6);
ffdbf3ba
VM
361
362 if (IS_ENABLED(CONFIG_IPV6)) {
363 ip6_make_lladdr(&net_link_local_ip6, net_ethaddr);
364 if (!memcmp(&net_ip6, &net_null_addr_ip6,
365 sizeof(struct in6_addr)))
366 memcpy(&net_ip6, &net_link_local_ip6,
367 sizeof(struct in6_addr));
368 }
369 }
c3f0278e
SA
370 else
371 /*
372 * Not ideal, but there's no way to get the actual error, and I
373 * don't feel like fixing all the users of eth_get_dev to deal
374 * with errors.
375 */
376 return -ENONET;
3c172c4f 377
6de98b60
EM
378 if (IS_ENABLED(CONFIG_IPV6_ROUTER_DISCOVERY))
379 if (first_call && use_ip6) {
380 first_call = false;
381 srand_mac(); /* This is for rand used in ip6_send_rs. */
382 net_loop(RS);
383 }
c3f0278e 384 return 0;
2f70c49e
HS
385}
386
ece223b5
JH
387static void net_clear_handlers(void)
388{
389 net_set_udp_handler(NULL);
390 net_set_arp_handler(NULL);
bc0571fc 391 net_set_timeout_handler(0, NULL);
ece223b5
JH
392}
393
394static void net_cleanup_loop(void)
395{
396 net_clear_handlers();
397}
398
c3f0278e 399int net_init(void)
46c495d5
JH
400{
401 static int first_call = 1;
402
403 if (first_call) {
404 /*
405 * Setup packet buffers, aligned correctly.
406 */
407 int i;
408
1203fcce
JH
409 net_tx_packet = &net_pkt_buf[0] + (PKTALIGN - 1);
410 net_tx_packet -= (ulong)net_tx_packet % PKTALIGN;
2a504df0 411 for (i = 0; i < PKTBUFSRX; i++) {
1203fcce
JH
412 net_rx_packets[i] = net_tx_packet +
413 (i + 1) * PKTSIZE_ALIGN;
2a504df0 414 }
85d25e0e 415 arp_init();
ffdbf3ba 416 ndisc_init();
46c495d5
JH
417 net_clear_handlers();
418
419 /* Only need to setup buffer pointers once. */
420 first_call = 0;
a3bf193b
YCLP
421 if (IS_ENABLED(CONFIG_PROT_TCP))
422 tcp_set_tcp_state(TCP_CLOSED);
46c495d5
JH
423 }
424
c3f0278e 425 return net_init_loop();
46c495d5
JH
426}
427
2d966958
WD
428/**********************************************************************/
429/*
430 * Main network processing loop.
431 */
432
bc0571fc 433int net_loop(enum proto_t protocol)
2d966958 434{
60304592 435 int ret = -EINVAL;
60177b26 436 enum net_loop_state prev_net_state = net_state;
2d966958 437
def7a5c0
MS
438#if defined(CONFIG_CMD_PING)
439 if (protocol != PING)
440 net_ping_ip.s_addr = 0;
441#endif
bc0571fc
JH
442 net_restarted = 0;
443 net_dev_exists = 0;
444 net_try_count = 1;
445 debug_cond(DEBUG_INT_STATE, "--- net_loop Entry\n");
73a8b27c 446
09bd3d0b
SMJ
447#ifdef CONFIG_PHY_NCSI
448 if (phy_interface_is_ncsi() && protocol != NCSI && !ncsi_active()) {
449 printf("%s: configuring NCSI first\n", __func__);
450 if (net_loop(NCSI) < 0)
451 return ret;
452 eth_init_state_only();
453 goto restart;
454 }
455#endif
456
573f14fe 457 bootstage_mark_name(BOOTSTAGE_ID_ETH_START, "eth_start");
46c495d5 458 net_init();
d9506cd4 459 if (eth_is_on_demand_init()) {
b1bf6f2c 460 eth_halt();
f8be7d65 461 eth_set_current();
60304592
JH
462 ret = eth_init();
463 if (ret < 0) {
f8be7d65 464 eth_halt();
60304592 465 return ret;
f8be7d65 466 }
bc0571fc 467 } else {
d2eaec60 468 eth_init_state_only();
bc0571fc 469 }
4b290d4a 470
2d966958 471restart:
b63056d6
JL
472#ifdef CONFIG_USB_KEYBOARD
473 net_busy_flag = 0;
474#endif
22f6e99d 475 net_set_state(NETLOOP_CONTINUE);
2d966958
WD
476
477 /*
478 * Start the ball rolling with the given start function. From
479 * here on, this code is a state machine driven by received
480 * packets and timer events.
481 */
bc0571fc
JH
482 debug_cond(DEBUG_INT_STATE, "--- net_loop Init\n");
483 net_init_loop();
2d966958 484
f43b2df3
SG
485 if (!test_eth_enabled())
486 return 0;
487
4f63acd0 488 switch (net_check_prereq(protocol)) {
2d966958
WD
489 case 1:
490 /* network not configured */
b1bf6f2c 491 eth_halt();
60177b26 492 net_set_state(prev_net_state);
60304592 493 return -ENODEV;
2d966958 494
2d966958
WD
495 case 2:
496 /* network device not configured */
497 break;
2d966958
WD
498
499 case 0:
bc0571fc 500 net_dev_exists = 1;
1411157d 501 net_boot_file_size = 0;
2d966958 502 switch (protocol) {
808f13d8 503#ifdef CONFIG_CMD_TFTPBOOT
e4bf0c5c 504 case TFTPGET:
1fb7cd49
SG
505#ifdef CONFIG_CMD_TFTPPUT
506 case TFTPPUT:
507#endif
2d966958 508 /* always use ARP to get server ethernet address */
8885c5fe 509 tftp_start(protocol);
2d966958 510 break;
808f13d8 511#endif
7a83af07
LC
512#ifdef CONFIG_CMD_TFTPSRV
513 case TFTPSRV:
8885c5fe 514 tftp_start_server();
7a83af07
LC
515 break;
516#endif
7d04986a 517#if CONFIG_IS_ENABLED(UDP_FUNCTION_FASTBOOT)
443d3191
DM
518 case FASTBOOT_UDP:
519 fastboot_udp_start_server();
520 break;
521#endif
7d04986a 522#if CONFIG_IS_ENABLED(TCP_FUNCTION_FASTBOOT)
443d3191
DM
523 case FASTBOOT_TCP:
524 fastboot_tcp_start_server();
f73a7df9
AK
525 break;
526#endif
643d1ab2 527#if defined(CONFIG_CMD_DHCP)
2d966958 528 case DHCP:
7044c6bb 529 bootp_reset();
049a95a7 530 net_ip.s_addr = 0;
7044c6bb 531 dhcp_request(); /* Basically same as BOOTP */
2d966958 532 break;
610f2e9c 533#endif
a0245818
SE
534 case DHCP6:
535 if (IS_ENABLED(CONFIG_CMD_DHCP6))
536 dhcp6_start();
537 break;
808f13d8 538#if defined(CONFIG_CMD_BOOTP)
2d966958 539 case BOOTP:
7044c6bb 540 bootp_reset();
049a95a7 541 net_ip.s_addr = 0;
7044c6bb 542 bootp_request();
2d966958 543 break;
808f13d8 544#endif
bf6cb247 545#if defined(CONFIG_CMD_RARP)
2d966958 546 case RARP:
698d78e5 547 rarp_try = 0;
049a95a7 548 net_ip.s_addr = 0;
698d78e5 549 rarp_request();
2d966958 550 break;
bf6cb247 551#endif
643d1ab2 552#if defined(CONFIG_CMD_PING)
73a8b27c 553 case PING:
a36b12f9 554 ping_start();
73a8b27c 555 break;
cbd8a35c 556#endif
eeb0a2c6
VM
557#if defined(CONFIG_CMD_PING6)
558 case PING6:
559 ping6_start();
560 break;
561#endif
a6ab4b54 562#if defined(CONFIG_CMD_NFS) && !defined(CONFIG_SPL_BUILD)
cbd8a35c 563 case NFS:
68c76a3a 564 nfs_start();
cbd8a35c 565 break;
a3d991bd 566#endif
cfbae482
YCLP
567#if defined(CONFIG_CMD_WGET)
568 case WGET:
569 wget_start();
570 break;
571#endif
643d1ab2 572#if defined(CONFIG_CMD_CDP)
a3d991bd 573 case CDP:
6aede5b7 574 cdp_start();
a3d991bd 575 break;
68ceb29e 576#endif
66c89ee3 577#if defined(CONFIG_NETCONSOLE) && !defined(CONFIG_SPL_BUILD)
68ceb29e 578 case NETCONS:
6a38a5f3 579 nc_start();
68ceb29e 580 break;
ea287deb 581#endif
1a32bf41
RG
582#if defined(CONFIG_CMD_DNS)
583 case DNS:
786eac5f 584 dns_start();
1a32bf41 585 break;
d22c338e
JH
586#endif
587#if defined(CONFIG_CMD_LINK_LOCAL)
588 case LINKLOCAL:
589 link_local_start();
590 break;
d8970dae
LF
591#endif
592#if defined(CONFIG_CMD_WOL)
593 case WOL:
594 wol_start();
595 break;
09bd3d0b
SMJ
596#endif
597#if defined(CONFIG_PHY_NCSI)
598 case NCSI:
599 ncsi_probe_packages();
600 break;
73a8b27c 601#endif
6de98b60
EM
602 case RS:
603 if (IS_ENABLED(CONFIG_IPV6_ROUTER_DISCOVERY))
604 ip6_send_rs();
605 break;
2d966958
WD
606 default:
607 break;
608 }
609
b43ea1bf
PR
610 if (IS_ENABLED(CONFIG_PROT_UDP) && protocol == UDP)
611 udp_start();
612
2d966958
WD
613 break;
614 }
615
643d1ab2 616#if defined(CONFIG_MII) || defined(CONFIG_CMD_MII)
3e38e429 617#if defined(CONFIG_SYS_FAULT_ECHO_LINK_DOWN) && \
2d8d190c
UM
618 defined(CONFIG_LED_STATUS) && \
619 defined(CONFIG_LED_STATUS_RED)
fc3e2165 620 /*
42d1f039 621 * Echo the inverted link state to the fault LED.
fc3e2165 622 */
d3c65b01 623 if (miiphy_link(eth_get_dev()->name, CONFIG_SYS_FAULT_MII_ADDR))
2d8d190c 624 status_led_set(CONFIG_LED_STATUS_RED, CONFIG_LED_STATUS_OFF);
d3c65b01 625 else
2d8d190c 626 status_led_set(CONFIG_LED_STATUS_RED, CONFIG_LED_STATUS_ON);
6d0f6bcf 627#endif /* CONFIG_SYS_FAULT_ECHO_LINK_DOWN, ... */
fc3e2165 628#endif /* CONFIG_MII, ... */
b63056d6
JL
629#ifdef CONFIG_USB_KEYBOARD
630 net_busy_flag = 1;
631#endif
2d966958
WD
632
633 /*
634 * Main packet reception loop. Loop receiving packets until
22f6e99d 635 * someone sets `net_state' to a state that terminates.
2d966958
WD
636 */
637 for (;;) {
29caf930 638 schedule();
c5a75339
JH
639 if (arp_timeout_check() > 0)
640 time_start = get_timer(0);
641
ffdbf3ba
VM
642 if (IS_ENABLED(CONFIG_IPV6)) {
643 if (use_ip6 && (ndisc_timeout_check() > 0))
644 time_start = get_timer(0);
645 }
646
2d966958
WD
647 /*
648 * Check the ethernet for a new packet. The ethernet
649 * receive routine will process it.
60304592
JH
650 * Most drivers return the most recent packet size, but not
651 * errors that may have happened.
2d966958 652 */
40cb90ee 653 eth_rx();
2d966958
WD
654
655 /*
656 * Abort if ctrl-c was pressed.
657 */
658 if (ctrlc()) {
e94070c4 659 /* cancel any ARP that may not have completed */
049a95a7 660 net_arp_wait_packet_ip.s_addr = 0;
e94070c4 661
ece223b5 662 net_cleanup_loop();
8bde7f77 663 eth_halt();
f8be7d65
JH
664 /* Invalidate the last protocol */
665 eth_set_last_protocol(BOOTP);
666
4f63acd0 667 puts("\nAbort\n");
4ef8d53c
JH
668 /* include a debug print as well incase the debug
669 messages are directed to stderr */
bc0571fc 670 debug_cond(DEBUG_INT_STATE, "--- net_loop Abort!\n");
19a4fbaa 671 ret = -EINTR;
4793ee65 672 goto done;
2d966958
WD
673 }
674
2d966958
WD
675 /*
676 * Check for a timeout, and run the timeout handler
677 * if we have one.
678 */
bc0571fc
JH
679 if (time_handler &&
680 ((get_timer(0) - time_start) > time_delta)) {
2d966958
WD
681 thand_f *x;
682
643d1ab2 683#if defined(CONFIG_MII) || defined(CONFIG_CMD_MII)
4f63acd0 684#if defined(CONFIG_SYS_FAULT_ECHO_LINK_DOWN) && \
2d8d190c
UM
685 defined(CONFIG_LED_STATUS) && \
686 defined(CONFIG_LED_STATUS_RED)
fc3e2165 687 /*
42d1f039 688 * Echo the inverted link state to the fault LED.
fc3e2165 689 */
4f63acd0 690 if (miiphy_link(eth_get_dev()->name,
bc0571fc 691 CONFIG_SYS_FAULT_MII_ADDR))
2d8d190c
UM
692 status_led_set(CONFIG_LED_STATUS_RED,
693 CONFIG_LED_STATUS_OFF);
bc0571fc 694 else
2d8d190c
UM
695 status_led_set(CONFIG_LED_STATUS_RED,
696 CONFIG_LED_STATUS_ON);
4f63acd0 697#endif /* CONFIG_SYS_FAULT_ECHO_LINK_DOWN, ... */
fc3e2165 698#endif /* CONFIG_MII, ... */
bc0571fc
JH
699 debug_cond(DEBUG_INT_STATE, "--- net_loop timeout\n");
700 x = time_handler;
701 time_handler = (thand_f *)0;
2d966958 702 (*x)();
6de98b60
EM
703 } else if (IS_ENABLED(CONFIG_IPV6_ROUTER_DISCOVERY))
704 if (time_handler && protocol == RS)
705 if (!ip6_is_unspecified_addr(&net_gateway6) &&
706 net_prefix_length != 0) {
707 net_set_state(NETLOOP_SUCCESS);
708 net_set_timeout_handler(0, NULL);
709 }
2d966958 710
5c421331 711 if (net_state == NETLOOP_FAIL)
bc0571fc 712 ret = net_start_again();
2d966958 713
22f6e99d 714 switch (net_state) {
2d966958 715 case NETLOOP_RESTART:
bc0571fc 716 net_restarted = 1;
2d966958
WD
717 goto restart;
718
719 case NETLOOP_SUCCESS:
ece223b5 720 net_cleanup_loop();
1411157d 721 if (net_boot_file_size > 0) {
8f911a7b 722 printf("Bytes transferred = %u (%x hex)\n",
1411157d 723 net_boot_file_size, net_boot_file_size);
018f5303 724 env_set_hex("filesize", net_boot_file_size);
bb872dd9 725 env_set_hex("fileaddr", image_load_addr);
2d966958 726 }
09bd3d0b 727 if (protocol != NETCONS && protocol != NCSI)
f8be7d65
JH
728 eth_halt();
729 else
730 eth_halt_state_only();
731
732 eth_set_last_protocol(protocol);
733
1411157d 734 ret = net_boot_file_size;
bc0571fc 735 debug_cond(DEBUG_INT_STATE, "--- net_loop Success!\n");
4793ee65 736 goto done;
2d966958
WD
737
738 case NETLOOP_FAIL:
ece223b5 739 net_cleanup_loop();
f8be7d65
JH
740 /* Invalidate the last protocol */
741 eth_set_last_protocol(BOOTP);
bc0571fc 742 debug_cond(DEBUG_INT_STATE, "--- net_loop Fail!\n");
a735e6e9 743 ret = -ENONET;
4793ee65 744 goto done;
22f6e99d
JH
745
746 case NETLOOP_CONTINUE:
747 continue;
2d966958
WD
748 }
749 }
4793ee65
SG
750
751done:
b63056d6
JL
752#ifdef CONFIG_USB_KEYBOARD
753 net_busy_flag = 0;
754#endif
39bccd21 755#ifdef CONFIG_CMD_TFTPPUT
4793ee65 756 /* Clear out the handlers */
ece223b5 757 net_set_udp_handler(NULL);
4793ee65 758 net_set_icmp_handler(NULL);
39bccd21 759#endif
60177b26 760 net_set_state(prev_net_state);
3eaac630
RF
761
762#if defined(CONFIG_CMD_PCAP)
763 if (pcap_active())
764 pcap_print_status();
765#endif
4793ee65 766 return ret;
2d966958
WD
767}
768
769/**********************************************************************/
770
bc0571fc 771static void start_again_timeout_handler(void)
2d966958 772{
22f6e99d 773 net_set_state(NETLOOP_RESTART);
2d966958
WD
774}
775
bc0571fc 776int net_start_again(void)
2d966958 777{
6e592385 778 char *nretry;
67b96e87
RB
779 int retry_forever = 0;
780 unsigned long retrycnt = 0;
60304592 781 int ret;
67b96e87 782
00caae6d 783 nretry = env_get("netretry");
67b96e87
RB
784 if (nretry) {
785 if (!strcmp(nretry, "yes"))
786 retry_forever = 1;
787 else if (!strcmp(nretry, "no"))
788 retrycnt = 0;
789 else if (!strcmp(nretry, "once"))
790 retrycnt = 1;
791 else
792 retrycnt = simple_strtoul(nretry, NULL, 0);
5c421331
JH
793 } else {
794 retrycnt = 0;
795 retry_forever = 0;
796 }
67b96e87 797
17d413b2 798 if ((!retry_forever) && (net_try_count > retrycnt)) {
67b96e87 799 eth_halt();
22f6e99d 800 net_set_state(NETLOOP_FAIL);
60304592
JH
801 /*
802 * We don't provide a way for the protocol to return an error,
803 * but this is almost always the reason.
804 */
805 return -ETIMEDOUT;
a3d991bd 806 }
67b96e87 807
bc0571fc 808 net_try_count++;
67b96e87 809
4f63acd0 810 eth_halt();
8b0c5c12 811#if !defined(CONFIG_NET_DO_NOT_TRY_ANOTHER)
bc0571fc 812 eth_try_another(!net_restarted);
8b0c5c12 813#endif
60304592 814 ret = eth_init();
bc0571fc
JH
815 if (net_restart_wrap) {
816 net_restart_wrap = 0;
817 if (net_dev_exists) {
818 net_set_timeout_handler(10000UL,
819 start_again_timeout_handler);
ece223b5 820 net_set_udp_handler(NULL);
6e592385 821 } else {
22f6e99d 822 net_set_state(NETLOOP_FAIL);
2d966958 823 }
6e592385 824 } else {
22f6e99d 825 net_set_state(NETLOOP_RESTART);
2d966958 826 }
60304592 827 return ret;
2d966958
WD
828}
829
830/**********************************************************************/
831/*
832 * Miscelaneous bits.
833 */
834
ece223b5 835static void dummy_handler(uchar *pkt, unsigned dport,
049a95a7 836 struct in_addr sip, unsigned sport,
ece223b5 837 unsigned len)
d280d3f4 838{
d280d3f4
JH
839}
840
ece223b5
JH
841rxhand_f *net_get_udp_handler(void)
842{
843 return udp_packet_handler;
844}
d280d3f4 845
ece223b5
JH
846void net_set_udp_handler(rxhand_f *f)
847{
bc0571fc 848 debug_cond(DEBUG_INT_STATE, "--- net_loop UDP handler set (%p)\n", f);
ece223b5
JH
849 if (f == NULL)
850 udp_packet_handler = dummy_handler;
851 else
852 udp_packet_handler = f;
853}
854
855rxhand_f *net_get_arp_handler(void)
2d966958 856{
ece223b5
JH
857 return arp_packet_handler;
858}
859
860void net_set_arp_handler(rxhand_f *f)
861{
bc0571fc 862 debug_cond(DEBUG_INT_STATE, "--- net_loop ARP handler set (%p)\n", f);
ece223b5
JH
863 if (f == NULL)
864 arp_packet_handler = dummy_handler;
865 else
866 arp_packet_handler = f;
2d966958
WD
867}
868
39bccd21 869#ifdef CONFIG_CMD_TFTPPUT
4793ee65
SG
870void net_set_icmp_handler(rxhand_icmp_f *f)
871{
872 packet_icmp_handler = f;
873}
39bccd21 874#endif
2d966958 875
bc0571fc 876void net_set_timeout_handler(ulong iv, thand_f *f)
2d966958
WD
877{
878 if (iv == 0) {
4ef8d53c 879 debug_cond(DEBUG_INT_STATE,
bc0571fc
JH
880 "--- net_loop timeout handler cancelled\n");
881 time_handler = (thand_f *)0;
2d966958 882 } else {
4ef8d53c 883 debug_cond(DEBUG_INT_STATE,
bc0571fc
JH
884 "--- net_loop timeout handler set (%p)\n", f);
885 time_handler = f;
886 time_start = get_timer(0);
887 time_delta = iv * CONFIG_SYS_HZ / 1000;
2d966958
WD
888 }
889}
890
ac3f26cc
JH
891uchar *net_get_async_tx_pkt_buf(void)
892{
893 if (arp_is_waiting())
894 return arp_tx_packet; /* If we are waiting, we already sent */
895 else
896 return net_tx_packet;
897}
898
1203fcce 899int net_send_udp_packet(uchar *ether, struct in_addr dest, int dport, int sport,
206d07fd 900 int payload_len)
5d457ecb
DH
901{
902 return net_send_ip_packet(ether, dest, dport, sport, payload_len,
903 IPPROTO_UDP, 0, 0, 0);
904}
905
a3bf193b
YCLP
906#if defined(CONFIG_PROT_TCP)
907int net_send_tcp_packet(int payload_len, int dport, int sport, u8 action,
908 u32 tcp_seq_num, u32 tcp_ack_num)
909{
910 return net_send_ip_packet(net_server_ethaddr, net_server_ip, dport,
911 sport, payload_len, IPPROTO_TCP, action,
912 tcp_seq_num, tcp_ack_num);
913}
914#endif
915
5d457ecb
DH
916int net_send_ip_packet(uchar *ether, struct in_addr dest, int dport, int sport,
917 int payload_len, int proto, u8 action, u32 tcp_seq_num,
918 u32 tcp_ack_num)
73a8b27c 919{
a3d991bd 920 uchar *pkt;
9214637a
JH
921 int eth_hdr_size;
922 int pkt_hdr_size;
a3d991bd 923
bc0571fc 924 /* make sure the net_tx_packet is initialized (net_init() was called) */
1203fcce
JH
925 assert(net_tx_packet != NULL);
926 if (net_tx_packet == NULL)
46c495d5
JH
927 return -1;
928
73a8b27c 929 /* convert to new style broadcast */
049a95a7
JH
930 if (dest.s_addr == 0)
931 dest.s_addr = 0xFFFFFFFF;
73a8b27c
WD
932
933 /* if broadcast, make the ether address a broadcast and don't do ARP */
049a95a7 934 if (dest.s_addr == 0xFFFFFFFF)
0adb5b76 935 ether = (uchar *)net_bcast_ethaddr;
73a8b27c 936
1203fcce 937 pkt = (uchar *)net_tx_packet;
9214637a 938
1203fcce 939 eth_hdr_size = net_set_ether(pkt, ether, PROT_IP);
5d457ecb
DH
940
941 switch (proto) {
942 case IPPROTO_UDP:
943 net_set_udp_header(pkt + eth_hdr_size, dest, dport, sport,
944 payload_len);
945 pkt_hdr_size = eth_hdr_size + IP_UDP_HDR_SIZE;
946 break;
a3bf193b
YCLP
947#if defined(CONFIG_PROT_TCP)
948 case IPPROTO_TCP:
949 pkt_hdr_size = eth_hdr_size
950 + tcp_set_tcp_header(pkt + eth_hdr_size, dport, sport,
951 payload_len, action, tcp_seq_num,
952 tcp_ack_num);
953 break;
954#endif
5d457ecb
DH
955 default:
956 return -EINVAL;
957 }
73a8b27c 958
e94070c4 959 /* if MAC address was not discovered yet, do an ARP request */
0adb5b76 960 if (memcmp(ether, net_null_ethaddr, 6) == 0) {
4ef8d53c 961 debug_cond(DEBUG_DEV_PKT, "sending ARP for %pI4\n", &dest);
0ebf04c6 962
9214637a 963 /* save the ip and eth addr for the packet to send after arp */
049a95a7 964 net_arp_wait_packet_ip = dest;
85d25e0e 965 arp_wait_packet_ethaddr = ether;
a3d991bd 966
73a8b27c 967 /* size of the waiting packet */
85d25e0e 968 arp_wait_tx_packet_size = pkt_hdr_size + payload_len;
73a8b27c
WD
969
970 /* and do the ARP request */
85d25e0e
JH
971 arp_wait_try = 1;
972 arp_wait_timer_start = get_timer(0);
973 arp_request();
73a8b27c 974 return 1; /* waiting */
9214637a 975 } else {
4ef8d53c 976 debug_cond(DEBUG_DEV_PKT, "sending UDP to %pI4/%pM\n",
bc0571fc 977 &dest, ether);
1203fcce 978 net_send_packet(net_tx_packet, pkt_hdr_size + payload_len);
9214637a 979 return 0; /* transmitted */
73a8b27c 980 }
73a8b27c
WD
981}
982
5cfaa4e5
AR
983#ifdef CONFIG_IP_DEFRAG
984/*
985 * This function collects fragments in a single packet, according
986 * to the algorithm in RFC815. It returns NULL or the pointer to
987 * a complete packet, in static storage
988 */
aa7a6487 989#define IP_PKTSIZE (CONFIG_NET_MAXDEFRAG)
5cfaa4e5 990
c5c59df0 991#define IP_MAXUDP (IP_PKTSIZE - IP_HDR_SIZE)
5cfaa4e5
AR
992
993/*
994 * this is the packet being assembled, either data or frag control.
995 * Fragments go by 8 bytes, so this union must be 8 bytes long
996 */
997struct hole {
998 /* first_byte is address of this structure */
999 u16 last_byte; /* last byte in this hole + 1 (begin of next hole) */
1000 u16 next_hole; /* index of next (in 8-b blocks), 0 == none */
1001 u16 prev_hole; /* index of prev, 0 == none */
1002 u16 unused;
1003};
1004
bc0571fc 1005static struct ip_udp_hdr *__net_defragment(struct ip_udp_hdr *ip, int *lenp)
5cfaa4e5 1006{
48522bb5 1007 static uchar pkt_buff[IP_PKTSIZE] __aligned(PKTALIGN);
5cfaa4e5
AR
1008 static u16 first_hole, total_len;
1009 struct hole *payload, *thisfrag, *h, *newh;
594c26f8 1010 struct ip_udp_hdr *localip = (struct ip_udp_hdr *)pkt_buff;
5cfaa4e5
AR
1011 uchar *indata = (uchar *)ip;
1012 int offset8, start, len, done = 0;
1013 u16 ip_off = ntohs(ip->ip_off);
1014
1817c382
RV
1015 /*
1016 * Calling code already rejected <, but we don't have to deal
1017 * with an IP fragment with no payload.
1018 */
1019 if (ntohs(ip->ip_len) <= IP_HDR_SIZE)
b85d130e
FE
1020 return NULL;
1021
5cfaa4e5 1022 /* payload starts after IP header, this fragment is in there */
c5c59df0 1023 payload = (struct hole *)(pkt_buff + IP_HDR_SIZE);
5cfaa4e5
AR
1024 offset8 = (ip_off & IP_OFFS);
1025 thisfrag = payload + offset8;
1026 start = offset8 * 8;
c5c59df0 1027 len = ntohs(ip->ip_len) - IP_HDR_SIZE;
5cfaa4e5 1028
1817c382
RV
1029 /* All but last fragment must have a multiple-of-8 payload. */
1030 if ((len & 7) && (ip_off & IP_FLAGS_MFRAG))
1031 return NULL;
1032
5cfaa4e5
AR
1033 if (start + len > IP_MAXUDP) /* fragment extends too far */
1034 return NULL;
1035
1036 if (!total_len || localip->ip_id != ip->ip_id) {
1037 /* new (or different) packet, reset structs */
1038 total_len = 0xffff;
1039 payload[0].last_byte = ~0;
1040 payload[0].next_hole = 0;
1041 payload[0].prev_hole = 0;
1042 first_hole = 0;
1043 /* any IP header will work, copy the first we received */
c5c59df0 1044 memcpy(localip, ip, IP_HDR_SIZE);
5cfaa4e5
AR
1045 }
1046
1047 /*
1048 * What follows is the reassembly algorithm. We use the payload
1049 * array as a linked list of hole descriptors, as each hole starts
1050 * at a multiple of 8 bytes. However, last byte can be whatever value,
1051 * so it is represented as byte count, not as 8-byte blocks.
1052 */
1053
1054 h = payload + first_hole;
1055 while (h->last_byte < start) {
1056 if (!h->next_hole) {
1057 /* no hole that far away */
1058 return NULL;
1059 }
1060 h = payload + h->next_hole;
1061 }
1062
e397e59e
FS
1063 /* last fragment may be 1..7 bytes, the "+7" forces acceptance */
1064 if (offset8 + ((len + 7) / 8) <= h - payload) {
5cfaa4e5
AR
1065 /* no overlap with holes (dup fragment?) */
1066 return NULL;
1067 }
1068
1069 if (!(ip_off & IP_FLAGS_MFRAG)) {
1070 /* no more fragmentss: truncate this (last) hole */
1071 total_len = start + len;
1072 h->last_byte = start + len;
1073 }
1074
1075 /*
06869686
RV
1076 * There is some overlap: fix the hole list. This code deals
1077 * with a fragment that overlaps with two different holes
1078 * (thus being a superset of a previously-received fragment)
1079 * by only using the part of the fragment that fits in the
1080 * first hole.
5cfaa4e5 1081 */
06869686
RV
1082 if (h->last_byte < start + len)
1083 len = h->last_byte - start;
5cfaa4e5 1084
4f63acd0 1085 if ((h >= thisfrag) && (h->last_byte <= start + len)) {
5cfaa4e5
AR
1086 /* complete overlap with hole: remove hole */
1087 if (!h->prev_hole && !h->next_hole) {
1088 /* last remaining hole */
1089 done = 1;
1090 } else if (!h->prev_hole) {
1091 /* first hole */
1092 first_hole = h->next_hole;
1093 payload[h->next_hole].prev_hole = 0;
1094 } else if (!h->next_hole) {
1095 /* last hole */
1096 payload[h->prev_hole].next_hole = 0;
1097 } else {
1098 /* in the middle of the list */
1099 payload[h->next_hole].prev_hole = h->prev_hole;
1100 payload[h->prev_hole].next_hole = h->next_hole;
1101 }
1102
1103 } else if (h->last_byte <= start + len) {
1104 /* overlaps with final part of the hole: shorten this hole */
1105 h->last_byte = start;
1106
1107 } else if (h >= thisfrag) {
1108 /* overlaps with initial part of the hole: move this hole */
1109 newh = thisfrag + (len / 8);
1110 *newh = *h;
1111 h = newh;
1112 if (h->next_hole)
1113 payload[h->next_hole].prev_hole = (h - payload);
1114 if (h->prev_hole)
1115 payload[h->prev_hole].next_hole = (h - payload);
1116 else
1117 first_hole = (h - payload);
1118
1119 } else {
1120 /* fragment sits in the middle: split the hole */
1121 newh = thisfrag + (len / 8);
1122 *newh = *h;
1123 h->last_byte = start;
1124 h->next_hole = (newh - payload);
1125 newh->prev_hole = (h - payload);
1126 if (newh->next_hole)
1127 payload[newh->next_hole].prev_hole = (newh - payload);
1128 }
1129
1130 /* finally copy this fragment and possibly return whole packet */
c5c59df0 1131 memcpy((uchar *)thisfrag, indata + IP_HDR_SIZE, len);
5cfaa4e5
AR
1132 if (!done)
1133 return NULL;
1134
c5c59df0 1135 *lenp = total_len + IP_HDR_SIZE;
06653c70 1136 localip->ip_len = htons(*lenp);
5cfaa4e5
AR
1137 return localip;
1138}
1139
bc0571fc
JH
1140static inline struct ip_udp_hdr *net_defragment(struct ip_udp_hdr *ip,
1141 int *lenp)
5cfaa4e5
AR
1142{
1143 u16 ip_off = ntohs(ip->ip_off);
1144 if (!(ip_off & (IP_OFFS | IP_FLAGS_MFRAG)))
1145 return ip; /* not a fragment */
bc0571fc 1146 return __net_defragment(ip, lenp);
5cfaa4e5
AR
1147}
1148
1149#else /* !CONFIG_IP_DEFRAG */
1150
bc0571fc
JH
1151static inline struct ip_udp_hdr *net_defragment(struct ip_udp_hdr *ip,
1152 int *lenp)
5cfaa4e5
AR
1153{
1154 u16 ip_off = ntohs(ip->ip_off);
1155 if (!(ip_off & (IP_OFFS | IP_FLAGS_MFRAG)))
1156 return ip; /* not a fragment */
1157 return NULL;
1158}
1159#endif
a3d991bd 1160
8f79bb17
SG
1161/**
1162 * Receive an ICMP packet. We deal with REDIRECT and PING here, and silently
1163 * drop others.
1164 *
1165 * @parma ip IP packet containing the ICMP
1166 */
594c26f8 1167static void receive_icmp(struct ip_udp_hdr *ip, int len,
049a95a7 1168 struct in_addr src_ip, struct ethernet_hdr *et)
8f79bb17 1169{
e0a63079 1170 struct icmp_hdr *icmph = (struct icmp_hdr *)&ip->udp_src;
8f79bb17
SG
1171
1172 switch (icmph->type) {
1173 case ICMP_REDIRECT:
1174 if (icmph->code != ICMP_REDIR_HOST)
1175 return;
1176 printf(" ICMP Host Redirect to %pI4 ",
bc0571fc 1177 &icmph->un.gateway);
8f79bb17 1178 break;
a36b12f9 1179 default:
8f79bb17 1180#if defined(CONFIG_CMD_PING)
a36b12f9 1181 ping_receive(et, ip, len);
8f79bb17 1182#endif
39bccd21 1183#ifdef CONFIG_CMD_TFTPPUT
4793ee65
SG
1184 if (packet_icmp_handler)
1185 packet_icmp_handler(icmph->type, icmph->code,
bc0571fc
JH
1186 ntohs(ip->udp_dst), src_ip,
1187 ntohs(ip->udp_src), icmph->un.data,
1188 ntohs(ip->udp_len));
39bccd21 1189#endif
8f79bb17
SG
1190 break;
1191 }
1192}
1193
2a504df0 1194void net_process_received_packet(uchar *in_packet, int len)
2d966958 1195{
cb487f56 1196 struct ethernet_hdr *et;
594c26f8 1197 struct ip_udp_hdr *ip;
049a95a7
JH
1198 struct in_addr dst_ip;
1199 struct in_addr src_ip;
8d353eb8 1200 int eth_proto;
643d1ab2 1201#if defined(CONFIG_CMD_CDP)
a3d991bd
WD
1202 int iscdp;
1203#endif
1204 ushort cti = 0, vlanid = VLAN_NONE, myvlanid, mynvlanid;
1205
4ef8d53c 1206 debug_cond(DEBUG_NET_PKT, "packet received\n");
798962ce
SA
1207 if (DEBUG_NET_PKT_TRACE)
1208 print_hex_dump_bytes("rx: ", DUMP_PREFIX_OFFSET, in_packet,
1209 len);
2d966958 1210
3eaac630
RF
1211#if defined(CONFIG_CMD_PCAP)
1212 pcap_post(in_packet, len, false);
1213#endif
1203fcce
JH
1214 net_rx_packet = in_packet;
1215 net_rx_packet_len = len;
2a504df0 1216 et = (struct ethernet_hdr *)in_packet;
a3d991bd
WD
1217
1218 /* too small packet? */
1219 if (len < ETHER_HDR_SIZE)
1220 return;
1221
0efe1bcf 1222#if defined(CONFIG_API) || defined(CONFIG_EFI_LOADER)
f85b6071 1223 if (push_packet) {
2a504df0 1224 (*push_packet)(in_packet, len);
f85b6071
RJ
1225 return;
1226 }
1227#endif
1228
643d1ab2 1229#if defined(CONFIG_CMD_CDP)
a3d991bd 1230 /* keep track if packet is CDP */
17351883 1231 iscdp = is_cdp_packet(et->et_dest);
a3d991bd
WD
1232#endif
1233
4fd5055f 1234 myvlanid = ntohs(net_our_vlan);
a3d991bd
WD
1235 if (myvlanid == (ushort)-1)
1236 myvlanid = VLAN_NONE;
4fd5055f 1237 mynvlanid = ntohs(net_native_vlan);
a3d991bd
WD
1238 if (mynvlanid == (ushort)-1)
1239 mynvlanid = VLAN_NONE;
2d966958 1240
8d353eb8 1241 eth_proto = ntohs(et->et_protlen);
2d966958 1242
8d353eb8 1243 if (eth_proto < 1514) {
cb487f56 1244 struct e802_hdr *et802 = (struct e802_hdr *)et;
2d966958 1245 /*
da5ebe2c
JH
1246 * Got a 802.2 packet. Check the other protocol field.
1247 * XXX VLAN over 802.2+SNAP not implemented!
2d966958 1248 */
8d353eb8 1249 eth_proto = ntohs(et802->et_prot);
a3d991bd 1250
2a504df0 1251 ip = (struct ip_udp_hdr *)(in_packet + E802_HDR_SIZE);
2d966958 1252 len -= E802_HDR_SIZE;
a3d991bd 1253
8d353eb8 1254 } else if (eth_proto != PROT_VLAN) { /* normal packet */
2a504df0 1255 ip = (struct ip_udp_hdr *)(in_packet + ETHER_HDR_SIZE);
2d966958 1256 len -= ETHER_HDR_SIZE;
a3d991bd
WD
1257
1258 } else { /* VLAN packet */
c68cca35
JH
1259 struct vlan_ethernet_hdr *vet =
1260 (struct vlan_ethernet_hdr *)et;
a3d991bd 1261
4ef8d53c 1262 debug_cond(DEBUG_NET_PKT, "VLAN packet received\n");
0ebf04c6 1263
a3d991bd
WD
1264 /* too small packet? */
1265 if (len < VLAN_ETHER_HDR_SIZE)
1266 return;
1267
1268 /* if no VLAN active */
4fd5055f 1269 if ((ntohs(net_our_vlan) & VLAN_IDMASK) == VLAN_NONE
643d1ab2 1270#if defined(CONFIG_CMD_CDP)
a3d991bd
WD
1271 && iscdp == 0
1272#endif
1273 )
1274 return;
1275
1276 cti = ntohs(vet->vet_tag);
1277 vlanid = cti & VLAN_IDMASK;
8d353eb8 1278 eth_proto = ntohs(vet->vet_type);
a3d991bd 1279
2a504df0 1280 ip = (struct ip_udp_hdr *)(in_packet + VLAN_ETHER_HDR_SIZE);
a3d991bd 1281 len -= VLAN_ETHER_HDR_SIZE;
2d966958
WD
1282 }
1283
4ef8d53c 1284 debug_cond(DEBUG_NET_PKT, "Receive from protocol 0x%x\n", eth_proto);
2d966958 1285
643d1ab2 1286#if defined(CONFIG_CMD_CDP)
a3d991bd 1287 if (iscdp) {
0b4c5ff4 1288 cdp_receive((uchar *)ip, len);
a3d991bd
WD
1289 return;
1290 }
1291#endif
1292
1293 if ((myvlanid & VLAN_IDMASK) != VLAN_NONE) {
1294 if (vlanid == VLAN_NONE)
1295 vlanid = (mynvlanid & VLAN_IDMASK);
1296 /* not matched? */
1297 if (vlanid != (myvlanid & VLAN_IDMASK))
1298 return;
1299 }
1300
8d353eb8 1301 switch (eth_proto) {
2d966958 1302 case PROT_ARP:
85d25e0e 1303 arp_receive(et, ip, len);
289f932c 1304 break;
2d966958 1305
bf6cb247 1306#ifdef CONFIG_CMD_RARP
2d966958 1307 case PROT_RARP:
8b9c5322 1308 rarp_receive(ip, len);
2d966958 1309 break;
ffdbf3ba
VM
1310#endif
1311#if IS_ENABLED(CONFIG_IPV6)
1312 case PROT_IP6:
1313 net_ip6_handler(et, (struct ip6_hdr *)ip, len);
0d6d5a4a 1314 break;
bf6cb247 1315#endif
2d966958 1316 case PROT_IP:
4ef8d53c 1317 debug_cond(DEBUG_NET_PKT, "Got IP\n");
5cfaa4e5 1318 /* Before we start poking the header, make sure it is there */
ad359d89 1319 if (len < IP_HDR_SIZE) {
594c26f8 1320 debug("len bad %d < %lu\n", len,
ad359d89 1321 (ulong)IP_HDR_SIZE);
2d966958
WD
1322 return;
1323 }
5cfaa4e5 1324 /* Check the packet length */
2d966958 1325 if (len < ntohs(ip->ip_len)) {
4ef8d53c 1326 debug("len bad %d < %d\n", len, ntohs(ip->ip_len));
2d966958
WD
1327 return;
1328 }
1329 len = ntohs(ip->ip_len);
ad359d89
RV
1330 if (len < IP_HDR_SIZE) {
1331 debug("bad ip->ip_len %d < %d\n", len, (int)IP_HDR_SIZE);
1332 return;
1333 }
4ef8d53c 1334 debug_cond(DEBUG_NET_PKT, "len=%d, v=%02x\n",
bc0571fc 1335 len, ip->ip_hl_v & 0xff);
0ebf04c6 1336
5cfaa4e5 1337 /* Can't deal with anything except IPv4 */
d3c65b01 1338 if ((ip->ip_hl_v & 0xf0) != 0x40)
2d966958 1339 return;
5cfaa4e5 1340 /* Can't deal with IP options (headers != 20 bytes) */
b0fcc48c 1341 if ((ip->ip_hl_v & 0x0f) != 0x05)
6b52cfe1 1342 return;
5cfaa4e5 1343 /* Check the Checksum of the header */
0da0fcd5 1344 if (!ip_checksum_ok((uchar *)ip, IP_HDR_SIZE)) {
4ef8d53c 1345 debug("checksum bad\n");
2d966958
WD
1346 return;
1347 }
5cfaa4e5 1348 /* If it is not for us, ignore it */
049a95a7
JH
1349 dst_ip = net_read_ip(&ip->ip_dst);
1350 if (net_ip.s_addr && dst_ip.s_addr != net_ip.s_addr &&
1351 dst_ip.s_addr != 0xFFFFFFFF) {
c819abee 1352 return;
2d966958 1353 }
03eb129f 1354 /* Read source IP address for later use */
049a95a7 1355 src_ip = net_read_ip(&ip->ip_src);
5cfaa4e5
AR
1356 /*
1357 * The function returns the unchanged packet if it's not
1358 * a fragment, and either the complete packet or NULL if
1359 * it is a fragment (if !CONFIG_IP_DEFRAG, it returns NULL)
1360 */
bc0571fc 1361 ip = net_defragment(ip, &len);
ccb9ebef 1362 if (!ip)
5cfaa4e5 1363 return;
2d966958
WD
1364 /*
1365 * watch for ICMP host redirects
1366 *
8bde7f77
WD
1367 * There is no real handler code (yet). We just watch
1368 * for ICMP host redirect messages. In case anybody
1369 * sees these messages: please contact me
1370 * ([email protected]), or - even better - send me the
1371 * necessary fixes :-)
2d966958 1372 *
8bde7f77
WD
1373 * Note: in all cases where I have seen this so far
1374 * it was a problem with the router configuration,
1375 * for instance when a router was configured in the
1376 * BOOTP reply, but the TFTP server was on the same
1377 * subnet. So this is probably a warning that your
1378 * configuration might be wrong. But I'm not really
1379 * sure if there aren't any other situations.
4793ee65
SG
1380 *
1381 * Simon Glass <[email protected]>: We get an ICMP when
1382 * we send a tftp packet to a dead connection, or when
1383 * there is no server at the other end.
2d966958
WD
1384 */
1385 if (ip->ip_p == IPPROTO_ICMP) {
8f79bb17
SG
1386 receive_icmp(ip, len, src_ip, et);
1387 return;
a3bf193b
YCLP
1388#if defined(CONFIG_PROT_TCP)
1389 } else if (ip->ip_p == IPPROTO_TCP) {
1390 debug_cond(DEBUG_DEV_PKT,
1391 "TCP PH (to=%pI4, from=%pI4, len=%d)\n",
1392 &dst_ip, &src_ip, len);
1393
1394 rxhand_tcp_f((union tcp_build_pkt *)ip, len);
1395 return;
1396#endif
2d966958
WD
1397 } else if (ip->ip_p != IPPROTO_UDP) { /* Only UDP packets */
1398 return;
1399 }
1400
06653c70 1401 if (ntohs(ip->udp_len) < UDP_HDR_SIZE || ntohs(ip->udp_len) > len - IP_HDR_SIZE)
fe728806 1402 return;
1403
4ef8d53c 1404 debug_cond(DEBUG_DEV_PKT,
bc0571fc
JH
1405 "received UDP (to=%pI4, from=%pI4, len=%d)\n",
1406 &dst_ip, &src_ip, len);
4ef8d53c 1407
4b37fd14 1408 if (IS_ENABLED(CONFIG_UDP_CHECKSUM) && ip->udp_xsum != 0) {
b2f50807 1409 ulong xsum;
8524423d 1410 u8 *sumptr;
8534bf9a
SR
1411 ushort sumlen;
1412
1413 xsum = ip->ip_p;
1414 xsum += (ntohs(ip->udp_len));
049a95a7
JH
1415 xsum += (ntohl(ip->ip_src.s_addr) >> 16) & 0x0000ffff;
1416 xsum += (ntohl(ip->ip_src.s_addr) >> 0) & 0x0000ffff;
1417 xsum += (ntohl(ip->ip_dst.s_addr) >> 16) & 0x0000ffff;
1418 xsum += (ntohl(ip->ip_dst.s_addr) >> 0) & 0x0000ffff;
8534bf9a
SR
1419
1420 sumlen = ntohs(ip->udp_len);
8524423d 1421 sumptr = (u8 *)&ip->udp_src;
8534bf9a
SR
1422
1423 while (sumlen > 1) {
8524423d
HS
1424 /* inlined ntohs() to avoid alignment errors */
1425 xsum += (sumptr[0] << 8) + sumptr[1];
1426 sumptr += 2;
8534bf9a
SR
1427 sumlen -= 2;
1428 }
8524423d
HS
1429 if (sumlen > 0)
1430 xsum += (sumptr[0] << 8) + sumptr[0];
8534bf9a 1431 while ((xsum >> 16) != 0) {
3e38e429
LC
1432 xsum = (xsum & 0x0000ffff) +
1433 ((xsum >> 16) & 0x0000ffff);
8534bf9a
SR
1434 }
1435 if ((xsum != 0x00000000) && (xsum != 0x0000ffff)) {
9b55a253 1436 printf(" UDP wrong checksum %08lx %08x\n",
bc0571fc 1437 xsum, ntohs(ip->udp_xsum));
8534bf9a
SR
1438 return;
1439 }
1440 }
8534bf9a 1441
66c89ee3 1442#if defined(CONFIG_NETCONSOLE) && !defined(CONFIG_SPL_BUILD)
594c26f8 1443 nc_input_packet((uchar *)ip + IP_UDP_HDR_SIZE,
bc0571fc
JH
1444 src_ip,
1445 ntohs(ip->udp_dst),
1446 ntohs(ip->udp_src),
1447 ntohs(ip->udp_len) - UDP_HDR_SIZE);
68ceb29e 1448#endif
2d966958 1449 /*
bc0571fc 1450 * IP header OK. Pass the packet to the current handler.
2d966958 1451 */
ece223b5 1452 (*udp_packet_handler)((uchar *)ip + IP_UDP_HDR_SIZE,
bc0571fc
JH
1453 ntohs(ip->udp_dst),
1454 src_ip,
1455 ntohs(ip->udp_src),
1456 ntohs(ip->udp_len) - UDP_HDR_SIZE);
2d966958 1457 break;
d8970dae
LF
1458#ifdef CONFIG_CMD_WOL
1459 case PROT_WOL:
1460 wol_receive(ip, len);
1461 break;
09bd3d0b
SMJ
1462#endif
1463#ifdef CONFIG_PHY_NCSI
1464 case PROT_NCSI:
1465 ncsi_receive(et, ip, len);
1466 break;
d8970dae 1467#endif
2d966958
WD
1468 }
1469}
1470
2d966958
WD
1471/**********************************************************************/
1472
e4bf0c5c 1473static int net_check_prereq(enum proto_t protocol)
2d966958
WD
1474{
1475 switch (protocol) {
6e592385 1476 /* Fall through */
643d1ab2 1477#if defined(CONFIG_CMD_PING)
73a8b27c 1478 case PING:
049a95a7 1479 if (net_ping_ip.s_addr == 0) {
4f63acd0 1480 puts("*** ERROR: ping address not given\n");
92895de9 1481 return 1;
6e592385
WD
1482 }
1483 goto common;
cbd8a35c 1484#endif
eeb0a2c6
VM
1485#if defined(CONFIG_CMD_PING6)
1486 case PING6:
1487 if (ip6_is_unspecified_addr(&net_ping_ip6)) {
1488 puts("*** ERROR: ping address not given\n");
1489 return 1;
1490 }
1491 goto common;
1492#endif
1a32bf41
RG
1493#if defined(CONFIG_CMD_DNS)
1494 case DNS:
049a95a7 1495 if (net_dns_server.s_addr == 0) {
1a32bf41
RG
1496 puts("*** ERROR: DNS server address not given\n");
1497 return 1;
1498 }
1499 goto common;
1500#endif
b43ea1bf
PR
1501#if defined(CONFIG_PROT_UDP)
1502 case UDP:
1503 if (udp_prereq())
1504 return 1;
1505 goto common;
1506#endif
1507
643d1ab2 1508#if defined(CONFIG_CMD_NFS)
cbd8a35c 1509 case NFS:
73a8b27c 1510#endif
bc0571fc 1511 /* Fall through */
e4bf0c5c 1512 case TFTPGET:
1fb7cd49 1513 case TFTPPUT:
7fbf230d
VM
1514 if (IS_ENABLED(CONFIG_IPV6) && use_ip6) {
1515 if (!memcmp(&net_server_ip6, &net_null_addr_ip6,
1516 sizeof(struct in6_addr)) &&
1517 !strchr(net_boot_file_name, '[')) {
1518 puts("*** ERROR: `serverip6' not set\n");
1519 return 1;
1520 }
1521 } else if (net_server_ip.s_addr == 0 && !is_serverip_in_cmd()) {
4f63acd0 1522 puts("*** ERROR: `serverip' not set\n");
92895de9 1523 return 1;
6e592385 1524 }
912ece4c 1525#if defined(CONFIG_CMD_PING) || \
b43ea1bf 1526 defined(CONFIG_CMD_DNS) || defined(CONFIG_PROT_UDP)
4f63acd0 1527common:
73a8b27c 1528#endif
8b6bbe10 1529 /* Fall through */
73a8b27c 1530
8b6bbe10 1531 case NETCONS:
443d3191
DM
1532 case FASTBOOT_UDP:
1533 case FASTBOOT_TCP:
7a83af07 1534 case TFTPSRV:
7fbf230d
VM
1535 if (IS_ENABLED(CONFIG_IPV6) && use_ip6) {
1536 if (!memcmp(&net_link_local_ip6, &net_null_addr_ip6,
1537 sizeof(struct in6_addr))) {
1538 puts("*** ERROR: `ip6addr` not set\n");
1539 return 1;
1540 }
1541 } else if (net_ip.s_addr == 0) {
4f63acd0 1542 puts("*** ERROR: `ipaddr' not set\n");
92895de9 1543 return 1;
6e592385
WD
1544 }
1545 /* Fall through */
2d966958 1546
bf6cb247 1547#ifdef CONFIG_CMD_RARP
2d966958 1548 case RARP:
09bd3d0b
SMJ
1549#endif
1550#ifdef CONFIG_PHY_NCSI
1551 case NCSI:
bf6cb247 1552#endif
2d966958 1553 case BOOTP:
a3d991bd 1554 case CDP:
bf6cb247 1555 case DHCP:
d22c338e 1556 case LINKLOCAL:
0adb5b76 1557 if (memcmp(net_ethaddr, "\0\0\0\0\0\0", 6) == 0) {
4f63acd0 1558 int num = eth_get_dev_index();
2d966958 1559
6e592385
WD
1560 switch (num) {
1561 case -1:
4f63acd0 1562 puts("*** ERROR: No ethernet found.\n");
92895de9 1563 return 1;
6e592385 1564 case 0:
4f63acd0 1565 puts("*** ERROR: `ethaddr' not set\n");
2d966958 1566 break;
6e592385 1567 default:
4f63acd0 1568 printf("*** ERROR: `eth%daddr' not set\n",
bc0571fc 1569 num);
2d966958 1570 break;
6e592385 1571 }
2d966958 1572
bc0571fc 1573 net_start_again();
92895de9 1574 return 2;
6e592385
WD
1575 }
1576 /* Fall through */
1577 default:
92895de9 1578 return 0;
2d966958 1579 }
92895de9 1580 return 0; /* OK */
2d966958
WD
1581}
1582/**********************************************************************/
1583
a3d991bd 1584int
1203fcce 1585net_eth_hdr_size(void)
a3d991bd
WD
1586{
1587 ushort myvlanid;
2d966958 1588
4fd5055f 1589 myvlanid = ntohs(net_our_vlan);
a3d991bd
WD
1590 if (myvlanid == (ushort)-1)
1591 myvlanid = VLAN_NONE;
1592
3e38e429
LC
1593 return ((myvlanid & VLAN_IDMASK) == VLAN_NONE) ? ETHER_HDR_SIZE :
1594 VLAN_ETHER_HDR_SIZE;
a3d991bd
WD
1595}
1596
1203fcce 1597int net_set_ether(uchar *xet, const uchar *dest_ethaddr, uint prot)
2d966958 1598{
cb487f56 1599 struct ethernet_hdr *et = (struct ethernet_hdr *)xet;
a3d991bd
WD
1600 ushort myvlanid;
1601
4fd5055f 1602 myvlanid = ntohs(net_our_vlan);
a3d991bd
WD
1603 if (myvlanid == (ushort)-1)
1604 myvlanid = VLAN_NONE;
2d966958 1605
0adb5b76
JH
1606 memcpy(et->et_dest, dest_ethaddr, 6);
1607 memcpy(et->et_src, net_ethaddr, 6);
a3d991bd 1608 if ((myvlanid & VLAN_IDMASK) == VLAN_NONE) {
c819abee 1609 et->et_protlen = htons(prot);
a3d991bd
WD
1610 return ETHER_HDR_SIZE;
1611 } else {
c68cca35
JH
1612 struct vlan_ethernet_hdr *vet =
1613 (struct vlan_ethernet_hdr *)xet;
2d966958 1614
a3d991bd
WD
1615 vet->vet_vlan_type = htons(PROT_VLAN);
1616 vet->vet_tag = htons((0 << 5) | (myvlanid & VLAN_IDMASK));
1617 vet->vet_type = htons(prot);
1618 return VLAN_ETHER_HDR_SIZE;
1619 }
1620}
2d966958 1621
e7111015
JH
1622int net_update_ether(struct ethernet_hdr *et, uchar *addr, uint prot)
1623{
1624 ushort protlen;
1625
1626 memcpy(et->et_dest, addr, 6);
0adb5b76 1627 memcpy(et->et_src, net_ethaddr, 6);
e7111015
JH
1628 protlen = ntohs(et->et_protlen);
1629 if (protlen == PROT_VLAN) {
1630 struct vlan_ethernet_hdr *vet =
1631 (struct vlan_ethernet_hdr *)et;
1632 vet->vet_type = htons(prot);
1633 return VLAN_ETHER_HDR_SIZE;
1634 } else if (protlen > 1514) {
1635 et->et_protlen = htons(prot);
1636 return ETHER_HDR_SIZE;
1637 } else {
1638 /* 802.2 + SNAP */
1639 struct e802_hdr *et802 = (struct e802_hdr *)et;
1640 et802->et_prot = htons(prot);
1641 return E802_HDR_SIZE;
1642 }
1643}
1644
5d457ecb
DH
1645void net_set_ip_header(uchar *pkt, struct in_addr dest, struct in_addr source,
1646 u16 pkt_len, u8 proto)
2d966958 1647{
4b11c916 1648 struct ip_udp_hdr *ip = (struct ip_udp_hdr *)pkt;
2d966958
WD
1649
1650 /*
4b11c916 1651 * Construct an IP header.
2d966958 1652 */
3e38e429
LC
1653 /* IP_HDR_SIZE / 4 (not including UDP) */
1654 ip->ip_hl_v = 0x45;
2d966958 1655 ip->ip_tos = 0;
5d457ecb
DH
1656 ip->ip_len = htons(pkt_len);
1657 ip->ip_p = proto;
bc0571fc 1658 ip->ip_id = htons(net_ip_id++);
e0c07b86 1659 ip->ip_off = htons(IP_FLAGS_DFRAG); /* Don't fragment */
2d966958 1660 ip->ip_ttl = 255;
2d966958 1661 ip->ip_sum = 0;
3e38e429 1662 /* already in network byte order */
049a95a7 1663 net_copy_ip((void *)&ip->ip_src, &source);
4b11c916 1664 /* already in network byte order */
049a95a7 1665 net_copy_ip((void *)&ip->ip_dst, &dest);
5d457ecb
DH
1666
1667 ip->ip_sum = compute_ip_checksum(ip, IP_HDR_SIZE);
4b11c916
JH
1668}
1669
049a95a7 1670void net_set_udp_header(uchar *pkt, struct in_addr dest, int dport, int sport,
4b11c916
JH
1671 int len)
1672{
1673 struct ip_udp_hdr *ip = (struct ip_udp_hdr *)pkt;
1674
1675 /*
1676 * If the data is an odd number of bytes, zero the
1677 * byte after the last byte so that the checksum
1678 * will work.
1679 */
1680 if (len & 1)
1681 pkt[IP_UDP_HDR_SIZE + len] = 0;
1682
5d457ecb
DH
1683 net_set_ip_header(pkt, dest, net_ip, IP_UDP_HDR_SIZE + len,
1684 IPPROTO_UDP);
4b11c916 1685
2d966958
WD
1686 ip->udp_src = htons(sport);
1687 ip->udp_dst = htons(dport);
594c26f8 1688 ip->udp_len = htons(UDP_HDR_SIZE + len);
2d966958 1689 ip->udp_xsum = 0;
2d966958
WD
1690}
1691
4f63acd0 1692void copy_filename(char *dst, const char *src, int size)
2d966958 1693{
16cf145f 1694 if (src && *src && (*src == '"')) {
2d966958
WD
1695 ++src;
1696 --size;
1697 }
1698
16cf145f 1699 while ((--size > 0) && src && *src && (*src != '"'))
2d966958 1700 *dst++ = *src++;
2d966958
WD
1701 *dst = '\0';
1702}
1703
3a66fcb7
JH
1704int is_serverip_in_cmd(void)
1705{
1706 return !!strchr(net_boot_file_name, ':');
1707}
1708
6ab12830
JH
1709int net_parse_bootfile(struct in_addr *ipaddr, char *filename, int max_len)
1710{
1711 char *colon;
85f8e03b
LF
1712 struct in_addr ip;
1713 ip.s_addr = 0;
6ab12830
JH
1714
1715 if (net_boot_file_name[0] == '\0')
1716 return 0;
1717
1718 colon = strchr(net_boot_file_name, ':');
1719 if (colon) {
85f8e03b
LF
1720 ip = string_to_ip(net_boot_file_name);
1721 if (ipaddr && ip.s_addr)
1722 *ipaddr = ip;
1723 }
1724 if (ip.s_addr) {
6ab12830
JH
1725 strncpy(filename, colon + 1, max_len);
1726 } else {
1727 strncpy(filename, net_boot_file_name, max_len);
1728 }
1729 filename[max_len - 1] = '\0';
1730
1731 return 1;
1732}
1733
049a95a7 1734void ip_to_string(struct in_addr x, char *s)
2d966958 1735{
049a95a7 1736 x.s_addr = ntohl(x.s_addr);
4f63acd0 1737 sprintf(s, "%d.%d.%d.%d",
049a95a7
JH
1738 (int) ((x.s_addr >> 24) & 0xff),
1739 (int) ((x.s_addr >> 16) & 0xff),
1740 (int) ((x.s_addr >> 8) & 0xff),
1741 (int) ((x.s_addr >> 0) & 0xff)
6e592385 1742 );
2d966958
WD
1743}
1744
4fd5055f 1745void vlan_to_string(ushort x, char *s)
a3d991bd
WD
1746{
1747 x = ntohs(x);
1748
1749 if (x == (ushort)-1)
1750 x = VLAN_NONE;
1751
1752 if (x == VLAN_NONE)
1753 strcpy(s, "none");
1754 else
1755 sprintf(s, "%d", x & VLAN_IDMASK);
1756}
1757
4fd5055f 1758ushort string_to_vlan(const char *s)
a3d991bd
WD
1759{
1760 ushort id;
1761
1762 if (s == NULL)
b9711de1 1763 return htons(VLAN_NONE);
a3d991bd
WD
1764
1765 if (*s < '0' || *s > '9')
1766 id = VLAN_NONE;
1767 else
0b1284eb 1768 id = (ushort)dectoul(s, NULL);
a3d991bd 1769
b9711de1 1770 return htons(id);
a3d991bd
WD
1771}
1772
723806cc 1773ushort env_get_vlan(char *var)
a3d991bd 1774{
00caae6d 1775 return string_to_vlan(env_get(var));
a3d991bd 1776}
This page took 0.916512 seconds and 4 git commands to generate.