cppcheck/tools/compare_ast_symdb.py

90 lines
2.8 KiB
Python

# Example usage:
# python3 compare_ast.py lib/token.cpp
# python3 compare_ast.py "--project=compile_commands.json --file-filter=*/lib/somefile.c"
# check all files in a compile_commands.json:
# grep '"file":' compile_commands.json | grep -v test | sed 's|.*/curl/|"--project=compile_commands.json --file-filter=*|' | xargs python3 ~/cppcheck/tools/compare_ast.py
import os
import re
import sys
import subprocess
CPPCHECK = os.path.expanduser('~/cppcheck/cppcheck')
def run_cppcheck(cppcheck_parameters:str, clang:str):
cmd = f'{CPPCHECK} {cppcheck_parameters} {clang} --debug --verbose'
#print(cmd)
p = subprocess.Popen(cmd.split(), stdout=subprocess.PIPE, stderr=subprocess.PIPE)
comm = p.communicate()
return comm[0].decode('utf-8', 'ignore')
def get_ast(debug):
debug = re.sub(r' f:0x[0-9a-fA-F]+', '', debug)
debug = re.sub(r" '[a-zA-Z0-9: *]+'", '', debug)
pos1 = debug.rfind('\n##AST\n')
if pos1 < 0:
return ''
pos1 = debug.find('\n', pos1) + 1
assert pos1 > 0
pos2 = debug.find("\n##", pos1)
if pos2 < 0:
return debug[pos1:]
return debug[pos1:pos2-1]
def get_symdb(debug):
debug = re.sub(r'0x[0-9a-fA-F]+', '0x12345678', debug)
debug = re.sub(r'nestedIn: Struct', 'nestedIn: Class', debug)
debug = re.sub(r'classDef: struct', 'classDef: class', debug)
debug = re.sub(r'isInline: [a-z]+', 'isInline: ---', debug)
debug = re.sub(r'definedType: .*', 'definedType: ---', debug)
debug = re.sub(r'needInitialization: .*', 'needInitialization: ---', debug)
debug = re.sub(r'functionOf: .*', 'functionOf: ---', debug)
debug = re.sub(r'0x12345678 Struct', '0x12345678 Class', debug)
pos1 = debug.rfind('\n### Symbol database ###\n')
if pos1 < 0:
return ''
pos1 = debug.find('\n', pos1) + 1
assert pos1 > 0
pos2 = debug.find("\n##", pos1)
if pos2 < 0:
return debug[pos1:]
return debug[pos1:pos2-1]
def compare_ast_symdb(cppcheck_parameters: str):
same = True
debug1 = run_cppcheck(cppcheck_parameters, '')
debug2 = run_cppcheck(cppcheck_parameters, '--clang')
ast1 = get_ast(debug1)
ast2 = get_ast(debug2)
if ast1 != ast2:
print(f"ast is not the same: {cppcheck_parameters}")
with open('cppcheck.ast', 'wt') as f:
f.write(ast1)
with open('clang.ast', 'wt') as f:
f.write(ast2)
same = False
symdb1 = get_symdb(debug1)
symdb2 = get_symdb(debug2)
if symdb1 != symdb2:
print(f"symdb is not the same: {cppcheck_parameters}")
with open('cppcheck.symdb', 'wt') as f:
f.write(symdb1)
with open('clang.symdb', 'wt') as f:
f.write(symdb2)
same = False
if not same:
sys.exit(1)
for arg in sys.argv[1:]:
print(arg)
compare_ast_symdb(arg)