| // 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 |