2015-03-02 18:41:37 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup sys_shell_commands
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Shell commands for interacting with network devices
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
2015-03-11 22:54:52 +01:00
|
|
|
* @author Martine Lenders <mlenders@inf.fu-berlin.de>
|
2015-03-02 18:41:37 +01:00
|
|
|
*/
|
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
#include <stdbool.h>
|
2015-03-02 18:41:37 +01:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2015-07-06 18:06:17 +02:00
|
|
|
#include <inttypes.h>
|
2015-03-02 18:41:37 +01:00
|
|
|
|
|
|
|
#include "thread.h"
|
2015-03-12 00:37:18 +01:00
|
|
|
#include "net/ng_ipv6/addr.h"
|
|
|
|
#include "net/ng_ipv6/netif.h"
|
2015-03-02 18:41:37 +01:00
|
|
|
#include "net/ng_netif.h"
|
|
|
|
#include "net/ng_netapi.h"
|
|
|
|
#include "net/ng_netconf.h"
|
2015-03-11 22:54:52 +01:00
|
|
|
#include "net/ng_pkt.h"
|
2015-03-02 18:41:37 +01:00
|
|
|
#include "net/ng_pktbuf.h"
|
|
|
|
#include "net/ng_netif/hdr.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief The maximal expected link layer address length in byte
|
|
|
|
*/
|
2015-03-11 22:54:52 +01:00
|
|
|
#define MAX_ADDR_LEN (8U)
|
2015-03-02 18:41:37 +01:00
|
|
|
|
|
|
|
/* utility functions */
|
2015-03-11 22:54:52 +01:00
|
|
|
static bool _is_number(char *str)
|
|
|
|
{
|
|
|
|
for (; *str; str++) {
|
|
|
|
if (*str < '0' || *str > '9') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool _is_iface(kernel_pid_t dev)
|
|
|
|
{
|
2015-05-25 22:00:44 +02:00
|
|
|
kernel_pid_t ifs[NG_NETIF_NUMOF];
|
|
|
|
size_t numof = ng_netif_get(ifs);
|
2015-03-11 22:54:52 +01:00
|
|
|
|
2015-07-17 12:18:07 +02:00
|
|
|
for (size_t i = 0; i < numof && i < NG_NETIF_NUMOF; i++) {
|
2015-03-11 22:54:52 +01:00
|
|
|
if (ifs[i] == dev) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _set_usage(char *cmd_name)
|
|
|
|
{
|
|
|
|
printf("usage: %s <if_id> set <key> <value>\n", cmd_name);
|
|
|
|
puts(" Sets an hardware specific specific value\n"
|
|
|
|
" <key> may be one of the following\n"
|
|
|
|
" * \"addr\" - sets (short) address\n"
|
|
|
|
" * \"addr_long\" - sets long address\n"
|
|
|
|
" * \"addr_short\" - alias for \"addr\"\n"
|
|
|
|
" * \"channel\" - sets the frequency channel\n"
|
|
|
|
" * \"chan\" - alias for \"channel\""
|
|
|
|
" * \"nid\" - sets the network identifier (or the PAN ID)\n"
|
|
|
|
" * \"pan\" - alias for \"nid\"\n"
|
|
|
|
" * \"pan_id\" - alias for \"nid\"\n"
|
2015-03-24 22:13:02 +01:00
|
|
|
" * \"power\" - TX power in dBm\n"
|
2015-03-24 23:15:29 +01:00
|
|
|
" * \"src_len\" - sets the source address length in byte\n"
|
|
|
|
" * \"state\" - set the device state\n");
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
|
2015-04-09 20:26:10 +02:00
|
|
|
static void _flag_usage(char *cmd_name)
|
|
|
|
{
|
2015-03-12 00:37:18 +01:00
|
|
|
printf("usage: %s <if_id> [-]{promisc|autoack|preload}\n", cmd_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _add_usage(char *cmd_name)
|
|
|
|
{
|
|
|
|
printf("usage: %s <if_id> add [anycast|multicast|unicast] "
|
|
|
|
"<ipv6_addr>[/prefix_len]\n", cmd_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _del_usage(char *cmd_name)
|
|
|
|
{
|
|
|
|
printf("usage: %s <if_id> del <ipv6_addr>\n",
|
|
|
|
cmd_name);
|
2015-04-09 20:26:10 +02:00
|
|
|
}
|
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
static void _print_netconf(ng_netconf_opt_t opt)
|
2015-03-02 18:41:37 +01:00
|
|
|
{
|
2015-03-11 22:54:52 +01:00
|
|
|
switch (opt) {
|
|
|
|
case NETCONF_OPT_ADDRESS:
|
|
|
|
printf("(short) address");
|
|
|
|
break;
|
2015-03-02 18:41:37 +01:00
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
case NETCONF_OPT_ADDRESS_LONG:
|
|
|
|
printf("long address");
|
|
|
|
break;
|
2015-03-02 18:41:37 +01:00
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
case NETCONF_OPT_SRC_LEN:
|
|
|
|
printf("source address length");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETCONF_OPT_CHANNEL:
|
|
|
|
printf("channel");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETCONF_OPT_NID:
|
|
|
|
printf("network identifier");
|
|
|
|
break;
|
|
|
|
|
2015-03-24 22:13:02 +01:00
|
|
|
case NETCONF_OPT_TX_POWER:
|
|
|
|
printf("TX power [in dBm]");
|
|
|
|
break;
|
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
default:
|
|
|
|
/* we don't serve these options here */
|
|
|
|
break;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-24 23:15:29 +01:00
|
|
|
static void _print_netconf_state(ng_netconf_state_t state)
|
|
|
|
{
|
|
|
|
switch (state) {
|
|
|
|
case NETCONF_STATE_OFF:
|
|
|
|
printf("OFF");
|
|
|
|
break;
|
|
|
|
case NETCONF_STATE_SLEEP:
|
|
|
|
printf("SLEEP");
|
|
|
|
break;
|
|
|
|
case NETCONF_STATE_IDLE:
|
|
|
|
printf("IDLE");
|
|
|
|
break;
|
|
|
|
case NETCONF_STATE_RX:
|
|
|
|
printf("RX");
|
|
|
|
break;
|
|
|
|
case NETCONF_STATE_TX:
|
|
|
|
printf("TX");
|
|
|
|
break;
|
|
|
|
case NETCONF_STATE_RESET:
|
|
|
|
printf("RESET");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* nothing to do then */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-29 19:07:48 +02:00
|
|
|
static void _netif_list(kernel_pid_t dev)
|
2015-03-02 18:41:37 +01:00
|
|
|
{
|
2015-03-11 22:54:52 +01:00
|
|
|
uint8_t hwaddr[MAX_ADDR_LEN];
|
|
|
|
uint16_t u16;
|
2015-03-24 22:13:02 +01:00
|
|
|
int16_t i16;
|
2015-03-11 22:54:52 +01:00
|
|
|
int res;
|
2015-03-24 23:15:29 +01:00
|
|
|
ng_netconf_state_t state;
|
2015-04-09 20:26:10 +02:00
|
|
|
ng_netconf_enable_t enable;
|
|
|
|
bool linebreak = false;
|
2015-03-12 00:37:18 +01:00
|
|
|
#ifdef MODULE_NG_IPV6_NETIF
|
|
|
|
ng_ipv6_netif_t *entry = ng_ipv6_netif_get(dev);
|
|
|
|
char ipv6_addr[NG_IPV6_ADDR_MAX_STR_LEN];
|
|
|
|
#endif
|
|
|
|
|
2015-03-02 18:41:37 +01:00
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
printf("Iface %2d ", dev);
|
2015-03-02 18:41:37 +01:00
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_ADDRESS, 0, hwaddr, sizeof(hwaddr));
|
|
|
|
|
|
|
|
if (res >= 0) {
|
2015-03-24 15:21:05 +01:00
|
|
|
char hwaddr_str[res * 3];
|
2015-03-11 22:54:52 +01:00
|
|
|
printf(" HWaddr: ");
|
2015-03-24 15:21:05 +01:00
|
|
|
printf("%s", ng_netif_addr_to_str(hwaddr_str, sizeof(hwaddr_str),
|
|
|
|
hwaddr, res));
|
2015-03-11 22:54:52 +01:00
|
|
|
printf(" ");
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_CHANNEL, 0, &u16, sizeof(u16));
|
|
|
|
|
|
|
|
if (res >= 0) {
|
|
|
|
printf(" Channel: %" PRIu16 " ", u16);
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_NID, 0, &u16, sizeof(u16));
|
|
|
|
|
|
|
|
if (res >= 0) {
|
2015-03-19 18:18:16 +01:00
|
|
|
printf(" NID: 0x%" PRIx16 " ", u16);
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
|
2015-03-24 22:13:02 +01:00
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_TX_POWER, 0, &i16, sizeof(i16));
|
|
|
|
|
|
|
|
if (res >= 0) {
|
|
|
|
printf(" TX-Power: %" PRIi16 "dBm ", i16);
|
|
|
|
}
|
|
|
|
|
2015-03-24 23:15:29 +01:00
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_STATE, 0, &state, sizeof(state));
|
|
|
|
|
|
|
|
if (res >= 0) {
|
|
|
|
printf(" State: ");
|
|
|
|
_print_netconf_state(state);
|
|
|
|
}
|
|
|
|
|
2015-03-29 19:07:48 +02:00
|
|
|
printf("\n ");
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_ADDRESS_LONG, 0, hwaddr, sizeof(hwaddr));
|
|
|
|
|
|
|
|
if (res >= 0) {
|
2015-03-24 15:21:05 +01:00
|
|
|
char hwaddr_str[res * 3];
|
2015-03-11 22:54:52 +01:00
|
|
|
printf("Long HWaddr: ");
|
2015-03-24 15:21:05 +01:00
|
|
|
printf("%s", ng_netif_addr_to_str(hwaddr_str, sizeof(hwaddr_str),
|
|
|
|
hwaddr, res));
|
2015-03-29 19:07:48 +02:00
|
|
|
printf("\n ");
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
|
2015-04-09 20:26:10 +02:00
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_PROMISCUOUSMODE, 0, &enable, sizeof(enable));
|
|
|
|
|
|
|
|
if ((res >= 0) && (enable == NETCONF_ENABLE)) {
|
|
|
|
printf("PROMISC ");
|
|
|
|
linebreak = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_AUTOACK, 0, &enable, sizeof(enable));
|
|
|
|
|
|
|
|
if ((res >= 0) && (enable == NETCONF_ENABLE)) {
|
|
|
|
printf("AUTOACK ");
|
|
|
|
linebreak = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_PRELOADING, 0, &enable, sizeof(enable));
|
|
|
|
|
|
|
|
if ((res >= 0) && (enable == NETCONF_ENABLE)) {
|
|
|
|
printf("PRELOAD ");
|
|
|
|
linebreak = true;
|
|
|
|
}
|
|
|
|
|
2015-04-22 15:45:20 +02:00
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_RAWMODE, 0, &enable, sizeof(enable));
|
|
|
|
|
|
|
|
if ((res >= 0) && (enable == NETCONF_ENABLE)) {
|
|
|
|
printf("RAWMODE ");
|
|
|
|
linebreak = true;
|
|
|
|
}
|
|
|
|
|
2015-04-09 21:35:05 +02:00
|
|
|
#ifdef MODULE_NG_IPV6_NETIF
|
|
|
|
if ((entry != NULL) && (entry->flags & NG_IPV6_NETIF_FLAGS_SIXLOWPAN)) {
|
|
|
|
printf("6LO ");
|
|
|
|
linebreak = true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-04-09 20:26:10 +02:00
|
|
|
if (linebreak) {
|
|
|
|
printf("\n ");
|
|
|
|
}
|
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
res = ng_netapi_get(dev, NETCONF_OPT_SRC_LEN, 0, &u16, sizeof(u16));
|
|
|
|
|
|
|
|
if (res >= 0) {
|
2015-05-09 16:13:12 +02:00
|
|
|
printf("Source address length: %" PRIu16 "\n ", u16);
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
|
2015-03-12 00:37:18 +01:00
|
|
|
#ifdef MODULE_NG_IPV6_NETIF
|
|
|
|
for (int i = 0; i < NG_IPV6_NETIF_ADDR_NUMOF; i++) {
|
|
|
|
if (!ng_ipv6_addr_is_unspecified(&entry->addrs[i].addr)) {
|
|
|
|
printf("inet6 addr: ");
|
|
|
|
|
|
|
|
if (ng_ipv6_addr_to_str(ipv6_addr, &entry->addrs[i].addr,
|
|
|
|
NG_IPV6_ADDR_MAX_STR_LEN)) {
|
|
|
|
printf("%s/%" PRIu8 " scope: ", ipv6_addr,
|
|
|
|
entry->addrs[i].prefix_len);
|
|
|
|
|
|
|
|
if ((ng_ipv6_addr_is_link_local(&entry->addrs[i].addr))) {
|
|
|
|
printf("local");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("global");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (entry->addrs[i].flags & NG_IPV6_NETIF_ADDR_FLAGS_NON_UNICAST) {
|
|
|
|
if (ng_ipv6_addr_is_multicast(&entry->addrs[i].addr)) {
|
|
|
|
printf(" [multicast]");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf(" [anycast]");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("error in conversion");
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("\n ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
puts("");
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
|
|
|
|
2015-03-29 19:07:48 +02:00
|
|
|
static int _netif_set_u16(kernel_pid_t dev, ng_netconf_opt_t opt,
|
|
|
|
char *u16_str)
|
2015-03-02 18:41:37 +01:00
|
|
|
{
|
2015-03-11 22:54:52 +01:00
|
|
|
unsigned int res;
|
|
|
|
bool hex = false;
|
2015-03-02 18:41:37 +01:00
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
if (_is_number(u16_str)) {
|
|
|
|
if ((res = strtoul(u16_str, NULL, 10)) == ULONG_MAX) {
|
|
|
|
puts("error: unable to parse value.\n"
|
|
|
|
"Must be a 16-bit unsigned integer (dec or hex)\n");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
else {
|
|
|
|
if ((res = strtoul(u16_str, NULL, 16)) == ULONG_MAX) {
|
|
|
|
puts("error: unable to parse value.\n"
|
|
|
|
"Must be a 16-bit unsigned integer (dec or hex)\n");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
hex = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res > 0xffff) {
|
|
|
|
puts("error: unable to parse value.\n"
|
|
|
|
"Must be a 16-bit unsigned integer (dec or hex)\n");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
if (ng_netapi_set(dev, opt, 0, (uint16_t *)&res, sizeof(uint16_t)) < 0) {
|
|
|
|
printf("error: unable to set ");
|
|
|
|
_print_netconf(opt);
|
|
|
|
puts("");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
printf("success: set ");
|
|
|
|
_print_netconf(opt);
|
|
|
|
printf(" on interface %" PRIkernel_pid " to ", dev);
|
|
|
|
|
|
|
|
if (hex) {
|
|
|
|
printf("0x%04x\n", res);
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
|
|
|
else {
|
2015-03-11 22:54:52 +01:00
|
|
|
printf("%u\n", res);
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-29 19:07:48 +02:00
|
|
|
|
|
|
|
return 0;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
|
|
|
|
2015-03-29 19:07:48 +02:00
|
|
|
static int _netif_set_i16(kernel_pid_t dev, ng_netconf_opt_t opt,
|
|
|
|
char *i16_str)
|
2015-03-24 22:13:02 +01:00
|
|
|
{
|
|
|
|
int16_t val = (int16_t)atoi(i16_str);
|
|
|
|
|
|
|
|
if (ng_netapi_set(dev, opt, 0, (int16_t *)&val, sizeof(int16_t)) < 0) {
|
|
|
|
printf("error: unable to set ");
|
|
|
|
_print_netconf(opt);
|
|
|
|
puts("");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-24 22:13:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
printf("success: set ");
|
|
|
|
_print_netconf(opt);
|
|
|
|
printf(" on interface %" PRIkernel_pid " to %i\n", dev, val);
|
2015-03-29 19:07:48 +02:00
|
|
|
|
|
|
|
return 0;
|
2015-03-24 22:13:02 +01:00
|
|
|
}
|
|
|
|
|
2015-03-29 19:07:48 +02:00
|
|
|
static int _netif_set_flag(kernel_pid_t dev, ng_netconf_opt_t opt,
|
|
|
|
ng_netconf_enable_t set)
|
2015-04-09 20:26:10 +02:00
|
|
|
{
|
|
|
|
if (ng_netapi_set(dev, opt, 0, &set, sizeof(ng_netconf_enable_t)) < 0) {
|
|
|
|
puts("error: unable to set option");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-04-09 20:26:10 +02:00
|
|
|
}
|
|
|
|
printf("success: %sset option\n", (set) ? "" : "un");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 0;
|
2015-04-09 20:26:10 +02:00
|
|
|
}
|
|
|
|
|
2015-03-29 19:07:48 +02:00
|
|
|
static int _netif_set_addr(kernel_pid_t dev, ng_netconf_opt_t opt,
|
|
|
|
char *addr_str)
|
2015-03-02 18:41:37 +01:00
|
|
|
{
|
2015-03-11 22:54:52 +01:00
|
|
|
uint8_t addr[MAX_ADDR_LEN];
|
2015-03-24 15:21:05 +01:00
|
|
|
size_t addr_len = ng_netif_addr_from_str(addr, sizeof(addr), addr_str);
|
2015-03-02 18:41:37 +01:00
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
if (addr_len == 0) {
|
|
|
|
puts("error: unable to parse address.\n"
|
|
|
|
"Must be of format [0-9a-fA-F]{2}(:[0-9a-fA-F]{2})*\n"
|
|
|
|
"(hex pairs delimited by colons)");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
if (ng_netapi_set(dev, opt, 0, addr, addr_len) < 0) {
|
|
|
|
printf("error: unable to set ");
|
|
|
|
_print_netconf(opt);
|
|
|
|
puts("");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
printf("success: set ");
|
|
|
|
_print_netconf(opt);
|
2015-03-24 15:21:05 +01:00
|
|
|
printf(" on interface %" PRIkernel_pid " to %s\n", dev, addr_str);
|
2015-03-29 19:07:48 +02:00
|
|
|
|
|
|
|
return 0;
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
|
2015-03-29 19:07:48 +02:00
|
|
|
static int _netif_set_state(kernel_pid_t dev, char *state_str)
|
2015-03-24 23:15:29 +01:00
|
|
|
{
|
|
|
|
ng_netconf_state_t state;
|
|
|
|
if ((strcmp("off", state_str) == 0) || (strcmp("OFF", state_str) == 0)) {
|
|
|
|
state = NETCONF_STATE_OFF;
|
|
|
|
}
|
|
|
|
else if ((strcmp("sleep", state_str) == 0) ||
|
|
|
|
(strcmp("SLEEP", state_str) == 0)) {
|
|
|
|
state = NETCONF_STATE_SLEEP;
|
|
|
|
}
|
|
|
|
else if ((strcmp("idle", state_str) == 0) ||
|
|
|
|
(strcmp("IDLE", state_str) == 0)) {
|
|
|
|
state = NETCONF_STATE_IDLE;
|
|
|
|
}
|
|
|
|
else if ((strcmp("reset", state_str) == 0) ||
|
|
|
|
(strcmp("RESET", state_str) == 0)) {
|
|
|
|
state = NETCONF_STATE_RESET;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("usage: ifconfig <if_id> set state [off|sleep|idle|reset]");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-24 23:15:29 +01:00
|
|
|
}
|
|
|
|
if (ng_netapi_set(dev, NETCONF_OPT_STATE, 0,
|
|
|
|
&state, sizeof(ng_netconf_state_t)) < 0) {
|
|
|
|
printf("error: unable to set state to ");
|
|
|
|
_print_netconf_state(state);
|
|
|
|
puts("");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-24 23:15:29 +01:00
|
|
|
}
|
|
|
|
printf("success: set state of interface %" PRIkernel_pid " to ", dev);
|
|
|
|
_print_netconf_state(state);
|
|
|
|
puts("");
|
2015-03-29 19:07:48 +02:00
|
|
|
|
|
|
|
return 0;
|
2015-03-24 23:15:29 +01:00
|
|
|
}
|
|
|
|
|
2015-03-29 19:07:48 +02:00
|
|
|
static int _netif_set(char *cmd_name, kernel_pid_t dev, char *key, char *value)
|
2015-03-11 22:54:52 +01:00
|
|
|
{
|
|
|
|
if ((strcmp("addr", key) == 0) || (strcmp("addr_short", key) == 0)) {
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set_addr(dev, NETCONF_OPT_ADDRESS, value);
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
else if (strcmp("addr_long", key) == 0) {
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set_addr(dev, NETCONF_OPT_ADDRESS_LONG, value);
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
else if ((strcmp("channel", key) == 0) || (strcmp("chan", key) == 0)) {
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set_u16(dev, NETCONF_OPT_CHANNEL, value);
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
else if ((strcmp("nid", key) == 0) || (strcmp("pan", key) == 0) ||
|
|
|
|
(strcmp("pan_id", key) == 0)) {
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set_u16(dev, NETCONF_OPT_NID, value);
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
2015-03-24 22:13:02 +01:00
|
|
|
else if (strcmp("power", key) == 0) {
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set_i16(dev, NETCONF_OPT_TX_POWER, value);
|
2015-03-24 22:13:02 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
else if (strcmp("src_len", key) == 0) {
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set_u16(dev, NETCONF_OPT_SRC_LEN, value);
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-24 23:15:29 +01:00
|
|
|
else if (strcmp("state", key) == 0) {
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set_state(dev, value);
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-29 19:07:48 +02:00
|
|
|
|
|
|
|
_set_usage(cmd_name);
|
|
|
|
return 1;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
|
|
|
|
2015-04-09 20:26:10 +02:00
|
|
|
static int _netif_flag(char *cmd, kernel_pid_t dev, char *flag)
|
|
|
|
{
|
|
|
|
ng_netconf_enable_t set = NETCONF_ENABLE;
|
|
|
|
|
|
|
|
if (flag[0] == '-') {
|
|
|
|
set = NETCONF_DISABLE;
|
|
|
|
flag++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(flag, "promisc") == 0) {
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set_flag(dev, NETCONF_OPT_PROMISCUOUSMODE, set);
|
2015-04-09 20:26:10 +02:00
|
|
|
}
|
|
|
|
else if (strcmp(flag, "preload") == 0) {
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set_flag(dev, NETCONF_OPT_PRELOADING, set);
|
2015-04-09 20:26:10 +02:00
|
|
|
}
|
|
|
|
else if (strcmp(flag, "autoack") == 0) {
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set_flag(dev, NETCONF_OPT_AUTOACK, set);
|
2015-04-09 20:26:10 +02:00
|
|
|
}
|
2015-04-22 15:45:20 +02:00
|
|
|
else if (strcmp(flag, "raw") == 0) {
|
|
|
|
return _netif_set_flag(dev, NETCONF_OPT_RAWMODE, set);
|
|
|
|
}
|
2015-04-09 21:35:05 +02:00
|
|
|
else if (strcmp(flag, "6lo") == 0) {
|
|
|
|
#ifdef MODULE_NG_IPV6_NETIF
|
|
|
|
ng_ipv6_netif_t *entry = ng_ipv6_netif_get(dev);
|
|
|
|
|
|
|
|
if (entry == NULL) {
|
|
|
|
puts("error: unable to (un)set 6LoWPAN support");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_lock(&entry->mutex);
|
|
|
|
|
|
|
|
if (set) {
|
|
|
|
entry->flags |= NG_IPV6_NETIF_FLAGS_SIXLOWPAN;
|
|
|
|
printf("success: set 6LoWPAN support on interface %" PRIkernel_pid
|
|
|
|
"\n", dev);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
entry->flags &= ~NG_IPV6_NETIF_FLAGS_SIXLOWPAN;
|
|
|
|
printf("success: unset 6LoWPAN support on interface %" PRIkernel_pid
|
|
|
|
"\n", dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_unlock(&entry->mutex);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
puts("error: unable to (un)set 6LoWPAN support");
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
#endif
|
|
|
|
}
|
2015-04-09 20:26:10 +02:00
|
|
|
|
2015-03-29 19:07:48 +02:00
|
|
|
_flag_usage(cmd);
|
|
|
|
return 1;
|
2015-04-09 20:26:10 +02:00
|
|
|
}
|
|
|
|
|
2015-03-12 00:37:18 +01:00
|
|
|
#ifdef MODULE_NG_IPV6_NETIF
|
|
|
|
static uint8_t _get_prefix_len(char *addr)
|
|
|
|
{
|
|
|
|
int prefix_len = 128;
|
|
|
|
|
|
|
|
while ((*addr != '/') && (*addr != '\0')) {
|
|
|
|
addr++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*addr == '/') {
|
|
|
|
*addr = '\0';
|
|
|
|
prefix_len = atoi(addr + 1);
|
|
|
|
|
|
|
|
if ((prefix_len < 1) || (prefix_len > 128)) {
|
|
|
|
prefix_len = 128;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return prefix_len;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static int _netif_add(char *cmd_name, kernel_pid_t dev, int argc, char **argv)
|
|
|
|
{
|
|
|
|
#ifdef MODULE_NG_IPV6_NETIF
|
|
|
|
enum {
|
|
|
|
_UNICAST = 0,
|
|
|
|
_MULTICAST, /* multicast value just to check if given addr is mc */
|
|
|
|
_ANYCAST
|
|
|
|
} type = _UNICAST;
|
|
|
|
char *addr_str = argv[0];
|
|
|
|
ng_ipv6_addr_t addr;
|
|
|
|
uint8_t prefix_len;
|
|
|
|
|
|
|
|
if (argc > 1) {
|
|
|
|
if (strcmp(argv[0], "anycast") == 0) {
|
|
|
|
type = _ANYCAST;
|
|
|
|
addr_str = argv[1];
|
|
|
|
}
|
|
|
|
else if (strcmp(argv[0], "multicast") == 0) {
|
|
|
|
type = _MULTICAST;
|
|
|
|
addr_str = argv[1];
|
|
|
|
}
|
|
|
|
else if (strcmp(argv[0], "unicast") == 0) {
|
|
|
|
/* type already set to unicast */
|
|
|
|
addr_str = argv[1];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
_add_usage(cmd_name);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
prefix_len = _get_prefix_len(addr_str);
|
|
|
|
|
|
|
|
if (ng_ipv6_addr_from_str(&addr, addr_str) == NULL) {
|
|
|
|
puts("error: unable to parse IPv6 address.");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((argc > 1) && (ng_ipv6_addr_is_multicast(&addr)) && (type != _MULTICAST)) {
|
|
|
|
puts("error: address was not a multicast address.");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-05-09 19:41:14 +02:00
|
|
|
if (ng_ipv6_netif_add_addr(dev, &addr, prefix_len,
|
|
|
|
(type == _ANYCAST) ?
|
|
|
|
NG_IPV6_NETIF_ADDR_FLAGS_NON_UNICAST :
|
|
|
|
NG_IPV6_NETIF_ADDR_FLAGS_UNICAST) == NULL) {
|
2015-03-12 00:37:18 +01:00
|
|
|
printf("error: unable to add IPv6 address\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("success: added %s/%d to interface %" PRIkernel_pid "\n", addr_str,
|
|
|
|
prefix_len, dev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
(void)cmd_name;
|
|
|
|
(void)dev;
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
puts("error: unable to add IPv6 address.");
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _netif_del(kernel_pid_t dev, char *addr_str)
|
|
|
|
{
|
|
|
|
#ifdef MODULE_NG_IPV6_NETIF
|
|
|
|
ng_ipv6_addr_t addr;
|
|
|
|
|
|
|
|
if (ng_ipv6_addr_from_str(&addr, addr_str) == NULL) {
|
|
|
|
puts("error: unable to parse IPv6 address.");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ng_ipv6_netif_remove_addr(dev, &addr);
|
|
|
|
|
|
|
|
printf("success: removed %s to interface %" PRIkernel_pid "\n", addr_str,
|
|
|
|
dev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
(void)dev;
|
|
|
|
(void)addr_str;
|
|
|
|
puts("error: unable to delete IPv6 address.");
|
|
|
|
return 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-03-11 22:54:52 +01:00
|
|
|
/* shell commands */
|
2015-03-20 08:51:45 +01:00
|
|
|
int _netif_send(int argc, char **argv)
|
2015-03-02 18:41:37 +01:00
|
|
|
{
|
|
|
|
kernel_pid_t dev;
|
2015-03-11 22:54:52 +01:00
|
|
|
uint8_t addr[MAX_ADDR_LEN];
|
|
|
|
size_t addr_len;
|
|
|
|
ng_pktsnip_t *pkt;
|
|
|
|
ng_netif_hdr_t *nethdr;
|
2015-03-27 22:30:42 +01:00
|
|
|
uint8_t flags = 0x00;
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
if (argc < 4) {
|
2015-03-27 22:30:42 +01:00
|
|
|
printf("usage: %s <if> [<addr>|bcast] <data>\n", argv[0]);
|
2015-03-20 08:51:45 +01:00
|
|
|
return 1;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
|
2015-03-02 18:41:37 +01:00
|
|
|
/* parse interface */
|
|
|
|
dev = (kernel_pid_t)atoi(argv[1]);
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
if (!_is_iface(dev)) {
|
2015-03-02 18:41:37 +01:00
|
|
|
puts("error: invalid interface given");
|
2015-03-20 08:51:45 +01:00
|
|
|
return 1;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
/* parse address */
|
2015-03-24 15:21:05 +01:00
|
|
|
addr_len = ng_netif_addr_from_str(addr, sizeof(addr), argv[2]);
|
2015-03-11 22:54:52 +01:00
|
|
|
|
|
|
|
if (addr_len == 0) {
|
2015-03-27 22:30:42 +01:00
|
|
|
if (strcmp(argv[2], "bcast") == 0) {
|
|
|
|
flags |= NG_NETIF_HDR_FLAGS_BROADCAST;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("error: invalid address given");
|
|
|
|
return 1;
|
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* put packet together */
|
|
|
|
pkt = ng_pktbuf_add(NULL, argv[3], strlen(argv[3]), NG_NETTYPE_UNDEF);
|
|
|
|
pkt = ng_pktbuf_add(pkt, NULL, sizeof(ng_netif_hdr_t) + addr_len,
|
2015-03-24 15:28:07 +01:00
|
|
|
NG_NETTYPE_NETIF);
|
2015-03-11 22:54:52 +01:00
|
|
|
nethdr = (ng_netif_hdr_t *)pkt->data;
|
|
|
|
ng_netif_hdr_init(nethdr, 0, addr_len);
|
|
|
|
ng_netif_hdr_set_dst_addr(nethdr, addr, addr_len);
|
2015-03-27 22:30:42 +01:00
|
|
|
nethdr->flags = flags;
|
2015-03-11 22:54:52 +01:00
|
|
|
/* and send it */
|
|
|
|
ng_netapi_send(dev, pkt);
|
2015-03-20 08:51:45 +01:00
|
|
|
|
|
|
|
return 0;
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
|
2015-03-20 08:51:45 +01:00
|
|
|
int _netif_config(int argc, char **argv)
|
2015-03-11 22:54:52 +01:00
|
|
|
{
|
|
|
|
if (argc < 2) {
|
2015-05-25 22:00:44 +02:00
|
|
|
kernel_pid_t ifs[NG_NETIF_NUMOF];
|
|
|
|
size_t numof = ng_netif_get(ifs);
|
2015-03-11 22:54:52 +01:00
|
|
|
|
2015-07-17 12:18:07 +02:00
|
|
|
for (size_t i = 0; i < numof && i < NG_NETIF_NUMOF; i++) {
|
2015-03-11 22:54:52 +01:00
|
|
|
_netif_list(ifs[i]);
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-29 19:07:48 +02:00
|
|
|
|
|
|
|
return 0;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
else if (_is_number(argv[1])) {
|
|
|
|
kernel_pid_t dev = (kernel_pid_t)atoi(argv[1]);
|
|
|
|
|
|
|
|
if (_is_iface(dev)) {
|
|
|
|
if (argc < 3) {
|
|
|
|
_netif_list(dev);
|
2015-03-20 08:51:45 +01:00
|
|
|
return 0;
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
else if (strcmp(argv[2], "set") == 0) {
|
|
|
|
if (argc < 5) {
|
|
|
|
_set_usage(argv[0]);
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
|
|
|
|
2015-03-29 19:07:48 +02:00
|
|
|
return _netif_set(argv[0], dev, argv[3], argv[4]);
|
2015-03-11 22:54:52 +01:00
|
|
|
}
|
2015-03-12 00:37:18 +01:00
|
|
|
else if (strcmp(argv[2], "add") == 0) {
|
|
|
|
if (argc < 4) {
|
|
|
|
_add_usage(argv[0]);
|
|
|
|
return 1;
|
|
|
|
}
|
2015-03-11 22:54:52 +01:00
|
|
|
|
2015-03-12 00:37:18 +01:00
|
|
|
return _netif_add(argv[0], (kernel_pid_t)dev, argc - 3, argv + 3);
|
|
|
|
}
|
|
|
|
else if (strcmp(argv[2], "del") == 0) {
|
|
|
|
if (argc < 4) {
|
|
|
|
_del_usage(argv[0]);
|
|
|
|
return 1;
|
|
|
|
}
|
2015-04-09 20:26:10 +02:00
|
|
|
|
2015-03-12 00:37:18 +01:00
|
|
|
return _netif_del((kernel_pid_t)dev, argv[3]);
|
|
|
|
}
|
2015-04-09 20:26:10 +02:00
|
|
|
else {
|
|
|
|
return _netif_flag(argv[0], dev, argv[2]);
|
|
|
|
}
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
|
|
|
else {
|
2015-03-11 22:54:52 +01:00
|
|
|
puts("error: invalid interface given");
|
2015-03-29 19:07:48 +02:00
|
|
|
return 1;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-12 00:37:18 +01:00
|
|
|
printf("usage: %s [<if_id>]\n", argv[0]);
|
2015-04-09 20:26:10 +02:00
|
|
|
_set_usage(argv[0]);
|
|
|
|
_flag_usage(argv[0]);
|
2015-03-12 00:37:18 +01:00
|
|
|
_add_usage(argv[0]);
|
|
|
|
_del_usage(argv[0]);
|
2015-03-20 08:51:45 +01:00
|
|
|
return 1;
|
2015-03-02 18:41:37 +01:00
|
|
|
}
|