1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/sys/net/routing/nhdp/nib_table.c

313 lines
9.5 KiB
C

/*
* Copyright (C) 2014 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.
*/
/**
* @ingroup nhdp
* @{
*
* @file
* @brief Neighbor Information Base implementation for NHDP
*
* @author Fabian Nack <nack@inf.fu-berlin.de>
*
* @}
*/
#include "timex.h"
#include "mutex.h"
#include "xtimer.h"
#include "utlist.h"
#include "rfc5444/rfc5444_iana.h"
#include "rfc5444/rfc5444_writer.h"
#include "nib_table.h"
#include "iib_table.h"
#include "nhdp_address.h"
#include "nhdp_writer.h"
/* Internal variables */
static mutex_t mtx_nib_access = MUTEX_INIT;
static nib_entry_t *nib_entry_head = NULL;
static nib_lost_address_entry_t *nib_lost_address_entry_head = NULL;
/* Internal function prototypes */
static nib_entry_t *add_nib_entry_for_nb_addr_list(void);
static void rem_nib_entry(nib_entry_t *nib_entry, timex_t *now);
static void clear_nb_addresses(nib_entry_t *nib_entry, timex_t *now);
static int add_lost_neighbor_address(nhdp_addr_t *lost_addr, timex_t *now);
static void rem_ln_entry(nib_lost_address_entry_t *ln_entry);
/*---------------------------------------------------------------------------*
* Neighbor Information Base API *
*---------------------------------------------------------------------------*/
nib_entry_t *nib_process_hello(void)
{
nib_entry_t *nb_match = NULL;
nib_entry_t *nib_elt = NULL, *nib_tmp = NULL;
timex_t now;
uint8_t matches = 0;
mutex_lock(&mtx_nib_access);
xtimer_now_timex(&now);
LL_FOREACH_SAFE(nib_entry_head, nib_elt, nib_tmp) {
nhdp_addr_entry_t *list_elt = NULL;
LL_FOREACH(nib_elt->address_list_head, list_elt) {
if (NHDP_ADDR_TMP_IN_NB_LIST(list_elt->address)) {
/* Matching neighbor tuple */
matches++;
if (matches > 1) {
/* Multiple matching nb tuples, delete the previous one */
iib_propagate_nb_entry_change(nb_match, nib_elt);
rem_nib_entry(nb_match, &now);
}
nb_match = nib_elt;
break;
}
}
}
/* Add or update nb tuple */
if (matches > 0) {
/* We found matching nb tuples, reuse the last one */
clear_nb_addresses(nb_match, &now);
if (matches > 1) {
nb_match->symmetric = 0;
}
nb_match->address_list_head = nhdp_generate_addr_list_from_tmp(NHDP_ADDR_TMP_NB_LIST);
if (!nb_match->address_list_head) {
/* Insufficient memory */
LL_DELETE(nib_entry_head, nb_match);
free(nb_match);
nb_match = NULL;
}
}
else {
nb_match = add_nib_entry_for_nb_addr_list();
}
mutex_unlock(&mtx_nib_access);
return nb_match;
}
void nib_fill_wr_addresses(struct rfc5444_writer *wr)
{
nib_entry_t *nib_elt = NULL;
nhdp_addr_entry_t *addr_elt = NULL;
nib_lost_address_entry_t *lost_elt = NULL, *lost_tmp = NULL;
timex_t now;
mutex_lock(&mtx_nib_access);
xtimer_now_timex(&now);
/* Add addresses of symmetric neighbors to HELLO msg */
LL_FOREACH(nib_entry_head, nib_elt) {
if (nib_elt->symmetric) {
LL_FOREACH(nib_elt->address_list_head, addr_elt) {
/* Check whether address is not already included with link status symmetric */
if (!NHDP_ADDR_TMP_IN_SYM(addr_elt->address)) {
nhdp_writer_add_addr(wr, addr_elt->address, RFC5444_ADDRTLV_OTHER_NEIGHB,
RFC5444_OTHERNEIGHB_SYMMETRIC,
rfc5444_metric_encode(nib_elt->metric_in),
rfc5444_metric_encode(nib_elt->metric_out));
addr_elt->address->in_tmp_table = NHDP_ADDR_TMP_SYM;
}
}
}
}
/* Add lost addresses of neighbors to HELLO msg */
LL_FOREACH_SAFE(nib_lost_address_entry_head, lost_elt, lost_tmp) {
if (timex_cmp(lost_elt->expiration_time, now) != 1) {
/* Entry expired, remove it */
rem_ln_entry(lost_elt);
}
else {
/* Check if address is not already present in one of the temporary lists */
if (!NHDP_ADDR_TMP_IN_ANY(lost_elt->address)) {
/* Address is not present in one of the lists, add it */
nhdp_writer_add_addr(wr, lost_elt->address, RFC5444_ADDRTLV_OTHER_NEIGHB,
RFC5444_OTHERNEIGHB_LOST, NHDP_METRIC_UNKNOWN,
NHDP_METRIC_UNKNOWN);
}
}
}
mutex_unlock(&mtx_nib_access);
}
void nib_rem_nb_entry(nib_entry_t *nib_entry)
{
nhdp_free_addr_list(nib_entry->address_list_head);
LL_DELETE(nib_entry_head, nib_entry);
free(nib_entry);
}
void nib_set_nb_entry_sym(nib_entry_t *nib_entry)
{
nib_lost_address_entry_t *ln_elt = NULL, *ln_tmp = NULL;
nhdp_addr_entry_t *nb_elt = NULL;
nib_entry->symmetric = 1;
LL_FOREACH(nib_entry->address_list_head, nb_elt) {
LL_FOREACH_SAFE(nib_lost_address_entry_head, ln_elt, ln_tmp) {
/* Remove all Lost Neighbor Tuples matching an address of the newly sym nb */
if (ln_elt->address == nb_elt->address) {
rem_ln_entry(ln_elt);
break;
}
}
}
}
void nib_reset_nb_entry_sym(nib_entry_t *nib_entry, timex_t *now)
{
nhdp_addr_entry_t *nb_elt = NULL;
nib_entry->symmetric = 0;
LL_FOREACH(nib_entry->address_list_head, nb_elt) {
/* Add a Lost Neighbor Tuple for each address of the neighbor */
if (add_lost_neighbor_address(nb_elt->address, now) == -1) {
/* Insufficient memory */
return;
}
}
}
/*------------------------------------------------------------------------------------*/
/* Internal functions */
/*------------------------------------------------------------------------------------*/
/**
* Add a Neighbor Tuple for the neighbor address list
*/
static nib_entry_t *add_nib_entry_for_nb_addr_list(void)
{
nib_entry_t *new_elem;
new_elem = malloc(sizeof(nib_entry_t));
if (!new_elem) {
/* Insufficient memory */
return NULL;
}
/* Copy neighbor address list to new neighbor tuple */
new_elem->address_list_head = nhdp_generate_addr_list_from_tmp(NHDP_ADDR_TMP_NB_LIST);
if (!new_elem->address_list_head) {
/* Insufficient memory */
free(new_elem);
return NULL;
}
new_elem->symmetric = 0;
new_elem->metric_in = NHDP_METRIC_UNKNOWN;
new_elem->metric_out = NHDP_METRIC_UNKNOWN;
LL_PREPEND(nib_entry_head, new_elem);
return new_elem;
}
/**
* Remove a given Neighbor Tuple
*/
static void rem_nib_entry(nib_entry_t *nib_entry, timex_t *now)
{
clear_nb_addresses(nib_entry, now);
LL_DELETE(nib_entry_head, nib_entry);
free(nib_entry);
}
/**
* Clear address list of a Neighbor Tuple and add Lost Neighbor Tuple for addresses
* no longer used by this neighbor
*/
static void clear_nb_addresses(nib_entry_t *nib_entry, timex_t *now)
{
nhdp_addr_entry_t *nib_elt = NULL, *nib_tmp = NULL;;
LL_FOREACH_SAFE(nib_entry->address_list_head, nib_elt, nib_tmp) {
/* Check whether address is still present in the new neighbor address list */
if (!NHDP_ADDR_TMP_IN_NB_LIST(nib_elt->address)) {
/* Address is not in the newly received address list of the neighbor */
/* Add it to the Removed Address List */
nib_elt->address->in_tmp_table |= NHDP_ADDR_TMP_REM_LIST;
/* Increment usage counter of address in central NHDP address storage */
nib_elt->address->usg_count++;
if (nib_entry->symmetric) {
/* Additionally create a Lost Neighbor Tuple for symmetric neighbors */
add_lost_neighbor_address(nib_elt->address, now);
}
}
/* Free the address entry */
nhdp_free_addr_entry(nib_elt);
}
nib_entry->address_list_head = NULL;
}
/**
* Add or update a Lost Neighbor Tuple
*/
static int add_lost_neighbor_address(nhdp_addr_t *lost_addr, timex_t *now)
{
nib_lost_address_entry_t *elt = NULL;
timex_t n_hold = timex_from_uint64(((uint64_t)NHDP_N_HOLD_TIME_MS) * US_PER_MS);
LL_FOREACH(nib_lost_address_entry_head, elt) {
if (elt->address == lost_addr) {
/* Existing entry for this address, no need to add a new one */
if (timex_cmp(elt->expiration_time, *now) == -1) {
/* Entry expired, so just update expiration time */
elt->expiration_time = timex_add(*now, n_hold);
}
return 0;
}
}
/* No existing entry, create a new one */
elt = malloc(sizeof(nib_lost_address_entry_t));
if (!elt) {
/* Insufficient memory */
return -1;
}
/* Increment usage counter of address in central NHDP address storage */
lost_addr->usg_count++;
elt->address = lost_addr;
elt->expiration_time = timex_add(*now, n_hold);
LL_PREPEND(nib_lost_address_entry_head, elt);
return 0;
}
/**
* Remove a given Lost Neighbor Tuple
*/
static void rem_ln_entry(nib_lost_address_entry_t *ln_entry)
{
nhdp_decrement_addr_usage(ln_entry->address);
LL_DELETE(nib_lost_address_entry_head, ln_entry);
free(ln_entry);
}