2015-04-23 18:36:55 +02: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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2015-08-09 21:24:55 +02:00
|
|
|
* @ingroup drivers_at86rf2xx
|
2015-04-23 18:36:55 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Netdev adaption for the AT86RF2xx drivers
|
|
|
|
*
|
|
|
|
* @author Thomas Eichinger <thomas.eichinger@fu-berlin.de>
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
2015-10-22 23:01:26 +02:00
|
|
|
* @author Kévin Roussel <Kevin.Roussel@inria.fr>
|
2016-03-18 10:26:33 +01:00
|
|
|
* @author Martine Lenders <mlenders@inf.fu-berlin.de>
|
2015-04-23 18:36:55 +02:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2016-02-11 23:06:07 +01:00
|
|
|
#include <string.h>
|
2016-03-18 10:26:33 +01:00
|
|
|
#include <assert.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
2015-06-03 17:08:03 +02:00
|
|
|
#include "net/eui64.h"
|
2015-08-07 14:36:04 +02:00
|
|
|
#include "net/ieee802154.h"
|
2017-02-15 13:07:34 +01:00
|
|
|
#include "net/netdev.h"
|
|
|
|
#include "net/netdev/ieee802154.h"
|
2016-03-18 10:26:33 +01:00
|
|
|
|
2015-08-09 21:24:55 +02:00
|
|
|
#include "at86rf2xx.h"
|
|
|
|
#include "at86rf2xx_netdev.h"
|
|
|
|
#include "at86rf2xx_internal.h"
|
|
|
|
#include "at86rf2xx_registers.h"
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2015-07-22 20:24:47 +02:00
|
|
|
#define _MAX_MHR_OVERHEAD (25)
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _send(netdev_t *netdev, const struct iovec *vector, unsigned count);
|
|
|
|
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info);
|
|
|
|
static int _init(netdev_t *netdev);
|
|
|
|
static void _isr(netdev_t *netdev);
|
|
|
|
static int _get(netdev_t *netdev, netopt_t opt, void *val, size_t max_len);
|
|
|
|
static int _set(netdev_t *netdev, netopt_t opt, void *val, size_t len);
|
2016-03-18 10:26:33 +01:00
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
const netdev_driver_t at86rf2xx_driver = {
|
2016-03-18 10:26:33 +01:00
|
|
|
.send = _send,
|
|
|
|
.recv = _recv,
|
|
|
|
.init = _init,
|
|
|
|
.isr = _isr,
|
|
|
|
.get = _get,
|
|
|
|
.set = _set,
|
|
|
|
};
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
static void _irq_handler(void *arg)
|
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_t *dev = (netdev_t *) arg;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
if (dev->event_callback) {
|
2017-02-15 13:07:34 +01:00
|
|
|
dev->event_callback(dev, NETDEV_EVENT_ISR);
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
2016-03-18 10:26:33 +01:00
|
|
|
}
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _init(netdev_t *netdev)
|
2016-03-18 10:26:33 +01:00
|
|
|
{
|
|
|
|
at86rf2xx_t *dev = (at86rf2xx_t *)netdev;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
/* initialise GPIOs */
|
2016-11-08 18:39:59 +01:00
|
|
|
spi_init_cs(dev->params.spi, dev->params.cs_pin);
|
2016-03-27 21:36:00 +02:00
|
|
|
gpio_init(dev->params.sleep_pin, GPIO_OUT);
|
|
|
|
gpio_clear(dev->params.sleep_pin);
|
|
|
|
gpio_init(dev->params.reset_pin, GPIO_OUT);
|
|
|
|
gpio_set(dev->params.reset_pin);
|
|
|
|
gpio_init_int(dev->params.int_pin, GPIO_IN, GPIO_RISING, _irq_handler, dev);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2017-07-07 12:48:46 +02:00
|
|
|
/* reset device to default values and put it into RX state */
|
|
|
|
at86rf2xx_reset(dev);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
/* test if the SPI is set up correctly and the device is responding */
|
|
|
|
if (at86rf2xx_reg_read(dev, AT86RF2XX_REG__PART_NUM) !=
|
|
|
|
AT86RF2XX_PARTNUM) {
|
|
|
|
DEBUG("[at86rf2xx] error: unable to read correct part number\n");
|
|
|
|
return -1;
|
|
|
|
}
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2016-02-11 23:06:07 +01:00
|
|
|
#ifdef MODULE_NETSTATS_L2
|
|
|
|
memset(&netdev->stats, 0, sizeof(netstats_t));
|
|
|
|
#endif
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _send(netdev_t *netdev, const struct iovec *vector, unsigned count)
|
2015-04-23 18:36:55 +02:00
|
|
|
{
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_t *dev = (at86rf2xx_t *)netdev;
|
2016-03-18 10:26:33 +01:00
|
|
|
const struct iovec *ptr = vector;
|
|
|
|
size_t len = 0;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_tx_prepare(dev);
|
2016-03-18 10:26:33 +01:00
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
/* load packet data into FIFO */
|
2016-08-02 16:56:15 +02:00
|
|
|
for (unsigned i = 0; i < count; i++, ptr++) {
|
2016-03-18 10:26:33 +01:00
|
|
|
/* current packet data + FCS too long */
|
|
|
|
if ((len + ptr->iov_len + 2) > AT86RF2XX_MAX_PKT_LENGTH) {
|
2016-04-11 02:19:33 +02:00
|
|
|
DEBUG("[at86rf2xx] error: packet too large (%u byte) to be send\n",
|
|
|
|
(unsigned)len + 2);
|
2016-03-18 10:26:33 +01:00
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
2016-02-11 23:06:07 +01:00
|
|
|
#ifdef MODULE_NETSTATS_L2
|
|
|
|
netdev->stats.tx_bytes += len;
|
|
|
|
#endif
|
2016-03-18 10:26:33 +01:00
|
|
|
len = at86rf2xx_tx_load(dev, ptr->iov_base, ptr->iov_len, len);
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
2016-03-18 10:26:33 +01:00
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
/* send data out directly if pre-loading id disabled */
|
2016-03-18 10:26:33 +01:00
|
|
|
if (!(dev->netdev.flags & AT86RF2XX_OPT_PRELOADING)) {
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_tx_exec(dev);
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
/* return the number of bytes that were actually send out */
|
|
|
|
return (int)len;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info)
|
2015-04-23 18:36:55 +02:00
|
|
|
{
|
2016-03-18 10:26:33 +01:00
|
|
|
at86rf2xx_t *dev = (at86rf2xx_t *)netdev;
|
2015-09-30 17:46:13 +02:00
|
|
|
uint8_t phr;
|
2016-03-18 10:26:33 +01:00
|
|
|
size_t pkt_len;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-09-30 17:46:13 +02:00
|
|
|
/* frame buffer protection will be unlocked as soon as at86rf2xx_fb_stop()
|
|
|
|
* is called*/
|
|
|
|
at86rf2xx_fb_start(dev);
|
|
|
|
|
|
|
|
/* get the size of the received packet */
|
|
|
|
at86rf2xx_fb_read(dev, &phr, 1);
|
|
|
|
|
2016-04-02 17:29:22 +02:00
|
|
|
/* ignore MSB (refer p.80) and substract length of FCS field */
|
|
|
|
pkt_len = (phr & 0x7f) - 2;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
/* just return length when buf == NULL */
|
|
|
|
if (buf == NULL) {
|
2015-09-30 17:46:13 +02:00
|
|
|
at86rf2xx_fb_stop(dev);
|
2016-03-18 10:26:33 +01:00
|
|
|
return pkt_len;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
2016-03-18 10:26:33 +01:00
|
|
|
/* not enough space in buf */
|
|
|
|
if (pkt_len > len) {
|
2015-09-30 17:46:13 +02:00
|
|
|
at86rf2xx_fb_stop(dev);
|
2016-03-18 10:26:33 +01:00
|
|
|
return -ENOBUFS;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
2016-10-21 21:57:32 +02:00
|
|
|
#ifdef MODULE_NETSTATS_L2
|
|
|
|
netdev->stats.rx_count++;
|
|
|
|
netdev->stats.rx_bytes += pkt_len;
|
|
|
|
#endif
|
2015-04-23 18:36:55 +02:00
|
|
|
/* copy payload */
|
2016-03-18 10:26:33 +01:00
|
|
|
at86rf2xx_fb_read(dev, (uint8_t *)buf, pkt_len);
|
2015-09-30 17:46:13 +02:00
|
|
|
|
2017-01-09 17:47:37 +01:00
|
|
|
/* Ignore FCS but advance fb read - we must give a temporary buffer here,
|
|
|
|
* as we are not allowed to issue SPI transfers without any buffer */
|
|
|
|
uint8_t tmp[2];
|
|
|
|
at86rf2xx_fb_read(dev, tmp, 2);
|
|
|
|
(void)tmp;
|
2015-09-30 17:46:13 +02:00
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
if (info != NULL) {
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_ieee802154_rx_info_t *radio_info = info;
|
2016-03-18 10:26:33 +01:00
|
|
|
at86rf2xx_fb_read(dev, &(radio_info->lqi), 1);
|
2015-09-30 17:46:13 +02:00
|
|
|
#ifndef MODULE_AT86RF231
|
2016-03-18 10:26:33 +01:00
|
|
|
at86rf2xx_fb_read(dev, &(radio_info->rssi), 1);
|
|
|
|
at86rf2xx_fb_stop(dev);
|
2015-09-30 17:46:13 +02:00
|
|
|
#else
|
2016-03-18 10:26:33 +01:00
|
|
|
at86rf2xx_fb_stop(dev);
|
|
|
|
radio_info->rssi = at86rf2xx_reg_read(dev, AT86RF2XX_REG__PHY_ED_LEVEL);
|
2015-09-30 17:46:13 +02:00
|
|
|
#endif
|
2016-03-18 10:26:33 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
at86rf2xx_fb_stop(dev);
|
|
|
|
}
|
2015-09-30 17:46:13 +02:00
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
return pkt_len;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
|
2015-08-09 21:24:55 +02:00
|
|
|
static int _set_state(at86rf2xx_t *dev, netopt_state_t state)
|
2015-04-23 18:36:55 +02:00
|
|
|
{
|
|
|
|
switch (state) {
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_STATE_SLEEP:
|
2015-09-16 18:48:10 +02:00
|
|
|
at86rf2xx_set_state(dev, AT86RF2XX_STATE_SLEEP);
|
2015-04-23 18:36:55 +02:00
|
|
|
break;
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_STATE_IDLE:
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_set_state(dev, AT86RF2XX_STATE_RX_AACK_ON);
|
2015-04-23 18:36:55 +02:00
|
|
|
break;
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_STATE_TX:
|
2016-03-18 10:26:33 +01:00
|
|
|
if (dev->netdev.flags & AT86RF2XX_OPT_PRELOADING) {
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_tx_exec(dev);
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
break;
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_STATE_RESET:
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_reset(dev);
|
2015-04-23 18:36:55 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
2015-08-06 15:36:56 +02:00
|
|
|
return sizeof(netopt_state_t);
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
|
2015-08-09 21:24:55 +02:00
|
|
|
netopt_state_t _get_state(at86rf2xx_t *dev)
|
2015-04-23 18:36:55 +02:00
|
|
|
{
|
2015-08-09 21:24:55 +02:00
|
|
|
switch (at86rf2xx_get_status(dev)) {
|
|
|
|
case AT86RF2XX_STATE_SLEEP:
|
2015-08-06 15:36:56 +02:00
|
|
|
return NETOPT_STATE_SLEEP;
|
2015-08-09 21:24:55 +02:00
|
|
|
case AT86RF2XX_STATE_BUSY_RX_AACK:
|
2015-08-06 15:36:56 +02:00
|
|
|
return NETOPT_STATE_RX;
|
2015-08-09 21:24:55 +02:00
|
|
|
case AT86RF2XX_STATE_BUSY_TX_ARET:
|
|
|
|
case AT86RF2XX_STATE_TX_ARET_ON:
|
2015-08-06 15:36:56 +02:00
|
|
|
return NETOPT_STATE_TX;
|
2015-08-09 21:24:55 +02:00
|
|
|
case AT86RF2XX_STATE_RX_AACK_ON:
|
2015-04-23 18:36:55 +02:00
|
|
|
default:
|
2015-08-06 15:36:56 +02:00
|
|
|
return NETOPT_STATE_IDLE;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _get(netdev_t *netdev, netopt_t opt, void *val, size_t max_len)
|
2015-04-23 18:36:55 +02:00
|
|
|
{
|
2016-03-18 10:26:33 +01:00
|
|
|
at86rf2xx_t *dev = (at86rf2xx_t *) netdev;
|
2015-09-16 18:48:10 +02:00
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
if (netdev == NULL) {
|
2015-04-23 18:36:55 +02:00
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
2015-09-16 18:48:10 +02:00
|
|
|
/* getting these options doesn't require the transceiver to be responsive */
|
2015-04-23 18:36:55 +02:00
|
|
|
switch (opt) {
|
2015-09-16 18:25:36 +02:00
|
|
|
case NETOPT_CHANNEL_PAGE:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(max_len >= sizeof(uint16_t));
|
2015-09-16 18:25:36 +02:00
|
|
|
((uint8_t *)val)[1] = 0;
|
|
|
|
((uint8_t *)val)[0] = at86rf2xx_get_page(dev);
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_MAX_PACKET_SIZE:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(max_len >= sizeof(int16_t));
|
2015-08-09 21:24:55 +02:00
|
|
|
*((uint16_t *)val) = AT86RF2XX_MAX_PKT_LENGTH - _MAX_MHR_OVERHEAD;
|
2015-04-23 18:36:55 +02:00
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_STATE:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(max_len >= sizeof(netopt_state_t));
|
2016-03-18 10:26:33 +01:00
|
|
|
*((netopt_state_t *)val) = _get_state(dev);
|
2015-09-16 18:48:10 +02:00
|
|
|
return sizeof(netopt_state_t);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_PRELOADING:
|
2016-03-18 10:26:33 +01:00
|
|
|
if (dev->netdev.flags & AT86RF2XX_OPT_PRELOADING) {
|
2015-08-06 15:36:56 +02:00
|
|
|
*((netopt_enable_t *)val) = NETOPT_ENABLE;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
else {
|
2015-08-06 15:36:56 +02:00
|
|
|
*((netopt_enable_t *)val) = NETOPT_DISABLE;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
2015-08-06 15:36:56 +02:00
|
|
|
return sizeof(netopt_enable_t);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_PROMISCUOUSMODE:
|
2016-03-18 10:26:33 +01:00
|
|
|
if (dev->netdev.flags & AT86RF2XX_OPT_PROMISCUOUS) {
|
2015-08-06 15:36:56 +02:00
|
|
|
*((netopt_enable_t *)val) = NETOPT_ENABLE;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
else {
|
2015-08-06 15:36:56 +02:00
|
|
|
*((netopt_enable_t *)val) = NETOPT_DISABLE;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
2015-08-06 15:36:56 +02:00
|
|
|
return sizeof(netopt_enable_t);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_RX_START_IRQ:
|
|
|
|
*((netopt_enable_t *)val) =
|
2016-03-18 10:26:33 +01:00
|
|
|
!!(dev->netdev.flags & AT86RF2XX_OPT_TELL_RX_START);
|
2015-08-06 15:36:56 +02:00
|
|
|
return sizeof(netopt_enable_t);
|
2015-05-05 22:50:14 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_RX_END_IRQ:
|
|
|
|
*((netopt_enable_t *)val) =
|
2016-03-18 10:26:33 +01:00
|
|
|
!!(dev->netdev.flags & AT86RF2XX_OPT_TELL_RX_END);
|
2015-08-06 15:36:56 +02:00
|
|
|
return sizeof(netopt_enable_t);
|
2015-05-05 22:50:14 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_TX_START_IRQ:
|
|
|
|
*((netopt_enable_t *)val) =
|
2016-03-18 10:26:33 +01:00
|
|
|
!!(dev->netdev.flags & AT86RF2XX_OPT_TELL_TX_START);
|
2015-08-06 15:36:56 +02:00
|
|
|
return sizeof(netopt_enable_t);
|
2015-05-05 22:50:14 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_TX_END_IRQ:
|
|
|
|
*((netopt_enable_t *)val) =
|
2016-03-18 10:26:33 +01:00
|
|
|
!!(dev->netdev.flags & AT86RF2XX_OPT_TELL_TX_END);
|
2015-08-06 15:36:56 +02:00
|
|
|
return sizeof(netopt_enable_t);
|
2015-05-05 22:49:04 +02:00
|
|
|
|
2015-06-01 15:51:50 +02:00
|
|
|
case NETOPT_CSMA:
|
|
|
|
*((netopt_enable_t *)val) =
|
2016-03-18 10:26:33 +01:00
|
|
|
!!(dev->netdev.flags & AT86RF2XX_OPT_CSMA);
|
2015-06-01 15:51:50 +02:00
|
|
|
return sizeof(netopt_enable_t);
|
|
|
|
|
2015-09-16 18:48:10 +02:00
|
|
|
default:
|
|
|
|
/* Can still be handled in second switch */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
int res;
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
if (((res = netdev_ieee802154_get((netdev_ieee802154_t *)netdev, opt, val,
|
2016-03-18 10:26:33 +01:00
|
|
|
max_len)) >= 0) || (res != -ENOTSUP)) {
|
|
|
|
return res;
|
|
|
|
}
|
2015-09-16 18:48:10 +02:00
|
|
|
|
|
|
|
uint8_t old_state = at86rf2xx_get_status(dev);
|
|
|
|
|
|
|
|
/* temporarily wake up if sleeping */
|
2016-03-18 10:26:33 +01:00
|
|
|
if (old_state == AT86RF2XX_STATE_SLEEP) {
|
2015-09-16 18:48:10 +02:00
|
|
|
at86rf2xx_assert_awake(dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* these options require the transceiver to be not sleeping*/
|
|
|
|
switch (opt) {
|
|
|
|
case NETOPT_TX_POWER:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(max_len >= sizeof(int16_t));
|
|
|
|
*((uint16_t *)val) = at86rf2xx_get_txpower(dev);
|
|
|
|
res = sizeof(uint16_t);
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NETOPT_RETRANS:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(max_len >= sizeof(uint8_t));
|
|
|
|
*((uint8_t *)val) = at86rf2xx_get_max_retries(dev);
|
|
|
|
res = sizeof(uint8_t);
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
|
|
|
|
2015-06-01 15:51:50 +02:00
|
|
|
case NETOPT_CSMA_RETRIES:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(max_len >= sizeof(uint8_t));
|
|
|
|
*((uint8_t *)val) = at86rf2xx_get_csma_max_retries(dev);
|
|
|
|
res = sizeof(uint8_t);
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
2015-06-01 15:51:50 +02:00
|
|
|
|
2015-10-22 23:01:26 +02:00
|
|
|
case NETOPT_CCA_THRESHOLD:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(max_len >= sizeof(int8_t));
|
|
|
|
*((int8_t *)val) = at86rf2xx_get_cca_threshold(dev);
|
|
|
|
res = sizeof(int8_t);
|
2015-10-22 23:01:26 +02:00
|
|
|
break;
|
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
default:
|
2015-09-16 18:48:10 +02:00
|
|
|
res = -ENOTSUP;
|
2017-03-15 16:14:09 +01:00
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
|
2015-09-16 18:48:10 +02:00
|
|
|
/* go back to sleep if were sleeping */
|
2016-03-18 10:26:33 +01:00
|
|
|
if (old_state == AT86RF2XX_STATE_SLEEP) {
|
2015-09-16 18:48:10 +02:00
|
|
|
at86rf2xx_set_state(dev, AT86RF2XX_STATE_SLEEP);
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _set(netdev_t *netdev, netopt_t opt, void *val, size_t len)
|
2015-04-23 18:36:55 +02:00
|
|
|
{
|
2016-03-18 10:26:33 +01:00
|
|
|
at86rf2xx_t *dev = (at86rf2xx_t *) netdev;
|
2015-09-16 18:48:10 +02:00
|
|
|
uint8_t old_state = at86rf2xx_get_status(dev);
|
2016-04-01 14:04:08 +02:00
|
|
|
int res = -ENOTSUP;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
|
|
if (dev == NULL) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
2017-06-29 21:30:21 +02:00
|
|
|
/* temporarily wake up if sleeping and opt != NETOPT_STATE.
|
|
|
|
* opt != NETOPT_STATE check prevents redundant wake-up.
|
|
|
|
* when opt == NETOPT_STATE, at86rf2xx_set_state() will wake up the
|
|
|
|
* radio if needed. */
|
|
|
|
if ((old_state == AT86RF2XX_STATE_SLEEP) && (opt != NETOPT_STATE)) {
|
2015-09-16 18:48:10 +02:00
|
|
|
at86rf2xx_assert_awake(dev);
|
|
|
|
}
|
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
switch (opt) {
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_ADDRESS:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(len <= sizeof(uint16_t));
|
|
|
|
at86rf2xx_set_addr_short(dev, *((uint16_t *)val));
|
2017-02-15 13:07:34 +01:00
|
|
|
/* don't set res to set netdev_ieee802154_t::short_addr */
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_ADDRESS_LONG:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(len <= sizeof(uint64_t));
|
|
|
|
at86rf2xx_set_addr_long(dev, *((uint64_t *)val));
|
2017-02-15 13:07:34 +01:00
|
|
|
/* don't set res to set netdev_ieee802154_t::long_addr */
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_NID:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(len <= sizeof(uint16_t));
|
|
|
|
at86rf2xx_set_pan(dev, *((uint16_t *)val));
|
2017-02-15 13:07:34 +01:00
|
|
|
/* don't set res to set netdev_ieee802154_t::pan */
|
2015-10-31 09:29:47 +01:00
|
|
|
break;
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_CHANNEL:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(len != sizeof(uint8_t));
|
|
|
|
uint8_t chan = ((uint8_t *)val)[0];
|
|
|
|
if (chan < AT86RF2XX_MIN_CHANNEL ||
|
|
|
|
chan > AT86RF2XX_MAX_CHANNEL) {
|
2015-09-16 18:48:10 +02:00
|
|
|
res = -EINVAL;
|
2015-12-06 15:09:00 +01:00
|
|
|
break;
|
2016-03-18 10:26:33 +01:00
|
|
|
}
|
2015-12-06 15:09:00 +01:00
|
|
|
at86rf2xx_set_chan(dev, chan);
|
2017-02-15 13:07:34 +01:00
|
|
|
/* don't set res to set netdev_ieee802154_t::chan */
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-09-16 18:25:36 +02:00
|
|
|
case NETOPT_CHANNEL_PAGE:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(len != sizeof(uint8_t));
|
|
|
|
uint8_t page = ((uint8_t *)val)[0];
|
|
|
|
#ifdef MODULE_AT86RF212B
|
|
|
|
if ((page != 0) && (page != 2)) {
|
2015-09-16 18:25:36 +02:00
|
|
|
res = -EINVAL;
|
2016-03-18 10:26:33 +01:00
|
|
|
}
|
|
|
|
else {
|
2015-12-06 15:09:00 +01:00
|
|
|
at86rf2xx_set_page(dev, page);
|
|
|
|
res = sizeof(uint8_t);
|
|
|
|
}
|
2015-09-16 18:25:36 +02:00
|
|
|
#else
|
2015-12-06 15:09:00 +01:00
|
|
|
/* rf23x only supports page 0, no need to configure anything in the driver. */
|
|
|
|
if (page != 0) {
|
|
|
|
res = -EINVAL;
|
2015-09-16 18:25:36 +02:00
|
|
|
}
|
2015-12-06 15:09:00 +01:00
|
|
|
else {
|
|
|
|
res = sizeof(uint8_t);
|
|
|
|
}
|
|
|
|
#endif
|
2015-09-16 18:25:36 +02:00
|
|
|
break;
|
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_TX_POWER:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(len <= sizeof(int16_t));
|
|
|
|
at86rf2xx_set_txpower(dev, *((int16_t *)val));
|
|
|
|
res = sizeof(uint16_t);
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_STATE:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(len <= sizeof(netopt_state_t));
|
|
|
|
res = _set_state(dev, *((netopt_state_t *)val));
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_AUTOACK:
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_AUTOACK,
|
|
|
|
((bool *)val)[0]);
|
2017-02-15 13:07:34 +01:00
|
|
|
/* don't set res to set netdev_ieee802154_t::flags */
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_RETRANS:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(len <= sizeof(uint8_t));
|
|
|
|
at86rf2xx_set_max_retries(dev, *((uint8_t *)val));
|
|
|
|
res = sizeof(uint8_t);
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
2015-05-18 21:19:03 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_PRELOADING:
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_PRELOADING,
|
|
|
|
((bool *)val)[0]);
|
2015-09-16 18:48:10 +02:00
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_PROMISCUOUSMODE:
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_PROMISCUOUS,
|
|
|
|
((bool *)val)[0]);
|
2015-09-16 18:48:10 +02:00
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_RX_START_IRQ:
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_TELL_RX_START,
|
|
|
|
((bool *)val)[0]);
|
2015-09-16 18:48:10 +02:00
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
break;
|
2015-05-05 22:50:14 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_RX_END_IRQ:
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_TELL_RX_END,
|
|
|
|
((bool *)val)[0]);
|
2015-09-16 18:48:10 +02:00
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
break;
|
2015-05-05 22:50:14 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_TX_START_IRQ:
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_TELL_TX_START,
|
|
|
|
((bool *)val)[0]);
|
2015-09-16 18:48:10 +02:00
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
break;
|
2015-05-05 22:50:14 +02:00
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
case NETOPT_TX_END_IRQ:
|
2015-08-09 21:24:55 +02:00
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_TELL_TX_END,
|
|
|
|
((bool *)val)[0]);
|
2015-09-16 18:48:10 +02:00
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
break;
|
2015-05-05 22:50:14 +02:00
|
|
|
|
2015-06-01 15:51:50 +02:00
|
|
|
case NETOPT_CSMA:
|
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_CSMA,
|
2016-03-18 10:26:33 +01:00
|
|
|
((bool *)val)[0]);
|
2015-09-16 18:48:10 +02:00
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
break;
|
2015-06-01 15:51:50 +02:00
|
|
|
|
|
|
|
case NETOPT_CSMA_RETRIES:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(len <= sizeof(uint8_t));
|
|
|
|
if( !(dev->netdev.flags & AT86RF2XX_OPT_CSMA ||
|
|
|
|
(*((uint8_t *)val) > 5)) ) {
|
|
|
|
/* If CSMA is disabled, don't allow setting retries */
|
|
|
|
res = -EINVAL;
|
2016-03-18 10:26:33 +01:00
|
|
|
}
|
2015-12-06 15:09:00 +01:00
|
|
|
else {
|
2015-09-16 18:48:10 +02:00
|
|
|
at86rf2xx_set_csma_max_retries(dev, *((uint8_t *)val));
|
|
|
|
res = sizeof(uint8_t);
|
2015-06-01 15:51:50 +02:00
|
|
|
}
|
2015-09-16 18:48:10 +02:00
|
|
|
break;
|
2015-06-01 15:51:50 +02:00
|
|
|
|
2015-10-22 23:01:26 +02:00
|
|
|
case NETOPT_CCA_THRESHOLD:
|
2015-12-06 15:09:00 +01:00
|
|
|
assert(len <= sizeof(int8_t));
|
|
|
|
at86rf2xx_set_cca_threshold(dev, *((int8_t *)val));
|
|
|
|
res = sizeof(int8_t);
|
2015-10-22 23:01:26 +02:00
|
|
|
break;
|
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
default:
|
2016-04-01 14:04:08 +02:00
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
|
2015-09-16 18:48:10 +02:00
|
|
|
/* go back to sleep if were sleeping and state hasn't been changed */
|
2016-03-18 10:26:33 +01:00
|
|
|
if ((old_state == AT86RF2XX_STATE_SLEEP) &&
|
|
|
|
(opt != NETOPT_STATE)) {
|
2015-09-16 18:48:10 +02:00
|
|
|
at86rf2xx_set_state(dev, AT86RF2XX_STATE_SLEEP);
|
|
|
|
}
|
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
if (res == -ENOTSUP) {
|
2017-02-15 13:07:34 +01:00
|
|
|
res = netdev_ieee802154_set((netdev_ieee802154_t *)netdev, opt,
|
2016-03-18 10:26:33 +01:00
|
|
|
val, len);
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
return res;
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static void _isr(netdev_t *netdev)
|
2015-04-23 18:36:55 +02:00
|
|
|
{
|
2016-03-18 10:26:33 +01:00
|
|
|
at86rf2xx_t *dev = (at86rf2xx_t *) netdev;
|
2015-04-23 18:36:55 +02:00
|
|
|
uint8_t irq_mask;
|
|
|
|
uint8_t state;
|
2015-08-04 17:46:50 +02:00
|
|
|
uint8_t trac_status;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-09-16 18:48:10 +02:00
|
|
|
/* If transceiver is sleeping register access is impossible and frames are
|
|
|
|
* lost anyway, so return immediately.
|
|
|
|
*/
|
|
|
|
state = at86rf2xx_get_status(dev);
|
2016-03-18 10:26:33 +01:00
|
|
|
if (state == AT86RF2XX_STATE_SLEEP) {
|
2015-09-16 18:48:10 +02:00
|
|
|
return;
|
2016-03-18 10:26:33 +01:00
|
|
|
}
|
2015-09-16 18:48:10 +02:00
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
/* read (consume) device status */
|
2015-08-09 21:24:55 +02:00
|
|
|
irq_mask = at86rf2xx_reg_read(dev, AT86RF2XX_REG__IRQ_STATUS);
|
2015-07-20 11:47:15 +02:00
|
|
|
|
2015-08-04 17:46:50 +02:00
|
|
|
trac_status = at86rf2xx_reg_read(dev, AT86RF2XX_REG__TRX_STATE) &
|
2016-03-18 10:26:33 +01:00
|
|
|
AT86RF2XX_TRX_STATE_MASK__TRAC;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
2015-08-09 21:24:55 +02:00
|
|
|
if (irq_mask & AT86RF2XX_IRQ_STATUS_MASK__RX_START) {
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_RX_STARTED);
|
2015-08-09 21:24:55 +02:00
|
|
|
DEBUG("[at86rf2xx] EVT - RX_START\n");
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
2015-08-04 17:46:50 +02:00
|
|
|
|
2015-08-09 21:24:55 +02:00
|
|
|
if (irq_mask & AT86RF2XX_IRQ_STATUS_MASK__TRX_END) {
|
2016-03-18 10:26:33 +01:00
|
|
|
if (state == AT86RF2XX_STATE_RX_AACK_ON ||
|
|
|
|
state == AT86RF2XX_STATE_BUSY_RX_AACK) {
|
2015-08-09 21:24:55 +02:00
|
|
|
DEBUG("[at86rf2xx] EVT - RX_END\n");
|
2016-03-18 10:26:33 +01:00
|
|
|
if (!(dev->netdev.flags & AT86RF2XX_OPT_TELL_RX_END)) {
|
2015-04-23 18:36:55 +02:00
|
|
|
return;
|
|
|
|
}
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_RX_COMPLETE);
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
2015-08-04 17:46:50 +02:00
|
|
|
else if (state == AT86RF2XX_STATE_TX_ARET_ON ||
|
|
|
|
state == AT86RF2XX_STATE_BUSY_TX_ARET) {
|
2016-04-07 16:53:34 +02:00
|
|
|
/* check for more pending TX calls and return to idle state if
|
|
|
|
* there are none */
|
2016-04-11 02:19:06 +02:00
|
|
|
assert(dev->pending_tx != 0);
|
|
|
|
if ((--dev->pending_tx) == 0) {
|
2016-04-07 16:53:34 +02:00
|
|
|
at86rf2xx_set_state(dev, dev->idle_state);
|
2016-04-28 10:51:47 +02:00
|
|
|
DEBUG("[at86rf2xx] return to state 0x%x\n", dev->idle_state);
|
2016-04-07 16:53:34 +02:00
|
|
|
}
|
|
|
|
|
2015-08-04 17:46:50 +02:00
|
|
|
DEBUG("[at86rf2xx] EVT - TX_END\n");
|
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
if (netdev->event_callback && (dev->netdev.flags & AT86RF2XX_OPT_TELL_TX_END)) {
|
|
|
|
switch (trac_status) {
|
2017-05-19 01:02:19 +02:00
|
|
|
#ifdef MODULE_OPENTHREAD
|
|
|
|
case AT86RF2XX_TRX_STATE__TRAC_SUCCESS:
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE);
|
|
|
|
DEBUG("[at86rf2xx] TX SUCCESS\n");
|
|
|
|
break;
|
|
|
|
case AT86RF2XX_TRX_STATE__TRAC_SUCCESS_DATA_PENDING:
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE_DATA_PENDING);
|
|
|
|
DEBUG("[at86rf2xx] TX SUCCESS DATA PENDING\n");
|
|
|
|
break;
|
|
|
|
#else
|
2016-03-18 10:26:33 +01:00
|
|
|
case AT86RF2XX_TRX_STATE__TRAC_SUCCESS:
|
|
|
|
case AT86RF2XX_TRX_STATE__TRAC_SUCCESS_DATA_PENDING:
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE);
|
2016-03-18 10:26:33 +01:00
|
|
|
DEBUG("[at86rf2xx] TX SUCCESS\n");
|
|
|
|
break;
|
2017-05-19 01:02:19 +02:00
|
|
|
#endif
|
2016-03-18 10:26:33 +01:00
|
|
|
case AT86RF2XX_TRX_STATE__TRAC_NO_ACK:
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_NOACK);
|
2016-03-18 10:26:33 +01:00
|
|
|
DEBUG("[at86rf2xx] TX NO_ACK\n");
|
|
|
|
break;
|
|
|
|
case AT86RF2XX_TRX_STATE__TRAC_CHANNEL_ACCESS_FAILURE:
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_MEDIUM_BUSY);
|
2016-03-18 10:26:33 +01:00
|
|
|
DEBUG("[at86rf2xx] TX_CHANNEL_ACCESS_FAILURE\n");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
DEBUG("[at86rf2xx] Unhandled TRAC_STATUS: %d\n",
|
|
|
|
trac_status >> 5);
|
2015-08-04 17:46:50 +02:00
|
|
|
}
|
2015-04-23 18:36:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|