root / firmware / zpu / lib / net_common.c @ e4d3f63c
History | View | Annotate | Download (14.6 KB)
| 1 |
/*
|
|---|---|
| 2 |
* Copyright 2009-2012 Ettus Research LLC
|
| 3 |
*
|
| 4 |
* This program is free software: you can redistribute it and/or modify
|
| 5 |
* it under the terms of the GNU General Public License as published by
|
| 6 |
* the Free Software Foundation, either version 3 of the License, or
|
| 7 |
* (at your option) any later version.
|
| 8 |
*
|
| 9 |
* This program is distributed in the hope that it will be useful,
|
| 10 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
| 11 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
| 12 |
* GNU General Public License for more details.
|
| 13 |
*
|
| 14 |
* You should have received a copy of the GNU General Public License
|
| 15 |
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
| 16 |
*/
|
| 17 |
|
| 18 |
#ifdef HAVE_CONFIG_H
|
| 19 |
#include <config.h> |
| 20 |
#endif
|
| 21 |
#include "net_common.h" |
| 22 |
#include "banal.h" |
| 23 |
#include <hal_io.h> |
| 24 |
#include <memory_map.h> |
| 25 |
#include <memcpy_wa.h> |
| 26 |
#include <ethernet.h> |
| 27 |
#include <net/padded_eth_hdr.h> |
| 28 |
#include <lwip/ip.h> |
| 29 |
#include <lwip/udp.h> |
| 30 |
#include <lwip/icmp.h> |
| 31 |
#include <stdlib.h> |
| 32 |
#include <nonstdio.h> |
| 33 |
#include "arp_cache.h" |
| 34 |
#include "if_arp.h" |
| 35 |
#include <ethertype.h> |
| 36 |
#include <string.h> |
| 37 |
#include "pkt_ctrl.h" |
| 38 |
|
| 39 |
/***********************************************************************
|
| 40 |
* Constants + Globals
|
| 41 |
**********************************************************************/
|
| 42 |
static const bool debug = false; |
| 43 |
static const size_t out_buff_size = 2048; |
| 44 |
static const eth_mac_addr_t BCAST_MAC_ADDR = {{0xff, 0xff, 0xff, 0xff, 0xff, 0xff}}; |
| 45 |
#define MAX_UDP_LISTENERS 10 |
| 46 |
|
| 47 |
/***********************************************************************
|
| 48 |
* 16-bit one's complement sum
|
| 49 |
**********************************************************************/
|
| 50 |
static uint32_t chksum_buffer(
|
| 51 |
uint16_t *buf, size_t nshorts, |
| 52 |
uint32_t initial_chksum |
| 53 |
){
|
| 54 |
uint32_t chksum = initial_chksum; |
| 55 |
for (size_t i = 0; i < nshorts; i++) chksum += buf[i]; |
| 56 |
|
| 57 |
while (chksum >> 16) chksum = (chksum & 0xffff) + (chksum >> 16); |
| 58 |
|
| 59 |
return chksum;
|
| 60 |
} |
| 61 |
|
| 62 |
/***********************************************************************
|
| 63 |
* Listener registry
|
| 64 |
**********************************************************************/
|
| 65 |
static eth_mac_addr_t _local_mac_addr;
|
| 66 |
static struct ip_addr _local_ip_addr; |
| 67 |
void register_addrs(const eth_mac_addr_t *mac_addr, const struct ip_addr *ip_addr){ |
| 68 |
_local_mac_addr = *mac_addr; |
| 69 |
_local_ip_addr = *ip_addr; |
| 70 |
} |
| 71 |
|
| 72 |
struct listener_entry {
|
| 73 |
unsigned short port; |
| 74 |
udp_receiver_t rcvr; |
| 75 |
}; |
| 76 |
|
| 77 |
static struct listener_entry listeners[MAX_UDP_LISTENERS]; |
| 78 |
|
| 79 |
void init_udp_listeners(void){ |
| 80 |
for (int i = 0; i < MAX_UDP_LISTENERS; i++){ |
| 81 |
listeners[i].rcvr = NULL;
|
| 82 |
} |
| 83 |
} |
| 84 |
|
| 85 |
static struct listener_entry * |
| 86 |
find_listener_by_port(unsigned short port) |
| 87 |
{
|
| 88 |
for (int i = 0; i < MAX_UDP_LISTENERS; i++){ |
| 89 |
if (port == listeners[i].port)
|
| 90 |
return &listeners[i];
|
| 91 |
} |
| 92 |
return 0; |
| 93 |
} |
| 94 |
|
| 95 |
static struct listener_entry * |
| 96 |
find_free_listener(void)
|
| 97 |
{
|
| 98 |
for (int i = 0; i < MAX_UDP_LISTENERS; i++){ |
| 99 |
if (listeners[i].rcvr == NULL) |
| 100 |
return &listeners[i];
|
| 101 |
} |
| 102 |
abort(); |
| 103 |
} |
| 104 |
|
| 105 |
void
|
| 106 |
register_udp_listener(int port, udp_receiver_t rcvr)
|
| 107 |
{
|
| 108 |
struct listener_entry *lx = find_listener_by_port(port);
|
| 109 |
if (lx)
|
| 110 |
lx->rcvr = rcvr; |
| 111 |
else {
|
| 112 |
lx = find_free_listener(); |
| 113 |
lx->port = port; |
| 114 |
lx->rcvr = rcvr; |
| 115 |
} |
| 116 |
} |
| 117 |
|
| 118 |
/***********************************************************************
|
| 119 |
* Protocol framer
|
| 120 |
**********************************************************************/
|
| 121 |
void setup_framer(
|
| 122 |
eth_mac_addr_t eth_dst, |
| 123 |
eth_mac_addr_t eth_src, |
| 124 |
struct socket_address sock_dst,
|
| 125 |
struct socket_address sock_src,
|
| 126 |
size_t which |
| 127 |
){
|
| 128 |
struct {
|
| 129 |
padded_eth_hdr_t eth; |
| 130 |
struct ip_hdr ip;
|
| 131 |
struct udp_hdr udp;
|
| 132 |
} frame; |
| 133 |
|
| 134 |
//-- load Ethernet header --//
|
| 135 |
frame.eth.dst = eth_dst; |
| 136 |
frame.eth.src = eth_src; |
| 137 |
frame.eth.ethertype = ETHERTYPE_IPV4; |
| 138 |
|
| 139 |
//-- load IPv4 header --//
|
| 140 |
IPH_VHLTOS_SET(&frame.ip, 4, 5, 0); |
| 141 |
IPH_LEN_SET(&frame.ip, 0);
|
| 142 |
IPH_ID_SET(&frame.ip, 0);
|
| 143 |
IPH_OFFSET_SET(&frame.ip, IP_DF); // don't fragment
|
| 144 |
IPH_TTL_SET(&frame.ip, 32);
|
| 145 |
IPH_PROTO_SET(&frame.ip, IP_PROTO_UDP); |
| 146 |
IPH_CHKSUM_SET(&frame.ip, 0);
|
| 147 |
frame.ip.src = sock_src.addr; |
| 148 |
frame.ip.dest = sock_dst.addr; |
| 149 |
IPH_CHKSUM_SET(&frame.ip, chksum_buffer( |
| 150 |
(unsigned short *) &frame.ip, |
| 151 |
sizeof(frame.ip)/sizeof(short), 0 |
| 152 |
)); |
| 153 |
|
| 154 |
//-- load UDP header --//
|
| 155 |
frame.udp.src = sock_src.port; |
| 156 |
frame.udp.dest = sock_dst.port; |
| 157 |
frame.udp.len = 0;
|
| 158 |
frame.udp.chksum = 0;
|
| 159 |
|
| 160 |
//copy into the framer table registers
|
| 161 |
memcpy_wa((void *)(sr_proto_framer_regs->table[which].entry + 1), &frame, sizeof(frame)); |
| 162 |
} |
| 163 |
|
| 164 |
/***********************************************************************
|
| 165 |
* Slow-path packet framing and transmission
|
| 166 |
**********************************************************************/
|
| 167 |
/*!
|
| 168 |
* low level routine to assembly an ethernet frame and send it.
|
| 169 |
*
|
| 170 |
* \param dst destination mac address
|
| 171 |
* \param ethertype ethertype field
|
| 172 |
* \param buf0 first part of data
|
| 173 |
* \param len0 length of first part of data
|
| 174 |
* \param buf1 second part of data
|
| 175 |
* \param len1 length of second part of data
|
| 176 |
* \param buf2 third part of data
|
| 177 |
* \param len2 length of third part of data
|
| 178 |
*/
|
| 179 |
static void |
| 180 |
send_pkt( |
| 181 |
eth_mac_addr_t dst, int ethertype,
|
| 182 |
const void *buf0, size_t len0, |
| 183 |
const void *buf1, size_t len1, |
| 184 |
const void *buf2, size_t len2 |
| 185 |
){
|
| 186 |
|
| 187 |
//control word for framed data
|
| 188 |
uint32_t ctrl_word = 0x0;
|
| 189 |
|
| 190 |
//assemble the ethernet header
|
| 191 |
padded_eth_hdr_t ehdr; |
| 192 |
ehdr.pad = 0;
|
| 193 |
ehdr.dst = dst; |
| 194 |
ehdr.src = _local_mac_addr; |
| 195 |
ehdr.ethertype = ethertype; |
| 196 |
|
| 197 |
//grab an out buffer and pointer
|
| 198 |
uint8_t *buff = (uint8_t *)pkt_ctrl_claim_outgoing_buffer(); |
| 199 |
uint8_t *p = buff; |
| 200 |
size_t total_len = 0;
|
| 201 |
|
| 202 |
//create a list of all buffers to copy
|
| 203 |
const void *buffs[] = {&ctrl_word, &ehdr, buf0, buf1, buf2}; |
| 204 |
size_t lens[] = {sizeof(ctrl_word), sizeof(ehdr), len0, len1, (len2 + 3) & ~3};
|
| 205 |
|
| 206 |
//copy each buffer into the out buffer
|
| 207 |
for (size_t i = 0; i < sizeof(buffs)/sizeof(buffs[0]); i++){ |
| 208 |
total_len += lens[i]; //use full length (not clipped)
|
| 209 |
size_t bytes_remaining = out_buff_size - (size_t)(p - buff); |
| 210 |
if (lens[i] > bytes_remaining) lens[i] = bytes_remaining;
|
| 211 |
if (lens[i] && ((lens[i] & 0x3) || (intptr_t) buffs[i] & 0x3)) |
| 212 |
printf("send_pkt: bad alignment of len and/or buf\n");
|
| 213 |
memcpy_wa(p, buffs[i], lens[i]); |
| 214 |
p += lens[i]; |
| 215 |
} |
| 216 |
|
| 217 |
//ensure that minimum length requirements are met
|
| 218 |
if (total_len < 64) total_len = 64; //60 + ctrl word |
| 219 |
|
| 220 |
pkt_ctrl_commit_outgoing_buffer(total_len/sizeof(uint32_t));
|
| 221 |
if (debug) printf("sent %d bytes\n", (int)total_len); |
| 222 |
} |
| 223 |
|
| 224 |
void
|
| 225 |
send_ip_pkt(struct ip_addr dst, int protocol, |
| 226 |
const void *buf0, size_t len0, |
| 227 |
const void *buf1, size_t len1) |
| 228 |
{
|
| 229 |
struct ip_hdr ip;
|
| 230 |
IPH_VHLTOS_SET(&ip, 4, 5, 0); |
| 231 |
IPH_LEN_SET(&ip, IP_HLEN + len0 + len1); |
| 232 |
IPH_ID_SET(&ip, 0);
|
| 233 |
IPH_OFFSET_SET(&ip, IP_DF); /* don't fragment */
|
| 234 |
IPH_TTL_SET(&ip, 32);
|
| 235 |
IPH_PROTO_SET(&ip, protocol); |
| 236 |
IPH_CHKSUM_SET(&ip, 0);
|
| 237 |
ip.src = _local_ip_addr; |
| 238 |
ip.dest = dst; |
| 239 |
|
| 240 |
IPH_CHKSUM_SET(&ip, ~chksum_buffer( |
| 241 |
(unsigned short *) &ip, sizeof(ip)/sizeof(short), 0 |
| 242 |
)); |
| 243 |
|
| 244 |
eth_mac_addr_t dst_mac; |
| 245 |
bool found = arp_cache_lookup_mac(&ip.dest, &dst_mac);
|
| 246 |
if (!found){
|
| 247 |
printf("net_common: failed to hit cache looking for ");
|
| 248 |
print_ip_addr(&ip.dest); |
| 249 |
newline(); |
| 250 |
return;
|
| 251 |
} |
| 252 |
|
| 253 |
send_pkt(dst_mac, ETHERTYPE_IPV4, |
| 254 |
&ip, sizeof(ip), buf0, len0, buf1, len1);
|
| 255 |
} |
| 256 |
|
| 257 |
void
|
| 258 |
send_udp_pkt(int src_port, struct socket_address dst, |
| 259 |
const void *buf, size_t len) |
| 260 |
{
|
| 261 |
struct udp_hdr udp _AL4;
|
| 262 |
udp.src = src_port; |
| 263 |
udp.dest = dst.port; |
| 264 |
udp.len = UDP_HLEN + len; |
| 265 |
udp.chksum = 0;
|
| 266 |
|
| 267 |
send_ip_pkt(dst.addr, IP_PROTO_UDP, |
| 268 |
&udp, sizeof(udp), buf, len);
|
| 269 |
} |
| 270 |
|
| 271 |
static void |
| 272 |
handle_udp_packet(struct ip_addr src_ip, struct ip_addr dst_ip, |
| 273 |
struct udp_hdr *udp, size_t len)
|
| 274 |
{
|
| 275 |
if (len != udp->len){
|
| 276 |
printf("UDP inconsistent lengths: %d %d\n", (int)len, udp->len); |
| 277 |
return;
|
| 278 |
} |
| 279 |
|
| 280 |
unsigned char *payload = ((unsigned char *) udp) + UDP_HLEN; |
| 281 |
int payload_len = len - UDP_HLEN;
|
| 282 |
|
| 283 |
if (0){ |
| 284 |
printf("\nUDP: src = %d dst = %d len = %d\n",
|
| 285 |
udp->src, udp->dest, udp->len); |
| 286 |
|
| 287 |
//print_bytes(0, payload, payload_len);
|
| 288 |
} |
| 289 |
|
| 290 |
struct listener_entry *lx = find_listener_by_port(udp->dest);
|
| 291 |
if (lx){
|
| 292 |
struct socket_address src = make_socket_address(src_ip, udp->src);
|
| 293 |
struct socket_address dst = make_socket_address(dst_ip, udp->dest);
|
| 294 |
lx->rcvr(src, dst, payload, payload_len); |
| 295 |
} |
| 296 |
} |
| 297 |
|
| 298 |
static void |
| 299 |
handle_icmp_packet(struct ip_addr src, struct ip_addr dst, |
| 300 |
struct icmp_echo_hdr *icmp, size_t len)
|
| 301 |
{
|
| 302 |
switch (icmp->type){
|
| 303 |
case ICMP_DUR: // Destinatino Unreachable |
| 304 |
if (icmp->code == ICMP_DUR_PORT){ // port unreachable |
| 305 |
//handle destination port unreachable (the host ctrl+c'd the app):
|
| 306 |
|
| 307 |
//filter out non udp data response
|
| 308 |
struct ip_hdr *ip = (struct ip_hdr *)(((uint8_t*)icmp) + sizeof(struct icmp_echo_hdr)); |
| 309 |
struct udp_hdr *udp = (struct udp_hdr *)(((char *)ip) + IP_HLEN); |
| 310 |
if (IPH_PROTO(ip) != IP_PROTO_UDP) break; |
| 311 |
|
| 312 |
struct listener_entry *lx = find_listener_by_port(udp->src);
|
| 313 |
if (lx){
|
| 314 |
struct socket_address src = make_socket_address(ip->src, udp->src);
|
| 315 |
struct socket_address dst = make_socket_address(ip->dest, udp->dest);
|
| 316 |
lx->rcvr(src, dst, NULL, 0); |
| 317 |
} |
| 318 |
|
| 319 |
putchar('i');
|
| 320 |
} |
| 321 |
else {
|
| 322 |
//printf("icmp dst unr (code: %d)", icmp->code);
|
| 323 |
putchar('i');
|
| 324 |
} |
| 325 |
break;
|
| 326 |
|
| 327 |
case ICMP_ECHO:{
|
| 328 |
const void *icmp_data_buff = ((uint8_t*)icmp) + sizeof(struct icmp_echo_hdr); |
| 329 |
size_t icmp_data_len = len - sizeof(struct icmp_echo_hdr); |
| 330 |
|
| 331 |
struct icmp_echo_hdr echo_reply;
|
| 332 |
echo_reply.type = 0;
|
| 333 |
echo_reply.code = 0;
|
| 334 |
echo_reply.chksum = 0;
|
| 335 |
echo_reply.id = icmp->id; |
| 336 |
echo_reply.seqno = icmp->seqno; |
| 337 |
echo_reply.chksum = ~chksum_buffer( //data checksum
|
| 338 |
(unsigned short *)icmp_data_buff, |
| 339 |
icmp_data_len/sizeof(short), |
| 340 |
chksum_buffer( //header checksum
|
| 341 |
(unsigned short *)&echo_reply, |
| 342 |
sizeof(echo_reply)/sizeof(short), |
| 343 |
0)
|
| 344 |
); |
| 345 |
|
| 346 |
send_ip_pkt( |
| 347 |
src, IP_PROTO_ICMP, |
| 348 |
&echo_reply, sizeof(echo_reply),
|
| 349 |
icmp_data_buff, icmp_data_len |
| 350 |
); |
| 351 |
break;
|
| 352 |
} |
| 353 |
default:
|
| 354 |
break;
|
| 355 |
} |
| 356 |
} |
| 357 |
|
| 358 |
static void |
| 359 |
send_arp_reply(struct arp_eth_ipv4 *req, eth_mac_addr_t our_mac)
|
| 360 |
{
|
| 361 |
struct arp_eth_ipv4 reply _AL4;
|
| 362 |
reply.ar_hrd = req->ar_hrd; |
| 363 |
reply.ar_pro = req->ar_pro; |
| 364 |
reply.ar_hln = req->ar_hln; |
| 365 |
reply.ar_pln = req->ar_pln; |
| 366 |
reply.ar_op = ARPOP_REPLY; |
| 367 |
memcpy(reply.ar_sha, &our_mac, 6);
|
| 368 |
memcpy(reply.ar_sip, req->ar_tip, 4);
|
| 369 |
memcpy(reply.ar_tha, req->ar_sha, 6);
|
| 370 |
memcpy(reply.ar_tip, req->ar_sip, 4);
|
| 371 |
|
| 372 |
eth_mac_addr_t t; |
| 373 |
memcpy(t.addr, reply.ar_tha, 6);
|
| 374 |
send_pkt(t, ETHERTYPE_ARP, &reply, sizeof(reply), 0, 0, 0, 0); |
| 375 |
} |
| 376 |
|
| 377 |
void net_common_send_arp_request(const struct ip_addr *addr){ |
| 378 |
struct arp_eth_ipv4 req _AL4;
|
| 379 |
req.ar_hrd = ARPHRD_ETHER; |
| 380 |
req.ar_pro = ETHERTYPE_IPV4; |
| 381 |
req.ar_hln = sizeof(eth_mac_addr_t);
|
| 382 |
req.ar_pln = sizeof(struct ip_addr); |
| 383 |
req.ar_op = ARPOP_REQUEST; |
| 384 |
memcpy(req.ar_sha, ethernet_mac_addr(), sizeof(eth_mac_addr_t));
|
| 385 |
memcpy(req.ar_sip, get_ip_addr(), sizeof(struct ip_addr)); |
| 386 |
memset(req.ar_tha, 0x00, sizeof(eth_mac_addr_t)); |
| 387 |
memcpy(req.ar_tip, addr, sizeof(struct ip_addr)); |
| 388 |
|
| 389 |
//send the request with a broadcast ethernet mac address
|
| 390 |
send_pkt(BCAST_MAC_ADDR, ETHERTYPE_ARP, &req, sizeof(req), 0, 0, 0, 0); |
| 391 |
} |
| 392 |
|
| 393 |
void send_gratuitous_arp(void){ |
| 394 |
struct arp_eth_ipv4 req _AL4;
|
| 395 |
req.ar_hrd = ARPHRD_ETHER; |
| 396 |
req.ar_pro = ETHERTYPE_IPV4; |
| 397 |
req.ar_hln = sizeof(eth_mac_addr_t);
|
| 398 |
req.ar_pln = sizeof(struct ip_addr); |
| 399 |
req.ar_op = ARPOP_REQUEST; |
| 400 |
memcpy(req.ar_sha, ethernet_mac_addr(), sizeof(eth_mac_addr_t));
|
| 401 |
memcpy(req.ar_sip, get_ip_addr(), sizeof(struct ip_addr)); |
| 402 |
memset(req.ar_tha, 0x00, sizeof(eth_mac_addr_t)); |
| 403 |
memcpy(req.ar_tip, get_ip_addr(), sizeof(struct ip_addr)); |
| 404 |
|
| 405 |
//send the request with a broadcast ethernet mac address
|
| 406 |
send_pkt(BCAST_MAC_ADDR, ETHERTYPE_ARP, &req, sizeof(req), 0, 0, 0, 0); |
| 407 |
} |
| 408 |
|
| 409 |
static void |
| 410 |
handle_arp_packet(struct arp_eth_ipv4 *p, size_t size)
|
| 411 |
{
|
| 412 |
if (size < sizeof(struct arp_eth_ipv4)){ |
| 413 |
printf("\nhandle_arp: weird size = %d\n", (int)size); |
| 414 |
return;
|
| 415 |
} |
| 416 |
|
| 417 |
if (0){ |
| 418 |
printf("ar_hrd = %d\n", p->ar_hrd);
|
| 419 |
printf("ar_pro = %d\n", p->ar_pro);
|
| 420 |
printf("ar_hln = %d\n", p->ar_hln);
|
| 421 |
printf("ar_pln = %d\n", p->ar_pln);
|
| 422 |
printf("ar_op = %d\n", p->ar_op);
|
| 423 |
printf("ar_sha = "); print_mac_addr(p->ar_sha); newline();
|
| 424 |
printf("ar_sip = "); print_ip_addr (p->ar_sip); newline();
|
| 425 |
printf("ar_tha = "); print_mac_addr(p->ar_tha); newline();
|
| 426 |
printf("ar_tip = "); print_ip_addr (p->ar_tip); newline();
|
| 427 |
} |
| 428 |
|
| 429 |
if (p->ar_hrd != ARPHRD_ETHER
|
| 430 |
|| p->ar_pro != ETHERTYPE_IPV4 |
| 431 |
|| p->ar_hln != 6
|
| 432 |
|| p->ar_pln != 4)
|
| 433 |
return;
|
| 434 |
|
| 435 |
if (p->ar_op == ARPOP_REPLY){
|
| 436 |
struct ip_addr ip_addr;
|
| 437 |
memcpy(&ip_addr, p->ar_sip, sizeof(ip_addr));
|
| 438 |
eth_mac_addr_t mac_addr; |
| 439 |
memcpy(&mac_addr, p->ar_sha, sizeof(mac_addr));
|
| 440 |
arp_cache_update(&ip_addr, &mac_addr); |
| 441 |
} |
| 442 |
|
| 443 |
if (p->ar_op != ARPOP_REQUEST)
|
| 444 |
return;
|
| 445 |
|
| 446 |
struct ip_addr sip;
|
| 447 |
struct ip_addr tip;
|
| 448 |
|
| 449 |
sip.addr = get_int32(p->ar_sip); |
| 450 |
tip.addr = get_int32(p->ar_tip); |
| 451 |
|
| 452 |
if (memcmp(&tip, &_local_ip_addr, sizeof(_local_ip_addr)) == 0){ // They're looking for us... |
| 453 |
send_arp_reply(p, _local_mac_addr); |
| 454 |
} |
| 455 |
} |
| 456 |
|
| 457 |
void
|
| 458 |
handle_eth_packet(uint32_t *p, size_t nlines) |
| 459 |
{
|
| 460 |
static size_t bcount = 0; |
| 461 |
if (debug) printf("===> %d\n", (int)bcount++); |
| 462 |
if (debug) print_buffer(p, nlines);
|
| 463 |
|
| 464 |
padded_eth_hdr_t *eth_hdr = (padded_eth_hdr_t *)p; |
| 465 |
|
| 466 |
if (eth_hdr->ethertype == ETHERTYPE_ARP){
|
| 467 |
struct arp_eth_ipv4 *arp = (struct arp_eth_ipv4 *)(p + 4); |
| 468 |
handle_arp_packet(arp, nlines*sizeof(uint32_t) - 14); |
| 469 |
} |
| 470 |
else if (eth_hdr->ethertype == ETHERTYPE_IPV4){ |
| 471 |
struct ip_hdr *ip = (struct ip_hdr *)(p + 4); |
| 472 |
if (IPH_V(ip) != 4 || IPH_HL(ip) != 5) // ignore pkts w/ bad version or options |
| 473 |
return;
|
| 474 |
|
| 475 |
if (IPH_OFFSET(ip) & (IP_MF | IP_OFFMASK)) // ignore fragmented packets |
| 476 |
return;
|
| 477 |
|
| 478 |
// filter on dest ip addr (should be broadcast or for us)
|
| 479 |
bool is_bcast = memcmp(ð_hdr->dst, &BCAST_MAC_ADDR, sizeof(BCAST_MAC_ADDR)) == 0; |
| 480 |
bool is_my_ip = memcmp(&ip->dest, &_local_ip_addr, sizeof(_local_ip_addr)) == 0; |
| 481 |
if (!is_bcast && !is_my_ip) return; |
| 482 |
|
| 483 |
arp_cache_update(&ip->src, (eth_mac_addr_t *)(((char *)p)+8)); |
| 484 |
|
| 485 |
int protocol = IPH_PROTO(ip);
|
| 486 |
int len = IPH_LEN(ip) - IP_HLEN;
|
| 487 |
|
| 488 |
switch (protocol){
|
| 489 |
case IP_PROTO_UDP:
|
| 490 |
handle_udp_packet(ip->src, ip->dest, (struct udp_hdr *)(((char *)ip) + IP_HLEN), len); |
| 491 |
break;
|
| 492 |
|
| 493 |
case IP_PROTO_ICMP:
|
| 494 |
handle_icmp_packet(ip->src, ip->dest, (struct icmp_echo_hdr *)(((char *)ip) + IP_HLEN), len); |
| 495 |
break;
|
| 496 |
|
| 497 |
default: // ignore |
| 498 |
break;
|
| 499 |
} |
| 500 |
} |
| 501 |
else
|
| 502 |
return; // Not ARP or IPV4, ignore |
| 503 |
} |