blob: a1c1df17199779a9aec4a5c6f0890068f77b40cb [file] [log] [blame]
// Copyright lowRISC contributors.
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0
//
// Hardened OR operation for life cycle values. To be used instead of lc_tx_or_*() if the output
// signal must be ActVal iff one of the two input signals is strictly ActVal.
//
`include "prim_assert.sv"
module prim_lc_or_hardened
import lc_ctrl_pkg::*;
#(
lc_tx_t ActVal = On
) (
// Clock is only used for SVAs in this module.
input clk_i,
input rst_ni,
input lc_tx_t lc_en_a_i,
input lc_tx_t lc_en_b_i,
output lc_tx_t lc_en_o
);
// For multibit or'ing we need to be careful if the resulting signal is consumed with
// lc_tx_test_*_strict(). I.e., due to the nature of the bitwise lc_tx_or_*() operation, two
// non-strictly ActVal values can produce a strictly ActVal value (this is not possible with the
// lc_tx_and_*() operation since there is only one strict ActVal output value in the truth table).
//
// To this end, we perform strict comparisons and make sure we only output ActVal if one of the
// two inputs is strictly ActVal. The comparisons are done redundantly so that the multibit
// aspect is preserved throughout instead of creating a single point of failure due to the
// reduction.
lc_tx_t [TxWidth-1:0] lc_en_a_copies;
prim_lc_sync #(
.NumCopies(TxWidth),
.AsyncOn(0) // no sync needed
) u_prim_lc_sync_a (
.clk_i,
.rst_ni,
.lc_en_i(lc_en_a_i),
.lc_en_o(lc_en_a_copies)
);
lc_tx_t [TxWidth-1:0] lc_en_b_copies;
prim_lc_sync #(
.NumCopies(TxWidth),
.AsyncOn(0) // no sync needed
) u_prim_lc_sync_b (
.clk_i,
.rst_ni,
.lc_en_i(lc_en_b_i),
.lc_en_o(lc_en_b_copies)
);
logic [TxWidth-1:0] lc_en_logic;
for (genvar k = 0; k < TxWidth; k++) begin : gen_hardened_or
assign lc_en_logic[k] = (lc_en_a_copies[k] == ActVal) || (lc_en_b_copies[k] == ActVal);
end
// So far all comparisons above produce the same value in lc_en_logic.
// X'oring with the inverse active value will flip the bits that need to be inverted.
assign lc_en_o = lc_tx_t'(lc_en_logic ^ lc_tx_inv(ActVal));
////////////////
// Assertions //
////////////////
// The outputs should be known at all times.
`ASSERT_KNOWN(OutputsKnown_A, lc_en_o)
`ASSERT(FunctionCheck_A, ((lc_en_a_i == ActVal) || (lc_en_b_i == ActVal)) == (lc_en_o == ActVal))
endmodule : prim_lc_or_hardened