]>
Commit | Line | Data |
---|---|---|
f739fcd8 | 1 | // SPDX-License-Identifier: GPL-2.0 |
a36b12f9 JH |
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 | #include "ping.h" | |
13 | #include "arp.h" | |
f7ae49fc | 14 | #include <log.h> |
90526e9f | 15 | #include <net.h> |
a36b12f9 | 16 | |
331db5a9 | 17 | static ushort ping_seq_number; |
a36b12f9 JH |
18 | |
19 | /* The ip address to ping */ | |
049a95a7 | 20 | struct in_addr net_ping_ip; |
a36b12f9 | 21 | |
049a95a7 | 22 | static void set_icmp_header(uchar *pkt, struct in_addr dest) |
4b11c916 JH |
23 | { |
24 | /* | |
25 | * Construct an IP and ICMP header. | |
26 | */ | |
4b11c916 JH |
27 | struct icmp_hdr *icmp = (struct icmp_hdr *)(pkt + IP_HDR_SIZE); |
28 | ||
5d457ecb | 29 | net_set_ip_header(pkt, dest, net_ip, IP_ICMP_HDR_SIZE, IPPROTO_ICMP); |
4b11c916 JH |
30 | |
31 | icmp->type = ICMP_ECHO_REQUEST; | |
32 | icmp->code = 0; | |
33 | icmp->checksum = 0; | |
34 | icmp->un.echo.id = 0; | |
331db5a9 | 35 | icmp->un.echo.sequence = htons(ping_seq_number++); |
0da0fcd5 | 36 | icmp->checksum = compute_ip_checksum(icmp, ICMP_HDR_SIZE); |
4b11c916 JH |
37 | } |
38 | ||
a36b12f9 JH |
39 | static int ping_send(void) |
40 | { | |
a36b12f9 | 41 | uchar *pkt; |
00f33268 | 42 | int eth_hdr_size; |
a36b12f9 JH |
43 | |
44 | /* XXX always send arp request */ | |
45 | ||
049a95a7 | 46 | debug_cond(DEBUG_DEV_PKT, "sending ARP for %pI4\n", &net_ping_ip); |
a36b12f9 | 47 | |
049a95a7 | 48 | net_arp_wait_packet_ip = net_ping_ip; |
a36b12f9 | 49 | |
1203fcce JH |
50 | eth_hdr_size = net_set_ether(net_tx_packet, net_null_ethaddr, PROT_IP); |
51 | pkt = (uchar *)net_tx_packet + eth_hdr_size; | |
a36b12f9 | 52 | |
049a95a7 | 53 | set_icmp_header(pkt, net_ping_ip); |
a36b12f9 JH |
54 | |
55 | /* size of the waiting packet */ | |
85d25e0e | 56 | arp_wait_tx_packet_size = eth_hdr_size + IP_ICMP_HDR_SIZE; |
a36b12f9 JH |
57 | |
58 | /* and do the ARP request */ | |
85d25e0e JH |
59 | arp_wait_try = 1; |
60 | arp_wait_timer_start = get_timer(0); | |
61 | arp_request(); | |
a36b12f9 JH |
62 | return 1; /* waiting */ |
63 | } | |
64 | ||
331db5a9 | 65 | static void ping_timeout_handler(void) |
a36b12f9 JH |
66 | { |
67 | eth_halt(); | |
22f6e99d | 68 | net_set_state(NETLOOP_FAIL); /* we did not get the reply */ |
a36b12f9 JH |
69 | } |
70 | ||
a36b12f9 JH |
71 | void ping_start(void) |
72 | { | |
73 | printf("Using %s device\n", eth_get_name()); | |
bc0571fc | 74 | net_set_timeout_handler(10000UL, ping_timeout_handler); |
a36b12f9 JH |
75 | |
76 | ping_send(); | |
77 | } | |
78 | ||
cb487f56 | 79 | void ping_receive(struct ethernet_hdr *et, struct ip_udp_hdr *ip, int len) |
a36b12f9 | 80 | { |
e0a63079 | 81 | struct icmp_hdr *icmph = (struct icmp_hdr *)&ip->udp_src; |
049a95a7 | 82 | struct in_addr src_ip; |
e7111015 | 83 | int eth_hdr_size; |
ac3f26cc | 84 | uchar *tx_packet; |
a36b12f9 JH |
85 | |
86 | switch (icmph->type) { | |
87 | case ICMP_ECHO_REPLY: | |
049a95a7 JH |
88 | src_ip = net_read_ip((void *)&ip->ip_src); |
89 | if (src_ip.s_addr == net_ping_ip.s_addr) | |
22f6e99d | 90 | net_set_state(NETLOOP_SUCCESS); |
a36b12f9 JH |
91 | return; |
92 | case ICMP_ECHO_REQUEST: | |
e7111015 | 93 | eth_hdr_size = net_update_ether(et, et->et_src, PROT_IP); |
a36b12f9 | 94 | |
331db5a9 JH |
95 | debug_cond(DEBUG_DEV_PKT, |
96 | "Got ICMP ECHO REQUEST, return %d bytes\n", | |
97 | eth_hdr_size + len); | |
a36b12f9 JH |
98 | |
99 | ip->ip_sum = 0; | |
100 | ip->ip_off = 0; | |
049a95a7 JH |
101 | net_copy_ip((void *)&ip->ip_dst, &ip->ip_src); |
102 | net_copy_ip((void *)&ip->ip_src, &net_ip); | |
0da0fcd5 | 103 | ip->ip_sum = compute_ip_checksum(ip, IP_HDR_SIZE); |
a36b12f9 JH |
104 | |
105 | icmph->type = ICMP_ECHO_REPLY; | |
106 | icmph->checksum = 0; | |
0da0fcd5 | 107 | icmph->checksum = compute_ip_checksum(icmph, len - IP_HDR_SIZE); |
ac3f26cc JH |
108 | |
109 | tx_packet = net_get_async_tx_pkt_buf(); | |
110 | memcpy(tx_packet, et, eth_hdr_size + len); | |
111 | net_send_packet(tx_packet, eth_hdr_size + len); | |
a36b12f9 JH |
112 | return; |
113 | /* default: | |
114 | return;*/ | |
115 | } | |
116 | } |