1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-15 23:53:08 +01:00
RIOT/cpu/nrf5x_common/radio/nrfble/nrfble.c

358 lines
9.5 KiB
C
Raw Normal View History

2018-04-05 10:58:52 +02:00
/*
* Copyright (C) 2017-2018 Freie Universität Berlin
*
* 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_nrf5x_nrfble
*
* @note This driver is not thread safe and should only be used by a
* single thread at once!
* @{
*
* @file
* @brief Bluetooth low energy radio driver for nRF5x SoCs
*
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
*
* @}
*/
#include <errno.h>
#include "cpu.h"
#include "assert.h"
#include "nrfble.h"
#include "net/netdev/ble.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
/* driver specific device configuration */
#define CONF_MODE RADIO_MODE_MODE_Ble_1Mbit
#define CONF_LEN (8U)
#define CONF_S0 (1U)
#define CONF_S1 (0U)
#define CONF_STATLEN (0U)
#define CONF_BASE_ADDR_LEN (3U)
#define CONF_ENDIAN RADIO_PCNF1_ENDIAN_Little
#define CONF_WHITENING RADIO_PCNF1_WHITEEN_Enabled
#define CONF_TIFS (150U)
#define CONF_CRC_LEN (0X3 | RADIO_CRCCNF_SKIPADDR_Msk)
#define CONF_CRC_POLY (0x00065b)
/* used shortcuts */
#define SHORTS_BASE (RADIO_SHORTS_READY_START_Msk | \
RADIO_SHORTS_END_DISABLE_Msk)
#define SHORTS_RX (SHORTS_BASE | RADIO_SHORTS_DISABLED_TXEN_Msk)
#define SHORTS_TX (SHORTS_BASE | RADIO_SHORTS_DISABLED_RXEN_Msk)
/* interrupt masks */
#define INT_DIS (RADIO_INTENCLR_DISABLED_Msk | \
RADIO_INTENCLR_ADDRESS_Msk)
#define INT_EN (RADIO_INTENSET_DISABLED_Msk | \
RADIO_INTENSET_ADDRESS_Msk)
/* driver internal radio states */
#define STATE_IDLE (0x00)
#define STATE_RX (0x01)
#define STATE_TX (0x02)
#define STATE_BUSY (0x80)
/* forward declaration of the netdev driver struct */
static const netdev_driver_t netdev_driver;
/* current radio state */
static volatile uint8_t _state = STATE_IDLE;
/* active radio context */
static netdev_ble_ctx_t *_ctx = NULL;
/* allocate the netdev device descriptor */
netdev_t _nrfble_dev;
/* map logical BLE channel values to actual radio frequencies */
static const uint8_t _ble_chan_map[40] = {
[ 0] = 4,
[ 1] = 6,
[ 2] = 8,
[ 3] = 10,
[ 4] = 12,
[ 5] = 14,
[ 6] = 16,
[ 7] = 18,
[ 8] = 20,
[ 9] = 22,
[10] = 24,
[11] = 28,
[12] = 30,
[13] = 32,
[14] = 34,
[15] = 36,
[16] = 38,
[17] = 40,
[18] = 42,
[19] = 44,
[20] = 46,
[21] = 48,
[22] = 50,
[23] = 52,
[24] = 54,
[25] = 56,
[26] = 58,
[27] = 60,
[28] = 62,
[29] = 64,
[30] = 66,
[31] = 68,
[32] = 70,
[33] = 72,
[34] = 74,
[35] = 76,
[36] = 78,
[37] = 2,
[38] = 26,
[39] = 80,
};
/**
* @brief Set radio into idle (DISABLED) state
*/
static void _go_idle(void)
{
if (!(_state & STATE_BUSY)) {
NRF_RADIO->INTENCLR = INT_DIS;
NRF_RADIO->SHORTS = 0;
NRF_RADIO->EVENTS_DISABLED = 0;
NRF_RADIO->TASKS_DISABLE = 1;
while (NRF_RADIO->EVENTS_DISABLED == 0) {}
_state = STATE_IDLE;
}
}
/**
* @brief Set radio context (channel, CRC, whitening, and access address)
*/
static void _set_context(netdev_ble_ctx_t *ctx)
{
if (ctx) {
assert(ctx->chan <= NRFBLE_CHAN_MAX);
_ctx = ctx;
NRF_RADIO->FREQUENCY = _ble_chan_map[ctx->chan];
NRF_RADIO->PREFIX0 = ctx->aa.raw[3];
NRF_RADIO->BASE0 = (uint32_t)((ctx->aa.raw[0] << 8) |
(ctx->aa.raw[1] << 16) |
(ctx->aa.raw[2] << 24));
NRF_RADIO->DATAWHITEIV = ctx->chan;
NRF_RADIO->CRCINIT = (ctx->crc & NETDEV_BLE_CRC_MASK);
}
else {
_go_idle();
}
}
static int16_t _nrfble_get_txpower(void)
{
int8_t p = (int8_t)NRF_RADIO->TXPOWER;
if (p < 0) {
return (int16_t)(0xff00 | p);
}
return (int16_t)p;
}
static void _nrfble_set_txpower(int16_t power)
{
if (power > 2) {
NRF_RADIO->TXPOWER = RADIO_TXPOWER_TXPOWER_Pos4dBm;
}
else if (power > -2) {
NRF_RADIO->TXPOWER = RADIO_TXPOWER_TXPOWER_0dBm;
}
else if (power > -6) {
NRF_RADIO->TXPOWER = RADIO_TXPOWER_TXPOWER_Neg4dBm;
}
else if (power > -10) {
NRF_RADIO->TXPOWER = RADIO_TXPOWER_TXPOWER_Neg8dBm;
}
else if (power > -14) {
NRF_RADIO->TXPOWER = RADIO_TXPOWER_TXPOWER_Neg12dBm;
}
else if (power > -18) {
NRF_RADIO->TXPOWER = RADIO_TXPOWER_TXPOWER_Neg16dBm;
}
else if (power > -25) {
NRF_RADIO->TXPOWER = RADIO_TXPOWER_TXPOWER_Neg20dBm;
}
else {
NRF_RADIO->TXPOWER = RADIO_TXPOWER_TXPOWER_Neg30dBm;
}
}
/**
* @brief Radio interrupt routine
*/
void isr_radio(void)
{
if (NRF_RADIO->EVENTS_ADDRESS) {
NRF_RADIO->EVENTS_ADDRESS = 0;
_state |= STATE_BUSY;
}
if (NRF_RADIO->EVENTS_DISABLED) {
NRF_RADIO->EVENTS_DISABLED = 0;
if (_state == (STATE_BUSY | STATE_RX)) {
_state = STATE_TX;
if (NRF_RADIO->CRCSTATUS) {
_ctx->crc |= NETDEV_BLE_CRC_OK;
}
else {
_ctx->crc &= ~(NETDEV_BLE_CRC_OK);
}
_nrfble_dev.event_callback(&_nrfble_dev, NETDEV_EVENT_RX_COMPLETE);
}
else { /* on TX done */
_state = STATE_RX;
_nrfble_dev.event_callback(&_nrfble_dev, NETDEV_EVENT_TX_COMPLETE);
}
}
cortexm_isr_end();
}
netdev_t *nrfble_setup(void)
{
_nrfble_dev.driver = &netdev_driver;
_nrfble_dev.event_callback = NULL;
_nrfble_dev.context = NULL;
return &_nrfble_dev;
}
static int _nrfble_init(netdev_t *dev)
{
(void)dev;
assert(_nrfble_dev.driver && _nrfble_dev.event_callback);
/* power on the NRFs radio */
NRF_RADIO->POWER = 1;
/* configure variable parameters to default values */
NRF_RADIO->TXPOWER = NRFBLE_TXPOWER_DEFAULT;
/* always send from and listen to logical address 0 */
NRF_RADIO->TXADDRESS = 0x00UL;
NRF_RADIO->RXADDRESSES = 0x01UL;
/* load driver specific configuration */
NRF_RADIO->MODE = CONF_MODE;
/* configure data fields and packet length whitening and endianness */
2018-04-05 10:58:52 +02:00
NRF_RADIO->PCNF0 = ((CONF_S1 << RADIO_PCNF0_S1LEN_Pos) |
(CONF_S0 << RADIO_PCNF0_S0LEN_Pos) |
(CONF_LEN << RADIO_PCNF0_LFLEN_Pos));
NRF_RADIO->PCNF1 = ((CONF_WHITENING << RADIO_PCNF1_WHITEEN_Pos) |
(CONF_ENDIAN << RADIO_PCNF1_ENDIAN_Pos) |
(CONF_BASE_ADDR_LEN << RADIO_PCNF1_BALEN_Pos) |
(CONF_STATLEN << RADIO_PCNF1_STATLEN_Pos) |
(NETDEV_BLE_PDU_MAXLEN << RADIO_PCNF1_MAXLEN_Pos));
/* set inter frame spacing to 150us */
NRF_RADIO->TIFS = CONF_TIFS;
/* configure CRC length and polynomial */
NRF_RADIO->CRCCNF = CONF_CRC_LEN;
NRF_RADIO->CRCPOLY = CONF_CRC_POLY;
/* enable global interrupts, but mask all local interrupts for now */
NRF_RADIO->INTENCLR = 0xffffffff;
NVIC_EnableIRQ(RADIO_IRQn);
DEBUG("[nrfble] initialization successful\n");
return 0;
}
static int _nrfble_send(netdev_t *dev, const iolist_t *data)
{
(void)dev;
assert(data);
NRF_RADIO->PACKETPTR = (uint32_t)data->iol_base;
NRF_RADIO->SHORTS = SHORTS_TX;
/* in case no trx sequence is active, we start a new one now */
if (_state == STATE_IDLE) {
_state = STATE_TX;
NRF_RADIO->EVENTS_DISABLED = 0;
NRF_RADIO->INTENSET = INT_EN;
NRF_RADIO->TASKS_TXEN = 1;
}
return 0;
}
static int _nrfble_recv(netdev_t *dev, void *buf, size_t len, void *info)
{
(void)dev;
(void)len;
(void)info;
assert(buf && (len == sizeof(netdev_ble_pkt_t)));
NRF_RADIO->PACKETPTR = (uint32_t)buf;
NRF_RADIO->SHORTS = SHORTS_RX;
/* in case no trx sequence is active, we start a new one now */
if (_state == STATE_IDLE) {
_state = STATE_RX;
NRF_RADIO->EVENTS_DISABLED = 0;
NRF_RADIO->INTENSET = INT_EN;
NRF_RADIO->TASKS_RXEN = 1;
}
return 0;
}
static int _nrfble_get(netdev_t *dev, netopt_t opt, void *val, size_t max_len)
{
(void)dev;
(void)max_len;
switch (opt) {
case NETOPT_TX_POWER:
assert(max_len >= sizeof(int16_t));
*((int16_t *)val) = _nrfble_get_txpower();
return sizeof(int16_t);
case NETOPT_DEVICE_TYPE:
assert(max_len >= sizeof(uint16_t));
*((uint16_t *)val) = NETDEV_TYPE_BLE;
return sizeof(uint16_t);
default:
return -ENOTSUP;
}
}
static int _nrfble_set(netdev_t *dev, netopt_t opt, const void *val, size_t len)
{
(void)dev;
(void)len;
switch (opt) {
case NETOPT_BLE_CTX:
_set_context((netdev_ble_ctx_t *)val);
return sizeof(netdev_ble_ctx_t);
case NETOPT_TX_POWER:
assert(len == sizeof(int16_t));
_nrfble_set_txpower(*((const int16_t *)val));
return sizeof(int16_t);
default:
return -ENOTSUP;
}
}
/* export of the netdev interface */
static const netdev_driver_t netdev_driver = {
.send = _nrfble_send,
.recv = _nrfble_recv,
.init = _nrfble_init,
.isr = NULL,
.get = _nrfble_get,
.set = _nrfble_set
};