mirror of
https://github.com/saymrwulf/uhd.git
synced 2026-05-16 21:10:10 +00:00
503 lines
15 KiB
C
503 lines
15 KiB
C
/*
|
|
* Copyright 2009-2012 Ettus Research LLC
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
#include "net_common.h"
|
|
#include "banal.h"
|
|
#include <hal_io.h>
|
|
#include <memory_map.h>
|
|
#include <memcpy_wa.h>
|
|
#include <ethernet.h>
|
|
#include <net/padded_eth_hdr.h>
|
|
#include <lwip/ip.h>
|
|
#include <lwip/udp.h>
|
|
#include <lwip/icmp.h>
|
|
#include <stdlib.h>
|
|
#include <nonstdio.h>
|
|
#include "arp_cache.h"
|
|
#include "if_arp.h"
|
|
#include <ethertype.h>
|
|
#include <string.h>
|
|
#include "pkt_ctrl.h"
|
|
|
|
/***********************************************************************
|
|
* Constants + Globals
|
|
**********************************************************************/
|
|
static const bool debug = false;
|
|
static const size_t out_buff_size = 2048;
|
|
static const eth_mac_addr_t BCAST_MAC_ADDR = {{0xff, 0xff, 0xff, 0xff, 0xff, 0xff}};
|
|
#define MAX_UDP_LISTENERS 10
|
|
|
|
/***********************************************************************
|
|
* 16-bit one's complement sum
|
|
**********************************************************************/
|
|
static uint32_t chksum_buffer(
|
|
uint16_t *buf, size_t nshorts,
|
|
uint32_t initial_chksum
|
|
){
|
|
uint32_t chksum = initial_chksum;
|
|
for (size_t i = 0; i < nshorts; i++) chksum += buf[i];
|
|
|
|
while (chksum >> 16) chksum = (chksum & 0xffff) + (chksum >> 16);
|
|
|
|
return chksum;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* Listener registry
|
|
**********************************************************************/
|
|
static eth_mac_addr_t _local_mac_addr;
|
|
static struct ip_addr _local_ip_addr;
|
|
void register_addrs(const eth_mac_addr_t *mac_addr, const struct ip_addr *ip_addr){
|
|
_local_mac_addr = *mac_addr;
|
|
_local_ip_addr = *ip_addr;
|
|
}
|
|
|
|
struct listener_entry {
|
|
unsigned short port;
|
|
udp_receiver_t rcvr;
|
|
};
|
|
|
|
static struct listener_entry listeners[MAX_UDP_LISTENERS];
|
|
|
|
void init_udp_listeners(void){
|
|
for (int i = 0; i < MAX_UDP_LISTENERS; i++){
|
|
listeners[i].rcvr = NULL;
|
|
}
|
|
}
|
|
|
|
static struct listener_entry *
|
|
find_listener_by_port(unsigned short port)
|
|
{
|
|
for (int i = 0; i < MAX_UDP_LISTENERS; i++){
|
|
if (port == listeners[i].port)
|
|
return &listeners[i];
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static struct listener_entry *
|
|
find_free_listener(void)
|
|
{
|
|
for (int i = 0; i < MAX_UDP_LISTENERS; i++){
|
|
if (listeners[i].rcvr == NULL)
|
|
return &listeners[i];
|
|
}
|
|
abort();
|
|
}
|
|
|
|
void
|
|
register_udp_listener(int port, udp_receiver_t rcvr)
|
|
{
|
|
struct listener_entry *lx = find_listener_by_port(port);
|
|
if (lx)
|
|
lx->rcvr = rcvr;
|
|
else {
|
|
lx = find_free_listener();
|
|
lx->port = port;
|
|
lx->rcvr = rcvr;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
* Protocol framer
|
|
**********************************************************************/
|
|
void setup_framer(
|
|
eth_mac_addr_t eth_dst,
|
|
eth_mac_addr_t eth_src,
|
|
struct socket_address sock_dst,
|
|
struct socket_address sock_src,
|
|
size_t which
|
|
){
|
|
struct {
|
|
padded_eth_hdr_t eth;
|
|
struct ip_hdr ip;
|
|
struct udp_hdr udp;
|
|
} frame;
|
|
|
|
//-- load Ethernet header --//
|
|
frame.eth.dst = eth_dst;
|
|
frame.eth.src = eth_src;
|
|
frame.eth.ethertype = ETHERTYPE_IPV4;
|
|
|
|
//-- load IPv4 header --//
|
|
IPH_VHLTOS_SET(&frame.ip, 4, 5, 0);
|
|
IPH_LEN_SET(&frame.ip, 0);
|
|
IPH_ID_SET(&frame.ip, 0);
|
|
IPH_OFFSET_SET(&frame.ip, IP_DF); // don't fragment
|
|
IPH_TTL_SET(&frame.ip, 32);
|
|
IPH_PROTO_SET(&frame.ip, IP_PROTO_UDP);
|
|
IPH_CHKSUM_SET(&frame.ip, 0);
|
|
frame.ip.src = sock_src.addr;
|
|
frame.ip.dest = sock_dst.addr;
|
|
IPH_CHKSUM_SET(&frame.ip, chksum_buffer(
|
|
(unsigned short *) &frame.ip,
|
|
sizeof(frame.ip)/sizeof(short), 0
|
|
));
|
|
|
|
//-- load UDP header --//
|
|
frame.udp.src = sock_src.port;
|
|
frame.udp.dest = sock_dst.port;
|
|
frame.udp.len = 0;
|
|
frame.udp.chksum = 0;
|
|
|
|
//copy into the framer table registers
|
|
memcpy_wa((void *)(sr_proto_framer_regs->table[which].entry + 1), &frame, sizeof(frame));
|
|
}
|
|
|
|
/***********************************************************************
|
|
* Slow-path packet framing and transmission
|
|
**********************************************************************/
|
|
/*!
|
|
* low level routine to assembly an ethernet frame and send it.
|
|
*
|
|
* \param dst destination mac address
|
|
* \param ethertype ethertype field
|
|
* \param buf0 first part of data
|
|
* \param len0 length of first part of data
|
|
* \param buf1 second part of data
|
|
* \param len1 length of second part of data
|
|
* \param buf2 third part of data
|
|
* \param len2 length of third part of data
|
|
*/
|
|
static void
|
|
send_pkt(
|
|
eth_mac_addr_t dst, int ethertype,
|
|
const void *buf0, size_t len0,
|
|
const void *buf1, size_t len1,
|
|
const void *buf2, size_t len2
|
|
){
|
|
|
|
//control word for framed data
|
|
uint32_t ctrl_word = 0x0;
|
|
|
|
//assemble the ethernet header
|
|
padded_eth_hdr_t ehdr;
|
|
ehdr.pad = 0;
|
|
ehdr.dst = dst;
|
|
ehdr.src = _local_mac_addr;
|
|
ehdr.ethertype = ethertype;
|
|
|
|
//grab an out buffer and pointer
|
|
uint8_t *buff = (uint8_t *)pkt_ctrl_claim_outgoing_buffer();
|
|
uint8_t *p = buff;
|
|
size_t total_len = 0;
|
|
|
|
//create a list of all buffers to copy
|
|
const void *buffs[] = {&ctrl_word, &ehdr, buf0, buf1, buf2};
|
|
size_t lens[] = {sizeof(ctrl_word), sizeof(ehdr), len0, len1, (len2 + 3) & ~3};
|
|
|
|
//copy each buffer into the out buffer
|
|
for (size_t i = 0; i < sizeof(buffs)/sizeof(buffs[0]); i++){
|
|
total_len += lens[i]; //use full length (not clipped)
|
|
size_t bytes_remaining = out_buff_size - (size_t)(p - buff);
|
|
if (lens[i] > bytes_remaining) lens[i] = bytes_remaining;
|
|
if (lens[i] && ((lens[i] & 0x3) || (intptr_t) buffs[i] & 0x3))
|
|
printf("send_pkt: bad alignment of len and/or buf\n");
|
|
memcpy_wa(p, buffs[i], lens[i]);
|
|
p += lens[i];
|
|
}
|
|
|
|
//ensure that minimum length requirements are met
|
|
if (total_len < 64) total_len = 64; //60 + ctrl word
|
|
|
|
pkt_ctrl_commit_outgoing_buffer(total_len/sizeof(uint32_t));
|
|
if (debug) printf("sent %d bytes\n", (int)total_len);
|
|
}
|
|
|
|
void
|
|
send_ip_pkt(struct ip_addr dst, int protocol,
|
|
const void *buf0, size_t len0,
|
|
const void *buf1, size_t len1)
|
|
{
|
|
struct ip_hdr ip;
|
|
IPH_VHLTOS_SET(&ip, 4, 5, 0);
|
|
IPH_LEN_SET(&ip, IP_HLEN + len0 + len1);
|
|
IPH_ID_SET(&ip, 0);
|
|
IPH_OFFSET_SET(&ip, IP_DF); /* don't fragment */
|
|
IPH_TTL_SET(&ip, 32);
|
|
IPH_PROTO_SET(&ip, protocol);
|
|
IPH_CHKSUM_SET(&ip, 0);
|
|
ip.src = _local_ip_addr;
|
|
ip.dest = dst;
|
|
|
|
IPH_CHKSUM_SET(&ip, ~chksum_buffer(
|
|
(unsigned short *) &ip, sizeof(ip)/sizeof(short), 0
|
|
));
|
|
|
|
eth_mac_addr_t dst_mac;
|
|
bool found = arp_cache_lookup_mac(&ip.dest, &dst_mac);
|
|
if (!found){
|
|
printf("net_common: failed to hit cache looking for ");
|
|
print_ip_addr(&ip.dest);
|
|
newline();
|
|
return;
|
|
}
|
|
|
|
send_pkt(dst_mac, ETHERTYPE_IPV4,
|
|
&ip, sizeof(ip), buf0, len0, buf1, len1);
|
|
}
|
|
|
|
void
|
|
send_udp_pkt(int src_port, struct socket_address dst,
|
|
const void *buf, size_t len)
|
|
{
|
|
struct udp_hdr udp _AL4;
|
|
udp.src = src_port;
|
|
udp.dest = dst.port;
|
|
udp.len = UDP_HLEN + len;
|
|
udp.chksum = 0;
|
|
|
|
send_ip_pkt(dst.addr, IP_PROTO_UDP,
|
|
&udp, sizeof(udp), buf, len);
|
|
}
|
|
|
|
static void
|
|
handle_udp_packet(struct ip_addr src_ip, struct ip_addr dst_ip,
|
|
struct udp_hdr *udp, size_t len)
|
|
{
|
|
if (len != udp->len){
|
|
printf("UDP inconsistent lengths: %d %d\n", (int)len, udp->len);
|
|
return;
|
|
}
|
|
|
|
unsigned char *payload = ((unsigned char *) udp) + UDP_HLEN;
|
|
int payload_len = len - UDP_HLEN;
|
|
|
|
if (0){
|
|
printf("\nUDP: src = %d dst = %d len = %d\n",
|
|
udp->src, udp->dest, udp->len);
|
|
|
|
//print_bytes(0, payload, payload_len);
|
|
}
|
|
|
|
struct listener_entry *lx = find_listener_by_port(udp->dest);
|
|
if (lx){
|
|
struct socket_address src = make_socket_address(src_ip, udp->src);
|
|
struct socket_address dst = make_socket_address(dst_ip, udp->dest);
|
|
lx->rcvr(src, dst, payload, payload_len);
|
|
}
|
|
}
|
|
|
|
static void
|
|
handle_icmp_packet(struct ip_addr src, struct ip_addr dst,
|
|
struct icmp_echo_hdr *icmp, size_t len)
|
|
{
|
|
switch (icmp->type){
|
|
case ICMP_DUR: // Destinatino Unreachable
|
|
if (icmp->code == ICMP_DUR_PORT){ // port unreachable
|
|
//handle destination port unreachable (the host ctrl+c'd the app):
|
|
|
|
//filter out non udp data response
|
|
struct ip_hdr *ip = (struct ip_hdr *)(((uint8_t*)icmp) + sizeof(struct icmp_echo_hdr));
|
|
struct udp_hdr *udp = (struct udp_hdr *)(((char *)ip) + IP_HLEN);
|
|
if (IPH_PROTO(ip) != IP_PROTO_UDP) break;
|
|
|
|
struct listener_entry *lx = find_listener_by_port(udp->src);
|
|
if (lx){
|
|
struct socket_address src = make_socket_address(ip->src, udp->src);
|
|
struct socket_address dst = make_socket_address(ip->dest, udp->dest);
|
|
lx->rcvr(src, dst, NULL, 0);
|
|
}
|
|
|
|
putchar('i');
|
|
}
|
|
else {
|
|
//printf("icmp dst unr (code: %d)", icmp->code);
|
|
putchar('i');
|
|
}
|
|
break;
|
|
|
|
case ICMP_ECHO:{
|
|
const void *icmp_data_buff = ((uint8_t*)icmp) + sizeof(struct icmp_echo_hdr);
|
|
size_t icmp_data_len = len - sizeof(struct icmp_echo_hdr);
|
|
|
|
struct icmp_echo_hdr echo_reply;
|
|
echo_reply.type = 0;
|
|
echo_reply.code = 0;
|
|
echo_reply.chksum = 0;
|
|
echo_reply.id = icmp->id;
|
|
echo_reply.seqno = icmp->seqno;
|
|
echo_reply.chksum = ~chksum_buffer( //data checksum
|
|
(unsigned short *)icmp_data_buff,
|
|
icmp_data_len/sizeof(short),
|
|
chksum_buffer( //header checksum
|
|
(unsigned short *)&echo_reply,
|
|
sizeof(echo_reply)/sizeof(short),
|
|
0)
|
|
);
|
|
|
|
send_ip_pkt(
|
|
src, IP_PROTO_ICMP,
|
|
&echo_reply, sizeof(echo_reply),
|
|
icmp_data_buff, icmp_data_len
|
|
);
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
send_arp_reply(struct arp_eth_ipv4 *req, eth_mac_addr_t our_mac)
|
|
{
|
|
struct arp_eth_ipv4 reply _AL4;
|
|
reply.ar_hrd = req->ar_hrd;
|
|
reply.ar_pro = req->ar_pro;
|
|
reply.ar_hln = req->ar_hln;
|
|
reply.ar_pln = req->ar_pln;
|
|
reply.ar_op = ARPOP_REPLY;
|
|
memcpy(reply.ar_sha, &our_mac, 6);
|
|
memcpy(reply.ar_sip, req->ar_tip, 4);
|
|
memcpy(reply.ar_tha, req->ar_sha, 6);
|
|
memcpy(reply.ar_tip, req->ar_sip, 4);
|
|
|
|
eth_mac_addr_t t;
|
|
memcpy(t.addr, reply.ar_tha, 6);
|
|
send_pkt(t, ETHERTYPE_ARP, &reply, sizeof(reply), 0, 0, 0, 0);
|
|
}
|
|
|
|
void net_common_send_arp_request(const struct ip_addr *addr){
|
|
struct arp_eth_ipv4 req _AL4;
|
|
req.ar_hrd = ARPHRD_ETHER;
|
|
req.ar_pro = ETHERTYPE_IPV4;
|
|
req.ar_hln = sizeof(eth_mac_addr_t);
|
|
req.ar_pln = sizeof(struct ip_addr);
|
|
req.ar_op = ARPOP_REQUEST;
|
|
memcpy(req.ar_sha, ethernet_mac_addr(), sizeof(eth_mac_addr_t));
|
|
memcpy(req.ar_sip, get_ip_addr(), sizeof(struct ip_addr));
|
|
memset(req.ar_tha, 0x00, sizeof(eth_mac_addr_t));
|
|
memcpy(req.ar_tip, addr, sizeof(struct ip_addr));
|
|
|
|
//send the request with a broadcast ethernet mac address
|
|
send_pkt(BCAST_MAC_ADDR, ETHERTYPE_ARP, &req, sizeof(req), 0, 0, 0, 0);
|
|
}
|
|
|
|
void send_gratuitous_arp(void){
|
|
struct arp_eth_ipv4 req _AL4;
|
|
req.ar_hrd = ARPHRD_ETHER;
|
|
req.ar_pro = ETHERTYPE_IPV4;
|
|
req.ar_hln = sizeof(eth_mac_addr_t);
|
|
req.ar_pln = sizeof(struct ip_addr);
|
|
req.ar_op = ARPOP_REQUEST;
|
|
memcpy(req.ar_sha, ethernet_mac_addr(), sizeof(eth_mac_addr_t));
|
|
memcpy(req.ar_sip, get_ip_addr(), sizeof(struct ip_addr));
|
|
memset(req.ar_tha, 0x00, sizeof(eth_mac_addr_t));
|
|
memcpy(req.ar_tip, get_ip_addr(), sizeof(struct ip_addr));
|
|
|
|
//send the request with a broadcast ethernet mac address
|
|
send_pkt(BCAST_MAC_ADDR, ETHERTYPE_ARP, &req, sizeof(req), 0, 0, 0, 0);
|
|
}
|
|
|
|
static void
|
|
handle_arp_packet(struct arp_eth_ipv4 *p, size_t size)
|
|
{
|
|
if (size < sizeof(struct arp_eth_ipv4)){
|
|
printf("\nhandle_arp: weird size = %d\n", (int)size);
|
|
return;
|
|
}
|
|
|
|
if (0){
|
|
printf("ar_hrd = %d\n", p->ar_hrd);
|
|
printf("ar_pro = %d\n", p->ar_pro);
|
|
printf("ar_hln = %d\n", p->ar_hln);
|
|
printf("ar_pln = %d\n", p->ar_pln);
|
|
printf("ar_op = %d\n", p->ar_op);
|
|
printf("ar_sha = "); print_mac_addr(p->ar_sha); newline();
|
|
printf("ar_sip = "); print_ip_addr (p->ar_sip); newline();
|
|
printf("ar_tha = "); print_mac_addr(p->ar_tha); newline();
|
|
printf("ar_tip = "); print_ip_addr (p->ar_tip); newline();
|
|
}
|
|
|
|
if (p->ar_hrd != ARPHRD_ETHER
|
|
|| p->ar_pro != ETHERTYPE_IPV4
|
|
|| p->ar_hln != 6
|
|
|| p->ar_pln != 4)
|
|
return;
|
|
|
|
if (p->ar_op == ARPOP_REPLY){
|
|
struct ip_addr ip_addr;
|
|
memcpy(&ip_addr, p->ar_sip, sizeof(ip_addr));
|
|
eth_mac_addr_t mac_addr;
|
|
memcpy(&mac_addr, p->ar_sha, sizeof(mac_addr));
|
|
arp_cache_update(&ip_addr, &mac_addr);
|
|
}
|
|
|
|
if (p->ar_op != ARPOP_REQUEST)
|
|
return;
|
|
|
|
struct ip_addr sip;
|
|
struct ip_addr tip;
|
|
|
|
sip.addr = get_int32(p->ar_sip);
|
|
tip.addr = get_int32(p->ar_tip);
|
|
|
|
if (memcmp(&tip, &_local_ip_addr, sizeof(_local_ip_addr)) == 0){ // They're looking for us...
|
|
send_arp_reply(p, _local_mac_addr);
|
|
}
|
|
}
|
|
|
|
void
|
|
handle_eth_packet(uint32_t *p, size_t nlines)
|
|
{
|
|
static size_t bcount = 0;
|
|
if (debug) printf("===> %d\n", (int)bcount++);
|
|
if (debug) print_buffer(p, nlines);
|
|
|
|
padded_eth_hdr_t *eth_hdr = (padded_eth_hdr_t *)p;
|
|
|
|
if (eth_hdr->ethertype == ETHERTYPE_ARP){
|
|
struct arp_eth_ipv4 *arp = (struct arp_eth_ipv4 *)(p + 4);
|
|
handle_arp_packet(arp, nlines*sizeof(uint32_t) - 14);
|
|
}
|
|
else if (eth_hdr->ethertype == ETHERTYPE_IPV4){
|
|
struct ip_hdr *ip = (struct ip_hdr *)(p + 4);
|
|
if (IPH_V(ip) != 4 || IPH_HL(ip) != 5) // ignore pkts w/ bad version or options
|
|
return;
|
|
|
|
if (IPH_OFFSET(ip) & (IP_MF | IP_OFFMASK)) // ignore fragmented packets
|
|
return;
|
|
|
|
// filter on dest ip addr (should be broadcast or for us)
|
|
bool is_bcast = memcmp(ð_hdr->dst, &BCAST_MAC_ADDR, sizeof(BCAST_MAC_ADDR)) == 0;
|
|
bool is_my_ip = memcmp(&ip->dest, &_local_ip_addr, sizeof(_local_ip_addr)) == 0;
|
|
if (!is_bcast && !is_my_ip) return;
|
|
|
|
arp_cache_update(&ip->src, (eth_mac_addr_t *)(((char *)p)+8));
|
|
|
|
int protocol = IPH_PROTO(ip);
|
|
int len = IPH_LEN(ip) - IP_HLEN;
|
|
|
|
switch (protocol){
|
|
case IP_PROTO_UDP:
|
|
handle_udp_packet(ip->src, ip->dest, (struct udp_hdr *)(((char *)ip) + IP_HLEN), len);
|
|
break;
|
|
|
|
case IP_PROTO_ICMP:
|
|
handle_icmp_packet(ip->src, ip->dest, (struct icmp_echo_hdr *)(((char *)ip) + IP_HLEN), len);
|
|
break;
|
|
|
|
default: // ignore
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
return; // Not ARP or IPV4, ignore
|
|
}
|