-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathbinutils-batch-build-asan.py
executable file
·269 lines (255 loc) · 6.12 KB
/
binutils-batch-build-asan.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
#!/usr/bin/env python3
import subprocess
import tempfile
import psutil
import os
import concurrent.futures
import sys
targets = '''
aarch64-elf
aarch64-linux
aarch64_be-linux-gnu_ilp32
alpha-dec-vms
alpha-linux
alpha-linuxecoff
alpha-netbsd
alpha-unknown-freebsd4.7
am33_2.0-linux
arc-elf
arc-linux-uclibc
arm-elf
arm-linuxeabi
arm-nacl
arm-netbsdelf
arm-nto
arm-pe
arm-symbianelf
arm-vxworks
arm-wince-pe
armeb-linuxeabi
avr-elf
bfin-elf
bfin-linux-uclibc
bpf-none
cr16-elf
cris-elf
cris-linux
crisv32-linux
crx-elf
csky-elf
csky-linux
d10v-elf
d30v-elf
dlx-elf
epiphany-elf
fr30-elf
frv-elf
frv-linux
ft32-elf
h8300-elf
h8300-linux
hppa-hp-hpux10
hppa-linux
hppa64-hp-hpux11.23
hppa64-linux
i386-bsd
i386-darwin
i386-lynxos
i386-msdos
i586-linux
i686-nto
i686-pc-beos
i686-pc-elf
i686-pe
i686-vxworks
ia64-elf
ia64-freebsd5
ia64-hpux
ia64-linux
ia64-netbsd
ia64-vms
ip2k-elf
iq2000-elf
lm32-elf
lm32-linux
m32c-elf
m32r-elf
m32r-linux
m68hc11-elf
m68hc12-elf
m68k-elf
m68k-linux
mcore-elf
mcore-pe
mep-elf
metag-linux
microblaze-elf
microblaze-linux
mips-linux
mips-sgi-irix6
mips-vxworks
mips64-linux
mips64-openbsd
mips64el-openbsd
mipsel-linux-gnu
mipsisa32el-linux
mipstx39-elf
mmix
mn10200-elf
mn10300-elf
moxie-elf
msp430-elf
mt-elf
nds32be-elf
nds32le-linux
nios2-linux
or1k-elf
or1k-linux
pdp11-dec-aout
pj-elf
powerpc-aix5.1
powerpc-aix5.2
powerpc-eabisim
powerpc-eabivle
powerpc-freebsd
powerpc-linux
powerpc-nto
powerpc-wrs-vxworks
powerpc64-freebsd
powerpc64-linux
powerpc64le-linux
powerpcle-elf
pru-elf
riscv32-elf
riscv64-linux
rl78-elf
rs6000-aix4.3.3
rs6000-aix5.1
rs6000-aix5.2
rx-elf
s12z-elf
s390-linux
s390x-linux
score-elf
sh-linux
sh-nto
sh-pe
sh-rtems
sh-vxworks
shle-unknown-netbsdelf
sparc-elf
sparc-linux
sparc-sun-solaris2
sparc-vxworks
sparc64-linux
spu-elf
tic30-unknown-coff
tic4x-coff
tic54x-coff
tic6x-elf
tilegx-linux
tilepro-linux
v850-elf
vax-netbsdelf
visium-elf
wasm32
x86_64-cloudabi
x86_64-linux
x86_64-pc-linux-gnux32
x86_64-rdos
x86_64-w64-mingw32
xgate-elf
xstormy16-elf
xtensa-elf
z80-coff
z80-elf
z8k-coff
'''
targets = targets.strip().split('\n')
cpu_count = psutil.cpu_count()
def build_and_test_target(target):
try:
folder = tempfile.TemporaryDirectory(prefix='/dev/shm/')
os.chdir(folder.name)
subprocess.check_output('%s/configure --build=x86_64-linux --disable-gdb --disable-gdbserver --disable-sim --enable-obsolete --target=%s CFLAGS="-g -O2 -fsanitize=address,undefined -Wno-error" CXXLAGS="-g -O2 -fsanitize=address,undefined -Wno-error" LDFLAGS="-ldl"'
% (sys.argv[1], target), shell=True, stderr=subprocess.DEVNULL)
env = os.environ.copy()
env['ASAN_OPTIONS'] = 'detect_leaks=0'
r = subprocess.run('make -j8', shell=True, stdout=subprocess.DEVNULL, stderr=subprocess.PIPE, encoding='utf8', env=env)
if r.returncode != 0:
errors = [l for l in r.stderr.split('\n') if 'error:' in l]
assert errors
return errors
subprocess.run('make check -k', shell=True, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
output = subprocess.check_output('find . -name "*.log" | xargs cat', shell=True, stderr=subprocess.DEVNULL, encoding='utf8').strip()
print('D', end='', flush=True)
return output.split('\n') if output else []
finally:
folder.cleanup()
seen = set()
def report_error(command, test_case, location, full_line, target):
prefix = 'ASAN error: ' if 'AddressSanitizer' in full_line else 'UBSAN error: '
if not location in seen:
seen.add(location)
print('-' * 80)
print(prefix + location)
print('The following fails when I build binutils with:')
print('configure --build=x86_64-linux --disable-gdb --disable-gdbserver --disable-werror --enable-obsolete --target=%s ' \
'CFLAGS="-g -O2 -fsanitize=address,undefined -Wno-error"' \
'CXXLAGS="-g -O2 -fsanitize=address,undefined -Wno-error"' \
'LDFLAGS="-ldl"' % target)
print('Target: %s' % target)
print(test_case)
print(command)
print(full_line)
print()
results = {}
print('.' * len(targets))
with concurrent.futures.ProcessPoolExecutor(max_workers=16) as executor:
future_map = {executor.submit(build_and_test_target, target): target for target in targets}
for future in concurrent.futures.as_completed(future_map):
target = future_map[future]
try:
errors = future.result()
results[target] = future.result()
except Exception as exc:
print(f'{target!r} generated an exception: {exc}')
print()
for target in targets:
errors = results[target]
print('=== %s ===' % target)
for i, err in enumerate(errors):
if err.startswith('succeeded with: ') or err.startswith('failed with: ') or err.startswith('line ') or err.startswith('extra line'):
continue
if 'ERROR: AddressSanitizer:' in err:
n = i + 1
bt = []
prefix = ' #'
while not errors[n].startswith(prefix):
n += 1
while errors[n].startswith(prefix):
bt.append(errors[n])
n += 1
location = bt[0]
location = location[location.find('in'):]
n = i - 1
while not 'Executing on host' in errors[n]:
n -= 1
command = errors[n]
while not 'PASS:' in errors[n] and not 'FAIL' in errors[n]:
n -= 1
test_case = errors[n]
report_error(command, test_case, location, '\n'.join([err] + bt), target)
elif 'runtime error' in err and 'conftest.c' not in err:
n = i - 1
while not 'Executing on host' in errors[n]:
n -= 1
command = errors[n]
while not 'PASS:' in errors[n] and not 'FAIL' in errors[n]:
n -= 1
test_case= errors[n]
location = err
location = location[:location.find('runtime error')]
report_error(command, test_case, location, err, target)
errors = [l for l in errors if 'runtime error:' in l or '' in l]
sys.exit(0 if not seen else 1)