#include #include #include "rpl_dodag.h" #include "trickle.h" #include "rpl.h" rpl_instance_t instances[RPL_MAX_INSTANCES]; rpl_dodag_t dodags[RPL_MAX_DODAGS]; rpl_parent_t parents[RPL_MAX_PARENTS]; rpl_instance_t *rpl_new_instance(uint8_t instanceid){ rpl_instance_t *inst; rpl_instance_t *end ; for(inst=&instances[0], end = inst+RPL_MAX_INSTANCES; inst < end;inst++){ if(inst->used == 0){ memset(inst, 0, sizeof(*inst)); inst->used = 1; inst->id = instanceid; return inst; } } return NULL; } rpl_instance_t *rpl_get_instance(uint8_t instanceid){ for(int i=0;iused == 0){ memset(dodag, 0,sizeof(*dodag)); dodag->instance = inst; dodag->my_rank = INFINITE_RANK; dodag->used = 1; memcpy(&dodag->dodag_id,dodagid,sizeof(*dodagid)); return dodag; } } return NULL; } rpl_dodag_t *rpl_get_dodag(ipv6_addr_t *id){ for(int i=0;ijoined = 0; dodag->my_preferred_parent = NULL; rpl_delete_all_parents(); } bool rpl_equal_id(ipv6_addr_t *id1, ipv6_addr_t *id2){ for(uint8_t i=0;i<4;i++){ if(id1->uint32[i] != id2->uint32[i]){ return false; } } return true; } 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; for(parent= &parents[0], end=parents+RPL_MAX_PARENTS; parent < end; parent++){ if(parent->used == 0){ memset(parent, 0, sizeof(*parent)); parent->used = 1; parent->addr = *address; parent->rank = rank; parent->dodag = dodag; //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); } rpl_parent_t *rpl_find_parent(ipv6_addr_t *address){ rpl_parent_t *parent; rpl_parent_t *end; for(parent= &parents[0], end=parents+RPL_MAX_PARENTS; parent < end; parent++){ if( (parent->used) && (rpl_equal_id(address, &parent->addr)) ){ return parent; } } return NULL; } void rpl_delete_parent(rpl_parent_t * parent){ rpl_dodag_t * my_dodag = rpl_get_my_dodag(); if( (my_dodag != NULL) && rpl_equal_id(&my_dodag->my_preferred_parent->addr, &parent->addr) ){ my_dodag->my_preferred_parent = NULL; } memset(parent,0,sizeof(*parent)); } void rpl_delete_worst_parent(void){ uint8_t worst = 0xFF; uint16_t max_rank = 0x0000; for(int i=0;i max_rank){ worst = i; max_rank = parents[i].rank; } } if(worst == 0xFF){ //Fehler, keine parents -> sollte nicht passieren return; } rpl_delete_parent(&parents[worst]); } void rpl_delete_all_parents(void){ rpl_dodag_t * my_dodag = rpl_get_my_dodag(); my_dodag->my_preferred_parent = NULL; for(int i=0;iof->which_parent(best, &parents[i]); } } } if(best == NULL){ return NULL; } if(!rpl_equal_id(&my_dodag->my_preferred_parent->addr, &best->addr)){ if(my_dodag->mop != NO_DOWNWARD_ROUTES){ //send DAO with ZERO_LIFETIME to old parent send_DAO(&my_dodag->my_preferred_parent->addr, 0, false); } my_dodag->my_preferred_parent = best; if(my_dodag->mop != NO_DOWNWARD_ROUTES){ delay_dao(); } reset_trickletimer(); } return best; } void rpl_parent_update(rpl_parent_t * parent){ rpl_dodag_t * my_dodag = rpl_get_my_dodag(); uint16_t old_rank = my_dodag->my_rank; if(rpl_find_preferred_parent() == NULL){ rpl_local_repair(); } if(rpl_calc_rank(old_rank, my_dodag->minhoprankincrease) != rpl_calc_rank(my_dodag->my_rank, my_dodag->minhoprankincrease)){ if(my_dodag->my_rank < my_dodag->min_rank){ my_dodag->min_rank = my_dodag->my_rank; } reset_trickletimer(); } } 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; my_dodag = rpl_new_dodag(dodag->instance->id, &dodag->dodag_id); if(my_dodag == NULL){ return; } preferred_parent = rpl_new_parent(my_dodag, parent, parent_rank); if(preferred_parent == NULL){ rpl_del_dodag(my_dodag); 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; my_dodag->joined = 1; my_dodag->my_preferred_parent = preferred_parent; 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; start_trickle(my_dodag->dio_min, my_dodag->dio_interval_doubling, my_dodag->dio_redundancy); delay_dao(); } void rpl_global_repair(rpl_dodag_t *dodag, ipv6_addr_t * p_addr, uint16_t rank){ rpl_dodag_t * my_dodag = rpl_get_my_dodag(); if(my_dodag == NULL){ printf("[Error] - no global repair possible, if not part of a DODAG\n"); return; } rpl_delete_all_parents(); my_dodag->version = dodag->version; my_dodag->dtsn++; my_dodag->my_preferred_parent = rpl_new_parent(my_dodag, p_addr, rank); if(my_dodag->my_preferred_parent == NULL){ printf("[Error] no more parent after global repair\n"); my_dodag->my_rank = INFINITE_RANK; }else{ //Calc new Rank my_dodag->my_rank = my_dodag->of->calc_rank(my_dodag->my_preferred_parent, my_dodag->my_rank); my_dodag->min_rank = my_dodag->my_rank; reset_trickletimer(); delay_dao(); } printf("Migrated to DODAG Version %d. My new Rank: %d\n", my_dodag->version, my_dodag->my_rank); } void rpl_local_repair(void){ rpl_dodag_t * my_dodag = rpl_get_my_dodag(); if(my_dodag == NULL){ printf("[Error] - no local repair possible, if not part of a DODAG\n"); return; } my_dodag->my_rank = INFINITE_RANK; my_dodag->dtsn++; rpl_delete_all_parents(); reset_trickletimer(); } ipv6_addr_t *rpl_get_my_preferred_parent(){ rpl_dodag_t * my_dodag = rpl_get_my_dodag(); if(my_dodag == NULL){ return NULL; } return &my_dodag->my_preferred_parent->addr; } uint16_t rpl_calc_rank(uint16_t abs_rank, uint16_t minhoprankincrease){ return abs_rank / minhoprankincrease; }