/* * 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 * @} */ #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); 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); }