Files
ratdeck/docs/ARCHITECTURE.md
2026-03-07 17:38:45 -07:00

200 lines
9.5 KiB
Markdown

# Ratdeck — Architecture
## Overview
Ratdeck is a standalone Reticulum mesh node with LXMF encrypted messaging, built for the LilyGo T-Deck Plus (ESP32-S3). It runs the full Reticulum protocol stack on-device — no host computer, no gateway, no KISS protocol. LoRa, WiFi, and BLE transports operate independently or bridged.
## Layer Diagram
```
+---------------------------------------------+
| UI Layer (LVGL v8.4) |
| Screens: Home, Friends, Msgs, Peers, Setup |
| LvStatusBar, LvTabBar, LvInput |
| Theme: Matrix green (#00FF41) on black |
+---------------------------------------------+
| Application Layer |
| LXMFManager AnnounceManager |
| IdentityManager UserConfig |
| MessageStore AudioNotify |
+---------------------------------------------+
| Reticulum Layer |
| ReticulumManager (microReticulum C++) |
| Identity, Destination, Transport |
+---------------------------------------------+
| Transport Layer |
| LoRaInterface WiFiInterface |
| TCPClientInterface BLESideband |
+---------------------------------------------+
| Storage Layer |
| FlashStore (LittleFS, 7.875 MB) |
| SDStore (FAT32 microSD) |
| NVS (ESP32 Preferences) |
+---------------------------------------------+
| Hardware Layer |
| SX1262 Radio ST7789V Display |
| I2C Keyboard GPIO Trackball |
| ESP32-S3 (8MB PSRAM, 16MB Flash) |
+---------------------------------------------+
```
## Directory Structure
```
src/
+-- main.cpp Setup + main loop (single-threaded, core 1)
+-- config/
| +-- BoardConfig.h GPIO pins, SPI config, hardware constants
| +-- Config.h Version, feature flags, storage paths, limits
| +-- UserConfig.* Runtime settings (JSON, dual SD+flash backend)
+-- radio/
| +-- SX1262.* SX1262 LoRa driver (register-level, async TX)
| +-- RadioConstants.h SX1262 register addresses and command bytes
+-- hal/
| +-- Display.* LovyanGFX ST7789V driver + LVGL flush callback
| +-- Power.* Screen dim/off/wake, battery ADC, brightness
| +-- Keyboard.* I2C keyboard (ESP32-C3 at 0x55)
| +-- Trackball.* GPIO trackball (ISR-based, 5 pins)
| +-- TouchInput.* GT911 capacitive touch (disabled)
+-- input/
| +-- InputManager.* Unified keyboard + trackball + touch polling
| +-- HotkeyManager.* Ctrl+key dispatch table
+-- ui/
| +-- Theme.h Color palette, layout constants
| +-- LvTheme.* LVGL theme application
| +-- LvStatusBar.* Signal bars + brand + battery %
| +-- LvTabBar.* 5-tab bar with unread badges
| +-- LvInput.* Key event routing to active screen
| +-- UIManager.* Screen lifecycle, boot/normal modes
| +-- screens/
| +-- LvBootScreen.* Boot animation with progress bar
| +-- LvHomeScreen.* Name, LXMF address, status, online nodes
| +-- LvContactsScreen.* Saved friends (display name only)
| +-- LvMessagesScreen.* Conversation list (sorted, previews, unread dots)
| +-- LvMessageView.* Chat bubbles + status colors + text input
| +-- LvNodesScreen.* Node browser (contacts + online sections)
| +-- LvSettingsScreen.* 7-category settings editor
| +-- LvHelpOverlay.* Hotkey reference modal
| +-- LvNameInputScreen.* First-boot name entry
+-- reticulum/
| +-- ReticulumManager.* microReticulum lifecycle, transport loop
| +-- AnnounceManager.* Node discovery, friend persistence
| +-- LXMFManager.* LXMF send/receive, queue, delivery tracking
| +-- LXMFMessage.* Wire format + JSON storage format
| +-- IdentityManager.* Multi-slot identity management
+-- transport/
| +-- LoRaInterface.* SX1262 <-> Reticulum (1-byte header)
| +-- WiFiInterface.* WiFi AP, TCP server on port 4242
| +-- TCPClientInterface.* WiFi STA, TCP client to remote nodes
| +-- BLEInterface.* BLE transport
| +-- BLESideband.* NimBLE Sideband interface
+-- storage/
| +-- FlashStore.* LittleFS with atomic writes
| +-- SDStore.* SD card (FAT32) with atomic writes
| +-- MessageStore.* Per-conversation dual-backend storage
+-- audio/
+-- AudioNotify.* Notification sounds
```
## Data Flow
### Incoming LoRa Packet
```
SX1262 IRQ (DIO1, GPIO 45) -> SX1262::receive() reads FIFO
-> LoRaInterface::loop() strips 1-byte header
-> RNS::InterfaceImpl::receive_incoming()
-> RNS::Transport processes packet
+-- Announce -> AnnounceManager callback -> UI update
+-- LXMF data -> LXMFManager -> MessageStore (flash + SD) -> UI notification
+-- Path/link -> Transport table update -> persist to flash + SD backup
```
### Outgoing LXMF Message
```
User types message -> LvMessageView -> LXMFManager::send()
-> Save to MessageStore (QUEUED status)
-> Pack: dest_hash(16) + src_hash(16) + signature(64) + msgpack([ts, title, content, fields])
-> RNS::Packet -> RNS::Transport selects interface
+-- LoRaInterface -> prepend 1-byte header -> SX1262::beginPacket/endPacket (async)
+-- TCPClient -> HDLC frame (0x7E delimit, 0x7D escape) -> TCP socket
-> StatusCallback fires -> re-save with SENT/DELIVERED/FAILED status
```
### Config Save
```
LvSettingsScreen -> UserConfig::save(sd, flash)
+-- serialize to JSON string
+-- SDStore::writeAtomic("/ratputer/config/user.json") -> .tmp -> verify -> .bak -> rename
+-- FlashStore::writeAtomic("/config/user.json") -> .tmp -> verify -> .bak -> rename
```
## Key Design Decisions
### Radio Driver
Register-level SX1262 driver with no library abstraction. Async TX mode (`endPacket(true)`) returns immediately — `isTxBusy()` polls completion in `LoRaInterface::loop()`. `waitOnBusy()` calls a yield callback so LVGL stays responsive during SPI waits. TCXO 1.8V, DIO2 as RF switch, 8 MHz SPI clock.
### Display
LovyanGFX drives the ST7789V over SPI at 15 MHz. LVGL v8.4 renders to a flush buffer with DMA transfer. 320x240 landscape, 20px status bar top, 20px tab bar bottom, 200px content area.
### Reticulum Integration
microReticulum C++ library with LittleFS-backed filesystem. Device runs as either an Endpoint (default) or Transport Node (configurable in Settings). All interfaces (LoRa, WiFi, TCP, BLE) register with `RNS::Transport`.
### LXMF Wire Format
```
[dest_hash:16][src_hash:16][signature:64][msgpack_payload]
```
MsgPack payload: `fixarray(4): [timestamp(float64), title(str), content(str), fields(map)]`
Signature covers: `dest_hash || src_hash || msgpack_payload`
Messages under MDU (~254 bytes for LoRa) are sent as single direct packets. Stored as JSON per-conversation in flash and SD.
### Shared SPI Bus
Display, SX1262 radio, and SD card all share SPI2_HOST (FSPI on ESP32-S3). Single-threaded access from `loop()` — no mutex needed. Arduino FSPI=0 maps to SPI2 hardware; do NOT use `SPI2_HOST` IDF constant directly.
### WiFi Transport
Three mutually exclusive modes:
- **OFF**: No WiFi — saves power and ~20 KB heap
- **AP**: Creates `ratdeck-XXXX` hotspot, TCP server on port 4242, HDLC framing
- **STA**: Connects to existing network, TCP client connections to configured endpoints
AP+STA concurrent mode was removed — consumed too much heap and caused instability.
### TCP Client Transport
Outbound TCP connections to remote Reticulum nodes. Created on WiFi STA connection, auto-reconnect on disconnect (15s interval). Settings changes apply live via `_tcpChangeCb` callback — `reloadTCPClients()` stops old clients and creates new ones. Switching servers clears transient nodes.
### Dual-Backend Storage
FlashStore (LittleFS, 7.875 MB partition) is the primary store. SDStore (FAT32 microSD) provides backup and extended capacity. Both use atomic writes (`.tmp` -> verify -> `.bak` -> rename) to prevent corruption on power loss. UserConfig, MessageStore, and AnnounceManager write to both backends.
### Identity System
Triple-redundant storage: Flash (LittleFS), SD card, and NVS (ESP32 Preferences). Load order: Flash -> SD -> NVS -> generate new. On any successful load or creation, saves to all three tiers. Multi-slot identity management with per-slot display names.
### Transport Reference Stability
`RNS::Transport::_interfaces` stores `Interface&` references (not copies). All `RNS::Interface` wrappers must outlive the transport — stored in `std::list` (not vector, which would invalidate references on reallocation).
### Power Management
Three states: ACTIVE -> DIMMED -> SCREEN_OFF. Strong activity (keyboard, trackball nav/click) wakes from any state. Weak activity (trackball raw movement) wakes from DIMMED only. Configurable timeouts via Settings.
### Boot Loop Recovery
NVS counter tracks consecutive boot failures. After 3 failures, WiFi is forced OFF on next boot. Counter resets to 0 at end of successful `setup()`.
### Input System
Trackball click uses deferred release with 80ms GPIO debounce. Long press fires at 1200ms hold, suppresses click on release. Nav events are rate-limited to 200ms with threshold of 3 accumulated deltas. I2C keyboard at address 0x55 with interrupt on GPIO 46.