Add flash utility, display fixes, and comprehensive README
- 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
This commit is contained in:
401
flash.py
Normal file
401
flash.py
Normal file
@@ -0,0 +1,401 @@
|
||||
#!/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()
|
||||
Reference in New Issue
Block a user