2021-03-11 17:28:33 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2021 Inria
|
|
|
|
*
|
|
|
|
* 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_sx126x
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Device driver implementation for the SX1261/2/8 and LLCC68 LoRa radio driver
|
|
|
|
*
|
|
|
|
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "sx126x_netdev.h"
|
|
|
|
|
|
|
|
#include "net/lora.h"
|
|
|
|
#include "periph/spi.h"
|
|
|
|
|
|
|
|
#include "sx126x_driver.h"
|
|
|
|
#include "sx126x.h"
|
|
|
|
#include "sx126x_params.h"
|
2021-06-30 13:34:20 +02:00
|
|
|
#include "sx126x_internal.h"
|
2021-03-11 17:28:33 +01:00
|
|
|
|
|
|
|
#define ENABLE_DEBUG 0
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
#ifndef CONFIG_SX126X_PKT_TYPE_DEFAULT
|
|
|
|
#define CONFIG_SX126X_PKT_TYPE_DEFAULT (SX126X_PKT_TYPE_LORA)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef CONFIG_SX126X_CHANNEL_DEFAULT
|
|
|
|
#define CONFIG_SX126X_CHANNEL_DEFAULT (868300000UL) /* in Hz */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef CONFIG_SX126X_TX_POWER_DEFAULT
|
|
|
|
#define CONFIG_SX126X_TX_POWER_DEFAULT (14U) /* in dBm */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef CONFIG_SX126X_RAMP_TIME_DEFAULT
|
|
|
|
#define CONFIG_SX126X_RAMP_TIME_DEFAULT (SX126X_RAMP_10_US)
|
|
|
|
#endif
|
|
|
|
|
2021-06-30 13:26:05 +02:00
|
|
|
const sx126x_pa_cfg_params_t sx1268_pa_cfg = {
|
|
|
|
.pa_duty_cycle = 0x04,
|
|
|
|
.hp_max = 0x06,
|
|
|
|
.device_sel = 0x00,
|
|
|
|
.pa_lut = 0x01
|
|
|
|
};
|
|
|
|
|
2021-11-04 21:54:40 +01:00
|
|
|
const sx126x_pa_cfg_params_t lpa_cfg = {
|
2021-06-30 13:26:05 +02:00
|
|
|
.pa_duty_cycle = 0x04,
|
|
|
|
.hp_max = 0x00,
|
|
|
|
.device_sel = 0x01,
|
|
|
|
.pa_lut = 0x01
|
|
|
|
};
|
|
|
|
|
2021-11-04 21:54:40 +01:00
|
|
|
const sx126x_pa_cfg_params_t hpa_cfg = {
|
|
|
|
.pa_duty_cycle = 0x02,
|
|
|
|
.hp_max = 0x02,
|
|
|
|
.device_sel = 0x00,
|
|
|
|
.pa_lut = 0x01
|
|
|
|
};
|
|
|
|
|
2021-03-11 17:28:33 +01:00
|
|
|
void sx126x_setup(sx126x_t *dev, const sx126x_params_t *params, uint8_t index)
|
|
|
|
{
|
2021-06-22 11:00:29 +02:00
|
|
|
netdev_t *netdev = &dev->netdev;
|
2021-03-11 17:28:33 +01:00
|
|
|
|
|
|
|
netdev->driver = &sx126x_driver;
|
|
|
|
dev->params = (sx126x_params_t *)params;
|
|
|
|
netdev_register(&dev->netdev, NETDEV_SX126X, index);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const uint16_t _bw_khz[3] = {
|
|
|
|
[LORA_BW_125_KHZ] = 125,
|
|
|
|
[LORA_BW_250_KHZ] = 250,
|
|
|
|
[LORA_BW_500_KHZ] = 500,
|
|
|
|
};
|
|
|
|
|
|
|
|
static uint8_t _compute_ldro(sx126x_t *dev)
|
|
|
|
{
|
2021-07-09 11:10:49 +02:00
|
|
|
uint32_t symbol_len =
|
|
|
|
(uint32_t)(1 << dev->mod_params.sf) / _bw_khz[dev->mod_params.bw - SX126X_LORA_BW_125];
|
|
|
|
|
2021-03-11 17:28:33 +01:00
|
|
|
if (symbol_len >= 16) {
|
|
|
|
return 0x01;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0x00;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sx126x_init_default_config(sx126x_t *dev)
|
|
|
|
{
|
|
|
|
/* packet type must be set first */
|
|
|
|
sx126x_set_pkt_type(dev, SX126X_PKT_TYPE_LORA);
|
|
|
|
sx126x_set_channel(dev, CONFIG_SX126X_CHANNEL_DEFAULT);
|
|
|
|
|
|
|
|
/* Configure PA optimal settings for maximum output power
|
|
|
|
* Values used here comes from the datasheet, section 13.1.14 SetPaConfig
|
|
|
|
* and are optimal for a TX output power of 14dBm.
|
|
|
|
*/
|
2021-06-30 13:34:20 +02:00
|
|
|
if (sx126x_is_llcc68(dev) || sx126x_is_sx1262(dev)) {
|
2021-11-04 21:54:40 +01:00
|
|
|
sx126x_set_pa_cfg(dev, &hpa_cfg);
|
2021-03-11 17:28:33 +01:00
|
|
|
}
|
2021-06-30 13:34:20 +02:00
|
|
|
else if (sx126x_is_sx1268(dev)) {
|
2021-06-30 13:26:05 +02:00
|
|
|
sx126x_set_pa_cfg(dev, &sx1268_pa_cfg);
|
2021-03-11 17:28:33 +01:00
|
|
|
}
|
2021-11-04 21:54:40 +01:00
|
|
|
else if (sx126x_is_sx1261(dev)) {
|
|
|
|
sx126x_set_pa_cfg(dev, &lpa_cfg);
|
|
|
|
}
|
|
|
|
#if IS_USED(MODULE_SX126X_RF_SWITCH)
|
|
|
|
if (dev->params->tx_pa_mode == SX126X_RF_MODE_TX_LPA){
|
|
|
|
sx126x_set_pa_cfg(dev, &lpa_cfg);
|
|
|
|
} else {
|
|
|
|
sx126x_set_pa_cfg(dev, &hpa_cfg);
|
2021-03-11 17:28:33 +01:00
|
|
|
}
|
2021-11-04 21:54:40 +01:00
|
|
|
#endif
|
2021-03-11 17:28:33 +01:00
|
|
|
sx126x_set_tx_params(dev, CONFIG_SX126X_TX_POWER_DEFAULT, CONFIG_SX126X_RAMP_TIME_DEFAULT);
|
|
|
|
|
|
|
|
dev->mod_params.bw = (sx126x_lora_bw_t)(CONFIG_LORA_BW_DEFAULT + SX126X_LORA_BW_125);
|
|
|
|
dev->mod_params.sf = (sx126x_lora_sf_t)CONFIG_LORA_SF_DEFAULT;
|
|
|
|
dev->mod_params.cr = (sx126x_lora_cr_t)CONFIG_LORA_CR_DEFAULT;
|
|
|
|
dev->mod_params.ldro = _compute_ldro(dev);
|
|
|
|
sx126x_set_lora_mod_params(dev, &dev->mod_params);
|
|
|
|
|
|
|
|
dev->pkt_params.pld_len_in_bytes = 0;
|
|
|
|
dev->pkt_params.crc_is_on = LORA_PAYLOAD_CRC_ON_DEFAULT;
|
|
|
|
dev->pkt_params.header_type = (
|
|
|
|
IS_ACTIVE(CONFIG_LORA_FIXED_HEADER_LEN_MODE_DEFAULT) ? true : false
|
|
|
|
);
|
|
|
|
dev->pkt_params.preamble_len_in_symb = CONFIG_LORA_PREAMBLE_LENGTH_DEFAULT;
|
|
|
|
dev->pkt_params.invert_iq_is_on = (
|
|
|
|
IS_ACTIVE(CONFIG_LORA_IQ_INVERTED_DEFAULT) ? true : false
|
|
|
|
);
|
|
|
|
sx126x_set_lora_pkt_params(dev, &dev->pkt_params);
|
|
|
|
}
|
|
|
|
|
2021-07-07 13:25:20 +02:00
|
|
|
#if IS_ACTIVE(SX126X_SPI)
|
2021-03-11 17:28:33 +01:00
|
|
|
static void _dio1_isr(void *arg)
|
|
|
|
{
|
2021-06-22 11:00:29 +02:00
|
|
|
netdev_trigger_event_isr(arg);
|
2021-03-11 17:28:33 +01:00
|
|
|
}
|
2021-07-07 13:25:20 +02:00
|
|
|
#endif
|
2021-03-11 17:28:33 +01:00
|
|
|
|
|
|
|
int sx126x_init(sx126x_t *dev)
|
|
|
|
{
|
|
|
|
/* Setup SPI for SX126X */
|
|
|
|
int res = spi_init_cs(dev->params->spi, dev->params->nss_pin);
|
|
|
|
|
|
|
|
if (res != SPI_OK) {
|
|
|
|
DEBUG("[sx126x] error: failed to initialize SPI_%i device (code %i)\n",
|
|
|
|
dev->params->spi, res);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG("[sx126x] init: SPI_%i initialized with success\n", dev->params->spi);
|
|
|
|
|
2021-07-07 13:25:20 +02:00
|
|
|
#if IS_ACTIVE(SX126X_SPI)
|
2021-03-11 17:28:33 +01:00
|
|
|
gpio_init(dev->params->reset_pin, GPIO_OUT);
|
|
|
|
gpio_init(dev->params->busy_pin, GPIO_IN_PD);
|
|
|
|
|
|
|
|
/* Initialize DIOs */
|
|
|
|
if (gpio_is_valid(dev->params->dio1_pin)) {
|
|
|
|
res = gpio_init_int(dev->params->dio1_pin, GPIO_IN, GPIO_RISING, _dio1_isr, dev);
|
|
|
|
if (res < 0) {
|
|
|
|
DEBUG("[sx126x] error: failed to initialize DIO1 pin\n");
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
DEBUG("[sx126x] error: no DIO1 pin defined\n");
|
|
|
|
return -EIO;
|
|
|
|
}
|
2021-07-07 13:25:20 +02:00
|
|
|
#endif
|
2021-03-11 17:28:33 +01:00
|
|
|
|
|
|
|
/* Reset the device */
|
|
|
|
sx126x_reset(dev);
|
|
|
|
|
|
|
|
/* Configure the power regulator mode */
|
|
|
|
sx126x_set_reg_mode(dev, dev->params->regulator);
|
|
|
|
|
|
|
|
/* Initialize radio with the default parameters */
|
|
|
|
sx126x_init_default_config(dev);
|
|
|
|
|
|
|
|
/* Configure available IRQs */
|
|
|
|
const uint16_t irq_mask = (
|
|
|
|
SX126X_IRQ_TX_DONE |
|
|
|
|
SX126X_IRQ_RX_DONE |
|
|
|
|
SX126X_IRQ_PREAMBLE_DETECTED |
|
2021-07-07 13:30:02 +02:00
|
|
|
SX126X_IRQ_SYNC_WORD_VALID |
|
2021-03-11 17:28:33 +01:00
|
|
|
SX126X_IRQ_HEADER_VALID |
|
|
|
|
SX126X_IRQ_HEADER_ERROR |
|
|
|
|
SX126X_IRQ_CRC_ERROR |
|
|
|
|
SX126X_IRQ_CAD_DONE |
|
|
|
|
SX126X_IRQ_CAD_DETECTED |
|
|
|
|
SX126X_IRQ_TIMEOUT
|
|
|
|
);
|
|
|
|
|
|
|
|
sx126x_set_dio_irq_params(dev, irq_mask, irq_mask, 0, 0);
|
|
|
|
|
|
|
|
if (IS_ACTIVE(ENABLE_DEBUG)) {
|
|
|
|
sx126x_pkt_type_t pkt_type;
|
|
|
|
sx126x_get_pkt_type(dev, &pkt_type);
|
|
|
|
DEBUG("[sx126x] init radio: pkt type: %d\n", pkt_type);
|
|
|
|
|
|
|
|
sx126x_chip_status_t radio_status;
|
|
|
|
sx126x_get_status(dev, &radio_status);
|
|
|
|
DEBUG("[sx126x] init: chip mode %d\n", radio_status.chip_mode);
|
|
|
|
DEBUG("[sx126x] init: cmd status %d\n", radio_status.cmd_status);
|
|
|
|
}
|
|
|
|
|
2021-06-25 17:57:23 +02:00
|
|
|
/* Radio Rx timeout timer stopped on preamble detection */
|
|
|
|
sx126x_stop_timer_on_preamble(dev, true);
|
|
|
|
|
2021-03-11 17:28:33 +01:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t sx126x_get_channel(const sx126x_t *dev)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_get_radio_status \n");
|
2021-03-11 17:28:33 +01:00
|
|
|
return dev->channel;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx126x_set_channel(sx126x_t *dev, uint32_t freq)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_set_channel %" PRIu32 "Hz \n", freq);
|
2021-03-11 17:28:33 +01:00
|
|
|
dev->channel = freq;
|
|
|
|
sx126x_set_rf_freq(dev, dev->channel);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t sx126x_get_bandwidth(const sx126x_t *dev)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_get_bandwidth \n");
|
2021-03-11 17:28:33 +01:00
|
|
|
return dev->mod_params.bw - SX126X_LORA_BW_125;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx126x_set_bandwidth(sx126x_t *dev, uint8_t bandwidth)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_set_bandwidth %02x\n", bandwidth);
|
2021-03-11 17:28:33 +01:00
|
|
|
dev->mod_params.bw = bandwidth + SX126X_LORA_BW_125;
|
|
|
|
dev->mod_params.ldro = _compute_ldro(dev);
|
|
|
|
sx126x_set_lora_mod_params(dev, &dev->mod_params);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t sx126x_get_spreading_factor(const sx126x_t *dev)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_get_spreading_factor \n");
|
2021-03-11 17:28:33 +01:00
|
|
|
return dev->mod_params.sf;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx126x_set_spreading_factor(sx126x_t *dev, uint8_t sf)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_set_spreading_factor : %02x\n", sf);
|
2021-03-11 17:28:33 +01:00
|
|
|
dev->mod_params.sf = (sx126x_lora_sf_t)sf;
|
|
|
|
dev->mod_params.ldro = _compute_ldro(dev);
|
|
|
|
sx126x_set_lora_mod_params(dev, &dev->mod_params);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t sx126x_get_coding_rate(const sx126x_t *dev)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_get_coding_rate \n");
|
2021-03-11 17:28:33 +01:00
|
|
|
return dev->mod_params.cr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx126x_set_coding_rate(sx126x_t *dev, uint8_t cr)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_set_coding_rate %01x\n", cr);
|
2021-03-11 17:28:33 +01:00
|
|
|
dev->mod_params.cr = (sx126x_lora_cr_t)cr;
|
|
|
|
sx126x_set_lora_mod_params(dev, &dev->mod_params);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t sx126x_get_lora_payload_length(const sx126x_t *dev)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_get_lora_payload_length \n");
|
2021-03-11 17:28:33 +01:00
|
|
|
sx126x_rx_buffer_status_t rx_buffer_status;
|
|
|
|
|
|
|
|
sx126x_get_rx_buffer_status(dev, &rx_buffer_status);
|
|
|
|
return rx_buffer_status.pld_len_in_bytes;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx126x_set_lora_payload_length(sx126x_t *dev, uint8_t len)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_set_lora_payload_length %d\n", len);
|
2021-03-11 17:28:33 +01:00
|
|
|
dev->pkt_params.pld_len_in_bytes = len;
|
|
|
|
sx126x_set_lora_pkt_params(dev, &dev->pkt_params);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool sx126x_get_lora_crc(const sx126x_t *dev)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_get_lora_crc \n");
|
2021-03-11 17:28:33 +01:00
|
|
|
return dev->pkt_params.crc_is_on;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx126x_set_lora_crc(sx126x_t *dev, bool crc)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_set_lora_crc %d\n", crc);
|
2021-03-11 17:28:33 +01:00
|
|
|
dev->pkt_params.crc_is_on = crc;
|
|
|
|
sx126x_set_lora_pkt_params(dev, &dev->pkt_params);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool sx126x_get_lora_implicit_header(const sx126x_t *dev)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_get_lora_implicit_header \n");
|
2021-03-11 17:28:33 +01:00
|
|
|
return dev->pkt_params.header_type == SX126X_LORA_PKT_IMPLICIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx126x_set_lora_implicit_header(sx126x_t *dev, bool mode)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_set_lora_implicit_header %d\n", mode);
|
2021-03-11 17:28:33 +01:00
|
|
|
dev->pkt_params.header_type = (mode ? SX126X_LORA_PKT_IMPLICIT : SX126X_LORA_PKT_EXPLICIT);
|
|
|
|
sx126x_set_lora_pkt_params(dev, &dev->pkt_params);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t sx126x_get_lora_preamble_length(const sx126x_t *dev)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_get_lora_preamble_length \n");
|
2021-03-11 17:28:33 +01:00
|
|
|
return dev->pkt_params.preamble_len_in_symb;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx126x_set_lora_preamble_length(sx126x_t *dev, uint16_t preamble)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_set_lora_preamble_length %" PRIu16 "\n", preamble);
|
2021-03-11 17:28:33 +01:00
|
|
|
dev->pkt_params.preamble_len_in_symb = preamble;
|
|
|
|
sx126x_set_lora_pkt_params(dev, &dev->pkt_params);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool sx126x_get_lora_iq_invert(const sx126x_t *dev)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_get_lora_iq_invert \n");
|
2021-03-11 17:28:33 +01:00
|
|
|
return dev->pkt_params.invert_iq_is_on;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx126x_set_lora_iq_invert(sx126x_t *dev, bool iq_invert)
|
|
|
|
{
|
2021-06-10 17:16:19 +02:00
|
|
|
DEBUG("[sx126x]: sx126x_set_lora_iq_invert %d\n", iq_invert);
|
2021-03-11 17:28:33 +01:00
|
|
|
dev->pkt_params.invert_iq_is_on = iq_invert;
|
|
|
|
sx126x_set_lora_pkt_params(dev, &dev->pkt_params);
|
|
|
|
}
|