ardupilot/Tools/scripts/build_bootloaders.py
2025-08-22 17:41:18 +10:00

159 lines
5.8 KiB
Python
Executable File

#!/usr/bin/env python3
# flake8: noqa
"""
script to build all of our bootloaders using AP_Bootloader and put the resulting binaries in Tools/bootloaders
"""
import os
import shutil
import subprocess
import sys
import fnmatch
import re
# get command line arguments
from argparse import ArgumentParser
parser = ArgumentParser(description='make_secure_bl')
parser.add_argument("--signing-key", type=str, default=None, help="signing key for secure bootloader")
parser.add_argument("--debug", action='store_true', default=False, help="build with debug symbols")
parser.add_argument("--periph-only", action='store_true', default=False, help="only build AP_Periph boards")
parser.add_argument("pattern", type=str, default='*', help="board wildcard pattern", nargs='?')
args = parser.parse_args()
if args.signing_key is not None and os.path.basename(args.signing_key).lower().find("private") != -1:
# prevent the easy mistake of using private key
print("You must use the public key in the bootloader")
sys.exit(1)
os.environ['PYTHONUNBUFFERED'] = '1'
failed_boards = set()
def has_hwdef_bl(board):
"""Return True if libraries/AP_HAL_ChibiOS/hwdef/<board>/hwdef-bl.dat exists"""
hwdef = os.path.join('libraries', 'AP_HAL_ChibiOS', 'hwdef', board, 'hwdef-bl.dat')
return os.path.exists(hwdef)
def read_hwdef(filepath):
'''read a hwdef file recursively'''
fh = open(filepath)
ret = []
text = fh.readlines()
for line in text:
m = re.match(r"^\s*include\s+(.+)\s*$", line)
if m is not None:
ret += read_hwdef(os.path.join(os.path.dirname(filepath), m.group(1)))
else:
ret += [line]
return ret
def is_ap_periph(hwdef):
'''return True if a hwdef is for a AP_Periph board'''
lines = read_hwdef(hwdef)
for line in lines:
if line.find('AP_PERIPH') != -1:
return True
return False
def get_board_list():
'''add boards based on existence of hwdef-bl.dat in subdirectories for ChibiOS'''
board_list = []
dirname, dirlist, filenames = next(os.walk('libraries/AP_HAL_ChibiOS/hwdef'))
for d in dirlist:
hwdef = os.path.join(dirname, d, 'hwdef-bl.dat')
if os.path.exists(hwdef):
if args.periph_only and not is_ap_periph(hwdef):
continue
board_list.append(d)
return board_list
def run_program(cmd_list):
print("Running (%s)" % " ".join(cmd_list))
retcode = subprocess.call(cmd_list)
if retcode != 0:
print("Build failed: %s" % ' '.join(cmd_list))
return False
return True
def build_board(board):
# do not attempt to build a bootloader for boards without hwdef-bl.dat
if not has_hwdef_bl(board):
print(f"Skipping {board}: no hwdef-bl.dat (no bootloader for this board)")
return True # treat as success, since there is nothing to build
configure_args = "--board %s --bootloader --no-submodule-update --Werror" % board
configure_args = configure_args.split()
if args.signing_key is not None:
print("Building secure bootloader")
configure_args.append("--signed-fw")
if args.debug:
print("Building with debug symbols")
configure_args.append("--debug")
if not run_program(["./waf", "configure"] + configure_args):
return False
if not run_program(["./waf", "clean"]):
return False
if not run_program(["./waf", "bootloader"]):
return False
return True
board_list = get_board_list()
def get_all_board_dirs():
dirname, dirlist, filenames = next(os.walk('libraries/AP_HAL_ChibiOS/hwdef'))
return dirlist
all_board_dirs = get_all_board_dirs()
# Determine pattern matches against *all* directories and against buildable boards
matches_all = [b for b in all_board_dirs if fnmatch.fnmatch(b, args.pattern)]
matches_buildable = [b for b in board_list if fnmatch.fnmatch(b, args.pattern)]
# If nothing matches any directory name at all, warn once
if args.pattern != '*' and not matches_all:
print(f"Warning: no board matches pattern '{args.pattern}'. Continuing.")
# For directories that match the pattern but *aren't* buildable (no hwdef-bl.dat),
# print the explicit skip message now so users see why their requested board didn't build.
for b in matches_all:
if b not in board_list:
print(f"Skipping {b}: no hwdef-bl.dat (no bootloader for this board)")
# check that the user-supplied board pattern matches something; if not, warn and exit
for board in board_list:
if not fnmatch.fnmatch(board, args.pattern):
continue
if not has_hwdef_bl(board):
print(f"Skipping {board}: no hwdef-bl.dat (no bootloader for this board)")
continue
print("Building for %s" % board)
if not build_board(board):
failed_boards.add(board)
continue
bl_file = 'Tools/bootloaders/%s_bl.bin' % board
hex_file = 'Tools/bootloaders/%s_bl.hex' % board
elf_file = 'Tools/bootloaders/%s_bl.elf' % board
shutil.copy('build/%s/bin/AP_Bootloader.bin' % board, bl_file)
print("Created %s" % bl_file)
shutil.copy('build/%s/bootloader/AP_Bootloader' % board, elf_file)
print("Created %s" % elf_file)
if args.signing_key is not None:
print("Signing bootloader with %s" % args.signing_key)
if not run_program(["./Tools/scripts/signing/make_secure_bl.py", bl_file, args.signing_key]):
print("Failed to sign bootloader for %s" % board)
sys.exit(1)
if not run_program(["./Tools/scripts/signing/make_secure_bl.py", elf_file, args.signing_key]):
print("Failed to sign ELF bootloader for %s" % board)
sys.exit(1)
if not run_program([sys.executable, "Tools/scripts/bin2hex.py", "--offset", "0x08000000", bl_file, hex_file]):
failed_boards.add(board)
continue
print("Created %s" % hex_file)
if len(failed_boards):
print("Failed boards: %s" % list(failed_boards))