2015-05-22 21:48:30 +02:00
|
|
|
|
/*
|
2017-11-30 21:17:11 +01:00
|
|
|
|
* Copyright (C) 2018 HAW Hamburg
|
|
|
|
|
* Copyright (C) 2015–2017 Cenk Gündoğan <mail-github@cgundogan.de>
|
2015-05-22 21:48:30 +02:00
|
|
|
|
*
|
|
|
|
|
* 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.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**
|
2018-06-01 13:19:39 +02:00
|
|
|
|
* @ingroup sys_shell_commands
|
2015-05-22 21:48:30 +02:00
|
|
|
|
* @{
|
|
|
|
|
*
|
|
|
|
|
* @file
|
|
|
|
|
*
|
2017-11-30 21:17:11 +01:00
|
|
|
|
* @author Cenk Gündoğan <cenk.guendogan@haw-hamburg.de>
|
2015-05-22 21:48:30 +02:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <stdio.h>
|
2022-05-31 23:27:27 +02:00
|
|
|
|
|
2017-11-16 18:06:46 +01:00
|
|
|
|
#include "net/gnrc/netif.h"
|
2015-08-17 15:41:29 +02:00
|
|
|
|
#include "net/gnrc/rpl.h"
|
|
|
|
|
#include "net/gnrc/rpl/dodag.h"
|
2022-05-31 23:27:27 +02:00
|
|
|
|
#include "net/gnrc/rpl/structs.h"
|
|
|
|
|
#include "shell.h"
|
2015-05-22 21:48:30 +02:00
|
|
|
|
#include "trickle.h"
|
2022-05-31 23:27:27 +02:00
|
|
|
|
#include "utlist.h"
|
2015-08-24 10:27:04 +02:00
|
|
|
|
#ifdef MODULE_GNRC_RPL_P2P
|
|
|
|
|
#include "net/gnrc/rpl/p2p.h"
|
|
|
|
|
#include "net/gnrc/rpl/p2p_dodag.h"
|
|
|
|
|
#include "net/gnrc/rpl/p2p_structs.h"
|
|
|
|
|
#endif
|
2015-05-22 21:48:30 +02:00
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
int _gnrc_rpl_init(char *arg)
|
2015-05-22 21:48:30 +02:00
|
|
|
|
{
|
2017-05-13 12:39:48 +02:00
|
|
|
|
kernel_pid_t iface_pid = atoi(arg);
|
2017-11-16 18:06:46 +01:00
|
|
|
|
if (gnrc_netif_get_by_pid(iface_pid) == NULL) {
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("unknown interface specified\n");
|
2015-05-22 21:48:30 +02:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
gnrc_rpl_init(iface_pid);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
printf("successfully initialized RPL on interface %d\n", iface_pid);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
int _gnrc_rpl_dodag_root(char *arg1, char *arg2)
|
2015-05-22 21:48:30 +02:00
|
|
|
|
{
|
2017-05-13 12:39:48 +02:00
|
|
|
|
uint8_t instance_id = atoi(arg1);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
ipv6_addr_t dodag_id;
|
|
|
|
|
|
|
|
|
|
if (ipv6_addr_from_str(&dodag_id, arg2) == NULL) {
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("error: <dodag_id> must be a valid IPv6 address\n");
|
2015-05-22 21:48:30 +02:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-07 17:33:54 +01:00
|
|
|
|
gnrc_rpl_instance_t *inst = gnrc_rpl_root_init(instance_id, &dodag_id, false, false);
|
2015-08-27 22:29:08 +02:00
|
|
|
|
if (inst == NULL) {
|
2015-05-22 21:48:30 +02:00
|
|
|
|
char addr_str[IPV6_ADDR_MAX_STR_LEN];
|
|
|
|
|
printf("error: could not add DODAG (%s) to instance (%d)\n",
|
|
|
|
|
ipv6_addr_to_str(addr_str, &dodag_id, sizeof(addr_str)), instance_id);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
printf("successfully added a new RPL DODAG\n");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-23 14:25:39 +02:00
|
|
|
|
#ifdef MODULE_GNRC_RPL_P2P
|
|
|
|
|
int _gnrc_rpl_find(char *arg1, char *arg2)
|
|
|
|
|
{
|
2017-05-13 12:39:48 +02:00
|
|
|
|
uint8_t instance_id = atoi(arg1);
|
2015-08-23 14:25:39 +02:00
|
|
|
|
ipv6_addr_t dodag_id;
|
|
|
|
|
ipv6_addr_t target;
|
|
|
|
|
|
|
|
|
|
if (ipv6_addr_from_str(&dodag_id, arg1) == NULL) {
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("<dodag_id> must be a valid IPv6 address\n");
|
2015-08-23 14:25:39 +02:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ipv6_addr_from_str(&target, arg2) == NULL) {
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("<target> must be a valid IPv6 address\n");
|
2015-08-23 14:25:39 +02:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-24 10:27:04 +02:00
|
|
|
|
if (gnrc_rpl_p2p_root_init(0, &dodag_id, &target, true) == NULL) {
|
2015-08-23 14:25:39 +02:00
|
|
|
|
char addr_str[IPV6_ADDR_MAX_STR_LEN];
|
|
|
|
|
printf("error: could not add DODAG (%s) to instance (%d)\n",
|
|
|
|
|
ipv6_addr_to_str(addr_str, &dodag_id, sizeof(addr_str)), instance_id);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
printf("successfully initiated a P2P-RPL Route Discovery\n");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
int _gnrc_rpl_instance_remove(char *arg1)
|
2015-05-22 21:48:30 +02:00
|
|
|
|
{
|
2017-05-13 12:39:48 +02:00
|
|
|
|
uint8_t instance_id = atoi(arg1);
|
2015-08-17 15:41:29 +02:00
|
|
|
|
gnrc_rpl_instance_t *inst;
|
2015-05-22 21:48:30 +02:00
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
if ((inst = gnrc_rpl_instance_get(instance_id)) == NULL) {
|
2015-05-22 21:48:30 +02:00
|
|
|
|
printf("error: could not find the instance (%d)\n", instance_id);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
if (gnrc_rpl_instance_remove(inst) == false) {
|
2015-05-22 21:48:30 +02:00
|
|
|
|
printf("error: could not remove instance (%d)\n", instance_id);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
printf("success: removed instance (%d)\n", instance_id);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-27 22:29:08 +02:00
|
|
|
|
int _gnrc_rpl_trickle_reset(char *arg1)
|
2015-05-22 21:48:30 +02:00
|
|
|
|
{
|
2017-05-13 12:39:48 +02:00
|
|
|
|
uint8_t instance_id = atoi(arg1);
|
2015-08-17 15:41:29 +02:00
|
|
|
|
gnrc_rpl_instance_t *inst;
|
2015-05-22 21:48:30 +02:00
|
|
|
|
char addr_str[IPV6_ADDR_MAX_STR_LEN];
|
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
if ((inst = gnrc_rpl_instance_get(instance_id)) == NULL) {
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("error: could not find the <instance_id>\n");
|
2015-05-22 21:48:30 +02:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-27 22:29:08 +02:00
|
|
|
|
trickle_reset_timer(&(inst->dodag.trickle));
|
2015-05-22 21:48:30 +02:00
|
|
|
|
|
2015-08-27 22:29:08 +02:00
|
|
|
|
printf("success: reset trickle timer of DODAG (%s) from instance (%d)\n",
|
|
|
|
|
ipv6_addr_to_str(addr_str, &(inst->dodag.dodag_id), sizeof(addr_str)),
|
2015-05-22 21:48:30 +02:00
|
|
|
|
instance_id);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-27 22:29:08 +02:00
|
|
|
|
int _gnrc_rpl_trickle_stop(char *arg1)
|
2015-05-22 21:48:30 +02:00
|
|
|
|
{
|
2017-05-13 12:39:48 +02:00
|
|
|
|
uint8_t instance_id = atoi(arg1);
|
2015-08-17 15:41:29 +02:00
|
|
|
|
gnrc_rpl_instance_t *inst;
|
2015-05-22 21:48:30 +02:00
|
|
|
|
char addr_str[IPV6_ADDR_MAX_STR_LEN];
|
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
if ((inst = gnrc_rpl_instance_get(instance_id)) == NULL) {
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("error: could not find the <instance_id>\n");
|
2015-05-22 21:48:30 +02:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
2015-08-27 22:29:08 +02:00
|
|
|
|
trickle_stop(&(inst->dodag.trickle));
|
2015-05-22 21:48:30 +02:00
|
|
|
|
|
|
|
|
|
printf("success: stopped trickle timer of DODAG (%s) from instance (%d)\n",
|
2015-08-27 22:29:08 +02:00
|
|
|
|
ipv6_addr_to_str(addr_str, &(inst->dodag.dodag_id), sizeof(addr_str)),
|
2015-05-22 21:48:30 +02:00
|
|
|
|
instance_id);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-27 22:29:08 +02:00
|
|
|
|
int _gnrc_rpl_trickle_start(char *arg1)
|
2015-05-22 21:48:30 +02:00
|
|
|
|
{
|
2017-05-13 12:39:48 +02:00
|
|
|
|
uint8_t instance_id = atoi(arg1);
|
2015-08-17 15:41:29 +02:00
|
|
|
|
gnrc_rpl_instance_t *inst;
|
2015-05-22 21:48:30 +02:00
|
|
|
|
char addr_str[IPV6_ADDR_MAX_STR_LEN];
|
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
if ((inst = gnrc_rpl_instance_get(instance_id)) == NULL) {
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("error: could not find the <instance_id>\n");
|
2015-05-22 21:48:30 +02:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2017-06-08 21:39:55 +02:00
|
|
|
|
trickle_start(gnrc_rpl_pid, &(inst->dodag.trickle), GNRC_RPL_MSG_TYPE_TRICKLE_MSG,
|
|
|
|
|
(1 << inst->dodag.dio_min), inst->dodag.dio_interval_doubl,
|
|
|
|
|
inst->dodag.dio_redun);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
|
|
|
|
|
printf("success: started trickle timer of DODAG (%s) from instance (%d)\n",
|
2015-08-27 22:29:08 +02:00
|
|
|
|
ipv6_addr_to_str(addr_str, &(inst->dodag.dodag_id), sizeof(addr_str)),
|
2015-05-22 21:48:30 +02:00
|
|
|
|
instance_id);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-18 09:52:34 +02:00
|
|
|
|
int _gnrc_rpl_send_dis_w_sol_opt(char* VID, char* version, char* instance, char* dodag)
|
|
|
|
|
{
|
|
|
|
|
uint8_t VID_flags = atoi(VID);
|
|
|
|
|
uint8_t version_number = atoi(version);
|
|
|
|
|
uint8_t instance_id = atoi(instance);
|
|
|
|
|
|
|
|
|
|
gnrc_rpl_internal_opt_dis_solicited_t sol;
|
|
|
|
|
sol.type = GNRC_RPL_OPT_SOLICITED_INFO;
|
|
|
|
|
sol.length = GNRC_RPL_DIS_SOLICITED_INFO_LENGTH;
|
|
|
|
|
sol.VID_flags = htons(VID_flags);
|
|
|
|
|
sol.version_number = version_number;
|
|
|
|
|
sol.instance_id = instance_id;
|
|
|
|
|
|
|
|
|
|
if (ipv6_addr_from_str(&sol.dodag_id, dodag))
|
|
|
|
|
{
|
|
|
|
|
gnrc_rpl_internal_opt_t* opt[] = {(gnrc_rpl_internal_opt_t*)&sol};
|
|
|
|
|
gnrc_rpl_send_DIS(NULL, (ipv6_addr_t *) &ipv6_addr_all_rpl_nodes, opt, 1);
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("success: send a DIS with SOL option\n\n");
|
2017-04-18 09:52:34 +02:00
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
int _gnrc_rpl_send_dis(void)
|
2015-05-22 21:48:30 +02:00
|
|
|
|
{
|
2017-04-18 09:52:34 +02:00
|
|
|
|
gnrc_rpl_send_DIS(NULL, (ipv6_addr_t *) &ipv6_addr_all_rpl_nodes, NULL, 0);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("success: send a DIS\n\n");
|
2015-05-22 21:48:30 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-01 14:05:30 +02:00
|
|
|
|
#ifdef MODULE_NETSTATS_RPL
|
2022-06-29 11:21:23 +02:00
|
|
|
|
static void _print_stats_block(netstats_rpl_block_t *block, const char *name)
|
|
|
|
|
{
|
2022-06-28 18:09:51 +02:00
|
|
|
|
/* In the following we need to sync with the RPL thread via disabling IRQs
|
|
|
|
|
* to avoid reading corrupted data. The simpler strategy would be to
|
|
|
|
|
* disable IRQs during the whole printing (so in _stats()_, but stdio could
|
|
|
|
|
* be via a slow UART and, hence, have severe impact on the real time
|
|
|
|
|
* capabilities of the system. The second and simplest strategy would be to
|
|
|
|
|
* memcpy the whole netstats_rpl_t on to the stack with IRQs disabled and
|
|
|
|
|
* print the stack copy with IRQs re-enabled. However, that structure is
|
|
|
|
|
* 128 B in size, so that we would easily provoke a stack-overflow.
|
|
|
|
|
*
|
|
|
|
|
* Our strategy instead is to read the data four values at a time with
|
|
|
|
|
* IRQs disabled, and print them with IRQs re-enabled. The disadvantage is
|
|
|
|
|
* that stats may get updated while printing one group of values. However,
|
|
|
|
|
* the stats are grouped such that closely related values are read together.
|
|
|
|
|
* Hence, related metrics will always refer to the same state of the stats.
|
|
|
|
|
*/
|
|
|
|
|
unsigned irq_state = irq_disable();
|
|
|
|
|
uint32_t rx_ucast = block->rx_ucast_count;
|
|
|
|
|
uint32_t tx_ucast = block->tx_ucast_count;
|
|
|
|
|
uint32_t rx_mcast = block->rx_mcast_count;
|
|
|
|
|
uint32_t tx_mcast = block->tx_mcast_count;
|
|
|
|
|
irq_restore(irq_state);
|
2022-06-29 11:21:23 +02:00
|
|
|
|
printf("%7s #packets: %10" PRIu32 " / %-10" PRIu32 " %10" PRIu32 " / %-10" PRIu32 "\n",
|
2022-06-28 18:09:51 +02:00
|
|
|
|
name, rx_ucast, tx_ucast, rx_mcast, tx_mcast);
|
|
|
|
|
|
|
|
|
|
irq_state = irq_disable();
|
|
|
|
|
rx_ucast = block->rx_ucast_bytes;
|
|
|
|
|
tx_ucast = block->tx_ucast_bytes;
|
|
|
|
|
rx_mcast = block->rx_mcast_bytes;
|
|
|
|
|
tx_mcast = block->tx_mcast_bytes;
|
|
|
|
|
irq_restore(irq_state);
|
2022-06-29 11:21:23 +02:00
|
|
|
|
printf("%7s #bytes: %10" PRIu32 " / %-10" PRIu32 " %10" PRIu32 " / %-10" PRIu32 "\n",
|
2022-06-28 18:09:51 +02:00
|
|
|
|
name, rx_ucast, tx_ucast, rx_mcast, tx_mcast);
|
2022-06-29 11:21:23 +02:00
|
|
|
|
}
|
|
|
|
|
|
2016-06-01 14:05:30 +02:00
|
|
|
|
int _stats(void)
|
|
|
|
|
{
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf( "Statistics (ucast) RX / TX RX / TX (mcast)\n");
|
2022-06-29 11:21:23 +02:00
|
|
|
|
_print_stats_block(&gnrc_rpl_netstats.dio, "DIO");
|
|
|
|
|
_print_stats_block(&gnrc_rpl_netstats.dis, "DIS");
|
|
|
|
|
_print_stats_block(&gnrc_rpl_netstats.dao, "DAO");
|
|
|
|
|
_print_stats_block(&gnrc_rpl_netstats.dao_ack, "DAO-ACK");
|
2016-06-01 14:05:30 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
int _gnrc_rpl_dodag_show(void)
|
2015-05-22 21:48:30 +02:00
|
|
|
|
{
|
2021-04-20 17:04:27 +02:00
|
|
|
|
if (gnrc_rpl_pid == KERNEL_PID_UNDEF) {
|
|
|
|
|
printf("RPL not initializied\n");
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2015-05-22 21:48:30 +02:00
|
|
|
|
printf("instance table:\t");
|
2015-08-17 15:41:29 +02:00
|
|
|
|
for (uint8_t i = 0; i < GNRC_RPL_INSTANCES_NUMOF; ++i) {
|
|
|
|
|
if (gnrc_rpl_instances[i].state == 0) {
|
2015-05-22 21:48:30 +02:00
|
|
|
|
printf("[ ]");
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
printf("[X]");
|
|
|
|
|
}
|
2015-11-09 13:07:10 +01:00
|
|
|
|
putchar('\t');
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
|
|
|
|
|
2015-11-09 13:07:10 +01:00
|
|
|
|
putchar('\n');
|
|
|
|
|
|
2015-05-22 21:48:30 +02:00
|
|
|
|
printf("parent table:\t");
|
2015-08-17 15:41:29 +02:00
|
|
|
|
for (uint8_t i = 0; i < GNRC_RPL_PARENTS_NUMOF; ++i) {
|
|
|
|
|
if (gnrc_rpl_parents[i].state == 0) {
|
2015-05-22 21:48:30 +02:00
|
|
|
|
printf("[ ]");
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
printf("[X]");
|
|
|
|
|
}
|
2015-11-09 13:07:10 +01:00
|
|
|
|
putchar('\t');
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
2015-08-24 10:27:04 +02:00
|
|
|
|
|
2015-05-22 21:48:30 +02:00
|
|
|
|
putchar('\n');
|
2015-08-24 10:27:04 +02:00
|
|
|
|
|
|
|
|
|
#ifdef MODULE_GNRC_RPL_P2P
|
|
|
|
|
printf("p2p-rpl table:\t");
|
|
|
|
|
for (int8_t i = 0; i < GNRC_RPL_P2P_EXTS_NUMOF; ++i) {
|
|
|
|
|
if (gnrc_rpl_p2p_exts[i].state == 0) {
|
|
|
|
|
printf("[ ]");
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
printf("[X]");
|
|
|
|
|
}
|
|
|
|
|
putchar('\t');
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2015-05-22 21:48:30 +02:00
|
|
|
|
putchar('\n');
|
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
gnrc_rpl_dodag_t *dodag = NULL;
|
2015-05-22 21:48:30 +02:00
|
|
|
|
char addr_str[IPV6_ADDR_MAX_STR_LEN];
|
2015-09-04 13:20:49 +02:00
|
|
|
|
|
2015-08-17 15:41:29 +02:00
|
|
|
|
for (uint8_t i = 0; i < GNRC_RPL_INSTANCES_NUMOF; ++i) {
|
|
|
|
|
if (gnrc_rpl_instances[i].state == 0) {
|
2015-05-22 21:48:30 +02:00
|
|
|
|
continue;
|
|
|
|
|
}
|
2015-08-27 22:29:08 +02:00
|
|
|
|
|
|
|
|
|
dodag = &gnrc_rpl_instances[i].dodag;
|
|
|
|
|
|
2016-03-23 09:57:25 +01:00
|
|
|
|
printf("instance [%d | Iface: %" PRIkernel_pid " | mop: %d | ocp: %d | mhri: %d | mri %d]\n",
|
|
|
|
|
gnrc_rpl_instances[i].id, dodag->iface,
|
|
|
|
|
gnrc_rpl_instances[i].mop, gnrc_rpl_instances[i].of->ocp,
|
|
|
|
|
gnrc_rpl_instances[i].min_hop_rank_inc, gnrc_rpl_instances[i].max_rank_inc);
|
|
|
|
|
|
2017-11-30 21:23:44 +01:00
|
|
|
|
printf("\tdodag [%s | R: %d | OP: %s | PIO: %s | "
|
2021-04-15 13:36:18 +02:00
|
|
|
|
"TR(I=[%d,%d], k=%d, c=%d)]\n",
|
2015-08-27 22:29:08 +02:00
|
|
|
|
ipv6_addr_to_str(addr_str, &dodag->dodag_id, sizeof(addr_str)),
|
|
|
|
|
dodag->my_rank, (dodag->node_status == GNRC_RPL_LEAF_NODE ? "Leaf" : "Router"),
|
2016-03-15 15:18:59 +01:00
|
|
|
|
((dodag->dio_opts & GNRC_RPL_REQ_DIO_OPT_PREFIX_INFO) ? "on" : "off"),
|
2017-11-30 21:23:44 +01:00
|
|
|
|
(1 << dodag->dio_min), dodag->dio_interval_doubl, dodag->trickle.k,
|
2021-04-15 13:36:18 +02:00
|
|
|
|
dodag->trickle.c);
|
2015-08-27 22:29:08 +02:00
|
|
|
|
|
2015-08-24 10:27:04 +02:00
|
|
|
|
#ifdef MODULE_GNRC_RPL_P2P
|
|
|
|
|
if (dodag->instance->mop == GNRC_RPL_P2P_MOP) {
|
|
|
|
|
gnrc_rpl_p2p_ext_t *p2p_ext = gnrc_rpl_p2p_ext_get(dodag);
|
|
|
|
|
printf("\tP2P-Ext [%s | HBH: %s | R: %s | L: %us]\n",
|
|
|
|
|
ipv6_addr_to_str(addr_str, &p2p_ext->target, sizeof(addr_str)),
|
|
|
|
|
p2p_ext->hop_by_hop ? "True" : "False",
|
|
|
|
|
p2p_ext->reply ? "True" : "False", p2p_ext->lifetime_sec);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2019-11-21 08:59:45 +01:00
|
|
|
|
gnrc_rpl_parent_t *parent = NULL;
|
2015-08-27 22:29:08 +02:00
|
|
|
|
LL_FOREACH(gnrc_rpl_instances[i].dodag.parents, parent) {
|
2017-11-30 21:23:44 +01:00
|
|
|
|
printf("\t\tparent [addr: %s | rank: %d]\n",
|
2015-08-27 22:29:08 +02:00
|
|
|
|
ipv6_addr_to_str(addr_str, &parent->addr, sizeof(addr_str)),
|
2017-11-30 21:23:44 +01:00
|
|
|
|
parent->rank);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-27 22:29:08 +02:00
|
|
|
|
int _gnrc_rpl_operation(bool leaf, char *arg1)
|
2015-08-22 09:56:43 +02:00
|
|
|
|
{
|
2017-05-13 12:39:48 +02:00
|
|
|
|
uint8_t instance_id = atoi(arg1);
|
2015-08-22 09:56:43 +02:00
|
|
|
|
gnrc_rpl_instance_t *inst;
|
|
|
|
|
|
|
|
|
|
if ((inst = gnrc_rpl_instance_get(instance_id)) == NULL) {
|
2015-08-27 22:29:08 +02:00
|
|
|
|
printf("error: could not find the instance (%d)\n", instance_id);
|
2015-08-22 09:56:43 +02:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (leaf) {
|
2015-08-27 22:29:08 +02:00
|
|
|
|
gnrc_rpl_leaf_operation(&inst->dodag);
|
2015-08-22 09:56:43 +02:00
|
|
|
|
}
|
|
|
|
|
else {
|
2015-08-27 22:29:08 +02:00
|
|
|
|
gnrc_rpl_router_operation(&inst->dodag);
|
2015-08-22 09:56:43 +02:00
|
|
|
|
}
|
|
|
|
|
|
2015-08-27 22:29:08 +02:00
|
|
|
|
printf("success: operate in instance (%d) as %s\n", instance_id, leaf ? "leaf" : "router");
|
2015-08-22 09:56:43 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-03 08:13:35 +01:00
|
|
|
|
int _gnrc_rpl_set_pio(char *inst_id, bool status)
|
|
|
|
|
{
|
2017-05-13 12:39:48 +02:00
|
|
|
|
uint8_t instance_id = atoi(inst_id);
|
2016-02-03 08:13:35 +01:00
|
|
|
|
gnrc_rpl_instance_t *inst;
|
|
|
|
|
|
|
|
|
|
if ((inst = gnrc_rpl_instance_get(instance_id)) == NULL) {
|
|
|
|
|
printf("error: could not find the instance (%d)\n", instance_id);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gnrc_rpl_config_pio(&inst->dodag, status);
|
|
|
|
|
|
|
|
|
|
printf("success: %sactivated PIO transmissions\n", status ? "" : "de");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-08 08:51:38 +02:00
|
|
|
|
static int _gnrc_rpl(int argc, char **argv)
|
2015-05-22 21:48:30 +02:00
|
|
|
|
{
|
|
|
|
|
if ((argc < 2) || (strcmp(argv[1], "show") == 0)) {
|
2015-08-17 15:41:29 +02:00
|
|
|
|
return _gnrc_rpl_dodag_show();
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
|
|
|
|
else if ((argc == 3) && strcmp(argv[1], "init") == 0) {
|
2015-08-17 15:41:29 +02:00
|
|
|
|
return _gnrc_rpl_init(argv[2]);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
|
|
|
|
else if ((argc == 4) && strcmp(argv[1], "root") == 0) {
|
2015-08-17 15:41:29 +02:00
|
|
|
|
return _gnrc_rpl_dodag_root(argv[2], argv[3]);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[1], "rm") == 0) {
|
2015-08-27 22:29:08 +02:00
|
|
|
|
if (argc == 3) {
|
2015-08-17 15:41:29 +02:00
|
|
|
|
return _gnrc_rpl_instance_remove(argv[2]);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[1], "trickle") == 0) {
|
2015-08-27 22:29:08 +02:00
|
|
|
|
if ((argc == 4) && (strcmp(argv[2], "reset") == 0)) {
|
|
|
|
|
return _gnrc_rpl_trickle_reset(argv[3]);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
2015-08-27 22:29:08 +02:00
|
|
|
|
else if ((argc == 4) && (strcmp(argv[2], "stop") == 0)) {
|
|
|
|
|
return _gnrc_rpl_trickle_stop(argv[3]);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
2015-08-27 22:29:08 +02:00
|
|
|
|
else if ((argc == 4) && (strcmp(argv[2], "start") == 0)) {
|
|
|
|
|
return _gnrc_rpl_trickle_start(argv[3]);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[1], "send") == 0) {
|
|
|
|
|
if ((argc == 3) && (strcmp(argv[2], "dis") == 0)) {
|
2015-08-17 15:41:29 +02:00
|
|
|
|
return _gnrc_rpl_send_dis();
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
2017-04-18 09:52:34 +02:00
|
|
|
|
if ((argc == 7) && (strcmp(argv[2], "dis") == 0)) {
|
|
|
|
|
return _gnrc_rpl_send_dis_w_sol_opt(argv[3], argv[4], argv[5], argv[6]);
|
|
|
|
|
}
|
2015-05-22 21:48:30 +02:00
|
|
|
|
}
|
2015-08-22 09:56:43 +02:00
|
|
|
|
else if (strcmp(argv[1], "leaf") == 0) {
|
2015-08-27 22:29:08 +02:00
|
|
|
|
if (argc == 3) {
|
|
|
|
|
return _gnrc_rpl_operation(true, argv[2]);
|
2015-08-22 09:56:43 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (strcmp(argv[1], "router") == 0) {
|
2015-08-27 22:29:08 +02:00
|
|
|
|
if (argc == 3) {
|
|
|
|
|
return _gnrc_rpl_operation(false, argv[2]);
|
2015-08-22 09:56:43 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2016-02-03 08:13:35 +01:00
|
|
|
|
else if (strcmp(argv[1], "set") == 0) {
|
2020-04-23 11:15:45 +02:00
|
|
|
|
if ((argc > 2) && !IS_ACTIVE(CONFIG_GNRC_RPL_WITHOUT_PIO)) {
|
2016-02-03 08:13:35 +01:00
|
|
|
|
if (strcmp(argv[2], "pio") == 0) {
|
|
|
|
|
if ((argc == 5) && (strcmp(argv[3], "on") == 0)) {
|
|
|
|
|
return _gnrc_rpl_set_pio(argv[4], true);
|
|
|
|
|
}
|
|
|
|
|
else if ((argc == 5) && (strcmp(argv[3], "off") == 0)) {
|
|
|
|
|
return _gnrc_rpl_set_pio(argv[4], false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2015-08-24 10:27:04 +02:00
|
|
|
|
#ifdef MODULE_GNRC_RPL_P2P
|
|
|
|
|
else if (strcmp(argv[1], "find") == 0) {
|
|
|
|
|
if (argc == 4) {
|
|
|
|
|
return _gnrc_rpl_find(argv[2], argv[3]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2016-06-01 14:05:30 +02:00
|
|
|
|
#ifdef MODULE_NETSTATS_RPL
|
|
|
|
|
else if (strcmp(argv[1], "stats") == 0) {
|
|
|
|
|
return _stats();
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2015-05-22 21:48:30 +02:00
|
|
|
|
|
2015-08-24 10:27:04 +02:00
|
|
|
|
#ifdef MODULE_GNRC_RPL_P2P
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("* find <dodag_id> <target>\t\t\t- initiate a P2P-RPL route discovery\n");
|
2015-08-24 10:27:04 +02:00
|
|
|
|
#endif
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("* help\t\t\t\t\t- show usage\n");
|
|
|
|
|
printf("* init <if_id>\t\t\t\t- initialize RPL on the given interface\n");
|
|
|
|
|
printf("* leaf <instance_id>\t\t\t- operate as leaf in the instance\n");
|
|
|
|
|
printf("* trickle reset <instance_id>\t\t- reset the trickle timer\n");
|
|
|
|
|
printf("* trickle start <instance_id>\t\t- start the trickle timer\n");
|
|
|
|
|
printf("* trickle stop <instance_id>\t\t- stop the trickle timer\n");
|
|
|
|
|
printf("* rm <instance_id>\t\t\t- delete the given instance and related dodag\n");
|
|
|
|
|
printf("* root <inst_id> <dodag_id>\t\t- add a dodag to a new or existing instance\n");
|
|
|
|
|
printf("* router <instance_id>\t\t\t- operate as router in the instance\n");
|
|
|
|
|
printf("* send dis\t\t\t\t- send a multicast DIS\n");
|
|
|
|
|
printf("* send dis <VID_flags> <version> <instance_id> <dodag_id> - send a multicast DIS with SOL option\n");
|
2020-04-23 11:15:45 +02:00
|
|
|
|
|
|
|
|
|
if (!IS_ACTIVE(CONFIG_GNRC_RPL_WITHOUT_PIO)) {
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("* set pio <on/off> <instance_id>\t- (de-)activate PIO transmissions in DIOs\n");
|
2020-04-23 11:15:45 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-02-27 15:21:04 +01:00
|
|
|
|
printf("* show\t\t\t\t\t- show instance and dodag tables\n");
|
2015-05-22 21:48:30 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
2022-05-31 23:27:27 +02:00
|
|
|
|
|
|
|
|
|
SHELL_COMMAND(rpl, "rpl configuration tool ('rpl help' for more information)", _gnrc_rpl);
|
2015-05-22 21:48:30 +02:00
|
|
|
|
/**
|
|
|
|
|
* @}
|
|
|
|
|
*/
|