Path: blob/master/Tools/scripts/build_script_base.py
9355 views
from __future__ import annotations12'''3Base class for ArduPilot build scripts providing common utilities45AP_FLAKE8_CLEAN6'''78import os9import pathlib10import string11import subprocess12import sys13import time141516class BuildScriptBase:17"""Base class for build scripts with common utilities for running programs"""1819def __init__(self):20self.tmpdir = None # Can be set by subclasses that need it2122def run_program(self, prefix, cmd_list, show_output=True, env=None, show_output_on_error=True, show_command=None, cwd="."):23if show_command is None:24show_command = True2526cmd = " ".join(cmd_list)27if cwd is None:28cwd = "."29command_debug = f"Running ({cmd}) in ({cwd})"30process_failure_content = command_debug + "\n"31if show_command:32self.progress(command_debug)3334p = subprocess.Popen(35cmd_list,36stdin=None,37stdout=subprocess.PIPE,38close_fds=True,39stderr=subprocess.STDOUT,40cwd=cwd,41env=env)42output = ""43while True:44x = p.stdout.readline()45if len(x) == 0:46returncode = os.waitpid(p.pid, 0)47if returncode:48break49# select not available on Windows... probably...50time.sleep(0.1)51continue52x = bytearray(x)53x = filter(lambda x : chr(x) in string.printable, x)54x = "".join([chr(c) for c in x])55output += x56process_failure_content += x57x = x.rstrip()58some_output = "%s: %s" % (prefix, x)59if show_output:60print(some_output)61(_, status) = returncode62if status != 0:63if not show_output and show_output_on_error:64# we were told not to show output, but we just65# failed... so show output...66print(output)67self.progress("Process failed (%s)" %68str(returncode))69try:70path = pathlib.Path(self.tmpdir, f"process-failure-{int(time.time())}")71path.write_text(process_failure_content)72self.progress("Wrote process failure file (%s)" % path)73except Exception:74self.progress("Writing process failure file failed")75raise subprocess.CalledProcessError(76returncode, cmd_list)77return output7879def find_current_git_branch_or_sha1(self):80try:81output = self.run_git(["symbolic-ref", "--short", "HEAD"])82output = output.strip()83return output84except subprocess.CalledProcessError:85pass8687# probably in a detached-head state. Get a sha1 instead:88output = self.run_git(["rev-parse", "--short", "HEAD"])89output = output.strip()90return output9192def find_git_branch_merge_base(self, branch, master_branch):93output = self.run_git(["merge-base", branch, master_branch])94output = output.strip()95return output9697def run_git(self, args, show_output=True, source_dir=None):98'''run git with args git_args; returns git's output'''99cmd_list = ["git"]100cmd_list.extend(args)101return self.run_program("SCB-GIT", cmd_list, show_output=show_output, cwd=source_dir)102103def run_waf(self, args, compiler=None, show_output=True, source_dir=None):104# try to modify the environment so we can consistent builds:105consistent_build_envs = {106"CHIBIOS_GIT_VERSION": "12345678",107"GIT_VERSION": "abcdef",108"GIT_VERSION_EXTENDED": "0123456789abcdef",109"GIT_VERSION_INT": "15",110}111for (n, v) in consistent_build_envs.items():112os.environ[n] = v113114if os.path.exists("waf"):115waf = "./waf"116else:117waf = os.path.join(".", "modules", "waf", "waf-light")118cmd_list = [waf]119cmd_list.extend(args)120env = None121if compiler is not None:122# default to $HOME/arm-gcc, but allow for any path with AP_GCC_HOME environment variable123gcc_home = os.environ.get("AP_GCC_HOME", os.path.join(os.environ["HOME"], "arm-gcc"))124gcc_path = os.path.join(gcc_home, compiler, "bin")125if os.path.exists(gcc_path):126# setup PATH to point at the right compiler, and setup to use ccache127env = os.environ.copy()128env["PATH"] = gcc_path + ":" + env["PATH"]129env["CC"] = "ccache arm-none-eabi-gcc"130env["CXX"] = "ccache arm-none-eabi-g++"131else:132raise Exception("BB-WAF: Missing compiler %s" % gcc_path)133self.run_program("SCB-WAF", cmd_list, env=env, show_output=show_output, cwd=source_dir)134135def progress(self, string):136'''pretty-print progress'''137print(f"{self.progress_prefix()}: {string}", file=sys.stderr)138139140