1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-15 11:32:44 +01:00
RIOT/drivers/cc110x/cc110x_settings.c

331 lines
12 KiB
C
Raw Normal View History

drivers/cc110x: Rewrite of the cc110x driver The cc110x driver has been re-written from scratch to overcome the limitations of the old driver. The main motivation of the rewrite was to achieve better maintainability by a detailed documentation, reduce the complexity and the overhead of the SPI communication with the device, and to allow to simultaneously use transceivers with different configuration regarding the used base band, the channel bandwidth, the modulation rate, and the channel map. Features of this driver include: - Support for the CC1100, CC1101, and the CC1100e sub-gigahertz transceivers. - Detailed documentation of every aspect of this driver. - An easy to use configuration API that allows setting the transceiver configuration (modulation rate, channel bandwidth, base frequency) and the channel map. - Fast channel hopping by pre-calibration of the channels during device configuration (so that no calibration is needed during hopping). - Simplified SPI communication: Only during start-up the MCU has to wait for the transceiver to be ready (for the power regulators and the crystal to stabilize). The old driver did this for every SPI transfer, which resulted in complex communication code. This driver will wait on start up for the transceiver to power up and then use RIOT's SPI API like every other driver. (Not only the data sheet states that this is fine, it also proved to be reliable in practise.) - Greatly reduced latency: The RTT on the old driver (@150 kbps data rate) was about 16ms, the new driver (@250 kbps data rate) has as RTT of ~3ms (depending on SPI clock and on CPU performance) (measured with ping6). - Increased reliability: The preamble size and the sync word size have been doubled compared to the old driver (preamble: 8 bytes instead of 4, sync word: 4 byte instead of 2). The new values are the once recommended by the data sheet for reliable communication. - Basic diagnostic during driver initialization to detect common issues as SPI communication issues and GDO pin configuration/wiring issues. - TX power configuration with netdev_driver_t::set() API-integration - Calls to netdev_driver_t::send() block until the transmission has completed to ease the use of the API (implemented without busy waiting, so that the MCU can enter lower power states or other threads can be executed).
2018-11-08 17:37:07 +01:00
/*
* Copyright (C) 2018 Otto-von-Guericke-Universität Magdeburg
*
* 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_cc110x
drivers/cc110x: Rewrite of the cc110x driver The cc110x driver has been re-written from scratch to overcome the limitations of the old driver. The main motivation of the rewrite was to achieve better maintainability by a detailed documentation, reduce the complexity and the overhead of the SPI communication with the device, and to allow to simultaneously use transceivers with different configuration regarding the used base band, the channel bandwidth, the modulation rate, and the channel map. Features of this driver include: - Support for the CC1100, CC1101, and the CC1100e sub-gigahertz transceivers. - Detailed documentation of every aspect of this driver. - An easy to use configuration API that allows setting the transceiver configuration (modulation rate, channel bandwidth, base frequency) and the channel map. - Fast channel hopping by pre-calibration of the channels during device configuration (so that no calibration is needed during hopping). - Simplified SPI communication: Only during start-up the MCU has to wait for the transceiver to be ready (for the power regulators and the crystal to stabilize). The old driver did this for every SPI transfer, which resulted in complex communication code. This driver will wait on start up for the transceiver to power up and then use RIOT's SPI API like every other driver. (Not only the data sheet states that this is fine, it also proved to be reliable in practise.) - Greatly reduced latency: The RTT on the old driver (@150 kbps data rate) was about 16ms, the new driver (@250 kbps data rate) has as RTT of ~3ms (depending on SPI clock and on CPU performance) (measured with ping6). - Increased reliability: The preamble size and the sync word size have been doubled compared to the old driver (preamble: 8 bytes instead of 4, sync word: 4 byte instead of 2). The new values are the once recommended by the data sheet for reliable communication. - Basic diagnostic during driver initialization to detect common issues as SPI communication issues and GDO pin configuration/wiring issues. - TX power configuration with netdev_driver_t::set() API-integration - Calls to netdev_driver_t::send() block until the transmission has completed to ease the use of the API (implemented without busy waiting, so that the MCU can enter lower power states or other threads can be executed).
2018-11-08 17:37:07 +01:00
* @{
*
* @file
* @brief On-chip settings for the TI CC1100/CC1101 transceiver
drivers/cc110x: Rewrite of the cc110x driver The cc110x driver has been re-written from scratch to overcome the limitations of the old driver. The main motivation of the rewrite was to achieve better maintainability by a detailed documentation, reduce the complexity and the overhead of the SPI communication with the device, and to allow to simultaneously use transceivers with different configuration regarding the used base band, the channel bandwidth, the modulation rate, and the channel map. Features of this driver include: - Support for the CC1100, CC1101, and the CC1100e sub-gigahertz transceivers. - Detailed documentation of every aspect of this driver. - An easy to use configuration API that allows setting the transceiver configuration (modulation rate, channel bandwidth, base frequency) and the channel map. - Fast channel hopping by pre-calibration of the channels during device configuration (so that no calibration is needed during hopping). - Simplified SPI communication: Only during start-up the MCU has to wait for the transceiver to be ready (for the power regulators and the crystal to stabilize). The old driver did this for every SPI transfer, which resulted in complex communication code. This driver will wait on start up for the transceiver to power up and then use RIOT's SPI API like every other driver. (Not only the data sheet states that this is fine, it also proved to be reliable in practise.) - Greatly reduced latency: The RTT on the old driver (@150 kbps data rate) was about 16ms, the new driver (@250 kbps data rate) has as RTT of ~3ms (depending on SPI clock and on CPU performance) (measured with ping6). - Increased reliability: The preamble size and the sync word size have been doubled compared to the old driver (preamble: 8 bytes instead of 4, sync word: 4 byte instead of 2). The new values are the once recommended by the data sheet for reliable communication. - Basic diagnostic during driver initialization to detect common issues as SPI communication issues and GDO pin configuration/wiring issues. - TX power configuration with netdev_driver_t::set() API-integration - Calls to netdev_driver_t::send() block until the transmission has completed to ease the use of the API (implemented without busy waiting, so that the MCU can enter lower power states or other threads can be executed).
2018-11-08 17:37:07 +01:00
*
* @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
drivers/cc110x: Rewrite of the cc110x driver The cc110x driver has been re-written from scratch to overcome the limitations of the old driver. The main motivation of the rewrite was to achieve better maintainability by a detailed documentation, reduce the complexity and the overhead of the SPI communication with the device, and to allow to simultaneously use transceivers with different configuration regarding the used base band, the channel bandwidth, the modulation rate, and the channel map. Features of this driver include: - Support for the CC1100, CC1101, and the CC1100e sub-gigahertz transceivers. - Detailed documentation of every aspect of this driver. - An easy to use configuration API that allows setting the transceiver configuration (modulation rate, channel bandwidth, base frequency) and the channel map. - Fast channel hopping by pre-calibration of the channels during device configuration (so that no calibration is needed during hopping). - Simplified SPI communication: Only during start-up the MCU has to wait for the transceiver to be ready (for the power regulators and the crystal to stabilize). The old driver did this for every SPI transfer, which resulted in complex communication code. This driver will wait on start up for the transceiver to power up and then use RIOT's SPI API like every other driver. (Not only the data sheet states that this is fine, it also proved to be reliable in practise.) - Greatly reduced latency: The RTT on the old driver (@150 kbps data rate) was about 16ms, the new driver (@250 kbps data rate) has as RTT of ~3ms (depending on SPI clock and on CPU performance) (measured with ping6). - Increased reliability: The preamble size and the sync word size have been doubled compared to the old driver (preamble: 8 bytes instead of 4, sync word: 4 byte instead of 2). The new values are the once recommended by the data sheet for reliable communication. - Basic diagnostic during driver initialization to detect common issues as SPI communication issues and GDO pin configuration/wiring issues. - TX power configuration with netdev_driver_t::set() API-integration - Calls to netdev_driver_t::send() block until the transmission has completed to ease the use of the API (implemented without busy waiting, so that the MCU can enter lower power states or other threads can be executed).
2018-11-08 17:37:07 +01:00
* @}
*/
#include "cc110x.h"
#include "cc110x_internal.h"
const char cc110x_conf[CC110X_CONF_SIZE] = {
/*
* IOCFG2; default: 0x29 (CHIP_RDYn)
* Invert GDO2: off,
* GDO2: Go high when RX data should be read
*
* Why not default?
* GDO2 will be used to be notified about FIFO events (e.g. refilling TX
* FIFO is needed during transmission, reading from RX FIFO is needed
* during reception)
*/
CC110X_GDO_ON_RX_DATA,
/*
* IOCFG1; default: 0x2E (3-state)
* Invert GDO1: off,
* GDO1: 3-state (required when SPI interface is shared with other devices)
*/
0x2E,
/*
* IOCFG0; default: 0x3F (CLK_XOSC/192)
* Invert GDO0: off,
* GDO0: Go high on PLL in lock
*
* Why not default?
* GDO0 will be used to be notified when a packet is coming while in RX
* mode (will go high during transmission) and when sending is completed
* while in TX (remains high during transmission and will go back low when
* done).
*/
CC110X_GDO_ON_TRANSMISSION,
/*
* FIFOTHR; default: 0x07
* TEST1 = 0x31 and TEST2 = 0x88 when waking up from SLEEP,
* 0dB RX attenuation,
* threshold for FIFOs: TX FIFO = 33, RX FIFO = 32
*/
0x07,
/*
* SYNC1, SYNC0; defaults: 0xD3, 0x91
* Use 0xD3,0x91 as sync word
*/
0xD3, /*< SYNC1 */
0x91, /*< SYNC0 */
/*
* PKTLEN; default: 0xFF
* Packet length in bytes in fixed length mode, else maximum length
*/
0xff,
/*
* PKTCTRL1; default: 0x04
* PQT: Accept all sync words, regardless of preamble quality
* CRC_AUTOFLUSH: Do not auto-flush RX FIFO on incorrect CRC
* APPEND_STATUS: Do not add 2 bytes of status information in RX FIFO
* ADDR_CHK: Filter incoming frames in hardware by address: Only frames
* with destination address 0x00 (broadcast) or with with the
* layer-2 address of the transceiver are accepted.
*
* Why not default?
* - The RSSI, LQI and CRC info are also available via status registers.
* Thus, it is not worth to sacrifice two bytes of RX FIFO for it.
* - Hardware address filtering could reduce the number IRQs generated
* (e.g. a huge frame is dropped before it fully received) which reduces
* the system's load. Thus, it is enabled.
*/
CC110X_PKTCTRL1_VALUE | CC110X_PKTCTRL1_ADDR_MATCH,
drivers/cc110x: Rewrite of the cc110x driver The cc110x driver has been re-written from scratch to overcome the limitations of the old driver. The main motivation of the rewrite was to achieve better maintainability by a detailed documentation, reduce the complexity and the overhead of the SPI communication with the device, and to allow to simultaneously use transceivers with different configuration regarding the used base band, the channel bandwidth, the modulation rate, and the channel map. Features of this driver include: - Support for the CC1100, CC1101, and the CC1100e sub-gigahertz transceivers. - Detailed documentation of every aspect of this driver. - An easy to use configuration API that allows setting the transceiver configuration (modulation rate, channel bandwidth, base frequency) and the channel map. - Fast channel hopping by pre-calibration of the channels during device configuration (so that no calibration is needed during hopping). - Simplified SPI communication: Only during start-up the MCU has to wait for the transceiver to be ready (for the power regulators and the crystal to stabilize). The old driver did this for every SPI transfer, which resulted in complex communication code. This driver will wait on start up for the transceiver to power up and then use RIOT's SPI API like every other driver. (Not only the data sheet states that this is fine, it also proved to be reliable in practise.) - Greatly reduced latency: The RTT on the old driver (@150 kbps data rate) was about 16ms, the new driver (@250 kbps data rate) has as RTT of ~3ms (depending on SPI clock and on CPU performance) (measured with ping6). - Increased reliability: The preamble size and the sync word size have been doubled compared to the old driver (preamble: 8 bytes instead of 4, sync word: 4 byte instead of 2). The new values are the once recommended by the data sheet for reliable communication. - Basic diagnostic during driver initialization to detect common issues as SPI communication issues and GDO pin configuration/wiring issues. - TX power configuration with netdev_driver_t::set() API-integration - Calls to netdev_driver_t::send() block until the transmission has completed to ease the use of the API (implemented without busy waiting, so that the MCU can enter lower power states or other threads can be executed).
2018-11-08 17:37:07 +01:00
/*
* PKTCTRL0; default: 0x45
* Data whitening enabled, use RX/TX FIFOs, CRC enabled,
* variable packet length
*/
0x45,
/*
* ADDR; default: 0x00
* Address will overwritten later
*
* Why not default?
* 0x00 is used as broadcast address. Using it would increase chance to
* receive message during device initialization and thus power consumption.
*/
0xFF,
/*
* CHANNR; default: 0x00
* Channel number 0 by default
*/
0x00,
/*
* FSCTRL1; default: 0x0C
* Intermediate frequency: 0x0C * 26MHz / 1024 = 304.7kHz
*
* Why not defaults?
* See MDMCFG4, MDMCFG3
*/
0x0C,
/*
* FSCTRL0; default: 0x00
* Frequency offset to base frequency: 0kHz
*/
0x00,
/*
* FREQ2, FREQ1, FREQ0; defaults: 0x1E, 0xC4, 0xEC
* 0x2146E4 * 26MHz / 65536 = 865.1998 MHz (LoRa Channel 10)
*
* Why not defaults?
* Default is 800.000 MHz, which is not in a license free frequency band.
* Using LoRa channel 10 instead.
*/
0x21, /*< FREQ2 */
0x46, /*< FREQ1 */
0xE4, /*< FREQ0 */
/*
* MDMCFG4, MDMCFG3; defaults: 0x8C, 0x22
* 541.67 kHz channel filter bandwidth,
* 249.94 kBaud symbol rate
*
* Why not defaults?
* Using 250 kBaud (==> 250kBit/s when 2-FSK/GFSK/ASK) to compete with
* 802.15.4 in data rate.
* Using settings listed in Table 7 (pages 12ff in the data sheet):
* - 250 kBaud
* - GFSK modulation
* - 304 kHz IF frequency
* - 540 kHz channel filter bandwidth
* - 127 kHz deviation
*/
0x2D, /*< MDMCFG4 */
0x3B, /*< MDMCFG3 */
/*
* MDMCFG2; default: 0x02
* DC blocking filter on,
* GFSK modulation,
* no manchester code,
* Enable RX when 30 bits of the 32 bits sync word are received correctly
*
* Why not default?
* Default expects all 16 bits of a two byte sync word to be correctly
* received. The data sheet recommends to expect 30 bits of a four byte
* sync word to be correctly received instead (see section 15 on page 37),
* so we go for this.
*
* Using GFSK instead of 2-FSK reduces the modulated spectrum width and is
* suggested in Table 7 of the datasheet, see MDMCFG4, MDMCFG3
*/
0x13,
/*
* MDMCFG1, MDMCFG0; defaults: 0x22, 0xF8
* FEC disabled,
* 4 preamble bytes,
* 49.99 kHz distance between channel centre frequencies (closest to 50kHz)
*
* Why not defaults?
* This driver uses an translation layer between physical channels (with
* 50 kHz distance) and "virtual" channel numbers as seen outside of the
* driver. This allows to set the frequency in a range of 12.75 MHz with
* a resolution of 50kHz - this seems to allow to configure all desired
* channel layouts.
*/
0x20, /*< MDMCFG1 */
0xF8, /*< MDMCFG0 */
/*
* DEVIATN; default: 0x47
* Deviation of frequency to encode data: +- 126.953kHz in 2-FSK/4-FSK/GFSK
*
* Why not default?
* Higher deviation required for reliable operation at 250 kbps data rate.
*/
0x62,
/*
* MCSM2; default: 0x07
* No direct RX termination on RSSI measurement (only for ASK/OOK),
* on RX timeout check for sync word and ignore PQI,
* no RX timeout
*/
0x07,
/*
* MCSM1; default: 0x30
* CCA: Enter TX even when channel is detected busy
* go to idle after packet received,
* go to idle after packet sent
*
* Why not default?
* By default the transceiver refuses to enter TX when the channel is
* detected busy. While this is desired, checking if TX was successfully
* entered is too slow and generated interrupts on the GDO2 pin are easily
* missed. However, reading the carrier sense value in the PKTSTATUS
* registers allows to implement this feature in software in a faster
* way. In addition to not missing GDO2 interrupts, this allows the send
* function to give feedback right away when the channel is busy.
*/
0x00,
/*
* MCSM0; default: 0x04
* Auto calibration: Disabled, driver has to manage
* delay to allow supply voltage to stabilize: 149-155 microseconds
* pin radio control option is off,
* oscillator is off in sleep state
*
* Why not default?
* Using 149µs-155µs instead of the default 37µs-39µs as PO_TIMEOUT is
* encouraged by the data sheet for robust operation
*/
0x08,
/*
* FOCCFG; default: 0x36
* Freeze frequency offset compensation etc until CS high: yes
* frequency compensation loop gain before sync word: 3K
* frequency compensation loop gain after sync word: K/2
* saturation point for frequency offset compensation: 25% channel bandwidth
* (incompatible with ASK/OOK)
*/
0x36,
/*
* BSCFG; default: 0x6C
* Clock recovery feedback loop integral gain before sync word: 2K_i,
* clock recovery feedback loop proportional gain before sync word: 3K_p,
* clock recovery feedback loop integral gain after sync word: K_i/2
* clock recovery feedback loop proportional gain after sync word: K_p,
* data rate offset compensation: Disabled
*/
0x6C,
/*
* AGCCTRL2; default: 0x03
* Maximum allowable DVGA gain: No limitation, maximum DVGA gain can be used
* Maximum allowable LNA + LNA2 gain: No limitation
* target amplitude from channel filter: 33 dB (default)
*/
0x03,
/*
* AGCCTRL1; default: 0x40
* LNA priority: Decrease LNA gain first, start decreasing LNA2 gain when
* LNA gain reached minimum
* Relative carrier sense threshold: Disabled
* Absolute carrier sense threshold: At MAGN_TARGET
*/
0x40,
/*
* AGCCTRL0; default: 0x91
* HYST_LEVEL: Medium hysteresis, medium asymmetric dead zone, medium gain
* Adjust gain after how many channel filter samples: After 16 samples
* Freeze AGC gain: Never; perform gain adjustments as required
* FILTER_LENGTH:
* - 16 channel filter samples when 2-FSK, 4-FSK, or MSK is used
* - 8 dB decision baundry when OOK/ASK is used
*/
0x91,
/*
* WOREVT1, WOREVT0, WORCTRL; defaults: 0x87, 0x6B, 0xF8
* Event0 Timeout: 1.000 seconds
* RC_PD: 1 (The datasheet is quite cryptic regarding this setting)
* Event1 Timeout: 1.333ms - 1.385ms (depending on crystal frequency)
* RC oscillator calibration: Enabled
* WOR_RES: 0 (Relevant for Event0 resolution and maximum timeout)
*/
0x87, /*< WOREVT1 */
0x6B, /*< WOREVT0 */
0xF8, /*< WORCTRL */
/*
* FREND1; default: 0x56
* LNA_CURRENT: 0b01
* LNA2MIX_CURRENT: 0b01
* LODIV_BUF_CURRENT_RW: 0b01
* MIX_CURRENT: 0b10
*/
0x56,
/*
* FREND0; default: 0x10
* LODIV_BUF_CURRENT_TX: 0b01
* Index in PA_POWER table (in 0..7, default is 0): 4 (0dBm)
*
* Why not default:
* Use a reasonable TX power level instead of the lowest.
*/
0x10 | CC110X_TX_POWER_0_DBM,
drivers/cc110x: Rewrite of the cc110x driver The cc110x driver has been re-written from scratch to overcome the limitations of the old driver. The main motivation of the rewrite was to achieve better maintainability by a detailed documentation, reduce the complexity and the overhead of the SPI communication with the device, and to allow to simultaneously use transceivers with different configuration regarding the used base band, the channel bandwidth, the modulation rate, and the channel map. Features of this driver include: - Support for the CC1100, CC1101, and the CC1100e sub-gigahertz transceivers. - Detailed documentation of every aspect of this driver. - An easy to use configuration API that allows setting the transceiver configuration (modulation rate, channel bandwidth, base frequency) and the channel map. - Fast channel hopping by pre-calibration of the channels during device configuration (so that no calibration is needed during hopping). - Simplified SPI communication: Only during start-up the MCU has to wait for the transceiver to be ready (for the power regulators and the crystal to stabilize). The old driver did this for every SPI transfer, which resulted in complex communication code. This driver will wait on start up for the transceiver to power up and then use RIOT's SPI API like every other driver. (Not only the data sheet states that this is fine, it also proved to be reliable in practise.) - Greatly reduced latency: The RTT on the old driver (@150 kbps data rate) was about 16ms, the new driver (@250 kbps data rate) has as RTT of ~3ms (depending on SPI clock and on CPU performance) (measured with ping6). - Increased reliability: The preamble size and the sync word size have been doubled compared to the old driver (preamble: 8 bytes instead of 4, sync word: 4 byte instead of 2). The new values are the once recommended by the data sheet for reliable communication. - Basic diagnostic during driver initialization to detect common issues as SPI communication issues and GDO pin configuration/wiring issues. - TX power configuration with netdev_driver_t::set() API-integration - Calls to netdev_driver_t::send() block until the transmission has completed to ease the use of the API (implemented without busy waiting, so that the MCU can enter lower power states or other threads can be executed).
2018-11-08 17:37:07 +01:00
/*
* FSCAL3, FSCAL2, FSCAL1, FSCAL0; defaults: 0xA9, 0x0A, 0x20, 0x0d
* These values store calibration date of the CC1100/CC1101 transceiver.
* Once the transceiver performs a calibration, those registers are updated
* with the new calibration data. In a "stable" environment (e.g. constant
* channel/frequency, stable humidity, temperature, supply voltage etc.)
* the obtained values could be written to the transceiver and calibration
* could be turned off completely.
*
* Fast channel hopping could be performed by obtaining the FSCAL1
* calibration data for each channel and storing it in the MCU's RAM.
* The other calibration values is frequency independent according to the
* data sheet, but depends on temperature etc.
*
* Once the FSCAL1 values for each channel are stored, the calibration can
* be disabled and the stored FSCAL1 data can be uploaded for each channel
* hop. A re-calibration from time to time is suggested to cope with changes
* in the environment, e.g. in temperature or supply voltage.
*
* Why not defaults?
* Using "magic" values obtained with SmartRF Studio software for 868 MHz
* band.
*/
0xEA, /*< FSCAL3: charge pump current calibration, frequency independent */
0x2A, /*< FSCAL2: VCO current calibration, frequency independent */
0x00, /*< FSCAL1: VCO capacitance calibration, frequency dependent */
0x1F, /*< FSCAL0: "Magic number", use SmartRF Studio to obtain */
/*
* RCCTRL1, RCCTRL0; defaults: 0x41, 0x00
* RC oscillator configuration, no explanation given in data sheet.
*/
0x41, /*< RCCTRL1 */
0x00, /*< RCCTRL0 */
};
const char cc110x_magic_registers[3] = { 0x88, 0x31, 0x09 };