1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/dist/tools/compile_test/compile_like_murdock.py

233 lines
7.5 KiB
Python
Raw Normal View History

#!/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
`./compile_like_murdock.py -a tests/driver_dht tests/saul`
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",
"tests/pkg_tinyusb_cdc_msc",
"tests/mtd_mapper",
"tests/shell",
"tests/saul"
]
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}")
def _all_apps(cwd):
cmd = ['make', 'info-applications', '--no-print-directory']
out = subprocess.check_output(cmd, cwd=cwd, stderr=subprocess.DEVNULL)
return out.decode("utf-8", errors="replace").split()
def _supported_boards(boards, env, cwd):
cmd = ['make', 'info-boards-supported', '--no-print-directory']
out = subprocess.check_output(cmd, env=env, cwd=cwd, stderr=subprocess.DEVNULL)
supported_boards = out.decode("utf-8", errors="replace").split()
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')
out = subprocess.check_output(cmd, shell=True, env=env, cwd=cwd, stderr=subprocess.DEVNULL)
return out.decode("utf-8", errors="replace").split()
def _build(app, board, jobs, env, cwd):
cmd = (f'/bin/bash -c "source .murdock; JOBS={jobs} '
f'compile {app} {board}:gnu"')
try:
out = subprocess.check_output(cmd, env=env, shell=True,
cwd=cwd, stderr=subprocess.STDOUT)
out = out.decode("utf-8", errors="replace")
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")
if lines[-3].startswith('< ') or lines[-3].startswith('> '):
print(f"{app: <30} {board: <30} FAIL: Kconfig module or pkg "
"mismatch")
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 "
"selected for you..."))
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."))
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)
else:
target_boards = _supported_boards(boards, full_env, test_dir)
for board in target_boards:
if args.dry_run:
print(f"{app: <30} {board: <30}")
else:
_build(app, board, args.jobs, full_env, riot_dir)
elapse_time = datetime.datetime.now() - start_time
_end(elapse_time.total_seconds(), args.jobs)
if __name__ == '__main__':
main()