| /** |
| * @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 "lwip/opt.h" |
| |
| #if LWIP_IPV6 /* don't build if not configured for use in lwipopts.h */ |
| |
| #include "lwip/pbuf.h" |
| #include "lwip/ip6.h" |
| #include "lwip/ip6_addr.h" |
| #include "lwip/netif.h" |
| |
| |
| #ifdef __cplusplus |
| extern "C" { |
| #endif |
| |
| /* Struct for tables. */ |
| struct nd6_neighbor_cache_entry { |
| ip6_addr_t next_hop_address; |
| struct netif * netif; |
| u8_t lladdr[NETIF_MAX_HWADDR_LEN]; |
| /*u32_t pmtu;*/ |
| #if LWIP_ND6_QUEUEING |
| /** Pointer to queue of pending outgoing packets on this entry. */ |
| struct nd6_q_entry *q; |
| #else /* LWIP_ND6_QUEUEING */ |
| /** Pointer to a single pending outgoing packet on this entry. */ |
| struct pbuf *q; |
| #endif /* LWIP_ND6_QUEUEING */ |
| u8_t state; |
| u8_t isrouter; |
| 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 "arch/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 "arch/epstruct.h" |
| #endif |
| |
| /** Neighbor advertisement message header. */ |
| #ifdef PACK_STRUCT_USE_INCLUDES |
| # include "arch/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 "arch/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 "arch/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 "arch/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 "arch/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 "arch/epstruct.h" |
| #endif |
| |
| /** Redirect message header. */ |
| #ifdef PACK_STRUCT_USE_INCLUDES |
| # include "arch/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 "arch/epstruct.h" |
| #endif |
| |
| /** Link-layer address option. */ |
| #define ND6_LLADDR_OPTION_TYPE_SOURCE (0x01) |
| #define ND6_LLADDR_OPTION_TYPE_TARGET (0x02) |
| #ifdef PACK_STRUCT_USE_INCLUDES |
| # include "arch/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[1]); |
| } PACK_STRUCT_STRUCT; |
| PACK_STRUCT_END |
| #ifdef PACK_STRUCT_USE_INCLUDES |
| # include "arch/epstruct.h" |
| #endif |
| |
| /** The address field is variable and not fixed in length, use the |
| * appropriate macros when accessing the length field or calculating |
| * the option size. |
| */ |
| #define __OFFSETOF(t, m) ((size_t)(&(((t *)0)->m))) |
| #define __ROUND_UP(n, m) ((((n) + ((m) - 1)) / (m)) * (m)) |
| #define ND6_LLADDR_OPTION_LENGTH_SHIFT 3 |
| #define ND6_LLADDR_OPTION_LENGTH_SIZE (1 << ND6_LLADDR_OPTION_LENGTH_SHIFT) |
| #define ND6_LLADDR_OPTION_LENGTH_MASK (ND6_LLADDR_OPTION_LENGTH_SIZE - 1) |
| #define ND6_LLADDR_OPTION_LENGTH_ENCODE(hwlen) (__ROUND_UP(hwlen, ND6_LLADDR_OPTION_LENGTH_SIZE) >> ND6_LLADDR_OPTION_LENGTH_SHIFT) |
| #define ND6_LLADDR_OPTION_LENGTH_DECODE(value) ((value) << ND6_LLADDR_OPTION_LENGTH_SHIFT) |
| #define ND6_LLADDR_OPTION_SIZE(hwlen) (__OFFSETOF(struct lladdr_option, addr) + (hwlen)) |
| |
| /** 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 "arch/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 |
| #ifdef PACK_STRUCT_USE_INCLUDES |
| # include "arch/epstruct.h" |
| #endif |
| |
| /** Redirected header option. */ |
| #define ND6_OPTION_TYPE_REDIR_HDR (0x04) |
| #ifdef PACK_STRUCT_USE_INCLUDES |
| # include "arch/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 "arch/epstruct.h" |
| #endif |
| |
| /** MTU option. */ |
| #define ND6_OPTION_TYPE_MTU (0x05) |
| #ifdef PACK_STRUCT_USE_INCLUDES |
| # include "arch/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 "arch/epstruct.h" |
| #endif |
| |
| /** Route information option. */ |
| #define ND6_OPTION_TYPE_ROUTE_INFO (24) |
| #ifdef PACK_STRUCT_USE_INCLUDES |
| # include "arch/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 "arch/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 */ |
| #define ND6_TMR_INTERVAL 1000 |
| |
| /* 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); |
| s8_t nd6_get_next_hop_entry(ip6_addr_t * ip6addr, struct netif * netif); |
| s8_t nd6_select_router(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 */ |
| |
| #ifdef __cplusplus |
| } |
| #endif |
| |
| #endif /* LWIP_IPV6 */ |
| |
| #endif /* __LWIP_ND6_H__ */ |