2006-06-18 00:41:10 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "contiki.h"
|
|
|
|
|
|
|
|
#include "net/uip.h"
|
|
|
|
#include "net/uip-fw.h"
|
|
|
|
#define BUF ((struct uip_tcpip_hdr *)&uip_buf[UIP_LLH_LEN])
|
|
|
|
|
|
|
|
#include "dev/leds.h"
|
|
|
|
|
|
|
|
#include "dev/cc2420.h"
|
|
|
|
|
|
|
|
#include "net/uaodv.h"
|
|
|
|
#include "net/uaodv-rt.h"
|
|
|
|
|
2007-02-01 15:02:19 +01:00
|
|
|
#if 0
|
|
|
|
#define PRINTF(...) printf(__VA_ARGS__)
|
|
|
|
#else
|
|
|
|
#define PRINTF(...) do {} while (0)
|
|
|
|
#endif
|
|
|
|
|
2006-06-18 00:41:10 +02:00
|
|
|
#define in_my_network(a) \
|
2006-08-09 18:13:39 +02:00
|
|
|
(((a[0] ^ cc2420if.ipaddr.u16[0]) & cc2420if.netmask.u16[0]) == 0 && \
|
|
|
|
((a[1] ^ cc2420if.ipaddr.u16[1]) & cc2420if.netmask.u16[1]) == 0)
|
2006-06-18 00:41:10 +02:00
|
|
|
|
|
|
|
u8_t
|
|
|
|
cc2420_send_uaodv(void)
|
|
|
|
{
|
2007-02-01 15:02:19 +01:00
|
|
|
int ret;
|
2006-06-18 00:41:10 +02:00
|
|
|
struct hdr_802_15 h;
|
|
|
|
|
|
|
|
h.len = MAC_HDR_LEN + uip_len + 2; /* Including footer[2]. */
|
|
|
|
h.fc0 = FC0_TYPE_DATA | FC0_REQ_ACK | FC0_INTRA_PAN;
|
|
|
|
h.fc1 = FC1_DST_16 | FC1_SRC_16;
|
|
|
|
|
2006-08-09 18:13:39 +02:00
|
|
|
h.src = uip_hostaddr.u16[1];
|
|
|
|
if (uip_ipaddr_cmp(&BUF->destipaddr, &uip_broadcast_addr))
|
2006-06-18 00:41:10 +02:00
|
|
|
h.dst = 0xffff;
|
|
|
|
else {
|
|
|
|
uip_ipaddr_t *next_gw;
|
|
|
|
|
2006-08-09 18:13:39 +02:00
|
|
|
if (in_my_network(BUF->destipaddr.u16))
|
2006-06-18 00:41:10 +02:00
|
|
|
next_gw = &BUF->destipaddr;
|
|
|
|
else
|
|
|
|
next_gw = &uip_draddr; /* Default router. */
|
|
|
|
|
2007-04-04 14:12:21 +02:00
|
|
|
if (cc2420_check_remote(next_gw->u16[1]) == REMOTE_NO)
|
2006-08-09 18:13:39 +02:00
|
|
|
h.dst = next_gw->u16[1]; /* local, use ucast */
|
2006-06-18 00:41:10 +02:00
|
|
|
else { /* remote or unknown */
|
|
|
|
struct uaodv_rt_entry *route = uaodv_request_route_to(next_gw);
|
|
|
|
|
|
|
|
if (route == NULL) {
|
2006-08-09 18:13:39 +02:00
|
|
|
h.dst = next_gw->u16[1]; /* try local while waiting for route */
|
2006-06-18 00:41:10 +02:00
|
|
|
} else {
|
2007-04-04 13:46:47 +02:00
|
|
|
if (cc2420_check_remote(route->nexthop.u16[1]) == REMOTE_YES) {
|
2007-02-01 15:02:19 +01:00
|
|
|
PRINTF("LOST 0x%04x\n", route->nexthop.u16[1]);
|
2006-06-18 00:41:10 +02:00
|
|
|
/* Send bad route notification? */
|
|
|
|
#ifdef UAODV_BAD_ROUTE
|
|
|
|
uaodv_bad_route(route);
|
|
|
|
#endif
|
|
|
|
uaodv_rt_remove(route);
|
|
|
|
h.dst = 0xffff; /* revert to bcast */
|
|
|
|
} else /* unknown */ {
|
|
|
|
/* This will implicitly update neigbour table. */
|
2006-08-09 18:13:39 +02:00
|
|
|
h.dst = route->nexthop.u16[1];
|
2006-06-18 00:41:10 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-04-04 13:46:47 +02:00
|
|
|
u8_t color = (cc2420_ack_received) ? LEDS_BLUE : LEDS_RED;
|
|
|
|
leds_toggle(color);
|
|
|
|
|
2006-06-18 00:41:10 +02:00
|
|
|
/* Don't request MAC level ACKs for broadcast packets. */
|
|
|
|
if (h.dst == 0xffff)
|
|
|
|
h.fc0 &= ~FC0_REQ_ACK;
|
|
|
|
|
2007-02-01 15:02:19 +01:00
|
|
|
ret = cc2420_send(&h, 10, &uip_buf[UIP_LLH_LEN], uip_len);
|
|
|
|
if (ret < 0) {
|
|
|
|
PRINTF("cc2420_send_uaodv uip_len=%d ret=%d\n", uip_len, ret);
|
2006-06-18 00:41:10 +02:00
|
|
|
leds_toggle(color);
|
|
|
|
return UIP_FW_TOOLARGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
leds_toggle(color);
|
|
|
|
|
|
|
|
return UIP_FW_OK;
|
|
|
|
}
|