421 lines
12 KiB
C
421 lines
12 KiB
C
/**
|
|
* @file
|
|
*
|
|
* Neighbor discovery and stateless address autoconfiguration for IPv6.
|
|
* Aims to be compliant with RFC 4861 (Neighbor discovery) and RFC 4862
|
|
* (Address autoconfiguration).
|
|
*/
|
|
|
|
/*
|
|
* Copyright (c) 2010 Inico Technologies Ltd.
|
|
* All rights reserved.
|
|
*
|
|
* 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. The name of the author may not be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
|
|
*
|
|
* This file is part of the lwIP TCP/IP stack.
|
|
*
|
|
* Author: Ivan Delamer <delamer@inicotech.com>
|
|
*
|
|
*
|
|
* Please coordinate changes and requests with Ivan Delamer
|
|
* <delamer@inicotech.com>
|
|
*/
|
|
|
|
#ifndef __LWIP_ND6_H__
|
|
#define __LWIP_ND6_H__
|
|
|
|
#include "opt.h"
|
|
|
|
#if LWIP_IPV6 /* don't build if not configured for use in lwipopts.h */
|
|
|
|
#include "pbuf.h"
|
|
#include "ip6.h"
|
|
#include "ip6_addr.h"
|
|
#include "netif.h"
|
|
#include "sys.h"
|
|
#include "etharp.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/** 1 second period */
|
|
#define ND6_TMR_RS 1000
|
|
#define ND6_TMR_RA 1000 * 10 /*set to 10s*/
|
|
|
|
#define ND6_TMR_RS_DEFAULT_INTERVAL 1 /* default interval times for rs */
|
|
#define ND6_TMR_RS_RETRY_COUNTER 6 /* default retry counter for rs*/
|
|
#define ND6_TMR_RS_RETRY_COUNTER_MAX 12 /* max retry counter for rs*/
|
|
|
|
typedef struct _ns_msg_pool //30B
|
|
{
|
|
u16_t ttl; // 2B
|
|
u16_t flag; // 2B
|
|
u16_t vlanid; // record vlanid
|
|
u16_t index; // record ns_msg_pool index for watch
|
|
struct netif * netif_lc; // 4B, local netif
|
|
ip6_addr_t ip6addr; // 16B, will store local or global address
|
|
char mac[ETHARP_HWADDR_LEN + ETH_VLANID_SIZE]; // 8B
|
|
}ns_msg_pool;
|
|
|
|
/* Struct for tables. */
|
|
struct nd6_neighbor_cache_entry {
|
|
ip6_addr_t next_hop_address;
|
|
struct netif * netif;
|
|
u8_t lladdr[NETIF_MAX_HWADDR_LEN];
|
|
u8_t state;
|
|
u8_t isrouter;
|
|
/*u32_t pmtu;*/
|
|
#if LWIP_ND6_QUEUEING
|
|
/** Pointer to queue of pending outgoing packets on this entry. */
|
|
struct nd6_q_entry *q;
|
|
u16_t queue_cnt; /*max set to 16 Now*/
|
|
u16_t vlanid; /*record vlanid*/
|
|
#else /* LWIP_ND6_QUEUEING */
|
|
/** Pointer to a single pending outgoing packet on this entry. */
|
|
struct pbuf *q;
|
|
#endif /* LWIP_ND6_QUEUEING */
|
|
union {
|
|
u32_t reachable_time;
|
|
u32_t delay_time;
|
|
u32_t probes_sent;
|
|
u32_t stale_time;
|
|
} counter;
|
|
};
|
|
|
|
struct nd6_destination_cache_entry {
|
|
ip6_addr_t destination_addr;
|
|
ip6_addr_t next_hop_addr;
|
|
u32_t pmtu;
|
|
u32_t age;
|
|
};
|
|
|
|
struct nd6_prefix_list_entry {
|
|
ip6_addr_t prefix;
|
|
struct netif * netif;
|
|
u32_t invalidation_timer;
|
|
#if LWIP_IPV6_AUTOCONFIG
|
|
u8_t flags;
|
|
#define ND6_PREFIX_AUTOCONFIG_AUTONOMOUS 0x01
|
|
#define ND6_PREFIX_AUTOCONFIG_ADDRESS_GENERATED 0x02
|
|
#define ND6_PREFIX_AUTOCONFIG_ADDRESS_DUPLICATE 0x04
|
|
#endif /* LWIP_IPV6_AUTOCONFIG */
|
|
};
|
|
|
|
struct nd6_router_list_entry {
|
|
struct nd6_neighbor_cache_entry * neighbor_entry;
|
|
u32_t invalidation_timer;
|
|
u8_t flags;
|
|
};
|
|
|
|
|
|
enum nd6_neighbor_cache_entry_state {
|
|
ND6_NO_ENTRY = 0,
|
|
ND6_INCOMPLETE,
|
|
ND6_REACHABLE,
|
|
ND6_STALE,
|
|
ND6_DELAY,
|
|
ND6_PROBE
|
|
};
|
|
|
|
#if LWIP_ND6_QUEUEING
|
|
/** struct for queueing outgoing packets for unknown address
|
|
* defined here to be accessed by memp.h
|
|
*/
|
|
struct nd6_q_entry {
|
|
struct nd6_q_entry *next;
|
|
struct pbuf *p;
|
|
};
|
|
#endif /* LWIP_ND6_QUEUEING */
|
|
|
|
/** Neighbor solicitation message header. */
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "bpstruct.h"
|
|
#endif
|
|
PACK_STRUCT_BEGIN
|
|
struct ns_header {
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t code);
|
|
PACK_STRUCT_FIELD(u16_t chksum);
|
|
PACK_STRUCT_FIELD(u32_t reserved);
|
|
PACK_STRUCT_FIELD(ip6_addr_p_t target_address);
|
|
/* Options follow. */
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "epstruct.h"
|
|
#endif
|
|
|
|
/** Neighbor advertisement message header. */
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "bpstruct.h"
|
|
#endif
|
|
PACK_STRUCT_BEGIN
|
|
struct na_header {
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t code);
|
|
PACK_STRUCT_FIELD(u16_t chksum);
|
|
PACK_STRUCT_FIELD(u8_t flags);
|
|
PACK_STRUCT_FIELD(u8_t reserved[3]);
|
|
PACK_STRUCT_FIELD(ip6_addr_p_t target_address);
|
|
/* Options follow. */
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "epstruct.h"
|
|
#endif
|
|
#define ND6_FLAG_ROUTER (0x80)
|
|
#define ND6_FLAG_SOLICITED (0x40)
|
|
#define ND6_FLAG_OVERRIDE (0x20)
|
|
|
|
/** Router solicitation message header. */
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "bpstruct.h"
|
|
#endif
|
|
PACK_STRUCT_BEGIN
|
|
struct rs_header {
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t code);
|
|
PACK_STRUCT_FIELD(u16_t chksum);
|
|
PACK_STRUCT_FIELD(u32_t reserved);
|
|
/* Options follow. */
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "epstruct.h"
|
|
#endif
|
|
|
|
/** Router advertisement message header. */
|
|
#define ND6_RA_FLAG_MANAGED_ADDR_CONFIG (0x80)
|
|
#define ND6_RA_FLAG_OTHER_STATEFUL_CONFIG (0x40)
|
|
#define ND6_RA_FLAG_HOME_AGENT (0x20)
|
|
#define ND6_RA_PREFERENCE_MASK (0x18)
|
|
#define ND6_RA_PREFERENCE_HIGH (0x08)
|
|
#define ND6_RA_PREFERENCE_MEDIUM (0x00)
|
|
#define ND6_RA_PREFERENCE_LOW (0x18)
|
|
#define ND6_RA_PREFERENCE_DISABLED (0x10)
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "bpstruct.h"
|
|
#endif
|
|
PACK_STRUCT_BEGIN
|
|
struct ra_header {
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t code);
|
|
PACK_STRUCT_FIELD(u16_t chksum);
|
|
PACK_STRUCT_FIELD(u8_t current_hop_limit);
|
|
PACK_STRUCT_FIELD(u8_t flags);
|
|
PACK_STRUCT_FIELD(u16_t router_lifetime);
|
|
PACK_STRUCT_FIELD(u32_t reachable_time);
|
|
PACK_STRUCT_FIELD(u32_t retrans_timer);
|
|
/* Options follow. */
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "epstruct.h"
|
|
#endif
|
|
|
|
/** Redirect message header. */
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "bpstruct.h"
|
|
#endif
|
|
PACK_STRUCT_BEGIN
|
|
struct redirect_header {
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t code);
|
|
PACK_STRUCT_FIELD(u16_t chksum);
|
|
PACK_STRUCT_FIELD(u32_t reserved);
|
|
PACK_STRUCT_FIELD(ip6_addr_p_t target_address);
|
|
PACK_STRUCT_FIELD(ip6_addr_p_t destination_address);
|
|
/* Options follow. */
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "epstruct.h"
|
|
#endif
|
|
|
|
/** Link-layer address option. */
|
|
#define ND6_OPTION_TYPE_SOURCE_LLADDR (0x01)
|
|
#define ND6_OPTION_TYPE_TARGET_LLADDR (0x02)
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "bpstruct.h"
|
|
#endif
|
|
PACK_STRUCT_BEGIN
|
|
struct lladdr_option {
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t length);
|
|
PACK_STRUCT_FIELD(u8_t addr[NETIF_MAX_HWADDR_LEN]);
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "epstruct.h"
|
|
#endif
|
|
|
|
/** Prefix information option. */
|
|
#define ND6_OPTION_TYPE_PREFIX_INFO (0x03)
|
|
#define ND6_PREFIX_FLAG_ON_LINK (0x80)
|
|
#define ND6_PREFIX_FLAG_AUTONOMOUS (0x40)
|
|
#define ND6_PREFIX_FLAG_ROUTER_ADDRESS (0x20)
|
|
#define ND6_PREFIX_FLAG_SITE_PREFIX (0x10)
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "bpstruct.h"
|
|
#endif
|
|
PACK_STRUCT_BEGIN
|
|
struct prefix_option {
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t length);
|
|
PACK_STRUCT_FIELD(u8_t prefix_length);
|
|
PACK_STRUCT_FIELD(u8_t flags);
|
|
PACK_STRUCT_FIELD(u32_t valid_lifetime);
|
|
PACK_STRUCT_FIELD(u32_t preferred_lifetime);
|
|
PACK_STRUCT_FIELD(u8_t reserved2[3]);
|
|
PACK_STRUCT_FIELD(u8_t site_prefix_length);
|
|
PACK_STRUCT_FIELD(ip6_addr_p_t prefix);
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
|
|
#define ND6_OPTION_TYPE_DNS (0x19)
|
|
PACK_STRUCT_BEGIN
|
|
struct dns_svr_option { //rfc 4339 & rfc 61016
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t length);
|
|
PACK_STRUCT_FIELD(u16_t reserved);
|
|
PACK_STRUCT_FIELD(u32_t lifetime);
|
|
PACK_STRUCT_FIELD(u32_t dns1[4]);
|
|
PACK_STRUCT_FIELD(u32_t dns2[4]);
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "epstruct.h"
|
|
#endif
|
|
|
|
/** Redirected header option. */
|
|
#define ND6_OPTION_TYPE_REDIR_HDR (0x04)
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "bpstruct.h"
|
|
#endif
|
|
PACK_STRUCT_BEGIN
|
|
struct redirected_header_option {
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t length);
|
|
PACK_STRUCT_FIELD(u8_t reserved[6]);
|
|
/* Portion of redirected packet follows. */
|
|
/* PACK_STRUCT_FIELD(u8_t redirected[8]); */
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "epstruct.h"
|
|
#endif
|
|
|
|
/** MTU option. */
|
|
#define ND6_OPTION_TYPE_MTU (0x05)
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "bpstruct.h"
|
|
#endif
|
|
PACK_STRUCT_BEGIN
|
|
struct mtu_option {
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t length);
|
|
PACK_STRUCT_FIELD(u16_t reserved);
|
|
PACK_STRUCT_FIELD(u32_t mtu);
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "epstruct.h"
|
|
#endif
|
|
|
|
/** Route information option. */
|
|
#define ND6_OPTION_TYPE_ROUTE_INFO (24)
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "bpstruct.h"
|
|
#endif
|
|
PACK_STRUCT_BEGIN
|
|
struct route_option {
|
|
PACK_STRUCT_FIELD(u8_t type);
|
|
PACK_STRUCT_FIELD(u8_t length);
|
|
PACK_STRUCT_FIELD(u8_t prefix_length);
|
|
PACK_STRUCT_FIELD(u8_t preference);
|
|
PACK_STRUCT_FIELD(u32_t route_lifetime);
|
|
PACK_STRUCT_FIELD(ip6_addr_p_t prefix);
|
|
} PACK_STRUCT_STRUCT;
|
|
PACK_STRUCT_END
|
|
#ifdef PACK_STRUCT_USE_INCLUDES
|
|
# include "epstruct.h"
|
|
#endif
|
|
|
|
/* the possible states of an IP address */
|
|
#define IP6_ADDRESS_STATE_INVALID (0)
|
|
#define IP6_ADDRESS_STATE_VALID (0x4)
|
|
#define IP6_ADDRESS_STATE_PREFERRED (0x5) /* includes valid */
|
|
#define IP6_ADDRESS_STATE_DEPRECATED (0x6) /* includes valid */
|
|
#define IP6_ADDRESS_STATE_TENTATIV (0x8)
|
|
|
|
/** 1 second period default, now set to 10s */
|
|
#define ND6_TMR_INTERVAL 10000
|
|
|
|
/* Router tables. */
|
|
/* TODO make these static? and entries accessible through API? */
|
|
extern struct nd6_neighbor_cache_entry neighbor_cache[];
|
|
extern struct nd6_destination_cache_entry destination_cache[];
|
|
extern struct nd6_prefix_list_entry prefix_list[];
|
|
extern struct nd6_router_list_entry default_router_list[];
|
|
|
|
/* Default values, can be updated by a RA message. */
|
|
extern u32_t reachable_time;
|
|
extern u32_t retrans_timer;
|
|
|
|
#define nd6_init() /* TODO should we init tables? */
|
|
void nd6_tmr(void);
|
|
void nd6_input(struct pbuf *p, struct netif *inp);
|
|
void nd6_clear_destination_cache(void);
|
|
s8_t nd6_get_next_hop_entry(ip6_addr_t * ip6addr, struct netif * netif);
|
|
u16_t nd6_get_destination_mtu(ip6_addr_t * ip6addr, struct netif * netif);
|
|
err_t nd6_queue_packet(s8_t neighbor_index, struct pbuf * p);
|
|
#if LWIP_ND6_TCP_REACHABILITY_HINTS
|
|
void nd6_reachability_hint(ip6_addr_t * ip6addr);
|
|
#endif /* LWIP_ND6_TCP_REACHABILITY_HINTS */
|
|
void nd6_cleanup_netif(struct netif *netif);
|
|
void nd6_restart_netif(struct netif *netif);
|
|
u32_t nd6_rs_get_total_time(void);
|
|
void reset_netif_ipv6_flg_all (struct netif *inp, char* callFunc);
|
|
void reset_netif_ipv6_flg_part (struct netif *inp, char* callFunc);
|
|
void nd6_set_prefix(struct netif * inp, u32_t p1, u32_t p2);
|
|
void nd6_set_dns6(struct netif * inp, ip6_addr_t *dns1, ip6_addr_t *dns2);
|
|
u8_t nd6_get_prefix_length(void);
|
|
|
|
#if LWIP_IPV6_FORWARD
|
|
char *ns_ip6_pool_get_mac(struct netif *netif, ip6_addr_t *ip6);
|
|
void ns_ip6_pool_na_recv(struct netif *inp, ip6_addr_t *ip6, char *mac);
|
|
void ns_ip6_pool_ip6_input(struct netif *inp, ip6_addr_t *ip6, char *mac);
|
|
ns_msg_pool *ns_ip6_pool_find_vlan(ip6_addr_t *ip6, u16_t vlanid);
|
|
struct netif *nd6_get_next_route(struct netif *inp, ip6_addr_t *ip6);
|
|
#endif
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* LWIP_IPV6 */
|
|
|
|
#endif /* __LWIP_ND6_H__ */
|