- flash.py: standalone flash utility with serial port listing, merge-bin, GitHub Releases download, and esptool flash support - Display.h: hide LAN row when Local TCP disabled, show local TCP port instead of backbone port - README.md: comprehensive documentation — Quick Start with 3 flash options, OLED display layout, interface modes, routing customizations, path table fix, interface name uniqueness, hardware rationale (PSRAM/flash) - Release/boot_app0.bin: bundled for flash.py standalone use - .gitignore: exclude merged firmware binary build artifact
402 lines
14 KiB
Python
402 lines
14 KiB
Python
#!/usr/bin/env python3
|
|
"""
|
|
RNodeTHV4 Flash Utility
|
|
|
|
Flash the RNodeTHV4 boundary node firmware to a Heltec WiFi LoRa 32 V4.
|
|
No PlatformIO required — just Python 3 and a USB cable.
|
|
|
|
Usage:
|
|
# Flash a pre-built merged binary (from GitHub Releases or local build)
|
|
python flash.py
|
|
|
|
# Flash a specific file
|
|
python flash.py --file rnodethv4_firmware.bin
|
|
|
|
# Download latest from GitHub and flash
|
|
python flash.py --download
|
|
|
|
# Specify serial port manually
|
|
python flash.py --port /dev/ttyACM0
|
|
|
|
# Just build the merged binary (requires PlatformIO build output)
|
|
python flash.py --merge-only
|
|
"""
|
|
|
|
import argparse
|
|
import glob
|
|
import hashlib
|
|
import os
|
|
import platform
|
|
import shutil
|
|
import subprocess
|
|
import sys
|
|
import time
|
|
|
|
# ── Configuration ──────────────────────────────────────────────────────────────
|
|
|
|
CHIP = "esp32s3"
|
|
FLASH_MODE = "qio"
|
|
FLASH_FREQ = "80m"
|
|
FLASH_SIZE = "16MB"
|
|
BAUD_RATE = "921600"
|
|
MERGED_FILENAME = "rnodethv4_firmware.bin"
|
|
GITHUB_REPO = "jrl290/RNodeTHV4"
|
|
|
|
# Flash addresses for ESP32-S3 Arduino framework
|
|
BOOTLOADER_ADDR = 0x0000
|
|
PARTITIONS_ADDR = 0x8000
|
|
BOOT_APP0_ADDR = 0xe000
|
|
APP_ADDR = 0x10000
|
|
|
|
# PlatformIO build output paths (relative to project root)
|
|
BUILD_DIR = ".pio/build/heltec_V4_boundary"
|
|
BOOTLOADER_BIN = os.path.join(BUILD_DIR, "bootloader.bin")
|
|
PARTITIONS_BIN = os.path.join(BUILD_DIR, "partitions.bin")
|
|
FIRMWARE_BIN = os.path.join(BUILD_DIR, "rnode_firmware_heltec32v4_boundary.bin")
|
|
BOOT_APP0_BIN = os.path.expanduser(
|
|
"~/.platformio/packages/framework-arduinoespressif32/tools/partitions/boot_app0.bin"
|
|
)
|
|
|
|
# ── Helpers ────────────────────────────────────────────────────────────────────
|
|
|
|
def find_esptool():
|
|
"""Find esptool.py — bundled, pip-installed, or PlatformIO's copy."""
|
|
# 1. Bundled in Release/
|
|
bundled = os.path.join(os.path.dirname(__file__), "Release", "esptool", "esptool.py")
|
|
if os.path.isfile(bundled):
|
|
return [sys.executable, bundled]
|
|
|
|
# 2. pip-installed esptool
|
|
if shutil.which("esptool.py"):
|
|
return ["esptool.py"]
|
|
if shutil.which("esptool"):
|
|
return ["esptool"]
|
|
|
|
# 3. PlatformIO's esptool
|
|
pio_esptool = os.path.expanduser(
|
|
"~/.platformio/packages/tool-esptoolpy/esptool.py"
|
|
)
|
|
if os.path.isfile(pio_esptool):
|
|
return [sys.executable, pio_esptool]
|
|
|
|
return None
|
|
|
|
|
|
def find_serial_port():
|
|
"""List available serial ports and let the user choose."""
|
|
system = platform.system()
|
|
|
|
# Gather ports from glob patterns
|
|
if system == "Darwin":
|
|
patterns = ["/dev/cu.usbmodem*", "/dev/tty.usbmodem*",
|
|
"/dev/cu.usbserial*", "/dev/cu.SLAB*"]
|
|
elif system == "Linux":
|
|
patterns = ["/dev/ttyACM*", "/dev/ttyUSB*"]
|
|
else:
|
|
patterns = []
|
|
|
|
ports = []
|
|
for pattern in patterns:
|
|
ports.extend(glob.glob(pattern))
|
|
|
|
# Also try pyserial's port enumeration (works on all platforms including Windows)
|
|
try:
|
|
import serial.tools.list_ports
|
|
for port in serial.tools.list_ports.comports():
|
|
if port.device not in ports:
|
|
ports.append(port.device)
|
|
except ImportError:
|
|
pass
|
|
|
|
# Sort for consistent ordering
|
|
ports.sort()
|
|
|
|
if not ports:
|
|
return None
|
|
|
|
print("\nAvailable serial ports:")
|
|
for i, p in enumerate(ports):
|
|
print(f" [{i+1}] {p}")
|
|
print()
|
|
|
|
while True:
|
|
try:
|
|
choice = input(f"Select port [1-{len(ports)}]: ").strip()
|
|
idx = int(choice) - 1
|
|
if 0 <= idx < len(ports):
|
|
return ports[idx]
|
|
except (ValueError, EOFError):
|
|
pass
|
|
print("Invalid selection, try again.")
|
|
|
|
|
|
def sha256_file(path):
|
|
"""Compute SHA-256 hash of a file."""
|
|
h = hashlib.sha256()
|
|
with open(path, "rb") as f:
|
|
for chunk in iter(lambda: f.read(65536), b""):
|
|
h.update(chunk)
|
|
return h.hexdigest()
|
|
|
|
|
|
def download_firmware(dest_path):
|
|
"""Download the latest merged firmware from GitHub Releases."""
|
|
try:
|
|
from urllib.request import urlretrieve, urlopen
|
|
import json
|
|
except ImportError:
|
|
print("Error: Python urllib not available.")
|
|
return False
|
|
|
|
api_url = f"https://api.github.com/repos/{GITHUB_REPO}/releases/latest"
|
|
print(f"Checking latest release from {GITHUB_REPO}...")
|
|
|
|
try:
|
|
with urlopen(api_url) as resp:
|
|
release = json.loads(resp.read())
|
|
except Exception as e:
|
|
print(f"Error fetching release info: {e}")
|
|
return False
|
|
|
|
# Find the merged firmware asset
|
|
asset_url = None
|
|
for asset in release.get("assets", []):
|
|
if asset["name"] == MERGED_FILENAME:
|
|
asset_url = asset["browser_download_url"]
|
|
break
|
|
|
|
if not asset_url:
|
|
print(f"Error: '{MERGED_FILENAME}' not found in latest release ({release.get('tag_name', '?')}).")
|
|
print("Available assets:", [a["name"] for a in release.get("assets", [])])
|
|
return False
|
|
|
|
print(f"Downloading {release['tag_name']} / {MERGED_FILENAME}...")
|
|
try:
|
|
urlretrieve(asset_url, dest_path)
|
|
except Exception as e:
|
|
print(f"Download failed: {e}")
|
|
return False
|
|
|
|
size = os.path.getsize(dest_path)
|
|
print(f"Downloaded {size:,} bytes SHA-256: {sha256_file(dest_path)[:16]}...")
|
|
return True
|
|
|
|
|
|
def merge_firmware(output_path, esptool_cmd):
|
|
"""Merge bootloader + partitions + boot_app0 + app into a single binary."""
|
|
# Check all required files exist
|
|
required = {
|
|
"bootloader": BOOTLOADER_BIN,
|
|
"partitions": PARTITIONS_BIN,
|
|
"firmware": FIRMWARE_BIN,
|
|
}
|
|
|
|
# boot_app0 can come from PlatformIO or be bundled
|
|
boot_app0 = BOOT_APP0_BIN
|
|
if not os.path.isfile(boot_app0):
|
|
# Check if bundled in Release/
|
|
alt = os.path.join(os.path.dirname(__file__), "Release", "boot_app0.bin")
|
|
if os.path.isfile(alt):
|
|
boot_app0 = alt
|
|
else:
|
|
print(f"Error: boot_app0.bin not found at {BOOT_APP0_BIN}")
|
|
print(" Run 'pio run -e heltec_V4_boundary' first, or install PlatformIO.")
|
|
return False
|
|
required["boot_app0"] = boot_app0
|
|
|
|
for name, path in required.items():
|
|
if not os.path.isfile(path):
|
|
print(f"Error: {name} not found: {path}")
|
|
print("Run 'pio run -e heltec_V4_boundary' to build first.")
|
|
return False
|
|
|
|
print("Merging firmware components...")
|
|
print(f" Bootloader: {BOOTLOADER_BIN} @ 0x{BOOTLOADER_ADDR:04x}")
|
|
print(f" Partitions: {PARTITIONS_BIN} @ 0x{PARTITIONS_ADDR:04x}")
|
|
print(f" boot_app0: {boot_app0} @ 0x{BOOT_APP0_ADDR:04x}")
|
|
print(f" Firmware: {FIRMWARE_BIN} @ 0x{APP_ADDR:05x}")
|
|
|
|
cmd = esptool_cmd + [
|
|
"--chip", CHIP,
|
|
"merge_bin",
|
|
"--flash_mode", FLASH_MODE,
|
|
"--flash_freq", FLASH_FREQ,
|
|
"--flash_size", FLASH_SIZE,
|
|
"-o", output_path,
|
|
f"0x{BOOTLOADER_ADDR:x}", BOOTLOADER_BIN,
|
|
f"0x{PARTITIONS_ADDR:x}", PARTITIONS_BIN,
|
|
f"0x{BOOT_APP0_ADDR:x}", boot_app0,
|
|
f"0x{APP_ADDR:x}", FIRMWARE_BIN,
|
|
]
|
|
|
|
result = subprocess.run(cmd, capture_output=True, text=True)
|
|
if result.returncode != 0:
|
|
print(f"Error merging: {result.stderr}{result.stdout}")
|
|
return False
|
|
|
|
size = os.path.getsize(output_path)
|
|
print(f"\nMerged binary: {output_path} ({size:,} bytes)")
|
|
print(f"SHA-256: {sha256_file(output_path)[:16]}...")
|
|
return True
|
|
|
|
|
|
def flash_firmware(firmware_path, port, esptool_cmd, baud=BAUD_RATE):
|
|
"""Flash firmware to the device."""
|
|
print(f"\nFlashing {firmware_path} to {port}...")
|
|
print(f" Chip: {CHIP} Baud: {baud} Flash: {FLASH_SIZE}\n")
|
|
|
|
# Determine if this is a merged binary (flash at 0x0) or app-only (flash at 0x10000)
|
|
size = os.path.getsize(firmware_path)
|
|
if size > 1500000:
|
|
# Merged binary — includes bootloader, partitions, etc.
|
|
flash_addr = f"0x{BOOTLOADER_ADDR:x}"
|
|
else:
|
|
# App-only binary
|
|
flash_addr = f"0x{APP_ADDR:x}"
|
|
|
|
cmd = esptool_cmd + [
|
|
"--chip", CHIP,
|
|
"--port", port,
|
|
"--baud", baud,
|
|
"--before", "default_reset",
|
|
"--after", "hard_reset",
|
|
"write_flash",
|
|
"-z",
|
|
"--flash_mode", FLASH_MODE,
|
|
"--flash_freq", FLASH_FREQ,
|
|
"--flash_size", FLASH_SIZE,
|
|
flash_addr, firmware_path,
|
|
]
|
|
|
|
print("Running: " + " ".join(cmd[-8:]))
|
|
result = subprocess.run(cmd)
|
|
return result.returncode == 0
|
|
|
|
|
|
# ── Main ───────────────────────────────────────────────────────────────────────
|
|
|
|
def main():
|
|
parser = argparse.ArgumentParser(
|
|
description="RNodeTHV4 Flash Utility — flash boundary node firmware to Heltec V4",
|
|
formatter_class=argparse.RawDescriptionHelpFormatter,
|
|
epilog="""
|
|
Examples:
|
|
python flash.py # Flash local merged binary
|
|
python flash.py --download # Download latest release and flash
|
|
python flash.py --file firmware.bin # Flash a specific file
|
|
python flash.py --merge-only # Build merged binary from PlatformIO output
|
|
python flash.py --port /dev/ttyACM0 # Specify serial port
|
|
""",
|
|
)
|
|
parser.add_argument("--file", "-f", help="Path to firmware binary to flash")
|
|
parser.add_argument("--port", "-p", help="Serial port (auto-detected if omitted)")
|
|
parser.add_argument("--baud", "-b", default=BAUD_RATE, help=f"Baud rate (default: {BAUD_RATE})")
|
|
parser.add_argument("--download", "-d", action="store_true",
|
|
help="Download latest firmware from GitHub Releases")
|
|
parser.add_argument("--merge-only", action="store_true",
|
|
help="Merge PlatformIO build output into single binary, don't flash")
|
|
parser.add_argument("--no-merge", action="store_true",
|
|
help="Skip merge step, use existing merged binary or --file")
|
|
|
|
args = parser.parse_args()
|
|
baud = args.baud
|
|
|
|
print("╔══════════════════════════════════════════╗")
|
|
print("║ RNodeTHV4 Flash Utility ║")
|
|
print("║ Heltec WiFi LoRa 32 V4 Boundary Node ║")
|
|
print("╚══════════════════════════════════════════╝")
|
|
print()
|
|
|
|
# Find esptool
|
|
esptool_cmd = find_esptool()
|
|
if not esptool_cmd:
|
|
print("Error: esptool not found!")
|
|
print("Install it with: pip install esptool")
|
|
sys.exit(1)
|
|
print(f"Using esptool: {' '.join(esptool_cmd)}")
|
|
|
|
# Determine firmware file
|
|
firmware_path = None
|
|
|
|
if args.file:
|
|
firmware_path = args.file
|
|
if not os.path.isfile(firmware_path):
|
|
print(f"Error: file not found: {firmware_path}")
|
|
sys.exit(1)
|
|
|
|
elif args.download:
|
|
firmware_path = MERGED_FILENAME
|
|
if not download_firmware(firmware_path):
|
|
sys.exit(1)
|
|
|
|
elif args.merge_only:
|
|
if merge_firmware(MERGED_FILENAME, esptool_cmd):
|
|
print(f"\nDone! Flash with: python flash.py --file {MERGED_FILENAME}")
|
|
else:
|
|
sys.exit(1)
|
|
return
|
|
|
|
else:
|
|
# Try to find or create a merged binary
|
|
if os.path.isfile(MERGED_FILENAME) and not args.no_merge:
|
|
# Check if PlatformIO build is newer
|
|
if os.path.isfile(FIRMWARE_BIN):
|
|
build_time = os.path.getmtime(FIRMWARE_BIN)
|
|
merge_time = os.path.getmtime(MERGED_FILENAME)
|
|
if build_time > merge_time:
|
|
print("Build output is newer than merged binary, re-merging...")
|
|
if not merge_firmware(MERGED_FILENAME, esptool_cmd):
|
|
sys.exit(1)
|
|
firmware_path = MERGED_FILENAME
|
|
elif os.path.isfile(FIRMWARE_BIN):
|
|
# Build exists but no merged binary — create one
|
|
print("Found PlatformIO build output, creating merged binary...")
|
|
if not merge_firmware(MERGED_FILENAME, esptool_cmd):
|
|
sys.exit(1)
|
|
firmware_path = MERGED_FILENAME
|
|
elif os.path.isfile(MERGED_FILENAME):
|
|
firmware_path = MERGED_FILENAME
|
|
else:
|
|
print("No firmware found!")
|
|
print()
|
|
print("Options:")
|
|
print(" 1. Build with PlatformIO first: pio run -e heltec_V4_boundary")
|
|
print(" 2. Download from GitHub: python flash.py --download")
|
|
print(" 3. Specify a file: python flash.py --file <path>")
|
|
sys.exit(1)
|
|
|
|
# Flash
|
|
port = args.port or find_serial_port()
|
|
if not port:
|
|
print("\nError: No serial port detected!")
|
|
print("Connect your Heltec V4 via USB and try again,")
|
|
print("or specify manually: python flash.py --port /dev/ttyACM0")
|
|
sys.exit(1)
|
|
|
|
print(f"\nSerial port: {port}")
|
|
print(f"Firmware: {firmware_path} ({os.path.getsize(firmware_path):,} bytes)")
|
|
print()
|
|
|
|
confirm = input("Flash firmware? [Y/n] ").strip().lower()
|
|
if confirm and confirm != "y":
|
|
print("Aborted.")
|
|
sys.exit(0)
|
|
|
|
if flash_firmware(firmware_path, port, esptool_cmd, baud):
|
|
print()
|
|
print("╔══════════════════════════════════════════╗")
|
|
print("║ Flash complete! ║")
|
|
print("║ Device will reboot automatically. ║")
|
|
print("║ ║")
|
|
print("║ On first boot, hold PRG > 5s to enter ║")
|
|
print("║ the configuration portal. ║")
|
|
print("╚══════════════════════════════════════════╝")
|
|
else:
|
|
print("\nFlash FAILED. Check connection and try again.")
|
|
print("You may need to hold BOOT while pressing RESET.")
|
|
sys.exit(1)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|