mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
399 lines
12 KiB
C
399 lines
12 KiB
C
/**
|
|
* Copyright (C) 2013, 2014 INRIA.
|
|
* Copyright (C) 2015 Cenk Gündoğan <cnkgndgn@gmail.com>
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
/**
|
|
* @{
|
|
*
|
|
* @file
|
|
* @author Eric Engel <eric.engel@fu-berlin.de>
|
|
* @author Cenk Gündoğan <cnkgndgn@gmail.com>
|
|
*/
|
|
|
|
#include <stdbool.h>
|
|
#include "net/ng_rpl/dodag.h"
|
|
#include "net/ng_rpl/structs.h"
|
|
#include "utlist.h"
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
#include "debug.h"
|
|
|
|
#if ENABLE_DEBUG && defined(MODULE_PV6_ADDR)
|
|
static char addr_str[IPV6_ADDR_MAX_STR_LEN];
|
|
#endif
|
|
|
|
void rpl_trickle_send_dio(void *args)
|
|
{
|
|
ng_rpl_dodag_t *dodag = (ng_rpl_dodag_t *) args;
|
|
ipv6_addr_t all_RPL_nodes = NG_IPV6_ADDR_ALL_RPL_NODES;
|
|
ng_rpl_send_DIO(dodag, &all_RPL_nodes);
|
|
DEBUG("trickle callback: Instance (%d) | DODAG: (%s)\n", dodag->instance->id,
|
|
ipv6_addr_to_str(addr_str,&dodag->dodag_id, sizeof(addr_str)));
|
|
}
|
|
|
|
bool ng_rpl_instance_add(uint8_t instance_id, ng_rpl_instance_t **inst)
|
|
{
|
|
*inst = NULL;
|
|
bool first = true;
|
|
for (uint8_t i = 0; i < NG_RPL_INSTANCES_NUMOF; ++i) {
|
|
/* save position to the first unused instance */
|
|
if ((ng_rpl_instances[i].state == 0) && first) {
|
|
*inst = &ng_rpl_instances[i];
|
|
first = false;
|
|
continue;
|
|
}
|
|
else if ((ng_rpl_instances[i].state != 0) && (ng_rpl_instances[i].id == instance_id)) {
|
|
DEBUG("Instance with id %d exists\n", instance_id);
|
|
*inst = &ng_rpl_instances[i];
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (*inst != NULL) {
|
|
(*inst)->id = instance_id;
|
|
(*inst)->state = 1;
|
|
(*inst)->max_rank_inc = NG_RPL_DEFAULT_MAX_RANK_INCREASE;
|
|
(*inst)->min_hop_rank_inc = NG_RPL_DEFAULT_MIN_HOP_RANK_INCREASE;
|
|
(*inst)->dodags = NULL;
|
|
return true;
|
|
}
|
|
|
|
/* no space available to allocate a new instance */
|
|
DEBUG("Could not allocate a new RPL instance\n");
|
|
*inst = NULL;
|
|
return false;
|
|
}
|
|
|
|
bool ng_rpl_instance_remove_by_id(uint8_t instance_id)
|
|
{
|
|
for(uint8_t i = 0; i < NG_RPL_INSTANCES_NUMOF; ++i) {
|
|
if (ng_rpl_instances[i].id == instance_id) {
|
|
ng_rpl_dodag_t *elt, *tmp;
|
|
LL_FOREACH_SAFE(ng_rpl_instances[i].dodags, elt, tmp) {
|
|
ng_rpl_dodag_remove(elt);
|
|
}
|
|
memset(&ng_rpl_instances[i], 0, sizeof(ng_rpl_instance_t));
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool ng_rpl_instance_remove(ng_rpl_instance_t *inst)
|
|
{
|
|
ng_rpl_dodag_t *elt, *tmp;
|
|
LL_FOREACH_SAFE(inst->dodags, elt, tmp) {
|
|
ng_rpl_dodag_remove(elt);
|
|
}
|
|
memset(inst, 0, sizeof(ng_rpl_instance_t));
|
|
return true;
|
|
}
|
|
|
|
ng_rpl_instance_t *ng_rpl_instance_get(uint8_t instance_id)
|
|
{
|
|
for (uint8_t i = 0; i < NG_RPL_INSTANCES_NUMOF; ++i) {
|
|
if (ng_rpl_instances[i].id == instance_id) {
|
|
return &ng_rpl_instances[i];
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
bool ng_rpl_dodag_add(ng_rpl_instance_t *instance, ipv6_addr_t *dodag_id, ng_rpl_dodag_t **dodag)
|
|
{
|
|
if ((instance == NULL) || instance->state == 0) {
|
|
DEBUG("Instance is NULL or unused\n");
|
|
return false;
|
|
}
|
|
|
|
*dodag = NULL;
|
|
bool first = true;
|
|
for (uint8_t i = 0; i < NG_RPL_DODAGS_NUMOF; ++i) {
|
|
/* save position to the first unused instance */
|
|
if ((ng_rpl_dodags[i].state == 0) && first) {
|
|
*dodag = &ng_rpl_dodags[i];
|
|
first = false;
|
|
continue;
|
|
}
|
|
else if ((ng_rpl_dodags[i].state != 0) &&
|
|
(ng_rpl_dodags[i].instance->id == instance->id) &&
|
|
ipv6_addr_equal(&ng_rpl_dodags[i].dodag_id, dodag_id)) {
|
|
DEBUG("DODAG with id: %s does exist\n", ipv6_addr_to_str(addr_str, dodag_id,
|
|
sizeof(addr_str)));
|
|
*dodag = &ng_rpl_dodags[i];
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (*dodag != NULL) {
|
|
(*dodag)->instance = instance;
|
|
LL_APPEND(instance->dodags, *dodag);
|
|
(*dodag)->state = 1;
|
|
(*dodag)->dodag_id = *dodag_id;
|
|
(*dodag)->my_rank = NG_RPL_INFINITE_RANK;
|
|
(*dodag)->trickle.callback.func = &rpl_trickle_send_dio;
|
|
(*dodag)->trickle.callback.args = *dodag;
|
|
(*dodag)->dio_interval_doubl = NG_RPL_DEFAULT_DIO_INTERVAL_DOUBLINGS;
|
|
(*dodag)->dio_min = NG_RPL_DEFAULT_DIO_INTERVAL_MIN;
|
|
(*dodag)->dio_redun = NG_RPL_DEFAULT_DIO_REDUNDANCY_CONSTANT;
|
|
(*dodag)->default_lifetime = NG_RPL_DEFAULT_LIFETIME;
|
|
(*dodag)->lifetime_unit = NG_RPL_LIFETIME_UNIT;
|
|
(*dodag)->node_status = NG_RPL_NORMAL_NODE;
|
|
(*dodag)->dao_seq = NG_RPL_COUNTER_INIT;
|
|
(*dodag)->dtsn = 0;
|
|
(*dodag)->dao_ack_received = false;
|
|
(*dodag)->dao_counter = 0;
|
|
(*dodag)->parents = NULL;
|
|
(*dodag)->cleanup_time = timex_set(NG_RPL_CLEANUP_TIME, 0);
|
|
return true;
|
|
}
|
|
|
|
/* no space available to allocate a new dodag */
|
|
DEBUG("Could not allocate a new RPL DODAG\n");
|
|
*dodag = NULL;
|
|
return false;
|
|
}
|
|
|
|
bool ng_rpl_dodag_remove(ng_rpl_dodag_t *dodag)
|
|
{
|
|
ng_rpl_dodag_remove_all_parents(dodag);
|
|
ng_rpl_instance_t *inst = dodag->instance;
|
|
LL_DELETE(inst->dodags, dodag);
|
|
trickle_stop(&dodag->trickle);
|
|
vtimer_remove(&dodag->dao_timer);
|
|
vtimer_remove(&dodag->cleanup_timer);
|
|
memset(dodag, 0, sizeof(ng_rpl_dodag_t));
|
|
if (inst->dodags == NULL) {
|
|
ng_rpl_instance_remove(inst);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void ng_rpl_dodag_remove_all_parents(ng_rpl_dodag_t *dodag)
|
|
{
|
|
ng_rpl_parent_t *elt, *tmp;
|
|
LL_FOREACH_SAFE(dodag->parents, elt, tmp) {
|
|
ng_rpl_parent_remove(elt);
|
|
}
|
|
vtimer_remove(&dodag->cleanup_timer);
|
|
vtimer_set_msg(&dodag->cleanup_timer, dodag->cleanup_time, ng_rpl_pid,
|
|
NG_RPL_MSG_TYPE_CLEANUP_HANDLE, dodag);
|
|
}
|
|
|
|
ng_rpl_dodag_t *ng_rpl_dodag_get(ng_rpl_instance_t *instance, ipv6_addr_t *dodag_id)
|
|
{
|
|
if ((instance == NULL) || (instance->state == 0)) {
|
|
DEBUG("Instance is NULL or unused\n");
|
|
return NULL;
|
|
}
|
|
|
|
ng_rpl_dodag_t *dodag = NULL;
|
|
LL_FOREACH(instance->dodags, dodag) {
|
|
if (ipv6_addr_equal(&dodag->dodag_id, dodag_id)) {
|
|
return dodag;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
bool ng_rpl_parent_add_by_addr(ng_rpl_dodag_t *dodag, ipv6_addr_t *addr, ng_rpl_parent_t **parent)
|
|
{
|
|
if ((dodag == NULL) || (dodag->state == 0)) {
|
|
DEBUG("DODAG is NULL or unused\n");
|
|
return false;
|
|
}
|
|
|
|
*parent = NULL;
|
|
bool first = true;
|
|
for (uint8_t i = 0; i < NG_RPL_PARENTS_NUMOF; ++i) {
|
|
/* save position to the first unused instance */
|
|
if ((ng_rpl_parents[i].state == 0) && first) {
|
|
*parent = &ng_rpl_parents[i];
|
|
first = false;
|
|
continue;
|
|
}
|
|
/* return false if parent exists */
|
|
else if ((ng_rpl_parents[i].state != 0) &&
|
|
(ng_rpl_parents[i].dodag->instance->id == dodag->instance->id) &&
|
|
ipv6_addr_equal(&ng_rpl_parents[i].dodag->dodag_id, &dodag->dodag_id) &&
|
|
ipv6_addr_equal(&ng_rpl_parents[i].addr, addr)) {
|
|
DEBUG("parent with addr: %s does exist\n", ipv6_addr_to_str(addr_str, addr,
|
|
sizeof(addr_str)));
|
|
*parent = &ng_rpl_parents[i];
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (*parent != NULL) {
|
|
(*parent)->dodag = dodag;
|
|
LL_APPEND(dodag->parents, *parent);
|
|
(*parent)->state = 1;
|
|
(*parent)->addr = *addr;
|
|
if ((*parent) == (*parent)->dodag->parents) {
|
|
ipv6_addr_t all_RPL_nodes = NG_IPV6_ADDR_ALL_RPL_NODES;
|
|
ipv6_addr_t def = IPV6_ADDR_UNSPECIFIED;
|
|
kernel_pid_t if_id = ng_ipv6_netif_find_by_addr(NULL, &all_RPL_nodes);
|
|
if (if_id == KERNEL_PID_UNDEF) {
|
|
DEBUG("RPL: no interface found for the parent addres\n");
|
|
return false;
|
|
}
|
|
if (fib_add_entry(if_id, def.u8, sizeof(ipv6_addr_t), AF_INET6, dodag->parents->addr.u8,
|
|
sizeof(ipv6_addr_t), AF_INET6,
|
|
(dodag->default_lifetime * dodag->lifetime_unit) * SEC_IN_MS) != 0) {
|
|
DEBUG("RPL: error adding parent to FIB\n");
|
|
ng_rpl_parent_remove(*parent);
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/* no space available to allocate a new parent */
|
|
DEBUG("Could not allocate a new parent\n");
|
|
*parent = NULL;
|
|
return false;
|
|
}
|
|
|
|
ng_rpl_parent_t *ng_rpl_parent_get(ng_rpl_dodag_t *dodag, ipv6_addr_t *addr)
|
|
{
|
|
if ((dodag == NULL) || (dodag->state == 0)) {
|
|
DEBUG("DODAG is NULL or unused\n");
|
|
return NULL;
|
|
}
|
|
|
|
ng_rpl_parent_t *parent = NULL;
|
|
LL_FOREACH(dodag->parents, parent) {
|
|
if (ipv6_addr_equal(&parent->addr, addr)) {
|
|
return parent;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
bool ng_rpl_parent_remove(ng_rpl_parent_t *parent)
|
|
{
|
|
if (parent == parent->dodag->parents) {
|
|
ipv6_addr_t def = { .u64 = {{0}, {0}} };
|
|
fib_remove_entry(def.u8, sizeof(ipv6_addr_t));
|
|
}
|
|
ng_rpl_dodag_t *dodag = parent->dodag;
|
|
LL_DELETE(dodag->parents, parent);
|
|
memset(parent, 0, sizeof(ng_rpl_parent_t));
|
|
return true;
|
|
}
|
|
|
|
void ng_rpl_local_repair(ng_rpl_dodag_t *dodag)
|
|
{
|
|
DEBUG("RPL: [INFO] Local Repair started\n");
|
|
|
|
dodag->dtsn++;
|
|
|
|
if (dodag->parents) {
|
|
ng_rpl_dodag_remove_all_parents(dodag);
|
|
ipv6_addr_t def = IPV6_ADDR_UNSPECIFIED;
|
|
fib_remove_entry(def.u8, sizeof(ipv6_addr_t));
|
|
}
|
|
|
|
if (dodag->my_rank != NG_RPL_INFINITE_RANK) {
|
|
trickle_reset_timer(&dodag->trickle);
|
|
vtimer_remove(&dodag->cleanup_timer);
|
|
vtimer_set_msg(&dodag->cleanup_timer, dodag->cleanup_time, ng_rpl_pid,
|
|
NG_RPL_MSG_TYPE_CLEANUP_HANDLE, dodag);
|
|
}
|
|
|
|
dodag->my_rank = NG_RPL_INFINITE_RANK;
|
|
}
|
|
|
|
void ng_rpl_parent_update(ng_rpl_dodag_t *dodag, ng_rpl_parent_t *parent)
|
|
{
|
|
uint16_t old_rank = dodag->my_rank;
|
|
timex_t now;
|
|
vtimer_now(&now);
|
|
ipv6_addr_t def = IPV6_ADDR_UNSPECIFIED;
|
|
|
|
/* update Parent lifetime */
|
|
if (parent != NULL) {
|
|
parent->lifetime.seconds = now.seconds + (dodag->default_lifetime * dodag->lifetime_unit);
|
|
parent->lifetime.microseconds = 0;
|
|
if (parent == dodag->parents) {
|
|
ipv6_addr_t all_RPL_nodes = NG_IPV6_ADDR_ALL_RPL_NODES;
|
|
kernel_pid_t if_id;
|
|
if ((if_id = ng_ipv6_netif_find_by_addr(NULL, &all_RPL_nodes)) != KERNEL_PID_UNDEF) {
|
|
fib_add_entry(if_id, def.u8, sizeof(ipv6_addr_t), AF_INET6,
|
|
dodag->parents->addr.u8, sizeof(ipv6_addr_t), AF_INET6,
|
|
(dodag->default_lifetime * dodag->lifetime_unit) * SEC_IN_MS);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ng_rpl_find_preferred_parent(dodag) == NULL) {
|
|
ng_rpl_local_repair(dodag);
|
|
}
|
|
|
|
if (dodag->parents && (old_rank != dodag->my_rank)) {
|
|
trickle_reset_timer(&dodag->trickle);
|
|
}
|
|
}
|
|
|
|
int _compare_parents(ng_rpl_parent_t *p1, ng_rpl_parent_t *p2)
|
|
{
|
|
return p1->dodag->instance->of->which_parent(p1, p2) == p1 ? -1 : 1;
|
|
}
|
|
|
|
ng_rpl_parent_t *ng_rpl_find_preferred_parent(ng_rpl_dodag_t *dodag)
|
|
{
|
|
ipv6_addr_t def = IPV6_ADDR_UNSPECIFIED;
|
|
ng_rpl_parent_t *old_best = dodag->parents;
|
|
|
|
LL_SORT(dodag->parents, _compare_parents);
|
|
|
|
if (dodag->parents == NULL) {
|
|
return NULL;
|
|
}
|
|
else if (dodag->parents->rank >= dodag->my_rank) {
|
|
ng_rpl_parent_remove(dodag->parents);
|
|
return NULL;
|
|
}
|
|
|
|
dodag->my_rank = dodag->instance->of->calc_rank(dodag->parents, 0);
|
|
ng_rpl_parent_t *elt, *tmp;
|
|
LL_FOREACH_SAFE(dodag->parents, elt, tmp) {
|
|
if (dodag->parents->rank < elt->rank) {
|
|
ng_rpl_parent_remove(elt);
|
|
}
|
|
}
|
|
|
|
if (old_best != dodag->parents) {
|
|
if (dodag->instance->mop != NG_RPL_MOP_NO_DOWNWARD_ROUTES) {
|
|
ng_rpl_send_DAO(dodag, &old_best->addr, 0);
|
|
ng_rpl_delay_dao(dodag);
|
|
}
|
|
trickle_reset_timer(&dodag->trickle);
|
|
|
|
fib_remove_entry(def.u8, sizeof(ipv6_addr_t));
|
|
ipv6_addr_t all_RPL_nodes = NG_IPV6_ADDR_ALL_RPL_NODES;
|
|
|
|
kernel_pid_t if_id = ng_ipv6_netif_find_by_addr(NULL, &all_RPL_nodes);
|
|
|
|
if (if_id == KERNEL_PID_UNDEF) {
|
|
DEBUG("RPL: no interface found for the parent addres\n");
|
|
return NULL;
|
|
}
|
|
|
|
fib_add_entry(if_id, def.u8, sizeof(ipv6_addr_t), AF_INET6, dodag->parents->addr.u8,
|
|
sizeof(ipv6_addr_t), AF_INET6,
|
|
(dodag->default_lifetime * dodag->lifetime_unit) * SEC_IN_MS);
|
|
}
|
|
|
|
return dodag->parents;
|
|
}
|
|
|
|
/**
|
|
* @}
|
|
*/
|