2015-04-23 18:36:55 +02:00
|
|
|
|
/*
|
2018-01-12 00:19:03 +01:00
|
|
|
|
* Copyright (C) 2018 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
|
* 2015 Freie Universität Berlin
|
2023-06-29 17:18:21 +02:00
|
|
|
|
* 2023 Gerson Fernando Budke
|
2015-04-23 18:36:55 +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.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**
|
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>
|
2018-01-12 00:19:03 +01:00
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
2019-10-21 01:07:44 +02:00
|
|
|
|
* @author Josua Arndt <jarndt@ias.rwth-aachen.de>
|
2023-06-29 17:18:21 +02:00
|
|
|
|
* @author Gerson Fernando Budke <nandojve@gmail.com>
|
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>
|
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
|
#include "iolist.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"
|
2022-11-28 15:57:38 +01:00
|
|
|
|
#if IS_USED(IEEE802154_SECURITY)
|
|
|
|
|
#include "net/ieee802154_security.h"
|
|
|
|
|
#endif
|
2020-10-02 19:39:43 +02:00
|
|
|
|
#if IS_USED(MODULE_AT86RF2XX_AES_SPI)
|
|
|
|
|
#include "at86rf2xx_aes.h"
|
|
|
|
|
#endif
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
|
#define ENABLE_DEBUG 0
|
2015-04-23 18:36:55 +02:00
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
|
static int _send(netdev_t *netdev, const iolist_t *iolist);
|
2017-02-15 13:07:34 +01:00
|
|
|
|
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);
|
2017-08-25 07:34:03 +02:00
|
|
|
|
static int _set(netdev_t *netdev, netopt_t opt, const 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
|
|
|
|
|
2023-01-30 18:37:58 +01:00
|
|
|
|
/* Default AT86RF2XX channel */
|
|
|
|
|
static const uint16_t at86rf2xx_chan_default = AT86RF2XX_DEFAULT_CHANNEL;
|
|
|
|
|
|
2022-11-28 15:57:38 +01:00
|
|
|
|
#if IS_USED(MODULE_AT86RF2XX_AES_SPI) && \
|
|
|
|
|
IS_USED(MODULE_IEEE802154_SECURITY)
|
|
|
|
|
/**
|
|
|
|
|
* @brief Pass the 802.15.4 encryption key to the transceiver hardware
|
|
|
|
|
*
|
|
|
|
|
* @param[in] dev Abstract security device descriptor
|
|
|
|
|
* @param[in] key Encryption key to be used
|
|
|
|
|
* @param[in] key_size Size of the encryption key in bytes
|
|
|
|
|
*/
|
|
|
|
|
static void _at86rf2xx_set_key(ieee802154_sec_dev_t *dev,
|
|
|
|
|
const uint8_t *key, uint8_t key_size)
|
|
|
|
|
{
|
|
|
|
|
(void)key_size;
|
|
|
|
|
at86rf2xx_aes_key_write_encrypt((at86rf2xx_t *)dev->ctx, key);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief Compute CBC-MAC from IEEE 802.15.4 security context
|
|
|
|
|
*
|
|
|
|
|
* @param[in] dev Abstract security device descriptor
|
|
|
|
|
* @param[out] cipher Buffer to store cipher blocks
|
|
|
|
|
* @param[in] iv Initial vector
|
|
|
|
|
* @param[in] plain Input data blocks
|
|
|
|
|
* @param[in] nblocks Number of blocks
|
|
|
|
|
*/
|
|
|
|
|
static void _at86rf2xx_cbc(const ieee802154_sec_dev_t *dev,
|
|
|
|
|
uint8_t *cipher,
|
|
|
|
|
uint8_t *iv,
|
|
|
|
|
const uint8_t *plain,
|
|
|
|
|
uint8_t nblocks)
|
|
|
|
|
{
|
|
|
|
|
at86rf2xx_aes_cbc_encrypt((at86rf2xx_t *)dev->ctx,
|
|
|
|
|
(aes_block_t *)cipher,
|
|
|
|
|
NULL,
|
|
|
|
|
iv,
|
|
|
|
|
(aes_block_t *)plain,
|
|
|
|
|
nblocks);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief Perform ECB encryption
|
|
|
|
|
*
|
|
|
|
|
* @param[in] dev Abstract security device descriptor
|
|
|
|
|
* @param[out] cipher Output cipher blocks
|
|
|
|
|
* @param[in] plain Plain blocks
|
|
|
|
|
* @param[in] nblocks Number of blocks
|
|
|
|
|
*/
|
|
|
|
|
static void _at86rf2xx_ecb(const ieee802154_sec_dev_t *dev,
|
|
|
|
|
uint8_t *cipher,
|
|
|
|
|
const uint8_t *plain,
|
|
|
|
|
uint8_t nblocks)
|
|
|
|
|
{
|
|
|
|
|
at86rf2xx_aes_ecb_encrypt((at86rf2xx_t *)dev->ctx,
|
|
|
|
|
(aes_block_t *)cipher,
|
|
|
|
|
NULL,
|
|
|
|
|
(aes_block_t *)plain,
|
|
|
|
|
nblocks);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
/**
|
|
|
|
|
* @brief Struct that contains IEEE 802.15.4 security operations
|
|
|
|
|
* which are implemented, using the transceiver´s hardware
|
|
|
|
|
* crypto capabilities
|
|
|
|
|
*/
|
|
|
|
|
static const ieee802154_radio_cipher_ops_t _at86rf2xx_cipher_ops = {
|
|
|
|
|
.set_key = _at86rf2xx_set_key,
|
|
|
|
|
.ecb = _at86rf2xx_ecb,
|
|
|
|
|
.cbc = _at86rf2xx_cbc
|
|
|
|
|
};
|
|
|
|
|
#endif /* IS_USED(MODULE_AT86RF2XX_AES_SPI) && \
|
|
|
|
|
IS_USED(MODULE_IEEE802154_SECURITY) */
|
|
|
|
|
|
2019-10-21 01:07:44 +02:00
|
|
|
|
/* SOC has radio interrupts, store reference to netdev */
|
|
|
|
|
static netdev_t *at86rfmega_dev;
|
2016-03-18 10:26:33 +01:00
|
|
|
|
static void _irq_handler(void *arg)
|
|
|
|
|
{
|
2020-03-05 15:11:20 +01:00
|
|
|
|
netdev_trigger_event_isr(arg);
|
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
|
|
|
|
{
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
2022-11-28 14:48:12 +01:00
|
|
|
|
if (AT86RF2XX_IS_PERIPH) {
|
|
|
|
|
at86rfmega_dev = netdev;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
at86rf2xx_spi_init(dev, _irq_handler);
|
2019-09-02 08:49:27 +02:00
|
|
|
|
}
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
2020-02-13 02:05:07 +01:00
|
|
|
|
/* reset hardware into a defined state */
|
|
|
|
|
at86rf2xx_hardware_reset(dev);
|
|
|
|
|
|
2019-09-02 08:49:27 +02:00
|
|
|
|
/* test if the device is responding */
|
2017-06-30 13:53:29 +02:00
|
|
|
|
if (at86rf2xx_reg_read(dev, AT86RF2XX_REG__PART_NUM) != AT86RF2XX_PARTNUM) {
|
2016-03-18 10:26:33 +01:00
|
|
|
|
DEBUG("[at86rf2xx] error: unable to read correct part number\n");
|
2019-09-02 08:49:27 +02:00
|
|
|
|
return -ENOTSUP;
|
2016-03-18 10:26:33 +01:00
|
|
|
|
}
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
2019-09-02 08:49:27 +02:00
|
|
|
|
/* reset device to default values and put it into RX state */
|
2022-11-28 15:48:23 +01:00
|
|
|
|
netdev_ieee802154_reset(&dev->netdev);
|
|
|
|
|
at86rf2xx_set_addr_long(dev, (eui64_t *)dev->netdev.long_addr);
|
|
|
|
|
at86rf2xx_set_addr_short(dev, (network_uint16_t *)dev->netdev.short_addr);
|
2023-01-30 18:37:58 +01:00
|
|
|
|
|
|
|
|
|
/* `netdev_ieee802154_reset` does not set the default channel. */
|
2023-06-30 22:41:44 +02:00
|
|
|
|
netdev_ieee802154_set(&dev->netdev, NETOPT_CHANNEL,
|
|
|
|
|
&at86rf2xx_chan_default, sizeof(at86rf2xx_chan_default));
|
2023-01-30 18:37:58 +01:00
|
|
|
|
|
2022-11-28 15:48:23 +01:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
2022-12-14 16:24:36 +01:00
|
|
|
|
static const netopt_enable_t ack_req =
|
|
|
|
|
IS_ACTIVE(CONFIG_IEEE802154_DEFAULT_ACK_REQ) ? NETOPT_ENABLE : NETOPT_DISABLE;
|
2022-11-28 15:48:23 +01:00
|
|
|
|
netdev_ieee802154_set(&dev->netdev, NETOPT_ACK_REQ,
|
2022-12-14 16:24:36 +01:00
|
|
|
|
&ack_req, sizeof(ack_req));
|
2022-11-28 15:48:23 +01:00
|
|
|
|
}
|
2022-11-28 15:57:38 +01:00
|
|
|
|
#if IS_USED(MODULE_IEEE802154_SECURITY) && \
|
|
|
|
|
IS_USED(MODULE_AT86RF2XX_AES_SPI)
|
|
|
|
|
dev->netdev.sec_ctx.dev.cipher_ops = &_at86rf2xx_cipher_ops;
|
|
|
|
|
dev->netdev.sec_ctx.dev.ctx = dev;
|
|
|
|
|
#endif
|
|
|
|
|
|
2019-09-02 08:49:27 +02:00
|
|
|
|
at86rf2xx_reset(dev);
|
|
|
|
|
|
2022-11-28 16:29:31 +01:00
|
|
|
|
/* Initialize CSMA seed with hardware address */
|
|
|
|
|
at86rf2xx_set_csma_seed(dev, dev->netdev.long_addr);
|
|
|
|
|
|
2022-08-18 12:02:47 +02:00
|
|
|
|
/* signal link UP */
|
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_LINK_UP);
|
|
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
|
static int _send(netdev_t *netdev, const iolist_t *iolist)
|
2015-04-23 18:36:55 +02:00
|
|
|
|
{
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
2016-03-18 10:26:33 +01:00
|
|
|
|
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 */
|
2018-01-12 00:19:03 +01:00
|
|
|
|
for (const iolist_t *iol = iolist; iol; iol = iol->iol_next) {
|
2016-03-18 10:26:33 +01:00
|
|
|
|
/* current packet data + FCS too long */
|
2018-01-12 00:19:03 +01:00
|
|
|
|
if ((len + iol->iol_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;
|
|
|
|
|
}
|
2019-03-15 15:34:38 +01:00
|
|
|
|
if (iol->iol_len) {
|
|
|
|
|
len = at86rf2xx_tx_load(dev, iol->iol_base, iol->iol_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 */
|
2018-07-11 14:10:50 +02:00
|
|
|
|
if (!(dev->flags & AT86RF2XX_OPT_PRELOADING)) {
|
2015-08-09 21:24:55 +02:00
|
|
|
|
at86rf2xx_tx_exec(dev);
|
2022-11-28 15:53:10 +01:00
|
|
|
|
if (netdev->event_callback) {
|
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_STARTED);
|
|
|
|
|
}
|
2015-04-23 18:36:55 +02:00
|
|
|
|
}
|
2017-09-07 14:38:47 +02:00
|
|
|
|
/* return the number of bytes that were actually loaded into the frame
|
|
|
|
|
* buffer/send out */
|
2015-04-23 18:36:55 +02:00
|
|
|
|
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
|
|
|
|
{
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, 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
|
|
|
|
|
2018-07-06 14:03:09 +02:00
|
|
|
|
/* frame buffer protection will be unlocked as soon as at86rf2xx_fb_stop() is called,
|
2019-10-21 01:07:44 +02:00
|
|
|
|
* Set receiver to PLL_ON state to be able to free the SPI bus and avoid losing data. */
|
2018-07-06 14:03:09 +02:00
|
|
|
|
at86rf2xx_set_state(dev, AT86RF2XX_STATE_PLL_ON);
|
|
|
|
|
|
|
|
|
|
/* start frame buffer access */
|
2015-09-30 17:46:13 +02:00
|
|
|
|
at86rf2xx_fb_start(dev);
|
|
|
|
|
|
|
|
|
|
/* get the size of the received packet */
|
2022-11-28 14:48:12 +01:00
|
|
|
|
phr = at86rf2xx_get_rx_len(dev);
|
2015-09-30 17:46:13 +02:00
|
|
|
|
|
2019-10-21 01:07:44 +02:00
|
|
|
|
/* ignore MSB (refer p.80) and subtract length of FCS field */
|
2016-04-02 17:29:22 +02:00
|
|
|
|
pkt_len = (phr & 0x7f) - 2;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
2018-07-06 14:03:09 +02:00
|
|
|
|
/* return length when buf == NULL */
|
2016-03-18 10:26:33 +01:00
|
|
|
|
if (buf == NULL) {
|
2018-07-06 14:03:09 +02:00
|
|
|
|
/* release SPI bus */
|
2015-09-30 17:46:13 +02:00
|
|
|
|
at86rf2xx_fb_stop(dev);
|
2018-07-06 14:03:09 +02:00
|
|
|
|
|
|
|
|
|
/* drop packet, continue receiving */
|
|
|
|
|
if (len > 0) {
|
|
|
|
|
/* set device back in operation state which was used before last transmission.
|
2018-11-16 18:47:22 +01:00
|
|
|
|
* This state is saved in at86rf2xx.c/at86rf2xx_tx_prepare() e.g RX_AACK_ON */
|
2018-07-06 14:03:09 +02:00
|
|
|
|
at86rf2xx_set_state(dev, dev->idle_state);
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
|
return pkt_len;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
}
|
2018-07-06 14:03:09 +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);
|
2018-07-06 14:03:09 +02:00
|
|
|
|
/* set device back in operation state which was used before last transmission.
|
2018-11-16 18:47:22 +01:00
|
|
|
|
* This state is saved in at86rf2xx.c/at86rf2xx_tx_prepare() e.g RX_AACK_ON */
|
2018-07-06 14:03:09 +02:00
|
|
|
|
at86rf2xx_set_state(dev, dev->idle_state);
|
2016-03-18 10:26:33 +01:00
|
|
|
|
return -ENOBUFS;
|
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
|
|
|
|
|
2018-10-27 01:12:58 +02:00
|
|
|
|
/* AT86RF212B RSSI_BASE_VAL + 1.03 * ED, base varies for diff. modulation and datarates
|
|
|
|
|
* AT86RF232 RSSI_BASE_VAL + ED, base -91dBm
|
|
|
|
|
* AT86RF233 RSSI_BASE_VAL + ED, base -94dBm
|
|
|
|
|
* AT86RF231 RSSI_BASE_VAL + ED, base -91dBm
|
2019-10-21 01:07:44 +02:00
|
|
|
|
* AT86RFA1 RSSI_BASE_VAL + ED, base -90dBm
|
|
|
|
|
* AT86RFR2 RSSI_BASE_VAL + ED, base -90dBm
|
2018-07-06 14:15:44 +02:00
|
|
|
|
*
|
2018-10-27 01:12:58 +02:00
|
|
|
|
* AT86RF231 MAN. p.92, 8.4.3 Data Interpretation
|
|
|
|
|
* AT86RF232 MAN. p.91, 8.4.3 Data Interpretation
|
|
|
|
|
* AT86RF233 MAN. p.102, 8.5.3 Data Interpretation
|
|
|
|
|
*
|
|
|
|
|
* for performance reasons we ignore the 1.03 scale factor on the 212B,
|
|
|
|
|
* which causes a slight error in the values, but the accuracy of the ED
|
|
|
|
|
* value is specified as +/- 5 dB, so it should not matter very much in real
|
|
|
|
|
* life.
|
2018-07-06 14:15:44 +02:00
|
|
|
|
*/
|
2016-03-18 10:26:33 +01:00
|
|
|
|
if (info != NULL) {
|
2018-10-27 01:12:58 +02:00
|
|
|
|
uint8_t ed = 0;
|
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);
|
2018-10-27 01:12:58 +02:00
|
|
|
|
|
2022-11-24 17:22:13 +01:00
|
|
|
|
#if AT86RF2XX_HAVE_ED_REGISTER
|
2018-10-27 01:12:58 +02:00
|
|
|
|
/* AT86RF231 does not provide ED at the end of the frame buffer, read
|
|
|
|
|
* from separate register instead */
|
2016-03-18 10:26:33 +01:00
|
|
|
|
at86rf2xx_fb_stop(dev);
|
2018-10-27 01:12:58 +02:00
|
|
|
|
ed = at86rf2xx_reg_read(dev, AT86RF2XX_REG__PHY_ED_LEVEL);
|
2015-09-30 17:46:13 +02:00
|
|
|
|
#else
|
2018-10-27 01:12:58 +02:00
|
|
|
|
at86rf2xx_fb_read(dev, &ed, 1);
|
2016-03-18 10:26:33 +01:00
|
|
|
|
at86rf2xx_fb_stop(dev);
|
2015-09-30 17:46:13 +02:00
|
|
|
|
#endif
|
2018-10-27 01:12:58 +02:00
|
|
|
|
radio_info->rssi = RSSI_BASE_VAL + ed;
|
2023-05-16 17:12:10 +02:00
|
|
|
|
DEBUG("[at86rf2xx] LQI:%d high is good, RSSI:%d high is either good or "
|
2018-07-06 14:15:44 +02:00
|
|
|
|
"too much interference.\n", radio_info->lqi, radio_info->rssi);
|
2023-05-16 17:12:10 +02:00
|
|
|
|
#if AT86RF2XX_IS_PERIPH && IS_USED(MODULE_NETDEV_IEEE802154_RX_TIMESTAMP)
|
|
|
|
|
/* AT86RF2XX_IS_PERIPH means the MCU is ATmegaRFR2 that has symbol counter */
|
|
|
|
|
{
|
|
|
|
|
uint32_t rx_sc;
|
|
|
|
|
rx_sc = at86rf2xx_get_sc(dev);
|
|
|
|
|
|
|
|
|
|
/* convert counter value to ns */
|
|
|
|
|
uint64_t res = SC_TO_NS * (uint64_t)rx_sc;
|
|
|
|
|
netdev_ieee802154_rx_info_set_timestamp(radio_info, res);
|
|
|
|
|
DEBUG("[at86rf2xx] CS: %" PRIu32 " timestamp: %" PRIu32 ".%09" PRIu32 " ",
|
|
|
|
|
rx_sc, (uint32_t)(radio_info->timestamp / NS_PER_SEC),
|
|
|
|
|
(uint32_t)(radio_info->timestamp % NS_PER_SEC));
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2016-03-18 10:26:33 +01:00
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
at86rf2xx_fb_stop(dev);
|
|
|
|
|
}
|
2015-09-30 17:46:13 +02:00
|
|
|
|
|
2018-07-06 14:03:09 +02:00
|
|
|
|
/* set device back in operation state which was used before last transmission.
|
2018-11-16 18:47:22 +01:00
|
|
|
|
* This state is saved in at86rf2xx.c/at86rf2xx_tx_prepare() e.g RX_AACK_ON */
|
2018-07-06 14:03:09 +02:00
|
|
|
|
at86rf2xx_set_state(dev, dev->idle_state);
|
|
|
|
|
|
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) {
|
2017-09-04 16:05:51 +02:00
|
|
|
|
case NETOPT_STATE_STANDBY:
|
|
|
|
|
at86rf2xx_set_state(dev, AT86RF2XX_STATE_TRX_OFF);
|
|
|
|
|
break;
|
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:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
at86rf2xx_set_state(dev, AT86RF2XX_PHY_STATE_RX);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
break;
|
2015-08-06 15:36:56 +02:00
|
|
|
|
case NETOPT_STATE_TX:
|
2018-07-11 14:10:50 +02:00
|
|
|
|
if (dev->flags & AT86RF2XX_OPT_PRELOADING) {
|
2017-09-07 14:38:47 +02:00
|
|
|
|
/* The netdev driver ISR switches the transceiver back to the
|
|
|
|
|
* previous idle state after a completed TX. If the user tries
|
|
|
|
|
* to initiate another transmission (retransmitting the same data)
|
|
|
|
|
* without first going to TX_ARET_ON, the command to start TX
|
|
|
|
|
* would be ignored, leading to a deadlock in this netdev driver
|
|
|
|
|
* thread.
|
|
|
|
|
* Additionally, avoids driver thread deadlock when PRELOADING
|
|
|
|
|
* is set and the user tries to initiate TX without first calling
|
|
|
|
|
* send() to write some frame data.
|
|
|
|
|
*/
|
|
|
|
|
if (dev->pending_tx == 0) {
|
|
|
|
|
/* retransmission of old data, at86rf2xx_tx_prepare normally
|
|
|
|
|
* increments this and the ISR for TX_END decrements it, to
|
|
|
|
|
* know when to switch back to the idle state. */
|
|
|
|
|
++dev->pending_tx;
|
|
|
|
|
}
|
2020-04-02 16:48:34 +02:00
|
|
|
|
at86rf2xx_set_state(dev, AT86RF2XX_PHY_STATE_TX);
|
2015-08-09 21:24:55 +02:00
|
|
|
|
at86rf2xx_tx_exec(dev);
|
2022-11-28 15:53:10 +01:00
|
|
|
|
if (dev->netdev.netdev.event_callback) {
|
|
|
|
|
dev->netdev.netdev.event_callback(&dev->netdev.netdev, NETDEV_EVENT_TX_STARTED);
|
|
|
|
|
}
|
2015-04-23 18:36:55 +02:00
|
|
|
|
}
|
|
|
|
|
break;
|
2015-08-06 15:36:56 +02:00
|
|
|
|
case NETOPT_STATE_RESET:
|
2020-02-13 02:05:07 +01:00
|
|
|
|
at86rf2xx_hardware_reset(dev);
|
2022-11-28 15:48:23 +01:00
|
|
|
|
netdev_ieee802154_reset(&dev->netdev);
|
|
|
|
|
/* set short and long address */
|
|
|
|
|
at86rf2xx_set_addr_long(dev, (eui64_t *)dev->netdev.long_addr);
|
|
|
|
|
at86rf2xx_set_addr_short(dev, (network_uint16_t *)dev->netdev.short_addr);
|
|
|
|
|
|
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
2022-12-14 16:24:36 +01:00
|
|
|
|
static const netopt_enable_t ack_req =
|
|
|
|
|
IS_ACTIVE(CONFIG_IEEE802154_DEFAULT_ACK_REQ) ? NETOPT_ENABLE : NETOPT_DISABLE;
|
2022-11-28 15:48:23 +01:00
|
|
|
|
netdev_ieee802154_set(&dev->netdev, NETOPT_ACK_REQ,
|
2022-12-14 16:24:36 +01:00
|
|
|
|
&ack_req, sizeof(ack_req));
|
2022-11-28 15:48:23 +01:00
|
|
|
|
}
|
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;
|
2017-09-04 16:05:51 +02:00
|
|
|
|
case AT86RF2XX_STATE_TRX_OFF:
|
|
|
|
|
return NETOPT_STATE_STANDBY;
|
2020-04-02 16:48:34 +02:00
|
|
|
|
case AT86RF2XX_PHY_STATE_RX_BUSY:
|
2015-08-06 15:36:56 +02:00
|
|
|
|
return NETOPT_STATE_RX;
|
2020-04-02 16:48:34 +02:00
|
|
|
|
case AT86RF2XX_PHY_STATE_TX:
|
|
|
|
|
case AT86RF2XX_PHY_STATE_TX_BUSY:
|
2015-08-06 15:36:56 +02:00
|
|
|
|
return NETOPT_STATE_TX;
|
2020-04-02 16:48:34 +02:00
|
|
|
|
case AT86RF2XX_PHY_STATE_RX:
|
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
|
|
|
|
{
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, 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) {
|
2022-11-28 14:51:34 +01:00
|
|
|
|
#if AT86RF2XX_HAVE_SUBGHZ
|
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;
|
2022-11-28 14:51:34 +01:00
|
|
|
|
((uint8_t *)val)[0] = dev->page;
|
2015-09-16 18:25:36 +02:00
|
|
|
|
return sizeof(uint16_t);
|
2022-11-28 14:51:34 +01:00
|
|
|
|
#endif
|
2015-09-16 18:25:36 +02:00
|
|
|
|
|
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:
|
2018-07-11 14:10:50 +02:00
|
|
|
|
if (dev->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:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
if (dev->flags & AT86RF2XX_OPT_PROMISCUOUS) {
|
|
|
|
|
*((netopt_enable_t *)val) = NETOPT_ENABLE;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
*((netopt_enable_t *)val) = NETOPT_DISABLE;
|
|
|
|
|
}
|
|
|
|
|
return sizeof(netopt_enable_t);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
}
|
2020-04-02 16:48:34 +02:00
|
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
|
case NETOPT_RX_START_IRQ:
|
|
|
|
|
case NETOPT_TX_START_IRQ:
|
|
|
|
|
case NETOPT_TX_END_IRQ:
|
2021-02-26 11:32:45 +01:00
|
|
|
|
*((netopt_enable_t *)val) = NETOPT_ENABLE;
|
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:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
*((netopt_enable_t *)val) =
|
|
|
|
|
!!(dev->flags & AT86RF2XX_OPT_CSMA);
|
|
|
|
|
return sizeof(netopt_enable_t);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2015-06-01 15:51:50 +02:00
|
|
|
|
|
2017-08-18 13:41:45 +02:00
|
|
|
|
/* Only radios with the XAH_CTRL_2 register support frame retry reporting */
|
|
|
|
|
#if AT86RF2XX_HAVE_RETRIES
|
|
|
|
|
case NETOPT_TX_RETRIES_NEEDED:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
assert(max_len >= sizeof(uint8_t));
|
|
|
|
|
*((uint8_t *)val) = dev->tx_retries;
|
|
|
|
|
return sizeof(uint8_t);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2017-08-18 13:41:45 +02:00
|
|
|
|
#endif
|
|
|
|
|
|
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;
|
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
|
if (((res = netdev_ieee802154_get(container_of(netdev, netdev_ieee802154_t, netdev),
|
|
|
|
|
opt, val, max_len)) >= 0)
|
|
|
|
|
|| (res != -ENOTSUP)) {
|
2016-03-18 10:26:33 +01:00
|
|
|
|
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));
|
2022-11-28 16:06:36 +01:00
|
|
|
|
*((uint16_t *)val) = netdev_ieee802154->txpower;
|
2015-12-06 15:09:00 +01:00
|
|
|
|
res = sizeof(uint16_t);
|
2015-09-16 18:48:10 +02:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case NETOPT_RETRANS:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
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:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
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;
|
|
|
|
|
|
2017-09-04 16:06:55 +02:00
|
|
|
|
case NETOPT_IS_CHANNEL_CLR:
|
|
|
|
|
assert(max_len >= sizeof(netopt_enable_t));
|
|
|
|
|
*((netopt_enable_t *)val) = at86rf2xx_cca(dev);
|
|
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
|
break;
|
|
|
|
|
|
2017-09-04 20:16:07 +02:00
|
|
|
|
case NETOPT_LAST_ED_LEVEL:
|
|
|
|
|
assert(max_len >= sizeof(int8_t));
|
|
|
|
|
*((int8_t *)val) = at86rf2xx_get_ed_level(dev);
|
|
|
|
|
res = sizeof(int8_t);
|
|
|
|
|
break;
|
|
|
|
|
|
2018-07-06 13:07:02 +02:00
|
|
|
|
case NETOPT_AUTOACK:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
assert(max_len >= sizeof(netopt_enable_t));
|
|
|
|
|
uint8_t tmp = at86rf2xx_reg_read(dev, AT86RF2XX_REG__CSMA_SEED_1);
|
2023-06-30 22:41:44 +02:00
|
|
|
|
*((netopt_enable_t *)val) = (tmp & AT86RF2XX_CSMA_SEED_1__AACK_DIS_ACK)
|
|
|
|
|
? false : true;
|
2020-04-02 16:48:34 +02:00
|
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
|
}
|
2017-12-11 15:57:23 +01:00
|
|
|
|
break;
|
|
|
|
|
|
2019-09-09 14:30:37 +02:00
|
|
|
|
#ifdef MODULE_NETDEV_IEEE802154_OQPSK
|
|
|
|
|
|
|
|
|
|
case NETOPT_IEEE802154_PHY:
|
|
|
|
|
assert(max_len >= sizeof(int8_t));
|
|
|
|
|
*(uint8_t *)val = at86rf2xx_get_phy_mode(dev);
|
|
|
|
|
return sizeof(uint8_t);
|
|
|
|
|
|
|
|
|
|
case NETOPT_OQPSK_RATE:
|
|
|
|
|
assert(max_len >= sizeof(int8_t));
|
|
|
|
|
*(uint8_t *)val = at86rf2xx_get_rate(dev);
|
|
|
|
|
return sizeof(uint8_t);
|
|
|
|
|
|
|
|
|
|
#endif /* MODULE_NETDEV_IEEE802154_OQPSK */
|
2022-07-01 00:13:13 +02:00
|
|
|
|
#if AT86RF2XX_RANDOM_NUMBER_GENERATOR
|
|
|
|
|
case NETOPT_RANDOM:
|
2022-07-11 08:04:18 +02:00
|
|
|
|
at86rf2xx_get_random(dev, (uint8_t*)val, max_len);
|
|
|
|
|
return max_len;
|
2022-07-01 00:13:13 +02:00
|
|
|
|
#endif
|
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-08-25 07:34:03 +02:00
|
|
|
|
static int _set(netdev_t *netdev, netopt_t opt, const void *val, size_t len)
|
2015-04-23 18:36:55 +02:00
|
|
|
|
{
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
if (dev == NULL) {
|
|
|
|
|
return -ENODEV;
|
|
|
|
|
}
|
2021-01-21 10:23:40 +01:00
|
|
|
|
uint8_t old_state = at86rf2xx_get_status(dev);
|
|
|
|
|
int res = -ENOTSUP;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
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:
|
2020-09-15 13:57:15 +02:00
|
|
|
|
assert(len == sizeof(network_uint16_t));
|
2022-11-28 15:11:29 +01:00
|
|
|
|
memcpy(dev->netdev.short_addr, val, len);
|
|
|
|
|
#ifdef MODULE_SIXLOWPAN
|
|
|
|
|
/* https://tools.ietf.org/html/rfc4944#section-12 requires the first bit to
|
|
|
|
|
* 0 for unicast addresses */
|
|
|
|
|
dev->netdev.short_addr[0] &= 0x7F;
|
|
|
|
|
#endif
|
2019-10-29 23:09:00 +01:00
|
|
|
|
at86rf2xx_set_addr_short(dev, 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:
|
2020-09-15 13:57:15 +02:00
|
|
|
|
assert(len == sizeof(eui64_t));
|
2022-11-28 15:11:29 +01:00
|
|
|
|
memcpy(dev->netdev.long_addr, val, len);
|
2019-10-29 23:09:00 +01:00
|
|
|
|
at86rf2xx_set_addr_long(dev, 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:
|
2020-09-15 13:57:15 +02:00
|
|
|
|
assert(len == sizeof(uint16_t));
|
2017-08-25 07:34:03 +02:00
|
|
|
|
at86rf2xx_set_pan(dev, *((const 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:
|
2017-08-21 12:14:39 +02:00
|
|
|
|
assert(len == sizeof(uint16_t));
|
|
|
|
|
uint8_t chan = (((const uint16_t *)val)[0]) & UINT8_MAX;
|
make: fix various compile errors with Wextra
pkg, nordic_softdevice_ble: disable CFLAGS to omit compiler error
sys, pm_layered: fix casting nonscalar to the same type
cpu, stm32_common: fix type-limits, remove always true assert
cpu, stm32f4: fix pointer arithmetic in periph/i2c
drivers, at86rf2xx: fix type-limits where condition always true
saul, gpio: fix if no gpio configured for saul
cpu, saml21: add frequency check to periph/timer
driver, cc110x: fix unused param and type-limts errors
boards, wsn430-common: fix old-style-declaration
make: fix old style definition
drivers, sdcard_spi: fix old style typedef
driver, at30tse: remove unnecessary check
driver, nrf24: fix type-limit
driver, pn532: change buffer from char to uint8_t
tests/driver_sdcard: fix type limits
boards, feather-m0: add missing field inits
driver, tcs37727: fix type limits
pkg, emb6: disable some compiler warnings
tests/emb6: disable some compiler warings
pkg, openthread: fix sign compare and unused params
tests/trickle: fix struct init
tests/pthread_cooperation: fix type limits
board, mips-malta: remove feature periph_uart
shell: fix var size for netif command
gnrc, netif: fix sign-compare
gnrc, nib: fix sign-compare
shell: fix output in netif command
posix: fix type-limits in pthread_cond
2017-10-31 11:52:18 +01:00
|
|
|
|
#if AT86RF2XX_MIN_CHANNEL
|
|
|
|
|
if (chan < AT86RF2XX_MIN_CHANNEL || chan > AT86RF2XX_MAX_CHANNEL) {
|
|
|
|
|
#else
|
|
|
|
|
if (chan > AT86RF2XX_MAX_CHANNEL) {
|
|
|
|
|
#endif /* AT86RF2XX_MIN_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
|
|
|
|
}
|
2022-11-28 15:35:36 +01:00
|
|
|
|
dev->netdev.chan = chan;
|
|
|
|
|
#if AT86RF2XX_HAVE_SUBGHZ
|
2022-11-28 16:06:36 +01:00
|
|
|
|
at86rf2xx_configure_phy(dev, chan, dev->page, dev->netdev.txpower);
|
2022-11-28 15:35:36 +01:00
|
|
|
|
#else
|
2022-11-28 16:06:36 +01:00
|
|
|
|
at86rf2xx_configure_phy(dev, chan, 0, dev->netdev.txpower);
|
2022-11-28 15:35:36 +01:00
|
|
|
|
#endif
|
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:
|
2017-08-21 12:14:39 +02:00
|
|
|
|
assert(len == sizeof(uint16_t));
|
|
|
|
|
uint8_t page = (((const uint16_t *)val)[0]) & UINT8_MAX;
|
2022-11-24 17:21:44 +01:00
|
|
|
|
#if AT86RF2XX_HAVE_SUBGHZ
|
2015-12-06 15:09:00 +01:00
|
|
|
|
if ((page != 0) && (page != 2)) {
|
2015-09-16 18:25:36 +02:00
|
|
|
|
res = -EINVAL;
|
2016-03-18 10:26:33 +01:00
|
|
|
|
}
|
|
|
|
|
else {
|
2022-11-28 15:35:36 +01:00
|
|
|
|
dev->page = page;
|
2022-11-28 16:06:36 +01:00
|
|
|
|
at86rf2xx_configure_phy(dev, dev->netdev.chan, page, dev->netdev.txpower);
|
2017-08-21 12:14:39 +02:00
|
|
|
|
res = sizeof(uint16_t);
|
2015-12-06 15:09:00 +01:00
|
|
|
|
}
|
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 {
|
2017-08-21 12:14:39 +02:00
|
|
|
|
res = sizeof(uint16_t);
|
2015-12-06 15:09:00 +01:00
|
|
|
|
}
|
|
|
|
|
#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));
|
2022-11-28 16:06:36 +01:00
|
|
|
|
netdev_ieee802154->txpower = *((const int16_t *)val);
|
2022-11-28 16:25:27 +01:00
|
|
|
|
#if AT86RF2XX_HAVE_SUBGHZ
|
|
|
|
|
at86rf2xx_configure_phy(dev, dev->netdev.chan, dev->page, *((const int16_t *)val));
|
|
|
|
|
#else
|
|
|
|
|
at86rf2xx_configure_phy(dev, dev->netdev.chan, 0, *((const int16_t *)val));
|
|
|
|
|
#endif
|
2015-12-06 15:09:00 +01:00
|
|
|
|
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));
|
2017-08-25 07:34:03 +02:00
|
|
|
|
res = _set_state(dev, *((const 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:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
|
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_AUTOACK,
|
|
|
|
|
((const bool *)val)[0]);
|
|
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
|
}
|
2015-09-16 18:48:10 +02:00
|
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
2018-01-17 00:51:07 +01:00
|
|
|
|
case NETOPT_ACK_PENDING:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_ACK_PENDING,
|
|
|
|
|
((const bool *)val)[0]);
|
|
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
|
}
|
2018-01-17 00:51:07 +01:00
|
|
|
|
break;
|
|
|
|
|
|
2015-08-06 15:36:56 +02:00
|
|
|
|
case NETOPT_RETRANS:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
assert(len <= sizeof(uint8_t));
|
|
|
|
|
at86rf2xx_set_max_retries(dev, *((const 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,
|
2017-08-25 07:34:03 +02:00
|
|
|
|
((const 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:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_PROMISCUOUS,
|
|
|
|
|
((const bool *)val)[0]);
|
|
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
|
}
|
2015-09-16 18:48:10 +02:00
|
|
|
|
break;
|
2015-04-23 18:36:55 +02:00
|
|
|
|
|
2015-06-01 15:51:50 +02:00
|
|
|
|
case NETOPT_CSMA:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
at86rf2xx_set_option(dev, AT86RF2XX_OPT_CSMA,
|
|
|
|
|
((const bool *)val)[0]);
|
|
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
|
}
|
2015-09-16 18:48:10 +02:00
|
|
|
|
break;
|
2015-06-01 15:51:50 +02:00
|
|
|
|
|
|
|
|
|
case NETOPT_CSMA_RETRIES:
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (!IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
assert(len <= sizeof(uint8_t));
|
|
|
|
|
if (!(dev->flags & AT86RF2XX_OPT_CSMA) ||
|
|
|
|
|
(*((uint8_t *)val) > 5)) {
|
|
|
|
|
/* If CSMA is disabled, don't allow setting retries */
|
|
|
|
|
res = -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
at86rf2xx_set_csma_max_retries(dev, *((const 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));
|
2017-08-25 07:34:03 +02:00
|
|
|
|
at86rf2xx_set_cca_threshold(dev, *((const int8_t *)val));
|
2015-12-06 15:09:00 +01:00
|
|
|
|
res = sizeof(int8_t);
|
2015-10-22 23:01:26 +02:00
|
|
|
|
break;
|
|
|
|
|
|
2019-09-09 14:30:37 +02:00
|
|
|
|
#ifdef MODULE_NETDEV_IEEE802154_OQPSK
|
|
|
|
|
|
|
|
|
|
case NETOPT_OQPSK_RATE:
|
|
|
|
|
assert(len <= sizeof(int8_t));
|
|
|
|
|
if (at86rf2xx_set_rate(dev, *((const uint8_t *)val)) < 0) {
|
|
|
|
|
res = -EINVAL;
|
|
|
|
|
} else {
|
|
|
|
|
res = sizeof(uint8_t);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
#endif /* MODULE_NETDEV_IEEE802154_OQPSK */
|
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 */
|
2017-06-30 13:53:29 +02: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) {
|
2021-06-21 16:32:16 +02:00
|
|
|
|
res = netdev_ieee802154_set(container_of(netdev, netdev_ieee802154_t, netdev),
|
|
|
|
|
opt, 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
|
|
|
|
}
|
|
|
|
|
|
2019-11-16 22:26:58 +01:00
|
|
|
|
static void _isr_send_complete(at86rf2xx_t *dev, uint8_t trac_status)
|
|
|
|
|
{
|
|
|
|
|
netdev_t *netdev = &dev->netdev.netdev;
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2019-11-16 22:26:58 +01:00
|
|
|
|
/* Only radios with the XAH_CTRL_2 register support frame retry reporting */
|
2022-11-24 17:22:33 +01:00
|
|
|
|
#if AT86RF2XX_HAVE_RETRIES_REG
|
2019-11-16 22:26:58 +01:00
|
|
|
|
dev->tx_retries = (at86rf2xx_reg_read(dev, AT86RF2XX_REG__XAH_CTRL_2)
|
|
|
|
|
& AT86RF2XX_XAH_CTRL_2__ARET_FRAME_RETRIES_MASK) >>
|
|
|
|
|
AT86RF2XX_XAH_CTRL_2__ARET_FRAME_RETRIES_OFFSET;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
DEBUG("[at86rf2xx] EVT - TX_END\n");
|
|
|
|
|
|
2021-02-26 11:32:45 +01:00
|
|
|
|
if (netdev->event_callback) {
|
2019-11-16 22:26:58 +01:00
|
|
|
|
switch (trac_status) {
|
|
|
|
|
#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
|
|
|
|
|
case AT86RF2XX_TRX_STATE__TRAC_SUCCESS:
|
|
|
|
|
case AT86RF2XX_TRX_STATE__TRAC_SUCCESS_DATA_PENDING:
|
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE);
|
|
|
|
|
DEBUG("[at86rf2xx] TX SUCCESS\n");
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
case AT86RF2XX_TRX_STATE__TRAC_NO_ACK:
|
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_NOACK);
|
|
|
|
|
DEBUG("[at86rf2xx] TX NO_ACK\n");
|
|
|
|
|
break;
|
|
|
|
|
case AT86RF2XX_TRX_STATE__TRAC_CHANNEL_ACCESS_FAILURE:
|
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_MEDIUM_BUSY);
|
|
|
|
|
DEBUG("[at86rf2xx] TX_CHANNEL_ACCESS_FAILURE\n");
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
DEBUG("[at86rf2xx] Unhandled TRAC_STATUS: %d\n",
|
|
|
|
|
trac_status >> 5);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-02 18:23:26 +02:00
|
|
|
|
static inline void _isr_recv_complete(netdev_t *netdev)
|
|
|
|
|
{
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
2021-02-26 11:32:45 +01:00
|
|
|
|
if (!netdev->event_callback) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-04-02 18:23:26 +02:00
|
|
|
|
if (IS_ACTIVE(AT86RF2XX_BASIC_MODE)) {
|
|
|
|
|
uint8_t phy_status = at86rf2xx_reg_read(dev, AT86RF2XX_REG__PHY_RSSI);
|
|
|
|
|
bool crc_ok = phy_status & AT86RF2XX_PHY_RSSI_MASK__RX_CRC_VALID;
|
|
|
|
|
|
|
|
|
|
if (crc_ok) {
|
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_RX_COMPLETE);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_CRC_ERROR);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_RX_COMPLETE);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
|
static void _isr(netdev_t *netdev)
|
2015-04-23 18:36:55 +02:00
|
|
|
|
{
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, 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 */
|
2022-11-28 14:48:12 +01:00
|
|
|
|
irq_mask = at86rf2xx_get_irq_flags(dev);
|
2015-07-20 11:47:15 +02:00
|
|
|
|
|
2017-06-30 13:53:29 +02:00
|
|
|
|
trac_status = at86rf2xx_reg_read(dev, AT86RF2XX_REG__TRX_STATE)
|
|
|
|
|
& 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) {
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if ((state == AT86RF2XX_PHY_STATE_RX)
|
|
|
|
|
|| (state == AT86RF2XX_PHY_STATE_RX_BUSY)) {
|
2015-08-09 21:24:55 +02:00
|
|
|
|
DEBUG("[at86rf2xx] EVT - RX_END\n");
|
2020-04-02 18:23:26 +02:00
|
|
|
|
|
|
|
|
|
_isr_recv_complete(netdev);
|
|
|
|
|
|
2015-04-23 18:36:55 +02:00
|
|
|
|
}
|
2020-04-02 16:48:34 +02:00
|
|
|
|
else if (state == AT86RF2XX_PHY_STATE_TX) {
|
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);
|
2019-11-16 22:28:42 +01:00
|
|
|
|
/* Radio is idle, any TX transaction is done */
|
|
|
|
|
dev->pending_tx = 0;
|
|
|
|
|
at86rf2xx_set_state(dev, dev->idle_state);
|
|
|
|
|
DEBUG("[at86rf2xx] return to idle state 0x%x\n", dev->idle_state);
|
2019-11-16 22:26:58 +01:00
|
|
|
|
_isr_send_complete(dev, trac_status);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
}
|
2019-11-16 22:28:42 +01:00
|
|
|
|
/* Only the case when an interrupt was received and the radio is busy
|
|
|
|
|
* with a next PDU transmission when _isr is called.
|
|
|
|
|
* dev->pending == 1 means a receive and immediately a send happened.
|
|
|
|
|
* The receive is discarded as the send already overwrote the internal
|
|
|
|
|
* buffer.
|
|
|
|
|
* dev->pending == 2 means two transmits occurred and this is the isr for
|
|
|
|
|
* the first.
|
|
|
|
|
*/
|
2020-04-02 16:48:34 +02:00
|
|
|
|
else if (state == AT86RF2XX_PHY_STATE_TX_BUSY) {
|
2019-11-16 22:28:42 +01:00
|
|
|
|
if (dev->pending_tx > 1) {
|
|
|
|
|
dev->pending_tx--;
|
|
|
|
|
_isr_send_complete(dev, trac_status);
|
2015-04-23 18:36:55 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-10-21 01:07:44 +02:00
|
|
|
|
|
2022-11-28 15:54:19 +01:00
|
|
|
|
void at86rf2xx_setup(at86rf2xx_t *dev, const at86rf2xx_params_t *params, uint8_t index)
|
|
|
|
|
{
|
|
|
|
|
netdev_t *netdev = &dev->netdev.netdev;
|
|
|
|
|
|
|
|
|
|
netdev->driver = &at86rf2xx_driver;
|
|
|
|
|
/* State to return after receiving or transmitting */
|
|
|
|
|
dev->idle_state = AT86RF2XX_STATE_TRX_OFF;
|
|
|
|
|
/* radio state is P_ON when first powered-on */
|
|
|
|
|
dev->state = AT86RF2XX_STATE_P_ON;
|
|
|
|
|
dev->pending_tx = 0;
|
|
|
|
|
|
|
|
|
|
#if AT86RF2XX_IS_PERIPH
|
|
|
|
|
(void) params;
|
|
|
|
|
/* set all interrupts off */
|
|
|
|
|
at86rf2xx_reg_write(dev, AT86RF2XX_REG__IRQ_MASK, 0x00);
|
|
|
|
|
#else
|
|
|
|
|
/* initialize device descriptor */
|
|
|
|
|
dev->params = *params;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
netdev_register(netdev, NETDEV_AT86RF2XX, index);
|
|
|
|
|
/* set device address */
|
|
|
|
|
netdev_ieee802154_setup(&dev->netdev);
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-24 16:50:58 +01:00
|
|
|
|
#if AT86RF2XX_IS_PERIPH
|
2019-10-21 01:07:44 +02:00
|
|
|
|
|
2020-07-23 19:50:08 +02:00
|
|
|
|
/**
|
|
|
|
|
* @brief ISR for transceiver's TX_START interrupt
|
|
|
|
|
*
|
|
|
|
|
* In procedure TX_ARET the TRX24_TX_START interrupt is issued separately for every
|
|
|
|
|
* frame transmission and frame retransmission.
|
|
|
|
|
* Indicates the frame start of a transmitted acknowledge frame in procedure RX_AACK.
|
|
|
|
|
*
|
|
|
|
|
* Flow Diagram Manual p. 52 / 63
|
|
|
|
|
*/
|
|
|
|
|
#if AT86RF2XX_HAVE_RETRIES
|
|
|
|
|
ISR(TRX24_TX_START_vect){
|
2020-10-02 08:01:49 +02:00
|
|
|
|
/* __enter_isr(); is not necessary as there is nothing which causes a
|
2020-07-23 19:50:08 +02:00
|
|
|
|
* thread_yield and the interrupt can not be interrupted by an other ISR */
|
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(at86rfmega_dev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
2020-07-23 19:50:08 +02:00
|
|
|
|
|
|
|
|
|
dev->tx_retries++;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-02-16 23:56:02 +01:00
|
|
|
|
/**
|
|
|
|
|
* @brief Transceiver PLL Lock
|
|
|
|
|
*
|
|
|
|
|
* Is triggered when PLL locked successfully.
|
|
|
|
|
*
|
|
|
|
|
* Manual p. 40
|
|
|
|
|
*/
|
2023-06-29 17:18:21 +02:00
|
|
|
|
static inline void txr24_pll_lock_handler(void)
|
2021-02-16 23:56:02 +01:00
|
|
|
|
{
|
|
|
|
|
DEBUG("TRX24_PLL_LOCK\n");
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(at86rfmega_dev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
|
|
|
|
dev->irq_status |= AT86RF2XX_IRQ_STATUS_MASK__PLL_LOCK;
|
2021-02-16 23:56:02 +01:00
|
|
|
|
}
|
2023-06-29 17:18:21 +02:00
|
|
|
|
AVR8_ISR(TRX24_PLL_LOCK_vect, txr24_pll_lock_handler);
|
2021-02-16 23:56:02 +01:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief Transceiver PLL Unlock
|
|
|
|
|
*
|
|
|
|
|
* Is triggered when PLL unlocked unexpectedly.
|
|
|
|
|
*
|
|
|
|
|
* Manual p. 89
|
|
|
|
|
*/
|
2023-06-29 17:18:21 +02:00
|
|
|
|
static inline void txr24_pll_unlock_handler(void)
|
2021-02-16 23:56:02 +01:00
|
|
|
|
{
|
|
|
|
|
DEBUG("TRX24_PLL_UNLOCK\n");
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(at86rfmega_dev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
|
|
|
|
dev->irq_status |= AT86RF2XX_IRQ_STATUS_MASK__PLL_UNLOCK;
|
2021-02-16 23:56:02 +01:00
|
|
|
|
}
|
2023-06-29 17:18:21 +02:00
|
|
|
|
AVR8_ISR(TRX24_PLL_UNLOCK_vect, txr24_pll_unlock_handler);
|
2021-02-16 23:56:02 +01:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief ISR for transceiver's receive start interrupt
|
|
|
|
|
*
|
|
|
|
|
* Is triggered when valid PHR is detected.
|
|
|
|
|
* Save IRQ status and inform upper layer of data reception.
|
|
|
|
|
*
|
|
|
|
|
* Flow Diagram Manual p. 52 / 63
|
|
|
|
|
*/
|
2023-06-29 17:18:21 +02:00
|
|
|
|
static inline void txr24_rx_start_handler(void)
|
2021-02-16 23:56:02 +01:00
|
|
|
|
{
|
|
|
|
|
uint8_t status = *AT86RF2XX_REG__TRX_STATE & AT86RF2XX_TRX_STATUS_MASK__TRX_STATUS;
|
|
|
|
|
DEBUG("TRX24_RX_START 0x%x\n", status);
|
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(at86rfmega_dev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
|
|
|
|
dev->irq_status |= AT86RF2XX_IRQ_STATUS_MASK__RX_START;
|
2021-02-16 23:56:02 +01:00
|
|
|
|
/* Call upper layer to process valid PHR */
|
|
|
|
|
netdev_trigger_event_isr(at86rfmega_dev);
|
|
|
|
|
}
|
2023-06-29 17:18:21 +02:00
|
|
|
|
AVR8_ISR(TRX24_RX_START_vect, txr24_rx_start_handler);
|
2021-02-16 23:56:02 +01:00
|
|
|
|
|
2019-10-21 01:07:44 +02:00
|
|
|
|
/**
|
|
|
|
|
* @brief ISR for transceiver's receive end interrupt
|
|
|
|
|
*
|
|
|
|
|
* Is triggered when valid data is received. FCS check passed.
|
|
|
|
|
* Save IRQ status and inform upper layer of data reception.
|
|
|
|
|
*
|
|
|
|
|
* Flow Diagram Manual p. 52 / 63
|
|
|
|
|
*/
|
2023-06-29 17:18:21 +02:00
|
|
|
|
static inline void txr24_rx_end_handler(void)
|
2019-10-21 01:07:44 +02:00
|
|
|
|
{
|
|
|
|
|
uint8_t status = *AT86RF2XX_REG__TRX_STATE & AT86RF2XX_TRX_STATUS_MASK__TRX_STATUS;
|
|
|
|
|
DEBUG("TRX24_RX_END 0x%x\n", status);
|
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(at86rfmega_dev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
|
|
|
|
dev->irq_status |= AT86RF2XX_IRQ_STATUS_MASK__RX_END;
|
2019-10-21 01:07:44 +02:00
|
|
|
|
/* Call upper layer to process received data */
|
2020-03-05 15:11:20 +01:00
|
|
|
|
netdev_trigger_event_isr(at86rfmega_dev);
|
2019-10-21 01:07:44 +02:00
|
|
|
|
}
|
2023-06-29 17:18:21 +02:00
|
|
|
|
AVR8_ISR(TRX24_RX_END_vect, txr24_rx_end_handler);
|
2019-10-21 01:07:44 +02:00
|
|
|
|
|
2021-02-16 23:56:02 +01:00
|
|
|
|
/**
|
|
|
|
|
* @brief Transceiver CCAED Measurement finished
|
|
|
|
|
*
|
|
|
|
|
* Is triggered when CCA or ED measurement completed.
|
|
|
|
|
*
|
|
|
|
|
* Manual p. 74 and p. 76
|
|
|
|
|
*/
|
2023-06-29 17:18:21 +02:00
|
|
|
|
static inline void txr24_cca_ed_done_handler(void)
|
2021-02-16 23:56:02 +01:00
|
|
|
|
{
|
|
|
|
|
DEBUG("TRX24_CCA_ED_DONE\n");
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(at86rfmega_dev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
|
|
|
|
dev->irq_status |= AT86RF2XX_IRQ_STATUS_MASK__CCA_ED_DONE;
|
2021-02-16 23:56:02 +01:00
|
|
|
|
}
|
2023-06-29 17:18:21 +02:00
|
|
|
|
AVR8_ISR(TRX24_CCA_ED_DONE_vect, txr24_cca_ed_done_handler);
|
2021-02-16 23:56:02 +01:00
|
|
|
|
|
2019-10-21 01:07:44 +02:00
|
|
|
|
/**
|
|
|
|
|
* @brief Transceiver Frame Address Match, indicates incoming frame
|
|
|
|
|
*
|
|
|
|
|
* Is triggered when Frame with valid Address is received.
|
|
|
|
|
* Can be used to wake up MCU from sleep, etc.
|
|
|
|
|
*
|
|
|
|
|
* Flow Diagram Manual p. 52 / 63
|
|
|
|
|
*/
|
2023-06-29 17:18:21 +02:00
|
|
|
|
static inline void txr24_xah_ami_handler(void)
|
2019-10-21 01:07:44 +02:00
|
|
|
|
{
|
|
|
|
|
DEBUG("TRX24_XAH_AMI\n");
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(at86rfmega_dev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
|
|
|
|
dev->irq_status |= AT86RF2XX_IRQ_STATUS_MASK__AMI;
|
2019-10-21 01:07:44 +02:00
|
|
|
|
}
|
2023-06-29 17:18:21 +02:00
|
|
|
|
AVR8_ISR(TRX24_XAH_AMI_vect, txr24_xah_ami_handler);
|
2019-10-21 01:07:44 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief ISR for transceiver's transmit end interrupt
|
|
|
|
|
*
|
|
|
|
|
* Is triggered when data or when acknowledge frames where send.
|
|
|
|
|
*
|
|
|
|
|
* Flow Diagram Manual p. 52 / 63
|
|
|
|
|
*/
|
2023-06-29 17:18:21 +02:00
|
|
|
|
static inline void txr24_tx_end_handler(void)
|
2019-10-21 01:07:44 +02:00
|
|
|
|
{
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(at86rfmega_dev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
2019-10-21 01:07:44 +02:00
|
|
|
|
uint8_t status = *AT86RF2XX_REG__TRX_STATE & AT86RF2XX_TRX_STATUS_MASK__TRX_STATUS;
|
|
|
|
|
DEBUG("TRX24_TX_END 0x%x\n", status);
|
|
|
|
|
|
|
|
|
|
/* only inform upper layer when a transmission was done,
|
|
|
|
|
* not for sending acknowledge frames if data was received. */
|
2020-04-02 16:48:34 +02:00
|
|
|
|
if (status != AT86RF2XX_PHY_STATE_RX) {
|
2019-10-21 01:07:44 +02:00
|
|
|
|
dev->irq_status |= AT86RF2XX_IRQ_STATUS_MASK__TX_END;
|
|
|
|
|
|
|
|
|
|
/* Call upper layer to process if data was send successful */
|
2020-03-05 15:11:20 +01:00
|
|
|
|
netdev_trigger_event_isr(at86rfmega_dev);
|
2019-10-21 01:07:44 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-06-29 17:18:21 +02:00
|
|
|
|
AVR8_ISR(TRX24_TX_END_vect, txr24_tx_end_handler);
|
2019-10-21 01:07:44 +02:00
|
|
|
|
|
2021-02-16 23:56:02 +01:00
|
|
|
|
/**
|
|
|
|
|
* @brief ISR for transceiver's wakeup finished interrupt
|
|
|
|
|
*
|
|
|
|
|
* Is triggered when transceiver module reset is finished.
|
|
|
|
|
* Save IRQ status and inform upper layer the transceiver is operational.
|
|
|
|
|
*
|
|
|
|
|
* Manual p. 40
|
|
|
|
|
*/
|
2023-06-29 17:18:21 +02:00
|
|
|
|
static inline void txr24_awake_handler(void)
|
2021-02-16 23:56:02 +01:00
|
|
|
|
{
|
|
|
|
|
DEBUG("TRX24_AWAKE\n");
|
|
|
|
|
|
2021-06-21 16:32:16 +02:00
|
|
|
|
netdev_ieee802154_t *netdev_ieee802154 = container_of(at86rfmega_dev,
|
|
|
|
|
netdev_ieee802154_t, netdev);
|
|
|
|
|
at86rf2xx_t *dev = container_of(netdev_ieee802154, at86rf2xx_t, netdev);
|
|
|
|
|
dev->irq_status |= AT86RF2XX_IRQ_STATUS_MASK__AWAKE;
|
2021-02-16 23:56:02 +01:00
|
|
|
|
/* Call upper layer to process transceiver wakeup finished */
|
|
|
|
|
netdev_trigger_event_isr(at86rfmega_dev);
|
|
|
|
|
}
|
2023-06-29 17:18:21 +02:00
|
|
|
|
AVR8_ISR(TRX24_AWAKE_vect, txr24_awake_handler);
|
2021-02-16 23:56:02 +01:00
|
|
|
|
|
2022-11-24 16:50:58 +01:00
|
|
|
|
#endif /* AT86RF2XX_IS_PERIPH */
|