1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
19351: tools/compile_like_murdock: Add features r=MrKevinWeiss a=MrKevinWeiss

### Contribution description

This adds a few nice-to-have features:
- The `-b all` allows all supported boards to build
- The `-v, -vv, -vvv` allows different levels of verbosity, the most useful might be `-v` which filters only the kconfig diffs
- All warning or STDERR based output from `info-*` are hidden
- The `-m` that only runs a diff of modules and packages, skipping any compilation

### Testing procedure

In induce a mismatch and check the output with `-v`, `-vv` options, clean it up and see the full output with `-vvv`:
```
./dist/tools/compile_test/compile_like_murdock.py -a APP -d -b BOARD -v
```

Check all supported boards are listed with
```
./dist/tools/compile_test/compile_like_murdock.py -a examples/hello-world/ -d -b all
```


### Issues/PRs references



Co-authored-by: MrKevinWeiss <weiss.kevin604@gmail.com>
This commit is contained in:
bors[bot] 2023-03-09 06:21:09 +00:00 committed by GitHub
commit 738e0c8318
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -134,41 +134,83 @@ def _end(sec, job):
print(f"You could have {val} {thing}") print(f"You could have {val} {thing}")
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")
def _all_apps(cwd): def _all_apps(cwd):
cmd = ['make', 'info-applications', '--no-print-directory'] cmd = ('make', 'info-applications', '--no-print-directory')
out = subprocess.check_output(cmd, cwd=cwd) return __exec_cmd(cmd, cwd=cwd).split()
return out.decode("utf-8", errors="replace").split()
def _supported_boards(boards, env, cwd): def _supported_boards(boards, env, cwd, all_boards=False):
cmd = ['make', 'info-boards-supported', '--no-print-directory'] cmd = ('make', 'info-boards-supported', '--no-print-directory')
out = subprocess.check_output(cmd, env=env, cwd=cwd) supported_boards = __exec_cmd(cmd, env=env, cwd=cwd).split()
supported_boards = out.decode("utf-8", errors="replace").split() if all_boards:
return supported_boards
return [brd for brd in supported_boards if brd in boards] return [brd for brd in supported_boards if brd in boards]
def _supported_boards_from_cpu(cpu, env, cwd): def _supported_boards_from_cpu(cpu, env, cwd):
cmd = (f'FEATURES_REQUIRED=cpu_{cpu} make info-boards-supported ' cmd = (f'FEATURES_REQUIRED=cpu_{cpu} make info-boards-supported '
'--no-print-directory') '--no-print-directory')
out = subprocess.check_output(cmd, shell=True, env=env, cwd=cwd) __exec_cmd(cmd, shell=True, env=env, cwd=cwd).split()
return out.decode("utf-8", errors="replace").split()
def _build(app, board, jobs, env, cwd): 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} ' cmd = (f'/bin/bash -c "source .murdock; JOBS={jobs} '
f'compile {app} {board}:gnu"') f'kconfig_module_packages_diff {board} {app}"')
try: try:
out = subprocess.check_output(cmd, env=env, shell=True, out = __exec_cmd(cmd, shell=True, env=env, cwd=cwd,
cwd=cwd, stderr=subprocess.STDOUT) stderr=subprocess.STDOUT)
out = out.decode("utf-8", errors="replace") if args.very_very_verbose:
print(out)
print(f"{app: <30} {board: <30} PASS") print(f"{app: <30} {board: <30} PASS")
except subprocess.CalledProcessError as err: except subprocess.CalledProcessError as err:
err.output = err.output.decode("utf-8", errors="replace") err.output = err.output.decode("utf-8", errors="replace")
lines = err.output.split("\n") lines = err.output.split("\n")
_print_module_or_pkg_mismatch(app, board, lines, args)
def _build(app, board, jobs, env, cwd, args):
cmd = (f'/bin/bash -c "source .murdock; JOBS={jobs} '
f'compile {app} {board}:gnu"')
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")
if args.very_very_verbose or args.very_verbose:
print(err.output)
# Check for known diff error outputs
if lines[-3].startswith('< ') or lines[-3].startswith('> '): if lines[-3].startswith('< ') or lines[-3].startswith('> '):
print(f"{app: <30} {board: <30} FAIL: Kconfig module or pkg " _print_module_or_pkg_mismatch(app, board, lines, args)
"mismatch")
elif "mismatch" in err.output: elif "mismatch" in err.output:
print(f"{app: <30} {board: <30} FAIL: Kconfig hash mismatch") print(f"{app: <30} {board: <30} FAIL: Kconfig hash mismatch")
else: else:
@ -181,7 +223,8 @@ def main():
help=("Optional boards list, will test all supported " help=("Optional boards list, will test all supported "
"boards on the list. Will override the cpu " "boards on the list. Will override the cpu "
"filter. If empty, a subset of boards will be " "filter. If empty, a subset of boards will be "
"selected for you...")) "selected for you. If 'all' then it will test "
"all supported boards."))
parser.add_argument("-c", "--cpu", type=str, parser.add_argument("-c", "--cpu", type=str,
help=("Optional filter for all supported boards " help=("Optional filter for all supported boards "
"belonging to the cpu family, for example, " "belonging to the cpu family, for example, "
@ -197,6 +240,15 @@ def main():
" without spending super long to compile them")) " without spending super long to compile them"))
parser.add_argument("-j", "--jobs", type=int, default=4, parser.add_argument("-j", "--jobs", type=int, default=4,
help=("The amount of jobs to use when compiling.")) help=("The amount of jobs to use when compiling."))
parser.add_argument("-m", "--modules-packages", action="store_true",
help=("Only check the diff of modules and packages."))
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."))
args = parser.parse_args() args = parser.parse_args()
start_time = datetime.datetime.now() start_time = datetime.datetime.now()
@ -217,13 +269,19 @@ def main():
if args.cpu: if args.cpu:
target_boards = _supported_boards_from_cpu(args.cpu, full_env, target_boards = _supported_boards_from_cpu(args.cpu, full_env,
test_dir) test_dir)
elif args.boards[0] == "all":
target_boards = _supported_boards(boards, full_env, test_dir, True)
else: else:
target_boards = _supported_boards(boards, full_env, test_dir) target_boards = _supported_boards(boards, full_env, test_dir,
False)
for board in target_boards: for board in target_boards:
if args.dry_run: if args.dry_run:
print(f"{app: <30} {board: <30}") print(f"{app: <30} {board: <30}")
elif args.modules_packages:
_modules_packages(app, board, args.jobs, full_env, riot_dir,
args)
else: else:
_build(app, board, args.jobs, full_env, riot_dir) _build(app, board, args.jobs, full_env, riot_dir, args)
elapse_time = datetime.datetime.now() - start_time elapse_time = datetime.datetime.now() - start_time
_end(elapse_time.total_seconds(), args.jobs) _end(elapse_time.total_seconds(), args.jobs)