From 82c5b2c74bca988e0c498981ca135d02b72aeb5b Mon Sep 17 00:00:00 2001 From: Michael Schaffner Date: Wed, 17 Jan 2024 17:38:58 -0800 Subject: [PATCH] [sram_ctrl] Add RW1C status for scr key rotation This adds a mubi RW1C CSR that indicates whether a scrambling key has successfully been rotated. Addresses #18663. Signed-off-by: Michael Schaffner --- hw/ip/sram_ctrl/data/sram_ctrl.hjson | 20 + hw/ip/sram_ctrl/doc/registers.md | 42 +- .../sram_ctrl/dv/env/sram_ctrl_scoreboard.sv | 12 + hw/ip/sram_ctrl/rtl/sram_ctrl.sv | 16 +- hw/ip/sram_ctrl/rtl/sram_ctrl_reg_pkg.sv | 17 +- hw/ip/sram_ctrl/rtl/sram_ctrl_reg_top.sv | 493 ------------------ hw/ip/sram_ctrl/rtl/sram_ctrl_regs_reg_top.sv | 49 +- 7 files changed, 134 insertions(+), 515 deletions(-) delete mode 100644 hw/ip/sram_ctrl/rtl/sram_ctrl_reg_top.sv diff --git a/hw/ip/sram_ctrl/data/sram_ctrl.hjson b/hw/ip/sram_ctrl/data/sram_ctrl.hjson index 08e279fb2f354a..2e360593da0a12 100644 --- a/hw/ip/sram_ctrl/data/sram_ctrl.hjson +++ b/hw/ip/sram_ctrl/data/sram_ctrl.hjson @@ -381,6 +381,26 @@ }, ] }, + { name: "SCR_KEY_ROTATED", + desc: "Clearable SRAM key request status.", + swaccess: "rw1c", + hwaccess: "hwo", + fields: [ + { bits: "3:0", + name: "SUCCESS", + mubi: true, + desc: ''' + This status register is similar to !!SCR_KEY_VALID with the difference that the status is multibit encoded, + SW clearable and sticky (i.e., HW does not auto-clear the register except during escalation). That way, + SW can use this for a hardened acknowledgement mechanism where it clears the register before requesting a key. + + kMultiBitBool4True indicates that a valid scrambling key has been obtained from OTP. + Write kMultiBitBool4True to clear. + ''', + resval: false + }, + ] + }, ], ram: [ diff --git a/hw/ip/sram_ctrl/doc/registers.md b/hw/ip/sram_ctrl/doc/registers.md index 6f468e43ac4fd6..0b3657fad2233a 100644 --- a/hw/ip/sram_ctrl/doc/registers.md +++ b/hw/ip/sram_ctrl/doc/registers.md @@ -3,14 +3,15 @@ ## Summary of the **`regs`** interface's registers -| Name | Offset | Length | Description | -|:----------------------------------------|:---------|---------:|:---------------------------------------------| -| sram_ctrl.[`ALERT_TEST`](#alert_test) | 0x0 | 4 | Alert Test Register | -| sram_ctrl.[`STATUS`](#status) | 0x4 | 4 | SRAM status register. | -| sram_ctrl.[`EXEC_REGWEN`](#exec_regwen) | 0x8 | 4 | Lock register for execution enable register. | -| sram_ctrl.[`EXEC`](#exec) | 0xc | 4 | Sram execution enable. | -| sram_ctrl.[`CTRL_REGWEN`](#ctrl_regwen) | 0x10 | 4 | Lock register for control register. | -| sram_ctrl.[`CTRL`](#ctrl) | 0x14 | 4 | SRAM ctrl register. | +| Name | Offset | Length | Description | +|:------------------------------------------------|:---------|---------:|:---------------------------------------------| +| sram_ctrl.[`ALERT_TEST`](#alert_test) | 0x0 | 4 | Alert Test Register | +| sram_ctrl.[`STATUS`](#status) | 0x4 | 4 | SRAM status register. | +| sram_ctrl.[`EXEC_REGWEN`](#exec_regwen) | 0x8 | 4 | Lock register for execution enable register. | +| sram_ctrl.[`EXEC`](#exec) | 0xc | 4 | Sram execution enable. | +| sram_ctrl.[`CTRL_REGWEN`](#ctrl_regwen) | 0x10 | 4 | Lock register for control register. | +| sram_ctrl.[`CTRL`](#ctrl) | 0x14 | 4 | SRAM ctrl register. | +| sram_ctrl.[`SCR_KEY_ROTATED`](#scr_key_rotated) | 0x18 | 4 | Clearable SRAM key request status. | ## ALERT_TEST Alert Test Register @@ -174,5 +175,30 @@ can poll its status. Note that requesting a new scrambling key takes ~200 OTP cy to ~800 CPU cycles (OTP runs at 24MHz, CPU runs at 100MHz). Note that writing 1 to this register while a key request is pending has no effect. +## SCR_KEY_ROTATED +Clearable SRAM key request status. +- Offset: `0x18` +- Reset default: `0x9` +- Reset mask: `0xf` + +### Fields + +```wavejson +{"reg": [{"name": "SUCCESS", "bits": 4, "attr": ["rw1c"], "rotate": -90}, {"bits": 28}], "config": {"lanes": 1, "fontsize": 10, "vspace": 90}} +``` + +| Bits | Type | Reset | Name | +|:------:|:------:|:-------:|:-------------------------------------| +| 31:4 | | | Reserved | +| 3:0 | rw1c | 0x9 | [SUCCESS](#scr_key_rotated--success) | + +### SCR_KEY_ROTATED . SUCCESS +This status register is similar to [`SCR_KEY_VALID`](#scr_key_valid) with the difference that the status is multibit encoded, +SW clearable and sticky (i.e., HW does not auto-clear the register except during escalation). That way, +SW can use this for a hardened acknowledgement mechanism where it clears the register before requesting a key. + +kMultiBitBool4True indicates that a valid scrambling key has been obtained from OTP. +Write kMultiBitBool4True to clear. + This interface does not expose any registers. diff --git a/hw/ip/sram_ctrl/dv/env/sram_ctrl_scoreboard.sv b/hw/ip/sram_ctrl/dv/env/sram_ctrl_scoreboard.sv index a6eb75ea0313c0..f377391e2f8fa4 100644 --- a/hw/ip/sram_ctrl/dv/env/sram_ctrl_scoreboard.sv +++ b/hw/ip/sram_ctrl/dv/env/sram_ctrl_scoreboard.sv @@ -13,6 +13,7 @@ class sram_ctrl_scoreboard #(parameter int AddrWidth = 10) extends cip_base_scor // local variables bit [TL_DW-1:0] exp_status = '0; + mubi4_t exp_scr_key_rotated = MuBi4False; bit in_key_req = 0; @@ -356,6 +357,7 @@ class sram_ctrl_scoreboard #(parameter int AddrWidth = 10) extends cip_base_scor exp_status[SramCtrlScrKeySeedValid] = 0; exp_status[SramCtrlScrKeyValid] = 0; exp_status[SramCtrlInitDone] = 0; + exp_scr_key_rotated = MuBi4False; // escalation resets the key and nonce back to defaults reset_key_nonce(); @@ -439,6 +441,7 @@ class sram_ctrl_scoreboard #(parameter int AddrWidth = 10) extends cip_base_scor // if we are in escalated state, key_valid and scr_key_seed_valid will remain low if (status_lc_esc == EscNone) begin + exp_scr_key_rotated = MuBi4True; exp_status[SramCtrlScrKeyValid] = 1; exp_status[SramCtrlScrKeySeedValid] = seed_valid; end @@ -523,6 +526,11 @@ class sram_ctrl_scoreboard #(parameter int AddrWidth = 10) extends cip_base_scor void'(ral.ctrl.init.predict(.value(0), .kind(UVM_PREDICT_READ))); end end + "scr_key_rotated": begin + if (addr_phase_read) begin + void'(ral.scr_key_rotated.predict(.value(exp_scr_key_rotated), .kind(UVM_PREDICT_READ))); + end + end default: begin `uvm_fatal(`gfn, $sformatf("invalid csr: %0s", csr.get_full_name())) end @@ -541,6 +549,9 @@ class sram_ctrl_scoreboard #(parameter int AddrWidth = 10) extends cip_base_scor mirrored_value[SramCtrlScrKeySeedValid] != item.d_data[SramCtrlScrKeySeedValid])) begin new_key_received = 0; + end else if (csr.get_name() == "scr_key_rotated" && new_key_received && ( + mirrored_value != item.d_data)) begin + new_key_received = 0; end else if (csr.get_name() == "status" && init_after_new_key && mirrored_value[SramCtrlInitDone] != item.d_data[SramCtrlInitDone]) begin @@ -571,6 +582,7 @@ class sram_ctrl_scoreboard #(parameter int AddrWidth = 10) extends cip_base_scor mem_bkdr_scb.reset(); mem_bkdr_scb.update_key(key, nonce); exp_status = '0; + exp_scr_key_rotated = MuBi4False; write_item_q.delete(); exp_mem[cfg.sram_ral_name].init(); diff --git a/hw/ip/sram_ctrl/rtl/sram_ctrl.sv b/hw/ip/sram_ctrl/rtl/sram_ctrl.sv index a0ee7ba9819d10..318f8d4f944179 100644 --- a/hw/ip/sram_ctrl/rtl/sram_ctrl.sv +++ b/hw/ip/sram_ctrl/rtl/sram_ctrl.sv @@ -60,6 +60,11 @@ module sram_ctrl import lc_ctrl_pkg::lc_tx_or_hi; import lc_ctrl_pkg::lc_tx_inv; import lc_ctrl_pkg::lc_to_mubi4; + import prim_mubi_pkg::mubi4_t; + import prim_mubi_pkg::mubi8_t; + import prim_mubi_pkg::MuBi4True; + import prim_mubi_pkg::MuBi4False; + import prim_mubi_pkg::mubi8_test_true_strict; // This is later on pruned to the correct width at the SRAM wrapper interface. parameter int unsigned Depth = MemSizeRam >> 2; @@ -264,6 +269,11 @@ module sram_ctrl assign hw2reg.status.scr_key_valid.d = key_ack & ~key_req & ~local_esc; assign hw2reg.status.scr_key_valid.de = key_req | key_ack | local_esc; + // As opposed to scr_key_valid, SW is responsible for clearing this register. + // It is not automatically cleared by HW, except when escalating. + assign hw2reg.scr_key_rotated.d = (key_ack & ~local_esc) ? MuBi4True : MuBi4False; + assign hw2reg.scr_key_rotated.de = key_ack | local_esc; + // Clear this bit on local escalation. logic key_seed_valid; assign hw2reg.status.scr_key_seed_valid.d = key_seed_valid & ~local_esc; @@ -321,12 +331,6 @@ module sram_ctrl // SRAM Execution // //////////////////// - import prim_mubi_pkg::mubi4_t; - import prim_mubi_pkg::mubi8_t; - import prim_mubi_pkg::MuBi4True; - import prim_mubi_pkg::MuBi4False; - import prim_mubi_pkg::mubi8_test_true_strict; - mubi4_t en_ifetch; if (InstrExec) begin : gen_instr_ctrl lc_tx_t lc_hw_debug_en; diff --git a/hw/ip/sram_ctrl/rtl/sram_ctrl_reg_pkg.sv b/hw/ip/sram_ctrl/rtl/sram_ctrl_reg_pkg.sv index 93e0c043de39ca..97937f0b87e2df 100644 --- a/hw/ip/sram_ctrl/rtl/sram_ctrl_reg_pkg.sv +++ b/hw/ip/sram_ctrl/rtl/sram_ctrl_reg_pkg.sv @@ -82,6 +82,11 @@ package sram_ctrl_reg_pkg; } init_done; } sram_ctrl_hw2reg_status_reg_t; + typedef struct packed { + logic [3:0] d; + logic de; + } sram_ctrl_hw2reg_scr_key_rotated_reg_t; + // Register -> HW type for regs interface typedef struct packed { sram_ctrl_reg2hw_alert_test_reg_t alert_test; // [14:13] @@ -92,7 +97,8 @@ package sram_ctrl_reg_pkg; // HW -> register type for regs interface typedef struct packed { - sram_ctrl_hw2reg_status_reg_t status; // [11:0] + sram_ctrl_hw2reg_status_reg_t status; // [16:5] + sram_ctrl_hw2reg_scr_key_rotated_reg_t scr_key_rotated; // [4:0] } sram_ctrl_regs_hw2reg_t; // Register offsets for regs interface @@ -102,6 +108,7 @@ package sram_ctrl_reg_pkg; parameter logic [RegsAw-1:0] SRAM_CTRL_EXEC_OFFSET = 5'h c; parameter logic [RegsAw-1:0] SRAM_CTRL_CTRL_REGWEN_OFFSET = 5'h 10; parameter logic [RegsAw-1:0] SRAM_CTRL_CTRL_OFFSET = 5'h 14; + parameter logic [RegsAw-1:0] SRAM_CTRL_SCR_KEY_ROTATED_OFFSET = 5'h 18; // Reset values for hwext registers and their fields for regs interface parameter logic [0:0] SRAM_CTRL_ALERT_TEST_RESVAL = 1'h 0; @@ -114,17 +121,19 @@ package sram_ctrl_reg_pkg; SRAM_CTRL_EXEC_REGWEN, SRAM_CTRL_EXEC, SRAM_CTRL_CTRL_REGWEN, - SRAM_CTRL_CTRL + SRAM_CTRL_CTRL, + SRAM_CTRL_SCR_KEY_ROTATED } sram_ctrl_regs_id_e; // Register width information to check illegal writes for regs interface - parameter logic [3:0] SRAM_CTRL_REGS_PERMIT [6] = '{ + parameter logic [3:0] SRAM_CTRL_REGS_PERMIT [7] = '{ 4'b 0001, // index[0] SRAM_CTRL_ALERT_TEST 4'b 0001, // index[1] SRAM_CTRL_STATUS 4'b 0001, // index[2] SRAM_CTRL_EXEC_REGWEN 4'b 0001, // index[3] SRAM_CTRL_EXEC 4'b 0001, // index[4] SRAM_CTRL_CTRL_REGWEN - 4'b 0001 // index[5] SRAM_CTRL_CTRL + 4'b 0001, // index[5] SRAM_CTRL_CTRL + 4'b 0001 // index[6] SRAM_CTRL_SCR_KEY_ROTATED }; endpackage diff --git a/hw/ip/sram_ctrl/rtl/sram_ctrl_reg_top.sv b/hw/ip/sram_ctrl/rtl/sram_ctrl_reg_top.sv deleted file mode 100644 index 0ddc50a05e521e..00000000000000 --- a/hw/ip/sram_ctrl/rtl/sram_ctrl_reg_top.sv +++ /dev/null @@ -1,493 +0,0 @@ -// Copyright lowRISC contributors. -// Licensed under the Apache License, Version 2.0, see LICENSE for details. -// SPDX-License-Identifier: Apache-2.0 -// -// Register Top module auto-generated by `reggen` - -`include "prim_assert.sv" - -module sram_ctrl_reg_top ( - input clk_i, - input rst_ni, - - input tlul_pkg::tl_h2d_t tl_i, - output tlul_pkg::tl_d2h_t tl_o, - // To HW - output sram_ctrl_reg_pkg::sram_ctrl_reg2hw_t reg2hw, // Write - input sram_ctrl_reg_pkg::sram_ctrl_hw2reg_t hw2reg, // Read - - // Integrity check errors - output logic intg_err_o -); - - import sram_ctrl_reg_pkg::* ; - - localparam int AW = 5; - localparam int DW = 32; - localparam int DBW = DW/8; // Byte Width - - // register signals - logic reg_we; - logic reg_re; - logic [AW-1:0] reg_addr; - logic [DW-1:0] reg_wdata; - logic [DBW-1:0] reg_be; - logic [DW-1:0] reg_rdata; - logic reg_error; - - logic addrmiss, wr_err; - - logic [DW-1:0] reg_rdata_next; - logic reg_busy; - - tlul_pkg::tl_h2d_t tl_reg_h2d; - tlul_pkg::tl_d2h_t tl_reg_d2h; - - - // incoming payload check - logic intg_err; - tlul_cmd_intg_chk u_chk ( - .tl_i(tl_i), - .err_o(intg_err) - ); - - logic intg_err_q; - always_ff @(posedge clk_i or negedge rst_ni) begin - if (!rst_ni) begin - intg_err_q <= '0; - end else if (intg_err) begin - intg_err_q <= 1'b1; - end - end - - // integrity error output is permanent and should be used for alert generation - // register errors are transactional - assign intg_err_o = intg_err_q | intg_err; - - // outgoing integrity generation - tlul_pkg::tl_d2h_t tl_o_pre; - tlul_rsp_intg_gen #( - .EnableRspIntgGen(1), - .EnableDataIntgGen(1) - ) u_rsp_intg_gen ( - .tl_i(tl_o_pre), - .tl_o(tl_o) - ); - - assign tl_reg_h2d = tl_i; - assign tl_o_pre = tl_reg_d2h; - - tlul_adapter_reg #( - .RegAw(AW), - .RegDw(DW), - .EnableDataIntgGen(0) - ) u_reg_if ( - .clk_i (clk_i), - .rst_ni (rst_ni), - - .tl_i (tl_reg_h2d), - .tl_o (tl_reg_d2h), - - .we_o (reg_we), - .re_o (reg_re), - .addr_o (reg_addr), - .wdata_o (reg_wdata), - .be_o (reg_be), - .busy_i (reg_busy), - .rdata_i (reg_rdata), - .error_i (reg_error) - ); - - // cdc oversampling signals - - assign reg_rdata = reg_rdata_next ; - assign reg_error = addrmiss | wr_err | intg_err; - - // Define SW related signals - // Format: __{wd|we|qs} - // or _{wd|we|qs} if field == 1 or 0 - logic alert_test_we; - logic alert_test_fatal_intg_error_wd; - logic alert_test_fatal_parity_error_wd; - logic status_re; - logic status_error_qs; - logic status_escalated_qs; - logic status_scr_key_valid_qs; - logic status_scr_key_seed_valid_qs; - logic exec_regwen_we; - logic exec_regwen_qs; - logic exec_regwen_wd; - logic exec_we; - logic [2:0] exec_qs; - logic [2:0] exec_wd; - logic ctrl_regwen_we; - logic ctrl_regwen_qs; - logic ctrl_regwen_wd; - logic ctrl_re; - logic ctrl_we; - logic ctrl_renew_scr_key_qs; - logic ctrl_renew_scr_key_wd; - logic ctrl_init_qs; - logic ctrl_init_wd; - logic [31:0] error_address_qs; - - // Register instances - // R[alert_test]: V(True) - - // F[fatal_intg_error]: 0:0 - prim_subreg_ext #( - .DW (1) - ) u_alert_test_fatal_intg_error ( - .re (1'b0), - .we (alert_test_we), - .wd (alert_test_fatal_intg_error_wd), - .d ('0), - .qre (), - .qe (reg2hw.alert_test.fatal_intg_error.qe), - .q (reg2hw.alert_test.fatal_intg_error.q), - .qs () - ); - - - // F[fatal_parity_error]: 1:1 - prim_subreg_ext #( - .DW (1) - ) u_alert_test_fatal_parity_error ( - .re (1'b0), - .we (alert_test_we), - .wd (alert_test_fatal_parity_error_wd), - .d ('0), - .qre (), - .qe (reg2hw.alert_test.fatal_parity_error.qe), - .q (reg2hw.alert_test.fatal_parity_error.q), - .qs () - ); - - - // R[status]: V(True) - - // F[error]: 0:0 - prim_subreg_ext #( - .DW (1) - ) u_status_error ( - .re (status_re), - .we (1'b0), - .wd ('0), - .d (hw2reg.status.error.d), - .qre (), - .qe (), - .q (), - .qs (status_error_qs) - ); - - - // F[escalated]: 1:1 - prim_subreg_ext #( - .DW (1) - ) u_status_escalated ( - .re (status_re), - .we (1'b0), - .wd ('0), - .d (hw2reg.status.escalated.d), - .qre (), - .qe (), - .q (), - .qs (status_escalated_qs) - ); - - - // F[scr_key_valid]: 2:2 - prim_subreg_ext #( - .DW (1) - ) u_status_scr_key_valid ( - .re (status_re), - .we (1'b0), - .wd ('0), - .d (hw2reg.status.scr_key_valid.d), - .qre (), - .qe (), - .q (), - .qs (status_scr_key_valid_qs) - ); - - - // F[scr_key_seed_valid]: 3:3 - prim_subreg_ext #( - .DW (1) - ) u_status_scr_key_seed_valid ( - .re (status_re), - .we (1'b0), - .wd ('0), - .d (hw2reg.status.scr_key_seed_valid.d), - .qre (), - .qe (), - .q (), - .qs (status_scr_key_seed_valid_qs) - ); - - - // R[exec_regwen]: V(False) - - prim_subreg #( - .DW (1), - .SwAccess(prim_subreg_pkg::SwAccessW0C), - .RESVAL (1'h1) - ) u_exec_regwen ( - .clk_i (clk_i), - .rst_ni (rst_ni), - - // from register interface - .we (exec_regwen_we), - .wd (exec_regwen_wd), - - // from internal hardware - .de (1'b0), - .d ('0), - - // to internal hardware - .qe (), - .q (), - - // to register interface (read) - .qs (exec_regwen_qs) - ); - - - // R[exec]: V(False) - - prim_subreg #( - .DW (3), - .SwAccess(prim_subreg_pkg::SwAccessRW), - .RESVAL (3'h0) - ) u_exec ( - .clk_i (clk_i), - .rst_ni (rst_ni), - - // from register interface - .we (exec_we & exec_regwen_qs), - .wd (exec_wd), - - // from internal hardware - .de (1'b0), - .d ('0), - - // to internal hardware - .qe (), - .q (reg2hw.exec.q), - - // to register interface (read) - .qs (exec_qs) - ); - - - // R[ctrl_regwen]: V(False) - - prim_subreg #( - .DW (1), - .SwAccess(prim_subreg_pkg::SwAccessW0C), - .RESVAL (1'h1) - ) u_ctrl_regwen ( - .clk_i (clk_i), - .rst_ni (rst_ni), - - // from register interface - .we (ctrl_regwen_we), - .wd (ctrl_regwen_wd), - - // from internal hardware - .de (1'b0), - .d ('0), - - // to internal hardware - .qe (), - .q (), - - // to register interface (read) - .qs (ctrl_regwen_qs) - ); - - - // R[ctrl]: V(True) - - // F[renew_scr_key]: 0:0 - prim_subreg_ext #( - .DW (1) - ) u_ctrl_renew_scr_key ( - .re (ctrl_re), - .we (ctrl_we & ctrl_regwen_qs), - .wd (ctrl_renew_scr_key_wd), - .d (hw2reg.ctrl.renew_scr_key.d), - .qre (), - .qe (reg2hw.ctrl.renew_scr_key.qe), - .q (reg2hw.ctrl.renew_scr_key.q), - .qs (ctrl_renew_scr_key_qs) - ); - - - // F[init]: 1:1 - prim_subreg_ext #( - .DW (1) - ) u_ctrl_init ( - .re (ctrl_re), - .we (ctrl_we & ctrl_regwen_qs), - .wd (ctrl_init_wd), - .d (hw2reg.ctrl.init.d), - .qre (), - .qe (reg2hw.ctrl.init.qe), - .q (reg2hw.ctrl.init.q), - .qs (ctrl_init_qs) - ); - - - // R[error_address]: V(False) - - prim_subreg #( - .DW (32), - .SwAccess(prim_subreg_pkg::SwAccessRO), - .RESVAL (32'h0) - ) u_error_address ( - .clk_i (clk_i), - .rst_ni (rst_ni), - - // from register interface - .we (1'b0), - .wd ('0), - - // from internal hardware - .de (hw2reg.error_address.de), - .d (hw2reg.error_address.d), - - // to internal hardware - .qe (), - .q (), - - // to register interface (read) - .qs (error_address_qs) - ); - - - - - logic [6:0] addr_hit; - always_comb begin - addr_hit = '0; - addr_hit[0] = (reg_addr == SRAM_CTRL_ALERT_TEST_OFFSET); - addr_hit[1] = (reg_addr == SRAM_CTRL_STATUS_OFFSET); - addr_hit[2] = (reg_addr == SRAM_CTRL_EXEC_REGWEN_OFFSET); - addr_hit[3] = (reg_addr == SRAM_CTRL_EXEC_OFFSET); - addr_hit[4] = (reg_addr == SRAM_CTRL_CTRL_REGWEN_OFFSET); - addr_hit[5] = (reg_addr == SRAM_CTRL_CTRL_OFFSET); - addr_hit[6] = (reg_addr == SRAM_CTRL_ERROR_ADDRESS_OFFSET); - end - - assign addrmiss = (reg_re || reg_we) ? ~|addr_hit : 1'b0 ; - - // Check sub-word write is permitted - always_comb begin - wr_err = (reg_we & - ((addr_hit[0] & (|(SRAM_CTRL_PERMIT[0] & ~reg_be))) | - (addr_hit[1] & (|(SRAM_CTRL_PERMIT[1] & ~reg_be))) | - (addr_hit[2] & (|(SRAM_CTRL_PERMIT[2] & ~reg_be))) | - (addr_hit[3] & (|(SRAM_CTRL_PERMIT[3] & ~reg_be))) | - (addr_hit[4] & (|(SRAM_CTRL_PERMIT[4] & ~reg_be))) | - (addr_hit[5] & (|(SRAM_CTRL_PERMIT[5] & ~reg_be))) | - (addr_hit[6] & (|(SRAM_CTRL_PERMIT[6] & ~reg_be))))); - end - assign alert_test_we = addr_hit[0] & reg_we & !reg_error; - - assign alert_test_fatal_intg_error_wd = reg_wdata[0]; - - assign alert_test_fatal_parity_error_wd = reg_wdata[1]; - assign status_re = addr_hit[1] & reg_re & !reg_error; - assign exec_regwen_we = addr_hit[2] & reg_we & !reg_error; - - assign exec_regwen_wd = reg_wdata[0]; - assign exec_we = addr_hit[3] & reg_we & !reg_error; - - assign exec_wd = reg_wdata[2:0]; - assign ctrl_regwen_we = addr_hit[4] & reg_we & !reg_error; - - assign ctrl_regwen_wd = reg_wdata[0]; - assign ctrl_re = addr_hit[5] & reg_re & !reg_error; - assign ctrl_we = addr_hit[5] & reg_we & !reg_error; - - assign ctrl_renew_scr_key_wd = reg_wdata[0]; - - assign ctrl_init_wd = reg_wdata[1]; - - // Read data return - always_comb begin - reg_rdata_next = '0; - unique case (1'b1) - addr_hit[0]: begin - reg_rdata_next[0] = '0; - reg_rdata_next[1] = '0; - end - - addr_hit[1]: begin - reg_rdata_next[0] = status_error_qs; - reg_rdata_next[1] = status_escalated_qs; - reg_rdata_next[2] = status_scr_key_valid_qs; - reg_rdata_next[3] = status_scr_key_seed_valid_qs; - end - - addr_hit[2]: begin - reg_rdata_next[0] = exec_regwen_qs; - end - - addr_hit[3]: begin - reg_rdata_next[2:0] = exec_qs; - end - - addr_hit[4]: begin - reg_rdata_next[0] = ctrl_regwen_qs; - end - - addr_hit[5]: begin - reg_rdata_next[0] = ctrl_renew_scr_key_qs; - reg_rdata_next[1] = ctrl_init_qs; - end - - addr_hit[6]: begin - reg_rdata_next[31:0] = error_address_qs; - end - - default: begin - reg_rdata_next = '1; - end - endcase - end - - // register busy - always_comb begin - reg_busy = '0; - unique case (1'b1) - default: begin - reg_busy = '0; - end - endcase - end - - - // Unused signal tieoff - - // wdata / byte enable are not always fully used - // add a blanket unused statement to handle lint waivers - logic unused_wdata; - logic unused_be; - assign unused_wdata = ^reg_wdata; - assign unused_be = ^reg_be; - - // Assertions for Register Interface - `ASSERT_PULSE(wePulse, reg_we, clk_i, !rst_ni) - `ASSERT_PULSE(rePulse, reg_re, clk_i, !rst_ni) - - `ASSERT(reAfterRv, $rose(reg_re || reg_we) |=> tl_o_pre.d_valid, clk_i, !rst_ni) - - `ASSERT(en2addrHit, (reg_we || reg_re) |-> $onehot0(addr_hit), clk_i, !rst_ni) - - // this is formulated as an assumption such that the FPV testbenches do disprove this - // property by mistake - //`ASSUME(reqParity, tl_reg_h2d.a_valid |-> tl_reg_h2d.a_user.chk_en == tlul_pkg::CheckDis) - -endmodule diff --git a/hw/ip/sram_ctrl/rtl/sram_ctrl_regs_reg_top.sv b/hw/ip/sram_ctrl/rtl/sram_ctrl_regs_reg_top.sv index 08962a6d207aab..91ccf158a26e48 100644 --- a/hw/ip/sram_ctrl/rtl/sram_ctrl_regs_reg_top.sv +++ b/hw/ip/sram_ctrl/rtl/sram_ctrl_regs_reg_top.sv @@ -52,9 +52,9 @@ module sram_ctrl_regs_reg_top ( // also check for spurious write enables logic reg_we_err; - logic [5:0] reg_we_check; + logic [6:0] reg_we_check; prim_reg_we_check #( - .OneHotWidth(6) + .OneHotWidth(7) ) u_prim_reg_we_check ( .clk_i(clk_i), .rst_ni(rst_ni), @@ -141,6 +141,9 @@ module sram_ctrl_regs_reg_top ( logic ctrl_we; logic ctrl_renew_scr_key_wd; logic ctrl_init_wd; + logic scr_key_rotated_we; + logic [3:0] scr_key_rotated_qs; + logic [3:0] scr_key_rotated_wd; // Register instances // R[alert_test]: V(True) @@ -486,8 +489,36 @@ module sram_ctrl_regs_reg_top ( assign reg2hw.ctrl.init.qe = ctrl_qe; + // R[scr_key_rotated]: V(False) + prim_subreg #( + .DW (4), + .SwAccess(prim_subreg_pkg::SwAccessW1C), + .RESVAL (4'h9), + .Mubi (1'b1) + ) u_scr_key_rotated ( + .clk_i (clk_i), + .rst_ni (rst_ni), - logic [5:0] addr_hit; + // from register interface + .we (scr_key_rotated_we), + .wd (scr_key_rotated_wd), + + // from internal hardware + .de (hw2reg.scr_key_rotated.de), + .d (hw2reg.scr_key_rotated.d), + + // to internal hardware + .qe (), + .q (), + .ds (), + + // to register interface (read) + .qs (scr_key_rotated_qs) + ); + + + + logic [6:0] addr_hit; always_comb begin addr_hit = '0; addr_hit[0] = (reg_addr == SRAM_CTRL_ALERT_TEST_OFFSET); @@ -496,6 +527,7 @@ module sram_ctrl_regs_reg_top ( addr_hit[3] = (reg_addr == SRAM_CTRL_EXEC_OFFSET); addr_hit[4] = (reg_addr == SRAM_CTRL_CTRL_REGWEN_OFFSET); addr_hit[5] = (reg_addr == SRAM_CTRL_CTRL_OFFSET); + addr_hit[6] = (reg_addr == SRAM_CTRL_SCR_KEY_ROTATED_OFFSET); end assign addrmiss = (reg_re || reg_we) ? ~|addr_hit : 1'b0 ; @@ -508,7 +540,8 @@ module sram_ctrl_regs_reg_top ( (addr_hit[2] & (|(SRAM_CTRL_REGS_PERMIT[2] & ~reg_be))) | (addr_hit[3] & (|(SRAM_CTRL_REGS_PERMIT[3] & ~reg_be))) | (addr_hit[4] & (|(SRAM_CTRL_REGS_PERMIT[4] & ~reg_be))) | - (addr_hit[5] & (|(SRAM_CTRL_REGS_PERMIT[5] & ~reg_be))))); + (addr_hit[5] & (|(SRAM_CTRL_REGS_PERMIT[5] & ~reg_be))) | + (addr_hit[6] & (|(SRAM_CTRL_REGS_PERMIT[6] & ~reg_be))))); end // Generate write-enables @@ -529,6 +562,9 @@ module sram_ctrl_regs_reg_top ( assign ctrl_renew_scr_key_wd = reg_wdata[0]; assign ctrl_init_wd = reg_wdata[1]; + assign scr_key_rotated_we = addr_hit[6] & reg_we & !reg_error; + + assign scr_key_rotated_wd = reg_wdata[3:0]; // Assign write-enables to checker logic vector. always_comb begin @@ -539,6 +575,7 @@ module sram_ctrl_regs_reg_top ( reg_we_check[3] = exec_gated_we; reg_we_check[4] = ctrl_regwen_we; reg_we_check[5] = ctrl_gated_we; + reg_we_check[6] = scr_key_rotated_we; end // Read data return @@ -575,6 +612,10 @@ module sram_ctrl_regs_reg_top ( reg_rdata_next[1] = '0; end + addr_hit[6]: begin + reg_rdata_next[3:0] = scr_key_rotated_qs; + end + default: begin reg_rdata_next = '1; end