mirror of
https://github.com/meshtastic/firmware.git
synced 2026-06-04 06:43:00 -04:00
* feat: add Nordic nRF54L15-DK variant (Zephyr + BLE + LoRa)
Adds a community hardware variant for the Nordic nRF54L15-DK (PCA10156)
with an external EBYTE E22-900M30S (SX1262) LoRa module. First Meshtastic
port running on the Zephyr RTOS; all other Nordic targets use the nRF5
SoftDevice stack.
Scope
-----
- New Zephyr-based platform layer under src/platform/nrf54l15/ providing
Arduino-compatible shims (Arduino.h, SPI, Wire, Print, Stream) over the
Zephyr APIs plus a LittleFS-backed InternalFileSystem on SPIM20.
- Bluetooth LE peripheral (NRF54L15Bluetooth.*) built on the Zephyr BT
host stack, exposing the Meshtastic GATT service with legacy
connectable advertising, just-works pairing, dynamic MTU exchange
(up to 247 bytes), and iOS connection-parameter tweaks.
- Variant directory variants/nrf54l15/nrf54l15dk/ with pin map for the
E22 module on connector J1, PlatformIO env (nrf54l15dk), Zephyr
DT overlay and a wiring README.
- Zephyr project config (zephyr/prj.conf + board overlay) tuned for
BT + LoRa: 16 KB main stack, 4 KB BT RX thread, RTT logging in
immediate mode, newlib-nano heap sized to leave room for the GATT
pools while still allowing ATT MTU=247.
- extra_scripts/nrf54l15_linker.py works around a PlatformIO + old Ninja
issue where Zephyr's two-pass linker script generation does not run
automatically; the post-script parses build.ninja and invokes the
gcc -E step directly before the final link.
- boards/nrf54l15dk.json board definition (PlatformIO needs it for the
DK; the Seeed platform only ships the XIAO variants).
- variants/rp2350/rp2350.ini excludes platform/nrf54l15/ from RP2350
build_src_filter so the shared platform tree does not leak between
targets.
- .gitignore: add nRF J-Link / RTT debug artifacts (flash.jlink,
rtt_*.txt).
Shared source changes
---------------------
- src/main.{cpp,h}, src/RedirectablePrint.cpp, src/FSCommon.{cpp,h},
src/mesh/{Channels,NodeDB,RadioLibInterface,MeshService,PhoneAPI}.cpp,
src/mesh/RadioLibInterface.h, src/modules/AdminModule.cpp: add small
guards / helpers so the Zephyr build compiles alongside the Arduino
targets. Behavior on existing boards is unchanged.
Hardware model
--------------
HW_VENDOR maps to meshtastic_HardwareModel_PRIVATE_HW until a dedicated
protobuf enum value is assigned upstream. The variant declares
custom_meshtastic_hw_model = 132 so the maintainers can wire the new
enum value through the protobufs repo after merge.
Hardware note
-------------
The E22-900M30S does not connect its DIO2 pin to TXEN internally — a
wire/solder bridge between DIO2 and TXEN on the module is required for
TX to work. Details and full pin map are in the variant README.
Validation
----------
Built clean against develop. On real hardware (April 2026) the port
passes end-to-end: iOS companion app pairs and connects, configuration
round-trip works, LoRa TX/RX reaches a canonical tbeam on the same mesh
channel, NodeDB updates propagate both ways, and traceroute completes.
* fix(nrf54l15): use atomic fs_rename instead of copy fallback
Zephyr LittleFS on nrf54l15 supports fs_rename natively, so route it
through the same atomic path as ESP32. The previous copyFile+remove
fallback truncated the destination before copying, leaving 0-byte files
if interrupted mid-write.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
* fix(nrf54l15): expand storage_partition from 36KB to 700KB
LittleFS on the default 9-block (36KB) storage_partition ran out of
space during copy-on-write of config.proto, causing fs_write to return
ENOSPC and pb_encode to surface "io error" when saving configuration
via the mobile app.
Reclaim slot1_partition (the MCUboot secondary slot — unused since we
flash directly via J-Link) and grow storage_partition to span
0xb6000..0x165000 (~175 blocks).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
* fix(nrf54l15): drop USERPREFS_LORACONFIG_* so LoRa config stays mutable
NodeDB rewrites LoRa config from USERPREFS_LORACONFIG_* on every boot,
which prevented reconfiguration via the BLE/serial app. Drop the
variant-level defaults; users configure region and modem preset through
the app like every other variant.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
* fix(nrf54l15): enforce MITM passkey pairing on GATT service
- Add MESH_PERM_READ/MESH_PERM_WRITE macros (READ_AUTHEN/WRITE_AUTHEN)
on all mesh service characteristics so clients must complete passkey
exchange before accessing fromNum/fromRadio/toRadio/logRadio.
- Wire FIXED_PIN mode to bt_passkey_set() so the device advertises a
known PIN (config.bluetooth.fixed_pin); RANDOM_PIN keeps default
per-pairing random passkey.
- Reduce BleDeferredThread HARD_WATCHDOG_MS from 3min to 1min.
- prj.conf: CONFIG_BT_SMP_ENFORCE_MITM=y, CONFIG_BT_FIXED_PASSKEY=y,
CONFIG_BT_SMP_SC_PAIR_ONLY=n (legacy fallback for clients that abort
SC pairing with reason 0x01 within 150ms).
* fix(nrf54l15): resolve develop-merge conflict + cppcheck warnings
The `Merge branch 'develop'` left two ~RadioLibInterface() declarations
in src/mesh/RadioLibInterface.h: the inline version added upstream by
PR #10254 (which independently applied the same UAF guard this PR was
carrying) and the out-of-line version this PR introduced. GCC rejects
the duplicate, breaking every platform build. Drop the out-of-line
declaration + definition; keep upstream's inline form.
Also silence the 13 cppcheck low warnings introduced by the new
nrf54l15 Arduino shim — Arduino's `String`/`SPISettings` API contract
relies on implicit single-arg constructors used pervasively by
existing Meshtastic code, so suppress `noExplicitConstructor` inline
with a comment instead of breaking the API. The few mechanical wins
(`const tmp[2]`, `const uint32_t *sp`) are applied directly.
* fmt: fix Trunk Check lint issues on nrf54l15-port
- extra_scripts/nrf54l15_linker.py: move regular imports above
Import("env") to silence E402, add trunk-ignore-all(F821) for the
PIO/SCons SConstruct injection (matches esp32_pre.py / nrf52_extra.py
convention)
- src/platform/nrf54l15/NRF54L15Bluetooth.cpp: clang-format 16.0.3
- boards/nrf54l15dk.json + variants/nrf54l15/nrf54l15dk/README.md:
prettier 3.8.3 (also resolves markdownlint MD060 on README tables)
No behavior change.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
* fix(nrf54l15): address Copilot review comments + correct clang-format style
Six review threads from the 2026-04-30 Copilot review:
- src/platform/nrf54l15/nrf54l15_main.cpp: validate PSP against the nRF54L15
SRAM range (0x20000000..0x20040000) and 4-byte alignment before walking the
faulting thread's stack, and clamp the walk so it never reads past the end
of RAM. Prevents a second fault inside the fatal handler when PSP is
corrupted (common in real faults).
- src/platform/nrf54l15/nrf54l15_arduino.cpp: gate the bring-up printk traces
in digitalWrite/digitalRead (CS/NRESET toggle log, BUSY-before-NRESET
snapshot, BUSY periodic timeline) behind a new -DNRF54L15_GPIO_DEBUG flag
that is off by default. The "dev NOT READY" message stays unconditional —
it indicates a genuine hardware/DTS misconfig.
- src/modules/AdminModule.cpp: don't mutate config.device.output_gpio_enabled
from handleGetConfig(). Reflect the live pin state in the response payload
only — a getter must not write back to disk-persisted state.
- src/platform/nrf54l15/InternalFileSystem.h: derive totalBytes() from
FIXED_PARTITION_SIZE(storage_partition) at compile time so it tracks the
DK overlay's ~700 KB partition instead of the stale 36 KB hard-coded value.
Updated the file header comment accordingly.
- extra_scripts/nrf54l15_linker.py: make _extract_gcc_command() handle the
POSIX Ninja COMMAND format (no `cmd.exe /C "..."` wrapper) in addition to
the Windows form, so the script doesn't hard-fail on Linux/macOS hosts.
- src/platform/nrf54l15/NRF54L15Bluetooth.cpp: clamp NO_PIN to RANDOM_PIN
with a one-shot LOG_WARN. The mesh GATT permissions are declared with
BT_GATT_PERM_*_AUTHEN and prj.conf sets CONFIG_BT_SMP_ENFORCE_MITM=y, so
NO_PIN with no auth callbacks would leave every characteristic returning
BT_ATT_ERR_AUTHENTICATION. Falling back to RANDOM_PIN keeps the link
usable instead of silently broken. Also re-formatted this file with the
project's .trunk/configs/.clang-format (Linux braces, 4-space indent,
130-col) — the previous lint-fix commit a2aca3234 accidentally used the
default LLVM style here, which CI's clang-format would have rejected.
Build verified: pio run -e nrf54l15dk passes, RAM 47.4%, Flash 28.6%.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
* fix(nrf54l15): address remaining Copilot review threads
Round 2/3 review fixes — bugs first, then docs/portability:
BLE concurrency (NRF54L15Bluetooth.cpp):
- onNowHasData / sendLog / BleDeferredThread / shutdown: acquire
active_conn under ble_mutex via new acquire_active_conn() helper so
disconnected_cb can't free the conn between the null check and
bt_conn_ref/bt_gatt_notify (use-after-unref).
- write_toradio: reject writes that exceed MAX_TO_FROM_RADIO_SIZE with
ATT_ERR_INVALID_ATTRIBUTE_LEN instead of returning success and silently
dropping the payload (would hide failed config writes from the phone).
- start_advertising: truncate the device name to fit the 31-byte legacy
scan-response limit and switch to BT_DATA_NAME_SHORTENED so
bt_le_adv_start() doesn't reject the payload when the name approaches
CONFIG_BT_DEVICE_NAME_MAX=32.
Linker / portability:
- main.h: drop the rp2040Loop() forward declaration that had no
definition and no callers — would surface as a link error if any RP2040
build added a call to the symbol.
- nrf54l15_arduino.cpp: transfer16() now uses static __aligned(4) DMA
buffers (matching transfer()), removing the EasyDMA-reachability hazard
of caller-stack buffers on this part.
Filesystem (InternalFileSystem.h):
- usedBytes(): return real usage from fs_statvfs() instead of 0 so OTA
/ range-test free-space guards work.
- rewindDirectory(): close the dir before reopening — Zephyr fs_dir_t has
no rewind, and re-fs_opendir on an open handle leaks LittleFS state.
Crash handler (nrf54l15_main.cpp):
- After the stack walk, busy-wait 50 ms to flush RTT/printk and call
sys_reboot(SYS_REBOOT_COLD) directly so the saved_crash record is
actually reported on the next boot. Default Zephyr config has
RESET_ON_FATAL_ERROR=n, so the previous k_fatal_halt() spun forever.
Generalization / config:
- PhoneAPI.cpp: replace the NRF54L15_DK ifdef with a
MESHTASTIC_EXCLUDE_FILES_MANIFEST capability flag (defined in the
nrf54l15dk env) so future variants can opt in/out without touching
shared code.
- variants/nrf54l15/nrf54l15.ini: parameterize libdeps include paths via
${PIOENV} so additional nRF54L15 envs sharing nrf54l15_base don't break.
- prj.conf: drop the stale "36 KB storage_partition" comments — the DK
overlay reclaims slot1 to ~700 KB and runtime size comes from
FIXED_PARTITION_SIZE.
- nrf54l15dk overlay: remove the zephyr,console / zephyr,shell-uart
chosen entries that conflicted with CONFIG_UART_CONSOLE=n + RTT
console; keep uart30 enabled so swapping the console is one Kconfig
flip away.
Build: nrf54l15dk SUCCESS (flash 28.6%, RAM 47.4%); wiznet_5500_evb_pico2
SUCCESS (verifies the shared main.h change).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
* fix(nrf54l15dk): use PRIVATE_HW (255) for custom_meshtastic_hw_model
Per @thebentern's review: the nRF54L15-DK is a development kit, not a
canonical Meshtastic SKU, so it falls under HardwareModel::PRIVATE_HW
(255) — the same enum value already used at runtime via HW_VENDOR. The
placeholder 132 is removed; no dedicated enum number will be assigned
for DK boards. Slug stays NRF54L15_DK as a human-readable identifier.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
* fix(nrf54l15): unstarve bt_long_wq so SC pairing completes
bt_pub_key_gen() runs the ECC P256 key generation on bt_long_wq. At default
prio=10 (preemptible) and stack=1400 it gets starved by Meshtastic app
threads at boot — sc_public_key stays NULL for minutes, smp_public_key()
defers with SMP_FLAG_PKEY_SEND, and every SC pairing attempt stalls right
after the public-key exchange. iOS shows "Connecting…" forever with no
PIN prompt; bleak/CLI fails the first CCC notify write with
"Protocol Error 0x05: Insufficient Authentication".
Set CONFIG_BT_LONG_WQ_PRIO=0 (highest preemptible, ties with main) and
CONFIG_BT_LONG_WQ_STACK_SIZE=4096 (margin for the P256M driver frames).
Validated E2E with iOS Meshtastic app: bt_smp_pkey_ready fires within ~40 s
of boot, 20 SC Passkey Entry rounds complete with matching pcnf/cfm,
encrypt 0x01 / sec_level 0x04 (Authenticated MITM), bonded=1.
* feat(nrf54l15): hardware I2C bus via TWIM30 + sensor telemetry
Adds the Arduino TwoWire layer for the nRF54L15-DK so Meshtastic's
sensor drivers can talk to external I2C devices over the hardware
TWIM30 peripheral.
Bus binding:
- &uart30 disabled in the board overlay (peripheral instance 30 is
shared between UARTE30 / TWIM30 / SPIM30 — pick one). Console stays
on RTT via CONFIG_RTT_CONSOLE.
- New i2c30_default / i2c30_sleep pinctrl with SDA=P0.03 / SCL=P0.04.
External 4.7 kOhm pull-ups required on both lines.
- &i2c30 enabled at I2C_BITRATE_FAST (400 kHz).
- button_3 (SW3, P0.04) deleted from DTS so the pad can be claimed by
i2c30 pinctrl; SW3 is still wired to the pad on the DK, do not press
it during I2C use or it will short SCL to GND.
Arduino layer:
- src/platform/nrf54l15/Wire.cpp resolves the DT node at compile time
via DEVICE_DT_GET(DT_NODELABEL(i2c30)) and dispatches Arduino's
beginTransmission / write / endTransmission / requestFrom to
i2c_write / i2c_write_read / i2c_read. Buffer is sized to 256 bytes
for forward compatibility with the SE050 secure element on the
custom PCB.
- Wire.h drops the prior compile-only stubs and exposes the real
TwoWire surface.
- Arduino.h: BitOrder becomes an enum (not #define) so Adafruit_BusIO's
`typedef BitOrder BusIOBitOrder;` compiles.
Variant + build flags:
- nrf54l15.ini flips HAS_WIRE / HAS_SENSOR / HAS_TELEMETRY from 0 to 1
and cherry-picks the sensor libs Meshtastic needs (BusIO, Sensor,
BMP280, BME280, INA219/226/260/3221, SHT4X). The full
environmental_base group is avoided because it pulls
Adafruit_SSD1306 / Adafruit_GFX which rely on Arduino pin macros the
Zephyr shim does not implement.
- nrf54l15dk variant.h defines PIN_WIRE_SDA / PIN_WIRE_SCL for parity
with the Arduino convention used by other variants. The actual bus
wiring is fixed by the overlay pinctrl above.
Validated 2026-05-14/15 on the DK with BMP280 @ 0x76 (temperature +
barometric pressure) and INA3221 @ 0x42 (rail voltage / current);
EnvironmentTelemetry / PowerTelemetry packets transmit successfully
over LoRa.
Footprint cost on nrf54l15dk: +45 KB flash, +1.7 KB RAM.
* feat(nodedb): honor USERPREFS for environment telemetry on first boot
installDefaultConfig() now respects two new compile-time prefs:
USERPREFS_CONFIG_ENV_TELEM_UPDATE_INTERVAL
USERPREFS_CONFIG_ENVIRONMENT_MEASUREMENT_ENABLED
The mobile apps enforce a 30 min floor on environment_update_interval
in the settings UI, which makes short-interval bring-up testing of new
sensor hardware painful — you have to wait half an hour for the first
LoRa packet to confirm wiring + driver. With these prefs baked into
the variant, the firmware can ship a freshly-flashed device that
broadcasts on a shorter cadence (e.g. 900 s) the moment storage_partition
is empty.
Both prefs are gated on #ifdef so the behavior is unchanged for any
variant that does not opt in. Documented in userPrefs.jsonc with the
existing telemetry-interval pref block.
* fix(nrf54l15): allow multiple bonded BLE peers
CONFIG_BT_MAX_PAIRED defaults to 1, so once the first peer (e.g. an
iOS phone) has paired and bonded, every subsequent pairing attempt
from a different MAC fails inside bt_keys_get_addr() with no free
key slot — the host returns BT_SECURITY_ERR_KEY_DOES_NOT_EXIST and
the second peer never gets past SMP.
Raise the slot count to 4 so the device can simultaneously hold an
iOS phone, a Windows host, a Linux host, and one spare bond. Add
BT_KEYS_OVERWRITE_OLDEST so that once the table fills, the LRU peer
is evicted on the next pairing rather than rejecting the new peer.
This matches the behavior other Meshtastic ports already provide
(nRF52 uses CONFIG_BT_PERIPHERAL_PRIO_CONN with similar semantics).
Discovered while bringing up the Python CLI on Windows alongside
the existing iOS bond.
* fix(nrf54l15): zero-initialize TwoWire buffers + clang-format Wire
cppcheck on every CI target (esp32s3, rp2040, rp2350, nrf52840, ...) was
failing the build with two `uninitMemberVar` warnings on TwoWire's
constructor: `txBuf` and `rxBuf` (256-byte arrays) were not initialized.
Even though the buffers are only read after txLen/rxLen is set, leaving
them uninitialized is a footgun if any future caller bypasses the
len-set step. Use C++11 value-initialization in the member initializer
list — costs ~512 B of memset at boot, gains a clean cppcheck pass and
defensive-against-future-changes semantics.
Also reformat Wire.{cpp,h} with the project's `.trunk/configs/.clang-format`
config so the Trunk Check Runner passes — clang-format moved the
`<errno.h>` include before the Zephyr-namespaced ones in Wire.cpp and
collapsed two inline overloads to single lines in Wire.h.
* fix(AdminModule): remove dead OUTPUT_GPIO_PIN/GpioOutputModule references
OUTPUT_GPIO_PIN is never defined and modules/GpioOutputModule.h doesn't
exist in the codebase; all #ifdef OUTPUT_GPIO_PIN branches were dead code
introduced by the nRF54L15-DK variant commit. Strips the include, the
output_gpio_enabled OFF→ON/ON→OFF transition logic in handleSetConfig(),
and the digitalRead() reflection in handleGetConfig().
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
---------
Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Co-authored-by: Ben Meadors <benmmeadors@gmail.com>
1232 lines
40 KiB
C++
1232 lines
40 KiB
C++
#include "configuration.h"
|
|
#if !MESHTASTIC_EXCLUDE_GPS
|
|
#include "GPS.h"
|
|
#endif
|
|
#if !MESHTASTIC_EXCLUDE_INPUTBROKER
|
|
#include "input/InputBroker.h"
|
|
#endif
|
|
#include "MeshRadio.h"
|
|
#include "MeshService.h"
|
|
#include "NodeDB.h"
|
|
#include "PowerFSM.h"
|
|
#include "PowerMon.h"
|
|
#include "RadioLibInterface.h"
|
|
#include "ReliableRouter.h"
|
|
#include "TransmitHistory.h"
|
|
#include "airtime.h"
|
|
#include "buzz.h"
|
|
#include "power/PowerHAL.h"
|
|
|
|
#include "FSCommon.h"
|
|
#include "RTC.h"
|
|
#include "SPILock.h"
|
|
#include "Throttle.h"
|
|
#include "concurrency/OSThread.h"
|
|
#include "concurrency/Periodic.h"
|
|
#include "detect/ScanI2C.h"
|
|
#include "error.h"
|
|
#include "power.h"
|
|
|
|
#if !MESHTASTIC_EXCLUDE_I2C
|
|
#include "detect/ScanI2CConsumer.h"
|
|
#include "detect/ScanI2CTwoWire.h"
|
|
#include <Wire.h>
|
|
#endif
|
|
#include "detect/einkScan.h"
|
|
#include "graphics/Screen.h"
|
|
#include "main.h"
|
|
#include "mesh/generated/meshtastic/config.pb.h"
|
|
#include "meshUtils.h"
|
|
#include "modules/Modules.h"
|
|
#include "sleep.h"
|
|
#include "target_specific.h"
|
|
#include <memory>
|
|
#include <utility>
|
|
#if HAS_SCREEN
|
|
#include "MessageStore.h"
|
|
#endif
|
|
|
|
#ifdef ARCH_ESP32
|
|
#include "freertosinc.h"
|
|
#if !MESHTASTIC_EXCLUDE_WEBSERVER
|
|
#include "mesh/http/WebServer.h"
|
|
#endif
|
|
#if !MESHTASTIC_EXCLUDE_BLUETOOTH
|
|
#include "nimble/NimbleBluetooth.h"
|
|
NimbleBluetooth *nimbleBluetooth = nullptr;
|
|
#endif
|
|
#endif
|
|
|
|
#ifdef ARCH_NRF52
|
|
#include "NRF52Bluetooth.h"
|
|
NRF52Bluetooth *nrf52Bluetooth = nullptr;
|
|
#endif
|
|
|
|
#ifdef ARCH_NRF54L15
|
|
void nrf54l15Setup();
|
|
void nrf54l15Loop();
|
|
NRF54L15Bluetooth *nrf54l15Bluetooth = nullptr;
|
|
#endif
|
|
|
|
#if HAS_WIFI || defined(USE_WS5500) || defined(USE_CH390D)
|
|
#include "mesh/api/WiFiServerAPI.h"
|
|
#include "mesh/wifi/WiFiAPClient.h"
|
|
#endif
|
|
|
|
#if HAS_ETHERNET && !defined(USE_WS5500) && !defined(USE_CH390D)
|
|
#include "mesh/api/ethServerAPI.h"
|
|
#include "mesh/eth/ethClient.h"
|
|
#endif
|
|
|
|
#if !MESHTASTIC_EXCLUDE_MQTT
|
|
#include "mqtt/MQTT.h"
|
|
#endif
|
|
|
|
#ifdef ARCH_PORTDUINO
|
|
#include "linux/LinuxHardwareI2C.h"
|
|
#include "mesh/raspihttp/PiWebServer.h"
|
|
#include "platform/portduino/PortduinoGlue.h"
|
|
#include <cstdlib>
|
|
#include <fstream>
|
|
#include <iostream>
|
|
#include <string>
|
|
#endif
|
|
|
|
#ifdef ARCH_ESP32
|
|
#ifdef DEBUG_PARTITION_TABLE
|
|
#include "esp_partition.h"
|
|
|
|
void printPartitionTable()
|
|
{
|
|
printf("\n--- Partition Table ---\n");
|
|
// Print Column Headers
|
|
printf("| %-16s | %-4s | %-7s | %-10s | %-10s |\n", "Label", "Type", "Subtype", "Offset", "Size");
|
|
printf("|------------------|------|---------|------------|------------|\n");
|
|
|
|
// Create an iterator to find ALL partitions (Type ANY, Subtype ANY)
|
|
esp_partition_iterator_t it = esp_partition_find(ESP_PARTITION_TYPE_ANY, ESP_PARTITION_SUBTYPE_ANY, NULL);
|
|
|
|
// Loop through the iterator
|
|
if (it != NULL) {
|
|
do {
|
|
const esp_partition_t *part = esp_partition_get(it);
|
|
|
|
// Print details: Label, Type (Hex), Subtype (Hex), Offset (Hex), Size (Hex)
|
|
printf("| %-16s | 0x%02x | 0x%02x | 0x%08x | 0x%08x |\n", part->label, part->type, part->subtype, part->address,
|
|
part->size);
|
|
|
|
// Move to next partition
|
|
it = esp_partition_next(it);
|
|
} while (it != NULL);
|
|
|
|
// Release the iterator memory
|
|
esp_partition_iterator_release(it);
|
|
} else {
|
|
printf("No partitions found.\n");
|
|
}
|
|
printf("-----------------------\n");
|
|
}
|
|
#endif // DEBUG_PARTITION_TABLE
|
|
#endif // ARCH_ESP32
|
|
|
|
#include "AmbientLightingThread.h"
|
|
#include "PowerFSMThread.h"
|
|
|
|
#if !defined(ARCH_STM32WL) && !MESHTASTIC_EXCLUDE_I2C && !MESHTASTIC_EXCLUDE_ACCELEROMETER
|
|
#include "motion/AccelerometerThread.h"
|
|
AccelerometerThread *accelerometerThread = nullptr;
|
|
#endif
|
|
|
|
#ifdef HAS_I2S
|
|
#include "AudioThread.h"
|
|
AudioThread *audioThread = nullptr;
|
|
#endif
|
|
|
|
#ifdef USE_XL9555
|
|
#include "ExtensionIOXL9555.hpp"
|
|
ExtensionIOXL9555 io;
|
|
#endif
|
|
|
|
#if HAS_TFT
|
|
extern void tftSetup(void);
|
|
#endif
|
|
|
|
#ifdef HAS_UDP_MULTICAST
|
|
#include "mesh/udp/UdpMulticastHandler.h"
|
|
UdpMulticastHandler *udpHandler = nullptr;
|
|
#endif
|
|
|
|
#if defined(TCXO_OPTIONAL)
|
|
float tcxoVoltage = SX126X_DIO3_TCXO_VOLTAGE; // if TCXO is optional, put this here so it can be changed further down.
|
|
#endif
|
|
|
|
#ifdef MESHTASTIC_INCLUDE_NICHE_GRAPHICS
|
|
void setupNicheGraphics();
|
|
#include "nicheGraphics.h"
|
|
#endif
|
|
|
|
#if defined(HW_SPI1_DEVICE) && defined(ARCH_ESP32)
|
|
SPIClass SPI1(HSPI);
|
|
#endif
|
|
|
|
using namespace concurrency;
|
|
|
|
volatile static const char slipstreamTZString[] = {USERPREFS_TZ_STRING};
|
|
|
|
// We always create a screen object, but we only init it if we find the hardware
|
|
graphics::Screen *screen = nullptr;
|
|
|
|
// Global power status
|
|
meshtastic::PowerStatus *powerStatus = new meshtastic::PowerStatus();
|
|
|
|
// Global GPS status
|
|
meshtastic::GPSStatus *gpsStatus = new meshtastic::GPSStatus();
|
|
|
|
// Global Node status
|
|
meshtastic::NodeStatus *nodeStatus = new meshtastic::NodeStatus();
|
|
|
|
// Global Bluetooth status
|
|
meshtastic::BluetoothStatus *bluetoothStatus = new meshtastic::BluetoothStatus();
|
|
|
|
// Scan for I2C Devices
|
|
|
|
/// The I2C address of our display (if found)
|
|
ScanI2C::DeviceAddress screen_found = ScanI2C::ADDRESS_NONE;
|
|
|
|
// The I2C address of the cardkb or RAK14004 (if found)
|
|
ScanI2C::DeviceAddress cardkb_found = ScanI2C::ADDRESS_NONE;
|
|
// 0x02 for RAK14004, 0x00 for cardkb, 0x10 for T-Deck
|
|
uint8_t kb_model;
|
|
// global bool to record that a kb is present
|
|
bool kb_found = false;
|
|
// global bool to record that on-screen keyboard (OSK) is present
|
|
bool osk_found = false;
|
|
|
|
// The I2C address of the RTC Module (if found)
|
|
ScanI2C::DeviceAddress rtc_found = ScanI2C::ADDRESS_NONE;
|
|
// The I2C address of the Accelerometer (if found)
|
|
ScanI2C::DeviceAddress accelerometer_found = ScanI2C::ADDRESS_NONE;
|
|
// The I2C address of the RGB LED (if found)
|
|
ScanI2C::FoundDevice rgb_found = ScanI2C::FoundDevice(ScanI2C::DeviceType::NONE, ScanI2C::ADDRESS_NONE);
|
|
/// The I2C address of our Air Quality Indicator (if found)
|
|
ScanI2C::DeviceAddress aqi_found = ScanI2C::ADDRESS_NONE;
|
|
|
|
#ifdef HAS_DRV2605
|
|
Adafruit_DRV2605 drv;
|
|
#endif
|
|
|
|
bool isVibrating = false;
|
|
|
|
bool eink_found = true;
|
|
|
|
uint32_t serialSinceMsec;
|
|
bool pauseBluetoothLogging = false;
|
|
|
|
bool pmu_found;
|
|
|
|
#if !MESHTASTIC_EXCLUDE_I2C
|
|
// Array map of sensor types with i2c address and wire as we'll find in the i2c scan
|
|
std::pair<uint8_t, TwoWire *> nodeTelemetrySensorsMap[_meshtastic_TelemetrySensorType_MAX + 1] = {};
|
|
#endif
|
|
|
|
Router *router = NULL; // Users of router don't care what sort of subclass implements that API
|
|
|
|
const char *firmware_version = optstr(APP_VERSION_SHORT);
|
|
|
|
const char *getDeviceName()
|
|
{
|
|
uint8_t dmac[6];
|
|
|
|
getMacAddr(dmac);
|
|
|
|
// Meshtastic_ab3c or Shortname_abcd
|
|
static char name[20];
|
|
snprintf(name, sizeof(name), "%02x%02x", dmac[4], dmac[5]);
|
|
// if the shortname exists and is NOT the new default of ab3c, use it for BLE name.
|
|
if (strcmp(owner.short_name, name) != 0) {
|
|
snprintf(name, sizeof(name), "%s_%02x%02x", owner.short_name, dmac[4], dmac[5]);
|
|
} else {
|
|
snprintf(name, sizeof(name), "Meshtastic_%02x%02x", dmac[4], dmac[5]);
|
|
}
|
|
return name;
|
|
}
|
|
|
|
uint32_t timeLastPowered = 0;
|
|
|
|
static OSThread *powerFSMthread;
|
|
AmbientLightingThread *ambientLightingThread;
|
|
|
|
RadioLibHal *RadioLibHAL = NULL;
|
|
|
|
/**
|
|
* Some platforms (nrf52) might provide an alterate version that suppresses calling delay from sleep.
|
|
*/
|
|
__attribute__((weak, noinline)) bool loopCanSleep()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
// Weak empty variant initialization function.
|
|
// May be redefined by variant files.
|
|
void lateInitVariant() __attribute__((weak));
|
|
void lateInitVariant() {}
|
|
|
|
void earlyInitVariant() __attribute__((weak));
|
|
void earlyInitVariant() {}
|
|
|
|
// NRF52 (and probably other platforms) can report when system is in power failure mode
|
|
// (eg. too low battery voltage) and operating it is unsafe (data corruption, bootloops, etc).
|
|
// For example NRF52 will prevent any flash writes in that case automatically
|
|
// (but it causes issues we need to handle).
|
|
// This detection is independent from whatever ADC or dividers used in Meshtastic
|
|
// boards and is internal to chip.
|
|
|
|
// we use powerHAL layer to get this info and delay booting until power level is safe
|
|
|
|
// wait until power level is safe to continue booting (to avoid bootloops)
|
|
// blink user led in 3 flashes sequence to indicate what is happening
|
|
void waitUntilPowerLevelSafe()
|
|
{
|
|
while (powerHAL_isPowerLevelSafe() == false) {
|
|
|
|
#ifdef LED_POWER
|
|
|
|
// 3x: blink for 300 ms, pause for 300 ms
|
|
|
|
for (int i = 0; i < 3; i++) {
|
|
digitalWrite(LED_POWER, LED_STATE_ON);
|
|
delay(300);
|
|
digitalWrite(LED_POWER, LED_STATE_OFF);
|
|
delay(300);
|
|
}
|
|
#endif
|
|
|
|
// sleep for 2s
|
|
delay(2000);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Print info as a structured log message (for automated log processing)
|
|
*/
|
|
void printInfo()
|
|
{
|
|
LOG_INFO("S:B:%d,%s,%s,%s", HW_VENDOR, optstr(APP_VERSION), optstr(APP_ENV), optstr(APP_REPO));
|
|
}
|
|
#ifndef PIO_UNIT_TESTING
|
|
void setup()
|
|
{
|
|
|
|
// initialize power HAL layer as early as possible
|
|
powerHAL_init();
|
|
|
|
#ifdef LED_POWER
|
|
pinMode(LED_POWER, OUTPUT);
|
|
digitalWrite(LED_POWER, LED_STATE_ON);
|
|
#endif
|
|
|
|
// prevent booting if device is in power failure mode
|
|
// boot sequence will follow when battery level raises to safe mode
|
|
waitUntilPowerLevelSafe();
|
|
|
|
// Defined in variant.cpp for early init code
|
|
earlyInitVariant();
|
|
|
|
#if defined(PIN_POWER_EN)
|
|
pinMode(PIN_POWER_EN, OUTPUT);
|
|
digitalWrite(PIN_POWER_EN, HIGH);
|
|
#endif
|
|
|
|
#ifdef LED_NOTIFICATION
|
|
pinMode(LED_NOTIFICATION, OUTPUT);
|
|
digitalWrite(LED_NOTIFICATION, HIGH ^ LED_STATE_ON);
|
|
#endif
|
|
|
|
#ifdef WIFI_LED
|
|
pinMode(WIFI_LED, OUTPUT);
|
|
digitalWrite(WIFI_LED, HIGH ^ WIFI_STATE_ON);
|
|
#endif
|
|
|
|
#ifdef BLE_LED
|
|
pinMode(BLE_LED, OUTPUT);
|
|
digitalWrite(BLE_LED, LED_STATE_OFF);
|
|
#endif
|
|
|
|
concurrency::hasBeenSetup = true;
|
|
#if HAS_SCREEN
|
|
meshtastic_Config_DisplayConfig_OledType screen_model =
|
|
meshtastic_Config_DisplayConfig_OledType::meshtastic_Config_DisplayConfig_OledType_OLED_AUTO;
|
|
#endif
|
|
OLEDDISPLAY_GEOMETRY screen_geometry = GEOMETRY_128_64;
|
|
|
|
#ifdef USE_SEGGER
|
|
auto mode = false ? SEGGER_RTT_MODE_BLOCK_IF_FIFO_FULL : SEGGER_RTT_MODE_NO_BLOCK_TRIM;
|
|
#ifdef NRF52840_XXAA
|
|
auto buflen = 4096; // this board has a fair amount of ram
|
|
#else
|
|
auto buflen = 256; // this board has a fair amount of ram
|
|
#endif
|
|
SEGGER_RTT_ConfigUpBuffer(SEGGER_STDOUT_CH, NULL, NULL, buflen, mode);
|
|
#endif
|
|
|
|
#ifdef DEBUG_PORT
|
|
consoleInit(); // Set serial baud rate and init our mesh console
|
|
#endif
|
|
|
|
#ifdef UNPHONE
|
|
unphone.printStore();
|
|
#endif
|
|
|
|
#if ARCH_PORTDUINO
|
|
RTCQuality ourQuality = RTCQualityDevice;
|
|
|
|
std::string timeCommandResult = exec("timedatectl status | grep synchronized | grep yes -c");
|
|
if (timeCommandResult[0] == '1') {
|
|
ourQuality = RTCQualityNTP;
|
|
}
|
|
|
|
struct timeval tv;
|
|
tv.tv_sec = time(NULL);
|
|
tv.tv_usec = 0;
|
|
perhapsSetRTC(ourQuality, &tv);
|
|
#endif
|
|
|
|
powerMonInit();
|
|
serialSinceMsec = millis();
|
|
|
|
LOG_INFO("\n\n//\\ E S H T /\\ S T / C\n");
|
|
|
|
#if defined(ARCH_ESP32) && defined(BOARD_HAS_PSRAM)
|
|
#ifndef SENSECAP_INDICATOR
|
|
// use PSRAM for malloc calls > 2048 bytes
|
|
heap_caps_malloc_extmem_enable(2048);
|
|
#endif
|
|
#endif
|
|
|
|
#if defined(DEBUG_MUTE) && defined(DEBUG_PORT)
|
|
DEBUG_PORT.printf("\r\n\r\n//\\ E S H T /\\ S T / C\r\n");
|
|
DEBUG_PORT.printf("Version %s for %s from %s\r\n", optstr(APP_VERSION), optstr(APP_ENV), optstr(APP_REPO));
|
|
DEBUG_PORT.printf("Debug mute is enabled, there will be no serial output.\r\n");
|
|
#endif
|
|
|
|
initDeepSleep();
|
|
|
|
#if defined(MODEM_POWER_EN)
|
|
pinMode(MODEM_POWER_EN, OUTPUT);
|
|
digitalWrite(MODEM_POWER_EN, LOW);
|
|
#endif
|
|
|
|
#if defined(MODEM_PWRKEY)
|
|
pinMode(MODEM_PWRKEY, OUTPUT);
|
|
digitalWrite(MODEM_PWRKEY, LOW);
|
|
#endif
|
|
|
|
#if defined(LORA_TCXO_GPIO)
|
|
pinMode(LORA_TCXO_GPIO, OUTPUT);
|
|
digitalWrite(LORA_TCXO_GPIO, HIGH);
|
|
#endif
|
|
|
|
#if defined(VEXT_ENABLE)
|
|
pinMode(VEXT_ENABLE, OUTPUT);
|
|
digitalWrite(VEXT_ENABLE, VEXT_ON_VALUE); // turn on the display power
|
|
#endif
|
|
|
|
#if defined(BIAS_T_ENABLE)
|
|
pinMode(BIAS_T_ENABLE, OUTPUT);
|
|
digitalWrite(BIAS_T_ENABLE, BIAS_T_VALUE); // turn on 5V for GPS Antenna
|
|
#endif
|
|
|
|
#if defined(VTFT_CTRL)
|
|
pinMode(VTFT_CTRL, OUTPUT);
|
|
digitalWrite(VTFT_CTRL, LOW);
|
|
#endif
|
|
|
|
#ifdef RESET_OLED
|
|
pinMode(RESET_OLED, OUTPUT);
|
|
digitalWrite(RESET_OLED, 1);
|
|
delay(2);
|
|
digitalWrite(RESET_OLED, 0);
|
|
delay(10);
|
|
digitalWrite(RESET_OLED, 1);
|
|
#endif
|
|
|
|
#ifdef SENSOR_POWER_CTRL_PIN
|
|
pinMode(SENSOR_POWER_CTRL_PIN, OUTPUT);
|
|
digitalWrite(SENSOR_POWER_CTRL_PIN, SENSOR_POWER_ON);
|
|
#endif
|
|
|
|
#ifdef SENSOR_GPS_CONFLICT
|
|
bool sensor_detected = false;
|
|
#endif
|
|
#ifdef PERIPHERAL_WARMUP_MS
|
|
// Some peripherals may require additional time to stabilize after power is connected
|
|
// e.g. I2C on Heltec Vision Master
|
|
LOG_INFO("Wait for peripherals to stabilize");
|
|
delay(PERIPHERAL_WARMUP_MS);
|
|
#endif
|
|
initSPI();
|
|
|
|
OSThread::setup();
|
|
|
|
fsInit();
|
|
|
|
#if !MESHTASTIC_EXCLUDE_I2C
|
|
#if defined(I2C_SDA1) && defined(ARCH_RP2040)
|
|
Wire1.setSDA(I2C_SDA1);
|
|
Wire1.setSCL(I2C_SCL1);
|
|
Wire1.begin();
|
|
#elif defined(I2C_SDA1) && !defined(ARCH_RP2040)
|
|
Wire1.begin(I2C_SDA1, I2C_SCL1);
|
|
#elif WIRE_INTERFACES_COUNT == 2
|
|
Wire1.begin();
|
|
#endif
|
|
|
|
#if defined(I2C_SDA) && defined(ARCH_RP2040)
|
|
Wire.setSDA(I2C_SDA);
|
|
Wire.setSCL(I2C_SCL);
|
|
Wire.begin();
|
|
#elif defined(I2C_SDA) && !defined(ARCH_RP2040)
|
|
LOG_INFO("Starting Bus with (SDA) %d and (SCL) %d: ", I2C_SDA, I2C_SCL);
|
|
Wire.begin(I2C_SDA, I2C_SCL);
|
|
#elif defined(ARCH_PORTDUINO)
|
|
if (portduino_config.i2cdev != "") {
|
|
LOG_INFO("Use %s as I2C device", portduino_config.i2cdev.c_str());
|
|
Wire.begin(portduino_config.i2cdev.c_str());
|
|
} else {
|
|
LOG_INFO("No I2C device configured, Skip");
|
|
}
|
|
#elif HAS_WIRE
|
|
Wire.begin();
|
|
#endif
|
|
#endif
|
|
|
|
#if defined(M5STACK_UNITC6L)
|
|
pinMode(LORA_CS, OUTPUT);
|
|
digitalWrite(LORA_CS, 1);
|
|
c6l_init();
|
|
#endif
|
|
|
|
#ifdef PIN_LCD_RESET
|
|
// FIXME - move this someplace better, LCD is at address 0x3F
|
|
pinMode(PIN_LCD_RESET, OUTPUT);
|
|
digitalWrite(PIN_LCD_RESET, 0);
|
|
delay(1);
|
|
digitalWrite(PIN_LCD_RESET, 1);
|
|
delay(1);
|
|
#endif
|
|
|
|
#ifdef AQ_SET_PIN
|
|
// RAK-12039 set pin for Air quality sensor. Detectable on I2C after ~3 seconds, so we need to rescan later
|
|
pinMode(AQ_SET_PIN, OUTPUT);
|
|
digitalWrite(AQ_SET_PIN, HIGH);
|
|
#endif
|
|
|
|
// Currently only the tbeam has a PMU
|
|
// PMU initialization needs to be placed before i2c scanning
|
|
power = new Power();
|
|
power->setStatusHandler(powerStatus);
|
|
powerStatus->observe(&power->newStatus);
|
|
power->setup(); // Must be after status handler is installed, so that handler gets notified of the initial configuration
|
|
|
|
#if !MESHTASTIC_EXCLUDE_I2C
|
|
// We need to scan here to decide if we have a screen for nodeDB.init() and because power has been applied to
|
|
// accessories
|
|
auto i2cScanner = std::unique_ptr<ScanI2CTwoWire>(new ScanI2CTwoWire());
|
|
#if HAS_WIRE
|
|
LOG_INFO("Scan for i2c devices");
|
|
#endif
|
|
|
|
#if defined(I2C_SDA1) || (defined(NRF52840_XXAA) && (WIRE_INTERFACES_COUNT == 2))
|
|
i2cScanner->scanPort(ScanI2C::I2CPort::WIRE1);
|
|
#endif
|
|
|
|
#if defined(I2C_SDA)
|
|
i2cScanner->scanPort(ScanI2C::I2CPort::WIRE);
|
|
#elif defined(ARCH_PORTDUINO)
|
|
if (portduino_config.i2cdev != "") {
|
|
LOG_INFO("Scan for i2c devices");
|
|
i2cScanner->scanPort(ScanI2C::I2CPort::WIRE);
|
|
}
|
|
#elif HAS_WIRE
|
|
i2cScanner->scanPort(ScanI2C::I2CPort::WIRE);
|
|
#endif
|
|
|
|
auto i2cCount = i2cScanner->countDevices();
|
|
if (i2cCount == 0) {
|
|
LOG_INFO("No I2C devices found");
|
|
} else {
|
|
LOG_INFO("%i I2C devices found", i2cCount);
|
|
#ifdef SENSOR_GPS_CONFLICT
|
|
sensor_detected = true;
|
|
#endif
|
|
}
|
|
#ifdef ARCH_ESP32
|
|
#ifdef DEBUG_PARTITION_TABLE
|
|
printPartitionTable();
|
|
#endif
|
|
#endif // ARCH_ESP32
|
|
#ifdef ARCH_ESP32
|
|
// Don't init display if we don't have one or we are waking headless due to a timer event
|
|
if (wakeCause == ESP_SLEEP_WAKEUP_TIMER) {
|
|
LOG_DEBUG("suppress screen wake because this is a headless timer wakeup");
|
|
i2cScanner->setSuppressScreen();
|
|
}
|
|
#endif
|
|
|
|
#if HAS_SCREEN
|
|
auto screenInfo = i2cScanner->firstScreen();
|
|
screen_found = screenInfo.type != ScanI2C::DeviceType::NONE ? screenInfo.address : ScanI2C::ADDRESS_NONE;
|
|
|
|
if (screen_found.port != ScanI2C::I2CPort::NO_I2C) {
|
|
switch (screenInfo.type) {
|
|
case ScanI2C::DeviceType::SCREEN_SH1106:
|
|
screen_model = meshtastic_Config_DisplayConfig_OledType::meshtastic_Config_DisplayConfig_OledType_OLED_SH1106;
|
|
break;
|
|
case ScanI2C::DeviceType::SCREEN_SSD1306:
|
|
screen_model = meshtastic_Config_DisplayConfig_OledType::meshtastic_Config_DisplayConfig_OledType_OLED_SSD1306;
|
|
break;
|
|
case ScanI2C::DeviceType::SCREEN_ST7567:
|
|
case ScanI2C::DeviceType::SCREEN_UNKNOWN:
|
|
default:
|
|
screen_model = meshtastic_Config_DisplayConfig_OledType::meshtastic_Config_DisplayConfig_OledType_OLED_AUTO;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#define UPDATE_FROM_SCANNER(FIND_FN)
|
|
#if defined(USE_VIRTUAL_KEYBOARD)
|
|
kb_found = true;
|
|
#endif
|
|
auto rtc_info = i2cScanner->firstRTC();
|
|
rtc_found = rtc_info.type != ScanI2C::DeviceType::NONE ? rtc_info.address : rtc_found;
|
|
|
|
auto kb_info = i2cScanner->firstKeyboard();
|
|
|
|
if (kb_info.type != ScanI2C::DeviceType::NONE) {
|
|
kb_found = true;
|
|
cardkb_found = kb_info.address;
|
|
switch (kb_info.type) {
|
|
case ScanI2C::DeviceType::RAK14004:
|
|
kb_model = 0x02;
|
|
break;
|
|
case ScanI2C::DeviceType::CARDKB:
|
|
kb_model = 0x00;
|
|
break;
|
|
case ScanI2C::DeviceType::TDECKKB:
|
|
// assign an arbitrary value to distinguish from other models
|
|
kb_model = 0x10;
|
|
break;
|
|
case ScanI2C::DeviceType::BBQ10KB:
|
|
// assign an arbitrary value to distinguish from other models
|
|
kb_model = 0x11;
|
|
break;
|
|
case ScanI2C::DeviceType::MPR121KB:
|
|
// assign an arbitrary value to distinguish from other models
|
|
kb_model = 0x37;
|
|
break;
|
|
case ScanI2C::DeviceType::TCA8418KB:
|
|
// assign an arbitrary value to distinguish from other models
|
|
kb_model = 0x84;
|
|
break;
|
|
default:
|
|
// use this as default since it's also just zero
|
|
LOG_WARN("kb_info.type is unknown(0x%02x), setting kb_model=0x00", kb_info.type);
|
|
kb_model = 0x00;
|
|
}
|
|
}
|
|
|
|
pmu_found = i2cScanner->exists(ScanI2C::DeviceType::PMU_AXP192_AXP2101);
|
|
|
|
auto aqiInfo = i2cScanner->firstAQI();
|
|
aqi_found = aqiInfo.type != ScanI2C::DeviceType::NONE ? aqiInfo.address : ScanI2C::ADDRESS_NONE;
|
|
|
|
/*
|
|
* There are a bunch of sensors that have no further logic than to be found and stuffed into the
|
|
* nodeTelemetrySensorsMap singleton. This wraps that logic in a temporary scope to declare the temporary field
|
|
* "found".
|
|
*/
|
|
|
|
// Two supported RGB LED currently
|
|
#ifdef HAS_RGB_LED
|
|
rgb_found = i2cScanner->firstRGBLED();
|
|
#endif
|
|
|
|
#ifdef HAS_TPS65233
|
|
// TPS65233 is a power management IC for satellite modems, used in the Dreamcatcher
|
|
// We are switching it off here since we don't use an LNB.
|
|
if (i2cScanner->exists(ScanI2C::DeviceType::TPS65233)) {
|
|
Wire.beginTransmission(TPS65233_ADDR);
|
|
Wire.write(0); // Register 0
|
|
Wire.write(128); // Turn off the LNB power, keep I2C Control enabled
|
|
Wire.endTransmission();
|
|
Wire.beginTransmission(TPS65233_ADDR);
|
|
Wire.write(1); // Register 1
|
|
Wire.write(0); // Turn off Tone Generator 22kHz
|
|
Wire.endTransmission();
|
|
}
|
|
#endif
|
|
|
|
#if !defined(ARCH_STM32WL) && !MESHTASTIC_EXCLUDE_ACCELEROMETER
|
|
auto acc_info = i2cScanner->firstAccelerometer();
|
|
accelerometer_found = acc_info.type != ScanI2C::DeviceType::NONE ? acc_info.address : accelerometer_found;
|
|
LOG_DEBUG("acc_info = %i", acc_info.type);
|
|
#endif
|
|
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::INA260, meshtastic_TelemetrySensorType_INA260);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::INA226, meshtastic_TelemetrySensorType_INA226);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::INA219, meshtastic_TelemetrySensorType_INA219);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::INA3221, meshtastic_TelemetrySensorType_INA3221);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::MAX17048, meshtastic_TelemetrySensorType_MAX17048);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::QMC6310U, meshtastic_TelemetrySensorType_QMC6310);
|
|
// TODO: Types need to be added meshtastic_TelemetrySensorType_QMC6310N
|
|
// scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::QMC6310N, meshtastic_TelemetrySensorType_QMC6310N);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::QMI8658, meshtastic_TelemetrySensorType_QMI8658);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::QMC5883L, meshtastic_TelemetrySensorType_QMC5883L);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::HMC5883L, meshtastic_TelemetrySensorType_QMC5883L);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::MLX90614, meshtastic_TelemetrySensorType_MLX90614);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::ICM20948, meshtastic_TelemetrySensorType_ICM20948);
|
|
scannerToSensorsMap(i2cScanner, ScanI2C::DeviceType::MAX30102, meshtastic_TelemetrySensorType_MAX30102);
|
|
#endif
|
|
|
|
#ifdef HAS_SDCARD
|
|
setupSDCard();
|
|
#endif
|
|
|
|
// Hello
|
|
printInfo();
|
|
#ifdef BUILD_EPOCH
|
|
LOG_INFO("Build timestamp: %ld", BUILD_EPOCH);
|
|
#endif
|
|
|
|
#ifdef ARCH_ESP32
|
|
esp32Setup();
|
|
#endif
|
|
|
|
#ifdef ARCH_NRF52
|
|
nrf52Setup();
|
|
#endif
|
|
#ifdef ARCH_NRF54L15
|
|
nrf54l15Setup();
|
|
#endif
|
|
|
|
#ifdef ARCH_RP2040
|
|
rp2040Setup();
|
|
#endif
|
|
|
|
// We do this as early as possible because this loads preferences from flash
|
|
// but we need to do this after main cpu init (esp32setup), because we need the random seed set
|
|
nodeDB = new NodeDB;
|
|
|
|
// Initialize transmit history to persist broadcast throttle timers across reboots
|
|
TransmitHistory::getInstance()->loadFromDisk();
|
|
#if HAS_TFT
|
|
if (config.display.displaymode == meshtastic_Config_DisplayConfig_DisplayMode_COLOR) {
|
|
tftSetup();
|
|
}
|
|
#endif
|
|
|
|
router = new ReliableRouter();
|
|
|
|
// only play start melody when role is not tracker or sensor
|
|
if (config.power.is_power_saving == true &&
|
|
IS_ONE_OF(config.device.role, meshtastic_Config_DeviceConfig_Role_TRACKER,
|
|
meshtastic_Config_DeviceConfig_Role_TAK_TRACKER, meshtastic_Config_DeviceConfig_Role_SENSOR))
|
|
LOG_DEBUG("Tracker/Sensor: Skip start melody");
|
|
else
|
|
playStartMelody();
|
|
|
|
#if HAS_SCREEN
|
|
// fixed screen override?
|
|
#if defined(USE_SH1107)
|
|
screen_model = meshtastic_Config_DisplayConfig_OledType_OLED_SH1107; // set dimension of 128x128
|
|
screen_geometry = GEOMETRY_128_128;
|
|
#elif defined(USE_SH1107_128_64)
|
|
screen_model = meshtastic_Config_DisplayConfig_OledType_OLED_SH1107; // keep dimension of 128x64
|
|
#else
|
|
if (config.display.oled != meshtastic_Config_DisplayConfig_OledType_OLED_AUTO) {
|
|
screen_model = config.display.oled;
|
|
|
|
// Fix: update geometry for SH1107 128x128 selected via menu
|
|
if (screen_model == meshtastic_Config_DisplayConfig_OledType_OLED_SH1107_128_128) {
|
|
screen_geometry = GEOMETRY_128_128;
|
|
screen_model = meshtastic_Config_DisplayConfig_OledType_OLED_SH1107; // normalize
|
|
}
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
#if !MESHTASTIC_EXCLUDE_I2C
|
|
#if !defined(ARCH_STM32WL) && !MESHTASTIC_EXCLUDE_ACCELEROMETER
|
|
if (acc_info.type != ScanI2C::DeviceType::NONE) {
|
|
accelerometerThread = new AccelerometerThread(acc_info.type);
|
|
}
|
|
#endif
|
|
|
|
#if defined(HAS_NEOPIXEL) || defined(UNPHONE) || defined(RGBLED_RED)
|
|
ambientLightingThread = new AmbientLightingThread(ScanI2C::DeviceType::NONE);
|
|
#elif !defined(ARCH_PORTDUINO) && !defined(ARCH_STM32WL)
|
|
if (rgb_found.type != ScanI2C::DeviceType::NONE) {
|
|
ambientLightingThread = new AmbientLightingThread(rgb_found.type);
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
#ifdef HAS_DRV2605
|
|
#if defined(PIN_DRV_EN)
|
|
pinMode(PIN_DRV_EN, OUTPUT);
|
|
digitalWrite(PIN_DRV_EN, HIGH);
|
|
delay(10);
|
|
#endif
|
|
drv.begin();
|
|
drv.selectLibrary(1);
|
|
// I2C trigger by sending 'go' command
|
|
drv.setMode(DRV2605_MODE_INTTRIG);
|
|
#endif
|
|
|
|
// Init our SPI controller (must be before screen and lora)
|
|
#ifdef ARCH_RP2040
|
|
#ifdef HW_SPI1_DEVICE
|
|
SPI1.setSCK(LORA_SCK);
|
|
SPI1.setTX(LORA_MOSI);
|
|
SPI1.setRX(LORA_MISO);
|
|
pinMode(LORA_CS, OUTPUT);
|
|
digitalWrite(LORA_CS, HIGH);
|
|
SPI1.begin(false);
|
|
#else // HW_SPI1_DEVICE
|
|
SPI.setSCK(LORA_SCK);
|
|
SPI.setTX(LORA_MOSI);
|
|
SPI.setRX(LORA_MISO);
|
|
SPI.begin(false);
|
|
#endif // HW_SPI1_DEVICE
|
|
#elif ARCH_PORTDUINO
|
|
if (portduino_config.lora_spi_dev != "ch341") {
|
|
SPI.begin();
|
|
}
|
|
#elif !defined(ARCH_ESP32) // ARCH_RP2040
|
|
#if defined(RAK3401) || defined(RAK13302)
|
|
pinMode(WB_IO2, OUTPUT);
|
|
digitalWrite(WB_IO2, HIGH);
|
|
SPI1.setPins(LORA_MISO, LORA_SCK, LORA_MOSI);
|
|
SPI1.begin();
|
|
#else
|
|
SPI.begin();
|
|
#endif
|
|
#else
|
|
// ESP32
|
|
#if defined(HW_SPI1_DEVICE)
|
|
SPI1.begin(LORA_SCK, LORA_MISO, LORA_MOSI, LORA_CS);
|
|
LOG_DEBUG("SPI1.begin(SCK=%d, MISO=%d, MOSI=%d, NSS=%d)", LORA_SCK, LORA_MISO, LORA_MOSI, LORA_CS);
|
|
SPI1.setFrequency(4000000);
|
|
#else
|
|
SPI.begin(LORA_SCK, LORA_MISO, LORA_MOSI, LORA_CS);
|
|
LOG_DEBUG("SPI.begin(SCK=%d, MISO=%d, MOSI=%d, NSS=%d)", LORA_SCK, LORA_MISO, LORA_MOSI, LORA_CS);
|
|
SPI.setFrequency(4000000);
|
|
#endif
|
|
#endif
|
|
|
|
// Initialize the screen first so we can show the logo while we start up everything else.
|
|
#if HAS_SCREEN
|
|
if (config.display.displaymode != meshtastic_Config_DisplayConfig_DisplayMode_COLOR) {
|
|
|
|
#if defined(ST7701_CS) || defined(ST7735_CS) || defined(USE_EINK) || defined(ILI9341_DRIVER) || defined(ILI9342_DRIVER) || \
|
|
defined(ST7789_CS) || defined(HX8357_CS) || defined(USE_ST7789) || defined(ILI9488_CS) || defined(ST7796_CS) || \
|
|
defined(USE_SPISSD1306) || defined(USE_ST7796) || defined(HACKADAY_COMMUNICATOR)
|
|
screen = new graphics::Screen(screen_found, screen_model, screen_geometry);
|
|
#elif defined(ARCH_PORTDUINO)
|
|
if ((screen_found.port != ScanI2C::I2CPort::NO_I2C || portduino_config.displayPanel) &&
|
|
config.display.displaymode != meshtastic_Config_DisplayConfig_DisplayMode_COLOR) {
|
|
screen = new graphics::Screen(screen_found, screen_model, screen_geometry);
|
|
}
|
|
#else
|
|
if (screen_found.port != ScanI2C::I2CPort::NO_I2C)
|
|
screen = new graphics::Screen(screen_found, screen_model, screen_geometry);
|
|
#endif
|
|
}
|
|
#endif // HAS_SCREEN
|
|
|
|
// TODO Remove magic string
|
|
// setup TZ prior to time actions.
|
|
#if !MESHTASTIC_EXCLUDE_TZ
|
|
LOG_DEBUG("Use compiled/slipstreamed %s", slipstreamTZString); // important, removing this clobbers our magic string
|
|
if (*config.device.tzdef && config.device.tzdef[0] != 0) {
|
|
LOG_DEBUG("Saved TZ: %s ", config.device.tzdef);
|
|
setenv("TZ", config.device.tzdef, 1);
|
|
} else {
|
|
if (strncmp((const char *)slipstreamTZString, "tzpl", 4) == 0) {
|
|
setenv("TZ", "GMT0", 1);
|
|
} else {
|
|
setenv("TZ", (const char *)slipstreamTZString, 1);
|
|
strcpy(config.device.tzdef, (const char *)slipstreamTZString);
|
|
}
|
|
}
|
|
tzset();
|
|
LOG_DEBUG("Set Timezone to %s", getenv("TZ"));
|
|
#endif
|
|
|
|
readFromRTC(); // read the main CPU RTC at first (in case we can't get GPS time)
|
|
|
|
#if !MESHTASTIC_EXCLUDE_GPS
|
|
// If we're taking on the repeater role, ignore GPS
|
|
#ifdef SENSOR_GPS_CONFLICT
|
|
if (sensor_detected == false) {
|
|
#endif
|
|
if (HAS_GPS) {
|
|
if (config.position.gps_mode != meshtastic_Config_PositionConfig_GpsMode_NOT_PRESENT) {
|
|
gps = GPS::createGps();
|
|
if (gps) {
|
|
gpsStatus->observe(&gps->newStatus);
|
|
} else {
|
|
LOG_DEBUG("Run without GPS");
|
|
}
|
|
}
|
|
}
|
|
#ifdef SENSOR_GPS_CONFLICT
|
|
}
|
|
#endif
|
|
|
|
#endif
|
|
|
|
nodeStatus->observe(&nodeDB->newStatus);
|
|
|
|
#ifdef HAS_I2S
|
|
LOG_DEBUG("Start audio thread");
|
|
audioThread = new AudioThread();
|
|
#endif
|
|
|
|
#ifdef HAS_UDP_MULTICAST
|
|
LOG_DEBUG("Start multicast thread");
|
|
udpHandler = new UdpMulticastHandler();
|
|
#ifdef ARCH_PORTDUINO
|
|
// FIXME: portduino does not ever call onNetworkConnected so call it here because I don't know what happen if I call
|
|
// onNetworkConnected there
|
|
if (config.network.enabled_protocols & meshtastic_Config_NetworkConfig_ProtocolFlags_UDP_BROADCAST) {
|
|
udpHandler->start();
|
|
}
|
|
#endif
|
|
#endif
|
|
service = new MeshService();
|
|
service->init();
|
|
|
|
// Set osk_found for trackball/encoder devices BEFORE setupModules so CannedMessageModule can detect it
|
|
#if defined(HAS_TRACKBALL) || (defined(INPUTDRIVER_ENCODER_TYPE) && INPUTDRIVER_ENCODER_TYPE == 2)
|
|
#ifndef HAS_PHYSICAL_KEYBOARD
|
|
osk_found = true;
|
|
#endif
|
|
#endif
|
|
|
|
// Now that the mesh service is created, create any modules
|
|
setupModules();
|
|
|
|
#if !MESHTASTIC_EXCLUDE_I2C
|
|
// Inform modules about I2C devices
|
|
ScanI2CCompleted(i2cScanner.get());
|
|
i2cScanner.reset();
|
|
#endif
|
|
|
|
#if !defined(MESHTASTIC_EXCLUDE_PKI)
|
|
// warn the user about a low entropy key
|
|
if (nodeDB->keyIsLowEntropy && !nodeDB->hasWarned) {
|
|
LOG_WARN(LOW_ENTROPY_WARNING);
|
|
meshtastic_ClientNotification *cn = clientNotificationPool.allocZeroed();
|
|
cn->level = meshtastic_LogRecord_Level_WARNING;
|
|
cn->time = getValidTime(RTCQualityFromNet);
|
|
sprintf(cn->message, LOW_ENTROPY_WARNING);
|
|
service->sendClientNotification(cn);
|
|
nodeDB->hasWarned = true;
|
|
}
|
|
#endif
|
|
#if !MESHTASTIC_EXCLUDE_INPUTBROKER
|
|
if (inputBroker)
|
|
inputBroker->Init();
|
|
#endif
|
|
|
|
#ifdef MESHTASTIC_INCLUDE_NICHE_GRAPHICS
|
|
// After modules are setup, so we can observe modules
|
|
setupNicheGraphics();
|
|
#endif
|
|
|
|
// Do this after service.init (because that clears error_code)
|
|
#ifdef HAS_PMU
|
|
if (!pmu_found)
|
|
RECORD_CRITICALERROR(meshtastic_CriticalErrorCode_NO_AXP192); // Record a hardware fault for missing hardware
|
|
#endif
|
|
|
|
#if !MESHTASTIC_EXCLUDE_I2C
|
|
// Don't call screen setup until after nodedb is setup (because we need
|
|
// the current region name)
|
|
#if defined(ST7701_CS) || defined(ST7735_CS) || defined(USE_EINK) || defined(ILI9341_DRIVER) || defined(ILI9342_DRIVER) || \
|
|
defined(ST7789_CS) || defined(HX8357_CS) || defined(USE_ST7789) || defined(ILI9488_CS) || defined(ST7796_CS) || \
|
|
defined(USE_ST7796) || defined(USE_SPISSD1306) || defined(HACKADAY_COMMUNICATOR)
|
|
if (screen)
|
|
screen->setup();
|
|
#elif defined(ARCH_PORTDUINO)
|
|
if ((screen_found.port != ScanI2C::I2CPort::NO_I2C || portduino_config.displayPanel) &&
|
|
config.display.displaymode != meshtastic_Config_DisplayConfig_DisplayMode_COLOR) {
|
|
screen->setup();
|
|
}
|
|
#else
|
|
if (screen_found.port != ScanI2C::I2CPort::NO_I2C && screen)
|
|
screen->setup();
|
|
#endif
|
|
#endif
|
|
|
|
auto rIf = initLoRa();
|
|
|
|
lateInitVariant(); // Do board specific init (see extra_variants/README.md for documentation)
|
|
|
|
#if !MESHTASTIC_EXCLUDE_MQTT
|
|
mqttInit();
|
|
#endif
|
|
|
|
#ifdef RF95_FAN_EN
|
|
// Ability to disable FAN if PIN has been set with RF95_FAN_EN.
|
|
// Make sure LoRa has been started before disabling FAN.
|
|
if (config.lora.pa_fan_disabled)
|
|
digitalWrite(RF95_FAN_EN, LOW ^ 0);
|
|
#endif
|
|
|
|
#ifndef ARCH_PORTDUINO
|
|
|
|
// Initialize Wifi
|
|
#if HAS_WIFI
|
|
initWifi();
|
|
#endif
|
|
|
|
#if HAS_ETHERNET
|
|
// Initialize Ethernet
|
|
initEthernet();
|
|
#endif
|
|
#endif
|
|
|
|
#if defined(ARCH_ESP32) && !MESHTASTIC_EXCLUDE_WEBSERVER
|
|
// Start web server thread.
|
|
webServerThread = new WebServerThread();
|
|
#endif
|
|
|
|
#ifdef ARCH_PORTDUINO
|
|
#if __has_include(<ulfius.h>)
|
|
if (portduino_config.webserverport != -1) {
|
|
piwebServerThread = new PiWebServerThread();
|
|
std::atexit([] { delete piwebServerThread; });
|
|
}
|
|
#endif
|
|
initApiServer(TCPPort);
|
|
#endif
|
|
|
|
// Start airtime logger thread.
|
|
airTime = new AirTime();
|
|
|
|
if (!rIf)
|
|
RECORD_CRITICALERROR(meshtastic_CriticalErrorCode_NO_RADIO);
|
|
else {
|
|
// Log bit rate to debug output
|
|
LOG_DEBUG("LoRA bitrate = %f bytes / sec", (float(meshtastic_Constants_DATA_PAYLOAD_LEN) /
|
|
(float(rIf->getPacketTime(meshtastic_Constants_DATA_PAYLOAD_LEN)))) *
|
|
1000);
|
|
|
|
router->addInterface(std::move(rIf));
|
|
}
|
|
|
|
// This must be _after_ service.init because we need our preferences loaded from flash to have proper timeout values
|
|
PowerFSM_setup(); // we will transition to ON in a couple of seconds, FIXME, only do this for cold boots, not waking from SDS
|
|
powerFSMthread = new PowerFSMThread();
|
|
|
|
#if !HAS_TFT
|
|
setCPUFast(false); // 80MHz is fine for our slow peripherals
|
|
#endif
|
|
|
|
#ifdef ARDUINO_ARCH_ESP32
|
|
LOG_DEBUG("Free heap : %7d bytes", ESP.getFreeHeap());
|
|
LOG_DEBUG("Free PSRAM : %7d bytes", ESP.getFreePsram());
|
|
#endif
|
|
|
|
// We manually run this to update the NodeStatus
|
|
nodeDB->notifyObservers(true);
|
|
}
|
|
|
|
#endif
|
|
uint32_t rebootAtMsec; // If not zero we will reboot at this time (used to reboot shortly after the update completes)
|
|
uint32_t shutdownAtMsec; // If not zero we will shutdown at this time (used to shutdown from python or mobile client)
|
|
bool suppressRebootBanner; // If true, suppress "Rebooting..." overlay (used for OTA handoff)
|
|
|
|
// If a thread does something that might need for it to be rescheduled ASAP it can set this flag
|
|
// This will suppress the current delay and instead try to run ASAP.
|
|
bool runASAP;
|
|
|
|
// TODO find better home than main.cpp
|
|
extern meshtastic_DeviceMetadata getDeviceMetadata()
|
|
{
|
|
meshtastic_DeviceMetadata deviceMetadata;
|
|
strncpy(deviceMetadata.firmware_version, optstr(APP_VERSION), sizeof(deviceMetadata.firmware_version));
|
|
deviceMetadata.device_state_version = DEVICESTATE_CUR_VER;
|
|
deviceMetadata.canShutdown = pmu_found || HAS_CPU_SHUTDOWN;
|
|
deviceMetadata.hasBluetooth = HAS_BLUETOOTH;
|
|
deviceMetadata.hasWifi = HAS_WIFI;
|
|
deviceMetadata.hasEthernet = HAS_ETHERNET;
|
|
deviceMetadata.role = config.device.role;
|
|
deviceMetadata.position_flags = config.position.position_flags;
|
|
deviceMetadata.hw_model = HW_VENDOR;
|
|
deviceMetadata.hasRemoteHardware = moduleConfig.remote_hardware.enabled;
|
|
deviceMetadata.excluded_modules = meshtastic_ExcludedModules_EXCLUDED_NONE;
|
|
#if MESHTASTIC_EXCLUDE_REMOTEHARDWARE
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_REMOTEHARDWARE_CONFIG;
|
|
#endif
|
|
#if MESHTASTIC_EXCLUDE_AUDIO
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_AUDIO_CONFIG;
|
|
#endif
|
|
// Option to explicitly include canned messages for edge cases, e.g. niche graphics
|
|
#if ((!HAS_SCREEN || NO_EXT_GPIO) || MESHTASTIC_EXCLUDE_CANNEDMESSAGES) && !defined(MESHTASTIC_INCLUDE_NICHE_GRAPHICS)
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_CANNEDMSG_CONFIG;
|
|
#endif
|
|
#if NO_EXT_GPIO || MESHTASTIC_EXCLUDE_EXTERNALNOTIFICATION
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_EXTNOTIF_CONFIG;
|
|
#endif
|
|
// Only edge case here is if we apply this a device with built in Accelerometer and want to detect interrupts
|
|
// We'll have to macro guard against those targets potentially
|
|
#if NO_EXT_GPIO || MESHTASTIC_EXCLUDE_DETECTIONSENSOR
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_DETECTIONSENSOR_CONFIG;
|
|
#endif
|
|
// If we don't have any GPIO and we don't have GPS OR we don't want too - no purpose in having serial config
|
|
#if NO_EXT_GPIO && NO_GPS || MESHTASTIC_EXCLUDE_SERIAL
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_SERIAL_CONFIG;
|
|
#endif
|
|
#ifndef ARCH_ESP32
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_PAXCOUNTER_CONFIG;
|
|
#endif
|
|
#if !defined(HAS_RGB_LED) && !RAK_4631
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_AMBIENTLIGHTING_CONFIG;
|
|
#endif
|
|
|
|
// No bluetooth on these targets (yet):
|
|
// Pico W / 2W may get it at some point
|
|
// Portduino and ESP32-C6 are excluded because we don't have a working bluetooth stacks integrated yet.
|
|
#if defined(ARCH_RP2040) || defined(ARCH_PORTDUINO) || defined(ARCH_STM32WL) || defined(CONFIG_IDF_TARGET_ESP32C6)
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_BLUETOOTH_CONFIG;
|
|
#endif
|
|
|
|
#if defined(ARCH_NRF52) && !HAS_ETHERNET // nrf52 doesn't have network unless it's a RAK ethernet gateway currently
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_NETWORK_CONFIG; // No network on nRF52
|
|
#elif defined(ARCH_RP2040) && !HAS_WIFI && !HAS_ETHERNET
|
|
deviceMetadata.excluded_modules |= meshtastic_ExcludedModules_NETWORK_CONFIG; // No network on RP2040
|
|
#endif
|
|
|
|
#if !(MESHTASTIC_EXCLUDE_PKI)
|
|
deviceMetadata.hasPKC = true;
|
|
#endif
|
|
return deviceMetadata;
|
|
}
|
|
|
|
#if !MESHTASTIC_EXCLUDE_I2C
|
|
void scannerToSensorsMap(const std::unique_ptr<ScanI2CTwoWire> &i2cScanner, ScanI2C::DeviceType deviceType,
|
|
meshtastic_TelemetrySensorType sensorType)
|
|
{
|
|
auto found = i2cScanner->find(deviceType);
|
|
if (found.type != ScanI2C::DeviceType::NONE) {
|
|
nodeTelemetrySensorsMap[sensorType].first = found.address.address;
|
|
nodeTelemetrySensorsMap[sensorType].second = i2cScanner->fetchI2CBus(found.address);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifndef PIO_UNIT_TESTING
|
|
void loop()
|
|
{
|
|
runASAP = false;
|
|
|
|
#ifdef ARCH_ESP32
|
|
esp32Loop();
|
|
#endif
|
|
#ifdef ARCH_NRF52
|
|
nrf52Loop();
|
|
#endif
|
|
#ifdef ARCH_NRF54L15
|
|
nrf54l15Loop();
|
|
#endif
|
|
power->powerCommandsCheck();
|
|
|
|
if (RadioLibInterface::instance != nullptr) {
|
|
static uint32_t lastRadioMissedIrqPoll;
|
|
if (!Throttle::isWithinTimespanMs(lastRadioMissedIrqPoll, 1000)) {
|
|
lastRadioMissedIrqPoll = millis();
|
|
RadioLibInterface::instance->pollMissedIrqs();
|
|
}
|
|
|
|
// Periodic AGC reset — warm sleep + recalibrate to prevent stuck AGC gain
|
|
static uint32_t lastAgcReset;
|
|
if (!Throttle::isWithinTimespanMs(lastAgcReset, AGC_RESET_INTERVAL_MS)) {
|
|
lastAgcReset = millis();
|
|
RadioLibInterface::instance->resetAGC();
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG_STACK
|
|
static uint32_t lastPrint = 0;
|
|
if (!Throttle::isWithinTimespanMs(lastPrint, 10 * 1000L)) {
|
|
lastPrint = millis();
|
|
meshtastic::printThreadInfo("main");
|
|
}
|
|
#endif
|
|
|
|
service->loop();
|
|
#if !MESHTASTIC_EXCLUDE_INPUTBROKER && defined(HAS_FREE_RTOS) && !defined(ARCH_RP2040)
|
|
if (inputBroker)
|
|
inputBroker->processInputEventQueue();
|
|
#endif
|
|
#if ARCH_PORTDUINO
|
|
if (portduino_config.lora_spi_dev == "ch341" && ch341Hal != nullptr) {
|
|
ch341Hal->checkError();
|
|
}
|
|
if (portduino_status.LoRa_in_error && rebootAtMsec == 0) {
|
|
LOG_ERROR("LoRa in error detected, attempting to recover");
|
|
router->addInterface(nullptr);
|
|
if (portduino_config.lora_spi_dev == "ch341") {
|
|
if (ch341Hal != nullptr) {
|
|
delete ch341Hal;
|
|
ch341Hal = nullptr;
|
|
sleep(3);
|
|
}
|
|
try {
|
|
ch341Hal = new Ch341Hal(0, portduino_config.lora_usb_serial_num, portduino_config.lora_usb_vid,
|
|
portduino_config.lora_usb_pid);
|
|
} catch (std::exception &e) {
|
|
std::cerr << e.what() << std::endl;
|
|
std::cerr << "Could not initialize CH341 device!" << std::endl;
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
}
|
|
auto rIf = initLoRa();
|
|
if (rIf) {
|
|
router->addInterface(std::move(rIf));
|
|
portduino_status.LoRa_in_error = false;
|
|
} else {
|
|
LOG_WARN("Reconfigure failed, rebooting");
|
|
if (screen) {
|
|
screen->showSimpleBanner("Rebooting...");
|
|
}
|
|
rebootAtMsec = millis() + 25;
|
|
}
|
|
}
|
|
#if HAS_TFT
|
|
if (screen && portduino_config.displayPanel == x11 &&
|
|
config.display.displaymode != meshtastic_Config_DisplayConfig_DisplayMode_COLOR) {
|
|
auto dispdev = screen->getDisplayDevice();
|
|
if (dispdev)
|
|
static_cast<TFTDisplay *>(dispdev)->sdlLoop();
|
|
}
|
|
#endif
|
|
#endif
|
|
#if HAS_SCREEN && ENABLE_MESSAGE_PERSISTENCE
|
|
messageStoreAutosaveTick();
|
|
#endif
|
|
long delayMsec = mainController.runOrDelay();
|
|
|
|
// We want to sleep as long as possible here - because it saves power
|
|
if (!runASAP && loopCanSleep()) {
|
|
#ifdef DEBUG_LOOP_TIMING
|
|
LOG_DEBUG("main loop delay: %d", delayMsec);
|
|
#endif
|
|
mainDelay.delay(delayMsec);
|
|
}
|
|
}
|
|
#endif
|