]>
Commit | Line | Data |
---|---|---|
83d290c5 | 1 | // SPDX-License-Identifier: GPL-2.0 |
a346ca79 JH |
2 | /* |
3 | * Copyright (c) 2015 National Instruments | |
4 | * | |
5 | * (C) Copyright 2015 | |
6 | * Joe Hershberger <[email protected]> | |
a346ca79 JH |
7 | */ |
8 | ||
f7ae49fc | 9 | #include <log.h> |
a346ca79 JH |
10 | #include <asm/eth-raw-os.h> |
11 | #include <common.h> | |
12 | #include <dm.h> | |
9fb625ce | 13 | #include <env.h> |
a346ca79 JH |
14 | #include <malloc.h> |
15 | #include <net.h> | |
401d1c4f | 16 | #include <asm/global_data.h> |
a346ca79 JH |
17 | |
18 | DECLARE_GLOBAL_DATA_PTR; | |
19 | ||
22f68524 | 20 | static int reply_arp; |
049a95a7 | 21 | static struct in_addr arp_ip; |
a346ca79 JH |
22 | |
23 | static int sb_eth_raw_start(struct udevice *dev) | |
24 | { | |
25 | struct eth_sandbox_raw_priv *priv = dev_get_priv(dev); | |
c69cda25 | 26 | struct eth_pdata *pdata = dev_get_plat(dev); |
8c7988b6 | 27 | int ret; |
a346ca79 JH |
28 | |
29 | debug("eth_sandbox_raw: Start\n"); | |
30 | ||
8c7988b6 JH |
31 | ret = sandbox_eth_raw_os_start(priv, pdata->enetaddr); |
32 | if (priv->local) { | |
382bee57 SG |
33 | env_set("ipaddr", "127.0.0.1"); |
34 | env_set("serverip", "127.0.0.1"); | |
ac13270b JH |
35 | net_ip = string_to_ip("127.0.0.1"); |
36 | net_server_ip = net_ip; | |
22f68524 | 37 | } |
8c7988b6 | 38 | return ret; |
a346ca79 JH |
39 | } |
40 | ||
41 | static int sb_eth_raw_send(struct udevice *dev, void *packet, int length) | |
42 | { | |
43 | struct eth_sandbox_raw_priv *priv = dev_get_priv(dev); | |
44 | ||
45 | debug("eth_sandbox_raw: Send packet %d\n", length); | |
46 | ||
22f68524 JH |
47 | if (priv->local) { |
48 | struct ethernet_hdr *eth = packet; | |
49 | ||
50 | if (ntohs(eth->et_protlen) == PROT_ARP) { | |
51 | struct arp_hdr *arp = packet + ETHER_HDR_SIZE; | |
52 | ||
53 | /** | |
54 | * localhost works on a higher-level API in Linux than | |
55 | * ARP packets, so fake it | |
56 | */ | |
049a95a7 | 57 | arp_ip = net_read_ip(&arp->ar_tpa); |
22f68524 JH |
58 | reply_arp = 1; |
59 | return 0; | |
60 | } | |
61 | packet += ETHER_HDR_SIZE; | |
62 | length -= ETHER_HDR_SIZE; | |
63 | } | |
a346ca79 JH |
64 | return sandbox_eth_raw_os_send(packet, length, priv); |
65 | } | |
66 | ||
a1ca92ea | 67 | static int sb_eth_raw_recv(struct udevice *dev, int flags, uchar **packetp) |
a346ca79 | 68 | { |
c69cda25 | 69 | struct eth_pdata *pdata = dev_get_plat(dev); |
a346ca79 | 70 | struct eth_sandbox_raw_priv *priv = dev_get_priv(dev); |
22f68524 | 71 | int retval = 0; |
a346ca79 JH |
72 | int length; |
73 | ||
22f68524 JH |
74 | if (reply_arp) { |
75 | struct arp_hdr *arp = (void *)net_rx_packets[0] + | |
76 | ETHER_HDR_SIZE; | |
77 | ||
78 | /* | |
79 | * Fake an ARP response. The u-boot network stack is sending an | |
80 | * ARP request (to find the MAC address to address the actual | |
81 | * packet to) and requires an ARP response to continue. Since | |
82 | * this is the localhost interface, there is no Etherent level | |
83 | * traffic at all, so there is no way to send an ARP request or | |
84 | * to get a response. For this reason we fake the response to | |
85 | * make the u-boot network stack happy. | |
86 | */ | |
87 | arp->ar_hrd = htons(ARP_ETHER); | |
88 | arp->ar_pro = htons(PROT_IP); | |
89 | arp->ar_hln = ARP_HLEN; | |
90 | arp->ar_pln = ARP_PLEN; | |
91 | arp->ar_op = htons(ARPOP_REPLY); | |
92 | /* Any non-zero MAC address will work */ | |
93 | memset(&arp->ar_sha, 0x01, ARP_HLEN); | |
94 | /* Use whatever IP we were looking for (always 127.0.0.1?) */ | |
049a95a7 | 95 | net_write_ip(&arp->ar_spa, arp_ip); |
22f68524 | 96 | memcpy(&arp->ar_tha, pdata->enetaddr, ARP_HLEN); |
049a95a7 | 97 | net_write_ip(&arp->ar_tpa, net_ip); |
22f68524 JH |
98 | length = ARP_HDR_SIZE; |
99 | } else { | |
100 | /* If local, the Ethernet header won't be included; skip it */ | |
101 | uchar *pktptr = priv->local ? | |
102 | net_rx_packets[0] + ETHER_HDR_SIZE : net_rx_packets[0]; | |
103 | ||
104 | retval = sandbox_eth_raw_os_recv(pktptr, &length, priv); | |
105 | } | |
a346ca79 JH |
106 | |
107 | if (!retval && length) { | |
22f68524 JH |
108 | if (priv->local) { |
109 | struct ethernet_hdr *eth = (void *)net_rx_packets[0]; | |
110 | ||
111 | /* Fill in enough of the missing Ethernet header */ | |
112 | memcpy(eth->et_dest, pdata->enetaddr, ARP_HLEN); | |
113 | memset(eth->et_src, 0x01, ARP_HLEN); | |
114 | eth->et_protlen = htons(reply_arp ? PROT_ARP : PROT_IP); | |
115 | reply_arp = 0; | |
116 | length += ETHER_HDR_SIZE; | |
117 | } | |
118 | ||
a346ca79 JH |
119 | debug("eth_sandbox_raw: received packet %d\n", |
120 | length); | |
121 | *packetp = net_rx_packets[0]; | |
122 | return length; | |
123 | } | |
124 | return retval; | |
125 | } | |
126 | ||
127 | static void sb_eth_raw_stop(struct udevice *dev) | |
128 | { | |
129 | struct eth_sandbox_raw_priv *priv = dev_get_priv(dev); | |
130 | ||
131 | debug("eth_sandbox_raw: Stop\n"); | |
132 | ||
133 | sandbox_eth_raw_os_stop(priv); | |
134 | } | |
135 | ||
b96ced9c JH |
136 | static int sb_eth_raw_read_rom_hwaddr(struct udevice *dev) |
137 | { | |
c69cda25 | 138 | struct eth_pdata *pdata = dev_get_plat(dev); |
b96ced9c JH |
139 | |
140 | net_random_ethaddr(pdata->enetaddr); | |
141 | ||
142 | return 0; | |
143 | } | |
144 | ||
a346ca79 JH |
145 | static const struct eth_ops sb_eth_raw_ops = { |
146 | .start = sb_eth_raw_start, | |
147 | .send = sb_eth_raw_send, | |
148 | .recv = sb_eth_raw_recv, | |
149 | .stop = sb_eth_raw_stop, | |
b96ced9c | 150 | .read_rom_hwaddr = sb_eth_raw_read_rom_hwaddr, |
a346ca79 JH |
151 | }; |
152 | ||
d1998a9f | 153 | static int sb_eth_raw_of_to_plat(struct udevice *dev) |
a346ca79 | 154 | { |
c69cda25 | 155 | struct eth_pdata *pdata = dev_get_plat(dev); |
8c7988b6 JH |
156 | struct eth_sandbox_raw_priv *priv = dev_get_priv(dev); |
157 | const char *ifname; | |
c9e2caff | 158 | int ret; |
8c7988b6 JH |
159 | |
160 | pdata->iobase = dev_read_addr(dev); | |
161 | ||
162 | ifname = dev_read_string(dev, "host-raw-interface"); | |
163 | if (ifname) { | |
164 | strncpy(priv->host_ifname, ifname, IFNAMSIZ); | |
165 | printf(": Using %s from DT\n", priv->host_ifname); | |
8c7988b6 | 166 | } |
c9e2caff JH |
167 | if (dev_read_u32(dev, "host-raw-interface-idx", |
168 | &priv->host_ifindex) < 0) { | |
169 | priv->host_ifindex = 0; | |
170 | } else { | |
171 | ret = sandbox_eth_raw_os_idx_to_name(priv); | |
172 | if (ret < 0) | |
173 | return ret; | |
174 | printf(": Using interface index %d from DT (%s)\n", | |
175 | priv->host_ifindex, priv->host_ifname); | |
176 | } | |
a346ca79 | 177 | |
8405452e SG |
178 | ret = sandbox_eth_raw_os_is_local(priv->host_ifname); |
179 | if (ret < 0) | |
180 | return ret; | |
181 | priv->local = ret; | |
ac13270b | 182 | |
a346ca79 JH |
183 | return 0; |
184 | } | |
185 | ||
186 | static const struct udevice_id sb_eth_raw_ids[] = { | |
187 | { .compatible = "sandbox,eth-raw" }, | |
188 | { } | |
189 | }; | |
190 | ||
191 | U_BOOT_DRIVER(eth_sandbox_raw) = { | |
192 | .name = "eth_sandbox_raw", | |
193 | .id = UCLASS_ETH, | |
194 | .of_match = sb_eth_raw_ids, | |
d1998a9f | 195 | .of_to_plat = sb_eth_raw_of_to_plat, |
a346ca79 | 196 | .ops = &sb_eth_raw_ops, |
41575d8e | 197 | .priv_auto = sizeof(struct eth_sandbox_raw_priv), |
caa4daa2 | 198 | .plat_auto = sizeof(struct eth_pdata), |
a346ca79 | 199 | }; |