1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-15 10:32:44 +01:00
RIOT/drivers/kw41zrf/kw41zrf_netdev.c
Joakim Nohlgård 5bd67d88a8 drivers/kw41zrf: Transceiver driver for the KW41Z radio
This is the radio found in NXP Kinetis KW41Z, KW21Z. Only 802.15.4 mode
is implemented (KW41Z also supports BLE on the same transceiver).

The driver uses vendor supplied initialization code for the low level
XCVR hardware, these files were imported from KSDK 2.2.0 (framework_5.3.5)
2020-03-19 17:00:04 -05:00

1259 lines
41 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_t *)thread_get(thread_getpid());
/* 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:
assert(len >= sizeof(netopt_enable_t));
*((netopt_enable_t *)value) =
!!(dev->flags & KW41ZRF_OPT_TELL_RX_START);
return sizeof(netopt_enable_t);
case NETOPT_RX_END_IRQ:
assert(len >= sizeof(netopt_enable_t));
*((netopt_enable_t *)value) =
!!(dev->flags & KW41ZRF_OPT_TELL_RX_END);
return sizeof(netopt_enable_t);
case NETOPT_TX_START_IRQ:
assert(len >= sizeof(netopt_enable_t));
*((netopt_enable_t *)value) =
!!(dev->flags & KW41ZRF_OPT_TELL_TX_START);
return sizeof(netopt_enable_t);
case NETOPT_TX_END_IRQ:
assert(len >= sizeof(netopt_enable_t));
*((netopt_enable_t *)value) =
!!(dev->flags & KW41ZRF_OPT_TELL_TX_END);
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_RX_END_IRQ:
assert(len <= sizeof(const netopt_enable_t));
kw41zrf_set_option(dev, KW41ZRF_OPT_TELL_RX_END,
*((const netopt_enable_t *)value));
res = sizeof(const netopt_enable_t);
break;
case NETOPT_TX_START_IRQ:
assert(len <= sizeof(const netopt_enable_t));
kw41zrf_set_option(dev, KW41ZRF_OPT_TELL_TX_START,
*((const netopt_enable_t *)value));
res = sizeof(const netopt_enable_t);
break;
case NETOPT_TX_END_IRQ:
assert(len <= sizeof(const netopt_enable_t));
kw41zrf_set_option(dev, KW41ZRF_OPT_TELL_TX_END,
*((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_RX_START_IRQ:
assert(len <= sizeof(const netopt_enable_t));
kw41zrf_set_option(dev, KW41ZRF_OPT_TELL_RX_START,
*((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->flags & KW41ZRF_OPT_TELL_TX_END) {
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->flags & KW41ZRF_OPT_TELL_TX_START) {
/* 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->flags & KW41ZRF_OPT_TELL_RX_START) {
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->flags & KW41ZRF_OPT_TELL_RX_END) {
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->flags & KW41ZRF_OPT_TELL_TX_END) {
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->flags & KW41ZRF_OPT_TELL_TX_END) {
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,
};
/** @} */