1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/drivers/at86rf215/at86rf215_ofdm.c
2020-11-03 10:51:21 +01:00

331 lines
9.3 KiB
C

/*
* Copyright (C) 2019 ML!PA Consulting GmbH
*
* 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_at86rf215
* @{
*
* @file
* @brief Configuration of the MR-OFDM PHY on the AT86RF215 chip
*
* @author Benjamin Valentin <benjamin.valentin@ml-pa.com>
* @}
*/
#include "at86rf215.h"
#include "at86rf215_internal.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
/* symbol time is always 120 µs for MR-OFDM */
#define OFDM_SYMBOL_TIME_US 120
/* IEEE Std 802.15.4g™-2012 Amendment 3
* Table 68d—Total number of channels and first channel center frequencies for SUN PHYs */
static uint32_t _channel_spacing_kHz(uint8_t option)
{
switch (option) {
case 1: return 1200;
case 2: return 800;
case 3: return 400;
case 4: return 200;
}
return 0;
}
/* IEEE Std 802.15.4g™-2012 Amendment 3
* Table 68d—Total number of channels and first channel center frequencies for SUN PHYs */
static uint32_t _channel_center_freq_kHz_868MHz(uint8_t option)
{
switch (option) {
case 1: return 863625;
case 2: return 863425;
case 3: return 863225;
case 4: return 863125;
}
return 0;
}
/* IEEE Std 802.15.4g™-2012 Amendment 3
* Table 68d—Total number of channels and first channel center frequencies for SUN PHYs */
static uint32_t _channel_center_freq_kHz_2400MHz(uint8_t option)
{
return 2400000 + _channel_spacing_kHz(option) - CCF0_24G_OFFSET;
}
/* IEEE Std 802.15.4g™-2012 Amendment 3
* Table 68d—Total number of channels and first channel center frequencies for SUN PHYs */
static uint16_t _get_max_chan(at86rf215_t *dev, uint8_t option)
{
if (is_subGHz(dev)) {
switch (option) {
case 1: return 5;
case 2: return 8;
case 3: return 17;
case 4: return 34;
}
}
else {
switch (option) {
case 1: return 64;
case 2: return 97;
case 3: return 207;
case 4: return 416;
}
}
return 0;
}
/* Table 6-90. Recommended Transmitter Frontend Configuration */
static uint32_t _TXCUTC_LPFCUT(uint8_t option)
{
switch (option) {
case 1: return 10 << TXCUTC_LPFCUT_SHIFT;
case 2: return 8 << TXCUTC_LPFCUT_SHIFT;
case 3: return 5 << TXCUTC_LPFCUT_SHIFT;
case 4: return 3 << TXCUTC_LPFCUT_SHIFT;
}
return 0;
}
/* Table 6-90. Recommended Transmitter Frontend Configuration */
static uint32_t _TXDFE_SR(uint8_t option)
{
switch (option) {
case 1:
case 2: return 3 << TXDFE_SR_SHIFT;
case 3:
case 4: return 6 << TXDFE_SR_SHIFT;
}
return 0;
}
/* Table 6-90. Recommended Transmitter Frontend Configuration */
static uint32_t _TXDFE_RCUT(uint8_t option)
{
switch (option) {
case 1: return 3 << TXDFE_RCUT_SHIFT;
case 2:
case 3: return 3 << TXDFE_RCUT_SHIFT;
case 4: return 2 << TXDFE_RCUT_SHIFT;
}
return 0;
}
/* Table 6-93. Recommended PHY Receiver and Digital Frontend Configuration */
static uint32_t _RXDFE_RCUT(uint8_t option, bool superGHz)
{
switch (option) {
case 1: return 4 << RXDFE_RCUT_SHIFT;
case 2: return 2 << RXDFE_RCUT_SHIFT;
case 3: return (2 + superGHz) << RXDFE_RCUT_SHIFT;
case 4: return 1 << RXDFE_RCUT_SHIFT;
}
return 0;
}
/* Table 6-93. Recommended PHY Receiver and Digital Frontend Configuration */
static uint32_t _RXBWC_BW(uint8_t option, bool superGHz)
{
switch (option) {
case 1: return (9 + superGHz) << RXBWC_BW_SHIFT;
case 2: return 7 << RXBWC_BW_SHIFT;
case 3: return (4 + superGHz) << RXBWC_BW_SHIFT;
case 4: return (2 + superGHz) << RXBWC_BW_SHIFT;
}
return 0;
}
/* Table 6-93. Recommended PHY Receiver and Digital Frontend Configuration */
static uint32_t _RXBWC_IFS(uint8_t option, bool superGHz)
{
switch (option) {
case 1:
case 2: return 1;
case 3: return superGHz;
case 4: return !superGHz;
}
return 0;
}
static void _set_option(at86rf215_t *dev, uint8_t option)
{
const bool superGHz = !is_subGHz(dev);
/* Set Receiver Bandwidth */
at86rf215_reg_write(dev, dev->RF->RG_RXBWC,
_RXBWC_BW(option, superGHz) | _RXBWC_IFS(option, superGHz));
/* Set fS (same as TX); fCUT for RX */
at86rf215_reg_write(dev, dev->RF->RG_RXDFE,
_TXDFE_SR(option) | _RXDFE_RCUT(option, superGHz));
/* Set Power Amplifier Ramp Time; fLPCUT */
at86rf215_reg_write(dev, dev->RF->RG_TXCUTC,
RF_PARAMP8U | _TXCUTC_LPFCUT(option));
/* Set fS; fCUT for TX */
at86rf215_reg_write(dev, dev->RF->RG_TXDFE,
_TXDFE_SR(option) | _TXDFE_RCUT(option));
/* set channel spacing with 25 kHz resolution */
at86rf215_reg_write(dev, dev->RF->RG_CS, _channel_spacing_kHz(option) / 25);
/* set channel center frequency with 25 kHz resolution */
if (superGHz) {
at86rf215_reg_write16(dev, dev->RF->RG_CCF0L,
1 + _channel_center_freq_kHz_2400MHz(option) / 25);
}
else {
at86rf215_reg_write16(dev, dev->RF->RG_CCF0L,
1 + _channel_center_freq_kHz_868MHz(option) / 25);
}
at86rf215_reg_write(dev, dev->BBC->RG_OFDMC, option - 1);
/* make sure channel config is still valid */
dev->num_chans = _get_max_chan(dev, option);
dev->netdev.chan = at86rf215_chan_valid(dev, dev->netdev.chan);
at86rf215_reg_write16(dev, dev->RF->RG_CNL, dev->netdev.chan);
}
static unsigned _get_frame_duration(uint8_t option, uint8_t scheme,
uint8_t bytes)
{
/* Table 150 - phySymbolsPerOctet values for MR-OFDM PHY, IEEE 802.15.4g-2012 */
static const uint8_t quot[] = { 3, 3, 6, 12, 18, 24, 36 };
--option;
/* phyMaxFrameDuration = phySHRDuration + phyPHRDuration + ceiling [(aMaxPHYPacketSize + 1) x phySymbolsPerOctet] */
const unsigned phySHRDuration = 6;
const unsigned phyPHRDuration = option ? 6 : 3;
const unsigned phyPDUDuration = ((bytes + 1) * (1 << option) + quot[scheme] - 1)
/ quot[scheme];
return (phySHRDuration + phyPHRDuration + phyPDUDuration) * OFDM_SYMBOL_TIME_US;
}
static void _set_ack_timeout(at86rf215_t *dev, uint8_t option, uint8_t scheme)
{
dev->ack_timeout_usec = dev->csma_backoff_period
+ IEEE802154G_ATURNAROUNDTIME_US
+ _get_frame_duration(option, scheme,
AT86RF215_ACK_PSDU_BYTES);
DEBUG("[%s] ACK timeout: %" PRIu32 " µs\n", "OFDM", dev->ack_timeout_usec);
}
static bool _option_mcs_valid(uint8_t option, uint8_t mcs)
{
if (option < 1 || option > 4) {
return false;
}
if (mcs > BB_MCS_16QAM_3BY4) {
return false;
}
if (mcs == BB_MCS_BPSK_REP4 && option > 2) {
return false;
}
if (mcs == BB_MCS_BPSK_REP2 && option == 4) {
return false;
}
return true;
}
int at86rf215_configure_OFDM(at86rf215_t *dev, uint8_t option, uint8_t scheme)
{
if (!_option_mcs_valid(option, scheme)) {
DEBUG("[%s] invalid option/MCS: %d | %d\n", __func__, option, scheme);
return -EINVAL;
}
at86rf215_await_state_end(dev, RF_STATE_TX);
/* disable radio */
at86rf215_reg_write(dev, dev->BBC->RG_PC, 0);
/* set receiver gain target according to data sheet */
at86rf215_reg_write(dev, dev->RF->RG_AGCS, 3 << AGCS_TGT_SHIFT);
/* enable automatic receiver gain */
at86rf215_reg_write(dev, dev->RF->RG_AGCC, AGCC_EN_MASK);
_set_option(dev, option);
at86rf215_reg_write(dev, dev->BBC->RG_OFDMPHRTX, scheme);
dev->csma_backoff_period = AT86RF215_BACKOFF_PERIOD_IN_SYMBOLS *
OFDM_SYMBOL_TIME_US;
DEBUG("[%s] CSMA BACKOFF: %" PRIu32 " µs\n", "OFDM",
dev->csma_backoff_period);
_set_ack_timeout(dev, option, scheme);
/* disable FSK preamble switching */
#ifdef MODULE_NETDEV_IEEE802154_MR_FSK
dev->fsk_pl = 0;
#endif
at86rf215_enable_radio(dev, BB_MROFDM);
return 0;
}
int at86rf215_OFDM_set_scheme(at86rf215_t *dev, uint8_t scheme)
{
uint8_t option = at86rf215_OFDM_get_option(dev);
if (!_option_mcs_valid(option, scheme)) {
DEBUG("[%s] invalid MCS: %d\n", __func__, scheme);
return -1;
}
at86rf215_await_state_end(dev, RF_STATE_TX);
at86rf215_reg_write(dev, dev->BBC->RG_OFDMPHRTX, scheme);
_set_ack_timeout(dev, at86rf215_OFDM_get_option(dev), scheme);
return 0;
}
uint8_t at86rf215_OFDM_get_scheme(at86rf215_t *dev)
{
return at86rf215_reg_read(dev, dev->BBC->RG_OFDMPHRTX) & OFDMPHRTX_MCS_MASK;
}
int at86rf215_OFDM_set_option(at86rf215_t *dev, uint8_t option)
{
uint8_t mcs = at86rf215_OFDM_get_scheme(dev);
if (!_option_mcs_valid(option, mcs)) {
DEBUG("[%s] invalid option: %d\n", __func__, option);
return -1;
}
at86rf215_await_state_end(dev, RF_STATE_TX);
_set_option(dev, option);
_set_ack_timeout(dev, option, mcs);
return 0;
}
uint8_t at86rf215_OFDM_get_option(at86rf215_t *dev)
{
return 1 + (at86rf215_reg_read(dev, dev->BBC->RG_OFDMC) & OFDMC_OPT_MASK);
}