QMK/lib/chibios-contrib/os/various/devices_lib/rf/nrf52_radio.c

1112 lines
36 KiB
C

/* Copyright (c) 2014 Nordic Semiconductor. All Rights Reserved.
*
* The information contained herein is property of Nordic Semiconductor ASA.
* Terms and conditions of usage are described in detail in NORDIC
* SEMICONDUCTOR STANDARD SOFTWARE LICENSE AGREEMENT.
*
* Licensees are granted free, non-transferable use of the information. NO
* WARRANTY of ANY KIND is provided. This heading must NOT be removed from
* the file.
*
* Enhanced ShockBurst proprietary protocol to ChibiOS port
*
* ported on: 25/10/2018, by andru
*
*/
#include <stdint.h>
#include <string.h>
#include "ch.h"
#include "hal.h"
#include "nrf52_radio.h"
#define BIT_MASK_UINT_8(x) (0xFF >> (8 - (x)))
#define NRF52_PIPE_COUNT 9
#define RADIO_SHORTS_COMMON ( RADIO_SHORTS_READY_START_Msk | RADIO_SHORTS_END_DISABLE_Msk | \
RADIO_SHORTS_ADDRESS_RSSISTART_Msk | RADIO_SHORTS_DISABLED_RSSISTOP_Msk )
// Constant parameters
#define RX_WAIT_FOR_ACK_TIMEOUT_US_2MBPS (48) /**< 2MBit RX wait for ack timeout value. Smallest reliable value - 43 */
#define RX_WAIT_FOR_ACK_TIMEOUT_US_1MBPS (64) /**< 1MBit RX wait for ack timeout value. Smallest reliable value - 59 */
#define NRF52_ADDR_UPDATE_MASK_BASE0 (1 << 0) /*< Mask value to signal updating BASE0 radio address. */
#define NRF52_ADDR_UPDATE_MASK_BASE1 (1 << 1) /*< Mask value to signal updating BASE1 radio address. */
#define NRF52_ADDR_UPDATE_MASK_PREFIX (1 << 2) /*< Mask value to signal updating radio prefixes */
#define NRF52_PID_RESET_VALUE 0xFF /**< Invalid PID value which is guaranteed to not collide with any valid PID value. */
#define NRF52_PID_MAX 3 /**< Maximum value for PID. */
#define NRF52_CRC_RESET_VALUE 0xFFFF /**< CRC reset value*/
#ifndef NRF52_RADIO_USE_TIMER0
#define NRF52_RADIO_USE_TIMER0 FALSE
#endif
#ifndef NRF52_RADIO_USE_TIMER1
#define NRF52_RADIO_USE_TIMER1 FALSE
#endif
#ifndef NRF52_RADIO_USE_TIMER2
#define NRF52_RADIO_USE_TIMER2 FALSE
#endif
#ifndef NRF52_RADIO_USE_TIMER3
#define NRF52_RADIO_USE_TIMER3 FALSE
#endif
#ifndef NRF52_RADIO_USE_TIMER4
#define NRF52_RADIO_USE_TIMER4 FALSE
#endif
#ifndef NRF52_RADIO_IRQ_PRIORITY
#define NRF52_RADIO_IRQ_PRIORITY 3 /**< RADIO interrupt priority. */
#endif
#ifndef NRF52_RADIO_PPI_TIMER_START
#error "PPI channel NRF52_RADIO_PPI_TIMER_START need to be defined"
#endif
#ifndef NRF52_RADIO_PPI_TIMER_STOP
#error "PPI channel NRF52_RADIO_PPI_TIMER_STOP need to be defined"
#endif
#ifndef NRF52_RADIO_PPI_RX_TIMEOUT
#error "PPI channel NRF52_RADIO_PPI_RX_TIMEOUT need to be defined"
#endif
#ifndef NRF52_RADIO_PPI_TX_START
#error "PPI channel NRF52_RADIO_PPI_TX_START need to be defined"
#endif
#if (NRF52_RADIO_USE_TIMER0 == FALSE) && (NRF52_RADIO_USE_TIMER1 == FALSE) && \
(NRF52_RADIO_USE_TIMER2 == FALSE) && (NRF52_RADIO_USE_TIMER3 == FALSE) && \
(NRF52_RADIO_USE_TIMER4 == FALSE)
#error "At least one hardware TIMER must be defined"
#endif
#ifndef NRF52_RADIO_INTTHD_PRIORITY
#error "Interrupt handle thread priority need to be defined"
#endif
#ifndef NRF52_RADIO_EVTTHD_PRIORITY
#error "Event thread priority need to be defined"
#endif
#define VERIFY_PAYLOAD_LENGTH(p) \
do \
{ \
if(p->length == 0 || \
p->length > NRF52_MAX_PAYLOAD_LENGTH || \
(RFD1.config.protocol == NRF52_PROTOCOL_ESB && \
p->length > RFD1.config.payload_length)) \
{ \
return NRF52_ERROR_INVALID_LENGTH; \
} \
}while(0)
//Structure holding pipe info PID and CRC and ack payload.
typedef struct
{
uint16_t m_crc;
uint8_t m_pid;
uint8_t m_ack_payload;
} pipe_info_t;
// First in first out queue of payloads to be transmitted.
typedef struct
{
nrf52_payload_t * p_payload[NRF52_TX_FIFO_SIZE]; /**< Pointer to the actual queue. */
uint32_t entry_point; /**< Current start of queue. */
uint32_t exit_point; /**< Current end of queue. */
uint32_t count; /**< Current number of elements in the queue. */
} nrf52_payload_tx_fifo_t;
// First in first out queue of received payloads.
typedef struct
{
nrf52_payload_t * p_payload[NRF52_RX_FIFO_SIZE]; /**< Pointer to the actual queue. */
uint32_t entry_point; /**< Current start of queue. */
uint32_t exit_point; /**< Current end of queue. */
uint32_t count; /**< Current number of elements in the queue. */
} nrf52_payload_rx_fifo_t;
// These function pointers are changed dynamically, depending on protocol configuration and state.
//static void (*on_radio_end)(RFDriver *rfp) = NULL;
static void (*set_rf_payload_format)(RFDriver *rfp, uint32_t payload_length) = NULL;
// The following functions are assigned to the function pointers above.
static void on_radio_disabled_tx_noack(RFDriver *rfp);
static void on_radio_disabled_tx(RFDriver *rfp);
static void on_radio_disabled_tx_wait_for_ack(RFDriver *rfp);
static void on_radio_disabled_rx(RFDriver *rfp);
static void on_radio_disabled_rx_ack(RFDriver *rfp);
static volatile uint16_t wait_for_ack_timeout_us;
static nrf52_payload_t * p_current_payload;
// TX FIFO
static nrf52_payload_t tx_fifo_payload[NRF52_TX_FIFO_SIZE];
static nrf52_payload_tx_fifo_t tx_fifo;
// RX FIFO
static nrf52_payload_t rx_fifo_payload[NRF52_RX_FIFO_SIZE];
static nrf52_payload_rx_fifo_t rx_fifo;
// Payload buffers
static uint8_t tx_payload_buffer[NRF52_MAX_PAYLOAD_LENGTH + 2];
static uint8_t rx_payload_buffer[NRF52_MAX_PAYLOAD_LENGTH + 2];
static uint8_t pids[NRF52_PIPE_COUNT];
static pipe_info_t rx_pipe_info[NRF52_PIPE_COUNT];
// disable and events semaphores.
static binary_semaphore_t disable_sem;
static binary_semaphore_t events_sem;
RFDriver RFD1;
// Function to do bytewise bit-swap on a unsigned 32 bit value
static uint32_t bytewise_bit_swap(uint8_t const * p_inp) {
uint32_t inp = (*(uint32_t*)p_inp);
return __REV((uint32_t)__RBIT(inp)); //lint -esym(628, __rev) -esym(526, __rev) -esym(628, __rbit) -esym(526, __rbit) */
}
// Internal function to convert base addresses from nRF24L type addressing to nRF52 type addressing
static uint32_t addr_conv(uint8_t const* p_addr) {
return __REV(bytewise_bit_swap(p_addr)); //lint -esym(628, __rev) -esym(526, __rev) */
}
static thread_t *rfEvtThread_p;
static THD_WORKING_AREA(waRFEvtThread, 64);
static THD_FUNCTION(rfEvtThread, arg) {
(void)arg;
chRegSetThreadName("rfevent");
while (!chThdShouldTerminateX()) {
chBSemWait(&events_sem);
nrf52_int_flags_t interrupts = RFD1.flags;
RFD1.flags = 0;
if (interrupts & NRF52_INT_TX_SUCCESS_MSK) {
chEvtBroadcastFlags(&RFD1.eventsrc, (eventflags_t) NRF52_EVENT_TX_SUCCESS);
}
if (interrupts & NRF52_INT_TX_FAILED_MSK) {
chEvtBroadcastFlags(&RFD1.eventsrc, (eventflags_t) NRF52_EVENT_TX_FAILED);
}
if (interrupts & NRF52_INT_RX_DR_MSK) {
chEvtBroadcastFlags(&RFD1.eventsrc, (eventflags_t) NRF52_EVENT_RX_RECEIVED);
}
}
chThdExit((msg_t) 0);
}
static thread_t *rfIntThread_p;
static THD_WORKING_AREA(waRFIntThread, 64);
static THD_FUNCTION(rfIntThread, arg) {
(void)arg;
chRegSetThreadName("rfint");
while (!chThdShouldTerminateX()) {
chBSemWait(&disable_sem);
switch (RFD1.state) {
case NRF52_STATE_PTX_TX:
on_radio_disabled_tx_noack(&RFD1);
break;
case NRF52_STATE_PTX_TX_ACK:
on_radio_disabled_tx(&RFD1);
break;
case NRF52_STATE_PTX_RX_ACK:
on_radio_disabled_tx_wait_for_ack(&RFD1);
break;
case NRF52_STATE_PRX:
on_radio_disabled_rx(&RFD1);
break;
case NRF52_STATE_PRX_SEND_ACK:
on_radio_disabled_rx_ack(&RFD1);
break;
default:
break;
}
}
chThdExit((msg_t) 0);
}
static void serve_radio_interrupt(RFDriver *rfp) {
(void) rfp;
if ((NRF_RADIO->INTENSET & RADIO_INTENSET_READY_Msk) && NRF_RADIO->EVENTS_READY) {
NRF_RADIO->EVENTS_READY = 0;
(void) NRF_RADIO->EVENTS_READY;
}
if ((NRF_RADIO->INTENSET & RADIO_INTENSET_DISABLED_Msk) && NRF_RADIO->EVENTS_DISABLED) {
NRF_RADIO->EVENTS_DISABLED = 0;
(void) NRF_RADIO->EVENTS_DISABLED;
chSysLockFromISR();
chBSemSignalI(&disable_sem);
chSysUnlockFromISR();
}
}
/**
* @brief RADIO events interrupt handler.
*
* @isr
*/
OSAL_IRQ_HANDLER(Vector44) {
OSAL_IRQ_PROLOGUE();
serve_radio_interrupt(&RFD1);
OSAL_IRQ_EPILOGUE();
}
static void set_rf_payload_format_esb_dpl(RFDriver *rfp, uint32_t payload_length) {
(void)payload_length;
#if (NRF52_MAX_PAYLOAD_LENGTH <= 32)
// Using 6 bits for length
NRF_RADIO->PCNF0 = (0 << RADIO_PCNF0_S0LEN_Pos) |
(6 << RADIO_PCNF0_LFLEN_Pos) |
(3 << RADIO_PCNF0_S1LEN_Pos) ;
#else
// Using 8 bits for length
NRF_RADIO->PCNF0 = (0 << RADIO_PCNF0_S0LEN_Pos) |
(8 << RADIO_PCNF0_LFLEN_Pos) |
(3 << RADIO_PCNF0_S1LEN_Pos) ;
#endif
NRF_RADIO->PCNF1 = (RADIO_PCNF1_WHITEEN_Disabled << RADIO_PCNF1_WHITEEN_Pos) |
(RADIO_PCNF1_ENDIAN_Big << RADIO_PCNF1_ENDIAN_Pos) |
((rfp->config.address.addr_length - 1) << RADIO_PCNF1_BALEN_Pos) |
(0 << RADIO_PCNF1_STATLEN_Pos) |
(NRF52_MAX_PAYLOAD_LENGTH << RADIO_PCNF1_MAXLEN_Pos);
}
static void set_rf_payload_format_esb(RFDriver *rfp, uint32_t payload_length) {
NRF_RADIO->PCNF0 = (1 << RADIO_PCNF0_S0LEN_Pos) |
(0 << RADIO_PCNF0_LFLEN_Pos) |
(1 << RADIO_PCNF0_S1LEN_Pos);
NRF_RADIO->PCNF1 = (RADIO_PCNF1_WHITEEN_Disabled << RADIO_PCNF1_WHITEEN_Pos) |
(RADIO_PCNF1_ENDIAN_Big << RADIO_PCNF1_ENDIAN_Pos) |
((rfp->config.address.addr_length - 1) << RADIO_PCNF1_BALEN_Pos) |
(payload_length << RADIO_PCNF1_STATLEN_Pos) |
(payload_length << RADIO_PCNF1_MAXLEN_Pos);
}
/* Set BASE0 and BASE1 addresses & prefixes registers
* NRF52 { prefixes[0], base0_addr[0], base0_addr[1], base0_addr[2], base0_addr[3] } ==
* NRF24 { addr[0], addr[1], addr[2], addr[3], addr[4] }
*/
static void set_addresses(RFDriver *rfp, uint8_t update_mask) {
if (update_mask & NRF52_ADDR_UPDATE_MASK_BASE0) {
NRF_RADIO->BASE0 = addr_conv(rfp->config.address.base_addr_p0);
NRF_RADIO->DAB[0] = addr_conv(rfp->config.address.base_addr_p0);
}
if (update_mask & NRF52_ADDR_UPDATE_MASK_BASE1) {
NRF_RADIO->BASE1 = addr_conv(rfp->config.address.base_addr_p1);
NRF_RADIO->DAB[1] = addr_conv(rfp->config.address.base_addr_p1);
}
if (update_mask & NRF52_ADDR_UPDATE_MASK_PREFIX) {
NRF_RADIO->PREFIX0 = bytewise_bit_swap(&rfp->config.address.pipe_prefixes[0]);
NRF_RADIO->DAP[0] = bytewise_bit_swap(&rfp->config.address.pipe_prefixes[0]);
NRF_RADIO->PREFIX1 = bytewise_bit_swap(&rfp->config.address.pipe_prefixes[4]);
NRF_RADIO->DAP[1] = bytewise_bit_swap(&rfp->config.address.pipe_prefixes[4]);
}
}
static void set_tx_power(RFDriver *rfp) {
NRF_RADIO->TXPOWER = rfp->config.tx_power << RADIO_TXPOWER_TXPOWER_Pos;
}
static void set_bitrate(RFDriver *rfp) {
NRF_RADIO->MODE = rfp->config.bitrate << RADIO_MODE_MODE_Pos;
switch (rfp->config.bitrate) {
case NRF52_BITRATE_2MBPS:
wait_for_ack_timeout_us = RX_WAIT_FOR_ACK_TIMEOUT_US_2MBPS;
break;
case NRF52_BITRATE_1MBPS:
wait_for_ack_timeout_us = RX_WAIT_FOR_ACK_TIMEOUT_US_1MBPS;
break;
}
}
static void set_protocol(RFDriver *rfp) {
switch (rfp->config.protocol) {
case NRF52_PROTOCOL_ESB_DPL:
set_rf_payload_format = set_rf_payload_format_esb_dpl;
break;
case NRF52_PROTOCOL_ESB:
set_rf_payload_format = set_rf_payload_format_esb;
break;
}
}
static void set_crc(RFDriver *rfp) {
NRF_RADIO->CRCCNF = rfp->config.crc << RADIO_CRCCNF_LEN_Pos;
if (rfp->config.crc == RADIO_CRCCNF_LEN_Two)
{
NRF_RADIO->CRCINIT = 0xFFFFUL; // Initial value
NRF_RADIO->CRCPOLY = 0x11021UL; // CRC poly: x^16+x^12^x^5+1
}
else if (rfp->config.crc == RADIO_CRCCNF_LEN_One)
{
NRF_RADIO->CRCINIT = 0xFFUL; // Initial value
NRF_RADIO->CRCPOLY = 0x107UL; // CRC poly: x^8+x^2^x^1+1
}
}
static void ppi_init(RFDriver *rfp) {
NRF_PPI->CH[NRF52_RADIO_PPI_TIMER_START].EEP = (uint32_t)&NRF_RADIO->EVENTS_READY;
NRF_PPI->CH[NRF52_RADIO_PPI_TIMER_START].TEP = (uint32_t)&rfp->timer->TASKS_START;
NRF_PPI->CH[NRF52_RADIO_PPI_TIMER_STOP].EEP = (uint32_t)&NRF_RADIO->EVENTS_ADDRESS;
NRF_PPI->CH[NRF52_RADIO_PPI_TIMER_STOP].TEP = (uint32_t)&rfp->timer->TASKS_STOP;
NRF_PPI->CH[NRF52_RADIO_PPI_RX_TIMEOUT].EEP = (uint32_t)&rfp->timer->EVENTS_COMPARE[0];
NRF_PPI->CH[NRF52_RADIO_PPI_RX_TIMEOUT].TEP = (uint32_t)&NRF_RADIO->TASKS_DISABLE;
NRF_PPI->CH[NRF52_RADIO_PPI_TX_START].EEP = (uint32_t)&rfp->timer->EVENTS_COMPARE[1];
NRF_PPI->CH[NRF52_RADIO_PPI_TX_START].TEP = (uint32_t)&NRF_RADIO->TASKS_TXEN;
}
static void set_parameters(RFDriver *rfp) {
set_tx_power(rfp);
set_bitrate(rfp);
set_protocol(rfp);
set_crc(rfp);
set_rf_payload_format(rfp, rfp->config.payload_length);
}
static void reset_fifo(void) {
tx_fifo.entry_point = 0;
tx_fifo.exit_point = 0;
tx_fifo.count = 0;
rx_fifo.entry_point = 0;
rx_fifo.exit_point = 0;
rx_fifo.count = 0;
}
static void init_fifo(void) {
uint8_t i;
reset_fifo();
for (i = 0; i < NRF52_TX_FIFO_SIZE; i++) {
tx_fifo.p_payload[i] = &tx_fifo_payload[i];
}
for (i = 0; i < NRF52_RX_FIFO_SIZE; i++) {
rx_fifo.p_payload[i] = &rx_fifo_payload[i];
}
}
static void tx_fifo_remove_last(void) {
if (tx_fifo.count > 0) {
nvicDisableVector(RADIO_IRQn);
tx_fifo.count--;
if (++tx_fifo.exit_point >= NRF52_TX_FIFO_SIZE) {
tx_fifo.exit_point = 0;
}
nvicEnableVector(RADIO_IRQn, NRF52_RADIO_IRQ_PRIORITY);
}
}
/** @brief Function to push the content of the rx_buffer to the RX FIFO.
*
* The module will point the register NRF_RADIO->PACKETPTR to a buffer for receiving packets.
* After receiving a packet the module will call this function to copy the received data to
* the RX FIFO.
*
* @param pipe Pipe number to set for the packet.
* @param pid Packet ID.
*
* @retval true Operation successful.
* @retval false Operation failed.
*/
static bool rx_fifo_push_rfbuf(RFDriver *rfp, uint8_t pipe, uint8_t pid) {
if (rx_fifo.count < NRF52_RX_FIFO_SIZE) {
if (rfp->config.protocol == NRF52_PROTOCOL_ESB_DPL) {
if (rx_payload_buffer[0] > NRF52_MAX_PAYLOAD_LENGTH) {
return false;
}
rx_fifo.p_payload[rx_fifo.entry_point]->length = rx_payload_buffer[0];
}
else if (rfp->state == NRF52_STATE_PTX_RX_ACK) {
// Received packet is an acknowledgment
rx_fifo.p_payload[rx_fifo.entry_point]->length = 0;
}
else {
rx_fifo.p_payload[rx_fifo.entry_point]->length = rfp->config.payload_length;
}
memcpy(rx_fifo.p_payload[rx_fifo.entry_point]->data, &rx_payload_buffer[2],
rx_fifo.p_payload[rx_fifo.entry_point]->length);
rx_fifo.p_payload[rx_fifo.entry_point]->pipe = pipe;
rx_fifo.p_payload[rx_fifo.entry_point]->rssi = NRF_RADIO->RSSISAMPLE;
rx_fifo.p_payload[rx_fifo.entry_point]->pid = pid;
if (++rx_fifo.entry_point >= NRF52_RX_FIFO_SIZE) {
rx_fifo.entry_point = 0;
}
rx_fifo.count++;
return true;
}
return false;
}
static void timer_init(RFDriver *rfp) {
// Configure the system timer with a 1 MHz base frequency
rfp->timer->PRESCALER = 4;
rfp->timer->BITMODE = TIMER_BITMODE_BITMODE_16Bit;
rfp->timer->SHORTS = TIMER_SHORTS_COMPARE1_CLEAR_Msk | TIMER_SHORTS_COMPARE1_STOP_Msk;
}
static void start_tx_transaction(RFDriver *rfp) {
bool ack;
rfp->tx_attempt = 1;
rfp->tx_remaining = rfp->config.retransmit.count;
// Prepare the payload
p_current_payload = tx_fifo.p_payload[tx_fifo.exit_point];
// Handling ack if noack is set to false or if selctive auto ack is turned turned off
ack = !p_current_payload->noack || !rfp->config.selective_auto_ack;
switch (rfp->config.protocol) {
case NRF52_PROTOCOL_ESB:
set_rf_payload_format(rfp, p_current_payload->length);
tx_payload_buffer[0] = p_current_payload->pid;
tx_payload_buffer[1] = 0;
memcpy(&tx_payload_buffer[2], p_current_payload->data, p_current_payload->length);
NRF_RADIO->SHORTS = RADIO_SHORTS_COMMON | RADIO_SHORTS_DISABLED_RXEN_Msk;
NRF_RADIO->INTENSET = RADIO_INTENSET_DISABLED_Msk | RADIO_INTENSET_READY_Msk;
// Configure the retransmit counter
rfp->tx_remaining = rfp->config.retransmit.count;
rfp->state = NRF52_STATE_PTX_TX_ACK;
break;
case NRF52_PROTOCOL_ESB_DPL:
tx_payload_buffer[0] = p_current_payload->length;
tx_payload_buffer[1] = p_current_payload->pid << 1;
tx_payload_buffer[1] |= ack ? 0x00 : 0x01;
memcpy(&tx_payload_buffer[2], p_current_payload->data, p_current_payload->length);
if (ack) {
NRF_RADIO->SHORTS = RADIO_SHORTS_COMMON | RADIO_SHORTS_DISABLED_RXEN_Msk;
NRF_RADIO->INTENSET = RADIO_INTENSET_DISABLED_Msk | RADIO_INTENSET_READY_Msk;
// Configure the retransmit counter
rfp->tx_remaining = rfp->config.retransmit.count;
rfp->state = NRF52_STATE_PTX_TX_ACK;
}
else {
NRF_RADIO->SHORTS = RADIO_SHORTS_COMMON;
NRF_RADIO->INTENSET = RADIO_INTENSET_DISABLED_Msk;
rfp->state = NRF52_STATE_PTX_TX;
}
break;
}
NRF_RADIO->TXADDRESS = p_current_payload->pipe;
NRF_RADIO->RXADDRESSES = 1 << p_current_payload->pipe;
NRF_RADIO->FREQUENCY = rfp->config.address.rf_channel;
NRF_RADIO->PACKETPTR = (uint32_t)tx_payload_buffer;
NRF_RADIO->EVENTS_READY = 0;
NRF_RADIO->EVENTS_DISABLED = 0;
(void)NRF_RADIO->EVENTS_READY;
(void)NRF_RADIO->EVENTS_DISABLED;
nvicClearPending(RADIO_IRQn);
nvicEnableVector(RADIO_IRQn, NRF52_RADIO_IRQ_PRIORITY);
NRF_RADIO->TASKS_TXEN = 1;
}
static void on_radio_disabled_tx_noack(RFDriver *rfp) {
rfp->flags |= NRF52_INT_TX_SUCCESS_MSK;
tx_fifo_remove_last();
chBSemSignal(&events_sem);
if (tx_fifo.count == 0) {
rfp->state = NRF52_STATE_IDLE;
}
else {
start_tx_transaction(rfp);
}
}
static void on_radio_disabled_tx(RFDriver *rfp) {
// Remove the DISABLED -> RXEN shortcut, to make sure the radio stays
// disabled after the RX window
NRF_RADIO->SHORTS = RADIO_SHORTS_COMMON;
// Make sure the timer is started the next time the radio is ready,
// and that it will disable the radio automatically if no packet is
// received by the time defined in m_wait_for_ack_timeout_us
rfp->timer->CC[0] = wait_for_ack_timeout_us + 130;
rfp->timer->CC[1] = rfp->config.retransmit.delay - 130;
rfp->timer->TASKS_CLEAR = 1;
rfp->timer->EVENTS_COMPARE[0] = 0;
rfp->timer->EVENTS_COMPARE[1] = 0;
(void)rfp->timer->EVENTS_COMPARE[0];
(void)rfp->timer->EVENTS_COMPARE[1];
NRF_PPI->CHENSET = (1 << NRF52_RADIO_PPI_TIMER_START) |
(1 << NRF52_RADIO_PPI_RX_TIMEOUT) |
(1 << NRF52_RADIO_PPI_TIMER_STOP);
NRF_PPI->CHENCLR = (1 << NRF52_RADIO_PPI_TX_START);
NRF_RADIO->EVENTS_END = 0;
(void)NRF_RADIO->EVENTS_END;
if (rfp->config.protocol == NRF52_PROTOCOL_ESB) {
set_rf_payload_format(rfp, 0);
}
NRF_RADIO->PACKETPTR = (uint32_t)rx_payload_buffer;
rfp->state = NRF52_STATE_PTX_RX_ACK;
}
static void on_radio_disabled_tx_wait_for_ack(RFDriver *rfp) {
// This marks the completion of a TX_RX sequence (TX with ACK)
// Make sure the timer will not deactivate the radio if a packet is received
NRF_PPI->CHENCLR = (1 << NRF52_RADIO_PPI_TIMER_START) |
(1 << NRF52_RADIO_PPI_RX_TIMEOUT) |
(1 << NRF52_RADIO_PPI_TIMER_STOP);
// If the radio has received a packet and the CRC status is OK
if (NRF_RADIO->EVENTS_END && NRF_RADIO->CRCSTATUS != 0) {
rfp->timer->TASKS_STOP = 1;
NRF_PPI->CHENCLR = (1 << NRF52_RADIO_PPI_TX_START);
rfp->flags |= NRF52_INT_TX_SUCCESS_MSK;
rfp->tx_attempt++;// = rfp->config.retransmit.count - rfp->tx_remaining + 1;
tx_fifo_remove_last();
if (rfp->config.protocol != NRF52_PROTOCOL_ESB && rx_payload_buffer[0] > 0) {
if (rx_fifo_push_rfbuf(rfp, (uint8_t)NRF_RADIO->TXADDRESS, 0)) {
rfp->flags |= NRF52_INT_RX_DR_MSK;
}
}
chBSemSignal(&events_sem);
if ((tx_fifo.count == 0) || (rfp->config.tx_mode == NRF52_TXMODE_MANUAL)) {
rfp->state = NRF52_STATE_IDLE;
}
else {
start_tx_transaction(rfp);
}
}
else {
if (rfp->tx_remaining-- == 0) {
rfp->timer->TASKS_STOP = 1;
NRF_PPI->CHENCLR = (1 << NRF52_RADIO_PPI_TX_START);
// All retransmits are expended, and the TX operation is suspended
rfp->tx_attempt = rfp->config.retransmit.count + 1;
rfp->flags |= NRF52_INT_TX_FAILED_MSK;
chBSemSignal(&events_sem);
rfp->state = NRF52_STATE_IDLE;
}
else {
// There are still have more retransmits left, TX mode should be
// entered again as soon as the system timer reaches CC[1].
NRF_RADIO->SHORTS = RADIO_SHORTS_COMMON | RADIO_SHORTS_DISABLED_RXEN_Msk;
set_rf_payload_format(rfp, p_current_payload->length);
NRF_RADIO->PACKETPTR = (uint32_t)tx_payload_buffer;
rfp->state = NRF52_STATE_PTX_TX_ACK;
rfp->timer->TASKS_START = 1;
NRF_PPI->CHENSET = (1 << NRF52_RADIO_PPI_TX_START);
if (rfp->timer->EVENTS_COMPARE[1])
NRF_RADIO->TASKS_TXEN = 1;
}
}
}
static void clear_events_restart_rx(RFDriver *rfp) {
NRF_RADIO->SHORTS = RADIO_SHORTS_COMMON;
set_rf_payload_format(rfp, rfp->config.payload_length);
NRF_RADIO->PACKETPTR = (uint32_t)rx_payload_buffer;
NRF_RADIO->INTENCLR = RADIO_INTENCLR_DISABLED_Msk;
NRF_RADIO->EVENTS_DISABLED = 0;
(void) NRF_RADIO->EVENTS_DISABLED;
NRF_RADIO->TASKS_DISABLE = 1;
while (NRF_RADIO->EVENTS_DISABLED == 0);
NRF_RADIO->EVENTS_DISABLED = 0;
(void) NRF_RADIO->EVENTS_DISABLED;
NRF_RADIO->INTENSET = RADIO_INTENSET_DISABLED_Msk;
NRF_RADIO->SHORTS = RADIO_SHORTS_COMMON | RADIO_SHORTS_DISABLED_TXEN_Msk;
NRF_RADIO->TASKS_RXEN = 1;
}
static void on_radio_disabled_rx(RFDriver *rfp) {
bool ack = false;
bool retransmit_payload = false;
bool send_rx_event = true;
pipe_info_t * p_pipe_info;
if (NRF_RADIO->CRCSTATUS == 0) {
clear_events_restart_rx(rfp);
return;
}
if(rx_fifo.count >= NRF52_RX_FIFO_SIZE) {
clear_events_restart_rx(rfp);
return;
}
p_pipe_info = &rx_pipe_info[NRF_RADIO->RXMATCH];
if (NRF_RADIO->RXCRC == p_pipe_info->m_crc &&
(rx_payload_buffer[1] >> 1) == p_pipe_info->m_pid ) {
retransmit_payload = true;
send_rx_event = false;
}
p_pipe_info->m_pid = rx_payload_buffer[1] >> 1;
p_pipe_info->m_crc = NRF_RADIO->RXCRC;
if(rfp->config.selective_auto_ack == false || ((rx_payload_buffer[1] & 0x01) == 0))
ack = true;
if(ack) {
NRF_RADIO->SHORTS = RADIO_SHORTS_COMMON | RADIO_SHORTS_DISABLED_RXEN_Msk;
switch(rfp->config.protocol) {
case NRF52_PROTOCOL_ESB_DPL:
{
if (tx_fifo.count > 0 &&
(tx_fifo.p_payload[tx_fifo.exit_point]->pipe == NRF_RADIO->RXMATCH))
{
// Pipe stays in ACK with payload until TX fifo is empty
// Do not report TX success on first ack payload or retransmit
if (p_pipe_info->m_ack_payload != 0 && !retransmit_payload) {
if(++tx_fifo.exit_point >= NRF52_TX_FIFO_SIZE) {
tx_fifo.exit_point = 0;
}
tx_fifo.count--;
// ACK payloads also require TX_DS
// (page 40 of the 'nRF24LE1_Product_Specification_rev1_6.pdf').
rfp->flags |= NRF52_INT_TX_SUCCESS_MSK;
}
p_pipe_info->m_ack_payload = 1;
p_current_payload = tx_fifo.p_payload[tx_fifo.exit_point];
set_rf_payload_format(rfp, p_current_payload->length);
tx_payload_buffer[0] = p_current_payload->length;
memcpy(&tx_payload_buffer[2],
p_current_payload->data,
p_current_payload->length);
}
else {
p_pipe_info->m_ack_payload = 0;
set_rf_payload_format(rfp, 0);
tx_payload_buffer[0] = 0;
}
tx_payload_buffer[1] = rx_payload_buffer[1];
}
break;
case NRF52_PROTOCOL_ESB:
{
set_rf_payload_format(rfp, 0);
tx_payload_buffer[0] = rx_payload_buffer[0];
tx_payload_buffer[1] = 0;
}
break;
}
rfp->state = NRF52_STATE_PRX_SEND_ACK;
NRF_RADIO->TXADDRESS = NRF_RADIO->RXMATCH;
NRF_RADIO->PACKETPTR = (uint32_t)tx_payload_buffer;
}
else {
clear_events_restart_rx(rfp);
}
if (send_rx_event) {
// Push the new packet to the RX buffer and trigger a received event if the operation was
// successful.
if (rx_fifo_push_rfbuf(rfp, NRF_RADIO->RXMATCH, p_pipe_info->m_pid)) {
rfp->flags |= NRF52_INT_RX_DR_MSK;
chBSemSignal(&events_sem);
}
}
}
static void on_radio_disabled_rx_ack(RFDriver *rfp) {
NRF_RADIO->SHORTS = RADIO_SHORTS_COMMON | RADIO_SHORTS_DISABLED_TXEN_Msk;
set_rf_payload_format(rfp, rfp->config.payload_length);
NRF_RADIO->PACKETPTR = (uint32_t)rx_payload_buffer;
rfp->state = NRF52_STATE_PRX;
}
nrf52_error_t radio_disable(void) {
RFD1.state = NRF52_STATE_IDLE;
// Clear PPI
NRF_PPI->CHENCLR = (1 << NRF52_RADIO_PPI_TIMER_START) |
(1 << NRF52_RADIO_PPI_TIMER_STOP) |
(1 << NRF52_RADIO_PPI_RX_TIMEOUT);
reset_fifo();
memset(rx_pipe_info, 0, sizeof(rx_pipe_info));
memset(pids, 0, sizeof(pids));
// Disable the radio
NRF_RADIO->SHORTS = RADIO_SHORTS_READY_START_Enabled << RADIO_SHORTS_READY_START_Pos |
RADIO_SHORTS_END_DISABLE_Enabled << RADIO_SHORTS_END_DISABLE_Pos;
nvicDisableVector(RADIO_IRQn);
// Terminate interrupts handle thread
chThdTerminate(rfIntThread_p);
chBSemSignal(&disable_sem);
chThdWait(rfIntThread_p);
// Terminate events handle thread
chThdTerminate(rfEvtThread_p);
RFD1.flags = 0;
chBSemSignal(&events_sem);
chThdWait(rfEvtThread_p);
RFD1.state = NRF52_STATE_UNINIT;
return NRF52_SUCCESS;
}
//
nrf52_error_t radio_init(nrf52_config_t const *config) {
osalDbgAssert(config != NULL,
"config must be defined");
osalDbgAssert(&config->address != NULL,
"address must be defined");
osalDbgAssert(NRF52_RADIO_IRQ_PRIORITY <= 7,
"wrong radio irq priority");
if (RFD1.state != NRF52_STATE_UNINIT) {
nrf52_error_t err = radio_disable();
if (err != NRF52_SUCCESS)
return err;
}
RFD1.radio = NRF_RADIO;
RFD1.config = *config;
RFD1.flags = 0;
init_fifo();
#if NRF52_RADIO_USE_TIMER0
RFD1.timer = NRF_TIMER0;
#endif
#if NRF52_RADIO_USE_TIMER1
RFD1.timer = NRF_TIMER1;
#endif
#if NRF52_RADIO_USE_TIMER2
RFD1.timer = NRF_TIMER2;
#endif
#if NRF52_RADIO_USE_TIMER3
RFD1.timer = NRF_TIMER3;
#endif
#if NRF52_RADIO_USE_TIMER4
RFD1.timer = NRF_TIMER4;
#endif
set_parameters(&RFD1);
set_addresses(&RFD1, NRF52_ADDR_UPDATE_MASK_BASE0);
set_addresses(&RFD1, NRF52_ADDR_UPDATE_MASK_BASE1);
set_addresses(&RFD1, NRF52_ADDR_UPDATE_MASK_PREFIX);
ppi_init(&RFD1);
timer_init(&RFD1);
chBSemObjectInit(&disable_sem, TRUE);
chBSemObjectInit(&events_sem, TRUE);
chEvtObjectInit(&RFD1.eventsrc);
// interrupt handle thread
rfIntThread_p = chThdCreateStatic(waRFIntThread, sizeof(waRFIntThread),
NRF52_RADIO_INTTHD_PRIORITY, rfIntThread, NULL);
// events handle thread
rfEvtThread_p = chThdCreateStatic(waRFEvtThread, sizeof(waRFEvtThread),
NRF52_RADIO_EVTTHD_PRIORITY, rfEvtThread, NULL);
nvicEnableVector(RADIO_IRQn, NRF52_RADIO_IRQ_PRIORITY);
RFD1.state = NRF52_STATE_IDLE;
return NRF52_SUCCESS;
}
nrf52_error_t radio_write_payload(nrf52_payload_t const * p_payload) {
if (RFD1.state == NRF52_STATE_UNINIT)
return NRF52_INVALID_STATE;
if(p_payload == NULL)
return NRF52_ERROR_NULL;
VERIFY_PAYLOAD_LENGTH(p_payload);
if (tx_fifo.count >= NRF52_TX_FIFO_SIZE)
return NRF52_ERROR_INVALID_LENGTH;
if (RFD1.config.mode == NRF52_MODE_PTX &&
p_payload->noack && !RFD1.config.selective_auto_ack )
{
return NRF52_ERROR_NOT_SUPPORTED;
}
nvicDisableVector(RADIO_IRQn);
memcpy(tx_fifo.p_payload[tx_fifo.entry_point], p_payload, sizeof(nrf52_payload_t));
pids[p_payload->pipe] = (pids[p_payload->pipe] + 1) % (NRF52_PID_MAX + 1);
tx_fifo.p_payload[tx_fifo.entry_point]->pid = pids[p_payload->pipe];
if (++tx_fifo.entry_point >= NRF52_TX_FIFO_SIZE) {
tx_fifo.entry_point = 0;
}
tx_fifo.count++;
nvicEnableVector(RADIO_IRQn, NRF52_RADIO_IRQ_PRIORITY);
if (RFD1.config.mode == NRF52_MODE_PTX &&
RFD1.config.tx_mode == NRF52_TXMODE_AUTO &&
RFD1.state == NRF52_STATE_IDLE)
{
start_tx_transaction(&RFD1);
}
return NRF52_SUCCESS;
}
nrf52_error_t radio_read_rx_payload(nrf52_payload_t * p_payload) {
if (RFD1.state == NRF52_STATE_UNINIT)
return NRF52_INVALID_STATE;
if (p_payload == NULL)
return NRF52_ERROR_NULL;
if (rx_fifo.count == 0) {
return NRF52_ERROR_INVALID_LENGTH;
}
nvicDisableVector(RADIO_IRQn);
p_payload->length = rx_fifo.p_payload[rx_fifo.exit_point]->length;
p_payload->pipe = rx_fifo.p_payload[rx_fifo.exit_point]->pipe;
p_payload->rssi = rx_fifo.p_payload[rx_fifo.exit_point]->rssi;
p_payload->pid = rx_fifo.p_payload[rx_fifo.exit_point]->pid;
memcpy(p_payload->data, rx_fifo.p_payload[rx_fifo.exit_point]->data, p_payload->length);
if (++rx_fifo.exit_point >= NRF52_RX_FIFO_SIZE) {
rx_fifo.exit_point = 0;
}
rx_fifo.count--;
nvicEnableVector(RADIO_IRQn, NRF52_RADIO_IRQ_PRIORITY);
return NRF52_SUCCESS;
}
nrf52_error_t radio_start_tx(void) {
if (RFD1.state != NRF52_STATE_IDLE)
return NRF52_ERROR_BUSY;
if (tx_fifo.count == 0) {
return NRF52_ERROR_INVALID_LENGTH;
}
start_tx_transaction(&RFD1);
return NRF52_SUCCESS;
}
nrf52_error_t radio_start_rx(void) {
if (RFD1.state != NRF52_STATE_IDLE)
return NRF52_ERROR_BUSY;
NRF_RADIO->INTENCLR = 0xFFFFFFFF;
NRF_RADIO->EVENTS_DISABLED = 0;
(void) NRF_RADIO->EVENTS_DISABLED;
NRF_RADIO->SHORTS = RADIO_SHORTS_COMMON | RADIO_SHORTS_DISABLED_TXEN_Msk;
NRF_RADIO->INTENSET = RADIO_INTENSET_DISABLED_Msk;
RFD1.state = NRF52_STATE_PRX;
NRF_RADIO->RXADDRESSES = RFD1.config.address.rx_pipes;
NRF_RADIO->FREQUENCY = RFD1.config.address.rf_channel;
NRF_RADIO->PACKETPTR = (uint32_t)rx_payload_buffer;
nvicClearPending(RADIO_IRQn);
nvicEnableVector(RADIO_IRQn, NRF52_RADIO_IRQ_PRIORITY);
NRF_RADIO->EVENTS_ADDRESS = 0;
NRF_RADIO->EVENTS_PAYLOAD = 0;
NRF_RADIO->EVENTS_DISABLED = 0;
(void) NRF_RADIO->EVENTS_ADDRESS;
(void) NRF_RADIO->EVENTS_PAYLOAD;
(void) NRF_RADIO->EVENTS_DISABLED;
NRF_RADIO->TASKS_RXEN = 1;
return NRF52_SUCCESS;
}
nrf52_error_t radio_stop_rx(void) {
if (RFD1.state != NRF52_STATE_PRX) {
return NRF52_INVALID_STATE;
}
NRF_RADIO->SHORTS = 0;
NRF_RADIO->INTENCLR = 0xFFFFFFFF;
NRF_RADIO->EVENTS_DISABLED = 0;
(void) NRF_RADIO->EVENTS_DISABLED;
NRF_RADIO->TASKS_DISABLE = 1;
while (NRF_RADIO->EVENTS_DISABLED == 0);
RFD1.state = NRF52_STATE_IDLE;
return NRF52_SUCCESS;
}
nrf52_error_t radio_flush_tx(void) {
if (RFD1.state == NRF52_STATE_UNINIT)
return NRF52_INVALID_STATE;
nvicDisableVector(RADIO_IRQn);
tx_fifo.count = 0;
tx_fifo.entry_point = 0;
tx_fifo.exit_point = 0;
nvicEnableVector(RADIO_IRQn, NRF52_RADIO_IRQ_PRIORITY);
return NRF52_SUCCESS;
}
nrf52_error_t radio_pop_tx(void) {
if (RFD1.state == NRF52_STATE_UNINIT)
return NRF52_INVALID_STATE;
if (tx_fifo.count == 0)
return NRF52_ERROR_INVALID_LENGTH;
nvicDisableVector(RADIO_IRQn);
if (++tx_fifo.entry_point >= NRF52_TX_FIFO_SIZE) {
tx_fifo.entry_point = 0;
}
tx_fifo.count--;
nvicEnableVector(RADIO_IRQn, NRF52_RADIO_IRQ_PRIORITY);
return NRF52_SUCCESS;
}
nrf52_error_t radio_flush_rx(void) {
if (RFD1.state == NRF52_STATE_UNINIT)
return NRF52_INVALID_STATE;
nvicDisableVector(RADIO_IRQn);
rx_fifo.count = 0;
rx_fifo.entry_point = 0;
rx_fifo.exit_point = 0;
memset(rx_pipe_info, 0, sizeof(rx_pipe_info));
nvicEnableVector(RADIO_IRQn, NRF52_RADIO_IRQ_PRIORITY);
return NRF52_SUCCESS;
}
nrf52_error_t radio_set_base_address_0(uint8_t const * p_addr) {
if (RFD1.state != NRF52_STATE_IDLE)
return NRF52_ERROR_BUSY;
if (p_addr == NULL)
return NRF52_ERROR_NULL;
memcpy(RFD1.config.address.base_addr_p0, p_addr, 4);
set_addresses(&RFD1, NRF52_ADDR_UPDATE_MASK_BASE0);
return NRF52_SUCCESS;
}
nrf52_error_t radio_set_base_address_1(uint8_t const * p_addr) {
if (RFD1.state != NRF52_STATE_IDLE)
return NRF52_ERROR_BUSY;
if (p_addr == NULL)
return NRF52_ERROR_NULL;
memcpy(RFD1.config.address.base_addr_p1, p_addr, 4);
set_addresses(&RFD1, NRF52_ADDR_UPDATE_MASK_BASE1);
return NRF52_SUCCESS;
}
nrf52_error_t radio_set_prefixes(uint8_t const * p_prefixes, uint8_t num_pipes) {
if (RFD1.state != NRF52_STATE_IDLE)
return NRF52_ERROR_BUSY;
if (p_prefixes == NULL)
return NRF52_ERROR_NULL;
if (num_pipes > 8)
return NRF52_ERROR_INVALID_PARAM;
memcpy(RFD1.config.address.pipe_prefixes, p_prefixes, num_pipes);
RFD1.config.address.num_pipes = num_pipes;
RFD1.config.address.rx_pipes = BIT_MASK_UINT_8(num_pipes);
set_addresses(&RFD1, NRF52_ADDR_UPDATE_MASK_PREFIX);
return NRF52_SUCCESS;
}
nrf52_error_t radio_set_prefix(uint8_t pipe, uint8_t prefix) {
if (RFD1.state != NRF52_STATE_IDLE)
return NRF52_ERROR_BUSY;
if (pipe > 8)
return NRF52_ERROR_INVALID_PARAM;
RFD1.config.address.pipe_prefixes[pipe] = prefix;
NRF_RADIO->PREFIX0 = bytewise_bit_swap(&RFD1.config.address.pipe_prefixes[0]);
NRF_RADIO->PREFIX1 = bytewise_bit_swap(&RFD1.config.address.pipe_prefixes[4]);
return NRF52_SUCCESS;
}