mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
609 lines
17 KiB
C
609 lines
17 KiB
C
/*
|
|
* Copyright 2020 Inria
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup sys_shell_commands
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Shell command implementation for OpenWSN
|
|
*
|
|
* @author Francisco Molina <francois-xavier.molina@inria.fr>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "shell.h"
|
|
#include "net/ieee802154.h"
|
|
#include "net/ipv6/addr.h"
|
|
#include "net/netif.h"
|
|
|
|
#include "openwsn.h"
|
|
#include "opendefs.h"
|
|
#include "scheduler.h"
|
|
#include "cross-layers/openqueue.h"
|
|
#include "cross-layers/idmanager.h"
|
|
#include "cross-layers/packetfunctions.h"
|
|
|
|
#include "02a-MAClow/IEEE802154E.h"
|
|
#include "02b-MAChigh/neighbors.h"
|
|
#include "02b-MAChigh/sixtop.h"
|
|
#include "02b-MAChigh/msf.h"
|
|
#include "03b-IPv6/icmpv6rpl.h"
|
|
#include "cjoin.h"
|
|
|
|
#define IEEE802154_LONG_ADDRESS_LEN_STR_MAX \
|
|
(sizeof("00:00:00:00:00:00:00:00"))
|
|
|
|
static const struct {
|
|
char *name;
|
|
int id;
|
|
} components[] = {
|
|
{ "?", COMPONENT_NULL },
|
|
{ "openwsn", COMPONENT_OPENWSN },
|
|
{ "idmanager", COMPONENT_IDMANAGER },
|
|
{ "openqueue", COMPONENT_OPENQUEUE },
|
|
{ "openserial", COMPONENT_OPENSERIAL },
|
|
{ "pktfuncs", COMPONENT_PACKETFUNCTIONS },
|
|
{ "random", COMPONENT_RANDOM },
|
|
{ "radio", COMPONENT_RADIO },
|
|
{ "154", COMPONENT_IEEE802154 },
|
|
{ "154e", COMPONENT_IEEE802154E },
|
|
{ "6top2154e", COMPONENT_SIXTOP_TO_IEEE802154E },
|
|
{ "154e26top", COMPONENT_IEEE802154E_TO_SIXTOP },
|
|
{ "6top", COMPONENT_SIXTOP },
|
|
{ "neigh", COMPONENT_NEIGHBORS },
|
|
{ "sched", COMPONENT_SCHEDULE },
|
|
{ "6topres", COMPONENT_SIXTOP_RES },
|
|
{ "bridge", COMPONENT_OPENBRIDGE },
|
|
{ "iphc", COMPONENT_IPHC },
|
|
{ "frag", COMPONENT_FRAG },
|
|
{ "fwd", COMPONENT_FORWARDING },
|
|
{ "icmpv6", COMPONENT_ICMPv6 },
|
|
{ "icmpv6echo", COMPONENT_ICMPv6ECHO },
|
|
{ "icmpv6router", COMPONENT_ICMPv6ROUTER },
|
|
{ "icmpv6rpl", COMPONENT_ICMPv6RPL },
|
|
{ "udp", COMPONENT_OPENUDP },
|
|
{ "coap", COMPONENT_OPENCOAP },
|
|
{ "cjoin", COMPONENT_CJOIN },
|
|
{ "oscore", COMPONENT_OSCORE },
|
|
{ "c6t", COMPONENT_C6T },
|
|
{ "uinject", COMPONENT_UINJECT },
|
|
};
|
|
|
|
static char *_get_component(int id)
|
|
{
|
|
for (unsigned i = 0; i < ARRAY_SIZE(components); i++) {
|
|
if (id == components[i].id) {
|
|
return components[i].name;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
int _openwsn_ifconfig(char *arg)
|
|
{
|
|
(void)arg;
|
|
|
|
open_addr_t *addr;
|
|
/* Use shared buffer for IEEE802154_LONG_ADDRES and IPV6_ADDR str
|
|
conversion */
|
|
#if IS_USED(MODULE_OPENWSN_IPV6)
|
|
char addr_str[IPV6_ADDR_MAX_STR_LEN];
|
|
#else
|
|
char addr_str[IEEE802154_LONG_ADDRESS_LEN_STR_MAX];
|
|
#endif
|
|
|
|
printf("Iface %d ", openwsn_get_pid());
|
|
|
|
addr = idmanager_getMyID(ADDR_16B);
|
|
printf("\tHWaddr: %s ",
|
|
netif_addr_to_str(addr->addr_16b, sizeof(addr->addr_16b),
|
|
addr_str));
|
|
|
|
addr = idmanager_getMyID(ADDR_PANID);
|
|
printf("NID: %s\n\n",
|
|
netif_addr_to_str(addr->panid, sizeof(addr->panid),
|
|
addr_str));
|
|
|
|
addr = idmanager_getMyID(ADDR_64B);
|
|
printf("\t\tLong HWaddr: %s\n",
|
|
netif_addr_to_str(addr->addr_64b, sizeof(addr->addr_64b),
|
|
addr_str));
|
|
|
|
if (IS_USED(MODULE_OPENWSN_IPV6)) {
|
|
/* Can't recover ADDR_128B directly, recover ADDR_64B
|
|
and ADDR_PREFIX to construct ADDR_128B */
|
|
open_addr_t addr_ipv6;
|
|
addr = idmanager_getMyID(ADDR_64B);
|
|
memcpy(&addr_ipv6.addr_128b[8], addr->addr_64b,
|
|
sizeof(addr->addr_64b));
|
|
addr = idmanager_getMyID(ADDR_PREFIX);
|
|
memcpy(&addr_ipv6.addr_128b[0], addr->prefix,
|
|
sizeof(addr->prefix));
|
|
ipv6_addr_to_str(addr_str, (ipv6_addr_t *)addr_ipv6.addr_128b,
|
|
IPV6_ADDR_MAX_STR_LEN);
|
|
printf("\t\tinet6 addr: %s\n\n", addr_str);
|
|
}
|
|
|
|
printf("\t\tIEEE802154E sync: %i\n", ieee154e_isSynch());
|
|
if (IS_USED(MODULE_OPENWSN_CJOIN)) {
|
|
printf("\t\t6TiSCH joined: %i\n", cjoin_getIsJoined());
|
|
}
|
|
puts("");
|
|
|
|
if (IS_USED(MODULE_OPENWSN_IPV6)) {
|
|
uint8_t index;
|
|
|
|
if (icmpv6rpl_getPreferredParentIndex(
|
|
&index) || idmanager_getIsDAGroot()) {
|
|
open_addr_t neighbor;
|
|
uint8_t addr_ipv6[IPV6_ADDR_BIT_LEN / 8];
|
|
printf("\t\tRPL rank: %i\n", icmpv6rpl_getMyDAGrank());
|
|
if (idmanager_getIsDAGroot()) {
|
|
puts("\t\tRPL parent: Node is DAG root");
|
|
}
|
|
else {
|
|
icmpv6rpl_getPreferredParentEui64(&neighbor);
|
|
printf("\t\tRPL parent: %s\n",
|
|
netif_addr_to_str(neighbor.addr_64b,
|
|
sizeof(neighbor.addr_64b),
|
|
addr_str));
|
|
}
|
|
printf("\t\tRPL children:\n");
|
|
for (uint8_t i = 0; i < MAXNUMNEIGHBORS; i++) {
|
|
if (neighbors_isNeighborWithHigherDAGrank(i)) {
|
|
neighbors_getNeighborEui64(&neighbor, ADDR_64B, i);
|
|
printf("\t\t\t%s\n",
|
|
netif_addr_to_str(neighbor.addr_64b,
|
|
sizeof(neighbor.addr_64b),
|
|
addr_str));
|
|
}
|
|
}
|
|
|
|
icmpv6rpl_getRPLDODAGid(addr_ipv6);
|
|
ipv6_addr_to_str(addr_str, (ipv6_addr_t *)addr_ipv6,
|
|
IPV6_ADDR_MAX_STR_LEN);
|
|
printf("\t\tRPL DODAG ID: %16s\n", addr_str);
|
|
}
|
|
else {
|
|
puts("\t\tNO RPL parent");
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int _neighbors_cmd(char *arg)
|
|
{
|
|
(void)arg;
|
|
|
|
char hwaddr_str[IEEE802154_LONG_ADDRESS_LEN_STR_MAX];
|
|
open_addr_t neighbor;
|
|
|
|
for (int i = 0; i < MAXNUMNEIGHBORS; i++) {
|
|
neighbors_getNeighborEui64(&neighbor, ADDR_64B, i);
|
|
netif_addr_to_str(neighbor.addr_64b, sizeof(neighbor.addr_64b),
|
|
hwaddr_str);
|
|
if (memcmp(hwaddr_str, "00:00:00:00:00:00:00:00",
|
|
IEEE802154_LONG_ADDRESS_LEN_STR_MAX)) {
|
|
printf("%02i. %s\n", i, hwaddr_str);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void _print_cell_usage(void)
|
|
{
|
|
puts("Usage:");
|
|
puts("\tcell list: show all active cell");
|
|
puts("\tcell add <slot_offset> <channel_offset> <adv|tx|rx>"
|
|
" [<address>]: add cell directly to schedule");
|
|
puts("\tcell rmv <slot_offset> <channel_offset> <adv|tx|rx>"
|
|
" [<address>]: remove cell directly in schedule ");
|
|
}
|
|
|
|
static int _cell_list_cmd(char *arg)
|
|
{
|
|
(void)arg;
|
|
|
|
extern schedule_vars_t schedule_vars;
|
|
char hwaddr_str[IEEE802154_LONG_ADDRESS_LEN_STR_MAX];
|
|
|
|
for (int i = 0; i < MAXACTIVESLOTS; i++) {
|
|
switch (schedule_vars.scheduleBuf[i].type) {
|
|
case CELLTYPE_TX:
|
|
printf("neigh: %s, slot: %03i, chan: %02i, type: TX\n",
|
|
netif_addr_to_str(
|
|
schedule_vars.scheduleBuf[i].neighbor.addr_64b,
|
|
IEEE802154_LONG_ADDRESS_LEN, hwaddr_str),
|
|
schedule_vars.scheduleBuf[i].slotOffset,
|
|
schedule_vars.scheduleBuf[i].channelOffset);
|
|
break;
|
|
case CELLTYPE_RX:
|
|
printf("slot: %03i, chan: %02i, type: RX\n",
|
|
schedule_vars.scheduleBuf[i].slotOffset,
|
|
schedule_vars.scheduleBuf[i].channelOffset);
|
|
break;
|
|
case CELLTYPE_TXRX:
|
|
printf("neigh: %s, slot: %03i, chan: %02i, type: RXTX\n",
|
|
netif_addr_to_str(
|
|
schedule_vars.scheduleBuf[i].neighbor.addr_64b,
|
|
IEEE802154_LONG_ADDRESS_LEN, hwaddr_str),
|
|
schedule_vars.scheduleBuf[i].slotOffset,
|
|
schedule_vars.scheduleBuf[i].channelOffset);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int _cell_manage_cmd(int argc, char **argv)
|
|
{
|
|
open_addr_t addr;
|
|
cellType_t type;
|
|
int res;
|
|
|
|
memset(&addr, 0, sizeof(addr));
|
|
|
|
if (argc < 5) {
|
|
_print_cell_usage();
|
|
return -1;
|
|
}
|
|
|
|
if (argc == 6) {
|
|
addr.type = ADDR_64B;
|
|
size_t len = netif_addr_from_str(argv[5], addr.addr_64b);
|
|
if (len == 0) {
|
|
puts("Error: invalid address");
|
|
return -1;
|
|
}
|
|
}
|
|
else {
|
|
addr.type = ADDR_ANYCAST;
|
|
}
|
|
|
|
if (!strcmp(argv[4], "adv")) {
|
|
type = CELLTYPE_TXRX;
|
|
}
|
|
else if (!strcmp(argv[4], "tx")) {
|
|
type = CELLTYPE_TX;
|
|
}
|
|
else if (!strcmp(argv[4], "rx")) {
|
|
type = CELLTYPE_RX;
|
|
}
|
|
else {
|
|
_print_cell_usage();
|
|
return -1;
|
|
}
|
|
|
|
if (!strcmp(argv[1], "add")) {
|
|
res =
|
|
schedule_addActiveSlot(atoi(argv[2]), type, true, false,
|
|
atoi(argv[3]), &addr);
|
|
}
|
|
else {
|
|
res = schedule_removeActiveSlot(atoi(argv[2]), type, true, &addr);
|
|
}
|
|
|
|
if (res == 0) {
|
|
puts("Successfully set link");
|
|
return 0;
|
|
}
|
|
else {
|
|
puts("Something went wrong (duplicate link?)");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
static int _cell_cmd(int argc, char **argv)
|
|
{
|
|
if (argc < 2) {
|
|
_print_cell_usage();
|
|
return -1;
|
|
}
|
|
|
|
if (!strcmp(argv[1], "list")) {
|
|
return _cell_list_cmd(NULL);
|
|
}
|
|
|
|
if (!strcmp(argv[1], "add") || !strcmp(argv[1], "rmv")) {
|
|
return _cell_manage_cmd(argc, argv);
|
|
}
|
|
|
|
_print_cell_usage();
|
|
return -1;
|
|
}
|
|
|
|
static void _print_6top_usage(void)
|
|
{
|
|
puts("Usage:");
|
|
puts("\t6top clear [<neighbor>]:"
|
|
" request neighbor to clear all cells");
|
|
puts("\t6top add <num> <adv|tx|rx> [<neighbor>]:"
|
|
" request parent to add num cells");
|
|
puts("\t6top rmv <num> <adv|tx|rx> [<neighbor>]:"
|
|
" request parent to remove num cells");
|
|
puts("\t6top rel <num> <adv|tx|rx> [<neighbor>]:"
|
|
" request parent to relocate num cells");
|
|
}
|
|
|
|
static int _6top_manage_cmd(int argc, char **argv)
|
|
{
|
|
cellInfo_ht cells_add[CELLLIST_MAX_LEN];
|
|
cellInfo_ht cells_rmv[CELLLIST_MAX_LEN];
|
|
open_addr_t neigh;
|
|
uint8_t cell_options;
|
|
uint8_t code = IANA_6TOP_CMD_NONE;
|
|
uint8_t num;
|
|
|
|
if (argc < 4) {
|
|
puts("Error: not enough args");
|
|
_print_6top_usage();
|
|
return -1;
|
|
}
|
|
|
|
if (argc == 5) {
|
|
neigh.type = ADDR_64B;
|
|
size_t len = netif_addr_from_str(argv[4], neigh.addr_64b);
|
|
if (len == 0) {
|
|
puts("Error: invalid address");
|
|
return -1;
|
|
}
|
|
}
|
|
else {
|
|
if (IS_USED(MODULE_OPENWSN_IPV6)) {
|
|
if (!icmpv6rpl_getPreferredParentEui64(&neigh)) {
|
|
puts("Error: no preferred parent");
|
|
return -1;
|
|
}
|
|
}
|
|
else {
|
|
puts("Error: a neighbor address must be supplied");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if (!strcmp(argv[3], "adv")) {
|
|
cell_options = CELLOPTIONS_TX | CELLOPTIONS_RX | CELLOPTIONS_SHARED;
|
|
}
|
|
else if (!strcmp(argv[3], "tx")) {
|
|
cell_options = CELLOPTIONS_TX;
|
|
}
|
|
else if (!strcmp(argv[3], "rx")) {
|
|
cell_options = CELLOPTIONS_RX;
|
|
}
|
|
else {
|
|
puts("Error: invalid cell option");
|
|
return -1;
|
|
}
|
|
|
|
num = atoi(argv[2]);
|
|
|
|
if (!strcmp(argv[1], "add")) {
|
|
code = IANA_6TOP_CMD_ADD;
|
|
if (!msf_candidateAddCellList(cells_add, num)) {
|
|
puts("Error: can't add that many cells");
|
|
return -1;
|
|
}
|
|
}
|
|
else if (!strcmp(argv[1], "rmv")) {
|
|
code = IANA_6TOP_CMD_DELETE;
|
|
if (!msf_candidateRemoveCellList(cells_rmv, &neigh, num,
|
|
cell_options)) {
|
|
puts("Error: can't remove the specified cells");
|
|
return -1;
|
|
}
|
|
}
|
|
else if (!strcmp(argv[1], "rel")) {
|
|
code = IANA_6TOP_CMD_RELOCATE;
|
|
if (!schedule_getCellsToBeRelocated(&neigh, cells_rmv) ||
|
|
!msf_candidateAddCellList(cells_add, num)) {
|
|
puts("Error: failed to get cells to relocate");
|
|
return -1;
|
|
}
|
|
}
|
|
else {
|
|
puts("Error: unknown 6top command");
|
|
return -1;
|
|
}
|
|
|
|
int ret = sixtop_request(code, &neigh, num, cell_options, cells_add,
|
|
cells_rmv, IANA_6TISCH_SFID_MSF, 0, 0);
|
|
|
|
if (ret) {
|
|
puts("Error: 6top request failed");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int _6top_cmd(int argc, char **argv)
|
|
{
|
|
if (argc < 2) {
|
|
_print_6top_usage();
|
|
return -1;
|
|
}
|
|
|
|
if (!strcmp(argv[1], "clear")) {
|
|
|
|
open_addr_t neighbor;
|
|
|
|
if (argc == 3) {
|
|
neighbor.type = ADDR_64B;
|
|
size_t len = netif_addr_from_str(argv[2], neighbor.addr_64b);
|
|
if (len == 0) {
|
|
puts("Error: invalid address");
|
|
return -1;
|
|
}
|
|
}
|
|
else {
|
|
if (IS_USED(MODULE_OPENWSN_IPV6)) {
|
|
if (!icmpv6rpl_getPreferredParentEui64(&neighbor)) {
|
|
puts("Error: no preferred parent");
|
|
return -1;
|
|
}
|
|
}
|
|
else {
|
|
puts("Error: a neighbor address must be supplied");
|
|
return -1;
|
|
}
|
|
}
|
|
int ret = sixtop_request(IANA_6TOP_CMD_CLEAR, &neighbor, 0, 0,
|
|
NULL, NULL, IANA_6TISCH_SFID_MSF, 0, 0);
|
|
if (ret) {
|
|
puts("Error: 6top request failed");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if (!strcmp(argv[1], "add") || !strcmp(argv[1], "rmv") ||
|
|
!strcmp(argv[1], "rel")) {
|
|
return _6top_manage_cmd(argc, argv);
|
|
}
|
|
|
|
_print_6top_usage();
|
|
return -1;
|
|
}
|
|
|
|
static void _print_queue_usage(void)
|
|
{
|
|
puts("Usage:");
|
|
puts("\tqueue list: show all Openqueue entries");
|
|
puts("\tqueue rmv <creator>: remove all entries"
|
|
" from <creator> in queue");
|
|
}
|
|
|
|
static int _queue_cmd(int argc, char **argv)
|
|
{
|
|
if (argc < 2) {
|
|
_print_queue_usage();
|
|
return -1;
|
|
}
|
|
|
|
if (!strcmp(argv[1], "list")) {
|
|
extern openqueue_vars_t openqueue_vars;
|
|
bool empty_queue = true;
|
|
|
|
for (uint8_t i = 0; i < QUEUELENGTH; i++) {
|
|
if (openqueue_vars.queue[i].creator ||
|
|
openqueue_vars.queue[i].owner) {
|
|
uint8_t creator = openqueue_vars.queue[i].creator;
|
|
uint8_t owner = openqueue_vars.queue[i].owner;
|
|
printf("Creator: %.9s [%d], ",
|
|
_get_component(creator), creator);
|
|
printf("Owner: %.9s [%d]\n",
|
|
_get_component(owner), owner);
|
|
empty_queue = false;
|
|
}
|
|
}
|
|
if (empty_queue) {
|
|
puts("Queue is empty");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
if (!strcmp(argv[1], "rmv")) {
|
|
if (argc != 3) {
|
|
_print_queue_usage();
|
|
return -1;
|
|
}
|
|
|
|
uint8_t creator = atoi(argv[2]);
|
|
|
|
if (creator == 0) {
|
|
puts("Error: invalid input value\n");
|
|
return -1;
|
|
}
|
|
else {
|
|
printf("Removing entries created by: %.9s [%d]\n",
|
|
_get_component(creator), creator);
|
|
openqueue_removeAllCreatedBy(creator);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
_print_queue_usage();
|
|
return -1;
|
|
}
|
|
|
|
#if SCHEDULER_DEBUG_ENABLE
|
|
static int _scheduler_cmd(char *arg)
|
|
{
|
|
(void)arg;
|
|
extern scheduler_dbg_t scheduler_dbg;
|
|
|
|
printf("Current tasks: %i\n", scheduler_dbg.numTasksCur);
|
|
printf("Max tasks: %i\n", scheduler_dbg.numTasksMax);
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
static void _print_usage(void)
|
|
{
|
|
puts("Usage:");
|
|
puts("\topenwsn neigh: show neighbor table");
|
|
puts("\topenwsn queue: Openqueue management commands");
|
|
puts("\topenwsn cell: cell management commands");
|
|
puts("\topenwsn 6top: 6top request commands");
|
|
#if SCHEDULER_DEBUG_ENABLE
|
|
puts("\topenwsn sched: show openos scheduler information");
|
|
#endif
|
|
}
|
|
|
|
int _openwsn_handler(int argc, char **argv)
|
|
{
|
|
if (argc < 2) {
|
|
_print_usage();
|
|
return -1;
|
|
}
|
|
|
|
if (!strcmp(argv[1], "neigh")) {
|
|
return _neighbors_cmd(NULL);
|
|
}
|
|
|
|
if (!strcmp(argv[1], "queue")) {
|
|
return _queue_cmd(argc - 1, &argv[1]);
|
|
}
|
|
|
|
if (!strcmp(argv[1], "cell")) {
|
|
if (!ieee154e_isSynch()) {
|
|
puts("Error: node is not synchronized");
|
|
return -1;
|
|
}
|
|
return _cell_cmd(argc - 1, &argv[1]);
|
|
}
|
|
|
|
#if SCHEDULER_DEBUG_ENABLE
|
|
if (!strcmp(argv[1], "sched")) {
|
|
return _scheduler_cmd(NULL);
|
|
}
|
|
#endif
|
|
|
|
if (!strcmp(argv[1], "6top")) {
|
|
if (!ieee154e_isSynch()) {
|
|
puts("Error: node is not synchronized");
|
|
return -1;
|
|
}
|
|
return _6top_cmd(argc - 1, &argv[1]);
|
|
}
|
|
|
|
_print_usage();
|
|
return -1;
|
|
}
|