2016-04-14 13:48:00 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015-2016 Freie Universität Berlin
|
2016-07-01 15:31:10 +02:00
|
|
|
* 2016 Inria
|
2016-04-14 13:48:00 +02:00
|
|
|
*
|
|
|
|
* 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_cc2420
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
2016-07-01 15:31:10 +02:00
|
|
|
* @brief Implementation of public functions for cc2420 driver
|
2016-04-14 13:48:00 +02:00
|
|
|
*
|
|
|
|
* @author Thomas Eichinger <thomas.eichinger@fu-berlin.de>
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "byteorder.h"
|
|
|
|
#include "net/ieee802154.h"
|
|
|
|
#include "net/gnrc.h"
|
|
|
|
|
|
|
|
#include "cc2420_internal.h"
|
2016-07-01 15:31:10 +02:00
|
|
|
#include "cc2420_netdev.h"
|
|
|
|
#include "cc2420_registers.h"
|
2016-04-14 13:48:00 +02:00
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2016-04-14 13:48:00 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
2020-09-11 16:16:50 +02:00
|
|
|
void cc2420_setup(cc2420_t * dev, const cc2420_params_t *params, uint8_t index)
|
2016-04-14 13:48:00 +02:00
|
|
|
{
|
2020-09-11 16:16:50 +02:00
|
|
|
netdev_t *netdev = &dev->netdev.netdev;
|
|
|
|
|
2016-04-14 13:48:00 +02:00
|
|
|
/* set pointer to the devices netdev functions */
|
2020-09-11 16:16:50 +02:00
|
|
|
netdev->driver = &cc2420_driver;
|
2016-04-14 13:48:00 +02:00
|
|
|
/* pull in device configuration parameters */
|
2019-01-06 22:54:44 +01:00
|
|
|
dev->params = *params;
|
2016-07-01 15:31:10 +02:00
|
|
|
dev->state = CC2420_STATE_IDLE;
|
2016-04-14 13:48:00 +02:00
|
|
|
/* reset device descriptor fields */
|
|
|
|
dev->options = 0;
|
2020-09-11 16:16:50 +02:00
|
|
|
|
|
|
|
netdev_register(netdev, NETDEV_CC2420, index);
|
|
|
|
netdev_ieee802154_setup(&dev->netdev);
|
2016-04-14 13:48:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int cc2420_init(cc2420_t *dev)
|
|
|
|
{
|
|
|
|
uint16_t reg;
|
|
|
|
|
2018-07-10 14:29:00 +02:00
|
|
|
netdev_ieee802154_reset(&dev->netdev);
|
2016-04-14 13:48:00 +02:00
|
|
|
|
2020-09-11 16:16:50 +02:00
|
|
|
cc2420_set_addr_short(dev, dev->netdev.short_addr);
|
|
|
|
cc2420_set_addr_long(dev, dev->netdev.long_addr);
|
2016-04-14 13:48:00 +02:00
|
|
|
cc2420_set_chan(dev, CC2420_CHAN_DEFAULT);
|
|
|
|
cc2420_set_txpower(dev, CC2420_TXPOWER_DEFAULT);
|
|
|
|
|
|
|
|
/* set default options */
|
|
|
|
cc2420_set_option(dev, CC2420_OPT_AUTOACK, true);
|
|
|
|
cc2420_set_option(dev, CC2420_OPT_CSMA, true);
|
2016-07-01 15:31:10 +02:00
|
|
|
|
2016-04-14 13:48:00 +02:00
|
|
|
/* change default RX bandpass filter to 1.3uA (as recommended) */
|
2016-07-01 15:31:10 +02:00
|
|
|
reg = cc2420_reg_read(dev, CC2420_REG_RXCTRL1);
|
2016-04-14 13:48:00 +02:00
|
|
|
reg |= CC2420_RXCTRL1_RXBPF_LOCUR;
|
2016-07-01 15:31:10 +02:00
|
|
|
cc2420_reg_write(dev, CC2420_REG_RXCTRL1, reg);
|
2016-04-14 13:48:00 +02:00
|
|
|
|
|
|
|
/* set the FIFOP threshold to maximum. */
|
2016-07-01 15:31:10 +02:00
|
|
|
cc2420_reg_write(dev, CC2420_REG_IOCFG0, CC2420_PKT_MAXLEN);
|
2016-04-14 13:48:00 +02:00
|
|
|
|
|
|
|
/* turn off "Security enable" (page 33). */
|
2016-07-01 15:31:10 +02:00
|
|
|
reg = cc2420_reg_read(dev, CC2420_REG_SECCTRL0);
|
2016-04-14 13:48:00 +02:00
|
|
|
reg &= ~CC2420_SECCTRL0_RXFIFO_PROT;
|
2016-07-01 15:31:10 +02:00
|
|
|
cc2420_reg_write(dev, CC2420_REG_SECCTRL0, reg);
|
2016-04-14 13:48:00 +02:00
|
|
|
|
|
|
|
/* set preamble length to 3 leading zero byte */
|
2016-10-28 23:46:23 +02:00
|
|
|
/* and turn on hardware CRC generation */
|
2016-07-01 15:31:10 +02:00
|
|
|
reg = cc2420_reg_read(dev, CC2420_REG_MDMCTRL0);
|
2016-04-14 13:48:00 +02:00
|
|
|
reg &= ~(CC2420_MDMCTRL0_PREAMBLE_M);
|
|
|
|
reg |= CC2420_MDMCTRL0_PREAMBLE_3B;
|
2016-10-28 23:46:23 +02:00
|
|
|
reg |= CC2420_MDMCTRL0_AUTOCRC;
|
2016-07-01 15:31:10 +02:00
|
|
|
cc2420_reg_write(dev, CC2420_REG_MDMCTRL0, reg);
|
2016-04-14 13:48:00 +02:00
|
|
|
|
2016-10-28 23:46:23 +02:00
|
|
|
/* go into RX state */
|
2020-07-29 18:51:13 +02:00
|
|
|
cc2420_set_state(dev, NETOPT_STATE_IDLE);
|
2016-10-28 23:46:23 +02:00
|
|
|
|
2016-04-14 13:48:00 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cc2420_cca(cc2420_t *dev)
|
|
|
|
{
|
2016-07-01 15:31:10 +02:00
|
|
|
while (!(cc2420_status(dev) & CC2420_STATUS_RSSI_VALID)) {}
|
2016-04-14 13:48:00 +02:00
|
|
|
return gpio_read(dev->params.pin_cca);
|
|
|
|
}
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
size_t cc2420_send(cc2420_t *dev, const iolist_t *iolist)
|
2016-04-14 13:48:00 +02:00
|
|
|
{
|
2018-01-12 00:19:03 +01:00
|
|
|
size_t n = cc2420_tx_prepare(dev, iolist);
|
2016-04-14 13:48:00 +02:00
|
|
|
|
|
|
|
if ((n > 0) && !(dev->options & CC2420_OPT_PRELOADING)) {
|
|
|
|
cc2420_tx_exec(dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
size_t cc2420_tx_prepare(cc2420_t *dev, const iolist_t *iolist)
|
2016-04-14 13:48:00 +02:00
|
|
|
{
|
|
|
|
size_t pkt_len = 2; /* include the FCS (frame check sequence) */
|
|
|
|
|
|
|
|
/* wait for any ongoing transmissions to be finished */
|
2016-09-27 17:16:57 +02:00
|
|
|
DEBUG("cc2420: tx_exec: waiting for any ongoing transmission\n");
|
2016-04-14 13:48:00 +02:00
|
|
|
while (cc2420_get_state(dev) & NETOPT_STATE_TX) {}
|
|
|
|
|
|
|
|
/* get and check the length of the packet */
|
2018-01-12 00:19:03 +01:00
|
|
|
for (const iolist_t *iol = iolist; iol; iol = iol->iol_next) {
|
|
|
|
pkt_len += iol->iol_len;
|
2016-04-14 13:48:00 +02:00
|
|
|
}
|
|
|
|
if (pkt_len >= CC2420_PKT_MAXLEN) {
|
2016-07-01 15:31:10 +02:00
|
|
|
DEBUG("cc2420: tx_prep: unable to send, pkt too large\n");
|
2016-04-14 13:48:00 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* flush TX FIFO and write new packet to it */
|
2016-07-01 15:31:10 +02:00
|
|
|
cc2420_strobe(dev, CC2420_STROBE_FLUSHTX);
|
2016-04-14 13:48:00 +02:00
|
|
|
/* push packet length to TX FIFO */
|
2016-07-01 15:31:10 +02:00
|
|
|
cc2420_fifo_write(dev, (uint8_t *)&pkt_len, 1);
|
2019-04-02 16:00:03 +02:00
|
|
|
/* push packet to TX FIFO, only if iol->iol_len > 0 */
|
2018-01-12 00:19:03 +01:00
|
|
|
for (const iolist_t *iol = iolist; iol; iol = iol->iol_next) {
|
2019-04-02 16:00:03 +02:00
|
|
|
if (iol->iol_len > 0) {
|
|
|
|
cc2420_fifo_write(dev, iol->iol_base, iol->iol_len);
|
|
|
|
}
|
2016-04-14 13:48:00 +02:00
|
|
|
}
|
2023-12-18 17:41:57 +01:00
|
|
|
DEBUG("cc2420: tx_prep: loaded %" PRIuSIZE " byte into the TX FIFO\n", pkt_len);
|
2016-04-14 13:48:00 +02:00
|
|
|
|
|
|
|
return pkt_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
void cc2420_tx_exec(cc2420_t *dev)
|
|
|
|
{
|
|
|
|
/* trigger the transmission */
|
2021-02-26 11:39:18 +01:00
|
|
|
if (dev->netdev.netdev.event_callback) {
|
2016-04-14 13:48:00 +02:00
|
|
|
dev->netdev.netdev.event_callback(&dev->netdev.netdev,
|
2017-02-15 13:07:34 +01:00
|
|
|
NETDEV_EVENT_TX_STARTED);
|
2016-04-14 13:48:00 +02:00
|
|
|
}
|
2016-07-01 15:31:10 +02:00
|
|
|
DEBUG("cc2420: tx_exec: TX_START\n");
|
2016-04-14 13:48:00 +02:00
|
|
|
if (dev->options & CC2420_OPT_CSMA) {
|
2016-07-01 15:31:10 +02:00
|
|
|
DEBUG("cc2420: tx_exec: triggering TX with CCA\n");
|
|
|
|
cc2420_strobe(dev, CC2420_STROBE_TXONCCA);
|
2016-04-14 13:48:00 +02:00
|
|
|
}
|
|
|
|
else {
|
2016-07-01 15:31:10 +02:00
|
|
|
DEBUG("cc2420: tx_exec: triggering TX without CCA\n");
|
|
|
|
cc2420_strobe(dev, CC2420_STROBE_TXON);
|
2016-04-14 13:48:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 17:08:25 +01:00
|
|
|
static inline void _flush_rx_fifo(cc2420_t *dev)
|
|
|
|
{
|
2019-01-07 17:10:42 +01:00
|
|
|
/* as stated in the CC2420 datasheet (section 14.3), the SFLUSHRX command
|
|
|
|
* strobe should be issued twice to ensure that the SFD pin goes back to its
|
|
|
|
* idle state */
|
2019-01-07 17:08:25 +01:00
|
|
|
cc2420_strobe(dev, CC2420_STROBE_FLUSHRX);
|
|
|
|
cc2420_strobe(dev, CC2420_STROBE_FLUSHRX);
|
|
|
|
}
|
|
|
|
|
2016-04-14 13:48:00 +02:00
|
|
|
int cc2420_rx(cc2420_t *dev, uint8_t *buf, size_t max_len, void *info)
|
|
|
|
{
|
build: fix unused parameter errors
cpu, sam0_common: fix unused parameter in periph/spi
cpu, kinetis_common: fix unused parameter in periph/spi
cpu, cc2538: fix unused param in periph/i2c
cpu, cc2538: fix unused param in periph/spi
cpu, sam3: fix unused param in periph/spi
cpu, stm32_common: fix unused param in periph/pm
cpu, stm32f3: fix unused params in periph/i2c
cpu, nrf5x_common: fix unused param in periph/gpio
cpu, nrf5x_common: fix unused param in periph/spi
cpu, lpc2387: fix unused params in periph/spi
cpu, cc2538: fix unused params in radio/netdev
cpu, cc2650: fix unused params in periph/uart
cpu, lm4f120: fix unused param in periph/spi
cpu, lm4f120: fix unused params in periph/timer
cpu, lm4f120: fix unused params in periph/uart
cpu, stm32_common: fix unused params in periph/dac
cpu, stm32l0: fix unused params in periph/i2c
cpu, msp430fxyz: fix unused params in periph/uart
cpu, mips: fix unused params
cpu, cc430: fix unused-params in periph/timer
cpu, msp430fxyz: fix unused params in periph/spi
drivers, cc2420: fix unused param
cpu, mips32r2_common: fix unused params in periph/timer
cpu, cc2538: fix unused-param in periph/i2c
cpu, mips32r2_common: fix unused-param in periph/timer
cpu, msp430fxyz: fix unused params in periph/timer
cpu, atmega_common: fix unused params in periph/spi
driver, nrfmin: fix unused params
cpu, cc2538_rf: fix unused params
driver, netdev_ieee802514: fix unused param
cpu, mip_pic32m: fix unused params
cpu, lpc2387: fix unused params in periph/pwm
tests/driver_sdcard_spi: fix unused params
cpu, sam3: fix unused param in periph/pwm
tests/driver_dynamixel: fix unused params, and style issues
cpu, cc430: fix unused param in periph/rtc
cpu, atmega_common: fix unused params in periph/i2c
2017-10-31 12:09:11 +01:00
|
|
|
(void)info;
|
|
|
|
|
2016-04-14 13:48:00 +02:00
|
|
|
uint8_t len;
|
2016-10-28 23:46:23 +02:00
|
|
|
uint8_t crc_corr;
|
2016-04-14 13:48:00 +02:00
|
|
|
|
2016-10-28 23:46:23 +02:00
|
|
|
/* without a provided buffer, only readout the length and return it */
|
|
|
|
if (buf == NULL) {
|
|
|
|
/* get the packet length (without dropping it) (first byte in RX FIFO) */
|
|
|
|
cc2420_ram_read(dev, CC2420_RAM_RXFIFO, &len, 1);
|
|
|
|
len -= 2; /* subtract RSSI and FCF */
|
2016-07-01 15:31:10 +02:00
|
|
|
DEBUG("cc2420: recv: packet of length %i in RX FIFO\n", (int)len);
|
2019-01-08 23:42:03 +01:00
|
|
|
if (max_len != 0) {
|
|
|
|
DEBUG("cc2420: recv: Dropping frame as requested\n");
|
|
|
|
_flush_rx_fifo(dev);
|
|
|
|
}
|
2016-04-14 13:48:00 +02:00
|
|
|
}
|
2016-10-28 23:46:23 +02:00
|
|
|
else {
|
|
|
|
/* read length byte */
|
2016-11-07 19:10:26 +01:00
|
|
|
cc2420_fifo_read(dev, &len, 1);
|
2016-10-28 23:46:23 +02:00
|
|
|
len -= 2; /* subtract RSSI and FCF */
|
2016-11-07 19:10:26 +01:00
|
|
|
|
2019-01-08 23:45:06 +01:00
|
|
|
if (len > max_len) {
|
|
|
|
DEBUG("cc2420: recv: Supplied buffer to small\n");
|
|
|
|
_flush_rx_fifo(dev);
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
2016-04-14 13:48:00 +02:00
|
|
|
|
|
|
|
/* read fifo contents */
|
2016-07-01 15:31:10 +02:00
|
|
|
DEBUG("cc2420: recv: reading %i byte of the packet\n", (int)len);
|
|
|
|
cc2420_fifo_read(dev, buf, len);
|
2016-04-14 13:48:00 +02:00
|
|
|
|
2017-08-30 21:09:21 +02:00
|
|
|
int8_t rssi;
|
|
|
|
cc2420_fifo_read(dev, (uint8_t*)&rssi, 1);
|
2016-07-01 15:31:10 +02:00
|
|
|
DEBUG("cc2420: recv: RSSI is %i\n", (int)rssi);
|
2016-10-28 23:46:23 +02:00
|
|
|
|
|
|
|
/* fetch and check if CRC_OK bit (MSB) is set */
|
|
|
|
cc2420_fifo_read(dev, &crc_corr, 1);
|
|
|
|
if (!(crc_corr & 0x80)) {
|
|
|
|
DEBUG("cc2420: recv: CRC_OK bit not set, dropping packet\n");
|
|
|
|
/* drop the corrupted frame from the RXFIFO */
|
|
|
|
len = 0;
|
|
|
|
}
|
2017-08-30 21:09:21 +02:00
|
|
|
if (info != NULL) {
|
|
|
|
netdev_ieee802154_rx_info_t *radio_info = info;
|
|
|
|
radio_info->rssi = CC2420_RSSI_OFFSET + rssi;
|
|
|
|
radio_info->lqi = crc_corr & CC2420_CRCCOR_COR_MASK;
|
|
|
|
}
|
2016-10-28 23:46:23 +02:00
|
|
|
|
2016-04-14 13:48:00 +02:00
|
|
|
/* finally flush the FIFO */
|
2019-01-07 17:08:25 +01:00
|
|
|
_flush_rx_fifo(dev);
|
2016-04-14 13:48:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return (int)len;
|
|
|
|
}
|