/* * 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 */ #include #include #include #include #include #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; netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); kw41zrf_t *dev = container_of(netdev_ieee802154, 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) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); kw41zrf_t *dev = container_of(netdev_ieee802154, 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(&dev->netdev.netdev); /* 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) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); kw41zrf_t *dev = container_of(netdev_ieee802154, 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) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); kw41zrf_t *dev = container_of(netdev_ieee802154, 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) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); kw41zrf_t *dev = container_of(netdev_ieee802154, 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_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(container_of(netdev, 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) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); kw41zrf_t *dev = container_of(netdev_ieee802154, 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(container_of(netdev, 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) { netdev_ieee802154_t *netdev_ieee802154 = container_of(netdev, netdev_ieee802154_t, netdev); kw41zrf_t *dev = container_of(netdev_ieee802154, 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, }; /** @} */