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 Basic functionality of sx127x driver
|
|
|
|
*
|
|
|
|
* @author Eugene P. <ep@unwds.com>
|
|
|
|
* @author José Ignacio Alamos <jose.alamos@inria.cl>
|
|
|
|
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "xtimer.h"
|
|
|
|
#include "thread.h"
|
|
|
|
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
#include "periph/spi.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_internal.h"
|
|
|
|
#include "sx127x_registers.h"
|
|
|
|
#include "sx127x_netdev.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/* Internal functions */
|
2018-04-12 09:59:51 +02:00
|
|
|
static int _init_spi(sx127x_t *dev);
|
|
|
|
static int _init_gpios(sx127x_t *dev);
|
2017-06-30 19:00:27 +02:00
|
|
|
static void _init_timers(sx127x_t *dev);
|
|
|
|
static void _on_tx_timeout(void *arg);
|
|
|
|
static void _on_rx_timeout(void *arg);
|
|
|
|
|
|
|
|
/* SX127X DIO interrupt handlers initialization */
|
|
|
|
static void sx127x_on_dio0_isr(void *arg);
|
|
|
|
static void sx127x_on_dio1_isr(void *arg);
|
|
|
|
static void sx127x_on_dio2_isr(void *arg);
|
|
|
|
static void sx127x_on_dio3_isr(void *arg);
|
|
|
|
|
|
|
|
|
|
|
|
void sx127x_setup(sx127x_t *dev, const sx127x_params_t *params)
|
|
|
|
{
|
|
|
|
netdev_t *netdev = (netdev_t*) dev;
|
|
|
|
netdev->driver = &sx127x_driver;
|
|
|
|
memcpy(&dev->params, params, sizeof(sx127x_params_t));
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx127x_reset(const sx127x_t *dev)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* This reset scheme complies with 7.2 chapter of the SX1272/1276 datasheet
|
|
|
|
* See http://www.semtech.com/images/datasheet/sx1276.pdf for SX1276
|
|
|
|
* See http://www.semtech.com/images/datasheet/sx1272.pdf for SX1272
|
|
|
|
*
|
|
|
|
* 1. Set NReset pin to LOW for at least 100 us
|
|
|
|
* 2. Set NReset in Hi-Z state
|
|
|
|
* 3. Wait at least 5 milliseconds
|
|
|
|
*/
|
|
|
|
gpio_init(dev->params.reset_pin, GPIO_OUT);
|
|
|
|
|
|
|
|
/* Set reset pin to 0 */
|
|
|
|
gpio_clear(dev->params.reset_pin);
|
|
|
|
|
|
|
|
/* Wait 1 ms */
|
|
|
|
xtimer_usleep(1000);
|
|
|
|
|
|
|
|
/* Put reset pin in High-Z */
|
|
|
|
gpio_init(dev->params.reset_pin, GPIO_IN);
|
|
|
|
|
|
|
|
/* Wait 10 ms */
|
|
|
|
xtimer_usleep(1000 * 10);
|
|
|
|
}
|
|
|
|
|
|
|
|
int sx127x_init(sx127x_t *dev)
|
|
|
|
{
|
|
|
|
/* Do internal initialization routines */
|
2018-04-12 09:59:51 +02:00
|
|
|
if (_init_spi(dev) < 0) {
|
|
|
|
DEBUG("[sx127x] error: failed to initialize SPI\n");
|
2017-06-30 19:00:27 +02:00
|
|
|
return -SX127X_ERR_SPI;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check presence of SX127X */
|
2018-04-12 09:59:51 +02:00
|
|
|
if (sx127x_check_version(dev) < 0) {
|
|
|
|
DEBUG("[sx127x] error: no valid device found\n");
|
|
|
|
return -SX127X_ERR_NODEV;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
_init_timers(dev);
|
|
|
|
xtimer_usleep(1000); /* wait 1 millisecond */
|
|
|
|
|
|
|
|
sx127x_reset(dev);
|
|
|
|
|
2018-04-11 17:51:11 +02:00
|
|
|
#if defined(MODULE_SX1276)
|
|
|
|
sx1276_rx_chain_calibration(dev);
|
|
|
|
#endif
|
2017-06-30 19:00:27 +02:00
|
|
|
sx127x_set_op_mode(dev, SX127X_RF_OPMODE_SLEEP);
|
|
|
|
|
2018-04-12 09:59:51 +02:00
|
|
|
if (_init_gpios(dev) < 0) {
|
|
|
|
DEBUG("[sx127x] error: failed to initialize GPIOs\n");
|
|
|
|
return -SX127X_ERR_GPIOS;
|
|
|
|
}
|
2017-06-30 19:00:27 +02:00
|
|
|
|
|
|
|
return SX127X_INIT_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sx127x_init_radio_settings(sx127x_t *dev)
|
|
|
|
{
|
2018-04-12 09:59:51 +02:00
|
|
|
DEBUG("[sx127x] initializing radio settings\n");
|
2018-03-30 16:55:28 +02:00
|
|
|
sx127x_set_channel(dev, SX127X_CHANNEL_DEFAULT);
|
2018-04-11 17:52:57 +02:00
|
|
|
sx127x_set_modem(dev, SX127X_MODEM_DEFAULT);
|
|
|
|
sx127x_set_tx_power(dev, SX127X_RADIO_TX_POWER);
|
2017-10-08 12:26:18 +02:00
|
|
|
sx127x_set_bandwidth(dev, LORA_BW_DEFAULT);
|
|
|
|
sx127x_set_spreading_factor(dev, LORA_SF_DEFAULT);
|
|
|
|
sx127x_set_coding_rate(dev, LORA_CR_DEFAULT);
|
|
|
|
sx127x_set_crc(dev, LORA_PAYLOAD_CRC_ON_DEFAULT);
|
2018-03-30 16:55:28 +02:00
|
|
|
sx127x_set_freq_hop(dev, LORA_FREQUENCY_HOPPING_DEFAULT);
|
2017-10-08 12:26:18 +02:00
|
|
|
sx127x_set_hop_period(dev, LORA_FREQUENCY_HOPPING_PERIOD_DEFAULT);
|
2018-03-30 16:55:28 +02:00
|
|
|
sx127x_set_fixed_header_len_mode(dev, LORA_FIXED_HEADER_LEN_MODE_DEFAULT);
|
|
|
|
sx127x_set_iq_invert(dev, LORA_IQ_INVERTED_DEFAULT);
|
|
|
|
sx127x_set_payload_length(dev, LORA_PAYLOAD_LENGTH_DEFAULT);
|
|
|
|
sx127x_set_preamble_length(dev, LORA_PREAMBLE_LENGTH_DEFAULT);
|
|
|
|
sx127x_set_symbol_timeout(dev, LORA_SYMBOL_TIMEOUT_DEFAULT);
|
2017-06-30 19:00:27 +02:00
|
|
|
sx127x_set_rx_single(dev, SX127X_RX_SINGLE);
|
|
|
|
sx127x_set_tx_timeout(dev, SX127X_TX_TIMEOUT_DEFAULT);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t sx127x_random(sx127x_t *dev)
|
|
|
|
{
|
|
|
|
uint32_t rnd = 0;
|
|
|
|
|
|
|
|
sx127x_set_modem(dev, SX127X_MODEM_LORA); /* Set LoRa modem ON */
|
|
|
|
|
|
|
|
/* Disable LoRa modem interrupts */
|
|
|
|
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);
|
|
|
|
|
|
|
|
/* Set radio in continuous reception */
|
|
|
|
sx127x_set_op_mode(dev, SX127X_RF_OPMODE_RECEIVER);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < 32; i++) {
|
|
|
|
xtimer_usleep(1000); /* wait for the chaos */
|
|
|
|
|
|
|
|
/* Non-filtered RSSI value reading. Only takes the LSB value */
|
|
|
|
rnd |= ((uint32_t) sx127x_reg_read(dev, SX127X_REG_LR_RSSIWIDEBAND) & 0x01) << i;
|
|
|
|
}
|
|
|
|
|
|
|
|
sx127x_set_sleep(dev);
|
|
|
|
|
|
|
|
return rnd;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* IRQ handlers
|
|
|
|
*/
|
|
|
|
void sx127x_isr(netdev_t *dev)
|
|
|
|
{
|
|
|
|
if (dev->event_callback) {
|
|
|
|
dev->event_callback(dev, NETDEV_EVENT_ISR);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sx127x_on_dio_isr(sx127x_t *dev, sx127x_flags_t flag)
|
|
|
|
{
|
|
|
|
dev->irq |= flag;
|
|
|
|
sx127x_isr((netdev_t *)dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sx127x_on_dio0_isr(void *arg)
|
|
|
|
{
|
|
|
|
sx127x_on_dio_isr((sx127x_t*) arg, SX127X_IRQ_DIO0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sx127x_on_dio1_isr(void *arg)
|
|
|
|
{
|
|
|
|
sx127x_on_dio_isr((sx127x_t*) arg, SX127X_IRQ_DIO1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sx127x_on_dio2_isr(void *arg)
|
|
|
|
{
|
|
|
|
sx127x_on_dio_isr((sx127x_t*) arg, SX127X_IRQ_DIO2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sx127x_on_dio3_isr(void *arg)
|
|
|
|
{
|
|
|
|
sx127x_on_dio_isr((sx127x_t*) arg, SX127X_IRQ_DIO3);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Internal event handlers */
|
2018-04-12 09:59:51 +02:00
|
|
|
static int _init_gpios(sx127x_t *dev)
|
2017-06-30 19:00:27 +02:00
|
|
|
{
|
2018-04-12 09:59:51 +02:00
|
|
|
int res = gpio_init_int(dev->params.dio0_pin, GPIO_IN, GPIO_RISING,
|
|
|
|
sx127x_on_dio0_isr, dev);
|
|
|
|
if (res < 0) {
|
|
|
|
DEBUG("[sx127x] error: failed to initialize DIO0 pin\n");
|
|
|
|
return res;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|
|
|
|
|
2018-04-12 09:59:51 +02:00
|
|
|
res = gpio_init_int(dev->params.dio1_pin, GPIO_IN, GPIO_RISING,
|
|
|
|
sx127x_on_dio1_isr, dev);
|
|
|
|
if (res < 0) {
|
|
|
|
DEBUG("[sx127x] error: failed to initialize DIO1 pin\n");
|
|
|
|
return res;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|
|
|
|
|
2018-04-12 09:59:51 +02:00
|
|
|
res = gpio_init_int(dev->params.dio2_pin, GPIO_IN, GPIO_RISING,
|
|
|
|
sx127x_on_dio2_isr, dev);
|
|
|
|
if (res < 0) {
|
|
|
|
DEBUG("[sx127x] error: failed to initialize DIO2 pin\n");
|
|
|
|
return res;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|
|
|
|
|
2018-04-12 09:59:51 +02:00
|
|
|
res = gpio_init_int(dev->params.dio3_pin, GPIO_IN, GPIO_RISING,
|
|
|
|
sx127x_on_dio3_isr, dev);
|
|
|
|
if (res < 0) {
|
|
|
|
DEBUG("[sx127x] error: failed to initialize DIO3 pin\n");
|
|
|
|
return res;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|
2018-04-12 09:59:51 +02:00
|
|
|
|
|
|
|
return res;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void _on_tx_timeout(void *arg)
|
|
|
|
{
|
|
|
|
netdev_t *dev = (netdev_t *) arg;
|
|
|
|
|
|
|
|
dev->event_callback(dev, NETDEV_EVENT_TX_TIMEOUT);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _on_rx_timeout(void *arg)
|
|
|
|
{
|
|
|
|
netdev_t *dev = (netdev_t *) arg;
|
|
|
|
|
|
|
|
dev->event_callback(dev, NETDEV_EVENT_RX_TIMEOUT);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _init_timers(sx127x_t *dev)
|
|
|
|
{
|
|
|
|
dev->_internal.tx_timeout_timer.arg = dev;
|
|
|
|
dev->_internal.tx_timeout_timer.callback = _on_tx_timeout;
|
|
|
|
|
|
|
|
dev->_internal.rx_timeout_timer.arg = dev;
|
|
|
|
dev->_internal.rx_timeout_timer.callback = _on_rx_timeout;
|
|
|
|
}
|
|
|
|
|
2018-04-12 09:59:51 +02:00
|
|
|
static int _init_spi(sx127x_t *dev)
|
2017-06-30 19:00:27 +02:00
|
|
|
{
|
|
|
|
int res;
|
|
|
|
|
|
|
|
/* Setup SPI for SX127X */
|
|
|
|
res = spi_init_cs(dev->params.spi, dev->params.nss_pin);
|
|
|
|
|
|
|
|
if (res != SPI_OK) {
|
2018-04-12 09:59:51 +02:00
|
|
|
DEBUG("[sx127x] error: failed to initialize SPI_%i device (code %i)\n",
|
|
|
|
dev->params.spi, res);
|
|
|
|
return -1;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|
|
|
|
|
2018-04-12 09:59:51 +02:00
|
|
|
DEBUG("[sx127x] SPI_%i initialized with success\n", dev->params.spi);
|
|
|
|
return 0;
|
2017-06-30 19:00:27 +02:00
|
|
|
}
|