147 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
			
		
		
	
	
			147 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
| #!/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()
 |