mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
474838ba15
This PR removes the RX Busy detection because it doesn't seem to be reliable enough to ensure the radio is not locked. Some radios such as nrf802154 don't provide RX Busy detection mechanisms, so this is not uncommon. It's intended that the MAC layer takes care of this (e.g slots, CSMA-CA with retransmissions, etc)
214 lines
6.0 KiB
C
214 lines
6.0 KiB
C
/*
|
|
* Copyright (C) 2016 MUTEX NZ Ltd.
|
|
* Copyright (C) 2015 Loci Controls Inc.
|
|
*
|
|
* 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 Low-level radio driver for the CC2538
|
|
*
|
|
* @author Aaron Sowry <aaron@mutex.nz>
|
|
* @author Ian Martin <ian@locicontrols.com>
|
|
* @}
|
|
*/
|
|
|
|
#include "periph_conf.h"
|
|
|
|
#include "cc2538_rf.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
/**
|
|
* @brief MAC timer period
|
|
*
|
|
* The period is set to the CSMA-CA Backoff Period Unit (20 symbols, 320 us).
|
|
* The system clock runs at 32 MHz. Thus, the timeout period is
|
|
* 320us * 32MHz = ~10738 (0x29F2)
|
|
*/
|
|
#define TIMER_PERIOD_LSB (0xF2)
|
|
#define TIMER_PERIOD_MSB (0x29)
|
|
|
|
/**
|
|
* @brief MAC timer period
|
|
*
|
|
* The period is set to the CSMA-CA Backoff Period Unit (20 symbols, 320 us).
|
|
* The system clock runs at 32 MHz. Thus, the timeout period is
|
|
* 320us * 32MHz = ~10738 (0x29F2)
|
|
*/
|
|
#define CCTEST_OBSSELX_EN (BIT(7))
|
|
|
|
static void _cc2538_setup_mac_timer(void)
|
|
{
|
|
RFCORE_SFR_MTMSEL &= ~CC2538_SFR_MTMSEL_MASK;
|
|
/* Select timer period */
|
|
RFCORE_SFR_MTMSEL |= CC2538_SFR_MTMSEL_TIMER_P;
|
|
|
|
/* Fix timer to Backoff period */
|
|
RFCORE_SFR_MTM0 |= TIMER_PERIOD_LSB;
|
|
RFCORE_SFR_MTM1 |= TIMER_PERIOD_MSB;
|
|
|
|
RFCORE_SFR_MTMSEL &= ~CC2538_SFR_MTMSEL_MASK;
|
|
RFCORE_SFR_MTCTRL |= CC2538_MCTRL_SYNC_MASK | CC2538_MCTRL_RUN_MASK;
|
|
}
|
|
|
|
static void _cc2538_observable_signals(void)
|
|
{
|
|
/* Select on which pin PC0:7 should the selected observable signals
|
|
be wired through, the signal is selected in CONFIG_CC2538_RF_OBS_%
|
|
and the pin in CONFIG_CC2538_RF_OBS_SIG_%_PCX */
|
|
if (IS_USED(MODULE_CC2538_RF_OBS_SIG)) {
|
|
if (CONFIG_CC2538_RF_OBS_0 != disabled) {
|
|
RFCORE_XREG_RFC_OBS_CTRL0 = CONFIG_CC2538_RF_OBS_0;
|
|
*(&CCTEST_OBSSEL0 + CONFIG_CC2538_RF_OBS_SIG_0_PCX) = \
|
|
CCTEST_OBSSELX_EN | rfc_obs_sig0;
|
|
}
|
|
if (CONFIG_CC2538_RF_OBS_1 != disabled) {
|
|
RFCORE_XREG_RFC_OBS_CTRL1 = CONFIG_CC2538_RF_OBS_1;
|
|
*(&CCTEST_OBSSEL0 + CONFIG_CC2538_RF_OBS_SIG_1_PCX) = \
|
|
CCTEST_OBSSELX_EN | rfc_obs_sig1;
|
|
}
|
|
if (CONFIG_CC2538_RF_OBS_2 != disabled) {
|
|
RFCORE_XREG_RFC_OBS_CTRL2 = CONFIG_CC2538_RF_OBS_2;
|
|
*(&CCTEST_OBSSEL0 + CONFIG_CC2538_RF_OBS_SIG_2_PCX) = \
|
|
CCTEST_OBSSELX_EN | rfc_obs_sig2;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool cc2538_channel_clear(void)
|
|
{
|
|
if (RFCORE->XREG_FSMSTAT0bits.FSM_FFCTRL_STATE == FSM_STATE_IDLE) {
|
|
bool result;
|
|
cc2538_on();
|
|
RFCORE_WAIT_UNTIL(RFCORE->XREG_RSSISTATbits.RSSI_VALID);
|
|
result = BOOLEAN(RFCORE->XREG_FSMSTAT1bits.CCA);
|
|
cc2538_off();
|
|
return result;
|
|
}
|
|
else {
|
|
RFCORE_WAIT_UNTIL(RFCORE->XREG_RSSISTATbits.RSSI_VALID);
|
|
return BOOLEAN(RFCORE->XREG_FSMSTAT1bits.CCA);
|
|
}
|
|
}
|
|
|
|
void cc2538_init(void)
|
|
{
|
|
/* Enable RF CORE clock in active mode */
|
|
SYS_CTRL_RCGCRFC = 1UL;
|
|
/* Enable RF CORE clock in sleep mode */
|
|
SYS_CTRL_SCGCRFC = 1UL;
|
|
/* Enable RF CORE clock in PM0 (system clock always powered down
|
|
in PM1-3) */
|
|
SYS_CTRL_DCGCRFC = 1UL;
|
|
/* Wait for the clock enabling to take effect */
|
|
while (!(SYS_CTRL_RCGCRFC & 1UL) || \
|
|
!(SYS_CTRL_SCGCRFC & 1UL) || \
|
|
!(SYS_CTRL_DCGCRFC & 1UL)
|
|
) {}
|
|
|
|
/* Register Setting updates for optimal performance, RM section 23.15 */
|
|
RFCORE_XREG_TXFILTCFG = 0x09;
|
|
RFCORE_XREG_AGCCTRL1 = 0x15;
|
|
RFCORE_XREG_FSCAL1 = 0x01;
|
|
ANA_REGS_IVCTRL = 0x0B;
|
|
|
|
/* Enable AUTOCRC and AUTOACK by default*/
|
|
RFCORE_XREG_FRMCTRL0 = AUTOCRC | AUTOACK;
|
|
|
|
/* Disable RX after TX, let upper layer change the state */
|
|
RFCORE_XREG_FRMCTRL1 = 0x00;
|
|
|
|
/* Disable source address matching and pending bits */
|
|
RFCORE_XREG_SRCMATCH = 0x00;
|
|
|
|
/* Set FIFOP_THR to its max value*/
|
|
RFCORE_XREG_FIFOPCTRL = CC2538_RF_MAX_DATA_LEN;
|
|
|
|
/* Set default IRQ */
|
|
cc2538_rf_enable_irq();
|
|
|
|
/* Enable all RF CORE error interrupts */
|
|
RFCORE_XREG_RFERRM = STROBE_ERR | TXUNDERF | TXOVERF | \
|
|
RXUNDERF | RXOVERF | NLOCK;
|
|
|
|
_cc2538_observable_signals();
|
|
|
|
/* Enable IRQs */
|
|
if (SYS_CTRL_I_MAP) {
|
|
NVIC_SetPriority(RF_RXTX_ALT_IRQn, CPU_DEFAULT_IRQ_PRIO);
|
|
NVIC_EnableIRQ(RF_RXTX_ALT_IRQn);
|
|
|
|
NVIC_SetPriority(RF_ERR_ALT_IRQn, CPU_DEFAULT_IRQ_PRIO);
|
|
NVIC_EnableIRQ(RF_ERR_ALT_IRQn);
|
|
|
|
NVIC_SetPriority(MAC_TIMER_ALT_IRQn, CPU_DEFAULT_IRQ_PRIO);
|
|
NVIC_EnableIRQ(MAC_TIMER_ALT_IRQn);
|
|
}
|
|
else {
|
|
NVIC_SetPriority(RF_RXTX_IRQn, CPU_DEFAULT_IRQ_PRIO);
|
|
NVIC_EnableIRQ(RF_RXTX_IRQn);
|
|
|
|
NVIC_SetPriority(RF_ERR_IRQn, CPU_DEFAULT_IRQ_PRIO);
|
|
NVIC_EnableIRQ(RF_ERR_IRQn);
|
|
|
|
NVIC_SetPriority(MACTIMER_IRQn, CPU_DEFAULT_IRQ_PRIO);
|
|
NVIC_EnableIRQ(MACTIMER_IRQn);
|
|
}
|
|
|
|
/* setup mac timer */
|
|
_cc2538_setup_mac_timer();
|
|
|
|
/* Enable Auto ACK */
|
|
RFCORE->XREG_FRMCTRL0bits.AUTOACK = !IS_ACTIVE(CONFIG_IEEE802154_AUTO_ACK_DISABLE);
|
|
|
|
/* Flush the receive and transmit FIFOs */
|
|
RFCORE_SFR_RFST = ISFLUSHTX;
|
|
RFCORE_SFR_RFST = ISFLUSHRX;
|
|
}
|
|
|
|
bool cc2538_is_on(void)
|
|
{
|
|
return RFCORE->XREG_FSMSTAT1bits.RX_ACTIVE || \
|
|
RFCORE->XREG_FSMSTAT1bits.TX_ACTIVE;
|
|
}
|
|
|
|
void cc2538_off(void)
|
|
{
|
|
/* Wait for ongoing TX to complete (e.g. this could be an outgoing ACK) */
|
|
RFCORE_WAIT_UNTIL(RFCORE->XREG_FSMSTAT1bits.TX_ACTIVE == 0);
|
|
|
|
/* Flush RX FIFO */
|
|
RFCORE_SFR_RFST = ISFLUSHRX;
|
|
|
|
/* Don't turn off if we are off as this will trigger a Strobe Error */
|
|
if (RFCORE_XREG_RXENABLE != 0) {
|
|
RFCORE_SFR_RFST = ISRFOFF;
|
|
}
|
|
}
|
|
|
|
bool cc2538_on(void)
|
|
{
|
|
/* Flush RX FIFO */
|
|
RFCORE_SFR_RFST = ISFLUSHRX;
|
|
|
|
/* Enable/calibrate RX */
|
|
RFCORE_SFR_RFST = ISRXON;
|
|
|
|
return true;
|
|
}
|
|
|
|
void cc2538_setup(cc2538_rf_t *dev)
|
|
{
|
|
(void) dev;
|
|
cc2538_init();
|
|
}
|