115 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
			
		
		
	
	
			115 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
#!/usr/bin/python
 | 
						|
 | 
						|
# [PR 11661] Note that we hardwire to /usr/bin/python because we
 | 
						|
# want to the use the system version of Python on Mac OS X.
 | 
						|
# This one has the scripting bridge enabled.
 | 
						|
 | 
						|
import sys
 | 
						|
if sys.version_info < (2, 7):
 | 
						|
    print "set-xcode-analyzer requires Python 2.7 or later"
 | 
						|
    sys.exit(1)
 | 
						|
 | 
						|
import os
 | 
						|
import subprocess
 | 
						|
import re
 | 
						|
import tempfile
 | 
						|
import shutil
 | 
						|
import stat
 | 
						|
from AppKit import *
 | 
						|
 | 
						|
def FindClangSpecs(path):
 | 
						|
  print "(+) Searching for xcspec file in: ", path
 | 
						|
  for root, dirs, files in os.walk(path):
 | 
						|
    for f in files:
 | 
						|
      if f.endswith(".xcspec") and f.startswith("Clang LLVM"):
 | 
						|
        yield os.path.join(root, f)
 | 
						|
 | 
						|
def ModifySpec(path, isBuiltinAnalyzer, pathToChecker):
 | 
						|
  t = tempfile.NamedTemporaryFile(delete=False)
 | 
						|
  foundAnalyzer = False
 | 
						|
  with open(path) as f:
 | 
						|
    if isBuiltinAnalyzer:
 | 
						|
      # First search for CLANG_ANALYZER_EXEC.  Newer
 | 
						|
      # versions of Xcode set EXEC_PATH to be CLANG_ANALYZER_EXEC.
 | 
						|
      with open(path) as f2:
 | 
						|
        for line in f2:
 | 
						|
          if line.find("CLANG_ANALYZER_EXEC") >= 0:
 | 
						|
            pathToChecker = "$(CLANG_ANALYZER_EXEC)"
 | 
						|
            break
 | 
						|
    # Now create a new file.
 | 
						|
    for line in f:
 | 
						|
      if not foundAnalyzer:
 | 
						|
        if line.find("Static Analyzer") >= 0:
 | 
						|
          foundAnalyzer = True
 | 
						|
      else:
 | 
						|
        m = re.search('^(\s*ExecPath\s*=\s*")', line)
 | 
						|
        if m:
 | 
						|
          line = "".join([m.group(0), pathToChecker, '";\n'])
 | 
						|
          # Do not modify further ExecPath's later in the xcspec.
 | 
						|
          foundAnalyzer = False
 | 
						|
      t.write(line)
 | 
						|
  t.close()
 | 
						|
  print "(+) processing:", path
 | 
						|
  try:
 | 
						|
    shutil.copy(t.name, path)
 | 
						|
    os.chmod(path, stat.S_IRUSR | stat.S_IWUSR | stat.S_IRGRP | stat.S_IROTH)
 | 
						|
  except IOError, why:
 | 
						|
    print "    (-) Cannot update file:", why, "\n"
 | 
						|
  except OSError, why:
 | 
						|
    print "    (-) Cannot update file:", why, "\n"
 | 
						|
  os.unlink(t.name)
 | 
						|
 | 
						|
def main():
 | 
						|
  from optparse import OptionParser
 | 
						|
  parser = OptionParser('usage: %prog [options]')
 | 
						|
  parser.set_description(__doc__)
 | 
						|
  parser.add_option("--use-checker-build", dest="path",
 | 
						|
                    help="Use the Clang located at the provided absolute path, e.g. /Users/foo/checker-1")
 | 
						|
  parser.add_option("--use-xcode-clang", action="store_const",
 | 
						|
                    const="$(CLANG)", dest="default",
 | 
						|
                    help="Use the Clang bundled with Xcode")
 | 
						|
  (options, args) = parser.parse_args()
 | 
						|
  if options.path is None and options.default is None:
 | 
						|
    parser.error("You must specify a version of Clang to use for static analysis.  Specify '-h' for details")
 | 
						|
 | 
						|
  # determine if Xcode is running
 | 
						|
  for x in NSWorkspace.sharedWorkspace().runningApplications():
 | 
						|
    if x.localizedName().find("Xcode") >= 0:
 | 
						|
      print "(-) You must quit Xcode first before modifying its configuration files."
 | 
						|
      sys.exit(1)
 | 
						|
 | 
						|
  isBuiltinAnalyzer = False
 | 
						|
  if options.path:
 | 
						|
    # Expand tildes.
 | 
						|
    path = os.path.expanduser(options.path)
 | 
						|
    if not path.endswith("clang"):
 | 
						|
      print "(+) Using Clang bundled with checker build:", path
 | 
						|
      path = os.path.join(path, "bin", "clang");
 | 
						|
    else:
 | 
						|
      print "(+) Using Clang located at:", path
 | 
						|
  else:
 | 
						|
    print "(+) Using the Clang bundled with Xcode"
 | 
						|
    path = options.default
 | 
						|
    isBuiltinAnalyzer = True
 | 
						|
 | 
						|
  try:
 | 
						|
    xcode_path = subprocess.check_output(["xcode-select", "-print-path"])
 | 
						|
  except AttributeError:
 | 
						|
    # Fall back to the default install location when using Python < 2.7.0
 | 
						|
    xcode_path = "/Developer"
 | 
						|
  if (xcode_path.find(".app/") != -1):
 | 
						|
    # Cut off the 'Developer' dir, as the xcspec lies in another part
 | 
						|
    # of the Xcode.app subtree.
 | 
						|
    xcode_path = xcode_path.rsplit('/Developer', 1)[0]
 | 
						|
 | 
						|
  foundSpec = False
 | 
						|
  for x in FindClangSpecs(xcode_path):
 | 
						|
    foundSpec = True
 | 
						|
    ModifySpec(x, isBuiltinAnalyzer, path)
 | 
						|
 | 
						|
  if foundSpec == False:
 | 
						|
      print "(-) No compiler configuration file was found.  Xcode's analyzer has not been updated."
 | 
						|
 | 
						|
if __name__ == '__main__':
 | 
						|
  main()
 |