[prim_mubi] Add mubi sender and sync primitives
Signed-off-by: Michael Schaffner <msf@google.com>
diff --git a/hw/ip/prim/rtl/prim_mubi8_sync.sv b/hw/ip/prim/rtl/prim_mubi8_sync.sv
new file mode 100644
index 0000000..157c709
--- /dev/null
+++ b/hw/ip/prim/rtl/prim_mubi8_sync.sv
@@ -0,0 +1,97 @@
+// Copyright lowRISC contributors.
+// Licensed under the Apache License, Version 2.0, see LICENSE for details.
+// SPDX-License-Identifier: Apache-2.0
+//
+// ------------------- W A R N I N G: A U T O - G E N E R A T E D C O D E !! -------------------//
+// PLEASE DO NOT HAND-EDIT THIS FILE. IT HAS BEEN AUTO-GENERATED WITH THE FOLLOWING COMMAND:
+//
+// hw/ip/prim/util/generate_prim_mubi.py
+//
+// Double-synchronizer flop for multibit signals with additional output buffers.
+
+`include "prim_assert.sv"
+
+module prim_mubi8_sync
+ import prim_mubi_pkg::*;
+#(
+ // Number of separately buffered output signals.
+ // The buffer cells have a don't touch constraint
+ // on them such that synthesis tools won't collapse
+ // all copies into one signal.
+ parameter int NumCopies = 1,
+ // This instantiates the synchronizer flops if set to 1.
+ // In special cases where the receiver is in the same clock domain as the sender,
+ // this can be set to 0. However, it is recommended to leave this at 1.
+ parameter bit AsyncOn = 1,
+ // Reset value for the sync flops
+ parameter mubi8_t ResetValue = MuBi8False
+) (
+ input clk_i,
+ input rst_ni,
+ input mubi8_t mubi_i,
+ output mubi8_t [NumCopies-1:0] mubi_o
+);
+
+ `ASSERT_INIT(NumCopiesMustBeGreaterZero_A, NumCopies > 0)
+
+ logic [MuBi8Width-1:0] mubi;
+ if (AsyncOn) begin : gen_flops
+ prim_flop_2sync #(
+ .Width(MuBi8Width),
+ .ResetValue(MuBi8Width'(ResetValue))
+ ) u_prim_flop_2sync (
+ .clk_i,
+ .rst_ni,
+ .d_i(mubi_i),
+ .q_o(mubi)
+ );
+ end else begin : gen_no_flops
+ logic unused_clk;
+ logic unused_rst;
+ assign unused_clk = clk_i;
+ assign unused_rst = rst_ni;
+ assign mubi = mubi_i;
+ end
+
+ for (genvar j = 0; j < NumCopies; j++) begin : gen_buffs
+ logic [MuBi8Width-1:0] mubi_out;
+ for (genvar k = 0; k < MuBi8Width; k++) begin : gen_bits
+ prim_buf u_prim_buf (
+ .in_i(mubi[k]),
+ .out_o(mubi_out[k])
+ );
+ end
+ assign mubi_o[j] = mubi8_t'(mubi_out);
+ end
+
+ ////////////////
+ // Assertions //
+ ////////////////
+
+ // The outputs should be known at all times.
+ `ASSERT_KNOWN(OutputsKnown_A, mubi_o)
+
+ // If the multibit signal is in a transient state, we expect it
+ // to be stable again within one clock cycle.
+ // DV will exclude these three assertions by name, thus added a module name prefix to make it
+ // harder to accidentally replicate in other modules.
+ `ASSERT(PrimMubi8SyncCheckTransients_A,
+ !(mubi_i inside {MuBi8True, MuBi8False})
+ |=>
+ (mubi_i inside {MuBi8True, MuBi8False}))
+
+ // If a signal departs from passive state, we expect it to move to the active state
+ // with only one transient cycle in between.
+ `ASSERT(PrimMubi8SyncCheckTransients0_A,
+ $past(mubi_i == MuBi8False) &&
+ !(mubi_i inside {MuBi8True, MuBi8False})
+ |=>
+ (mubi_i == MuBi8True))
+
+ `ASSERT(PrimMubi8SyncCheckTransients1_A,
+ $past(mubi_i == MuBi8True) &&
+ !(mubi_i inside {MuBi8True, MuBi8False})
+ |=>
+ (mubi_i == MuBi8False))
+
+endmodule : prim_mubi8_sync