2016-07-08 03:33:54 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016 MUTEX NZ Ltd
|
|
|
|
*
|
|
|
|
* 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 cpu_cc2538
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Netdev adaption for the CC2538 RF driver
|
|
|
|
*
|
|
|
|
* @author Aaron Sowry <aaron@mutex.nz>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "net/gnrc.h"
|
2017-02-15 13:07:34 +01:00
|
|
|
#include "net/netdev.h"
|
2016-07-08 03:33:54 +02:00
|
|
|
|
|
|
|
#include "cc2538_rf.h"
|
|
|
|
#include "cc2538_rf_netdev.h"
|
|
|
|
#include "cc2538_rf_internal.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/* Reference pointer for the IRQ handler */
|
2017-02-15 13:07:34 +01:00
|
|
|
static netdev_t *_dev;
|
2016-07-08 03:33:54 +02:00
|
|
|
|
|
|
|
void _irq_handler(void)
|
|
|
|
{
|
2020-06-23 15:37:04 +02:00
|
|
|
RFCORE_SFR_RFIRQF0 = 0;
|
|
|
|
RFCORE_SFR_RFIRQF1 = 0;
|
|
|
|
|
2020-03-05 15:11:20 +01:00
|
|
|
netdev_trigger_event_isr(_dev);
|
2016-07-08 03:33:54 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _get(netdev_t *netdev, netopt_t opt, void *value, size_t max_len)
|
2016-07-08 03:33:54 +02:00
|
|
|
{
|
|
|
|
cc2538_rf_t *dev = (cc2538_rf_t *)netdev;
|
|
|
|
|
|
|
|
if (dev == NULL) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (opt) {
|
2018-11-17 22:35:01 +01:00
|
|
|
case NETOPT_ADDRESS:
|
|
|
|
if (max_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
else {
|
2020-09-10 00:26:18 +02:00
|
|
|
cc2538_get_addr_short(value);
|
2018-11-17 22:35:01 +01:00
|
|
|
}
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETOPT_ADDRESS_LONG:
|
|
|
|
if (max_len < sizeof(uint64_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
else {
|
2020-09-10 00:26:18 +02:00
|
|
|
cc2538_get_addr_long(value);
|
2018-11-17 22:35:01 +01:00
|
|
|
}
|
|
|
|
return sizeof(uint64_t);
|
|
|
|
|
2016-07-08 03:33:54 +02:00
|
|
|
case NETOPT_AUTOACK:
|
|
|
|
if (RFCORE->XREG_FRMCTRL0bits.AUTOACK) {
|
|
|
|
*((netopt_enable_t *)value) = NETOPT_ENABLE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*((netopt_enable_t *)value) = NETOPT_DISABLE;
|
|
|
|
}
|
|
|
|
return sizeof(netopt_enable_t);
|
|
|
|
|
2018-11-17 22:51:35 +01:00
|
|
|
case NETOPT_CHANNEL:
|
|
|
|
if (max_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((uint16_t *)value) = (uint16_t)cc2538_get_chan();
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
2016-07-08 03:33:54 +02:00
|
|
|
case NETOPT_CHANNEL_PAGE:
|
|
|
|
if (max_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
2019-10-23 21:13:52 +02:00
|
|
|
/* This transceiver only supports page 0 */
|
2016-07-08 03:33:54 +02:00
|
|
|
*((uint16_t *)value) = 0;
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETOPT_DEVICE_TYPE:
|
|
|
|
if (max_len < sizeof(uint16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
2017-02-15 13:07:34 +01:00
|
|
|
*((uint16_t *) value) = NETDEV_TYPE_IEEE802154;
|
2016-07-08 03:33:54 +02:00
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
case NETOPT_IS_CHANNEL_CLR:
|
|
|
|
if (cc2538_channel_clear()) {
|
|
|
|
*((netopt_enable_t *)value) = NETOPT_ENABLE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*((netopt_enable_t *)value) = NETOPT_DISABLE;
|
|
|
|
}
|
|
|
|
return sizeof(netopt_enable_t);
|
|
|
|
|
|
|
|
case NETOPT_IS_WIRED:
|
|
|
|
return -ENOTSUP;
|
|
|
|
|
|
|
|
case NETOPT_PROMISCUOUSMODE:
|
|
|
|
if (cc2538_get_monitor()) {
|
|
|
|
*((netopt_enable_t *)value) = NETOPT_ENABLE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*((netopt_enable_t *)value) = NETOPT_DISABLE;
|
|
|
|
}
|
|
|
|
return sizeof(netopt_enable_t);
|
|
|
|
|
|
|
|
case NETOPT_STATE:
|
|
|
|
if (max_len < sizeof(netopt_state_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((netopt_state_t *)value) = dev->state;
|
|
|
|
return sizeof(netopt_state_t);
|
|
|
|
|
|
|
|
case NETOPT_TX_POWER:
|
|
|
|
if (max_len < sizeof(int16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
*((uint16_t *)value) = cc2538_get_tx_power();
|
|
|
|
return sizeof(uint16_t);
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
int res;
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
if (((res = netdev_ieee802154_get((netdev_ieee802154_t *)netdev, opt, value,
|
|
|
|
max_len)) >= 0) || (res != -ENOTSUP)) {
|
2016-07-08 03:33:54 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
2017-08-25 07:34:03 +02:00
|
|
|
static int _set(netdev_t *netdev, netopt_t opt, const void *value, size_t value_len)
|
2016-07-08 03:33:54 +02:00
|
|
|
{
|
|
|
|
cc2538_rf_t *dev = (cc2538_rf_t *)netdev;
|
|
|
|
int res = -ENOTSUP;
|
|
|
|
|
|
|
|
if (dev == NULL) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
case NETOPT_ADDRESS:
|
|
|
|
if (value_len > sizeof(uint16_t)) {
|
|
|
|
res = -EOVERFLOW;
|
|
|
|
}
|
|
|
|
else {
|
2020-09-10 00:26:18 +02:00
|
|
|
cc2538_set_addr_short(value);
|
2018-11-17 22:35:34 +01:00
|
|
|
res = sizeof(uint16_t);
|
2016-07-08 03:33:54 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETOPT_ADDRESS_LONG:
|
|
|
|
if (value_len > sizeof(uint64_t)) {
|
|
|
|
res = -EOVERFLOW;
|
|
|
|
}
|
|
|
|
else {
|
2020-09-10 00:26:18 +02:00
|
|
|
cc2538_set_addr_long(value);
|
2018-11-17 22:35:34 +01:00
|
|
|
res = sizeof(uint64_t);
|
2016-07-08 03:33:54 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETOPT_AUTOACK:
|
2017-08-25 07:34:03 +02:00
|
|
|
RFCORE->XREG_FRMCTRL0bits.AUTOACK = ((const bool *)value)[0];
|
2016-07-08 03:33:54 +02:00
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETOPT_CHANNEL:
|
|
|
|
if (value_len != sizeof(uint16_t)) {
|
|
|
|
res = -EINVAL;
|
|
|
|
}
|
|
|
|
else {
|
2017-08-25 07:34:03 +02:00
|
|
|
uint8_t chan = ((const uint8_t *)value)[0];
|
2016-10-21 12:58:50 +02:00
|
|
|
if (chan < IEEE802154_CHANNEL_MIN ||
|
|
|
|
chan > IEEE802154_CHANNEL_MAX) {
|
2016-07-08 03:33:54 +02:00
|
|
|
res = -EINVAL;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
cc2538_set_chan(chan);
|
2018-11-17 22:51:59 +01:00
|
|
|
res = sizeof(uint16_t);
|
2016-07-08 03:33:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETOPT_CHANNEL_PAGE:
|
2019-10-23 21:13:52 +02:00
|
|
|
/* This transceiver only supports page 0 */
|
2016-07-08 03:33:54 +02:00
|
|
|
if (value_len != sizeof(uint16_t) ||
|
2017-08-25 07:34:03 +02:00
|
|
|
*((const uint16_t *)value) != 0 ) {
|
2016-07-08 03:33:54 +02:00
|
|
|
res = -EINVAL;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
res = sizeof(uint16_t);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETOPT_IS_WIRED:
|
|
|
|
return -ENOTSUP;
|
|
|
|
|
|
|
|
case NETOPT_NID:
|
|
|
|
if (value_len > sizeof(uint16_t)) {
|
|
|
|
res = -EOVERFLOW;
|
|
|
|
}
|
|
|
|
else {
|
2017-08-25 07:34:03 +02:00
|
|
|
cc2538_set_pan(*((const uint16_t *)value));
|
2016-07-08 03:33:54 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETOPT_PROMISCUOUSMODE:
|
2017-08-25 07:34:03 +02:00
|
|
|
cc2538_set_monitor(((const bool *)value)[0]);
|
2016-07-08 03:33:54 +02:00
|
|
|
res = sizeof(netopt_enable_t);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETOPT_STATE:
|
|
|
|
if (value_len > sizeof(netopt_state_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
2017-08-25 07:34:03 +02:00
|
|
|
cc2538_set_state(dev, *((const netopt_state_t *)value));
|
2016-07-08 03:33:54 +02:00
|
|
|
res = sizeof(netopt_state_t);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NETOPT_TX_POWER:
|
|
|
|
if (value_len > sizeof(int16_t)) {
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
2017-08-25 07:34:03 +02:00
|
|
|
cc2538_set_tx_power(*((const int16_t *)value));
|
2016-07-08 03:33:54 +02:00
|
|
|
res = sizeof(uint16_t);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res == -ENOTSUP) {
|
2017-02-15 13:07:34 +01:00
|
|
|
res = netdev_ieee802154_set((netdev_ieee802154_t *)netdev, opt,
|
|
|
|
value, value_len);
|
2016-07-08 03:33:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
static int _send(netdev_t *netdev, const iolist_t *iolist)
|
2016-07-08 03:33:54 +02:00
|
|
|
{
|
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) netdev;
|
|
|
|
|
2016-07-08 03:33:54 +02:00
|
|
|
int pkt_len = 0;
|
|
|
|
|
2016-08-05 05:23:27 +02:00
|
|
|
/* Flush TX FIFO once no transmission in progress */
|
|
|
|
RFCORE_WAIT_UNTIL(RFCORE->XREG_FSMSTAT1bits.TX_ACTIVE == 0);
|
2016-07-08 03:33:54 +02:00
|
|
|
RFCORE_SFR_RFST = ISFLUSHTX;
|
|
|
|
|
|
|
|
/* The first byte of the TX FIFO must be the packet length,
|
|
|
|
but we don't know what it is yet. Write a null byte to the
|
|
|
|
start of the FIFO, so we can come back and update it later */
|
|
|
|
rfcore_write_byte(0);
|
|
|
|
|
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-07-08 03:33:54 +02:00
|
|
|
|
|
|
|
if (pkt_len > CC2538_RF_MAX_DATA_LEN) {
|
|
|
|
DEBUG("cc2538_rf: packet too large (%u > %u)\n",
|
|
|
|
pkt_len, CC2538_RF_MAX_DATA_LEN);
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
|
2018-01-12 00:19:03 +01:00
|
|
|
rfcore_write_fifo(iol->iol_base, iol->iol_len);
|
2016-07-08 03:33:54 +02:00
|
|
|
}
|
2016-10-26 08:48:34 +02:00
|
|
|
|
2016-07-08 03:33:54 +02:00
|
|
|
/* Set first byte of TX FIFO to the packet length */
|
|
|
|
rfcore_poke_tx_fifo(0, pkt_len + CC2538_AUTOCRC_LEN);
|
|
|
|
|
|
|
|
RFCORE_SFR_RFST = ISTXON;
|
|
|
|
|
2016-10-07 01:25:38 +02:00
|
|
|
/* Wait for transmission to complete */
|
|
|
|
RFCORE_WAIT_UNTIL(RFCORE->XREG_FSMSTAT1bits.TX_ACTIVE == 0);
|
|
|
|
|
2016-07-08 03:33:54 +02:00
|
|
|
return pkt_len;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info)
|
2016-07-08 03:33:54 +02:00
|
|
|
{
|
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) netdev;
|
|
|
|
|
2016-07-08 03:33:54 +02:00
|
|
|
size_t pkt_len;
|
|
|
|
|
|
|
|
if (buf == NULL) {
|
2020-04-20 13:36:52 +02:00
|
|
|
/* Check that the last byte of a new frame has been received */
|
|
|
|
if (RFCORE->XREG_FSMSTAT1bits.FIFOP == 0) {
|
|
|
|
DEBUG_PRINT("cc2538_rf: Frame has not finished being received\n");
|
|
|
|
return -EAGAIN;
|
|
|
|
}
|
|
|
|
|
2016-07-08 03:33:54 +02:00
|
|
|
/* GNRC wants to know how much data we've got for it */
|
|
|
|
pkt_len = rfcore_read_byte();
|
|
|
|
|
2016-07-25 18:34:44 +02:00
|
|
|
/* Make sure pkt_len is sane */
|
|
|
|
if (pkt_len > CC2538_RF_MAX_DATA_LEN) {
|
2020-04-20 13:37:39 +02:00
|
|
|
DEBUG_PRINT("cc2538_rf: pkt_len > CC2538_RF_MAX_DATA_LEN\n");
|
2016-07-08 03:33:54 +02:00
|
|
|
RFCORE_SFR_RFST = ISFLUSHRX;
|
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
|
|
|
|
2020-04-20 13:37:39 +02:00
|
|
|
/* Make sure pkt_len is not too short.
|
|
|
|
* There are at least 2 bytes (FCS). */
|
|
|
|
if (pkt_len < IEEE802154_FCS_LEN) {
|
|
|
|
DEBUG_PRINT("cc2538_rf: pkt_len < IEEE802154_FCS_LEN\n");
|
|
|
|
RFCORE_SFR_RFST = ISFLUSHRX;
|
|
|
|
return -ENODATA;
|
|
|
|
}
|
|
|
|
|
2016-07-08 03:33:54 +02:00
|
|
|
if (len > 0) {
|
|
|
|
/* GNRC wants us to drop the packet */
|
|
|
|
RFCORE_SFR_RFST = ISFLUSHRX;
|
|
|
|
}
|
|
|
|
|
|
|
|
return pkt_len - IEEE802154_FCS_LEN;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* GNRC is expecting len bytes of data */
|
|
|
|
pkt_len = len;
|
|
|
|
}
|
2016-10-26 08:48:34 +02:00
|
|
|
|
2016-07-08 03:33:54 +02:00
|
|
|
rfcore_read_fifo(buf, pkt_len);
|
|
|
|
|
2020-04-20 13:36:52 +02:00
|
|
|
int8_t rssi_val = rfcore_read_byte() + CC2538_RSSI_OFFSET;
|
|
|
|
uint8_t crc_corr_val = rfcore_read_byte();
|
|
|
|
|
2020-04-20 13:39:36 +02:00
|
|
|
/* CRC check */
|
|
|
|
if (!(crc_corr_val & CC2538_CRC_BIT_MASK)) {
|
|
|
|
/* CRC failed; discard packet */
|
|
|
|
RFCORE_SFR_RFST = ISFLUSHRX;
|
|
|
|
return -ENODATA;
|
|
|
|
}
|
|
|
|
|
2020-04-20 13:40:32 +02:00
|
|
|
if (info != NULL) {
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_ieee802154_rx_info_t *radio_info = info;
|
2016-07-16 11:26:46 +02:00
|
|
|
|
|
|
|
/* The number of dB above maximum sensitivity detected for the
|
|
|
|
* received packet */
|
2020-07-31 15:50:48 +02:00
|
|
|
radio_info->rssi = rssi_val;
|
2016-07-16 11:26:46 +02:00
|
|
|
|
2020-04-20 13:36:52 +02:00
|
|
|
uint8_t corr_val = crc_corr_val & CC2538_CORR_VAL_MASK;
|
2016-07-16 11:26:46 +02:00
|
|
|
|
|
|
|
if (corr_val < CC2538_CORR_VAL_MIN) {
|
|
|
|
corr_val = CC2538_CORR_VAL_MIN;
|
|
|
|
}
|
|
|
|
else if (corr_val > CC2538_CORR_VAL_MAX) {
|
|
|
|
corr_val = CC2538_CORR_VAL_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Interpolate the correlation value between 0 - 255
|
|
|
|
* to provide an LQI value */
|
|
|
|
radio_info->lqi = 255 * (corr_val - CC2538_CORR_VAL_MIN) /
|
|
|
|
(CC2538_CORR_VAL_MAX - CC2538_CORR_VAL_MIN);
|
2016-07-08 03:33:54 +02:00
|
|
|
}
|
|
|
|
|
2020-04-20 13:41:24 +02:00
|
|
|
/* Check for overflow of the rx fifo */
|
|
|
|
if (RFCORE->XREG_FSMSTAT1bits.FIFOP != 0 &&
|
|
|
|
RFCORE->XREG_FSMSTAT1bits.FIFO == 0)
|
|
|
|
{
|
|
|
|
DEBUG_PRINT("cc2538_rf: RXFIFO Overflow\n");
|
|
|
|
RFCORE_SFR_RFST = ISFLUSHRX;
|
|
|
|
}
|
2016-07-08 03:33:54 +02:00
|
|
|
|
|
|
|
return pkt_len;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static void _isr(netdev_t *netdev)
|
2016-07-08 03:33:54 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_RX_COMPLETE);
|
2016-07-08 03:33:54 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _init(netdev_t *netdev)
|
2016-07-08 03:33:54 +02:00
|
|
|
{
|
|
|
|
cc2538_rf_t *dev = (cc2538_rf_t *) netdev;
|
|
|
|
_dev = netdev;
|
|
|
|
|
|
|
|
uint16_t chan = cc2538_get_chan();
|
|
|
|
|
2018-07-11 11:00:07 +02:00
|
|
|
netdev_ieee802154_reset(&dev->netdev);
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
/* Initialise netdev_ieee802154_t struct */
|
2018-07-11 11:02:11 +02:00
|
|
|
netdev_ieee802154_set(&dev->netdev, NETOPT_CHANNEL,
|
|
|
|
&chan, sizeof(chan));
|
2016-07-08 03:33:54 +02:00
|
|
|
|
|
|
|
cc2538_set_state(dev, NETOPT_STATE_IDLE);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2018-01-12 00:19:03 +01:00
|
|
|
|
|
|
|
const netdev_driver_t cc2538_rf_driver = {
|
|
|
|
.get = _get,
|
|
|
|
.set = _set,
|
|
|
|
.send = _send,
|
|
|
|
.recv = _recv,
|
|
|
|
.isr = _isr,
|
|
|
|
.init = _init,
|
|
|
|
};
|