1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/dist/tools/compile_commands/compile_commands.py
Marian Buschsieweke a07dac9ad1
build system: add new compile-commands make target
By running make compile-commands a `compile_commands.json` in the RIOT base
directory. With the environment variable `COMPILE_COMMANDS` the path of
this file can be changed to a custom location.

The `compile_commands.json` will contain the exact compile command, but
as additional flag `-I/usr/$(TARGET)/include` is added to work around
`clangd` not being able to locate the newlib system headers. The
additional includes can be overwritten using the environment variable
`COMPILE_COMMANDS_EXTRA_INCLUDES`.
2021-04-14 14:51:46 +02:00

293 lines
11 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 = "Compiler {} not found, not adding system include paths\n".format(compiler)
sys.stderr.write(msg)
return []
stderrdata = stderrdata.decode("utf-8")
version = REGEX_VERSION.search(stderrdata).group(1)
includes = [os.path.abspath(p) for p in REGEX_INCLUDES.search(stderrdata).group(1).split()]
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 = "Warning: Cannot detect default include search paths for {}\n".format(compiler)
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
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
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 ' +
'--filter-out=-Wformat-truncation --filter-out=-Wformat-overflow ' +
'--filter-out=-mno-thumb-interwork')
_args = parser.parse_args()
if _args.clangd:
_args.add_built_in_includes = True
_args.add_libstdcxx_includes = True
_args.filter_out = ['-Wformat-truncation', '-Wformat-overflow', '-mno-thumb-interwork']
generate_compile_commands(_args)