2020-05-04 03:25:22 +02:00
|
|
|
/*
|
|
|
|
* 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"
|
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2020-05-04 03:25:22 +02:00
|
|
|
#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);
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|