mirror of
https://github.com/torlando-tech/pyxis.git
synced 2026-04-01 14:45:43 +00:00
Split T-Deck firmware from microReticulum examples/lxmf_tdeck/ into its own repo. microReticulum is consumed as a git submodule dependency pinned to feat/t-deck. All include paths updated from relative symlinks to bare includes resolved via library build flags. Both tdeck (NimBLE) and tdeck-bluedroid environments compile successfully. Licensed under AGPLv3. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
497 lines
16 KiB
C++
497 lines
16 KiB
C++
#include "TCPClientInterface.h"
|
|
#include "HDLC.h"
|
|
|
|
#include <Transport.h>
|
|
#include <Log.h>
|
|
|
|
#include <memory>
|
|
|
|
#ifdef ARDUINO
|
|
// ESP32 lwIP socket headers
|
|
#include <lwip/sockets.h>
|
|
#include <lwip/netdb.h>
|
|
#else
|
|
#include <sys/socket.h>
|
|
#include <netinet/in.h>
|
|
#include <netinet/tcp.h>
|
|
#include <arpa/inet.h>
|
|
#include <netdb.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#endif
|
|
|
|
using namespace RNS;
|
|
|
|
TCPClientInterface::TCPClientInterface(const char* name /*= "TCPClientInterface"*/)
|
|
: RNS::InterfaceImpl(name) {
|
|
|
|
_IN = true;
|
|
_OUT = true;
|
|
_bitrate = BITRATE_GUESS;
|
|
_HW_MTU = HW_MTU;
|
|
}
|
|
|
|
/*virtual*/ TCPClientInterface::~TCPClientInterface() {
|
|
stop();
|
|
}
|
|
|
|
/*virtual*/ bool TCPClientInterface::start() {
|
|
_online = false;
|
|
|
|
TRACE("TCPClientInterface: target host: " + _target_host);
|
|
TRACE("TCPClientInterface: target port: " + std::to_string(_target_port));
|
|
|
|
if (_target_host.empty()) {
|
|
ERROR("TCPClientInterface: No target host configured");
|
|
return false;
|
|
}
|
|
|
|
// WiFi connection is handled externally (in main.cpp)
|
|
// Attempt initial connection
|
|
if (!connect()) {
|
|
INFO("TCPClientInterface: Initial connection failed, will retry in background");
|
|
// Don't return false - we'll reconnect in loop()
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool TCPClientInterface::connect() {
|
|
TRACE("TCPClientInterface: Connecting to " + _target_host + ":" + std::to_string(_target_port));
|
|
|
|
#ifdef ARDUINO
|
|
// Set connection timeout
|
|
_client.setTimeout(CONNECT_TIMEOUT_MS);
|
|
|
|
if (!_client.connect(_target_host.c_str(), _target_port)) {
|
|
DEBUG("TCPClientInterface: Connection failed");
|
|
return false;
|
|
}
|
|
|
|
// Configure socket options
|
|
configure_socket();
|
|
|
|
INFO("TCPClientInterface: Connected to " + _target_host + ":" + std::to_string(_target_port));
|
|
_online = true;
|
|
_reconnected = true; // Signal that we (re)connected - main loop should announce
|
|
_last_data_received = millis(); // Reset stale timer
|
|
_frame_buffer.clear();
|
|
return true;
|
|
|
|
#else
|
|
// Resolve target host
|
|
struct in_addr target_addr;
|
|
if (inet_aton(_target_host.c_str(), &target_addr) == 0) {
|
|
struct hostent* host_ent = gethostbyname(_target_host.c_str());
|
|
if (host_ent == nullptr || host_ent->h_addr_list[0] == nullptr) {
|
|
ERROR("TCPClientInterface: Unable to resolve host " + _target_host);
|
|
return false;
|
|
}
|
|
_target_address = *((in_addr_t*)(host_ent->h_addr_list[0]));
|
|
} else {
|
|
_target_address = target_addr.s_addr;
|
|
}
|
|
|
|
// Create TCP socket
|
|
_socket = socket(PF_INET, SOCK_STREAM, 0);
|
|
if (_socket < 0) {
|
|
ERROR("TCPClientInterface: Unable to create socket, error " + std::to_string(errno));
|
|
return false;
|
|
}
|
|
|
|
// Set non-blocking for connect timeout
|
|
int flags = fcntl(_socket, F_GETFL, 0);
|
|
fcntl(_socket, F_SETFL, flags | O_NONBLOCK);
|
|
|
|
// Connect to server
|
|
sockaddr_in server_addr;
|
|
server_addr.sin_family = AF_INET;
|
|
server_addr.sin_addr.s_addr = _target_address;
|
|
server_addr.sin_port = htons(_target_port);
|
|
|
|
int result = ::connect(_socket, (struct sockaddr*)&server_addr, sizeof(server_addr));
|
|
if (result < 0 && errno != EINPROGRESS) {
|
|
close(_socket);
|
|
_socket = -1;
|
|
ERROR("TCPClientInterface: Connect failed, error " + std::to_string(errno));
|
|
return false;
|
|
}
|
|
|
|
// Wait for connection with timeout
|
|
fd_set write_fds;
|
|
FD_ZERO(&write_fds);
|
|
FD_SET(_socket, &write_fds);
|
|
struct timeval timeout;
|
|
timeout.tv_sec = CONNECT_TIMEOUT_MS / 1000;
|
|
timeout.tv_usec = (CONNECT_TIMEOUT_MS % 1000) * 1000;
|
|
|
|
result = select(_socket + 1, nullptr, &write_fds, nullptr, &timeout);
|
|
if (result <= 0) {
|
|
close(_socket);
|
|
_socket = -1;
|
|
DEBUG("TCPClientInterface: Connection timeout");
|
|
return false;
|
|
}
|
|
|
|
// Check if connection succeeded
|
|
int sock_error = 0;
|
|
socklen_t len = sizeof(sock_error);
|
|
getsockopt(_socket, SOL_SOCKET, SO_ERROR, &sock_error, &len);
|
|
if (sock_error != 0) {
|
|
close(_socket);
|
|
_socket = -1;
|
|
DEBUG("TCPClientInterface: Connection failed, error " + std::to_string(sock_error));
|
|
return false;
|
|
}
|
|
|
|
// Restore blocking mode for normal operation
|
|
fcntl(_socket, F_SETFL, flags);
|
|
|
|
// Configure socket options
|
|
configure_socket();
|
|
|
|
INFO("TCPClientInterface: Connected to " + _target_host + ":" + std::to_string(_target_port));
|
|
_online = true;
|
|
_frame_buffer.clear();
|
|
return true;
|
|
#endif
|
|
}
|
|
|
|
void TCPClientInterface::configure_socket() {
|
|
#ifdef ARDUINO
|
|
// Get underlying socket fd for setsockopt
|
|
int fd = _client.fd();
|
|
if (fd < 0) {
|
|
DEBUG("TCPClientInterface: Could not get socket fd for configuration");
|
|
return;
|
|
}
|
|
|
|
// TCP_NODELAY - disable Nagle's algorithm
|
|
int flag = 1;
|
|
setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &flag, sizeof(flag));
|
|
|
|
// Enable TCP keepalive
|
|
setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE, &flag, sizeof(flag));
|
|
|
|
// Keepalive parameters (may not all be available on ESP32 lwIP)
|
|
#ifdef TCP_KEEPIDLE
|
|
int keepidle = TCP_KEEPIDLE_SEC;
|
|
setsockopt(fd, IPPROTO_TCP, TCP_KEEPIDLE, &keepidle, sizeof(keepidle));
|
|
#endif
|
|
#ifdef TCP_KEEPINTVL
|
|
int keepintvl = TCP_KEEPINTVL_SEC;
|
|
setsockopt(fd, IPPROTO_TCP, TCP_KEEPINTVL, &keepintvl, sizeof(keepintvl));
|
|
#endif
|
|
#ifdef TCP_KEEPCNT
|
|
int keepcnt = TCP_KEEPCNT_PROBES;
|
|
setsockopt(fd, IPPROTO_TCP, TCP_KEEPCNT, &keepcnt, sizeof(keepcnt));
|
|
#endif
|
|
|
|
TRACE("TCPClientInterface: Socket configured with TCP_NODELAY and keepalive");
|
|
|
|
#else
|
|
// TCP_NODELAY
|
|
int flag = 1;
|
|
setsockopt(_socket, IPPROTO_TCP, TCP_NODELAY, &flag, sizeof(flag));
|
|
|
|
// Enable TCP keepalive
|
|
setsockopt(_socket, SOL_SOCKET, SO_KEEPALIVE, &flag, sizeof(flag));
|
|
|
|
// Keepalive parameters
|
|
int keepidle = TCP_KEEPIDLE_SEC;
|
|
int keepintvl = TCP_KEEPINTVL_SEC;
|
|
int keepcnt = TCP_KEEPCNT_PROBES;
|
|
setsockopt(_socket, IPPROTO_TCP, TCP_KEEPIDLE, &keepidle, sizeof(keepidle));
|
|
setsockopt(_socket, IPPROTO_TCP, TCP_KEEPINTVL, &keepintvl, sizeof(keepintvl));
|
|
setsockopt(_socket, IPPROTO_TCP, TCP_KEEPCNT, &keepcnt, sizeof(keepcnt));
|
|
|
|
// TCP_USER_TIMEOUT (Linux 2.6.37+)
|
|
#ifdef TCP_USER_TIMEOUT
|
|
int user_timeout = 24000; // 24 seconds, matches Python RNS
|
|
setsockopt(_socket, IPPROTO_TCP, TCP_USER_TIMEOUT, &user_timeout, sizeof(user_timeout));
|
|
#endif
|
|
|
|
TRACE("TCPClientInterface: Socket configured with TCP_NODELAY, keepalive, and timeouts");
|
|
#endif
|
|
}
|
|
|
|
void TCPClientInterface::disconnect() {
|
|
DEBUG("TCPClientInterface: Disconnecting");
|
|
|
|
#ifdef ARDUINO
|
|
_client.stop();
|
|
#else
|
|
if (_socket >= 0) {
|
|
close(_socket);
|
|
_socket = -1;
|
|
}
|
|
#endif
|
|
|
|
_online = false;
|
|
_frame_buffer.clear();
|
|
}
|
|
|
|
void TCPClientInterface::handle_disconnect() {
|
|
if (_online) {
|
|
INFO("TCPClientInterface: Connection lost, will attempt reconnection");
|
|
disconnect();
|
|
// Reset connect attempt timer to enforce wait before reconnection
|
|
_last_connect_attempt = millis();
|
|
}
|
|
}
|
|
|
|
/*virtual*/ void TCPClientInterface::stop() {
|
|
disconnect();
|
|
}
|
|
|
|
/*virtual*/ void TCPClientInterface::loop() {
|
|
// Periodic status logging
|
|
static uint32_t last_status_log = 0;
|
|
static uint32_t loop_count = 0;
|
|
static uint32_t total_rx = 0;
|
|
loop_count++;
|
|
uint32_t now = millis();
|
|
if (now - last_status_log >= 5000) { // Every 5 seconds
|
|
last_status_log = now;
|
|
int avail = _client.available();
|
|
Serial.printf("[TCP] connected=%d online=%d avail=%d loops=%u rx=%u buf=%d\n",
|
|
_client.connected(), _online, avail, loop_count, total_rx, (int)_frame_buffer.size());
|
|
loop_count = 0;
|
|
}
|
|
|
|
// Handle reconnection if not connected
|
|
if (!_online) {
|
|
if (_initiator) {
|
|
#ifdef ARDUINO
|
|
uint32_t now = millis();
|
|
#else
|
|
uint32_t now = static_cast<uint32_t>(Utilities::OS::time() * 1000);
|
|
#endif
|
|
if (now - _last_connect_attempt >= RECONNECT_WAIT_MS) {
|
|
_last_connect_attempt = now;
|
|
// Skip reconnection if memory is too low - prevents fragmentation
|
|
uint32_t max_block = ESP.getMaxAllocHeap();
|
|
if (max_block < 20000) {
|
|
Serial.printf("[TCP] Skipping reconnect - low memory (max_block=%u)\n", max_block);
|
|
} else {
|
|
DEBUG("TCPClientInterface: Attempting reconnection...");
|
|
connect();
|
|
}
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Check connection status
|
|
// Note: ESP32 WiFiClient.connected() has known bugs where it returns false incorrectly
|
|
// See: https://github.com/espressif/arduino-esp32/issues/1714
|
|
// Workaround: only disconnect if connected() is false AND no data available
|
|
#ifdef ARDUINO
|
|
if (!_client.connected() && _client.available() == 0) {
|
|
Serial.printf("[TCP] Connection closed (connected=false, available=0)\n");
|
|
handle_disconnect();
|
|
return;
|
|
}
|
|
|
|
// Stale connection detection disabled - was causing frequent reconnects
|
|
// TODO: investigate why this triggers even when receiving data
|
|
// if (_last_data_received > 0 && (now - _last_data_received) > STALE_CONNECTION_MS) {
|
|
// WARNING("TCPClientInterface: Connection appears stale, forcing reconnection");
|
|
// handle_disconnect();
|
|
// return;
|
|
// }
|
|
|
|
// Read available data
|
|
int avail = _client.available();
|
|
if (avail > 0) {
|
|
Serial.printf("[TCP] Reading %d bytes\n", avail);
|
|
total_rx += avail;
|
|
_last_data_received = now; // Update stale timer on any data receipt
|
|
size_t start_pos = _frame_buffer.size();
|
|
while (_client.available() > 0) {
|
|
uint8_t byte = _client.read();
|
|
_frame_buffer.append(byte);
|
|
}
|
|
// Dump first 20 bytes of new data
|
|
Serial.printf("[TCP] First bytes: ");
|
|
size_t dump_len = (_frame_buffer.size() - start_pos);
|
|
if (dump_len > 20) dump_len = 20;
|
|
for (size_t i = 0; i < dump_len; ++i) {
|
|
Serial.printf("%02x ", _frame_buffer.data()[start_pos + i]);
|
|
}
|
|
Serial.printf("\n");
|
|
}
|
|
#else
|
|
// Non-blocking read
|
|
uint8_t buf[4096];
|
|
ssize_t len = recv(_socket, buf, sizeof(buf), MSG_DONTWAIT);
|
|
if (len > 0) {
|
|
DEBUG("TCPClientInterface: Received " + std::to_string(len) + " bytes");
|
|
_frame_buffer.append(buf, len);
|
|
} else if (len == 0) {
|
|
// Connection closed by peer
|
|
DEBUG("TCPClientInterface: recv returned 0 - connection closed");
|
|
handle_disconnect();
|
|
return;
|
|
} else {
|
|
int err = errno;
|
|
if (err != EAGAIN && err != EWOULDBLOCK) {
|
|
// Socket error
|
|
ERROR("TCPClientInterface: recv error " + std::to_string(err));
|
|
handle_disconnect();
|
|
return;
|
|
}
|
|
// EAGAIN/EWOULDBLOCK - normal for non-blocking, just no data yet
|
|
}
|
|
#endif
|
|
|
|
// Process any complete frames
|
|
extract_and_process_frames();
|
|
}
|
|
|
|
void TCPClientInterface::extract_and_process_frames() {
|
|
// Find and process complete HDLC frames: [FLAG][data][FLAG]
|
|
static uint32_t frame_count = 0;
|
|
|
|
while (true) {
|
|
if (_frame_buffer.size() == 0) break;
|
|
|
|
// Find first FLAG byte
|
|
int start = -1;
|
|
for (size_t i = 0; i < _frame_buffer.size(); ++i) {
|
|
if (_frame_buffer.data()[i] == HDLC::FLAG) {
|
|
start = static_cast<int>(i);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (start < 0) {
|
|
// No FLAG found, discard buffer (garbage data before any frame)
|
|
Serial.printf("[HDLC] No FLAG in %d bytes, clearing\n", (int)_frame_buffer.size());
|
|
_frame_buffer.clear();
|
|
break;
|
|
}
|
|
|
|
// Discard data before first FLAG
|
|
if (start > 0) {
|
|
Serial.printf("[HDLC] Discarding %d bytes before FLAG\n", start);
|
|
_frame_buffer = _frame_buffer.mid(start);
|
|
}
|
|
|
|
// Find end FLAG (skip the start FLAG at position 0)
|
|
int end = -1;
|
|
for (size_t i = 1; i < _frame_buffer.size(); ++i) {
|
|
if (_frame_buffer.data()[i] == HDLC::FLAG) {
|
|
end = static_cast<int>(i);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (end < 0) {
|
|
// Incomplete frame, wait for more data
|
|
break;
|
|
}
|
|
|
|
// Extract frame content between FLAGS (excluding the FLAGS)
|
|
Bytes frame_content = _frame_buffer.mid(1, end - 1);
|
|
frame_count++;
|
|
Serial.printf("[HDLC] Frame #%u: %d escaped bytes\n", frame_count, (int)frame_content.size());
|
|
|
|
// Remove processed frame from buffer (keep data after end FLAG)
|
|
_frame_buffer = _frame_buffer.mid(end);
|
|
|
|
// Skip empty frames (consecutive FLAGs)
|
|
if (frame_content.size() == 0) {
|
|
Serial.printf("[HDLC] Empty frame, skipping\n");
|
|
continue;
|
|
}
|
|
|
|
// Unescape frame
|
|
Bytes unescaped = HDLC::unescape(frame_content);
|
|
if (unescaped.size() == 0) {
|
|
Serial.printf("[HDLC] Unescape failed!\n");
|
|
DEBUG("TCPClientInterface: HDLC unescape error, discarding frame");
|
|
continue;
|
|
}
|
|
|
|
// Validate minimum frame size (matches Python RNS HEADER_MINSIZE check)
|
|
if (unescaped.size() < Type::Reticulum::HEADER_MINSIZE) {
|
|
TRACE("TCPClientInterface: Frame too small (" + std::to_string(unescaped.size()) + " bytes), discarding");
|
|
continue;
|
|
}
|
|
|
|
// Pass to transport layer
|
|
Serial.printf("[TCP] Processing frame: %d bytes\n", (int)unescaped.size());
|
|
DEBUG(toString() + ": Received frame, " + std::to_string(unescaped.size()) + " bytes");
|
|
InterfaceImpl::handle_incoming(unescaped);
|
|
}
|
|
}
|
|
|
|
/*virtual*/ void TCPClientInterface::send_outgoing(const Bytes& data) {
|
|
DEBUG(toString() + ".send_outgoing: data: " + std::to_string(data.size()) + " bytes");
|
|
|
|
// Log first 50 bytes of raw packet (before HDLC framing)
|
|
std::string hex_preview;
|
|
size_t preview_len = (data.size() < 50) ? data.size() : 50;
|
|
for (size_t i = 0; i < preview_len; ++i) {
|
|
char buf[4];
|
|
snprintf(buf, sizeof(buf), "%02x", data.data()[i]);
|
|
hex_preview += buf;
|
|
}
|
|
if (data.size() > 50) hex_preview += "...";
|
|
INFO("WIRE TX raw (" + std::to_string(data.size()) + " bytes): " + hex_preview);
|
|
|
|
if (!_online) {
|
|
DEBUG("TCPClientInterface: Not connected, cannot send");
|
|
return;
|
|
}
|
|
|
|
try {
|
|
// Frame with HDLC
|
|
Bytes framed = HDLC::frame(data);
|
|
|
|
// Log HDLC framed output for debugging
|
|
std::string framed_hex;
|
|
size_t flen = (framed.size() < 30) ? framed.size() : 30;
|
|
for (size_t i = 0; i < flen; ++i) {
|
|
char buf[4];
|
|
snprintf(buf, sizeof(buf), "%02x", framed.data()[i]);
|
|
framed_hex += buf;
|
|
}
|
|
if (framed.size() > 30) framed_hex += "...";
|
|
INFO("WIRE TX framed (" + std::to_string(framed.size()) + " bytes): " + framed_hex);
|
|
|
|
#ifdef ARDUINO
|
|
size_t written = _client.write(framed.data(), framed.size());
|
|
if (written != framed.size()) {
|
|
ERROR("TCPClientInterface: Write incomplete, " + std::to_string(written) +
|
|
" of " + std::to_string(framed.size()) + " bytes");
|
|
handle_disconnect();
|
|
return;
|
|
}
|
|
_client.flush();
|
|
#else
|
|
ssize_t written = send(_socket, framed.data(), framed.size(), MSG_NOSIGNAL);
|
|
if (written < 0) {
|
|
ERROR("TCPClientInterface: send error " + std::to_string(errno));
|
|
handle_disconnect();
|
|
return;
|
|
}
|
|
if (static_cast<size_t>(written) != framed.size()) {
|
|
ERROR("TCPClientInterface: Write incomplete, " + std::to_string(written) +
|
|
" of " + std::to_string(framed.size()) + " bytes");
|
|
handle_disconnect();
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
// Perform post-send housekeeping
|
|
InterfaceImpl::handle_outgoing(data);
|
|
|
|
} catch (std::exception& e) {
|
|
ERROR("TCPClientInterface: Exception during send: " + std::string(e.what()));
|
|
handle_disconnect();
|
|
}
|
|
}
|