2012-12-15 02:27:40 -08:00
|
|
|
#!/usr/bin/env python
|
2012-11-09 10:43:11 +01:00
|
|
|
# Automated script to run the pspautotests test suite in PPSSPP.
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import io
|
|
|
|
import os
|
|
|
|
import subprocess
|
2012-11-18 22:35:08 +01:00
|
|
|
import threading
|
2012-11-09 10:43:11 +01:00
|
|
|
|
|
|
|
|
2013-01-04 16:45:31 +01:00
|
|
|
PPSSPP_EXECUTABLES = [ "Windows\\Release\\PPSSPPHeadless.exe", "build/PPSSPPHeadless" ]
|
2012-11-09 10:43:11 +01:00
|
|
|
PPSSPP_EXE = None
|
|
|
|
TEST_ROOT = "pspautotests/tests/"
|
2012-11-12 03:12:01 +01:00
|
|
|
teamcity_mode = False
|
2012-11-18 22:35:08 +01:00
|
|
|
TIMEOUT = 5
|
|
|
|
|
|
|
|
class Command(object):
|
|
|
|
def __init__(self, cmd):
|
|
|
|
self.cmd = cmd
|
|
|
|
self.process = None
|
|
|
|
self.output = None
|
|
|
|
self.timeout = False
|
|
|
|
|
|
|
|
def run(self, timeout):
|
|
|
|
def target():
|
|
|
|
self.process = subprocess.Popen(self.cmd, stdin=subprocess.PIPE, stdout=subprocess.PIPE)
|
|
|
|
self.output, _ = self.process.communicate()
|
2012-12-15 02:27:40 -08:00
|
|
|
self.output = self.output.decode("utf-8")
|
2012-11-18 22:35:08 +01:00
|
|
|
|
|
|
|
thread = threading.Thread(target=target)
|
|
|
|
thread.start()
|
|
|
|
|
|
|
|
thread.join(timeout)
|
|
|
|
if thread.is_alive():
|
|
|
|
self.timeout = True
|
|
|
|
self.process.terminate()
|
|
|
|
thread.join()
|
2012-11-09 10:43:11 +01:00
|
|
|
|
|
|
|
# Test names are the C files without the .c extension.
|
|
|
|
# These have worked and should keep working always - regression tests.
|
|
|
|
tests_good = [
|
2012-11-29 14:40:11 +01:00
|
|
|
"cpu/cpu_alu/cpu_alu",
|
2012-11-14 00:47:00 +01:00
|
|
|
"cpu/vfpu/base/vfpu",
|
2012-11-11 18:44:20 +01:00
|
|
|
"cpu/vfpu/convert/vfpu_convert",
|
|
|
|
"cpu/vfpu/prefixes/vfpu_prefixes",
|
|
|
|
"cpu/vfpu/colors/vfpu_colors",
|
2012-11-09 10:43:11 +01:00
|
|
|
"cpu/icache/icache",
|
|
|
|
"cpu/lsu/lsu",
|
2012-11-09 11:20:39 +01:00
|
|
|
"cpu/fpu/fpu",
|
2012-11-09 11:33:22 +01:00
|
|
|
|
2012-12-02 17:19:57 -08:00
|
|
|
"ctrl/ctrl",
|
2012-12-22 21:28:01 -08:00
|
|
|
"ctrl/idle/idle",
|
2012-12-02 17:19:57 -08:00
|
|
|
"ctrl/sampling/sampling",
|
2012-12-22 21:28:01 -08:00
|
|
|
"ctrl/sampling2/sampling2",
|
2012-11-09 11:33:22 +01:00
|
|
|
"display/display",
|
2013-03-09 13:41:42 -08:00
|
|
|
"display/vblankmulti",
|
2012-11-09 11:33:22 +01:00
|
|
|
"dmac/dmactest",
|
2012-11-30 21:50:52 +01:00
|
|
|
"loader/bss/bss",
|
2012-11-09 11:33:22 +01:00
|
|
|
"intr/intr",
|
|
|
|
"intr/vblank/vblank",
|
|
|
|
"misc/testgp",
|
2013-01-05 17:24:47 -08:00
|
|
|
"misc/libc",
|
|
|
|
"misc/dcache",
|
2013-02-23 19:31:03 -08:00
|
|
|
"mstick/mstick",
|
2012-11-09 11:33:22 +01:00
|
|
|
"string/string",
|
2012-11-09 13:00:36 +01:00
|
|
|
"gpu/callbacks/ge_callbacks",
|
2012-12-21 00:23:55 -08:00
|
|
|
"threads/alarm/alarm",
|
2012-12-21 12:38:12 -08:00
|
|
|
"threads/alarm/cancel/cancel",
|
|
|
|
"threads/alarm/refer/refer",
|
|
|
|
"threads/alarm/set/set",
|
2013-02-23 19:31:03 -08:00
|
|
|
"threads/callbacks/callbacks",
|
2012-12-14 21:25:29 -08:00
|
|
|
"threads/events/events",
|
2012-12-14 21:32:06 -08:00
|
|
|
"threads/events/cancel/cancel",
|
|
|
|
"threads/events/clear/clear",
|
|
|
|
"threads/events/create/create",
|
|
|
|
"threads/events/delete/delete",
|
|
|
|
"threads/events/poll/poll",
|
|
|
|
"threads/events/refer/refer",
|
|
|
|
"threads/events/set/set",
|
|
|
|
"threads/events/wait/wait",
|
2013-01-04 17:06:36 +01:00
|
|
|
"threads/k0/k0",
|
2013-02-24 22:51:55 -08:00
|
|
|
"threads/lwmutex/create",
|
|
|
|
"threads/lwmutex/delete",
|
|
|
|
"threads/lwmutex/lock",
|
|
|
|
"threads/lwmutex/priority",
|
|
|
|
"threads/lwmutex/refer",
|
|
|
|
"threads/lwmutex/try",
|
|
|
|
"threads/lwmutex/try600",
|
|
|
|
"threads/lwmutex/unlock",
|
2012-11-10 23:17:37 +01:00
|
|
|
"threads/mbx/mbx",
|
2012-12-16 23:01:43 -08:00
|
|
|
"threads/mbx/cancel/cancel",
|
|
|
|
"threads/mbx/create/create",
|
|
|
|
"threads/mbx/delete/delete",
|
|
|
|
"threads/mbx/poll/poll",
|
|
|
|
"threads/mbx/priority/priority",
|
|
|
|
"threads/mbx/receive/receive",
|
|
|
|
"threads/mbx/refer/refer",
|
|
|
|
"threads/mbx/send/send",
|
2013-02-24 22:51:55 -08:00
|
|
|
"threads/mutex/create",
|
|
|
|
"threads/mutex/delete",
|
|
|
|
"threads/mutex/lock",
|
2012-12-07 19:01:31 -08:00
|
|
|
"threads/mutex/mutex",
|
2013-02-24 22:51:55 -08:00
|
|
|
"threads/mutex/priority",
|
|
|
|
"threads/mutex/refer",
|
|
|
|
"threads/mutex/try",
|
|
|
|
"threads/mutex/unlock",
|
2012-11-12 08:16:16 -08:00
|
|
|
"threads/semaphores/semaphores",
|
2012-11-30 23:22:55 -08:00
|
|
|
"threads/semaphores/cancel/cancel",
|
|
|
|
"threads/semaphores/create/create",
|
2012-11-17 01:43:01 -08:00
|
|
|
"threads/semaphores/delete/delete",
|
|
|
|
"threads/semaphores/poll/poll",
|
2012-12-07 19:31:59 -08:00
|
|
|
"threads/semaphores/priority/priority",
|
2012-11-17 01:43:01 -08:00
|
|
|
"threads/semaphores/refer/refer",
|
|
|
|
"threads/semaphores/signal/signal",
|
2012-11-30 23:22:55 -08:00
|
|
|
"threads/semaphores/wait/wait",
|
2013-02-23 19:31:03 -08:00
|
|
|
"threads/threads/release",
|
|
|
|
"threads/threads/rotate",
|
|
|
|
"threads/threads/threadend",
|
|
|
|
"threads/threads/threads",
|
|
|
|
"threads/vpl/cancel",
|
2013-02-09 01:19:02 -08:00
|
|
|
"threads/vpl/delete",
|
|
|
|
"threads/vpl/free",
|
|
|
|
"threads/vpl/priority",
|
|
|
|
"threads/vpl/refer",
|
|
|
|
"threads/vpl/try",
|
2013-02-23 19:31:03 -08:00
|
|
|
"threads/vpl/vpl",
|
2012-11-11 19:30:48 +01:00
|
|
|
"power/power",
|
2012-11-11 22:38:19 +01:00
|
|
|
"umd/callbacks/umd",
|
2012-12-14 21:32:06 -08:00
|
|
|
"umd/wait/wait",
|
2012-11-14 21:24:57 +00:00
|
|
|
"io/directory/directory",
|
2012-11-12 14:35:10 +01:00
|
|
|
]
|
2012-11-09 10:43:11 +01:00
|
|
|
|
2012-11-12 14:35:10 +01:00
|
|
|
tests_next = [
|
2013-02-23 19:31:03 -08:00
|
|
|
# These are the next tests up for fixing. These run by default.
|
|
|
|
"audio/atrac/atractest",
|
|
|
|
"audio/mp3/mp3test",
|
2012-11-30 21:50:52 +01:00
|
|
|
"audio/sascore/sascore",
|
2012-11-17 17:46:05 +01:00
|
|
|
"malloc/malloc",
|
2012-11-12 14:35:10 +01:00
|
|
|
"threads/fpl/fpl",
|
2012-11-30 21:50:52 +01:00
|
|
|
"threads/k0/k0",
|
2012-11-12 14:35:10 +01:00
|
|
|
"threads/msgpipe/msgpipe",
|
|
|
|
"threads/scheduling/scheduling",
|
2013-02-23 19:31:03 -08:00
|
|
|
"threads/threads/create",
|
2012-11-12 14:35:10 +01:00
|
|
|
"threads/vtimers/vtimer",
|
2013-02-09 01:19:02 -08:00
|
|
|
"threads/vpl/allocate",
|
|
|
|
"threads/vpl/create",
|
2012-11-12 14:35:10 +01:00
|
|
|
"threads/wakeup/wakeup",
|
2013-02-23 19:31:03 -08:00
|
|
|
"gpu/commands/basic",
|
|
|
|
"gpu/complex/complex",
|
|
|
|
"gpu/displaylist/state",
|
|
|
|
"gpu/reflection/reflection",
|
|
|
|
"gpu/rendertarget/rendertarget",
|
|
|
|
"gpu/signals/jumps",
|
|
|
|
"gpu/signals/simple",
|
2012-11-09 13:00:36 +01:00
|
|
|
"gpu/simple/simple",
|
|
|
|
"gpu/triangle/triangle",
|
|
|
|
"hle/check_not_used_uids",
|
|
|
|
"font/fonttest",
|
2012-11-09 11:33:22 +01:00
|
|
|
"io/cwd/cwd",
|
2013-02-23 19:31:03 -08:00
|
|
|
"io/file/file",
|
2012-11-09 11:33:22 +01:00
|
|
|
"io/io/io",
|
|
|
|
"io/iodrv/iodrv",
|
|
|
|
"modules/loadexec/loader",
|
2012-11-22 23:46:38 -08:00
|
|
|
"rtc/rtc",
|
2013-02-27 22:54:41 -08:00
|
|
|
"sysmem/partition",
|
2013-02-23 19:31:03 -08:00
|
|
|
"sysmem/sysmem",
|
2012-11-09 11:33:22 +01:00
|
|
|
"umd/io/umd_io",
|
2012-11-30 21:50:52 +01:00
|
|
|
"umd/raw_access/raw_access",
|
2012-11-12 14:35:10 +01:00
|
|
|
"utility/systemparam/systemparam",
|
2012-11-30 21:50:52 +01:00
|
|
|
"video/pmf/pmf",
|
|
|
|
"video/pmf_simple/pmf_simple",
|
2012-11-12 03:12:01 +01:00
|
|
|
]
|
|
|
|
|
2012-11-11 18:44:20 +01:00
|
|
|
# These don't even run (or run correctly) on the real PSP
|
|
|
|
test_broken = [
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2012-11-09 10:43:11 +01:00
|
|
|
# These are the tests we ignore (not important, or impossible to run)
|
|
|
|
tests_ignored = [
|
2012-11-09 11:33:22 +01:00
|
|
|
"kirk/kirk",
|
|
|
|
"me/me",
|
2012-11-09 12:32:35 +01:00
|
|
|
|
|
|
|
"umd/umd", # mostly fixed but output seems broken? (first retval of unregister...)
|
2012-11-09 10:43:11 +01:00
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def init():
|
|
|
|
global PPSSPP_EXE
|
|
|
|
if not os.path.exists("pspautotests"):
|
2012-12-15 02:27:40 -08:00
|
|
|
print("Please run git submodule init; git submodule update;")
|
2012-11-09 10:43:11 +01:00
|
|
|
sys.exit(1)
|
|
|
|
|
2012-11-29 14:40:11 +01:00
|
|
|
if not os.path.exists(TEST_ROOT + "cpu/cpu_alu/cpu_alu.prx"):
|
2012-12-15 02:27:40 -08:00
|
|
|
print("Please install the pspsdk and run make in common/ and in all the tests")
|
|
|
|
print("(checked for existence of cpu/cpu_alu/cpu_alu.prx)")
|
2012-11-09 10:43:11 +01:00
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
for p in PPSSPP_EXECUTABLES:
|
|
|
|
if os.path.exists(p):
|
|
|
|
PPSSPP_EXE = p
|
|
|
|
break
|
|
|
|
|
|
|
|
if not PPSSPP_EXE:
|
2012-12-15 02:27:40 -08:00
|
|
|
print("PPSSPP executable missing, please build one.")
|
2012-11-09 10:43:11 +01:00
|
|
|
sys.exit(1)
|
|
|
|
|
2012-11-12 03:12:01 +01:00
|
|
|
def tcprint(arg):
|
|
|
|
global teamcity_mode
|
|
|
|
if teamcity_mode:
|
2012-12-15 02:27:40 -08:00
|
|
|
print(arg)
|
2012-11-09 10:43:11 +01:00
|
|
|
|
2012-11-09 13:00:36 +01:00
|
|
|
def run_tests(test_list, args):
|
2012-11-18 22:35:08 +01:00
|
|
|
global PPSSPP_EXE, TIMEOUT
|
2012-11-09 11:03:01 +01:00
|
|
|
tests_passed = []
|
|
|
|
tests_failed = []
|
2012-12-15 02:27:40 -08:00
|
|
|
|
2012-11-09 10:43:11 +01:00
|
|
|
for test in test_list:
|
2012-11-11 15:05:40 +01:00
|
|
|
# Try prx first
|
2012-11-09 10:43:11 +01:00
|
|
|
expected_filename = TEST_ROOT + test + ".expected"
|
2012-11-11 16:46:25 +01:00
|
|
|
|
2012-11-11 15:05:40 +01:00
|
|
|
elf_filename = TEST_ROOT + test + ".prx"
|
2012-12-15 02:27:40 -08:00
|
|
|
print(elf_filename)
|
2012-11-11 15:05:40 +01:00
|
|
|
|
|
|
|
if not os.path.exists(elf_filename):
|
2012-12-15 02:27:40 -08:00
|
|
|
print("WARNING: no prx, trying elf")
|
2012-11-11 15:05:40 +01:00
|
|
|
elf_filename = TEST_ROOT + test + ".elf"
|
|
|
|
|
|
|
|
if not os.path.exists(elf_filename):
|
|
|
|
print("ERROR: PRX/ELF file missing, failing test: " + test)
|
|
|
|
tests_failed.append(test)
|
2012-11-12 03:12:01 +01:00
|
|
|
tcprint("##teamcity[testIgnored name='%s' message='PRX/ELF missing']" % test)
|
2012-11-11 15:05:40 +01:00
|
|
|
continue
|
|
|
|
|
2012-11-09 10:43:11 +01:00
|
|
|
if not os.path.exists(expected_filename):
|
2012-11-11 15:05:40 +01:00
|
|
|
print("WARNING: expects file missing, failing test: " + test)
|
2012-11-09 11:03:01 +01:00
|
|
|
tests_failed.append(test)
|
2012-11-12 03:12:01 +01:00
|
|
|
tcprint("##teamcity[testIgnored name='%s' message='Expects file missing']" % test)
|
2012-11-09 10:43:11 +01:00
|
|
|
continue
|
|
|
|
|
2012-11-23 18:41:46 +01:00
|
|
|
expected_output = open(expected_filename).read().strip()
|
2012-12-15 02:27:40 -08:00
|
|
|
|
2012-11-12 03:12:01 +01:00
|
|
|
tcprint("##teamcity[testStarted name='%s' captureStandardOutput='true']" % test)
|
2012-11-09 10:43:11 +01:00
|
|
|
|
2012-11-18 22:35:08 +01:00
|
|
|
cmdline = [PPSSPP_EXE, elf_filename]
|
2012-11-24 21:45:32 -08:00
|
|
|
cmdline.extend([i for i in args if i not in ['-v', '-g']])
|
2013-01-13 16:35:34 -08:00
|
|
|
if os.path.exists(expected_filename + ".bmp"):
|
|
|
|
cmdline.extend(["--screenshot=" + expected_filename + ".bmp", "--graphics"])
|
2012-11-18 22:35:08 +01:00
|
|
|
|
|
|
|
c = Command(cmdline)
|
|
|
|
c.run(TIMEOUT)
|
|
|
|
|
|
|
|
output = c.output.strip()
|
2012-12-15 02:27:40 -08:00
|
|
|
|
2012-11-18 22:35:08 +01:00
|
|
|
if c.timeout:
|
2012-12-15 02:27:40 -08:00
|
|
|
print(output)
|
|
|
|
print("Test exceded limit of %d seconds." % TIMEOUT)
|
2013-02-09 01:07:45 -08:00
|
|
|
tests_failed.append(test)
|
2012-11-18 22:35:08 +01:00
|
|
|
tcprint("##teamcity[testFailed name='%s' message='Test timeout']" % test)
|
|
|
|
tcprint("##teamcity[testFinished name='%s']" % test)
|
|
|
|
continue
|
2012-11-09 11:03:01 +01:00
|
|
|
|
|
|
|
if output.startswith("TESTERROR"):
|
2012-12-15 02:27:40 -08:00
|
|
|
print("Failed to run test " + elf_filename + "!")
|
2012-11-09 11:03:01 +01:00
|
|
|
tests_failed.append(test)
|
2012-11-12 03:12:01 +01:00
|
|
|
tcprint("##teamcity[testFailed name='%s' message='Failed to run test']" % test)
|
|
|
|
tcprint("##teamcity[testFinished name='%s']" % test)
|
2012-11-09 11:03:01 +01:00
|
|
|
continue
|
|
|
|
|
|
|
|
different = False
|
2012-11-23 18:41:46 +01:00
|
|
|
expected_lines = [x.strip() for x in expected_output.splitlines()]
|
|
|
|
output_lines = [x.strip() for x in output.splitlines()]
|
2012-12-15 02:27:40 -08:00
|
|
|
|
2012-11-11 15:05:40 +01:00
|
|
|
for i in range(0, min(len(output_lines), len(expected_lines))):
|
2012-11-09 11:03:01 +01:00
|
|
|
if output_lines[i] != expected_lines[i]:
|
2012-12-15 02:27:40 -08:00
|
|
|
print("E%i < %s" % (i + 1, expected_lines[i]))
|
|
|
|
print("O%i > %s" % (i + 1, output_lines[i]))
|
2012-11-09 11:03:01 +01:00
|
|
|
different = True
|
|
|
|
|
2012-11-11 15:05:40 +01:00
|
|
|
if len(output_lines) != len(expected_lines):
|
2012-11-23 18:33:40 +01:00
|
|
|
for i in range(len(output_lines), len(expected_lines)):
|
2012-12-15 02:27:40 -08:00
|
|
|
print("E%i < %s" % (i + 1, expected_lines[i]))
|
2012-11-23 18:33:40 +01:00
|
|
|
for i in range(len(expected_lines), len(output_lines)):
|
2012-12-15 02:27:40 -08:00
|
|
|
print("O%i > %s" % (i + 1, output_lines[i]))
|
|
|
|
print("*** Different number of lines!")
|
2012-11-11 15:05:40 +01:00
|
|
|
different = True
|
|
|
|
|
2012-11-09 11:03:01 +01:00
|
|
|
if not different:
|
2012-11-11 22:38:19 +01:00
|
|
|
if '-v' in args:
|
2012-12-15 02:27:40 -08:00
|
|
|
print("++++++++++++++ The Equal Output +++++++++++++")
|
|
|
|
print("\n".join(output_lines))
|
|
|
|
print("+++++++++++++++++++++++++++++++++++++++++++++")
|
|
|
|
print(" " + test + " - passed!")
|
2012-11-09 11:03:01 +01:00
|
|
|
tests_passed.append(test)
|
2012-11-12 03:12:01 +01:00
|
|
|
tcprint("##teamcity[testFinished name='%s']" % test)
|
2012-11-09 10:43:11 +01:00
|
|
|
else:
|
2012-11-11 19:30:48 +01:00
|
|
|
if '-v' in args:
|
2012-12-15 02:27:40 -08:00
|
|
|
print("============== output from failed " + test + " :")
|
|
|
|
print(output)
|
|
|
|
print("============== expected output:")
|
|
|
|
print(expected_output)
|
|
|
|
print("===============================")
|
2012-11-09 11:03:01 +01:00
|
|
|
tests_failed.append(test)
|
2012-11-12 03:12:01 +01:00
|
|
|
tcprint("##teamcity[testFailed name='%s' message='Output different from expected file']" % test)
|
|
|
|
tcprint("##teamcity[testFinished name='%s']" % test)
|
2012-11-09 11:03:01 +01:00
|
|
|
|
2012-12-15 02:27:40 -08:00
|
|
|
print("%i tests passed, %i tests failed." % (
|
|
|
|
len(tests_passed), len(tests_failed)))
|
2012-11-09 13:00:36 +01:00
|
|
|
|
2012-11-09 11:03:01 +01:00
|
|
|
if len(tests_failed):
|
2012-12-15 02:27:40 -08:00
|
|
|
print("Failed tests:")
|
2012-11-09 11:03:01 +01:00
|
|
|
for t in tests_failed:
|
2012-12-15 02:27:40 -08:00
|
|
|
print(" " + t)
|
|
|
|
print("Ran " + PPSSPP_EXE)
|
2012-11-09 10:43:11 +01:00
|
|
|
|
2012-11-09 13:00:36 +01:00
|
|
|
|
2012-11-09 10:43:11 +01:00
|
|
|
def main():
|
2012-11-12 03:12:01 +01:00
|
|
|
global teamcity_mode
|
2012-11-09 10:43:11 +01:00
|
|
|
init()
|
2012-11-09 13:00:36 +01:00
|
|
|
tests = []
|
|
|
|
args = []
|
|
|
|
for arg in sys.argv[1:]:
|
2012-11-12 03:12:01 +01:00
|
|
|
if arg == '--teamcity':
|
|
|
|
teamcity_mode = True
|
|
|
|
elif arg[0] == '-':
|
2012-11-09 13:00:36 +01:00
|
|
|
args.append(arg)
|
|
|
|
else:
|
|
|
|
tests.append(arg)
|
|
|
|
|
|
|
|
if not tests:
|
2012-11-12 14:35:10 +01:00
|
|
|
if '-g' in args:
|
|
|
|
tests = tests_good
|
|
|
|
else:
|
|
|
|
tests = tests_next + tests_good
|
2012-11-09 13:00:36 +01:00
|
|
|
|
|
|
|
run_tests(tests, args)
|
2012-11-09 10:43:11 +01:00
|
|
|
|
|
|
|
main()
|