|
|
|
|
@@ -2,11 +2,12 @@
|
|
|
|
|
# trunk-ignore-all(ruff/F821)
|
|
|
|
|
# trunk-ignore-all(flake8/F821): For SConstruct imports
|
|
|
|
|
import sys
|
|
|
|
|
from os.path import join, basename, isfile
|
|
|
|
|
from os.path import join
|
|
|
|
|
import subprocess
|
|
|
|
|
import json
|
|
|
|
|
import re
|
|
|
|
|
from datetime import datetime
|
|
|
|
|
from typing import Dict
|
|
|
|
|
|
|
|
|
|
from readprops import readProps
|
|
|
|
|
|
|
|
|
|
@@ -14,8 +15,47 @@ Import("env")
|
|
|
|
|
platform = env.PioPlatform()
|
|
|
|
|
progname = env.get("PROGNAME")
|
|
|
|
|
lfsbin = f"{progname.replace('firmware-', 'littlefs-')}.bin"
|
|
|
|
|
manifest_ran = False
|
|
|
|
|
|
|
|
|
|
def infer_architecture(board_cfg):
|
|
|
|
|
try:
|
|
|
|
|
mcu = board_cfg.get("build.mcu") if board_cfg else None
|
|
|
|
|
except KeyError:
|
|
|
|
|
mcu = None
|
|
|
|
|
except Exception:
|
|
|
|
|
mcu = None
|
|
|
|
|
if not mcu:
|
|
|
|
|
return None
|
|
|
|
|
mcu_l = str(mcu).lower()
|
|
|
|
|
if "esp32s3" in mcu_l:
|
|
|
|
|
return "esp32-s3"
|
|
|
|
|
if "esp32c6" in mcu_l:
|
|
|
|
|
return "esp32-c6"
|
|
|
|
|
if "esp32c3" in mcu_l:
|
|
|
|
|
return "esp32-c3"
|
|
|
|
|
if "esp32" in mcu_l:
|
|
|
|
|
return "esp32"
|
|
|
|
|
if "rp2040" in mcu_l:
|
|
|
|
|
return "rp2040"
|
|
|
|
|
if "rp2350" in mcu_l:
|
|
|
|
|
return "rp2350"
|
|
|
|
|
if "nrf52" in mcu_l or "nrf52840" in mcu_l:
|
|
|
|
|
return "nrf52840"
|
|
|
|
|
if "stm32" in mcu_l:
|
|
|
|
|
return "stm32"
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
def manifest_gather(source, target, env):
|
|
|
|
|
global manifest_ran
|
|
|
|
|
if manifest_ran:
|
|
|
|
|
return
|
|
|
|
|
# Skip manifest generation if we cannot determine architecture (host/native builds)
|
|
|
|
|
board_arch = infer_architecture(env.BoardConfig())
|
|
|
|
|
if not board_arch:
|
|
|
|
|
print(f"Skipping mtjson generation for unknown architecture (env={env.get('PIOENV')})")
|
|
|
|
|
manifest_ran = True
|
|
|
|
|
return
|
|
|
|
|
manifest_ran = True
|
|
|
|
|
out = []
|
|
|
|
|
board_platform = env.BoardConfig().get("platform")
|
|
|
|
|
needs_ota_suffix = board_platform == "nordicnrf52"
|
|
|
|
|
@@ -47,14 +87,39 @@ def manifest_gather(source, target, env):
|
|
|
|
|
manifest_write(out, env)
|
|
|
|
|
|
|
|
|
|
def manifest_write(files, env):
|
|
|
|
|
# Defensive: also skip manifest writing if we cannot determine architecture
|
|
|
|
|
def get_project_option(name):
|
|
|
|
|
try:
|
|
|
|
|
return env.GetProjectOption(name)
|
|
|
|
|
except Exception:
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
def get_project_option_any(names):
|
|
|
|
|
for name in names:
|
|
|
|
|
val = get_project_option(name)
|
|
|
|
|
if val is not None:
|
|
|
|
|
return val
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
def as_bool(val):
|
|
|
|
|
return str(val).strip().lower() in ("1", "true", "yes", "on")
|
|
|
|
|
|
|
|
|
|
def as_int(val):
|
|
|
|
|
try:
|
|
|
|
|
return int(str(val), 10)
|
|
|
|
|
except (TypeError, ValueError):
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
def as_list(val):
|
|
|
|
|
return [item.strip() for item in str(val).split(",") if item.strip()]
|
|
|
|
|
|
|
|
|
|
manifest = {
|
|
|
|
|
"version": verObj["long"],
|
|
|
|
|
"build_epoch": build_epoch,
|
|
|
|
|
"board": env.get("PIOENV"),
|
|
|
|
|
"platformioTarget": env.get("PIOENV"),
|
|
|
|
|
"mcu": env.get("BOARD_MCU"),
|
|
|
|
|
"repo": repo_owner,
|
|
|
|
|
"files": files,
|
|
|
|
|
"part": None,
|
|
|
|
|
"has_mui": False,
|
|
|
|
|
"has_inkhud": False,
|
|
|
|
|
}
|
|
|
|
|
@@ -69,6 +134,51 @@ def manifest_write(files, env):
|
|
|
|
|
if "MESHTASTIC_INCLUDE_INKHUD" in env.get("CPPDEFINES", []):
|
|
|
|
|
manifest["has_inkhud"] = True
|
|
|
|
|
|
|
|
|
|
pioenv = env.get("PIOENV")
|
|
|
|
|
device_meta = {}
|
|
|
|
|
device_meta_fields = [
|
|
|
|
|
("hwModel", ["custom_meshtastic_hw_model"], as_int),
|
|
|
|
|
("hwModelSlug", ["custom_meshtastic_hw_model_slug"], str),
|
|
|
|
|
("architecture", ["custom_meshtastic_architecture"], str),
|
|
|
|
|
("activelySupported", ["custom_meshtastic_actively_supported"], as_bool),
|
|
|
|
|
("displayName", ["custom_meshtastic_display_name"], str),
|
|
|
|
|
("supportLevel", ["custom_meshtastic_support_level"], as_int),
|
|
|
|
|
("images", ["custom_meshtastic_images"], as_list),
|
|
|
|
|
("tags", ["custom_meshtastic_tags"], as_list),
|
|
|
|
|
("requiresDfu", ["custom_meshtastic_requires_dfu"], as_bool),
|
|
|
|
|
("partitionScheme", ["custom_meshtastic_partition_scheme"], str),
|
|
|
|
|
("url", ["custom_meshtastic_url"], str),
|
|
|
|
|
("key", ["custom_meshtastic_key"], str),
|
|
|
|
|
("variant", ["custom_meshtastic_variant"], str),
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for manifest_key, option_keys, caster in device_meta_fields:
|
|
|
|
|
raw_val = get_project_option_any(option_keys)
|
|
|
|
|
if raw_val is None:
|
|
|
|
|
continue
|
|
|
|
|
parsed = caster(raw_val) if callable(caster) else raw_val
|
|
|
|
|
if parsed is not None and parsed != "":
|
|
|
|
|
device_meta[manifest_key] = parsed
|
|
|
|
|
|
|
|
|
|
# Determine architecture once; if we can't infer it, skip manifest generation
|
|
|
|
|
board_arch = device_meta.get("architecture") or infer_architecture(env.BoardConfig())
|
|
|
|
|
if not board_arch:
|
|
|
|
|
print(f"Skipping mtjson write for unknown architecture (env={env.get('PIOENV')})")
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
device_meta["architecture"] = board_arch
|
|
|
|
|
|
|
|
|
|
# Always set requiresDfu: true for nrf52840 targets
|
|
|
|
|
if board_arch == "nrf52840":
|
|
|
|
|
device_meta["requiresDfu"] = True
|
|
|
|
|
|
|
|
|
|
device_meta.setdefault("displayName", pioenv)
|
|
|
|
|
device_meta.setdefault("activelySupported", False)
|
|
|
|
|
|
|
|
|
|
if device_meta:
|
|
|
|
|
manifest.update(device_meta)
|
|
|
|
|
|
|
|
|
|
# Write the manifest to the build directory
|
|
|
|
|
with open(env.subst("$BUILD_DIR/${PROGNAME}.mt.json"), "w") as f:
|
|
|
|
|
json.dump(manifest, f, indent=2)
|
|
|
|
|
@@ -166,8 +276,12 @@ def load_boot_logo(source, target, env):
|
|
|
|
|
if ("HAS_TFT", 1) in env.get("CPPDEFINES", []):
|
|
|
|
|
env.AddPreAction(f"$BUILD_DIR/{lfsbin}", load_boot_logo)
|
|
|
|
|
|
|
|
|
|
mtjson_deps = ["buildprog"]
|
|
|
|
|
if platform.name == "espressif32":
|
|
|
|
|
board_arch = infer_architecture(env.BoardConfig())
|
|
|
|
|
should_skip_manifest = board_arch is None
|
|
|
|
|
|
|
|
|
|
# For host/native envs, avoid depending on 'buildprog' (some targets don't define it)
|
|
|
|
|
mtjson_deps = [] if should_skip_manifest else ["buildprog"]
|
|
|
|
|
if not should_skip_manifest and platform.name == "espressif32":
|
|
|
|
|
# Build littlefs image as part of mtjson target
|
|
|
|
|
# Equivalent to `pio run -t buildfs`
|
|
|
|
|
target_lfs = env.DataToBin(
|
|
|
|
|
@@ -175,11 +289,27 @@ if platform.name == "espressif32":
|
|
|
|
|
)
|
|
|
|
|
mtjson_deps.append(target_lfs)
|
|
|
|
|
|
|
|
|
|
env.AddCustomTarget(
|
|
|
|
|
name="mtjson",
|
|
|
|
|
dependencies=mtjson_deps,
|
|
|
|
|
actions=[manifest_gather],
|
|
|
|
|
title="Meshtastic Manifest",
|
|
|
|
|
description="Generating Meshtastic manifest JSON + Checksums",
|
|
|
|
|
always_build=False,
|
|
|
|
|
)
|
|
|
|
|
if should_skip_manifest:
|
|
|
|
|
def skip_manifest(source, target, env):
|
|
|
|
|
print(f"mtjson: skipped for native environment: {env.get('PIOENV')}")
|
|
|
|
|
|
|
|
|
|
env.AddCustomTarget(
|
|
|
|
|
name="mtjson",
|
|
|
|
|
dependencies=mtjson_deps,
|
|
|
|
|
actions=[skip_manifest],
|
|
|
|
|
title="Meshtastic Manifest (skipped)",
|
|
|
|
|
description="mtjson generation is skipped for native environments",
|
|
|
|
|
always_build=True,
|
|
|
|
|
)
|
|
|
|
|
else:
|
|
|
|
|
env.AddCustomTarget(
|
|
|
|
|
name="mtjson",
|
|
|
|
|
dependencies=mtjson_deps,
|
|
|
|
|
actions=[manifest_gather],
|
|
|
|
|
title="Meshtastic Manifest",
|
|
|
|
|
description="Generating Meshtastic manifest JSON + Checksums",
|
|
|
|
|
always_build=True,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Run manifest generation as part of the default build pipeline for non-native builds.
|
|
|
|
|
env.Default("mtjson")
|
|
|
|
|
|