#!/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()