blob: ea4350dc22265ee5e84fe1dded17d225e40e30c3 [file] [log] [blame]
#!/usr/bin/env python3
# Copyright 2023 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Runs test within Spike, Qemu and Renode simulators."""
import argparse
import os
import re
import sys
import tempfile
import io
import pexpect
parser = argparse.ArgumentParser(
description="Run a springbok test on an simulator.")
parser.add_argument("simulator",
help="Select a simulator",
choices=["renode", "qemu", "spike"])
parser.add_argument("elf",
help="Elf to execute on a simulator")
parser.add_argument("--renode-path",
help="Path to renode simulator")
parser.add_argument("--trace-output",
help="Path to trace output file")
parser.add_argument("--qemu-path",
help="Path to qemu simulator")
parser.add_argument("--spike-path",
help="Path to spike simulator")
parser.add_argument("--timeout", type=int,
help="Timeout for test", default=1000)
parser.add_argument("--quick_test",
help="allow quickest test time", action="store_true")
args = parser.parse_args()
class Simulation: # pylint: disable=too-few-public-methods
""" Base class for simulation """
def __init__(self, simulator_cmd):
self.simulator_cmd = simulator_cmd
self.buffer = io.StringIO()
self.child = None
self.termination_strings = [
"main returned",
"Exception occurred",
"ReadByte from non existing peripheral",
"File does not exist"
]
# stats collected command for renode
self.renode_end_command = None
def run(self, timeout=1000):
""" Run the simulation command and quit the simulation."""
self.child = pexpect.spawn(self.simulator_cmd, encoding="utf-8")
self.child.logfile = self.buffer
try:
self.child.expect(self.termination_strings, timeout=timeout)
except pexpect.exceptions.EOF as run_hit_eof:
self.buffer.seek(0)
message = ("Runner reach EOF with the execution log: \n\n" +
cleanup_message(self.buffer.read()))
exc = pexpect.exceptions.EOF(message)
exc.__cause__ = None
raise exc from run_hit_eof
except pexpect.exceptions.TIMEOUT as run_hit_timeout:
self.buffer.seek(0)
message = ("Runner times out with the execution log: \n\n" +
cleanup_message(self.buffer.read()))
exc = pexpect.exceptions.EOF(message)
exc.__cause__ = None
raise exc from run_hit_timeout
if self.renode_end_command:
self.child.send(self.renode_end_command)
self.child.expect("(springbok)", timeout=timeout)
self.child.send("\nq\n")
self.child.expect(pexpect.EOF, timeout=timeout)
self.child.close()
self.buffer.seek(0)
return self.buffer.read()
class QemuSimulation(Simulation): # pylint: disable=too-few-public-methods
""" Qemu simulation """
def __init__(self, path, elf):
self.qemu_simulator_cmd = (
"%(sim)s -M springbok -nographic -d springbok "
"-device loader,file=%(elf)s")
self.sim_params = {"sim": path, "elf": elf}
super().__init__(self.qemu_simulator_cmd % self.sim_params)
class RenodeSimulation(Simulation): # pylint: disable=too-few-public-methods
""" Renode Simulation """
def __init__(self, path, elf):
# Get the ROOTDIR path if it exists
self.rootdir = os.environ.get("ROOTDIR", default=None)
if self.rootdir is None:
parser.error("ROOTDIR environment variable not set.")
renode_script = """
$bin=@%(elf)s
path set @%(rootdir)s
include @sim/config/springbok.resc"""
if args.quick_test:
renode_script += """
sysbus.cpu2 PerformanceInMips 2000
emulation SetGlobalQuantum "1" """
if args.trace_output:
renode_script += """
sysbus.cpu2 EnableExecutionTracing @%(trace_file)s PCAndOpcode """
renode_script += """
sysbus.cpu2 EnableExternalWindowMmu false
start
sysbus.vec_controlblock WriteDoubleWord 0xc 0"""
trace_file = ""
if args.trace_output:
trace_file = os.path.realpath(args.trace_output)
self.script_params = {
"elf": os.path.realpath(elf),
"rootdir": self.rootdir,
"trace_file": trace_file
}
self.renode_script = renode_script % self.script_params
self.renode_args = [
f"{path}",
"--disable-xwt",
" --console",
"--plain",
]
self.renode_simulator_cmd = " ".join(self.renode_args)
super().__init__(self.renode_simulator_cmd)
self.renode_end_command = "\nsysbus.cpu2 ExecutedInstructions\n"
def run(self, timeout=120):
file_desc, script_path = tempfile.mkstemp(suffix=".resc")
try:
with os.fdopen(file_desc, "w") as tmp:
tmp.write(self.renode_script)
tmp.flush()
self.simulator_cmd += f" {script_path}"
test_output = super().run(timeout=timeout)
finally:
os.remove(script_path)
return test_output
class SpikeSimulation(Simulation): # pylint: disable=too-few-public-methods
""" Spike Simulation """
def __init__(self, path, elf):
trace_file = ""
if args.trace_output:
trace_file = os.path.realpath(args.trace_output)
self.sim_params = {
"path": path,
"elf": elf,
"trace_file": trace_file
}
self.spike_simulator_cmd = (
"%(path)s -m0x34000000:0x1000000 "
"--pc=0x34000000 ")
if args.trace_output:
self.spike_simulator_cmd += " -l --log=%(trace_file)s "
self.spike_simulator_cmd += " %(elf)s"
super().__init__(self.spike_simulator_cmd % self.sim_params)
Simulators = {
"qemu": QemuSimulation,
"renode": RenodeSimulation,
"spike": SpikeSimulation,
}
simulators_paths = {
"renode": args.renode_path,
"qemu": args.qemu_path,
"spike": args.spike_path,
}
def cleanup_message(message: str) -> str:
""" Clean up the message generated by Mono.
The non-ascii code generated by Mono.
Convert the opcode count from hex to decimal.
"""
ansi_escape = re.compile(r"\x1B(?:[@-Z\\-_]|\[[0-?]*[ -/]*[@-~])")
output = ansi_escape.sub("", message)
op_count_out = re.compile(
r"(?P<op_count>0x[0-9A-Fa-f]+)\r\r\r\n\(springbok\)")
op_count = op_count_out.search(output)
if op_count:
op_count = int(op_count.group(1), 16)
output = op_count_out.sub(
f"Renode total instruction count: {op_count}\n", output)
return output
def main():
""" Run a test and check for Pass or Fail """
simulator_path = simulators_paths[args.simulator]
if simulator_path is None:
parser.error(
f"Must provide path to simulator {args.simulator}, "
f"use argument --{args.simulator}-path")
simulator_class = Simulators[args.simulator]
simulator = simulator_class(simulator_path, args.elf)
output = simulator.run(timeout=args.timeout)
output = cleanup_message(output)
print(output)
failure_strings = [
"FAILED",
"Exception occurred",
"ReadByte from non existing peripheral",
"File does not exist"
]
if any(x in output for x in failure_strings):
sys.exit(1)
# Grab the return code from the output string with regex
# Syntax: "main returned: ", <code> (<hex_code>)
return_string = re.compile(
r"\"main returned:\s\",(?P<ret_code>\s[0-9]+\s*)")
code = return_string.search(output)
sys.exit(int(code.group(1)))
if __name__ == "__main__":
main()