/** * Copyright (C) 2013, 2014 INRIA. * Copyright (C) 2015 Cenk Gündoğan * * 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 * @author Cenk Gündoğan */ #include #include "inet_pton.h" #include "net/ng_rpl/dodag.h" #include "net/ng_rpl/structs.h" #include "utlist.h" #include "net/ng_rpl.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)->prefix_len = NG_RPL_DEFAULT_PREFIX_LEN; (*dodag)->addr_preferred = NG_RPL_DEFAULT_PREFIX_LIFETIME; (*dodag)->addr_valid = NG_RPL_DEFAULT_PREFIX_LIFETIME; (*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; } /** * @} */