blob: ebb5ae409e978caedc19276f220509ff18b2faac [file] [log] [blame]
# Copyright lowRISC contributors.
# Licensed under the Apache License, Version 2.0, see LICENSE for details.
# SPDX-License-Identifier: Apache-2.0
load("@//rules:opentitan.bzl", "opentitan_flash_binary", "opentitan_rom_binary")
load("@bazel_skylib//lib:shell.bzl", "shell")
OTTF_SUCCESS_MSG = r"PASS.*\n"
OTTF_FAILURE_MSG = r"(FAIL|FAULT).*\n"
ROM_BOOT_FAILURE_MSG = "BFV:[0-9a-f]{8}"
# These are defined for positive test cases and should be flipped for negative
# test cases, i.e., when a test failure is the expected outcome.
DEFAULT_TEST_SUCCESS_MSG = OTTF_SUCCESS_MSG
DEFAULT_TEST_FAILURE_MSG = "({})|({})".format(
OTTF_FAILURE_MSG,
ROM_BOOT_FAILURE_MSG,
)
_BASE_PARAMS = {
"args": [], # Passed to test runner as arguments.
"data": [],
"local": True,
"otp": "@//hw/ip/otp_ctrl/data:img_rma",
"rom": "@//sw/device/lib/testing/test_rom",
"tags": [],
"test_runner": "@//util:opentitan_functest_runner.sh",
"test_cmds": [], # Passed to test_runner via TEST_CMDS env var.
"timeout": "moderate", # 5 minutes
"exit_success": DEFAULT_TEST_SUCCESS_MSG,
"exit_failure": DEFAULT_TEST_FAILURE_MSG,
}
def dv_params(
# Base Parameters
args = _BASE_PARAMS["args"] + [
"$(location {dvsim_config})",
],
data = _BASE_PARAMS["data"],
local = _BASE_PARAMS["local"],
otp = _BASE_PARAMS["otp"],
rom = _BASE_PARAMS["rom"],
tags = _BASE_PARAMS["tags"],
timeout = _BASE_PARAMS["timeout"],
test_runner = "@//util:dvsim_test_runner.sh",
test_cmds = [],
# DV-specific Parameters
dvsim_config = "@//hw/top_earlgrey/dv:chip_sim_cfg.hjson",
**kwargs):
"""A macro to create DV sim parameters for OpenTitan functional tests.
This macro emits a dictionary of parameters which are pasted into the DV
simulation specific test rule.
Parameters:
@param args: Extra arguments to pass to the test runner (`dvsim.py`).
@param data: Data dependencies of the test.
@param local: Whether the test should be run locally without sandboxing.
@param otp: The OTP image to use.
@param rom: The ROM image to use.
@param tags: The test tags to apply to the test rule.
@param timeout: The timeout to apply to the test rule.
@param test_cmds: A list of required commands and args that make up the
immutable portion of the harness invocation.
@param dvsim_config: The dvsim.py Hjson config file for the toplevel.
"""
required_test_cmds = [
"-i",
"chip_sw_{name}",
"--",
]
required_data = [
dvsim_config,
"@//util/dvsim",
"@//hw:all_files",
"@//hw:fusesoc_ignore",
]
required_tags = ["dv"]
kwargs.update(
args = args,
data = required_data + data,
local = local,
otp = otp,
rom = rom,
tags = required_tags + tags,
test_runner = test_runner,
test_cmds = required_test_cmds + test_cmds,
timeout = timeout,
dvsim_config = dvsim_config,
)
return kwargs
def verilator_params(
# Base Parameters
args = _BASE_PARAMS["args"],
data = _BASE_PARAMS["data"],
exit_success = _BASE_PARAMS["exit_success"],
exit_failure = _BASE_PARAMS["exit_failure"],
local = _BASE_PARAMS["local"],
otp = _BASE_PARAMS["otp"],
rom = _BASE_PARAMS["rom"],
tags = _BASE_PARAMS["tags"] + ["cpu:4"],
timeout = _BASE_PARAMS["timeout"],
test_runner = _BASE_PARAMS["test_runner"],
test_cmds = _BASE_PARAMS["test_cmds"] + [
"console",
"--exit-success={exit_success}",
"--exit-failure={exit_failure}",
],
# Verilator-specific Parameters
# None
**kwargs):
"""A macro to create Verilator sim parameters for OpenTitan functional tests.
This macro emits a dictionary of parameters which are pasted into the
Verilator simulation specific test rule.
Parameters:
@param args: Extra arguments to pass to the test runner (`opentitantool`).
@param data: Data dependencies of the test.
@param local: Whether the test should be run locally without sandboxing.
@param otp: The OTP image to use.
@param rom: The ROM image to use.
@param tags: The test tags to apply to the test rule.
@param timeout: The timeout to apply to the test rule.
@param test_cmds: A list of required commands and args that make up the
immutable portion of the harness invocation.
"""
default_args = [
"--rcfile=",
"--logging=info",
]
required_test_cmds = [
"--interface=verilator",
"--verilator-bin=$(location @//hw:verilator)",
"--verilator-rom=$(location {rom})",
"--verilator-flash=$(location {flash})",
"--verilator-otp=$(location {otp})",
]
required_data = [
"@//hw:verilator",
"@//hw:fusesoc_ignore",
"@//sw/host/opentitantool:test_resources",
]
required_tags = ["verilator"]
kwargs.update(
args = default_args + args,
data = required_data + data,
exit_success = exit_success,
exit_failure = exit_failure,
local = local,
otp = otp,
rom = rom,
tags = required_tags + tags,
test_runner = test_runner,
test_cmds = required_test_cmds + test_cmds,
timeout = timeout,
)
return kwargs
def cw310_params(
# Base Parameters
args = _BASE_PARAMS["args"],
data = _BASE_PARAMS["data"] + ["{bitstream}"],
exit_success = _BASE_PARAMS["exit_success"],
exit_failure = _BASE_PARAMS["exit_failure"],
local = _BASE_PARAMS["local"],
otp = _BASE_PARAMS["otp"],
tags = _BASE_PARAMS["tags"],
test_runner = _BASE_PARAMS["test_runner"],
test_cmds = [
"--exec=\"load-bitstream --rom-kind={rom_kind} $(location {bitstream})\"",
"--exec=\"bootstrap --clear-uart=true $(location {flash})\"",
"console",
"--exit-success={exit_success}",
"--exit-failure={exit_failure}",
],
# CW310-specific Parameters
bitstream = "@//hw/bitstream:test_rom",
rom_kind = None,
# None
timeout = "short",
**kwargs):
"""A macro to create CW310 parameters for OpenTitan functional tests.
This macro emits a dictionary of parameters which are pasted into the
ChipWhisperer-310 FPGA specific test rule.
Parameters:
@param args: Extra arguments to pass the test runner `opentitantool`.
@param data: Data dependencies of the test.
@param local: Whether the test should be run locally without sandboxing.
@param otp: The OTP image to use.
@param tags: The test tags to apply to the test rule.
@param test_cmds: A list of required commands and args that make up the
immutable portion of the harness invocation.
@param timeout: The timeout to apply to the test rule.
@param bitstream: The bitstream to load into the FPGA (this specifies
the ROM image that is also used, since the ROM is
baked into the bitstream).
@param rom_kind: The ROM type (test or mask) that is baked into the
bitstream that is loaded into the FPGA.
"""
default_args = [
"--rcfile=",
"--logging=info",
]
required_test_cmds = [
"--interface=cw310",
]
required_data = [
"@//sw/host/opentitantool:test_resources",
]
required_tags = [
"cw310",
"exclusive",
]
kwargs.update(
args = default_args + args,
data = required_data + data,
exit_success = exit_success,
exit_failure = exit_failure,
local = local,
otp = otp,
tags = required_tags + tags,
test_runner = test_runner,
test_cmds = required_test_cmds + test_cmds,
timeout = timeout,
bitstream = bitstream,
rom_kind = rom_kind,
)
return kwargs
def opentitan_functest(
name,
targets = ["dv", "verilator", "cw310"],
args = [],
data = [],
test_in_rom = False,
ot_flash_binary = None,
signed = False,
test_harness = "@//sw/host/opentitantool",
key = "test_key_0",
dv = None,
verilator = None,
cw310 = None,
**kwargs):
"""A helper macro for generating OpenTitan functional tests.
This macro is mostly a wrapper around `opentitan_flash_binary`, but creates
testing artifacts for each of the hardware targets in `targets`. The testing
artifacts are then given to an `sh_test` rule which dispatches the test to
the corresponding hardware target via `opentitantool`.
Parameters:
@param name: The name of this rule.
@param targets: A list of hardware targets on which to dispatch tests.
@param args: Extra arguments (in addition to those defined in the target-
specific parameter dictionary) to pass to the test harness
(e.g., `opentitantool`).
@param data: Extra data dependencies (in addition to those defined in the
target-specific parameter dictionary) needed while executing
the test.
@param test_in_rom: Whether to run the test from ROM, runs from flash by
default.
@param ot_flash_binary: Use the named `opentitan_flash_binary` as the
flash image for the test rather than building one
from srcs/deps.
@param signed: Whether to sign the test image. Unsigned by default.
@param test_harness: The binary on the host side that runs the test.
@param key: Which signed test image (by key) to use.
@param dv: DV test parameters.
@param verilator: Verilator test parameters.
@param cw310: CW310 test parameters.
@param **kwargs: Arguments to forward to `opentitan_flash_binary`.
This macro emits the following rules:
if `test_in_rom`:
opentitan_rom_binary named: {name}_rom_prog (and all emitted rules).
opentitan_flash_binary named: {name}_prog (and all emitted rules).
sh_test named: {name}_sim_dv
sh_test named: {name}_sim_verilator
sh_test named: {name}_fpga_cw310
test_suite named: {name}
"""
deps = kwargs.pop("deps", [])
all_tests = []
# Handle the special case were the test is run at the ROM stage.
if test_in_rom:
if ot_flash_binary:
fail("Tests that run in ROM stage cannot use pre-built flash binary.")
if "cw310" in targets:
fail("Tests that run in ROM stage cannot run on FPGA.")
ot_flash_binary = name + "_rom_prog"
opentitan_rom_binary(
name = ot_flash_binary,
deps = deps,
**kwargs
)
# Generate SW artifacts for the tests.
if not ot_flash_binary:
ot_flash_binary = name + "_prog"
opentitan_flash_binary(
name = ot_flash_binary,
signed = signed,
deps = deps,
**kwargs
)
target_params = {
"sim_dv": dv_params() if not dv else dv,
"sim_verilator": verilator_params() if not verilator else verilator,
"fpga_cw310": cw310_params() if not cw310 else cw310,
}
for target, params in target_params.items():
# We only emit `sh_test` rules for hardware targets that we want to run
# this test on.
if target.split("_")[-1] not in targets:
continue
# Set test name.
test_name = "{}_{}".format(name, target)
if "manual" not in params.get("tags"):
all_tests.append(test_name)
# The args/data lists will hold all of the test arguments/artifacts that
# need to be present at runtime.
target_args = args
target_data = data + [test_harness]
target_test_cmds = []
########################################################################
# Retrieve host-side test components.
########################################################################
# Environment variables to pass to the `sh_test`.
env = {}
# Set the test runner (i.e., the script that invokes the "test harness").
test_runner = params.pop("test_runner")
# Set the test harness (i.e., the host-side test component, or
# `opentitantool` in most cases).
env["TEST_HARNESS"] = "$(location {})".format(test_harness)
########################################################################
# Retrieve device-side test components.
########################################################################
# Set ROM image.
# Note: FPGA targets will not specify a ROM image as the ROM imaged is
# specified via the `bitstream` parameter (since the ROM is baked into
# the bitstream).
rom = params.pop("rom", None)
if rom:
if test_in_rom:
rom_filegroup = "{}_rom_prog_{}".format(name, target)
rom = "{}_scr_vmem".format(rom_filegroup)
else:
rom_label = Label(rom)
rom_filegroup = "@{}//{}:{}_{}".format(
rom_label.workspace_name,
rom_label.package,
rom_label.name,
target,
)
rom = "{}_scr_vmem".format(rom_filegroup)
target_data.append(rom)
target_data.append(rom_filegroup)
# Set flash image.
if target in ["sim_dv", "sim_verilator"]:
flash = "{}_{}_scr_vmem64".format(ot_flash_binary, target)
else:
flash = "{}_{}_bin".format(ot_flash_binary, target)
if signed:
flash += "_signed_{}".format(key)
# If test is to be run in ROM we load the same image into flash as a
# as a placeholder (since execution will never reach flash). Moreover,
# there is no need to update the data dependencies list as the ROM
# targets were already added above.
if test_in_rom:
flash = rom
else:
target_data.append(flash)
target_data.append("{}_{}".format(ot_flash_binary, target))
# Set OTP image.
otp = params.pop("otp")
target_data.append(otp)
########################################################################
# Retrieve hardware-target-specific parameters.
########################################################################
# Set Bitstream (for FPGA targets).
bitstream = params.pop("bitstream", None)
rom_kind = params.pop("rom_kind", None)
if bitstream and not rom_kind:
if "test_rom" in bitstream:
rom_kind = "testrom"
elif "rom" in bitstream:
rom_kind = "rom"
else:
fail("Unknown bitstream type. Expected the bitstream label to contain the string 'test_rom' or 'rom'.")
# Set success/failure strings for target platforms that print test
# results over the UART (e.g., Verilator and FPGA).
exit_strings_kwargs = {}
if target in ["fpga_cw310", "sim_verilator"]:
exit_strings_kwargs = {
"exit_success": shell.quote(params.pop("exit_success")),
"exit_failure": shell.quote(params.pop("exit_failure")),
}
# Set dvsim configuration file (for DV simulation target).
dvsim_config = params.pop("dvsim_config", None)
########################################################################
# Format arguments to send to the test running scripts/tools.
########################################################################
# Concatenate args / data passed into the opentitan_functest macro with
# args / data from device-specific params.
target_args = target_args + params.pop("args")
target_data = target_data + params.pop("data")
target_test_cmds = target_test_cmds + params.pop("test_cmds")
# Fill placeholders in arg/data lists.
format_dict = {
"name": name,
"flash": flash,
"rom": rom,
"otp": otp,
"dvsim_config": dvsim_config,
"bitstream": bitstream,
"rom_kind": rom_kind,
}
format_dict.update(exit_strings_kwargs)
target_args = [a.format(**format_dict) for a in target_args]
target_data = [d.format(**format_dict) for d in target_data]
target_test_cmds = [s.format(**format_dict) for s in target_test_cmds]
env["TEST_CMDS"] = " ".join(target_test_cmds)
if target == "fpga_cw310":
# We attach the UART configuration to the front of the command line
# so that they'll be parsed as global options rather than
# command-specific options.
target_args = select({
"@//ci:lowrisc_fpga_cw310": ["--cw310-uarts=/dev/ttyACM_CW310_1,/dev/ttyACM_CW310_0"],
"//conditions:default": [],
}) + target_args
########################################################################
# Instantiate the test rule.
########################################################################
native.sh_test(
name = test_name,
srcs = [test_runner],
args = target_args,
data = target_data,
env = env,
**params
)
############################################################################
# Instantiate a suite of tests to run the same test on each hardware target.
############################################################################
native.test_suite(
name = name,
tests = all_tests,
# In test_suites, tags perform a filtering function and will select
# matching tags internally instead of allowing filters to select
# test_suites.
# There are exceptions: "small", "medium", "large", "enourmous", "manual"
tags = [
# The manual tag is a special case and is applied to the test suite
# it prevents it from being included in wildcards so that
# --build_tag_filters=-verilator works as expected and excludes
# building verilator tests so the verilator build wont be invoked.
"manual",
# For more see https://bazel.build/reference/be/general#test_suite.tags
],
)