1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 11:52:44 +01:00
RIOT/dist/tools/compile_commands/compile_commands.py
Marian Buschsieweke a5f52cbbb7
dist/tools/compile_commands: fix error handling
detect_includes_and_version_gcc() previously only detected the includes,
but has been extended to also return the version. This is done by
returning a tuple, with the first item being the list of include paths,
and the second being the version. In the error handling the script still
returns only an empty list of includes, but not an empty version. This
fixes the issue.
2022-06-14 12:32:22 +02:00

329 lines
12 KiB
Python
Executable File

#!/usr/bin/python3
"""
Command line utility to generate compile_commands.json for RIOT applications
"""
import argparse
import json
import os
import re
import shlex
import subprocess
import sys
REGEX_VERSION = re.compile(r"\ngcc version ([^ ]+)")
REGEX_INCLUDES = r"^#include <\.\.\.> search starts here:$((?:\n|\r|.)*?)^End of search list\.$"
REGEX_INCLUDES = re.compile(REGEX_INCLUDES, re.MULTILINE)
def detect_includes_and_version_gcc(compiler):
"""
Runs the given compiler with -v -E on an no-op compilation unit and parses
the built-in include search directories and the GCC version from the output
:param compiler: name / path of the compiler to run
:type compiler: str
:return: (list_of_include_paths, version)
:rtype: tuple
"""
try:
with subprocess.Popen([compiler, "-v", "-E", "-"],
stdin=subprocess.PIPE, stdout=subprocess.PIPE,
stderr=subprocess.PIPE) as proc:
inputdata = b"typedef int dont_be_pedantic;"
_, stderrdata = proc.communicate(input=inputdata)
except FileNotFoundError:
msg = f"Compiler {compiler} not found, not adding system include paths\n"
sys.stderr.write(msg)
return ([], "")
stderrdata = stderrdata.decode("utf-8")
version = REGEX_VERSION.search(stderrdata).group(1)
tmp = [os.path.abspath(p) for p in REGEX_INCLUDES.search(stderrdata).group(1).split()]
# the include path containing newlib.h must come prior to the gcc headers
# in the include list to avoid mischief from happening
newlib_path = ""
includes = []
for path in tmp:
if os.path.exists(os.path.join(path, "newlib.h")):
newlib_path = path
includes.append(path)
break
for path in tmp:
if path != newlib_path:
includes.append(path)
return (includes, version)
def detect_libstdcxx_includes(compiler, includes, version):
"""
Tries to detect the g++ libstdc++ built-in include search directories using
black magic and adds them to the list given in includes
:param compiler: Name or path of the compiler
:type compiler: str
:param includes: List of include directories
:type includes: list of str
:param version: Version of g++
:type version: str
"""
for path in includes:
cxx_lib = os.path.join(path, "c++", version)
if os.path.exists(cxx_lib):
includes.append(cxx_lib)
triple = os.path.basename(compiler)[0:-4]
cxx_extra = os.path.join(cxx_lib, triple)
if os.path.exists(cxx_extra):
includes.append(cxx_extra)
break
def detect_built_in_includes(compiler, args):
"""
Tries to detect the built-in include search directories of the given
compiler
:param compiler: Name or path of the compiler
:type compiler: str
:param args: Command line arguments
:return: List of built-in include directories
:rtype: list of str
"""
if compiler.endswith('-gcc'):
includes, version = detect_includes_and_version_gcc(compiler)
elif compiler.endswith('-g++'):
includes, version = detect_includes_and_version_gcc(compiler)
if args.add_libstdcxx_includes:
detect_libstdcxx_includes(compiler, includes, version)
elif compiler in ('clang', 'clang++', 'gcc', 'g++'):
# clang / clang++ doesn't have any magic include search dirs built in,
# so we don't need to detect them.
# for host gcc/g++ we don't need to detect magic include dirs either.
includes = []
else:
msg = f"Warning: Cannot detect default include search paths for {compiler}\n"
sys.stderr.write(msg)
includes = []
return includes
class CompilationDetails: # pylint: disable=too-few-public-methods,too-many-instance-attributes
"""
Representation of the compilation details stored by RIOT's build system.
:param path: Path of the module to read compilation details from
:type path: str
"""
def __init__(self, path): # pylint: disable=too-many-branches
with open(os.path.join(path, "compile_cmds.txt"), "r") as file:
for line in file.read().splitlines():
if line.startswith("SRC: "):
self.src_c = line.lstrip("SRC: ").split()
elif line.startswith("SRC_NO_LTO: "):
self.src_c_no_lto = line.lstrip("SRC_NO_LTO: ").split()
elif line.startswith("SRCXX: "):
self.src_cxx = line.lstrip("SRCXX: ").split()
elif line.startswith("CURDIR: "):
self.dir = line.lstrip("CURDIR: ").strip()
elif line.startswith("CFLAGS: "):
self.cflags = shlex.split(line.lstrip("CFLAGS: "))
elif line.startswith("LTOFLAGS: "):
self.ltoflags = shlex.split(line.lstrip("LTOFLAGS: "))
elif line.startswith("INCLUDES: "):
self.includes = shlex.split(line.lstrip("INCLUDES: "))
elif line.startswith("CXXFLAGS: "):
self.cxxflags = shlex.split(line.lstrip("CXXFLAGS: "))
elif line.startswith("CXXINCLUDES: "):
self.cxxincludes = shlex.split(line.lstrip("CXXINCLUDES: "))
elif line.startswith("CC: "):
self.cc = line.lstrip("CC: ").strip() # pylint: disable=invalid-name
elif line.startswith("CXX: "):
self.cxx = line.lstrip("CXX: ").strip()
elif line.startswith("TARGET_ARCH: "):
self.target_arch = line.lstrip("TARGET_ARCH: ").strip()
elif line.startswith("TARGET_ARCH_LLVM: "):
self.target_arch_llvm = line.lstrip("TARGET_ARCH_LLVM: ").strip()
class State: # pylint: disable=too-few-public-methods
"""
Entity to store the current programs state
"""
def __init__(self):
self.def_includes = dict()
self.is_first = True
def get_built_in_include_flags(compiler, state, args):
"""
Get built-in include search directories as parameter list.
:param compiler: Name or path of the compiler to get the include search
dirs from
:type compiler: str
:param state: state of the program
:param args: command line arguments
:return: The -isystem <...> compiler flags for the built-in include search
dirs as list
:rtype: list
"""
result = []
if compiler not in state.def_includes:
state.def_includes[compiler] = detect_built_in_includes(compiler, args)
for include in state.def_includes[compiler]:
result.append('-isystem')
result.append(include)
return result
def write_compile_command(state, compiler, src, flags, cdetails, path):
"""
Write the compile command for the given source file with the given
parameters to stdout
:param state: state of the program
:param compiler: the C/C++ compiler used
:type compiler: str
:param src: the file to compiler
:type src: str
:param flags: flags used for compiler invocation
:type flags: list of str
:param cetails: compilation details
:type cdetails: CompilationDetails
:param path: the output path
"type path: str
"""
if state.is_first:
state.is_first = False
else:
sys.stdout.write(",\n")
obj = os.path.splitext(src)[0] + ".o"
arguments = [compiler, '-DRIOT_FILE_RELATIVE="' + os.path.join(cdetails.dir, src) + '"',
'-DRIOT_FILE_NOPATH="' + src + '"']
arguments += flags
if '-c' in arguments:
# bindgen is unhappy with multiple -c (that would be created by the -c
# added later) and even with the -c showing up anywhere between other
# arguments.
assert arguments.count('-c') == 1, "Spurious duplicate -c arguments"
arguments.remove('-c')
arguments += ['-MQ', obj, '-MD', '-MP', '-c', '-o', obj, src]
entry = {
'arguments': arguments,
'directory': cdetails.dir,
'file': os.path.join(cdetails.dir, src),
'output': os.path.join(path, obj)
}
sys.stdout.write(json.dumps(entry, indent=2))
def generate_module_compile_commands(path, state, args):
"""
Generate section of compile_commands.json for the module in path and write
it to stdout.
:param path: path of the module's bin folder to emit the
compile_commands.json chunk for
:type path: str
:param state: state of the program
:param args: command line arguments
"""
cdetails = CompilationDetails(path)
for flag in args.filter_out:
try:
cdetails.cflags.remove(flag)
except ValueError:
pass
try:
cdetails.cxxflags.remove(flag)
except ValueError:
pass
if args.clangd:
cdetails.cflags.append('-Wno-unknown-warning-option')
c_extra_includes = []
cxx_extra_includes = []
if args.add_built_in_includes:
c_extra_includes = get_built_in_include_flags(cdetails.cc, state, args)
cxx_extra_includes = get_built_in_include_flags(cdetails.cxx, state, args)
if args.clangd:
if cdetails.target_arch_llvm:
cdetails.cflags += ['-target', cdetails.target_arch_llvm]
cdetails.cxxflags += ['-target', cdetails.target_arch_llvm]
elif cdetails.target_arch:
cdetails.cflags += ['-target', cdetails.target_arch]
cdetails.cxxflags += ['-target', cdetails.target_arch]
for src in cdetails.src_c:
compiler = 'clang' if args.clangd else cdetails.cc
flags = cdetails.cflags + cdetails.ltoflags + cdetails.includes + c_extra_includes
write_compile_command(state, compiler, src, flags, cdetails, path)
for src in cdetails.src_c_no_lto:
compiler = 'clang' if args.clangd else cdetails.cc
flags = cdetails.cflags + cdetails.includes + c_extra_includes
write_compile_command(state, compiler, src, flags, cdetails, path)
for src in cdetails.src_cxx:
compiler = 'clang++' if args.clangd else cdetails.cxx
flags = cdetails.cxxflags + cdetails.cxxincludes + cdetails.includes + cxx_extra_includes
write_compile_command(state, compiler, src, flags, cdetails, path)
def generate_compile_commands(args):
"""
Generate the compile_commands.json content and write them to stdout
:param args: command line arguments
"""
state = State()
sys.stdout.write("[\n")
for module in os.scandir(args.path):
if module.is_dir() and os.path.isfile(os.path.join(module.path, 'compile_cmds.txt')):
generate_module_compile_commands(module.path, state, args)
sys.stdout.write("\n]\n")
if __name__ == '__main__':
parser = argparse.ArgumentParser(description='Generate compile_commands.json for RIOT apps')
parser.add_argument('path', metavar='PATH', type=str,
help='Bin path, usually \'<APP>/bin/<BOARD>\'')
parser.add_argument('--add-built-in-includes', default=False, action='store_const', const=True,
help='Explicitly add built in include search directories with -I<PATH> ' +
'options')
parser.add_argument('--add-libstdcxx-includes', default=False, action='store_const', const=True,
help='Explicitly add libstdc++ include search directories with -I<PATH> ' +
'options')
parser.add_argument('--filter-out', type=str, default=[], action='append',
help='Drop the given flag, if present (repeatable)')
parser.add_argument('--clangd', default=False, action='store_const', const=True,
help='Shorthand for --add-built-in-includes --add-libstdxx-includes ' +
'and some CFLAG adjustments throughy --filter-out, and ignores ' +
'unknown warning flags')
_args = parser.parse_args()
if _args.clangd:
_args.add_built_in_includes = True
_args.add_libstdcxx_includes = True
_args.filter_out = ['-mno-thumb-interwork',
# Only even included for versions of GCC that support it
'-misa-spec=2.2',
'-malign-data=natural',
# Only supported starting with clang 11
'-msmall-data-limit=8',
]
generate_compile_commands(_args)