diff options
| author | Wade Fife <wade.fife@ettus.com> | 2020-06-19 15:40:12 -0500 | 
|---|---|---|
| committer | Aaron Rossetto <aaron.rossetto@ni.com> | 2020-07-30 12:51:41 -0500 | 
| commit | 1e94f85b8bafc3f9acab7ef35d2675fa7e61f6f4 (patch) | |
| tree | 12ba30a59c8057e355971797d5cc7bf6910f520b /fpga/usrp3/lib/rfnoc/blocks | |
| parent | b0b3849a18e1f2d3cb255a507b01ac5e7a9416a0 (diff) | |
| download | uhd-1e94f85b8bafc3f9acab7ef35d2675fa7e61f6f4.tar.gz uhd-1e94f85b8bafc3f9acab7ef35d2675fa7e61f6f4.tar.bz2 uhd-1e94f85b8bafc3f9acab7ef35d2675fa7e61f6f4.zip | |
fpga: rfnoc: Add Signal Generator RFNoC block
Diffstat (limited to 'fpga/usrp3/lib/rfnoc/blocks')
8 files changed, 1743 insertions, 0 deletions
| diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/Makefile b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/Makefile new file mode 100644 index 000000000..8fffd1c2e --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/Makefile @@ -0,0 +1,49 @@ +# +# Copyright 2020 Ettus Research, a National Instruments Brand +# +# SPDX-License-Identifier: LGPL-3.0-or-later +# + +#------------------------------------------------- +# Top-of-Makefile +#------------------------------------------------- +# Define BASE_DIR to point to the "top" dir. Note: +# UHD_FPGA_DIR must be passed into this Makefile. +BASE_DIR = ../../../../top +# Include viv_sim_preample after defining BASE_DIR +include $(BASE_DIR)/../tools/make/viv_sim_preamble.mak + +#------------------------------------------------- +# Design Specific +#------------------------------------------------- +# Include makefiles and sources for the DUT and its +# dependencies. +include $(BASE_DIR)/../lib/rfnoc/core/Makefile.srcs +include $(BASE_DIR)/../lib/rfnoc/utils/Makefile.srcs +include $(LIB_IP_DIR)/cordic_rotator/Makefile.inc +include Makefile.srcs + +DESIGN_SRCS += $(abspath \ +$(RFNOC_CORE_SRCS) \ +$(RFNOC_UTIL_SRCS) \ +$(RFNOC_OOT_SRCS)  \ +$(LIB_IP_CORDIC_ROTATOR_SRCS) \ +) + +#------------------------------------------------- +# Testbench Specific +#------------------------------------------------- +SIM_TOP = rfnoc_block_siggen_all_tb glbl +SIM_SRCS = \ +$(abspath $(IP_BUILD_DIR)/cordic_rotator/sim/cordic_rotator.vhd) \ +$(VIVADO_PATH)/data/verilog/src/glbl.v \ +$(abspath rfnoc_block_siggen_tb.sv) \ +$(abspath rfnoc_block_siggen_all_tb.sv) \ + +#------------------------------------------------- +# Bottom-of-Makefile +#------------------------------------------------- +# Include all simulator specific makefiles here +# Each should define a unique target to simulate +# e.g. xsim, vsim, etc and a common "clean" target +include $(BASE_DIR)/../tools/make/viv_simulator.mak diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/Makefile.srcs b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/Makefile.srcs new file mode 100644 index 000000000..336cc8cfa --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/Makefile.srcs @@ -0,0 +1,24 @@ +# +# Copyright 2020 Ettus Research, a National Instruments Brand +# +# SPDX-License-Identifier: LGPL-3.0-or-later +# + +################################################## +# RFNoC Block Sources +################################################## +# Here, list all the files that are necessary to synthesize this block. Don't +# include testbenches! +# Make sure that the source files are nicely detectable by a regex. Best to put +# one on each line. +# The first argument to addprefix is the current path to this Makefile, so the +# path list is always absolute, regardless of from where we're including or +# calling this file. RFNOC_OOT_SRCS needs to be a simply expanded variable +# (not a recursively expanded variable), and we take care of that in the build +# infrastructure. +RFNOC_OOT_SRCS += $(addprefix $(dir $(abspath $(lastword $(MAKEFILE_LIST)))), \ +noc_shell_siggen.v \ +rfnoc_siggen_core.v \ +rfnoc_block_siggen_regs.vh \ +rfnoc_block_siggen.v \ +) diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/noc_shell_siggen.v b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/noc_shell_siggen.v new file mode 100644 index 000000000..6f14430b7 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/noc_shell_siggen.v @@ -0,0 +1,263 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: noc_shell_siggen +// +// Description: +// +//   This is a tool-generated NoC-shell for the siggen block. +//   See the RFNoC specification for more information about NoC shells. +// +// Parameters: +// +//   THIS_PORTID : Control crossbar port to which this block is connected +//   CHDR_W      : AXIS-CHDR data bus width +//   MTU         : Maximum transmission unit (i.e., maximum packet size in +// + +`default_nettype none + + +module noc_shell_siggen #( +  parameter [9:0] THIS_PORTID     = 10'd0, +  parameter       CHDR_W          = 64, +  parameter [5:0] MTU             = 10, +  parameter       NUM_PORTS       = 1 +) ( +  //--------------------- +  // Framework Interface +  //--------------------- + +  // RFNoC Framework Clocks +  input  wire rfnoc_chdr_clk, +  input  wire rfnoc_ctrl_clk, +  input  wire ce_clk, + +  // NoC Shell Generated Resets +  output wire rfnoc_chdr_rst, +  output wire rfnoc_ctrl_rst, +  output wire ce_rst, + +  // RFNoC Backend Interface +  input  wire [511:0]          rfnoc_core_config, +  output wire [511:0]          rfnoc_core_status, + +  // AXIS-CHDR Input Ports (from framework) +  input  wire [(1)*CHDR_W-1:0] s_rfnoc_chdr_tdata, +  input  wire [(1)-1:0]        s_rfnoc_chdr_tlast, +  input  wire [(1)-1:0]        s_rfnoc_chdr_tvalid, +  output wire [(1)-1:0]        s_rfnoc_chdr_tready, +  // AXIS-CHDR Output Ports (to framework) +  output wire [(0+NUM_PORTS)*CHDR_W-1:0] m_rfnoc_chdr_tdata, +  output wire [(0+NUM_PORTS)-1:0]        m_rfnoc_chdr_tlast, +  output wire [(0+NUM_PORTS)-1:0]        m_rfnoc_chdr_tvalid, +  input  wire [(0+NUM_PORTS)-1:0]        m_rfnoc_chdr_tready, + +  // AXIS-Ctrl Control Input Port (from framework) +  input  wire [31:0]           s_rfnoc_ctrl_tdata, +  input  wire                  s_rfnoc_ctrl_tlast, +  input  wire                  s_rfnoc_ctrl_tvalid, +  output wire                  s_rfnoc_ctrl_tready, +  // AXIS-Ctrl Control Output Port (to framework) +  output wire [31:0]           m_rfnoc_ctrl_tdata, +  output wire                  m_rfnoc_ctrl_tlast, +  output wire                  m_rfnoc_ctrl_tvalid, +  input  wire                  m_rfnoc_ctrl_tready, + +  //--------------------- +  // Client Interface +  //--------------------- + +  // CtrlPort Clock and Reset +  output wire               ctrlport_clk, +  output wire               ctrlport_rst, +  // CtrlPort Master +  output wire               m_ctrlport_req_wr, +  output wire               m_ctrlport_req_rd, +  output wire [19:0]        m_ctrlport_req_addr, +  output wire [31:0]        m_ctrlport_req_data, +  input  wire               m_ctrlport_resp_ack, +  input  wire [31:0]        m_ctrlport_resp_data, + +  // AXI-Stream Data Clock and Reset +  output wire               axis_data_clk, +  output wire               axis_data_rst, +  // Data Stream to User Logic: out +  input  wire [NUM_PORTS*32*1-1:0]   s_out_axis_tdata, +  input  wire [NUM_PORTS*1-1:0]      s_out_axis_tkeep, +  input  wire [NUM_PORTS-1:0]        s_out_axis_tlast, +  input  wire [NUM_PORTS-1:0]        s_out_axis_tvalid, +  output wire [NUM_PORTS-1:0]        s_out_axis_tready, +  input  wire [NUM_PORTS*64-1:0]     s_out_axis_ttimestamp, +  input  wire [NUM_PORTS-1:0]        s_out_axis_thas_time, +  input  wire [NUM_PORTS*16-1:0]     s_out_axis_tlength, +  input  wire [NUM_PORTS-1:0]        s_out_axis_teov, +  input  wire [NUM_PORTS-1:0]        s_out_axis_teob +); + +  //--------------------------------------------------------------------------- +  //  Backend Interface +  //--------------------------------------------------------------------------- + +  wire         data_i_flush_en; +  wire [31:0]  data_i_flush_timeout; +  wire [63:0]  data_i_flush_active; +  wire [63:0]  data_i_flush_done; +  wire         data_o_flush_en; +  wire [31:0]  data_o_flush_timeout; +  wire [63:0]  data_o_flush_active; +  wire [63:0]  data_o_flush_done; + +  backend_iface #( +    .NOC_ID        (32'h51663110), +    .NUM_DATA_I    (1), +    .NUM_DATA_O    (0+NUM_PORTS), +    .CTRL_FIFOSIZE ($clog2(32)), +    .MTU           (MTU) +  ) backend_iface_i ( +    .rfnoc_chdr_clk       (rfnoc_chdr_clk), +    .rfnoc_chdr_rst       (rfnoc_chdr_rst), +    .rfnoc_ctrl_clk       (rfnoc_ctrl_clk), +    .rfnoc_ctrl_rst       (rfnoc_ctrl_rst), +    .rfnoc_core_config    (rfnoc_core_config), +    .rfnoc_core_status    (rfnoc_core_status), +    .data_i_flush_en      (data_i_flush_en), +    .data_i_flush_timeout (data_i_flush_timeout), +    .data_i_flush_active  (data_i_flush_active), +    .data_i_flush_done    (data_i_flush_done), +    .data_o_flush_en      (data_o_flush_en), +    .data_o_flush_timeout (data_o_flush_timeout), +    .data_o_flush_active  (data_o_flush_active), +    .data_o_flush_done    (data_o_flush_done) +  ); + +  //--------------------------------------------------------------------------- +  //  Reset Generation +  //--------------------------------------------------------------------------- + +  wire ce_rst_pulse; + +  pulse_synchronizer #(.MODE ("POSEDGE")) pulse_synchronizer_ce ( +    .clk_a(rfnoc_chdr_clk), .rst_a(1'b0), .pulse_a (rfnoc_chdr_rst), .busy_a (), +    .clk_b(ce_clk), .pulse_b (ce_rst_pulse) +  ); + +  pulse_stretch_min #(.LENGTH(32)) pulse_stretch_min_ce ( +    .clk(ce_clk), .rst(1'b0), +    .pulse_in(ce_rst_pulse), .pulse_out(ce_rst) +  ); + +  //--------------------------------------------------------------------------- +  //  Control Path +  //--------------------------------------------------------------------------- + +  assign ctrlport_clk = ce_clk; +  assign ctrlport_rst = ce_rst; + +  ctrlport_endpoint #( +    .THIS_PORTID      (THIS_PORTID), +    .SYNC_CLKS        (0), +    .AXIS_CTRL_MST_EN (0), +    .AXIS_CTRL_SLV_EN (1), +    .SLAVE_FIFO_SIZE  ($clog2(32)) +  ) ctrlport_endpoint_i ( +    .rfnoc_ctrl_clk            (rfnoc_ctrl_clk), +    .rfnoc_ctrl_rst            (rfnoc_ctrl_rst), +    .ctrlport_clk              (ctrlport_clk), +    .ctrlport_rst              (ctrlport_rst), +    .s_rfnoc_ctrl_tdata        (s_rfnoc_ctrl_tdata), +    .s_rfnoc_ctrl_tlast        (s_rfnoc_ctrl_tlast), +    .s_rfnoc_ctrl_tvalid       (s_rfnoc_ctrl_tvalid), +    .s_rfnoc_ctrl_tready       (s_rfnoc_ctrl_tready), +    .m_rfnoc_ctrl_tdata        (m_rfnoc_ctrl_tdata), +    .m_rfnoc_ctrl_tlast        (m_rfnoc_ctrl_tlast), +    .m_rfnoc_ctrl_tvalid       (m_rfnoc_ctrl_tvalid), +    .m_rfnoc_ctrl_tready       (m_rfnoc_ctrl_tready), +    .m_ctrlport_req_wr         (m_ctrlport_req_wr), +    .m_ctrlport_req_rd         (m_ctrlport_req_rd), +    .m_ctrlport_req_addr       (m_ctrlport_req_addr), +    .m_ctrlport_req_data       (m_ctrlport_req_data), +    .m_ctrlport_req_byte_en    (), +    .m_ctrlport_req_has_time   (), +    .m_ctrlport_req_time       (), +    .m_ctrlport_resp_ack       (m_ctrlport_resp_ack), +    .m_ctrlport_resp_status    (2'b0), +    .m_ctrlport_resp_data      (m_ctrlport_resp_data), +    .s_ctrlport_req_wr         (1'b0), +    .s_ctrlport_req_rd         (1'b0), +    .s_ctrlport_req_addr       (20'b0), +    .s_ctrlport_req_portid     (10'b0), +    .s_ctrlport_req_rem_epid   (16'b0), +    .s_ctrlport_req_rem_portid (10'b0), +    .s_ctrlport_req_data       (32'b0), +    .s_ctrlport_req_byte_en    (4'hF), +    .s_ctrlport_req_has_time   (1'b0), +    .s_ctrlport_req_time       (64'b0), +    .s_ctrlport_resp_ack       (), +    .s_ctrlport_resp_status    (), +    .s_ctrlport_resp_data      () +  ); + +  //--------------------------------------------------------------------------- +  //  Data Path +  //--------------------------------------------------------------------------- + +  genvar i; + +  assign axis_data_clk = ce_clk; +  assign axis_data_rst = ce_rst; + +  //--------------------- +  // Input Data Paths +  //--------------------- + +  // No input data paths for this block +  assign s_rfnoc_chdr_tready[0] = 1'b1; +  assign data_i_flush_done[0] = 1'b1; + +  //--------------------- +  // Output Data Paths +  //--------------------- + +  for (i = 0; i < NUM_PORTS; i = i + 1) begin: gen_output_out +    axis_data_to_chdr #( +      .CHDR_W          (CHDR_W), +      .ITEM_W          (32), +      .NIPC            (1), +      .SYNC_CLKS       (0), +      .INFO_FIFO_SIZE  ($clog2(32)), +      .PYLD_FIFO_SIZE  ($clog2(32)), +      .MTU             (MTU), +      .SIDEBAND_AT_END (0) +    ) axis_data_to_chdr_out_out ( +      .axis_chdr_clk      (rfnoc_chdr_clk), +      .axis_chdr_rst      (rfnoc_chdr_rst), +      .axis_data_clk      (axis_data_clk), +      .axis_data_rst      (axis_data_rst), +      .m_axis_chdr_tdata  (m_rfnoc_chdr_tdata[(0+i)*CHDR_W+:CHDR_W]), +      .m_axis_chdr_tlast  (m_rfnoc_chdr_tlast[0+i]), +      .m_axis_chdr_tvalid (m_rfnoc_chdr_tvalid[0+i]), +      .m_axis_chdr_tready (m_rfnoc_chdr_tready[0+i]), +      .s_axis_tdata       (s_out_axis_tdata[(32*1)*i+:(32*1)]), +      .s_axis_tkeep       (s_out_axis_tkeep[1*i+:1]), +      .s_axis_tlast       (s_out_axis_tlast[i]), +      .s_axis_tvalid      (s_out_axis_tvalid[i]), +      .s_axis_tready      (s_out_axis_tready[i]), +      .s_axis_ttimestamp  (s_out_axis_ttimestamp[64*i+:64]), +      .s_axis_thas_time   (s_out_axis_thas_time[i]), +      .s_axis_tlength     (s_out_axis_tlength[16*i+:16]), +      .s_axis_teov        (s_out_axis_teov[i]), +      .s_axis_teob        (s_out_axis_teob[i]), +      .flush_en           (data_o_flush_en), +      .flush_timeout      (data_o_flush_timeout), +      .flush_active       (data_o_flush_active[0+i]), +      .flush_done         (data_o_flush_done[0+i]) +    ); +  end + +endmodule // noc_shell_siggen + + +`default_nettype wire diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen.v b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen.v new file mode 100644 index 000000000..8a90c3e28 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen.v @@ -0,0 +1,242 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: rfnoc_block_siggen +// +// Description: +// +//   Signal generator RFNoC block. This block outputs packets of one of three +//   output types based on the REG_WAVEFORM register setting. Supported modes +//   include constant, sinusoidal, and noise/random. The output is also run +//   through a gain stage that is configurable using the REG_GAIN register. +//   See the register descriptions in rfnoc_block_siggen_regs.vh for details. +// +//   The sine output is based on the Xilinx CORDIC IP, configured for the +//   rotate function, with scaled radians as the units. See the CORDIC user +//   guide (PG105) and register descriptions for details. +// +// Parameters: +// +//   THIS_PORTID : Control crossbar port to which this block is connected +//   CHDR_W      : AXIS-CHDR data bus width +//   MTU         : Maximum transmission unit (i.e., maximum packet size in +//                 CHDR words is 2**MTU). +//   NUM_PORTS   : Number of siggen cores to instantiate. +// + +`default_nettype none + + +module rfnoc_block_siggen #( +  parameter [9:0] THIS_PORTID = 10  'd0, +  parameter       CHDR_W      = 64, +  parameter [5:0] MTU         = 10, +  parameter       NUM_PORTS   = 1 +) ( +  // RFNoC Framework Clocks and Resets +  input  wire                        rfnoc_chdr_clk, +  input  wire                        rfnoc_ctrl_clk, +  input  wire                        ce_clk, +  // RFNoC Backend Interface +  input  wire [               511:0] rfnoc_core_config, +  output wire [               511:0] rfnoc_core_status, +  // AXIS-CHDR Input Ports (from framework) +  input  wire [          CHDR_W-1:0] s_rfnoc_chdr_tdata, +  input  wire                        s_rfnoc_chdr_tlast, +  input  wire                        s_rfnoc_chdr_tvalid, +  output wire                        s_rfnoc_chdr_tready, +  // AXIS-CHDR Output Ports (to framework) +  output wire [NUM_PORTS*CHDR_W-1:0] m_rfnoc_chdr_tdata, +  output wire [       NUM_PORTS-1:0] m_rfnoc_chdr_tlast, +  output wire [       NUM_PORTS-1:0] m_rfnoc_chdr_tvalid, +  input  wire [       NUM_PORTS-1:0] m_rfnoc_chdr_tready, +  // AXIS-Ctrl Input Port (from framework) +  input  wire [                31:0] s_rfnoc_ctrl_tdata, +  input  wire                        s_rfnoc_ctrl_tlast, +  input  wire                        s_rfnoc_ctrl_tvalid, +  output wire                        s_rfnoc_ctrl_tready, +  // AXIS-Ctrl Output Port (to framework) +  output wire [                31:0] m_rfnoc_ctrl_tdata, +  output wire                        m_rfnoc_ctrl_tlast, +  output wire                        m_rfnoc_ctrl_tvalid, +  input  wire                        m_rfnoc_ctrl_tready +); + +  `include "rfnoc_block_siggen_regs.vh" + + +  //--------------------------------------------------------------------------- +  // Signal Declarations +  //--------------------------------------------------------------------------- + +  // CtrlPort Master +  wire        m_ctrlport_req_wr; +  wire        m_ctrlport_req_rd; +  wire [19:0] m_ctrlport_req_addr; +  wire [31:0] m_ctrlport_req_data; +  wire        m_ctrlport_resp_ack; +  wire [31:0] m_ctrlport_resp_data; +  // Data Stream to User Logic: out +  wire [NUM_PORTS*32*1-1:0] s_out_axis_tdata; +  wire [     NUM_PORTS-1:0] s_out_axis_tlast; +  wire [     NUM_PORTS-1:0] s_out_axis_tvalid; +  wire [     NUM_PORTS-1:0] s_out_axis_tready; +  wire [  NUM_PORTS*16-1:0] s_out_axis_tlength; + + +  //--------------------------------------------------------------------------- +  // NoC Shell +  //--------------------------------------------------------------------------- + +  wire ce_rst; + +  noc_shell_siggen #( +    .CHDR_W      (CHDR_W), +    .THIS_PORTID (THIS_PORTID), +    .MTU         (MTU), +    .NUM_PORTS   (NUM_PORTS) +  ) noc_shell_siggen_i ( +    //--------------------- +    // Framework Interface +    //--------------------- + +    // Clock Inputs +    .rfnoc_chdr_clk        (rfnoc_chdr_clk), +    .rfnoc_ctrl_clk        (rfnoc_ctrl_clk), +    .ce_clk                (ce_clk), +    // Reset Outputs +    .rfnoc_chdr_rst        (), +    .rfnoc_ctrl_rst        (), +    .ce_rst                (ce_rst), +    // RFNoC Backend Interface +    .rfnoc_core_config     (rfnoc_core_config), +    .rfnoc_core_status     (rfnoc_core_status), +    // CHDR Input Ports  (from framework) +    .s_rfnoc_chdr_tdata    (s_rfnoc_chdr_tdata), +    .s_rfnoc_chdr_tlast    (s_rfnoc_chdr_tlast), +    .s_rfnoc_chdr_tvalid   (s_rfnoc_chdr_tvalid), +    .s_rfnoc_chdr_tready   (s_rfnoc_chdr_tready), +    // CHDR Output Ports (to framework) +    .m_rfnoc_chdr_tdata    (m_rfnoc_chdr_tdata), +    .m_rfnoc_chdr_tlast    (m_rfnoc_chdr_tlast), +    .m_rfnoc_chdr_tvalid   (m_rfnoc_chdr_tvalid), +    .m_rfnoc_chdr_tready   (m_rfnoc_chdr_tready), +    // AXIS-Ctrl Input Port (from framework) +    .s_rfnoc_ctrl_tdata    (s_rfnoc_ctrl_tdata), +    .s_rfnoc_ctrl_tlast    (s_rfnoc_ctrl_tlast), +    .s_rfnoc_ctrl_tvalid   (s_rfnoc_ctrl_tvalid), +    .s_rfnoc_ctrl_tready   (s_rfnoc_ctrl_tready), +    // AXIS-Ctrl Output Port (to framework) +    .m_rfnoc_ctrl_tdata    (m_rfnoc_ctrl_tdata), +    .m_rfnoc_ctrl_tlast    (m_rfnoc_ctrl_tlast), +    .m_rfnoc_ctrl_tvalid   (m_rfnoc_ctrl_tvalid), +    .m_rfnoc_ctrl_tready   (m_rfnoc_ctrl_tready), + +    //--------------------- +    // Client Interface +    //--------------------- + +    // CtrlPort Clock and Reset +    .ctrlport_clk          (), +    .ctrlport_rst          (), +    // CtrlPort Master +    .m_ctrlport_req_wr     (m_ctrlport_req_wr), +    .m_ctrlport_req_rd     (m_ctrlport_req_rd), +    .m_ctrlport_req_addr   (m_ctrlport_req_addr), +    .m_ctrlport_req_data   (m_ctrlport_req_data), +    .m_ctrlport_resp_ack   (m_ctrlport_resp_ack), +    .m_ctrlport_resp_data  (m_ctrlport_resp_data), + +    // AXI-Stream Clock and Reset +    .axis_data_clk         (), +    .axis_data_rst         (), +    // Data Stream from User Logic: out +    .s_out_axis_tdata      (s_out_axis_tdata), +    .s_out_axis_tkeep      ({NUM_PORTS{1'b1}}), +    .s_out_axis_tlast      (s_out_axis_tlast), +    .s_out_axis_tvalid     (s_out_axis_tvalid), +    .s_out_axis_tready     (s_out_axis_tready), +    .s_out_axis_ttimestamp ({NUM_PORTS{64'b0}}), +    .s_out_axis_thas_time  ({NUM_PORTS{1'b0}}), +    .s_out_axis_tlength    (s_out_axis_tlength), +    .s_out_axis_teov       ({NUM_PORTS{1'b0}}), +    .s_out_axis_teob       ({NUM_PORTS{1'b0}}) +  ); + + +  //--------------------------------------------------------------------------- +  // CtrlPort Splitter +  //--------------------------------------------------------------------------- + +  // Create a CtrlPort bus for each port instance + +  wire [ 1*NUM_PORTS-1:0] ctrlport_req_wr; +  wire [ 1*NUM_PORTS-1:0] ctrlport_req_rd; +  wire [20*NUM_PORTS-1:0] ctrlport_req_addr; +  wire [32*NUM_PORTS-1:0] ctrlport_req_data; +  wire [ 1*NUM_PORTS-1:0] ctrlport_resp_ack; +  wire [32*NUM_PORTS-1:0] ctrlport_resp_data; + +  ctrlport_decoder #( +    .NUM_SLAVES   (NUM_PORTS), +    .BASE_ADDR    (0), +    .SLAVE_ADDR_W (SIGGEN_ADDR_W) +  ) ctrlport_decoder_i ( +    .ctrlport_clk            (ce_clk), +    .ctrlport_rst            (ce_rst), +    .s_ctrlport_req_wr       (m_ctrlport_req_wr), +    .s_ctrlport_req_rd       (m_ctrlport_req_rd), +    .s_ctrlport_req_addr     (m_ctrlport_req_addr), +    .s_ctrlport_req_data     (m_ctrlport_req_data), +    .s_ctrlport_req_byte_en  (4'hF), +    .s_ctrlport_req_has_time (1'b0), +    .s_ctrlport_req_time     (64'b0), +    .s_ctrlport_resp_ack     (m_ctrlport_resp_ack), +    .s_ctrlport_resp_status  (), +    .s_ctrlport_resp_data    (m_ctrlport_resp_data), +    .m_ctrlport_req_wr       (ctrlport_req_wr), +    .m_ctrlport_req_rd       (ctrlport_req_rd), +    .m_ctrlport_req_addr     (ctrlport_req_addr), +    .m_ctrlport_req_data     (ctrlport_req_data), +    .m_ctrlport_req_byte_en  (), +    .m_ctrlport_req_has_time (), +    .m_ctrlport_req_time     (), +    .m_ctrlport_resp_ack     (ctrlport_resp_ack), +    .m_ctrlport_resp_status  ({NUM_PORTS{2'b0}}), +    .m_ctrlport_resp_data    (ctrlport_resp_data) +  ); + + +  //--------------------------------------------------------------------------- +  // Port Instances +  //--------------------------------------------------------------------------- + +  genvar port; +  generate +    for (port = 0; port < NUM_PORTS; port = port+1) begin : gen_ports + +      rfnoc_siggen_core rfnoc_siggen_core_i ( +        .clk                  (ce_clk), +        .rst                  (ce_rst), +        .s_ctrlport_req_wr    (ctrlport_req_wr    [port* 1 +:  1]), +        .s_ctrlport_req_rd    (ctrlport_req_rd    [port* 1 +:  1]), +        .s_ctrlport_req_addr  (ctrlport_req_addr  [port*20 +: 20]), +        .s_ctrlport_req_data  (ctrlport_req_data  [port*32 +: 32]), +        .s_ctrlport_resp_ack  (ctrlport_resp_ack  [port* 1 +:  1]), +        .s_ctrlport_resp_data (ctrlport_resp_data [port*32 +: 32]), +        .m_tdata              (s_out_axis_tdata   [port*32 +: 32]), +        .m_tlast              (s_out_axis_tlast   [port* 1 +:  1]), +        .m_tvalid             (s_out_axis_tvalid  [port* 1 +:  1]), +        .m_tready             (s_out_axis_tready  [port* 1 +:  1]), +        .m_tlength            (s_out_axis_tlength [port*16 +: 16]) +      ); + +    end +  endgenerate + +endmodule // rfnoc_block_siggen + + +`default_nettype wire diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen_all_tb.sv b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen_all_tb.sv new file mode 100644 index 000000000..0b23dcd0f --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen_all_tb.sv @@ -0,0 +1,28 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: rfnoc_block_logpwr_all_tb +// +// Description: Top-level testbench for the Signal Generator RFNoC block. This +// instantiates rfnoc_block_siggen_tb with different parameters to test +// multiple configurations. +// + +`default_nettype none + + +module rfnoc_block_siggen_all_tb; + +  // Test multiple CHDR widths +  rfnoc_block_siggen_tb #(.CHDR_W(64),  .NUM_PORTS(1))  test_siggen_0(); +  rfnoc_block_siggen_tb #(.CHDR_W(64),  .NUM_PORTS(2))  test_siggen_1(); +  rfnoc_block_siggen_tb #(.CHDR_W(64),  .NUM_PORTS(3))  test_siggen_2(); +  rfnoc_block_siggen_tb #(.CHDR_W(128), .NUM_PORTS(2))  test_siggen_3(); +  rfnoc_block_siggen_tb #(.CHDR_W(256), .NUM_PORTS(1))  test_siggen_4(); +   +endmodule : rfnoc_block_siggen_all_tb + + +`default_nettype wire diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen_regs.vh b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen_regs.vh new file mode 100644 index 000000000..10cda5425 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen_regs.vh @@ -0,0 +1,134 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module:  rfnoc_block_siggen_regs (Header) +// +// Description: RFNoC Signal Generator block register descriptions +// + + +// Address space size, per signal generator core. That is, each signal +// generator core's address space is separated in the CtrlPort address space +// by 2^SIGGEN_ADDR_W bytes. +localparam SIGGEN_ADDR_W = 5; + + + +// REG_ENABLE (R/W) +// +// Starts or stops the waveform output. Write a 1 to enable waveform output, 0 +// to disable waveform output. Starting and stopping occurs on packet +// boundaries. +// +// [31:1] Reserved +// [0]    Enable bit +// +localparam REG_ENABLE = 'h00; +// +localparam REG_ENABLE_LEN = 1; + + +// REG_SPP (R/W) +// +// The number of samples per packet to output for the selected waveform. This +// is read at the start of each new packet. +// +localparam REG_SPP = 'h04; +// +localparam REG_SPP_LEN = 14; + + +// REG_WAVEFORM (R/W) +// +// Selects the type of waveform to output. The possible values are: +// +//   0 : (WAVE_CONST) Constant data +//   1 : (WAVE_SINE)  Sine wave +//   2 : (WAVE_NOISE) Noise / random data +// +localparam REG_WAVEFORM = 'h08; +// +localparam REG_WAVEFORM_LEN = 2; +// +localparam WAVE_CONST = 2'h0; +localparam WAVE_SINE  = 2'h1; +localparam WAVE_NOISE = 2'h2; + + +// REG_GAIN (R/W) +// +// Sets the gain for the output. This is a 16-bit signed fixed point value +// with 15 fractional bits. The gain is applied to both the real and imaginary +// parts of each output sample. This gain is applied to all waveform output +// types. +// +localparam REG_GAIN = 'h0C; +// +localparam REG_GAIN_LEN = 16; + + +// REG_CONSTANT (R/W) +// +// Sets the value for the sample to output for the constant waveform. Both the +// real and imaginary components are treated as 16-bit signed fixed point +// values with 15 fractional bits. +// +// [31:16] Real/I component +// [15: 0] Imaginary/Q component +// +localparam REG_CONSTANT = 'h10; +// +localparam REG_CONSTANT_LEN = 32; + + +// REG_PHASE_INC (R/W) +// +// Sets the phase increment, in "scaled radians", for the sine waveform +// generator. This is the amount by which REG_CARTESIAN is rotated each clock +// cycle. In other words, it controls the rate of rotation, or the frequency, +// of the sine wave. The range of the phase value is -1.0 to +1.0. In scaled +// radians, the value range -1 to +1 corresponds to -Pi to Pi in radians. +// +// In other words, the normalized frequency (in cycles/sample) of the +// sinusoidal output is equal to 0.5*REG_PHASE_INC. +// +// [31:16] : Reserved +// [15: 0] : Signed fixed-point phase value with 3 integer bits and 13 +//           fractional bits. +// +localparam REG_PHASE_INC = 'h14; +// +localparam REG_PHASE_INC_LEN = 16; + + +// REG_CARTESIAN (R/W) +// +// Sets the (X,Y) Cartesian coordinate that will be rotated to generate the +// sine output. The rate of rotation is controlled by REG_PHASE_INC. Note that +// this input vector is also scaled by a "CORDIC scale factor" that equals +// about 1.16444 (the product of sqrt(1 + 2^(-2i)) for i = 1 to n, where n = +// 14, the number of fractional bits used by the CORDIC IP). +// +// Both the X and Y coordinates are signed fixed-point values with 15 +// fractional bits. +// +// For example, supposed you wanted a sinusoidal output with an amplitude of +// about 0.9. In that case, you could set the Y coordinate to 0 and the X +// coordinate to 0.9/1.16444 = 0.7729. In fixed-point, that's 0.7729 * 2^15 = +// 0x62EE. +// +// NOTE: The Xilinx CORDIC IP describes the input and output as 16-bit signed +// fixed point with 2 integer and 14 fractional bits, which is accurate. +// However, since we treat the output as sc16 (15 fractional bits), we need to +// double the value of the CARTESIAN inputs to get the output we want for sc16. +// This is mathematically inequivalent to simply saying the CARTESIAN inputs +// have 15 fractional bits instead of 14. +// +// [31:16] : Y (Imaginary) component +// [15: 0] : X (Real) component +// +localparam REG_CARTESIAN = 'h18; +// +localparam REG_CARTESIAN_LEN = 32; diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen_tb.sv b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen_tb.sv new file mode 100644 index 000000000..4cb701aaf --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_block_siggen_tb.sv @@ -0,0 +1,719 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: rfnoc_block_siggen_tb +// +// Description: Testbench for the siggen RFNoC block. +// + +`default_nettype none + + +module rfnoc_block_siggen_tb #( +  parameter CHDR_W    = 64, +  parameter NUM_PORTS = 1 +); + +  `include "test_exec.svh" + +  import PkgTestExec::*; +  import PkgChdrUtils::*; +  import PkgRfnocBlockCtrlBfm::*; +  import PkgRfnocItemUtils::*; + +  `include "rfnoc_block_siggen_regs.vh" + + +  //--------------------------------------------------------------------------- +  // Testbench Configuration +  //--------------------------------------------------------------------------- + +  localparam [31:0] NOC_ID          = 32'h51663110; +  localparam [ 9:0] THIS_PORTID     = 10'h123; +  localparam int    MTU             = 10;    // Log2 of max transmission unit in CHDR words +  localparam int    NUM_PORTS_I     = 1; +  localparam int    NUM_PORTS_O     = 0+NUM_PORTS; +  localparam int    ITEM_W          = 32;    // Sample size in bits +  localparam int    SPP             = 64;    // Samples per packet +  localparam int    PKT_SIZE_BYTES  = SPP * (ITEM_W/8); +  localparam int    STALL_PROB      = 25;    // Default BFM stall probability +  localparam real   CHDR_CLK_PER    = 5.0;   // 200 MHz +  localparam real   CTRL_CLK_PER    = 8.0;   // 125 MHz +  localparam real   CE_CLK_PER      = 4.0;   // 250 MHz + +  localparam real PI = 2*$acos(0); + +  // Number of fractional bits used for fixed point values of the different +  // settings (derived from the DUT). +  localparam int GAIN_FRAC  = 15; +  localparam int CONST_FRAC = 15; +  localparam int PHASE_FRAC = 13; +  localparam int CART_FRAC  = 14; + +  // Maximum real (floating point) values allowed for the different fixed +  // point formats (for range checking). All of the fixed point values are +  // signed 16-bit.  +  localparam real MAX_GAIN_R  =  (2.0**15-1) / (2.0**GAIN_FRAC); +  localparam real MIN_GAIN_R  = -(2.0**15)   / (2.0**GAIN_FRAC); +  localparam real MAX_CONST_R =  (2.0**15-1) / (2.0**CONST_FRAC); +  localparam real MIN_CONST_R = -(2.0**15)   / (2.0**CONST_FRAC); +  localparam real MAX_CART_R  =  (2.0**15-1) / (2.0**CART_FRAC); +  localparam real MIN_CART_R  = -(2.0**15)   / (2.0**CART_FRAC); +  // Note that the CORDIC only supports phase values from -1.0 to +1.0. +  localparam real MAX_PHASE_R = +1.0; +  localparam real MIN_PHASE_R = -1.0; + + +  //--------------------------------------------------------------------------- +  // Clocks and Resets +  //--------------------------------------------------------------------------- + +  bit rfnoc_chdr_clk; +  bit rfnoc_ctrl_clk; +  bit ce_clk; + +  sim_clock_gen #(.PERIOD(CHDR_CLK_PER), .AUTOSTART(0)) +   rfnoc_chdr_clk_gen (.clk(rfnoc_chdr_clk), .rst()); +  sim_clock_gen #(.PERIOD(CTRL_CLK_PER), .AUTOSTART(0)) +   rfnoc_ctrl_clk_gen (.clk(rfnoc_ctrl_clk), .rst()); +  sim_clock_gen #(.PERIOD(CE_CLK_PER), .AUTOSTART(0)) +     ce_clk_gen (.clk(ce_clk), .rst()); + + +  //--------------------------------------------------------------------------- +  // Bus Functional Models +  //--------------------------------------------------------------------------- + +  // Backend Interface +  RfnocBackendIf backend (rfnoc_chdr_clk, rfnoc_ctrl_clk); + +  // AXIS-Ctrl Interface +  AxiStreamIf #(32) m_ctrl (rfnoc_ctrl_clk, 1'b0); +  AxiStreamIf #(32) s_ctrl (rfnoc_ctrl_clk, 1'b0); + +  // AXIS-CHDR Interfaces +  AxiStreamIf #(CHDR_W) m_chdr [NUM_PORTS_I] (rfnoc_chdr_clk, 1'b0); +  AxiStreamIf #(CHDR_W) s_chdr [NUM_PORTS_O] (rfnoc_chdr_clk, 1'b0); + +  // Block Controller BFM +  RfnocBlockCtrlBfm #(CHDR_W, ITEM_W) blk_ctrl = new(backend, m_ctrl, s_ctrl); + +  // CHDR word and item/sample data types +  typedef ChdrData #(CHDR_W, ITEM_W)::chdr_word_t chdr_word_t; +  typedef ChdrData #(CHDR_W, ITEM_W)::item_t      item_t; + +  // Connect block controller to BFMs +  for (genvar i = 0; i < NUM_PORTS_I; i++) begin : gen_bfm_input_connections +    initial begin +      blk_ctrl.connect_master_data_port(i, m_chdr[i], PKT_SIZE_BYTES); +      blk_ctrl.set_master_stall_prob(i, STALL_PROB); +    end +  end +  for (genvar i = 0; i < NUM_PORTS_O; i++) begin : gen_bfm_output_connections +    initial begin +      blk_ctrl.connect_slave_data_port(i, s_chdr[i]); +      blk_ctrl.set_slave_stall_prob(i, STALL_PROB); +    end +  end + + +  //--------------------------------------------------------------------------- +  // Device Under Test (DUT) +  //--------------------------------------------------------------------------- + +  // DUT Slave (Input) Port Signals +  logic [CHDR_W*NUM_PORTS_I-1:0] s_rfnoc_chdr_tdata; +  logic [       NUM_PORTS_I-1:0] s_rfnoc_chdr_tlast; +  logic [       NUM_PORTS_I-1:0] s_rfnoc_chdr_tvalid; +  logic [       NUM_PORTS_I-1:0] s_rfnoc_chdr_tready; + +  // DUT Master (Output) Port Signals +  logic [CHDR_W*NUM_PORTS_O-1:0] m_rfnoc_chdr_tdata; +  logic [       NUM_PORTS_O-1:0] m_rfnoc_chdr_tlast; +  logic [       NUM_PORTS_O-1:0] m_rfnoc_chdr_tvalid; +  logic [       NUM_PORTS_O-1:0] m_rfnoc_chdr_tready; + +  // Map the array of BFMs to a flat vector for the DUT connections +  for (genvar i = 0; i < NUM_PORTS_I; i++) begin : gen_dut_input_connections +    // Connect BFM master to DUT slave port +    assign s_rfnoc_chdr_tdata[CHDR_W*i+:CHDR_W] = m_chdr[i].tdata; +    assign s_rfnoc_chdr_tlast[i]                = m_chdr[i].tlast; +    assign s_rfnoc_chdr_tvalid[i]               = m_chdr[i].tvalid; +    assign m_chdr[i].tready                     = s_rfnoc_chdr_tready[i]; +  end +  for (genvar i = 0; i < NUM_PORTS_O; i++) begin : gen_dut_output_connections +    // Connect BFM slave to DUT master port +    assign s_chdr[i].tdata        = m_rfnoc_chdr_tdata[CHDR_W*i+:CHDR_W]; +    assign s_chdr[i].tlast        = m_rfnoc_chdr_tlast[i]; +    assign s_chdr[i].tvalid       = m_rfnoc_chdr_tvalid[i]; +    assign m_rfnoc_chdr_tready[i] = s_chdr[i].tready; +  end + +  rfnoc_block_siggen #( +    .THIS_PORTID         (THIS_PORTID), +    .CHDR_W              (CHDR_W), +    .MTU                 (MTU), +    .NUM_PORTS           (NUM_PORTS) +  ) dut ( +    .rfnoc_chdr_clk      (rfnoc_chdr_clk), +    .rfnoc_ctrl_clk      (rfnoc_ctrl_clk), +    .ce_clk              (ce_clk), +    .rfnoc_core_config   (backend.cfg), +    .rfnoc_core_status   (backend.sts), +    .s_rfnoc_chdr_tdata  (s_rfnoc_chdr_tdata), +    .s_rfnoc_chdr_tlast  (s_rfnoc_chdr_tlast), +    .s_rfnoc_chdr_tvalid (s_rfnoc_chdr_tvalid), +    .s_rfnoc_chdr_tready (s_rfnoc_chdr_tready), +    .m_rfnoc_chdr_tdata  (m_rfnoc_chdr_tdata), +    .m_rfnoc_chdr_tlast  (m_rfnoc_chdr_tlast), +    .m_rfnoc_chdr_tvalid (m_rfnoc_chdr_tvalid), +    .m_rfnoc_chdr_tready (m_rfnoc_chdr_tready), +    .s_rfnoc_ctrl_tdata  (m_ctrl.tdata), +    .s_rfnoc_ctrl_tlast  (m_ctrl.tlast), +    .s_rfnoc_ctrl_tvalid (m_ctrl.tvalid), +    .s_rfnoc_ctrl_tready (m_ctrl.tready), +    .m_rfnoc_ctrl_tdata  (s_ctrl.tdata), +    .m_rfnoc_ctrl_tlast  (s_ctrl.tlast), +    .m_rfnoc_ctrl_tvalid (s_ctrl.tvalid), +    .m_rfnoc_ctrl_tready (s_ctrl.tready) +  ); + + +  //--------------------------------------------------------------------------- +  // Helper Tasks +  //--------------------------------------------------------------------------- + +  // Write a 32-bit register +  task automatic write_reg(int port, bit [19:0] addr, bit [31:0] value); +    blk_ctrl.reg_write(port * (2**SIGGEN_ADDR_W) + addr, value); +  endtask : write_reg + +  // Read a 32-bit register +  task automatic read_reg(int port, bit [19:0] addr, output logic [31:0] value); +    blk_ctrl.reg_read(port * (2**SIGGEN_ADDR_W) + addr, value); +  endtask : read_reg + + +  // Check if two samples are within a given distance from each other (i.e., +  // check if the Cartesian distance is < threshold). +  function bit samples_are_close( +    logic [31:0] samp_a, samp_b, +    real         threshold = 3.0 +  ); +    real ax, ay, bx, by; +    real distance; + +    // Treat the samples and signed 16-bit numbers (not fixed point) +    ax = signed'(samp_a[31:16]); +    ay = signed'(samp_a[15: 0]); +    bx = signed'(samp_b[31:16]); +    by = signed'(samp_b[15: 0]); + +    distance = $sqrt( (ax-bx)*(ax-bx) + (ay-by)*(ay-by) ); + +    return distance <= threshold; +  endfunction : samples_are_close + + +  // Convert real to signed 16-bit fixed point with "frac" fractional bits +  function automatic logic [15:0] real_to_fixed(real value, int frac = 15); +    // Convert to fixed point value +    value = value * 2.0**frac; + +    // Round +    value = $floor(value + 0.5); + +    // Saturate +    if (value > 16'sh7FFF) value = 16'sh7FFF; +    if (value < 16'sh8000) value = 16'sh8000; +    return int'(value); +  endfunction : real_to_fixed + + +  // Convert signed 16-bit fixed point to real, where the fixed point has +  // "frac" fractional bits. +  function automatic real fixed_to_real( +    logic signed [15:0] value, +    int                 frac = 15 +  ); +    return real'(value) / (2.0 ** frac); +  endfunction : fixed_to_real + + +  // Compute the next sine value we expect based on the previous +  function automatic logic [31:0] next_sine_value( +    logic [31:0] sample, +    logic [15:0] phase_inc +  ); +    real x, y, phase, new_x, new_y; +    x        = fixed_to_real(sample[31:16], CART_FRAC); +    y        = fixed_to_real(sample[15: 0], CART_FRAC); +    phase    = fixed_to_real(phase_inc, PHASE_FRAC) * PI; + +    // Compute the rotated coordinates +    new_x =  x*$cos(phase) + y*$sin(phase); +    new_y = -x*$sin(phase) + y*$cos(phase); +     +    return { real_to_fixed(new_x, CART_FRAC), real_to_fixed(new_y, CART_FRAC) }; +  endfunction : next_sine_value + + +  // Apply a gain to an input value, then round and clip the same way the DUT +  // does. +  function automatic logic [15:0] apply_gain( +    logic signed [15:0] gain, +    logic signed [15:0] value +  ); +    logic signed [31:0] result; +    bit round; + +    // Apply gain +    result = gain * value; + +    // Now we "round and clip". The round and clip block is configured with +    // 32-bit input, 16-bit output, and one "clip_bit". This means it takes +    // the upper 17-bits of the result, rounded, then converts that to a +    // 16-bit result, saturated. + +    // Round the value in the upper 17 bits to nearest (biased towards +inf, +    // but don't allow overflow). +    if (result[31:15] != 17'h0FFFF) begin +      round = result[14]; +    end else begin +      round = 0; +    end +    result = result >>> 15;   // Arithmetic right shift +    result = result + round;  // Round the result + +    // Saturate to 16-bit number +    if (result < 16'sh8000) begin +      result = 16'sh8000; +    end else if (result > 16'sh7FFF) begin +      result = 16'sh7FFF; +    end + +    return result[15:0]; +  endfunction : apply_gain + + +  // Flush (drop) any queued up packets on the output +  task automatic flush_output(int port, timeout = 100); +    item_t items[$]; + +    forever begin +      fork +        begin : wait_for_data_fork +          // Wait for tvalid to rise for up to "timeout" clock cycles +          if (m_rfnoc_chdr_tvalid[port]) +            wait(!m_rfnoc_chdr_tvalid[port]); +          wait(m_rfnoc_chdr_tvalid[port]); +        end +        begin : wait_for_timeout_fork +          #(CHDR_CLK_PER*timeout); +        end +      join_any + +      // Check if we timed out or if new data arrived +      if (!m_rfnoc_chdr_tvalid[port]) break; +    end + +    // Dump all the packets that were received +    while (blk_ctrl.num_received(port)) begin  +      blk_ctrl.recv_items(port, items); +    end +  endtask : flush_output + + +  // Test a read/write register for correct functionality +  // +  //   port          : Replay block port to use +  //   addr          : Register byte address +  //   mask          : Mask of the bits we expect to be writable +  //   initial_value : Value we expect to read initially +  // +  task automatic test_read_write_reg( +    int          port, +    bit   [19:0] addr, +    bit   [31:0] mask = 32'hFFFFFFFF, +    logic [31:0] initial_value = '0 +  ); +    string       err_msg; +    logic [31:0] value; +    logic [31:0] expected; + +    err_msg = $sformatf("Register 0x%X failed read/write test: ", addr); + +    // Check initial value +    expected = initial_value; +    read_reg(port, addr, value); +    `ASSERT_ERROR(value === expected, {err_msg, "initial value"}); + +    // Write maximum value +    expected = (initial_value & ~mask) | mask; +    write_reg(port, addr, '1); +    read_reg(port, addr, value); +    `ASSERT_ERROR(value === expected, {err_msg, "write max value"}); + +    // Test writing 0 +    expected = (initial_value & ~mask); +    write_reg(port, addr, '0); +    read_reg(port, addr, value); +    `ASSERT_ERROR(value === expected, {err_msg, "write zero"}); + +    // Restore original value +    write_reg(port, addr, initial_value); +  endtask : test_read_write_reg + + +  // Run the block using the indicated settings and verify the output. +  task automatic run_waveform( +    int                 port, +    logic signed [15:0] gain        = 16'h7FFF,  // 0.99997 +    logic         [2:0] mode        = WAVE_CONST, +    int                 num_packets = 1, +    int                 spp         = SPP, +    logic signed [15:0] const_re    = 16'h7FFF,  // 0.99997 +    logic signed [15:0] const_im    = 16'h7FFF,  // 0.99997 +    logic signed [15:0] phase_inc   = real_to_fixed(0.5, 13), //real_to_fixed(2.0/16, 13), // 2*pi/16 radians +    logic signed [15:0] cart_x      = real_to_fixed(1.0, 14), +    logic signed [15:0] cart_y      = real_to_fixed(0.0, 14) +  ); +    write_reg(port, REG_SPP, spp); +    write_reg(port, REG_WAVEFORM, mode); +    write_reg(port, REG_GAIN, gain); +    if (mode == WAVE_CONST) begin +      write_reg(port, REG_CONSTANT, {const_re, const_im}); +    end else if (mode == WAVE_SINE) begin +      write_reg(port, REG_PHASE_INC, phase_inc); +      write_reg(port, REG_CARTESIAN, {cart_y, cart_x}); +    end +    write_reg(port, REG_ENABLE, 1); + +    for (int packet_count = 0; packet_count < num_packets; packet_count++) begin +      item_t items[$]; +      item_t expected_const, expected_sine, actual; + +      // Receive the next packet +      blk_ctrl.recv_items(port, items); + +      // Verify the length +      `ASSERT_ERROR( +        items.size() == spp,  +        "Packet length didn't match configured SPP" +      ); + +      // Verify the payload +      foreach (items[i]) begin +        actual = items[i]; + +        // Determine the expected constant output +        expected_const[31:16] = apply_gain(gain, const_re); +        expected_const[15: 0] = apply_gain(gain, const_im); + +        // Determine the expected sine output +        if (i == 0) begin +          // We have no basis for comparison on the first sample, so don't +          // check it. It will be used to compute the next output. +          expected_sine = actual; +        end else begin +          expected_sine = next_sine_value(items[i-1], phase_inc); +        end + +        // Check the output +        if (mode == WAVE_CONST) begin +          // For the constant, we expect the output to match exactly +          `ASSERT_ERROR( +            actual == expected_const, +            $sformatf("Incorrect constant sample on packet %0d. Expected 0x%X, received 0x%X.", +              packet_count, expected_const, actual) +          ); +        end else if (mode == WAVE_SINE) begin +          // For sine, it's hard to reproduce the rounding behavior of the IP +          // exactly, so we just check if we're close to the expected answer. +          `ASSERT_ERROR( +            samples_are_close(actual, expected_sine), +            $sformatf("Incorrect sine sample on packet %0d. Expected 0x%X, received 0x%X.", +              packet_count, expected_sine, actual) +          ); +        end else if (mode == WAVE_NOISE) begin +          if (i != 0) begin +            // For noise, it's hard to even estimate the output, so make sure +            // it's changing. +            `ASSERT_ERROR(items[i] !== items[i-1], +              $sformatf("Noise output didn't update on packet %0d.Received 0x%X.", +                packet_count, actual) +            ); +          end +        end +        +      end +    end + +    // Disable the output and flush any output +    write_reg(port, REG_ENABLE, 0); +    flush_output(port); +  endtask : run_waveform + + +  // Run the block using the "constant" waveform mode using the indicated +  // settings and verify the output. +  task automatic run_const( +    int  port, +    int  num_packets = 50, +    int  spp         = SPP, +    real gain, +    real re, +    real im +  ); +    logic signed [15:0] fgain, fre, fim;   // Fixed-point versions + +    // Check the ranges +    `ASSERT_FATAL(gain <= MAX_GAIN_R   || gain >= MIN_GAIN_R,  "Gain out of range"); +    `ASSERT_FATAL(re   <= MAX_CONST_R  || re   >= MIN_CONST_R, "Real out of range"); +    `ASSERT_FATAL(im   <= MAX_CONST_R  || im   >= MIN_CONST_R, "Imag out of range"); + +    // Convert arguments to fixed point +    fgain  = real_to_fixed(gain, GAIN_FRAC); +    fre    = real_to_fixed(re, CONST_FRAC); +    fim    = real_to_fixed(im, CONST_FRAC); + +    // Test the waveform +    run_waveform( +      .port(port), +      .gain(fgain), +      .mode(WAVE_CONST), +      .num_packets(num_packets), +      .spp(spp), +      .const_re(fre), +      .const_im(fim) +    ); +  endtask : run_const + + +  // Run the block using the "sine" waveform mode using the indicated settings +  // and verify the output. +  task automatic run_sine( +    int  port, +    int  num_packets = 50, +    int  spp         = SPP, +    real gain, +    real x, +    real y, +    real phase +  ); +    logic signed [15:0] fgain, fx, fy, fphase;   // Fixed-point versions + +    // Check the ranges +    `ASSERT_FATAL(gain  <= MAX_GAIN_R  || gain  >= MIN_GAIN_R,  "Gain out of range"); +    `ASSERT_FATAL(x     <= MAX_CART_R  || x     >= MIN_CART_R,  "X out of range"); +    `ASSERT_FATAL(y     <= MAX_CART_R  || y     >= MIN_CART_R,  "Y out of range"); +    `ASSERT_FATAL(phase <= MAX_PHASE_R || phase >= MIN_PHASE_R, "Phase out of range"); + +    // Convert arguments to fixed point. +    fgain  = real_to_fixed(gain, GAIN_FRAC); +    fx     = real_to_fixed(x, CART_FRAC); +    fy     = real_to_fixed(y, CART_FRAC); +    fphase = real_to_fixed(phase, PHASE_FRAC); + +    // Test the waveform +    run_waveform( +      .port(port), +      .gain(fgain), +      .mode(WAVE_SINE), +      .num_packets(num_packets), +      .spp(spp), +      .cart_x(fx), +      .cart_y(fy), +      .phase_inc(fphase) +    ); +  endtask : run_sine + + +  // Run the block using the "noise" waveform mode using the indicated +  // settings and verify the output. +  task automatic run_noise( +    int  port, +    int  num_packets = 50, +    int  spp         = SPP, +    real gain +  ); +    logic signed [15:0] fgain;   // Fixed-point versions + +    // Check the ranges +    `ASSERT_FATAL(gain <= MAX_GAIN_R || gain >= MIN_GAIN_R, "Gain out of range"); + +    // Convert arguments to fixed point +    fgain = real_to_fixed(gain, GAIN_FRAC); + +    // Test the waveform +    run_waveform( +      .port(port), +      .gain(fgain), +      .mode(WAVE_NOISE), +      .num_packets(num_packets), +      .spp(spp) +    ); +  endtask : run_noise + + +  //--------------------------------------------------------------------------- +  // Test Procedures +  //--------------------------------------------------------------------------- + +  // Test the min and max allowed values on all registers +  task automatic test_registers(int port); +    test.start_test($sformatf("Test registers (port %0d)", port), 1ms); +    // REG_ENABLE and REG_WAVEFORM will be tested during the other tests +    test_read_write_reg(port, REG_SPP,       {REG_SPP_LEN{1'b1}},       32'd16); +    test_read_write_reg(port, REG_GAIN,      {REG_GAIN_LEN{1'b1}},      32'h7FFF); +    test_read_write_reg(port, REG_CONSTANT,  {REG_CONSTANT_LEN{1'b1}},  32'h0); +    test_read_write_reg(port, REG_PHASE_INC, {REG_PHASE_INC_LEN{1'b1}}, {REG_PHASE_INC_LEN{1'bX}}); +    test_read_write_reg(port, REG_CARTESIAN, {REG_CARTESIAN_LEN{1'b1}}, {REG_CARTESIAN_LEN{1'bX}}); +    test.end_test(); +  endtask : test_registers + + +  // Run through all the waveform modes to make sure they work as expected +  task automatic test_waveforms(int port); +    test.start_test($sformatf("Test waveforms (port %0d)", port), 1ms); +    run_const(.port(port), .gain(0.5), .re(0.25), .im(0.5)); +    run_sine(.port(port), .gain(0.75), .x(0.25), .y(0.5), .phase(2.0/64)); +    run_noise(.port(port), .gain(0.999)); +    test.end_test(); +  endtask : test_waveforms + + +  // Use the constant waveform to test the gain. The gain logic is shared by +  // all modes, but using "const" waveform makes it easy to control the values +  // we're testing. +  task automatic test_gain(int port); +    logic signed [15:0] min_val; +    logic signed [15:0] max_val; + +    test.start_test($sformatf("Test gain (port %0d)", port), 1ms); + +    max_val = 16'sh7FFF; +    min_val = 16'sh8000; +     +    // Test max gain with min and max sample values +    run_waveform(.port(port), .mode(WAVE_CONST), .gain(max_val), +      .const_re(max_val), .const_im(min_val)); +    // Test min gain with max and min sample values +    run_waveform(.port(port), .mode(WAVE_CONST), .gain(min_val), +      .const_re(min_val), .const_im(max_val)); +    // Test zero +    run_waveform(.port(port), .mode(WAVE_CONST), .gain(0), +      .const_re(max_val), .const_im(min_val)); +    // Test 0.5 * 0.5 = 0.25 and 0.25 * 0.5 = 0.125 +    run_waveform( +      .port(port), +      .mode(WAVE_CONST), +      .const_re(real_to_fixed(0.5, CONST_FRAC)),   +      .const_im(real_to_fixed(0.25, CONST_FRAC)), +      .gain(real_to_fixed(0.5, GAIN_FRAC)) +      ); +    test.end_test(); +  endtask : test_gain + + +  // Test the phase setting for the sine waveform +  task automatic test_phase(int port); +    test.start_test($sformatf("Test phase (port %0d)", port), 1ms); +    // Test typical phase +    run_sine(.port(port), .gain(0.5), .x(1.0), .y(0.0), .phase(2.0/16), .num_packets(2)); +    // Test max phase +    run_sine(.port(port), .gain(0.5), .x(1.0), .y(0.0), .phase(MAX_PHASE_R), .num_packets(2)); +    // Test min phase +    run_sine(.port(port), .gain(0.5), .x(1.0), .y(0.0), .phase(MIN_PHASE_R), .num_packets(2)); +    test.end_test(); +  endtask : test_phase + + +  // Use constant waveform to test min and max packet lengths +  task automatic test_packet_length(int port); +    test.start_test($sformatf("Test packet length (port %0d)", port), 1ms); +    run_waveform(.port(port), .spp(2)); +    run_waveform(.port(port), .spp(SPP)); +    run_waveform(.port(port), .spp((2**MTU-1) * (CHDR_W / ITEM_W))); // Test MTU size +    test.end_test(); +  endtask : test_packet_length + + +  //--------------------------------------------------------------------------- +  // Main Test Process +  //--------------------------------------------------------------------------- + +  initial begin : tb_main +    int port; + +    // Initialize the test exec object for this testbench +    test.start_tb( +      $sformatf("rfnoc_block_siggen_tb (CHDR_W = %0d, NUM_PORTS = %0d)", +      CHDR_W, NUM_PORTS)); + +    // Don't start the clocks until after start_tb() returns. This ensures that +    // the clocks aren't toggling while other instances of this testbench are +    // running, which speeds up simulation time. +    rfnoc_chdr_clk_gen.start(); +    rfnoc_ctrl_clk_gen.start(); +    ce_clk_gen.start(); + +    // Start the BFMs running +    blk_ctrl.run(); + +    //-------------------------------- +    // Reset +    //-------------------------------- + +    test.start_test("Flush block then reset it", 10us); +    blk_ctrl.flush_and_reset(); +    test.end_test(); + +    //-------------------------------- +    // Verify Block Info +    //-------------------------------- + +    test.start_test("Verify Block Info", 2us); +    `ASSERT_ERROR(blk_ctrl.get_noc_id() == NOC_ID, "Incorrect NOC_ID Value"); +    `ASSERT_ERROR(blk_ctrl.get_num_data_i() == NUM_PORTS_I, "Incorrect NUM_DATA_I Value"); +    `ASSERT_ERROR(blk_ctrl.get_num_data_o() == NUM_PORTS_O, "Incorrect NUM_DATA_O Value"); +    `ASSERT_ERROR(blk_ctrl.get_mtu() == MTU, "Incorrect MTU Value"); +    test.end_test(); + +    //-------------------------------- +    // Test Sequences +    //-------------------------------- + +    // Run basic test all ports +    for(port = 0; port < NUM_PORTS; port++) begin +      test_registers(port); +      test_waveforms(port); +    end + +    // Run remaining tests on single port +    port = 0; +    test_gain(port); +    test_packet_length(port); +    test_phase(port); + +    //-------------------------------- +    // Finish Up +    //-------------------------------- + +    // Display final statistics and results, but don't call $finish, since we +    // don't want to kill other instances of this testbench that may be +    // running. +    test.end_tb(0); + +    // Kill the clocks to end this instance of the testbench +    rfnoc_chdr_clk_gen.kill(); +    rfnoc_ctrl_clk_gen.kill(); +    ce_clk_gen.kill(); +  end : tb_main + +endmodule : rfnoc_block_siggen_tb + + +`default_nettype wire diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_siggen_core.v b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_siggen_core.v new file mode 100644 index 000000000..bcb664fe0 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_siggen/rfnoc_siggen_core.v @@ -0,0 +1,284 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: rfnoc_siggen_core +// +// Description: +// +//   This module contains the registers and core logic for a single RFNoC +//   Signal Generator module instance. +// + + +module rfnoc_siggen_core ( +  input wire clk, +  input wire rst, + +  // CtrlPort Slave +  input  wire        s_ctrlport_req_wr, +  input  wire        s_ctrlport_req_rd, +  input  wire [19:0] s_ctrlport_req_addr, +  input  wire [31:0] s_ctrlport_req_data, +  output reg         s_ctrlport_resp_ack, +  output reg  [31:0] s_ctrlport_resp_data, + +  // Output data stream +  output wire [31:0] m_tdata, +  output wire        m_tlast, +  output wire        m_tvalid, +  input  wire        m_tready, +  output wire [15:0] m_tlength +); + +  `include "rfnoc_block_siggen_regs.vh" + + +  //--------------------------------------------------------------------------- +  // Registers +  //--------------------------------------------------------------------------- + +  // Define maximum fixed point value for the gain, equal to about 0.9999 +  localparam MAX_GAIN = {REG_GAIN_LEN-1{1'b1}}; + +  reg [   REG_ENABLE_LEN-1:0] reg_enable    = 0; +  reg [      REG_SPP_LEN-1:0] reg_spp       = 16; +  reg [ REG_WAVEFORM_LEN-1:0] reg_waveform  = WAVE_CONST; +  reg [     REG_GAIN_LEN-1:0] reg_gain      = MAX_GAIN; +  reg [ REG_CONSTANT_LEN-1:0] reg_constant  = 0; +  reg [REG_PHASE_INC_LEN-1:0] reg_phase_inc; +  reg [REG_CARTESIAN_LEN-1:0] reg_cartesian; + +  reg reg_phase_inc_stb; +  reg reg_cartesian_stb; + +  always @(posedge clk) begin +    if (rst) begin +      reg_enable           <= 0; +      reg_spp              <= 16; +      reg_waveform         <= WAVE_CONST; +      reg_gain             <= MAX_GAIN; +      reg_constant         <= 0; +      reg_phase_inc        <= 'bX; +      reg_cartesian        <= 'bX; +      s_ctrlport_resp_ack  <= 1'b0; +      s_ctrlport_resp_data <= 'bX; +      reg_phase_inc_stb    <= 1'b0; +      reg_cartesian_stb    <= 1'b0; +    end else begin + +      // Default assignments +      s_ctrlport_resp_ack  <= 1'b0; +      s_ctrlport_resp_data <= 0; +      reg_phase_inc_stb    <= 1'b0; +      reg_cartesian_stb    <= 1'b0; + +      // Handle register writes +      if (s_ctrlport_req_wr) begin +        s_ctrlport_resp_ack <= 1; +        case (s_ctrlport_req_addr) +          REG_ENABLE    : reg_enable    <= s_ctrlport_req_data[REG_ENABLE_LEN-1:0]; +          REG_SPP       : reg_spp       <= s_ctrlport_req_data[REG_SPP_LEN-1:0]; +          REG_WAVEFORM  : reg_waveform  <= s_ctrlport_req_data[REG_WAVEFORM_LEN-1:0]; +          REG_GAIN      : reg_gain      <= s_ctrlport_req_data[REG_GAIN_LEN-1:0]; +          REG_CONSTANT  : reg_constant  <= s_ctrlport_req_data[REG_CONSTANT_LEN-1:0]; +          REG_PHASE_INC : begin +            reg_phase_inc     <= s_ctrlport_req_data[REG_PHASE_INC_LEN-1:0]; +            reg_phase_inc_stb <= 1'b1; +          end +          REG_CARTESIAN : begin +            reg_cartesian     <= s_ctrlport_req_data[REG_CARTESIAN_LEN-1:0]; +            reg_cartesian_stb <= 1'b1; +          end +        endcase +      end + +      // Handle register reads +      if (s_ctrlport_req_rd) begin +        s_ctrlport_resp_ack <= 1; +        case (s_ctrlport_req_addr) +          REG_ENABLE    : s_ctrlport_resp_data[REG_ENABLE_LEN-1:0]     <= reg_enable; +          REG_SPP       : s_ctrlport_resp_data[REG_SPP_LEN-1:0]        <= reg_spp; +          REG_WAVEFORM  : s_ctrlport_resp_data[REG_WAVEFORM_LEN-1:0]   <= reg_waveform; +          REG_GAIN      : s_ctrlport_resp_data[REG_GAIN_LEN-1:0]       <= reg_gain; +          REG_CONSTANT  : s_ctrlport_resp_data[REG_CONSTANT_LEN-1:0]   <= reg_constant; +          REG_PHASE_INC : s_ctrlport_resp_data[REG_PHASE_INC_LEN-1:0]  <= reg_phase_inc; +          REG_CARTESIAN : s_ctrlport_resp_data[REG_CARTESIAN_LEN-1:0]  <= reg_cartesian; +        endcase +      end +    end +  end + + +  //--------------------------------------------------------------------------- +  // Waveform Generation +  //--------------------------------------------------------------------------- + +  wire [31:0]  axis_sine_tdata; +  wire         axis_sine_tvalid; +  wire         axis_sine_tready; +  wire [31:0]  axis_const_tdata; +  wire         axis_const_tvalid; +  wire         axis_const_tready; +  wire [31:0]  axis_noise_tdata; +  wire         axis_noise_tvalid; +  wire         axis_noise_tready; + +  //------------------------------------ +  // Sine waveform generation +  //------------------------------------ + +  // Convert the registers writes to settings bus transactions. Only one +  // register strobe will assert at a time. +  wire        sine_set_stb  = reg_cartesian_stb | reg_phase_inc_stb; +  wire [31:0] sine_set_data = reg_cartesian_stb ? reg_cartesian : reg_phase_inc; +  wire [ 7:0] sine_set_addr = reg_cartesian_stb; + +  sine_tone #( +    .WIDTH             (32), +    .SR_PHASE_INC_ADDR (0), +    .SR_CARTESIAN_ADDR (1) +  ) sine_tone_i ( +    .clk      (clk), +    .reset    (rst), +    .clear    (1'b0), +    .enable   (1'b1), +    .set_stb  (sine_set_stb), +    .set_data (sine_set_data), +    .set_addr (sine_set_addr), +    .o_tdata  (axis_sine_tdata), +    .o_tlast  (), +    .o_tvalid (axis_sine_tvalid), +    .o_tready (axis_sine_tready) +  ); + +  //------------------------------------ +  // Constant waveform generation +  //------------------------------------ + +  assign axis_const_tdata  = reg_constant; +  assign axis_const_tvalid = 1'b1; + +  //------------------------------------ +  // Noise waveform generation +  //------------------------------------ + +  assign axis_noise_tvalid = 1'b1; + +  // Random number generator +  rng rng_i ( +    .clk (clk), +    .rst (rst), +    .out (axis_noise_tdata) +  ); + + +  //--------------------------------------------------------------------------- +  // Waveform Selection +  //--------------------------------------------------------------------------- + +  wire [31:0]  axis_mux_tdata; +  wire         axis_mux_tvalid; +  wire         axis_mux_tready; + +  axi_mux_select #( +    .WIDTH          (32), +    .SIZE           (3), +    .SWITCH_ON_LAST (0) +  ) axi_mux_select_i ( +    .clk      (clk), +    .reset    (rst), +    .clear    (1'b0), +    .select   (reg_waveform), +    .i_tdata  ({axis_noise_tdata, axis_sine_tdata, axis_const_tdata}), +    .i_tlast  ({3'd0}),   // Length controlled by SPP register +    .i_tvalid ({axis_noise_tvalid, axis_sine_tvalid, axis_const_tvalid}), +    .i_tready ({axis_noise_tready, axis_sine_tready, axis_const_tready}), +    .o_tdata  (axis_mux_tdata), +    .o_tlast  (), +    .o_tvalid (axis_mux_tvalid), +    .o_tready (axis_mux_tready) +  ); + + +  //--------------------------------------------------------------------------- +  // Gain +  //--------------------------------------------------------------------------- + +  wire [63:0]  axis_gain_tdata; +  wire         axis_gain_tvalid; +  wire         axis_gain_tready; +  wire [31:0]  axis_round_tdata; +  wire         axis_round_tvalid; +  wire         axis_round_tready; + +  mult_rc #( +    .WIDTH_REAL (16), +    .WIDTH_CPLX (16), +    .WIDTH_P    (32), +    .DROP_TOP_P (5), +    .LATENCY    (4) +  ) mult_rc_i ( +    .clk         (clk), +    .reset       (rst), +    .real_tdata  (reg_gain), +    .real_tlast  (1'b0), +    .real_tvalid (1'b1), +    .real_tready (), +    .cplx_tdata  (axis_mux_tdata), +    .cplx_tlast  (1'b0), +    .cplx_tvalid (axis_mux_tvalid), +    .cplx_tready (axis_mux_tready), +    .p_tdata     (axis_gain_tdata), +    .p_tlast     (), +    .p_tvalid    (axis_gain_tvalid), +    .p_tready    (axis_gain_tready) +  ); + +  axi_round_and_clip_complex #( +    .WIDTH_IN  (32), +    .WIDTH_OUT (16), +    .CLIP_BITS (1) +  ) axi_round_and_clip_complex_i ( +    .clk      (clk), +    .reset    (rst), +    .i_tdata  (axis_gain_tdata), +    .i_tlast  (1'b0), +    .i_tvalid (axis_gain_tvalid), +    .i_tready (axis_gain_tready), +    .o_tdata  (axis_round_tdata), +    .o_tlast  (), +    .o_tvalid (axis_round_tvalid), +    .o_tready (axis_round_tready) +  ); + + +  //--------------------------------------------------------------------------- +  // Packet Length Control +  //--------------------------------------------------------------------------- + +  wire [REG_SPP_LEN-1:0] m_tlength_samples; + +  assign m_tlength = { m_tlength_samples, 2'b0 };   // 4 bytes per sample + +  axis_packetize #( +    .DATA_W  (32), +    .SIZE_W  (REG_SPP_LEN), +    .FLUSH   (1) +  ) axis_packetize_i ( +    .clk      (clk), +    .rst      (rst), +    .gate     (~reg_enable), +    .size     (reg_spp), +    .i_tdata  (axis_round_tdata), +    .i_tvalid (axis_round_tvalid), +    .i_tready (axis_round_tready), +    .o_tdata  (m_tdata), +    .o_tlast  (m_tlast), +    .o_tvalid (m_tvalid), +    .o_tready (m_tready), +    .o_tuser  (m_tlength_samples) +  ); + +endmodule | 
