blob: f323100b5f9979a3f5b6e9fc51d0f7f702f6590b [file] [log] [blame]
// Copyright lowRISC contributors.
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0
//
// Component handling register CDC
//
// Currently, this module only works correctly when paired with tlul_adapter_reg.
// This is because tlul_adapter_reg does not emit a new transaction to the same
// register if it discovers it is currently busy. Please see the BusySrcReqChk_A
// assertion below for more details.
//
// If in the future this assumption changes, we can modify this module easily to
// support the new behavior.
`include "prim_assert.sv"
module prim_reg_cdc #(
parameter int DataWidth = 32,
parameter logic [DataWidth-1:0] ResetVal = 32'h0,
parameter logic [DataWidth-1:0] BitMask = 32'hFFFFFFFF,
// whether this instance needs to support independent hardware writes
parameter bit DstWrReq = 0
) (
input clk_src_i,
input rst_src_ni,
input clk_dst_i,
input rst_dst_ni,
input src_regwen_i,
input src_we_i,
input src_re_i,
input [DataWidth-1:0] src_wd_i,
output logic src_busy_o,
output logic [DataWidth-1:0] src_qs_o,
input [DataWidth-1:0] dst_ds_i,
input [DataWidth-1:0] dst_qs_i,
input dst_update_i,
output logic dst_we_o,
output logic dst_re_o,
output logic dst_regwen_o,
output logic [DataWidth-1:0] dst_wd_o
);
////////////////////////////
// Source domain
////////////////////////////
localparam int TxnWidth = 3;
logic src_ack;
logic src_busy_q;
logic [DataWidth-1:0] src_q;
logic [TxnWidth-1:0] txn_bits_q;
logic src_req;
assign src_req = src_we_i | src_re_i;
// busy indication back-pressures upstream if the register is accessed
// again. The busy indication is also used as a "commit" indication for
// resolving software and hardware write conflicts
always_ff @(posedge clk_src_i or negedge rst_src_ni) begin
if (!rst_src_ni) begin
src_busy_q <= '0;
end else if (src_req) begin
src_busy_q <= 1'b1;
end else if (src_ack) begin
src_busy_q <= 1'b0;
end
end
// A src_ack should only be sent if there was a src_req.
// src_busy_q asserts whenever there is a src_req. By association,
// whenever src_ack is seen, then src_busy must be high.
`ASSERT(SrcAckBusyChk_A, src_ack |-> src_busy_q, clk_src_i, !rst_src_ni)
assign src_busy_o = src_busy_q;
// src_q acts as both the write holding register and the software read back
// register.
// When software performs a write, the write data is captured in src_q for
// CDC purposes. When not performing a write, the src_q reflects the most recent
// hardware value. For registes with no hardware access, this is simply the
// the value programmed by software (or in the case R1C, W1C etc) the value after
// the operation. For registers with hardware access, this reflects a potentially
// delayed version of the real value, as the software facing updates lag real
// time updates.
//
// To resolve software and hardware conflicts, the process is as follows:
// When software issues a write, this module asserts "busy". While busy,
// src_q does not take on destination value updates. Since the
// logic has committed to updating based on software command, there is an irreversible
// window from which hardware writes are ignored. Once the busy window completes,
// the cdc portion then begins sampling once more.
//
// This is consistent with prim_subreg_arb where during software / hardware conflicts,
// software is always prioritized. The main difference is the conflict resolution window
// is now larger instead of just one destination clock cycle.
logic busy;
assign busy = src_busy_q & !src_ack;
// This is the current destination value
logic [DataWidth-1:0] dst_qs;
logic src_update;
always_ff @(posedge clk_src_i or negedge rst_src_ni) begin
if (!rst_src_ni) begin
src_q <= ResetVal;
txn_bits_q <= '0;
end else if (src_req) begin
// See assertion below
// At the beginning of a software initiated transaction, the following
// values are captured in the src_q/txn_bits_q flops to ensure they cannot
// change for the duration of the synchronization operation.
src_q <= src_wd_i & BitMask;
txn_bits_q <= {src_we_i, src_re_i, src_regwen_i};
end else if (src_busy_q && src_ack || src_update && !busy) begin
// sample data whenever a busy transaction finishes OR
// when an update pulse is seen.
// TODO: We should add a cover group to test different sync timings
// between src_ack and src_update. Ie, there can be 3 scearios:
// 1. update one cycle before ack
// 2. ack one cycle before update
// 3. update / ack on the same cycle
// During all 3 cases the read data should be correct
src_q <= dst_qs;
txn_bits_q <= '0;
end
end
// The current design (tlul_adapter_reg) does not spit out a request if the destination it chooses
// (decoded from address) is busy. So this creates a situation in the current design where
// src_req_i and busy can never be high at the same time.
// While the code above could be coded directly to be expressed as `src_req & !busy`, which makes
// the intent clearer, it ends up causing coverage holes from the tool's perspective since that
// condition cannot be met.
// Thus we add an assertion here to ensure the condition is always satisfied.
`ASSERT(BusySrcReqChk_A, busy |-> !src_req, clk_src_i, !rst_src_ni)
// reserved bits are not used
logic unused_wd;
assign unused_wd = ^src_wd_i;
// src_q is always updated in the clk_src domain.
// when performing an update to the destination domain, it is guaranteed
// to not change by protocol.
assign src_qs_o = src_q;
assign dst_wd_o = src_q;
////////////////////////////
// CDC handling
////////////////////////////
logic dst_req_from_src;
logic dst_req;
// the software transaction is pulse synced across the domain.
// the prim_reg_cdc_arb module handles conflicts with ongoing hardware updates.
prim_pulse_sync u_src_to_dst_req (
.clk_src_i,
.rst_src_ni,
.clk_dst_i,
.rst_dst_ni,
.src_pulse_i(src_req),
.dst_pulse_o(dst_req_from_src)
);
prim_reg_cdc_arb #(
.DataWidth(DataWidth),
.ResetVal(ResetVal),
.DstWrReq(DstWrReq)
) u_arb (
.clk_src_i,
.rst_src_ni,
.clk_dst_i,
.rst_dst_ni,
.src_ack_o(src_ack),
.src_update_o(src_update),
.dst_req_i(dst_req_from_src),
.dst_req_o(dst_req),
.dst_update_i,
.dst_ds_i,
.dst_qs_i,
.dst_qs_o(dst_qs)
);
// Each is valid only when destination request pulse is high
assign {dst_we_o, dst_re_o, dst_regwen_o} = txn_bits_q & {TxnWidth{dst_req}};
`ASSERT_KNOWN(SrcBusyKnown_A, src_busy_o, clk_src_i, !rst_src_ni)
`ASSERT_KNOWN(DstReqKnown_A, dst_req, clk_dst_i, !rst_dst_ni)
// If busy goes high, we must eventually see an ack
`ifdef FPV_ON
`ASSERT(HungHandShake_A, $rose(src_req) |-> strong(##[0:$] src_ack), clk_src_i, !rst_src_ni)
// TODO: #14913 check if we can add additional sim assertions.
`endif
endmodule // prim_subreg_cdc