Skip to content

Commit 7cc9b9e

Browse files
authored
Merge pull request #4456 from bruntib/verbatim_args
[feat] Introduce cc-verbatim-args-file
2 parents d0c86b9 + cf42610 commit 7cc9b9e

File tree

14 files changed

+280
-163
lines changed

14 files changed

+280
-163
lines changed

analyzer/codechecker_analyzer/analyzers/clangsa/analyzer.py

+27-22
Original file line numberDiff line numberDiff line change
@@ -12,11 +12,11 @@
1212
import os
1313
import plistlib
1414
import re
15-
import shlex
1615
import subprocess
16+
import sys
17+
from typing import List, Tuple
1718

18-
from typing import List
19-
19+
from codechecker_common import util
2020
from codechecker_common.logger import get_logger
2121

2222
from codechecker_analyzer import analyzer_context, env
@@ -42,7 +42,7 @@
4242
def parse_clang_help_page(
4343
command: List[str],
4444
start_label: str
45-
) -> List[str]:
45+
) -> List[Tuple[str, str]]:
4646
"""
4747
Parse the clang help page starting from a specific label.
4848
Returns a list of (flag, description) tuples.
@@ -121,6 +121,12 @@ class ClangSA(analyzer_base.SourceAnalyzer):
121121

122122
__ctu_autodetection = None
123123

124+
__additional_analyzer_config = {
125+
'cc-verbatim-args-file':
126+
'A file path containing flags that are forwarded verbatim to the '
127+
'analyzer tool. E.g.: cc-verbatim-args-file=<filepath>'
128+
}
129+
124130
def __init__(self, cfg_handler, buildaction):
125131
super().__init__(cfg_handler, buildaction)
126132
self.__disable_ctu = False
@@ -308,15 +314,17 @@ def get_checker_config(cls) -> List[str]:
308314
return parse_clang_help_page(command, 'OPTIONS:')
309315

310316
@classmethod
311-
def get_analyzer_config(cls) -> List[str]:
317+
def get_analyzer_config(cls) -> List[Tuple[str, str]]:
312318
"""Return the list of analyzer config options."""
313319
command = [cls.analyzer_binary(), "-cc1"]
314320

315321
cls.__add_plugin_load_flags(command)
316322

317323
command.append("-analyzer-config-help")
318324

319-
return parse_clang_help_page(command, 'OPTIONS:')
325+
native_config = parse_clang_help_page(command, 'OPTIONS:')
326+
327+
return native_config + list(cls.__additional_analyzer_config.items())
320328

321329
def post_analyze(self, result_handler):
322330
"""
@@ -636,21 +644,6 @@ def construct_config_handler(cls, args):
636644
'ctu_ast_mode' in args and \
637645
args.ctu_ast_mode == 'parse-on-demand'
638646

639-
try:
640-
with open(args.clangsa_args_cfg_file, 'r', encoding='utf8',
641-
errors='ignore') as sa_cfg:
642-
handler.analyzer_extra_arguments = \
643-
re.sub(r'\$\((.*?)\)',
644-
env.replace_env_var(args.clangsa_args_cfg_file),
645-
sa_cfg.read().strip())
646-
handler.analyzer_extra_arguments = \
647-
shlex.split(handler.analyzer_extra_arguments)
648-
except IOError as ioerr:
649-
LOG.debug_analyzer(ioerr)
650-
except AttributeError as aerr:
651-
# No clangsa arguments file was given in the command line.
652-
LOG.debug_analyzer(aerr)
653-
654647
checkers = ClangSA.get_analyzer_checkers()
655648

656649
try:
@@ -716,7 +709,19 @@ def construct_config_handler(cls, args):
716709
if 'analyzer_config' in args and \
717710
isinstance(args.analyzer_config, list):
718711
for cfg in args.analyzer_config:
719-
if cfg.analyzer == cls.ANALYZER_NAME:
712+
# TODO: The analyzer plugin should get only its own analyzer
713+
# config options from outside.
714+
if cfg.analyzer != cls.ANALYZER_NAME:
715+
continue
716+
717+
if cfg.option == 'cc-verbatim-args-file':
718+
try:
719+
handler.analyzer_extra_arguments = \
720+
util.load_args_from_file(cfg.value)
721+
except FileNotFoundError:
722+
LOG.error(f"File not found: {cfg.value}")
723+
sys.exit(1)
724+
else:
720725
handler.checker_config.append(f"{cfg.option}={cfg.value}")
721726

722727
return handler

analyzer/codechecker_analyzer/analyzers/clangtidy/analyzer.py

+25-22
Original file line numberDiff line numberDiff line change
@@ -15,13 +15,14 @@
1515
import os
1616
from pathlib import Path
1717
import re
18-
import shlex
1918
import shutil
2019
import subprocess
20+
import sys
2121
from typing import Iterable, List, Set, Tuple
2222

2323
import yaml
2424

25+
from codechecker_common import util
2526
from codechecker_common.logger import get_logger
2627

2728
from codechecker_analyzer import analyzer_context, env
@@ -250,6 +251,12 @@ class ClangTidy(analyzer_base.SourceAnalyzer):
250251
# Cache object for get_analyzer_checkers().
251252
__analyzer_checkers = None
252253

254+
__additional_analyzer_config = {
255+
'cc-verbatim-args-file':
256+
'A file path containing flags that are forwarded verbatim to the '
257+
'analyzer tool. E.g.: cc-verbatim-args-file=<filepath>'
258+
}
259+
253260
@classmethod
254261
def analyzer_binary(cls):
255262
return analyzer_context.get_context() \
@@ -348,9 +355,11 @@ def get_analyzer_config(cls):
348355
universal_newlines=True,
349356
encoding="utf-8",
350357
errors="ignore")
351-
return parse_analyzer_config(result)
358+
native_config = parse_analyzer_config(result)
352359
except (subprocess.CalledProcessError, OSError):
353-
return []
360+
native_config = []
361+
362+
return native_config + list(cls.__additional_analyzer_config.items())
354363

355364
def get_checker_list(self, config) -> Tuple[List[str], List[str]]:
356365
"""
@@ -623,32 +632,26 @@ def construct_config_handler(cls, args):
623632
'add_gcc_include_dirs_with_isystem' in args and \
624633
args.add_gcc_include_dirs_with_isystem
625634

626-
# FIXME We cannot get the resource dir from the clang-tidy binary,
627-
# therefore we get a sibling clang binary which of clang-tidy.
628-
# TODO Support "clang-tidy -print-resource-dir" .
629-
try:
630-
with open(args.tidy_args_cfg_file, 'r', encoding='utf-8',
631-
errors='ignore') as tidy_cfg:
632-
handler.analyzer_extra_arguments = \
633-
re.sub(r'\$\((.*?)\)',
634-
env.replace_env_var(args.tidy_args_cfg_file),
635-
tidy_cfg.read().strip())
636-
handler.analyzer_extra_arguments = \
637-
shlex.split(handler.analyzer_extra_arguments)
638-
except IOError as ioerr:
639-
LOG.debug_analyzer(ioerr)
640-
except AttributeError as aerr:
641-
# No clang tidy arguments file was given in the command line.
642-
LOG.debug_analyzer(aerr)
643-
644635
analyzer_config = {}
645636
# TODO: This extra "isinstance" check is needed for
646637
# CodeChecker analyzers --analyzer-config. This command also
647638
# runs this function in order to construct a config handler.
648639
if 'analyzer_config' in args and \
649640
isinstance(args.analyzer_config, list):
650641
for cfg in args.analyzer_config:
651-
if cfg.analyzer == cls.ANALYZER_NAME:
642+
# TODO: The analyzer plugin should get only its own analyzer
643+
# config options from outside.
644+
if cfg.analyzer != cls.ANALYZER_NAME:
645+
continue
646+
647+
if cfg.option == 'cc-verbatim-args-file':
648+
try:
649+
handler.analyzer_extra_arguments = \
650+
util.load_args_from_file(cfg.value)
651+
except FileNotFoundError:
652+
LOG.error(f"File not found: {cfg.value}")
653+
sys.exit(1)
654+
else:
652655
analyzer_config[cfg.option] = cfg.value
653656

654657
# Reports in headers are hidden by default in clang-tidy. Re-enable it

analyzer/codechecker_analyzer/analyzers/cppcheck/analyzer.py

+22-21
Original file line numberDiff line numberDiff line change
@@ -10,19 +10,19 @@
1010
"""
1111

1212
from collections import defaultdict
13+
import sys
1314
from packaging.version import Version
1415
from pathlib import Path
1516
import os
1617
import re
17-
import shlex
1818
import shutil
1919
import subprocess
2020
import xml.etree.ElementTree as ET
2121

2222
from codechecker_common.logger import get_logger
2323
from codechecker_common import util
2424

25-
from codechecker_analyzer import analyzer_context, env
25+
from codechecker_analyzer import analyzer_context
2626
from codechecker_analyzer.env import get_binary_in_path
2727

2828
from .. import analyzer_base
@@ -303,11 +303,15 @@ def get_analyzer_config(cls):
303303
"""
304304
Config options for cppcheck.
305305
"""
306-
return [("addons", "A list of cppcheck addon files."),
307-
("libraries", "A list of cppcheck library definiton files."),
308-
("platform", "The platform configuration .xml file."),
309-
("inconclusive", "Enable inconclusive reports.")
310-
]
306+
return [
307+
("addons", "A list of cppcheck addon files."),
308+
("libraries", "A list of cppcheck library definiton files."),
309+
("platform", "The platform configuration .xml file."),
310+
("inconclusive", "Enable inconclusive reports."),
311+
("cc-verbatim-args-file",
312+
"A file path containing flags that are forwarded verbatim to the "
313+
"analyzer tool. E.g.: cc-verbatim-args-file=<filepath>")
314+
]
311315

312316
@classmethod
313317
def get_checker_config(cls):
@@ -410,20 +414,17 @@ def construct_config_handler(cls, args):
410414

411415
handler.analyzer_config = analyzer_config
412416

413-
try:
414-
with open(args.cppcheck_args_cfg_file, 'r', encoding='utf8',
415-
errors='ignore') as sa_cfg:
416-
handler.analyzer_extra_arguments = \
417-
re.sub(r'\$\((.*?)\)',
418-
env.replace_env_var(args.cppcheck_args_cfg_file),
419-
sa_cfg.read().strip())
420-
handler.analyzer_extra_arguments = \
421-
shlex.split(handler.analyzer_extra_arguments)
422-
except IOError as ioerr:
423-
LOG.debug_analyzer(ioerr)
424-
except AttributeError as aerr:
425-
# No cppcheck arguments file was given in the command line.
426-
LOG.debug_analyzer(aerr)
417+
if 'analyzer_config' in args and \
418+
isinstance(args.analyzer_config, list):
419+
for cfg in args.analyzer_config:
420+
if cfg.analyzer == cls.ANALYZER_NAME and \
421+
cfg.option == 'cc-verbatim-args-file':
422+
try:
423+
handler.analyzer_extra_arguments = \
424+
util.load_args_from_file(cfg.value)
425+
except FileNotFoundError:
426+
LOG.error(f"File not found: {cfg.value}")
427+
sys.exit(1)
427428

428429
checkers = cls.get_analyzer_checkers()
429430

analyzer/codechecker_analyzer/analyzers/gcc/analyzer.py

+21-8
Original file line numberDiff line numberDiff line change
@@ -5,12 +5,13 @@
55
# SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
66
#
77
# -------------------------------------------------------------------------
8-
from collections import defaultdict
98
from packaging.version import Version
109
import shlex
1110
import subprocess
11+
import sys
1212

1313
from codechecker_common.logger import get_logger
14+
from codechecker_common import util
1415

1516
from codechecker_analyzer import analyzer_context
1617

@@ -80,6 +81,8 @@ def construct_analyzer_cmd(self, result_handler):
8081
if not has_flag('-x', analyzer_cmd):
8182
analyzer_cmd.extend(['-x', compile_lang])
8283

84+
analyzer_cmd.extend(config.analyzer_extra_arguments)
85+
8386
analyzer_cmd.append(self.source_file)
8487

8588
LOG.debug_analyzer("Running analysis command "
@@ -126,7 +129,11 @@ def get_analyzer_config(cls):
126129
Config options for gcc.
127130
"""
128131
# TODO
129-
return []
132+
return [
133+
('cc-verbatim-args-file',
134+
'A file path containing flags that are forwarded verbatim to the '
135+
'analyzer tool. E.g.: cc-verbatim-args-file=<filepath>')
136+
]
130137

131138
@classmethod
132139
def get_checker_config(cls):
@@ -216,15 +223,21 @@ def construct_result_handler(self, buildaction, report_output,
216223
def construct_config_handler(cls, args):
217224
handler = GccConfigHandler()
218225

219-
analyzer_config = defaultdict(list)
220-
221226
if 'analyzer_config' in args and \
222227
isinstance(args.analyzer_config, list):
223228
for cfg in args.analyzer_config:
224-
if cfg.analyzer == cls.ANALYZER_NAME:
225-
analyzer_config[cfg.option].append(cfg.value)
226-
227-
handler.analyzer_config = analyzer_config
229+
# TODO: The analyzer plugin should get only its own analyzer
230+
# config options from outside.
231+
if cfg.analyzer != cls.ANALYZER_NAME:
232+
continue
233+
234+
if cfg.option == 'cc-verbatim-args-file':
235+
try:
236+
handler.analyzer_extra_arguments = \
237+
util.load_args_from_file(cfg.value)
238+
except FileNotFoundError:
239+
LOG.error(f"File not found: {cfg.value}")
240+
sys.exit(1)
228241

229242
checkers = cls.get_analyzer_checkers()
230243

0 commit comments

Comments
 (0)