Files
UnrealEngine/Engine/Source/ThirdParty/Intel/ISPC/ispc-1.19.0/run_tests.py
2025-05-18 13:04:45 +08:00

1062 lines
41 KiB
Python

#!/usr/bin/env python3
#
# Copyright (c) 2013-2022, Intel Corporation
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are
# met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
#
# * Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
#
# * Neither the name of Intel Corporation nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
# IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
# TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
# PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
# OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
# PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
# LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
# NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
# Supported operating systems
from enum import Enum, unique
@unique
class OS(Enum):
Unknown = 0
Windows = 1
Linux = 2
Mac = 3
FreeBSD = 4
@unique
class Status(Enum):
Success = 0
Compfail = 1
Runfail = 2
Skip = 3
StatusStr = {Status.Success: "PASSED",
Status.Compfail: "FAILED compilation",
Status.Runfail: "FAILED execution",
Status.Skip: "SKIPPED",
}
# The description of host testing system
class Host(object):
def set_os(self, system):
if system == 'Windows' or 'CYGWIN_NT' in system:
self.os = OS.Windows
elif system == 'Darwin':
self.os = OS.Mac
elif system == 'Linux':
self.os = OS.Linux
elif system == 'FreeBSD':
self.os = OS.FreeBSD
else:
self.os = OS.Unknown
# set ispc exe using ISPC_HOME or PATH environment variables
def set_ispc_exe(self):
ispc_exe = ""
ispc_ext = ""
if self.is_windows():
ispc_ext = ".exe"
if "ISPC_HOME" in os.environ:
if os.path.exists(os.environ["ISPC_HOME"] + os.sep + "ispc" + ispc_ext):
ispc_exe = os.environ["ISPC_HOME"] + os.sep + "ispc" + ispc_ext
PATH_dir = os.environ["PATH"].split(os.pathsep)
for counter in PATH_dir:
if ispc_exe == "":
if os.path.exists(counter + os.sep + "ispc" + ispc_ext):
ispc_exe = counter + os.sep + "ispc" + ispc_ext
# checks the required ispc compiler otherwise prints an error message
if ispc_exe == "":
error("ISPC compiler not found.\nAdd path to ispc compiler to your PATH or ISPC_HOME env variable\n", 1)
# use relative path
self.ispc_exe = os.path.relpath(ispc_exe, os.getcwd())
def __init__(self, system):
self.set_os(system)
self.set_ispc_exe()
def is_windows(self):
return self.os == OS.Windows
def is_linux(self):
return self.os == OS.Linux
def is_mac(self):
return self.os == OS.Mac
def is_freebsd(self):
return self.os == OS.FreeBSD
def set_ispc_cmd(self, ispc_flags):
self.ispc_cmd = self.ispc_exe + " " + ispc_flags
# The description of testing target configuration
class TargetConfig(object):
def __init__(self, arch, target, cpu):
if arch == "x86_64" or arch == "x86-64":
self.arch = "x86-64"
else:
self.arch = arch
self.target = target
self.xe = target.find("gen9") != -1 or target.find("xe") != -1
self.set_cpu(cpu)
self.set_target()
def is_xe(self):
return self.xe
def set_cpu(self, cpu):
if cpu is not None:
self.cpu = cpu
# Alias all of acm-* devices to dg2.
if cpu.startswith("acm-"):
self.cpu = "dg2"
else:
self.cpu = "unspec"
# set arch/target
def set_target(self):
if self.target == 'neon':
self.arch = 'aarch64'
# Representation of output which goes to console.
# It consist of stdout & stderr.
class Output(object):
stdout: str
stderr: str
def __init__(self, stdout, stderr):
self.stdout = stdout
self.stderr = stderr
# By default we can get string from instance of this class
# which will return merged stdout with stderr.
def __str__(self):
return self.stdout + self.stderr
# test-running driver for ispc
# utility routine to print an update on the number of tests that have been
# finished. Should be called with the lock held..
def update_progress(fn, total_tests_arg, counter, max_test_length_arg):
counter.value += 1
if options.non_interactive == False:
progress_str = " Done %d / %d [%s]" % (counter.value, total_tests_arg, fn)
# spaces to clear out detrius from previous printing...
spaces_needed = max_test_length_arg - len(fn)
for x in range(spaces_needed):
progress_str += ' '
progress_str += '\r'
sys.stdout.write(progress_str)
sys.stdout.flush()
def run_command(cmd, timeout=600, cwd="."):
if options.verbose:
print_debug("Running: %s\n" % cmd, s, run_tests_log)
# Here's a bit tricky part. To pass a command for execution we should
# break down the line in to arguments. shlex class is designed exactly
# for this purpose, but by default it interprets escape sequences.
# On Windows backslaches are all over the place and they are treates as
# ESC-sequences, so we have to set manually to not interpret them.
lexer = shlex.shlex(cmd, posix=True)
lexer.whitespace_split = True
lexer.escape = ''
arg_list = list(lexer)
# prepare for OSError exceptions raised in the child process (re-raised in the parent)
try:
proc = subprocess.Popen(arg_list, stdin=None, stdout=subprocess.PIPE, stderr=subprocess.PIPE, cwd=cwd)
except:
print_debug("ERROR: The child (%s) raised an exception: %s\n" % (arg_list, sys.exc_info()[1]), s, run_tests_log)
raise
is_timeout = False
# read data from stdout and stderr
try:
out = proc.communicate(timeout=timeout)
except subprocess.TimeoutExpired:
proc.kill()
out = proc.communicate()
is_timeout = True
except:
print_debug("ERROR: The child (%s) raised an exception: %s\n" % (arg_list, sys.exc_info()[1]), s, run_tests_log)
raise
output = Output(out[0].decode("utf-8"), out[1].decode("utf-8"))
return (proc.returncode, output, is_timeout)
# checks whether print ouput is correct
# (whether test and reference outputs are same)
# NOTE: output contains both test and reference lines
def check_print_output(output):
lines = output.splitlines()
if len(lines) == 0 or len(lines) % 2:
return False
else:
return lines[0:len(lines)//2] == lines[len(lines)//2:len(lines)]
# run the commands in cmd_list
def run_cmds(compile_cmds, run_cmd, filename, expect_failure, sig, exe_wd="."):
for cmd in compile_cmds:
(return_code, output, timeout) = run_command(cmd, options.test_time)
compile_failed = (return_code != 0)
if compile_failed:
print_debug("Compilation of test %s failed %s \n" % (filename, "due to TIMEOUT" if timeout else ""), s, run_tests_log)
if output != "":
print_debug("%s" % output, s, run_tests_log)
return Status.Compfail
if not options.save_bin:
(return_code, output, timeout) = run_command(run_cmd, options.test_time, cwd=exe_wd)
if sig < 32:
run_failed = (return_code != 0) or timeout
else:
# check only stdout
output_equality = check_print_output(output.stdout)
if not output_equality:
print_debug("Print outputs check failed\n", s, run_tests_log)
run_failed = (return_code != 0) or not output_equality or timeout
else:
run_failed = 0
surprise = ((expect_failure and not run_failed) or
(not expect_failure and run_failed))
if surprise == True:
print_debug("Test %s %s (return code %d) \n" % \
(filename, "unexpectedly passed" if expect_failure else "failed",
return_code), s, run_tests_log)
if str(output):
print_debug("%s\n" % output, s, run_tests_log)
if surprise == True:
return Status.Runfail
else:
return Status.Success
def add_prefix(path, host, target):
if host.is_windows():
# On Windows we run tests in tmp dir, so the root is one level up.
input_prefix = "..\\"
else:
# For Xe target we run tests in tmp dir since output file has
# the same name for all tests, so the root is one level up
if target.is_xe():
input_prefix = "../"
else:
input_prefix = ""
path = input_prefix + path
path = os.path.abspath(path)
return path
# Return True if test should be skipped,
# return False otherwise.
#
# Default rules are to run always:
# //rule: run on <key>=*
#
# Rules can be overrriden by putting
# comments to test file:
# //rule: run on <key>=<value>
# //rule: skip on <key>=<value>
#
# Currently supported keys are:
# [arch, OS, cpu].
#
# * (asterisk) represent any value.
#
# Rules order is important,
# rule may override all previous rules.
#
# Examples:
#
# 1. Run only on arch xe32 or arch xe64:
# // rule: skip on arch=*
# // rule: run on arch=xe32
# // rule: run on arch=xe64
#
# 2. Run only on Linux OS:
# // rule: skip on OS=*
# // rule: run on OS=Linux
#
def check_if_skip_test(filename, host, target):
# by default we're not skipping test
skip = False
if host.is_windows():
oss = "windows"
elif host.is_linux():
oss = "linux"
elif host.is_mac():
oss = "mac"
elif host.is_freebsd():
oss = "freebsd"
else:
oss = "unknown"
target_cpu = target.cpu
rule_values = {"arch": target.arch, "OS": oss, "cpu": target_cpu}
test_file_path = add_prefix(filename, host, target)
with open(test_file_path) as test_file:
# scan test file line by line
while True:
test_line = test_file.readline()
# EOF
if not test_line:
break;
rule = re.search('// *rule: (run|skip) on (arch|OS|cpu)=(.*)', test_line)
# no match for this line -> look at next line
if rule == None:
continue
rule_action = rule.group(1)
rule_key = rule.group(2)
rule_value = rule.group(3)
for key in rule_values.keys():
if rule_key == key:
if rule_value == rule_values[key] or rule_value == "*":
if rule_action == "run":
skip = False
elif rule_action == "skip":
skip = True
return skip
def run_test(testname, host, target):
# testname is a path to the test from the root of ispc dir
# filename is a path to the test from the current dir
# ispc_exe_rel is a relative path to ispc
filename = add_prefix(testname, host, target)
# Debug check is now supported only for xe
if options.debug_check and target.is_xe():
ispc_exe_rel = add_prefix(host.ispc_cmd + " -g", host, target)
else:
ispc_exe_rel = add_prefix(host.ispc_cmd, host, target)
# is this a test to make sure an error is issued?
want_error = (filename.find("tests_errors") != -1)
if want_error == True:
ispc_cmd = ispc_exe_rel + " --werror --nowrap %s --arch=%s --target=%s" % \
(filename, target.arch, target.target)
(return_code, output, timeout) = run_command(ispc_cmd, options.test_time)
got_error = (return_code != 0) or timeout
# figure out the error message we're expecting
file = open(filename, 'r')
firstline = file.readline()
firstline = firstline.replace("//", "")
firstline = firstline.lstrip()
firstline = firstline.rstrip()
file.close()
if re.search(firstline, output.__str__()) == None:
print_debug("Didn't see expected error message %s from test %s.\nActual output:\n%s\n" % \
(firstline, testname, output), s, run_tests_log)
return Status.Compfail
elif got_error == False:
print_debug("Unexpectedly no errors issued from test %s\n" % testname, s, run_tests_log)
return Status.Compfail
else:
return Status.Success
else:
# do we expect this test to fail?
should_fail = (testname.find("failing_") != -1)
# We need to figure out the signature of the test
# function that this test has.
sig2def = { "f_v(" : 0, "f_f(" : 1, "f_fu(" : 2, "f_fi(" : 3,
"f_du(" : 4, "f_duf(" : 5, "f_di(" : 6, "f_sz" : 7,
"f_t(" : 8, "print_uf(" : 32, "print_f(" : 33,
"print_fuf(" : 34, "print_no(" : 35 }
file = open(filename, 'r')
match = -1
for line in file:
# look for lines with 'export'...
if line.find("task") == -1 and line.find("export") == -1:
continue
# one of them should have a function with one of the
# declarations in sig2def
for pattern, ident in list(sig2def.items()):
if line.find(pattern) != -1:
match = ident
break
file.close()
# Figure out target width
width = -1
target_match = re.match('.*-(i[0-9]*)?x([0-9]*)', options.target)
# If target does not contain width in a standard way:
if target_match == None:
error("Unable to detect the target width for target %s\nOnly canonical form of the target names is supported, deprecated forms are not supported" % options.target, 0)
return Status.Compfail
width = int(target_match.group(2))
if match == -1:
error("Unable to find function signature in test %s\n" % testname, 0)
return Status.Compfail
else:
xe_target = options.target
if host.is_windows():
if target.is_xe():
obj_name = "test_xe.bin" if options.ispc_output == "ze" else "test_xe.spv"
else:
obj_name = "%s.obj" % os.path.basename(filename)
if target.arch == "wasm32":
exe_name = "%s.js" % os.path.realpath(filename)
else:
exe_name = "%s.exe" % os.path.basename(filename)
cc_cmd = "%s /I. /Zi /nologo /DTEST_SIG=%d /DTEST_WIDTH=%d %s %s /Fe%s" % \
(options.compiler_exe, match, width, add_prefix("test_static.cpp", host, target), obj_name, exe_name)
if target.is_xe():
cc_cmd = "%s /I. /I%s\\include /nologo /DTEST_SIG=%d /DTEST_WIDTH=%d %s %s /Fe%s ze_loader.lib /link /LIBPATH:%s\\lib" % \
(options.compiler_exe, options.l0loader, match, width, " /DTEST_ZEBIN" if options.ispc_output == "ze" else " /DTEST_SPV", \
add_prefix("test_static_l0.cpp", host, target), exe_name, options.l0loader)
if should_fail:
cc_cmd += " /DEXPECT_FAILURE"
else:
if target.is_xe():
obj_name = "test_xe.bin" if options.ispc_output == "ze" else "test_xe.spv"
else:
obj_name = "%s.o" % testname
if target.arch == "wasm32":
exe_name = "%s.js" % os.path.realpath(testname)
else:
exe_name = "%s.run" % testname
if target.arch == 'arm':
gcc_arch = '--with-fpu=hardfp -marm -mfpu=neon -mfloat-abi=hard'
elif target.arch == 'x86' or target.arch == "wasm32" or target.arch == 'xe32':
gcc_arch = '-m32'
elif target.arch == 'aarch64':
gcc_arch = '-march=armv8-a'
else:
gcc_arch = '-m64'
cc_cmd = "%s -O2 -I. %s test_static.cpp -DTEST_SIG=%d -DTEST_WIDTH=%d %s -o %s" % \
(options.compiler_exe, gcc_arch, match, width, obj_name, exe_name)
# Produce position independent code for both c++ and ispc compilations.
# The motivation for this is that Clang 15 changed default
# from "-mrelocation-model static" to "-mrelocation-model pic", so
# we enable PIC compilation to have it consistently regardless compiler version.
cc_cmd += ' -fPIE'
if should_fail:
cc_cmd += " -DEXPECT_FAILURE"
if target.is_xe():
exe_name = "%s.run" % os.path.basename(testname)
cc_cmd = "%s -O0 -I. -I %s/include -lze_loader -L %s/lib \
%s %s -DTEST_SIG=%d -DTEST_WIDTH=%d -o %s" % \
(options.compiler_exe, options.l0loader, options.l0loader, gcc_arch, add_prefix("test_static_l0.cpp", host, target),
match, width, exe_name)
exe_name = "./" + exe_name
cc_cmd += " -DTEST_ZEBIN" if options.ispc_output == "ze" else " -DTEST_SPV"
ispc_cmd = ispc_exe_rel + " --pic --woff %s -o %s --arch=%s --target=%s -DTEST_SIG=%d" % \
(filename, obj_name, options.arch, xe_target if target.is_xe() else options.target, match)
if target.is_xe():
ispc_cmd += " --emit-zebin" if options.ispc_output == "ze" else " --emit-spirv"
ispc_cmd += " -DISPC_GPU"
if options.device != None:
ispc_cmd += " --device="+ options.device
if options.opt == 'O0':
ispc_cmd += " -O0"
elif options.opt == 'O1':
ispc_cmd += " -O1"
elif options.opt == 'O2':
ispc_cmd += " -O2"
exe_wd = "."
if target.arch == "wasm32":
cc_cmd += " -D__WASM__"
options.wrapexe = "v8 --experimental-wasm-simd"
exe_wd = os.path.realpath("./tests")
# compile the ispc code, make the executable, and run it...
ispc_cmd += " -h " + filename + ".h"
cc_cmd += " -DTEST_HEADER=\"<" + filename + ".h>\""
status = run_cmds([ispc_cmd, cc_cmd], options.wrapexe + " " + exe_name,
testname, should_fail, match, exe_wd=exe_wd)
# clean up after running the test
try:
os.unlink(filename + ".h")
if not options.save_bin:
if status != Status.Runfail:
os.unlink(exe_name)
if host.is_windows():
basename = os.path.basename(filename)
os.unlink("%s.pdb" % basename)
os.unlink("%s.ilk" % basename)
os.unlink(obj_name)
os.unlink(filename + ".wasm")
os.unlink(filename + ".js")
os.unlink(filename + ".html")
except:
None
return status
# pull tests to run from the given queue and run them. Multiple copies of
# this function will be running in parallel across all of the CPU cores of
# the system.
def run_tasks_from_queue(queue, queue_ret, total_tests_arg, max_test_length_arg, counter, mutex, glob_var):
# This is needed on windows because windows doesn't copy globals from parent process while multiprocessing
host = glob_var[0]
global options
options = glob_var[1]
global s
s = glob_var[2]
target = glob_var[3]
global run_tests_log
run_tests_log = glob_var[4]
if host.is_windows() or target.is_xe():
tmpdir = "tmp%d" % os.getpid()
while os.access(tmpdir, os.F_OK):
tmpdir = "%sx" % tmpdir
os.mkdir(tmpdir)
os.chdir(tmpdir)
else:
olddir = ""
for filename in iter(queue.get, 'STOP'):
status = Status.Skip
if not check_if_skip_test(filename, host, target):
try:
status = run_test(filename, host, target)
except:
# This is in case the child has unexpectedly died or some other exception happened
# Count it as runfail and continue with next test.
exc_type, exc_value, exc_traceback = sys.exc_info()
traceback.print_tb(exc_traceback, file=sys.stderr)
print_debug("ERROR: run_test function raised an exception: %s\n" % (sys.exc_info()[1]), s, run_tests_log)
status = Status.Runfail
queue_ret.put((filename, status))
with mutex:
update_progress(filename, total_tests_arg, counter, max_test_length_arg)
# Task done for the test.
queue.task_done()
if host.is_windows():
try:
os.remove("test_static.obj")
# vc*.pdb trick is in anticipaton of new versions of VS.
vcpdb = glob.glob("vc*.pdb")[0]
os.remove(vcpdb)
os.chdir("..")
# This will fail if there were failing tests or
# Windows is in bad mood.
os.rmdir(tmpdir)
except:
None
else:
if target.is_xe():
try:
os.chdir("..")
os.rmdir(tmpdir)
except:
None
# Task done for terminating `STOP`.
queue.task_done()
def sigint(signum, frame):
for t in task_threads:
t.terminate()
sys.exit(1)
def file_check(results, host, target):
global exit_code
exit_code = 0
compfails = [fname for fname, status in results if status == Status.Compfail]
runfails = [fname for fname, status in results if status == Status.Runfail]
errors = len(compfails) + len(runfails)
new_compfails = []
new_runfails = []
new_passes_compfails = []
new_passes_runfails = []
# Open fail db file
f = open(options.fail_db, 'r')
f_lines = f.readlines()
f.close()
# Detect OS
if platform.system() == 'Windows' or 'CYGWIN_NT' in platform.system():
OS = "Windows"
else:
if platform.system() == 'Darwin':
OS = "Mac"
else:
OS = "Linux"
# Detect opt_set
opt = options.opt
# Detect testing output
ispc_output = options.ispc_output
# Detect LLVM version
temp1 = common.take_lines(host.ispc_exe + " --version", "first")
temp2 = re.search('LLVM [0-9]*\.[0-9]*', temp1)
if temp2 != None:
llvm_version = temp2.group()
else:
llvm_version = "unknown LLVM"
# Detect compiler version
if OS != "Windows":
temp1 = common.take_lines(options.compiler_exe + " --version", "first")
temp2 = re.search("[0-9]*\.[0-9]*\.[0-9]", temp1)
if temp2 == None:
temp3 = re.search("[0-9]*\.[0-9]*", temp1)
else:
temp3 = re.search("[0-9]*\.[0-9]*", temp2.group())
compiler_version = options.compiler_exe + temp3.group()
else:
compiler_version = "cl"
cpu = target.cpu
possible_compilers=set()
for x in f_lines:
if x.startswith("."):
possible_compilers.add(x.split(' ')[-3])
#if not compiler_version in possible_compilers:
# error("\n**********\nWe don't have history of fails for compiler " +
# compiler_version +
# "\nAll fails will be new!!!\n**********", 2)
new_line = " "+target.arch.rjust(6)+" "+target.target.rjust(14)+" "+cpu+" "+OS.rjust(7)+" "+llvm_version+" "+compiler_version.rjust(10)+" "+opt+ " " + ispc_output + " *\n"
new_compfails = compfails[:]
new_runfails = runfails[:]
new_f_lines = f_lines[:]
for j in range(0, len(f_lines)):
if (((" "+target.arch+" ") in f_lines[j]) and
((" "+target.target+" ") in f_lines[j]) and
((" "+cpu+" ") in f_lines[j]) and
((" "+OS+" ") in f_lines[j]) and
((" "+llvm_version+" ") in f_lines[j]) and
((" "+compiler_version+" ") in f_lines[j]) and
((" "+opt+" ") in f_lines[j]) and
((" "+ispc_output+" ") in f_lines[j])):
if (" compfail " in f_lines[j]):
f = 0
for i in range(0, len(compfails)):
if compfails[i] in f_lines[j]:
new_compfails.remove(compfails[i])
else:
f = f + 1
if f == len(compfails):
temp3 = f_lines[j].split(" ")
new_passes_compfails.append(temp3[0])
if options.update == "FP":
new_f_lines.remove(f_lines[j])
if (" runfail " in f_lines[j]):
f = 0
for i in range(0, len(runfails)):
if runfails[i] in f_lines[j]:
new_runfails.remove(runfails[i])
else:
f = f + 1
if f == len(runfails):
temp3 = f_lines[j].split(" ")
new_passes_runfails.append(temp3[0])
if options.update == "FP":
new_f_lines.remove(f_lines[j])
if len(new_runfails) != 0:
new_runfails.sort()
print_debug("NEW RUNFAILS:\n", s, run_tests_log)
exit_code = 1
for i in range (0,len(new_runfails)):
new_f_lines.append(new_runfails[i] + " runfail " + new_line)
print_debug("\t" + new_runfails[i] + "\n", s, run_tests_log)
if len(new_compfails) != 0:
new_compfails.sort()
print_debug("NEW COMPFAILS:\n", s, run_tests_log)
exit_code = 1
for i in range (0,len(new_compfails)):
new_f_lines.append(new_compfails[i] + " compfail " + new_line)
print_debug("\t" + new_compfails[i] + "\n", s, run_tests_log)
if len(new_runfails) == 0 and len(new_compfails) == 0:
print_debug("No new fails\n", s, run_tests_log)
if len(new_passes_runfails) != 0:
new_passes_runfails.sort()
print_debug("NEW PASSES after RUNFAILS:\n", s, run_tests_log)
for i in range (0,len(new_passes_runfails)):
print_debug("\t" + new_passes_runfails[i] + "\n", s, run_tests_log)
if len(new_passes_compfails) != 0:
new_passes_compfails.sort()
print_debug("NEW PASSES after COMPFAILS:\n", s, run_tests_log)
for i in range (0,len(new_passes_compfails)):
print_debug("\t" + new_passes_compfails[i] + "\n", s, run_tests_log)
if options.update != "":
output = open(options.fail_db, 'w')
output.writelines(new_f_lines)
output.close()
return [new_runfails, new_compfails, new_passes_runfails, new_passes_compfails, new_line, errors]
# TODO: This function is out of date, it needs update and test coverage.
def verify():
# Open fail db file
f = open(options.fail_db, 'r')
f_lines = f.readlines()
f.close()
check = [["g++", "clang++", "cl"],["-O0", "-O2"],["x86","x86-64"],
["Linux","Windows","Mac"],["LLVM 3.2","LLVM 3.3","LLVM 3.4","LLVM 3.5","LLVM 3.6","LLVM trunk"],
["sse2-i32x4", "sse2-i32x8",
"sse4-i32x4", "sse4-i32x8", "sse4-i16x8", "sse4-i8x16",
"avx1-i32x4", "avx1-i32x8", "avx1-i32x16", "avx1-i64x4",
"avx2-i32x4", "avx2-i32x8", "avx2-i32x16", "avx2-i64x4",
"avx512knl-x16", "avx512skx-x16", "avx512skx-x8", "avx512skx-x4", "avx512skx-x64", "avx512skx-x32"]]
for i in range (0,len(f_lines)):
if f_lines[i][0] == "%":
continue
for j in range(0,len(check)):
temp = 0
for t in range(0,len(check[j])):
if " " + check[j][t] + " " in f_lines[i]:
temp = temp + 1
if temp != 1:
print_debug("error in line " + str(i) + "\n", False, run_tests_log)
break
# populate ex_state test table and run info with testing results
def populate_ex_state(options, target, total_tests, test_result):
# Detect opt_set
opt = options.opt
try:
common.ex_state.add_to_rinf_testall(total_tests)
for fname, status in test_result:
# one-hot encoding
succ = status == Status.Success
runf = status == Status.Runfail
comp = status == Status.Compfail
skip = status == Status.Skip
# We do not add skipped tests to test table as we do not know the test result
if status != Status.Skip:
common.ex_state.add_to_tt(fname, target.arch, opt, target.target, runf, comp)
common.ex_state.add_to_rinf(target.arch, opt, target.target, succ, runf, comp, skip)
except:
print_debug("Exception in ex_state. Skipping...\n", s, run_tests_log)
# set compiler exe depending on the OS
def set_compiler_exe(host, options):
if options.compiler_exe == None:
if options.arch == "wasm32":
options.compiler_exe = "emcc"
elif host.is_windows():
options.compiler_exe = "cl.exe"
else:
options.compiler_exe = "clang++"
# checks the required compiler otherwise prints an error message
check_compiler_exists(options.compiler_exe)
# set ISPC output format
def set_ispc_output(target, options):
if options.ispc_output == None:
if target.is_xe():
options.ispc_output = "spv"
else:
options.ispc_output = "obj"
else:
if not target.is_xe() and not options.ispc_output=="obj" or target.is_xe() and options.ispc_output=="obj":
error("unsupported test output \"%s\" is specified for target: %s \n" % (options.ispc_output, target.target), 1)
# returns the list of test files
def get_test_files(host, args):
if len(args) == 0:
ispc_root = "."
files = glob.glob(ispc_root + os.sep + "tests" + os.sep + "*ispc") + \
glob.glob(ispc_root + os.sep + "tests_errors" + os.sep + "*ispc")
else:
if host.is_windows():
argfiles = [ ]
for f in args:
# we have to glob ourselves if this is being run under a DOS
# shell, as it passes wildcard as is.
argfiles += glob.glob(f)
else:
argfiles = args
files = [ ]
for f in argfiles:
if os.path.splitext(f.lower())[1] != ".ispc":
error("Ignoring file %s, which doesn't have an .ispc extension.\n" % f, 2)
else:
files += [ f ]
return files
# checks the required compiler in PATH otherwise prints an error message
def check_compiler_exists(compiler_exe):
for path in os.environ["PATH"].split(os.pathsep):
if os.path.exists(path + os.sep + compiler_exe):
return
error("missing the required compiler: %s \n" % compiler_exe, 1)
def print_result(status, results, s, run_tests_log, csv):
title = StatusStr[status]
file_list = [fname for fname, fstatus in results if status == fstatus]
total_tests = len(results)
print_debug("%d / %d tests %s\n" % (len(file_list), total_tests, title), s, run_tests_log)
if status == Status.Success:
return
for f in sorted(file_list):
print_debug("\t%s\n" % f, s, run_tests_log)
print_debug("%s;%s\n" % (f, title), csv, csv) # dump result to csv if filename is non-empty
def run_tests(options1, args, print_version):
global exit_code
global options
options = options1
global s
s = options.silent
# prepare run_tests_log and fail_db file
if len(args) != 0 and not os.path.exists(options.fail_db):
print("Fail database file not found!")
exit_code = 1
return 0
global run_tests_log
if options.in_file:
run_tests_log = os.getcwd() + os.sep + options.in_file
if print_version == 1:
common.remove_if_exists(run_tests_log)
else:
run_tests_log = ""
if options.verify:
verify()
return 0
# disable fancy error/warning printing with ANSI colors, so grepping for error
# messages doesn't get confused
os.environ["TERM"] = "dumb"
host = Host(platform.system())
host.set_ispc_cmd(options.ispc_flags)
target = TargetConfig(options.arch, options.target, options.device)
if options.debug_check and (not target.is_xe() or not host.is_linux()):
print("--debug_check is supported only for xe target and only on Linux OS")
exit_code = 1
return 0
print_debug("Testing ISPC compiler: " + host.ispc_exe + "\n", s, run_tests_log)
print_debug("Testing ISPC target: %s\n" % options.target, s, run_tests_log)
print_debug("Testing ISPC arch: %s\n" % options.arch, s, run_tests_log)
set_compiler_exe(host, options)
set_ispc_output(target, options)
# print compilers versions
if print_version > 0:
common.print_version(host.ispc_exe, "", options.compiler_exe, False, run_tests_log, host.is_windows())
# if no specific test files are specified, run all of the tests in tests/
# and tests_errors/
files = get_test_files(host, args)
# max_test_length is used to issue exact number of whitespace characters when
# updating status. Otherwise update causes new lines standard 80 char terminal
# on both Linux and Windows.
max_test_length = max([len(f) for f in files])
# randomly shuffle the tests if asked to do so
if (options.random):
random.seed()
random.shuffle(files)
# counter
total_tests = len(files)
results = []
nthreads = min([multiprocessing.cpu_count(), options.num_jobs, len(files)])
print_debug("Running %d jobs in parallel. Running %d tests.\n" % (nthreads, total_tests), s, run_tests_log)
# put each of the test filenames into a queue
test_queue = multiprocessing.JoinableQueue()
for fn in files:
test_queue.put(fn)
for x in range(nthreads):
test_queue.put('STOP')
# qret is a queue for returned data
qret = multiprocessing.Queue()
# need to catch sigint so that we can terminate all of the tasks if
# we're interrupted
signal.signal(signal.SIGINT, sigint)
finished_tests_counter = multiprocessing.Value('i') # 'i' is typecode of ctypes.c_int
# lock to protect counter increment and stdout printing
lock = multiprocessing.Lock()
start_time = time.time()
# launch jobs to run tests
glob_var = [host, options, s, target, run_tests_log]
# task_threads has to be global as it is used in sigint handler
global task_threads
task_threads = [0] * nthreads
for x in range(nthreads):
task_threads[x] = multiprocessing.Process(target=run_tasks_from_queue, args=(test_queue, qret, total_tests,
max_test_length, finished_tests_counter, lock, glob_var))
task_threads[x].start()
# wait for them all to finish and rid the queue of STOPs
# join() here just waits for synchronization
test_queue.join()
if options.non_interactive == False:
print_debug("\n", s, run_tests_log)
temp_time = (time.time() - start_time)
elapsed_time = time.strftime('%Hh%Mm%Ssec.', time.gmtime(temp_time))
while not qret.empty():
results.append(qret.get())
# populate ex_state test table and run info with testing results
populate_ex_state(options, target, total_tests, results)
run_succeed_files = [fname for fname, fstatus in results if fstatus == Status.Success]
skip_files = [fname for fname, fstatus in results if fstatus == Status.Skip]
total_tests_executed = total_tests-len(skip_files)
if options.non_interactive:
print_debug(" Done %d / %d\n" % (finished_tests_counter.value, total_tests), s, run_tests_log)
print_debug("\nExecuted %d / %d (%d skipped)\n\n" % (total_tests_executed, total_tests, len(skip_files)), s, run_tests_log)
# Pass rate
if (total_tests_executed) > 0:
pass_rate = (len(run_succeed_files)/total_tests_executed)*100
else:
pass_rate = -1
print_debug("PASSRATE (%d/%d) = %d%% \n\n" % (len(run_succeed_files), total_tests_executed, pass_rate), s, run_tests_log)
for status in Status:
print_result(status, results, s, run_tests_log, options.csv)
fails = [status != Status.Compfail and status != Status.Runfail for _, status in results]
if sum(fails) == 0:
print_debug("No fails\n", s, run_tests_log)
if len(args) == 0:
R = file_check(results, host, target)
else:
error("don't check new fails for incomplete suite of tests", 2)
R = 0
if options.time:
print_debug("Elapsed time: " + elapsed_time + "\n", s, run_tests_log)
return [R, elapsed_time]
from optparse import OptionParser
import multiprocessing
import os
import sys
import glob
import re
import signal
import random
import threading
import subprocess
import shlex
import platform
import tempfile
import os.path
import time
# our functions
import common
import traceback
print_debug = common.print_debug
error = common.error
exit_code = 0
# Use different default targets on different architectures.
default_target = "sse4-i32x4"
default_arch = "x86-64"
if platform.machine() == "arm":
default_target = "neon-i32x4"
default_arch = "arm"
elif platform.machine() == "aarch64":
default_target = "neon-i32x4"
default_arch = "aarch64"
elif platform.machine() == "arm64":
default_target = "neon-i32x4"
default_arch = "aarch64"
elif "86" in platform.machine() or platform.machine() == "AMD64":
# Some variant of x86: x86_64, i386, i486, i586, i686
# Windows reports platform as AMD64
pass
else:
print_debug("WARNING: host machine was not recognized - " + str(platform.machine()), False, "")
if __name__ == "__main__":
parser = OptionParser()
parser.add_option("-r", "--random-shuffle", dest="random", help="Randomly order tests",
default=False, action="store_true")
parser.add_option("-f", "--ispc-flags", dest="ispc_flags", help="Additional flags for ispc (-g, -O1, ...)",
default="")
parser.add_option('-t', '--target', dest='target',
help=('Set compilation target. For example: sse4-i32x4, avx2-i32x8, avx512skx-x16, etc.'), default=default_target)
parser.add_option('-a', '--arch', dest='arch',
help='Set architecture (arm, aarch64, x86, x86-64, xe32, xe64)', default=default_arch)
parser.add_option("-c", "--compiler", dest="compiler_exe", help="C/C++ compiler binary to use to run tests",
default=None)
parser.add_option('-o', '--opt', dest='opt', choices=['', 'O0', 'O1', 'O2'], help='Set optimization level passed to the compiler (O0, O1, O2).',
default='O2')
parser.add_option('-j', '--jobs', dest='num_jobs', help='Maximum number of jobs to run in parallel',
default="1024", type="int")
parser.add_option('-v', '--verbose', dest='verbose', help='Enable verbose output',
default=False, action="store_true")
parser.add_option('--wrap-exe', dest='wrapexe',
help='Executable to wrap test runs with (e.g. "valgrind" or "sde -knl -- ")',
default="")
parser.add_option('--time', dest='time', help='Enable time output',
default=False, action="store_true")
parser.add_option('--non-interactive', dest='non_interactive', help='Disable interactive status updates',
default=False, action="store_true")
parser.add_option('-u', "--update-errors", dest='update', help='Update file with fails (F of FP)', default="")
parser.add_option('-s', "--silent", dest='silent', help='enable silent mode without any output', default=False,
action = "store_true")
parser.add_option("--file", dest='in_file', help='file to save run_tests output', default="")
parser.add_option("--l0loader", dest='l0loader', help='Path to L0 loader', default="")
parser.add_option("--device", dest='device', help='Specify target ISPC device. For example: core2, skx, cortex-a9, skl, tgllp, acm-g11, etc.', default=None)
parser.add_option("--ispc_output", dest='ispc_output', choices=['obj', 'spv', 'ze'], help='Specify ISPC output', default=None)
parser.add_option("--fail_db", dest='fail_db', help='File to use as a fail database', default='fail_db.txt', type=str)
parser.add_option("--debug_check", dest='debug_check', help='Run tests in debug mode with validating debug info', default=False, action="store_true")
parser.add_option("--verify", dest='verify', help='verify the fail database file', default=False, action="store_true")
parser.add_option("--save-bin", dest='save_bin', help='compile and create bin, but don\'t execute it',
default=False, action="store_true")
parser.add_option('--csv', dest="csv", help="file to save testing results", default="")
parser.add_option('--test_time', dest="test_time", help="time needed for each test", default=600, type="int", action="store")
(options, args) = parser.parse_args()
L = run_tests(options, args, 1)
exit(exit_code)