2023-12-26 20:37:54 +01:00
|
|
|
#include <RadioLib.h>
|
2023-11-26 18:30:57 +01:00
|
|
|
#include <WiFi.h>
|
2023-06-06 17:21:59 +02:00
|
|
|
#include "configuration.h"
|
2023-10-08 14:39:44 +02:00
|
|
|
#include "aprs_is_utils.h"
|
2025-03-03 15:19:25 +01:00
|
|
|
#include "station_utils.h"
|
2024-11-05 18:41:41 +01:00
|
|
|
#include "board_pinout.h"
|
2023-06-12 05:21:52 +02:00
|
|
|
#include "syslog_utils.h"
|
2024-10-14 16:44:22 +02:00
|
|
|
#include "ntp_utils.h"
|
2023-06-04 16:10:39 +02:00
|
|
|
#include "display.h"
|
2024-03-17 12:21:11 +01:00
|
|
|
#include "utils.h"
|
2023-06-04 16:10:39 +02:00
|
|
|
|
2024-04-20 15:37:22 +02:00
|
|
|
extern Configuration Config;
|
|
|
|
|
extern uint32_t lastRxTime;
|
|
|
|
|
|
2024-04-13 18:20:10 +02:00
|
|
|
extern std::vector<ReceivedPacket> receivedPackets;
|
|
|
|
|
|
2024-05-17 21:11:59 +02:00
|
|
|
bool operationDone = true;
|
|
|
|
|
bool transmitFlag = true;
|
2024-03-29 09:55:25 +01:00
|
|
|
|
|
|
|
|
#ifdef HAS_SX1262
|
2024-05-11 18:59:07 +02:00
|
|
|
SX1262 radio = new Module(RADIO_CS_PIN, RADIO_DIO1_PIN, RADIO_RST_PIN, RADIO_BUSY_PIN);
|
2023-12-26 20:37:54 +01:00
|
|
|
#endif
|
2024-03-29 09:55:25 +01:00
|
|
|
#ifdef HAS_SX1268
|
2024-10-25 15:42:15 +02:00
|
|
|
#if defined(LIGHTGATEWAY_1_0)
|
|
|
|
|
SPIClass loraSPI(FSPI);
|
|
|
|
|
SX1268 radio = new Module(RADIO_CS_PIN, RADIO_DIO1_PIN, RADIO_RST_PIN, RADIO_BUSY_PIN, loraSPI);
|
|
|
|
|
#else
|
|
|
|
|
SX1268 radio = new Module(RADIO_CS_PIN, RADIO_DIO1_PIN, RADIO_RST_PIN, RADIO_BUSY_PIN);
|
|
|
|
|
#endif
|
2024-03-29 09:55:25 +01:00
|
|
|
#endif
|
|
|
|
|
#ifdef HAS_SX1278
|
2024-05-11 18:59:07 +02:00
|
|
|
SX1278 radio = new Module(RADIO_CS_PIN, RADIO_BUSY_PIN, RADIO_RST_PIN);
|
2024-01-03 02:41:54 +01:00
|
|
|
#endif
|
2024-04-24 04:25:20 +02:00
|
|
|
#ifdef HAS_SX1276
|
2024-05-11 18:59:07 +02:00
|
|
|
SX1276 radio = new Module(RADIO_CS_PIN, RADIO_BUSY_PIN, RADIO_RST_PIN);
|
2024-04-24 04:25:20 +02:00
|
|
|
#endif
|
2024-09-23 17:37:35 +02:00
|
|
|
#if defined(HAS_LLCC68) //LLCC68 supports spreading factor only in range of 5-11!
|
|
|
|
|
LLCC68 radio = new Module(RADIO_CS_PIN, RADIO_DIO1_PIN, RADIO_RST_PIN, RADIO_BUSY_PIN);
|
|
|
|
|
#endif
|
2024-04-24 04:25:20 +02:00
|
|
|
|
2023-06-20 01:44:55 +02:00
|
|
|
int rssi, freqError;
|
|
|
|
|
float snr;
|
|
|
|
|
|
2024-11-06 16:47:42 +01:00
|
|
|
|
2023-06-07 23:25:50 +02:00
|
|
|
namespace LoRa_Utils {
|
2023-06-04 16:10:39 +02:00
|
|
|
|
2024-02-24 14:09:05 +01:00
|
|
|
void setFlag(void) {
|
2024-05-17 05:22:27 +02:00
|
|
|
operationDone = true;
|
2023-12-27 00:47:59 +01:00
|
|
|
}
|
2023-06-04 16:10:39 +02:00
|
|
|
|
2024-02-24 14:09:05 +01:00
|
|
|
void setup() {
|
2024-10-25 15:42:15 +02:00
|
|
|
#ifdef LIGHTGATEWAY_1_0
|
|
|
|
|
pinMode(RADIO_VCC_PIN,OUTPUT);
|
|
|
|
|
digitalWrite(RADIO_VCC_PIN,HIGH);
|
|
|
|
|
loraSPI.begin(RADIO_SCLK_PIN, RADIO_MISO_PIN, RADIO_MOSI_PIN, RADIO_CS_PIN);
|
|
|
|
|
#else
|
|
|
|
|
SPI.begin(RADIO_SCLK_PIN, RADIO_MISO_PIN, RADIO_MOSI_PIN);
|
|
|
|
|
#endif
|
2024-03-17 12:21:11 +01:00
|
|
|
float freq = (float)Config.loramodule.rxFreq / 1000000;
|
2024-07-18 20:41:09 +02:00
|
|
|
#if defined(RADIO_HAS_XTAL)
|
|
|
|
|
radio.XTAL = true;
|
|
|
|
|
#endif
|
2024-02-24 14:09:05 +01:00
|
|
|
int state = radio.begin(freq);
|
|
|
|
|
if (state == RADIOLIB_ERR_NONE) {
|
2024-03-29 09:55:25 +01:00
|
|
|
Utils::println("Initializing LoRa Module");
|
|
|
|
|
} else {
|
2024-07-18 20:41:09 +02:00
|
|
|
Utils::println("Starting LoRa failed! State: " + String(state));
|
2024-03-17 12:21:11 +01:00
|
|
|
while (true);
|
2024-02-24 14:09:05 +01:00
|
|
|
}
|
2024-09-23 17:37:35 +02:00
|
|
|
#if defined(HAS_SX1262) || defined(HAS_SX1268) || defined(HAS_LLCC68)
|
2024-05-11 18:59:07 +02:00
|
|
|
if (!Config.lowPowerMode) {
|
|
|
|
|
radio.setDio1Action(setFlag);
|
|
|
|
|
} else {
|
|
|
|
|
radio.setDIOMapping(1, RADIOLIB_SX126X_IRQ_RX_DONE);
|
|
|
|
|
}
|
2024-03-29 09:55:25 +01:00
|
|
|
#endif
|
2024-05-16 23:52:25 +02:00
|
|
|
#if defined(HAS_SX1278) || defined(HAS_SX1276)
|
|
|
|
|
radio.setDio0Action(setFlag, RISING);
|
|
|
|
|
#endif
|
2024-02-24 14:09:05 +01:00
|
|
|
radio.setSpreadingFactor(Config.loramodule.spreadingFactor);
|
2024-03-20 02:41:07 +01:00
|
|
|
float signalBandwidth = Config.loramodule.signalBandwidth/1000;
|
|
|
|
|
radio.setBandwidth(signalBandwidth);
|
2024-02-24 14:09:05 +01:00
|
|
|
radio.setCodingRate(Config.loramodule.codingRate4);
|
2024-02-25 16:55:38 +01:00
|
|
|
radio.setCRC(true);
|
2024-03-29 09:55:25 +01:00
|
|
|
|
2024-10-25 15:42:15 +02:00
|
|
|
#if (defined(RADIO_RXEN) && defined(RADIO_TXEN)) || defined(LIGHTGATEWAY_1_0) // QRP Labs LightGateway has 400M22S (SX1268)
|
2024-05-11 18:59:07 +02:00
|
|
|
radio.setRfSwitchPins(RADIO_RXEN, RADIO_TXEN);
|
2024-06-10 20:09:30 +02:00
|
|
|
#endif
|
|
|
|
|
|
2024-09-23 17:37:35 +02:00
|
|
|
#ifdef HAS_1W_LORA // Ebyte E22 400M30S (SX1268) / 900M30S (SX1262) / Ebyte E220 400M30S (LLCC68)
|
2024-06-19 01:30:34 +02:00
|
|
|
state = radio.setOutputPower(Config.loramodule.power); // max value 20dB for 1W modules as they have Low Noise Amp
|
2024-05-17 21:11:59 +02:00
|
|
|
radio.setCurrentLimit(140); // to be validated (100 , 120, 140)?
|
2024-03-29 09:55:25 +01:00
|
|
|
#endif
|
2024-05-17 21:11:59 +02:00
|
|
|
#if defined(HAS_SX1278) || defined(HAS_SX1276)
|
2024-05-11 18:59:07 +02:00
|
|
|
state = radio.setOutputPower(Config.loramodule.power); // max value 20dB for 400M30S as it has Low Noise Amp
|
2024-05-17 21:11:59 +02:00
|
|
|
radio.setCurrentLimit(100); // to be validated (80 , 100)?
|
|
|
|
|
#endif
|
2024-06-19 01:30:34 +02:00
|
|
|
#if (defined(HAS_SX1268) || defined(HAS_SX1262)) && !defined(HAS_1W_LORA)
|
2024-05-11 18:59:07 +02:00
|
|
|
state = radio.setOutputPower(Config.loramodule.power + 2); // values available: 10, 17, 22 --> if 20 in tracker_conf.json it will be updated to 22.
|
2024-05-17 21:11:59 +02:00
|
|
|
radio.setCurrentLimit(140);
|
2024-03-29 09:55:25 +01:00
|
|
|
#endif
|
2024-05-16 23:52:25 +02:00
|
|
|
|
2024-09-23 17:37:35 +02:00
|
|
|
#if defined(HAS_SX1262) || defined(HAS_SX1268) || defined(HAS_LLCC68)
|
2024-05-11 18:59:07 +02:00
|
|
|
radio.setRxBoostedGainMode(true);
|
2024-05-11 05:18:27 +02:00
|
|
|
#endif
|
2024-05-16 23:52:25 +02:00
|
|
|
|
2024-02-24 14:09:05 +01:00
|
|
|
if (state == RADIOLIB_ERR_NONE) {
|
2024-03-29 09:55:25 +01:00
|
|
|
Utils::println("init : LoRa Module ... done!");
|
2024-03-24 03:36:18 +01:00
|
|
|
} else {
|
2024-08-06 15:12:01 +02:00
|
|
|
Utils::println("Starting LoRa failed! State: " + String(state));
|
2024-02-24 14:09:05 +01:00
|
|
|
while (true);
|
|
|
|
|
}
|
2023-12-27 00:47:59 +01:00
|
|
|
}
|
2023-06-06 20:37:47 +02:00
|
|
|
|
2024-03-07 17:46:38 +01:00
|
|
|
void changeFreqTx() {
|
|
|
|
|
delay(500);
|
2024-03-17 12:21:11 +01:00
|
|
|
float freq = (float)Config.loramodule.txFreq / 1000000;
|
2024-03-07 17:46:38 +01:00
|
|
|
radio.setFrequency(freq);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void changeFreqRx() {
|
|
|
|
|
delay(500);
|
2024-03-17 12:21:11 +01:00
|
|
|
float freq = (float)Config.loramodule.rxFreq / 1000000;
|
2024-03-07 17:46:38 +01:00
|
|
|
radio.setFrequency(freq);
|
|
|
|
|
}
|
2024-03-17 12:21:11 +01:00
|
|
|
|
2024-04-20 15:27:20 +02:00
|
|
|
void sendNewPacket(const String& newPacket) {
|
2024-05-17 15:01:15 +02:00
|
|
|
if (!Config.loramodule.txActive) return;
|
2024-03-07 17:46:38 +01:00
|
|
|
|
2024-05-17 15:01:15 +02:00
|
|
|
if (Config.loramodule.txFreq != Config.loramodule.rxFreq) {
|
|
|
|
|
changeFreqTx();
|
|
|
|
|
}
|
2024-10-08 06:03:58 +02:00
|
|
|
|
2024-05-17 15:01:15 +02:00
|
|
|
#ifdef INTERNAL_LED_PIN
|
2024-10-08 06:03:58 +02:00
|
|
|
if (!Config.digi.ecoMode) digitalWrite(INTERNAL_LED_PIN, HIGH);
|
2024-05-17 15:01:15 +02:00
|
|
|
#endif
|
|
|
|
|
int state = radio.transmit("\x3c\xff\x01" + newPacket);
|
|
|
|
|
transmitFlag = true;
|
|
|
|
|
if (state == RADIOLIB_ERR_NONE) {
|
|
|
|
|
if (Config.syslog.active && WiFi.status() == WL_CONNECTED) {
|
2024-05-30 21:12:34 +02:00
|
|
|
SYSLOG_Utils::log(3, newPacket, 0, 0.0, 0); // TX
|
2024-05-17 05:22:27 +02:00
|
|
|
}
|
2025-02-25 17:54:36 +01:00
|
|
|
Utils::print("---> LoRa Packet Tx : ");
|
2024-05-17 15:01:15 +02:00
|
|
|
Utils::println(newPacket);
|
|
|
|
|
} else {
|
|
|
|
|
Utils::print(F("failed, code "));
|
|
|
|
|
Utils::println(String(state));
|
|
|
|
|
}
|
|
|
|
|
#ifdef INTERNAL_LED_PIN
|
2024-10-08 06:03:58 +02:00
|
|
|
if (!Config.digi.ecoMode) digitalWrite(INTERNAL_LED_PIN, LOW);
|
2024-05-17 15:01:15 +02:00
|
|
|
#endif
|
|
|
|
|
if (Config.loramodule.txFreq != Config.loramodule.rxFreq) {
|
|
|
|
|
changeFreqRx();
|
|
|
|
|
}
|
2024-01-26 02:35:15 +01:00
|
|
|
}
|
|
|
|
|
|
2024-06-06 05:49:16 +02:00
|
|
|
/*String packetSanitization(const String& packet) {
|
2024-05-15 23:47:29 +02:00
|
|
|
String sanitizedPacket = packet;
|
2024-03-17 12:21:11 +01:00
|
|
|
if (packet.indexOf("\0") > 0) {
|
2024-05-15 23:47:29 +02:00
|
|
|
sanitizedPacket.replace("\0", "");
|
2024-02-24 14:09:05 +01:00
|
|
|
}
|
2024-03-17 12:21:11 +01:00
|
|
|
if (packet.indexOf("\r") > 0) {
|
2024-05-15 23:47:29 +02:00
|
|
|
sanitizedPacket.replace("\r", "");
|
2024-02-24 14:09:05 +01:00
|
|
|
}
|
2024-03-17 12:21:11 +01:00
|
|
|
if (packet.indexOf("\n") > 0) {
|
2024-05-15 23:47:29 +02:00
|
|
|
sanitizedPacket.replace("\n", "");
|
2024-02-24 14:09:05 +01:00
|
|
|
}
|
2024-05-15 23:47:29 +02:00
|
|
|
return sanitizedPacket;
|
2024-06-06 05:49:16 +02:00
|
|
|
}*/
|
2024-02-24 14:09:05 +01:00
|
|
|
|
2024-03-28 18:00:46 +01:00
|
|
|
void startReceive() {
|
|
|
|
|
radio.startReceive();
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-24 14:09:05 +01:00
|
|
|
String receivePacket() {
|
2024-05-16 23:52:25 +02:00
|
|
|
String packet = "";
|
2024-08-13 18:36:37 +02:00
|
|
|
if (operationDone || Config.lowPowerMode) {
|
2024-05-17 05:22:27 +02:00
|
|
|
operationDone = false;
|
2024-08-13 18:36:37 +02:00
|
|
|
if (transmitFlag && !Config.lowPowerMode) {
|
2024-05-17 05:22:27 +02:00
|
|
|
radio.startReceive();
|
|
|
|
|
transmitFlag = false;
|
|
|
|
|
} else {
|
|
|
|
|
int state = radio.readData(packet);
|
|
|
|
|
if (state == RADIOLIB_ERR_NONE) {
|
|
|
|
|
if (packet != "") {
|
2025-03-03 15:19:25 +01:00
|
|
|
|
|
|
|
|
String sender = packet.substring(3, packet.indexOf(">"));
|
2025-03-09 14:23:34 +01:00
|
|
|
if (packet.substring(0,3) == "\x3c\xff\x01" && !STATION_Utils::isBlacklisted(sender)){ // avoid processing BlackListed stations
|
2025-03-03 15:19:25 +01:00
|
|
|
rssi = radio.getRSSI();
|
|
|
|
|
snr = radio.getSNR();
|
|
|
|
|
freqError = radio.getFrequencyError();
|
|
|
|
|
Utils::println("<--- LoRa Packet Rx : " + packet.substring(3));
|
|
|
|
|
Utils::println("(RSSI:" + String(rssi) + " / SNR:" + String(snr) + " / FreqErr:" + String(freqError) + ")");
|
|
|
|
|
|
|
|
|
|
if (!Config.lowPowerMode && !Config.digi.ecoMode) {
|
|
|
|
|
if (receivedPackets.size() >= 10) {
|
|
|
|
|
receivedPackets.erase(receivedPackets.begin());
|
|
|
|
|
}
|
|
|
|
|
ReceivedPacket receivedPacket;
|
|
|
|
|
receivedPacket.rxTime = NTP_Utils::getFormatedTime();
|
|
|
|
|
receivedPacket.packet = packet.substring(3);
|
|
|
|
|
receivedPacket.RSSI = rssi;
|
|
|
|
|
receivedPacket.SNR = snr;
|
|
|
|
|
receivedPackets.push_back(receivedPacket);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (Config.syslog.active && WiFi.status() == WL_CONNECTED) {
|
|
|
|
|
SYSLOG_Utils::log(1, packet, rssi, snr, freqError); // RX
|
2024-10-14 17:49:01 +02:00
|
|
|
}
|
2025-03-03 15:19:25 +01:00
|
|
|
} else {
|
|
|
|
|
packet = "";
|
|
|
|
|
}
|
2024-05-17 05:22:27 +02:00
|
|
|
lastRxTime = millis();
|
|
|
|
|
return packet;
|
2024-11-06 16:47:42 +01:00
|
|
|
}
|
2024-05-17 05:22:27 +02:00
|
|
|
} else if (state == RADIOLIB_ERR_CRC_MISMATCH) {
|
2024-05-30 22:27:07 +02:00
|
|
|
rssi = radio.getRSSI();
|
|
|
|
|
snr = radio.getSNR();
|
|
|
|
|
freqError = radio.getFrequencyError();
|
2024-05-17 05:22:27 +02:00
|
|
|
Utils::println(F("CRC error!"));
|
2024-04-09 18:51:51 +02:00
|
|
|
if (Config.syslog.active && WiFi.status() == WL_CONNECTED) {
|
2024-05-17 05:22:27 +02:00
|
|
|
SYSLOG_Utils::log(0, packet, rssi, snr, freqError); // CRC
|
2024-03-23 13:48:12 +01:00
|
|
|
}
|
2024-05-17 05:22:27 +02:00
|
|
|
packet = "";
|
|
|
|
|
} else {
|
|
|
|
|
Utils::print(F("failed, code "));
|
|
|
|
|
Utils::println(String(state));
|
|
|
|
|
packet = "";
|
2024-03-23 13:48:12 +01:00
|
|
|
}
|
2024-02-24 14:09:05 +01:00
|
|
|
}
|
|
|
|
|
}
|
2024-05-16 23:52:25 +02:00
|
|
|
return packet;
|
2023-12-27 00:47:59 +01:00
|
|
|
}
|
2023-06-04 16:10:39 +02:00
|
|
|
|
2024-05-29 21:43:23 +02:00
|
|
|
void sleepRadio() {
|
|
|
|
|
radio.sleep();
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-04 16:10:39 +02:00
|
|
|
}
|