From: Sascha Hauer <sha@pengutronix.de>
To: Ahmad Fatoum <ahmad@a3f.at>
Cc: barebox@lists.infradead.org
Subject: Re: [PATCH] Kbuild: add compile_commands.json target
Date: Thu, 4 Feb 2021 11:21:05 +0100 [thread overview]
Message-ID: <20210204102105.GZ19583@pengutronix.de> (raw)
In-Reply-To: <20210131194004.3369551-1-ahmad@a3f.at>
On Sun, Jan 31, 2021 at 08:40:04PM +0100, Ahmad Fatoum wrote:
> The JSON compilation database format specification describes a
> compile_commands.json file that lists how translation units are
> compiled by a build system. This makes integration of external tools,
> like IDEs, LSP servers and static analyzers easier.
>
> Import the Linux bits. The database can now be manually generated
> with make compile_commands.json.
>
> Signed-off-by: Ahmad Fatoum <ahmad@a3f.at>
> ---
> Cc: Rouven Czerwinski <rcz@pengutronix.de>
> ---
Applied, thanks
Sascha
> .gitignore | 1 +
> Makefile | 16 +-
> scripts/clang-tools/gen_compile_commands.py | 237 ++++++++++++++++++++
> 3 files changed, 252 insertions(+), 2 deletions(-)
> create mode 100755 scripts/clang-tools/gen_compile_commands.py
>
> diff --git a/.gitignore b/.gitignore
> index 7fa2948bf4a4..d7a37b3c9b39 100644
> --- a/.gitignore
> +++ b/.gitignore
> @@ -87,3 +87,4 @@ GTAGS
> /allno.config
> /allrandom.config
> /allyes.config
> +/compile_commands.json
> diff --git a/Makefile b/Makefile
> index ea1d5dae1c5e..f3c85cff9430 100644
> --- a/Makefile
> +++ b/Makefile
> @@ -541,7 +541,7 @@ endif
> # in addition to whatever we do anyway.
> # Just "make" or "make all" shall build modules as well
>
> -ifneq ($(filter all _all modules,$(MAKECMDGOALS)),)
> +ifneq ($(filter all _all modules %compile_commands.json,$(MAKECMDGOALS)),)
> KBUILD_MODULES := 1
> endif
>
> @@ -1104,7 +1104,7 @@ endif # CONFIG_MODULES
> CLEAN_DIRS += $(MODVERDIR)
> CLEAN_FILES += barebox System.map include/generated/barebox_default_env.h \
> .tmp_version .tmp_barebox* barebox.bin barebox.map barebox.S \
> - .tmp_kallsyms* barebox.ldr \
> + .tmp_kallsyms* barebox.ldr compile_commands.json \
> scripts/bareboxenv-target barebox-flash-image \
> barebox.srec barebox.s5p barebox.ubl barebox.zynq \
> barebox.uimage barebox.spi barebox.kwb barebox.kwbuart \
> @@ -1162,6 +1162,18 @@ distclean: mrproper
> -o -name 'core' \) \
> -type f -print | xargs rm -f
>
> +# Clang Tooling
> +# ---------------------------------------------------------------------------
> +
> +quiet_cmd_gen_compile_commands = GEN $@
> + cmd_gen_compile_commands = $(PYTHON3) $< -a $(AR) -o $@ $(filter-out $<, $(real-prereqs))
> +
> +compile_commands.json: scripts/clang-tools/gen_compile_commands.py \
> + $(BAREBOX_OBJS) $(if $(CONFIG_PBL_IMAGE),$(BAREBOX_PBL_OBJS),) FORCE
> + $(call if_changed,gen_compile_commands)
> +
> +PHONY += compile_commands.json
> +
> # Brief documentation of the typical targets used
> # ---------------------------------------------------------------------------
>
> diff --git a/scripts/clang-tools/gen_compile_commands.py b/scripts/clang-tools/gen_compile_commands.py
> new file mode 100755
> index 000000000000..7ed3919f453a
> --- /dev/null
> +++ b/scripts/clang-tools/gen_compile_commands.py
> @@ -0,0 +1,237 @@
> +#!/usr/bin/env python
> +# SPDX-License-Identifier: GPL-2.0
> +#
> +# Copyright (C) Google LLC, 2018
> +#
> +# Author: Tom Roeder <tmroeder@google.com>
> +#
> +"""A tool for generating compile_commands.json in the Linux kernel."""
> +
> +import argparse
> +import json
> +import logging
> +import os
> +import sys
> +import re
> +import subprocess
> +
> +_DEFAULT_OUTPUT = 'compile_commands.json'
> +_DEFAULT_LOG_LEVEL = 'WARNING'
> +
> +_FILENAME_PATTERN = r'^\..*\.cmd$'
> +_LINE_PATTERN = r'^cmd_[^ ]*\.o := (.* )([^ ]*\.c)$'
> +_VALID_LOG_LEVELS = ['DEBUG', 'INFO', 'WARNING', 'ERROR', 'CRITICAL']
> +
> +
> +def parse_arguments():
> + """Sets up and parses command-line arguments.
> +
> + Returns:
> + log_level: A logging level to filter log output.
> + directory: The work directory where the objects were built.
> + ar: Command used for parsing .a archives.
> + output: Where to write the compile-commands JSON file.
> + paths: The list of files/directories to handle to find .cmd files.
> + """
> + usage = 'Creates a compile_commands.json database from kernel .cmd files'
> + parser = argparse.ArgumentParser(description=usage)
> +
> + directory_help = ('specify the output directory used for the kernel build '
> + '(defaults to the working directory)')
> + parser.add_argument('-d', '--directory', type=str, default='.',
> + help=directory_help)
> +
> + output_help = ('path to the output command database (defaults to ' +
> + _DEFAULT_OUTPUT + ')')
> + parser.add_argument('-o', '--output', type=str, default=_DEFAULT_OUTPUT,
> + help=output_help)
> +
> + log_level_help = ('the level of log messages to produce (defaults to ' +
> + _DEFAULT_LOG_LEVEL + ')')
> + parser.add_argument('--log_level', choices=_VALID_LOG_LEVELS,
> + default=_DEFAULT_LOG_LEVEL, help=log_level_help)
> +
> + ar_help = 'command used for parsing .a archives'
> + parser.add_argument('-a', '--ar', type=str, default='llvm-ar', help=ar_help)
> +
> + paths_help = ('directories to search or files to parse '
> + '(files should be *.o, *.a, or modules.order). '
> + 'If nothing is specified, the current directory is searched')
> + parser.add_argument('paths', type=str, nargs='*', help=paths_help)
> +
> + args = parser.parse_args()
> +
> + return (args.log_level,
> + os.path.abspath(args.directory),
> + args.output,
> + args.ar,
> + args.paths if len(args.paths) > 0 else [args.directory])
> +
> +
> +def cmdfiles_in_dir(directory):
> + """Generate the iterator of .cmd files found under the directory.
> +
> + Walk under the given directory, and yield every .cmd file found.
> +
> + Args:
> + directory: The directory to search for .cmd files.
> +
> + Yields:
> + The path to a .cmd file.
> + """
> +
> + filename_matcher = re.compile(_FILENAME_PATTERN)
> +
> + for dirpath, _, filenames in os.walk(directory):
> + for filename in filenames:
> + if filename_matcher.match(filename):
> + yield os.path.join(dirpath, filename)
> +
> +
> +def to_cmdfile(path):
> + """Return the path of .cmd file used for the given build artifact
> +
> + Args:
> + Path: file path
> +
> + Returns:
> + The path to .cmd file
> + """
> + dir, base = os.path.split(path)
> + return os.path.join(dir, '.' + base + '.cmd')
> +
> +
> +def cmdfiles_for_o(obj):
> + """Generate the iterator of .cmd files associated with the object
> +
> + Yield the .cmd file used to build the given object
> +
> + Args:
> + obj: The object path
> +
> + Yields:
> + The path to .cmd file
> + """
> + yield to_cmdfile(obj)
> +
> +
> +def cmdfiles_for_a(archive, ar):
> + """Generate the iterator of .cmd files associated with the archive.
> +
> + Parse the given archive, and yield every .cmd file used to build it.
> +
> + Args:
> + archive: The archive to parse
> +
> + Yields:
> + The path to every .cmd file found
> + """
> + for obj in subprocess.check_output([ar, '-t', archive]).decode().split():
> + yield to_cmdfile(obj)
> +
> +
> +def cmdfiles_for_modorder(modorder):
> + """Generate the iterator of .cmd files associated with the modules.order.
> +
> + Parse the given modules.order, and yield every .cmd file used to build the
> + contained modules.
> +
> + Args:
> + modorder: The modules.order file to parse
> +
> + Yields:
> + The path to every .cmd file found
> + """
> + with open(modorder) as f:
> + for line in f:
> + ko = line.rstrip()
> + base, ext = os.path.splitext(ko)
> + if ext != '.ko':
> + sys.exit('{}: module path must end with .ko'.format(ko))
> + mod = base + '.mod'
> + # The first line of *.mod lists the objects that compose the module.
> + with open(mod) as m:
> + for obj in m.readline().split():
> + yield to_cmdfile(obj)
> +
> +
> +def process_line(root_directory, command_prefix, file_path):
> + """Extracts information from a .cmd line and creates an entry from it.
> +
> + Args:
> + root_directory: The directory that was searched for .cmd files. Usually
> + used directly in the "directory" entry in compile_commands.json.
> + command_prefix: The extracted command line, up to the last element.
> + file_path: The .c file from the end of the extracted command.
> + Usually relative to root_directory, but sometimes absolute.
> +
> + Returns:
> + An entry to append to compile_commands.
> +
> + Raises:
> + ValueError: Could not find the extracted file based on file_path and
> + root_directory or file_directory.
> + """
> + # The .cmd files are intended to be included directly by Make, so they
> + # escape the pound sign '#', either as '\#' or '$(pound)' (depending on the
> + # kernel version). The compile_commands.json file is not interepreted
> + # by Make, so this code replaces the escaped version with '#'.
> + prefix = command_prefix.replace('\#', '#').replace('$(pound)', '#')
> +
> + # Use os.path.abspath() to normalize the path resolving '.' and '..' .
> + abs_path = os.path.abspath(os.path.join(root_directory, file_path))
> + if not os.path.exists(abs_path):
> + raise ValueError('File %s not found' % abs_path)
> + return {
> + 'directory': root_directory,
> + 'file': abs_path,
> + 'command': prefix + file_path,
> + }
> +
> +
> +def main():
> + """Walks through the directory and finds and parses .cmd files."""
> + log_level, directory, output, ar, paths = parse_arguments()
> +
> + level = getattr(logging, log_level)
> + logging.basicConfig(format='%(levelname)s: %(message)s', level=level)
> +
> + line_matcher = re.compile(_LINE_PATTERN)
> +
> + compile_commands = []
> +
> + for path in paths:
> + # If 'path' is a directory, handle all .cmd files under it.
> + # Otherwise, handle .cmd files associated with the file.
> + # Most of built-in objects are linked via archives (built-in.a or lib.a)
> + # but some objects are linked to vmlinux directly.
> + # Modules are listed in modules.order.
> + if os.path.isdir(path):
> + cmdfiles = cmdfiles_in_dir(path)
> + elif path.endswith('.o'):
> + cmdfiles = cmdfiles_for_o(path)
> + elif path.endswith('.a'):
> + cmdfiles = cmdfiles_for_a(path, ar)
> + elif path.endswith('modules.order'):
> + cmdfiles = cmdfiles_for_modorder(path)
> + else:
> + sys.exit('{}: unknown file type'.format(path))
> +
> + for cmdfile in cmdfiles:
> + with open(cmdfile, 'rt') as f:
> + result = line_matcher.match(f.readline())
> + if result:
> + try:
> + entry = process_line(directory, result.group(1),
> + result.group(2))
> + compile_commands.append(entry)
> + except ValueError as err:
> + logging.info('Could not add line from %s: %s',
> + cmdfile, err)
> +
> + with open(output, 'wt') as f:
> + json.dump(compile_commands, f, indent=2, sort_keys=True)
> +
> +
> +if __name__ == '__main__':
> + main()
> --
> 2.30.0
>
>
> _______________________________________________
> barebox mailing list
> barebox@lists.infradead.org
> http://lists.infradead.org/mailman/listinfo/barebox
>
--
Pengutronix e.K. | |
Steuerwalder Str. 21 | http://www.pengutronix.de/ |
31137 Hildesheim, Germany | Phone: +49-5121-206917-0 |
Amtsgericht Hildesheim, HRA 2686 | Fax: +49-5121-206917-5555 |
_______________________________________________
barebox mailing list
barebox@lists.infradead.org
http://lists.infradead.org/mailman/listinfo/barebox
prev parent reply other threads:[~2021-02-04 10:21 UTC|newest]
Thread overview: 2+ messages / expand[flat|nested] mbox.gz Atom feed top
2021-01-31 19:40 Ahmad Fatoum
2021-02-04 10:21 ` Sascha Hauer [this message]
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20210204102105.GZ19583@pengutronix.de \
--to=sha@pengutronix.de \
--cc=ahmad@a3f.at \
--cc=barebox@lists.infradead.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox