mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2024-11-23 13:50:11 +00:00
b71edfaa4e
This is the first commit in a series that will reformat all the python files in the LLVM repository. Reformatting is done with `black`. See more information here: https://discourse.llvm.org/t/rfc-document-and-standardize-python-code-style Reviewed By: jhenderson, JDevlieghere, MatzeB Differential Revision: https://reviews.llvm.org/D150545
153 lines
3.8 KiB
Python
Executable File
153 lines
3.8 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()
|