mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
634 lines
27 KiB
C
634 lines
27 KiB
C
|
/*
|
||
|
* Copyright (C) 2013 INRIA
|
||
|
* 2014 Freie Universität Berlin
|
||
|
* 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
||
|
* 2018,2019 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.
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* @defgroup drivers_cc110x CC1100/CC1100e/CC1101 Sub-GHz transceiver driver
|
||
|
* @ingroup drivers_netdev
|
||
|
*
|
||
|
* This module contains the driver for the TI CC1100/CC110e/CC1101 Sub-GHz
|
||
|
* transceivers.
|
||
|
*
|
||
|
* @warning How the CC1100/CC1101 operate can be configured quite sophistically.
|
||
|
* This has the drawback, that configurations breaking laws and rules
|
||
|
* are complete possible. Please make sure that the configured output
|
||
|
* power, duty cycle, frequency range, etc. conform to the rules,
|
||
|
* standards and laws that apply in your use case.
|
||
|
*
|
||
|
*
|
||
|
* Supported Hardware and how to obtain
|
||
|
* ====================================
|
||
|
*
|
||
|
* This driver has been developed for the CC1101 and the older CC1100
|
||
|
* transceiver and tested for both. However, it should work with the CC1100e
|
||
|
* as well - but this has *NOT* been tested at all.
|
||
|
*
|
||
|
* It is suggested to go for the CC1101 when considering to buy one of the
|
||
|
* supported transceivers. The easiest way is to obtain CC1101 break out boards
|
||
|
* with a complete antenna circuit & antenna that can be connected via jumper
|
||
|
* wires using an 8 pin DIP pin header. These are sold in various flavours start
|
||
|
* from less than 2€ at quantity one at your favourite Far East store. Beware
|
||
|
* that while the CC1101 chip can operate a various base frequencies, the
|
||
|
* antenna circuit will only work for a single frequency band. Most break out
|
||
|
* boards will operate at 433 MHz, which is license free in many countries (but
|
||
|
* verify that for your country before buying!). EU citizens might prefer the
|
||
|
* 868 MHz band over the 433 MHz, as more license free bandwidth is available
|
||
|
* in the 868 MHz band in the EU. (But when deploying only a few dozens of
|
||
|
* devices, the 433 MHz band is also fine for EU citizens.) US citizens should
|
||
|
* go for the 900 MHz band (as 868 MHz is not license free in the USA), which
|
||
|
* even contains more bandwidth than the 868 MHz band. (However, the 900 MHz
|
||
|
* band has not been tested, as using it would be illegal in the EU.)
|
||
|
*
|
||
|
*
|
||
|
* Packet Format
|
||
|
* =============
|
||
|
*
|
||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
* 0 1 2 3
|
||
|
* 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
|
||
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||
|
* | Preamble (4 bytes, handled by hardware, see MDMCFG1) |
|
||
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||
|
* | Sync Word (4 bytes, handled by hardware, see MDMCFG2) |
|
||
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||
|
* | Length Field | Destination | Source | Payload...
|
||
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||
|
* ... (see Length Field) | CRC (handled by hardware) |
|
||
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
*
|
||
|
* | Field | Description |
|
||
|
* |--------------|------------------------------------------------------------|
|
||
|
* | Preamble | 4 bytes, handled by hardware |
|
||
|
* | Sync Word | 4 bytes, handled by hardware |
|
||
|
* | Length Field | Handled by software & hardware, length of payload + 2 |
|
||
|
* | Destination | Handled by software & hardware, destination MAC address |
|
||
|
* | Source | Handled by software only, source MAC address |
|
||
|
* | Payload | Handled by software only, the payload to send |
|
||
|
* | CRC | 2 bytes, handled by hardware |
|
||
|
*
|
||
|
* The Length Field contains the length of the driver supplied data in bytes,
|
||
|
* not counting the Length Field. Thus, it contains the length of the payload
|
||
|
* plus the length of the Destination and Source address.
|
||
|
*
|
||
|
*
|
||
|
* Layer-2 Addresses
|
||
|
* -----------------
|
||
|
*
|
||
|
* The layer 2 addresses of the CC110x transceivers is a single byte long and
|
||
|
* the special value `0x00` for the destination address is used in broadcast
|
||
|
* transmissions. The transceiver is configured by this driver to ignore all
|
||
|
* packets unless the destination address matches the address of the transceiver
|
||
|
* or the destination address is `0x00`.
|
||
|
*
|
||
|
* Please note that the layer 2 address by default is derived from the CPU ID.
|
||
|
* Due to the birthday paradox with only 20 devices the probability of a
|
||
|
* collision is already bigger than 50%. Thus, manual address assignment is
|
||
|
* supported by defining `C110X_PARAM_L2ADDR`.
|
||
|
*
|
||
|
*
|
||
|
* Base Band, Data Rate, Channel Bandwidth and Channel Map Configuration
|
||
|
* =====================================================================
|
||
|
*
|
||
|
* This driver allows to configure the base band, the data rate and the channel
|
||
|
* bandwidth using an @ref cc110x_config_t data structure. Default
|
||
|
* configurations are supplied and name using the following scheme:
|
||
|
* `cc110x_config_<BASE-BAND>_<DATA-RATE>_<CHANNEL-BANDWIDTH>`. (E.g.
|
||
|
* @ref cc110x_config_868mhz_250kbps_300khz is the default configuration used by
|
||
|
* the MSB-A2 and the MSB-IoT boards.)
|
||
|
*
|
||
|
* Using the @ref cc110x_chanmap_t data structure the channel layout can be
|
||
|
* defined. This map contains 8 entries, each defines the offset from the base
|
||
|
* frequency defined in the @ref cc110x_config_t data structure for each
|
||
|
* channel in steps of 50kHz. E.g. @ref cc110x_chanmap_868mhz_lora provides
|
||
|
* the LoRa channels 10 to 17 in the 868MHz band. (The RIOT channel numbers
|
||
|
* will always start from 0, and currently only up to eight channels are
|
||
|
* supported. A special value of 255 as offset from the base frequency in the
|
||
|
* channel map is used mark the channel as disabled. This can be used if less
|
||
|
* than 8 non-overlapping channels are possible in the license free band.)
|
||
|
*
|
||
|
* Please note that the channel map (@ref cc110x_chanmap_t) must match the
|
||
|
* base configuration (@ref cc110x_config_t), as the channel map is relative
|
||
|
* to the configured base frequency. Also, the distance between the channels
|
||
|
* in the channel map should match the channel bandwidth of the configuration,
|
||
|
* as otherwise channels could overlap.
|
||
|
*
|
||
|
* Both configuration and matching channel map can be applied using
|
||
|
* @ref cc110x_apply_config. Please consider this as a slow operation, as the
|
||
|
* transceiver needs to be calibrated for each channel in the channel map.
|
||
|
*
|
||
|
*
|
||
|
* Calibration of the Frequency Generator
|
||
|
* ======================================
|
||
|
*
|
||
|
* The CC110x transceivers use a voltage controlled oscillator (VCO) and a
|
||
|
* phase locked loop (PLL) for frequency generation. However, they need to be
|
||
|
* calibrated to work correctly with the given supply voltage and the current
|
||
|
* temperature. The driver will perform this calibration during startup, but
|
||
|
* when the supply voltage or the temperature is not stable, a recalibration is
|
||
|
* required whenever the supply voltage of temperature has changed too much since
|
||
|
* the last calibration. This can be done by calling
|
||
|
* @ref cc110x_full_calibration. It is left to the application developer to
|
||
|
* perform this calibration when needed. During a test of about 2 hours of
|
||
|
* operation in an in-door environment with a stable temperature the CC1101 has
|
||
|
* worked reliable without any calibration at all (except for the automatic
|
||
|
* calibration at start up). So there are use cases which do not require any
|
||
|
* recalibration at all.
|
||
|
*
|
||
|
*
|
||
|
* Troubleshooting
|
||
|
* ===============
|
||
|
*
|
||
|
* The Driver Does Not Initialize Properly
|
||
|
* ---------------------------------------
|
||
|
* Set `ENABLE_DEBUG` in `cc110x_netdev.c` to `1` to get debug output, which
|
||
|
* will likely tell you what is going wrong. There are basically two things
|
||
|
* that can fail:
|
||
|
*
|
||
|
* Upon initialization the driver will read out the part number and version of
|
||
|
* the transceiver. If those do not match the ones expected for the CC1100,
|
||
|
* CC1100E, or the CC1101 the driver will refuse to initialize. If this fails,
|
||
|
* most likely incorrect values are read out and the SPI communication does not
|
||
|
* work correctly. However, future revisions of the CC110X transceivers might
|
||
|
* be produced and might have different values for the part number or version.
|
||
|
* If this should happen and they remain compatible with the driver, their
|
||
|
* part number & revision needs to be added to the driver.
|
||
|
*
|
||
|
* After uploading the configuration, the driver will read back the
|
||
|
* configuration to verify it. If the SPI communication is not reliable (e.g.
|
||
|
* sporadically bits flip), this will fail from time to time. E.g. on the
|
||
|
* MSB-IoT boards this is the case when the SPI interface operates at a clock of
|
||
|
* 5MHz, but it becomes reliable when clocked at 1MHz.
|
||
|
*
|
||
|
* The Driver Initializes, but Communication Is Impossible
|
||
|
* -------------------------------------------------------
|
||
|
* If two transceivers are too close to each other and TX power is at maximum,
|
||
|
* the signal is just too strong to be received correctly. Reducing TX power
|
||
|
* or increasing the distance (about half a meter should be fine) will solve
|
||
|
* this issue.
|
||
|
*
|
||
|
* While the chips can operate at any base frequency offered by the driver,
|
||
|
* the circuit the chip is connected to and the antenna are build for a single
|
||
|
* base band. Check if your configuration matches the frequency range the
|
||
|
* board is build for. E.g. most break out boards operate at 433MHz, but there
|
||
|
* are also boards for 868MHz.
|
||
|
*
|
||
|
* @{
|
||
|
*
|
||
|
* @file
|
||
|
* @brief Interface definition for the CC1100/CC1101 driver
|
||
|
*
|
||
|
* @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
|
||
|
* @author Oliver Hahm <oliver.hahm@inria.fr>
|
||
|
* @author Fabian Nack <nack@inf.fu-berlin.de>
|
||
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
||
|
*/
|
||
|
|
||
|
#ifndef CC110X_H
|
||
|
#define CC110X_H
|
||
|
|
||
|
#include <stdint.h>
|
||
|
|
||
|
#include "cc1xxx_common.h"
|
||
|
#include "mutex.h"
|
||
|
#include "net/gnrc/nettype.h"
|
||
|
#include "net/netdev.h"
|
||
|
#include "periph/adc.h"
|
||
|
#include "periph/gpio.h"
|
||
|
#include "periph/spi.h"
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
extern "C" {
|
||
|
#endif
|
||
|
|
||
|
/**
|
||
|
* @brief Length of a layer 2 frame
|
||
|
*
|
||
|
* This does not include the preamble, sync word, CRC field, and length field.
|
||
|
*/
|
||
|
#define CC110X_MAX_FRAME_SIZE 0xFF
|
||
|
|
||
|
/**
|
||
|
* @brief Maximum (layer 2) payload size supported by the driver
|
||
|
*/
|
||
|
#define CC110X_MAX_PAYLOAD_SIZE (CC110X_MAX_FRAME_SIZE - CC1XXX_HEADER_SIZE)
|
||
|
|
||
|
/**
|
||
|
* @brief Maximum number of channels supported by the driver
|
||
|
*/
|
||
|
#define CC110X_MAX_CHANNELS 8
|
||
|
|
||
|
/**
|
||
|
* @brief Special value to indicate that layer 2 address should be derived
|
||
|
* from the CPU-ID
|
||
|
*/
|
||
|
#define CC110X_L2ADDR_AUTO 0x00
|
||
|
|
||
|
/**
|
||
|
* @brief Default protocol for data that is coming in
|
||
|
*/
|
||
|
#ifdef MODULE_GNRC_SIXLOWPAN
|
||
|
#define CC110X_DEFAULT_PROTOCOL (GNRC_NETTYPE_SIXLOWPAN)
|
||
|
#else
|
||
|
#define CC110X_DEFAULT_PROTOCOL (GNRC_NETTYPE_UNDEF)
|
||
|
#endif
|
||
|
|
||
|
/**
|
||
|
* @brief The state of the CC1100/CC1101 transceiver
|
||
|
*
|
||
|
* The three least significant bytes match the representation of the matching
|
||
|
* transceiver state given in the status byte of the hardware. See Table 32 on
|
||
|
* page 31 in the data sheet for the possible states in the status byte.
|
||
|
*/
|
||
|
typedef enum {
|
||
|
CC110X_STATE_IDLE = 0b00000000, /**< IDLE state */
|
||
|
/**
|
||
|
* @brief Frame received, waiting for upper layer to retrieve it
|
||
|
*
|
||
|
* Transceiver is in IDLE state.
|
||
|
*/
|
||
|
CC110X_STATE_FRAME_READY = 0b00001000,
|
||
|
/**
|
||
|
* @brief Frame received, waiting for upper layer to retrieve it
|
||
|
*
|
||
|
* Transceiver is in SLEEP state. There is no matching representation in the
|
||
|
* status byte, as reading the status byte will power up the transceiver in
|
||
|
* bring it in the IDLE state. Thus, we set the three least significant bits
|
||
|
* to the IDLE state
|
||
|
*/
|
||
|
CC110X_STATE_OFF = 0b00010000,
|
||
|
CC110X_STATE_RX_MODE = 0b00000001, /**< Listening for frames */
|
||
|
/**
|
||
|
* @brief Receiving a frame just now
|
||
|
*
|
||
|
* Transceiver is in RX state.
|
||
|
*/
|
||
|
CC110X_STATE_RECEIVING = 0b00001001,
|
||
|
CC110X_STATE_TX_MODE = 0b00000010, /**< Transmit mode */
|
||
|
/**
|
||
|
* @brief Waiting for transceiver to complete outgoing transmission
|
||
|
*
|
||
|
* Transceiver is in TX state
|
||
|
*/
|
||
|
CC110X_STATE_TX_COMPLETING = 0b00001010,
|
||
|
CC110X_STATE_FSTXON = 0b00000011, /**< Fast TX ready */
|
||
|
CC110X_STATE_CALIBRATE = 0b00000100, /**< Device is calibrating */
|
||
|
CC110X_STATE_SETTLING = 0b00000101, /**< PLL is settling */
|
||
|
CC110X_STATE_RXFIFO_OVERFLOW = 0b00000110, /**< RX FIFO overflown */
|
||
|
CC110X_STATE_TXFIFO_UNDERFLOW = 0b00000111, /**< TX FIFO underflown */
|
||
|
} cc110x_state_t;
|
||
|
|
||
|
/**
|
||
|
* @brief Enumeration over the possible TX power settings the driver offers
|
||
|
*/
|
||
|
typedef enum {
|
||
|
CC110X_TX_POWER_MINUS_30_DBM, /**< -30 dBm */
|
||
|
CC110X_TX_POWER_MINUS_20_DBM, /**< -20 dBm */
|
||
|
CC110X_TX_POWER_MINUS_15_DBM, /**< -15 dBm */
|
||
|
CC110X_TX_POWER_MINUS_10_DBM, /**< -10 dBm */
|
||
|
CC110X_TX_POWER_0_DBM, /**< 0 dBm */
|
||
|
CC110X_TX_POWER_PLUS_5_DBM, /**< 5 dBm */
|
||
|
CC110X_TX_POWER_PLUS_7_DBM, /**< 7 dBm */
|
||
|
CC110X_TX_POWER_PLUS_10_DBM, /**< 1 dBm */
|
||
|
CC110X_TX_POWER_NUMOF, /**< Number of TX power options */
|
||
|
} cc110x_tx_power_t;
|
||
|
|
||
|
/**
|
||
|
* @brief Structure that holds the PATABLE, which allows to configure the
|
||
|
* 8 available output power levels using a magic number for each level.
|
||
|
*
|
||
|
* See Section "24 Output Power Programming" on page 59ff in the data sheet.
|
||
|
* The values suggested in Table 39 on page 60 in the data sheet are already
|
||
|
* available by this driver, but will only be linked in (8 bytes of ROM size)
|
||
|
* when they are referenced.
|
||
|
*
|
||
|
* @see cc110x_patable_433mhz
|
||
|
* @see cc110x_patable_868mhz
|
||
|
* @see cc110x_patable_915mhz
|
||
|
*/
|
||
|
typedef struct {
|
||
|
uint8_t data[8]; /**< Magic number to store in the configuration register */
|
||
|
} cc110x_patable_t;
|
||
|
|
||
|
|
||
|
/**
|
||
|
* @brief Configuration of the transceiver to use
|
||
|
*
|
||
|
* @warning Two transceivers with different configurations will be unable
|
||
|
* to communicate.
|
||
|
*
|
||
|
* The data uploaded into configuration registers are stored in
|
||
|
* @ref cc110x_conf. Most of them cannot be changed, as the driver relies on
|
||
|
* their values. However, the base frequency, the symbol rate (which equals
|
||
|
* the bit rate for the chosen modulation and error correction) and the
|
||
|
* channel bandwidth can be configured using this data structure.
|
||
|
*
|
||
|
* Please note that while the CC1100/CC1101 chip is compatible with a huge
|
||
|
* frequency range (300 MHz - 928 MHz), the complete circuit is optimized to
|
||
|
* a narrow frequency band. So make sure the configured base frequency is within
|
||
|
* that frequency band that is compatible with that circuit. (Most break out
|
||
|
* board will operate at the 433 MHz band. In the EU the 868 MHz band would be
|
||
|
* more interesting, but 433 MHz is license free as well. In the USA the 915 MHz
|
||
|
* band is license free.
|
||
|
*
|
||
|
* Please verify that the driver is configured in a way that allows legal
|
||
|
* operation according to rules and laws that apply for you.
|
||
|
*/
|
||
|
typedef struct {
|
||
|
uint8_t base_freq[3]; /**< Base frequency to use */
|
||
|
/**
|
||
|
* @brief FSCTRL1 configuration register value that affects the
|
||
|
* intermediate frequency of the transceiver to use
|
||
|
* @note The 4 most significant bits have to be 0.
|
||
|
*
|
||
|
* Assuming a 26 MHz crystal the IF is calculated as follows (in kHz):
|
||
|
*
|
||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
* double intermediate_frequency = 26000 / 1024 * fsctrl1;
|
||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
*/
|
||
|
uint8_t fsctrl1;
|
||
|
/**
|
||
|
* @brief MDMCFG4 configuration register value that affects channel filter
|
||
|
* bandwidth and the data rate
|
||
|
*
|
||
|
* See page 76 in the data sheet.
|
||
|
*
|
||
|
* Assuming a 26 MHz crystal the channel filter bandwidth is calculated
|
||
|
* as follows (in kHz):
|
||
|
*
|
||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
* uint8_t exponent = mdmcfg4 >> 6;
|
||
|
* uint8_t mantissa = (mdmcfg4 >> 4) & 0x03;
|
||
|
* double bandwidth = 26000.0 / (8 * (4 + mantissa) * (1L << exponent));
|
||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
*/
|
||
|
uint8_t mdmcfg4;
|
||
|
/**
|
||
|
* @brief MDMCFG3 configuration register value that affects the data rate
|
||
|
*
|
||
|
* @see cc110x_config_t::mdmcfg4
|
||
|
*
|
||
|
* See page 76 in the data sheet.
|
||
|
*
|
||
|
* Assuming a 26 MHz crystal the symbol rate of the transceiver is calculated
|
||
|
* as fallows (in kBaud):
|
||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
* uint8_t exponent = mdmcfg4 & 0x0f;
|
||
|
* int32_t mantissa = mdmcfg3;
|
||
|
* double baudrate = (256 + mantissa) * 26000.0 / (1L << (28 - exponent));
|
||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
*/
|
||
|
uint8_t mdmcfg3;
|
||
|
/**
|
||
|
* @brief DEVIANT configuration register that affects the amount by which
|
||
|
* the radio frequency is shifted in FSK/GFSK modulation
|
||
|
*
|
||
|
* @see cc110x_config_t::mdmcfg4
|
||
|
*
|
||
|
* See page 79 in the data sheet.
|
||
|
*
|
||
|
* In an ideal world the channel bandwidth would be twice the channel
|
||
|
* deviation. In the real world the used channel bandwidth is higher.
|
||
|
* Assuming a 26 MHz crystal and GFSK modulation (the driver will configure
|
||
|
* the transceiver to use GFSK) the deviation
|
||
|
*
|
||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
* uint8_t exponent = (deviatn >> 4) & 0x07;
|
||
|
* int32_t mantissa = deviatn & 0x07;
|
||
|
* double deviation = (8 + mantissa) * 26000.0 / (1L << (17 - exponent));
|
||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
*
|
||
|
* For reliable operation at high symbol rates, the deviation has to be
|
||
|
* increased as well.
|
||
|
*/
|
||
|
uint8_t deviatn;
|
||
|
} cc110x_config_t;
|
||
|
|
||
|
/**
|
||
|
* @brief Structure to hold mapping between virtual and physical channel numbers
|
||
|
*
|
||
|
* This driver will provide "virtual" channel numbers 0 to 7, which will be
|
||
|
* translated to "physical" channel numbers before being send to the
|
||
|
* transceiver. This is used to overcome the following limitations:
|
||
|
*
|
||
|
* - The transceiver does not support channel maps with varying distance between
|
||
|
* channels. However, e.g. the LoRa channels 10 - 16 in the 868 MHz band have
|
||
|
* a distance of 300 kHz, but channel 16 and 17 have a distance of 1 MHz.
|
||
|
* - The transceiver does not supports channel distances higher than 405.46 kHz.
|
||
|
*
|
||
|
* This mapping overcomes both limitations be using 50kHz physical channel
|
||
|
* spacing and use the map to translate to the correct physical channel. This
|
||
|
* also allows to keep the same MDMCFG1 and MDMCFG0 configuration register
|
||
|
* values for all channel layouts. Finally, different channel sets can be
|
||
|
* used by different groups of IoT device in the same environment to limit
|
||
|
* collisions between those groups - assuming that enough non-overlapping
|
||
|
* channels are available.
|
||
|
*
|
||
|
* The "virtual" channel (the channel number presented to RIOT) will be used
|
||
|
* as index in @ref cc110x_chanmap_t::map, the value in there will give the
|
||
|
* corresponding "physical" channel number, or 255 if this virtual channel
|
||
|
* number is not available.
|
||
|
*/
|
||
|
typedef struct {
|
||
|
uint8_t map[CC110X_MAX_CHANNELS]; /**< "Physical" channel numbers */
|
||
|
} cc110x_chanmap_t;
|
||
|
|
||
|
/**
|
||
|
* @brief Structure holding all parameter for driver initialization
|
||
|
*/
|
||
|
typedef struct {
|
||
|
const cc110x_patable_t *patable; /**< Pointer to the PATABLE to use */
|
||
|
/**
|
||
|
* @brief Pointer to the configuration of the base frequency, data rate and
|
||
|
* channel bandwidth; or `NULL` to keep the default.
|
||
|
*/
|
||
|
const cc110x_config_t *config;
|
||
|
const cc110x_chanmap_t *channels; /**< Pointer to the default channel map */
|
||
|
spi_t spi; /**< SPI bus connected to the device */
|
||
|
spi_clk_t spi_clk; /**< SPI clock to use (max 6.5 MHz) */
|
||
|
spi_cs_t cs; /**< GPIO pin connected to chip select */
|
||
|
gpio_t gdo0; /**< GPIO pin connected to GDO0 */
|
||
|
gpio_t gdo2; /**< GPIO pin connected to GDO2 */
|
||
|
/**
|
||
|
* @brief Layer-2 address to use or `CC110X_L2ADDR_AUTO` to derive it from
|
||
|
* the CPU ID
|
||
|
*/
|
||
|
uint8_t l2addr;
|
||
|
} cc110x_params_t;
|
||
|
|
||
|
/**
|
||
|
* @brief Structure holding the calibration data of the frequency synthesizer
|
||
|
*/
|
||
|
typedef struct {
|
||
|
/**
|
||
|
* @brief VCO capacitance calibration, which depends on the frequency and,
|
||
|
* thus, has to be stored for each channel
|
||
|
*/
|
||
|
char fscal1[CC110X_MAX_CHANNELS];
|
||
|
char fscal2; /**< VCO current calibration, independent of channel */
|
||
|
char fscal3; /**< charge pump current calibration, independent of channel */
|
||
|
} cc110x_fs_calibration_t;
|
||
|
|
||
|
/**
|
||
|
* @brief Buffer to temporary store incoming/outgoing packet
|
||
|
*
|
||
|
* The CC1100/CC1101 transceiver's FIFO sadly is only 64 bytes in size. To
|
||
|
* support frames bigger than that, chunks of the frame have to be
|
||
|
* transferred between the MCU and the CC1100/CC1101 transceiver while the
|
||
|
* frame is in transit.
|
||
|
*/
|
||
|
typedef struct __attribute__((packed)) {
|
||
|
uint8_t len; /**< Length of the frame in bytes */
|
||
|
/**
|
||
|
* @brief The payload data of the frame
|
||
|
*/
|
||
|
uint8_t data[CC110X_MAX_FRAME_SIZE];
|
||
|
/**
|
||
|
* @brief Index of the next @ref cc110x_framebuf_t::data element to transfer
|
||
|
*
|
||
|
* In RX mode: Index of the next @ref cc110x_framebuf_t::data element to store
|
||
|
* data read from the RX-FIFO into.
|
||
|
*
|
||
|
* In TX mode: Index of the next @ref cc110x_framebuf_t::data element to write
|
||
|
* to the TX-FIFO.
|
||
|
*/
|
||
|
uint8_t pos;
|
||
|
} cc110x_framebuf_t;
|
||
|
|
||
|
/**
|
||
|
* @brief Device descriptor for CC1100/CC1101 transceivers
|
||
|
*/
|
||
|
typedef struct {
|
||
|
netdev_t netdev; /**< RIOT's interface to this driver */
|
||
|
uint8_t addr; /**< Layer 2 address of this device */
|
||
|
/* Keep above in sync with cc1xx_t members, as they must overlap! */
|
||
|
cc110x_state_t state; /**< State of the transceiver */
|
||
|
cc110x_tx_power_t tx_power; /**< TX power of the receiver */
|
||
|
uint8_t channel; /**< Currently tuned (virtual) channel */
|
||
|
/* Struct packing: addr, state, tx_power and channel add up to 32 bit */
|
||
|
const cc110x_chanmap_t *channels; /**< Pointer to the channel map to use. */
|
||
|
cc110x_params_t params; /**< Configuration of the driver */
|
||
|
cc110x_framebuf_t buf; /**< Temporary frame buffer */
|
||
|
/**
|
||
|
* @brief RSSI and LQI of the last received frame
|
||
|
*/
|
||
|
cc1xxx_rx_info_t rx_info;
|
||
|
/**
|
||
|
* @brief Frequency synthesizer calibration data
|
||
|
*/
|
||
|
cc110x_fs_calibration_t fscal;
|
||
|
/**
|
||
|
* @brief Use mutex to block during TX and unblock from ISR when ISR
|
||
|
* needs to be handled from thread-context
|
||
|
*
|
||
|
* Blocking during TX within the driver prevents the upper layers from
|
||
|
* calling @ref netdev_driver_t::send while already transmitting a frame.
|
||
|
*/
|
||
|
mutex_t isr_signal;
|
||
|
uint8_t rssi_offset; /**< dBm to subtract from raw RSSI data */
|
||
|
} cc110x_t;
|
||
|
|
||
|
/**
|
||
|
* @brief Setup the CC1100/CC1101 driver, but perform no initialization
|
||
|
*
|
||
|
* @ref netdev_driver_t::init can be used after this call to initialize the
|
||
|
* transceiver.
|
||
|
*
|
||
|
* @param dev Device descriptor to use
|
||
|
* @param params Parameter of the device to setup
|
||
|
*
|
||
|
* @retval 0 Device successfully set up
|
||
|
* @retval -EINVAL @p dev or @p params is `NULL`, or @p params is invalid
|
||
|
*/
|
||
|
int cc110x_setup(cc110x_t *dev, const cc110x_params_t *params);
|
||
|
|
||
|
/**
|
||
|
* @brief Apply the given configuration and the given channel map and performs
|
||
|
* a recalibration
|
||
|
*
|
||
|
* @param dev Device descriptor of the transceiver
|
||
|
* @param conf Configuration to apply or `NULL` to only change channel map
|
||
|
* @param chanmap Channel map to apply (must be compatible with @p conf)
|
||
|
*
|
||
|
* @retval 0 Success
|
||
|
* @retval -EINVAL Called with invalid argument
|
||
|
* @retval -EIO Communication with the transceiver failed
|
||
|
*
|
||
|
* @pre The application developer checked in the documentation that the channel
|
||
|
* map in @p chanmap is compatible with the configuration in @p conf
|
||
|
*
|
||
|
* Because the configuration (potentially) changes the channel bandwidth, the
|
||
|
* old channel map is rendered invalid. This API therefore asks for both to make
|
||
|
* sure an application developer does not forget to update the channel map.
|
||
|
* Because the old calibration data is also rendered invalid,
|
||
|
* @ref cc110x_full_calibration is called to update it.
|
||
|
*/
|
||
|
int cc110x_apply_config(cc110x_t *dev, const cc110x_config_t *conf,
|
||
|
const cc110x_chanmap_t *chanmap);
|
||
|
|
||
|
/**
|
||
|
* @brief Perform a calibration of the frequency generator for each supported
|
||
|
* channel
|
||
|
*
|
||
|
* @param dev Device descriptor of the transceiver
|
||
|
*
|
||
|
* @retval 0 Success
|
||
|
* @retval -EINVAL Called with invalid argument
|
||
|
* @retval -EAGAIN Current state prevents deliberate calibration
|
||
|
* @retval -EIO Communication with the transceiver failed
|
||
|
*
|
||
|
* Tunes in each supported channel and calibrates the transceiver. The
|
||
|
* calibration data is stored so that @ref cc110x_set_channel can skip the
|
||
|
* calibration phase and use the stored calibration data instead.
|
||
|
*/
|
||
|
int cc110x_full_calibration(cc110x_t *dev);
|
||
|
|
||
|
/**
|
||
|
* @brief Hops to the specified channel
|
||
|
*
|
||
|
* @param dev Device descriptor of the transceiver
|
||
|
* @param channel Channel to hop to
|
||
|
*
|
||
|
* @retval 0 Success
|
||
|
* @retval -EINVAL Called with `NULL` as @p dev
|
||
|
* @retval -ERANGE Channel out of range or not supported by channel map
|
||
|
* @retval -EAGAIN Currently in a state that does not allow hopping, e.g.
|
||
|
* sending/receiving a packet, calibrating or handling
|
||
|
* transmission errors
|
||
|
* @retval -EIO Communication with the transceiver failed
|
||
|
*
|
||
|
* This function implements the fact channel hopping approach outlined in
|
||
|
* section 28.2 on page 64 in the data sheet, which skips the calibration phase
|
||
|
* by storing the calibration date for each channel in the driver.
|
||
|
*/
|
||
|
int cc110x_set_channel(cc110x_t *dev, uint8_t channel);
|
||
|
|
||
|
/**
|
||
|
* @brief Set the TX power to the specified value
|
||
|
*
|
||
|
* @param dev Device descriptor of the transceiver
|
||
|
* @param power Output power to apply
|
||
|
*
|
||
|
* @retval 0 Success
|
||
|
* @retval -EINVAL Called with `NULL` as @p dev
|
||
|
* @retval -ERANGE Called with an invalid value for @p power
|
||
|
* @retval -EAGAIN Changing the TX power is in the current state not possible
|
||
|
* @retval -EIO Communication with the transceiver failed
|
||
|
*/
|
||
|
int cc110x_set_tx_power(cc110x_t *dev, cc110x_tx_power_t power);
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#endif /* CC110X_H */
|
||
|
/** @} */
|