/* -*- c -*- */
/*
* Copyright 2009,2010 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 .
*/
#ifdef HAVE_CONFIG_H
#include
#endif
#include "net_common.h"
#include "banal.h"
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "arp_cache.h"
#include "if_arp.h"
#include
#include
int cpu_tx_buf_dest_port = PORT_ETH;
// If this is non-zero, this dbsm could be writing to the ethernet
dbsm_t *ac_could_be_sending_to_eth;
static inline bool
ip_addr_eq(const struct ip_addr a, const struct ip_addr b)
{
return a.addr == b.addr;
}
// ------------------------------------------------------------------------
get_eth_mac_addr_t _get_eth_mac_addr = NULL;
void register_get_eth_mac_addr(get_eth_mac_addr_t get_eth_mac_addr){
_get_eth_mac_addr = get_eth_mac_addr;
}
get_ip_addr_t _get_ip_addr = NULL;
void register_get_ip_addr(get_ip_addr_t get_ip_addr){
_get_ip_addr = get_ip_addr;
}
//-------------------------------------------------------------------------
#define MAX_UDP_LISTENERS 6
struct listener_entry {
unsigned short port;
udp_receiver_t rcvr;
};
static struct listener_entry listeners[MAX_UDP_LISTENERS];
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 == 0)
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;
}
}
// ------------------------------------------------------------------------
/*!
* 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)
{
// Wait for buffer to become idle
// FIXME can this ever not be ready?
//hal_set_leds(LED_BUF_BUSY, LED_BUF_BUSY);
while((buffer_pool_status->status & BPS_IDLE(CPU_TX_BUF)) == 0)
;
//hal_set_leds(0, LED_BUF_BUSY);
// Assemble the header
padded_eth_hdr_t ehdr;
ehdr.pad = 0;
ehdr.dst = dst;
ehdr.src = _get_eth_mac_addr();
ehdr.ethertype = ethertype;
uint32_t *p = buffer_ram(CPU_TX_BUF);
// Copy the pieces into the buffer
*p++ = 0x0; // slow path
memcpy_wa(p, &ehdr, sizeof(ehdr)); // 4 lines
p += sizeof(ehdr)/sizeof(uint32_t);
// FIXME modify memcpy_wa to do read/modify/write if reqd
if (len0 && ((len0 & 0x3) || (intptr_t) buf0 & 0x3))
printf("send_pkt: bad alignment of len0 and/or buf0\n");
if (len1 && ((len1 & 0x3) || (intptr_t) buf1 & 0x3))
printf("send_pkt: bad alignment of len1 and/or buf1\n");
if (len2 && ((len2 & 0x3) || (intptr_t) buf2 & 0x3))
printf("send_pkt: bad alignment of len2 and/or buf2\n");
if (len0){
memcpy_wa(p, buf0, len0);
p += len0/sizeof(uint32_t);
}
if (len1){
memcpy_wa(p, buf1, len1);
p += len1/sizeof(uint32_t);
}
if (len2){
memcpy_wa(p, buf2, len2);
p += len2/sizeof(uint32_t);
}
size_t total_len = (p - buffer_ram(CPU_TX_BUF)) * sizeof(uint32_t);
if (total_len < 60) // ensure that we don't try to send a short packet
total_len = 60;
// wait until nobody else is sending to the ethernet
if (ac_could_be_sending_to_eth){
//hal_set_leds(LED_ETH_BUSY, LED_ETH_BUSY);
dbsm_wait_for_opening(ac_could_be_sending_to_eth);
//hal_set_leds(0x0, LED_ETH_BUSY);
}
if (0){
printf("send_pkt to port %d, len = %d\n",
cpu_tx_buf_dest_port, (int) total_len);
print_buffer(buffer_ram(CPU_TX_BUF), total_len/4);
}
// fire it off
bp_send_from_buf(CPU_TX_BUF, cpu_tx_buf_dest_port, 1, 0, total_len/4);
// wait for it to complete (not long, it's a small pkt)
while((buffer_pool_status->status & (BPS_DONE(CPU_TX_BUF) | BPS_ERROR(CPU_TX_BUF))) == 0)
;
bp_clear_buf(CPU_TX_BUF);
}
unsigned int
chksum_buffer(unsigned short *buf, int nshorts, unsigned int initial_chksum)
{
unsigned int chksum = initial_chksum;
for (int i = 0; i < nshorts; i++)
CHKSUM(buf[i], &chksum);
return chksum;
}
void
send_ip_pkt(struct ip_addr dst, int protocol,
const void *buf0, size_t len0,
const void *buf1, size_t len1)
{
struct ip_addr src = _get_ip_addr();
int ttl = 32;
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 */
ip._ttl_proto = (ttl << 8) | (protocol & 0xff);
ip._chksum = 0;
ip.src = src;
ip.dest = dst;
ip._chksum = ~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(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
//stop_streaming(); //FIXME
if (icmp->code == ICMP_DUR_PORT){ // port unreachable
//struct udp_hdr *udp = (struct udp_hdr *)((char *)icmp + 28);
//printf("icmp port unr %d\n", udp->dest);
putchar('i');
}
else {
//printf("icmp dst unr (code: %d)", icmp->code);
putchar('i');
}
break;
case ICMP_ECHO:
default:
break;
}
}
static void __attribute__((unused))
print_arp_ip(const unsigned char ip[4])
{
printf("%d.%d.%d.%d", ip[0], ip[1], ip[2],ip[3]);
}
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);
}
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_arp_ip(p->ar_sip); newline();
printf("ar_tha = "); print_mac_addr(p->ar_tha); newline();
printf("ar_tip = "); print_arp_ip(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_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 (ip_addr_eq(tip, _get_ip_addr())){ // They're looking for us...
send_arp_reply(p, _get_eth_mac_addr());
}
}
void
handle_eth_packet(uint32_t *p, size_t nlines)
{
//print_buffer(p, nlines);
int ethertype = p[3] & 0xffff;
if (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 (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;
// FIXME filter on dest ip addr (should be broadcast or for us)
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
}
// ------------------------------------------------------------------------
void
print_ip(struct ip_addr ip)
{
unsigned int t = ntohl(ip.addr);
printf("%d.%d.%d.%d",
(t >> 24) & 0xff,
(t >> 16) & 0xff,
(t >> 8) & 0xff,
t & 0xff);
}