diff --git a/src/fpnew_opgroup_block.sv b/src/fpnew_opgroup_block.sv index db2c3032..628beaa9 100644 --- a/src/fpnew_opgroup_block.sv +++ b/src/fpnew_opgroup_block.sv @@ -61,6 +61,7 @@ module fpnew_opgroup_block #( // Output handshake output logic out_valid_o, input logic out_ready_i, + input logic out_lock_i, // Indication of valid data in flight output logic busy_o ); @@ -223,7 +224,7 @@ module fpnew_opgroup_block #( output_t arbiter_output; // Round-Robin arbiter to decide which result to use - rr_arb_tree #( + rr_arb_tree_lock #( .NumIn ( NUM_FORMATS ), .DataType ( output_t ), .AxiVldRdy ( 1'b1 ) @@ -232,6 +233,7 @@ module fpnew_opgroup_block #( .rst_ni, .flush_i, .rr_i ( '0 ), + .lock_rr_i ( out_lock_i ), .req_i ( fmt_out_valid ), .gnt_o ( fmt_out_ready ), .data_i ( fmt_outputs ), diff --git a/src/fpnew_pkg.sv b/src/fpnew_pkg.sv index 42d0df6b..46050b4c 100644 --- a/src/fpnew_pkg.sv +++ b/src/fpnew_pkg.sv @@ -325,6 +325,31 @@ package fpnew_pkg; LfsrInternalPrecision: 32 }; + // Different kinds of Redundancy that might be used + typedef enum logic [2:0] { + NONE, // No redundancy module is generated - redundancy can not be enabled + TMR_FAST, // Operands will be tripplicated in time - if nothing goes wrong output after 2 cycles (longer critical path) + TMR_SMALL, // Operands will be tripplicated in time - always output after 3 cycles (shorter critical path) + DMR, // Operands will be duplicated in time and are retried on failure + DMR_INORDER // Operands will be duplicated in time and are retried on failure - always keeps the order of outputs the same + } redundancy_type_t; + + // FPU configuration: redundancy + typedef struct packed { + logic TripplicateRepetition; // Whether to tripplicate the state machines for redundant operations + redundancy_type_t RedundancyType; + } redundancy_features_t; + + localparam redundancy_features_t DEFAULT_NO_REDUNDANCY = '{ + TripplicateRepetition: 1'b0, + RedundancyType: NONE + }; + + localparam redundancy_features_t DEFAULT_REDUNDANCY = '{ + TripplicateRepetition: 1'b1, + RedundancyType: TMR_FAST + }; + // ----------------------- // Synthesis optimization // ----------------------- @@ -589,4 +614,30 @@ package fpnew_pkg; return res; endfunction + // Returns the number data elements in the longest path of the FPU + function automatic int unsigned longest_path(fmt_unsigned_t regs, fmt_logic_t cfg); + automatic int unsigned res = 0; + for (int unsigned i = 0; i < NUM_FP_FORMATS; i++) begin + if (cfg[i]) res = maximum(res, regs[i]); + end + return res + 1; + endfunction + + // Returns the number data elements in the shortest path of the FPU + function automatic int unsigned shortest_path(fmt_unsigned_t regs, fmt_logic_t cfg); + automatic int unsigned res = 0; + for (int unsigned i = 0; i < NUM_FP_FORMATS; i++) begin + if (cfg[i]) res = minimum(res, regs[i]); + end + return res + 1; + endfunction + + // Return whether any active format is set as MERGED + function automatic logic division_enabled(opgrp_fmt_unit_types_t unit_types); + for (int unsigned i = 0; i < NUM_FP_FORMATS; i++) begin + if (unit_types[DIVSQRT][i] != DISABLED) return 1'b1; + end + return 1'b0; + endfunction + endpackage diff --git a/src/fpnew_top.sv b/src/fpnew_top.sv index b564286d..a2fcbdc5 100644 --- a/src/fpnew_top.sv +++ b/src/fpnew_top.sv @@ -13,17 +13,20 @@ // Author: Stefan Mach +`include "common_cells/registers.svh" + module fpnew_top #( // FPU configuration - parameter fpnew_pkg::fpu_features_t Features = fpnew_pkg::RV64D_Xsflt, - parameter fpnew_pkg::fpu_implementation_t Implementation = fpnew_pkg::DEFAULT_NOREGS, + parameter fpnew_pkg::fpu_features_t Features = fpnew_pkg::RV64D_Xsflt, + parameter fpnew_pkg::fpu_implementation_t Implementation = fpnew_pkg::DEFAULT_NOREGS, // DivSqrtSel chooses among PULP, TH32, or THMULTI (see documentation and fpnew_pkg.sv for further details) - parameter fpnew_pkg::divsqrt_unit_t DivSqrtSel = fpnew_pkg::THMULTI, - parameter type TagType = logic, - parameter logic TrueSIMDClass = 1'b0, - parameter logic EnableSIMDMask = 1'b0, - parameter logic CompressedVecCmpResult = 1'b0, // conceived for RV32FD cores + parameter fpnew_pkg::divsqrt_unit_t DivSqrtSel = fpnew_pkg::THMULTI, + parameter type TagType = logic, + parameter logic TrueSIMDClass = 1'b0, + parameter logic EnableSIMDMask = 1'b0, + parameter logic CompressedVecCmpResult = 1'b0, // conceived for RV32FD cores parameter fpnew_pkg::rsr_impl_t StochasticRndImplementation = fpnew_pkg::DEFAULT_NO_RSR, + parameter fpnew_pkg::redundancy_features_t RedundancyFeatures = fpnew_pkg::DEFAULT_NO_REDUNDANCY, // Do not change localparam int unsigned NumLanes = fpnew_pkg::max_num_lanes(Features.Width, Features.FpFmtMask, Features.EnableVectors), localparam type MaskType = logic [NumLanes-1:0], @@ -33,6 +36,7 @@ module fpnew_top #( input logic clk_i, input logic rst_ni, input logic [31:0] hart_id_i, + input logic redundancy_enable_i, // Input signals input logic [NUM_OPERANDS-1:0][WIDTH-1:0] operands_i, input fpnew_pkg::roundmode_e rnd_mode_i, @@ -56,31 +60,250 @@ module fpnew_top #( output logic out_valid_o, input logic out_ready_i, // Indication of valid data in flight - output logic busy_o + output logic busy_o, + output logic fault_detected_o ); localparam int unsigned NUM_OPGROUPS = fpnew_pkg::NUM_OPGROUPS; localparam int unsigned NUM_FORMATS = fpnew_pkg::NUM_FP_FORMATS; + localparam int LOCK_TIMEOUT = fpnew_pkg::division_enabled(Implementation.UnitTypes) ? 60: 5; + + localparam int MAX_DELAY = + // Base formula for how long something can stay in chain + 2 * fpnew_pkg::longest_path(Implementation.PipeRegs, Implementation.PipeConfig) + - fpnew_pkg::shortest_path(Implementation.PipeRegs, Implementation.PipeConfig) + // In case of a DMR based approach the retry has another storage element that we need to account for + + ((RedundancyFeatures.RedundancyType == fpnew_pkg::DMR) || (RedundancyFeatures.RedundancyType == fpnew_pkg::DMR_INORDER)) ? 1 : 0; + // The ternary operator ? 1 : 0 is needed since True / False might not evaluate to 1 / 0 in all tools + // For example in synopsys-2022.03-kgf dc_shell the True evaluates to 2 or 3 in this line + + // Based of the max delay we can not calculate how big of an ID is needed to ensure ids are locally unique + localparam int unsigned ID_SIZE_BASE = fpnew_pkg::maximum( + 1, + $clog2(MAX_DELAY) + + // In case of a TMR approach we add extra ID Bits for the Division since it can take up to 12 cycles + // For DMR this is not needed as we stall the unit instead + ((RedundancyFeatures.RedundancyType == fpnew_pkg::TMR_FAST || RedundancyFeatures.RedundancyType == fpnew_pkg::TMR_SMALL) + && fpnew_pkg::division_enabled(Implementation.UnitTypes) + ) ? 4 : 0 + ); + + // We have an extra bit for DMR methods to do error detection + localparam int unsigned ID_SIZE = ID_SIZE_BASE + ( + ((RedundancyFeatures.RedundancyType == fpnew_pkg::DMR) || (RedundancyFeatures.RedundancyType == fpnew_pkg::DMR_INORDER)) ? 1 : 0 + ); // ---------------- // Type Definition // ---------------- typedef struct packed { - logic [WIDTH-1:0] result; - fpnew_pkg::status_t status; - TagType tag; - } output_t; + logic [NUM_OPERANDS-1:0][WIDTH-1:0] operands; + fpnew_pkg::roundmode_e rnd_mode; + fpnew_pkg::operation_e op; + logic op_mod; + fpnew_pkg::fp_format_e src_fmt; + fpnew_pkg::fp_format_e dst_fmt; + fpnew_pkg::int_format_e int_fmt; + logic vectorial_op; + TagType tag; + MaskType simd_mask; + } tmr_in_stacked_t; + + typedef struct packed { + TagType tag; + logic [ID_SIZE-1:0] opid; + } submodules_stacked_t; + + typedef struct packed { + logic [WIDTH-1:0] result; + fpnew_pkg::status_t status; + TagType tag; + logic [ID_SIZE-1:0] opid; + } rr_stacked_t; + + typedef struct packed { + logic [WIDTH-1:0] result; + fpnew_pkg::status_t status; + TagType tag; + } tmr_out_stacked_t; + + // ---------------- + // Enable / Disable Redundancy + // ---------------- + + logic in_gated_valid, in_gated_ready; + logic internal_busy, gated_redundancy_enable, division_busy; + + if (RedundancyFeatures.RedundancyType == fpnew_pkg::NONE) begin : gen_no_redundandcy_controller + assign in_gated_valid = in_valid_i; + assign in_ready_o = in_gated_ready; + assign busy_o = internal_busy; + assign gated_redundancy_enable = 0; + end else begin: gen_redundancy_controller + + logic in_intermediate_valid, in_intermediate_ready; + + redundancy_controller # ( + .InternalRedundancy ( RedundancyFeatures.TripplicateRepetition ), + .LockTimeout ( LOCK_TIMEOUT ) + ) i_redundancy_controller ( + .clk_i, + .rst_ni, + .enable_i ( redundancy_enable_i ), + .busy_o ( busy_o ), + .busy_i ( internal_busy ), + .enable_o ( gated_redundancy_enable ), + .valid_i ( in_valid_i ), + .ready_o ( in_ready_o ), + .valid_o ( in_intermediate_valid ), + .ready_i ( in_intermediate_ready ) + ); + + // Stall Handshake when a division is going on and DMR is enabled + logic division_stall; + + if (RedundancyFeatures.RedundancyType == fpnew_pkg::TMR_FAST || RedundancyFeatures.RedundancyType == fpnew_pkg::TMR_SMALL) begin: gen_no_division_stall + assign division_stall = 0; + end else begin: gen_division_stall + logic division_busy_q; + `FF(division_busy_q, division_busy, 1'b0); + assign division_stall = division_busy_q & gated_redundancy_enable; + end + + assign in_gated_valid = in_intermediate_valid & ~division_stall; + assign in_intermediate_ready = in_gated_ready & ~division_stall; + end + + // ----------- + // Repeat Signals for Redundancy + // ----------- + tmr_in_stacked_t in_data, in_redundant_data; + logic [ID_SIZE-1:0] in_redundant_opid; + logic in_redundant_valid, in_redundant_ready; + + assign in_data.operands = operands_i; + assign in_data.rnd_mode = rnd_mode_i; + assign in_data.op = op_i; + assign in_data.op_mod = op_mod_i; + assign in_data.src_fmt = src_fmt_i; + assign in_data.dst_fmt = dst_fmt_i; + assign in_data.int_fmt = int_fmt_i; + assign in_data.vectorial_op = vectorial_op_i; + assign in_data.tag = tag_i; + assign in_data.simd_mask = simd_mask_i | ~{NumLanes{EnableSIMDMask}}; // Filter out the mask if not used + + // Connection down to counterpart + logic [ID_SIZE-1:0] dmr_next_id; + retry_interface #(.IDSize(ID_SIZE)) retry_connection (); + + if (RedundancyFeatures.RedundancyType == fpnew_pkg::NONE) begin: gen_in_no_redundancy + assign in_redundant_data = in_data; + assign in_redundant_valid = in_gated_valid; + assign in_gated_ready = in_redundant_ready; + assign in_redundant_opid = 0; + + assign dmr_next_id = fpnew_pkg::DONT_CARE; + assign retry_ready = fpnew_pkg::DONT_CARE; + assign retry_replacement_id = fpnew_pkg::DONT_CARE; + + end else if (RedundancyFeatures.RedundancyType == fpnew_pkg::TMR_FAST || RedundancyFeatures.RedundancyType == fpnew_pkg::TMR_SMALL) begin: gen_in_tmr + time_TMR_start #( + .DataType ( tmr_in_stacked_t ), + .IDSize ( ID_SIZE ), + .InternalRedundancy ( RedundancyFeatures.TripplicateRepetition ) + ) i_time_TMR_start ( + .clk_i, + .rst_ni, + .enable_i( gated_redundancy_enable ), + .data_i ( in_data ), + .valid_i ( in_gated_valid ), + .ready_o ( in_gated_ready ), + .data_o ( in_redundant_data ), + .id_o ( in_redundant_opid ), + .valid_o ( in_redundant_valid ), + .ready_i ( in_redundant_ready ) + ); + + assign dmr_next_id = fpnew_pkg::DONT_CARE; + assign retry_ready = fpnew_pkg::DONT_CARE; + assign retry_replacement_id = fpnew_pkg::DONT_CARE; + + end else begin: gen_in_dmr + // Connection directly to next module + tmr_in_stacked_t retry2dmr_data; + logic [ID_SIZE-1:0] retry2dmr_opid; + logic retry2dmr_valid, retry2dmr_ready; + + if (RedundancyFeatures.RedundancyType == fpnew_pkg::DMR) begin: gen_in_oo_retry + retry_start #( + .DataType ( tmr_in_stacked_t ), + .IDSize ( ID_SIZE ) + ) i_retry_start ( + .clk_i, + .rst_ni, + .data_i ( in_data ), + .valid_i ( in_gated_valid ), + .ready_o ( in_gated_ready ), + .data_o ( retry2dmr_data ), + .id_o ( retry2dmr_opid ), + .valid_o ( retry2dmr_valid ), + .ready_i ( retry2dmr_ready ), + .retry ( retry_connection ) + ); + assign retry_replacement_id = fpnew_pkg::DONT_CARE; + + end else begin: gen_in_io_retry + retry_inorder_start #( + .DataType ( tmr_in_stacked_t ), + .IDSize ( ID_SIZE ) + ) i_retry_inorder_start ( + .clk_i, + .rst_ni, + .data_i ( in_data ), + .valid_i ( in_gated_valid ), + .ready_o ( in_gated_ready ), + .data_o ( retry2dmr_data ), + .id_o ( retry2dmr_opid ), + .valid_o ( retry2dmr_valid ), + .ready_i ( retry2dmr_ready ), + .retry ( retry_connection ) + ); + end + + time_DMR_start #( + .DataType ( tmr_in_stacked_t ), + .IDSize ( ID_SIZE ), + .InternalRedundancy ( RedundancyFeatures.TripplicateRepetition ), + .UseExternalId ( 1 ) + ) i_time_DMR_start ( + .clk_i, + .rst_ni, + .enable_i ( gated_redundancy_enable ), + .next_id_o ( dmr_next_id ), + .data_i ( retry2dmr_data ), + .id_i ( retry2dmr_opid ), + .valid_i ( retry2dmr_valid ), + .ready_o ( retry2dmr_ready ), + .data_o ( in_redundant_data ), + .id_o ( in_redundant_opid ), + .valid_o ( in_redundant_valid ), + .ready_i ( in_redundant_ready ) + ); + end // Handshake signals for the blocks - logic [NUM_OPGROUPS-1:0] opgrp_in_ready, opgrp_out_valid, opgrp_out_ready, opgrp_ext, opgrp_busy; - output_t [NUM_OPGROUPS-1:0] opgrp_outputs; + logic [NUM_OPGROUPS-1:0] in_opgrp_ready, out_opgrp_valid, out_opgrp_ready, out_opgrp_ext, opgrp_busy; + rr_stacked_t [NUM_OPGROUPS-1:0] out_opgrp_data; + logic out_rr_lock; logic [NUM_FORMATS-1:0][NUM_OPERANDS-1:0] is_boxed; // ----------- // Input Side // ----------- - assign in_ready_o = in_valid_i & opgrp_in_ready[fpnew_pkg::get_opgroup(op_i)]; + assign in_redundant_ready = in_redundant_valid & in_opgrp_ready[fpnew_pkg::get_opgroup(in_redundant_data.op)]; + assign internal_busy = (| opgrp_busy); + assign division_busy = opgrp_busy[fpnew_pkg::DIVSQRT]; // NaN-boxing check for (genvar fmt = 0; fmt < int'(NUM_FORMATS); fmt++) begin : gen_nanbox_check @@ -88,8 +311,8 @@ module fpnew_top #( // NaN boxing is only generated if it's enabled and needed if (Features.EnableNanBox && (FP_WIDTH < WIDTH)) begin : check for (genvar op = 0; op < int'(NUM_OPERANDS); op++) begin : operands - assign is_boxed[fmt][op] = (!vectorial_op_i) - ? operands_i[op][WIDTH-1:FP_WIDTH] == '1 + assign is_boxed[fmt][op] = (!in_redundant_data.vectorial_op) + ? in_redundant_data.operands[op][WIDTH-1:FP_WIDTH] == '1 : 1'b1; end end else begin : no_check @@ -97,10 +320,6 @@ module fpnew_top #( end end - // Filter out the mask if not used - MaskType simd_mask; - assign simd_mask = simd_mask_i | ~{NumLanes{EnableSIMDMask}}; - // ------------------------- // Generate Operation Blocks // ------------------------- @@ -110,7 +329,7 @@ module fpnew_top #( logic in_valid; logic [NUM_FORMATS-1:0][NUM_OPS-1:0] input_boxed; - assign in_valid = in_valid_i & (fpnew_pkg::get_opgroup(op_i) == fpnew_pkg::opgroup_e'(opgrp)); + assign in_valid = in_redundant_valid & (fpnew_pkg::get_opgroup(in_redundant_data.op) == fpnew_pkg::opgroup_e'(opgrp)); // slice out input boxing always_comb begin : slice_inputs @@ -118,6 +337,11 @@ module fpnew_top #( input_boxed[fmt] = is_boxed[fmt][NUM_OPS-1:0]; end + submodules_stacked_t in_tag, out_tag; + + assign in_tag.tag = in_redundant_data.tag; + assign in_tag.opid = in_redundant_opid; + fpnew_opgroup_block #( .OpGroup ( fpnew_pkg::opgroup_e'(opgrp) ), .Width ( WIDTH ), @@ -128,67 +352,187 @@ module fpnew_top #( .FmtPipeRegs ( Implementation.PipeRegs[opgrp] ), .FmtUnitTypes ( Implementation.UnitTypes[opgrp] ), .PipeConfig ( Implementation.PipeConfig ), - .TagType ( TagType ), + .TagType ( submodules_stacked_t ), .TrueSIMDClass ( TrueSIMDClass ), .CompressedVecCmpResult ( CompressedVecCmpResult ), .StochasticRndImplementation ( StochasticRndImplementation ) ) i_opgroup_block ( .clk_i, .rst_ni, - .hart_id_i, - .operands_i ( operands_i[NUM_OPS-1:0] ), - .is_boxed_i ( input_boxed ), - .rnd_mode_i, - .op_i, - .op_mod_i, - .src_fmt_i, - .dst_fmt_i, - .int_fmt_i, - .vectorial_op_i, - .tag_i, - .simd_mask_i ( simd_mask ), - .in_valid_i ( in_valid ), - .in_ready_o ( opgrp_in_ready[opgrp] ), + .hart_id_i ( hart_id_i ), + .operands_i ( in_redundant_data.operands[NUM_OPS-1:0] ), + .is_boxed_i ( input_boxed ), + .rnd_mode_i ( in_redundant_data.rnd_mode ), + .op_i ( in_redundant_data.op ), + .op_mod_i ( in_redundant_data.op_mod ), + .src_fmt_i ( in_redundant_data.src_fmt ), + .dst_fmt_i ( in_redundant_data.dst_fmt ), + .int_fmt_i ( in_redundant_data.int_fmt ), + .vectorial_op_i ( in_redundant_data.vectorial_op ), + .tag_i ( in_tag ), + .simd_mask_i ( in_redundant_data.simd_mask ), + .in_valid_i ( in_valid ), + .in_ready_o ( in_opgrp_ready[opgrp] ), .flush_i, - .result_o ( opgrp_outputs[opgrp].result ), - .status_o ( opgrp_outputs[opgrp].status ), - .extension_bit_o ( opgrp_ext[opgrp] ), - .tag_o ( opgrp_outputs[opgrp].tag ), - .out_valid_o ( opgrp_out_valid[opgrp] ), - .out_ready_i ( opgrp_out_ready[opgrp] ), - .busy_o ( opgrp_busy[opgrp] ) + .result_o ( out_opgrp_data[opgrp].result ), + .status_o ( out_opgrp_data[opgrp].status ), + .extension_bit_o ( out_opgrp_ext[opgrp] ), + .tag_o ( out_tag ), + .out_valid_o ( out_opgrp_valid[opgrp] ), + .out_lock_i ( out_rr_lock ), + .out_ready_i ( out_opgrp_ready[opgrp] ), + .busy_o ( opgrp_busy[opgrp] ) ); + + assign out_opgrp_data[opgrp].tag = out_tag.tag; + assign out_opgrp_data[opgrp].opid = out_tag.opid; + end // ------------------ // Arbitrate Outputs // ------------------ - output_t arbiter_output; + logic out_redundant_valid, out_redundant_ready; + rr_stacked_t out_redundant_data; // Round-Robin arbiter to decide which result to use - rr_arb_tree #( + rr_arb_tree_lock #( .NumIn ( NUM_OPGROUPS ), - .DataType ( output_t ), - .AxiVldRdy ( 1'b1 ) + .DataType ( rr_stacked_t ), + .AxiVldRdy ( 1'b1 ), + .FairArb ( 1'b1 ) ) i_arbiter ( .clk_i, .rst_ni, .flush_i, - .rr_i ( '0 ), - .req_i ( opgrp_out_valid ), - .gnt_o ( opgrp_out_ready ), - .data_i ( opgrp_outputs ), - .gnt_i ( out_ready_i ), - .req_o ( out_valid_o ), - .data_o ( arbiter_output ), - .idx_o ( /* unused */ ) + .rr_i ( '0 ), + .lock_rr_i ( out_rr_lock ), + .req_i ( out_opgrp_valid ), + .gnt_o ( out_opgrp_ready ), + .data_i ( out_opgrp_data ), + .gnt_i ( out_redundant_ready ), + .req_o ( out_redundant_valid ), + .data_o ( out_redundant_data ), + .idx_o ( /* Unused */ ) ); - // Unpack output - assign result_o = arbiter_output.result; - assign status_o = arbiter_output.status; - assign tag_o = arbiter_output.tag; + // ------------------ + // Unrepeat Outputs + // ------------------ + + tmr_out_stacked_t out_data, out_redundant_data_noid; + assign out_redundant_data_noid.tag = out_redundant_data.tag; + assign out_redundant_data_noid.status = out_redundant_data.status; + assign out_redundant_data_noid.result = out_redundant_data.result; + + if (RedundancyFeatures.RedundancyType == fpnew_pkg::NONE) begin : gen_out_no_redundancy + assign out_data = out_redundant_data_noid; + assign out_valid_o = out_redundant_valid; + assign out_redundant_ready = out_ready_i; + assign out_rr_lock = 0; + assign fault_detected_o = 0; + + assign retry_opid = fpnew_pkg::DONT_CARE; + assign retry_valid = fpnew_pkg::DONT_CARE; + assign retry_lock = fpnew_pkg::DONT_CARE; + + end else if (RedundancyFeatures.RedundancyType == fpnew_pkg::TMR_FAST || RedundancyFeatures.RedundancyType == fpnew_pkg::TMR_SMALL) begin : gen_out_tmr + time_TMR_end #( + .DataType ( tmr_out_stacked_t ), + .LockTimeout ( LOCK_TIMEOUT ), + .IDSize ( ID_SIZE ), + .InternalRedundancy ( RedundancyFeatures.TripplicateRepetition ), + .EarlyValidEnable ( (RedundancyFeatures.RedundancyType == fpnew_pkg::TMR_FAST) ? 1 : 0 ) + // The ternary operator ? 1 : 0 is needed since True / False might not evaluate to 1 / 0 in all tools + ) i_time_TMR_end ( + .clk_i, + .rst_ni, + .enable_i ( gated_redundancy_enable ), + .data_i ( out_redundant_data_noid ), + .id_i ( out_redundant_data.opid ), + .valid_i ( out_redundant_valid ), + .ready_o ( out_redundant_ready ), + .lock_o ( out_rr_lock ), + .data_o ( out_data ), + .valid_o ( out_valid_o ), + .ready_i ( out_ready_i ), + .fault_detected_o ( fault_detected_o ) + ); + + assign retry_opid = fpnew_pkg::DONT_CARE; + assign retry_valid = fpnew_pkg::DONT_CARE; + assign retry_lock = fpnew_pkg::DONT_CARE; + + end else begin : gen_out_dmr + tmr_out_stacked_t dmr2retry_data; + logic [ID_SIZE-1:0] dmr2retry_opid; + logic dmr2retry_valid, dmr2retry_ready, dmr2retry_needs_retry; - assign busy_o = (| opgrp_busy); + time_DMR_end #( + .DataType ( tmr_out_stacked_t ), + .LockTimeout ( LOCK_TIMEOUT ), + .IDSize ( ID_SIZE ), + .InternalRedundancy ( RedundancyFeatures.TripplicateRepetition ) + ) i_time_DMR_end ( + .clk_i, + .rst_ni, + .enable_i ( gated_redundancy_enable ), + .next_id_i ( dmr_next_id ), + .data_i ( out_redundant_data_noid ), + .id_i ( out_redundant_data.opid ), + .valid_i ( out_redundant_valid ), + .ready_o ( out_redundant_ready ), + .lock_o ( out_rr_lock ), + .data_o ( dmr2retry_data ), + .id_o ( dmr2retry_opid ), + .needs_retry_o ( dmr2retry_needs_retry ), + .valid_o ( dmr2retry_valid ), + .ready_i ( dmr2retry_ready ), + .fault_detected_o ( fault_detected_o ) + ); + + if (RedundancyFeatures.RedundancyType == fpnew_pkg::DMR) begin: gen_out_oo_retry + retry_end #( + .DataType ( tmr_out_stacked_t ), + .IDSize ( ID_SIZE ) + ) i_retry_end ( + .clk_i, + .rst_ni, + .data_i ( dmr2retry_data ), + .id_i ( dmr2retry_opid ), + .needs_retry_i ( dmr2retry_needs_retry ), + .valid_i ( dmr2retry_valid ), + .ready_o ( dmr2retry_ready ), + .data_o ( out_data ), + .valid_o ( out_valid_o ), + .ready_i ( out_ready_i ), + .retry ( retry_connection ) + ); + assign retry_lock = fpnew_pkg::DONT_CARE; + + end else begin: gen_out_io_retry + retry_inorder_end #( + .DataType ( tmr_out_stacked_t ), + .IDSize ( ID_SIZE ) + ) i_retry_inorder_end ( + .clk_i, + .rst_ni, + .data_i ( dmr2retry_data ), + .id_i ( dmr2retry_opid ), + .needs_retry_i ( dmr2retry_needs_retry ), + .valid_i ( dmr2retry_valid ), + .ready_o ( dmr2retry_ready ), + .data_o ( out_data ), + .valid_o ( out_valid_o ), + .ready_i ( out_ready_i ), + .retry ( retry_connection ) + ); + end + end + + // Unpack output + assign result_o = out_data.result; + assign status_o = out_data.status; + assign tag_o = out_data.tag; endmodule