OLD | NEW |
---|---|
1 #!/usr/bin/env python2 | 1 #!/usr/bin/env python2 |
2 | 2 |
3 import argparse | 3 import argparse |
4 import os | 4 import os |
5 import re | |
6 import subprocess | 5 import subprocess |
7 import sys | 6 import sys |
8 import tempfile | 7 import tempfile |
9 | 8 |
10 from utils import shellcmd | 9 from utils import shellcmd |
11 from utils import FindBaseNaCl | 10 from utils import FindBaseNaCl |
12 | 11 |
13 if __name__ == '__main__': | 12 def main(): |
14 """Builds a cross-test binary that allows functions translated by | 13 """Builds a cross-test binary that allows functions translated by |
15 Subzero and llc to be compared. | 14 Subzero and llc to be compared. |
16 | 15 |
17 Each --test argument is compiled once by llc and once by Subzero. | 16 Each --test argument is compiled once by llc and once by Subzero. |
18 C/C++ tests are first compiled down to PNaCl bitcode by the | 17 C/C++ tests are first compiled down to PNaCl bitcode by the |
19 build-pnacl-ir.py script. The --prefix argument ensures that | 18 build-pnacl-ir.py script. The --prefix argument ensures that |
20 symbol names are different between the two object files, to avoid | 19 symbol names are different between the two object files, to avoid |
21 linking errors. | 20 linking errors. |
22 | 21 |
23 There is also a --driver argument that specifies the C/C++ file | 22 There is also a --driver argument that specifies the C/C++ file |
(...skipping 18 matching lines...) Expand all Loading... | |
42 argparser.add_argument('-O', required=False, default='2', dest='optlevel', | 41 argparser.add_argument('-O', required=False, default='2', dest='optlevel', |
43 choices=['m1', '-1', '0', '1', '2'], | 42 choices=['m1', '-1', '0', '1', '2'], |
44 metavar='OPTLEVEL', | 43 metavar='OPTLEVEL', |
45 help='Optimization level ' + | 44 help='Optimization level ' + |
46 '(m1 and -1 are equivalent).' + | 45 '(m1 and -1 are equivalent).' + |
47 ' Default %(default)s.') | 46 ' Default %(default)s.') |
48 argparser.add_argument('--mattr', required=False, default='sse2', | 47 argparser.add_argument('--mattr', required=False, default='sse2', |
49 dest='attr', choices=['sse2', 'sse4.1'], | 48 dest='attr', choices=['sse2', 'sse4.1'], |
50 metavar='ATTRIBUTE', | 49 metavar='ATTRIBUTE', |
51 help='Target attribute. Default %(default)s.') | 50 help='Target attribute. Default %(default)s.') |
51 argparser.add_argument('--sandbox', required=False, default=0, type=int, | |
52 dest='sandbox', | |
53 help='Use sandboxing. Default "%(default)s".') | |
52 argparser.add_argument('--prefix', required=True, | 54 argparser.add_argument('--prefix', required=True, |
53 metavar='SZ_PREFIX', | 55 metavar='SZ_PREFIX', |
54 help='String prepended to Subzero symbol names') | 56 help='String prepended to Subzero symbol names') |
55 argparser.add_argument('--output', '-o', required=True, | 57 argparser.add_argument('--output', '-o', required=True, |
56 metavar='EXECUTABLE', | 58 metavar='EXECUTABLE', |
57 help='Executable to produce') | 59 help='Executable to produce') |
58 argparser.add_argument('--dir', required=False, default='.', | 60 argparser.add_argument('--dir', required=False, default='.', |
59 metavar='OUTPUT_DIR', | 61 metavar='OUTPUT_DIR', |
60 help='Output directory for all files.' + | 62 help='Output directory for all files.' + |
61 ' Default "%(default)s".') | 63 ' Default "%(default)s".') |
62 argparser.add_argument('--crosstest-bitcode', required=False, | 64 argparser.add_argument('--crosstest-bitcode', required=False, |
63 default=1, type=int, | 65 default=1, type=int, |
64 help='Compile non-subzero crosstest object file ' + | 66 help='Compile non-subzero crosstest object file ' + |
65 'from the same bitcode as the subzero object. ' + | 67 'from the same bitcode as the subzero object. ' + |
66 'If 0, then compile it straight from source.' + | 68 'If 0, then compile it straight from source.' + |
67 ' Default %(default)d.') | 69 ' Default %(default)d.') |
68 argparser.add_argument('--filetype', default='obj', dest='filetype', | 70 argparser.add_argument('--filetype', default='obj', dest='filetype', |
69 choices=['obj', 'asm', 'iasm'], | 71 choices=['obj', 'asm', 'iasm'], |
70 help='Output file type. Default %(default)s.') | 72 help='Output file type. Default %(default)s.') |
71 args = argparser.parse_args() | 73 args = argparser.parse_args() |
72 | 74 |
73 nacl_root = FindBaseNaCl() | 75 nacl_root = FindBaseNaCl() |
74 # Prepend PNaCl bin to $PATH. | 76 bindir = ('{root}/toolchain/linux_x86/pnacl_newlib/bin' |
75 os.environ['PATH'] = nacl_root + \ | 77 .format(root=nacl_root)) |
76 '/toolchain/linux_x86/pnacl_newlib/bin' + \ | 78 triple = arch_map[args.target] + ('-nacl' if args.sandbox else '') |
77 os.pathsep + os.environ['PATH'] | |
78 | 79 |
79 objs = [] | 80 objs = [] |
80 remove_internal = re.compile('^define internal ') | |
81 fix_target = re.compile('le32-unknown-nacl') | |
82 for arg in args.test: | 81 for arg in args.test: |
83 base, ext = os.path.splitext(arg) | 82 base, ext = os.path.splitext(arg) |
84 if ext == '.ll': | 83 if ext == '.ll': |
85 bitcode = arg | 84 bitcode = arg |
86 else: | 85 else: |
87 bitcode = os.path.join(args.dir, base + '.pnacl.ll') | 86 bitcode = os.path.join(args.dir, base + '.pnacl.ll') |
88 shellcmd(['../pydir/build-pnacl-ir.py', '--disable-verify', | 87 shellcmd(['../pydir/build-pnacl-ir.py', '--disable-verify', |
89 '--dir', args.dir, arg]) | 88 '--dir', args.dir, arg]) |
90 # Read in the bitcode file, fix it up, and rewrite the file. | |
91 f = open(bitcode) | |
92 ll_lines = f.readlines() | |
93 f.close() | |
94 f = open(bitcode, 'w') | |
95 for line in ll_lines: | |
96 line = remove_internal.sub('define ', line) | |
97 line = fix_target.sub('i686-pc-linux-gnu', line) | |
jvoung (off chromium)
2015/03/03 22:58:02
nice =)
| |
98 f.write(line) | |
99 f.close() | |
100 | 89 |
101 base_sz = '%s.O%s.%s.%s' % (base, args.optlevel, args.attr, args.target) | 90 base_sz = '{base}.{sb}.O{opt}.{attr}.{target}'.format( |
91 base=base, sb='sb' if args.sandbox else 'nat', opt=args.optlevel, | |
92 attr=args.attr, target=args.target) | |
102 asm_sz = os.path.join(args.dir, base_sz + '.sz.s') | 93 asm_sz = os.path.join(args.dir, base_sz + '.sz.s') |
103 obj_sz = os.path.join(args.dir, base_sz + '.sz.o') | 94 obj_sz = os.path.join(args.dir, base_sz + '.sz.o') |
104 obj_llc = os.path.join(args.dir, base + '.llc.o') | 95 obj_llc = os.path.join(args.dir, base_sz + '.llc.o') |
105 shellcmd(['../pnacl-sz', | 96 shellcmd(['../pnacl-sz', |
106 '-O' + args.optlevel, | 97 '-O' + args.optlevel, |
107 '-mattr=' + args.attr, | 98 '-mattr=' + args.attr, |
108 '--target=' + args.target, | 99 '--target=' + args.target, |
100 '--sandbox=' + str(args.sandbox), | |
109 '--prefix=' + args.prefix, | 101 '--prefix=' + args.prefix, |
110 '-allow-uninitialized-globals', | 102 '-allow-uninitialized-globals', |
103 '-externalize', | |
111 '-filetype=' + args.filetype, | 104 '-filetype=' + args.filetype, |
112 '-o=' + (obj_sz if args.filetype == 'obj' else asm_sz), | 105 '-o=' + (obj_sz if args.filetype == 'obj' else asm_sz), |
113 bitcode]) | 106 bitcode]) |
114 if args.filetype != 'obj': | 107 if args.filetype != 'obj': |
115 shellcmd(['llvm-mc', | 108 shellcmd(['{bin}/llvm-mc'.format(bin=bindir), |
116 '-triple=' + arch_map[args.target], | 109 '-triple=' + triple, |
117 '-filetype=obj', | 110 '-filetype=obj', |
118 '-o=' + obj_sz, | 111 '-o=' + obj_sz, |
119 asm_sz]) | 112 asm_sz]) |
120 objs.append(obj_sz) | 113 objs.append(obj_sz) |
121 # Each original bitcode file needs to be translated by the | 114 if args.crosstest_bitcode: |
122 # LLVM toolchain and have its object file linked in. There | 115 shellcmd(['{bin}/pnacl-llc'.format(bin=bindir), |
123 # are two ways to do this: explicitly use llc, or include the | 116 '-mtriple=' + triple, |
124 # .ll file in the link command. It turns out that these two | 117 # Use sse2 instructions regardless of input -mattr |
125 # approaches can produce different semantics on some undefined | 118 # argument to avoid differences in (undefined) behavior of |
126 # bitcode behavior. Specifically, LLVM produces different | 119 # converting NaN to int. |
127 # results for overflowing fptoui instructions for i32 and i64 | 120 '-mattr=sse2', |
128 # on x86-32. As it turns out, Subzero lowering was based on | 121 '-externalize', |
129 # inspecting the object code produced by the direct llc | |
130 # command, so we need to directly run llc on the bitcode, even | |
131 # though it makes this script longer, to avoid spurious | |
132 # failures. This behavior can be inspected by switching | |
133 # use_llc between True and False. | |
134 use_llc = False | |
135 pure_c = os.path.splitext(args.driver)[1] == '.c' | |
136 if not args.crosstest_bitcode: | |
137 objs.append(arg) | |
138 elif use_llc: | |
139 shellcmd(['llc' | |
140 '-filetype=obj', | 122 '-filetype=obj', |
141 '-o=' + obj_llc, | 123 '-o=' + obj_llc, |
142 bitcode]) | 124 bitcode]) |
143 objs.append(obj_llc) | 125 objs.append(obj_llc) |
144 else: | 126 else: |
145 objs.append(bitcode) | 127 objs.append(arg) |
146 | 128 |
129 # Add szrt_sb_x8632.o or szrt_native_x8632.o. | |
147 objs.append(( | 130 objs.append(( |
148 '{root}/toolchain_build/src/subzero/build/runtime/' + | 131 '{root}/toolchain_build/src/subzero/build/runtime/' + |
149 'szrt_native_x8632.o' | 132 'szrt_{sb}_' + args.target + '.o' |
150 ).format(root=nacl_root)) | 133 ).format(root=nacl_root, sb='sb' if args.sandbox else 'native')) |
151 linker = 'clang' if pure_c else 'clang++' | 134 pure_c = os.path.splitext(args.driver)[1] == '.c' |
152 shellcmd([linker, '-g', '-m32', args.driver] + | 135 # Set compiler to clang, clang++, pnacl-clang, or pnacl-clang++. |
153 objs + | 136 compiler = '{bin}/{prefix}{cc}'.format( |
154 ['-lm', '-lpthread', '-o', os.path.join(args.dir, args.output)]) | 137 bin=bindir, prefix='pnacl-' if args.sandbox else '', |
138 cc='clang' if pure_c else 'clang++') | |
139 sb_native_args = (['-O0', '--pnacl-allow-native', '-arch', 'x8632'] | |
140 if args.sandbox else | |
141 ['-g', '-m32', '-lm', '-lpthread']) | |
142 shellcmd([compiler, args.driver] + objs + | |
143 ['-o', os.path.join(args.dir, args.output)] + sb_native_args) | |
144 | |
145 if __name__ == '__main__': | |
146 main() | |
OLD | NEW |