mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-15 19:32:46 +01:00
672 lines
25 KiB
C
672 lines
25 KiB
C
|
/*
|
|||
|
* Copyright (C) 2019 Otto-von-Guericke-Universität Magdeburg
|
|||
|
*
|
|||
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|||
|
* General Public License v2.1. See the file LICENSE in the top level
|
|||
|
* directory for more details.
|
|||
|
*/
|
|||
|
/**
|
|||
|
* @ingroup drivers_nrf24l01p_ng
|
|||
|
* @{
|
|||
|
*
|
|||
|
* @file
|
|||
|
* @brief Implementation of RIOT's netdev_driver API
|
|||
|
* for the NRF24L01+ (NG) transceiver
|
|||
|
*
|
|||
|
* @author Fabian Hüßler <fabian.huessler@ovgu.de>
|
|||
|
* @}
|
|||
|
*/
|
|||
|
#include <errno.h>
|
|||
|
#include <string.h>
|
|||
|
#include <assert.h>
|
|||
|
|
|||
|
#define ENABLE_DEBUG (0)
|
|||
|
#include "debug.h"
|
|||
|
|
|||
|
#include "kernel_defines.h"
|
|||
|
#include "iolist.h"
|
|||
|
#include "irq.h"
|
|||
|
#include "luid.h"
|
|||
|
#include "mutex.h"
|
|||
|
#include "net/eui64.h"
|
|||
|
#include "net/netdev.h"
|
|||
|
#include "xtimer.h"
|
|||
|
|
|||
|
#include "gnrc_netif_nrf24l01p_ng.h"
|
|||
|
#include "nrf24l01p_ng_constants.h"
|
|||
|
#include "nrf24l01p_ng_registers.h"
|
|||
|
#include "nrf24l01p_ng_communication.h"
|
|||
|
#include "nrf24l01p_ng_states.h"
|
|||
|
#include "nrf24l01p_ng_netdev.h"
|
|||
|
#if IS_USED(MODULE_NRF24L01P_NG_DIAGNOSTICS)
|
|||
|
#include "nrf24l01p_ng_diagnostics.h"
|
|||
|
#endif
|
|||
|
|
|||
|
#define NRF24L01P_NG_FLG_IRQ \
|
|||
|
(NRF24L01P_NG_FLG_MAX_RT | NRF24L01P_NG_FLG_TX_DS | NRF24L01P_NG_FLG_RX_DR)
|
|||
|
|
|||
|
static int _init(netdev_t *netdev);
|
|||
|
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info);
|
|||
|
static int _send(netdev_t *netdev, const iolist_t *iolist);
|
|||
|
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, const void *val, size_t max_len);
|
|||
|
|
|||
|
const netdev_driver_t nrf24l01p_ng_driver = {
|
|||
|
.init = _init,
|
|||
|
.recv = _recv,
|
|||
|
.send = _send,
|
|||
|
.isr = _isr,
|
|||
|
.get = _get,
|
|||
|
.set = _set
|
|||
|
};
|
|||
|
|
|||
|
static inline
|
|||
|
void _trigger_send(const nrf24l01p_ng_t *dev)
|
|||
|
{
|
|||
|
gpio_set(dev->params.pin_ce);
|
|||
|
xtimer_usleep(NRF24L01P_NG_DELAY_US_CE_HIGH_PULSE);
|
|||
|
gpio_clear(dev->params.pin_ce);
|
|||
|
xtimer_usleep(NRF24L01P_NG_DELAY_US_TX_SETTLING);
|
|||
|
}
|
|||
|
|
|||
|
static int _assert_awake(const nrf24l01p_ng_t *dev)
|
|||
|
{
|
|||
|
return nrf24l01p_ng_reg8_read(dev, NRF24L01P_NG_REG_CONFIG) &
|
|||
|
NRF24L01P_NG_FLG_PWR_UP;
|
|||
|
}
|
|||
|
|
|||
|
static netopt_state_t _state_to_netif(nrf24l01p_ng_state_t state)
|
|||
|
{
|
|||
|
if (state == NRF24L01P_NG_STATE_POWER_DOWN) {
|
|||
|
return NETOPT_STATE_SLEEP;
|
|||
|
}
|
|||
|
if (state == NRF24L01P_NG_STATE_STANDBY_1) {
|
|||
|
return NETOPT_STATE_STANDBY;
|
|||
|
}
|
|||
|
if (state == NRF24L01P_NG_STATE_STANDBY_2) {
|
|||
|
return NETOPT_STATE_TX;
|
|||
|
}
|
|||
|
if (state == NRF24L01P_NG_STATE_TX_MODE) {
|
|||
|
return NETOPT_STATE_TX;
|
|||
|
}
|
|||
|
if (state == NRF24L01P_NG_STATE_RX_MODE) {
|
|||
|
return NETOPT_STATE_RX;
|
|||
|
}
|
|||
|
return NETOPT_STATE_OFF; /* error */
|
|||
|
}
|
|||
|
|
|||
|
nrf24l01p_ng_state_t _state_from_netif(netopt_state_t state)
|
|||
|
{
|
|||
|
if (state == NETOPT_STATE_SLEEP) {
|
|||
|
return NRF24L01P_NG_STATE_POWER_DOWN;
|
|||
|
}
|
|||
|
if (state == NETOPT_STATE_STANDBY) {
|
|||
|
return NRF24L01P_NG_STATE_STANDBY_1;
|
|||
|
}
|
|||
|
if (state == NETOPT_STATE_TX) {
|
|||
|
return NRF24L01P_NG_STATE_TX_MODE;
|
|||
|
}
|
|||
|
if (state == NETOPT_STATE_RX) {
|
|||
|
return NRF24L01P_NG_STATE_RX_MODE;
|
|||
|
}
|
|||
|
return NRF24L01P_NG_STATE_UNDEFINED;
|
|||
|
}
|
|||
|
|
|||
|
static void _nrf24l01p_ng_irq_handler(void *_dev)
|
|||
|
{
|
|||
|
nrf24l01p_ng_t *dev = (nrf24l01p_ng_t *)_dev;
|
|||
|
/* Once the IRQ pin has triggered,
|
|||
|
do not congest the thread´s
|
|||
|
message queue with IRQ events */
|
|||
|
gpio_irq_disable(dev->params.pin_irq);
|
|||
|
netdev_trigger_event_isr((netdev_t *)dev);
|
|||
|
}
|
|||
|
|
|||
|
static void _isr_max_rt(nrf24l01p_ng_t *dev)
|
|||
|
{
|
|||
|
assert(dev->state == NRF24L01P_NG_STATE_STANDBY_1 ||
|
|||
|
dev->state == NRF24L01P_NG_STATE_STANDBY_2 ||
|
|||
|
dev->state == NRF24L01P_NG_STATE_RX_MODE ||
|
|||
|
dev->state == NRF24L01P_NG_STATE_TX_MODE);
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] IRS MAX_RT");
|
|||
|
nrf24l01p_ng_acquire(dev);
|
|||
|
nrf24l01p_ng_flush_tx(dev);
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
dev->netdev.event_callback(&dev->netdev, NETDEV_EVENT_TX_NOACK);
|
|||
|
}
|
|||
|
|
|||
|
static void _isr_rx_dr(nrf24l01p_ng_t *dev)
|
|||
|
{
|
|||
|
assert(dev->state == NRF24L01P_NG_STATE_STANDBY_1 ||
|
|||
|
dev->state == NRF24L01P_NG_STATE_STANDBY_2 ||
|
|||
|
dev->state == NRF24L01P_NG_STATE_RX_MODE ||
|
|||
|
dev->state == NRF24L01P_NG_STATE_TX_MODE);
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] IRS RX_DR");
|
|||
|
/* read all RX data */
|
|||
|
nrf24l01p_ng_acquire(dev);
|
|||
|
while (!(nrf24l01p_ng_reg8_read(dev, NRF24L01P_NG_REG_FIFO_STATUS) &
|
|||
|
NRF24L01P_NG_FLG_RX_EMPTY)) {
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] ISR: read pending Rx frames");
|
|||
|
dev->netdev.event_callback(&dev->netdev, NETDEV_EVENT_RX_COMPLETE);
|
|||
|
}
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
}
|
|||
|
|
|||
|
static void _isr_tx_ds(nrf24l01p_ng_t *dev)
|
|||
|
{
|
|||
|
assert(dev->state == NRF24L01P_NG_STATE_STANDBY_1 ||
|
|||
|
dev->state == NRF24L01P_NG_STATE_STANDBY_2 ||
|
|||
|
dev->state == NRF24L01P_NG_STATE_RX_MODE ||
|
|||
|
dev->state == NRF24L01P_NG_STATE_TX_MODE);
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] IRS TX_DS");
|
|||
|
dev->netdev.event_callback(&dev->netdev, NETDEV_EVENT_TX_COMPLETE);
|
|||
|
}
|
|||
|
|
|||
|
static int _init(netdev_t *netdev)
|
|||
|
{
|
|||
|
nrf24l01p_ng_t *dev = (nrf24l01p_ng_t *)netdev;
|
|||
|
if (dev->params.config.cfg_data_rate >= NRF24L01P_NG_RF_DR_NUM_OF ||
|
|||
|
dev->params.config.cfg_crc == NRF24L01P_NG_CRC_0BYTE ||
|
|||
|
dev->params.config.cfg_channel >= NRF24L01P_NG_NUM_CHANNELS) {
|
|||
|
return -ENOTSUP;
|
|||
|
}
|
|||
|
if (spi_init_cs(dev->params.spi, dev->params.pin_cs) != SPI_OK) {
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] _init(): spi_init_cs() failed");
|
|||
|
return -EIO;
|
|||
|
}
|
|||
|
if (gpio_init(dev->params.pin_ce, GPIO_OUT) < 0) {
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] _init(): gpio_init() failed");
|
|||
|
return -EIO;
|
|||
|
}
|
|||
|
gpio_clear(dev->params.pin_ce);
|
|||
|
if (nrf24l01p_ng_acquire(dev) < 0) {
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] _init(): nrf24l01p_ng_acquire() failed");
|
|||
|
return -EIO;
|
|||
|
}
|
|||
|
if (dev->state != NRF24L01P_NG_STATE_POWER_DOWN) {
|
|||
|
nrf24l01p_ng_transition_to_power_down(dev);
|
|||
|
}
|
|||
|
/* flush internal Tx and Rx FIFO */
|
|||
|
nrf24l01p_ng_flush_tx(dev);
|
|||
|
nrf24l01p_ng_flush_rx(dev);
|
|||
|
uint8_t aw = NRF24L01P_NG_ADDR_WIDTH;
|
|||
|
const uint8_t bc[] = NRF24L01P_NG_BROADCAST_ADDR;
|
|||
|
memcpy(NRF24L01P_NG_ADDR_P0(dev), bc, aw);
|
|||
|
/* assign to pipe 0 the broadcast address*/
|
|||
|
nrf24l01p_ng_write_reg(dev, NRF24L01P_NG_REG_RX_ADDR_P0,
|
|||
|
NRF24L01P_NG_ADDR_P0(dev), aw);
|
|||
|
luid_get_lb(NRF24L01P_NG_ADDR_P1(dev), aw);
|
|||
|
/* "The LSByte must be unique for all six pipes" [datasheet p.38] */
|
|||
|
if (NRF24L01P_NG_ADDR_P1(dev)[aw - 1] == bc[aw - 1]) {
|
|||
|
luid_get_lb(NRF24L01P_NG_ADDR_P1(dev), aw);
|
|||
|
}
|
|||
|
/* assign to pipe 0 the "main" listening address */
|
|||
|
nrf24l01p_ng_write_reg(dev, NRF24L01P_NG_REG_RX_ADDR_P1,
|
|||
|
NRF24L01P_NG_ADDR_P1(dev), aw);
|
|||
|
/* set the address width */
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_SETUP_AW,
|
|||
|
NRF24L01P_NG_FLG_AW(nrf24l01p_ng_valtoe_aw(aw)));
|
|||
|
/* set Tx power and Tx data rate */
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_RF_SETUP,
|
|||
|
NRF24L01P_NG_FLG_RF_DR(dev->params.config.cfg_data_rate) |
|
|||
|
NRF24L01P_NG_FLG_RF_PWR(dev->params.config.cfg_tx_power));
|
|||
|
/* set retransmission delay and the maximum number of retransmisisons */
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_SETUP_RETR,
|
|||
|
NRF24L01P_NG_FLG_ARD(dev->params.config.cfg_retr_delay) |
|
|||
|
NRF24L01P_NG_FLG_ARC(dev->params.config.cfg_max_retr));
|
|||
|
/* set the radio channel */
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_RF_CH,
|
|||
|
NRF24L01P_NG_FLG_RF_CH(dev->params.config.cfg_channel));
|
|||
|
/* enable pipe 0 and pipe 1 */
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_EN_RXADDR,
|
|||
|
NRF24L01P_NG_FLG_ERX_P0 |
|
|||
|
NRF24L01P_NG_FLG_ERX_P1);
|
|||
|
/* set CRC length */
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_CONFIG,
|
|||
|
NRF24L01P_NG_FLG_CRCO(dev->params.config.cfg_crc));
|
|||
|
/* enable NRF24L01+ features:
|
|||
|
automatic acknowledgements,
|
|||
|
dynamic payload lengths,
|
|||
|
piggyback acknowledgements */
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_FEATURES,
|
|||
|
NRF24L01P_NG_FLG_EN_DYN_ACK |
|
|||
|
NRF24L01P_NG_FLG_EN_DPL |
|
|||
|
NRF24L01P_NG_FLG_EN_ACK_PAY);
|
|||
|
/* enable automatic acknowledgements for pipe 0 and pipe 1 */
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_EN_AA,
|
|||
|
NRF24L01P_NG_FLG_ENAA_P0 |
|
|||
|
NRF24L01P_NG_FLG_ENAA_P1);
|
|||
|
/* enable dynamic payload lengths for pipe 0 and pipe 1 */
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_DYNPD,
|
|||
|
NRF24L01P_NG_FLG_DPL_P0 |
|
|||
|
NRF24L01P_NG_FLG_DPL_P1);
|
|||
|
/* clear interrupts */
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_STATUS, NRF24L01P_NG_FLG_IRQ);
|
|||
|
/* go to standby */
|
|||
|
nrf24l01p_ng_transition_to_standby_1(dev);
|
|||
|
#if IS_USED(MODULE_NRF24L01P_NG_DIAGNOSTICS) && ENABLE_DEBUG
|
|||
|
nrf24l01p_ng_diagnostics_print_all_regs(dev);
|
|||
|
nrf24l01p_ng_diagnostics_print_dev_info(dev);
|
|||
|
#endif
|
|||
|
/* check if the transceiver responds */
|
|||
|
if (!_assert_awake(dev)) {
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
return -ENODEV;
|
|||
|
}
|
|||
|
/* go to listening state */
|
|||
|
nrf24l01p_ng_transition_to_rx_mode(dev);
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
/* enable interrupt pins */
|
|||
|
if (gpio_init_int(dev->params.pin_irq, GPIO_IN, GPIO_FALLING,
|
|||
|
_nrf24l01p_ng_irq_handler, dev) < 0) {
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] _init(): gpio_init_int() failed");
|
|||
|
return -EIO;
|
|||
|
}
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* @brief NRF24L01+ @ref netdev_driver_t::recv routine
|
|||
|
*
|
|||
|
* @pre @see nrf24l01p_ng_acquire must have been called before.
|
|||
|
* @pre Interrupts should be disabled
|
|||
|
*
|
|||
|
* The SPI bus is not acquired in this function because it is called from
|
|||
|
* @ref netdev_driver_t::isr, possibly for multiple times. If another
|
|||
|
* device acquired the SPI bus within the ISR, the ISR would block
|
|||
|
* until that device releases the bus.
|
|||
|
*
|
|||
|
* @param[in] netdev Abstract network device handle
|
|||
|
* @param[out] buf Rx buffer
|
|||
|
* @param[in] len Size of Rx buffer
|
|||
|
* @param[out] LQI and RSSI information (unused)
|
|||
|
*
|
|||
|
* @return Size of received frame in @p buf
|
|||
|
* @return Upper estimation of the frame width,
|
|||
|
* if @p buf == NULL and len == 0
|
|||
|
* @return Actual frame width,
|
|||
|
* if @p buf == NULL and @p len != 0
|
|||
|
* (frame is NOT dropped)
|
|||
|
* @retval -ENOBUFS @p buf != NULL and @p len < actual frame width
|
|||
|
* (frame is dropped)
|
|||
|
* @retval -EINVAL @p buf == NULL
|
|||
|
* (and none of the above cases are true)
|
|||
|
* @retval -ENOTSUP Malformed header
|
|||
|
* @retval 0 No data to read from Rx FIFO
|
|||
|
*/
|
|||
|
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info)
|
|||
|
{
|
|||
|
(void)info; /* nrf24l01+ supports neither lqi nor rssi */
|
|||
|
|
|||
|
/* return upper estaimation bound of frame size */
|
|||
|
if (!buf && !len) {
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] Return upper frame estimation");
|
|||
|
return NRF24L01P_NG_ADDR_WIDTH + NRF24L01P_NG_MAX_PAYLOAD_WIDTH;
|
|||
|
}
|
|||
|
nrf24l01p_ng_t *dev = (nrf24l01p_ng_t *)netdev;
|
|||
|
uint8_t pl_width;
|
|||
|
uint8_t status = nrf24l01p_ng_read_rx_pl_width(dev, &pl_width);
|
|||
|
uint8_t pno = NRF24L01P_NG_VAL_RX_P_NO(status);
|
|||
|
if (!pl_width ||
|
|||
|
pl_width > NRF24L01P_NG_MAX_PAYLOAD_WIDTH ||
|
|||
|
pno >= NRF24L01P_NG_PX_NUM_OF) {
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] RX error, flush RX FIFO");
|
|||
|
/* In some rare cases the RX payload width (R_RX_PL_WID) exceeds
|
|||
|
the maximum of 32 bytes. In that case it must be flushed.
|
|||
|
See https://devzone.nordicsemi.com/f/nordic-q-a/26489/nrf24l01-the-length-of-received-data-exceed-32
|
|||
|
and https://www.mikrocontroller.net/articles/NRF24L01_Tutorial */
|
|||
|
nrf24l01p_ng_flush_rx(dev);
|
|||
|
return 0;
|
|||
|
}
|
|||
|
uint8_t frame_len = NRF24L01P_NG_ADDR_WIDTH + pl_width;
|
|||
|
/* do NOT drop frame and return exact frame size */
|
|||
|
if (!buf) {
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] Return exact frame length");
|
|||
|
return frame_len;
|
|||
|
}
|
|||
|
/* drop frame, content in buf becomes invalid and return -ENOBUFS */
|
|||
|
if (len < frame_len) {
|
|||
|
DEBUG("[nrf24l01p_ng] Buffer too small: %u < %u, dropping frame\n",
|
|||
|
len, frame_len);
|
|||
|
uint8_t garbage[pl_width];
|
|||
|
nrf24l01p_ng_read_rx_payload(dev, garbage, pl_width);
|
|||
|
return -ENOBUFS;
|
|||
|
}
|
|||
|
/* get received frame */
|
|||
|
uint8_t dst_addr[NRF24L01P_NG_ADDR_WIDTH];
|
|||
|
if (pno == NRF24L01P_NG_P0) {
|
|||
|
memcpy(dst_addr, NRF24L01P_NG_ADDR_P0(dev), sizeof(dst_addr));
|
|||
|
}
|
|||
|
else {
|
|||
|
memcpy(dst_addr, NRF24L01P_NG_ADDR_P1(dev), sizeof(dst_addr));
|
|||
|
if (pno > NRF24L01P_NG_P1) {
|
|||
|
dst_addr[NRF24L01P_NG_ADDR_WIDTH - 1] =
|
|||
|
NRF24L01P_NG_ADDR_PX_LSB(dev, pno);
|
|||
|
}
|
|||
|
}
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] Handle received frame");
|
|||
|
uint8_t *frame = (uint8_t *)buf;
|
|||
|
memcpy(frame, dst_addr, sizeof(dst_addr));
|
|||
|
frame += sizeof(dst_addr);
|
|||
|
nrf24l01p_ng_read_rx_payload(dev, frame, pl_width);
|
|||
|
#if IS_USED(MODULE_NRF24L01P_NG_DIAGNOSTICS) && ENABLE_DEBUG
|
|||
|
nrf24l01p_ng_diagnostics_print_frame(dev, (uint8_t *)buf, frame_len);
|
|||
|
#endif
|
|||
|
DEBUG("[nrf24l01p_ng] Received frame length: %u\n", frame_len);
|
|||
|
return (int)frame_len;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* @brief NRF24L01+ @ref netdev_driver_t::send routine
|
|||
|
*
|
|||
|
* @param[in] netdev Abstract network device handle
|
|||
|
* @param[in] iolist Linked list of data to be sent, where
|
|||
|
* the base must be the destination address
|
|||
|
*
|
|||
|
* @return Size of sent payload
|
|||
|
* @retval -ENOTSUP @p iolist had no base and no next link,
|
|||
|
* or address was too big, or too short
|
|||
|
* @retval -EAGAIN Pending interrupts have been handled first
|
|||
|
* @retval -EBUSY The internal Tx FIFO is full
|
|||
|
* @retval -E2BIG Resulting frame from iolist was too big to be sent
|
|||
|
*/
|
|||
|
static int _send(netdev_t *netdev, const iolist_t *iolist)
|
|||
|
{
|
|||
|
assert(netdev && iolist);
|
|||
|
if (!(iolist->iol_base) || !(iolist->iol_next)) {
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] No Tx address or no payload");
|
|||
|
return -ENOTSUP;
|
|||
|
}
|
|||
|
nrf24l01p_ng_t *dev = (nrf24l01p_ng_t *)netdev;
|
|||
|
uint8_t pl_width = 0;
|
|||
|
const uint8_t bcast_addr[] = NRF24L01P_NG_BROADCAST_ADDR;
|
|||
|
uint8_t payload[NRF24L01P_NG_MAX_PAYLOAD_WIDTH];
|
|||
|
nrf24l01p_ng_acquire(dev);
|
|||
|
uint8_t fifo_status;
|
|||
|
uint8_t status =
|
|||
|
nrf24l01p_ng_read_reg(dev, NRF24L01P_NG_REG_FIFO_STATUS,
|
|||
|
&fifo_status, 1);
|
|||
|
if (status & NRF24L01P_NG_FLG_IRQ) {
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] Handle pending IRQ, before sending new data");
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
_isr(&dev->netdev);
|
|||
|
return -EAGAIN;
|
|||
|
}
|
|||
|
if (fifo_status & NRF24L01P_NG_FLG_TX_FULL_) {
|
|||
|
/* If the TX FIFO is full, but no ACK has arrived yet,
|
|||
|
so no TX_DS / MAX_RT interrupt has triggered so far that
|
|||
|
could clean the TX FIFO. So we need to wait until an interrupt
|
|||
|
occurs, before we can send a new frame. This is done
|
|||
|
while this _send() function is called in a loop and the
|
|||
|
interrupt status is polled.
|
|||
|
If you flush the FIFO here, pending content will
|
|||
|
be lost. */
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] TX FIFO full");
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
return -EBUSY; /* for gnrc_netif_pktq */
|
|||
|
}
|
|||
|
uint8_t *dst_addr = iolist->iol_base;
|
|||
|
uint8_t dst_addr_len = iolist->iol_len;
|
|||
|
if (dst_addr_len > NRF24L01P_NG_MAX_ADDR_WIDTH ||
|
|||
|
dst_addr_len < NRF24L01P_NG_MIN_ADDR_WIDTH) {
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
DEBUG("[nrf24l01p_ng] Destination address has an invalid length: %u\n",
|
|||
|
dst_addr_len);
|
|||
|
return -ENOTSUP;
|
|||
|
}
|
|||
|
for (const iolist_t *iol = iolist->iol_next; iol; iol = iol->iol_next) {
|
|||
|
if (pl_width + iol->iol_len > sizeof(payload)) {
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
DEBUG_PUTS("[nrf24l01p_ng] frame too big");
|
|||
|
return -E2BIG;
|
|||
|
}
|
|||
|
memcpy(payload + pl_width, iol->iol_base, iol->iol_len);
|
|||
|
pl_width += iol->iol_len;
|
|||
|
}
|
|||
|
nrf24l01p_ng_write_reg(dev, NRF24L01P_NG_REG_TX_ADDR,
|
|||
|
dst_addr, dst_addr_len);
|
|||
|
if (!memcmp(dst_addr, bcast_addr, dst_addr_len)) {
|
|||
|
/* do not expect ACK for broadcast */
|
|||
|
nrf24l01p_ng_write_tx_pl_no_ack(dev, payload, pl_width);
|
|||
|
}
|
|||
|
else {
|
|||
|
nrf24l01p_ng_write_tx_payload(dev, payload, pl_width);
|
|||
|
/* A PTX node must change pipe 0 Rx address to Tx address
|
|||
|
* in order to receive ACKs.
|
|||
|
* If node switches back to Rx mode, pipe 0 Rx address
|
|||
|
* must be restored from params. */
|
|||
|
nrf24l01p_ng_write_reg(dev, NRF24L01P_NG_REG_RX_ADDR_P0,
|
|||
|
dst_addr, dst_addr_len);
|
|||
|
nrf24l01p_ng_reg8_write(dev, NRF24L01P_NG_REG_SETUP_AW,
|
|||
|
NRF24L01P_NG_FLG_AW(nrf24l01p_ng_valtoe_aw(dst_addr_len)));
|
|||
|
}
|
|||
|
if (dev->state != NRF24L01P_NG_STATE_TX_MODE &&
|
|||
|
dev->state != NRF24L01P_NG_STATE_STANDBY_2) {
|
|||
|
dev->idle_state = dev->state;
|
|||
|
if (dev->state != NRF24L01P_NG_STATE_STANDBY_1) {
|
|||
|
nrf24l01p_ng_transition_to_standby_1(dev);
|
|||
|
}
|
|||
|
nrf24l01p_ng_transition_to_tx_mode(dev);
|
|||
|
}
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
_trigger_send(dev);
|
|||
|
DEBUG("[nrf24l01p_ng] Sending %u bytes\n", pl_width);
|
|||
|
return (int)pl_width;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* @brief NRF24L01+ @ref netdev_driver_t::isr
|
|||
|
*
|
|||
|
* @param[in] netdev Abstract network device
|
|||
|
*/
|
|||
|
static void _isr(netdev_t *netdev)
|
|||
|
{
|
|||
|
nrf24l01p_ng_t *dev = (nrf24l01p_ng_t *)netdev;
|
|||
|
|
|||
|
nrf24l01p_ng_acquire(dev);
|
|||
|
gpio_irq_enable(dev->params.pin_irq);
|
|||
|
uint8_t status = nrf24l01p_ng_get_status(dev);
|
|||
|
/* clear interrupt flags */
|
|||
|
nrf24l01p_ng_write_reg(dev, NRF24L01P_NG_REG_STATUS, &status, 1);
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
|
|||
|
if (status & NRF24L01P_NG_FLG_RX_DR) {
|
|||
|
_isr_rx_dr(dev);
|
|||
|
}
|
|||
|
if (status & NRF24L01P_NG_FLG_MAX_RT) {
|
|||
|
_isr_max_rt(dev);
|
|||
|
}
|
|||
|
if (status & NRF24L01P_NG_FLG_TX_DS) {
|
|||
|
_isr_tx_ds(dev);
|
|||
|
}
|
|||
|
|
|||
|
nrf24l01p_ng_acquire(dev);
|
|||
|
if (dev->state == NRF24L01P_NG_STATE_TX_MODE ||
|
|||
|
dev->state == NRF24L01P_NG_STATE_STANDBY_2) {
|
|||
|
/* frame in FIFO is not an ACK */
|
|||
|
if (!(nrf24l01p_ng_reg8_read(dev, NRF24L01P_NG_REG_FIFO_STATUS) &
|
|||
|
NRF24L01P_NG_FLG_TX_EMPTY)) {
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
_trigger_send(dev);
|
|||
|
return;
|
|||
|
}
|
|||
|
}
|
|||
|
/* no more data to transmit */
|
|||
|
if (dev->state != NRF24L01P_NG_STATE_STANDBY_1) {
|
|||
|
nrf24l01p_ng_transition_to_standby_1(dev);
|
|||
|
}
|
|||
|
/* go to idle state */
|
|||
|
if (dev->idle_state != NRF24L01P_NG_STATE_STANDBY_1) {
|
|||
|
if (dev->idle_state == NRF24L01P_NG_STATE_POWER_DOWN) {
|
|||
|
nrf24l01p_ng_transition_to_power_down(dev);
|
|||
|
}
|
|||
|
else {
|
|||
|
dev->idle_state = NRF24L01P_NG_STATE_RX_MODE;
|
|||
|
nrf24l01p_ng_transition_to_rx_mode(dev);
|
|||
|
}
|
|||
|
}
|
|||
|
nrf24l01p_ng_release(dev);
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* @brief @ref netdev_driver_t::get
|
|||
|
*
|
|||
|
* @param[in] netdev Abstract network device
|
|||
|
* @param[in] opt netdev option type
|
|||
|
* @param[out] val Option value
|
|||
|
* @param[in] max_len Maximum option length
|
|||
|
*
|
|||
|
* @return Size of written option value
|
|||
|
* @retval -ENOTSUP Unsupported netdev option @p opt
|
|||
|
*/
|
|||
|
static int _get(netdev_t *netdev, netopt_t opt, void *val, size_t max_len)
|
|||
|
{
|
|||
|
nrf24l01p_ng_t *dev = (nrf24l01p_ng_t *)netdev;
|
|||
|
|
|||
|
(void)max_len; /* only used in assert() */
|
|||
|
switch (opt) {
|
|||
|
case NETOPT_ADDR_LEN:
|
|||
|
case NETOPT_SRC_LEN: {
|
|||
|
assert(max_len == sizeof(uint16_t));
|
|||
|
*((uint16_t *)val) = NRF24L01P_NG_ADDR_WIDTH;
|
|||
|
return sizeof(uint16_t);
|
|||
|
} break;
|
|||
|
case NETOPT_ADDRESS: {
|
|||
|
assert(max_len >= NRF24L01P_NG_ADDR_WIDTH);
|
|||
|
memcpy(val, NRF24L01P_NG_ADDR_P1(dev), NRF24L01P_NG_ADDR_WIDTH);
|
|||
|
return NRF24L01P_NG_ADDR_WIDTH;
|
|||
|
} break;
|
|||
|
case NETOPT_AUTOACK: {
|
|||
|
assert(max_len == sizeof(netopt_enable_t));
|
|||
|
/* mandatory for Enhanced ShockBurst */
|
|||
|
*((netopt_enable_t *)val) = NETOPT_ENABLE;
|
|||
|
return sizeof(netopt_enable_t);
|
|||
|
} break;
|
|||
|
case NETOPT_CHANNEL: {
|
|||
|
assert(max_len == sizeof(uint16_t));
|
|||
|
*((uint16_t *)val) = (uint16_t)nrf24l01p_ng_get_channel(dev);
|
|||
|
return sizeof(uint16_t);
|
|||
|
} break;
|
|||
|
case NETOPT_CHECKSUM:
|
|||
|
case NETOPT_INTEGRITY_CHECK: {
|
|||
|
assert(max_len == sizeof(netopt_enable_t));
|
|||
|
/* mandatory for Enhanced ShockBurst */
|
|||
|
*((netopt_enable_t *)val) = NETOPT_ENABLE;
|
|||
|
return sizeof(netopt_enable_t);
|
|||
|
} break;
|
|||
|
case NETOPT_DEVICE_TYPE: {
|
|||
|
assert(max_len == sizeof(uint16_t));
|
|||
|
*((uint16_t *)val) = NETDEV_TYPE_NRF24L01P_NG;
|
|||
|
return sizeof(uint16_t);
|
|||
|
} break;
|
|||
|
case NETOPT_PROTO: {
|
|||
|
assert(max_len == sizeof(gnrc_nettype_t));
|
|||
|
*((gnrc_nettype_t *)val) = NRF24L01P_NG_UPPER_LAYER_PROTOCOL;
|
|||
|
return sizeof(gnrc_nettype_t);
|
|||
|
} break;
|
|||
|
case NETOPT_MAX_PDU_SIZE: {
|
|||
|
assert(max_len == sizeof(uint16_t));
|
|||
|
*((uint16_t *)val) = NRF24L01P_NG_MAX_PAYLOAD_WIDTH -
|
|||
|
NRF24L01P_NG_ADDR_WIDTH
|
|||
|
- 1;
|
|||
|
return sizeof(uint16_t);
|
|||
|
}
|
|||
|
case NETOPT_RETRANS: {
|
|||
|
assert(max_len == sizeof(uint8_t));
|
|||
|
*((uint8_t *)val) = nrf24l01p_ng_get_max_retransm(dev);
|
|||
|
return sizeof(uint8_t);
|
|||
|
}
|
|||
|
case NETOPT_RX_TIMEOUT: {
|
|||
|
assert(max_len == sizeof(uint32_t));
|
|||
|
*((uint32_t *)val) =
|
|||
|
(uint32_t)nrf24l01p_ng_get_retransm_delay(dev, NULL);
|
|||
|
return sizeof(uint32_t);
|
|||
|
} break;
|
|||
|
case NETOPT_STATE: {
|
|||
|
assert(max_len == sizeof(netopt_state_t));
|
|||
|
*((netopt_state_t *)val) = _state_to_netif(dev->state);
|
|||
|
return sizeof(netopt_state_t);
|
|||
|
} break;
|
|||
|
case NETOPT_TX_POWER: {
|
|||
|
assert(max_len == sizeof(int16_t));
|
|||
|
*((int16_t *)val) = (int16_t)nrf24l01p_ng_get_tx_power(dev, NULL);
|
|||
|
return sizeof(uint16_t);
|
|||
|
} break;
|
|||
|
default:
|
|||
|
DEBUG("[nrf24l01p_ng] Unsupported netdev option %d\n", opt);
|
|||
|
return -ENOTSUP;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* @brief @ref netdev_driver_t::set
|
|||
|
*
|
|||
|
* @param[in] netdev Abstract network device handle
|
|||
|
* @param[in] opt netdev option type
|
|||
|
* @param[in] val Option value
|
|||
|
* @param[in] len Size of option value
|
|||
|
*
|
|||
|
* @return Size of written option value
|
|||
|
* @return negative number, on failure
|
|||
|
* @retval -ENOTSUP Unsupported netdev option @p opt
|
|||
|
*/
|
|||
|
static int _set(netdev_t *netdev, netopt_t opt, const void *val, size_t len)
|
|||
|
{
|
|||
|
nrf24l01p_ng_t *dev = (nrf24l01p_ng_t *)netdev;
|
|||
|
|
|||
|
switch (opt) {
|
|||
|
case NETOPT_ADDRESS: {
|
|||
|
/* common address length for all pipes */
|
|||
|
assert(len == NRF24L01P_NG_ADDR_WIDTH);
|
|||
|
int ret = nrf24l01p_ng_set_rx_address(dev, val, NRF24L01P_NG_P0);
|
|||
|
return ret ? ret : (int)len;
|
|||
|
} break;
|
|||
|
case NETOPT_CHANNEL: {
|
|||
|
assert(len == sizeof(uint16_t));
|
|||
|
uint16_t ch = *((uint16_t *)val);
|
|||
|
int ret = nrf24l01p_ng_set_channel(dev, (uint8_t)ch);
|
|||
|
return ret ? ret : (int)sizeof(uint16_t);
|
|||
|
} break;
|
|||
|
case NETOPT_CHECKSUM:
|
|||
|
case NETOPT_INTEGRITY_CHECK: {
|
|||
|
assert(len == sizeof(netopt_enable_t));
|
|||
|
nrf24l01p_ng_crc_t crc =
|
|||
|
(*((netopt_enable_t *)val) == NETOPT_ENABLE)
|
|||
|
? NRF24L01P_NG_CRC_2BYTE : NRF24L01P_NG_CRC_0BYTE;
|
|||
|
int ret = nrf24l01p_ng_set_crc(dev, crc);
|
|||
|
return ret ? ret : (int)sizeof(netopt_enable_t);
|
|||
|
} break;
|
|||
|
case NETOPT_RETRANS: {
|
|||
|
assert(len == sizeof(uint8_t));
|
|||
|
uint8_t n = *((uint8_t *)val);
|
|||
|
int ret = nrf24l01p_ng_set_max_retransm(dev, n);
|
|||
|
return ret ? ret : (int)sizeof(uint8_t);
|
|||
|
} break;
|
|||
|
case NETOPT_RX_TIMEOUT: {
|
|||
|
assert(len == sizeof(uint32_t));
|
|||
|
uint32_t us = *((uint32_t *)val);
|
|||
|
nrf24l01p_ng_ard_t ard = nrf24l01p_ng_valtoe_ard(us);
|
|||
|
int ret = nrf24l01p_ng_set_retransm_delay(dev, ard);
|
|||
|
return ret ? ret : (int)sizeof(uint32_t);
|
|||
|
} break;
|
|||
|
case NETOPT_STATE: {
|
|||
|
assert(len == sizeof(netopt_state_t));
|
|||
|
nrf24l01p_ng_state_t s =
|
|||
|
_state_from_netif(*((netopt_state_t *)val));
|
|||
|
int ret = nrf24l01p_ng_set_state(dev, s);
|
|||
|
return ret < 0 ? ret : (int)sizeof(netopt_state_t);
|
|||
|
} break;
|
|||
|
case NETOPT_TX_POWER: {
|
|||
|
assert(len == sizeof(int16_t));
|
|||
|
int16_t dbm = *((int16_t *)val);
|
|||
|
nrf24l01p_ng_tx_power_t txp = nrf24l01p_ng_valtoe_tx_power(dbm);
|
|||
|
int ret = nrf24l01p_ng_set_tx_power(dev, txp);
|
|||
|
return ret ? ret : (int)sizeof(int16_t);
|
|||
|
} break;
|
|||
|
default:
|
|||
|
DEBUG("[nrf24l01p_ng] Unsupported netdev option %d\n", opt);
|
|||
|
return -ENOTSUP;
|
|||
|
}
|
|||
|
}
|