2017-06-30 19:00:27 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2016 Unwired Devices <info@unwds.com>
|
|
|
|
* 2017 Inria Chile
|
|
|
|
* 2017 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_sx127x
|
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @brief implementation of internal functions for sx127x
|
|
|
|
*
|
|
|
|
* @author Eugene P. <ep@unwds.com>
|
|
|
|
* @author José Ignacio Alamos <jose.alamos@inria.cl>
|
|
|
|
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
#include <math.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
|
2020-07-17 15:35:29 +02:00
|
|
|
#include "ztimer.h"
|
|
|
|
|
2017-10-08 12:26:18 +02:00
|
|
|
#include "net/lora.h"
|
|
|
|
|
2017-06-30 19:00:27 +02:00
|
|
|
#include "sx127x.h"
|
|
|
|
#include "sx127x_registers.h"
|
|
|
|
#include "sx127x_internal.h"
|
|
|
|
#include "sx127x_params.h"
|
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2017-06-30 19:00:27 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
|
2017-07-17 11:57:22 +02:00
|
|
|
#define SX127X_SPI_SPEED (SPI_CLK_1MHZ)
|
|
|
|
#define SX127X_SPI_MODE (SPI_MODE_0)
|
|
|
|
|
|
|
|
|
2018-04-12 09:59:51 +02:00
|
|
|
int sx127x_check_version(const sx127x_t *dev)
|
2017-06-30 19:00:27 +02:00
|
|
|
{
|
|
|
|
/* Read version number and compare with sx127x assigned revision */
|
|
|
|
uint8_t version = sx127x_reg_read(dev, SX127X_REG_VERSION);
|
|
|
|
|
|
|
|
#if defined(MODULE_SX1272)
|
|
|
|
if (version != VERSION_SX1272) {
|
2018-04-11 17:52:03 +02:00
|
|
|
DEBUG("[sx127x] sx1272 test failed, invalid version number: %d\n",
|
2017-06-30 19:00:27 +02:00
|
|
|
version);
|
2018-04-12 09:59:51 +02:00
|
|
|
return -1;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|
2018-04-12 09:59:51 +02:00
|
|
|
DEBUG("[sx127x] SX1272 transceiver detected\n");
|
2017-06-30 19:00:27 +02:00
|
|
|
#else /* MODULE_SX1276) */
|
|
|
|
if (version != VERSION_SX1276) {
|
2018-04-11 17:52:03 +02:00
|
|
|
DEBUG("[sx127x] sx1276 test failed, invalid version number: %d\n",
|
2017-06-30 19:00:27 +02:00
|
|
|
version);
|
2018-04-12 09:59:51 +02:00
|
|
|
return -1;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|
2018-04-12 09:59:51 +02:00
|
|
|
DEBUG("[sx127x] SX1276 transceiver detected\n");
|
2017-06-30 19:00:27 +02:00
|
|
|
#endif
|
|
|
|
|
2018-04-12 09:59:51 +02:00
|
|
|
return 0;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void sx127x_reg_write(const sx127x_t *dev, uint8_t addr, uint8_t data)
|
|
|
|
{
|
|
|
|
sx127x_reg_write_burst(dev, addr, &data, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t sx127x_reg_read(const sx127x_t *dev, uint8_t addr)
|
|
|
|
{
|
|
|
|
uint8_t data;
|
|
|
|
|
|
|
|
sx127x_reg_read_burst(dev, addr, &data, 1);
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx127x_reg_write_burst(const sx127x_t *dev, uint8_t addr, uint8_t *buffer,
|
|
|
|
uint8_t size)
|
|
|
|
{
|
2017-07-17 11:57:22 +02:00
|
|
|
spi_acquire(dev->params.spi, SPI_CS_UNDEF, SX127X_SPI_MODE, SX127X_SPI_SPEED);
|
2017-06-30 19:00:27 +02:00
|
|
|
|
|
|
|
gpio_clear(dev->params.nss_pin);
|
|
|
|
spi_transfer_regs(dev->params.spi, SPI_CS_UNDEF, addr | 0x80, (char *) buffer, NULL, size);
|
|
|
|
gpio_set(dev->params.nss_pin);
|
|
|
|
|
|
|
|
spi_release(dev->params.spi);
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx127x_reg_read_burst(const sx127x_t *dev, uint8_t addr, uint8_t *buffer,
|
|
|
|
uint8_t size)
|
|
|
|
{
|
2017-07-17 11:57:22 +02:00
|
|
|
spi_acquire(dev->params.spi, SPI_CS_UNDEF, SX127X_SPI_MODE, SX127X_SPI_SPEED);
|
2017-06-30 19:00:27 +02:00
|
|
|
|
|
|
|
gpio_clear(dev->params.nss_pin);
|
|
|
|
spi_transfer_regs(dev->params.spi, SPI_CS_UNDEF, addr & 0x7F, NULL, (char *) buffer, size);
|
|
|
|
gpio_set(dev->params.nss_pin);
|
|
|
|
|
|
|
|
spi_release(dev->params.spi);
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx127x_write_fifo(const sx127x_t *dev, uint8_t *buffer, uint8_t size)
|
|
|
|
{
|
|
|
|
sx127x_reg_write_burst(dev, 0, buffer, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx127x_read_fifo(const sx127x_t *dev, uint8_t *buffer, uint8_t size)
|
|
|
|
{
|
|
|
|
sx127x_reg_read_burst(dev, 0, buffer, size);
|
|
|
|
}
|
|
|
|
|
2018-04-11 17:51:11 +02:00
|
|
|
#if defined(MODULE_SX1276)
|
|
|
|
void sx1276_rx_chain_calibration(sx127x_t *dev)
|
2017-06-30 19:00:27 +02:00
|
|
|
{
|
|
|
|
uint8_t reg_pa_config_init_val;
|
|
|
|
uint32_t initial_freq;
|
|
|
|
|
|
|
|
/* Save context */
|
|
|
|
reg_pa_config_init_val = sx127x_reg_read(dev, SX127X_REG_PACONFIG);
|
|
|
|
initial_freq = (double) (((uint32_t) sx127x_reg_read(dev, SX127X_REG_FRFMSB) << 16)
|
|
|
|
| ((uint32_t) sx127x_reg_read(dev, SX127X_REG_FRFMID) << 8)
|
2017-10-08 12:26:18 +02:00
|
|
|
| ((uint32_t) sx127x_reg_read(dev, SX127X_REG_FRFLSB))) * (double)LORA_FREQUENCY_RESOLUTION_DEFAULT;
|
2017-06-30 19:00:27 +02:00
|
|
|
|
|
|
|
/* Cut the PA just in case, RFO output, power = -1 dBm */
|
|
|
|
sx127x_reg_write(dev, SX127X_REG_PACONFIG, 0x00);
|
|
|
|
|
|
|
|
/* Launch Rx chain calibration for LF band */
|
|
|
|
sx127x_reg_write(dev,
|
|
|
|
SX127X_REG_IMAGECAL,
|
|
|
|
(sx127x_reg_read(dev, SX127X_REG_IMAGECAL) & SX127X_RF_IMAGECAL_IMAGECAL_MASK)
|
|
|
|
| SX127X_RF_IMAGECAL_IMAGECAL_START);
|
|
|
|
|
|
|
|
while ((sx127x_reg_read(dev, SX127X_REG_IMAGECAL) & SX127X_RF_IMAGECAL_IMAGECAL_RUNNING)
|
|
|
|
== SX127X_RF_IMAGECAL_IMAGECAL_RUNNING) {
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set a frequency in HF band */
|
|
|
|
sx127x_set_channel(dev, SX127X_HF_CHANNEL_DEFAULT);
|
|
|
|
|
|
|
|
/* Launch Rx chain calibration for HF band */
|
|
|
|
sx127x_reg_write(dev,
|
|
|
|
SX127X_REG_IMAGECAL,
|
|
|
|
(sx127x_reg_read(dev, SX127X_REG_IMAGECAL) & SX127X_RF_IMAGECAL_IMAGECAL_MASK)
|
|
|
|
| SX127X_RF_IMAGECAL_IMAGECAL_START);
|
|
|
|
while ((sx127x_reg_read(dev, SX127X_REG_IMAGECAL) & SX127X_RF_IMAGECAL_IMAGECAL_RUNNING)
|
|
|
|
== SX127X_RF_IMAGECAL_IMAGECAL_RUNNING) {
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Restore context */
|
|
|
|
sx127x_reg_write(dev, SX127X_REG_PACONFIG, reg_pa_config_init_val);
|
|
|
|
sx127x_set_channel(dev, initial_freq);
|
|
|
|
}
|
2018-04-11 17:51:11 +02:00
|
|
|
#endif
|
2017-06-30 19:00:27 +02:00
|
|
|
|
|
|
|
int16_t sx127x_read_rssi(const sx127x_t *dev)
|
|
|
|
{
|
|
|
|
int16_t rssi = 0;
|
|
|
|
|
|
|
|
switch (dev->settings.modem) {
|
|
|
|
case SX127X_MODEM_FSK:
|
|
|
|
rssi = -(sx127x_reg_read(dev, SX127X_REG_RSSIVALUE) >> 1);
|
|
|
|
break;
|
|
|
|
case SX127X_MODEM_LORA:
|
|
|
|
#if defined(MODULE_SX1272)
|
|
|
|
rssi = SX127X_RSSI_OFFSET + sx127x_reg_read(dev, SX127X_REG_LR_RSSIVALUE);
|
|
|
|
#else /* MODULE_SX1276 */
|
|
|
|
if (dev->settings.channel > SX127X_RF_MID_BAND_THRESH) {
|
|
|
|
rssi = SX127X_RSSI_OFFSET_HF + sx127x_reg_read(dev, SX127X_REG_LR_RSSIVALUE);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
rssi = SX127X_RSSI_OFFSET_LF + sx127x_reg_read(dev, SX127X_REG_LR_RSSIVALUE);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
rssi = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rssi;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx127x_start_cad(sx127x_t *dev)
|
|
|
|
{
|
|
|
|
switch (dev->settings.modem) {
|
|
|
|
case SX127X_MODEM_FSK:
|
|
|
|
break;
|
|
|
|
case SX127X_MODEM_LORA:
|
|
|
|
/* Disable all interrupts except CAD-related */
|
|
|
|
sx127x_reg_write(dev, SX127X_REG_LR_IRQFLAGSMASK,
|
|
|
|
SX127X_RF_LORA_IRQFLAGS_RXTIMEOUT |
|
|
|
|
SX127X_RF_LORA_IRQFLAGS_RXDONE |
|
|
|
|
SX127X_RF_LORA_IRQFLAGS_PAYLOADCRCERROR |
|
|
|
|
SX127X_RF_LORA_IRQFLAGS_VALIDHEADER |
|
|
|
|
SX127X_RF_LORA_IRQFLAGS_TXDONE |
|
|
|
|
/*SX127X_RF_LORA_IRQFLAGS_CADDONE |*/
|
|
|
|
SX127X_RF_LORA_IRQFLAGS_FHSSCHANGEDCHANNEL
|
|
|
|
/* | SX127X_RF_LORA_IRQFLAGS_CADDETECTED*/
|
|
|
|
);
|
|
|
|
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(dev->params.dio3_pin)) {
|
2020-03-10 10:04:09 +01:00
|
|
|
/* DIO3 = CADDone */
|
|
|
|
sx127x_reg_write(dev, SX127X_REG_DIOMAPPING1,
|
|
|
|
(sx127x_reg_read(dev, SX127X_REG_DIOMAPPING1) &
|
|
|
|
SX127X_RF_LORA_DIOMAPPING1_DIO3_MASK) |
|
|
|
|
SX127X_RF_LORA_DIOMAPPING1_DIO3_00);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* DIO0 = CADDone */
|
|
|
|
sx127x_reg_write(dev, SX127X_REG_DIOMAPPING1,
|
|
|
|
(sx127x_reg_read(dev, SX127X_REG_DIOMAPPING1) &
|
|
|
|
SX127X_RF_LORA_DIOMAPPING1_DIO0_MASK) |
|
|
|
|
SX127X_RF_LORA_DIOMAPPING1_DIO0_10);
|
|
|
|
}
|
2017-06-30 19:00:27 +02:00
|
|
|
|
|
|
|
sx127x_set_state(dev, SX127X_RF_CAD);
|
|
|
|
sx127x_set_op_mode(dev, SX127X_RF_LORA_OPMODE_CAD);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-11-20 17:28:07 +01:00
|
|
|
|
|
|
|
bool sx127x_is_channel_free(sx127x_t *dev, uint32_t freq, int16_t rssi_threshold)
|
|
|
|
{
|
|
|
|
int16_t rssi = 0;
|
|
|
|
|
|
|
|
sx127x_set_channel(dev, freq);
|
|
|
|
sx127x_set_op_mode(dev, SX127X_RF_OPMODE_RECEIVER);
|
|
|
|
|
2020-07-17 15:35:29 +02:00
|
|
|
ztimer_sleep(ZTIMER_MSEC, 1); /* wait 1 millisecond */
|
2017-11-20 17:28:07 +01:00
|
|
|
|
|
|
|
rssi = sx127x_read_rssi(dev);
|
|
|
|
sx127x_set_sleep(dev);
|
|
|
|
|
|
|
|
return (rssi <= rssi_threshold);
|
|
|
|
}
|