|  | #!/usr/bin/env python2 | 
|  |  | 
|  | import argparse | 
|  | import os | 
|  | import shutil | 
|  | import tempfile | 
|  |  | 
|  | import targets | 
|  | from utils import FindBaseNaCl, GetObjcopyCmd, shellcmd | 
|  |  | 
|  |  | 
|  | def Translate(ll_files, extra_args, obj, verbose, target): | 
|  | """Translate a set of input bitcode files into a single object file. | 
|  |  | 
|  | Use pnacl-llc to translate textual bitcode input ll_files into object file | 
|  | obj, using extra_args as the architectural flags. | 
|  | """ | 
|  | externalize = ['-externalize'] | 
|  | shellcmd(['cat'] + ll_files + ['|', | 
|  | 'pnacl-llc', | 
|  | '-function-sections', | 
|  | '-O2', | 
|  | '-filetype=obj', | 
|  | '-bitcode-format=llvm', | 
|  | '-o', obj | 
|  | ] + extra_args + externalize, echo=verbose) | 
|  | localize_syms = ['nacl_tp_tdb_offset', 'nacl_tp_tls_offset'] | 
|  |  | 
|  | shellcmd([GetObjcopyCmd(target), obj] + | 
|  | [('--localize-symbol=' + sym) for sym in localize_syms]) | 
|  |  | 
|  |  | 
|  | def PartialLink(obj_files, extra_args, lib, verbose): | 
|  | """Partially links a set of obj files into a final obj library.""" | 
|  | shellcmd(['le32-nacl-ld', | 
|  | '-o', lib, | 
|  | '-r', | 
|  | ] + extra_args + obj_files, echo=verbose) | 
|  |  | 
|  |  | 
|  | def MakeRuntimesForTarget(target_info, ll_files, | 
|  | srcdir, tempdir, rtdir, verbose, excluded_targets): | 
|  | """Builds native, sandboxed, and nonsfi runtimes for the given target.""" | 
|  | if target_info.target in excluded_targets: | 
|  | return | 
|  | # File-mangling helper functions. | 
|  | def TmpFile(template): | 
|  | return template.format(dir=tempdir, target=target_info.target) | 
|  | def OutFile(template): | 
|  | return template.format(rtdir=rtdir, target=target_info.target) | 
|  | # Helper function for building the native unsandboxed runtime. | 
|  | def MakeNativeRuntime(): | 
|  | """Builds just the native runtime.""" | 
|  | # Translate tempdir/szrt.ll and tempdir/szrt_ll.ll to | 
|  | # szrt_native_{target}.tmp.o. | 
|  | Translate(ll_files, | 
|  | ['-mtriple=' + target_info.triple] + target_info.llc_flags, | 
|  | TmpFile('{dir}/szrt_native_{target}.tmp.o'), | 
|  | verbose, target_info.target) | 
|  | # Compile srcdir/szrt_profiler.c to | 
|  | # tempdir/szrt_profiler_native_{target}.o. | 
|  | shellcmd(['clang', | 
|  | '-O2', | 
|  | '-target=' + target_info.triple, | 
|  | '-c', | 
|  | '{srcdir}/szrt_profiler.c'.format(srcdir=srcdir), | 
|  | '-o', TmpFile('{dir}/szrt_native_profiler_{target}.o') | 
|  | ], echo=verbose) | 
|  | # Assemble srcdir/szrt_asm_{target}.s to tempdir/szrt_asm_{target}.o. | 
|  | shellcmd(['llvm-mc', | 
|  | '-triple=' + target_info.triple, '--defsym NATIVE=1', | 
|  | '-filetype=obj', | 
|  | '-o', TmpFile('{dir}/szrt_native_asm_{target}.o'), | 
|  | '{srcdir}/szrt_asm_{target}.s'.format( | 
|  | srcdir=srcdir, target=target_info.target) | 
|  | ], echo=verbose) | 
|  | # Write full szrt_native_{target}.o. | 
|  | PartialLink([TmpFile('{dir}/szrt_native_{target}.tmp.o'), | 
|  | TmpFile('{dir}/szrt_native_asm_{target}.o'), | 
|  | TmpFile('{dir}/szrt_native_profiler_{target}.o')], | 
|  | ['-m {ld_emu}'.format(ld_emu=target_info.ld_emu)], | 
|  | OutFile('{rtdir}/szrt_native_{target}.o'), | 
|  | verbose) | 
|  | shellcmd([GetObjcopyCmd(target_info.target), | 
|  | '--strip-symbol=NATIVE', | 
|  | OutFile('{rtdir}/szrt_native_{target}.o')]) | 
|  | # Compile srcdir/szrt_asan.c to szrt_asan_{target}.o | 
|  | shellcmd(['clang', | 
|  | '-O2', | 
|  | '-target=' + target_info.triple, | 
|  | '-c', | 
|  | '{srcdir}/szrt_asan.c'.format(srcdir=srcdir), | 
|  | '-o', OutFile('{rtdir}/szrt_asan_{target}.o') | 
|  | ], echo=verbose) | 
|  |  | 
|  | # Helper function for building the sandboxed runtime. | 
|  | def MakeSandboxedRuntime(): | 
|  | """Builds just the sandboxed runtime.""" | 
|  | # Translate tempdir/szrt.ll and tempdir/szrt_ll.ll to szrt_sb_{target}.o. | 
|  | # The sandboxed library does not get the profiler helper function as the | 
|  | # binaries are linked with -nostdlib. | 
|  | Translate(ll_files, | 
|  | ['-mtriple=' + targets.ConvertTripleToNaCl(target_info.triple)] + | 
|  | target_info.llc_flags, | 
|  | TmpFile('{dir}/szrt_sb_{target}.tmp.o'), | 
|  | verbose,target_info.target) | 
|  | # Assemble srcdir/szrt_asm_{target}.s to tempdir/szrt_asm_{target}.o. | 
|  | shellcmd(['llvm-mc', | 
|  | '-triple=' + targets.ConvertTripleToNaCl(target_info.triple), | 
|  | '--defsym NACL=1', | 
|  | '-filetype=obj', | 
|  | '-o', TmpFile('{dir}/szrt_sb_asm_{target}.o'), | 
|  | '{srcdir}/szrt_asm_{target}.s'.format( | 
|  | srcdir=srcdir, target=target_info.target) | 
|  | ], echo=verbose) | 
|  | PartialLink([TmpFile('{dir}/szrt_sb_{target}.tmp.o'), | 
|  | TmpFile('{dir}/szrt_sb_asm_{target}.o')], | 
|  | ['-m {ld_emu}'.format(ld_emu=target_info.sb_emu)], | 
|  | OutFile('{rtdir}/szrt_sb_{target}.o'), | 
|  | verbose) | 
|  | shellcmd([GetObjcopyCmd(target_info.target), | 
|  | '--strip-symbol=NACL', | 
|  | OutFile('{rtdir}/szrt_sb_{target}.o')]) | 
|  |  | 
|  | # Helper function for building the Non-SFI runtime. | 
|  | def MakeNonsfiRuntime(): | 
|  | """Builds just the nonsfi runtime.""" | 
|  | # Translate tempdir/szrt.ll and tempdir/szrt_ll.ll to | 
|  | # szrt_nonsfi_{target}.tmp.o. | 
|  | Translate(ll_files, | 
|  | ['-mtriple=' + target_info.triple] + target_info.llc_flags + | 
|  | ['-relocation-model=pic', '-force-tls-non-pic', '-malign-double'], | 
|  | TmpFile('{dir}/szrt_nonsfi_{target}.tmp.o'), | 
|  | verbose, target_info.target) | 
|  | # Assemble srcdir/szrt_asm_{target}.s to tempdir/szrt_asm_{target}.o. | 
|  | shellcmd(['llvm-mc', | 
|  | '-triple=' + target_info.triple, '--defsym NONSFI=1', | 
|  | '-filetype=obj', | 
|  | '-o', TmpFile('{dir}/szrt_nonsfi_asm_{target}.o'), | 
|  | '{srcdir}/szrt_asm_{target}.s'.format( | 
|  | srcdir=srcdir, target=target_info.target) | 
|  | ], echo=verbose) | 
|  | # Write full szrt_nonsfi_{target}.o. | 
|  | PartialLink([TmpFile('{dir}/szrt_nonsfi_{target}.tmp.o'), | 
|  | TmpFile('{dir}/szrt_nonsfi_asm_{target}.o')], | 
|  | ['-m {ld_emu}'.format(ld_emu=target_info.ld_emu)], | 
|  | OutFile('{rtdir}/szrt_nonsfi_{target}.o'), | 
|  | verbose) | 
|  | shellcmd([GetObjcopyCmd(target_info.target), | 
|  | '--strip-symbol=NONSFI', | 
|  | OutFile('{rtdir}/szrt_nonsfi_{target}.o')]) | 
|  |  | 
|  |  | 
|  | # Run the helper functions. | 
|  | MakeNativeRuntime() | 
|  | MakeSandboxedRuntime() | 
|  | MakeNonsfiRuntime() | 
|  |  | 
|  |  | 
|  | def main(): | 
|  | """Build the Subzero runtime support library for all architectures. | 
|  | """ | 
|  | nacl_root = FindBaseNaCl() | 
|  | argparser = argparse.ArgumentParser( | 
|  | description='    ' + main.__doc__, | 
|  | formatter_class=argparse.RawTextHelpFormatter) | 
|  | argparser.add_argument('--verbose', '-v', dest='verbose', | 
|  | action='store_true', | 
|  | help='Display some extra debugging output') | 
|  | argparser.add_argument('--pnacl-root', dest='pnacl_root', | 
|  | default=( | 
|  | '{root}/toolchain/linux_x86/pnacl_newlib_raw' | 
|  | ).format(root=nacl_root), | 
|  | help='Path to PNaCl toolchain binaries.') | 
|  | argparser.add_argument('--exclude-target', dest='excluded_targets', | 
|  | default=[], action='append', | 
|  | help='Target whose runtime should not be built') | 
|  | args = argparser.parse_args() | 
|  | os.environ['PATH'] = ('{root}/bin{sep}{path}' | 
|  | ).format(root=args.pnacl_root, sep=os.pathsep, path=os.environ['PATH']) | 
|  | srcdir = ( | 
|  | '{root}/toolchain_build/src/subzero/runtime' | 
|  | ).format(root=nacl_root) | 
|  | rtdir = ( | 
|  | '{root}/toolchain_build/src/subzero/build/runtime' | 
|  | ).format(root=nacl_root) | 
|  | try: | 
|  | tempdir = tempfile.mkdtemp() | 
|  | if os.path.exists(rtdir) and not os.path.isdir(rtdir): | 
|  | os.remove(rtdir) | 
|  | if not os.path.exists(rtdir): | 
|  | os.makedirs(rtdir) | 
|  | # Compile srcdir/szrt.c to tempdir/szrt.ll | 
|  | shellcmd(['pnacl-clang', | 
|  | '-O2', | 
|  | '-c', | 
|  | '{srcdir}/szrt.c'.format(srcdir=srcdir), | 
|  | '-o', '{dir}/szrt.tmp.bc'.format(dir=tempdir) | 
|  | ], echo=args.verbose) | 
|  | shellcmd(['pnacl-opt', | 
|  | '-pnacl-abi-simplify-preopt', | 
|  | '-pnacl-abi-simplify-postopt', | 
|  | '-pnaclabi-allow-debug-metadata', | 
|  | '{dir}/szrt.tmp.bc'.format(dir=tempdir), | 
|  | '-S', | 
|  | '-o', '{dir}/szrt.ll'.format(dir=tempdir) | 
|  | ], echo=args.verbose) | 
|  | ll_files = ['{dir}/szrt.ll'.format(dir=tempdir), | 
|  | '{srcdir}/szrt_ll.ll'.format(srcdir=srcdir)] | 
|  |  | 
|  | MakeRuntimesForTarget(targets.X8632Target, ll_files, | 
|  | srcdir, tempdir, rtdir, args.verbose, | 
|  | args.excluded_targets) | 
|  | MakeRuntimesForTarget(targets.X8664Target, ll_files, | 
|  | srcdir, tempdir, rtdir, args.verbose, | 
|  | args.excluded_targets) | 
|  | MakeRuntimesForTarget(targets.ARM32Target, ll_files, | 
|  | srcdir, tempdir, rtdir, args.verbose, | 
|  | args.excluded_targets) | 
|  | MakeRuntimesForTarget(targets.MIPS32Target, ll_files, | 
|  | srcdir, tempdir, rtdir, args.verbose, | 
|  | args.excluded_targets) | 
|  |  | 
|  | finally: | 
|  | try: | 
|  | shutil.rmtree(tempdir) | 
|  | except OSError as exc: | 
|  | if exc.errno != errno.ENOENT: # ENOENT - no such file or directory | 
|  | raise # re-raise exception | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | main() |