2015-05-22 21:47:37 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2013 - 2014 INRIA.
|
|
|
|
* Copyright (C) 2015 Cenk Gündoğan <cnkgndgn@gmail.com>
|
|
|
|
*
|
|
|
|
* 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 Cenk Gündoğan <cnkgndgn@gmail.com>
|
|
|
|
*/
|
|
|
|
|
2015-08-18 22:50:51 +02:00
|
|
|
#include "net/af.h"
|
2015-08-10 13:22:24 +02:00
|
|
|
#include "net/icmpv6.h"
|
2015-08-17 15:41:29 +02:00
|
|
|
#include "net/ipv6/hdr.h"
|
|
|
|
#include "net/gnrc/icmpv6.h"
|
|
|
|
#include "net/gnrc/ipv6.h"
|
|
|
|
#include "net/gnrc.h"
|
2015-07-23 13:23:42 +02:00
|
|
|
#include "net/eui64.h"
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
#include "net/gnrc/rpl.h"
|
|
|
|
|
2015-05-22 21:47:37 +02:00
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2015-08-19 22:08:07 +02:00
|
|
|
#if ENABLE_DEBUG
|
2015-05-22 21:47:37 +02:00
|
|
|
static char addr_str[IPV6_ADDR_MAX_STR_LEN];
|
|
|
|
#endif
|
|
|
|
|
2015-08-18 21:13:53 +02:00
|
|
|
#define GNRC_RPL_GROUNDED_SHIFT (7)
|
|
|
|
#define GNRC_RPL_MOP_SHIFT (3)
|
|
|
|
#define GNRC_RPL_OPT_DODAG_CONF_LEN (14)
|
|
|
|
#define GNRC_RPL_OPT_PREFIX_INFO_LEN (30)
|
2015-08-19 20:57:28 +02:00
|
|
|
#define GNRC_RPL_OPT_TARGET_LEN (18)
|
|
|
|
#define GNRC_RPL_OPT_TRANSIT_INFO_LEN (4)
|
2015-08-18 21:13:53 +02:00
|
|
|
#define GNRC_RPL_SHIFTED_MOP_MASK (0x7)
|
|
|
|
#define GNRC_RPL_PRF_MASK (0x7)
|
|
|
|
#define GNRC_RPL_PREFIX_AUTO_ADDRESS_BIT (1 << 6)
|
|
|
|
#define GNRC_RPL_DAO_D_BIT (1 << 6)
|
|
|
|
#define GNRC_RPL_DAO_K_BIT (1 << 7)
|
|
|
|
#define GNRC_RPL_DAO_ACK_D_BIT (1 << 7)
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-23 18:33:51 +02:00
|
|
|
void gnrc_rpl_send(gnrc_pktsnip_t *pkt, ipv6_addr_t *src, ipv6_addr_t *dst, ipv6_addr_t *dodag_id)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
2015-09-22 16:26:19 +02:00
|
|
|
(void) dodag_id;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_pktsnip_t *hdr;
|
|
|
|
ipv6_addr_t all_RPL_nodes = GNRC_RPL_ALL_NODES_ADDR, ll_addr;
|
|
|
|
kernel_pid_t iface = gnrc_ipv6_netif_find_by_addr(NULL, &all_RPL_nodes);
|
2015-05-22 21:47:37 +02:00
|
|
|
if (iface == KERNEL_PID_UNDEF) {
|
|
|
|
DEBUG("RPL: no suitable interface found for this destination address\n");
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_pktbuf_release(pkt);
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (src == NULL) {
|
2015-09-22 16:26:19 +02:00
|
|
|
ipv6_addr_set_link_local_prefix(&ll_addr);
|
|
|
|
src = gnrc_ipv6_netif_match_prefix(iface, &ll_addr);
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-09-22 16:26:19 +02:00
|
|
|
if (src == NULL) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: no suitable src address found\n");
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_pktbuf_release(pkt);
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dst == NULL) {
|
|
|
|
dst = &all_RPL_nodes;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
hdr = gnrc_ipv6_hdr_build(pkt, (uint8_t *)src, sizeof(ipv6_addr_t), (uint8_t *)dst,
|
2015-05-22 21:47:37 +02:00
|
|
|
sizeof(ipv6_addr_t));
|
|
|
|
|
|
|
|
if (hdr == NULL) {
|
|
|
|
DEBUG("RPL: Send - no space left in packet buffer\n");
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_pktbuf_release(pkt);
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if (!gnrc_netapi_dispatch_send(GNRC_NETTYPE_IPV6, GNRC_NETREG_DEMUX_CTX_ALL,hdr)) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: cannot send packet: no subscribers found.\n");
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_pktbuf_release(hdr);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
void gnrc_rpl_send_DIO(gnrc_rpl_dodag_t *dodag, ipv6_addr_t *destination)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
|
|
|
if (dodag == NULL) {
|
|
|
|
DEBUG("RPL: Error - trying to send DIO without being part of a dodag.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_pktsnip_t *pkt;
|
2015-08-10 13:22:24 +02:00
|
|
|
icmpv6_hdr_t *icmp;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_dio_t *dio;
|
2015-05-22 21:47:37 +02:00
|
|
|
uint8_t *pos;
|
2015-08-17 15:41:29 +02:00
|
|
|
int size = sizeof(icmpv6_hdr_t) + sizeof(gnrc_rpl_dio_t);
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-25 11:03:28 +02:00
|
|
|
if (dodag->dodag_conf_requested) {
|
2015-08-17 15:41:29 +02:00
|
|
|
size += sizeof(gnrc_rpl_opt_dodag_conf_t);
|
2015-08-25 11:03:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (dodag->prefix_info_requested) {
|
2015-08-17 15:41:29 +02:00
|
|
|
size += sizeof(gnrc_rpl_opt_prefix_info_t);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if ((pkt = gnrc_icmpv6_build(NULL, ICMPV6_RPL_CTRL, GNRC_RPL_ICMPV6_CODE_DIO, size)) == NULL) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: Send DIO - no space left in packet buffer\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-10 13:22:24 +02:00
|
|
|
icmp = (icmpv6_hdr_t *)pkt->data;
|
2015-08-17 15:41:29 +02:00
|
|
|
dio = (gnrc_rpl_dio_t *)(icmp + 1);
|
2015-05-22 21:47:37 +02:00
|
|
|
pos = (uint8_t *) dio;
|
|
|
|
dio->instance_id = dodag->instance->id;
|
|
|
|
dio->version_number = dodag->version;
|
|
|
|
dio->rank = byteorder_htons(dodag->my_rank);
|
2015-08-17 15:41:29 +02:00
|
|
|
dio->g_mop_prf = (dodag->grounded << GNRC_RPL_GROUNDED_SHIFT) |
|
|
|
|
(dodag->instance->mop << GNRC_RPL_MOP_SHIFT) | dodag->prf;
|
2015-05-22 21:47:37 +02:00
|
|
|
dio->dtsn = dodag->dtsn;
|
|
|
|
dio->flags = 0;
|
|
|
|
dio->reserved = 0;
|
|
|
|
dio->dodag_id = dodag->dodag_id;
|
|
|
|
|
|
|
|
pos += sizeof(*dio);
|
|
|
|
|
2015-08-25 11:03:28 +02:00
|
|
|
if (dodag->dodag_conf_requested) {
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_opt_dodag_conf_t *dodag_conf;
|
|
|
|
dodag_conf = (gnrc_rpl_opt_dodag_conf_t *) pos;
|
|
|
|
dodag_conf->type = GNRC_RPL_OPT_DODAG_CONF;
|
|
|
|
dodag_conf->length = GNRC_RPL_OPT_DODAG_CONF_LEN;
|
2015-05-22 21:47:37 +02:00
|
|
|
dodag_conf->flags_a_pcs = 0;
|
|
|
|
dodag_conf->dio_int_doubl = dodag->dio_interval_doubl;
|
|
|
|
dodag_conf->dio_int_min = dodag->dio_min;
|
|
|
|
dodag_conf->dio_redun = dodag->dio_redun;
|
|
|
|
dodag_conf->max_rank_inc = byteorder_htons(dodag->instance->max_rank_inc);
|
|
|
|
dodag_conf->min_hop_rank_inc = byteorder_htons(dodag->instance->min_hop_rank_inc);
|
|
|
|
dodag_conf->ocp = byteorder_htons(dodag->instance->of->ocp);
|
|
|
|
dodag_conf->reserved = 0;
|
|
|
|
dodag_conf->default_lifetime = dodag->default_lifetime;
|
|
|
|
dodag_conf->lifetime_unit = byteorder_htons(dodag->lifetime_unit);
|
2015-07-23 13:23:42 +02:00
|
|
|
pos += sizeof(*dodag_conf);
|
2015-08-25 11:03:28 +02:00
|
|
|
dodag->dodag_conf_requested = false;
|
|
|
|
}
|
2015-07-23 13:23:42 +02:00
|
|
|
|
2015-08-25 11:03:28 +02:00
|
|
|
if (dodag->prefix_info_requested) {
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_opt_prefix_info_t *prefix_info;
|
|
|
|
prefix_info = (gnrc_rpl_opt_prefix_info_t *) pos;
|
|
|
|
prefix_info->type = GNRC_RPL_OPT_PREFIX_INFO;
|
|
|
|
prefix_info->length = GNRC_RPL_OPT_PREFIX_INFO_LEN;
|
2015-07-23 13:23:42 +02:00
|
|
|
/* auto-address configuration */
|
2015-08-17 15:41:29 +02:00
|
|
|
prefix_info->LAR_flags = GNRC_RPL_PREFIX_AUTO_ADDRESS_BIT;
|
2015-07-23 13:23:42 +02:00
|
|
|
prefix_info->valid_lifetime = dodag->addr_valid;
|
|
|
|
prefix_info->pref_lifetime = dodag->addr_preferred;
|
|
|
|
prefix_info->prefix_len = dodag->prefix_len;
|
|
|
|
prefix_info->reserved = 0;
|
|
|
|
|
|
|
|
memset(&prefix_info->prefix, 0, sizeof(prefix_info->prefix));
|
|
|
|
ipv6_addr_init_prefix(&prefix_info->prefix, &dodag->dodag_id, dodag->prefix_len);
|
2015-08-25 11:03:28 +02:00
|
|
|
dodag->prefix_info_requested = false;
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
2015-08-23 18:33:51 +02:00
|
|
|
gnrc_rpl_send(pkt, NULL, destination, &dodag->dodag_id);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
void gnrc_rpl_send_DIS(gnrc_rpl_dodag_t *dodag, ipv6_addr_t *destination)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_pktsnip_t *pkt;
|
2015-08-10 13:22:24 +02:00
|
|
|
icmpv6_hdr_t *icmp;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_dis_t *dis;
|
2015-09-08 09:26:48 +02:00
|
|
|
|
2015-05-22 21:47:37 +02:00
|
|
|
/* TODO: Currently the DIS is too small so that wireshark complains about an incorrect
|
|
|
|
* ethernet frame check sequence. In order to prevent this, 4 PAD1 options are added.
|
|
|
|
* This will be addressed in follow-up PRs */
|
2015-09-08 09:26:48 +02:00
|
|
|
uint8_t padding[] = {
|
|
|
|
0x01, 0x02, 0x00, 0x00
|
|
|
|
};
|
|
|
|
|
|
|
|
int size = sizeof(icmpv6_hdr_t) + sizeof(gnrc_rpl_dis_t) + sizeof(padding);
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if ((pkt = gnrc_icmpv6_build(NULL, ICMPV6_RPL_CTRL, GNRC_RPL_ICMPV6_CODE_DIS, size)) == NULL) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: Send DIS - no space left in packet buffer\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-10 13:22:24 +02:00
|
|
|
icmp = (icmpv6_hdr_t *)pkt->data;
|
2015-08-17 15:41:29 +02:00
|
|
|
dis = (gnrc_rpl_dis_t *)(icmp + 1);
|
2015-05-22 21:47:37 +02:00
|
|
|
dis->flags = 0;
|
|
|
|
dis->reserved = 0;
|
2015-09-08 09:26:48 +02:00
|
|
|
|
|
|
|
/* TODO add padding may be removed if packet size grows */
|
|
|
|
memcpy((dis + 1), padding, sizeof(padding));
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-23 18:33:51 +02:00
|
|
|
gnrc_rpl_send(pkt, NULL, destination, (dodag ? &dodag->dodag_id : NULL));
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
2015-08-19 19:43:06 +02:00
|
|
|
static bool _gnrc_rpl_check_DIS_validity(gnrc_rpl_dis_t *dis, uint16_t len)
|
|
|
|
{
|
|
|
|
uint16_t expected_len = sizeof(*dis) + sizeof(icmpv6_hdr_t);
|
|
|
|
|
|
|
|
if (expected_len <= len) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG("RPL: wrong DIS len: %d, expected: %d\n", len, expected_len);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
void gnrc_rpl_recv_DIS(gnrc_rpl_dis_t *dis, ipv6_addr_t *src, ipv6_addr_t *dst, uint16_t len)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
|
|
|
/* TODO handle Solicited Information Option */
|
|
|
|
(void) dis;
|
|
|
|
(void) len;
|
|
|
|
|
2015-08-19 19:43:06 +02:00
|
|
|
if (!_gnrc_rpl_check_DIS_validity(dis, len)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-05-22 21:47:37 +02:00
|
|
|
if (ipv6_addr_is_multicast(dst)) {
|
2015-08-17 15:41:29 +02:00
|
|
|
for (uint8_t i = 0; i < GNRC_RPL_DODAGS_NUMOF; ++i) {
|
|
|
|
if (gnrc_rpl_dodags[i].state != 0) {
|
|
|
|
trickle_reset_timer(&gnrc_rpl_dodags[i].trickle);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2015-08-17 15:41:29 +02:00
|
|
|
for (uint8_t i = 0; i < GNRC_RPL_DODAGS_NUMOF; ++i) {
|
|
|
|
if (gnrc_rpl_dodags[i].state != 0) {
|
2015-08-25 11:03:28 +02:00
|
|
|
gnrc_rpl_dodags[i].dodag_conf_requested = true;
|
|
|
|
gnrc_rpl_dodags[i].prefix_info_requested = true;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_send_DIO(&gnrc_rpl_dodags[i], src);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-19 20:57:28 +02:00
|
|
|
static bool _gnrc_rpl_check_options_validity(int msg_type, gnrc_rpl_dodag_t *dodag,
|
|
|
|
gnrc_rpl_opt_t *opt, uint16_t len)
|
|
|
|
{
|
|
|
|
uint16_t expected_len = 0;
|
|
|
|
|
|
|
|
while(expected_len < len) {
|
|
|
|
switch(opt->type) {
|
|
|
|
case (GNRC_RPL_OPT_PAD1):
|
|
|
|
expected_len += 1;
|
|
|
|
opt = (gnrc_rpl_opt_t *) (((uint8_t *) opt) + 1);
|
|
|
|
continue;
|
|
|
|
|
|
|
|
case (GNRC_RPL_OPT_DODAG_CONF):
|
|
|
|
if (msg_type != GNRC_RPL_ICMPV6_CODE_DIO) {
|
|
|
|
DEBUG("RPL: DODAG CONF DIO option not expected\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt->length != GNRC_RPL_OPT_DODAG_CONF_LEN) {
|
|
|
|
DEBUG("RPL: wrong DIO option (DODAG CONF) len: %d, expected: %d\n",
|
|
|
|
opt->length, GNRC_RPL_OPT_DODAG_CONF_LEN);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case (GNRC_RPL_OPT_PREFIX_INFO):
|
|
|
|
if (msg_type != GNRC_RPL_ICMPV6_CODE_DIO) {
|
|
|
|
DEBUG("RPL: PREFIX INFO DIO option not expected\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt->length != GNRC_RPL_OPT_PREFIX_INFO_LEN) {
|
|
|
|
DEBUG("RPL: wrong DIO option (PREFIX INFO) len: %d, expected: %d\n",
|
|
|
|
opt->length, GNRC_RPL_OPT_PREFIX_INFO_LEN);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case (GNRC_RPL_OPT_TARGET):
|
|
|
|
if (msg_type != GNRC_RPL_ICMPV6_CODE_DAO) {
|
|
|
|
DEBUG("RPL: RPL TARGET DAO option not expected\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt->length > GNRC_RPL_OPT_TARGET_LEN) {
|
|
|
|
DEBUG("RPL: wrong DAO option (RPL TARGET) len: %d, expected (max): %d\n",
|
|
|
|
opt->length, GNRC_RPL_OPT_TARGET_LEN);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case (GNRC_RPL_OPT_TRANSIT):
|
|
|
|
if (msg_type != GNRC_RPL_ICMPV6_CODE_DAO) {
|
|
|
|
DEBUG("RPL: RPL TRANSIT INFO DAO option not expected\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t parent_addr = 0;
|
|
|
|
if (dodag->instance->mop == GNRC_RPL_MOP_NON_STORING_MODE) {
|
|
|
|
parent_addr = sizeof(ipv6_addr_t);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt->length != (GNRC_RPL_OPT_TRANSIT_INFO_LEN + parent_addr)) {
|
|
|
|
DEBUG("RPL: wrong DAO option (TRANSIT INFO) len: %d, expected: %d\n",
|
|
|
|
opt->length, (GNRC_RPL_OPT_TRANSIT_INFO_LEN + parent_addr));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
expected_len += opt->length + sizeof(gnrc_rpl_opt_t);
|
|
|
|
opt = (gnrc_rpl_opt_t *) (((uint8_t *) (opt + 1)) + opt->length);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (expected_len == len) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG("RPL: wrong options len: %d, expected: %d\n", len, expected_len);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-05-22 21:47:37 +02:00
|
|
|
/** @todo allow target prefixes in target options to be of variable length */
|
2015-08-17 15:41:29 +02:00
|
|
|
bool _parse_options(int msg_type, gnrc_rpl_dodag_t *dodag, gnrc_rpl_opt_t *opt, uint16_t len,
|
2015-08-25 11:03:28 +02:00
|
|
|
ipv6_addr_t *src, uint32_t *included_opts)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
|
|
|
uint16_t l = 0;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_opt_target_t *first_target = NULL;
|
2015-07-23 13:23:42 +02:00
|
|
|
eui64_t iid;
|
2015-08-19 20:57:28 +02:00
|
|
|
kernel_pid_t if_id = KERNEL_PID_UNDEF;
|
2015-08-25 11:03:28 +02:00
|
|
|
*included_opts = 0;
|
2015-08-19 20:57:28 +02:00
|
|
|
|
|
|
|
if (!_gnrc_rpl_check_options_validity(msg_type, dodag, opt, len)) {
|
|
|
|
return false;
|
|
|
|
}
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-19 20:57:28 +02:00
|
|
|
while(l < len) {
|
2015-05-22 21:47:37 +02:00
|
|
|
switch(opt->type) {
|
2015-08-19 20:57:28 +02:00
|
|
|
case (GNRC_RPL_OPT_PAD1):
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: PAD1 option parsed\n");
|
2015-08-25 11:03:28 +02:00
|
|
|
*included_opts |= ((uint32_t) 1) << GNRC_RPL_OPT_PAD1;
|
2015-05-22 21:47:37 +02:00
|
|
|
l += 1;
|
2015-08-17 15:41:29 +02:00
|
|
|
opt = (gnrc_rpl_opt_t *) (((uint8_t *) opt) + 1);
|
2015-05-22 21:47:37 +02:00
|
|
|
continue;
|
2015-08-19 20:57:28 +02:00
|
|
|
|
|
|
|
case (GNRC_RPL_OPT_PADN):
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: PADN option parsed\n");
|
2015-08-25 11:03:28 +02:00
|
|
|
*included_opts |= ((uint32_t) 1) << GNRC_RPL_OPT_PADN;
|
2015-05-22 21:47:37 +02:00
|
|
|
break;
|
2015-08-19 20:57:28 +02:00
|
|
|
|
|
|
|
case (GNRC_RPL_OPT_DODAG_CONF):
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: DODAG CONF DIO option parsed\n");
|
2015-08-25 11:03:28 +02:00
|
|
|
*included_opts |= ((uint32_t) 1) << GNRC_RPL_OPT_DODAG_CONF;
|
|
|
|
dodag->dodag_conf_requested = true;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_opt_dodag_conf_t *dc = (gnrc_rpl_opt_dodag_conf_t *) opt;
|
|
|
|
gnrc_rpl_of_t *of = gnrc_rpl_get_of_for_ocp(byteorder_ntohs(dc->ocp));
|
2015-05-22 21:47:37 +02:00
|
|
|
if (of != NULL) {
|
|
|
|
dodag->instance->of = of;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
DEBUG("RPL: Unsupported OCP 0x%02x\n", byteorder_ntohs(dc->ocp));
|
2015-08-17 15:41:29 +02:00
|
|
|
dodag->instance->of = gnrc_rpl_get_of_for_ocp(GNRC_RPL_DEFAULT_OCP);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
dodag->dio_interval_doubl = dc->dio_int_doubl;
|
|
|
|
dodag->dio_min = dc->dio_int_min;
|
|
|
|
dodag->dio_redun = dc->dio_redun;
|
|
|
|
dodag->instance->max_rank_inc = byteorder_ntohs(dc->max_rank_inc);
|
|
|
|
dodag->instance->min_hop_rank_inc = byteorder_ntohs(dc->min_hop_rank_inc);
|
|
|
|
dodag->default_lifetime = dc->default_lifetime;
|
|
|
|
dodag->lifetime_unit = byteorder_ntohs(dc->lifetime_unit);
|
|
|
|
dodag->trickle.Imin = (1 << dodag->dio_min);
|
|
|
|
dodag->trickle.Imax = dodag->dio_interval_doubl;
|
|
|
|
dodag->trickle.k = dodag->dio_redun;
|
|
|
|
break;
|
2015-08-19 20:57:28 +02:00
|
|
|
|
|
|
|
case (GNRC_RPL_OPT_PREFIX_INFO):
|
2015-07-23 13:23:42 +02:00
|
|
|
DEBUG("RPL: Prefix Information DIO option parsed\n");
|
2015-08-25 11:03:28 +02:00
|
|
|
*included_opts |= ((uint32_t) 1) << GNRC_RPL_OPT_PREFIX_INFO;
|
|
|
|
dodag->prefix_info_requested = true;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_opt_prefix_info_t *pi = (gnrc_rpl_opt_prefix_info_t *) opt;
|
|
|
|
ipv6_addr_t all_RPL_nodes = GNRC_RPL_ALL_NODES_ADDR;
|
2015-08-19 20:57:28 +02:00
|
|
|
if_id = gnrc_ipv6_netif_find_by_addr(NULL, &all_RPL_nodes);
|
2015-07-23 13:23:42 +02:00
|
|
|
/* check for the auto address-configuration flag */
|
2015-08-17 15:41:29 +02:00
|
|
|
if ((gnrc_netapi_get(if_id, NETOPT_IPV6_IID, 0, &iid, sizeof(eui64_t)) < 0) &&
|
|
|
|
!(pi->LAR_flags & GNRC_RPL_PREFIX_AUTO_ADDRESS_BIT)) {
|
2015-07-23 13:23:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
ipv6_addr_set_aiid(&pi->prefix, iid.uint8);
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_ipv6_netif_add_addr(if_id, &pi->prefix, pi->prefix_len, 0);
|
2015-07-23 13:23:42 +02:00
|
|
|
|
|
|
|
break;
|
2015-08-19 20:57:28 +02:00
|
|
|
|
|
|
|
case (GNRC_RPL_OPT_TARGET):
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: RPL TARGET DAO option parsed\n");
|
2015-08-25 11:03:28 +02:00
|
|
|
*included_opts |= ((uint32_t) 1) << GNRC_RPL_OPT_TARGET;
|
2015-08-19 20:57:28 +02:00
|
|
|
if_id = gnrc_ipv6_netif_find_by_prefix(NULL, &dodag->dodag_id);
|
2015-05-22 21:47:37 +02:00
|
|
|
if (if_id == KERNEL_PID_UNDEF) {
|
|
|
|
DEBUG("RPL: no interface found for the configured DODAG id\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_opt_target_t *target = (gnrc_rpl_opt_target_t *) opt;
|
2015-05-22 21:47:37 +02:00
|
|
|
if (first_target == NULL) {
|
|
|
|
first_target = target;
|
|
|
|
}
|
|
|
|
|
2015-08-25 15:22:01 +02:00
|
|
|
fib_add_entry(&gnrc_ipv6_fib_table, if_id, target->target.u8,
|
2015-08-12 19:04:43 +02:00
|
|
|
sizeof(ipv6_addr_t), AF_INET6, src->u8,
|
|
|
|
sizeof(ipv6_addr_t), AF_INET6,
|
|
|
|
(dodag->default_lifetime * dodag->lifetime_unit) *
|
|
|
|
SEC_IN_MS);
|
2015-05-22 21:47:37 +02:00
|
|
|
break;
|
2015-08-19 20:57:28 +02:00
|
|
|
|
|
|
|
case (GNRC_RPL_OPT_TRANSIT):
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: RPL TRANSIT INFO DAO option parsed\n");
|
2015-08-25 11:03:28 +02:00
|
|
|
*included_opts |= ((uint32_t) 1) << GNRC_RPL_OPT_TRANSIT;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_opt_transit_t *transit = (gnrc_rpl_opt_transit_t *) opt;
|
2015-05-22 21:47:37 +02:00
|
|
|
if (first_target == NULL) {
|
|
|
|
DEBUG("RPL: Encountered a RPL TRANSIT DAO option without \
|
|
|
|
a preceding RPL TARGET DAO option\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
2015-08-25 15:22:01 +02:00
|
|
|
fib_update_entry(&gnrc_ipv6_fib_table,
|
|
|
|
first_target->target.u8,
|
2015-08-12 19:04:43 +02:00
|
|
|
sizeof(ipv6_addr_t), src->u8,
|
|
|
|
sizeof(ipv6_addr_t), AF_INET6,
|
2015-08-25 15:22:01 +02:00
|
|
|
(transit->path_lifetime *
|
|
|
|
dodag->lifetime_unit * SEC_IN_MS));
|
2015-08-17 15:41:29 +02:00
|
|
|
first_target = (gnrc_rpl_opt_target_t *) (((uint8_t *) (first_target)) +
|
|
|
|
sizeof(gnrc_rpl_opt_t) + first_target->length);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
2015-08-17 15:41:29 +02:00
|
|
|
while (first_target->type == GNRC_RPL_OPT_TARGET);
|
2015-05-22 21:47:37 +02:00
|
|
|
|
|
|
|
first_target = NULL;
|
|
|
|
break;
|
2015-08-19 20:57:28 +02:00
|
|
|
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
2015-08-17 15:41:29 +02:00
|
|
|
l += opt->length + sizeof(gnrc_rpl_opt_t);
|
|
|
|
opt = (gnrc_rpl_opt_t *) (((uint8_t *) (opt + 1)) + opt->length);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-08-19 19:53:09 +02:00
|
|
|
static bool _gnrc_rpl_check_DIO_validity(gnrc_rpl_dio_t *dio, uint16_t len)
|
|
|
|
{
|
|
|
|
uint16_t expected_len = sizeof(*dio) + sizeof(icmpv6_hdr_t);
|
|
|
|
|
|
|
|
if (expected_len <= len) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG("RPL: wrong DIO len: %d, expected: %d\n", len, expected_len);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
void gnrc_rpl_recv_DIO(gnrc_rpl_dio_t *dio, ipv6_addr_t *src, uint16_t len)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_instance_t *inst = NULL;
|
|
|
|
gnrc_rpl_dodag_t *dodag = NULL;
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-19 19:53:09 +02:00
|
|
|
if (!_gnrc_rpl_check_DIO_validity(dio, len)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
len -= (sizeof(gnrc_rpl_dio_t) + sizeof(icmpv6_hdr_t));
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if (gnrc_rpl_instance_add(dio->instance_id, &inst)) {
|
|
|
|
inst->mop = (dio->g_mop_prf >> GNRC_RPL_MOP_SHIFT) & GNRC_RPL_SHIFTED_MOP_MASK;
|
|
|
|
inst->of = gnrc_rpl_get_of_for_ocp(GNRC_RPL_DEFAULT_OCP);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
else if (inst == NULL) {
|
|
|
|
DEBUG("RPL: Could not allocate a new instance.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if ((byteorder_ntohs(dio->rank) == GNRC_RPL_INFINITE_RANK) &&
|
|
|
|
(gnrc_rpl_dodag_get(inst, &dio->dodag_id) == NULL)) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: ignore INFINITE_RANK DIO when we are not part of this DODAG\n");
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_instance_remove(inst);
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if (gnrc_rpl_dodag_add(inst, &dio->dodag_id, &dodag)) {
|
|
|
|
gnrc_rpl_parent_t *parent = NULL;
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if (!gnrc_rpl_parent_add_by_addr(dodag, src, &parent) && (parent == NULL)) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: Could not allocate new parent.\n");
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_dodag_remove(dodag);
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dodag->version = dio->version_number;
|
|
|
|
|
|
|
|
parent->rank = byteorder_ntohs(dio->rank);
|
|
|
|
|
2015-08-25 11:03:28 +02:00
|
|
|
uint32_t included_opts = 0;
|
|
|
|
if(!_parse_options(GNRC_RPL_ICMPV6_CODE_DIO, dodag, (gnrc_rpl_opt_t *)(dio + 1), len, NULL,
|
|
|
|
&included_opts)) {
|
2015-08-18 21:13:53 +02:00
|
|
|
DEBUG("RPL: Error encountered during DIO option parsing - remove DODAG\n");
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_dodag_remove(dodag);
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-25 11:03:28 +02:00
|
|
|
if (!(included_opts & (((uint32_t) 1) << GNRC_RPL_OPT_DODAG_CONF))) {
|
|
|
|
DEBUG("RPL: DIO without DODAG_CONF option - remove DODAG and request new DIO\n");
|
|
|
|
gnrc_rpl_dodag_remove(dodag);
|
|
|
|
gnrc_rpl_send_DIS(NULL, src);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_delay_dao(dodag);
|
|
|
|
trickle_start(gnrc_rpl_pid, &dodag->trickle, GNRC_RPL_MSG_TYPE_TRICKLE_INTERVAL,
|
|
|
|
GNRC_RPL_MSG_TYPE_TRICKLE_CALLBACK, (1 << dodag->dio_min),
|
2015-05-22 21:47:37 +02:00
|
|
|
dodag->dio_interval_doubl, dodag->dio_redun);
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_parent_update(dodag, parent);
|
2015-08-25 11:03:28 +02:00
|
|
|
DEBUG("RPL: Joined DODAG (%s).\n",
|
|
|
|
ipv6_addr_to_str(addr_str, &dio->dodag_id, sizeof(addr_str)));
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
else if (dodag == NULL) {
|
|
|
|
DEBUG("RPL: Could not allocate a new DODAG.\n");
|
|
|
|
if (inst->dodags == NULL) {
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_instance_remove(inst);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dodag->instance->mop !=
|
2015-08-17 15:41:29 +02:00
|
|
|
((dio->g_mop_prf >> GNRC_RPL_MOP_SHIFT) & GNRC_RPL_SHIFTED_MOP_MASK)) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: invalid MOP for this instance.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if (GNRC_RPL_COUNTER_GREATER_THAN(dio->version_number, dodag->version)) {
|
|
|
|
if (dodag->node_status == GNRC_RPL_ROOT_NODE) {
|
|
|
|
dodag->version = GNRC_RPL_COUNTER_INCREMENT(dio->version_number);
|
2015-05-22 21:47:37 +02:00
|
|
|
trickle_reset_timer(&dodag->trickle);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dodag->version = dio->version_number;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_local_repair(dodag);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
}
|
2015-08-17 15:41:29 +02:00
|
|
|
else if (GNRC_RPL_COUNTER_GREATER_THAN(dodag->version, dio->version_number)) {
|
2015-05-22 21:47:37 +02:00
|
|
|
trickle_reset_timer(&dodag->trickle);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if (dodag->node_status == GNRC_RPL_ROOT_NODE) {
|
|
|
|
if (byteorder_ntohs(dio->rank) != GNRC_RPL_INFINITE_RANK) {
|
2015-05-22 21:47:37 +02:00
|
|
|
trickle_increment_counter(&dodag->trickle);
|
|
|
|
}
|
2015-08-21 13:36:37 +02:00
|
|
|
else {
|
|
|
|
trickle_reset_timer(&dodag->trickle);
|
|
|
|
}
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_parent_t *parent = NULL;
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if (!gnrc_rpl_parent_add_by_addr(dodag, src, &parent) && (parent == NULL)) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: Could not allocate new parent.\n");
|
|
|
|
if (dodag->parents == NULL) {
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_dodag_remove(dodag);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* cppcheck-suppress nullPointer */
|
|
|
|
else if (parent != NULL) {
|
|
|
|
trickle_increment_counter(&dodag->trickle);
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
/* gnrc_rpl_parent_add_by_addr should have set this already */
|
2015-08-18 17:53:11 +02:00
|
|
|
assert(parent != NULL);
|
|
|
|
|
2015-05-22 21:47:37 +02:00
|
|
|
parent->rank = byteorder_ntohs(dio->rank);
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_parent_update(dodag, parent);
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-21 13:36:37 +02:00
|
|
|
/* sender of incoming DIO is not a parent of mine (anymore) and has an INFINITE rank
|
|
|
|
and I have a rank != INFINITE_RANK */
|
|
|
|
if (parent->state == 0) {
|
|
|
|
if ((byteorder_ntohs(dio->rank) == GNRC_RPL_INFINITE_RANK)
|
|
|
|
&& (dodag->my_rank != GNRC_RPL_INFINITE_RANK)) {
|
|
|
|
trickle_reset_timer(&dodag->trickle);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2015-08-21 12:43:41 +02:00
|
|
|
/* incoming DIO is from pref. parent */
|
2015-08-21 13:36:37 +02:00
|
|
|
else if (parent == dodag->parents) {
|
2015-08-21 12:43:41 +02:00
|
|
|
if (parent->dtsn != dio->dtsn) {
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_delay_dao(dodag);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
parent->dtsn = dio->dtsn;
|
2015-08-21 12:43:41 +02:00
|
|
|
dodag->grounded = dio->g_mop_prf >> GNRC_RPL_GROUNDED_SHIFT;
|
|
|
|
dodag->prf = dio->g_mop_prf & GNRC_RPL_PRF_MASK;
|
2015-08-25 11:03:28 +02:00
|
|
|
uint32_t included_opts = 0;
|
|
|
|
if(!_parse_options(GNRC_RPL_ICMPV6_CODE_DIO, dodag, (gnrc_rpl_opt_t *)(dio + 1), len, NULL,
|
|
|
|
&included_opts)) {
|
2015-08-21 12:43:41 +02:00
|
|
|
DEBUG("RPL: Error encountered during DIO option parsing - remove DODAG\n");
|
|
|
|
gnrc_rpl_dodag_remove(dodag);
|
|
|
|
return;
|
|
|
|
}
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
void _dao_fill_target(gnrc_rpl_opt_target_t *target, ipv6_addr_t *addr)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
2015-08-17 15:41:29 +02:00
|
|
|
target->type = GNRC_RPL_OPT_TARGET;
|
2015-05-22 21:47:37 +02:00
|
|
|
target->length = sizeof(target->flags) + sizeof(target->prefix_length) + sizeof(target->target);
|
|
|
|
target->flags = 0;
|
|
|
|
target->prefix_length = 128;
|
|
|
|
target->target = *addr;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
void gnrc_rpl_send_DAO(gnrc_rpl_dodag_t *dodag, ipv6_addr_t *destination, uint8_t lifetime)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
2015-09-30 18:21:14 +02:00
|
|
|
size_t dst_size = GNRC_IPV6_FIB_TABLE_SIZE;
|
|
|
|
fib_destination_set_entry_t fib_dest_set[GNRC_IPV6_FIB_TABLE_SIZE];
|
2015-05-22 21:47:37 +02:00
|
|
|
|
|
|
|
if (dodag == NULL) {
|
|
|
|
DEBUG("RPL: Error - trying to send DAO without being part of a dodag.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if (dodag->node_status == GNRC_RPL_ROOT_NODE) {
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (destination == NULL) {
|
|
|
|
if (dodag->parents == NULL) {
|
|
|
|
DEBUG("RPL: dodag has no preferred parent\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
destination = &(dodag->parents->addr);
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_pktsnip_t *pkt;
|
2015-08-10 13:22:24 +02:00
|
|
|
icmpv6_hdr_t *icmp;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_dao_t *dao;
|
|
|
|
gnrc_rpl_opt_target_t *target;
|
|
|
|
gnrc_rpl_opt_transit_t *transit;
|
2015-05-22 21:47:37 +02:00
|
|
|
|
|
|
|
/* find my address */
|
|
|
|
ipv6_addr_t *me = NULL;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_ipv6_netif_find_by_prefix(&me, &dodag->dodag_id);
|
2015-05-22 21:47:37 +02:00
|
|
|
if (me == NULL) {
|
|
|
|
DEBUG("RPL: no address configured\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_ipv6_netif_addr_t *me_netif = gnrc_ipv6_netif_addr_get(me);
|
2015-05-22 21:47:37 +02:00
|
|
|
if (me_netif == NULL) {
|
|
|
|
DEBUG("RPL: no netif address found for %s\n", ipv6_addr_to_str(addr_str, me,
|
|
|
|
sizeof(addr_str)));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* find prefix for my address */
|
|
|
|
ipv6_addr_t prefix;
|
2015-09-10 20:31:01 +02:00
|
|
|
memset(&prefix, 0, sizeof(prefix));
|
2015-05-22 21:47:37 +02:00
|
|
|
ipv6_addr_init_prefix(&prefix, me, me_netif->prefix_len);
|
2015-08-25 15:22:01 +02:00
|
|
|
fib_get_destination_set(&gnrc_ipv6_fib_table, prefix.u8,
|
|
|
|
sizeof(ipv6_addr_t), fib_dest_set, &dst_size);
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
int size = sizeof(icmpv6_hdr_t) + sizeof(gnrc_rpl_dao_t) +
|
|
|
|
(sizeof(gnrc_rpl_opt_target_t) * (dst_size + 1)) + sizeof(gnrc_rpl_opt_transit_t);
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-23 18:26:00 +02:00
|
|
|
bool local_instance = (dodag->instance->id & GNRC_RPL_INSTANCE_ID_MSB) ? true : false;
|
2015-08-18 21:13:53 +02:00
|
|
|
|
|
|
|
if (local_instance) {
|
|
|
|
size += sizeof(ipv6_addr_t);
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if ((pkt = gnrc_icmpv6_build(NULL, ICMPV6_RPL_CTRL, GNRC_RPL_ICMPV6_CODE_DAO, size)) == NULL) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: Send DAO - no space left in packet buffer\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-10 13:22:24 +02:00
|
|
|
icmp = (icmpv6_hdr_t *)pkt->data;
|
2015-08-17 15:41:29 +02:00
|
|
|
dao = (gnrc_rpl_dao_t *)(icmp + 1);
|
2015-08-18 21:13:53 +02:00
|
|
|
target = (gnrc_rpl_opt_target_t *) (dao + 1);
|
2015-05-22 21:47:37 +02:00
|
|
|
|
|
|
|
dao->instance_id = dodag->instance->id;
|
2015-08-18 21:13:53 +02:00
|
|
|
if (local_instance) {
|
|
|
|
/* set the D flag to indicate that a DODAG id is present */
|
|
|
|
dao->k_d_flags = GNRC_RPL_DAO_D_BIT;
|
|
|
|
memcpy((dao + 1), &dodag->dodag_id, sizeof(ipv6_addr_t));
|
|
|
|
target = (gnrc_rpl_opt_target_t *)(((uint8_t *) target) + sizeof(ipv6_addr_t));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dao->k_d_flags = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set the K flag to indicate that ACKs are required */
|
|
|
|
dao->k_d_flags |= GNRC_RPL_DAO_K_BIT;
|
2015-05-22 21:47:37 +02:00
|
|
|
dao->dao_sequence = dodag->dao_seq;
|
|
|
|
dao->reserved = 0;
|
|
|
|
|
|
|
|
/* add own address */
|
|
|
|
_dao_fill_target(target, me);
|
2015-08-18 21:13:53 +02:00
|
|
|
|
2015-05-22 21:47:37 +02:00
|
|
|
/* add children */
|
|
|
|
for (size_t i = 0; i < dst_size; ++i) {
|
|
|
|
target = (target + 1);
|
|
|
|
_dao_fill_target(target, ((ipv6_addr_t *) fib_dest_set[i].dest));
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
transit = (gnrc_rpl_opt_transit_t *) (target + 1);
|
|
|
|
transit->type = GNRC_RPL_OPT_TRANSIT;
|
2015-05-22 21:47:37 +02:00
|
|
|
transit->length = sizeof(transit->e_flags) + sizeof(transit->path_control) +
|
|
|
|
sizeof(transit->path_sequence) + sizeof(transit->path_lifetime);
|
|
|
|
transit->e_flags = 0;
|
|
|
|
transit->path_control = 0;
|
|
|
|
transit->path_sequence = 0;
|
|
|
|
transit->path_lifetime = lifetime;
|
|
|
|
|
2015-08-23 18:33:51 +02:00
|
|
|
gnrc_rpl_send(pkt, NULL, destination, &dodag->dodag_id);
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
GNRC_RPL_COUNTER_INCREMENT(dodag->dao_seq);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
void gnrc_rpl_send_DAO_ACK(gnrc_rpl_dodag_t *dodag, ipv6_addr_t *destination, uint8_t seq)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
|
|
|
if (dodag == NULL) {
|
|
|
|
DEBUG("RPL: Error - trying to send DAO-ACK without being part of a dodag.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_pktsnip_t *pkt;
|
2015-08-10 13:22:24 +02:00
|
|
|
icmpv6_hdr_t *icmp;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_dao_ack_t *dao_ack;
|
|
|
|
int size = sizeof(icmpv6_hdr_t) + sizeof(gnrc_rpl_dao_ack_t);
|
2015-08-23 18:26:00 +02:00
|
|
|
bool local_instance = (dodag->instance->id & GNRC_RPL_INSTANCE_ID_MSB) ? true : false;
|
2015-08-18 21:13:53 +02:00
|
|
|
|
|
|
|
if (local_instance) {
|
|
|
|
size += sizeof(ipv6_addr_t);
|
|
|
|
}
|
2015-05-22 21:47:37 +02:00
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if ((pkt = gnrc_icmpv6_build(NULL, ICMPV6_RPL_CTRL, GNRC_RPL_ICMPV6_CODE_DAO_ACK, size)) == NULL) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: Send DAOACK - no space left in packet buffer\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-10 13:22:24 +02:00
|
|
|
icmp = (icmpv6_hdr_t *)pkt->data;
|
2015-08-17 15:41:29 +02:00
|
|
|
dao_ack = (gnrc_rpl_dao_ack_t *)(icmp + 1);
|
2015-05-22 21:47:37 +02:00
|
|
|
|
|
|
|
dao_ack->instance_id = dodag->instance->id;
|
2015-08-18 21:13:53 +02:00
|
|
|
if (local_instance) {
|
|
|
|
/* set the D flag to indicate that a DODAG id is present */
|
|
|
|
dao_ack->d_reserved = GNRC_RPL_DAO_ACK_D_BIT;
|
|
|
|
memcpy((dao_ack + 1), &dodag->dodag_id, sizeof(ipv6_addr_t));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dao_ack->d_reserved = 0;
|
|
|
|
}
|
|
|
|
|
2015-05-22 21:47:37 +02:00
|
|
|
dao_ack->dao_sequence = seq;
|
|
|
|
dao_ack->status = 0;
|
|
|
|
|
2015-08-23 18:33:51 +02:00
|
|
|
gnrc_rpl_send(pkt, NULL, destination, &dodag->dodag_id);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
2015-08-19 19:18:15 +02:00
|
|
|
static bool _gnrc_rpl_check_DAO_validity(gnrc_rpl_dao_t *dao, uint16_t len)
|
|
|
|
{
|
|
|
|
uint16_t expected_len = sizeof(*dao) + sizeof(icmpv6_hdr_t);
|
|
|
|
|
|
|
|
if ((dao->k_d_flags & GNRC_RPL_DAO_D_BIT)) {
|
|
|
|
expected_len += sizeof(ipv6_addr_t);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (expected_len <= len) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG("RPL: wrong DAO len: %d, expected: %d\n", len, expected_len);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
void gnrc_rpl_recv_DAO(gnrc_rpl_dao_t *dao, ipv6_addr_t *src, uint16_t len)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_instance_t *inst = NULL;
|
|
|
|
gnrc_rpl_dodag_t *dodag = NULL;
|
2015-08-19 19:18:15 +02:00
|
|
|
|
|
|
|
if (!_gnrc_rpl_check_DAO_validity(dao, len)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-18 21:13:53 +02:00
|
|
|
gnrc_rpl_opt_t *opts = (gnrc_rpl_opt_t *) (dao + 1);
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if ((inst = gnrc_rpl_instance_get(dao->instance_id)) == NULL) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: DAO with unknown instance id (%d) received\n", dao->instance_id);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-18 21:13:53 +02:00
|
|
|
len -= (sizeof(gnrc_rpl_dao_t) + sizeof(icmpv6_hdr_t));
|
|
|
|
|
2015-05-22 21:47:37 +02:00
|
|
|
/* check if the D flag is set before accessing the DODAG id */
|
2015-08-18 21:13:53 +02:00
|
|
|
if ((dao->k_d_flags & GNRC_RPL_DAO_D_BIT)) {
|
|
|
|
if ((dodag = gnrc_rpl_dodag_get(inst, (ipv6_addr_t *)(dao + 1))) == NULL) {
|
|
|
|
DEBUG("RPL: DAO with unknown DODAG id (%s)\n", ipv6_addr_to_str(addr_str,
|
|
|
|
(ipv6_addr_t *)(dao + 1), sizeof(addr_str)));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
opts = (gnrc_rpl_opt_t *)(((uint8_t *) opts) + sizeof(ipv6_addr_t));
|
|
|
|
len -= sizeof(ipv6_addr_t);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
2015-08-18 21:13:53 +02:00
|
|
|
else {
|
|
|
|
if ((dodag = gnrc_rpl_dodag_get(inst, NULL)) == NULL) {
|
|
|
|
DEBUG("RPL: DAO for instance (%d) without DODAGs\n", dao->instance_id);
|
|
|
|
return;
|
|
|
|
}
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
2015-08-25 11:03:28 +02:00
|
|
|
uint32_t included_opts = 0;
|
|
|
|
if(!_parse_options(GNRC_RPL_ICMPV6_CODE_DAO, dodag, opts, len, src, &included_opts)) {
|
2015-08-19 20:57:28 +02:00
|
|
|
DEBUG("RPL: Error encountered during DAO option parsing - ignore DAO\n");
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* send a DAO-ACK if K flag is set */
|
2015-08-18 21:13:53 +02:00
|
|
|
if (dao->k_d_flags & GNRC_RPL_DAO_K_BIT) {
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_send_DAO_ACK(dodag, src, dao->dao_sequence);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_delay_dao(dodag);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
2015-08-19 19:29:17 +02:00
|
|
|
static bool _gnrc_rpl_check_DAO_ACK_validity(gnrc_rpl_dao_ack_t *dao_ack, uint16_t len)
|
|
|
|
{
|
|
|
|
uint16_t expected_len = sizeof(*dao_ack) + sizeof(icmpv6_hdr_t);
|
|
|
|
|
|
|
|
if ((dao_ack->d_reserved & GNRC_RPL_DAO_ACK_D_BIT)) {
|
|
|
|
expected_len += sizeof(ipv6_addr_t);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (expected_len == len) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG("RPL: wrong DAO-ACK len: %d, expected: %d\n", len, expected_len);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gnrc_rpl_recv_DAO_ACK(gnrc_rpl_dao_ack_t *dao_ack, uint16_t len)
|
2015-05-22 21:47:37 +02:00
|
|
|
{
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_instance_t *inst = NULL;
|
|
|
|
gnrc_rpl_dodag_t *dodag = NULL;
|
2015-08-19 19:29:17 +02:00
|
|
|
|
|
|
|
if (!_gnrc_rpl_check_DAO_ACK_validity(dao_ack, len)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
if ((inst = gnrc_rpl_instance_get(dao_ack->instance_id)) == NULL) {
|
2015-05-22 21:47:37 +02:00
|
|
|
DEBUG("RPL: DAO-ACK with unknown instance id (%d) received\n", dao_ack->instance_id);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check if the D flag is set before accessing the DODAG id */
|
2015-08-18 21:13:53 +02:00
|
|
|
if ((dao_ack->d_reserved & GNRC_RPL_DAO_ACK_D_BIT)) {
|
|
|
|
if ((dodag = gnrc_rpl_dodag_get(inst, (ipv6_addr_t *)(dao_ack + 1))) == NULL) {
|
|
|
|
DEBUG("RPL: DAO-ACK with unknown DODAG id (%s)\n", ipv6_addr_to_str(addr_str,
|
|
|
|
(ipv6_addr_t *)(dao_ack + 1), sizeof(addr_str)));
|
|
|
|
return;
|
|
|
|
}
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
2015-08-18 21:13:53 +02:00
|
|
|
/* D flag not set - global instance id */
|
|
|
|
else if ((dodag = gnrc_rpl_dodag_get(inst, NULL)) == NULL) {
|
|
|
|
DEBUG("RPL: DAO-ACK for instance (%d) without DODAGs\n", dao_ack->instance_id);
|
2015-05-22 21:47:37 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((dao_ack->status != 0) && (dao_ack->dao_sequence != dodag->dao_seq)) {
|
|
|
|
DEBUG("RPL: DAO-ACK sequence (%d) does not match expected sequence (%d)\n",
|
|
|
|
dao_ack->dao_sequence, dodag->dao_seq);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dodag->dao_ack_received = true;
|
2015-08-17 15:41:29 +02:00
|
|
|
gnrc_rpl_long_delay_dao(dodag);
|
2015-05-22 21:47:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @}
|
|
|
|
*/
|