blob: 7c2ec1505fd6f84c56423d5b9f3470d9ea868a26 [file] [log] [blame]
// Copyright lowRISC contributors.
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0
//
module tb;
// dep packages
import uvm_pkg::*;
import dv_utils_pkg::*;
import otp_ctrl_env_pkg::*;
import otp_ctrl_test_pkg::*;
import otp_ctrl_reg_pkg::*;
import mem_bkdr_util_pkg::mem_bkdr_util;
// macro includes
`include "uvm_macros.svh"
`include "dv_macros.svh"
// TB base test ENV_T & CFG_T specification
//
// Specify the parameters for the otp_ctrl_base_test
// This will invoke the UVM registry and link this test type to
// the name 'otp_ctrl_base_test' as a test name passed by UVM_TESTNAME
//
// This is done explicitly only for the prim_pkg::ImplGeneric implementation
// since partner base tests inherit from otp_ctrl_base_test#(CFG_T, ENV_T) and
// specify directly (CFG_T, ENV_T) via the class extension and use a different
// UVM_TESTNAME
if (`PRIM_DEFAULT_IMPL == prim_pkg::ImplGeneric) begin : gen_spec_base_test_params
typedef otp_ctrl_base_test #(.CFG_T(otp_ctrl_env_cfg),
.ENV_T(otp_ctrl_env)) otp_ctrl_base_test_t;
end
wire clk, rst_n;
wire devmode;
wire otp_ctrl_pkg::flash_otp_key_req_t flash_req;
wire otp_ctrl_pkg::flash_otp_key_rsp_t flash_rsp;
wire otp_ctrl_pkg::otbn_otp_key_req_t otbn_req;
wire otp_ctrl_pkg::otbn_otp_key_rsp_t otbn_rsp;
wire otp_ctrl_pkg::sram_otp_key_req_t[NumSramKeyReqSlots-1:0] sram_req;
wire otp_ctrl_pkg::sram_otp_key_rsp_t[NumSramKeyReqSlots-1:0] sram_rsp;
wire [NUM_MAX_INTERRUPTS-1:0] interrupts;
wire intr_otp_operation_done, intr_otp_error;
wire otp_ctrl_pkg::otp_ast_req_t ast_req;
tlul_pkg::tl_d2h_t prim_tl_o;
// interfaces
clk_rst_if clk_rst_if(.clk(clk), .rst_n(rst_n));
pins_if #(NUM_MAX_INTERRUPTS) intr_if(interrupts);
pins_if #(1) devmode_if(devmode);
// lc_otp interfaces
push_pull_if #(.HostDataWidth(LC_PROG_DATA_SIZE), .DeviceDataWidth(1))
lc_prog_if(.clk(clk), .rst_n(rst_n));
push_pull_if #(.DeviceDataWidth(SRAM_DATA_SIZE))
sram_if[NumSramKeyReqSlots](.clk(clk), .rst_n(rst_n));
push_pull_if #(.DeviceDataWidth(OTBN_DATA_SIZE)) otbn_if(.clk(clk), .rst_n(rst_n));
push_pull_if #(.DeviceDataWidth(FLASH_DATA_SIZE)) flash_addr_if(.clk(clk), .rst_n(rst_n));
push_pull_if #(.DeviceDataWidth(FLASH_DATA_SIZE)) flash_data_if(.clk(clk), .rst_n(rst_n));
tl_if tl_if(.clk(clk), .rst_n(rst_n));
tl_if prim_tl_if(.clk(clk), .rst_n(rst_n));
otp_ctrl_if otp_ctrl_if(.clk_i(clk), .rst_ni(rst_n));
`DV_ALERT_IF_CONNECT
// edn_clk, edn_rst_n and edn_if are defined and driven in below macro
`DV_EDN_IF_CONNECT
assign otp_ctrl_if.lc_prog_req = lc_prog_if.req;
assign otp_ctrl_if.lc_prog_err = lc_prog_if.d_data;
// Assign to otp_ctrl_if for assertion checks.
assign otp_ctrl_if.lc_prog_ack = lc_prog_if.ack;
assign otp_ctrl_if.flash_acks = flash_data_if.ack;
assign otp_ctrl_if.otbn_ack = otbn_if.ack;
// This signal probes design's alert request to avoid additional logic for triggering alert and
// disable assertions.
// Alert checkings are done independently in otp_ctrl's scb.
// The correctness of this probed signal is checked in otp_ctrl's scb as well.
assign otp_ctrl_if.alert_reqs = dut.alerts[0] | dut.alerts[1];
// connected to interface
wire otp_ext_voltage_h = otp_ctrl_if.ext_voltage_h_io;
// dut
otp_ctrl dut (
.clk_i (clk ),
.rst_ni (rst_n ),
// edn
.clk_edn_i (edn_clk ),
.rst_edn_ni (edn_rst_n ),
.edn_o (edn_if[0].req ),
.edn_i ({edn_if[0].ack, edn_if[0].d_data}),
// bus interfaces
.core_tl_i (tl_if.h2d ),
.core_tl_o (tl_if.d2h ),
.prim_tl_i (prim_tl_if.h2d),
.prim_tl_o (prim_tl_if.d2h),
// interrupt
.intr_otp_operation_done_o (intr_otp_operation_done),
.intr_otp_error_o (intr_otp_error),
// alert
.alert_rx_i (alert_rx ),
.alert_tx_o (alert_tx ),
// ast
.otp_ast_pwr_seq_o (ast_req),
.otp_ast_pwr_seq_h_i (otp_ctrl_if.otp_ast_pwr_seq_h_i),
.otp_alert_o (otp_ctrl_if.otp_alert_o),
// pwrmgr
.pwr_otp_i (otp_ctrl_if.pwr_otp_init_i),
.pwr_otp_o ({otp_ctrl_if.pwr_otp_done_o, otp_ctrl_if.pwr_otp_idle_o}),
// lc
.lc_otp_vendor_test_o (otp_ctrl_if.otp_vendor_test_status_o),
.lc_otp_vendor_test_i (otp_ctrl_if.otp_vendor_test_ctrl_i),
.lc_otp_program_i ({lc_prog_if.req, lc_prog_if.h_data}),
.lc_otp_program_o ({lc_prog_if.d_data, lc_prog_if.ack}),
.lc_creator_seed_sw_rw_en_i (otp_ctrl_if.lc_creator_seed_sw_rw_en_i),
.lc_seed_hw_rd_en_i (otp_ctrl_if.lc_seed_hw_rd_en_i),
.lc_dft_en_i (otp_ctrl_if.lc_dft_en_i),
.lc_escalate_en_i (otp_ctrl_if.lc_escalate_en_i),
.lc_check_byp_en_i (otp_ctrl_if.lc_check_byp_en_i),
.otp_lc_data_o (otp_ctrl_if.lc_data_o),
// keymgr
.otp_keymgr_key_o (otp_ctrl_if.keymgr_key_o),
// flash
.flash_otp_key_i (flash_req),
.flash_otp_key_o (flash_rsp),
// sram
.sram_otp_key_i (sram_req),
.sram_otp_key_o (sram_rsp),
// otbn
.otbn_otp_key_i (otbn_req),
.otbn_otp_key_o (otbn_rsp),
.otp_hw_cfg_o (otp_ctrl_if.otp_hw_cfg_o),
.otp_ext_voltage_h_io (otp_ext_voltage_h),
//scan
.scan_en_i (otp_ctrl_if.scan_en_i),
.scan_rst_ni (otp_ctrl_if.scan_rst_ni),
.scanmode_i (otp_ctrl_if.scanmode_i),
// Test-related GPIO output
.cio_test_o (otp_ctrl_if.cio_test_o),
.cio_test_en_o (otp_ctrl_if.cio_test_en_o)
);
for (genvar i = 0; i < NumSramKeyReqSlots; i++) begin : gen_sram_pull_if
assign sram_req[i] = sram_if[i].req;
assign sram_if[i].ack = sram_rsp[i].ack;
assign sram_if[i].d_data = {sram_rsp[i].key, sram_rsp[i].nonce, sram_rsp[i].seed_valid};
assign otp_ctrl_if.sram_acks[i] = sram_rsp[i].ack;
initial begin
uvm_config_db#(virtual push_pull_if#(.DeviceDataWidth(SRAM_DATA_SIZE)))::set(null,
$sformatf("*env.m_sram_pull_agent[%0d]*", i), "vif", sram_if[i]);
end
end
assign otbn_req = otbn_if.req;
assign otbn_if.ack = otbn_rsp.ack;
assign otbn_if.d_data = {otbn_rsp.key, otbn_rsp.nonce, otbn_rsp.seed_valid};
assign flash_req = {flash_data_if.req, flash_addr_if.req};
assign flash_data_if.ack = flash_rsp.data_ack;
assign flash_addr_if.ack = flash_rsp.addr_ack;
assign flash_data_if.d_data = {flash_rsp.key, flash_rsp.seed_valid};
assign flash_addr_if.d_data = {flash_rsp.key, flash_rsp.seed_valid};
assign interrupts[OtpOperationDone] = intr_otp_operation_done;
assign interrupts[OtpErr] = intr_otp_error;
// Instantitate the memory backdoor util instance only for OS implementation
// Proprietary IP will instantiate their own backdoor util
if (`PRIM_DEFAULT_IMPL == prim_pkg::ImplGeneric) begin : gen_impl_generic
`define MEM_MODULE_PATH \
tb.dut.u_otp.gen_generic.u_impl_generic.u_prim_ram_1p_adv
`define MEM_ARRAY_PATH \
`MEM_MODULE_PATH.u_mem.gen_generic.u_impl_generic.mem
initial begin : mem_bkdr_util_gen
mem_bkdr_util m_mem_bkdr_util;
m_mem_bkdr_util = new(.name("mem_bkdr_util"),
.path(`DV_STRINGIFY(`MEM_ARRAY_PATH)),
.depth($size(`MEM_ARRAY_PATH)),
.n_bits($bits(`MEM_ARRAY_PATH)),
.err_detection_scheme(mem_bkdr_util_pkg::EccHamming_22_16));
uvm_config_db#(mem_bkdr_util)::set(null, "*.env", "mem_bkdr_util", m_mem_bkdr_util);
end : mem_bkdr_util_gen
`undef MEM_ARRAY_PATH
`undef MEM_MODULE_PATH
end : gen_impl_generic
initial begin
// These SVA checks the lc_escalate_en is either Off or On, we will use more than these
// 2 values.
// If the value is not lc_ctrl_pkg::Off, design will treat it as lc_ctrl_pkg::On.
$assertoff(0, tb.dut.u_prim_lc_sync_escalate_en.PrimLcSyncCheckTransients_A);
$assertoff(0, tb.dut.u_prim_lc_sync_escalate_en.PrimLcSyncCheckTransients0_A);
$assertoff(0, tb.dut.u_prim_lc_sync_escalate_en.PrimLcSyncCheckTransients1_A);
// These SVA checks the lc_sync_seed_hw_rd_en is either Off or On, we will use more than these
// 2 values.
// If the value is not lc_ctrl_pkg::On, design will treat it as lc_ctrl_pkg::Off.
$assertoff(0, tb.dut.u_prim_lc_sync_seed_hw_rd_en.PrimLcSyncCheckTransients_A);
$assertoff(0, tb.dut.u_prim_lc_sync_seed_hw_rd_en.PrimLcSyncCheckTransients0_A);
$assertoff(0, tb.dut.u_prim_lc_sync_seed_hw_rd_en.PrimLcSyncCheckTransients1_A);
// These SVA checks the lc_check_byp_en is either Off or On, we will use more than these
// 2 values.
// If the value is not lc_ctrl_pkg::On, design will treat it as lc_ctrl_pkg::Off.
$assertoff(0, tb.dut.u_prim_lc_sync_check_byp_en.PrimLcSyncCheckTransients_A);
$assertoff(0, tb.dut.u_prim_lc_sync_check_byp_en.PrimLcSyncCheckTransients0_A);
$assertoff(0, tb.dut.u_prim_lc_sync_check_byp_en.PrimLcSyncCheckTransients1_A);
// These SVA checks the lc_dft_en is either Off or On, we will use more than these 2 values.
// If the value is not lc_ctrl_pkg::On, design will treat it as lc_ctrl_pkg::Off.
$assertoff(0, tb.dut.u_prim_lc_sync_dft_en.PrimLcSyncCheckTransients_A);
$assertoff(0, tb.dut.u_prim_lc_sync_dft_en.PrimLcSyncCheckTransients0_A);
$assertoff(0, tb.dut.u_prim_lc_sync_dft_en.PrimLcSyncCheckTransients1_A);
$assertoff(0, tb.dut.u_tlul_lc_gate.u_err_en_sync.PrimLcSyncCheckTransients_A);
$assertoff(0, tb.dut.u_tlul_lc_gate.u_err_en_sync.PrimLcSyncCheckTransients0_A);
$assertoff(0, tb.dut.u_tlul_lc_gate.u_err_en_sync.PrimLcSyncCheckTransients1_A);
// These SVA checks the lc_sync_creator_seed_sw_rw_en is either Off or On, we will use more
// than these 2 values.
// If the value is not lc_ctrl_pkg::On, design will treat it as lc_ctrl_pkg::Off.
$assertoff(0, tb.dut.u_prim_lc_sync_creator_seed_sw_rw_en.PrimLcSyncCheckTransients_A);
$assertoff(0, tb.dut.u_prim_lc_sync_creator_seed_sw_rw_en.PrimLcSyncCheckTransients0_A);
$assertoff(0, tb.dut.u_prim_lc_sync_creator_seed_sw_rw_en.PrimLcSyncCheckTransients1_A);
// DV forced otp_cmd_i to reach invalid state, thus violate the assertions
$assertoff(0, tb.dut.gen_partitions[3].gen_buffered.u_part_buf.OtpErrorState_A);
$assertoff(0, tb.dut.gen_partitions[4].gen_buffered.u_part_buf.OtpErrorState_A);
$assertoff(0, tb.dut.gen_partitions[5].gen_buffered.u_part_buf.OtpErrorState_A);
$assertoff(0, tb.dut.gen_partitions[6].gen_buffered.u_part_buf.OtpErrorState_A);
// drive clk and rst_n from clk_if
clk_rst_if.set_active();
uvm_config_db#(virtual clk_rst_if)::set(null, "*.env", "clk_rst_vif", clk_rst_if);
uvm_config_db#(virtual clk_rst_if)::set(null, "*.env",
"clk_rst_vif_otp_ctrl_prim_reg_block", clk_rst_if);
uvm_config_db#(virtual tl_if)::set(null, "*.env.m_tl_agent_otp_ctrl_core_reg_block*",
"vif", tl_if);
uvm_config_db#(virtual tl_if)::set(null, "*.env.m_tl_agent_otp_ctrl_prim_reg_block",
"vif", prim_tl_if);
uvm_config_db#(virtual push_pull_if#(.DeviceDataWidth(OTBN_DATA_SIZE)))::set(null,
"*env.m_otbn_pull_agent*", "vif", otbn_if);
uvm_config_db#(virtual push_pull_if#(.DeviceDataWidth(FLASH_DATA_SIZE)))::set(null,
"*env.m_flash_data_pull_agent*", "vif", flash_data_if);
uvm_config_db#(virtual push_pull_if#(.DeviceDataWidth(FLASH_DATA_SIZE)))::set(null,
"*env.m_flash_addr_pull_agent*", "vif", flash_addr_if);
uvm_config_db#(virtual push_pull_if#(.HostDataWidth(LC_PROG_DATA_SIZE), .DeviceDataWidth(1)))::
set(null, "*env.m_lc_prog_pull_agent*", "vif", lc_prog_if);
uvm_config_db#(intr_vif)::set(null, "*.env", "intr_vif", intr_if);
uvm_config_db#(devmode_vif)::set(null, "*.env", "devmode_vif", devmode_if);
uvm_config_db#(virtual otp_ctrl_if)::set(null, "*.env", "otp_ctrl_vif", otp_ctrl_if);
$timeformat(-12, 0, " ps", 12);
run_test();
end
endmodule