2025-01-13 14:07:48 +11:00
|
|
|
#include <Arduino.h> // needed for PlatformIO
|
|
|
|
|
#include <Mesh.h>
|
2025-01-20 20:22:40 +11:00
|
|
|
|
2025-09-08 21:46:19 +10:00
|
|
|
#include "MyMesh.h"
|
2025-06-03 19:01:03 +10:00
|
|
|
|
2025-03-05 12:52:29 +11:00
|
|
|
#ifdef DISPLAY_CLASS
|
|
|
|
|
#include "UITask.h"
|
|
|
|
|
static UITask ui_task(display);
|
|
|
|
|
#endif
|
|
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
#ifdef ETHERNET_ENABLED
|
2026-03-09 14:52:01 -07:00
|
|
|
#include <SPI.h>
|
|
|
|
|
#include <RAK13800_W5100S.h>
|
2026-03-10 22:35:31 -07:00
|
|
|
#include <helpers/nrf52/EthernetMac.h>
|
2026-03-09 14:52:01 -07:00
|
|
|
|
|
|
|
|
#define PIN_SPI1_MISO (29)
|
|
|
|
|
#define PIN_SPI1_MOSI (30)
|
|
|
|
|
#define PIN_SPI1_SCK (3)
|
2026-03-10 22:35:31 -07:00
|
|
|
SPIClass ETHERNET_SPI_PORT(NRF_SPIM1, PIN_SPI1_MISO, PIN_SPI1_SCK, PIN_SPI1_MOSI);
|
2026-03-09 14:52:01 -07:00
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
#define PIN_ETHERNET_POWER_EN WB_IO2
|
2026-03-09 14:52:01 -07:00
|
|
|
#define PIN_ETHERNET_RESET 21
|
|
|
|
|
#define PIN_ETHERNET_SS 26
|
|
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
#ifndef ETHERNET_TCP_PORT
|
|
|
|
|
#define ETHERNET_TCP_PORT 23 // telnet port for CLI access
|
2026-03-09 14:52:01 -07:00
|
|
|
#endif
|
|
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
#define ETHERNET_RETRY_INTERVAL_MS 30000
|
2026-03-09 14:52:01 -07:00
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
static EthernetServer ethernet_server(ETHERNET_TCP_PORT);
|
|
|
|
|
static EthernetClient ethernet_client;
|
|
|
|
|
static volatile bool ethernet_running = false;
|
2026-03-09 14:52:01 -07:00
|
|
|
|
|
|
|
|
// FreeRTOS task: handles hw init, DHCP, and retries in the background
|
2026-03-10 22:35:31 -07:00
|
|
|
static void ethernet_task(void* param) {
|
2026-03-09 14:52:01 -07:00
|
|
|
(void)param;
|
|
|
|
|
|
|
|
|
|
// Hardware init
|
|
|
|
|
Serial.println("ETH: Initializing hardware");
|
2026-03-10 22:35:31 -07:00
|
|
|
pinMode(PIN_ETHERNET_POWER_EN, OUTPUT);
|
|
|
|
|
digitalWrite(PIN_ETHERNET_POWER_EN, HIGH);
|
2026-03-09 14:52:01 -07:00
|
|
|
vTaskDelay(pdMS_TO_TICKS(100));
|
|
|
|
|
|
|
|
|
|
pinMode(PIN_ETHERNET_RESET, OUTPUT);
|
|
|
|
|
digitalWrite(PIN_ETHERNET_RESET, LOW);
|
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(100));
|
|
|
|
|
digitalWrite(PIN_ETHERNET_RESET, HIGH);
|
|
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
ETHERNET_SPI_PORT.begin();
|
|
|
|
|
Ethernet.init(ETHERNET_SPI_PORT, PIN_ETHERNET_SS);
|
2026-03-09 14:52:01 -07:00
|
|
|
|
|
|
|
|
uint8_t mac[6];
|
2026-03-11 12:43:08 -07:00
|
|
|
generateEthernetMac(mac);
|
2026-03-09 14:52:01 -07:00
|
|
|
Serial.printf("ETH: MAC: %02X:%02X:%02X:%02X:%02X:%02X\n",
|
|
|
|
|
mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
|
|
|
|
|
|
|
|
|
|
// Retry loop: keep trying until we get an IP
|
2026-03-10 22:35:31 -07:00
|
|
|
while (!ethernet_running) {
|
2026-03-09 14:52:01 -07:00
|
|
|
if (Ethernet.hardwareStatus() == EthernetNoHardware) {
|
|
|
|
|
Serial.println("ETH: Hardware not found, giving up");
|
|
|
|
|
vTaskDelete(NULL);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (Ethernet.linkStatus() == LinkOFF) {
|
2026-03-10 22:35:31 -07:00
|
|
|
vTaskDelay(pdMS_TO_TICKS(ETHERNET_RETRY_INTERVAL_MS));
|
2026-03-09 14:52:01 -07:00
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Serial.println("ETH: Link detected, attempting DHCP...");
|
|
|
|
|
if (Ethernet.begin(mac, 10000, 2000) == 0) {
|
|
|
|
|
Serial.println("ETH: DHCP failed, will retry");
|
2026-03-10 22:35:31 -07:00
|
|
|
vTaskDelay(pdMS_TO_TICKS(ETHERNET_RETRY_INTERVAL_MS));
|
2026-03-09 14:52:01 -07:00
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IPAddress ip = Ethernet.localIP();
|
|
|
|
|
Serial.printf("ETH: IP: %u.%u.%u.%u\n", ip[0], ip[1], ip[2], ip[3]);
|
2026-03-10 22:35:31 -07:00
|
|
|
Serial.printf("ETH: Listening on TCP port %d\n", ETHERNET_TCP_PORT);
|
|
|
|
|
ethernet_server.begin();
|
|
|
|
|
ethernet_running = true;
|
2026-03-09 14:52:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DHCP succeeded, task is done
|
|
|
|
|
vTaskDelete(NULL);
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
static void ethernet_start_task() {
|
|
|
|
|
xTaskCreate(ethernet_task, "eth_init", 1024, NULL, 1, NULL);
|
2026-03-09 14:52:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Format ethernet status into reply buffer. Returns true if command was handled.
|
2026-03-10 22:35:31 -07:00
|
|
|
static bool ethernet_handle_command(const char* command, char* reply) {
|
2026-03-11 12:43:08 -07:00
|
|
|
if (strcmp(command, "eth.status") == 0) {
|
|
|
|
|
if (!ethernet_running) {
|
|
|
|
|
strcpy(reply, "ETH: not connected");
|
|
|
|
|
} else {
|
|
|
|
|
IPAddress ip = Ethernet.localIP();
|
|
|
|
|
sprintf(reply, "ETH: %u.%u.%u.%u:%d", ip[0], ip[1], ip[2], ip[3], ETHERNET_TCP_PORT);
|
|
|
|
|
}
|
|
|
|
|
return true;
|
2026-03-09 14:52:01 -07:00
|
|
|
}
|
2026-03-11 12:43:08 -07:00
|
|
|
return false;
|
2026-03-09 14:52:01 -07:00
|
|
|
}
|
|
|
|
|
|
2026-03-11 12:43:08 -07:00
|
|
|
// Check for new TCP client connections, replacing any existing connection
|
2026-03-10 22:35:31 -07:00
|
|
|
static void ethernet_check_client() {
|
|
|
|
|
auto newClient = ethernet_server.available();
|
2026-03-09 14:52:01 -07:00
|
|
|
if (newClient) {
|
2026-03-10 22:35:31 -07:00
|
|
|
if (ethernet_client) ethernet_client.stop();
|
|
|
|
|
ethernet_client = newClient;
|
|
|
|
|
IPAddress ip = ethernet_client.remoteIP();
|
2026-03-09 14:52:01 -07:00
|
|
|
Serial.printf("ETH: Client connected from %u.%u.%u.%u\n", ip[0], ip[1], ip[2], ip[3]);
|
2026-03-10 22:35:31 -07:00
|
|
|
ethernet_client.println("MeshCore Repeater CLI");
|
2026-03-09 14:52:01 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2025-01-13 14:07:48 +11:00
|
|
|
StdRNG fast_rng;
|
|
|
|
|
SimpleMeshTables tables;
|
2025-01-22 19:47:53 +11:00
|
|
|
|
2025-03-27 19:34:16 +11:00
|
|
|
MyMesh the_mesh(board, radio_driver, *new ArduinoMillis(), fast_rng, rtc_clock, tables);
|
2025-01-13 14:07:48 +11:00
|
|
|
|
|
|
|
|
void halt() {
|
|
|
|
|
while (1) ;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-01 19:28:44 +10:00
|
|
|
static char command[160];
|
2026-03-10 22:35:31 -07:00
|
|
|
#ifdef ETHERNET_ENABLED
|
|
|
|
|
static char ethernet_command[160];
|
2026-03-09 14:52:01 -07:00
|
|
|
#endif
|
2025-01-15 17:14:08 +11:00
|
|
|
|
2025-12-23 12:48:08 +07:00
|
|
|
// For power saving
|
|
|
|
|
unsigned long lastActive = 0; // mark last active time
|
|
|
|
|
unsigned long nextSleepinSecs = 120; // next sleep in seconds. The first sleep (if enabled) is after 2 minutes from boot
|
|
|
|
|
|
2025-01-13 14:07:48 +11:00
|
|
|
void setup() {
|
|
|
|
|
Serial.begin(115200);
|
2025-01-19 20:22:02 +11:00
|
|
|
delay(1000);
|
2025-01-13 14:07:48 +11:00
|
|
|
|
|
|
|
|
board.begin();
|
2025-01-15 02:14:52 +11:00
|
|
|
|
2026-02-05 13:35:04 +00:00
|
|
|
#if defined(MESH_DEBUG) && defined(NRF52_PLATFORM)
|
|
|
|
|
// give some extra time for serial to settle so
|
|
|
|
|
// boot debug messages can be seen on terminal
|
|
|
|
|
delay(5000);
|
|
|
|
|
#endif
|
|
|
|
|
|
2025-12-23 12:48:08 +07:00
|
|
|
// For power saving
|
|
|
|
|
lastActive = millis(); // mark last active time since boot
|
|
|
|
|
|
2025-04-01 16:00:09 +13:00
|
|
|
#ifdef DISPLAY_CLASS
|
2025-05-19 14:16:55 +10:00
|
|
|
if (display.begin()) {
|
2025-04-01 16:00:09 +13:00
|
|
|
display.startFrame();
|
2025-09-02 13:56:24 +08:00
|
|
|
display.setCursor(0, 0);
|
2025-04-01 16:00:09 +13:00
|
|
|
display.print("Please wait...");
|
|
|
|
|
display.endFrame();
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2025-06-27 19:55:17 +01:00
|
|
|
if (!radio_init()) {
|
2026-02-05 13:35:04 +00:00
|
|
|
MESH_DEBUG_PRINTLN("Radio init failed!");
|
2025-06-27 19:55:17 +01:00
|
|
|
halt();
|
|
|
|
|
}
|
2025-01-15 02:14:52 +11:00
|
|
|
|
2025-03-27 19:34:16 +11:00
|
|
|
fast_rng.begin(radio_get_rng_seed());
|
2025-02-02 11:03:23 +11:00
|
|
|
|
2025-01-27 11:13:37 +11:00
|
|
|
FILESYSTEM* fs;
|
2025-05-11 09:26:32 +02:00
|
|
|
#if defined(NRF52_PLATFORM) || defined(STM32_PLATFORM)
|
2025-01-20 20:22:40 +11:00
|
|
|
InternalFS.begin();
|
2025-01-27 11:13:37 +11:00
|
|
|
fs = &InternalFS;
|
2025-02-04 01:35:04 +11:00
|
|
|
IdentityStore store(InternalFS, "");
|
2025-01-20 20:22:40 +11:00
|
|
|
#elif defined(ESP32)
|
2025-01-13 14:07:48 +11:00
|
|
|
SPIFFS.begin(true);
|
2025-01-27 11:13:37 +11:00
|
|
|
fs = &SPIFFS;
|
2025-01-13 14:07:48 +11:00
|
|
|
IdentityStore store(SPIFFS, "/identity");
|
2025-04-21 21:17:03 +01:00
|
|
|
#elif defined(RP2040_PLATFORM)
|
|
|
|
|
LittleFS.begin();
|
|
|
|
|
fs = &LittleFS;
|
|
|
|
|
IdentityStore store(LittleFS, "/identity");
|
2025-04-22 15:26:04 +10:00
|
|
|
store.begin();
|
2025-01-20 20:22:40 +11:00
|
|
|
#else
|
|
|
|
|
#error "need to define filesystem"
|
|
|
|
|
#endif
|
2025-01-13 14:07:48 +11:00
|
|
|
if (!store.load("_main", the_mesh.self_id)) {
|
2025-02-04 01:35:04 +11:00
|
|
|
MESH_DEBUG_PRINTLN("Generating new keypair");
|
2025-03-27 19:34:16 +11:00
|
|
|
the_mesh.self_id = radio_new_identity(); // create new random identity
|
2025-04-06 12:34:09 +10:00
|
|
|
int count = 0;
|
|
|
|
|
while (count < 10 && (the_mesh.self_id.pub_key[0] == 0x00 || the_mesh.self_id.pub_key[0] == 0xFF)) { // reserved id hashes
|
|
|
|
|
the_mesh.self_id = radio_new_identity(); count++;
|
|
|
|
|
}
|
2025-01-13 14:07:48 +11:00
|
|
|
store.save("_main", the_mesh.self_id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Serial.print("Repeater ID: ");
|
|
|
|
|
mesh::Utils::printHex(Serial, the_mesh.self_id.pub_key, PUB_KEY_SIZE); Serial.println();
|
|
|
|
|
|
2025-01-15 17:14:08 +11:00
|
|
|
command[0] = 0;
|
2026-03-10 22:35:31 -07:00
|
|
|
#ifdef ETHERNET_ENABLED
|
|
|
|
|
ethernet_command[0] = 0;
|
2026-03-09 14:52:01 -07:00
|
|
|
#endif
|
2025-01-15 17:14:08 +11:00
|
|
|
|
2025-05-03 13:14:03 +10:00
|
|
|
sensors.begin();
|
|
|
|
|
|
2025-01-27 11:13:37 +11:00
|
|
|
the_mesh.begin(fs);
|
2025-01-13 14:07:48 +11:00
|
|
|
|
2025-03-05 12:52:29 +11:00
|
|
|
#ifdef DISPLAY_CLASS
|
2025-04-20 17:40:58 -07:00
|
|
|
ui_task.begin(the_mesh.getNodePrefs(), FIRMWARE_BUILD_DATE, FIRMWARE_VERSION);
|
2025-03-05 12:52:29 +11:00
|
|
|
#endif
|
|
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
#ifdef ETHERNET_ENABLED
|
|
|
|
|
ethernet_start_task();
|
2026-03-09 14:52:01 -07:00
|
|
|
#endif
|
|
|
|
|
|
2026-01-26 22:20:36 +13:00
|
|
|
// send out initial zero hop Advertisement to the mesh
|
2026-01-26 22:39:39 +13:00
|
|
|
#if ENABLE_ADVERT_ON_BOOT == 1
|
2026-01-26 22:20:36 +13:00
|
|
|
the_mesh.sendSelfAdvertisement(16000, false);
|
2026-01-26 22:39:39 +13:00
|
|
|
#endif
|
2025-01-13 14:07:48 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void loop() {
|
2026-03-09 14:52:01 -07:00
|
|
|
// Handle Serial CLI
|
2025-01-15 17:14:08 +11:00
|
|
|
int len = strlen(command);
|
|
|
|
|
while (Serial.available() && len < sizeof(command)-1) {
|
|
|
|
|
char c = Serial.read();
|
2025-03-07 00:16:22 +01:00
|
|
|
if (c != '\n') {
|
2025-01-15 17:14:08 +11:00
|
|
|
command[len++] = c;
|
|
|
|
|
command[len] = 0;
|
2025-11-03 22:53:14 +11:00
|
|
|
Serial.print(c);
|
2025-01-15 17:14:08 +11:00
|
|
|
}
|
2025-11-03 22:53:14 +11:00
|
|
|
if (c == '\r') break;
|
2025-01-15 17:14:08 +11:00
|
|
|
}
|
|
|
|
|
if (len == sizeof(command)-1) { // command buffer full
|
|
|
|
|
command[sizeof(command)-1] = '\r';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (len > 0 && command[len - 1] == '\r') { // received complete line
|
2025-11-03 22:53:14 +11:00
|
|
|
Serial.print('\n');
|
2025-01-15 17:14:08 +11:00
|
|
|
command[len - 1] = 0; // replace newline with C string null terminator
|
2025-01-22 12:11:43 +11:00
|
|
|
char reply[160];
|
2026-03-09 14:52:01 -07:00
|
|
|
reply[0] = 0;
|
2026-03-10 22:35:31 -07:00
|
|
|
#ifdef ETHERNET_ENABLED
|
|
|
|
|
if (!ethernet_handle_command(command, reply)) {
|
|
|
|
|
the_mesh.handleCommand(0, command, reply);
|
|
|
|
|
}
|
|
|
|
|
#else
|
2025-07-08 17:50:06 +10:00
|
|
|
the_mesh.handleCommand(0, command, reply); // NOTE: there is no sender_timestamp via serial!
|
2026-03-10 22:35:31 -07:00
|
|
|
#endif
|
2025-01-22 12:11:43 +11:00
|
|
|
if (reply[0]) {
|
|
|
|
|
Serial.print(" -> "); Serial.println(reply);
|
2025-01-15 17:14:08 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
command[0] = 0; // reset command buffer
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
#ifdef ETHERNET_ENABLED
|
|
|
|
|
if (ethernet_running) {
|
|
|
|
|
ethernet_check_client();
|
2026-03-09 14:52:01 -07:00
|
|
|
Ethernet.maintain();
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
if (ethernet_running && ethernet_client && ethernet_client.connected()) {
|
|
|
|
|
int elen = strlen(ethernet_command);
|
|
|
|
|
while (ethernet_client.available() && elen < (int)sizeof(ethernet_command)-1) {
|
|
|
|
|
char c = ethernet_client.read();
|
|
|
|
|
if (c == '\n' && elen == 0) continue; // ignore leading LF (from CR+LF)
|
|
|
|
|
if (c == '\r' || c == '\n') { ethernet_command[elen++] = '\r'; break; }
|
|
|
|
|
ethernet_command[elen++] = c;
|
|
|
|
|
ethernet_command[elen] = 0;
|
2026-03-09 14:52:01 -07:00
|
|
|
}
|
2026-03-10 22:35:31 -07:00
|
|
|
if (elen == sizeof(ethernet_command)-1) {
|
|
|
|
|
ethernet_command[sizeof(ethernet_command)-1] = '\r';
|
2026-03-09 14:52:01 -07:00
|
|
|
}
|
|
|
|
|
|
2026-03-10 22:35:31 -07:00
|
|
|
if (elen > 0 && ethernet_command[elen - 1] == '\r') {
|
|
|
|
|
ethernet_command[elen - 1] = 0;
|
|
|
|
|
ethernet_client.println();
|
2026-03-09 14:52:01 -07:00
|
|
|
char reply[160];
|
|
|
|
|
reply[0] = 0;
|
2026-03-10 22:35:31 -07:00
|
|
|
if (!ethernet_handle_command(ethernet_command, reply)) {
|
|
|
|
|
the_mesh.handleCommand(0, ethernet_command, reply);
|
|
|
|
|
}
|
2026-03-09 14:52:01 -07:00
|
|
|
if (reply[0]) {
|
2026-03-10 22:35:31 -07:00
|
|
|
ethernet_client.print(" -> "); ethernet_client.println(reply);
|
2026-03-09 14:52:01 -07:00
|
|
|
}
|
2026-03-10 22:35:31 -07:00
|
|
|
ethernet_command[0] = 0;
|
2026-03-09 14:52:01 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2025-01-13 14:07:48 +11:00
|
|
|
the_mesh.loop();
|
2025-05-03 13:14:03 +10:00
|
|
|
sensors.loop();
|
2025-09-08 21:46:19 +10:00
|
|
|
#ifdef DISPLAY_CLASS
|
|
|
|
|
ui_task.loop();
|
|
|
|
|
#endif
|
2025-10-30 16:45:50 +11:00
|
|
|
rtc_clock.tick();
|
2025-12-23 12:48:08 +07:00
|
|
|
|
2026-02-01 14:46:55 +11:00
|
|
|
if (the_mesh.getNodePrefs()->powersaving_enabled && !the_mesh.hasPendingWork()) {
|
|
|
|
|
#if defined(NRF52_PLATFORM)
|
|
|
|
|
board.sleep(1800); // nrf ignores seconds param, sleeps whenever possible
|
|
|
|
|
#else
|
|
|
|
|
if (the_mesh.millisHasNowPassed(lastActive + nextSleepinSecs * 1000)) { // To check if it is time to sleep
|
2025-12-23 12:48:08 +07:00
|
|
|
board.sleep(1800); // To sleep. Wake up after 30 minutes or when receiving a LoRa packet
|
|
|
|
|
lastActive = millis();
|
|
|
|
|
nextSleepinSecs = 5; // Default: To work for 5s and sleep again
|
|
|
|
|
} else {
|
|
|
|
|
nextSleepinSecs += 5; // When there is pending work, to work another 5s
|
|
|
|
|
}
|
2026-02-01 14:46:55 +11:00
|
|
|
#endif
|
2025-12-23 12:48:08 +07:00
|
|
|
}
|
2025-01-13 14:07:48 +11:00
|
|
|
}
|