1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-15 23:53:08 +01:00
RIOT/drivers/cc110x/cc110x_rx_tx.c
2020-10-23 01:26:09 +02:00

327 lines
11 KiB
C

/*
* Copyright (C) 2018 Otto-von-Guericke-Universität Magdeburg
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup drivers_cc110x
* @{
*
* @file
* @brief Functions to manage sending/receiving frames with the CC110x
*
* @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
* @}
*/
#include "xtimer.h"
#include "cc110x.h"
#include "cc110x_internal.h"
#define ENABLE_DEBUG 0
#include "debug.h"
/* Use NETDEV_EVENT_ISR to indicate that no event needs to be passed to upper
* layer at end of ISR, as ISR will never need this event
*/
#define NETDEV_NO_EVENT NETDEV_EVENT_ISR
void cc110x_enter_rx_mode(cc110x_t *dev)
{
DEBUG("[cc110x] Going to RX\n");
/* bring device to IDLE state and flush FIFOs (just in case) */
gpio_irq_disable(dev->params.gdo0);
gpio_irq_disable(dev->params.gdo2);
cc110x_cmd(dev, CC110X_STROBE_IDLE);
cc110x_cmd(dev, CC110X_STROBE_FLUSH_RX);
cc110x_cmd(dev, CC110X_STROBE_FLUSH_TX);
dev->buf.pos = dev->buf.len = 0;
/* Apply GDO2 config and go to RX */
cc110x_write(dev, CC110X_REG_IOCFG2, CC110X_GDO_ON_RX_DATA);
cc110x_write(dev, CC110X_REG_IOCFG0, CC110X_GDO_ON_TRANSMISSION);
cc110x_cmd(dev, CC110X_STROBE_RX);
dev->state = CC110X_STATE_RX_MODE;
gpio_irq_enable(dev->params.gdo2);
gpio_irq_enable(dev->params.gdo0);
}
/**
* @brief Function to run when frame is fully received
*
* @param dev Device descriptor of the transceiver
*
* Intended to be called from @ref cc110x_rx_continue
*/
static netdev_event_t cc110x_rx_done(cc110x_t *dev)
{
uint8_t lqi_crc;
int8_t rssi;
cc110x_read(dev, CC110X_REG_LQI, &lqi_crc);
cc110x_read(dev, CC110X_REG_RSSI, (uint8_t *)&rssi);
/* CRC_OK bit is most significant bit, see page 92 in the data sheet */
if (!(lqi_crc & 0x80)) {
DEBUG("[cc110x] ISR: CRC error, dropping frame\n");
/* Drop frame and go back to RX */
cc110x_enter_rx_mode(dev);
return NETDEV_EVENT_CRC_ERROR;
}
/* Copy all but the CRC_OK bit */
dev->rx_info.lqi = (uint8_t)lqi_crc & 0x7f;
/* Use the formula in section 17.3 on page 44 in the data sheet to obtain
* the correct RSSI value in dBm.
*/
dev->rx_info.rssi = (int16_t)(rssi / 2) - (int16_t)dev->rssi_offset;
/* Transceiver has automatically gone to IDLE. We keep it in IDLE until
* upper layer fetched the frame
*/
dev->state = CC110X_STATE_FRAME_READY;
return NETDEV_EVENT_RX_COMPLETE;
}
/**
* @brief Read a chunk of data from the RX-FIFO
*
* @param dev Device descriptor of the transceiver
*
* This function should be called from the ISR when data in the RX-FIFO is
* available or the last byte of the frame was received
*/
static netdev_event_t cc110x_rx_continue(cc110x_t *dev)
{
uint8_t in_fifo;
netdev_event_t retval = NETDEV_NO_EVENT;
while (gpio_read(dev->params.gdo2)) {
cc110x_read_reliable(dev, CC110X_REG_RXBYTES, &in_fifo);
if (in_fifo & 0x80) {
/* RXFIFO_OVERFLOW bit is set (see RXBYTES on page 94) */
DEBUG("[cc110x] ISR: RX-FIFO overflown, ISR too slow\n");
/* Drop frame and go to RX */
cc110x_enter_rx_mode(dev);
return NETDEV_EVENT_RX_TIMEOUT;
}
if (!in_fifo) {
/* GDO2 will be high when data is present *or* at end of packet */
break;
}
/* Handle first read from RX FIFO differently from subsequent reads, as
* in first reads the Length Field is read as well
*/
if (!dev->buf.len) {
if (in_fifo < sizeof(cc1xxx_l2hdr_t) + 1) {
/* At least a frame header + Length Field (1B) is expected */
DEBUG("[cc110x] ISR: Incoming frame smaller than header "
"--> drop\n");
cc110x_enter_rx_mode(dev);
/* Not exactly CRC, but incorrect CRC indicates a broken frame*/
return NETDEV_EVENT_CRC_ERROR;
}
cc110x_burst_read(dev, CC110X_MULTIREG_FIFO, &dev->buf,
in_fifo - 1);
/* Update read position in payload, that is number of bytes read
* minus the Length Filed and minus the byte left in the FIFO to not
* trigger a silicon bug
*/
dev->buf.pos = in_fifo - 2;
retval = NETDEV_EVENT_RX_STARTED;
}
else {
/* Prevent overflow of buffer */
if (dev->buf.pos + in_fifo > CC110X_MAX_FRAME_SIZE) {
DEBUG("[cc110x] ISR: Incoming frame exceeds maximum size\n");
cc110x_enter_rx_mode(dev);
/* Not exactly CRC, but incorrect CRC indicates a broken frame */
return NETDEV_EVENT_CRC_ERROR;
}
if (dev->buf.pos + in_fifo < dev->buf.len) {
/* Frame not fully received yet, keeping one byte in RX FIFO
* to prevent triggering a silicon bug
*/
in_fifo--;
}
/* Continue reading data */
cc110x_burst_read(dev, CC110X_MULTIREG_FIFO,
dev->buf.data + dev->buf.pos, in_fifo);
dev->buf.pos += in_fifo;
}
}
if (dev->buf.pos > dev->buf.len) {
DEBUG("[cc110x] ISR: Incoming frame larger than Length Field "
"--> drop\n");
cc110x_enter_rx_mode(dev);
/* Not exactly CRC, but incorrect CRC indicates a broken frame */
return NETDEV_EVENT_CRC_ERROR;
}
if (!gpio_read(dev->params.gdo0)) {
/* GDO0 is low when transmission is over ==> RX complete or corrupt
frame */
if (dev->buf.pos == dev->buf.len) {
return cc110x_rx_done(dev);
}
else {
DEBUG("[cc110x] ISR: Incoming frame smaller than Length Field "
"--> drop\n");
cc110x_enter_rx_mode(dev);
/* Not exactly CRC, but incorrect CRC indicates a broken frame */
return NETDEV_EVENT_CRC_ERROR;
}
}
return retval;
}
/**
* @brief Function to run when frame is fully send
*
* @param dev Device descriptor of the transceiver
*/
static netdev_event_t cc110x_tx_done(cc110x_t *dev)
{
uint8_t status = cc110x_status(dev);
cc110x_state_t state = cc110x_state_from_status(status);
switch (state){
case CC110X_STATE_SETTLING:
case CC110X_STATE_CALIBRATE:
case CC110X_STATE_TX_MODE:
/* TX still in progress, or hasn't even started yet */
return NETDEV_NO_EVENT;
case CC110X_STATE_IDLE:
cc110x_enter_rx_mode(dev);
return NETDEV_EVENT_TX_COMPLETE;
case CC110X_STATE_TXFIFO_UNDERFLOW:
DEBUG("[cc110x] ISR: TX FIFO underflown.\n");
break;
default:
DEBUG("[cc110x] ISR: Unknown state during TX.\n");
break;
}
cc110x_enter_rx_mode(dev);
/* TX timeout is the only TX-related error event known to RIOT */
return NETDEV_EVENT_TX_TIMEOUT;
}
/**
* @brief Refill the TX-FIFO
*
* @param dev Device descriptor of the transceiver
*/
static netdev_event_t cc110x_tx_continue(cc110x_t *dev)
{
uint8_t in_fifo;
cc110x_read_reliable(dev, CC110X_REG_TXBYTES, &in_fifo);
/* most significant bit indicates TXFIFO underflow, see page 94 in the
* data sheet
*/
if (in_fifo & 0x80) {
DEBUG("[cc110x] ISR: ERROR: TX-FIFO underflown, ISR too slow\n");
/* Abort: Flush TX and go back to RX */
cc110x_cmd(dev, CC110X_STROBE_IDLE);
cc110x_cmd(dev, CC110X_STROBE_FLUSH_TX);
cc110x_enter_rx_mode(dev);
return NETDEV_EVENT_TX_TIMEOUT;
}
uint8_t to_write = CC110X_FIFO_SIZE - in_fifo;
if (to_write == 0) {
/* ISR came to early, nothing to do yet */
return NETDEV_NO_EVENT;
}
uint8_t left = dev->buf.len - dev->buf.pos;
to_write = (left < to_write) ? left : to_write;
cc110x_burst_write(dev, CC110X_MULTIREG_FIFO,
dev->buf.data + dev->buf.pos, to_write);
dev->buf.pos += to_write;
if (dev->buf.pos == dev->buf.len) {
/* All data send to the transceiver, now waiting for transceiver to
* complete transmission
*/
dev->state = CC110X_STATE_TX_COMPLETING;
/* Disable GDO2, as we do not need to further feed TX FIFO */
cc110x_write(dev, CC110X_REG_IOCFG2, CC110X_GDO_CONSTANT_LOW);
}
return NETDEV_NO_EVENT;
}
void cc110x_isr(netdev_t *netdev)
{
cc110x_t *dev = (cc110x_t *)netdev;
/* We don't want to create events while device descriptor is acquired, to
* prevent a dead lock. (Currently e.g. on NETDEV_EVENT_RX_COMPLETE the
* upper layer will immediately call netdev_driver_t::recv(), which in
* turn wants to operate on the device descriptor. We could rely on this
* behaviour by skipping cc110x_acquire() there, but the driver would break
* when upper layer behaviour is changed. By moving the event notification
* at the end of the ISR (end after cc110x_release()), the driver becomes
* agnostic to which behaviour the upper layer implements.)
*/
netdev_event_t post_isr_event = NETDEV_NO_EVENT;
if (cc110x_acquire(dev) != SPI_OK) {
DEBUG("[cc110x] ISR: CRITICAL ERROR: Couldn't acquire device\n");
return;
}
/* Disable IRQs in a coarse manner, instead of doing so any time the
* IOCFGx configuration registers are changed. (This should be less
* bug prone.)
*/
gpio_irq_disable(dev->params.gdo0);
gpio_irq_disable(dev->params.gdo2);
switch (dev->state) {
case CC110X_STATE_RX_MODE:
if (gpio_read(dev->params.gdo0) || gpio_read(dev->params.gdo2)) {
dev->state = CC110X_STATE_RECEIVING;
dev->buf.pos = dev->buf.len = 0;
}
break;
case CC110X_STATE_RECEIVING:
post_isr_event = cc110x_rx_continue(dev);
break;
case CC110X_STATE_TX_MODE:
post_isr_event = cc110x_tx_continue(dev);
break;
case CC110X_STATE_TX_COMPLETING:
post_isr_event = cc110x_tx_done(dev);
break;
default:
DEBUG("[cc110x] ISR: CRITICAL ERROR: No interrupt expected "
"for current state\n");
/* Go back to RX and pray that solved the problem */
cc110x_enter_rx_mode(dev);
}
/* Re-enable IRQs again, unless device state */
gpio_irq_enable(dev->params.gdo0);
gpio_irq_enable(dev->params.gdo2);
cc110x_release(dev);
/* Pass event to uper layer, if needed */
if (post_isr_event != NETDEV_NO_EVENT) {
dev->netdev.event_callback(&dev->netdev, post_isr_event);
}
}