1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/drivers/kw41zrf/kw41zrf_netdev.c
Marian Buschsieweke 134f323fb1
drivers/kw41zrf: make TX/RX IRQs read only
This brings the implementation in sync with the API.
2021-03-22 08:15:54 +01:00

1215 lines
39 KiB
C

/*
* Copyright (C) 2017 SKF AB
*
* 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_kw41zrf
* @{
*
* @file
* @brief Netdev interface for kw41zrf drivers
*
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
*/
#include <errno.h>
#include <stdint.h>
#include <string.h>
#include <assert.h>
#include <stdatomic.h>
#include "log.h"
#include "random.h"
#include "thread_flags.h"
#include "net/eui64.h"
#include "net/ieee802154.h"
#include "net/netdev.h"
#include "net/netdev/ieee802154.h"
#include "kw41zrf.h"
#include "kw41zrf_netdev.h"
#include "kw41zrf_intern.h"
#include "kw41zrf_getset.h"
#include "vendor/MKW41Z4.h"
#ifdef MODULE_OD
#include "od.h"
#endif
#define ENABLE_DEBUG 0
#include "debug.h"
#define _MAX_MHR_OVERHEAD (25)
/* Timing units are in radio timer ticks (16 usec per tick) */
#define KW41ZRF_CCA_TIME 8
#define KW41ZRF_SHR_PHY_TIME 12
#define KW41ZRF_PER_BYTE_TIME 2
#define KW41ZRF_ACK_WAIT_TIME 54
#define KW41ZRF_CSMA_UNIT_TIME 20
static void kw41zrf_netdev_isr(netdev_t *netdev);
/* True while we've already queued a callback and don't need to queue another */
static atomic_bool irq_is_queued = false;
/* True while blocking the netdev thread waiting for some operation to finish */
static bool blocking_for_irq = false;
/* Set this to a flag bit that is not used by the MAC implementation */
#define KW41ZRF_THREAD_FLAG_ISR (1u << 8)
static void kw41zrf_irq_handler(void *arg)
{
netdev_t *netdev = arg;
kw41zrf_t *dev = (kw41zrf_t *)netdev;
KW41ZRF_LED_IRQ_ON;
kw41zrf_mask_irqs();
/* Signal to the thread that an IRQ has arrived, if it is waiting */
thread_flags_set(dev->thread, KW41ZRF_THREAD_FLAG_ISR);
/* Avoid filling the message queue with redundant ISR events */
if (!irq_is_queued) {
irq_is_queued = true;
if (netdev->event_callback) {
netdev->event_callback(netdev, NETDEV_EVENT_ISR);
}
}
}
static int kw41zrf_netdev_init(netdev_t *netdev)
{
kw41zrf_t *dev = (kw41zrf_t *)netdev;
dev->thread = thread_get_active();
/* initialize hardware */
if (kw41zrf_init(dev, kw41zrf_irq_handler)) {
LOG_ERROR("[kw41zrf] unable to initialize device\n");
return -1;
}
return 0;
}
/**
* @brief Generate a random number for using as a CSMA delay value
*/
static inline uint32_t kw41zrf_csma_random_delay(kw41zrf_t *dev)
{
/* Use topmost csma_be bits of the random number */
uint32_t rnd = random_uint32() >> (32 - dev->csma_be);
return (rnd * KW41ZRF_CSMA_UNIT_TIME);
}
static inline size_t kw41zrf_tx_load(const void *buf, size_t len, size_t offset)
{
/* Array bounds are checked in the kw41zrf_netdev_send loop. */
/* offset + 1 is used because buf[0] contains the frame length byte. */
/* Don't use memcpy to work around a presumed compiler bug in
* arm-none-eabi-gcc 7.3.1 2018-q2-6 */
for (unsigned i = 0; i < len; i++) {
((uint8_t *)ZLL->PKT_BUFFER_TX)[i + offset + 1] = ((uint8_t *)buf)[i];
}
return offset + len;
}
/**
* @brief set up TMR2 to trigger the TX sequence from the ISR
*/
static void kw41zrf_tx_exec(kw41zrf_t *dev)
{
kw41zrf_abort_sequence(dev);
if (dev->flags & KW41ZRF_OPT_CSMA) {
/* Use CSMA/CA random delay in the interval [0, 2**dev->csma_be) */
dev->backoff_delay = kw41zrf_csma_random_delay(dev);
}
/* less than 2 is sometimes in the past */
if (dev->backoff_delay < 2) {
dev->backoff_delay = 2;
}
/* Avoid risk of setting a timer in the past */
int irq = irq_disable();
kw41zrf_timer_set(dev, &ZLL->T2CMP, ~0ul);
/* enable TMR2 */
bit_clear32(&ZLL->PHY_CTRL, ZLL_PHY_CTRL_TMRTRIGEN_SHIFT);
bit_set32(&ZLL->PHY_CTRL, ZLL_PHY_CTRL_TMR2CMP_EN_SHIFT);
/* Set real trigger time */
kw41zrf_timer_set(dev, &ZLL->T2CMP, dev->backoff_delay);
irq_restore(irq);
}
/**
* @brief Block the current thread until the radio is idle
*
* Any ongoing TX or CCA sequence will have finished when this function returns.
*
* @param[in] dev kw41zrf device descriptor
*/
static void kw41zrf_wait_idle(kw41zrf_t *dev)
{
/* things that trigger this must run only from the netdev thread */
assert(thread_getpid() == dev->thread->pid);
/* make sure any ongoing T or TR sequence is finished */
if (kw41zrf_can_switch_to_idle(dev) && dev->backoff_delay == 0) {
return;
}
DEBUG("[kw41zrf] waiting for idle\n");
/* we exit this wait using an IRQ so we can't do it from IRQ */
assert(!irq_is_in());
/* in case we're servicing an IRQ currently, IRQs will be masked */
kw41zrf_unmask_irqs();
assert(!blocking_for_irq);
blocking_for_irq = true;
PM_BLOCK(KW41ZRF_PM_BLOCKER);
while (1) {
/* TX or CCA or CSMA backoff in progress */
/* Block until we get an IRQ */
thread_flags_wait_any(KW41ZRF_THREAD_FLAG_ISR);
/* Handle the IRQ */
kw41zrf_netdev_isr((netdev_t *)dev);
/* kw41zrf_netdev_isr() will switch the transceiver back to idle
* after handling the sequence complete IRQ */
if (kw41zrf_can_switch_to_idle(dev) && dev->backoff_delay == 0) {
break;
}
}
DEBUG("[kw41zrf] waited ISR\n");
PM_UNBLOCK(KW41ZRF_PM_BLOCKER);
blocking_for_irq = false;
}
int kw41zrf_cca(kw41zrf_t *dev)
{
kw41zrf_wait_idle(dev);
if (kw41zrf_is_dsm()) {
/* bring the device out of DSM */
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_IDLE);
}
kw41zrf_abort_sequence(dev);
kw41zrf_unmask_irqs();
KW41ZRF_LED_RX_ON;
kw41zrf_set_sequence(dev, XCVSEQ_CCA);
/* Wait for the CCA to finish, it will take exactly RX warmup time + 128 µs */
kw41zrf_wait_idle(dev);
KW41ZRF_LED_RX_OFF;
DEBUG("[kw41zrf] CCA: %u RSSI: %d\n", (unsigned)dev->cca_result,
kw41zrf_get_ed_level(dev));
return dev->cca_result;
}
static int kw41zrf_netdev_send(netdev_t *netdev, const iolist_t *iolist)
{
kw41zrf_t *dev = (kw41zrf_t *)netdev;
size_t len = 0;
kw41zrf_wait_idle(dev);
if (kw41zrf_is_dsm()) {
/* bring the device out of DSM */
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_IDLE);
}
/* load packet data into buffer */
for (const iolist_t *iol = iolist; iol; iol = iol->iol_next) {
/* current packet data + FCS too long */
if ((len + iol->iol_len) > (KW41ZRF_MAX_PKT_LENGTH - IEEE802154_FCS_LEN)) {
LOG_ERROR("[kw41zrf] packet too large (%u byte) to fit\n",
(unsigned)len + IEEE802154_FCS_LEN);
return -EOVERFLOW;
}
len = kw41zrf_tx_load(iol->iol_base, iol->iol_len, len);
}
DEBUG("[kw41zrf] TX %u bytes\n", len);
/*
* First octet in the TX buffer contains the frame length.
* Nbytes = FRAME_LEN - 2 -> FRAME_LEN = Nbytes + 2
* MKW41Z ref. man. 44.6.2.6.3.1.3 Sequence T (Transmit), p. 2147
*/
*((volatile uint8_t *)&ZLL->PKT_BUFFER_TX[0]) = len + IEEE802154_FCS_LEN;
#if defined(MODULE_OD) && ENABLE_DEBUG
DEBUG("[kw41zrf] send:\n");
od_hex_dump((const uint8_t *)ZLL->PKT_BUFFER_TX, len, OD_WIDTH_DEFAULT);
#endif
/* send data out directly if pre-loading is disabled */
if (!(dev->flags & KW41ZRF_OPT_PRELOADING)) {
dev->csma_be = dev->csma_min_be;
dev->csma_num_backoffs = 0;
dev->num_retrans = 0;
kw41zrf_tx_exec(dev);
}
return (int)len;
}
static inline void kw41zrf_unblock_rx(kw41zrf_t *dev)
{
dev->recv_blocked = 0;
if (kw41zrf_can_switch_to_idle(dev)) {
kw41zrf_abort_sequence(dev);
kw41zrf_set_sequence(dev, dev->idle_seq);
}
KW41ZRF_LED_RX_OFF;
}
static int kw41zrf_netdev_recv(netdev_t *netdev, void *buf, size_t len, void *info)
{
kw41zrf_t *dev = (kw41zrf_t *)netdev;
if (kw41zrf_is_dsm()) {
/* bring the device out of DSM, sleep will be restored before returning */
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_IDLE);
}
/* get size of the received packet */
uint8_t pkt_len = (ZLL->IRQSTS & ZLL_IRQSTS_RX_FRAME_LENGTH_MASK) >> ZLL_IRQSTS_RX_FRAME_LENGTH_SHIFT;
if (pkt_len < IEEE802154_FCS_LEN) {
kw41zrf_unblock_rx(dev);
return -EAGAIN;
}
/* skip FCS */
pkt_len -= IEEE802154_FCS_LEN;
DEBUG("[kw41zrf] RX %u bytes\n", pkt_len);
/* just return length when buf == NULL */
if (buf == NULL) {
if (len > 0) {
/* discard what we have stored in the buffer, unblock RX */
kw41zrf_unblock_rx(dev);
}
/* No set_sequence(idle_seq) here, keep transceiver turned on if the
* buffer was not discarded, we expect the higher layer to call again
* shortly with a proper buffer */
return pkt_len;
}
#if defined(MODULE_OD) && ENABLE_DEBUG
DEBUG("[kw41zrf] recv:\n");
od_hex_dump((const uint8_t *)ZLL->PKT_BUFFER_RX, pkt_len, OD_WIDTH_DEFAULT);
#endif
if (pkt_len > len) {
/* not enough space in buf */
/* discard what we have stored in the buffer, unblock RX */
kw41zrf_unblock_rx(dev);
return -ENOBUFS;
}
/* Read packet buffer. */
/* Don't use memcpy to work around a presumed compiler bug in
* arm-none-eabi-gcc 7.3.1 2018-q2-6 */
for (int i = 0; i < pkt_len; i++) {
((uint8_t *)buf)[i] = ((uint8_t *)ZLL->PKT_BUFFER_RX)[i];
}
if (info != NULL) {
netdev_ieee802154_rx_info_t *radio_info = info;
radio_info->lqi = (ZLL->LQI_AND_RSSI & ZLL_LQI_AND_RSSI_LQI_VALUE_MASK)
>> ZLL_LQI_AND_RSSI_LQI_VALUE_SHIFT;
radio_info->rssi = (int8_t)((ZLL->LQI_AND_RSSI & ZLL_LQI_AND_RSSI_RSSI_MASK)
>> ZLL_LQI_AND_RSSI_RSSI_SHIFT);
}
/* Go back to RX mode */
kw41zrf_unblock_rx(dev);
return pkt_len;
}
static int kw41zrf_netdev_set_state(kw41zrf_t *dev, netopt_state_t state)
{
kw41zrf_wait_idle(dev);
switch (state) {
case NETOPT_STATE_OFF:
/* There is no deeper 'off' mode than deep sleep mode */
/* fall through */
case NETOPT_STATE_SLEEP:
if (kw41zrf_is_dsm()) {
/* Transceiver is already in deep sleep mode */
break;
}
kw41zrf_abort_sequence(dev);
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_DSM);
dev->idle_seq = XCVSEQ_DSM_IDLE;
break;
case NETOPT_STATE_STANDBY:
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_IDLE);
kw41zrf_abort_sequence(dev);
dev->idle_seq = XCVSEQ_IDLE;
kw41zrf_set_sequence(dev, dev->idle_seq);
break;
case NETOPT_STATE_IDLE:
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_IDLE);
kw41zrf_abort_sequence(dev);
dev->idle_seq = XCVSEQ_RECEIVE;
kw41zrf_set_sequence(dev, dev->idle_seq);
break;
case NETOPT_STATE_TX:
if (dev->flags & KW41ZRF_OPT_PRELOADING) {
kw41zrf_wait_idle(dev);
if (kw41zrf_is_dsm()) {
/* bring the device out of DSM */
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_IDLE);
}
dev->csma_be = dev->csma_min_be;
dev->csma_num_backoffs = 0;
dev->num_retrans = 0;
kw41zrf_tx_exec(dev);
}
break;
case NETOPT_STATE_RESET:
kw41zrf_reset(dev);
break;
default:
return -ENOTSUP;
}
return sizeof(netopt_state_t);
}
static netopt_state_t kw41zrf_netdev_get_state(kw41zrf_t *dev)
{
(void) dev;
/* ZLL register access require that the transceiver is powered on and not in
* deep sleep mode */
if (kw41zrf_is_dsm()) {
/* Transceiver is in deep sleep mode */
return NETOPT_STATE_SLEEP;
}
uint32_t seq = (ZLL->PHY_CTRL & ZLL_PHY_CTRL_XCVSEQ_MASK) >> ZLL_PHY_CTRL_XCVSEQ_SHIFT;
switch (seq) {
case XCVSEQ_TRANSMIT:
case XCVSEQ_TX_RX:
return NETOPT_STATE_TX;
case XCVSEQ_CCA:
case XCVSEQ_CONTINUOUS_CCA:
return NETOPT_STATE_RX;
case XCVSEQ_RECEIVE:
{
uint32_t seq_state = ZLL->SEQ_STATE;
if (seq_state & ZLL_SEQ_STATE_SFD_DET_MASK) {
/* SFD detection has been triggered */
if (seq_state & ZLL_SEQ_STATE_RX_BYTE_COUNT_MASK) {
/* packet reception is in progress */
return NETOPT_STATE_RX;
}
}
/* NETOPT_STATE_IDLE means on, and listening for incoming packets */
return NETOPT_STATE_IDLE;
}
case XCVSEQ_IDLE:
/* SEQ_IDLE in kw41z means on, but not listening for incoming traffic */
return NETOPT_STATE_STANDBY;
default:
/* Unknown state */
LOG_ERROR("[kw41z] in unknown sequence: 0x%02" PRIx32 "\n", seq);
return NETOPT_STATE_OFF;
}
}
int kw41zrf_netdev_get(netdev_t *netdev, netopt_t opt, void *value, size_t len)
{
kw41zrf_t *dev = (kw41zrf_t *)netdev;
if (dev == NULL) {
return -ENODEV;
}
/* These settings do not require the transceiver to be powered on */
switch (opt) {
case NETOPT_STATE:
assert(len >= sizeof(netopt_state_t));
*((netopt_state_t *)value) = kw41zrf_netdev_get_state(dev);
return sizeof(netopt_state_t);
case NETOPT_PRELOADING:
assert(len >= sizeof(netopt_enable_t));
*((netopt_enable_t *)value) =
!!(dev->flags & KW41ZRF_OPT_PRELOADING);
return sizeof(netopt_enable_t);
case NETOPT_PROMISCUOUSMODE:
assert(len >= sizeof(netopt_enable_t));
*((netopt_enable_t *)value) =
!!(dev->flags & KW41ZRF_OPT_PROMISCUOUS);
return sizeof(netopt_enable_t);
case NETOPT_RX_START_IRQ:
case NETOPT_RX_END_IRQ:
case NETOPT_TX_START_IRQ:
case NETOPT_TX_END_IRQ:
assert(len >= sizeof(netopt_enable_t));
*((netopt_enable_t *)value) = NETOPT_ENABLE;
return sizeof(netopt_enable_t);
case NETOPT_CSMA:
assert(len >= sizeof(netopt_enable_t));
*((netopt_enable_t *)value) =
!!(dev->flags & KW41ZRF_OPT_CSMA);
return sizeof(netopt_enable_t);
case NETOPT_CSMA_RETRIES:
assert(len >= sizeof(uint8_t));
*((uint8_t *)value) = dev->csma_max_backoffs;
return sizeof(uint8_t);
case NETOPT_CSMA_MAXBE:
assert(len >= sizeof(uint8_t));
*((uint8_t *)value) = dev->csma_max_be;
return sizeof(uint8_t);
case NETOPT_CSMA_MINBE:
assert(len >= sizeof(uint8_t));
*((uint8_t *)value) = dev->csma_min_be;
return sizeof(uint8_t);
case NETOPT_RETRANS:
assert(len >= sizeof(uint8_t));
*((uint8_t *)value) = dev->max_retrans;
return sizeof(uint8_t);
case NETOPT_TX_RETRIES_NEEDED:
assert(len >= sizeof(uint8_t));
*((uint8_t *)value) = dev->num_retrans;
return sizeof(uint8_t);
case NETOPT_CHANNEL_PAGE:
assert(len >= sizeof(uint16_t));
*((uint16_t *)value) = 0;
return sizeof(uint16_t);
default:
break;
}
/* The below settings require the transceiver to be powered on */
unsigned put_to_sleep_when_done = 0;
if (kw41zrf_is_dsm()) {
/* Transceiver is in deep sleep mode */
switch (opt) {
case NETOPT_CHANNEL:
case NETOPT_NID:
case NETOPT_ADDRESS:
case NETOPT_ADDRESS_LONG:
case NETOPT_TX_POWER:
case NETOPT_IS_CHANNEL_CLR:
case NETOPT_CCA_THRESHOLD:
case NETOPT_CCA_MODE:
case NETOPT_LAST_ED_LEVEL:
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_IDLE);
#ifdef MODULE_NETOPT
DEBUG("[kw41zrf] Wake to get opt %s\n", netopt2str(opt));
#else
DEBUG("[kw41zrf] Wake to get opt %d\n", (int)opt);
#endif
put_to_sleep_when_done = 1;
break;
default:
break;
}
}
else {
/* Wait for oscillator ready signal if the CPU is coming out of low
* power mode */
while ((RSIM->CONTROL & RSIM_CONTROL_RF_OSC_READY_MASK) == 0) {}
}
int res = -ENOTSUP;
switch (opt) {
case NETOPT_CHANNEL:
assert(len >= sizeof(uint16_t));
*((uint16_t *)value) = (uint16_t)kw41zrf_get_channel(dev);
res = sizeof(uint16_t);
break;
case NETOPT_NID:
assert(len >= sizeof(uint16_t));
*((uint16_t *)value) = kw41zrf_get_pan(dev);
res = sizeof(uint16_t);
break;
case NETOPT_ADDRESS:
assert(len >= sizeof(network_uint16_t));
kw41zrf_get_addr_short(dev, value);
res = sizeof(network_uint16_t);
break;
case NETOPT_ADDRESS_LONG:
assert(len >= sizeof(eui64_t));
kw41zrf_get_addr_long(dev, value);
*(uint64_t*)value = byteorder_swapll(*(uint64_t*)value);
res = sizeof(eui64_t);
break;
case NETOPT_TX_POWER:
assert(len >= sizeof(int16_t));
*((int16_t *)value) = kw41zrf_get_txpower(dev);
res = sizeof(int16_t);
break;
case NETOPT_IS_CHANNEL_CLR:
assert(len >= sizeof(netopt_enable_t));
*((netopt_enable_t *)value) = !(kw41zrf_cca(dev));
res = sizeof(netopt_enable_t);
break;
case NETOPT_CCA_THRESHOLD:
assert(len >= sizeof(int8_t));
*((int8_t *)value) = kw41zrf_get_cca_threshold(dev);
res = sizeof(int8_t);
break;
case NETOPT_CCA_MODE:
assert(len >= sizeof(uint8_t));
uint8_t mode = kw41zrf_get_cca_mode(dev);
switch (mode) {
case NETDEV_IEEE802154_CCA_MODE_1:
case NETDEV_IEEE802154_CCA_MODE_2:
case NETDEV_IEEE802154_CCA_MODE_3:
*((uint8_t *)value) = mode;
res = sizeof(uint8_t);
break;
default:
res = -EINVAL;
break;
}
break;
case NETOPT_LAST_ED_LEVEL:
assert(len >= sizeof(int8_t));
*((int8_t *)value) = kw41zrf_get_ed_level(dev);
res = sizeof(int8_t);
break;
default:
break;
}
if (put_to_sleep_when_done) {
DEBUG("[kw41zrf] Go back to sleep\n");
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_DSM);
}
if (res == -ENOTSUP) {
res = netdev_ieee802154_get((netdev_ieee802154_t *)netdev, opt, value, len);
}
return res;
}
static int kw41zrf_netdev_set(netdev_t *netdev, netopt_t opt, const void *value, size_t len)
{
kw41zrf_t *dev = (kw41zrf_t *)netdev;
int res = -ENOTSUP;
if (dev == NULL) {
return -ENODEV;
}
kw41zrf_wait_idle(dev);
/* These settings do not require the transceiver to be awake */
switch (opt) {
case NETOPT_STATE:
assert(len <= sizeof(const netopt_state_t));
res = kw41zrf_netdev_set_state(dev, *((const netopt_state_t *)value));
break;
case NETOPT_PRELOADING:
assert(len <= sizeof(const netopt_enable_t));
kw41zrf_set_option(dev, KW41ZRF_OPT_PRELOADING,
*((const netopt_enable_t *)value));
res = sizeof(const netopt_enable_t);
break;
case NETOPT_CSMA_RETRIES:
assert(len <= sizeof(uint8_t));
dev->csma_max_backoffs = *((const uint8_t*)value);
res = sizeof(uint8_t);
break;
case NETOPT_CSMA_MAXBE:
assert(len <= sizeof(uint8_t));
dev->csma_max_be = *((const uint8_t*)value);
res = sizeof(uint8_t);
break;
case NETOPT_CSMA_MINBE:
assert(len <= sizeof(uint8_t));
dev->csma_min_be = *((const uint8_t*)value);
res = sizeof(uint8_t);
break;
case NETOPT_RETRANS:
assert(len <= sizeof(uint8_t));
dev->max_retrans = *((const uint8_t *)value);
res = sizeof(uint8_t);
break;
default:
break;
}
unsigned put_to_sleep_when_done = 0;
if (kw41zrf_is_dsm()) {
/* Transceiver is in deep sleep mode, check if setting the option
* requires the radio powered on */
switch (opt) {
case NETOPT_AUTOACK:
case NETOPT_PROMISCUOUSMODE:
case NETOPT_RX_START_IRQ:
case NETOPT_CSMA:
case NETOPT_ADDRESS:
case NETOPT_ADDRESS_LONG:
case NETOPT_NID:
case NETOPT_CHANNEL:
case NETOPT_TX_POWER:
case NETOPT_CCA_THRESHOLD:
case NETOPT_CCA_MODE:
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_IDLE);
#ifdef MODULE_NETOPT
DEBUG("[kw41zrf] Wake to set opt %s\n", netopt2str(opt));
#else
DEBUG("[kw41zrf] Wake to set opt %d\n", (int)opt);
#endif
put_to_sleep_when_done = 1;
break;
default:
break;
}
}
switch (opt) {
case NETOPT_AUTOACK:
/* Set up HW generated automatic ACK after Receive */
assert(len <= sizeof(const netopt_enable_t));
kw41zrf_set_option(dev, KW41ZRF_OPT_AUTOACK,
*((const netopt_enable_t *)value));
res = sizeof(const netopt_enable_t);
break;
case NETOPT_ACK_PENDING:
/* Enable pending bit on automatic hardware ACKs */
assert(len <= sizeof(const netopt_enable_t));
kw41zrf_set_option(dev, KW41ZRF_OPT_ACK_PENDING,
*((const netopt_enable_t *)value));
res = sizeof(const netopt_enable_t);
break;
case NETOPT_PROMISCUOUSMODE:
assert(len <= sizeof(const netopt_enable_t));
kw41zrf_set_option(dev, KW41ZRF_OPT_PROMISCUOUS,
*((const netopt_enable_t *)value));
res = sizeof(const netopt_enable_t);
break;
case NETOPT_CSMA:
assert(len <= sizeof(const netopt_enable_t));
kw41zrf_set_option(dev, KW41ZRF_OPT_CSMA,
((const netopt_enable_t *)value)[0]);
res = sizeof(const netopt_enable_t);
break;
case NETOPT_ADDRESS:
assert(len <= sizeof(const network_uint16_t));
kw41zrf_set_addr_short(dev, value);
res = sizeof(const network_uint16_t);
break;
case NETOPT_ADDRESS_LONG: {
eui64_t addr;
assert(len <= sizeof(const eui64_t));
addr.uint64.u64 = byteorder_swapll(*(uint64_t*)value);
kw41zrf_set_addr_long(dev, &addr);
res = sizeof(const eui64_t);
break;
}
case NETOPT_NID:
assert(len <= sizeof(const uint16_t));
kw41zrf_set_pan(dev, *((const uint16_t *)value));
/* TODO uncomment this when the upper layer code is refactored to
* not need netdev_t::pan */
//~ res = sizeof(const uint16_t);
break;
case NETOPT_CHANNEL:
assert(len <= sizeof(const uint16_t));
if (kw41zrf_set_channel(dev, *((const uint16_t *)value))) {
res = -EINVAL;
break;
}
res = sizeof(const uint16_t);
kw41zrf_abort_sequence(dev);
kw41zrf_set_sequence(dev, dev->idle_seq);
break;
case NETOPT_TX_POWER:
assert(len <= sizeof(const int16_t));
kw41zrf_set_tx_power(dev, *(const int16_t *)value);
res = sizeof(const int16_t);
break;
case NETOPT_CCA_THRESHOLD:
assert(len <= sizeof(const uint8_t));
kw41zrf_set_cca_threshold(dev, *((const uint8_t*)value));
res = sizeof(const uint8_t);
break;
case NETOPT_CCA_MODE:
assert(len <= sizeof(const uint8_t));
uint8_t mode = *((const uint8_t*)value);
switch (mode) {
case NETDEV_IEEE802154_CCA_MODE_1:
case NETDEV_IEEE802154_CCA_MODE_2:
case NETDEV_IEEE802154_CCA_MODE_3:
kw41zrf_set_cca_mode(dev, mode);
res = sizeof(const uint8_t);
break;
case NETDEV_IEEE802154_CCA_MODE_4:
case NETDEV_IEEE802154_CCA_MODE_5:
case NETDEV_IEEE802154_CCA_MODE_6:
default:
res = -EINVAL;
break;
}
break;
default:
break;
}
if (put_to_sleep_when_done) {
DEBUG("[kw41zrf] Go back to sleep\n");
kw41zrf_set_power_mode(dev, KW41ZRF_POWER_DSM);
}
if (res == -ENOTSUP) {
res = netdev_ieee802154_set((netdev_ieee802154_t *)netdev, opt, value, len);
}
return res;
}
/* Common CCA check handler code for sequences Transmit and Transmit/Receive */
static uint32_t _isr_event_seq_t_ccairq(kw41zrf_t *dev, uint32_t irqsts)
{
uint32_t handled_irqs = 0;
if (irqsts & ZLL_IRQSTS_TMR2IRQ_MASK) {
assert(!(irqsts & ZLL_IRQSTS_CCAIRQ_MASK));
handled_irqs |= ZLL_IRQSTS_TMR2IRQ_MASK;
dev->tx_timeout = 0;
/* Check FCF field in the TX buffer to see if the ACK_REQ flag was set in
* the packet that is queued for transmission */
uint16_t len_fcf = ZLL->PKT_BUFFER_TX[0];
uint8_t fcf = (len_fcf >> 8) & 0xff;
if ((fcf & IEEE802154_FCF_ACK_REQ) &&
(dev->netdev.flags & NETDEV_IEEE802154_ACK_REQ)) {
uint8_t payload_len = len_fcf & 0xff;
dev->tx_timeout = dev->backoff_delay + dev->tx_warmup_time +
KW41ZRF_SHR_PHY_TIME + payload_len * KW41ZRF_PER_BYTE_TIME +
KW41ZRF_ACK_WAIT_TIME + 2;
}
KW41ZRF_LED_TX_ON;
if (dev->tx_timeout) {
kw41zrf_set_sequence(dev, XCVSEQ_TX_RX | ZLL_PHY_CTRL_TC3TMOUT_MASK);
}
else {
kw41zrf_set_sequence(dev, XCVSEQ_TRANSMIT);
}
if (dev->tx_timeout > 0) {
/* Set real timeout for RX ACK */
kw41zrf_timer_set(dev, &ZLL->T3CMP, dev->tx_timeout);
}
dev->backoff_delay = 0;
/* disable TMR2 match */
bit_clear32(&ZLL->PHY_CTRL, ZLL_PHY_CTRL_TMR2CMP_EN_SHIFT);
}
if (irqsts & ZLL_IRQSTS_CCAIRQ_MASK) {
/* CCA before TX has completed */
handled_irqs |= ZLL_IRQSTS_CCAIRQ_MASK;
if (irqsts & ZLL_IRQSTS_CCA_MASK) {
/* Channel was determined busy */
if (irqsts & ZLL_IRQSTS_SEQIRQ_MASK) {
handled_irqs |= ZLL_IRQSTS_SEQIRQ_MASK;
kw41zrf_abort_sequence(dev);
KW41ZRF_LED_TX_OFF;
}
LOG_DEBUG("[kw41zrf] CCA ch busy (RSSI: %d retry: %u)\n",
(int8_t)((ZLL->LQI_AND_RSSI & ZLL_LQI_AND_RSSI_CCA1_ED_FNL_MASK) >>
ZLL_LQI_AND_RSSI_CCA1_ED_FNL_SHIFT),
dev->csma_num_backoffs
);
if (dev->csma_num_backoffs < dev->csma_max_backoffs) {
/* Perform CSMA/CA backoff algorithm */
++dev->csma_num_backoffs;
if (dev->csma_be < dev->csma_max_be) {
/* Increase delay exponent */
++dev->csma_be;
}
/* Resubmit the frame for transmission */
kw41zrf_tx_exec(dev);
return handled_irqs;
}
/* If we get here we've used up the csma retries and we're done */
kw41zrf_abort_sequence(dev);
kw41zrf_set_sequence(dev, dev->idle_seq);
if (dev->netdev.netdev.event_callback) {
dev->netdev.netdev.event_callback(&dev->netdev.netdev, NETDEV_EVENT_TX_MEDIUM_BUSY);
LOG_INFO("[kw41zrf] dropping frame after %u backoffs\n",
dev->csma_num_backoffs);
}
}
else {
/* Channel is idle */
DEBUG("[kw41zrf] CCA ch idle (RSSI: %d retries: %u)\n",
(int8_t)((ZLL->LQI_AND_RSSI & ZLL_LQI_AND_RSSI_CCA1_ED_FNL_MASK) >>
ZLL_LQI_AND_RSSI_CCA1_ED_FNL_SHIFT),
dev->csma_num_backoffs
);
if (dev->netdev.netdev.event_callback) {
/* TX will start automatically after CCA check succeeded */
dev->netdev.netdev.event_callback(&dev->netdev.netdev, NETDEV_EVENT_TX_STARTED);
}
}
}
return handled_irqs;
}
/* Handler for Receive sequence */
static uint32_t _isr_event_seq_r(kw41zrf_t *dev, uint32_t irqsts)
{
uint32_t handled_irqs = 0;
if (irqsts & ZLL_IRQSTS_RXWTRMRKIRQ_MASK) {
DEBUG("[kw41zrf] RXWTRMRKIRQ (R)\n");
handled_irqs |= ZLL_IRQSTS_RXWTRMRKIRQ_MASK;
if (dev->netdev.netdev.event_callback) {
dev->netdev.netdev.event_callback(&dev->netdev.netdev, NETDEV_EVENT_RX_STARTED);
}
}
if (irqsts & ZLL_IRQSTS_FILTERFAIL_IRQ_MASK) {
KW41ZRF_LED_RX_OFF;
DEBUG("[kw41zrf] FILTERFAILIRQ: %04"PRIx32"\n", ZLL->FILTERFAIL_CODE);
handled_irqs |= ZLL_IRQSTS_FILTERFAIL_IRQ_MASK;
}
if (irqsts & ZLL_IRQSTS_RXIRQ_MASK) {
KW41ZRF_LED_RX_ON;
DEBUG("[kw41zrf] finished RX\n");
handled_irqs |= ZLL_IRQSTS_RXIRQ_MASK;
DEBUG("[kw41zrf] RX len: %3u\n",
(unsigned int)((ZLL->IRQSTS & ZLL_IRQSTS_RX_FRAME_LENGTH_MASK) >>
ZLL_IRQSTS_RX_FRAME_LENGTH_SHIFT));
if (ZLL->PHY_CTRL & ZLL_PHY_CTRL_AUTOACK_MASK) {
KW41ZRF_LED_TX_ON;
DEBUG("[kw41zrf] perform TXACK\n");
}
}
if (irqsts & ZLL_IRQSTS_TXIRQ_MASK) {
KW41ZRF_LED_TX_OFF;
DEBUG("[kw41zrf] finished TXACK\n");
handled_irqs |= ZLL_IRQSTS_TXIRQ_MASK;
}
if (irqsts & ZLL_IRQSTS_SEQIRQ_MASK) {
uint32_t seq_ctrl_sts = ZLL->SEQ_CTRL_STS;
kw41zrf_abort_sequence(dev);
DEBUG("[kw41zrf] SEQIRQ (R)\n");
handled_irqs |= ZLL_IRQSTS_SEQIRQ_MASK;
KW41ZRF_LED_TX_OFF;
if ((irqsts & ZLL_IRQSTS_CRCVALID_MASK) == 0) {
DEBUG("[kw41zrf] CRC failure (R)\n");
}
else if (seq_ctrl_sts & ZLL_SEQ_CTRL_STS_TC3_ABORTED_MASK) {
DEBUG("[kw41zrf] RX timeout (R)\n");
}
else if (seq_ctrl_sts & ZLL_SEQ_CTRL_STS_PLL_ABORTED_MASK) {
LOG_ERROR("[kw41zrf] PLL unlock (R)\n");
}
else if (seq_ctrl_sts & ZLL_SEQ_CTRL_STS_SW_ABORTED_MASK) {
LOG_ERROR("[kw41zrf] SW abort (R)\n");
}
else {
/* No error reported */
DEBUG("[kw41zrf] success (R)\n");
/* Block XCVSEQ_RECEIVE until netdev->recv has been called */
dev->recv_blocked = 1;
kw41zrf_set_sequence(dev, dev->idle_seq);
if (dev->netdev.netdev.event_callback) {
dev->netdev.netdev.event_callback(&dev->netdev.netdev, NETDEV_EVENT_RX_COMPLETE);
}
return handled_irqs;
}
kw41zrf_set_sequence(dev, dev->idle_seq);
}
return handled_irqs;
}
/* Handler for Transmit sequence */
static uint32_t _isr_event_seq_t(kw41zrf_t *dev, uint32_t irqsts)
{
uint32_t handled_irqs = 0;
if (irqsts & ZLL_IRQSTS_TXIRQ_MASK) {
DEBUG("[kw41zrf] finished TX (T)\n");
handled_irqs |= ZLL_IRQSTS_TXIRQ_MASK;
}
if (irqsts & ZLL_IRQSTS_SEQIRQ_MASK) {
/* Finished T sequence */
kw41zrf_abort_sequence(dev);
/* Go back to being idle */
kw41zrf_set_sequence(dev, dev->idle_seq);
DEBUG("[kw41zrf] SEQIRQ (T)\n");
handled_irqs |= ZLL_IRQSTS_SEQIRQ_MASK;
if (dev->netdev.netdev.event_callback) {
dev->netdev.netdev.event_callback(&dev->netdev.netdev, NETDEV_EVENT_TX_COMPLETE);
}
KW41ZRF_LED_TX_OFF;
}
return handled_irqs;
}
/* Handler for standalone CCA */
static uint32_t _isr_event_seq_cca(kw41zrf_t *dev, uint32_t irqsts)
{
uint32_t handled_irqs = 0;
if (irqsts & ZLL_IRQSTS_SEQIRQ_MASK) {
/* Finished CCA sequence */
kw41zrf_abort_sequence(dev);
kw41zrf_set_sequence(dev, dev->idle_seq);
DEBUG("[kw41zrf] SEQIRQ (C)\n");
handled_irqs |= ZLL_IRQSTS_SEQIRQ_MASK;
if (irqsts & ZLL_IRQSTS_CCA_MASK) {
DEBUG("[kw41zrf] CCA ch busy\n");
dev->cca_result = 1;
}
else {
DEBUG("[kw41zrf] CCA ch idle\n");
dev->cca_result = 0;
}
}
return handled_irqs;
}
/* Handler for Transmit/Receive sequence */
static uint32_t _isr_event_seq_tr(kw41zrf_t *dev, uint32_t irqsts)
{
uint32_t handled_irqs = 0;
if (irqsts & ZLL_IRQSTS_TXIRQ_MASK) {
KW41ZRF_LED_RX_ON;
DEBUG("[kw41zrf] finished TX (TR)\n");
handled_irqs |= ZLL_IRQSTS_TXIRQ_MASK;
DEBUG("[kw41zrf] wait for RX ACK\n");
}
if (irqsts & ZLL_IRQSTS_RXIRQ_MASK) {
KW41ZRF_LED_RX_OFF;
DEBUG("[kw41zrf] got RX ACK\n");
handled_irqs |= ZLL_IRQSTS_RXIRQ_MASK;
}
if (irqsts & ZLL_IRQSTS_FILTERFAIL_IRQ_MASK) {
DEBUG("[kw41zrf] FILTERFAILIRQ (TR): %04"PRIx32"\n", ZLL->FILTERFAIL_CODE);
handled_irqs |= ZLL_IRQSTS_FILTERFAIL_IRQ_MASK;
}
if (irqsts & ZLL_IRQSTS_SEQIRQ_MASK) {
uint32_t seq_ctrl_sts = ZLL->SEQ_CTRL_STS;
kw41zrf_abort_sequence(dev);
DEBUG("[kw41zrf] SEQIRQ (TR)\n");
handled_irqs |= ZLL_IRQSTS_SEQIRQ_MASK;
KW41ZRF_LED_TX_OFF;
KW41ZRF_LED_RX_OFF;
if (seq_ctrl_sts & ZLL_SEQ_CTRL_STS_TC3_ABORTED_MASK) {
if (dev->num_retrans < dev->max_retrans) {
/* Perform frame retransmission */
++dev->num_retrans;
DEBUG("[kw41zrf] TX retry %u\n", (unsigned)dev->num_retrans);
/* Reset CSMA counters for backoff handling */
dev->csma_be = dev->csma_min_be;
dev->csma_num_backoffs = 0;
/* Resubmit the frame for transmission */
kw41zrf_tx_exec(dev);
return handled_irqs;
}
}
assert(!kw41zrf_is_dsm());
kw41zrf_set_sequence(dev, dev->idle_seq);
if (dev->netdev.netdev.event_callback) {
if (seq_ctrl_sts & ZLL_SEQ_CTRL_STS_TC3_ABORTED_MASK) {
LOG_DEBUG("[kw41zrf] RXACK timeout (TR)\n");
dev->netdev.netdev.event_callback(&dev->netdev.netdev, NETDEV_EVENT_TX_NOACK);
}
else if (seq_ctrl_sts & ZLL_SEQ_CTRL_STS_PLL_ABORTED_MASK) {
LOG_ERROR("[kw41zrf] PLL unlock (TR)\n");
/* TODO: there is no other error event for TX failures */
dev->netdev.netdev.event_callback(&dev->netdev.netdev, NETDEV_EVENT_TX_MEDIUM_BUSY);
/* if this does happen in development, it's worth checking why */
assert(0);
}
else if (seq_ctrl_sts & ZLL_SEQ_CTRL_STS_SW_ABORTED_MASK) {
DEBUG("[kw41zrf] SW abort (TR)\n");
/* TODO: there is no other error event for TX failures */
dev->netdev.netdev.event_callback(&dev->netdev.netdev, NETDEV_EVENT_TX_MEDIUM_BUSY);
}
else {
/* No error reported */
DEBUG("[kw41zrf] TX success (TR)\n");
dev->netdev.netdev.event_callback(&dev->netdev.netdev, NETDEV_EVENT_TX_COMPLETE);
}
}
}
return handled_irqs;
}
/* Handler for Continuous CCA */
static uint32_t _isr_event_seq_ccca(kw41zrf_t *dev, uint32_t irqsts)
{
uint32_t handled_irqs = 0;
if (irqsts & ZLL_IRQSTS_SEQIRQ_MASK) {
DEBUG("[kw41zrf] SEQIRQ (CCCA)\n");
handled_irqs |= ZLL_IRQSTS_SEQIRQ_MASK;
if (irqsts & ZLL_IRQSTS_CCA_MASK) {
DEBUG("[kw41zrf] CCCA ch busy\n");
}
else {
DEBUG("[kw41zrf] CCCA ch idle\n");
}
kw41zrf_abort_sequence(dev);
kw41zrf_set_sequence(dev, dev->idle_seq);
}
return handled_irqs;
}
static void kw41zrf_netdev_isr(netdev_t *netdev)
{
kw41zrf_t *dev = (kw41zrf_t *)netdev;
irq_is_queued = false;
thread_flags_clear(KW41ZRF_THREAD_FLAG_ISR);
/* ZLL register access requires that the transceiver is not in deep sleep mode */
if (kw41zrf_is_dsm()) {
/* Transceiver is sleeping, the IRQ must have occurred before entering
* sleep, discard the call */
DEBUG("kw41zrf: unexpected IRQ while sleeping\n");
kw41zrf_unmask_irqs();
return;
}
uint32_t irqsts = ZLL->IRQSTS;
/* Clear all IRQ flags now */
ZLL->IRQSTS = irqsts;
uint32_t handled_irqs = 0;
DEBUG("[kw41zrf] CTRL %08" PRIx32 ", IRQSTS %08" PRIx32 ", FILTERFAIL %08" PRIx32 "\n",
ZLL->PHY_CTRL, irqsts, ZLL->FILTERFAIL_CODE);
uint8_t seq = (ZLL->PHY_CTRL & ZLL_PHY_CTRL_XCVSEQ_MASK)
>> ZLL_PHY_CTRL_XCVSEQ_SHIFT;
switch (seq) {
case XCVSEQ_RECEIVE:
handled_irqs |= _isr_event_seq_r(dev, irqsts);
break;
case XCVSEQ_IDLE: /* XCVSEQ is idle during csma backoff */
case XCVSEQ_TRANSMIT:
/* First check CCA flags */
handled_irqs |= _isr_event_seq_t_ccairq(dev, irqsts);
/* Then TX flags */
handled_irqs |= _isr_event_seq_t(dev, irqsts & ~handled_irqs);
break;
case XCVSEQ_CCA:
handled_irqs |= _isr_event_seq_cca(dev, irqsts);
break;
case XCVSEQ_TX_RX:
/* First check CCA flags */
handled_irqs |= _isr_event_seq_t_ccairq(dev, irqsts);
/* Then TX/RX flags */
handled_irqs |= _isr_event_seq_tr(dev, irqsts & ~handled_irqs);
break;
case XCVSEQ_CONTINUOUS_CCA:
handled_irqs |= _isr_event_seq_ccca(dev, irqsts);
break;
default:
assert(0);
break;
}
irqsts &= ~handled_irqs;
/* doesn't need handling; just prevent outputting an error below */
irqsts &= ~ZLL_IRQSTS_RXWTRMRKIRQ_MASK;
if (irqsts & 0x000f017ful) {
LOG_ERROR("[kw41zrf] Unhandled IRQs: 0x%08lx\n", (irqsts & 0x000f017ful));
}
kw41zrf_unmask_irqs();
}
const netdev_driver_t kw41zrf_driver = {
.init = kw41zrf_netdev_init,
.send = kw41zrf_netdev_send,
.recv = kw41zrf_netdev_recv,
.get = kw41zrf_netdev_get,
.set = kw41zrf_netdev_set,
.isr = kw41zrf_netdev_isr,
};
/** @} */