2016-01-25 00:17:49 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @author Martine Lenders <mlenders@inf.fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2022-07-01 00:13:13 +02:00
|
|
|
#include "at86rf2xx_internal.h"
|
2016-01-25 00:17:49 +01:00
|
|
|
#include "common.h"
|
2022-07-04 13:55:31 +02:00
|
|
|
#include "net/ieee802154.h"
|
|
|
|
#include "net/netdev/ieee802154.h"
|
2016-01-25 00:17:49 +01:00
|
|
|
#include "od.h"
|
2022-07-04 13:55:31 +02:00
|
|
|
#include "test_utils/expect.h"
|
2016-01-25 00:17:49 +01:00
|
|
|
|
2017-10-26 00:26:48 +02:00
|
|
|
#define _MAX_ADDR_LEN (8)
|
|
|
|
#define MAC_VECTOR_SIZE (2) /* mhr + payload */
|
2016-01-25 00:17:49 +01:00
|
|
|
|
2017-10-26 00:26:48 +02:00
|
|
|
static size_t _parse_addr(uint8_t *out, size_t out_len, const char *in);
|
2016-01-25 00:17:49 +01:00
|
|
|
static int send(int iface, le_uint16_t dst_pan, uint8_t *dst_addr,
|
|
|
|
size_t dst_len, char *data);
|
|
|
|
|
|
|
|
int ifconfig_list(int idx)
|
|
|
|
{
|
|
|
|
int res;
|
2021-06-21 16:32:16 +02:00
|
|
|
netdev_ieee802154_t *dev = &devs[idx].netdev;
|
2016-01-25 00:17:49 +01:00
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
int (*get)(netdev_t *, netopt_t, void *, size_t) = dev->netdev.driver->get;
|
2016-01-25 00:17:49 +01:00
|
|
|
netopt_enable_t enable_val;
|
|
|
|
uint16_t u16_val;
|
|
|
|
|
|
|
|
printf("Iface %3d HWaddr: ", idx);
|
|
|
|
print_addr(dev->short_addr, IEEE802154_SHORT_ADDRESS_LEN);
|
|
|
|
printf(", Long HWaddr: ");
|
|
|
|
print_addr(dev->long_addr, IEEE802154_LONG_ADDRESS_LEN);
|
|
|
|
printf(", PAN: 0x%04x", dev->pan);
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_ADDR_LEN, &u16_val, sizeof(u16_val));
|
2016-01-25 00:17:49 +01:00
|
|
|
if (res < 0) {
|
|
|
|
puts("(err)");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
printf("\n Address length: %u", (unsigned)u16_val);
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_SRC_LEN, &u16_val, sizeof(u16_val));
|
2016-01-25 00:17:49 +01:00
|
|
|
if (res < 0) {
|
|
|
|
puts("(err)");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
printf(", Source address length: %u", (unsigned)u16_val);
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_MAX_PDU_SIZE, &u16_val,
|
2016-01-25 00:17:49 +01:00
|
|
|
sizeof(u16_val));
|
|
|
|
if (res < 0) {
|
|
|
|
puts("(err)");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
printf(", Max.Payload: %u", (unsigned)u16_val);
|
|
|
|
printf("\n Channel: %u", dev->chan);
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_CHANNEL_PAGE, &u16_val, sizeof(u16_val));
|
2016-01-25 00:17:49 +01:00
|
|
|
if (res < 0) {
|
|
|
|
puts("(err)");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
printf(", Ch.page: %u", (unsigned)u16_val);
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_TX_POWER, &u16_val, sizeof(u16_val));
|
2016-01-25 00:17:49 +01:00
|
|
|
if (res < 0) {
|
|
|
|
puts("(err)");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
printf(", TXPower: %d dBm", (int)u16_val);
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_IS_WIRED, &u16_val, sizeof(u16_val));
|
2016-01-25 00:17:49 +01:00
|
|
|
if (res < 0) {
|
|
|
|
puts(", wireless");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts(", wired");
|
|
|
|
}
|
|
|
|
|
|
|
|
printf(" ");
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_PRELOADING, &enable_val,
|
2016-01-25 00:17:49 +01:00
|
|
|
sizeof(netopt_enable_t));
|
|
|
|
if ((res > 0) && (enable_val == NETOPT_ENABLE)) {
|
|
|
|
printf(" PRELOAD");
|
|
|
|
}
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_AUTOACK, &enable_val,
|
2016-01-25 00:17:49 +01:00
|
|
|
sizeof(netopt_enable_t));
|
|
|
|
if ((res > 0) && (enable_val == NETOPT_ENABLE)) {
|
|
|
|
printf(" AUTOACK");
|
|
|
|
}
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_RAWMODE, &enable_val,
|
2016-01-25 00:17:49 +01:00
|
|
|
sizeof(netopt_enable_t));
|
|
|
|
if ((res > 0) && (enable_val == NETOPT_ENABLE)) {
|
|
|
|
printf(" RAW");
|
|
|
|
}
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_AUTOCCA, &enable_val,
|
2016-01-25 00:17:49 +01:00
|
|
|
sizeof(netopt_enable_t));
|
|
|
|
if ((res > 0) && (enable_val == NETOPT_ENABLE)) {
|
|
|
|
printf(" AUTOCCA");
|
|
|
|
}
|
2021-06-21 16:32:16 +02:00
|
|
|
res = get(&dev->netdev, NETOPT_CSMA, &enable_val,
|
2016-01-25 00:17:49 +01:00
|
|
|
sizeof(netopt_enable_t));
|
|
|
|
if ((res > 0) && (enable_val == NETOPT_ENABLE)) {
|
|
|
|
printf(" CSMA");
|
|
|
|
}
|
|
|
|
puts("");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ifconfig(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
2017-10-26 00:26:48 +02:00
|
|
|
for (unsigned int i = 0; i < AT86RF2XX_NUM; i++) {
|
2016-01-25 00:17:49 +01:00
|
|
|
ifconfig_list(i);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void txtsnd_usage(char *cmd_name)
|
|
|
|
{
|
|
|
|
printf("usage: %s <iface> [<pan>] <addr> <text>\n", cmd_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
int txtsnd(int argc, char **argv)
|
|
|
|
{
|
|
|
|
char *text;
|
|
|
|
uint8_t addr[_MAX_ADDR_LEN];
|
2017-10-26 00:26:48 +02:00
|
|
|
int iface, idx = 2;
|
|
|
|
size_t res;
|
2016-01-25 00:17:49 +01:00
|
|
|
le_uint16_t pan = { 0 };
|
|
|
|
|
|
|
|
switch (argc) {
|
|
|
|
case 4:
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
res = _parse_addr((uint8_t *)&pan, sizeof(pan), argv[idx++]);
|
2017-10-26 00:26:48 +02:00
|
|
|
if ((res == 0) || (res > sizeof(pan))) {
|
2016-01-25 00:17:49 +01:00
|
|
|
txtsnd_usage(argv[0]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
pan.u16 = byteorder_swaps(pan.u16);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
txtsnd_usage(argv[0]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
iface = atoi(argv[1]);
|
|
|
|
res = _parse_addr(addr, sizeof(addr), argv[idx++]);
|
2017-10-26 00:26:48 +02:00
|
|
|
if (res == 0) {
|
2016-01-25 00:17:49 +01:00
|
|
|
txtsnd_usage(argv[0]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
text = argv[idx++];
|
2017-10-26 00:26:48 +02:00
|
|
|
return send(iface, pan, addr, res, text);
|
2016-01-25 00:17:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline int _dehex(char c, int default_)
|
|
|
|
{
|
|
|
|
if ('0' <= c && c <= '9') {
|
|
|
|
return c - '0';
|
|
|
|
}
|
|
|
|
else if ('A' <= c && c <= 'F') {
|
|
|
|
return c - 'A' + 10;
|
|
|
|
}
|
|
|
|
else if ('a' <= c && c <= 'f') {
|
|
|
|
return c - 'a' + 10;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return default_;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-26 00:26:48 +02:00
|
|
|
static size_t _parse_addr(uint8_t *out, size_t out_len, const char *in)
|
2016-01-25 00:17:49 +01:00
|
|
|
{
|
|
|
|
const char *end_str = in;
|
|
|
|
uint8_t *out_end = out;
|
|
|
|
size_t count = 0;
|
|
|
|
int assert_cell = 1;
|
|
|
|
|
|
|
|
if (!in || !*in) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
while (end_str[1]) {
|
|
|
|
++end_str;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (end_str >= in) {
|
|
|
|
int a = 0, b = _dehex(*end_str--, -1);
|
|
|
|
if (b < 0) {
|
|
|
|
if (assert_cell) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
assert_cell = 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert_cell = 0;
|
|
|
|
|
|
|
|
if (end_str >= in) {
|
|
|
|
a = _dehex(*end_str--, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (++count > out_len) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
*out_end++ = (a << 4) | b;
|
|
|
|
}
|
|
|
|
if (assert_cell) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* out is reversed */
|
|
|
|
|
|
|
|
while (out < --out_end) {
|
|
|
|
uint8_t tmp = *out_end;
|
|
|
|
*out_end = *out;
|
|
|
|
*out++ = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int send(int iface, le_uint16_t dst_pan, uint8_t *dst, size_t dst_len,
|
|
|
|
char *data)
|
|
|
|
{
|
|
|
|
int res;
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_ieee802154_t *dev;
|
2016-01-25 00:17:49 +01:00
|
|
|
uint8_t *src;
|
|
|
|
size_t src_len;
|
|
|
|
uint8_t mhr[IEEE802154_MAX_HDR_LEN];
|
|
|
|
uint8_t flags;
|
|
|
|
le_uint16_t src_pan;
|
|
|
|
|
|
|
|
if (((unsigned)iface) > (AT86RF2XX_NUM - 1)) {
|
|
|
|
printf("txtsnd: %d is not an interface\n", iface);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
iolist_t iol_data = {
|
|
|
|
.iol_base = data,
|
|
|
|
.iol_len = strlen(data)
|
|
|
|
};
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
dev = &devs[iface].netdev;
|
2017-02-15 13:07:34 +01:00
|
|
|
flags = (uint8_t)(dev->flags & NETDEV_IEEE802154_SEND_MASK);
|
2016-01-25 00:17:49 +01:00
|
|
|
flags |= IEEE802154_FCF_TYPE_DATA;
|
|
|
|
src_pan = byteorder_btols(byteorder_htons(dev->pan));
|
|
|
|
if (dst_pan.u16 == 0) {
|
|
|
|
dst_pan = src_pan;
|
|
|
|
}
|
2017-02-15 13:07:34 +01:00
|
|
|
if (dev->flags & NETDEV_IEEE802154_SRC_MODE_LONG) {
|
2016-01-25 00:17:49 +01:00
|
|
|
src_len = 8;
|
|
|
|
src = dev->long_addr;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
src_len = 2;
|
|
|
|
src = dev->short_addr;
|
|
|
|
}
|
|
|
|
/* fill MAC header, seq should be set by device */
|
|
|
|
if ((res = ieee802154_set_frame_hdr(mhr, src, src_len,
|
|
|
|
dst, dst_len,
|
|
|
|
src_pan, dst_pan,
|
|
|
|
flags, dev->seq++)) < 0) {
|
|
|
|
puts("txtsnd: Error preperaring frame");
|
|
|
|
return 1;
|
|
|
|
}
|
2018-01-12 00:19:03 +01:00
|
|
|
|
|
|
|
iolist_t iol_hdr = {
|
|
|
|
.iol_next = &iol_data,
|
|
|
|
.iol_base = mhr,
|
|
|
|
.iol_len = (size_t)res
|
|
|
|
};
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
res = dev->netdev.driver->send(&dev->netdev, &iol_hdr);
|
2016-01-25 00:17:49 +01:00
|
|
|
if (res < 0) {
|
|
|
|
puts("txtsnd: Error on sending");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
2018-01-12 00:19:03 +01:00
|
|
|
printf("txtsnd: send %u bytes to ", (unsigned)iol_data.iol_len);
|
2016-01-25 00:17:49 +01:00
|
|
|
print_addr(dst, dst_len);
|
|
|
|
printf(" (PAN: ");
|
|
|
|
print_addr((uint8_t *)&dst_pan, sizeof(dst_pan));
|
|
|
|
puts(")");
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-07-04 13:55:31 +02:00
|
|
|
#if AT86RF2XX_RANDOM_NUMBER_GENERATOR
|
2022-07-01 00:13:13 +02:00
|
|
|
void random_net_api(uint8_t idx, uint32_t *value)
|
|
|
|
{
|
|
|
|
netdev_ieee802154_t *dev = &devs[idx].netdev;
|
2022-07-04 13:55:31 +02:00
|
|
|
int retval = dev->netdev.driver->get(&dev->netdev, NETOPT_RANDOM,
|
|
|
|
value, sizeof(uint32_t));
|
|
|
|
if (retval < 0) {
|
|
|
|
printf("get(NETOPT_RANDOM) failed: %s\n", strerror(-retval));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
expect(retval == sizeof(*value));
|
|
|
|
}
|
2022-07-01 00:13:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int random_by_at86rf2xx(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
for (unsigned int i = 0; i < AT86RF2XX_NUM; i++) {
|
|
|
|
uint32_t test = 0;
|
|
|
|
at86rf2xx_get_random(&devs[i], (uint8_t *)&test, sizeof(test));
|
|
|
|
printf("Random number for device %u via native API: %" PRIx32 "\n", i, test);
|
|
|
|
test = 0;
|
|
|
|
random_net_api(i, &test);
|
|
|
|
printf("Random number for device %u via netopt: %" PRIx32 "\n", i, test);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
2016-01-25 00:17:49 +01:00
|
|
|
/** @} */
|