/* * Copyright (C) 2017 Freie Universität Berlin * * 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 Martine Lenders */ #include #include #include #include #include "log.h" #include "net/ipv6/addr.h" #include "net/gnrc/icmpv6/error.h" #include "net/gnrc/nettype.h" #include "net/gnrc/netif/internal.h" #include "net/gnrc/ipv6/nib.h" #include "net/gnrc/ndp.h" #include "net/gnrc/netreg.h" #include "net/gnrc/pktqueue.h" #include "net/gnrc/sixlowpan/nd.h" #include "net/ndp.h" #include "net/sixlowpan/nd.h" #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_DNS) #include "net/sock/dns.h" #endif #if IS_ACTIVE(MODULE_DHCPV6_CLIENT) #include "net/dhcpv6/client.h" #endif #include "_nib-internal.h" #include "_nib-arsm.h" #include "_nib-router.h" #include "_nib-6ln.h" #include "_nib-6lr.h" #include "_nib-slaac.h" #define ENABLE_DEBUG 0 #include "debug.h" #if IS_ACTIVE(ENABLE_DEBUG) #include "evtimer.h" #endif static char addr_str[IPV6_ADDR_MAX_STR_LEN]; #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_QUEUE_PKT) static gnrc_pktqueue_t _queue_pool[CONFIG_GNRC_IPV6_NIB_NUMOF]; #endif /* CONFIG_GNRC_IPV6_NIB_QUEUE_PKT */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_DNS) static evtimer_msg_event_t _rdnss_timeout; #endif /** * @internal * @{ */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ROUTER) static void _handle_rtr_sol(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const ndp_rtr_sol_t *rtr_sol, size_t icmpv6_len); #endif /* CONFIG_GNRC_IPV6_NIB_ROUTER */ static void _handle_rtr_adv(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const ndp_rtr_adv_t *rtr_adv, size_t icmpv6_len); static void _handle_nbr_sol(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const ndp_nbr_sol_t *nbr_sol, size_t icmpv6_len); static void _handle_nbr_adv(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const ndp_nbr_adv_t *nbr_adv, size_t icmpv6_len); static bool _resolve_addr(const ipv6_addr_t *dst, gnrc_netif_t *netif, gnrc_pktsnip_t *pkt, gnrc_ipv6_nib_nc_t *nce, _nib_onl_entry_t *entry); static void _handle_pfx_timeout(_nib_offl_entry_t *pfx); static void _handle_rtr_timeout(_nib_dr_entry_t *router); static void _handle_snd_na(gnrc_pktsnip_t *pkt); /* needs to be exported for 6LN's ARO handling */ void _handle_search_rtr(gnrc_netif_t *netif); #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_DNS) static void _handle_rdnss_timeout(sock_udp_ep_t *dns_server); #endif /** @} */ static inline bool _should_search_rtr(const gnrc_netif_t *netif) { /* 6LBR interface does not send RS. A non-advertising router sends RS or a 6LN that is advertising or not has to refetch router information */ return !gnrc_netif_is_6lbr(netif) && (!gnrc_netif_is_rtr_adv(netif) || gnrc_netif_is_6ln(netif)); } void gnrc_ipv6_nib_init(void) { evtimer_event_t *tmp; _nib_acquire(); for (evtimer_event_t *ptr = _nib_evtimer.events; (ptr != NULL) && (tmp = (ptr->next), 1); ptr = tmp) { evtimer_del((evtimer_t *)(&_nib_evtimer), ptr); } _nib_init(); _nib_release(); } static void _add_static_lladdr(gnrc_netif_t *netif) { #ifdef GNRC_IPV6_STATIC_LLADDR /* parse addr from string and explicitly set a link local prefix * if ifnum > 1 each interface will get its own link local address * with GNRC_IPV6_STATIC_LLADDR + i */ char lladdr_str[] = GNRC_IPV6_STATIC_LLADDR; ipv6_addr_t lladdr; if (ipv6_addr_from_str(&lladdr, lladdr_str) != NULL) { lladdr.u8[15] += netif->pid; assert(ipv6_addr_is_link_local(&lladdr)); gnrc_netif_ipv6_addr_add_internal( netif, &lladdr, 64U, GNRC_NETIF_IPV6_ADDRS_FLAGS_STATE_VALID ); } #else (void)netif; #endif } void gnrc_ipv6_nib_iface_up(gnrc_netif_t *netif) { assert(netif != NULL); gnrc_netif_acquire(netif); _init_iface_arsm(netif); netif->ipv6.rs_sent = 0; netif->ipv6.na_sent = 0; _auto_configure_addr(netif, &ipv6_addr_link_local_prefix, 64U); if (_should_search_rtr(netif)) { uint32_t next_rs_time = random_uint32_range(0, NDP_MAX_RS_MS_DELAY); _evtimer_add(netif, GNRC_IPV6_NIB_SEARCH_RTR, &netif->ipv6.search_rtr, next_rs_time); } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ROUTER) else { _handle_snd_mc_ra(netif); } #endif /* CONFIG_GNRC_IPV6_NIB_ROUTER */ gnrc_netif_release(netif); } void gnrc_ipv6_nib_iface_down(gnrc_netif_t *netif, bool send_final_ra) { assert(netif != NULL); DEBUG("nib: Deinitialize interface %u\n", netif->pid); gnrc_netif_acquire(netif); _deinit_iface_arsm(netif); if (_should_search_rtr(netif)) { _evtimer_del(&netif->ipv6.search_rtr); } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ROUTER) else { _evtimer_del(&netif->ipv6.snd_mc_ra); if (send_final_ra) { /* trigger final RA with lifetime set to zero */ netif->ipv6.ra_sent = (UINT8_MAX - NDP_MAX_FIN_RA_NUMOF) + 1; _handle_snd_mc_ra(netif); } } #else (void)send_final_ra; #endif for (unsigned i = 0; i < CONFIG_GNRC_NETIF_IPV6_ADDRS_NUMOF; i++) { if (ipv6_addr_is_link_local(&netif->ipv6.addrs[i])) { /* link-local address might change on reconnect */ gnrc_netif_ipv6_addr_remove_internal(netif, &netif->ipv6.addrs[i]); } } gnrc_netif_release(netif); } void gnrc_ipv6_nib_init_iface(gnrc_netif_t *netif) { assert(netif != NULL); DEBUG("nib: Initialize interface %u\n", netif->pid); gnrc_netif_acquire(netif); netif->ipv6.retrans_time = NDP_RETRANS_TIMER_MS; #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_SLAAC) || IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LN) /* TODO: set differently dependent on CONFIG_GNRC_IPV6_NIB_SLAAC if * alternatives exist */ netif->ipv6.aac_mode |= GNRC_NETIF_AAC_AUTO; #endif /* CONFIG_GNRC_IPV6_NIB_SLAAC || CONFIG_GNRC_IPV6_NIB_6LN */ _init_iface_router(netif); gnrc_netif_init_6ln(netif); if (gnrc_netif_ipv6_group_join_internal(netif, &ipv6_addr_all_nodes_link_local) < 0) { DEBUG("nib: Can't join link-local all-nodes on interface %u\n", netif->pid); gnrc_netif_release(netif); return; } _add_static_lladdr(netif); gnrc_netif_release(netif); } static bool _on_link(const ipv6_addr_t *dst, unsigned *iface) { _nib_offl_entry_t *entry = NULL; _nib_offl_entry_t *match = NULL; if (ipv6_addr_is_link_local(dst)) { return true; } while ((entry = _nib_offl_iter(entry))) { if ((ipv6_addr_match_prefix(dst, &entry->pfx) >= entry->pfx_len) && ((match == NULL) || (entry->pfx_len > match->pfx_len))) { match = entry; } } if (match) { *iface = _nib_onl_get_if(match->next_hop); /* check if prefix is on-link */ return (match->mode & _PL) && (match->flags & _PFX_ON_LINK); } return false; } static gnrc_netif_t *_acquire_new_iface(unsigned iface) { gnrc_netif_t *netif = gnrc_netif_get_by_pid(iface); /* release NIB, in case other thread calls a NIB function while we wait for * the netif */ _nib_release(); gnrc_netif_acquire(netif); /* re-acquire NIB */ _nib_acquire(); return netif; } int gnrc_ipv6_nib_get_next_hop_l2addr(const ipv6_addr_t *dst, gnrc_netif_t *netif, gnrc_pktsnip_t *pkt, gnrc_ipv6_nib_nc_t *nce) { int res = 0; DEBUG("nib: get next hop link-layer address of %s%%%u\n", ipv6_addr_to_str(addr_str, dst, sizeof(addr_str)), (netif != NULL) ? (unsigned)netif->pid : 0U); gnrc_netif_acquire(netif); _nib_acquire(); do { /* XXX: hidden goto ;-) */ _nib_onl_entry_t *node = _nib_onl_nc_get(dst, (netif == NULL) ? 0 : netif->pid); /* consider neighbor cache entries first */ unsigned iface = (node == NULL) ? 0 : _nib_onl_get_if(node); if ((node != NULL) || _on_link(dst, &iface)) { DEBUG("nib: %s is %s, start address resolution\n", ipv6_addr_to_str(addr_str, dst, sizeof(addr_str)), node ? "in NC" : "on-link"); /* on-link prefixes return their interface */ if (!ipv6_addr_is_link_local(dst) && (iface != 0)) { /* release pre-assumed netif */ gnrc_netif_release(netif); netif = _acquire_new_iface(iface); /* get node from proper interface */ if (netif != NULL) { node = _nib_onl_nc_get(dst, netif->pid); } } if ((netif == NULL) || !_resolve_addr(dst, netif, pkt, nce, node)) { DEBUG("nib: host unreachable\n"); /* _resolve_addr releases pkt only if not queued (in which case * we also shouldn't release), but if netif is not defined we * should release in any case. */ if ((netif == NULL) && (pkt != NULL)) { gnrc_icmpv6_error_dst_unr_send(ICMPV6_ERROR_DST_UNR_ADDR, pkt); gnrc_pktbuf_release_error(pkt, EHOSTUNREACH); } res = -EHOSTUNREACH; break; } } else { gnrc_ipv6_nib_ft_t route; DEBUG("nib: %s is off-link, resolve route\n", ipv6_addr_to_str(addr_str, dst, sizeof(addr_str))); res = _nib_get_route(dst, pkt, &route); if ((res < 0) || ipv6_addr_is_unspecified(&route.next_hop)) { DEBUG("nib: no route to %s found or is prefix list entry, " "search neighbor cache\n", ipv6_addr_to_str(addr_str, dst, sizeof(addr_str))); if ((res == 0) && /* If ARSM is not active only use link-local as next hop */ (IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ARSM) || ipv6_addr_is_link_local(dst))) { DEBUG("nib: prefix list entry => taking dst as next hop\n"); memcpy(&route.next_hop, dst, sizeof(route.next_hop)); } else { res = -ENETUNREACH; if (pkt != NULL) { gnrc_icmpv6_error_dst_unr_send( ICMPV6_ERROR_DST_UNR_NO_ROUTE, pkt ); gnrc_pktbuf_release_error(pkt, ENETUNREACH); } break; } } if ((netif != NULL) && (netif->pid != (int)route.iface)) { /* release pre-assumed netif */ gnrc_netif_release(netif); } if ((netif == NULL) || (netif->pid != (int)route.iface)) { /* get actual netif */ netif = _acquire_new_iface(route.iface); } node = _nib_onl_nc_get(&route.next_hop, (netif != NULL) ? netif->pid : 0); if (_resolve_addr(&route.next_hop, netif, pkt, nce, node)) { _call_route_info_cb(netif, GNRC_IPV6_NIB_ROUTE_INFO_TYPE_RN, &route.dst, (void *)((intptr_t)route.dst_len)); #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_DC) _nib_dc_add(&route.next_hop, netif->pid, dst); #endif /* CONFIG_GNRC_IPV6_NIB_DC */ } else { /* _resolve_addr releases pkt if not queued (in which case * we also shouldn't release */ res = -EHOSTUNREACH; } } } while (0); _nib_release(); gnrc_netif_release(netif); return res; } void gnrc_ipv6_nib_handle_pkt(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const icmpv6_hdr_t *icmpv6, size_t icmpv6_len) { DEBUG("nib: Handle packet (icmpv6->type = %u)\n", icmpv6->type); assert(netif != NULL); gnrc_netif_acquire(netif); _nib_acquire(); switch (icmpv6->type) { #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ROUTER) case ICMPV6_RTR_SOL: _handle_rtr_sol(netif, ipv6, (ndp_rtr_sol_t *)icmpv6, icmpv6_len); break; #endif /* CONFIG_GNRC_IPV6_NIB_ROUTER */ case ICMPV6_RTR_ADV: _handle_rtr_adv(netif, ipv6, (ndp_rtr_adv_t *)icmpv6, icmpv6_len); break; case ICMPV6_NBR_SOL: _handle_nbr_sol(netif, ipv6, (ndp_nbr_sol_t *)icmpv6, icmpv6_len); break; case ICMPV6_NBR_ADV: _handle_nbr_adv(netif, ipv6, (ndp_nbr_adv_t *)icmpv6, icmpv6_len); break; #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_REDIRECT) case ICMPV6_REDIRECT: /* TODO */ break; #endif /* CONFIG_GNRC_IPV6_NIB_REDIRECT */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_DAD) case ICMPV6_DAR: /* TODO */ break; case ICMPV6_DAC: /* TODO */ break; #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_DAD */ } _nib_release(); gnrc_netif_release(netif); } void gnrc_ipv6_nib_handle_timer_event(void *ctx, uint16_t type) { DEBUG("nib: Handle timer event (ctx = %p, type = 0x%04x, now = %ums)\n", ctx, type, (unsigned)evtimer_now_msec()); _nib_acquire(); switch (type) { #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ARSM) case GNRC_IPV6_NIB_SND_UC_NS: case GNRC_IPV6_NIB_SND_MC_NS: _handle_snd_ns(ctx); break; case GNRC_IPV6_NIB_REACH_TIMEOUT: case GNRC_IPV6_NIB_DELAY_TIMEOUT: _handle_state_timeout(ctx); break; case GNRC_IPV6_NIB_RECALC_REACH_TIME: _recalc_reach_time(ctx); break; #endif /* CONFIG_GNRC_IPV6_NIB_ARSM */ case GNRC_IPV6_NIB_SND_NA: _handle_snd_na(ctx); break; case GNRC_IPV6_NIB_SEARCH_RTR: _handle_search_rtr(ctx); break; #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ROUTER) case GNRC_IPV6_NIB_REPLY_RS: _handle_reply_rs(ctx); break; case GNRC_IPV6_NIB_SND_MC_RA: _handle_snd_mc_ra(ctx); break; case GNRC_IPV6_NIB_ROUTE_TIMEOUT: _nib_ft_remove(ctx); break; #endif /* CONFIG_GNRC_IPV6_NIB_ROUTER */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LR) case GNRC_IPV6_NIB_ADDR_REG_TIMEOUT: _nib_nc_remove(ctx); break; #endif /* CONFIG_GNRC_IPV6_NIB_6LR */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C) case GNRC_IPV6_NIB_ABR_TIMEOUT: _nib_abr_remove(&((_nib_abr_entry_t *)ctx)->addr); break; #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ case GNRC_IPV6_NIB_PFX_TIMEOUT: _handle_pfx_timeout(ctx); break; case GNRC_IPV6_NIB_RTR_TIMEOUT: _handle_rtr_timeout(ctx); break; #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LN) case GNRC_IPV6_NIB_REREG_ADDRESS: _handle_rereg_address(ctx); break; #endif /* CONFIG_GNRC_IPV6_NIB_6LN */ case GNRC_IPV6_NIB_DAD: _handle_dad(ctx); break; case GNRC_IPV6_NIB_VALID_ADDR: _handle_valid_addr(ctx); break; #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_DNS) case GNRC_IPV6_NIB_RDNSS_TIMEOUT: _handle_rdnss_timeout(ctx); #endif default: break; } _nib_release(); } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ROUTER) void gnrc_ipv6_nib_change_rtr_adv_iface(gnrc_netif_t *netif, bool enable) { gnrc_netif_acquire(netif); if (enable) { _set_rtr_adv(netif); } else { uint32_t next_rs_time = random_uint32_range(0, NDP_MAX_RS_MS_DELAY); netif->ipv6.ra_sent = (UINT8_MAX - NDP_MAX_FIN_RA_NUMOF) + 1; netif->flags &= ~GNRC_NETIF_FLAGS_IPV6_RTR_ADV; /* send final router advertisements */ _handle_snd_mc_ra(netif); if (!gnrc_netif_is_6lbr(netif)) { _evtimer_add(netif, GNRC_IPV6_NIB_SEARCH_RTR, &netif->ipv6.search_rtr, next_rs_time); } } gnrc_netif_release(netif); } #endif /* CONFIG_GNRC_IPV6_NIB_ROUTER */ /* * @internal * @{ */ static void _handle_mtuo(gnrc_netif_t *netif, const icmpv6_hdr_t *icmpv6, const ndp_opt_mtu_t *mtuo); #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_DNS) static uint32_t _handle_rdnsso(gnrc_netif_t *netif, const icmpv6_hdr_t *icmpv6, const ndp_opt_rdnss_impl_t *rdnsso); #endif #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C) static uint32_t _handle_pio(gnrc_netif_t *netif, const icmpv6_hdr_t *icmpv6, const ndp_opt_pi_t *pio, _nib_abr_entry_t *abr); #else /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ static uint32_t _handle_pio(gnrc_netif_t *netif, const icmpv6_hdr_t *icmpv6, const ndp_opt_pi_t *pio); #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ static uint32_t _handle_rio(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const ndp_opt_ri_t *pio); /** @} */ /* Iterator for NDP options in a packet */ #define FOREACH_OPT(ndp_pkt, opt, icmpv6_len) \ for (opt = (ndp_opt_t *)(ndp_pkt + 1); \ icmpv6_len > 0; \ icmpv6_len -= (opt->len << 3), \ opt = (ndp_opt_t *)(((uint8_t *)opt) + (opt->len << 3))) #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ROUTER) static void _handle_rtr_sol(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const ndp_rtr_sol_t *rtr_sol, size_t icmpv6_len) { size_t tmp_len = icmpv6_len - sizeof(ndp_rtr_sol_t); _nib_onl_entry_t *nce = NULL; ndp_opt_t *opt; assert(netif != NULL); /* check validity, see: https://tools.ietf.org/html/rfc4861#section-6.1.1 */ /* checksum is checked by GNRC's ICMPv6 module */ if (!(gnrc_netif_is_rtr(netif)) || (ipv6->hl != NDP_HOP_LIMIT) || (rtr_sol->code != 0U) || (icmpv6_len < sizeof(ndp_rtr_sol_t))) { DEBUG("nib: Received router solicitation is invalid (or interface %i " "is not a forwarding interface). Discarding silently\n", netif->pid); DEBUG(" - IP Hop Limit: %u (should be %u)\n", ipv6->hl, NDP_HOP_LIMIT); DEBUG(" - ICMP code: %u (should be 0)\n", rtr_sol->code); DEBUG(" - ICMP length: %u (should > %u)\n", (unsigned)icmpv6_len, (unsigned)sizeof(ndp_rtr_sol_t)); return; } /* pre-check option length */ FOREACH_OPT(rtr_sol, opt, tmp_len) { if (tmp_len > icmpv6_len) { DEBUG("nib: Payload length (%u) of RS doesn't align with options\n", (unsigned)icmpv6_len); return; } if (opt->len == 0U) { DEBUG("nib: Option of length 0 detected. " "Discarding router solicitation silently\n"); return; } if ((opt->type == NDP_OPT_SL2A) && ipv6_addr_is_unspecified(&ipv6->src)) { DEBUG("nib: RS contains SLLAO, but source was unspecfied. " "Discarding router solicitation silently\n"); return; } } DEBUG("nib: Received valid router solicitation:\n"); DEBUG(" - Source address: %s\n", ipv6_addr_to_str(addr_str, &ipv6->src, sizeof(addr_str))); DEBUG(" - Destination address: %s\n", ipv6_addr_to_str(addr_str, &ipv6->dst, sizeof(addr_str))); if (!ipv6_addr_is_unspecified(&ipv6->src)) { tmp_len = icmpv6_len - sizeof(ndp_rtr_sol_t); FOREACH_OPT(rtr_sol, opt, tmp_len) { switch (opt->type) { case NDP_OPT_SL2A: if (!gnrc_netif_is_6ln(netif)) { _handle_sl2ao(netif, ipv6, (const icmpv6_hdr_t *)rtr_sol, opt); } break; default: break; } } nce = _nib_onl_nc_get(&ipv6->src, netif->pid); } if (!gnrc_netif_is_6ln(netif)) { uint32_t next_ra_delay = random_uint32_range(0, NDP_MAX_RA_DELAY); uint32_t next_ra_scheduled = _evtimer_lookup(netif, GNRC_IPV6_NIB_SND_MC_RA); if (next_ra_scheduled < next_ra_delay) { DEBUG("nib: There is a MC RA scheduled within the next %" PRIu32 "ms. " "Using that to advertise router\n", next_ra_scheduled); return; } else if (nce != NULL) { /* we send unicast RAs so we do not need to rate-limit as * https://tools.ietf.org/html/rfc4861#section-6.2.6 asks for */ _evtimer_add(nce, GNRC_IPV6_NIB_REPLY_RS, &nce->reply_rs, next_ra_delay); } else { uint32_t now = evtimer_now_msec(); /* check for integer overflows and initial value of last_ra */ if (((netif->ipv6.last_ra > (UINT32_MAX - NDP_MIN_MS_DELAY_BETWEEN_RAS) && (now < NDP_MIN_MS_DELAY_BETWEEN_RAS))) || ((now - NDP_MIN_MS_DELAY_BETWEEN_RAS) > netif->ipv6.last_ra)) { next_ra_delay += NDP_MIN_MS_DELAY_BETWEEN_RAS; } _evtimer_add(netif, GNRC_IPV6_NIB_SND_MC_RA, &netif->ipv6.snd_mc_ra, next_ra_delay); } } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LR) else if (gnrc_netif_is_rtr(netif) && gnrc_netif_is_rtr_adv(netif)) { _snd_rtr_advs(netif, &ipv6->src, false); } #endif /* CONFIG_GNRC_IPV6_NIB_6LR */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LN) (void)nce; /* NCE is not used */ #endif } #endif /* CONFIG_GNRC_IPV6_NIB_ROUTER */ static inline uint32_t _min(uint32_t a, uint32_t b) { return (a < b) ? a : b; } static void _handle_rtr_adv(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const ndp_rtr_adv_t *rtr_adv, size_t icmpv6_len) { size_t tmp_len = icmpv6_len - sizeof(ndp_rtr_adv_t); _nib_dr_entry_t *dr = NULL; ndp_opt_t *opt; #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C) sixlowpan_nd_opt_abr_t *abro = NULL; _nib_abr_entry_t *abr = NULL; #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ uint32_t next_timeout = UINT32_MAX; assert(netif != NULL); /* check validity, see: https://tools.ietf.org/html/rfc4861#section-6.1.1 */ /* checksum is checked by GNRC's ICMPv6 module */ if (!(ipv6_addr_is_link_local(&ipv6->src)) || (ipv6->hl != NDP_HOP_LIMIT) || (rtr_adv->code != 0U) || (icmpv6_len < sizeof(ndp_rtr_adv_t)) || (!gnrc_netif_is_6ln(netif) && (byteorder_ntohs(rtr_adv->ltime) > NDP_RTR_ADV_LTIME_SEC_MAX))) { DEBUG("nib: Received router advertisement is invalid. " "Discarding silently\n"); DEBUG(" - IP Hop Limit: %u (should be %u)\n", ipv6->hl, NDP_HOP_LIMIT); DEBUG(" - ICMP code: %u (should be 0)\n", rtr_adv->code); DEBUG(" - ICMP length: %u (should > %u)\n", (unsigned)icmpv6_len, (unsigned)sizeof(ndp_rtr_adv_t)); DEBUG(" - Source address: %s (should be link-local)\n", ipv6_addr_to_str(addr_str, &ipv6->src, sizeof(addr_str))); DEBUG(" - Router lifetime: %u (should be <= 9000 on non-6LN)\n", byteorder_ntohs(rtr_adv->ltime)); return; } /* pre-check option length */ FOREACH_OPT(rtr_adv, opt, tmp_len) { if (tmp_len > icmpv6_len) { DEBUG("nib: Payload length (%u) of RA doesn't align with options\n", (unsigned)icmpv6_len); return; } if (opt->len == 0U) { DEBUG("nib: Option of length 0 detected. " "Discarding router advertisement silently\n"); return; } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C) if (opt->type == NDP_OPT_ABR) { if (abro != NULL) { DEBUG("nib: More than one ABRO. " "Discarding router advertisement silently\n"); return; } abro = (sixlowpan_nd_opt_abr_t *)opt; } #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ } DEBUG("nib: Received valid router advertisement:\n"); DEBUG(" - Source address: %s\n", ipv6_addr_to_str(addr_str, &ipv6->src, sizeof(addr_str))); DEBUG(" - Destination address: %s\n", ipv6_addr_to_str(addr_str, &ipv6->dst, sizeof(addr_str))); DEBUG(" - Cur Hop Limit: %u\n", rtr_adv->cur_hl); DEBUG(" - Flags: %c%c\n", (rtr_adv->flags & NDP_RTR_ADV_FLAGS_M) ? 'M' : '-', (rtr_adv->flags & NDP_RTR_ADV_FLAGS_O) ? 'O' : '-'); DEBUG(" - Router Lifetime: %us\n", byteorder_ntohs(rtr_adv->ltime)); DEBUG(" - Reachable Time: %" PRIu32 "ms\n", byteorder_ntohl(rtr_adv->reach_time)); DEBUG(" - Retrans Timer: %" PRIu32 "ms\n", byteorder_ntohl(rtr_adv->retrans_timer)); #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C) if (abro != NULL) { if ((abr = _handle_abro(abro)) == NULL) { DEBUG("nib: could not allocate space for new border router or " "there is no new information in the RA. " "Discarding silently\n"); return; } /* UINT16_MAX * 60 * 1000 < UINT32_MAX so there are no overflows */ next_timeout = _min(next_timeout, MS_PER_SEC * SEC_PER_MIN * gnrc_sixlowpan_nd_opt_get_ltime(abro)); } #if !IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LBR) else if (gnrc_netif_is_6lr(netif)) { DEBUG("nib: multihop prefix and context dissemination on router activated,\n" " but no ABRO found. Discarding router advertisement silently\n"); return; } #endif /* !CONFIG_GNRC_IPV6_NIB_6LBR */ #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ if (!gnrc_netif_is_6lbr(netif) && rtr_adv->ltime.u16 != 0) { uint16_t rtr_ltime = byteorder_ntohs(rtr_adv->ltime); dr = _nib_drl_add(&ipv6->src, netif->pid); if (dr != NULL) { _evtimer_add(dr, GNRC_IPV6_NIB_RTR_TIMEOUT, &dr->rtr_timeout, rtr_ltime * MS_PER_SEC); } else { DEBUG("nib: default router list is full. Ignoring RA from %s\n", ipv6_addr_to_str(addr_str, &ipv6->src, sizeof(addr_str))); return; } /* UINT16_MAX * 1000 < UINT32_MAX so there are no overflows */ next_timeout = _min(next_timeout, rtr_ltime * MS_PER_SEC); } else { dr = _nib_drl_get(&ipv6->src, netif->pid); DEBUG("nib: router lifetime was 0. Removing router and routes via it."); if (dr != NULL) { _handle_rtr_timeout(dr); } dr = NULL; } if (rtr_adv->cur_hl != 0) { netif->cur_hl = rtr_adv->cur_hl; } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ARSM) if (rtr_adv->reach_time.u32 != 0) { uint32_t reach_time = byteorder_ntohl(rtr_adv->reach_time); if (reach_time != netif->ipv6.reach_time_base) { _evtimer_add(netif, GNRC_IPV6_NIB_RECALC_REACH_TIME, &netif->ipv6.recalc_reach_time, CONFIG_GNRC_IPV6_NIB_REACH_TIME_RESET); netif->ipv6.reach_time_base = reach_time; _recalc_reach_time(&netif->ipv6); } } #endif /* CONFIG_GNRC_IPV6_NIB_ARSM */ if (rtr_adv->retrans_timer.u32 != 0) { netif->ipv6.retrans_time = byteorder_ntohl(rtr_adv->retrans_timer); } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LN) if ((dr != NULL) && gnrc_netif_is_6ln(netif) && !gnrc_netif_is_6lbr(netif)) { /* (register addresses already assigned but not valid yet)*/ for (int i = 0; i < CONFIG_GNRC_NETIF_IPV6_ADDRS_NUMOF; i++) { if ((netif->ipv6.addrs_flags[i] != 0) && (netif->ipv6.addrs_flags[i] != GNRC_NETIF_IPV6_ADDRS_FLAGS_STATE_VALID)) { _handle_rereg_address(&netif->ipv6.addrs[i]); } } } #endif /* CONFIG_GNRC_IPV6_NIB_6LN */ tmp_len = icmpv6_len - sizeof(ndp_rtr_adv_t); FOREACH_OPT(rtr_adv, opt, tmp_len) { switch (opt->type) { case NDP_OPT_SL2A: _handle_sl2ao(netif, ipv6, (const icmpv6_hdr_t *)rtr_adv, opt); break; case NDP_OPT_MTU: _handle_mtuo(netif, (const icmpv6_hdr_t *)rtr_adv, (ndp_opt_mtu_t *)opt); break; case NDP_OPT_RI: _handle_rio(netif, ipv6, (ndp_opt_ri_t *)opt); break; case NDP_OPT_PI: { uint32_t min_pfx_timeout; #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C) min_pfx_timeout = _handle_pio(netif, (const icmpv6_hdr_t *)rtr_adv, (ndp_opt_pi_t *)opt, abr); #else /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ min_pfx_timeout = _handle_pio(netif, (const icmpv6_hdr_t *)rtr_adv, (ndp_opt_pi_t *)opt); #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ next_timeout = _min(next_timeout, min_pfx_timeout); /* notify optional PIO consumer */ if (IS_USED(MODULE_GNRC_IPV6_NIB_RTR_ADV_PIO_CB)) { extern void gnrc_ipv6_nib_rtr_adv_pio_cb(gnrc_netif_t *netif, const ndp_opt_pi_t *pio); gnrc_ipv6_nib_rtr_adv_pio_cb(netif, (ndp_opt_pi_t *)opt); } break; } /* ABRO was already secured in the option check above */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LN) case NDP_OPT_6CTX: #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C) next_timeout = _min(_handle_6co((icmpv6_hdr_t *)rtr_adv, (sixlowpan_nd_opt_6ctx_t *)opt, abr), next_timeout); #else /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ next_timeout = _min(_handle_6co((icmpv6_hdr_t *)rtr_adv, (sixlowpan_nd_opt_6ctx_t *)opt), next_timeout); #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ break; #endif /* CONFIG_GNRC_IPV6_NIB_6LN */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_DNS) case NDP_OPT_RDNSS: next_timeout = _min(_handle_rdnsso(netif, (icmpv6_hdr_t *)rtr_adv, (ndp_opt_rdnss_impl_t *)opt), next_timeout); break; #endif default: break; } } /* we still don't have a default router */ if (dr == NULL) { return; } #if IS_ACTIVE(MODULE_DHCPV6_CLIENT) uint8_t current_conf_mode = dhcpv6_client_get_conf_mode(); if (rtr_adv->flags & NDP_RTR_ADV_FLAGS_M) { if (IS_USED(MODULE_DHCPV6_CLIENT_IA_NA)) { netif->ipv6.aac_mode |= GNRC_NETIF_AAC_DHCP; dhcpv6_client_req_ia_na(netif->pid); } else { dhcpv6_client_set_conf_mode(DHCPV6_CLIENT_CONF_MODE_STATEFUL); } } else if ((rtr_adv->flags & NDP_RTR_ADV_FLAGS_O) && (current_conf_mode != DHCPV6_CLIENT_CONF_MODE_STATEFUL)) { dhcpv6_client_set_conf_mode(DHCPV6_CLIENT_CONF_MODE_STATELESS); } #endif /* MODULE_DHCPV6_CLIENT */ /* stop sending router solicitations * see https://tools.ietf.org/html/rfc4861#section-6.3.7 */ if (!gnrc_netif_is_6lbr(netif)) { evtimer_del(&_nib_evtimer, &netif->ipv6.search_rtr.event); } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LN) if (gnrc_netif_is_6ln(netif) && !gnrc_netif_is_6lbr(netif)) { if (IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C)) { _set_rtr_adv(netif); } /* but re-fetch information from router in time */ _evtimer_add(netif, GNRC_IPV6_NIB_SEARCH_RTR, &netif->ipv6.search_rtr, (next_timeout >> 2) * 3); /* i.e. 3/4 of the time before the earliest expires */ } #endif /* CONFIG_GNRC_IPV6_NIB_6LN */ } static inline size_t _get_l2src(const gnrc_netif_t *netif, uint8_t *l2src) { #if GNRC_NETIF_L2ADDR_MAXLEN > 0 memcpy(l2src, netif->l2addr, netif->l2addr_len); return netif->l2addr_len; #else /* GNRC_NETIF_L2ADDR_MAXLEN > 0 */ (void)netif; (void)l2src; return 0; #endif /* GNRC_NETIF_L2ADDR_MAXLEN > 0 */ } static gnrc_pktsnip_t *_check_release_pkt(gnrc_pktsnip_t *pkt, gnrc_pktsnip_t *payload) { if (pkt == NULL) { DEBUG("nib: No space left in packet buffer. Not replying NS"); gnrc_pktbuf_release(payload); } return pkt; } static void _send_delayed_nbr_adv(const gnrc_netif_t *netif, const ipv6_addr_t *tgt, const ipv6_hdr_t *ipv6_hdr, gnrc_pktsnip_t *payload) { gnrc_pktsnip_t *pkt; _nib_onl_entry_t *nce; uint8_t reply_flags = NDP_NBR_ADV_FLAGS_S; nce = _nib_onl_nc_get(tgt, netif->pid); if (nce == NULL) { /* usually this should be the case, but when NCE is full, just * ignore the sending. Other nodes in this anycast group are * then preferred */ gnrc_pktbuf_release(payload); return; } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ROUTER) if (gnrc_netif_is_rtr(netif)) { reply_flags |= NDP_NBR_ADV_FLAGS_R; } #endif /* CONFIG_GNRC_IPV6_NIB_ROUTER */ #if GNRC_NETIF_L2ADDR_MAXLEN > 0 if (ipv6_addr_is_multicast(&ipv6_hdr->dst)) { uint8_t l2addr[GNRC_NETIF_L2ADDR_MAXLEN]; size_t l2addr_len = _get_l2src(netif, l2addr); if (l2addr_len > 0) { pkt = gnrc_ndp_opt_tl2a_build(l2addr, l2addr_len, payload); if ((payload = _check_release_pkt(pkt, payload)) == NULL) { return; } } else { reply_flags |= NDP_NBR_ADV_FLAGS_O; } } else { reply_flags |= NDP_NBR_ADV_FLAGS_O; } #else /* GNRC_NETIF_L2ADDR_MAXLEN > 0 */ reply_flags |= NDP_NBR_ADV_FLAGS_O; #endif /* GNRC_NETIF_L2ADDR_MAXLEN > 0 */ /* discard const qualifier */ pkt = gnrc_ndp_nbr_adv_build(tgt, reply_flags, payload); if ((payload = _check_release_pkt(pkt, payload)) == NULL) { return; } pkt = gnrc_ipv6_hdr_build(payload, NULL, &ipv6_hdr->src); if ((payload = _check_release_pkt(pkt, payload)) == NULL) { return; } ((ipv6_hdr_t *)payload->data)->hl = NDP_HOP_LIMIT; pkt = gnrc_netif_hdr_build(NULL, 0, NULL, 0); if ((pkt = _check_release_pkt(pkt, payload)) == NULL) { return; } gnrc_netif_hdr_set_netif(pkt->data, netif); pkt = gnrc_pkt_prepend(payload, pkt); _evtimer_add(pkt, GNRC_IPV6_NIB_SND_NA, &nce->snd_na, random_uint32_range(0, NDP_MAX_ANYCAST_MS_DELAY)); } static void _handle_nbr_sol(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const ndp_nbr_sol_t *nbr_sol, size_t icmpv6_len) { size_t tmp_len = icmpv6_len - sizeof(ndp_nbr_sol_t); int tgt_idx; ndp_opt_t *opt; /* check validity, see: https://tools.ietf.org/html/rfc4861#section-7.1.1 */ /* checksum is checked by GNRC's ICMPv6 module */ if ((ipv6->hl != NDP_HOP_LIMIT) || (nbr_sol->code != 0U) || (icmpv6_len < sizeof(ndp_nbr_sol_t)) || ipv6_addr_is_multicast(&nbr_sol->tgt) || (ipv6_addr_is_unspecified(&ipv6->src) && !ipv6_addr_is_solicited_node(&ipv6->dst))) { DEBUG("nib: Received neighbor solicitation is invalid. Discarding silently\n"); DEBUG(" - IP Hop Limit: %u (should be %u)\n", ipv6->hl, NDP_HOP_LIMIT); DEBUG(" - ICMP code: %u (should be 0)\n", nbr_sol->code); DEBUG(" - ICMP length: %u (should > %u)\n", (unsigned)icmpv6_len, (unsigned)sizeof(ndp_nbr_sol_t)); DEBUG(" - Target address: %s (should not be multicast)\n", ipv6_addr_to_str(addr_str, &nbr_sol->tgt, sizeof(addr_str))); DEBUG(" - Source address: %s\n", ipv6_addr_to_str(addr_str, &ipv6->src, sizeof(addr_str))); DEBUG(" - Destination address: %s (should be of format " "ff02::1:ffxx:xxxx if source address is ::)\n", ipv6_addr_to_str(addr_str, &ipv6->dst, sizeof(addr_str))); return; } /* check if target is assigned only now in case the length was wrong */ tgt_idx = gnrc_netif_ipv6_addr_idx(netif, &nbr_sol->tgt); if (tgt_idx < 0) { DEBUG("nib: Target address %s is not assigned to the local interface\n", ipv6_addr_to_str(addr_str, &nbr_sol->tgt, sizeof(addr_str))); return; } /* pre-check option length */ FOREACH_OPT(nbr_sol, opt, tmp_len) { if (tmp_len > icmpv6_len) { DEBUG("nib: Payload length (%u) of NS doesn't align with options\n", (unsigned)icmpv6_len); return; } if (opt->len == 0U) { DEBUG("nib: Option of length 0 detected. " "Discarding neighbor solicitation silently\n"); return; } } DEBUG("nib: Received valid neighbor solicitation:\n"); DEBUG(" - Target address: %s\n", ipv6_addr_to_str(addr_str, &nbr_sol->tgt, sizeof(addr_str))); DEBUG(" - Source address: %s\n", ipv6_addr_to_str(addr_str, &ipv6->src, sizeof(addr_str))); DEBUG(" - Destination address: %s\n", ipv6_addr_to_str(addr_str, &ipv6->dst, sizeof(addr_str))); #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_SLAAC) gnrc_netif_t *tgt_netif = gnrc_netif_get_by_ipv6_addr(&nbr_sol->tgt); if (tgt_netif != NULL) { int idx; gnrc_netif_acquire(tgt_netif); idx = gnrc_netif_ipv6_addr_idx(tgt_netif, &nbr_sol->tgt); /* if idx < 0: * nbr_sol->tgt was removed between getting tgt_netif by nbr_sol->tgt * and gnrc_netif_acquire(tgt_netif). This is like `tgt_netif` would * have been NULL in the first place so just continue as if it would * have. */ if ((idx >= 0) && gnrc_netif_ipv6_addr_dad_trans(tgt_netif, idx)) { if (!ipv6_addr_is_unspecified(&ipv6->src)) { /* (see https://tools.ietf.org/html/rfc4862#section-5.4.3) */ DEBUG("nib: Neighbor is performing AR, but target address is " "still TENTATIVE for us => Ignoring NS\n"); gnrc_netif_release(tgt_netif); return; } /* cancel validation timer */ evtimer_del(&_nib_evtimer, &tgt_netif->ipv6.addrs_timers[idx].event); /* _remove_tentative_addr() context switches to `tgt_netif->pid` so * release `tgt_netif`. We are done here anyway. */ gnrc_netif_release(tgt_netif); _remove_tentative_addr(tgt_netif, &nbr_sol->tgt); return; } gnrc_netif_release(tgt_netif); } #endif /* CONFIG_GNRC_IPV6_NIB_SLAAC */ if (ipv6_addr_is_unspecified(&ipv6->src)) { gnrc_ndp_nbr_adv_send(&nbr_sol->tgt, netif, &ipv6->src, false, NULL); } else { #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LR) ndp_opt_t *sl2ao = NULL; sixlowpan_nd_opt_ar_t *aro = NULL; #else /* CONFIG_GNRC_IPV6_NIB_6LR */ #define sl2ao (NULL) #define aro (NULL) #endif /* CONFIG_GNRC_IPV6_NIB_6LR */ tmp_len = icmpv6_len - sizeof(ndp_nbr_sol_t); if (!(netif->flags & GNRC_NETIF_FLAGS_HAS_L2ADDR)) { /* Set STALE NCE if link-layer has no addresses */ _nib_nc_add(&ipv6->src, netif->pid, GNRC_IPV6_NIB_NC_INFO_NUD_STATE_STALE); } FOREACH_OPT(nbr_sol, opt, tmp_len) { switch (opt->type) { case NDP_OPT_SL2A: #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LR) if (gnrc_netif_is_6lr(netif)) { DEBUG("nib: Storing SL2AO for later handling\n"); sl2ao = opt; break; /* SL2AO is handled below together with an ARO */ } #endif /* CONFIG_GNRC_IPV6_NIB_6LR */ _handle_sl2ao(netif, ipv6, (const icmpv6_hdr_t *)nbr_sol, opt); break; #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LR) case NDP_OPT_AR: DEBUG("nib: Storing ARO for later handling\n"); aro = (sixlowpan_nd_opt_ar_t *)opt; break; #endif /* CONFIG_GNRC_IPV6_NIB_6LR */ default: DEBUG("nib: Ignoring unrecognized option type %u for NS\n", opt->type); break; } } gnrc_pktsnip_t *reply_aro = NULL; if (aro && sl2ao) { /* If no SLLAO is included, then any included ARO is ignored. */ if (!(reply_aro = _copy_and_handle_aro(netif, ipv6, nbr_sol, aro, sl2ao))) { /* If the Length field is not two, or if the Status field is not zero, then the NS is silently ignored.*/ return; } } else if (sl2ao) { _handle_sl2ao(netif, ipv6, (const icmpv6_hdr_t *)nbr_sol, sl2ao); } /* check if target address is anycast */ if (netif->ipv6.addrs_flags[tgt_idx] & GNRC_NETIF_IPV6_ADDRS_FLAGS_ANYCAST) { _send_delayed_nbr_adv(netif, &nbr_sol->tgt, ipv6, reply_aro); } else { gnrc_ndp_nbr_adv_send(&nbr_sol->tgt, netif, &ipv6->src, ipv6_addr_is_multicast(&ipv6->dst), reply_aro); } } } static void _handle_nbr_adv(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const ndp_nbr_adv_t *nbr_adv, size_t icmpv6_len) { size_t tmp_len = icmpv6_len - sizeof(ndp_nbr_adv_t); ndp_opt_t *opt; _nib_onl_entry_t *nce; /* check validity, see: https://tools.ietf.org/html/rfc4861#section-7.1.2 */ /* checksum is checked by GNRC's ICMPv6 module */ if ((ipv6->hl != NDP_HOP_LIMIT) || (nbr_adv->code != 0U) || (icmpv6_len < sizeof(ndp_nbr_adv_t)) || ipv6_addr_is_multicast(&nbr_adv->tgt) || (ipv6_addr_is_multicast(&ipv6->dst) && (nbr_adv->flags & NDP_NBR_ADV_FLAGS_S))) { DEBUG("nib: Received neighbor advertisement is invalid. Discarding silently\n"); DEBUG(" - IP Hop Limit: %u (should be %u)\n", ipv6->hl, NDP_HOP_LIMIT); DEBUG(" - ICMP code: %u (should be 0)\n", nbr_adv->code); DEBUG(" - ICMP length: %u (should > %u)\n", (unsigned)icmpv6_len, (unsigned)sizeof(ndp_nbr_adv_t)); DEBUG(" - Target address: %s (should not be multicast)\n", ipv6_addr_to_str(addr_str, &nbr_adv->tgt, sizeof(addr_str))); DEBUG(" - Destination address: %s\n", ipv6_addr_to_str(addr_str, &ipv6->dst, sizeof(addr_str))); DEBUG(" - Flags: %c%c%c (S must not be set if destination is multicast)\n", (nbr_adv->flags & NDP_NBR_ADV_FLAGS_R) ? 'R' : '-', (nbr_adv->flags & NDP_NBR_ADV_FLAGS_S) ? 'S' : '-', (nbr_adv->flags & NDP_NBR_ADV_FLAGS_O) ? 'O' : '-'); return; } /* pre-check option length */ FOREACH_OPT(nbr_adv, opt, tmp_len) { if (tmp_len > icmpv6_len) { DEBUG("nib: Payload length (%u) of NA doesn't align with options\n", (unsigned)icmpv6_len); return; } if (opt->len == 0U) { DEBUG("nib: Option of length 0 detected. " "Discarding neighbor advertisement silently\n"); return; } } DEBUG("nib: Received valid neighbor advertisement:\n"); DEBUG(" - Target address: %s\n", ipv6_addr_to_str(addr_str, &nbr_adv->tgt, sizeof(addr_str))); DEBUG(" - Source address: %s\n", ipv6_addr_to_str(addr_str, &ipv6->src, sizeof(addr_str))); DEBUG(" - Destination address: %s\n", ipv6_addr_to_str(addr_str, &ipv6->dst, sizeof(addr_str))); DEBUG(" - Flags: %c%c%c\n", (nbr_adv->flags & NDP_NBR_ADV_FLAGS_R) ? 'R' : '-', (nbr_adv->flags & NDP_NBR_ADV_FLAGS_S) ? 'S' : '-', (nbr_adv->flags & NDP_NBR_ADV_FLAGS_O) ? 'O' : '-'); #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_SLAAC) gnrc_netif_t *tgt_netif = gnrc_netif_get_by_ipv6_addr(&nbr_adv->tgt); if (tgt_netif != NULL) { int idx; gnrc_netif_acquire(tgt_netif); idx = gnrc_netif_ipv6_addr_idx(tgt_netif, &nbr_adv->tgt); /* if idx < 0: * nbr_sol->tgt was removed between getting tgt_netif by nbr_sol->tgt * and gnrc_netif_acquire(tgt_netif). This is like `tgt_netif` would * have been NULL in the first place so just continue as if it would * have. */ if ((idx >= 0) && gnrc_netif_ipv6_addr_dad_trans(tgt_netif, idx)) { DEBUG("nib: duplicate address detected, removing target address " "from this interface\n"); /* cancel validation timer */ evtimer_del(&_nib_evtimer, &tgt_netif->ipv6.addrs_timers[idx].event); /* _remove_tentative_addr() context switches to `tgt_netif->pid` so * release `tgt_netif`. We are done here anyway. */ gnrc_netif_release(tgt_netif); _remove_tentative_addr(tgt_netif, &nbr_adv->tgt); return; } /* else case beyond scope of RFC4862: * https://tools.ietf.org/html/rfc4862#section-5.4.4 */ gnrc_netif_release(tgt_netif); } #endif /* CONFIG_GNRC_IPV6_NIB_SLAAC */ if ((nce = _nib_onl_nc_get(&nbr_adv->tgt, netif->pid)) != NULL) { #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ARSM) bool tl2ao_avail = false; #endif /* CONFIG_GNRC_IPV6_NIB_ARSM */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LN) uint8_t aro_status = _ADDR_REG_STATUS_UNAVAIL; #endif tmp_len = icmpv6_len - sizeof(ndp_nbr_adv_t); FOREACH_OPT(nbr_adv, opt, tmp_len) { switch (opt->type) { #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ARSM) case NDP_OPT_TL2A: _handle_adv_l2(netif, nce, (icmpv6_hdr_t *)nbr_adv, opt); tl2ao_avail = true; break; #endif /* CONFIG_GNRC_IPV6_NIB_ARSM */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LN) case NDP_OPT_AR: aro_status = _handle_aro(netif, ipv6, (const icmpv6_hdr_t *)nbr_adv, (const sixlowpan_nd_opt_ar_t *)opt, opt, nce); break; #endif /* CONFIG_GNRC_IPV6_NIB_6LN */ default: DEBUG("nib: Ignoring unrecognized option type %u for NA\n", opt->type); } } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ARSM) if (!tl2ao_avail && (nbr_adv->flags & NDP_NBR_ADV_FLAGS_S) && (_get_nud_state(nce) != GNRC_IPV6_NIB_NC_INFO_NUD_STATE_INCOMPLETE)) { /* reachability confirmed without TL2AO */ _set_reachable(netif, nce); } if (!(netif->flags & GNRC_NETIF_FLAGS_HAS_L2ADDR)) { _handle_adv_l2(netif, nce, (icmpv6_hdr_t *)nbr_adv, NULL); } #endif /* CONFIG_GNRC_IPV6_NIB_ARSM */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_SLAAC) && IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LN) /* 6Lo-ND duplicate address detection (DAD) was ignored by neighbor, try * traditional DAD */ if ((aro_status == _ADDR_REG_STATUS_UNAVAIL) && gnrc_netif_is_6ln(netif)) { DEBUG("nib: No ARO in NA, falling back to classic DAD\n"); _handle_dad(&ipv6->dst); } #elif IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_6LN) (void)aro_status; #endif } } static gnrc_pktqueue_t *_alloc_queue_entry(gnrc_pktsnip_t *pkt) { #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_QUEUE_PKT) for (int i = 0; i < CONFIG_GNRC_IPV6_NIB_NUMOF; i++) { if (_queue_pool[i].pkt == NULL) { _queue_pool[i].pkt = pkt; return &_queue_pool[i]; } } #else (void)pkt; #endif /* CONFIG_GNRC_IPV6_NIB_QUEUE_PKT */ return NULL; } static bool _resolve_addr_from_nc(_nib_onl_entry_t *entry, gnrc_netif_t *netif, gnrc_ipv6_nib_nc_t *nce) { if (entry == NULL) { return false; } if (IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ARSM)) { if (!_is_reachable(entry)) { return false; } if (_get_nud_state(entry) == GNRC_IPV6_NIB_NC_INFO_NUD_STATE_STALE) { _set_nud_state(netif, entry, GNRC_IPV6_NIB_NC_INFO_NUD_STATE_DELAY); _evtimer_add(entry, GNRC_IPV6_NIB_DELAY_TIMEOUT, &entry->nud_timeout, NDP_DELAY_FIRST_PROBE_MS); } } _nib_nc_get(entry, nce); return true; } static bool _enqueue_for_resolve(gnrc_netif_t *netif, gnrc_pktsnip_t *pkt, _nib_onl_entry_t *entry) { if (!IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_QUEUE_PKT) || _get_nud_state(entry) != GNRC_IPV6_NIB_NC_INFO_NUD_STATE_INCOMPLETE) { gnrc_icmpv6_error_dst_unr_send(ICMPV6_ERROR_DST_UNR_ADDR, pkt); gnrc_pktbuf_release_error(pkt, EHOSTUNREACH); return true; } gnrc_pktqueue_t *queue_entry = _alloc_queue_entry(pkt); if (queue_entry == NULL) { DEBUG("nib: can't allocate entry for packet queue " "dropping packet\n"); gnrc_pktbuf_release(pkt); return false; } if (netif != NULL) { gnrc_pktsnip_t *netif_hdr = gnrc_netif_hdr_build(NULL, 0, NULL, 0); if (netif_hdr == NULL) { DEBUG("nib: can't allocate netif header for queue\n"); gnrc_pktbuf_release(pkt); queue_entry->pkt = NULL; return false; } gnrc_netif_hdr_set_netif(netif_hdr->data, netif); queue_entry->pkt = gnrc_pkt_prepend(queue_entry->pkt, netif_hdr); } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_QUEUE_PKT) gnrc_pktqueue_add(&entry->pktqueue, queue_entry); #else (void)entry; #endif return true; } static bool _resolve_addr(const ipv6_addr_t *dst, gnrc_netif_t *netif, gnrc_pktsnip_t *pkt, gnrc_ipv6_nib_nc_t *nce, _nib_onl_entry_t *entry) { if ((netif != NULL) && (netif->device_type == NETDEV_TYPE_SLIP)) { /* XXX: Linux doesn't do neighbor discovery for SLIP so no use sending * NS and since SLIP doesn't have link-layer addresses anyway, we can * just return the interface without any link-layer addresses */ memcpy(&nce->ipv6, dst, sizeof(nce->ipv6)); nce->info = (netif->pid << _NIB_IF_POS); nce->l2addr_len = 0; return true; } /* first check if address is cached */ if (_resolve_addr_from_nc(entry, netif, nce)) { DEBUG("nib: resolve address %s%%%u from neighbor cache\n", ipv6_addr_to_str(addr_str, &entry->ipv6, sizeof(addr_str)), _nib_onl_get_if(entry)); return true; } /* directly resolve address if it uses 6lo addressing mode */ if (_resolve_addr_from_ipv6(dst, netif, nce)) { DEBUG("nib: resolve l2 address from IPv6 address\n"); return true; } bool reset = false; DEBUG("nib: resolve address %s by probing neighbors\n", ipv6_addr_to_str(addr_str, dst, sizeof(addr_str))); if (entry == NULL) { entry = _nib_nc_add(dst, netif ? netif->pid : 0, GNRC_IPV6_NIB_NC_INFO_NUD_STATE_INCOMPLETE); if (entry == NULL) { DEBUG("nib: can't resolve address, neighbor cache full\n"); gnrc_pktbuf_release(pkt); return false; } if (IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ROUTER) && netif != NULL) { _call_route_info_cb(netif, GNRC_IPV6_NIB_ROUTE_INFO_TYPE_NSC, dst, (void *)GNRC_IPV6_NIB_NC_INFO_NUD_STATE_INCOMPLETE); } reset = true; } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_ARSM) else if (_get_nud_state(entry) == GNRC_IPV6_NIB_NC_INFO_NUD_STATE_UNREACHABLE) { /* reduce back-off to possibly resolve neighbor sooner again */ entry->ns_sent = 3; } #endif /* queue packet as we have to do address resolution first */ if (pkt != NULL && !_enqueue_for_resolve(netif, pkt, entry)) { return false; } /* don't do multicast address resolution on 6lo */ if (!gnrc_netif_is_6ln(netif)) { _probe_nbr(entry, reset); } return false; } static void _handle_snd_na(gnrc_pktsnip_t *pkt) { #ifdef MODULE_GNRC_IPV6 DEBUG("nib: Send delayed neighbor advertisement\n"); if (!gnrc_netapi_dispatch_send(GNRC_NETTYPE_IPV6, GNRC_NETREG_DEMUX_CTX_ALL, pkt)) { DEBUG("nib: No receivers for neighbor advertisement\n"); gnrc_pktbuf_release_error(pkt, EBADF); } #else /* MODULE_GNRC_IPV6 */ (void)pkt; DEBUG("nib: No IPv6 module to send delayed neighbor advertisement\n"); #endif /* MODULE_GNRC_IPV6 */ } static void _handle_pfx_timeout(_nib_offl_entry_t *pfx) { gnrc_netif_t *netif = gnrc_netif_get_by_pid(_nib_onl_get_if(pfx->next_hop)); if (netif == NULL) { return; } uint32_t now = evtimer_now_msec(); gnrc_netif_acquire(netif); if (now >= pfx->valid_until) { _nib_offl_remove_prefix(pfx); } else if (now >= pfx->pref_until) { for (int i = 0; i < CONFIG_GNRC_NETIF_IPV6_ADDRS_NUMOF; i++) { if (ipv6_addr_match_prefix(&netif->ipv6.addrs[i], &pfx->pfx) >= pfx->pfx_len) { netif->ipv6.addrs_flags[i] &= ~GNRC_NETIF_IPV6_ADDRS_FLAGS_STATE_MASK; netif->ipv6.addrs_flags[i] |= GNRC_NETIF_IPV6_ADDRS_FLAGS_STATE_DEPRECATED; } } _evtimer_add(pfx, GNRC_IPV6_NIB_PFX_TIMEOUT, &pfx->pfx_timeout, pfx->valid_until - now); } gnrc_netif_release(netif); } static void _handle_rtr_timeout(_nib_dr_entry_t *router) { if ((router->next_hop != NULL) && (router->next_hop->mode & _DRL)) { _nib_offl_entry_t *route = NULL; unsigned iface = _nib_onl_get_if(router->next_hop); ipv6_addr_t addr = router->next_hop->ipv6; _nib_drl_remove(router); /* also remove all routes to that router */ while ((route = _nib_offl_iter(route))) { if ((route->next_hop != NULL) && (_nib_onl_get_if(route->next_hop) == iface) && (ipv6_addr_equal(&route->next_hop->ipv6, &addr))) { route->mode = _EMPTY; route->next_hop->mode &= ~_DST; _nib_offl_clear(route); /* XXX routing protocol gets informed in case NUD * determines ipv6->src (still in neighbor cache) to be * unreachable */ } } } } void _handle_search_rtr(gnrc_netif_t *netif) { #if !IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_NO_RTR_SOL) gnrc_netif_acquire(netif); if (_should_search_rtr(netif)) { uint32_t next_rs = _evtimer_lookup(netif, GNRC_IPV6_NIB_SEARCH_RTR); uint32_t interval = _get_next_rs_interval(netif); if (next_rs > interval) { gnrc_ndp_rtr_sol_send(netif, &ipv6_addr_all_routers_link_local); if (netif->ipv6.rs_sent < 10U) { /* with more the backoff (required in RFC 6775) is truncated * anyway and this way we prevent overflows. 10 is arbitrary, so * we do not need a define here */ netif->ipv6.rs_sent++; } if ((netif->ipv6.rs_sent < NDP_MAX_RS_NUMOF) || gnrc_netif_is_6ln(netif)) { /* 6LN will solicitate indefinitely */ _evtimer_add(netif, GNRC_IPV6_NIB_SEARCH_RTR, &netif->ipv6.search_rtr, interval); } } } gnrc_netif_release(netif); #else (void)netif; #endif /* !CONFIG_GNRC_IPV6_NIB_NO_RTR_SOL */ } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_DNS) static void _handle_rdnss_timeout(sock_udp_ep_t *dns_server) { memset(dns_server, 0, sizeof(sock_udp_ep_t)); } #endif static void _handle_mtuo(gnrc_netif_t *netif, const icmpv6_hdr_t *icmpv6, const ndp_opt_mtu_t *mtuo) { if ((mtuo->len != NDP_OPT_MTU_LEN) || (icmpv6->type != ICMPV6_RTR_ADV)) { return; } if (byteorder_ntohl(mtuo->mtu) >= IPV6_MIN_MTU) { netif->ipv6.mtu = byteorder_ntohl(mtuo->mtu); } } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_DNS) static uint32_t _handle_rdnsso(gnrc_netif_t *netif, const icmpv6_hdr_t *icmpv6, const ndp_opt_rdnss_impl_t *rdnsso) { uint32_t ltime = UINT32_MAX; const ipv6_addr_t *addr; if ((rdnsso->len < NDP_OPT_RDNSS_MIN_LEN) || (icmpv6->type != ICMPV6_RTR_ADV)) { return ltime; } /* select first if unassigned, search possible address otherwise */ addr = (sock_dns_server.port == 0) ? &rdnsso->addrs[0] : NULL; if (addr == NULL) { unsigned addrs_num = (rdnsso->len - 1) / 2; for (unsigned i = 0; i < addrs_num; i++) { if (memcmp(sock_dns_server.addr.ipv6, &rdnsso->addrs[i], sizeof(rdnsso->addrs[i])) == 0) { addr = &rdnsso->addrs[i]; break; } } } #if SOCK_HAS_IPV6 ltime = byteorder_ntohl(rdnsso->ltime); if (addr != NULL) { if (ltime > 0) { sock_dns_server.port = SOCK_DNS_PORT; sock_dns_server.family = AF_INET6; sock_dns_server.netif = netif->pid; memcpy(sock_dns_server.addr.ipv6, rdnsso->addrs, sizeof(sock_dns_server.addr.ipv6)); if (ltime < UINT32_MAX) { /* the valid lifetime is given in seconds, but our timers work * in milliseconds, so we have to scale down to the smallest * possible value (UINT32_MAX - 1). This is however alright * since we ask for a new router advertisement before this * timeout expires */ ltime = (ltime > (UINT32_MAX / MS_PER_SEC)) ? (UINT32_MAX - 1) : ltime * MS_PER_SEC; _evtimer_add(&sock_dns_server, GNRC_IPV6_NIB_RDNSS_TIMEOUT, &_rdnss_timeout, ltime); } } else { evtimer_del(&_nib_evtimer, &_rdnss_timeout.event); _handle_rdnss_timeout(&sock_dns_server); } } #else (void)addr; #endif return ltime; } #endif static void _remove_prefix(const ipv6_addr_t *pfx, unsigned pfx_len) { _nib_offl_entry_t *offl = NULL; while ((offl = _nib_offl_iter(offl))) { if ((offl->mode & _PL) && (offl->pfx_len == pfx_len) && (ipv6_addr_match_prefix(&offl->pfx, pfx) >= pfx_len)) { _nib_offl_remove_prefix(offl); } } } #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C) static inline bool _multihop_p6c(gnrc_netif_t *netif, _nib_abr_entry_t *abr) { (void)netif; /* gnrc_netif_is_6lr() might resolve to a NOP */ return (gnrc_netif_is_6lr(netif) && (abr != NULL)); } #else /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ #define _multihop_p6c(netif, abr) (false) #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C) static uint32_t _handle_pio(gnrc_netif_t *netif, const icmpv6_hdr_t *icmpv6, const ndp_opt_pi_t *pio, _nib_abr_entry_t *abr) #else /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ static uint32_t _handle_pio(gnrc_netif_t *netif, const icmpv6_hdr_t *icmpv6, const ndp_opt_pi_t *pio) #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ { uint32_t valid_ltime; uint32_t pref_ltime; valid_ltime = byteorder_ntohl(pio->valid_ltime); pref_ltime = byteorder_ntohl(pio->pref_ltime); if ((pio->len != NDP_OPT_PI_LEN) || (icmpv6->type != ICMPV6_RTR_ADV) || ipv6_addr_is_link_local(&pio->prefix) || (valid_ltime < pref_ltime)) { DEBUG("nib: ignoring PIO with invalid data\n"); return UINT32_MAX; } DEBUG("nib: received valid Prefix Information option:\n"); DEBUG(" - Prefix: %s/%u\n", ipv6_addr_to_str(addr_str, &pio->prefix, sizeof(addr_str)), pio->prefix_len); DEBUG(" - Flags: %c%c\n", (pio->flags & NDP_OPT_PI_FLAGS_L) ? 'L' : '-', (pio->flags & NDP_OPT_PI_FLAGS_A) ? 'A' : '-'); DEBUG(" - Valid lifetime: %" PRIu32 "\n", byteorder_ntohl(pio->valid_ltime)); DEBUG(" - Preferred lifetime: %" PRIu32 "\n", byteorder_ntohl(pio->pref_ltime)); if (pio->flags & NDP_OPT_PI_FLAGS_A) { _auto_configure_addr(netif, &pio->prefix, pio->prefix_len); } if ((pio->flags & NDP_OPT_PI_FLAGS_L) || _multihop_p6c(netif, abr)) { _nib_offl_entry_t *pfx; if (pio->valid_ltime.u32 == 0) { DEBUG("nib: PIO for %s/%u with lifetime 0. Removing prefix.\n", ipv6_addr_to_str(addr_str, &pio->prefix, sizeof(addr_str)), pio->prefix_len); _remove_prefix(&pio->prefix, pio->prefix_len); return UINT32_MAX; } if (valid_ltime < UINT32_MAX) { /* UINT32_MAX means infinite lifetime */ /* the valid lifetime is given in seconds, but our timers work in * microseconds, so we have to scale down to the smallest possible * value (UINT32_MAX - 1). This is however alright since we ask for * a new router advertisement before this timeout expires */ valid_ltime = (valid_ltime > (UINT32_MAX / MS_PER_SEC)) ? (UINT32_MAX - 1) : valid_ltime * MS_PER_SEC; } if (pref_ltime < UINT32_MAX) { /* UINT32_MAX means infinite lifetime */ /* same treatment for pref_ltime */ pref_ltime = (pref_ltime > (UINT32_MAX / MS_PER_SEC)) ? (UINT32_MAX - 1) : pref_ltime * MS_PER_SEC; } if ((pfx = _nib_pl_add(netif->pid, &pio->prefix, pio->prefix_len, valid_ltime, pref_ltime))) { #if IS_ACTIVE(CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C) if (abr != NULL) { _nib_abr_add_pfx(abr, pfx); } #endif /* CONFIG_GNRC_IPV6_NIB_MULTIHOP_P6C */ if (pio->flags & NDP_OPT_PI_FLAGS_L) { pfx->flags |= _PFX_ON_LINK; } if (pio->flags & NDP_OPT_PI_FLAGS_A) { pfx->flags |= _PFX_SLAAC; } return _min(pref_ltime, valid_ltime); } } return UINT32_MAX; } static const char *_prio_string(uint8_t prio) { switch (prio & NDP_OPT_RI_FLAGS_MASK) { case NDP_OPT_RI_FLAGS_PRF_NONE: return "none"; case NDP_OPT_RI_FLAGS_PRF_NEG: return "-1"; case NDP_OPT_RI_FLAGS_PRF_ZERO: return "0"; case NDP_OPT_RI_FLAGS_PRF_POS: return "1"; } return "invalid"; } static uint32_t _handle_rio(gnrc_netif_t *netif, const ipv6_hdr_t *ipv6, const ndp_opt_ri_t *rio) { if (!IS_USED(MODULE_GNRC_IPV6_NIB_RIO)) { return 0; } uint32_t route_ltime = byteorder_ntohl(rio->route_ltime); if (ipv6_addr_is_link_local(&rio->prefix)) { DEBUG("nib: ignoring RIO with invalid data\n"); return UINT32_MAX; } DEBUG("nib: received valid Route Information option:\n"); DEBUG(" - Prefix: %s/%u\n", ipv6_addr_to_str(addr_str, &rio->prefix, sizeof(addr_str)), rio->prefix_len); DEBUG(" - Priority: %s\n", _prio_string(rio->flags)); DEBUG(" - Route lifetime: %" PRIu32 "\n", byteorder_ntohl(rio->route_ltime)); if (route_ltime < UINT32_MAX) { /* UINT32_MAX means infinite lifetime */ /* the valid lifetime is given in seconds, but our timers work in * microseconds, so we have to scale down to the smallest possible * value (UINT32_MAX - 1). This is however alright since we ask for * a new router advertisement before this timeout expires */ route_ltime = (route_ltime > (UINT32_MAX / MS_PER_SEC)) ? (UINT32_MAX - 1) : route_ltime * MS_PER_SEC; } if (route_ltime == 0) { gnrc_ipv6_nib_ft_del(&rio->prefix, rio->prefix_len); } else { gnrc_ipv6_nib_ft_add(&rio->prefix, rio->prefix_len, &ipv6->src, netif->pid, route_ltime); } return route_ltime; } /** @} */