| # 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 |
| ], |
| ) |