47 #include "net/rpl/rpl-private.h"
50 #include "net/nbr-table.h"
59 #define DEBUG DEBUG_NONE
64 extern rpl_of_t RPL_OF;
65 static rpl_of_t *
const objective_functions[] = {&RPL_OF};
70 #ifndef RPL_CONF_GROUNDED
71 #define RPL_GROUNDED 0
73 #define RPL_GROUNDED RPL_CONF_GROUNDED
78 NBR_TABLE(rpl_parent_t, rpl_parents);
81 rpl_instance_t instance_table[RPL_MAX_INSTANCES];
82 rpl_instance_t *default_instance;
85 nbr_callback(
void *ptr)
87 rpl_remove_parent(ptr);
93 nbr_table_register(rpl_parents, (nbr_table_callback *)nbr_callback);
99 rpl_parent_t *p = nbr_table_get_from_lladdr(rpl_parents, (linkaddr_t *)addr);
110 rpl_parent_t *p = nbr_table_get_from_lladdr(rpl_parents, (
const linkaddr_t *)addr);
112 return p->link_metric;
119 rpl_get_parent_ipaddr(rpl_parent_t *p)
121 linkaddr_t *lladdr = nbr_table_get_lladdr(rpl_parents, p);
122 return uip_ds6_nbr_ipaddr_from_lladdr((
uip_lladdr_t *)lladdr);
126 rpl_set_preferred_parent(rpl_dag_t *dag, rpl_parent_t *p)
128 if(dag !=
NULL && dag->preferred_parent != p) {
129 PRINTF(
"RPL: rpl_set_preferred_parent ");
131 PRINT6ADDR(rpl_get_parent_ipaddr(p));
135 PRINTF(
" used to be ");
136 if(dag->preferred_parent !=
NULL) {
137 PRINT6ADDR(rpl_get_parent_ipaddr(dag->preferred_parent));
145 nbr_table_unlock(rpl_parents, dag->preferred_parent);
146 nbr_table_lock(rpl_parents, p);
147 dag->preferred_parent = p;
154 lollipop_greater_than(
int a,
int b)
157 if(a > RPL_LOLLIPOP_CIRCULAR_REGION && b <= RPL_LOLLIPOP_CIRCULAR_REGION) {
158 return (RPL_LOLLIPOP_MAX_VALUE + 1 + b - a) > RPL_LOLLIPOP_SEQUENCE_WINDOWS;
162 return (a > b && (a - b) < RPL_LOLLIPOP_SEQUENCE_WINDOWS) ||
163 (a < b && (b - a) > (RPL_LOLLIPOP_CIRCULAR_REGION + 1-
164 RPL_LOLLIPOP_SEQUENCE_WINDOWS));
169 remove_parents(rpl_dag_t *dag, rpl_rank_t minimum_rank)
173 PRINTF(
"RPL: Removing parents (minimum rank %u)\n",
176 p = nbr_table_head(rpl_parents);
178 if(dag == p->dag && p->rank >= minimum_rank) {
179 rpl_remove_parent(p);
181 p = nbr_table_next(rpl_parents, p);
186 nullify_parents(rpl_dag_t *dag, rpl_rank_t minimum_rank)
190 PRINTF(
"RPL: Nullifying parents (minimum rank %u)\n",
193 p = nbr_table_head(rpl_parents);
195 if(dag == p->dag && p->rank >= minimum_rank) {
196 rpl_nullify_parent(p);
198 p = nbr_table_next(rpl_parents, p);
203 should_send_dao(rpl_instance_t *instance, rpl_dio_t *dio, rpl_parent_t *p)
206 if(instance->mop == RPL_MOP_NO_DOWNWARD_ROUTES) {
210 return p == instance->current_dag->preferred_parent &&
211 (lollipop_greater_than(dio->dtsn, p->dtsn));
215 acceptable_rank(rpl_dag_t *dag, rpl_rank_t rank)
217 return rank != INFINITE_RANK &&
218 ((dag->instance->max_rankinc == 0) ||
219 DAG_RANK(rank, dag->instance) <= DAG_RANK(dag->min_rank + dag->instance->max_rankinc, dag->instance));
223 get_dag(uint8_t instance_id, uip_ipaddr_t *dag_id)
225 rpl_instance_t *instance;
229 instance = rpl_get_instance(instance_id);
230 if(instance ==
NULL) {
234 for(i = 0; i < RPL_MAX_DAG_PER_INSTANCE; ++i) {
235 dag = &instance->dag_table[i];
236 if(dag->used && uip_ipaddr_cmp(&dag->dag_id, dag_id)) {
245 rpl_set_root(uint8_t instance_id, uip_ipaddr_t *dag_id)
248 rpl_instance_t *instance;
251 version = RPL_LOLLIPOP_INIT;
252 dag = get_dag(instance_id, dag_id);
254 version = dag->version;
255 RPL_LOLLIPOP_INCREMENT(version);
256 PRINTF(
"RPL: Dropping a joined DAG when setting this node as root");
257 if(dag == dag->instance->current_dag) {
258 dag->instance->current_dag =
NULL;
263 dag = rpl_alloc_dag(instance_id, dag_id);
265 PRINTF(
"RPL: Failed to allocate a DAG\n");
269 instance = dag->instance;
271 dag->version = version;
273 dag->grounded = RPL_GROUNDED;
274 dag->preference = RPL_PREFERENCE;
275 instance->mop = RPL_MOP_DEFAULT;
276 instance->of = &RPL_OF;
277 rpl_set_preferred_parent(dag,
NULL);
279 memcpy(&dag->dag_id, dag_id,
sizeof(dag->dag_id));
281 instance->dio_intdoubl = RPL_DIO_INTERVAL_DOUBLINGS;
282 instance->dio_intmin = RPL_DIO_INTERVAL_MIN;
285 instance->dio_intcurrent = RPL_DIO_INTERVAL_MIN +
286 RPL_DIO_INTERVAL_DOUBLINGS;
287 instance->dio_redundancy = RPL_DIO_REDUNDANCY;
288 instance->max_rankinc = RPL_MAX_RANKINC;
289 instance->min_hoprankinc = RPL_MIN_HOPRANKINC;
290 instance->default_lifetime = RPL_DEFAULT_LIFETIME;
291 instance->lifetime_unit = RPL_DEFAULT_LIFETIME_UNIT;
293 dag->rank = ROOT_RANK(instance);
295 if(instance->current_dag != dag && instance->current_dag !=
NULL) {
297 rpl_remove_routes(instance->current_dag);
299 instance->current_dag->joined = 0;
302 instance->current_dag = dag;
303 instance->dtsn_out = RPL_LOLLIPOP_INIT;
304 instance->of->update_metric_container(instance);
305 default_instance = instance;
307 PRINTF(
"RPL: Node set to be a DAG root with DAG ID ");
308 PRINT6ADDR(&dag->dag_id);
311 ANNOTATE(
"#A root=%u\n", dag->dag_id.u8[
sizeof(dag->dag_id) - 1]);
313 rpl_reset_dio_timer(instance);
319 rpl_repair_root(uint8_t instance_id)
321 rpl_instance_t *instance;
323 instance = rpl_get_instance(instance_id);
324 if(instance ==
NULL ||
325 instance->current_dag->rank != ROOT_RANK(instance)) {
326 PRINTF(
"RPL: rpl_repair_root triggered but not root\n");
330 RPL_LOLLIPOP_INCREMENT(instance->current_dag->version);
331 RPL_LOLLIPOP_INCREMENT(instance->dtsn_out);
332 PRINTF(
"RPL: rpl_repair_root initiating global repair with version %d\n", instance->current_dag->version);
333 rpl_reset_dio_timer(instance);
338 set_ip_from_prefix(uip_ipaddr_t *ipaddr, rpl_prefix_t *prefix)
340 memset(ipaddr, 0,
sizeof(uip_ipaddr_t));
341 memcpy(ipaddr, &prefix->prefix, (prefix->length + 7) / 8);
346 check_prefix(rpl_prefix_t *last_prefix, rpl_prefix_t *new_prefix)
351 if(last_prefix !=
NULL && new_prefix !=
NULL &&
352 last_prefix->length == new_prefix->length &&
353 uip_ipaddr_prefixcmp(&last_prefix->prefix, &new_prefix->prefix, new_prefix->length) &&
354 last_prefix->flags == new_prefix->flags) {
359 if(last_prefix !=
NULL) {
360 set_ip_from_prefix(&ipaddr, last_prefix);
361 rep = uip_ds6_addr_lookup(&ipaddr);
363 PRINTF(
"RPL: removing global IP address ");
366 uip_ds6_addr_rm(rep);
370 if(new_prefix !=
NULL) {
371 set_ip_from_prefix(&ipaddr, new_prefix);
372 if(uip_ds6_addr_lookup(&ipaddr) ==
NULL) {
373 PRINTF(
"RPL: adding global IP address ");
376 uip_ds6_addr_add(&ipaddr, 0, ADDR_AUTOCONF);
382 rpl_set_prefix(rpl_dag_t *dag, uip_ipaddr_t *prefix,
unsigned len)
384 rpl_prefix_t last_prefix;
385 uint8_t last_len = dag->prefix_info.length;
390 if(dag->prefix_info.length != 0) {
391 memcpy(&last_prefix, &dag->prefix_info,
sizeof(rpl_prefix_t));
393 memset(&dag->prefix_info.prefix, 0,
sizeof(dag->prefix_info.prefix));
394 memcpy(&dag->prefix_info.prefix, prefix, (len + 7) / 8);
395 dag->prefix_info.length = len;
396 dag->prefix_info.flags = UIP_ND6_RA_FLAG_AUTONOMOUS;
397 PRINTF(
"RPL: Prefix set - will announce this in DIOs\n");
401 PRINTF(
"rpl_set_prefix - prefix NULL\n");
402 check_prefix(
NULL, &dag->prefix_info);
404 PRINTF(
"rpl_set_prefix - prefix NON-NULL\n");
405 check_prefix(&last_prefix, &dag->prefix_info);
411 rpl_set_default_route(rpl_instance_t *instance, uip_ipaddr_t *from)
413 if(instance->def_route !=
NULL) {
414 PRINTF(
"RPL: Removing default route through ");
415 PRINT6ADDR(&instance->def_route->ipaddr);
417 uip_ds6_defrt_rm(instance->def_route);
418 instance->def_route =
NULL;
422 PRINTF(
"RPL: Adding default route through ");
425 instance->def_route = uip_ds6_defrt_add(from,
426 RPL_LIFETIME(instance,
427 instance->default_lifetime));
428 if(instance->def_route ==
NULL) {
432 PRINTF(
"RPL: Removing default route\n");
433 if(instance->def_route !=
NULL) {
434 uip_ds6_defrt_rm(instance->def_route);
436 PRINTF(
"RPL: Not actually removing default route, since instance had no default route\n");
443 rpl_alloc_instance(uint8_t instance_id)
445 rpl_instance_t *instance, *end;
447 for(instance = &instance_table[0], end = instance + RPL_MAX_INSTANCES;
448 instance < end; ++instance) {
449 if(instance->used == 0) {
450 memset(instance, 0,
sizeof(*instance));
451 instance->instance_id = instance_id;
452 instance->def_route =
NULL;
461 rpl_alloc_dag(uint8_t instance_id, uip_ipaddr_t *dag_id)
463 rpl_dag_t *dag, *end;
464 rpl_instance_t *instance;
466 instance = rpl_get_instance(instance_id);
467 if(instance ==
NULL) {
468 instance = rpl_alloc_instance(instance_id);
469 if(instance ==
NULL) {
470 RPL_STAT(rpl_stats.mem_overflows++);
475 for(dag = &instance->dag_table[0], end = dag + RPL_MAX_DAG_PER_INSTANCE; dag < end; ++dag) {
477 memset(dag, 0,
sizeof(*dag));
479 dag->rank = INFINITE_RANK;
480 dag->min_rank = INFINITE_RANK;
481 dag->instance = instance;
486 RPL_STAT(rpl_stats.mem_overflows++);
487 rpl_free_instance(instance);
492 rpl_set_default_instance(rpl_instance_t *instance)
494 default_instance = instance;
498 rpl_free_instance(rpl_instance_t *instance)
503 PRINTF(
"RPL: Leaving the instance %u\n", instance->instance_id);
506 for(dag = &instance->dag_table[0], end = dag + RPL_MAX_DAG_PER_INSTANCE; dag < end; ++dag) {
512 rpl_set_default_route(instance,
NULL);
517 if(default_instance == instance) {
518 default_instance =
NULL;
525 rpl_free_dag(rpl_dag_t *dag)
528 PRINTF(
"RPL: Leaving the DAG ");
529 PRINT6ADDR(&dag->dag_id);
534 rpl_remove_routes(dag);
537 if((dag->prefix_info.flags & UIP_ND6_RA_FLAG_AUTONOMOUS)) {
538 check_prefix(&dag->prefix_info,
NULL);
541 remove_parents(dag, 0);
547 rpl_add_parent(rpl_dag_t *dag, rpl_dio_t *dio, uip_ipaddr_t *addr)
549 rpl_parent_t *p =
NULL;
552 const uip_lladdr_t *lladdr = uip_ds6_nbr_lladdr_from_ipaddr(addr);
554 PRINTF(
"RPL: rpl_add_parent lladdr %p ", lladdr);
559 p = nbr_table_add_lladdr(rpl_parents, (linkaddr_t *)lladdr);
561 PRINTF(
"RPL: rpl_add_parent p NULL\n");
566 p->link_metric = RPL_INIT_LINK_METRIC * RPL_DAG_MC_ETX_DIVISOR;
567 #if RPL_DAG_MC != RPL_DAG_MC_NONE
568 memcpy(&p->mc, &dio->mc,
sizeof(p->mc));
576 static rpl_parent_t *
577 find_parent_any_dag_any_instance(uip_ipaddr_t *addr)
580 const uip_lladdr_t *lladdr = uip_ds6_nbr_get_ll(ds6_nbr);
581 return nbr_table_get_from_lladdr(rpl_parents, (linkaddr_t *)lladdr);
585 rpl_find_parent(rpl_dag_t *dag, uip_ipaddr_t *addr)
587 rpl_parent_t *p = find_parent_any_dag_any_instance(addr);
588 if(p !=
NULL && p->dag == dag) {
596 find_parent_dag(rpl_instance_t *instance, uip_ipaddr_t *addr)
598 rpl_parent_t *p = find_parent_any_dag_any_instance(addr);
607 rpl_find_parent_any_dag(rpl_instance_t *instance, uip_ipaddr_t *addr)
609 rpl_parent_t *p = find_parent_any_dag_any_instance(addr);
610 if(p && p->dag && p->dag->instance == instance) {
618 rpl_select_dag(rpl_instance_t *instance, rpl_parent_t *p)
620 rpl_parent_t *last_parent;
621 rpl_dag_t *dag, *end, *best_dag;
624 old_rank = instance->current_dag->rank;
625 last_parent = instance->current_dag->preferred_parent;
627 best_dag = instance->current_dag;
628 if(best_dag->rank != ROOT_RANK(instance)) {
629 if(rpl_select_parent(p->dag) !=
NULL) {
630 if(p->dag != best_dag) {
631 best_dag = instance->of->best_dag(best_dag, p->dag);
633 }
else if(p->dag == best_dag) {
635 for(dag = &instance->dag_table[0], end = dag + RPL_MAX_DAG_PER_INSTANCE; dag < end; ++dag) {
636 if(dag->used && dag->preferred_parent !=
NULL && dag->preferred_parent->rank != INFINITE_RANK) {
637 if(best_dag ==
NULL) {
640 best_dag = instance->of->best_dag(best_dag, dag);
647 if(best_dag ==
NULL) {
652 if(instance->current_dag != best_dag) {
654 rpl_remove_routes(instance->current_dag);
656 PRINTF(
"RPL: New preferred DAG: ");
657 PRINT6ADDR(&best_dag->dag_id);
660 if(best_dag->prefix_info.flags & UIP_ND6_RA_FLAG_AUTONOMOUS) {
661 check_prefix(&instance->current_dag->prefix_info, &best_dag->prefix_info);
662 }
else if(instance->current_dag->prefix_info.flags & UIP_ND6_RA_FLAG_AUTONOMOUS) {
663 check_prefix(&instance->current_dag->prefix_info,
NULL);
666 best_dag->joined = 1;
667 instance->current_dag->joined = 0;
668 instance->current_dag = best_dag;
671 instance->of->update_metric_container(instance);
673 best_dag->rank = instance->of->calculate_rank(best_dag->preferred_parent, 0);
674 if(last_parent ==
NULL || best_dag->rank < best_dag->min_rank) {
675 best_dag->min_rank = best_dag->rank;
676 }
else if(!acceptable_rank(best_dag, best_dag->rank)) {
677 PRINTF(
"RPL: New rank unacceptable!\n");
678 rpl_set_preferred_parent(instance->current_dag,
NULL);
679 if(instance->mop != RPL_MOP_NO_DOWNWARD_ROUTES && last_parent !=
NULL) {
681 dao_output(last_parent, RPL_ZERO_LIFETIME);
686 if(best_dag->preferred_parent != last_parent) {
687 rpl_set_default_route(instance, rpl_get_parent_ipaddr(best_dag->preferred_parent));
688 PRINTF(
"RPL: Changed preferred parent, rank changed from %u to %u\n",
689 (
unsigned)old_rank, best_dag->rank);
690 RPL_STAT(rpl_stats.parent_switch++);
691 if(instance->mop != RPL_MOP_NO_DOWNWARD_ROUTES) {
692 if(last_parent !=
NULL) {
694 dao_output(last_parent, RPL_ZERO_LIFETIME);
697 RPL_LOLLIPOP_INCREMENT(instance->dtsn_out);
698 rpl_schedule_dao(instance);
700 rpl_reset_dio_timer(instance);
701 }
else if(best_dag->rank != old_rank) {
702 PRINTF(
"RPL: Preferred parent update, rank changed from %u to %u\n",
703 (
unsigned)old_rank, best_dag->rank);
708 static rpl_parent_t *
709 best_parent(rpl_dag_t *dag)
711 rpl_parent_t *p, *best;
715 p = nbr_table_head(rpl_parents);
717 if(p->dag != dag || p->rank == INFINITE_RANK) {
719 }
else if(best ==
NULL) {
722 best = dag->instance->of->best_parent(best, p);
724 p = nbr_table_next(rpl_parents, p);
731 rpl_select_parent(rpl_dag_t *dag)
733 rpl_parent_t *best = best_parent(dag);
736 rpl_set_preferred_parent(dag, best);
743 rpl_remove_parent(rpl_parent_t *parent)
745 PRINTF(
"RPL: Removing parent ");
746 PRINT6ADDR(rpl_get_parent_ipaddr(parent));
749 rpl_nullify_parent(parent);
751 nbr_table_remove(rpl_parents, parent);
755 rpl_nullify_parent(rpl_parent_t *parent)
757 rpl_dag_t *dag = parent->dag;
760 if(parent == dag->preferred_parent || dag->preferred_parent ==
NULL) {
761 rpl_set_preferred_parent(dag,
NULL);
762 dag->rank = INFINITE_RANK;
764 if(dag->instance->def_route !=
NULL) {
765 PRINTF(
"RPL: Removing default route ");
766 PRINT6ADDR(rpl_get_parent_ipaddr(parent));
768 uip_ds6_defrt_rm(dag->instance->def_route);
769 dag->instance->def_route =
NULL;
771 dao_output(parent, RPL_ZERO_LIFETIME);
775 PRINTF(
"RPL: Nullifying parent ");
776 PRINT6ADDR(rpl_get_parent_ipaddr(parent));
781 rpl_move_parent(rpl_dag_t *dag_src, rpl_dag_t *dag_dst, rpl_parent_t *parent)
783 if(parent == dag_src->preferred_parent) {
784 rpl_set_preferred_parent(dag_src,
NULL);
785 dag_src->rank = INFINITE_RANK;
786 if(dag_src->joined && dag_src->instance->def_route !=
NULL) {
787 PRINTF(
"RPL: Removing default route ");
788 PRINT6ADDR(rpl_get_parent_ipaddr(parent));
790 PRINTF(
"rpl_move_parent\n");
791 uip_ds6_defrt_rm(dag_src->instance->def_route);
792 dag_src->instance->def_route =
NULL;
794 }
else if(dag_src->joined) {
796 rpl_remove_routes_by_nexthop(rpl_get_parent_ipaddr(parent), dag_src);
799 PRINTF(
"RPL: Moving parent ");
800 PRINT6ADDR(rpl_get_parent_ipaddr(parent));
803 parent->dag = dag_dst;
807 rpl_get_any_dag(
void)
811 for(i = 0; i < RPL_MAX_INSTANCES; ++i) {
812 if(instance_table[i].used && instance_table[i].current_dag->joined) {
813 return instance_table[i].current_dag;
820 rpl_get_instance(uint8_t instance_id)
824 for(i = 0; i < RPL_MAX_INSTANCES; ++i) {
825 if(instance_table[i].used && instance_table[i].instance_id == instance_id) {
826 return &instance_table[i];
833 rpl_find_of(rpl_ocp_t ocp)
838 i <
sizeof(objective_functions) /
sizeof(objective_functions[0]);
840 if(objective_functions[i]->ocp == ocp) {
841 return objective_functions[i];
849 rpl_join_instance(uip_ipaddr_t *from, rpl_dio_t *dio)
851 rpl_instance_t *instance;
856 dag = rpl_alloc_dag(dio->instance_id, &dio->dag_id);
858 PRINTF(
"RPL: Failed to allocate a DAG object!\n");
862 instance = dag->instance;
864 p = rpl_add_parent(dag, dio, from);
865 PRINTF(
"RPL: Adding ");
867 PRINTF(
" as a parent: ");
874 PRINTF(
"succeeded\n");
878 of = rpl_find_of(dio->ocp);
880 PRINTF(
"RPL: DIO for DAG instance %u does not specify a supported OF\n",
882 rpl_remove_parent(p);
889 if(dio->prefix_info.flags & UIP_ND6_RA_FLAG_AUTONOMOUS) {
890 check_prefix(
NULL, &dio->prefix_info);
894 dag->preference = dio->preference;
895 dag->grounded = dio->grounded;
896 dag->version = dio->version;
899 instance->mop = dio->mop;
900 instance->current_dag = dag;
901 instance->dtsn_out = RPL_LOLLIPOP_INIT;
903 instance->max_rankinc = dio->dag_max_rankinc;
904 instance->min_hoprankinc = dio->dag_min_hoprankinc;
905 instance->dio_intdoubl = dio->dag_intdoubl;
906 instance->dio_intmin = dio->dag_intmin;
907 instance->dio_intcurrent = instance->dio_intmin + instance->dio_intdoubl;
908 instance->dio_redundancy = dio->dag_redund;
909 instance->default_lifetime = dio->default_lifetime;
910 instance->lifetime_unit = dio->lifetime_unit;
912 memcpy(&dag->dag_id, &dio->dag_id,
sizeof(dio->dag_id));
915 memcpy(&dag->prefix_info, &dio->prefix_info,
sizeof(rpl_prefix_t));
917 rpl_set_preferred_parent(dag, p);
918 instance->of->update_metric_container(instance);
919 dag->rank = instance->of->calculate_rank(p, 0);
921 dag->min_rank = dag->rank;
923 if(default_instance ==
NULL) {
924 default_instance = instance;
927 PRINTF(
"RPL: Joined DAG with instance ID %u, rank %hu, DAG ID ",
928 dio->instance_id, dag->rank);
929 PRINT6ADDR(&dag->dag_id);
932 ANNOTATE(
"#A join=%u\n", dag->dag_id.u8[
sizeof(dag->dag_id) - 1]);
934 rpl_reset_dio_timer(instance);
935 rpl_set_default_route(instance, from);
937 if(instance->mop != RPL_MOP_NO_DOWNWARD_ROUTES) {
938 rpl_schedule_dao(instance);
940 PRINTF(
"RPL: The DIO does not meet the prerequisites for sending a DAO\n");
946 rpl_add_dag(uip_ipaddr_t *from, rpl_dio_t *dio)
948 rpl_instance_t *instance;
949 rpl_dag_t *dag, *previous_dag;
953 dag = rpl_alloc_dag(dio->instance_id, &dio->dag_id);
955 PRINTF(
"RPL: Failed to allocate a DAG object!\n");
959 instance = dag->instance;
961 previous_dag = find_parent_dag(instance, from);
962 if(previous_dag ==
NULL) {
963 PRINTF(
"RPL: Adding ");
965 PRINTF(
" as a parent: ");
966 p = rpl_add_parent(dag, dio, from);
972 PRINTF(
"succeeded\n");
974 p = rpl_find_parent(previous_dag, from);
976 rpl_move_parent(previous_dag, dag, p);
982 of = rpl_find_of(dio->ocp);
983 if(of != instance->of ||
984 instance->mop != dio->mop ||
985 instance->max_rankinc != dio->dag_max_rankinc ||
986 instance->min_hoprankinc != dio->dag_min_hoprankinc ||
987 instance->dio_intdoubl != dio->dag_intdoubl ||
988 instance->dio_intmin != dio->dag_intmin ||
989 instance->dio_redundancy != dio->dag_redund ||
990 instance->default_lifetime != dio->default_lifetime ||
991 instance->lifetime_unit != dio->lifetime_unit) {
992 PRINTF(
"RPL: DIO for DAG instance %u incompatible with previous DIO\n",
994 rpl_remove_parent(p);
1000 dag->grounded = dio->grounded;
1001 dag->preference = dio->preference;
1002 dag->version = dio->version;
1004 memcpy(&dag->dag_id, &dio->dag_id,
sizeof(dio->dag_id));
1007 memcpy(&dag->prefix_info, &dio->prefix_info,
sizeof(rpl_prefix_t));
1009 rpl_set_preferred_parent(dag, p);
1010 dag->rank = instance->of->calculate_rank(p, 0);
1011 dag->min_rank = dag->rank;
1013 PRINTF(
"RPL: Joined DAG with instance ID %u, rank %hu, DAG ID ",
1014 dio->instance_id, dag->rank);
1015 PRINT6ADDR(&dag->dag_id);
1018 ANNOTATE(
"#A join=%u\n", dag->dag_id.u8[
sizeof(dag->dag_id) - 1]);
1020 rpl_process_parent_event(instance, p);
1021 p->dtsn = dio->dtsn;
1026 global_repair(uip_ipaddr_t *from, rpl_dag_t *dag, rpl_dio_t *dio)
1030 remove_parents(dag, 0);
1031 dag->version = dio->version;
1032 dag->instance->of->reset(dag);
1033 dag->min_rank = INFINITE_RANK;
1034 RPL_LOLLIPOP_INCREMENT(dag->instance->dtsn_out);
1036 p = rpl_add_parent(dag, dio, from);
1038 PRINTF(
"RPL: Failed to add a parent during the global repair\n");
1039 dag->rank = INFINITE_RANK;
1041 dag->rank = dag->instance->of->calculate_rank(p, 0);
1042 dag->min_rank = dag->rank;
1043 PRINTF(
"RPL: rpl_process_parent_event global repair\n");
1044 rpl_process_parent_event(dag->instance, p);
1047 PRINTF(
"RPL: Participating in a global repair (version=%u, rank=%hu)\n",
1048 dag->version, dag->rank);
1050 RPL_STAT(rpl_stats.global_repairs++);
1054 rpl_local_repair(rpl_instance_t *instance)
1058 if(instance ==
NULL) {
1059 PRINTF(
"RPL: local repair requested for instance NULL\n");
1062 PRINTF(
"RPL: Starting a local instance repair\n");
1063 for(i = 0; i < RPL_MAX_DAG_PER_INSTANCE; i++) {
1064 if(instance->dag_table[i].used) {
1065 instance->dag_table[i].rank = INFINITE_RANK;
1066 nullify_parents(&instance->dag_table[i], 0);
1070 rpl_reset_dio_timer(instance);
1072 RPL_STAT(rpl_stats.local_repairs++);
1076 rpl_recalculate_ranks(
void)
1085 p = nbr_table_head(rpl_parents);
1087 if(p->dag !=
NULL && p->dag->instance && (p->flags & RPL_PARENT_FLAG_UPDATED)) {
1088 p->flags &= ~RPL_PARENT_FLAG_UPDATED;
1089 PRINTF(
"RPL: rpl_process_parent_event recalculate_ranks\n");
1090 if(!rpl_process_parent_event(p->dag->instance, p)) {
1091 PRINTF(
"RPL: A parent was dropped\n");
1094 p = nbr_table_next(rpl_parents, p);
1099 rpl_process_parent_event(rpl_instance_t *instance, rpl_parent_t *p)
1104 rpl_rank_t old_rank;
1105 old_rank = instance->current_dag->rank;
1110 if(!acceptable_rank(p->dag, p->rank)) {
1113 PRINTF(
"RPL: Unacceptable rank %u\n", (
unsigned)p->rank);
1114 rpl_nullify_parent(p);
1115 if(p != instance->current_dag->preferred_parent) {
1122 if(rpl_select_dag(instance, p) ==
NULL) {
1124 PRINTF(
"RPL: No parents found in any DAG\n");
1125 rpl_local_repair(instance);
1130 if(DAG_RANK(old_rank, instance) != DAG_RANK(instance->current_dag->rank, instance)) {
1131 PRINTF(
"RPL: Moving in the instance from rank %hu to %hu\n",
1132 DAG_RANK(old_rank, instance), DAG_RANK(instance->current_dag->rank, instance));
1133 if(instance->current_dag->rank != INFINITE_RANK) {
1134 PRINTF(
"RPL: The preferred parent is ");
1135 PRINT6ADDR(rpl_get_parent_ipaddr(instance->current_dag->preferred_parent));
1136 PRINTF(
" (rank %u)\n",
1137 (
unsigned)DAG_RANK(instance->current_dag->preferred_parent->rank, instance));
1139 PRINTF(
"RPL: We don't have any parent");
1144 return return_value;
1148 rpl_process_dio(uip_ipaddr_t *from, rpl_dio_t *dio)
1150 rpl_instance_t *instance;
1151 rpl_dag_t *dag, *previous_dag;
1154 #if RPL_CONF_MULTICAST
1157 if(dio->mop < RPL_MOP_STORING_NO_MULTICAST) {
1159 if(dio->mop != RPL_MOP_DEFAULT) {
1161 PRINTF(
"RPL: Ignoring a DIO with an unsupported MOP: %d\n", dio->mop);
1165 dag = get_dag(dio->instance_id, &dio->dag_id);
1166 instance = rpl_get_instance(dio->instance_id);
1168 if(dag !=
NULL && instance !=
NULL) {
1169 if(lollipop_greater_than(dio->version, dag->version)) {
1170 if(dag->rank == ROOT_RANK(instance)) {
1171 PRINTF(
"RPL: Root received inconsistent DIO version number\n");
1172 dag->version = dio->version;
1173 RPL_LOLLIPOP_INCREMENT(dag->version);
1174 rpl_reset_dio_timer(instance);
1176 PRINTF(
"RPL: Global repair\n");
1177 if(dio->prefix_info.length != 0) {
1178 if(dio->prefix_info.flags & UIP_ND6_RA_FLAG_AUTONOMOUS) {
1179 PRINTF(
"RPL : Prefix announced in DIO\n");
1180 rpl_set_prefix(dag, &dio->prefix_info.prefix, dio->prefix_info.length);
1183 global_repair(from, dag, dio);
1188 if(lollipop_greater_than(dag->version, dio->version)) {
1190 PRINTF(
"RPL: old version received => inconsistency detected\n");
1192 rpl_reset_dio_timer(instance);
1198 if(instance ==
NULL) {
1199 PRINTF(
"RPL: New instance detected: Joining...\n");
1200 rpl_join_instance(from, dio);
1204 if(instance->current_dag->rank == ROOT_RANK(instance) && instance->current_dag != dag) {
1205 PRINTF(
"RPL: Root ignored DIO for different DAG\n");
1210 PRINTF(
"RPL: Adding new DAG to known instance.\n");
1211 rpl_add_dag(from, dio);
1216 if(dio->rank < ROOT_RANK(instance)) {
1217 PRINTF(
"RPL: Ignoring DIO with too low rank: %u\n",
1218 (
unsigned)dio->rank);
1220 }
else if(dio->rank == INFINITE_RANK && dag->joined) {
1221 rpl_reset_dio_timer(instance);
1225 if(dio->prefix_info.length != 0) {
1226 if(dio->prefix_info.flags & UIP_ND6_RA_FLAG_AUTONOMOUS) {
1227 PRINTF(
"RPL : Prefix announced in DIO\n");
1228 rpl_set_prefix(dag, &dio->prefix_info.prefix, dio->prefix_info.length);
1232 if(dag->rank == ROOT_RANK(instance)) {
1233 if(dio->rank != INFINITE_RANK) {
1234 instance->dio_counter++;
1246 p = rpl_find_parent(dag, from);
1248 previous_dag = find_parent_dag(instance, from);
1249 if(previous_dag ==
NULL) {
1251 p = rpl_add_parent(dag, dio, from);
1253 PRINTF(
"RPL: Failed to add a new parent (");
1258 PRINTF(
"RPL: New candidate parent with rank %u: ", (
unsigned)p->rank);
1262 p = rpl_find_parent(previous_dag, from);
1264 rpl_move_parent(previous_dag, dag, p);
1268 if(p->rank == dio->rank) {
1269 PRINTF(
"RPL: Received consistent DIO\n");
1271 instance->dio_counter++;
1278 PRINTF(
"RPL: preferred DAG ");
1279 PRINT6ADDR(&instance->current_dag->dag_id);
1280 PRINTF(
", rank %u, min_rank %u, ",
1281 instance->current_dag->rank, instance->current_dag->min_rank);
1282 PRINTF(
"parent rank %u, parent etx %u, link metric %u, instance etx %u\n",
1283 p->rank, -1, p->link_metric, instance->mc.obj.etx);
1287 #if RPL_DAG_MC != RPL_DAG_MC_NONE
1288 memcpy(&p->mc, &dio->mc,
sizeof(p->mc));
1290 if(rpl_process_parent_event(instance, p) == 0) {
1291 PRINTF(
"RPL: The candidate parent is rejected\n");
1296 if(dag->joined && p == dag->preferred_parent) {
1297 if(should_send_dao(instance, dio, p)) {
1298 RPL_LOLLIPOP_INCREMENT(instance->dtsn_out);
1299 rpl_schedule_dao(instance);
1303 uip_ds6_defrt_add(from, RPL_LIFETIME(instance, instance->default_lifetime));
1305 p->dtsn = dio->dtsn;
1309 rpl_lock_parent(rpl_parent_t *p)
1311 nbr_table_lock(rpl_parents, p);
Linked list manipulation routines.
Neighbor discovery (RFC 4861)
CCIF uip_lladdr_t uip_lladdr
Host L2 address.
This header file contains configuration directives for uIPv6 multicast support.
Unicast address structure.
Header file for the uIP TCP/IP stack.
void uip_ds6_set_addr_iid(uip_ipaddr_t *ipaddr, uip_lladdr_t *lladdr)
set the last 64 bits of an IP address based on the MAC address
#define NULL
The null pointer.
Header file for the callback timer
A set of debugging macros.
Memory block allocation routines.
void ctimer_stop(struct ctimer *c)
Stop a pending callback timer.
An entry in the nbr cache.