2022-10-25 22:10:16 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
# Copyright (C) 2022 HAW Hamburg
|
|
|
|
#
|
|
|
|
# This library is free software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU Lesser General Public
|
|
|
|
# License as published by the Free Software Foundation; either
|
|
|
|
# version 2.1 of the License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This library is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
# Lesser General Public License for more details.
|
|
|
|
#
|
|
|
|
"""Helper script to pre-test murdock build conditions.
|
|
|
|
|
|
|
|
This is intended to compile like murdock giving control for only a subset of
|
|
|
|
boards or applications. One can use this if only a limited change should be
|
|
|
|
build tested. Defaults boards and apps are selected to be an early warning
|
|
|
|
if something is wrong.
|
|
|
|
|
|
|
|
This should be used before triggering much larger murdock builds.
|
|
|
|
|
|
|
|
The following use cases are:
|
|
|
|
|
|
|
|
I made a change to something in the stm32 clocks...
|
|
|
|
`./compile_like_murdock.py -c stm32`
|
|
|
|
|
|
|
|
I changed a driver the DHT driver
|
2023-05-13 18:45:55 +02:00
|
|
|
`./compile_like_murdock.py -a tests/drivers/dht tests/drivers/saul`
|
2022-10-25 22:10:16 +02:00
|
|
|
|
|
|
|
I changed a nucleo-f103rb board...
|
|
|
|
`./compile_like_murdock.py -a all -b nucleo-f103rb`
|
|
|
|
"""
|
|
|
|
import argparse
|
|
|
|
import os
|
|
|
|
import subprocess
|
|
|
|
import pathlib
|
|
|
|
import random
|
|
|
|
import datetime
|
|
|
|
|
|
|
|
|
|
|
|
_RATIOS = [
|
|
|
|
('digits of pi calculated', 4423213),
|
|
|
|
('bitcoins mined', 0.000000077),
|
|
|
|
('meters driven in a Tesla', 5),
|
|
|
|
('liters of maple syrup refined', 0.00004)
|
|
|
|
]
|
|
|
|
|
|
|
|
DEFAULT_APPS = [
|
|
|
|
"examples/hello-world",
|
2023-05-13 18:45:55 +02:00
|
|
|
"tests/drivers/mtd_mapper",
|
|
|
|
"tests/drivers/saul",
|
2023-05-06 07:48:58 +02:00
|
|
|
"tests/pkg/tinyusb_cdc_msc",
|
2023-05-13 18:23:57 +02:00
|
|
|
"tests/sys/shell/",
|
2022-10-25 22:10:16 +02:00
|
|
|
]
|
|
|
|
|
|
|
|
DEFAULT_BOARDS = [
|
|
|
|
'adafruit-itsybitsy-m4',
|
|
|
|
'arduino-due',
|
|
|
|
'arduino-leonardo',
|
|
|
|
'arduino-mkrzero',
|
|
|
|
'arduino-mega2560',
|
|
|
|
'arduino-nano',
|
|
|
|
'atxmega-a1-xplained',
|
|
|
|
'atxmega-a3bu-xplained',
|
|
|
|
'avr-rss2',
|
|
|
|
'avsextrem',
|
|
|
|
'bastwan',
|
|
|
|
'bluepill',
|
|
|
|
'b-l475e-iot01a',
|
|
|
|
'cc1352-launchpad',
|
|
|
|
'cc2650-launchpad',
|
|
|
|
'derfmega128',
|
|
|
|
'dwm1001',
|
|
|
|
'esp32-ci',
|
|
|
|
'esp32-heltec-lora32-v2',
|
|
|
|
'esp32-olimex-evb',
|
|
|
|
'esp32c3-ci',
|
|
|
|
'esp32c3-devkit',
|
|
|
|
'esp32s3-ci',
|
|
|
|
'esp32s3-devkit',
|
|
|
|
'esp32s2-ci',
|
|
|
|
'esp32s2-devkit',
|
|
|
|
'esp8266-ci',
|
|
|
|
'esp8266-esp-12x',
|
|
|
|
'hamilton',
|
|
|
|
'hifive1',
|
|
|
|
'mbed_lpc1768',
|
|
|
|
'mega-xplained',
|
|
|
|
'microbit',
|
|
|
|
'mulle',
|
|
|
|
'native',
|
|
|
|
'nrf52840dk',
|
|
|
|
'nrf9160dk',
|
|
|
|
'nucleo-f072rb',
|
|
|
|
'nucleo-f103rb',
|
|
|
|
'nucleo-f207zg',
|
|
|
|
'nucleo-f334r8',
|
|
|
|
'nucleo-f429zi',
|
|
|
|
'nucleo-f767zi',
|
|
|
|
'nucleo-g071rb',
|
|
|
|
'nucleo-g474re',
|
|
|
|
'nucleo-l011k4',
|
|
|
|
'nucleo-l073rz',
|
|
|
|
'nucleo-l152re',
|
|
|
|
'nucleo-l433rc',
|
|
|
|
'nucleo-l552ze-q',
|
|
|
|
'pba-d-01-kw2x',
|
|
|
|
'p-nucleo-wb55',
|
|
|
|
'qn9080dk',
|
|
|
|
'remote-revb',
|
|
|
|
'same54-xpro',
|
|
|
|
'samr21-xpro',
|
|
|
|
'seeedstudio-gd32',
|
|
|
|
'slstk3400a',
|
|
|
|
'sltb001a',
|
|
|
|
'slwstk6220a',
|
|
|
|
'stm32f723e-disco',
|
|
|
|
'stm32mp157c-dk2',
|
|
|
|
'waspmote-pro',
|
|
|
|
'waveshare-nrf52840-eval-kit',
|
|
|
|
'weact-f401ce',
|
|
|
|
'z1'
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
def _end(sec, job):
|
|
|
|
time_str = str(datetime.timedelta(seconds=sec))
|
|
|
|
thing, ratio = random.choice(_RATIOS)
|
|
|
|
val = ratio * sec * job
|
|
|
|
print(f"This took {time_str}...")
|
|
|
|
print(f"You could have {val} {thing}")
|
|
|
|
|
|
|
|
|
2023-03-06 15:13:46 +01:00
|
|
|
def __exec_cmd(cmd, shell=False, env=None, cwd=None,
|
|
|
|
stderr=subprocess.DEVNULL):
|
|
|
|
out = subprocess.run(
|
|
|
|
cmd,
|
|
|
|
shell=shell,
|
|
|
|
env=env,
|
|
|
|
cwd=cwd,
|
|
|
|
stderr=stderr,
|
|
|
|
check=True,
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
).stdout
|
|
|
|
return out.decode("utf-8", errors="replace")
|
|
|
|
|
|
|
|
|
2022-10-25 22:10:16 +02:00
|
|
|
def _all_apps(cwd):
|
2023-03-06 15:13:46 +01:00
|
|
|
cmd = ('make', 'info-applications', '--no-print-directory')
|
|
|
|
return __exec_cmd(cmd, cwd=cwd).split()
|
2022-10-25 22:10:16 +02:00
|
|
|
|
|
|
|
|
2023-03-06 13:59:46 +01:00
|
|
|
def _supported_boards(boards, env, cwd, all_boards=False):
|
2023-03-06 15:13:46 +01:00
|
|
|
cmd = ('make', 'info-boards-supported', '--no-print-directory')
|
|
|
|
supported_boards = __exec_cmd(cmd, env=env, cwd=cwd).split()
|
2023-03-06 13:59:46 +01:00
|
|
|
if all_boards:
|
|
|
|
return supported_boards
|
2022-10-25 22:10:16 +02:00
|
|
|
return [brd for brd in supported_boards if brd in boards]
|
|
|
|
|
|
|
|
|
|
|
|
def _supported_boards_from_cpu(cpu, env, cwd):
|
|
|
|
cmd = (f'FEATURES_REQUIRED=cpu_{cpu} make info-boards-supported '
|
|
|
|
'--no-print-directory')
|
2023-03-06 15:13:46 +01:00
|
|
|
__exec_cmd(cmd, shell=True, env=env, cwd=cwd).split()
|
|
|
|
|
|
|
|
|
|
|
|
def _print_module_or_pkg_mismatch(app, board, lines, args):
|
|
|
|
if args.verbose:
|
|
|
|
for line in lines:
|
|
|
|
# Reprint the < and > from diff to show if kconfig or make are
|
|
|
|
# responsible
|
|
|
|
if line.startswith('< '):
|
|
|
|
print(" make has:", line[2:])
|
|
|
|
if line.startswith('> '):
|
|
|
|
print(" kconfig has:", line[2:])
|
|
|
|
print(f"{app: <30} {board: <30} FAIL: Kconfig module or pkg mismatch")
|
|
|
|
|
|
|
|
|
|
|
|
def _modules_packages(app, board, jobs, env, cwd, args):
|
|
|
|
cmd = (f'/bin/bash -c "source .murdock; JOBS={jobs} '
|
|
|
|
f'kconfig_module_packages_diff {board} {app}"')
|
|
|
|
try:
|
|
|
|
out = __exec_cmd(cmd, shell=True, env=env, cwd=cwd,
|
|
|
|
stderr=subprocess.STDOUT)
|
|
|
|
if args.very_very_verbose:
|
|
|
|
print(out)
|
|
|
|
print(f"{app: <30} {board: <30} PASS")
|
|
|
|
except subprocess.CalledProcessError as err:
|
|
|
|
err.output = err.output.decode("utf-8", errors="replace")
|
|
|
|
lines = err.output.split("\n")
|
|
|
|
_print_module_or_pkg_mismatch(app, board, lines, args)
|
2022-10-25 22:10:16 +02:00
|
|
|
|
|
|
|
|
2023-03-06 14:00:21 +01:00
|
|
|
def _build(app, board, jobs, env, cwd, args):
|
2022-10-25 22:10:16 +02:00
|
|
|
cmd = (f'/bin/bash -c "source .murdock; JOBS={jobs} '
|
|
|
|
f'compile {app} {board}:gnu"')
|
|
|
|
try:
|
2023-03-06 15:13:46 +01:00
|
|
|
out = __exec_cmd(cmd, shell=True, env=env, cwd=cwd,
|
|
|
|
stderr=subprocess.STDOUT)
|
2023-03-06 14:00:21 +01:00
|
|
|
if args.very_very_verbose:
|
|
|
|
print(out)
|
2022-10-25 22:10:16 +02:00
|
|
|
print(f"{app: <30} {board: <30} PASS")
|
|
|
|
except subprocess.CalledProcessError as err:
|
|
|
|
err.output = err.output.decode("utf-8", errors="replace")
|
|
|
|
lines = err.output.split("\n")
|
2023-03-06 14:00:21 +01:00
|
|
|
if args.very_very_verbose or args.very_verbose:
|
|
|
|
print(err.output)
|
2023-03-06 15:13:46 +01:00
|
|
|
# Check for known diff error outputs
|
2022-10-25 22:10:16 +02:00
|
|
|
if lines[-3].startswith('< ') or lines[-3].startswith('> '):
|
2023-03-06 15:13:46 +01:00
|
|
|
_print_module_or_pkg_mismatch(app, board, lines, args)
|
2022-10-25 22:10:16 +02:00
|
|
|
elif "mismatch" in err.output:
|
|
|
|
print(f"{app: <30} {board: <30} FAIL: Kconfig hash mismatch")
|
|
|
|
else:
|
|
|
|
print(f"{app: <30} {board: <30} FAIL")
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument("-b", "--boards", nargs="+",
|
|
|
|
help=("Optional boards list, will test all supported "
|
|
|
|
"boards on the list. Will override the cpu "
|
|
|
|
"filter. If empty, a subset of boards will be "
|
2023-03-06 13:59:46 +01:00
|
|
|
"selected for you. If 'all' then it will test "
|
|
|
|
"all supported boards."))
|
2022-10-25 22:10:16 +02:00
|
|
|
parser.add_argument("-c", "--cpu", type=str,
|
|
|
|
help=("Optional filter for all supported boards "
|
|
|
|
"belonging to the cpu family, for example, "
|
|
|
|
"stm32 or esp32."))
|
|
|
|
parser.add_argument("-n", "--no-nightly", action="store_false",
|
|
|
|
help=("Prevent running as nightly, by default NIGHTLY "
|
|
|
|
"is set to 1, making more extensive testing."))
|
|
|
|
parser.add_argument("-a", "--apps", nargs="+",
|
|
|
|
help=("A list of apps to test on the supported boards."
|
|
|
|
" If empty we will choose what is tested."))
|
|
|
|
parser.add_argument("-d", "--dry-run", action="store_true",
|
|
|
|
help=("Show each of the boards and apps to be compiled"
|
|
|
|
" without spending super long to compile them"))
|
|
|
|
parser.add_argument("-j", "--jobs", type=int, default=4,
|
|
|
|
help=("The amount of jobs to use when compiling."))
|
2023-03-06 15:13:46 +01:00
|
|
|
parser.add_argument("-m", "--modules-packages", action="store_true",
|
|
|
|
help=("Only check the diff of modules and packages."))
|
2023-03-06 14:00:21 +01:00
|
|
|
parser.add_argument("-v", "--verbose", action="store_true",
|
|
|
|
help=("Shows mismatch info."))
|
|
|
|
parser.add_argument("-vv", "--very-verbose", action="store_true",
|
|
|
|
help=("Shows extra output on failures."))
|
|
|
|
parser.add_argument("-vvv", "--very-very-verbose", action="store_true",
|
|
|
|
help=("Shows all output info."))
|
|
|
|
|
2022-10-25 22:10:16 +02:00
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
start_time = datetime.datetime.now()
|
|
|
|
full_env = os.environ.copy()
|
|
|
|
full_env["NIGHTLY"] = str(int(args.no_nightly))
|
|
|
|
dir = pathlib.Path(__file__).parent.resolve()
|
|
|
|
riot_dir = pathlib.Path(pathlib.PurePath(dir, "../../../")).resolve()
|
|
|
|
boards = args.boards or DEFAULT_BOARDS
|
|
|
|
apps = args.apps or DEFAULT_APPS
|
|
|
|
if 'all' in apps:
|
|
|
|
apps = _all_apps(riot_dir)
|
|
|
|
|
|
|
|
for app in apps:
|
|
|
|
test_dir = str(pathlib.PurePath(riot_dir, app))
|
|
|
|
if not pathlib.Path(test_dir).exists():
|
|
|
|
print(f"{test_dir: <60}SKIP: Does not exists (typo?)")
|
|
|
|
continue
|
|
|
|
if args.cpu:
|
|
|
|
target_boards = _supported_boards_from_cpu(args.cpu, full_env,
|
|
|
|
test_dir)
|
2023-03-09 14:10:48 +01:00
|
|
|
elif args.boards and args.boards[0] == "all":
|
2023-03-06 13:59:46 +01:00
|
|
|
target_boards = _supported_boards(boards, full_env, test_dir, True)
|
2022-10-25 22:10:16 +02:00
|
|
|
else:
|
2023-03-06 15:13:46 +01:00
|
|
|
target_boards = _supported_boards(boards, full_env, test_dir,
|
|
|
|
False)
|
2022-10-25 22:10:16 +02:00
|
|
|
for board in target_boards:
|
|
|
|
if args.dry_run:
|
|
|
|
print(f"{app: <30} {board: <30}")
|
2023-03-06 15:13:46 +01:00
|
|
|
elif args.modules_packages:
|
|
|
|
_modules_packages(app, board, args.jobs, full_env, riot_dir,
|
|
|
|
args)
|
2022-10-25 22:10:16 +02:00
|
|
|
else:
|
2023-03-06 14:00:21 +01:00
|
|
|
_build(app, board, args.jobs, full_env, riot_dir, args)
|
2022-10-25 22:10:16 +02:00
|
|
|
elapse_time = datetime.datetime.now() - start_time
|
|
|
|
_end(elapse_time.total_seconds(), args.jobs)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|