mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-15 14:12:44 +01:00
315 lines
8.7 KiB
C
315 lines
8.7 KiB
C
/*
|
|
* Copyright (C) 2017 Freie Universität Berlin
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @defgroup net_gnrc_ipv6_nib_nc Neighbor Cache
|
|
* @ingroup net_gnrc_ipv6_nib
|
|
* @brief Neighbor cache component of neighbor information base
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Neighbor cache definitions
|
|
*
|
|
* @author Martine Lenders <m.lenders@fu-berlin.de>
|
|
*/
|
|
#ifndef NET_GNRC_IPV6_NIB_NC_H
|
|
#define NET_GNRC_IPV6_NIB_NC_H
|
|
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
|
|
#include "net/eui64.h"
|
|
#include "net/gnrc/netif/hdr.h"
|
|
#include "net/gnrc/ipv6/nib/conf.h"
|
|
#include "net/ipv6/addr.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/**
|
|
* @brief Values for gnrc_ipv6_nib_nc_t::info
|
|
* @anchor net_gnrc_ipv6_nib_nc_info
|
|
* @name Info values
|
|
* @{
|
|
*/
|
|
/**
|
|
* @brief Mask for neighbor unreachability detection (NUD) states
|
|
*
|
|
* @see [RFC 4861, section 7.3.2](https://tools.ietf.org/html/rfc4861#section-7.3.2)
|
|
* @see [RFC 7048](https://tools.ietf.org/html/rfc7048)
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_NUD_STATE_MASK (0x0007)
|
|
|
|
/**
|
|
* @brief Not managed by NUD
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_NUD_STATE_UNMANAGED (0x0000)
|
|
|
|
/**
|
|
* @brief Entry is not reachable
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_NUD_STATE_UNREACHABLE (0x0001)
|
|
|
|
/**
|
|
* @brief Address resolution is currently performed
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_NUD_STATE_INCOMPLETE (0x0002)
|
|
|
|
/**
|
|
* @brief Address might not be reachable
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_NUD_STATE_STALE (0x0003)
|
|
|
|
/**
|
|
* @brief NUD will be performed in a moment
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_NUD_STATE_DELAY (0x0004)
|
|
|
|
/**
|
|
* @brief NUD is performed
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_NUD_STATE_PROBE (0x0005)
|
|
|
|
/**
|
|
* @brief Entry is reachable
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_NUD_STATE_REACHABLE (0x0006)
|
|
|
|
/**
|
|
* @brief gnrc_ipv6_nib_t::next_hop is router
|
|
*
|
|
* This flag indicates that gnrc_ipv6_nib_t::next_hop is a router, but it does
|
|
* not necessarily indicate that it is in the default router list! A neighbor
|
|
* that has this flag unset however **must not** appear in the default router
|
|
* list.
|
|
*
|
|
* @see [RFC 4861, Appendix D](https://tools.ietf.org/html/rfc4861#page-91)
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_IS_ROUTER (0x0008)
|
|
|
|
/**
|
|
* @brief Mask for interface identifier
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_IFACE_MASK (0x01f0)
|
|
|
|
/**
|
|
* @brief Shift position of interface identifier
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_IFACE_POS (4)
|
|
|
|
/**
|
|
* @brief Mask for 6LoWPAN address registration (6Lo-AR) states
|
|
*
|
|
* @see [RFC 6775, section 3.5](https://tools.ietf.org/html/rfc6775#section-3.5)
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_AR_STATE_MASK (0x0600)
|
|
|
|
/**
|
|
* @brief Shift position of address registration states
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_AR_STATE_POS (9)
|
|
|
|
/**
|
|
* @brief Not managed by 6Lo-AR (address can be removed when memory is low
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_AR_STATE_GC (0x0000)
|
|
|
|
/**
|
|
* @brief Address registration still pending at upstream router
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_AR_STATE_TENTATIVE (0x0200)
|
|
|
|
/**
|
|
* @brief Address is registered
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_AR_STATE_REGISTERED (0x0400)
|
|
|
|
/**
|
|
* @brief Address was added manually
|
|
*/
|
|
#define GNRC_IPV6_NIB_NC_INFO_AR_STATE_MANUAL (0x0600)
|
|
/** @} */
|
|
|
|
/**
|
|
* @brief Neighbor cache entry view on NIB
|
|
*/
|
|
typedef struct {
|
|
ipv6_addr_t ipv6; /**< Neighbor's IPv6 address */
|
|
/**
|
|
* @brief Neighbor's link-layer address
|
|
*/
|
|
uint8_t l2addr[GNRC_IPV6_NIB_L2ADDR_MAX_LEN];
|
|
/**
|
|
* @brief Neighbor information as defined in
|
|
* @ref net_gnrc_ipv6_nib_nc_info "info values"
|
|
*/
|
|
uint16_t info;
|
|
uint8_t l2addr_len; /**< Length of gnrc_ipv6_nib_nc_t::l2addr in bytes */
|
|
} gnrc_ipv6_nib_nc_t;
|
|
|
|
/**
|
|
* @brief Gets neighbor unreachability state from entry
|
|
*
|
|
* @param[in] entry A neighbor cache entry.
|
|
*
|
|
* @return The neighbor unreachability state of @p entry.
|
|
*/
|
|
static inline unsigned gnrc_ipv6_nib_nc_get_nud_state(const gnrc_ipv6_nib_nc_t *entry)
|
|
{
|
|
return (entry->info & GNRC_IPV6_NIB_NC_INFO_NUD_STATE_MASK);
|
|
}
|
|
|
|
/**
|
|
* @brief Gets router flag of a neighbor.
|
|
*
|
|
* @param[in] entry A neighbor cache entry.
|
|
*
|
|
* @return true, if @p entry is a router.
|
|
* @return false, if @p entry is not a router.
|
|
*/
|
|
static inline bool gnrc_ipv6_nib_nc_is_router(const gnrc_ipv6_nib_nc_t *entry)
|
|
{
|
|
return (entry->info & GNRC_IPV6_NIB_NC_INFO_IS_ROUTER);
|
|
}
|
|
|
|
/**
|
|
* @brief Gets interface from entry
|
|
*
|
|
* @param[in] entry A neighbor cache entry
|
|
*
|
|
* @return The interface identifier of @p entry.
|
|
* @return 0 if no interface is identified for @p entry.
|
|
*/
|
|
static inline unsigned gnrc_ipv6_nib_nc_get_iface(const gnrc_ipv6_nib_nc_t *entry)
|
|
{
|
|
return (entry->info & GNRC_IPV6_NIB_NC_INFO_IFACE_MASK) >>
|
|
GNRC_IPV6_NIB_NC_INFO_IFACE_POS;
|
|
}
|
|
|
|
/**
|
|
* @brief Gets address registration state of an entry
|
|
*
|
|
* @param[in] entry A neighbor cache entry
|
|
*
|
|
* @return The address registration state of @p entry.
|
|
*/
|
|
static inline unsigned gnrc_ipv6_nib_nc_get_ar_state(const gnrc_ipv6_nib_nc_t *entry)
|
|
{
|
|
return (entry->info & GNRC_IPV6_NIB_NC_INFO_AR_STATE_MASK);
|
|
}
|
|
|
|
/**
|
|
* @brief Adds an unmanaged neighbor entry to NIB
|
|
*
|
|
* @pre `ipv6 != NULL`
|
|
* @pre `l2addr_len <= GNRC_IPV6_NIB_L2ADDR_MAX_LEN`
|
|
* @pre `(iface > KERNEL_PID_UNDEF) && (iface <= KERNEL_PID_LAST)`
|
|
*
|
|
* @param[in] ipv6 The neighbor's IPv6 address.
|
|
* @param[in] iface The interface to the neighbor.
|
|
* @param[in] l2addr The neighbor's L2 address.
|
|
* @param[in] l2addr_len Length of @p l2addr.
|
|
*
|
|
* A neighbor cache entry created this way is marked as persistent.
|
|
* Also, a non-persistent neighbor or destination cache entry already in the
|
|
* NIB might be removed to make room for the new entry.
|
|
* If an entry pointing to the same IPv6 address as @p ipv6 exists already it
|
|
* will be overwritten and marked as unmanaged.
|
|
*
|
|
* If @ref GNRC_IPV6_NIB_CONF_ARSM != 0 @p l2addr and @p l2addr_len won't be set.
|
|
*
|
|
* @return 0 on success.
|
|
* @return -ENOMEM, if no space is left in neighbor cache.
|
|
*/
|
|
int gnrc_ipv6_nib_nc_set(const ipv6_addr_t *ipv6, unsigned iface,
|
|
const uint8_t *l2addr, size_t l2addr_len);
|
|
|
|
/**
|
|
* @brief Deletes neighbor with address @p ipv6 from NIB
|
|
*
|
|
* @pre `ipv6 != NULL`
|
|
*
|
|
* @param[in] ipv6 The neighbor's IPv6 address.
|
|
* @param[in] iface The interface to the neighbor.
|
|
*
|
|
* If the @p ipv6 can't be found for a neighbor in the NIB nothing happens.
|
|
*/
|
|
void gnrc_ipv6_nib_nc_del(const ipv6_addr_t *ipv6, unsigned iface);
|
|
|
|
/**
|
|
* @brief Mark neighbor with address @p ipv6 as reachable
|
|
*
|
|
* @pre `ipv6 != NULL`
|
|
*
|
|
* @param[in] ipv6 A neighbor's IPv6 address. May not be NULL.
|
|
*
|
|
* This function shall be called if an upper layer gets reachability
|
|
* confirmation via its own means (e.g. a TCP connection build-up or
|
|
* confirmation). Unmanaged neighbor cache entries (i.e. entries created using
|
|
* @ref gnrc_ipv6_nib_nc_set()) or entries whose next-hop are not yet in the
|
|
* neighbor cache are ignored.
|
|
*
|
|
* Entries in state @ref GNRC_IPV6_NIB_NC_INFO_NUD_STATE_UNMANAGED are not
|
|
* affected by this, since they are assumed to always be reachable and kept out
|
|
* of the NUD state-machine
|
|
*/
|
|
void gnrc_ipv6_nib_nc_mark_reachable(const ipv6_addr_t *ipv6);
|
|
|
|
/**
|
|
* @brief Iterates over all neighbor cache entries in the NIB
|
|
*
|
|
* @pre `(state != NULL) && (nce != NULL)`
|
|
*
|
|
* @param[in] iface Restrict iteration to entries on this interface.
|
|
* 0 for any interface.
|
|
* @param[in,out] state Iteration state of the neighbor cache. Must point to
|
|
* a NULL pointer to start iteration.
|
|
* @param[out] nce The next neighbor cache entry.
|
|
*
|
|
* Usage example:
|
|
*
|
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.c}
|
|
* #include "net/gnrc/ipv6/nib/nc.h"
|
|
*
|
|
* int main(void) {
|
|
* void *state = NULL;
|
|
* gnrc_ipv6_nib_nc_t nce;
|
|
*
|
|
* puts("My neighbors:");
|
|
* while (gnrc_ipv6_nib_nc_iter(0, &state, &nce)) {
|
|
* gnrc_ipv6_nib_nc_print(&nce);
|
|
* }
|
|
* return 0;
|
|
* }
|
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
*
|
|
* @note The list may change during iteration.
|
|
*
|
|
* @return true, if iteration can be continued.
|
|
* @return false, if @p nce is the last neighbor cache entry in the NIB.
|
|
*/
|
|
bool gnrc_ipv6_nib_nc_iter(unsigned iface, void **state,
|
|
gnrc_ipv6_nib_nc_t *nce);
|
|
|
|
/**
|
|
* @brief Prints a neighbor cache entry
|
|
*
|
|
* @pre `nce != NULL`
|
|
*
|
|
* @param[in] nce A neighbor cache entry.
|
|
*/
|
|
void gnrc_ipv6_nib_nc_print(gnrc_ipv6_nib_nc_t *nce);
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* NET_GNRC_IPV6_NIB_NC_H */
|
|
/** @} */
|