blob: c4ef5e321497a16ab532fc824a6d0b4a5b77757e [file] [log] [blame]
// Copyright lowRISC contributors.
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0
//
// REQ/ACK synchronizer with associated data.
//
// This module synchronizes a REQ/ACK handshake with associated data across a clock domain
// crossing (CDC). Both domains will see a handshake with the duration of one clock cycle. By
// default, the data itself is not registered. The main purpose of feeding the data through this
// module to have an anchor point for waiving CDC violations. If the data is configured to flow
// from the destination (DST) to the source (SRC) domain, an additional register stage can be
// inserted for data buffering.
//
// Under the hood, this module uses a prim_sync_reqack primitive for synchronizing the
// REQ/ACK handshake. See prim_sync_reqack.sv for more details.
`include "prim_assert.sv"
module prim_sync_reqack_data #(
parameter int unsigned Width = 1,
parameter bit EnRstChks = 1'b0, // Enable reset-related assertion checks, disabled by
// default.
parameter bit DataSrc2Dst = 1'b1, // Direction of data flow: 1'b1 = SRC to DST,
// 1'b0 = DST to SRC
parameter bit DataReg = 1'b0, // Enable optional register stage for data,
// only usable with DataSrc2Dst == 1'b0.
parameter bit EnRzHs = 1'b0 // By Default, we the faster NRZ handshake protocol
// (EnRzHs = 0) is used. Enable the RZ handshake
// protocol if the FSMs need to be partial-reset-safe.
) (
input clk_src_i, // REQ side, SRC domain
input rst_src_ni, // REQ side, SRC domain
input clk_dst_i, // ACK side, DST domain
input rst_dst_ni, // ACK side, DST domain
input logic req_chk_i, // Used for gating assertions. Drive to 1 during normal operation.
input logic src_req_i, // REQ side, SRC domain
output logic src_ack_o, // REQ side, SRC domain
output logic dst_req_o, // ACK side, DST domain
input logic dst_ack_i, // ACK side, DST domain
input logic [Width-1:0] data_i,
output logic [Width-1:0] data_o
);
////////////////////////////////////
// REQ/ACK synchronizer primitive //
////////////////////////////////////
prim_sync_reqack #(
.EnRstChks(EnRstChks),
.EnRzHs(EnRzHs)
) u_prim_sync_reqack (
.clk_src_i,
.rst_src_ni,
.clk_dst_i,
.rst_dst_ni,
.req_chk_i,
.src_req_i,
.src_ack_o,
.dst_req_o,
.dst_ack_i
);
/////////////////////////
// Data register stage //
/////////////////////////
// Optional - Only relevant if the data flows from DST to SRC. In this case, it must be ensured
// that the data remains stable until the ACK becomes visible in the SRC domain.
//
// Note that for larger data widths, it is recommended to adjust the data sender to hold the data
// stable until the next REQ in order to save the cost of this register stage.
if (DataSrc2Dst == 1'b0 && DataReg == 1'b1) begin : gen_data_reg
logic data_we;
logic [Width-1:0] data_d, data_q;
// Sample the data when seing the REQ/ACK handshake in the DST domain.
assign data_we = dst_req_o & dst_ack_i;
assign data_d = data_i;
always_ff @(posedge clk_dst_i or negedge rst_dst_ni) begin
if (!rst_dst_ni) begin
data_q <= '0;
end else if (data_we) begin
data_q <= data_d;
end
end
assign data_o = data_q;
end else begin : gen_no_data_reg
// Just feed through the data.
assign data_o = data_i;
end
////////////////
// Assertions //
////////////////
if (DataSrc2Dst == 1'b1) begin : gen_assert_data_src2dst
`ifdef INC_ASSERT
//VCS coverage off
// pragma coverage off
logic effective_rst_n;
assign effective_rst_n = rst_src_ni && rst_dst_ni;
logic chk_flag_d, chk_flag_q;
assign chk_flag_d = src_req_i && !chk_flag_q ? 1'b1 : chk_flag_q;
always_ff @(posedge clk_src_i or negedge effective_rst_n) begin
if (!effective_rst_n) begin
chk_flag_q <= '0;
end else begin
chk_flag_q <= chk_flag_d;
end
end
//VCS coverage on
// pragma coverage on
// SRC domain cannot change data while waiting for ACK.
`ASSERT(SyncReqAckDataHoldSrc2Dst, !$stable(data_i) && chk_flag_q |->
(!src_req_i || (src_req_i && src_ack_o)),
clk_src_i, !rst_src_ni || !rst_dst_ni || !chk_flag_q)
// Register stage cannot be used.
`ASSERT_INIT(SyncReqAckDataReg, DataSrc2Dst && !DataReg)
`endif
end else if (DataSrc2Dst == 1'b0 && DataReg == 1'b0) begin : gen_assert_data_dst2src
// DST domain shall not change data while waiting for SRC domain to latch it (together with
// receiving ACK). It takes 2 SRC cycles for ACK to cross over from DST to SRC, and 1 SRC cycle
// for the next REQ to cross over from SRC to DST.
//
// Denote the src clock where REQ & ACK as time zero. The data flowing through the CDC could be
// corrupted if data_o was not stable over the previous 2 clock cycles (so we want to check time
// points -2, -1 and 0). Moreover, the DST domain cannot know that it is allowed to change value
// until at least one more SRC cycle (the time taken for REQ to cross back from SRC to DST).
//
// To make this assertion, we will sample at each of 4 time points (-2, -1, 0 and +1), asserting
// that data_o is equal at each of these times. Note this won't detect glitches at intermediate
// timepoints.
//
// The SVAs below are designed not to consume time, which means that they can be disabled with
// an $assertoff(..) and won't linger to fail later. This wouldn't work properly if we used
// something like |=> instead of the $past(...) function. That means that we have to trigger at
// the "end" of the window. To make sure we don't miss a situation where the value changed at
// time -1 (causing corruption), but reset was asserted between time 0 and 1, we split the
// assertion into two pieces. The first (SyncReqAckDataHoldDst2SrcA) checks that data doesn't
// change in a way that could cause data corruption. The second (SyncReqAckDataHoldDst2SrcB)
// checks that the DST side doesn't do anything that it shouldn't know is safe.
`ifdef INC_ASSERT
//VCS coverage off
// pragma coverage off
logic effective_rst_n;
assign effective_rst_n = rst_src_ni && rst_dst_ni;
logic chk_flag_d, chk_flag_q;
assign chk_flag_d = src_req_i && !chk_flag_q ? 1'b1 : chk_flag_q;
always_ff @(posedge clk_src_i or negedge effective_rst_n) begin
if (!effective_rst_n) begin
chk_flag_q <= '0;
end else begin
chk_flag_q <= chk_flag_d;
end
end
//VCS coverage on
// pragma coverage on
`ASSERT(SyncReqAckDataHoldDst2SrcA,
chk_flag_q && src_req_i && src_ack_o |->
$past(data_o, 2) == data_o && $past(data_o) == data_o,
clk_src_i, !rst_src_ni || !rst_dst_ni || !chk_flag_q)
`ASSERT(SyncReqAckDataHoldDst2SrcB,
chk_flag_q && $past(src_req_i && src_ack_o) |-> $past(data_o) == data_o,
clk_src_i, !rst_src_ni || !rst_dst_ni || !chk_flag_q)
`endif
end
endmodule