Main uIPv6 files addition - more
This commit is contained in:
parent
f95cdc5abb
commit
2e8264010b
10 changed files with 6832 additions and 0 deletions
445
core/net/uip-netif.c
Normal file
445
core/net/uip-netif.c
Normal file
|
@ -0,0 +1,445 @@
|
|||
/**
|
||||
* \addtogroup uip6
|
||||
* @{
|
||||
*/
|
||||
|
||||
/**
|
||||
* \file
|
||||
* Network interface and stateless autoconfiguration (RFC 4862)
|
||||
* \author Mathilde Durvy <mdurvy@cisco.com>
|
||||
* \author Julien Abeille <jabeille@cisco.com>
|
||||
*
|
||||
*/
|
||||
|
||||
/*
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions
|
||||
* are met:
|
||||
* 1. Redistributions of source code must retain the above copyright
|
||||
* notice, this list of conditions and the following disclaimer.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice, this list of conditions and the following disclaimer in the
|
||||
* documentation and/or other materials provided with the distribution.
|
||||
* 3. Neither the name of the Institute nor the names of its contributors
|
||||
* may be used to endorse or promote products derived from this software
|
||||
* without specific prior written permission.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
|
||||
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||
* ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
|
||||
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
||||
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||||
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
||||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||
* SUCH DAMAGE.
|
||||
*
|
||||
*
|
||||
*/
|
||||
|
||||
#include "net/uip-nd6.h"
|
||||
#include "net/uip-netif.h"
|
||||
#include "lib/random.h"
|
||||
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
|
||||
#define DEBUG 0
|
||||
#if DEBUG
|
||||
#include <stdio.h>
|
||||
#define PRINTF(...) printf(__VA_ARGS__)
|
||||
#define PRINT6ADDR(addr) PRINTF(" %02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x ", ((u8_t *)addr)[0], ((u8_t *)addr)[1], ((u8_t *)addr)[2], ((u8_t *)addr)[3], ((u8_t *)addr)[4], ((u8_t *)addr)[5], ((u8_t *)addr)[6], ((u8_t *)addr)[7], ((u8_t *)addr)[8], ((u8_t *)addr)[9], ((u8_t *)addr)[10], ((u8_t *)addr)[11], ((u8_t *)addr)[12], ((u8_t *)addr)[13], ((u8_t *)addr)[14], ((u8_t *)addr)[15])
|
||||
#define PRINTLLADDR(lladdr) PRINTF(" %02x:%02x:%02x:%02x:%02x:%02x ",lladdr->addr[0], lladdr->addr[1], lladdr->addr[2], lladdr->addr[3],lladdr->addr[4], lladdr->addr[5])
|
||||
#else
|
||||
#define PRINTF(...)
|
||||
#define PRINT6ADDR(addr)
|
||||
#define PRINTLLADDR(lladdr)
|
||||
#endif
|
||||
|
||||
#if UIP_LOGGING
|
||||
#include <stdio.h>
|
||||
void uip_log(char *msg);
|
||||
#define UIP_LOG(m) uip_log(m)
|
||||
#else
|
||||
#define UIP_LOG(m)
|
||||
#endif /* UIP_LOGGING == 1 */
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
/** \brief The single physical interface */
|
||||
struct uip_netif uip_netif_physical_if;
|
||||
/** \brief DAD timer */
|
||||
struct etimer uip_netif_timer_dad;
|
||||
/** \brief The interface address on which dad is being performed */
|
||||
static struct uip_netif_addr *dad_ifaddr;
|
||||
/** \brief Number of ns already sent for DAD*/
|
||||
static u8_t dad_ns;
|
||||
/** \brief RS timer, to schedule RS sending */
|
||||
struct etimer uip_netif_timer_rs;
|
||||
/** \brief number of rs already sent */
|
||||
static u8_t rs_count;
|
||||
/** \brief index for loops */
|
||||
static u8_t i;
|
||||
/** \brief timer to check the address states */
|
||||
struct etimer uip_netif_timer_periodic;
|
||||
|
||||
|
||||
/** \brief remove an interface address, argument type is uip_netif_addr* */
|
||||
#define uip_netif_addr_rm(addr) do { \
|
||||
PRINTF("Removing address"); \
|
||||
PRINT6ADDR(&addr->ipaddr); \
|
||||
PRINTF("\n"); \
|
||||
addr->state = NOT_USED; \
|
||||
} while(0)
|
||||
/*---------------------------------------------------------------------------*/
|
||||
void
|
||||
uip_netif_init(void)
|
||||
{
|
||||
/* INITIALIZE INTERFACE (default values for now) */
|
||||
uip_netif_physical_if.link_mtu = UIP_LINK_MTU;
|
||||
uip_netif_physical_if.cur_hop_limit = UIP_TTL;
|
||||
uip_netif_physical_if.base_reachable_time = UIP_ND6_REACHABLE_TIME;
|
||||
uip_netif_physical_if.reachable_time = uip_netif_compute_reachable_time();
|
||||
uip_netif_physical_if.retrans_timer = UIP_ND6_RETRANS_TIMER;
|
||||
uip_netif_physical_if.dup_addr_detect_transmit = 1;
|
||||
|
||||
/*
|
||||
* STATELESS AUTOCONFIGURATION of the link local address. We set it to
|
||||
* infinite (this will become really true once DAD succeeds)
|
||||
*/
|
||||
uip_ip6addr(&(uip_netif_physical_if.addresses[0].ipaddr),
|
||||
0xfe80,0,0,0,0,0,0,0);
|
||||
uip_netif_addr_autoconf_set(&(uip_netif_physical_if.addresses[0].ipaddr), &uip_lladdr);
|
||||
uip_netif_physical_if.addresses[0].state = TENTATIVE;
|
||||
uip_netif_physical_if.addresses[0].type = MANUAL;
|
||||
uip_netif_physical_if.addresses[0].is_infinite = 1;
|
||||
|
||||
/* set all other addresses to NOT_USED initialy */
|
||||
for(i = 1; i < UIP_CONF_NETIF_MAX_ADDRESSES; i ++)
|
||||
uip_netif_physical_if.addresses[i].state = NOT_USED;
|
||||
|
||||
uip_ip6addr_u8(&(uip_netif_physical_if.solicited_node_mcastaddr),
|
||||
0xff, 0x02, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0x01, 0xff,
|
||||
uip_lladdr.addr[UIP_LLADDR_LEN - 3],
|
||||
uip_lladdr.addr[UIP_LLADDR_LEN - 2],
|
||||
uip_lladdr.addr[UIP_LLADDR_LEN - 1]);
|
||||
/* Start DAD */
|
||||
uip_netif_sched_dad(&(uip_netif_physical_if.addresses[0]));
|
||||
|
||||
/* Find router (send rs to all-routers multicast group)) */
|
||||
uip_netif_sched_send_rs();
|
||||
|
||||
/* Reset the timer */
|
||||
etimer_set(&uip_netif_timer_periodic, CLOCK_SECOND);
|
||||
}
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
void
|
||||
uip_netif_periodic(void) {
|
||||
for(i = 1; i < UIP_CONF_NETIF_MAX_ADDRESSES; i++) {
|
||||
if((uip_netif_physical_if.addresses[i].state != NOT_USED) &&
|
||||
(uip_netif_physical_if.addresses[i].is_infinite != 1) &&
|
||||
(timer_expired(&uip_netif_physical_if.addresses[i].vlifetime))) {
|
||||
uip_netif_addr_rm((&uip_netif_physical_if.addresses[i]));
|
||||
}
|
||||
}
|
||||
etimer_reset(&uip_netif_timer_periodic);
|
||||
return;
|
||||
}
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
u32_t
|
||||
uip_netif_compute_reachable_time(void)
|
||||
{
|
||||
return (u32_t)(uip_netif_physical_if.base_reachable_time * UIP_ND6_MIN_RANDOM_FACTOR) + ((u16_t)(random_rand() << 8) + (u16_t)random_rand()) % (u32_t)(uip_netif_physical_if.base_reachable_time *(UIP_ND6_MAX_RANDOM_FACTOR - UIP_ND6_MIN_RANDOM_FACTOR));
|
||||
}
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
u8_t
|
||||
uip_netif_is_addr_my_solicited(uip_ipaddr_t *ipaddr)
|
||||
{
|
||||
if(uip_ipaddr_cmp(ipaddr, &uip_netif_physical_if.solicited_node_mcastaddr))
|
||||
return 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
struct uip_netif_addr *
|
||||
uip_netif_addr_lookup(uip_ipaddr_t *ipaddr, u8_t length, uip_netif_type type) {
|
||||
for(i = 0; i < UIP_CONF_NETIF_MAX_ADDRESSES; i ++) {
|
||||
if((uip_netif_physical_if.addresses[i].state != NOT_USED) &&
|
||||
(uip_netif_physical_if.addresses[i].type == type || type == 0) &&
|
||||
(uip_ipaddr_prefixcmp(&(uip_netif_physical_if.addresses[i].ipaddr), ipaddr, length))) {
|
||||
return &uip_netif_physical_if.addresses[i];
|
||||
}
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
void
|
||||
uip_netif_addr_add(uip_ipaddr_t *ipaddr, u8_t length, clock_time_t vlifetime, uip_netif_type type) {
|
||||
|
||||
/* check prefix has the right length if we are doing autoconf */
|
||||
if((type == AUTOCONF) && (length != UIP_DEFAULT_PREFIX_LEN)) {
|
||||
UIP_LOG("Error: UNSUPPORTED PREFIX LENGTH");
|
||||
return;
|
||||
}
|
||||
|
||||
/* check if addr does not already exist and find a free entry */
|
||||
for(i = 0; i < UIP_CONF_NETIF_MAX_ADDRESSES; ++i) {
|
||||
if(uip_netif_physical_if.addresses[i].state == NOT_USED){
|
||||
/*
|
||||
* Copying address
|
||||
* If we are doing autoconf, ipaddr is a prefix, we copy the 128 bits
|
||||
* of it, then overwrite the last 64 bits with the interface ID at
|
||||
* next if statement.
|
||||
* Otherwise ipaddr is an address, we just copy it
|
||||
*/
|
||||
uip_ipaddr_copy(&uip_netif_physical_if.addresses[i].ipaddr, ipaddr);
|
||||
if(type == AUTOCONF) {
|
||||
/* construct address from prefix and layer2 id */
|
||||
uip_netif_addr_autoconf_set(&uip_netif_physical_if.addresses[i].ipaddr, &uip_lladdr);
|
||||
}
|
||||
/* setting state, type */
|
||||
uip_netif_physical_if.addresses[i].state = TENTATIVE;
|
||||
uip_netif_physical_if.addresses[i].type = type;
|
||||
/* setting lifetime timer if lieftime is not infinite */
|
||||
if(vlifetime != 0) {
|
||||
timer_set(&(uip_netif_physical_if.addresses[i].vlifetime), vlifetime);
|
||||
uip_netif_physical_if.addresses[i].is_infinite = 0;
|
||||
} else {
|
||||
uip_netif_physical_if.addresses[i].is_infinite = 1;
|
||||
}
|
||||
PRINTF("Created new address");
|
||||
PRINT6ADDR(&uip_netif_physical_if.addresses[i].ipaddr);
|
||||
PRINTF("for interface\n");
|
||||
|
||||
/* schedule DAD */
|
||||
uip_netif_sched_dad(&uip_netif_physical_if.addresses[i]);
|
||||
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
/* If we did not find space, log */
|
||||
UIP_LOG("ADDRESS LIST FULL");
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
void
|
||||
uip_netif_addr_autoconf_set(uip_ipaddr_t *ipaddr, uip_lladdr_t *lladdr)
|
||||
{
|
||||
/* We consider only links with IEEE EUI-64 identifier or
|
||||
IEEE 48-bit MAC addresses */
|
||||
#if (UIP_LLADDR_LEN == 8)
|
||||
memcpy(ipaddr->u8 + 8, lladdr, UIP_LLADDR_LEN);
|
||||
ipaddr->u8[8] ^= 0x02;
|
||||
#elif (UIP_LLADDR_LEN == 6)
|
||||
memcpy(ipaddr->u8 + 8, lladdr, 3);
|
||||
ipaddr->u8[11] = 0xff;
|
||||
ipaddr->u8[12] = 0xfe;
|
||||
memcpy(ipaddr->u8 + 13, lladdr + 3, 3);
|
||||
ipaddr->u8[8] ^= 0x02;
|
||||
#else
|
||||
UIP_LOG("CAN NOT BUIL INTERFACE IDENTIFIER");
|
||||
UIP_LOG("THE STACK IS GOING TO SHUT DOWN");
|
||||
UIP_LOG("THE HOST WILL BE UNREACHABLE");
|
||||
exit(-1);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
u8_t
|
||||
get_match_length(uip_ipaddr_t *src, uip_ipaddr_t *dst)
|
||||
{
|
||||
u8_t j, k, x_or;
|
||||
u8_t len = 0;
|
||||
for(j = 0; j < 16; j ++) {
|
||||
if(src->u8[j] == dst->u8[j]) {
|
||||
len += 8;
|
||||
} else {
|
||||
x_or = src->u8[j] ^ dst->u8[j];
|
||||
for(k = 0; k < 8; k ++) {
|
||||
if((x_or & 0x80) == 0){
|
||||
len ++;
|
||||
x_or <<= 1;
|
||||
}
|
||||
else {
|
||||
break;
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
return len;
|
||||
}
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
void
|
||||
uip_netif_select_src(uip_ipaddr_t *src, uip_ipaddr_t *dst)
|
||||
{
|
||||
u8_t best = 0; /* number of bit in common with best match*/
|
||||
u8_t n = 0;
|
||||
u8_t index = 0;
|
||||
|
||||
if(!uip_is_addr_link_local(dst) && !uip_is_addr_mcast(dst)) {
|
||||
for(i = 1; i < UIP_CONF_NETIF_MAX_ADDRESSES; ++i) {
|
||||
if(uip_netif_physical_if.addresses[i].state == PREFERRED){
|
||||
n = get_match_length(dst, &(uip_netif_physical_if.addresses[i].ipaddr));
|
||||
if(n >= best){
|
||||
best = n;
|
||||
index = i;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
uip_ipaddr_copy(src, &(uip_netif_physical_if.addresses[index].ipaddr));
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
void
|
||||
uip_netif_sched_dad(struct uip_netif_addr *ifaddr)
|
||||
{
|
||||
if(ifaddr->state != TENTATIVE){
|
||||
UIP_LOG("DAD called with non tentative address");
|
||||
return;
|
||||
}
|
||||
/*
|
||||
* check dad is not running
|
||||
*/
|
||||
if(dad_ifaddr != NULL){
|
||||
UIP_LOG("DAD already running");
|
||||
return;
|
||||
}
|
||||
/*
|
||||
* Set the interface address that is going through DAD
|
||||
*/
|
||||
dad_ifaddr = ifaddr;
|
||||
|
||||
PRINTF("Scheduling DAD for ipaddr:");
|
||||
PRINT6ADDR(&dad_ifaddr->ipaddr);
|
||||
PRINTF("\n");
|
||||
|
||||
etimer_set(&uip_netif_timer_dad, random_rand()%(UIP_ND6_MAX_RTR_SOLICITATION_DELAY*CLOCK_SECOND));
|
||||
}
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
void
|
||||
uip_netif_dad(void)
|
||||
{
|
||||
/*
|
||||
* check if dad is running
|
||||
*/
|
||||
if(dad_ifaddr == NULL){
|
||||
PRINTF("uip_netif_dad: DAD not running\n");
|
||||
return;
|
||||
}
|
||||
/*
|
||||
* send dup_addr_detect_transmit NS for DAD
|
||||
*/
|
||||
if(dad_ns < uip_netif_physical_if.dup_addr_detect_transmit) {
|
||||
uip_nd6_io_ns_output(NULL, NULL, &dad_ifaddr->ipaddr);
|
||||
dad_ns++;
|
||||
etimer_set(&uip_netif_timer_dad, uip_netif_physical_if.retrans_timer/1000*CLOCK_SECOND);
|
||||
return;
|
||||
}
|
||||
/*
|
||||
* If we arrive here it means DAD succeeded, otherwise the dad process
|
||||
* would have been interrupted in nd6_dad_ns/na_input
|
||||
*/
|
||||
PRINTF("DAD succeeded for ipaddr:");
|
||||
PRINT6ADDR(&(dad_ifaddr->ipaddr));
|
||||
PRINTF("\n");
|
||||
|
||||
etimer_stop(&uip_netif_timer_dad);
|
||||
dad_ifaddr->state = PREFERRED;
|
||||
dad_ifaddr = NULL;
|
||||
dad_ns = 0;
|
||||
/*
|
||||
* check if we need to run DAD on another address
|
||||
* This is necessary because if you receive a RA,
|
||||
* you might want to run DAD for several addresses
|
||||
* Considering that we have structures to do DAD
|
||||
* for one address only, we start DAD for the subsequent
|
||||
* addresses here
|
||||
*/
|
||||
PRINTF("END of DAD\n");
|
||||
for(i = 0; i < UIP_CONF_NETIF_MAX_ADDRESSES; i ++){
|
||||
if(uip_netif_physical_if.addresses[i].state != NOT_USED){
|
||||
PRINTF("address %d : ",i);
|
||||
PRINT6ADDR(&(uip_netif_physical_if.addresses[i].ipaddr));
|
||||
PRINTF("\n");
|
||||
}
|
||||
if(uip_netif_physical_if.addresses[i].state == TENTATIVE){
|
||||
uip_netif_sched_dad(&uip_netif_physical_if.addresses[i]);
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
void
|
||||
uip_netif_dad_failed(uip_ipaddr_t *ipaddr)
|
||||
{
|
||||
|
||||
UIP_LOG("DAD FAILED");
|
||||
UIP_LOG("THE STACK IS GOING TO SHUT DOWN");
|
||||
UIP_LOG("THE HOST WILL BE UNREACHABLE");
|
||||
|
||||
if(uip_ipaddr_cmp(&dad_ifaddr->ipaddr, ipaddr)){
|
||||
etimer_stop(&uip_netif_timer_dad);
|
||||
dad_ifaddr->state = NOT_USED;
|
||||
dad_ifaddr = NULL;
|
||||
dad_ns = 0;
|
||||
}
|
||||
|
||||
exit(-1);
|
||||
}
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
void
|
||||
uip_netif_sched_send_rs(void)
|
||||
{
|
||||
/* before a host sends an initial solicitation, it SHOULD delay the
|
||||
transmission for a random amount of time between 0 and
|
||||
UIP_ND6_MAX_RTR_SOLICITATION_DELAY. */
|
||||
if(rs_count == 0){
|
||||
etimer_set(&uip_netif_timer_rs, random_rand()%(UIP_ND6_MAX_RTR_SOLICITATION_DELAY*CLOCK_SECOND));
|
||||
PRINTF("Scheduling RS\n");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------*/
|
||||
void
|
||||
uip_netif_send_rs(void)
|
||||
{
|
||||
if((uip_nd6_choose_defrouter() == NULL) && (rs_count < UIP_ND6_MAX_RTR_SOLICITATIONS)){
|
||||
//PRINTF("Sending RS %d\n", rs_count);
|
||||
uip_nd6_io_rs_output();
|
||||
rs_count++;
|
||||
etimer_set(&uip_netif_timer_rs, UIP_ND6_RTR_SOLICITATION_INTERVAL * CLOCK_SECOND);
|
||||
} else {
|
||||
PRINTF("Router found ? (boolean): %d\n", (uip_nd6_choose_defrouter() != NULL));
|
||||
etimer_stop(&uip_netif_timer_rs);
|
||||
rs_count = 0;
|
||||
}
|
||||
}
|
||||
|
||||
/** @} */
|
Loading…
Add table
Add a link
Reference in a new issue