Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Move Handshake and Aux Chain out of Opgroups #133

Open
wants to merge 14 commits into
base: develop
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions Bender.yml
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,8 @@ sources:
- vendor/openc910/C910_RTL_FACTORY/gen_rtl/vfdsu/rtl/ct_vfdsu_srt_radix16_with_sqrt.v
- vendor/openc910/C910_RTL_FACTORY/gen_rtl/vfdsu/rtl/ct_vfdsu_srt.v
- vendor/openc910/C910_RTL_FACTORY/gen_rtl/vfdsu/rtl/ct_vfdsu_top.v
- src/fpnew_aux.sv
- src/fpnew_aux_fsm.sv
- src/fpnew_divsqrt_th_32.sv
- src/fpnew_divsqrt_th_64_multi.sv
- src/fpnew_divsqrt_multi.sv
Expand Down
100 changes: 100 additions & 0 deletions src/fpnew_aux.sv
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
// Copyright 2024 ETH Zurich and University of Bologna.
//
// Copyright and related rights are licensed under the Solderpad Hardware
// License, Version 0.51 (the "License"); you may not use this file except in
// compliance with the License. You may obtain a copy of the License at
// http://solderpad.org/licenses/SHL-0.51. Unless required by applicable law
// or agreed to in writing, software, hardware and materials distributed under
// this License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
// CONDITIONS OF ANY KIND, either express or implied. See the License for the
// specific language governing permissions and limitations under the License.
//
// SPDX-License-Identifier: SHL-0.51

// Author: Maurus Item <itemm@student.ethz.ch>
//
// Description Aux chain for FPNew, handles transmitting of shared handshake and aux data
// And enables the correct lanes so they always stay in sync.

`include "common_cells/registers.svh"

module fpnew_aux #(
parameter int unsigned NumPipeRegs = 0,
parameter type TagType = logic,
parameter type AuxType = logic
) (
input logic clk_i,
input logic rst_ni,
// Input signals
input TagType tag_i,
input AuxType aux_i,
// Input Handshake
input logic in_valid_i,
output logic in_ready_o,
input logic flush_i,
// Output signals
output TagType tag_o,
output AuxType aux_o,
// Output handshake
output logic out_valid_o,
input logic out_ready_i,
// Register Enable for Lanes
output logic [NumPipeRegs-1:0] reg_enable_o,
// External register enable override
input logic [NumPipeRegs-1:0] reg_ena_i,
// Indication of valid data in flight
output logic busy_o
);


// ---------------
// Input pipeline
// ---------------
// Input pipeline signals, index i holds signal after i register stages
TagType [0:NumPipeRegs] tag;
AuxType [0:NumPipeRegs] aux;
logic [0:NumPipeRegs] valid;

// Ready signal is combinatorial for all stages
logic [0:NumPipeRegs] ready;

// First element of pipeline is taken from inputs
assign tag [0] = tag_i;
assign aux [0] = aux_i;
assign valid [0] = in_valid_i;

// Propagate pipeline ready signal to upstream circuitry
assign in_ready_o = ready[0];

// Generate the register stages
for (genvar i = 0; i < NumPipeRegs; i++) begin : gen_input_pipeline

// Internal register enable for this stage
logic reg_ena;
// Determine the ready signal of the current stage - advance the pipeline:
// 1. if the next stage is ready for our data
// 2. if the next stage only holds a bubble (not valid) -> we can pop it
assign ready[i] = ready[i+1] | ~valid[i+1];

// Valid: enabled by ready signal, synchronous clear with the flush signal
`FFLARNC(valid[i+1], valid[i], ready[i], flush_i, 1'b0, clk_i, rst_ni)

// Enable register if pipeline ready and a valid data item is present
assign reg_enable_o[i] = ready[i] & valid[i] | reg_ena_i[i];

// Generate the pipeline registers within the stages, use enable-registers
`FFL( tag[i+1], tag[i], reg_enable_o[i], TagType'('0))
`FFL( aux[i+1], aux[i], reg_enable_o[i], AuxType'('0))
end

// Ready travels backwards from output side, driven by downstream circuitry
assign ready[NumPipeRegs] = out_ready_i;

// Assign module outputs
assign tag_o = tag [NumPipeRegs];
assign aux_o = aux [NumPipeRegs];
assign out_valid_o = valid [NumPipeRegs];

// Assign output Flags: Busy if any element inside the pipe is valid
assign busy_o = |valid;
endmodule
278 changes: 278 additions & 0 deletions src/fpnew_aux_fsm.sv
Original file line number Diff line number Diff line change
@@ -0,0 +1,278 @@
// Copyright 2024 ETH Zurich and University of Bologna.
//
// Copyright and related rights are licensed under the Solderpad Hardware
// License, Version 0.51 (the "License"); you may not use this file except in
// compliance with the License. You may obtain a copy of the License at
// http://solderpad.org/licenses/SHL-0.51. Unless required by applicable law
// or agreed to in writing, software, hardware and materials distributed under
// this License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
// CONDITIONS OF ANY KIND, either express or implied. See the License for the
// specific language governing permissions and limitations under the License.
//
// SPDX-License-Identifier: SHL-0.51

// Author: Maurus Item <itemm@student.ethz.ch>
//
// Description Aux chain for FPNew, handles transmitting of shared handshake and aux data
// And enables the correct lanes so they always stay in sync.
// This version can be used for lanes that have some form of FSM in them and only eventually are ready

`include "common_cells/registers.svh"

module fpnew_aux_fsm #(
parameter int unsigned NumPipeRegs = 0,
parameter fpnew_pkg::pipe_config_t PipeConfig = fpnew_pkg::BEFORE,
parameter type TagType = logic,
parameter type AuxType = logic
) (
input logic clk_i,
input logic rst_ni,
// Input signals
input TagType tag_i,
input AuxType aux_i,
// Input Handshake
input logic in_valid_i,
output logic in_ready_o,
input logic flush_i,
// Output signals
output TagType tag_o,
output AuxType aux_o,
// Output handshake
output logic out_valid_o,
input logic out_ready_i,
// Register Enable for Lanes
output logic [NumPipeRegs-1:0] reg_enable_o,
// Signals for the Lane FSMs
// Signal to start the FSM, will be asserted for one cycle
output logic fsm_start_o,
output logic fsm_kill_o,
input logic fsm_ready_i,
// External register enable override
input logic [NumPipeRegs-1:0] reg_ena_i,
// Indication of valid data in flight
output logic busy_o
);

// ----------
// Pipeline Distribution
// ----------
// This must match between this module and modules that use this module as reg enable input!
localparam NUM_INP_REGS = (PipeConfig == fpnew_pkg::BEFORE)
? NumPipeRegs
: (PipeConfig == fpnew_pkg::DISTRIBUTED
? (NumPipeRegs / 2) // Last to get distributed regs
: 0); // Always have one reg to use for FSM Input
localparam NUM_OUT_REGS = (PipeConfig == fpnew_pkg::AFTER || PipeConfig == fpnew_pkg::INSIDE)
? NumPipeRegs
: (PipeConfig == fpnew_pkg::DISTRIBUTED
? ((NumPipeRegs + 1) / 2) // First to get distributed regs
: 0); // no regs here otherwise

// ---------------
// Input pipeline
// ---------------
// Input pipeline signals, index i holds signal after i register stages
TagType [0:NUM_INP_REGS] in_tag;
AuxType [0:NUM_INP_REGS] in_aux;
logic [0:NUM_INP_REGS] in_valid;

// Ready signal is combinatorial for all stages
logic [0:NUM_INP_REGS] in_ready;

// First element of pipeline is taken from inputs
assign in_tag [0] = tag_i;
assign in_aux [0] = aux_i;
assign in_valid [0] = in_valid_i;

// Propagate pipeline ready signal to upstream circuitry
assign in_ready_o = in_ready[0];

// Generate the register stages
for (genvar i = 0; i < NUM_INP_REGS; i++) begin : gen_input_pipeline

// Internal register enable for this stage
logic reg_ena;
// Determine the ready signal of the current stage - advance the pipeline:
// 1. if the next stage is ready for our data
// 2. if the next stage only holds a bubble (not valid) -> we can pop it
assign in_ready[i] = in_ready[i+1] | ~in_valid[i+1];

// Valid: enabled by ready signal, synchronous clear with the flush signal
`FFLARNC(in_valid[i+1], in_valid[i], in_ready[i], flush_i, 1'b0, clk_i, rst_ni)

// Enable register if pipleine ready and a valid data item is present
assign reg_enable_o[i] = in_ready[i] & in_valid[i] | reg_ena_i[i];

// Generate the pipeline registers within the stages, use enable-registers
`FFL( in_tag[i+1], in_tag[i], reg_enable_o[i], TagType'('0))
`FFL( in_aux[i+1], in_aux[i], reg_enable_o[i], AuxType'('0))
end

// ----------
// Global FSM
// ----------

// FSM states
typedef enum logic [1:0] {IDLE, BUSY, HOLD} fsm_state_e;
fsm_state_e state_q, state_d;

// Input & Output Handshake
logic fsm_in_valid, fsm_in_ready;
logic fsm_out_valid, fsm_out_ready;

logic fsm_start, fsm_busy;

// Data holding signals
TagType held_tag;
AuxType held_aux;

// Upstream Handshake Connection
assign fsm_in_valid = in_valid[NUM_INP_REGS];
assign in_ready[NUM_INP_REGS] = fsm_in_ready;

// FSM to safely apply and receive data from DIVSQRT unit
always_comb begin : flag_fsm
// Default assignments
fsm_out_valid = 1'b0;
fsm_in_ready = 1'b0;
fsm_start = 1'b0;
fsm_busy = 1'b0;
state_d = state_q;

unique case (state_q)
IDLE: begin
fsm_in_ready = '1;
if (fsm_in_valid) begin
state_d = BUSY;
fsm_start = 1'b1;
end
end
BUSY: begin
fsm_busy = 1'b1;
// If all active lanes are done send data down chain
if (fsm_ready_i) begin
fsm_out_valid = 1'b1;
if (fsm_out_ready) begin
fsm_in_ready = 1'b1;
if (fsm_in_valid) begin
state_d = BUSY;
fsm_start = 1'b1;
end else begin
state_d = IDLE;
end
end else begin
state_d = HOLD;
end
end
end
HOLD: begin
// Exact same as BUSY, but outer condition is already given
fsm_out_valid = 1'b1;
if (fsm_out_ready) begin
fsm_in_ready = 1'b1;
if (fsm_in_valid) begin
state_d = BUSY;
fsm_start = 1'b1;
end else begin
state_d = IDLE;
end
end else begin
state_d = HOLD;
end
end

// fall into idle state otherwise
default: state_d = IDLE;
endcase

// Flushing overrides the other actions
if (flush_i) begin
fsm_out_valid = 1'b0;
state_d = IDLE;
end
end

`FF(state_q, state_d, IDLE);

// Mini FSM for external reg enable. If external reg enable is set:
// 1. Kill any ongoing operations
// 2. On the next cycle start new operations
logic ext_fsm_start_d, ext_fsm_start_q;

if (NUM_INP_REGS > 0) begin
assign ext_fsm_start_d = reg_ena_i[NUM_INP_REGS - 1];
end else begin
assign ext_fsm_start_d = 1'b0;
end

`FF(ext_fsm_start_q, ext_fsm_start_d, 1'b0);

assign fsm_kill_o = ext_fsm_start_d;
assign fsm_start_o = (fsm_start || ext_fsm_start_q);

// ----------------
// Data Holding FFs
// ----------------

logic hold_reg_enable;
assign hold_reg_enable = fsm_start || ext_fsm_start_d;

`FFL( held_tag, in_tag[NUM_INP_REGS], hold_reg_enable, TagType'('0));
`FFL( held_aux, in_aux[NUM_INP_REGS], hold_reg_enable, AuxType'('0));

// ---------------
// Output pipeline
// ---------------

// Output pipeline signals, index i holds signal after i register stages
TagType [0:NUM_OUT_REGS] out_tag;
AuxType [0:NUM_OUT_REGS] out_aux;
logic [0:NUM_OUT_REGS] out_valid;

// Ready signal is combinatorial for all stages
logic [0:NUM_OUT_REGS] out_ready;

// Connect to upstream Handshake
assign out_valid[0] = fsm_out_valid;
assign fsm_out_ready = out_ready[0];

// Connect to Hold Register
assign out_tag [0] = held_tag;
assign out_aux [0] = held_aux;

// Generate the register stages
for (genvar i = 0; i < NUM_OUT_REGS; i++) begin : gen_output_pipeline

// Internal register enable for this stage
logic reg_ena;
// Determine the ready signal of the current stage - advance the pipeline:
// 1. if the next stage is ready for our data
// 2. if the next stage only holds a bubble (not valid) -> we can pop it
assign out_ready[i] = out_ready[i+1] | ~out_valid[i+1];

// Valid: enabled by ready signal, synchronous clear with the flush signal
`FFLARNC(out_valid[i+1], out_valid[i], out_ready[i], flush_i, 1'b0, clk_i, rst_ni)

// Enable register if pipleine ready and a valid data item is present
assign reg_ena = (out_ready[i] & out_valid[i]) | reg_ena_i[NUM_INP_REGS + i];;

// Drive external registers with reg enable
assign reg_enable_o[NUM_INP_REGS + i] = reg_ena;

// Generate the pipeline registers within the stages, use enable-registers
`FFL( out_tag[i+1], out_tag[i], reg_ena, TagType'('0))
`FFL( out_aux[i+1], out_aux[i], reg_ena, AuxType'('0))
end

// Ready travels backwards from output side, driven by downstream circuitry
assign out_ready[NUM_OUT_REGS] = out_ready_i;

// Assign module outputs
assign tag_o = out_tag [NUM_OUT_REGS];
assign aux_o = out_aux [NUM_OUT_REGS];
assign out_valid_o = out_valid [NUM_OUT_REGS];

// Assign output Flags: Busy if any element inside the pipe is valid
assign busy_o = |in_valid | |out_valid | fsm_busy;

endmodule
Loading