1#!/usr/bin/env python 2# Copyright (c) 2012 The Chromium Authors. All rights reserved. 3# Use of this source code is governed by a BSD-style license that can be 4# found in the LICENSE file. 5 6"""This script is used to download prebuilt clang binaries. 7 8It is also used by package.py to build the prebuilt clang binaries.""" 9 10import argparse 11import distutils.spawn 12import glob 13import os 14import pipes 15import re 16import shutil 17import subprocess 18import stat 19import sys 20import tarfile 21import tempfile 22import time 23import urllib2 24import zipfile 25 26 27# Do NOT CHANGE this if you don't know what you're doing -- see 28# https://chromium.googlesource.com/chromium/src/+/master/docs/updating_clang.md 29# Reverting problematic clang rolls is safe, though. 30CLANG_REVISION = '282487' 31 32use_head_revision = 'LLVM_FORCE_HEAD_REVISION' in os.environ 33if use_head_revision: 34 CLANG_REVISION = 'HEAD' 35 36# This is incremented when pushing a new build of Clang at the same revision. 37CLANG_SUB_REVISION=1 38 39PACKAGE_VERSION = "%s-%s" % (CLANG_REVISION, CLANG_SUB_REVISION) 40 41# Path constants. (All of these should be absolute paths.) 42THIS_DIR = os.path.abspath(os.path.dirname(__file__)) 43CHROMIUM_DIR = os.path.abspath(os.path.join(THIS_DIR, '..', '..', '..')) 44THIRD_PARTY_DIR = os.path.join(CHROMIUM_DIR, 'third_party') 45LLVM_DIR = os.path.join(THIRD_PARTY_DIR, 'llvm') 46LLVM_BOOTSTRAP_DIR = os.path.join(THIRD_PARTY_DIR, 'llvm-bootstrap') 47LLVM_BOOTSTRAP_INSTALL_DIR = os.path.join(THIRD_PARTY_DIR, 48 'llvm-bootstrap-install') 49LLVM_LTO_GOLD_PLUGIN_DIR = os.path.join(THIRD_PARTY_DIR, 'llvm-lto-gold-plugin') 50CHROME_TOOLS_SHIM_DIR = os.path.join(LLVM_DIR, 'tools', 'chrometools') 51LLVM_BUILD_DIR = os.path.join(CHROMIUM_DIR, 'third_party', 'llvm-build', 52 'Release+Asserts') 53COMPILER_RT_BUILD_DIR = os.path.join(LLVM_BUILD_DIR, 'compiler-rt') 54CLANG_DIR = os.path.join(LLVM_DIR, 'tools', 'clang') 55LLD_DIR = os.path.join(LLVM_DIR, 'tools', 'lld') 56# compiler-rt is built as part of the regular LLVM build on Windows to get 57# the 64-bit runtime, and out-of-tree elsewhere. 58# TODO(thakis): Try to unify this. 59if sys.platform == 'win32': 60 COMPILER_RT_DIR = os.path.join(LLVM_DIR, 'projects', 'compiler-rt') 61else: 62 COMPILER_RT_DIR = os.path.join(LLVM_DIR, 'compiler-rt') 63LIBCXX_DIR = os.path.join(LLVM_DIR, 'projects', 'libcxx') 64LIBCXXABI_DIR = os.path.join(LLVM_DIR, 'projects', 'libcxxabi') 65LLVM_BUILD_TOOLS_DIR = os.path.abspath( 66 os.path.join(LLVM_DIR, '..', 'llvm-build-tools')) 67STAMP_FILE = os.path.normpath( 68 os.path.join(LLVM_DIR, '..', 'llvm-build', 'cr_build_revision')) 69BINUTILS_DIR = os.path.join(THIRD_PARTY_DIR, 'binutils') 70BINUTILS_BIN_DIR = os.path.join(BINUTILS_DIR, BINUTILS_DIR, 71 'Linux_x64', 'Release', 'bin') 72BFD_PLUGINS_DIR = os.path.join(BINUTILS_DIR, 'Linux_x64', 'Release', 73 'lib', 'bfd-plugins') 74VERSION = '4.0.0' 75ANDROID_NDK_DIR = os.path.join( 76 CHROMIUM_DIR, 'third_party', 'android_tools', 'ndk') 77 78# URL for pre-built binaries. 79CDS_URL = os.environ.get('CDS_CLANG_BUCKET_OVERRIDE', 80 'https://commondatastorage.googleapis.com/chromium-browser-clang') 81 82LLVM_REPO_URL='https://llvm.org/svn/llvm-project' 83if 'LLVM_REPO_URL' in os.environ: 84 LLVM_REPO_URL = os.environ['LLVM_REPO_URL'] 85 86# Bump after VC updates. 87DIA_DLL = { 88 '2013': 'msdia120.dll', 89 '2015': 'msdia140.dll', 90} 91 92 93def DownloadUrl(url, output_file): 94 """Download url into output_file.""" 95 CHUNK_SIZE = 4096 96 TOTAL_DOTS = 10 97 num_retries = 3 98 retry_wait_s = 5 # Doubled at each retry. 99 100 while True: 101 try: 102 sys.stdout.write('Downloading %s ' % url) 103 sys.stdout.flush() 104 response = urllib2.urlopen(url) 105 total_size = int(response.info().getheader('Content-Length').strip()) 106 bytes_done = 0 107 dots_printed = 0 108 while True: 109 chunk = response.read(CHUNK_SIZE) 110 if not chunk: 111 break 112 output_file.write(chunk) 113 bytes_done += len(chunk) 114 num_dots = TOTAL_DOTS * bytes_done / total_size 115 sys.stdout.write('.' * (num_dots - dots_printed)) 116 sys.stdout.flush() 117 dots_printed = num_dots 118 if bytes_done != total_size: 119 raise urllib2.URLError("only got %d of %d bytes" % 120 (bytes_done, total_size)) 121 print ' Done.' 122 return 123 except urllib2.URLError as e: 124 sys.stdout.write('\n') 125 print e 126 if num_retries == 0 or isinstance(e, urllib2.HTTPError) and e.code == 404: 127 raise e 128 num_retries -= 1 129 print 'Retrying in %d s ...' % retry_wait_s 130 time.sleep(retry_wait_s) 131 retry_wait_s *= 2 132 133 134def EnsureDirExists(path): 135 if not os.path.exists(path): 136 print "Creating directory %s" % path 137 os.makedirs(path) 138 139 140def DownloadAndUnpack(url, output_dir): 141 with tempfile.TemporaryFile() as f: 142 DownloadUrl(url, f) 143 f.seek(0) 144 EnsureDirExists(output_dir) 145 if url.endswith('.zip'): 146 zipfile.ZipFile(f).extractall(path=output_dir) 147 else: 148 tarfile.open(mode='r:gz', fileobj=f).extractall(path=output_dir) 149 150 151def ReadStampFile(path=STAMP_FILE): 152 """Return the contents of the stamp file, or '' if it doesn't exist.""" 153 try: 154 with open(path, 'r') as f: 155 return f.read().rstrip() 156 except IOError: 157 return '' 158 159 160def WriteStampFile(s, path=STAMP_FILE): 161 """Write s to the stamp file.""" 162 EnsureDirExists(os.path.dirname(path)) 163 with open(path, 'w') as f: 164 f.write(s) 165 f.write('\n') 166 167 168def GetSvnRevision(svn_repo): 169 """Returns current revision of the svn repo at svn_repo.""" 170 if sys.platform == 'darwin': 171 # mac_files toolchain must be set for hermetic builds. 172 root = os.path.dirname(os.path.dirname(os.path.dirname( 173 os.path.dirname(__file__)))) 174 sys.path.append(os.path.join(root, 'build')) 175 import mac_toolchain 176 177 mac_toolchain.SetToolchainEnvironment() 178 svn_info = subprocess.check_output('svn info ' + svn_repo, shell=True) 179 m = re.search(r'Revision: (\d+)', svn_info) 180 return m.group(1) 181 182 183def RmTree(dir): 184 """Delete dir.""" 185 def ChmodAndRetry(func, path, _): 186 # Subversion can leave read-only files around. 187 if not os.access(path, os.W_OK): 188 os.chmod(path, stat.S_IWUSR) 189 return func(path) 190 raise 191 192 shutil.rmtree(dir, onerror=ChmodAndRetry) 193 194 195def RmCmakeCache(dir): 196 """Delete CMake cache related files from dir.""" 197 for dirpath, dirs, files in os.walk(dir): 198 if 'CMakeCache.txt' in files: 199 os.remove(os.path.join(dirpath, 'CMakeCache.txt')) 200 if 'CMakeFiles' in dirs: 201 RmTree(os.path.join(dirpath, 'CMakeFiles')) 202 203 204def RunCommand(command, msvc_arch=None, env=None, fail_hard=True): 205 """Run command and return success (True) or failure; or if fail_hard is 206 True, exit on failure. If msvc_arch is set, runs the command in a 207 shell with the msvc tools for that architecture.""" 208 209 if msvc_arch and sys.platform == 'win32': 210 command = GetVSVersion().SetupScript(msvc_arch) + ['&&'] + command 211 212 # https://docs.python.org/2/library/subprocess.html: 213 # "On Unix with shell=True [...] if args is a sequence, the first item 214 # specifies the command string, and any additional items will be treated as 215 # additional arguments to the shell itself. That is to say, Popen does the 216 # equivalent of: 217 # Popen(['/bin/sh', '-c', args[0], args[1], ...])" 218 # 219 # We want to pass additional arguments to command[0], not to the shell, 220 # so manually join everything into a single string. 221 # Annoyingly, for "svn co url c:\path", pipes.quote() thinks that it should 222 # quote c:\path but svn can't handle quoted paths on Windows. Since on 223 # Windows follow-on args are passed to args[0] instead of the shell, don't 224 # do the single-string transformation there. 225 if sys.platform != 'win32': 226 command = ' '.join([pipes.quote(c) for c in command]) 227 print 'Running', command 228 if subprocess.call(command, env=env, shell=True) == 0: 229 return True 230 print 'Failed.' 231 if fail_hard: 232 sys.exit(1) 233 return False 234 235 236def CopyFile(src, dst): 237 """Copy a file from src to dst.""" 238 print "Copying %s to %s" % (src, dst) 239 shutil.copy(src, dst) 240 241 242def CopyDirectoryContents(src, dst, filename_filter=None): 243 """Copy the files from directory src to dst 244 with an optional filename filter.""" 245 dst = os.path.realpath(dst) # realpath() in case dst ends in /.. 246 EnsureDirExists(dst) 247 for root, _, files in os.walk(src): 248 for f in files: 249 if filename_filter and not re.match(filename_filter, f): 250 continue 251 CopyFile(os.path.join(root, f), dst) 252 253 254def Checkout(name, url, dir): 255 """Checkout the SVN module at url into dir. Use name for the log message.""" 256 print "Checking out %s r%s into '%s'" % (name, CLANG_REVISION, dir) 257 258 command = ['svn', 'checkout', '--force', url + '@' + CLANG_REVISION, dir] 259 if RunCommand(command, fail_hard=False): 260 return 261 262 if os.path.isdir(dir): 263 print "Removing %s." % (dir) 264 RmTree(dir) 265 266 print "Retrying." 267 RunCommand(command) 268 269 270def DeleteChromeToolsShim(): 271 OLD_SHIM_DIR = os.path.join(LLVM_DIR, 'tools', 'zzz-chrometools') 272 shutil.rmtree(OLD_SHIM_DIR, ignore_errors=True) 273 shutil.rmtree(CHROME_TOOLS_SHIM_DIR, ignore_errors=True) 274 275 276def CreateChromeToolsShim(): 277 """Hooks the Chrome tools into the LLVM build. 278 279 Several Chrome tools have dependencies on LLVM/Clang libraries. The LLVM build 280 detects implicit tools in the tools subdirectory, so this helper install a 281 shim CMakeLists.txt that forwards to the real directory for the Chrome tools. 282 283 Note that the shim directory name intentionally has no - or _. The implicit 284 tool detection logic munges them in a weird way.""" 285 assert not any(i in os.path.basename(CHROME_TOOLS_SHIM_DIR) for i in '-_') 286 os.mkdir(CHROME_TOOLS_SHIM_DIR) 287 with file(os.path.join(CHROME_TOOLS_SHIM_DIR, 'CMakeLists.txt'), 'w') as f: 288 f.write('# Automatically generated by tools/clang/scripts/update.py. ' + 289 'Do not edit.\n') 290 f.write('# Since tools/clang is located in another directory, use the \n') 291 f.write('# two arg version to specify where build artifacts go. CMake\n') 292 f.write('# disallows reuse of the same binary dir for multiple source\n') 293 f.write('# dirs, so the build artifacts need to go into a subdirectory.\n') 294 f.write('# dirs, so the build artifacts need to go into a subdirectory.\n') 295 f.write('if (CHROMIUM_TOOLS_SRC)\n') 296 f.write(' add_subdirectory(${CHROMIUM_TOOLS_SRC} ' + 297 '${CMAKE_CURRENT_BINARY_DIR}/a)\n') 298 f.write('endif (CHROMIUM_TOOLS_SRC)\n') 299 300 301def DownloadHostGcc(args): 302 """Downloads gcc 4.8.5 and makes sure args.gcc_toolchain is set.""" 303 if not sys.platform.startswith('linux') or args.gcc_toolchain: 304 return 305 # Unconditionally download a prebuilt gcc to guarantee the included libstdc++ 306 # works on Ubuntu Precise. 307 gcc_dir = os.path.join(LLVM_BUILD_TOOLS_DIR, 'gcc485precise') 308 if not os.path.exists(gcc_dir): 309 print 'Downloading pre-built GCC 4.8.5...' 310 DownloadAndUnpack( 311 CDS_URL + '/tools/gcc485precise.tgz', LLVM_BUILD_TOOLS_DIR) 312 args.gcc_toolchain = gcc_dir 313 314 315def AddCMakeToPath(): 316 """Download CMake and add it to PATH.""" 317 if sys.platform == 'win32': 318 zip_name = 'cmake-3.4.3-win32-x86.zip' 319 cmake_dir = os.path.join(LLVM_BUILD_TOOLS_DIR, 320 'cmake-3.4.3-win32-x86', 'bin') 321 else: 322 suffix = 'Darwin' if sys.platform == 'darwin' else 'Linux' 323 zip_name = 'cmake343_%s.tgz' % suffix 324 cmake_dir = os.path.join(LLVM_BUILD_TOOLS_DIR, 'cmake343', 'bin') 325 if not os.path.exists(cmake_dir): 326 DownloadAndUnpack(CDS_URL + '/tools/' + zip_name, LLVM_BUILD_TOOLS_DIR) 327 os.environ['PATH'] = cmake_dir + os.pathsep + os.environ.get('PATH', '') 328 329 330def AddGnuWinToPath(): 331 """Download some GNU win tools and add them to PATH.""" 332 if sys.platform != 'win32': 333 return 334 335 gnuwin_dir = os.path.join(LLVM_BUILD_TOOLS_DIR, 'gnuwin') 336 GNUWIN_VERSION = '5' 337 GNUWIN_STAMP = os.path.join(gnuwin_dir, 'stamp') 338 if ReadStampFile(GNUWIN_STAMP) == GNUWIN_VERSION: 339 print 'GNU Win tools already up to date.' 340 else: 341 zip_name = 'gnuwin-%s.zip' % GNUWIN_VERSION 342 DownloadAndUnpack(CDS_URL + '/tools/' + zip_name, LLVM_BUILD_TOOLS_DIR) 343 WriteStampFile(GNUWIN_VERSION, GNUWIN_STAMP) 344 345 os.environ['PATH'] = gnuwin_dir + os.pathsep + os.environ.get('PATH', '') 346 347 348vs_version = None 349def GetVSVersion(): 350 global vs_version 351 if vs_version: 352 return vs_version 353 354 # Try using the toolchain in depot_tools. 355 # This sets environment variables used by SelectVisualStudioVersion below. 356 sys.path.append(os.path.join(CHROMIUM_DIR, 'build')) 357 import vs_toolchain 358 vs_toolchain.SetEnvironmentAndGetRuntimeDllDirs() 359 360 # Use gyp to find the MSVS installation, either in depot_tools as per above, 361 # or a system-wide installation otherwise. 362 sys.path.append(os.path.join(CHROMIUM_DIR, 'tools', 'gyp', 'pylib')) 363 import gyp.MSVSVersion 364 vs_version = gyp.MSVSVersion.SelectVisualStudioVersion( 365 vs_toolchain.GetVisualStudioVersion()) 366 return vs_version 367 368 369def CopyDiaDllTo(target_dir): 370 # This script always wants to use the 64-bit msdia*.dll. 371 dia_path = os.path.join(GetVSVersion().Path(), 'DIA SDK', 'bin', 'amd64') 372 dia_dll = os.path.join(dia_path, DIA_DLL[GetVSVersion().ShortName()]) 373 CopyFile(dia_dll, target_dir) 374 375 376def VeryifyVersionOfBuiltClangMatchesVERSION(): 377 """Checks that `clang --version` outputs VERSION. If this fails, VERSION 378 in this file is out-of-date and needs to be updated (possibly in an 379 `if use_head_revision:` block in main() first).""" 380 clang = os.path.join(LLVM_BUILD_DIR, 'bin', 'clang') 381 if sys.platform == 'win32': 382 # TODO: Parse `clang-cl /?` output for built clang's version and check that 383 # to check the binary we're actually shipping? But clang-cl.exe is just 384 # a copy of clang.exe, so this does check the same thing. 385 clang += '.exe' 386 version_out = subprocess.check_output([clang, '--version']) 387 version_out = re.match(r'clang version ([0-9.]+)', version_out).group(1) 388 if version_out != VERSION: 389 print ('unexpected clang version %s (not %s), update VERSION in update.py' 390 % (version_out, VERSION)) 391 sys.exit(1) 392 393 394def UpdateClang(args): 395 print 'Updating Clang to %s...' % PACKAGE_VERSION 396 397 need_gold_plugin = 'LLVM_DOWNLOAD_GOLD_PLUGIN' in os.environ or ( 398 sys.platform.startswith('linux') and 399 'buildtype=Official' in os.environ.get('GYP_DEFINES', '') and 400 'branding=Chrome' in os.environ.get('GYP_DEFINES', '')) 401 402 if ReadStampFile() == PACKAGE_VERSION and not args.force_local_build: 403 print 'Clang is already up to date.' 404 if not need_gold_plugin or os.path.exists( 405 os.path.join(LLVM_BUILD_DIR, "lib/LLVMgold.so")): 406 return 0 407 408 # Reset the stamp file in case the build is unsuccessful. 409 WriteStampFile('') 410 411 if not args.force_local_build: 412 cds_file = "clang-%s.tgz" % PACKAGE_VERSION 413 if sys.platform == 'win32' or sys.platform == 'cygwin': 414 cds_full_url = CDS_URL + '/Win/' + cds_file 415 elif sys.platform == 'darwin': 416 cds_full_url = CDS_URL + '/Mac/' + cds_file 417 else: 418 assert sys.platform.startswith('linux') 419 cds_full_url = CDS_URL + '/Linux_x64/' + cds_file 420 421 print 'Downloading prebuilt clang' 422 if os.path.exists(LLVM_BUILD_DIR): 423 RmTree(LLVM_BUILD_DIR) 424 try: 425 DownloadAndUnpack(cds_full_url, LLVM_BUILD_DIR) 426 print 'clang %s unpacked' % PACKAGE_VERSION 427 if sys.platform == 'win32': 428 CopyDiaDllTo(os.path.join(LLVM_BUILD_DIR, 'bin')) 429 # Download the gold plugin if requested to by an environment variable. 430 # This is used by the CFI ClusterFuzz bot, and it's required for official 431 # builds on linux. 432 if need_gold_plugin: 433 RunCommand(['python', CHROMIUM_DIR+'/build/download_gold_plugin.py']) 434 WriteStampFile(PACKAGE_VERSION) 435 return 0 436 except urllib2.URLError: 437 print 'Failed to download prebuilt clang %s' % cds_file 438 print 'Use --force-local-build if you want to build locally.' 439 print 'Exiting.' 440 return 1 441 442 if args.with_android and not os.path.exists(ANDROID_NDK_DIR): 443 print 'Android NDK not found at ' + ANDROID_NDK_DIR 444 print 'The Android NDK is needed to build a Clang whose -fsanitize=address' 445 print 'works on Android. See ' 446 print 'https://www.chromium.org/developers/how-tos/android-build-instructions' 447 print 'for how to install the NDK, or pass --without-android.' 448 return 1 449 450 DownloadHostGcc(args) 451 AddCMakeToPath() 452 AddGnuWinToPath() 453 454 DeleteChromeToolsShim() 455 456 Checkout('LLVM', LLVM_REPO_URL + '/llvm/trunk', LLVM_DIR) 457 Checkout('Clang', LLVM_REPO_URL + '/cfe/trunk', CLANG_DIR) 458 if sys.platform == 'win32' or use_head_revision: 459 Checkout('LLD', LLVM_REPO_URL + '/lld/trunk', LLD_DIR) 460 Checkout('compiler-rt', LLVM_REPO_URL + '/compiler-rt/trunk', COMPILER_RT_DIR) 461 if sys.platform == 'darwin': 462 # clang needs a libc++ checkout, else -stdlib=libc++ won't find includes 463 # (i.e. this is needed for bootstrap builds). 464 Checkout('libcxx', LLVM_REPO_URL + '/libcxx/trunk', LIBCXX_DIR) 465 # We used to check out libcxxabi on OS X; we no longer need that. 466 if os.path.exists(LIBCXXABI_DIR): 467 RmTree(LIBCXXABI_DIR) 468 469 cc, cxx = None, None 470 libstdcpp = None 471 if args.gcc_toolchain: # This option is only used on Linux. 472 # Use the specified gcc installation for building. 473 cc = os.path.join(args.gcc_toolchain, 'bin', 'gcc') 474 cxx = os.path.join(args.gcc_toolchain, 'bin', 'g++') 475 476 if not os.access(cc, os.X_OK): 477 print 'Invalid --gcc-toolchain: "%s"' % args.gcc_toolchain 478 print '"%s" does not appear to be valid.' % cc 479 return 1 480 481 # Set LD_LIBRARY_PATH to make auxiliary targets (tablegen, bootstrap 482 # compiler, etc.) find the .so. 483 libstdcpp = subprocess.check_output( 484 [cxx, '-print-file-name=libstdc++.so.6']).rstrip() 485 os.environ['LD_LIBRARY_PATH'] = os.path.dirname(libstdcpp) 486 487 cflags = [] 488 cxxflags = [] 489 ldflags = [] 490 491 base_cmake_args = ['-GNinja', 492 '-DCMAKE_BUILD_TYPE=Release', 493 '-DLLVM_ENABLE_ASSERTIONS=ON', 494 '-DLLVM_ENABLE_THREADS=OFF', 495 '-DLLVM_ENABLE_TIMESTAMPS=OFF', 496 # Statically link MSVCRT to avoid DLL dependencies. 497 '-DLLVM_USE_CRT_RELEASE=MT', 498 ] 499 500 binutils_incdir = '' 501 if sys.platform.startswith('linux'): 502 binutils_incdir = os.path.join(BINUTILS_DIR, 'Linux_x64/Release/include') 503 504 if args.bootstrap: 505 print 'Building bootstrap compiler' 506 EnsureDirExists(LLVM_BOOTSTRAP_DIR) 507 os.chdir(LLVM_BOOTSTRAP_DIR) 508 bootstrap_args = base_cmake_args + [ 509 '-DLLVM_BINUTILS_INCDIR=' + binutils_incdir, 510 '-DLLVM_TARGETS_TO_BUILD=host', 511 '-DCMAKE_INSTALL_PREFIX=' + LLVM_BOOTSTRAP_INSTALL_DIR, 512 '-DCMAKE_C_FLAGS=' + ' '.join(cflags), 513 '-DCMAKE_CXX_FLAGS=' + ' '.join(cxxflags), 514 ] 515 if cc is not None: bootstrap_args.append('-DCMAKE_C_COMPILER=' + cc) 516 if cxx is not None: bootstrap_args.append('-DCMAKE_CXX_COMPILER=' + cxx) 517 RmCmakeCache('.') 518 RunCommand(['cmake'] + bootstrap_args + [LLVM_DIR], msvc_arch='x64') 519 RunCommand(['ninja'], msvc_arch='x64') 520 if args.run_tests: 521 if sys.platform == 'win32': 522 CopyDiaDllTo(os.path.join(LLVM_BOOTSTRAP_DIR, 'bin')) 523 RunCommand(['ninja', 'check-all'], msvc_arch='x64') 524 RunCommand(['ninja', 'install'], msvc_arch='x64') 525 if args.gcc_toolchain: 526 # Copy that gcc's stdlibc++.so.6 to the build dir, so the bootstrap 527 # compiler can start. 528 CopyFile(libstdcpp, os.path.join(LLVM_BOOTSTRAP_INSTALL_DIR, 'lib')) 529 530 if sys.platform == 'win32': 531 cc = os.path.join(LLVM_BOOTSTRAP_INSTALL_DIR, 'bin', 'clang-cl.exe') 532 cxx = os.path.join(LLVM_BOOTSTRAP_INSTALL_DIR, 'bin', 'clang-cl.exe') 533 # CMake has a hard time with backslashes in compiler paths: 534 # https://stackoverflow.com/questions/13050827 535 cc = cc.replace('\\', '/') 536 cxx = cxx.replace('\\', '/') 537 else: 538 cc = os.path.join(LLVM_BOOTSTRAP_INSTALL_DIR, 'bin', 'clang') 539 cxx = os.path.join(LLVM_BOOTSTRAP_INSTALL_DIR, 'bin', 'clang++') 540 541 if args.gcc_toolchain: 542 # Tell the bootstrap compiler to use a specific gcc prefix to search 543 # for standard library headers and shared object files. 544 cflags = ['--gcc-toolchain=' + args.gcc_toolchain] 545 cxxflags = ['--gcc-toolchain=' + args.gcc_toolchain] 546 print 'Building final compiler' 547 548 # Build LLVM gold plugin with LTO. That speeds up the linker by ~10%. 549 # We only use LTO for Linux now. 550 if args.bootstrap and args.lto_gold_plugin: 551 print 'Building LTO LLVM Gold plugin' 552 if os.path.exists(LLVM_LTO_GOLD_PLUGIN_DIR): 553 RmTree(LLVM_LTO_GOLD_PLUGIN_DIR) 554 EnsureDirExists(LLVM_LTO_GOLD_PLUGIN_DIR) 555 os.chdir(LLVM_LTO_GOLD_PLUGIN_DIR) 556 557 # Create a symlink to LLVMgold.so build in the previous step so that ar 558 # and ranlib could find it while linking LLVMgold.so with LTO. 559 EnsureDirExists(BFD_PLUGINS_DIR) 560 RunCommand(['ln', '-sf', 561 os.path.join(LLVM_BOOTSTRAP_INSTALL_DIR, 'lib', 'LLVMgold.so'), 562 os.path.join(BFD_PLUGINS_DIR, 'LLVMgold.so')]) 563 564 lto_cflags = ['-flto'] 565 lto_ldflags = ['-fuse-ld=gold'] 566 if args.gcc_toolchain: 567 # Tell the bootstrap compiler to use a specific gcc prefix to search 568 # for standard library headers and shared object files. 569 lto_cflags += ['--gcc-toolchain=' + args.gcc_toolchain] 570 lto_cmake_args = base_cmake_args + [ 571 '-DLLVM_BINUTILS_INCDIR=' + binutils_incdir, 572 '-DCMAKE_C_COMPILER=' + cc, 573 '-DCMAKE_CXX_COMPILER=' + cxx, 574 '-DCMAKE_C_FLAGS=' + ' '.join(lto_cflags), 575 '-DCMAKE_CXX_FLAGS=' + ' '.join(lto_cflags), 576 '-DCMAKE_EXE_LINKER_FLAGS=' + ' '.join(lto_ldflags), 577 '-DCMAKE_SHARED_LINKER_FLAGS=' + ' '.join(lto_ldflags), 578 '-DCMAKE_MODULE_LINKER_FLAGS=' + ' '.join(lto_ldflags)] 579 580 # We need to use the proper binutils which support LLVM Gold plugin. 581 lto_env = os.environ.copy() 582 lto_env['PATH'] = BINUTILS_BIN_DIR + os.pathsep + lto_env.get('PATH', '') 583 584 RmCmakeCache('.') 585 RunCommand(['cmake'] + lto_cmake_args + [LLVM_DIR], env=lto_env) 586 RunCommand(['ninja', 'LLVMgold'], env=lto_env) 587 588 589 # LLVM uses C++11 starting in llvm 3.5. On Linux, this means libstdc++4.7+ is 590 # needed, on OS X it requires libc++. clang only automatically links to libc++ 591 # when targeting OS X 10.9+, so add stdlib=libc++ explicitly so clang can run 592 # on OS X versions as old as 10.7. 593 deployment_target = '' 594 595 if sys.platform == 'darwin' and args.bootstrap: 596 # When building on 10.9, /usr/include usually doesn't exist, and while 597 # Xcode's clang automatically sets a sysroot, self-built clangs don't. 598 cflags = ['-isysroot', subprocess.check_output( 599 ['xcrun', '--show-sdk-path']).rstrip()] 600 cxxflags = ['-stdlib=libc++'] + cflags 601 ldflags += ['-stdlib=libc++'] 602 deployment_target = '10.7' 603 # Running libc++ tests takes a long time. Since it was only needed for 604 # the install step above, don't build it as part of the main build. 605 # This makes running package.py over 10% faster (30 min instead of 34 min) 606 RmTree(LIBCXX_DIR) 607 608 # Build clang. 609 610 # If building at head, define a macro that plugins can use for #ifdefing 611 # out code that builds at head, but not at CLANG_REVISION or vice versa. 612 if use_head_revision: 613 cflags += ['-DLLVM_FORCE_HEAD_REVISION'] 614 cxxflags += ['-DLLVM_FORCE_HEAD_REVISION'] 615 616 CreateChromeToolsShim() 617 618 deployment_env = None 619 if deployment_target: 620 deployment_env = os.environ.copy() 621 deployment_env['MACOSX_DEPLOYMENT_TARGET'] = deployment_target 622 623 cmake_args = [] 624 # TODO(thakis): Unconditionally append this to base_cmake_args instead once 625 # compiler-rt can build with clang-cl on Windows (http://llvm.org/PR23698) 626 cc_args = base_cmake_args if sys.platform != 'win32' else cmake_args 627 if cc is not None: cc_args.append('-DCMAKE_C_COMPILER=' + cc) 628 if cxx is not None: cc_args.append('-DCMAKE_CXX_COMPILER=' + cxx) 629 cmake_args += base_cmake_args + [ 630 '-DLLVM_BINUTILS_INCDIR=' + binutils_incdir, 631 '-DCMAKE_C_FLAGS=' + ' '.join(cflags), 632 '-DCMAKE_CXX_FLAGS=' + ' '.join(cxxflags), 633 '-DCMAKE_EXE_LINKER_FLAGS=' + ' '.join(ldflags), 634 '-DCMAKE_SHARED_LINKER_FLAGS=' + ' '.join(ldflags), 635 '-DCMAKE_MODULE_LINKER_FLAGS=' + ' '.join(ldflags), 636 '-DCMAKE_INSTALL_PREFIX=' + LLVM_BUILD_DIR, 637 # TODO(thakis): Remove this once official builds pass -Wl,--build-id 638 # explicitly, https://crbug.com/622775 639 '-DENABLE_LINKER_BUILD_ID=ON', 640 '-DCHROMIUM_TOOLS_SRC=%s' % os.path.join(CHROMIUM_DIR, 'tools', 'clang'), 641 '-DCHROMIUM_TOOLS=%s' % ';'.join(args.tools)] 642 643 EnsureDirExists(LLVM_BUILD_DIR) 644 os.chdir(LLVM_BUILD_DIR) 645 RmCmakeCache('.') 646 RunCommand(['cmake'] + cmake_args + [LLVM_DIR], 647 msvc_arch='x64', env=deployment_env) 648 649 if args.gcc_toolchain: 650 # Copy in the right stdlibc++.so.6 so clang can start. 651 if not os.path.exists(os.path.join(LLVM_BUILD_DIR, 'lib')): 652 os.mkdir(os.path.join(LLVM_BUILD_DIR, 'lib')) 653 libstdcpp = subprocess.check_output( 654 [cxx] + cxxflags + ['-print-file-name=libstdc++.so.6']).rstrip() 655 CopyFile(libstdcpp, os.path.join(LLVM_BUILD_DIR, 'lib')) 656 657 RunCommand(['ninja'], msvc_arch='x64') 658 659 if args.tools: 660 # If any Chromium tools were built, install those now. 661 RunCommand(['ninja', 'cr-install'], msvc_arch='x64') 662 663 if sys.platform == 'darwin': 664 # See http://crbug.com/256342 665 RunCommand(['strip', '-x', os.path.join(LLVM_BUILD_DIR, 'bin', 'clang')]) 666 elif sys.platform.startswith('linux'): 667 RunCommand(['strip', os.path.join(LLVM_BUILD_DIR, 'bin', 'clang')]) 668 669 VeryifyVersionOfBuiltClangMatchesVERSION() 670 671 # Do an out-of-tree build of compiler-rt. 672 # On Windows, this is used to get the 32-bit ASan run-time. 673 # TODO(hans): Remove once the regular build above produces this. 674 # On Mac and Linux, this is used to get the regular 64-bit run-time. 675 # Do a clobbered build due to cmake changes. 676 if os.path.isdir(COMPILER_RT_BUILD_DIR): 677 RmTree(COMPILER_RT_BUILD_DIR) 678 os.makedirs(COMPILER_RT_BUILD_DIR) 679 os.chdir(COMPILER_RT_BUILD_DIR) 680 # TODO(thakis): Add this once compiler-rt can build with clang-cl (see 681 # above). 682 #if args.bootstrap and sys.platform == 'win32': 683 # The bootstrap compiler produces 64-bit binaries by default. 684 #cflags += ['-m32'] 685 #cxxflags += ['-m32'] 686 compiler_rt_args = base_cmake_args + [ 687 '-DCMAKE_C_FLAGS=' + ' '.join(cflags), 688 '-DCMAKE_CXX_FLAGS=' + ' '.join(cxxflags)] 689 if sys.platform == 'darwin': 690 compiler_rt_args += ['-DCOMPILER_RT_ENABLE_IOS=ON'] 691 if sys.platform != 'win32': 692 compiler_rt_args += ['-DLLVM_CONFIG_PATH=' + 693 os.path.join(LLVM_BUILD_DIR, 'bin', 'llvm-config'), 694 '-DSANITIZER_MIN_OSX_VERSION="10.7"'] 695 # compiler-rt is part of the llvm checkout on Windows but a stand-alone 696 # directory elsewhere, see the TODO above COMPILER_RT_DIR. 697 RmCmakeCache('.') 698 RunCommand(['cmake'] + compiler_rt_args + 699 [LLVM_DIR if sys.platform == 'win32' else COMPILER_RT_DIR], 700 msvc_arch='x86', env=deployment_env) 701 RunCommand(['ninja', 'compiler-rt'], msvc_arch='x86') 702 703 # Copy select output to the main tree. 704 # TODO(hans): Make this (and the .gypi and .isolate files) version number 705 # independent. 706 if sys.platform == 'win32': 707 platform = 'windows' 708 elif sys.platform == 'darwin': 709 platform = 'darwin' 710 else: 711 assert sys.platform.startswith('linux') 712 platform = 'linux' 713 asan_rt_lib_src_dir = os.path.join(COMPILER_RT_BUILD_DIR, 'lib', platform) 714 if sys.platform == 'win32': 715 # TODO(thakis): This too is due to compiler-rt being part of the checkout 716 # on Windows, see TODO above COMPILER_RT_DIR. 717 asan_rt_lib_src_dir = os.path.join(COMPILER_RT_BUILD_DIR, 'lib', 'clang', 718 VERSION, 'lib', platform) 719 asan_rt_lib_dst_dir = os.path.join(LLVM_BUILD_DIR, 'lib', 'clang', 720 VERSION, 'lib', platform) 721 # Blacklists: 722 CopyDirectoryContents(os.path.join(asan_rt_lib_src_dir, '..', '..'), 723 os.path.join(asan_rt_lib_dst_dir, '..', '..'), 724 r'^.*blacklist\.txt$') 725 # Headers: 726 if sys.platform != 'win32': 727 CopyDirectoryContents( 728 os.path.join(COMPILER_RT_BUILD_DIR, 'include/sanitizer'), 729 os.path.join(LLVM_BUILD_DIR, 'lib/clang', VERSION, 'include/sanitizer')) 730 # Static and dynamic libraries: 731 CopyDirectoryContents(asan_rt_lib_src_dir, asan_rt_lib_dst_dir) 732 if sys.platform == 'darwin': 733 for dylib in glob.glob(os.path.join(asan_rt_lib_dst_dir, '*.dylib')): 734 # Fix LC_ID_DYLIB for the ASan dynamic libraries to be relative to 735 # @executable_path. 736 # TODO(glider): this is transitional. We'll need to fix the dylib 737 # name either in our build system, or in Clang. See also 738 # http://crbug.com/344836. 739 subprocess.call(['install_name_tool', '-id', 740 '@executable_path/' + os.path.basename(dylib), dylib]) 741 742 743 if sys.platform == 'win32': 744 # Make an extra copy of the sanitizer headers, to be put on the include path 745 # of the fallback compiler. 746 sanitizer_include_dir = os.path.join(LLVM_BUILD_DIR, 'lib', 'clang', 747 VERSION, 'include', 'sanitizer') 748 aux_sanitizer_include_dir = os.path.join(LLVM_BUILD_DIR, 'lib', 'clang', 749 VERSION, 'include_sanitizer', 750 'sanitizer') 751 EnsureDirExists(aux_sanitizer_include_dir) 752 for _, _, files in os.walk(sanitizer_include_dir): 753 for f in files: 754 CopyFile(os.path.join(sanitizer_include_dir, f), 755 aux_sanitizer_include_dir) 756 757 if args.with_android: 758 make_toolchain = os.path.join( 759 ANDROID_NDK_DIR, 'build', 'tools', 'make_standalone_toolchain.py') 760 for target_arch in ['aarch64', 'arm', 'i686']: 761 # Make standalone Android toolchain for target_arch. 762 toolchain_dir = os.path.join( 763 LLVM_BUILD_DIR, 'android-toolchain-' + target_arch) 764 RunCommand([ 765 make_toolchain, 766 '--api=' + ('21' if target_arch == 'aarch64' else '19'), 767 '--force', 768 '--install-dir=%s' % toolchain_dir, 769 '--stl=stlport', 770 '--arch=' + { 771 'aarch64': 'arm64', 772 'arm': 'arm', 773 'i686': 'x86', 774 }[target_arch]]) 775 # Android NDK r9d copies a broken unwind.h into the toolchain, see 776 # http://crbug.com/357890 777 for f in glob.glob(os.path.join(toolchain_dir, 'include/c++/*/unwind.h')): 778 os.remove(f) 779 780 # Build ASan runtime for Android in a separate build tree. 781 build_dir = os.path.join(LLVM_BUILD_DIR, 'android-' + target_arch) 782 if not os.path.exists(build_dir): 783 os.mkdir(os.path.join(build_dir)) 784 os.chdir(build_dir) 785 cflags = ['--target=%s-linux-androideabi' % target_arch, 786 '--sysroot=%s/sysroot' % toolchain_dir, 787 '-B%s' % toolchain_dir] 788 android_args = base_cmake_args + [ 789 '-DCMAKE_C_COMPILER=' + os.path.join(LLVM_BUILD_DIR, 'bin/clang'), 790 '-DCMAKE_CXX_COMPILER=' + os.path.join(LLVM_BUILD_DIR, 'bin/clang++'), 791 '-DLLVM_CONFIG_PATH=' + os.path.join(LLVM_BUILD_DIR, 'bin/llvm-config'), 792 '-DCMAKE_C_FLAGS=' + ' '.join(cflags), 793 '-DCMAKE_CXX_FLAGS=' + ' '.join(cflags), 794 '-DANDROID=1'] 795 RmCmakeCache('.') 796 RunCommand(['cmake'] + android_args + [COMPILER_RT_DIR]) 797 RunCommand(['ninja', 'libclang_rt.asan-%s-android.so' % target_arch]) 798 799 # And copy it into the main build tree. 800 runtime = 'libclang_rt.asan-%s-android.so' % target_arch 801 for root, _, files in os.walk(build_dir): 802 if runtime in files: 803 shutil.copy(os.path.join(root, runtime), asan_rt_lib_dst_dir) 804 805 # Run tests. 806 if args.run_tests or use_head_revision: 807 os.chdir(LLVM_BUILD_DIR) 808 RunCommand(['ninja', 'cr-check-all'], msvc_arch='x64') 809 if args.run_tests: 810 if sys.platform == 'win32': 811 CopyDiaDllTo(os.path.join(LLVM_BUILD_DIR, 'bin')) 812 os.chdir(LLVM_BUILD_DIR) 813 RunCommand(['ninja', 'check-all'], msvc_arch='x64') 814 815 WriteStampFile(PACKAGE_VERSION) 816 print 'Clang update was successful.' 817 return 0 818 819 820def main(): 821 parser = argparse.ArgumentParser(description='Build Clang.') 822 parser.add_argument('--bootstrap', action='store_true', 823 help='first build clang with CC, then with itself.') 824 parser.add_argument('--if-needed', action='store_true', 825 help="run only if the script thinks clang is needed") 826 parser.add_argument('--force-local-build', action='store_true', 827 help="don't try to download prebuild binaries") 828 parser.add_argument('--gcc-toolchain', help='set the version for which gcc ' 829 'version be used for building; --gcc-toolchain=/opt/foo ' 830 'picks /opt/foo/bin/gcc') 831 parser.add_argument('--lto-gold-plugin', action='store_true', 832 help='build LLVM Gold plugin with LTO') 833 parser.add_argument('--llvm-force-head-revision', action='store_true', 834 help=('use the revision in the repo when printing ' 835 'the revision')) 836 parser.add_argument('--print-revision', action='store_true', 837 help='print current clang revision and exit.') 838 parser.add_argument('--print-clang-version', action='store_true', 839 help='print current clang version (e.g. x.y.z) and exit.') 840 parser.add_argument('--run-tests', action='store_true', 841 help='run tests after building; only for local builds') 842 parser.add_argument('--tools', nargs='*', 843 help='select which chrome tools to build', 844 default=['plugins', 'blink_gc_plugin']) 845 parser.add_argument('--without-android', action='store_false', 846 help='don\'t build Android ASan runtime (linux only)', 847 dest='with_android', 848 default=sys.platform.startswith('linux')) 849 args = parser.parse_args() 850 851 if args.lto_gold_plugin and not args.bootstrap: 852 print '--lto-gold-plugin requires --bootstrap' 853 return 1 854 if args.lto_gold_plugin and not sys.platform.startswith('linux'): 855 print '--lto-gold-plugin is only effective on Linux. Ignoring the option.' 856 args.lto_gold_plugin = False 857 858 if args.if_needed: 859 is_clang_required = False 860 # clang is always used on Mac and Linux. 861 if sys.platform == 'darwin' or sys.platform.startswith('linux'): 862 is_clang_required = True 863 # clang requested via $GYP_DEFINES. 864 if re.search(r'\b(clang|asan|lsan|msan|tsan)=1', 865 os.environ.get('GYP_DEFINES', '')): 866 is_clang_required = True 867 # clang previously downloaded, keep it up to date. 868 # If you don't want this, delete third_party/llvm-build on your machine. 869 if os.path.isdir(LLVM_BUILD_DIR): 870 is_clang_required = True 871 if not is_clang_required: 872 return 0 873 if re.search(r'\b(make_clang_dir)=', os.environ.get('GYP_DEFINES', '')): 874 print 'Skipping Clang update (make_clang_dir= was set in GYP_DEFINES).' 875 return 0 876 877 global CLANG_REVISION, PACKAGE_VERSION 878 if args.print_revision: 879 if use_head_revision or args.llvm_force_head_revision: 880 print GetSvnRevision(LLVM_DIR) 881 else: 882 print PACKAGE_VERSION 883 return 0 884 885 if args.print_clang_version: 886 sys.stdout.write(VERSION) 887 return 0 888 889 # Don't buffer stdout, so that print statements are immediately flushed. 890 # Do this only after --print-revision has been handled, else we'll get 891 # an error message when this script is run from gn for some reason. 892 sys.stdout = os.fdopen(sys.stdout.fileno(), 'w', 0) 893 894 if use_head_revision: 895 # Use a real revision number rather than HEAD to make sure that the stamp 896 # file logic works. 897 CLANG_REVISION = GetSvnRevision(LLVM_REPO_URL) 898 PACKAGE_VERSION = CLANG_REVISION + '-0' 899 900 args.force_local_build = True 901 if 'OS=android' not in os.environ.get('GYP_DEFINES', ''): 902 # Only build the Android ASan rt on ToT bots when targetting Android. 903 args.with_android = False 904 905 return UpdateClang(args) 906 907 908if __name__ == '__main__': 909 sys.exit(main()) 910