blob: 60286e5223a2695b9ca1109c3cc3a95826a9b1a8 [file] [log] [blame]
#!/usr/bin/env python
#
# The LLVM Compiler Infrastructure
#
# This file is distributed under the University of Illinois Open Source
# License. See LICENSE.TXT for details.
#
##===----------------------------------------------------------------------===##
#
# A reduced version of the 'ccc' script that is designed to handle off
# actual compilation to gcc, but run the code passed to gcc through the
# static analyzer.
#
##===----------------------------------------------------------------------===##
import sys
import subprocess
import os
def error(message):
print >> sys.stderr, 'ccc: ' + message
sys.exit(1)
def quote(arg):
if '"' in arg:
return repr(arg)
return arg
def run(args):
code = subprocess.call(args)
if code > 255:
code = 1
if code:
sys.exit(code)
def compile(args):
command = 'gcc'.split()
run(command + args)
def remove_pch_extension(path):
i = path.rfind('.gch')
if i < 0:
return path
return path[:i]
def analyze(clang, args,language,output,files,verbose,htmldir,file,analysis_type):
if language.rfind("c++") >= 0:
return
print_args = []
if verbose:
# We MUST print to stderr. Some clients use the stdout output of
# gcc for various purposes.
print >> sys.stderr, ' '.join(['\n[LOCATION]:', os.getcwd(), '\n' ])
i = 0
while i < len(args):
print_args.append(''.join([ '\'', args[i], '\'' ]))
i += 1
RunAnalyzer = 0;
if language.find("header") > 0:
target = remove_pch_extension(output)
command = 'cp'.split()
args = command + files + target.split()
else:
command = clang.split() + analysis_type.split()
args = command + args;
RunAnalyzer = 1
if verbose == 2:
print >> sys.stderr, '#SHELL (cd ' + os.getcwd() + ' && ' + ' '.join(command + print_args) + ')\n'
if RunAnalyzer and htmldir is not None:
args.append('-o')
print_args.append('-o')
args.append(htmldir)
print_args.append(htmldir)
if verbose == 1:
# We MUST print to stderr. Some clients use the stdout output of
# gcc for various purposes.
print >> sys.stderr, ' '.join(command+print_args)
print >> sys.stderr, '\n'
subprocess.call(args)
def extension(path):
return path.split(".")[-1]
def changeextension(path, newext):
i = path.rfind('.')
if i < 0:
return path
j = path.rfind('/', 0, i)
print path
if j < 0:
return path[:i] + "." + newext
return path[j+1:i] + "." + newext
def inferlanguage(extension):
if extension == "c":
return "c"
elif extension in ["cpp", "cc"]:
return "c++"
elif extension == "i":
return "c-cpp-output"
elif extension == "m":
return "objective-c"
elif extension == "mi":
return "objective-c-cpp-output"
elif extension == "s" or extension == "o":
return "skip"
else:
return "unknown"
def main(args):
old_args = args
action = 'link'
output = ''
compile_opts = [ ]
link_opts = [ ]
files = []
save_temps = 0
language = ''
verbose = 0
clang = "clang"
# Forward to GCC.
compile(args)
# Set the analyzer flag.
analysis_type = os.environ.get('CCC_ANALYZER_ANALYSIS')
if analysis_type is not None:
analysis_type = "-" + analysis_type
else:
analysis_type = "-checker-cfref"
# Determine the level of verbosity.
if os.environ.get('CCC_ANALYZER_VERBOSE') is not None:
verbose = 1
if os.environ.get('CCC_ANALYZER_LOG') is not None:
verbose = 2
# Determine what clang executable to use.
clang_env = os.environ.get('CLANG')
if clang_env is not None:
clang = clang_env
# Get the HTML output directory.
htmldir = None
if analysis_type == "-checker-cfref":
htmldir = os.environ.get('CCC_ANALYZER_HTML')
# Process the arguments.
i = 0
while i < len(args):
arg = args[i]
# Modes ccc supports
if arg == '-E':
action = 'preprocess'
if arg == '-c':
action = 'compile'
if arg.startswith('-print-prog-name'):
action = 'print-prog-name'
if arg == '-save-temps':
save_temps = 1
# Options with no arguments that should pass through
if arg in ['-v']:
compile_opts.append(arg)
link_opts.append(arg)
# Options with one argument that should be ignored
if arg in ['--param', '-u']:
i += 1
# Prefix matches for the compile mode
if arg[:2] in ['-D', '-I', '-U', '-F' ]:
if not arg[2:]:
arg += args[i+1]
i += 1
compile_opts.append(arg)
if arg[:5] in ['-std=']:
compile_opts.append(arg)
# Options with one argument that should pass through to compiler
if arg in [ '-include', '-idirafter', '-iprefix',
'-iquote', '-isystem', '-iwithprefix',
'-iwithprefixbefore']:
compile_opts.append(arg)
compile_opts.append(args[i+1])
i += 1
# Options with no argument that should pass through to compiler
if arg in [ '-nostdinc', '-fobjc-gc-only', '-fobjc-gc' ]:
compile_opts.append(arg)
# Options with one argument that should pass through to linker
if arg == '-framework':
link_opts.append(arg)
link_opts.append(args[i+1])
i += 1
# Options with one argument that should pass through to both
if arg in ['-isysroot', '-arch']:
compile_opts.append(arg)
compile_opts.append(args[i+1])
link_opts.append(arg)
link_opts.append(args[i+1])
i += 1
# Prefix matches for the link mode
if arg[:2] in ['-l', '-L', '-O', '-F']:
if arg == '-O': arg = '-O1'
if arg == '-Os': arg = '-O2'
link_opts.append(arg)
# Input files
if arg == '-filelist':
f = open(args[i+1])
for line in f:
files.append(line.strip())
f.close()
i += 1
if arg == '-x':
language = args[i+1]
i += 1
if arg[0] != '-':
files.append(arg)
# Output file
if arg == '-o':
output = args[i+1]
i += 1
i += 1
if action == 'print-prog-name':
# assume we can handle everything
print sys.argv[0]
return
if not files:
error('no input files')
if action == 'compile' or save_temps or action == 'link':
for i, file in enumerate(files):
file_language = language
if not language:
file_language = inferlanguage(extension(file))
if file_language == "skip":
continue
if save_temps and action != "compile":
# Need a temporary output file
coutput = changeextension(file, "o");
files[i] = coutput
elif not output:
coutput = changeextension(file, "o")
else:
coutput = output
analyze_args = [ file ]
if language != 'unknown':
analyze_args = [ '-x', language ] + analyze_args
analyze_args = analyze_args + compile_opts
analyze(clang, analyze_args, language, output, files, verbose, htmldir, file, analysis_type)
if __name__ == '__main__':
main(sys.argv[1:])