diff options
Diffstat (limited to 'fpga/usrp3')
-rw-r--r-- | fpga/usrp3/lib/fifo/axi_mux.v | 251 |
1 files changed, 160 insertions, 91 deletions
diff --git a/fpga/usrp3/lib/fifo/axi_mux.v b/fpga/usrp3/lib/fifo/axi_mux.v index 72a771348..29727ad4b 100644 --- a/fpga/usrp3/lib/fifo/axi_mux.v +++ b/fpga/usrp3/lib/fifo/axi_mux.v @@ -1,110 +1,179 @@ - -// Copyright 2016 Ettus Research -// Copyright 2018 Ettus Research, a National Instruments Company +// +// Copyright 2021 Ettus Research, a National Instruments Brand // // SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: axi_mux +// +// Description: +// +// Takes arbitrary number of AXI streams and merges them to into a single +// output channel. Bubble cycles are inserted after each packet. +// +// Parameters: +// +// PRIO : Controls the arbitration scheme. +// 0 - Round-robin +// 1 - Priority (lower number ports get priority) +// WIDTH : Width of each AXI-Stream (width of TDATA). +// PRE_FIFO_SIZE : Log2 of the input buffer FIFO. Set to 0 for no FIFO. +// POST_FIFO_SIZE : Log2 of the output buffer FIFO. Set to 0 for no FIFO. +// SIZE : Number of input ports to the multiplexer. +// + +`default_nettype none -// axi_mux -- takes arbitrary number of AXI stream, merges them to 1 output channel -// Round-robin if PRIO=0, priority if PRIO=1 (lower number ports get priority) -// Bubble cycles are inserted after each packet -module axi_mux - #(parameter PRIO=0, - parameter WIDTH=64, - parameter PRE_FIFO_SIZE=0, - parameter POST_FIFO_SIZE=0, - parameter SIZE=4) - (input clk, input reset, input clear, - input [(WIDTH*SIZE)-1:0] i_tdata, input [SIZE-1:0] i_tlast, input [SIZE-1:0] i_tvalid, output [SIZE-1:0] i_tready, - output [WIDTH-1:0] o_tdata, output o_tlast, output o_tvalid, input o_tready); +module axi_mux #( + parameter PRIO = 0, + parameter WIDTH = 64, + parameter PRE_FIFO_SIZE = 0, + parameter POST_FIFO_SIZE = 0, + parameter SIZE = 4 +) ( + input wire clk, + input wire reset, + input wire clear, - wire [WIDTH*SIZE-1:0] i_tdata_int; - wire [SIZE-1:0] i_tlast_int, i_tvalid_int, i_tready_int; + // Input streams + input wire [WIDTH*SIZE-1:0] i_tdata, + input wire [ SIZE-1:0] i_tlast, + input wire [ SIZE-1:0] i_tvalid, + output wire [ SIZE-1:0] i_tready, - wire [WIDTH-1:0] o_tdata_int; - wire o_tlast_int, o_tvalid_int, o_tready_int; + // Single output stream + output wire [ WIDTH-1:0] o_tdata, + output wire o_tlast, + output wire o_tvalid, + input wire o_tready +); - reg [$clog2(SIZE)-1:0] st_port; - reg st_active; + wire [WIDTH*SIZE-1:0] i_tdata_int; + wire [ SIZE-1:0] i_tlast_int; + wire [ SIZE-1:0] i_tvalid_int; + wire [ SIZE-1:0] i_tready_int; + + wire [WIDTH-1:0] o_tdata_int; + wire o_tlast_int; + wire o_tvalid_int; + wire o_tready_int; + + reg [$clog2(SIZE)-1:0] st_port; + reg st_active; + + //--------------------------------------------------------------------------- + // Input FIFO + //--------------------------------------------------------------------------- genvar n; generate - if (PRE_FIFO_SIZE == 0) begin + if (PRE_FIFO_SIZE == 0) begin : gen_no_pre_fifo assign i_tdata_int = i_tdata; assign i_tlast_int = i_tlast; assign i_tvalid_int = i_tvalid; assign i_tready = i_tready_int; - end else begin + end else begin : gen_pre_fifo for (n = 0; n < SIZE; n = n + 1) begin - axi_fifo #(.WIDTH(WIDTH+1), .SIZE(PRE_FIFO_SIZE)) axi_fifo ( - .clk(clk), .reset(reset), .clear(clear), - .i_tdata({i_tlast[n],i_tdata[WIDTH*(n+1)-1:WIDTH*n]}), .i_tvalid(i_tvalid[n]), .i_tready(i_tready[n]), - .o_tdata({i_tlast_int[n],i_tdata_int[WIDTH*(n+1)-1:WIDTH*n]}), .o_tvalid(i_tvalid_int[n]), .o_tready(i_tready_int[n]), - .space(), .occupied()); + axi_fifo #( + .WIDTH(WIDTH+1 ), + .SIZE (PRE_FIFO_SIZE) + ) axi_fifo ( + .clk (clk ), + .reset (reset ), + .clear (clear ), + .i_tdata ({i_tlast[n],i_tdata[WIDTH*(n+1)-1:WIDTH*n]} ), + .i_tvalid(i_tvalid[n] ), + .i_tready(i_tready[n] ), + .o_tdata ({i_tlast_int[n],i_tdata_int[WIDTH*(n+1)-1:WIDTH*n]}), + .o_tvalid(i_tvalid_int[n] ), + .o_tready(i_tready_int[n] ), + .space ( ), + .occupied( ) + ); end end endgenerate - always @(posedge clk) - if(reset) - begin - st_port <= 0; - st_active <= 1'b0; - end - else - if(st_active) - begin - if(o_tlast_int & o_tvalid_int & o_tready_int) - begin - st_active <= 1'b0; - if((PRIO != 0) | (st_port == (SIZE-1))) - st_port <= 0; - else - st_port <= st_port + 1; - end - end // if (st_active) - else - if(i_tvalid_int[st_port]) - st_active <= 1'b1; - else - if(st_port == (SIZE-1)) - st_port <= 0; - else - st_port <= st_port + 1; - - genvar i; - generate - for(i=0;i<SIZE;i=i+1) - begin : gen1 - assign i_tready_int[i] = st_active & o_tready_int & (st_port == i); - end - endgenerate - - assign o_tvalid_int = st_active & i_tvalid_int[st_port]; - assign o_tlast_int = i_tlast_int[st_port]; - - genvar j; - generate - for (j=0;j<WIDTH;j=j+1) - begin : gen2 - assign o_tdata_int[j] = i_tdata_int[st_port*WIDTH+j]; - end - endgenerate - - generate - if(POST_FIFO_SIZE == 0) - begin - assign o_tdata = o_tdata_int; - assign o_tlast = o_tlast_int; - assign o_tvalid = o_tvalid_int; - assign o_tready_int = o_tready; - end - else - axi_fifo #(.WIDTH(WIDTH+1),.SIZE(POST_FIFO_SIZE)) axi_fifo - (.clk(clk), .reset(reset), .clear(clear), - .i_tdata({o_tlast_int,o_tdata_int}), .i_tvalid(o_tvalid_int), .i_tready(o_tready_int), - .o_tdata({o_tlast,o_tdata}), .o_tvalid(o_tvalid), .o_tready(o_tready), - .space(), .occupied()); - endgenerate - -endmodule // axi__mux + //--------------------------------------------------------------------------- + // Multiplexer Logic + //--------------------------------------------------------------------------- + + always @(posedge clk) begin + if (reset) begin + st_port <= 0; + st_active <= 1'b0; + end else begin + if (st_active) begin + if (o_tlast_int & o_tvalid_int & o_tready_int) begin + st_active <= 1'b0; + if ((PRIO != 0) | (st_port == (SIZE-1))) begin + st_port <= 0; + end else begin + st_port <= st_port + 1; + end + end + end else begin + if (i_tvalid_int[st_port]) begin + st_active <= 1'b1; + end else begin + if (st_port == (SIZE-1)) begin + st_port <= 0; + end else begin + st_port <= st_port + 1; + end + end + end + end + end + + genvar i; + generate + for (i=0; i<SIZE; i=i+1) begin : gen_tready + assign i_tready_int[i] = st_active & o_tready_int & (st_port == i); + end + endgenerate + + assign o_tvalid_int = st_active & i_tvalid_int[st_port]; + assign o_tlast_int = i_tlast_int[st_port]; + + genvar j; + generate + for (j=0; j<WIDTH; j=j+1) begin : gen_tdata + assign o_tdata_int[j] = i_tdata_int[st_port*WIDTH+j]; + end + endgenerate + + //--------------------------------------------------------------------------- + // Output FIFO + //--------------------------------------------------------------------------- + + generate + if (POST_FIFO_SIZE == 0) begin + assign o_tdata = o_tdata_int; + assign o_tlast = o_tlast_int; + assign o_tvalid = o_tvalid_int; + assign o_tready_int = o_tready; + end else begin + axi_fifo #( + .WIDTH(WIDTH+1 ), + .SIZE (POST_FIFO_SIZE) + ) axi_fifo ( + .clk (clk ), + .reset (reset ), + .clear (clear ), + .i_tdata ({o_tlast_int,o_tdata_int}), + .i_tvalid(o_tvalid_int ), + .i_tready(o_tready_int ), + .o_tdata ({o_tlast,o_tdata} ), + .o_tvalid(o_tvalid ), + .o_tready(o_tready ), + .space ( ), + .occupied( ) + ); + end + endgenerate + +endmodule + + +`default_nettype wire |