mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-22 02:33:06 +01:00
Add a utility to reduce GlobalISel tests
Summary: This patch adds a script to reduce GlobalISel failures using bugpoint. Reviewers: bogner Reviewed By: bogner Subscribers: MatzeB, qcolombet, rovka, kristof.beyls, llvm-commits Differential Revision: https://reviews.llvm.org/D41876 llvm-svn: 323248
This commit is contained in:
parent
e68727c0a3
commit
2b414f4473
146
utils/bugpoint_gisel_reducer.py
Executable file
146
utils/bugpoint_gisel_reducer.py
Executable file
@ -0,0 +1,146 @@
|
||||
#!/usr/bin/env python
|
||||
|
||||
"""Reduces GlobalISel failures.
|
||||
|
||||
This script is a utility to reduce tests that GlobalISel
|
||||
fails to compile.
|
||||
|
||||
It runs llc to get the error message using a regex and creates
|
||||
a custom command to check that specific error. Then, it runs bugpoint
|
||||
with the custom command.
|
||||
|
||||
"""
|
||||
from __future__ import print_function
|
||||
import argparse
|
||||
import re
|
||||
import subprocess
|
||||
import sys
|
||||
import tempfile
|
||||
import os
|
||||
|
||||
|
||||
def log(msg):
|
||||
print(msg)
|
||||
|
||||
|
||||
def hr():
|
||||
log('-' * 50)
|
||||
|
||||
|
||||
def log_err(msg):
|
||||
print('ERROR: {}'.format(msg), file=sys.stderr)
|
||||
|
||||
|
||||
def check_path(path):
|
||||
if not os.path.exists(path):
|
||||
log_err('{} does not exist.'.format(path))
|
||||
raise
|
||||
return path
|
||||
|
||||
|
||||
def check_bin(build_dir, bin_name):
|
||||
file_name = '{}/bin/{}'.format(build_dir, bin_name)
|
||||
return check_path(file_name)
|
||||
|
||||
|
||||
def run_llc(llc, irfile):
|
||||
pr = subprocess.Popen([llc,
|
||||
'-o',
|
||||
'-',
|
||||
'-global-isel',
|
||||
'-pass-remarks-missed=gisel',
|
||||
irfile],
|
||||
stdout=subprocess.PIPE,
|
||||
stderr=subprocess.PIPE)
|
||||
out, err = pr.communicate()
|
||||
res = pr.wait()
|
||||
if res == 0:
|
||||
return 0
|
||||
re_err = re.compile(
|
||||
r'LLVM ERROR: ([a-z\s]+):.*(G_INTRINSIC[_A-Z]* <intrinsic:@[a-zA-Z0-9\.]+>|G_[A-Z_]+)')
|
||||
match = re_err.match(err)
|
||||
if not match:
|
||||
return 0
|
||||
else:
|
||||
return [match.group(1), match.group(2)]
|
||||
|
||||
|
||||
def run_bugpoint(bugpoint_bin, llc_bin, opt_bin, tmp, ir_file):
|
||||
compileCmd = '-compile-command={} -c {} {}'.format(
|
||||
os.path.realpath(__file__), llc_bin, tmp)
|
||||
pr = subprocess.Popen([bugpoint_bin,
|
||||
'-compile-custom',
|
||||
compileCmd,
|
||||
'-opt-command={}'.format(opt_bin),
|
||||
ir_file])
|
||||
res = pr.wait()
|
||||
if res != 0:
|
||||
log_err("Unable to reduce the test.")
|
||||
raise
|
||||
|
||||
|
||||
def run_bugpoint_check():
|
||||
path_to_llc = sys.argv[2]
|
||||
path_to_err = sys.argv[3]
|
||||
path_to_ir = sys.argv[4]
|
||||
with open(path_to_err, 'r') as f:
|
||||
err = f.read()
|
||||
res = run_llc(path_to_llc, path_to_ir)
|
||||
if res == 0:
|
||||
return 0
|
||||
log('GlobalISed failed, {}: {}'.format(res[0], res[1]))
|
||||
if res != err.split(';'):
|
||||
return 0
|
||||
else:
|
||||
return 1
|
||||
|
||||
|
||||
def main():
|
||||
# Check if this is called by bugpoint.
|
||||
if len(sys.argv) == 5 and sys.argv[1] == '-c':
|
||||
sys.exit(run_bugpoint_check())
|
||||
|
||||
# Parse arguments.
|
||||
parser = argparse.ArgumentParser(
|
||||
description=__doc__, formatter_class=argparse.RawTextHelpFormatter)
|
||||
parser.add_argument('BuildDir', help="Path to LLVM build directory")
|
||||
parser.add_argument('IRFile', help="Path to the input IR file")
|
||||
args = parser.parse_args()
|
||||
|
||||
# Check if the binaries exist.
|
||||
build_dir = check_path(args.BuildDir)
|
||||
ir_file = check_path(args.IRFile)
|
||||
llc_bin = check_bin(build_dir, 'llc')
|
||||
opt_bin = check_bin(build_dir, 'opt')
|
||||
bugpoint_bin = check_bin(build_dir, 'bugpoint')
|
||||
|
||||
# Run llc to see if GlobalISel fails.
|
||||
log('Running llc...')
|
||||
res = run_llc(llc_bin, ir_file)
|
||||
if res == 0:
|
||||
log_err("Expected failure")
|
||||
raise
|
||||
hr()
|
||||
log('GlobalISel failed, {}: {}.'.format(res[0], res[1]))
|
||||
tmp = tempfile.NamedTemporaryFile()
|
||||
log('Writing error to {} for bugpoint.'.format(tmp.name))
|
||||
tmp.write(';'.join(res))
|
||||
tmp.flush()
|
||||
hr()
|
||||
|
||||
# Run bugpoint.
|
||||
log('Running bugpoint...')
|
||||
run_bugpoint(bugpoint_bin, llc_bin, opt_bin, tmp.name, ir_file)
|
||||
hr()
|
||||
log('Done!')
|
||||
hr()
|
||||
output_file = 'bugpoint-reduced-simplified.bc'
|
||||
log('Run llvm-dis to disassemble the output:')
|
||||
log('$ {}/bin/llvm-dis -o - {}'.format(build_dir, output_file))
|
||||
log('Run llc to reproduce the problem:')
|
||||
log('$ {}/bin/llc -o - -global-isel '
|
||||
'-pass-remarks-missed=gisel {}'.format(build_dir, output_file))
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
Loading…
Reference in New Issue
Block a user