2013-06-22 05:11:53 +02:00
|
|
|
/**
|
2014-07-08 16:54:54 +02:00
|
|
|
* Copyright (C) 2013, 2014 INRIA.
|
2013-06-22 05:11:53 +02:00
|
|
|
*
|
2014-07-31 19:45:27 +02:00
|
|
|
* 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.
|
2013-06-22 05:11:53 +02:00
|
|
|
*
|
2013-08-08 13:39:00 +02:00
|
|
|
* @ingroup rpl
|
2013-06-22 05:11:53 +02:00
|
|
|
* @{
|
2014-07-08 16:54:54 +02:00
|
|
|
*
|
2015-05-22 07:34:41 +02:00
|
|
|
* @file
|
2014-07-08 16:54:54 +02:00
|
|
|
* @brief RPL DODAG
|
|
|
|
*
|
|
|
|
* Implementation of a DODAG for usage with RPL.
|
|
|
|
*
|
|
|
|
* @author Eric Engel <eric.engel@fu-berlin.de>
|
2013-06-22 05:11:53 +02:00
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
2012-01-19 17:35:50 +01:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2014-08-06 13:48:18 +02:00
|
|
|
#include "rpl/rpl_dodag.h"
|
2012-02-14 22:22:01 +01:00
|
|
|
#include "trickle.h"
|
2012-02-26 19:30:48 +01:00
|
|
|
#include "rpl.h"
|
2012-01-19 17:35:50 +01:00
|
|
|
|
2013-11-21 20:47:44 +01:00
|
|
|
#define ENABLE_DEBUG (0)
|
2013-11-22 23:59:33 +01:00
|
|
|
#if ENABLE_DEBUG
|
|
|
|
char addr_str[IPV6_MAX_ADDR_STR_LEN];
|
|
|
|
#endif
|
2013-11-21 20:47:44 +01:00
|
|
|
#include "debug.h"
|
|
|
|
|
2015-02-01 16:01:56 +01:00
|
|
|
static rpl_instance_t instances[RPL_MAX_INSTANCES];
|
2015-03-17 11:38:00 +01:00
|
|
|
rpl_dodag_t rpl_dodags[RPL_MAX_DODAGS];
|
2015-02-01 16:01:56 +01:00
|
|
|
static rpl_parent_t parents[RPL_MAX_PARENTS];
|
2012-01-19 17:35:50 +01:00
|
|
|
|
2015-01-22 12:56:04 +01:00
|
|
|
void rpl_trickle_send_dio(void *args)
|
|
|
|
{
|
2014-10-14 08:14:45 +02:00
|
|
|
ipv6_addr_t mcast;
|
|
|
|
|
2015-02-25 20:33:13 +01:00
|
|
|
ipv6_addr_set_all_rpl_nodes_addr(&mcast);
|
2015-03-16 12:26:17 +01:00
|
|
|
rpl_send_DIO((rpl_dodag_t *) args, &mcast);
|
2014-10-14 08:14:45 +02:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
rpl_instance_t *rpl_new_instance(uint8_t instanceid)
|
|
|
|
{
|
|
|
|
rpl_instance_t *inst;
|
|
|
|
rpl_instance_t *end ;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (inst = &instances[0], end = inst + RPL_MAX_INSTANCES; inst < end; inst++) {
|
|
|
|
if (inst->used == 0) {
|
2013-06-22 05:11:53 +02:00
|
|
|
memset(inst, 0, sizeof(*inst));
|
|
|
|
inst->used = 1;
|
|
|
|
inst->id = instanceid;
|
|
|
|
return inst;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2012-01-19 17:35:50 +01:00
|
|
|
}
|
2015-03-21 16:24:47 +01:00
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
rpl_instance_t *rpl_get_instance(uint8_t instanceid)
|
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
for (int i = 0; i < RPL_MAX_INSTANCES; i++) {
|
|
|
|
if (instances[i].used && (instances[i].id == instanceid)) {
|
2013-06-22 05:11:53 +02:00
|
|
|
return &instances[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2012-01-19 17:35:50 +01:00
|
|
|
}
|
|
|
|
|
2014-05-07 12:36:32 +02:00
|
|
|
rpl_instance_t *rpl_get_my_instance(void)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
for (int i = 0; i < RPL_MAX_INSTANCES; i++) {
|
|
|
|
if (instances[i].joined) {
|
2013-06-22 05:11:53 +02:00
|
|
|
return &instances[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2012-01-19 17:35:50 +01:00
|
|
|
}
|
2012-01-31 19:36:26 +01:00
|
|
|
|
2015-03-16 09:35:14 +01:00
|
|
|
rpl_dodag_t *rpl_new_dodag(rpl_instance_t *inst, ipv6_addr_t *dodagid)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
if (inst == NULL) {
|
2015-03-22 17:19:38 +01:00
|
|
|
DEBUGF("Error - No instance specified\n");
|
2013-06-22 05:11:53 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
rpl_dodag_t *dodag;
|
|
|
|
rpl_dodag_t *end;
|
|
|
|
|
2015-03-17 11:38:00 +01:00
|
|
|
for (dodag = &rpl_dodags[0], end = dodag + RPL_MAX_DODAGS; dodag < end; dodag++) {
|
2013-06-24 22:37:35 +02:00
|
|
|
if (dodag->used == 0) {
|
2013-06-22 05:11:53 +02:00
|
|
|
memset(dodag, 0, sizeof(*dodag));
|
|
|
|
dodag->instance = inst;
|
|
|
|
dodag->my_rank = INFINITE_RANK;
|
|
|
|
dodag->used = 1;
|
2014-10-14 08:14:45 +02:00
|
|
|
dodag->ack_received = true;
|
|
|
|
dodag->dao_counter = 0;
|
|
|
|
dodag->trickle.callback.func = &rpl_trickle_send_dio;
|
2015-03-16 12:26:17 +01:00
|
|
|
dodag->trickle.callback.args = dodag;
|
2013-06-22 05:11:53 +02:00
|
|
|
memcpy(&dodag->dodag_id, dodagid, sizeof(*dodagid));
|
|
|
|
return dodag;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2012-01-19 17:35:50 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-03-16 13:15:05 +01:00
|
|
|
rpl_dodag_t *rpl_get_joined_dodag(uint8_t instanceid)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
for (int i = 0; i < RPL_MAX_DODAGS; i++) {
|
2015-03-16 13:15:05 +01:00
|
|
|
if (rpl_dodags[i].joined && rpl_dodags[i].instance->id == instanceid) {
|
2015-03-17 11:38:00 +01:00
|
|
|
return &rpl_dodags[i];
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2012-01-19 17:35:50 +01:00
|
|
|
}
|
2015-03-21 16:24:47 +01:00
|
|
|
|
2014-05-07 12:36:32 +02:00
|
|
|
rpl_dodag_t *rpl_get_my_dodag(void)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
for (int i = 0; i < RPL_MAX_DODAGS; i++) {
|
2015-03-17 11:38:00 +01:00
|
|
|
if (rpl_dodags[i].joined) {
|
|
|
|
return &rpl_dodags[i];
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2012-01-19 17:35:50 +01:00
|
|
|
}
|
2015-03-21 16:24:47 +01:00
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void rpl_del_dodag(rpl_dodag_t *dodag)
|
|
|
|
{
|
2014-10-14 08:14:45 +02:00
|
|
|
rpl_leave_dodag(dodag);
|
2013-06-22 05:11:53 +02:00
|
|
|
memset(dodag, 0, sizeof(*dodag));
|
2012-01-19 17:35:50 +01:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void rpl_leave_dodag(rpl_dodag_t *dodag)
|
|
|
|
{
|
|
|
|
dodag->joined = 0;
|
|
|
|
dodag->my_preferred_parent = NULL;
|
2015-03-17 19:42:23 +01:00
|
|
|
rpl_delete_all_parents(dodag);
|
2014-10-14 08:14:45 +02:00
|
|
|
trickle_stop(&dodag->trickle);
|
|
|
|
vtimer_remove(&dodag->dao_timer);
|
2012-01-19 17:35:50 +01:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
bool rpl_equal_id(ipv6_addr_t *id1, ipv6_addr_t *id2)
|
|
|
|
{
|
2014-07-08 16:54:54 +02:00
|
|
|
for (uint8_t i = 0; i < 2; i++) {
|
|
|
|
DEBUGF("ID1: %d ID2: %d\n", id1->uint8[12 + i], id2->uint8[12 + i]);
|
|
|
|
|
|
|
|
if (id1->uint8[14 + i] != id2->uint8[14 + i]) {
|
2013-06-22 05:11:53 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2012-02-22 00:50:40 +01:00
|
|
|
|
2012-01-19 17:35:50 +01:00
|
|
|
}
|
2012-01-26 20:26:55 +01:00
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
rpl_parent_t *rpl_new_parent(rpl_dodag_t *dodag, ipv6_addr_t *address, uint16_t rank)
|
|
|
|
{
|
|
|
|
rpl_parent_t *parent;
|
|
|
|
rpl_parent_t *end;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (parent = &parents[0], end = parents + RPL_MAX_PARENTS; parent < end; parent++) {
|
|
|
|
if (parent->used == 0) {
|
2013-06-22 05:11:53 +02:00
|
|
|
memset(parent, 0, sizeof(*parent));
|
|
|
|
parent->used = 1;
|
|
|
|
parent->addr = *address;
|
|
|
|
parent->rank = rank;
|
|
|
|
parent->dodag = dodag;
|
2013-11-23 00:03:12 +01:00
|
|
|
parent->lifetime = dodag->default_lifetime * dodag->lifetime_unit;
|
2013-06-22 05:11:53 +02:00
|
|
|
/* dtsn is set at the end of recv_dio function */
|
|
|
|
parent->dtsn = 0;
|
|
|
|
return parent;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rpl_delete_worst_parent();
|
|
|
|
return rpl_new_parent(dodag, address, rank);
|
2012-01-26 20:26:55 +01:00
|
|
|
}
|
2012-01-31 19:36:26 +01:00
|
|
|
|
2015-03-19 13:06:36 +01:00
|
|
|
rpl_parent_t *rpl_find_parent(rpl_dodag_t *dodag, ipv6_addr_t *address)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
|
|
|
rpl_parent_t *parent;
|
|
|
|
rpl_parent_t *end;
|
2012-01-31 19:36:26 +01:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (parent = &parents[0], end = parents + RPL_MAX_PARENTS; parent < end; parent++) {
|
2015-03-19 13:06:36 +01:00
|
|
|
if ((parent->used) && (rpl_equal_id(address, &parent->addr)
|
|
|
|
&& (parent->dodag->instance->id == dodag->instance->id)
|
|
|
|
&& (!memcmp(&parent->dodag->dodag_id,
|
|
|
|
&dodag->dodag_id, sizeof(ipv6_addr_t))))) {
|
2013-06-22 05:11:53 +02:00
|
|
|
return parent;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2012-01-31 19:36:26 +01:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void rpl_delete_parent(rpl_parent_t *parent)
|
|
|
|
{
|
2015-03-17 19:42:23 +01:00
|
|
|
if (parent == parent->dodag->my_preferred_parent) {
|
|
|
|
parent->dodag->my_preferred_parent = NULL;
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
memset(parent, 0, sizeof(*parent));
|
2012-02-22 00:50:40 +01:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void rpl_delete_worst_parent(void)
|
|
|
|
{
|
|
|
|
uint8_t worst = 0xFF;
|
|
|
|
uint16_t max_rank = 0x0000;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (int i = 0; i < RPL_MAX_PARENTS; i++) {
|
|
|
|
if (parents[i].rank > max_rank) {
|
2013-06-22 05:11:53 +02:00
|
|
|
worst = i;
|
|
|
|
max_rank = parents[i].rank;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (worst == 0xFF) {
|
2013-06-22 05:11:53 +02:00
|
|
|
/* Fehler, keine parents -> sollte nicht passieren */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
rpl_delete_parent(&parents[worst]);
|
2012-02-14 22:22:01 +01:00
|
|
|
|
2012-02-22 00:50:40 +01:00
|
|
|
}
|
2012-02-16 23:23:15 +01:00
|
|
|
|
2015-03-17 19:42:23 +01:00
|
|
|
void rpl_delete_all_parents(rpl_dodag_t *dodag)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
|
|
|
|
2015-03-17 19:42:23 +01:00
|
|
|
dodag->my_preferred_parent = NULL;
|
2013-06-24 22:37:35 +02:00
|
|
|
for (int i = 0; i < RPL_MAX_PARENTS; i++) {
|
2015-03-17 19:42:23 +01:00
|
|
|
if (parents[i].dodag && (dodag->instance->id == parents[i].dodag->instance->id) &&
|
|
|
|
(!memcmp(&dodag->dodag_id, &parents[i].dodag->dodag_id, sizeof(ipv6_addr_t)))) {
|
|
|
|
memset(&parents[i], 0, sizeof(parents[i]));
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
2012-02-16 23:23:15 +01:00
|
|
|
}
|
|
|
|
|
2015-03-19 18:43:46 +01:00
|
|
|
rpl_parent_t *rpl_find_preferred_parent(rpl_dodag_t *my_dodag)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
|
|
|
rpl_parent_t *best = NULL;
|
|
|
|
|
2013-11-22 02:11:27 +01:00
|
|
|
if (my_dodag == NULL) {
|
|
|
|
DEBUG("Not part of a dodag\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (uint8_t i = 0; i < RPL_MAX_PARENTS; i++) {
|
2015-03-19 18:43:46 +01:00
|
|
|
if (parents[i].used
|
|
|
|
&& (parents[i].dodag->instance->id == my_dodag->instance->id)
|
|
|
|
&& (!memcmp(&parents[i].dodag->dodag_id,
|
|
|
|
&my_dodag->dodag_id, sizeof(ipv6_addr_t)))) {
|
2013-06-24 22:37:35 +02:00
|
|
|
if ((parents[i].rank == INFINITE_RANK) || (parents[i].lifetime <= 1)) {
|
2013-11-21 20:47:44 +01:00
|
|
|
DEBUG("Infinite rank, bad parent\n");
|
2013-06-22 05:11:53 +02:00
|
|
|
continue;
|
|
|
|
}
|
2013-06-24 22:37:35 +02:00
|
|
|
else if (best == NULL) {
|
2013-11-21 20:47:44 +01:00
|
|
|
DEBUG("possible parent\n");
|
2013-06-22 05:11:53 +02:00
|
|
|
best = &parents[i];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
best = my_dodag->of->which_parent(best, &parents[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (best == NULL) {
|
2013-06-22 05:11:53 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-11-21 18:50:56 +01:00
|
|
|
if (my_dodag->my_preferred_parent == NULL) {
|
|
|
|
my_dodag->my_preferred_parent = best;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (!rpl_equal_id(&my_dodag->my_preferred_parent->addr, &best->addr)) {
|
2015-01-06 08:29:27 +01:00
|
|
|
if (my_dodag->mop != RPL_MOP_NO_DOWNWARD_ROUTES) {
|
2013-06-22 05:11:53 +02:00
|
|
|
/* send DAO with ZERO_LIFETIME to old parent */
|
2015-03-16 10:52:32 +01:00
|
|
|
rpl_send_DAO(my_dodag, &my_dodag->my_preferred_parent->addr, 0, false, 0);
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
my_dodag->my_preferred_parent = best;
|
|
|
|
|
2015-01-06 08:29:27 +01:00
|
|
|
if (my_dodag->mop != RPL_MOP_NO_DOWNWARD_ROUTES) {
|
2014-10-14 08:14:45 +02:00
|
|
|
rpl_delay_dao(my_dodag);
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
|
2014-10-14 08:14:45 +02:00
|
|
|
trickle_reset_timer(&my_dodag->trickle);
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return best;
|
2012-02-26 19:30:48 +01:00
|
|
|
}
|
|
|
|
|
2015-03-19 18:43:46 +01:00
|
|
|
void rpl_parent_update(rpl_dodag_t *my_dodag, rpl_parent_t *parent)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2014-02-20 02:29:33 +01:00
|
|
|
uint16_t old_rank;
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-11-22 02:11:27 +01:00
|
|
|
if (my_dodag == NULL) {
|
|
|
|
DEBUG("Not part of a dodag - this should not happen");
|
|
|
|
return;
|
|
|
|
}
|
2014-02-20 02:29:33 +01:00
|
|
|
|
|
|
|
old_rank = my_dodag->my_rank;
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
/* update Parent lifetime */
|
2013-06-24 22:37:35 +02:00
|
|
|
if (parent != NULL) {
|
2013-06-22 05:11:53 +02:00
|
|
|
parent->lifetime = my_dodag->default_lifetime * my_dodag->lifetime_unit;
|
|
|
|
}
|
|
|
|
|
2015-03-19 18:43:46 +01:00
|
|
|
if (rpl_find_preferred_parent(my_dodag) == NULL) {
|
|
|
|
rpl_local_repair(my_dodag);
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (rpl_calc_rank(old_rank, my_dodag->minhoprankincrease) !=
|
2013-08-08 13:39:00 +02:00
|
|
|
rpl_calc_rank(my_dodag->my_rank, my_dodag->minhoprankincrease)) {
|
2013-06-24 22:37:35 +02:00
|
|
|
if (my_dodag->my_rank < my_dodag->min_rank) {
|
2013-06-22 05:11:53 +02:00
|
|
|
my_dodag->min_rank = my_dodag->my_rank;
|
|
|
|
}
|
|
|
|
|
2014-10-14 08:14:45 +02:00
|
|
|
trickle_reset_timer(&my_dodag->trickle);
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
2012-02-26 19:30:48 +01:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
void rpl_join_dodag(rpl_dodag_t *dodag, ipv6_addr_t *parent, uint16_t parent_rank)
|
|
|
|
{
|
|
|
|
rpl_dodag_t *my_dodag;
|
|
|
|
rpl_parent_t *preferred_parent;
|
2015-03-16 09:35:14 +01:00
|
|
|
my_dodag = rpl_new_dodag(dodag->instance, &dodag->dodag_id);
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (my_dodag == NULL) {
|
2013-06-22 05:11:53 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
my_dodag->instance->joined = 1;
|
|
|
|
my_dodag->of = dodag->of;
|
|
|
|
my_dodag->mop = dodag->mop;
|
|
|
|
my_dodag->dtsn = dodag->dtsn;
|
|
|
|
my_dodag->prf = dodag->prf;
|
|
|
|
my_dodag->dio_interval_doubling = dodag->dio_interval_doubling;
|
|
|
|
my_dodag->dio_min = dodag->dio_min;
|
|
|
|
my_dodag->dio_redundancy = dodag->dio_redundancy;
|
|
|
|
my_dodag->maxrankincrease = dodag->maxrankincrease;
|
|
|
|
my_dodag->minhoprankincrease = dodag->minhoprankincrease;
|
|
|
|
my_dodag->default_lifetime = dodag->default_lifetime;
|
|
|
|
my_dodag->lifetime_unit = dodag->lifetime_unit;
|
|
|
|
my_dodag->version = dodag->version;
|
|
|
|
my_dodag->grounded = dodag->grounded;
|
2015-03-03 19:36:14 +01:00
|
|
|
my_dodag->prefix_length = dodag->prefix_length;
|
|
|
|
my_dodag->prefix = dodag->prefix;
|
|
|
|
my_dodag->prefix_valid_lifetime = dodag->prefix_valid_lifetime;
|
|
|
|
my_dodag->prefix_preferred_lifetime = dodag->prefix_preferred_lifetime;
|
|
|
|
my_dodag->prefix_flags = dodag->prefix_flags;
|
2013-06-22 05:11:53 +02:00
|
|
|
my_dodag->joined = 1;
|
2014-12-14 09:20:50 +01:00
|
|
|
|
|
|
|
preferred_parent = rpl_new_parent(my_dodag, parent, parent_rank);
|
|
|
|
|
|
|
|
if (preferred_parent == NULL) {
|
|
|
|
rpl_del_dodag(my_dodag);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
my_dodag->my_preferred_parent = preferred_parent;
|
2013-06-28 17:53:21 +02:00
|
|
|
my_dodag->node_status = (uint8_t) NORMAL_NODE;
|
2013-06-22 05:11:53 +02:00
|
|
|
my_dodag->my_rank = dodag->of->calc_rank(preferred_parent, dodag->my_rank);
|
|
|
|
my_dodag->dao_seq = RPL_COUNTER_INIT;
|
|
|
|
my_dodag->min_rank = my_dodag->my_rank;
|
2013-11-22 23:59:33 +01:00
|
|
|
DEBUG("Joint DODAG:\n");
|
|
|
|
DEBUG("\tMOP:\t%02X\n", my_dodag->mop);
|
|
|
|
DEBUG("\tminhoprankincrease :\t%04X\n", my_dodag->minhoprankincrease);
|
|
|
|
DEBUG("\tdefault_lifetime:\t%02X\n", my_dodag->default_lifetime);
|
|
|
|
DEBUG("\tgrounded:\t%02X\n", my_dodag->grounded);
|
2014-02-13 13:57:59 +01:00
|
|
|
DEBUG("\tmy_preferred_parent:\t%s\n",
|
|
|
|
ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN,
|
|
|
|
&my_dodag->my_preferred_parent->addr));
|
2013-11-22 23:59:33 +01:00
|
|
|
DEBUG("\tmy_preferred_parent rank\t%02X\n", my_dodag->my_preferred_parent->rank);
|
|
|
|
DEBUG("\tmy_preferred_parent lifetime\t%04X\n", my_dodag->my_preferred_parent->lifetime);
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2014-10-14 08:14:45 +02:00
|
|
|
trickle_start(rpl_process_pid, &my_dodag->trickle, RPL_MSG_TYPE_TRICKLE_INTERVAL,
|
2015-01-22 12:56:04 +01:00
|
|
|
RPL_MSG_TYPE_TRICKLE_CALLBACK, (1 << my_dodag->dio_min), my_dodag->dio_interval_doubling,
|
|
|
|
my_dodag->dio_redundancy);
|
2014-10-14 08:14:45 +02:00
|
|
|
rpl_delay_dao(my_dodag);
|
2012-01-31 19:36:26 +01:00
|
|
|
}
|
|
|
|
|
2015-03-19 18:43:46 +01:00
|
|
|
void rpl_global_repair(rpl_dodag_t *my_dodag, ipv6_addr_t *p_addr, uint16_t rank)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2014-08-01 17:39:51 +02:00
|
|
|
DEBUGF("[INFO] Global repair started\n");
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (my_dodag == NULL) {
|
2014-08-01 17:39:51 +02:00
|
|
|
DEBUGF("[Error] - no global repair possible, if not part of a DODAG\n");
|
2013-06-22 05:11:53 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-03-17 19:42:23 +01:00
|
|
|
rpl_delete_all_parents(my_dodag);
|
2013-06-22 05:11:53 +02:00
|
|
|
my_dodag->dtsn++;
|
|
|
|
my_dodag->my_preferred_parent = rpl_new_parent(my_dodag, p_addr, rank);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (my_dodag->my_preferred_parent == NULL) {
|
2014-08-01 17:39:51 +02:00
|
|
|
DEBUGF("[Error] no more parent after global repair\n");
|
2013-06-22 05:11:53 +02:00
|
|
|
my_dodag->my_rank = INFINITE_RANK;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Calc new Rank */
|
|
|
|
my_dodag->my_rank = my_dodag->of->calc_rank(my_dodag->my_preferred_parent,
|
2013-08-08 13:39:00 +02:00
|
|
|
my_dodag->my_rank);
|
2013-06-22 05:11:53 +02:00
|
|
|
my_dodag->min_rank = my_dodag->my_rank;
|
2014-10-14 08:14:45 +02:00
|
|
|
trickle_reset_timer(&my_dodag->trickle);
|
|
|
|
rpl_delay_dao(my_dodag);
|
2013-06-22 05:11:53 +02:00
|
|
|
}
|
|
|
|
|
2014-08-01 17:39:51 +02:00
|
|
|
DEBUGF("Migrated to DODAG Version %d. My new Rank: %d\n", my_dodag->version,
|
2013-06-22 05:11:53 +02:00
|
|
|
my_dodag->my_rank);
|
2012-02-26 19:30:48 +01:00
|
|
|
}
|
|
|
|
|
2015-03-19 18:43:46 +01:00
|
|
|
void rpl_local_repair(rpl_dodag_t *my_dodag)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
2014-08-01 17:39:51 +02:00
|
|
|
DEBUGF("[INFO] Local Repair started\n");
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (my_dodag == NULL) {
|
2014-08-01 17:39:51 +02:00
|
|
|
DEBUGF("[Error] - no local repair possible, if not part of a DODAG\n");
|
2013-06-22 05:11:53 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
my_dodag->my_rank = INFINITE_RANK;
|
|
|
|
my_dodag->dtsn++;
|
2015-03-17 19:42:23 +01:00
|
|
|
rpl_delete_all_parents(my_dodag);
|
2014-10-14 08:14:45 +02:00
|
|
|
trickle_reset_timer(&my_dodag->trickle);
|
2012-02-26 19:30:48 +01:00
|
|
|
|
2012-01-31 19:36:26 +01:00
|
|
|
}
|
2012-02-14 22:22:01 +01:00
|
|
|
|
2014-05-07 12:36:32 +02:00
|
|
|
ipv6_addr_t *rpl_get_my_preferred_parent(void)
|
2013-06-22 05:11:53 +02:00
|
|
|
{
|
|
|
|
rpl_dodag_t *my_dodag = rpl_get_my_dodag();
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (my_dodag == NULL) {
|
2013-06-22 05:11:53 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return &my_dodag->my_preferred_parent->addr;
|
2012-02-14 22:22:01 +01:00
|
|
|
}
|
2012-02-26 19:30:48 +01:00
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
uint16_t rpl_calc_rank(uint16_t abs_rank, uint16_t minhoprankincrease)
|
|
|
|
{
|
|
|
|
return abs_rank / minhoprankincrease;
|
2012-02-26 19:30:48 +01:00
|
|
|
}
|