mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-01 00:58:39 +02:00 
			
		
		
		
	 b24520ffa9
			
		
	
	
		b24520ffa9
		
	
	
	
	
		
			
			Add a -checks argument to allow the checks passed to the clang-tool to be set on the command line. Add a pass through -header-filter option. Don't run analysis on non-C or CPP files. Signed-off-by: Ian Rogers <irogers@google.com> Reviewed-by: Nick Desaulniers <ndesaulniers@google.com> Cc: Ravi Bangoria <ravi.bangoria@amd.com> Cc: Mark Rutland <mark.rutland@arm.com> Cc: Yang Jihong <yangjihong1@huawei.com> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Adrian Hunter <adrian.hunter@intel.com> Cc: Arnaldo Carvalho de Melo <acme@kernel.org> Cc: Huacai Chen <chenhuacai@kernel.org> Cc: Jiri Olsa <jolsa@kernel.org> Cc: Nathan Chancellor <nathan@kernel.org> Cc: Alexander Shishkin <alexander.shishkin@linux.intel.com> Cc: Kan Liang <kan.liang@linux.intel.com> Cc: llvm@lists.linux.dev Cc: Ming Wang <wangming01@loongson.cn> Cc: Ingo Molnar <mingo@redhat.com> Cc: Tom Rix <trix@redhat.com> Cc: bpf@vger.kernel.org Cc: linux-kernel@vger.kernel.org Cc: linux-perf-users@vger.kernel.org Link: https://lore.kernel.org/r/20231009183920.200859-4-irogers@google.com Signed-off-by: Namhyung Kim <namhyung@kernel.org>
		
			
				
	
	
		
			100 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			100 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
| #!/usr/bin/env python3
 | |
| # SPDX-License-Identifier: GPL-2.0
 | |
| #
 | |
| # Copyright (C) Google LLC, 2020
 | |
| #
 | |
| # Author: Nathan Huckleberry <nhuck@google.com>
 | |
| #
 | |
| """A helper routine run clang-tidy and the clang static-analyzer on
 | |
| compile_commands.json.
 | |
| """
 | |
| 
 | |
| import argparse
 | |
| import json
 | |
| import multiprocessing
 | |
| import subprocess
 | |
| import sys
 | |
| 
 | |
| 
 | |
| def parse_arguments():
 | |
|     """Set up and parses command-line arguments.
 | |
|     Returns:
 | |
|         args: Dict of parsed args
 | |
|         Has keys: [path, type]
 | |
|     """
 | |
|     usage = """Run clang-tidy or the clang static-analyzer on a
 | |
|         compilation database."""
 | |
|     parser = argparse.ArgumentParser(description=usage)
 | |
| 
 | |
|     type_help = "Type of analysis to be performed"
 | |
|     parser.add_argument("type",
 | |
|                         choices=["clang-tidy", "clang-analyzer"],
 | |
|                         help=type_help)
 | |
|     path_help = "Path to the compilation database to parse"
 | |
|     parser.add_argument("path", type=str, help=path_help)
 | |
| 
 | |
|     checks_help = "Checks to pass to the analysis"
 | |
|     parser.add_argument("-checks", type=str, default=None, help=checks_help)
 | |
|     header_filter_help = "Pass the -header-filter value to the tool"
 | |
|     parser.add_argument("-header-filter", type=str, default=None, help=header_filter_help)
 | |
| 
 | |
|     return parser.parse_args()
 | |
| 
 | |
| 
 | |
| def init(l, a):
 | |
|     global lock
 | |
|     global args
 | |
|     lock = l
 | |
|     args = a
 | |
| 
 | |
| 
 | |
| def run_analysis(entry):
 | |
|     # Disable all checks, then re-enable the ones we want
 | |
|     global args
 | |
|     checks = None
 | |
|     if args.checks:
 | |
|         checks = args.checks.split(',')
 | |
|     else:
 | |
|         checks = ["-*"]
 | |
|         if args.type == "clang-tidy":
 | |
|             checks.append("linuxkernel-*")
 | |
|         else:
 | |
|             checks.append("clang-analyzer-*")
 | |
|             checks.append("-clang-analyzer-security.insecureAPI.DeprecatedOrUnsafeBufferHandling")
 | |
|     file = entry["file"]
 | |
|     if not file.endswith(".c") and not file.endswith(".cpp"):
 | |
|         with lock:
 | |
|             print(f"Skipping non-C file: '{file}'", file=sys.stderr)
 | |
|         return
 | |
|     pargs = ["clang-tidy", "-p", args.path, "-checks=" + ",".join(checks)]
 | |
|     if args.header_filter:
 | |
|         pargs.append("-header-filter=" + args.header_filter)
 | |
|     pargs.append(file)
 | |
|     p = subprocess.run(pargs,
 | |
|                        stdout=subprocess.PIPE,
 | |
|                        stderr=subprocess.STDOUT,
 | |
|                        cwd=entry["directory"])
 | |
|     with lock:
 | |
|         sys.stderr.buffer.write(p.stdout)
 | |
| 
 | |
| 
 | |
| def main():
 | |
|     try:
 | |
|         args = parse_arguments()
 | |
| 
 | |
|         lock = multiprocessing.Lock()
 | |
|         pool = multiprocessing.Pool(initializer=init, initargs=(lock, args))
 | |
|         # Read JSON data into the datastore variable
 | |
|         with open(args.path, "r") as f:
 | |
|             datastore = json.load(f)
 | |
|             pool.map(run_analysis, datastore)
 | |
|     except BrokenPipeError:
 | |
|         # Python flushes standard streams on exit; redirect remaining output
 | |
|         # to devnull to avoid another BrokenPipeError at shutdown
 | |
|         devnull = os.open(os.devnull, os.O_WRONLY)
 | |
|         os.dup2(devnull, sys.stdout.fileno())
 | |
|         sys.exit(1)  # Python exits with error code 1 on EPIPE
 | |
| 
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|     main()
 |