aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorWade Fife <wade.fife@ettus.com>2020-04-14 16:23:59 -0500
committerWade Fife <wade.fife@ettus.com>2020-04-14 16:37:43 -0500
commitd386c750074f6da4ab86038e2c30a3fe6e0f9d47 (patch)
treee4258b2744e24bf6e829910e66c93a74b8d7603a
parenta8c4f021277cf3b0a0897fa9da0252541512f3a6 (diff)
downloaduhd-d386c750074f6da4ab86038e2c30a3fe6e0f9d47.tar.gz
uhd-d386c750074f6da4ab86038e2c30a3fe6e0f9d47.tar.bz2
uhd-d386c750074f6da4ab86038e2c30a3fe6e0f9d47.zip
rfnoc: Add RFNoC fosphor block
-rw-r--r--fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/Makefile44
-rw-r--r--fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/Makefile.srcs22
-rw-r--r--fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/noc_shell_fosphor.v344
-rw-r--r--fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor.v398
-rw-r--r--fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor_regs.vh186
-rw-r--r--fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor_tb.sv590
-rw-r--r--fpga/usrp3/lib/rfnoc/fosphor/fifo_srl.v2
-rw-r--r--host/include/uhd/rfnoc/CMakeLists.txt1
-rw-r--r--host/include/uhd/rfnoc/blocks/fosphor.yml56
-rw-r--r--host/include/uhd/rfnoc/defaults.hpp2
-rw-r--r--host/include/uhd/rfnoc/fosphor_block_control.hpp383
-rw-r--r--host/lib/rfnoc/CMakeLists.txt1
-rw-r--r--host/lib/rfnoc/fosphor_block_control.cpp418
13 files changed, 2446 insertions, 1 deletions
diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/Makefile b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/Makefile
new file mode 100644
index 000000000..58a2cde0e
--- /dev/null
+++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/Makefile
@@ -0,0 +1,44 @@
+#
+# 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
+BASE_DIR = $(abspath ../../../../top)
+# Include viv_sim_preamble 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 Makefile.srcs
+
+DESIGN_SRCS += $(abspath \
+$(RFNOC_CORE_SRCS) \
+$(RFNOC_UTIL_SRCS) \
+$(RFNOC_OOT_SRCS) \
+)
+
+#-------------------------------------------------
+# Testbench Specific
+#-------------------------------------------------
+SIM_TOP = rfnoc_block_fosphor_tb glbl
+SIM_SRCS = \
+$(abspath rfnoc_block_fosphor_tb.sv) \
+$(VIVADO_PATH)/data/verilog/src/glbl.v \
+
+#-------------------------------------------------
+# 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_fosphor/Makefile.srcs b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/Makefile.srcs
new file mode 100644
index 000000000..620e993e2
--- /dev/null
+++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/Makefile.srcs
@@ -0,0 +1,22 @@
+#
+# 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)))), \
+rfnoc_block_fosphor.v \
+noc_shell_fosphor.v \
+)
diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/noc_shell_fosphor.v b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/noc_shell_fosphor.v
new file mode 100644
index 000000000..dd1845b40
--- /dev/null
+++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/noc_shell_fosphor.v
@@ -0,0 +1,344 @@
+//
+// Copyright 2020 Ettus Research, A National Instruments Brand
+//
+// SPDX-License-Identifier: LGPL-3.0-or-later
+//
+// Module: noc_shell_fosphor
+//
+// Description:
+//
+// This is a tool-generated NoC-shell for the fosphor 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_fosphor #(
+ parameter [9:0] THIS_PORTID = 10'd0,
+ parameter CHDR_W = 64,
+ parameter [5:0] MTU = 10
+) (
+ //---------------------
+ // 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 [(2)*CHDR_W-1:0] m_rfnoc_chdr_tdata,
+ output wire [(2)-1:0] m_rfnoc_chdr_tlast,
+ output wire [(2)-1:0] m_rfnoc_chdr_tvalid,
+ input wire [(2)-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: fft_in
+ output wire [32*1-1:0] m_fft_in_axis_tdata,
+ output wire [1-1:0] m_fft_in_axis_tkeep,
+ output wire m_fft_in_axis_tlast,
+ output wire m_fft_in_axis_tvalid,
+ input wire m_fft_in_axis_tready,
+ output wire [63:0] m_fft_in_axis_ttimestamp,
+ output wire m_fft_in_axis_thas_time,
+ output wire [15:0] m_fft_in_axis_tlength,
+ output wire m_fft_in_axis_teov,
+ output wire m_fft_in_axis_teob,
+ // Data Stream from User Logic: hist
+ input wire [8*4-1:0] s_hist_axis_tdata,
+ input wire [3:0] s_hist_axis_tkeep,
+ input wire s_hist_axis_tlast,
+ input wire s_hist_axis_tvalid,
+ output wire s_hist_axis_tready,
+ input wire [63:0] s_hist_axis_ttimestamp,
+ input wire s_hist_axis_thas_time,
+ input wire [15:0] s_hist_axis_tlength,
+ input wire s_hist_axis_teov,
+ input wire s_hist_axis_teob,
+ // Data Stream from User Logic: wf
+ input wire [8*4-1:0] s_wf_axis_tdata,
+ input wire [3:0] s_wf_axis_tkeep,
+ input wire s_wf_axis_tlast,
+ input wire s_wf_axis_tvalid,
+ output wire s_wf_axis_tready,
+ input wire [63:0] s_wf_axis_ttimestamp,
+ input wire s_wf_axis_thas_time,
+ input wire [15:0] s_wf_axis_tlength,
+ input wire s_wf_axis_teov,
+ input wire s_wf_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'h666F0000),
+ .NUM_DATA_I (1),
+ .NUM_DATA_O (2),
+ .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
+ //---------------------
+
+ chdr_to_axis_data #(
+ .CHDR_W (CHDR_W),
+ .ITEM_W (32),
+ .NIPC (1),
+ .SYNC_CLKS (0),
+ .INFO_FIFO_SIZE ($clog2(32)),
+ .PYLD_FIFO_SIZE ($clog2(32))
+ ) chdr_to_axis_data_in_fft_in (
+ .axis_chdr_clk (rfnoc_chdr_clk),
+ .axis_chdr_rst (rfnoc_chdr_rst),
+ .axis_data_clk (axis_data_clk),
+ .axis_data_rst (axis_data_rst),
+ .s_axis_chdr_tdata (s_rfnoc_chdr_tdata[(0)*CHDR_W+:CHDR_W]),
+ .s_axis_chdr_tlast (s_rfnoc_chdr_tlast[0]),
+ .s_axis_chdr_tvalid (s_rfnoc_chdr_tvalid[0]),
+ .s_axis_chdr_tready (s_rfnoc_chdr_tready[0]),
+ .m_axis_tdata (m_fft_in_axis_tdata),
+ .m_axis_tkeep (m_fft_in_axis_tkeep),
+ .m_axis_tlast (m_fft_in_axis_tlast),
+ .m_axis_tvalid (m_fft_in_axis_tvalid),
+ .m_axis_tready (m_fft_in_axis_tready),
+ .m_axis_ttimestamp (m_fft_in_axis_ttimestamp),
+ .m_axis_thas_time (m_fft_in_axis_thas_time),
+ .m_axis_tlength (m_fft_in_axis_tlength),
+ .m_axis_teov (m_fft_in_axis_teov),
+ .m_axis_teob (m_fft_in_axis_teob),
+ .flush_en (data_i_flush_en),
+ .flush_timeout (data_i_flush_timeout),
+ .flush_active (data_i_flush_active[0]),
+ .flush_done (data_i_flush_done[0])
+ );
+
+ //---------------------
+ // Output Data Paths
+ //---------------------
+
+ axis_data_to_chdr #(
+ .CHDR_W (CHDR_W),
+ .ITEM_W (8),
+ .NIPC (4),
+ .SYNC_CLKS (0),
+ .INFO_FIFO_SIZE ($clog2(32)),
+ .PYLD_FIFO_SIZE ($clog2(32)),
+ .MTU (MTU),
+ .SIDEBAND_AT_END (0)
+ ) axis_data_to_chdr_out_hist (
+ .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)*CHDR_W+:CHDR_W]),
+ .m_axis_chdr_tlast (m_rfnoc_chdr_tlast[0]),
+ .m_axis_chdr_tvalid (m_rfnoc_chdr_tvalid[0]),
+ .m_axis_chdr_tready (m_rfnoc_chdr_tready[0]),
+ .s_axis_tdata (s_hist_axis_tdata),
+ .s_axis_tkeep (s_hist_axis_tkeep),
+ .s_axis_tlast (s_hist_axis_tlast),
+ .s_axis_tvalid (s_hist_axis_tvalid),
+ .s_axis_tready (s_hist_axis_tready),
+ .s_axis_ttimestamp (s_hist_axis_ttimestamp),
+ .s_axis_thas_time (s_hist_axis_thas_time),
+ .s_axis_tlength (s_hist_axis_tlength),
+ .s_axis_teov (s_hist_axis_teov),
+ .s_axis_teob (s_hist_axis_teob),
+ .flush_en (data_o_flush_en),
+ .flush_timeout (data_o_flush_timeout),
+ .flush_active (data_o_flush_active[0]),
+ .flush_done (data_o_flush_done[0])
+ );
+
+ axis_data_to_chdr #(
+ .CHDR_W (CHDR_W),
+ .ITEM_W (8),
+ .NIPC (4),
+ .SYNC_CLKS (0),
+ .INFO_FIFO_SIZE ($clog2(32)),
+ .PYLD_FIFO_SIZE ($clog2(32)),
+ .MTU (MTU),
+ .SIDEBAND_AT_END (0)
+ ) axis_data_to_chdr_out_wf (
+ .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[(1)*CHDR_W+:CHDR_W]),
+ .m_axis_chdr_tlast (m_rfnoc_chdr_tlast[1]),
+ .m_axis_chdr_tvalid (m_rfnoc_chdr_tvalid[1]),
+ .m_axis_chdr_tready (m_rfnoc_chdr_tready[1]),
+ .s_axis_tdata (s_wf_axis_tdata),
+ .s_axis_tkeep (s_wf_axis_tkeep),
+ .s_axis_tlast (s_wf_axis_tlast),
+ .s_axis_tvalid (s_wf_axis_tvalid),
+ .s_axis_tready (s_wf_axis_tready),
+ .s_axis_ttimestamp (s_wf_axis_ttimestamp),
+ .s_axis_thas_time (s_wf_axis_thas_time),
+ .s_axis_tlength (s_wf_axis_tlength),
+ .s_axis_teov (s_wf_axis_teov),
+ .s_axis_teob (s_wf_axis_teob),
+ .flush_en (data_o_flush_en),
+ .flush_timeout (data_o_flush_timeout),
+ .flush_active (data_o_flush_active[1]),
+ .flush_done (data_o_flush_done[1])
+ );
+
+endmodule // noc_shell_fosphor
+
+
+`default_nettype wire
diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor.v b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor.v
new file mode 100644
index 000000000..40112b05e
--- /dev/null
+++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor.v
@@ -0,0 +1,398 @@
+//
+// Copyright 2020 Ettus Research, A National Instruments Brand
+//
+// SPDX-License-Identifier: LGPL-3.0-or-later
+//
+// Module: rfnoc_block_fosphor
+//
+// Description:
+//
+// Fosphor RFNoC block. This block accepts packets containing FFT data (one
+// FFT output per packet) and generates two output data streams, one
+// containing histogram data and the other containing waterfall plot data.
+//
+// Each set of waterfall data is output as a single packet. The frequency of
+// waterfall output depends on the waterfall decimation register setting.
+//
+// Each set of histogram data is output as a burst of 64 packets, followed by
+// a single packet of max values and then a single packet of average values.
+// The frequency of waterfall output bursts depends on the waterfall
+// decimation register setting.
+//
+// For all outputs, the packets contain byte values, and the number of bytes
+// in each packet matches the number of 4-byte sc16 samples in the input
+// packets (i.e., the FFT size). In other words, the output packet size is
+// 1/4th the input packet size.
+//
+// Many registers control the visual effects and behavior of the waterfall
+// and histogram. See the register descriptions in
+// rfnoc_block_fosphor_regs.vh 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).
+//
+
+`default_nettype none
+
+
+module rfnoc_block_fosphor #(
+ parameter [9:0] THIS_PORTID = 10'd0,
+ parameter CHDR_W = 64,
+ parameter [5:0] MTU = 10
+) (
+ // 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 [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 [2*CHDR_W-1:0] m_rfnoc_chdr_tdata,
+ output wire [ (2)-1:0] m_rfnoc_chdr_tlast,
+ output wire [ (2)-1:0] m_rfnoc_chdr_tvalid,
+ input wire [ (2)-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 "../../core/rfnoc_chdr_utils.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;
+ reg m_ctrlport_resp_ack;
+ reg [31:0] m_ctrlport_resp_data;
+ // Data Stream to User Logic: in
+ wire [32*1-1:0] in_tdata;
+ wire in_tlast;
+ wire in_tvalid;
+ wire in_tready;
+ wire [15:0] in_tlength;
+ // Data Stream from User Logic: hist
+ wire [8*4-1:0] hist_tdata;
+ wire hist_tlast;
+ wire hist_tvalid;
+ wire hist_tready;
+ wire [15:0] hist_tlength;
+ wire hist_teob;
+ // Data Stream from User Logic: wf
+ wire [8*4-1:0] wf_tdata;
+ wire wf_tlast;
+ wire wf_tvalid;
+ wire wf_tready;
+ wire [15:0] wf_tlength;
+
+ //---------------------------------------------------------------------------
+ // NoC Shell
+ //---------------------------------------------------------------------------
+
+ wire ce_rst;
+
+ noc_shell_fosphor #(
+ .CHDR_W (CHDR_W),
+ .THIS_PORTID (THIS_PORTID),
+ .MTU (MTU)
+ ) noc_shell_fosphor_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 to User Logic: in
+ .m_fft_in_axis_tdata (in_tdata),
+ .m_fft_in_axis_tkeep (),
+ .m_fft_in_axis_tlast (in_tlast),
+ .m_fft_in_axis_tvalid (in_tvalid),
+ .m_fft_in_axis_tready (in_tready),
+ .m_fft_in_axis_ttimestamp (),
+ .m_fft_in_axis_thas_time (),
+ .m_fft_in_axis_tlength (in_tlength),
+ .m_fft_in_axis_teov (),
+ .m_fft_in_axis_teob (),
+ // Data Stream from User Logic: hist
+ .s_hist_axis_tdata (hist_tdata),
+ .s_hist_axis_tkeep (4'hF),
+ .s_hist_axis_tlast (hist_tlast),
+ .s_hist_axis_tvalid (hist_tvalid),
+ .s_hist_axis_tready (hist_tready),
+ .s_hist_axis_ttimestamp (64'b0),
+ .s_hist_axis_thas_time (1'b0),
+ .s_hist_axis_tlength (hist_tlength),
+ .s_hist_axis_teov (1'b0),
+ .s_hist_axis_teob (hist_teob),
+ // Data Stream from User Logic: wf
+ .s_wf_axis_tdata (wf_tdata),
+ .s_wf_axis_tkeep (4'hF),
+ .s_wf_axis_tlast (wf_tlast),
+ .s_wf_axis_tvalid (wf_tvalid),
+ .s_wf_axis_tready (wf_tready),
+ .s_wf_axis_ttimestamp (64'b0),
+ .s_wf_axis_thas_time (1'b0),
+ .s_wf_axis_tlength (wf_tlength),
+ .s_wf_axis_teov (1'b0),
+ .s_wf_axis_teob (1'b0)
+ );
+
+
+ //---------------------------------------------------------------------------
+ // Registers
+ //---------------------------------------------------------------------------
+
+ `include "rfnoc_block_fosphor_regs.vh"
+
+ // Configuration registers
+ reg [REG_ENABLE_LEN-1:0] cfg_enable;
+ reg clear_req;
+ reg fosphor_rst = 1;
+ reg [REG_RANDOM_LEN-1:0] cfg_random;
+ reg [REG_HIST_DECIM_LEN-1:0] cfg_hist_decim;
+ reg [REG_OFFSET_LEN-1:0] cfg_offset;
+ reg [REG_SCALE_LEN-1:0] cfg_scale;
+ reg [REG_TRISE_LEN-1:0] cfg_trise;
+ reg [REG_TDECAY_LEN-1:0] cfg_tdecay;
+ reg [REG_ALPHA_LEN-1:0] cfg_alpha;
+ reg [REG_EPSILON_LEN-1:0] cfg_epsilon;
+ reg [REG_WF_DIV_LEN-1:0] cfg_wf_div;
+ reg cfg_wf_mode;
+ reg [REG_WF_DECIM_LEN-1:0] cfg_wf_decim;
+ reg cfg_hist_decim_changed;
+ reg cfg_wf_decim_changed;
+
+ always @(posedge ce_clk) begin
+ if (ce_rst) begin
+ m_ctrlport_resp_ack <= 0;
+ m_ctrlport_resp_data <= 'bX;
+ cfg_enable <= 0;
+ clear_req <= 0;
+ fosphor_rst <= 1;
+ cfg_random <= 0;
+ cfg_hist_decim <= 0;
+ cfg_hist_decim_changed <= 0;
+ cfg_offset <= 0;
+ cfg_scale <= 0;
+ cfg_trise <= 0;
+ cfg_tdecay <= 0;
+ cfg_alpha <= 0;
+ cfg_epsilon <= 0;
+ cfg_wf_div <= 0;
+ cfg_wf_mode <= 0;
+ cfg_wf_decim <= 0;
+ cfg_wf_decim_changed <= 0;
+ end else begin
+ // Default assignments
+ m_ctrlport_resp_ack <= 0;
+ m_ctrlport_resp_data <= 0;
+ cfg_hist_decim_changed <= 0;
+ cfg_wf_decim_changed <= 0;
+ clear_req <= 0;
+ fosphor_rst <= 0;
+ m_ctrlport_resp_data <= 0;
+ m_ctrlport_resp_ack <= 0;
+
+ // Handle register writes
+ if (m_ctrlport_req_wr) begin
+ m_ctrlport_resp_ack <= 1;
+ case (m_ctrlport_req_addr)
+ REG_ENABLE : cfg_enable <= m_ctrlport_req_data[0+:REG_ENABLE_LEN];
+ REG_CLEAR : begin
+ fosphor_rst <= m_ctrlport_req_data[REG_RESET_POS];
+ clear_req <= m_ctrlport_req_data[REG_CLEAR_POS];
+ end
+ REG_RANDOM : cfg_random <= m_ctrlport_req_data[0+:REG_RANDOM_LEN];
+ REG_HIST_DECIM : begin
+ cfg_hist_decim <= m_ctrlport_req_data[0+:REG_HIST_DECIM_LEN];
+ cfg_hist_decim_changed <= 1'b1;
+ end
+ REG_OFFSET : cfg_offset <= m_ctrlport_req_data[0+:REG_OFFSET_LEN];
+ REG_SCALE : cfg_scale <= m_ctrlport_req_data[0+:REG_SCALE_LEN];
+ REG_TRISE : cfg_trise <= m_ctrlport_req_data[0+:REG_TRISE_LEN];
+ REG_TDECAY : cfg_tdecay <= m_ctrlport_req_data[0+:REG_TDECAY_LEN];
+ REG_ALPHA : cfg_alpha <= m_ctrlport_req_data[0+:REG_ALPHA_LEN];
+ REG_EPSILON : cfg_epsilon <= m_ctrlport_req_data[0+:REG_EPSILON_LEN];
+ REG_WF_CTRL : begin
+ cfg_wf_mode <= m_ctrlport_req_data[REG_WF_MODE_POS];
+ cfg_wf_div <= m_ctrlport_req_data[REG_WF_DIV_POS+:REG_WF_DIV_LEN];
+ end
+ REG_WF_DECIM : begin
+ cfg_wf_decim <= m_ctrlport_req_data[0+:REG_WF_DECIM_LEN];
+ cfg_wf_decim_changed <= 1'b1;
+ end
+ endcase
+
+ // Handle register reads
+ end else if (m_ctrlport_req_rd) begin
+ m_ctrlport_resp_ack <= 1;
+ case (m_ctrlport_req_addr)
+ REG_ENABLE : m_ctrlport_resp_data[0+:REG_ENABLE_LEN] <= cfg_enable;
+ REG_RANDOM : m_ctrlport_resp_data[0+:REG_RANDOM_LEN] <= cfg_random;
+ REG_HIST_DECIM : m_ctrlport_resp_data[0+:REG_HIST_DECIM_LEN] <= cfg_hist_decim;
+ REG_OFFSET : m_ctrlport_resp_data[0+:REG_OFFSET_LEN] <= cfg_offset;
+ REG_SCALE : m_ctrlport_resp_data[0+:REG_SCALE_LEN] <= cfg_scale;
+ REG_TRISE : m_ctrlport_resp_data[0+:REG_TRISE_LEN] <= cfg_trise;
+ REG_TDECAY : m_ctrlport_resp_data[0+:REG_TDECAY_LEN] <= cfg_tdecay;
+ REG_ALPHA : m_ctrlport_resp_data[0+:REG_ALPHA_LEN] <= cfg_alpha;
+ REG_EPSILON : m_ctrlport_resp_data[0+:REG_EPSILON_LEN] <= cfg_epsilon;
+ REG_WF_CTRL : begin
+ m_ctrlport_resp_data[REG_WF_MODE_POS] <= cfg_wf_mode;
+ m_ctrlport_resp_data[REG_WF_DIV_POS+:REG_WF_DIV_LEN] <= cfg_wf_div;
+ end
+ REG_WF_DECIM : m_ctrlport_resp_data[0+:REG_WF_DECIM_LEN] <= cfg_wf_decim;
+ endcase
+ end
+ end
+ end
+
+
+ //---------------------------------------------------------------------------
+ // Output Packet Length Register
+ //---------------------------------------------------------------------------
+
+ // The output length is always 1/4th the input length, since
+ // we output one byte for each sc16 input.
+ reg [15:0] out_packet_length;
+ reg start_of_packet = 1'b1;
+
+ assign wf_tlength = out_packet_length;
+ assign hist_tlength = out_packet_length;
+
+ always @(posedge ce_clk) begin
+ if (ce_rst) begin
+ start_of_packet <= 1'b1;
+ out_packet_length <= 'bX;
+ end else begin
+ if (in_tvalid && in_tready) begin
+ start_of_packet <= in_tlast;
+ if (start_of_packet) begin
+ out_packet_length <= in_tlength / 4;
+ end
+ end
+ end
+ end
+
+
+ //---------------------------------------------------------------------------
+ // Fosphor Core
+ //---------------------------------------------------------------------------
+
+ wire hist_tvalid_tmp;
+ wire hist_tready_tmp;
+ wire wf_tvalid_tmp;
+ wire wf_tready_tmp;
+
+ f15_core f15_core_inst (
+ .clk (ce_clk),
+ .reset (fosphor_rst),
+ .clear_req (clear_req),
+ .cfg_random (cfg_random),
+ .cfg_offset (cfg_offset),
+ .cfg_scale (cfg_scale),
+ .cfg_trise (cfg_trise),
+ .cfg_tdecay (cfg_tdecay),
+ .cfg_alpha (cfg_alpha),
+ .cfg_epsilon (cfg_epsilon),
+ .cfg_decim (cfg_hist_decim),
+ .cfg_decim_changed (cfg_hist_decim_changed),
+ .cfg_wf_div (cfg_wf_div),
+ .cfg_wf_mode (cfg_wf_mode),
+ .cfg_wf_decim (cfg_wf_decim),
+ .cfg_wf_decim_changed (cfg_wf_decim_changed),
+ .i_tdata (in_tdata),
+ .i_tlast (in_tlast),
+ .i_tvalid (in_tvalid),
+ .i_tready (in_tready),
+ .o_hist_tdata (hist_tdata),
+ .o_hist_tlast (hist_tlast),
+ .o_hist_tvalid (hist_tvalid_tmp),
+ .o_hist_tready (hist_tready_tmp),
+ .o_hist_teob (hist_teob),
+ .o_wf_tdata (wf_tdata),
+ .o_wf_tlast (wf_tlast),
+ .o_wf_tvalid (wf_tvalid_tmp),
+ .o_wf_tready (wf_tready_tmp)
+ );
+
+ // Enable/disable logic. All we're doing here is discarding the output for
+ // the "disabled" output. It is still generated internally.
+ assign hist_tready_tmp = hist_tready | ~cfg_enable[0];
+ assign hist_tvalid = hist_tvalid_tmp & cfg_enable[0];
+ assign wf_tready_tmp = wf_tready | ~cfg_enable[1];
+ assign wf_tvalid = wf_tvalid_tmp & cfg_enable[1];
+
+endmodule // rfnoc_block_fosphor
+
+
+`default_nettype wire
diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor_regs.vh b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor_regs.vh
new file mode 100644
index 000000000..eca372169
--- /dev/null
+++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor_regs.vh
@@ -0,0 +1,186 @@
+//
+// Copyright 2020 Ettus Research, A National Instruments Brand
+//
+// SPDX-License-Identifier: LGPL-3.0-or-later
+//
+// Module: rfnoc_block_fosphor_vh (Header)
+//
+// Description:
+//
+// Fosphor RFNoC block register descriptions. See the block controller
+// (fosphor_block_control.hpp) for additional documentation.
+//
+
+
+// REG_ENABLE (R/W)
+//
+// This register enables or disables the stream of histogram data from the
+// block. The streams are always generated internally and this register causes
+// them to be discarded or not. This register should only be updated when the
+// Fosphor block is idle to avoid enabling it while packets are in flight.
+//
+// [1] : Enable waterfall output stream
+// [0] : Enable histogram output stream
+//
+localparam REG_ENABLE = 'h00;
+//
+localparam REG_ENABLE_LEN = 2;
+
+// REG_CLEAR (W)
+//
+// Controls reset of the Fosphor IP and clearing of the accumulated history
+// (average and max hold values). Note that reset is not a superset of clear,
+// and both reset and clear should not be asserted in the same write. To reset
+// and clear, set only the reset bit in the first write then set only the clear
+// bit in a second write.
+//
+// [1] : Reset (strobe). This is a self-clearing strobe bit to reset the
+// internal Fosphor core.
+// [0] : Clear (strobe). This is a self-clearing strobe bit to clear the
+// history of the fosphor core.
+//
+localparam REG_CLEAR = 'h04;
+//
+localparam REG_CLEAR_LEN = 2;
+//
+localparam REG_RESET_POS = 1;
+localparam REG_CLEAR_POS = 0;
+
+// REG_RANDOM (R/W)
+//
+// Enables or disables the addition of random noise and/or dithering to the
+// incoming signal.
+//
+// [1] : Noise enable. Adds random numbers to the signal.
+// [0] : Dither enable. Randomizes the least-significant bits of the signal.
+//
+localparam REG_RANDOM = 'h08;
+//
+localparam REG_RANDOM_LEN = 2;
+
+// REG_HIST_DECIM (R/W)
+//
+// [11:0] : Histogram decimation. This determines the amount of histogram data
+// that is output relative to the amount of input FFT data. The actual
+// decimation is N:1 where N=VALUE+2. That is, you'll get 1 histogram
+// output packet for ever N FFT packets received, on average. However,
+// histogram data is always output as a burst of 66 packets (64
+// histogram, 1 maximum value, 1 average value).
+//
+localparam REG_HIST_DECIM = 'h0C;
+//
+localparam REG_HIST_DECIM_LEN = 12;
+
+// REG_OFFSET (R/W)
+//
+// Histogram offset to apply to the FFT power levels before determining the
+// appropriate histogram bin.
+//
+// [15:0] : Offset
+//
+localparam REG_OFFSET = 'h10;
+//
+localparam REG_OFFSET_LEN = 16;
+
+// REG_SCALE (R/W)
+//
+// Histogram scaling factor. Controls the scaling factor to apply to FFT power
+// levels before determining the appropriate histogram bin. The scaling factor
+// is scale / 256.
+//
+// [15:0] : Scale
+//
+localparam REG_SCALE = 'h14;
+//
+localparam REG_SCALE_LEN = 16;
+
+// REG_TRISE (R/W)
+//
+// Histogram rise rate. Controls the rate at which the hit count in each
+// frequency bin increases when there are hits in the particular bin. The
+// higher the value, the more quickly the values increase.
+//
+// [15:0] : Rise time
+//
+localparam REG_TRISE = 'h18;
+//
+localparam REG_TRISE_LEN = 16;
+
+// REG_TDECAY (R/W)
+//
+// Histogram decay rate. Controls the rate at which the hit count in each
+// frequency and power bin decreases when there are no hits in a particular
+// bin. The higher the value, the more quickly the values decrease.
+//
+// [15:0] : Decay time
+//
+localparam REG_TDECAY = 'h1C;
+//
+localparam REG_TDECAY_LEN = 16;
+
+// REG_ALPHA (R/W)
+//
+// Controls the weighting to be applied to the average power level value for
+// each FFT frequency bin. The higher the value, the higher the weight that is
+// given to older samples and the more slowly the average values change over
+// time in each bin.
+//
+// [15:0] : Alpha
+//
+localparam REG_ALPHA = 'h20;
+//
+localparam REG_ALPHA_LEN = 16;
+
+// REG_EPSILON (R/W)
+//
+// Controls the rate at which the maximum value for each FFT frequency bin
+// decays. The higher the value, the faster the decay rate. A value of 0
+// retains the maximum values indefinitely.
+//
+// [15:0] : Epsilon
+//
+localparam REG_EPSILON = 'h24;
+//
+localparam REG_EPSILON_LEN = 16;
+
+// REG_WF_CTRL (R/W)
+//
+// Waterfall Control register
+//
+// [7] : Waterfall mode. Controls the source of the waterfall history data.
+// When set to "Max Hold", the waterfall data is comprised of the max
+// power values from each frequency bin. When set to "Average", the
+// waterfall data is comprised of the accumulated average value from
+// each frequency bin between waterfall output packets. Can take on the
+// following values:
+//
+// 0 = Max Hold
+// 1 = Average
+//
+// [1:0] : Waterfall pre-division. Controls the scaling factor applied to
+// waterfall values. Can take on the following values:
+//
+// 0 = 1:1
+// 1 = 1:8
+// 2 = 1:64
+// 3 = 1:256
+//
+localparam REG_WF_CTRL = 'h28;
+//
+localparam REG_WF_CTRL_LEN = 8;
+//
+localparam REG_WF_MODE_POS = 7;
+//
+localparam REG_WF_DIV_POS = 0;
+localparam REG_WF_DIV_LEN = 2;
+
+// REG_WF_DECIM (R/W)
+//
+// [7:0] : Waterfall decimation. This controls the amount of waterfall data
+// that is output relative to the amount of input FFT data. The actual
+// decimation is N:1 where N=VALUE+2. That is, you'll get 1 waterfall
+// output packet for ever N FFT packets received.
+//
+localparam REG_WF_DECIM = 'h2C;
+//
+localparam REG_WF_DECIM_LEN = 8;
diff --git a/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor_tb.sv b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor_tb.sv
new file mode 100644
index 000000000..3f46e4383
--- /dev/null
+++ b/fpga/usrp3/lib/rfnoc/blocks/rfnoc_block_fosphor/rfnoc_block_fosphor_tb.sv
@@ -0,0 +1,590 @@
+//
+// Copyright 2020 Ettus Research, A National Instruments Company
+//
+// SPDX-License-Identifier: LGPL-3.0-or-later
+//
+// Module: rfnoc_block_fosphor_tb
+//
+// Description: Testbench for the fosphor RFNoC block.
+//
+
+`default_nettype none
+
+
+module rfnoc_block_fosphor_tb;
+
+ `include "test_exec.svh"
+
+ import PkgTestExec::*;
+ import PkgChdrUtils::*;
+ import PkgRfnocBlockCtrlBfm::*;
+ import PkgRfnocItemUtils::*;
+
+ `include "rfnoc_block_fosphor_regs.vh"
+
+ //---------------------------------------------------------------------------
+ // Testbench Configuration
+ //---------------------------------------------------------------------------
+
+ localparam [ 9:0] THIS_PORTID = 10'h123;
+ localparam [31:0] NOC_ID = 32'h666F0000;
+ localparam int CHDR_W = 64;
+ localparam int ITEM_W = 32;
+ localparam int NUM_PORTS_I = 1;
+ localparam int NUM_PORTS_O = 2;
+ localparam int MTU = 13;
+ localparam int SPP = 128;
+ localparam int PKT_SIZE_BYTES = SPP * (ITEM_W/8);
+ localparam int STALL_PROB = 60; // Default BFM stall probability
+ localparam real CHDR_CLK_PER = 5.0; // 200 MHz
+ localparam real CTRL_CLK_PER = 25.0; // 40 MHz
+ localparam real CE_CLK_PER = 5.0; // 200 MHz
+
+ localparam int HIST_PKT_PER_BURST = 66; // Always 64 hist + 1 max + 1 avg
+ localparam int HIST_PORT = 0;
+ localparam int WF_PORT = 1;
+
+ //---------------------------------------------------------------------------
+ // Clocks and Resets
+ //---------------------------------------------------------------------------
+
+ bit rfnoc_chdr_clk;
+ bit rfnoc_ctrl_clk;
+ bit ce_clk;
+
+ sim_clock_gen #(CHDR_CLK_PER) rfnoc_chdr_clk_gen (.clk(rfnoc_chdr_clk), .rst());
+ sim_clock_gen #(CTRL_CLK_PER) rfnoc_ctrl_clk_gen (.clk(rfnoc_ctrl_clk), .rst());
+ sim_clock_gen #(CE_CLK_PER) 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_fosphor #(
+ .THIS_PORTID (THIS_PORTID),
+ .CHDR_W (CHDR_W),
+ .MTU (MTU)
+ ) 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 Functions
+ //---------------------------------------------------------------------------
+
+ typedef enum bit { WF_MAX_HOLD, WF_AVERAGE } wf_mode_t;
+ typedef enum bit [1:0] { WF_1_1, WF_1_8, WF_1_64, WF_1_256 } wf_div_t;
+
+ // Data structure to hold the Fosphor configuration state
+ typedef struct packed {
+ bit en_wf;
+ bit en_hist;
+ bit en_noise;
+ bit en_dither;
+ bit [11:0] hist_decim;
+ bit [15:0] offset;
+ bit [15:0] scale;
+ bit [15:0] trise;
+ bit [15:0] tdecay;
+ bit [15:0] alpha;
+ bit [15:0] epsilon;
+ wf_mode_t wf_mode;
+ wf_div_t wf_div;
+ bit [ 7:0] wf_decim;
+ } fosphor_config_t;
+
+ // Default configuration copied from GNURadio
+ const fosphor_config_t DEFAULT_CONFG = '{
+ en_wf : 1,
+ en_hist : 1,
+ en_noise : 0,
+ en_dither : 0,
+ hist_decim : 2,
+ offset : 0,
+ scale : 256,
+ trise : 4096,
+ tdecay : 16384,
+ alpha : 65280,
+ epsilon : 2,
+ wf_mode : WF_MAX_HOLD,
+ wf_div : WF_1_8,
+ wf_decim : 2
+ };
+
+
+ // Rand#(WIDTH)::rand_logic() returns a WIDTH-bit random number. We avoid
+ // std::randomize() due to license requirements and limited tool support.
+ class Rand #(WIDTH = 32);
+
+ static function logic [WIDTH-1:0] rand_bits();
+ bit [WIDTH-1:0] result;
+ int num_rand32 = (WIDTH + 31) / 32;
+ for (int i = 0; i < num_rand32; i++) begin
+ result = {result, $urandom()};
+ end
+ return result;
+ endfunction : rand_bits
+
+ endclass : Rand
+
+
+ // Set all Fosphor registers based off the cfg data structure
+ task automatic set_registers(fosphor_config_t cfg);
+ blk_ctrl.reg_write(REG_ENABLE, (int'(cfg.en_wf) << 1) |
+ (int'(cfg.en_hist) << 0));
+ blk_ctrl.reg_write(REG_RANDOM, (int'(cfg.en_noise) << 1) |
+ (int'(cfg.en_dither) << 0));
+ blk_ctrl.reg_write(REG_HIST_DECIM, cfg.hist_decim);
+ blk_ctrl.reg_write(REG_OFFSET, cfg.offset);
+ blk_ctrl.reg_write(REG_SCALE, cfg.scale);
+ blk_ctrl.reg_write(REG_TRISE, cfg.trise);
+ blk_ctrl.reg_write(REG_TDECAY, cfg.tdecay);
+ blk_ctrl.reg_write(REG_ALPHA, cfg.alpha);
+ blk_ctrl.reg_write(REG_EPSILON, cfg.epsilon);
+ blk_ctrl.reg_write(REG_WF_CTRL, (int'(cfg.wf_mode) << 7) | int'(cfg.wf_div));
+ blk_ctrl.reg_write(REG_WF_DECIM, cfg.wf_decim);
+ endtask : set_registers;
+
+
+ // Verify that all the Fosphor registers match the cfg data structure
+ task automatic verify_registers(fosphor_config_t cfg);
+ bit [31:0] value;
+
+ blk_ctrl.reg_read(REG_ENABLE, value);
+ `ASSERT_ERROR(value[1] == cfg.en_wf, "REG_ENABLE[1] didn't have expected value");
+ `ASSERT_ERROR(value[0] == cfg.en_hist, "REG_ENABLE[0] didn't have expected value");
+
+ blk_ctrl.reg_read(REG_CLEAR, value);
+ `ASSERT_ERROR(value == 0, "REG_CLEAR didn't have expected value");
+
+ blk_ctrl.reg_read(REG_RANDOM, value);
+ `ASSERT_ERROR(value[1] == cfg.en_noise, "REG_RANDOM[1] didn't have expected value");
+ `ASSERT_ERROR(value[0] == cfg.en_dither, "REG_RANDOM[0] didn't have expected value");
+
+ blk_ctrl.reg_read(REG_HIST_DECIM, value);
+ `ASSERT_ERROR(value == cfg.hist_decim, "REG_HIST_DECIM didn't have expected value");
+
+ blk_ctrl.reg_read(REG_OFFSET, value);
+ `ASSERT_ERROR(value == cfg.offset, "REG_OFFSET didn't have expected value");
+
+ blk_ctrl.reg_read(REG_SCALE, value);
+ `ASSERT_ERROR(value == cfg.scale, "REG_SCALE didn't have expected value");
+
+ blk_ctrl.reg_read(REG_TRISE, value);
+ `ASSERT_ERROR(value == cfg.trise, "REG_TRISE didn't have expected value");
+
+ blk_ctrl.reg_read(REG_TDECAY, value);
+ `ASSERT_ERROR(value == cfg.tdecay, "REG_TDECAY didn't have expected value");
+
+ blk_ctrl.reg_read(REG_ALPHA, value);
+ `ASSERT_ERROR(value == cfg.alpha, "REG_ALPHA didn't have expected value");
+
+ blk_ctrl.reg_read(REG_EPSILON, value);
+ `ASSERT_ERROR(value == cfg.epsilon, "REG_EPSILON didn't have expected value");
+
+ blk_ctrl.reg_read(REG_WF_CTRL, value);
+ `ASSERT_ERROR(value[7] == cfg.wf_mode, "REG_WF_CTRL didn't have expected value");
+ `ASSERT_ERROR(value[1:0] == cfg.wf_div, "REG_WF_CTRL didn't have expected value");
+
+ blk_ctrl.reg_read(REG_WF_DECIM, value);
+ `ASSERT_ERROR(value == cfg.wf_decim, "REG_WF_DECIM didn't have expected value");
+ endtask : verify_registers;
+
+
+ // Generate a random Fosphor configuration to test
+ task automatic randomize_cfg(output fosphor_config_t cfg, output int spp);
+ // Chase a random SPP size, but make it a power of 2 (like the FFT) up to
+ // the define SPP value.
+ spp = 2**$urandom_range(4, $clog2(SPP));
+
+ // Start by randomizing the entire fosphor configuration, but then
+ cfg = Rand #($bits(cfg))::rand_bits();
+
+ // Keep decimation relatively small to decrease simulation time
+ cfg.hist_decim = $urandom_range(0, 8);
+
+ // Make sure wf_mode and wf_div are valid values
+ cfg.wf_mode = wf_mode_t'($urandom_range(cfg.wf_mode.num()-1));
+ cfg.wf_div = wf_div_t'($urandom_range(cfg.wf_div.num()-1));
+ endtask : randomize_cfg
+
+
+ // Test the passed Fosphor configuration. This updates the registers, inputs
+ // num_packets of data (spp-samples each) and verifies the output.
+ task automatic test_config(fosphor_config_t cfg, int num_packets, int spp);
+ item_t fft_items[$];
+
+ $display("Testing . . .");
+ $display(" packets: %0d", num_packets);
+ $display(" spp: %0d", spp);
+ $display(" en_wf %0d", cfg.en_wf);
+ $display(" en_hist %0d", cfg.en_hist);
+ $display(" hist_decim: %0d", cfg.hist_decim);
+ $display(" wf_decim: %0d", cfg.wf_decim);
+
+ // Clear any existing data
+ blk_ctrl.reg_write(REG_CLEAR, 1);
+
+ // Configure all the core's registers
+ set_registers(cfg);
+
+ // Generate packets to send
+ fft_items = {};
+ for (int i = 0; i < spp; i++) begin
+ fft_items.push_back({
+ shortint'(i),
+ shortint'(0)
+ });
+ end
+
+ // Send the packets
+ for (int i = 0; i < num_packets; i++) begin
+ blk_ctrl.send_items(0, fft_items);
+ end
+
+ fork
+ begin : fork_waterfall
+ item_t recv_items[$];
+ int exp_num_packets;
+
+ if (cfg.en_wf) begin
+ // Calculate expected number of packets
+ exp_num_packets = num_packets / (cfg.wf_decim + 2);
+ end else begin
+ exp_num_packets = 0;
+ end
+
+ $display("Expecting %0d waterfall packets of length %0d bytes",
+ exp_num_packets, spp);
+
+ if (exp_num_packets > 0) begin
+ for (int i = 0; i < exp_num_packets; i++) begin
+ string err_string;
+ blk_ctrl.recv_items(WF_PORT, recv_items);
+
+ // We expect one byte output per sample input
+ $sformat(
+ err_string,
+ "Waterfall packet of %0d bytes didn't match expected length of %0d bytes",
+ recv_items.size()*4, spp
+ );
+ `ASSERT_ERROR(recv_items.size()*4 == spp, err_string);
+ end
+ $display("All waterfall packets received!");
+ end
+ end
+
+ begin : fork_histogram
+ item_t recv_items[$];
+ chdr_word_t mdata[$];
+ int exp_num_packets;
+ packet_info_t pkt_info;
+
+ if(cfg.en_hist) begin
+ // Calculate expected number of packets
+ exp_num_packets = num_packets / (cfg.hist_decim + 2);
+ // Round it down to a multiple of HIST_PKT_PER_BURST, since it always
+ // outputs HIST_PKT_PER_BURST packets at a time.
+ exp_num_packets = (exp_num_packets / HIST_PKT_PER_BURST) * HIST_PKT_PER_BURST;
+ end else begin
+ exp_num_packets = 0;
+ end
+
+ $display("Expecting %0d histogram packets of length %0d bytes",
+ exp_num_packets, spp);
+
+ if (exp_num_packets > 0) begin
+ for (int i = 0; i < exp_num_packets; i++) begin
+ string err_string;
+ blk_ctrl.recv_items_adv(HIST_PORT, recv_items, mdata, pkt_info);
+ //$display("Recvd hist pkt %0d", i);
+
+ // We expect one byte output per sample input
+ $sformat(
+ err_string,
+ "Histogram packet of %0d bytes didn't match expected length of %0d bytes",
+ recv_items.size()*4, spp
+ );
+ `ASSERT_ERROR(recv_items.size()*4 == spp, err_string);
+
+ // Check that the last packet of each burst has EOB set
+ if ((i+1) % HIST_PKT_PER_BURST == 0) begin
+ `ASSERT_ERROR(pkt_info.eob == 1, "EOB was not set on last packet of histogram");
+ end else begin
+ `ASSERT_ERROR(pkt_info.eob == 0, "EOB was set on middle packet histogram");
+ end
+ end
+ $display("All histogram packets received!");
+ end
+ end
+ join
+
+ // Wait until all input packets were accepted before moving on, since we
+ // don't want any output from these packets to be confused with the next
+ // test.
+ blk_ctrl.wait_complete(0);
+ #(CE_CLK_PER * SPP * 2);
+
+ // The current Fosphor core doesn't cleanly handle transitions between
+ // settings, so we reset the core before each test.
+ blk_ctrl.reg_write(REG_CLEAR, 2);
+
+ endtask : test_config
+
+
+ //---------------------------------------------------------------------------
+ // Test Sequences
+ //---------------------------------------------------------------------------
+
+ // Test that all the registers read/write as expected
+ task automatic test_registers();
+ fosphor_config_t cfg;
+
+ // All registers reset to 0
+ test.start_test("Test Registers (reset values)", 50us);
+ cfg = '0;
+ verify_registers(cfg);
+ test.end_test();
+
+ test.start_test("Test Registers (max values)", 50us);
+ cfg = '{
+ en_wf : 'h1,
+ en_hist : 'h1,
+ en_noise : 'h1,
+ en_dither : 'h1,
+ hist_decim : 'hFFF,
+ offset : 'hFFFF,
+ scale : 'hFFFF,
+ trise : 'hFFFF,
+ tdecay : 'hFFFF,
+ alpha : 'hFFFF,
+ epsilon : 'hFFFF,
+ wf_mode : wf_mode_t'('h1),
+ wf_div : wf_div_t'('h3),
+ wf_decim : 'hFF
+ };
+ set_registers(cfg);
+ verify_registers(cfg);
+ test.end_test();
+
+ test.start_test("Test Registers (default values)", 50us);
+ cfg = DEFAULT_CONFG;
+ set_registers(cfg);
+ verify_registers(cfg);
+ test.end_test();
+ endtask : test_registers;
+
+
+ // Test waterfall decimation settings
+ task automatic test_wf_decimation();
+ const int spp = 16;
+ const int num_wf = 4;
+ fosphor_config_t cfg;
+ int num_packets;
+
+ test.start_test("Test waterfall decimation", 1ms);
+ cfg = DEFAULT_CONFG;
+ cfg.en_hist = 0;
+ for (int wf_decim = 0; wf_decim < 5; wf_decim++) begin
+ cfg.wf_decim = wf_decim;
+ // Input enough packets to get num_wf packets out
+ num_packets = (wf_decim+2) * (num_wf+1) - 1;
+ test_config(cfg, num_packets, spp);
+ end
+ test.end_test();
+ endtask : test_wf_decimation
+
+
+ // Test histogram decimation settings
+ task automatic test_hist_decimation();
+ const int spp = 16;
+ const int num_hist = HIST_PKT_PER_BURST * 4;
+ fosphor_config_t cfg;
+ int num_packets;
+
+ test.start_test("Test histogram decimation", 1ms);
+ cfg = DEFAULT_CONFG;
+ cfg.en_wf = 0;
+ for (int hist_decim = 0; hist_decim < 5; hist_decim++) begin
+ cfg.hist_decim = hist_decim;
+ // Input enough packets to get num_hist packets out
+ num_packets = (hist_decim+2) * (num_hist+HIST_PKT_PER_BURST/2);
+ test_config(cfg, num_packets, spp);
+ end
+ test.end_test();
+ endtask : test_hist_decimation
+
+
+ // Choose num_iter random configurations and test each one
+ task automatic test_rand_config(int num_iter);
+ int num_packets, num_packets_wf, num_packets_hist;
+ int spp;
+ fosphor_config_t cfg;
+ const int num_wf = 2; // Get 2 waterfall packets
+ const int num_hist = HIST_PKT_PER_BURST * 2; // Get 2 histogram bursts
+
+ test.start_test("Test random configurations", num_iter * 10ms);
+ for(int i = 0; i < num_iter; i++) begin
+ string str;
+ $display("<<<<<<<< RANDOM ITERATION %0d >>>>>>>>", i);
+ //test.current_test = $sformatf("%0d", i);
+
+ // Choose a random configuration
+ randomize_cfg(cfg, spp);
+
+ // Only allow the output of waterfall or histogram at one time. Because
+ // they operate independently and their outputs overlap, we only check
+ // one at a time. This way we can end testing cleanly between output
+ // packets without cutting off either the waterfall or histogram output.
+ if (($urandom() & 1) == 0) begin
+ cfg.en_wf = 1;
+ cfg.en_hist = 0;
+ num_packets = (cfg.wf_decim+2) * (num_wf+1) - 1;
+ end else begin
+ cfg.en_wf = 0;
+ cfg.en_hist = 1;
+ num_packets = (cfg.hist_decim+2) * (num_hist+HIST_PKT_PER_BURST/2);
+ end
+ test_config(cfg, num_packets, spp);
+ end
+ test.end_test();
+ endtask : test_rand_config
+
+
+ //---------------------------------------------------------------------------
+ // Main Test Process
+ //---------------------------------------------------------------------------
+
+ initial begin : tb_main
+
+ // Initialize the test exec object for this testbench
+ test.start_tb("rfnoc_block_fosphor_tb");
+
+ // 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
+ //--------------------------------
+
+ test_registers();
+ test_wf_decimation();
+ test_hist_decimation();
+ test_rand_config(16);
+
+ //--------------------------------
+ // Finish Up
+ //--------------------------------
+
+ // Display final statistics and results
+ test.end_tb();
+ end : tb_main
+
+endmodule : rfnoc_block_fosphor_tb
+
+
+`default_nettype wire
diff --git a/fpga/usrp3/lib/rfnoc/fosphor/fifo_srl.v b/fpga/usrp3/lib/rfnoc/fosphor/fifo_srl.v
index 700da18d3..822a1b769 100644
--- a/fpga/usrp3/lib/rfnoc/fosphor/fifo_srl.v
+++ b/fpga/usrp3/lib/rfnoc/fosphor/fifo_srl.v
@@ -28,7 +28,7 @@ module fifo_srl #(
output wire full,
output wire afull,
- output reg [WIDTH-1:0] do,
+ output reg [WIDTH-1:0] do = {WIDTH{1'b0}},
input wire rden,
output reg empty,
diff --git a/host/include/uhd/rfnoc/CMakeLists.txt b/host/include/uhd/rfnoc/CMakeLists.txt
index 6435ec2a6..098c79562 100644
--- a/host/include/uhd/rfnoc/CMakeLists.txt
+++ b/host/include/uhd/rfnoc/CMakeLists.txt
@@ -36,6 +36,7 @@ UHD_INSTALL(FILES
dmafifo_block_control.hpp
fft_block_control.hpp
fir_filter_block_control.hpp
+ fosphor_block_control.hpp
null_block_control.hpp
radio_control.hpp
diff --git a/host/include/uhd/rfnoc/blocks/fosphor.yml b/host/include/uhd/rfnoc/blocks/fosphor.yml
new file mode 100644
index 000000000..d803dc602
--- /dev/null
+++ b/host/include/uhd/rfnoc/blocks/fosphor.yml
@@ -0,0 +1,56 @@
+schema: rfnoc_modtool_args
+module_name: fosphor
+version: 1.0
+rfnoc_version: 1.0
+chdr_width: 64
+noc_id: 0x666F0000
+
+clocks:
+ - name: rfnoc_chdr
+ freq: "[]"
+ - name: rfnoc_ctrl
+ freq: "[]"
+ - name: ce
+ freq: "[]"
+
+control:
+ sw_iface: nocscript
+ fpga_iface: ctrlport
+ interface_direction: slave
+ fifo_depth: 32
+ clk_domain: ce
+ ctrlport:
+ byte_mode: False
+ timed: False
+ has_status: False
+
+data:
+ fpga_iface: axis_data
+ clk_domain: ce
+ inputs:
+ fft_in:
+ item_width: 32
+ nipc: 1
+ info_fifo_depth: 32
+ payload_fifo_depth: 32
+ format: sc16
+ mdata_sig: ~
+ outputs:
+ hist:
+ item_width: 8
+ nipc: 4
+ info_fifo_depth: 32
+ payload_fifo_depth: 32
+ format: u8
+ mdata_sig: ~
+ wf:
+ item_width: 8
+ nipc: 4
+ info_fifo_depth: 32
+ payload_fifo_depth: 32
+ format: u8
+ mdata_sig: ~
+
+registers:
+
+properties:
diff --git a/host/include/uhd/rfnoc/defaults.hpp b/host/include/uhd/rfnoc/defaults.hpp
index bc97fd9f5..38ae57242 100644
--- a/host/include/uhd/rfnoc/defaults.hpp
+++ b/host/include/uhd/rfnoc/defaults.hpp
@@ -30,6 +30,7 @@ static const std::string NODE_ID_SEP("SEP");
using io_type_t = std::string;
static const io_type_t IO_TYPE_SC16 = "sc16";
+static const io_type_t IO_TYPE_U8 = "u8";
static const std::string ACTION_KEY_STREAM_CMD("stream_cmd");
static const std::string ACTION_KEY_RX_EVENT("rx_event");
@@ -75,5 +76,6 @@ static const noc_id_t RADIO_BLOCK = 0x12AD1000;
static const noc_id_t DUC_BLOCK = 0xD0C00000;
static const noc_id_t DDC_BLOCK = 0xDDC00000;
static const noc_id_t FIR_FILTER_BLOCK = 0xf1120000;
+static const noc_id_t FOSPHOR_BLOCK = 0x666F0000;
}} // namespace uhd::rfnoc
diff --git a/host/include/uhd/rfnoc/fosphor_block_control.hpp b/host/include/uhd/rfnoc/fosphor_block_control.hpp
new file mode 100644
index 000000000..965b733a7
--- /dev/null
+++ b/host/include/uhd/rfnoc/fosphor_block_control.hpp
@@ -0,0 +1,383 @@
+//
+// Copyright 2020 Ettus Research, a National Instruments Brand
+//
+// SPDX-License-Identifier: GPL-3.0-or-later
+//
+
+#pragma once
+
+#include <uhd/config.hpp>
+#include <uhd/rfnoc/noc_block_base.hpp>
+
+namespace uhd { namespace rfnoc {
+
+enum class fosphor_waterfall_mode { MAX_HOLD, AVERAGE };
+enum class fosphor_waterfall_predivision_ratio {
+ RATIO_1_1,
+ RATIO_1_8,
+ RATIO_1_64,
+ RATIO_1_256
+};
+
+/*!
+ * Fosphor Control Class
+ *
+ * The Fosphor Block is an RFNoC block that accepts FFT data as signed
+ * complex 16-bit data and produces two streams of eight-bit data, a
+ * stream of histogram data and a stream of waterfall data.
+ *
+ * \section Histogram
+ *
+ * Each time the Fosphor block receives an FFT input packet, the power values
+ * in each of the N frequency bins are quantized into one of 64 power bins
+ * (X axis represents individual FFT frequency bins; Y axis represents the
+ * power bins):
+ *
+ * 63 .
+ * : . .
+ * : . . . . . .
+ * 0 . . . . . . . . . . . . . . . . . . . .
+ * 0 1 2 3 4 5 - - - - - - - - - - - - - - - - - - - - - - N-1
+ *
+ * Each time an FFT power value is quantized to a bin, the bin count
+ * is increased by one (illustrated by a '+'):
+ *
+ * 63 +
+ * : + +
+ * : + + + + + +
+ * 0 + + + + + + + + + + + + + + + + + + + +
+ * 0 1 2 3 4 5 - - - - - - - - - - - - - - - - - - - - - - N-1
+ *
+ * As more FFT packets are received, the counts in each bin accumulate.
+ * Over time, the count in the 'closest' power bin to each sample in the FFT
+ * accumulates at the highest rate. However, adjacent power bins' counts may
+ * increase due to quantization noise and variances in the input FFT signal
+ * (highest counts illustrated by '*', followed by '+' and '.'):
+ *
+ * 63 . * +
+ * : . . . + * . * +
+ * : + + . . + + . + * * + + + * + * + + * * . + . + + .
+ * 0 * * * * * * * * + + * * * + + * * + + * * * * * *
+ * 0 1 2 3 4 5 - - - - - - - - - - - - - - - - - - - - - - N-1
+ *
+ * The Fosphor block also calculates the average power level and maximum
+ * power level encountered in each FFT frequency bin. The rate at which
+ * the accumulated counts, average power level, and maximum power level
+ * values rise and fall over time is configurable.
+ *
+ * An instance of histogram data output consists of 66 packets:
+ *
+ * * 64 packets, one per quantized power level, of N values, representing the
+ * accumulated count for each frequency bin for that particular quantized
+ * power level;
+ * * One packet of N values, representing the average power level in each
+ * frequency bin; and
+ * * One packet of N values, representing the maximum power level in each
+ * frequency bin.
+ *
+ * \section Waterfall
+ *
+ * The waterfall stream consists of history data of either the average or
+ * maximum power level values in each bin, depending on the selected waterfall
+ * mode. In max hold mode, each waterfall packet consists of N values,
+ * representing the maximum power level in each frequency bin. The rate
+ * that packets are produced relative to the number of input FFT packets is
+ * configurable via the waterfall decimation parameter.
+ *
+ * In average mode, each waterfall packet consists of N values, representing
+ * the _sum_ of the average power level in each frequency bin accumulated
+ * between packets. (Thus, if the decimation rate is increased, the values
+ * returned are higher than if the decimation rate is decreased.) The
+ * waterfall predivision ratio parameter can be used to scale the values
+ * prior to accumulation to counteract this effect.
+ *
+ * These streams are intended to be inputs to the GNU Radio Fosphor
+ * display block, which renders the streams in a entertaining graphical
+ * format.
+ */
+class UHD_API fosphor_block_control : public noc_block_base
+{
+public:
+ RFNOC_DECLARE_BLOCK(fosphor_block_control)
+
+ // Block registers
+ static const uint32_t REG_ENABLE_ADDR;
+ static const uint32_t REG_CLEAR_ADDR;
+ static const uint32_t REG_RANDOM_ADDR;
+ static const uint32_t REG_DECIM_ADDR;
+ static const uint32_t REG_OFFSET_ADDR;
+ static const uint32_t REG_SCALE_ADDR;
+ static const uint32_t REG_TRISE_ADDR;
+ static const uint32_t REG_TDECAY_ADDR;
+ static const uint32_t REG_ALPHA_ADDR;
+ static const uint32_t REG_EPSILON_ADDR;
+ static const uint32_t REG_WF_CTRL_ADDR;
+ static const uint32_t REG_WF_DECIM_ADDR;
+
+ /*! Set the histogram stream enable flag
+ *
+ * Enables or disables the stream of histogram data from the block.
+ *
+ * \param enable_histogram Histogram stream enable/disable flag
+ */
+ virtual void set_enable_histogram(const bool enable_histogram) = 0;
+
+ /*! Get the histogram stream enable flag
+ *
+ * Returns the current histogram enable value.
+ *
+ * \returns Histogram stream enable/disable flag
+ */
+ virtual bool get_enable_histogram() const = 0;
+
+ /*! Set the waterfall stream enable flag
+ *
+ * Enables or disables the stream of waterfall data from the block.
+ *
+ * \param enable_waterfall Histogram stream enable/disable flag
+ */
+ virtual void set_enable_waterfall(const bool enable_waterfall) = 0;
+
+ /*! Get the waterfall stream enable flag
+ *
+ * Returns the current waterfall enable value.
+ *
+ * \returns Histogram stream enable/disable flag
+ */
+ virtual bool get_enable_waterfall() const = 0;
+
+ /*! Clear the Fosphor block's stored history
+ *
+ * Clears the accumulated history in the Fosphor block, resetting
+ * average and max hold values.
+ */
+ virtual void clear_history() = 0;
+
+ /*! Set the dither enable flag
+ *
+ * Enables or disables dithering. Dithering adds quantization error
+ * to the incoming signal.
+ *
+ * \param enable_dither Dither enable/disable flag
+ */
+ virtual void set_enable_dither(const bool enable_dither) = 0;
+
+ /*! Get the dither enable flag
+ *
+ * Returns the current dither enable value.
+ *
+ * \returns Dither enable/disable flag
+ */
+ virtual bool get_enable_dither() const = 0;
+
+ /*! Set the noise enable flag
+ *
+ * Enables or disables the addition of random noise to the incoming
+ * signal.
+ *
+ * \param enable_noise Noise enable/disable flag
+ */
+ virtual void set_enable_noise(const bool enable_noise) = 0;
+
+ /*! Get the noise enable flag
+ *
+ * Returns the current noise enable value.
+ *
+ * \returns Noise enable/disable flag
+ */
+ virtual bool get_enable_noise() const = 0;
+
+ /*! Set the histogram decimation factor
+ *
+ * Sets the ratio of histogram outputs to FFT packet inputs.
+ * For every \p decimation FFT input packets, one histogram
+ * output cluster (64 histogram packets, plus a maximum and
+ * average values packet) is produced. The minimum value for
+ * \p decimation is 2.
+ *
+ * \param decimation Histogram decimation factor
+ */
+ virtual void set_histogram_decimation(const uint16_t decimation) = 0;
+
+ /*! Get the histogram decimation factor
+ *
+ * Returns the current histogram decimation factor.
+ *
+ * \returns Histogram decimation factor
+ */
+ virtual uint16_t get_histogram_decimation() const = 0;
+
+ /*! Set the histogram offset factor
+ *
+ * Sets the offset factor to apply to FFT power levels before determining
+ * the appropriate histogram bin.
+ *
+ * \param offset The histogram offset factor to apply
+ */
+ virtual void set_histogram_offset(const uint16_t offset) = 0;
+
+ /*! Get the histogram offset factor
+ *
+ * Returns the current histogram offset factor.
+ *
+ * \returns The histogram offset factor
+ */
+ virtual uint16_t get_histogram_offset() const = 0;
+
+ /*! Set the histogram scale factor
+ *
+ * Sets the scale factor to apply to FFT power levels before determining
+ * the appropriate histogram bin. The scaling factor is \p scale / 256.
+ *
+ * \param scale The histogram scale factor to apply
+ */
+ virtual void set_histogram_scale(const uint16_t scale) = 0;
+
+ /*! Get the history scale factor
+ *
+ * Returns the current histogram scale factor.
+ *
+ * \returns The histogram scale factor
+ */
+ virtual uint16_t get_histogram_scale() const = 0;
+
+ /*! Set the histogram rise rate factor
+ *
+ * Sets the rate at which the hit count in each frequency and power bin
+ * increases when accumulating (i.e., there are hits in the particular
+ * bin). The higher the value, the more quickly the values increase,
+ * leading to a phosphorescent-like effect on the Fosphor display similar
+ * to the gradual illumination of a CRT display in the area where the
+ * electron beam is pointing.
+ *
+ * \param rise_rate The histogram rise rate factor to apply
+ */
+ virtual void set_histogram_rise_rate(const uint16_t rise_rate) = 0;
+
+ /*! Get the histogram rise rate factor
+ *
+ * Returns the current histogram rise rate factor.
+ *
+ * \returns The histogram rise rate factor
+ */
+ virtual uint16_t get_histogram_rise_rate() const = 0;
+
+ /*! Set the histogram decay rate factor
+ *
+ * Sets the rate at which the hit count in each frequency and power bin
+ * decreases when not accumulating (i.e., there are no hits in the
+ * particular bin). The lower the value, the more slowly the values
+ * decrease, leading to a phosphorescent-like effect on the Fosphor
+ * display similar to the gradual fading of a CRT display when the
+ * electron beam is extinguished.
+ *
+ * \param decay_rate The histogram decay rate factor to apply
+ */
+ virtual void set_histogram_decay_rate(const uint16_t decay_rate) = 0;
+
+ /*! Get the histogram decay rate factor
+ *
+ * Returns the current histogram decay rate factor.
+ *
+ * \returns The histogram decay rate factor
+ */
+ virtual uint16_t get_histogram_decay_rate() const = 0;
+
+ /*! Set the power level moving average weighting
+ *
+ * Sets the weighing to be applied to the average power level value
+ * for each FFT frequency bin. The higher the value, the higher the
+ * weight is given to older samples (and thus the more slowly the average
+ * values change over time in each bin).
+ *
+ * \param alpha The power level moving average weighting to apply
+ */
+ virtual void set_spectrum_alpha(const uint16_t alpha) = 0;
+
+ /*! Get the power level moving average weighting
+ *
+ * Returns the weighting that is applied to older samples when calculating
+ * the average power level for each FFT frequency bin.
+ *
+ * \returns The power level moving average weighting
+ */
+ virtual uint16_t get_spectrum_alpha() const = 0;
+
+ /*! Set the maximum hold decay rate
+ *
+ * Sets the rate at which the maximum value for each FFT frequency
+ * bin decays. The higher the value, the faster the decay rate.
+ * A value of 0 retains the maximum values indefinitely.
+ *
+ * \param epsilon The histogram scale factor to apply
+ */
+ virtual void set_spectrum_max_hold_decay(const uint16_t epsilon) = 0;
+
+ /*! Get the maximum hold decay rate
+ *
+ * Returns the rate at which the maximum value for each FFT frequency
+ * bin decays.
+ *
+ * \returns The maximum hold decay rate
+ */
+ virtual uint16_t get_spectrum_max_hold_decay() const = 0;
+
+ /*! Set the waterfall predivision ratio
+ *
+ * Sets the scaling factor applied to waterfall values.
+ *
+ * \param waterfall_predivision The waterfall predivision ratio to apply
+ */
+ virtual void set_waterfall_predivision(
+ const fosphor_waterfall_predivision_ratio waterfall_predivision) = 0;
+
+ /*! Get the waterfall predivision ratio
+ *
+ * Returns the current waterfall predivision ratio.
+ *
+ * \returns The waterfall predivision ratio
+ */
+ virtual fosphor_waterfall_predivision_ratio get_waterfall_predivision() const = 0;
+
+ /*! Set the waterfall mode setting
+ *
+ * Sets the source of the waterfall history data. When \p waterfall_mode
+ * is set to `MAX_HOLD`, the waterfall data is comprised of the max
+ * power values from each FFT frequency bin. When \p waterfall_mode is set
+ * to `AVERAGE`, the waterfall data is comprised of the accumulated
+ * average value from each FFT frequency bin between waterfall output
+ * packets.
+ *
+ * \param waterfall_mode The waterfall mode setting
+ */
+ virtual void set_waterfall_mode(const fosphor_waterfall_mode waterfall_mode) = 0;
+
+ /*! Get the waterfall mode setting
+ *
+ * Returns the current waterfall mode setting.
+ *
+ * \returns The waterfall mode setting
+ */
+ virtual fosphor_waterfall_mode get_waterfall_mode() const = 0;
+
+ /*! Set the waterfall decimation factor
+ *
+ * Sets the ratio of waterfall outputs to FFT packet inputs.
+ * For every \p waterfall_decimation FFT input packets, one waterfall
+ * output packet is produced. The minimum value for
+ * \p waterfall_decimation is 2.
+ *
+ * \param waterfall_decimation The waterfall decimation factor to apply
+ */
+ virtual void set_waterfall_decimation(const uint16_t waterfall_decimation) = 0;
+
+ /*! Get the histogram decimation factor
+ *
+ * Returns the current waterfall decimation factor.
+ *
+ * \returns The waterfall decimation factor
+ */
+ virtual uint16_t get_waterfall_decimation() const = 0;
+};
+
+}} // namespace uhd::rfnoc
diff --git a/host/lib/rfnoc/CMakeLists.txt b/host/lib/rfnoc/CMakeLists.txt
index 4edfe4009..6881a57e8 100644
--- a/host/lib/rfnoc/CMakeLists.txt
+++ b/host/lib/rfnoc/CMakeLists.txt
@@ -46,6 +46,7 @@ LIBUHD_APPEND_SOURCES(
${CMAKE_CURRENT_SOURCE_DIR}/dmafifo_block_control.cpp
${CMAKE_CURRENT_SOURCE_DIR}/fft_block_control.cpp
${CMAKE_CURRENT_SOURCE_DIR}/fir_filter_block_control.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/fosphor_block_control.cpp
${CMAKE_CURRENT_SOURCE_DIR}/null_block_control.cpp
${CMAKE_CURRENT_SOURCE_DIR}/radio_control_impl.cpp
)
diff --git a/host/lib/rfnoc/fosphor_block_control.cpp b/host/lib/rfnoc/fosphor_block_control.cpp
new file mode 100644
index 000000000..6bf621982
--- /dev/null
+++ b/host/lib/rfnoc/fosphor_block_control.cpp
@@ -0,0 +1,418 @@
+//
+// Copyright 2019 Ettus Research, a National Instruments Brand
+//
+// SPDX-License-Identifier: GPL-3.0-or-later
+//
+
+#include <uhd/exception.hpp>
+#include <uhd/rfnoc/defaults.hpp>
+#include <uhd/rfnoc/fosphor_block_control.hpp>
+#include <uhd/rfnoc/property.hpp>
+#include <uhd/rfnoc/registry.hpp>
+#include <string>
+
+using namespace uhd::rfnoc;
+
+
+// Register offsets
+const uint32_t fosphor_block_control::REG_ENABLE_ADDR = 0x00;
+const uint32_t fosphor_block_control::REG_CLEAR_ADDR = 0x04;
+const uint32_t fosphor_block_control::REG_RANDOM_ADDR = 0x08;
+const uint32_t fosphor_block_control::REG_DECIM_ADDR = 0x0c;
+const uint32_t fosphor_block_control::REG_OFFSET_ADDR = 0x10;
+const uint32_t fosphor_block_control::REG_SCALE_ADDR = 0x14;
+const uint32_t fosphor_block_control::REG_TRISE_ADDR = 0x18;
+const uint32_t fosphor_block_control::REG_TDECAY_ADDR = 0x1c;
+const uint32_t fosphor_block_control::REG_ALPHA_ADDR = 0x20;
+const uint32_t fosphor_block_control::REG_EPSILON_ADDR = 0x24;
+const uint32_t fosphor_block_control::REG_WF_CTRL_ADDR = 0x28;
+const uint32_t fosphor_block_control::REG_WF_DECIM_ADDR = 0x2c;
+
+// Mask bits
+constexpr uint32_t RESET_HISTORY_BIT = 0;
+constexpr uint32_t RESET_HISTORY_MASK = (1 << RESET_HISTORY_BIT);
+constexpr uint32_t RESET_CORE_BIT = 1;
+constexpr uint32_t RESET_CORE_MASK = (1 << RESET_CORE_BIT);
+
+constexpr uint32_t HISTOGRAM_ENABLE_BIT = 0;
+constexpr uint32_t HISTOGRAM_ENABLE_MASK = (1 << HISTOGRAM_ENABLE_BIT);
+constexpr uint32_t WATERFALL_ENABLE_BIT = 1;
+constexpr uint32_t WATERFALL_ENABLE_MASK = (1 << WATERFALL_ENABLE_BIT);
+
+constexpr uint32_t DITHER_ENABLE_BIT = 0;
+constexpr uint32_t DITHER_ENABLE_MASK = (1 << DITHER_ENABLE_BIT);
+constexpr uint32_t NOISE_ENABLE_BIT = 1;
+constexpr uint32_t NOISE_ENABLE_MASK = (1 << NOISE_ENABLE_BIT);
+
+constexpr uint32_t WATERFALL_MODE_BIT = 7;
+constexpr uint32_t WATERFALL_MODE_MASK = (1 << WATERFALL_MODE_BIT);
+constexpr uint32_t PREDIV_RATIO_MASK = (1 << 0) | (1 << 1);
+
+// User property names
+const char* const PROP_KEY_ENABLE_HISTOGRAM = "enable_histogram";
+const char* const PROP_KEY_ENABLE_WATERFALL = "enable_waterfall";
+const char* const PROP_KEY_CLEAR_HISTORY = "clear_history";
+const char* const PROP_KEY_ENABLE_DITHER = "enable_dither";
+const char* const PROP_KEY_ENABLE_NOISE = "enable_noise";
+const char* const PROP_KEY_HIST_DECIMATION = "hist_decimation";
+const char* const PROP_KEY_OFFSET = "offset";
+const char* const PROP_KEY_SCALE = "scale";
+const char* const PROP_KEY_RISE_TIME = "trise";
+const char* const PROP_KEY_DECAY_TIME = "tdecay";
+const char* const PROP_KEY_ALPHA = "alpha";
+const char* const PROP_KEY_EPSILON = "epsilon";
+const char* const PROP_KEY_WF_PREDIVISION_RATIO = "wf_predivision_ratio";
+const char* const PROP_KEY_WF_MODE = "wf_mode";
+const char* const PROP_KEY_WF_DECIMATION = "wf_decimation";
+
+// Edge property details
+constexpr uint32_t HISTOGRAM_PORT = 0;
+constexpr uint32_t WATERFALL_PORT = 1;
+
+
+class fosphor_block_control_impl : public fosphor_block_control
+{
+public:
+ RFNOC_BLOCK_CONSTRUCTOR(fosphor_block_control)
+ {
+ // reset the core upon block construction
+ this->regs().poke32(REG_CLEAR_ADDR, RESET_CORE_MASK);
+ _register_props();
+ }
+
+ void set_enable_histogram(const bool enable_histogram)
+ {
+ set_property<bool>(PROP_KEY_ENABLE_HISTOGRAM, enable_histogram);
+ }
+
+ bool get_enable_histogram() const
+ {
+ return _prop_enable_histogram.get();
+ }
+
+ void set_enable_waterfall(const bool enable_waterfall)
+ {
+ set_property<bool>(PROP_KEY_ENABLE_WATERFALL, enable_waterfall);
+ }
+
+ bool get_enable_waterfall() const
+ {
+ return _prop_enable_waterfall.get();
+ }
+
+ void clear_history()
+ {
+ set_property<bool>(PROP_KEY_CLEAR_HISTORY, true);
+ }
+
+ void set_enable_dither(const bool enable_dither)
+ {
+ set_property<bool>(PROP_KEY_ENABLE_DITHER, enable_dither);
+ }
+
+ bool get_enable_dither() const
+ {
+ return _prop_enable_dither.get();
+ }
+
+ void set_enable_noise(const bool enable_noise)
+ {
+ set_property<bool>(PROP_KEY_ENABLE_NOISE, enable_noise);
+ }
+
+ bool get_enable_noise() const
+ {
+ return _prop_enable_noise.get();
+ }
+
+ void set_histogram_decimation(const uint16_t decimation)
+ {
+ set_property<int>(PROP_KEY_HIST_DECIMATION, decimation);
+ }
+
+ uint16_t get_histogram_decimation() const
+ {
+ return _prop_hist_decimation.get();
+ }
+
+ void set_histogram_offset(const uint16_t offset)
+ {
+ set_property<int>(PROP_KEY_OFFSET, offset);
+ }
+
+ uint16_t get_histogram_offset() const
+ {
+ return _prop_offset.get();
+ }
+
+ void set_histogram_scale(const uint16_t scale)
+ {
+ set_property<int>(PROP_KEY_SCALE, scale);
+ }
+
+ uint16_t get_histogram_scale() const
+ {
+ return _prop_scale.get();
+ }
+
+ void set_histogram_rise_rate(const uint16_t rise_rate)
+ {
+ set_property<int>(PROP_KEY_RISE_TIME, rise_rate);
+ }
+
+ uint16_t get_histogram_rise_rate() const
+ {
+ return _prop_trise.get();
+ }
+
+ void set_histogram_decay_rate(const uint16_t decay_rate)
+ {
+ set_property<int>(PROP_KEY_DECAY_TIME, decay_rate);
+ }
+
+ uint16_t get_histogram_decay_rate() const
+ {
+ return _prop_tdecay.get();
+ }
+
+ void set_spectrum_alpha(const uint16_t alpha)
+ {
+ set_property<int>(PROP_KEY_ALPHA, alpha);
+ }
+
+ uint16_t get_spectrum_alpha() const
+ {
+ return _prop_alpha.get();
+ }
+
+ void set_spectrum_max_hold_decay(const uint16_t epsilon)
+ {
+ set_property<int>(PROP_KEY_EPSILON, epsilon);
+ }
+
+ uint16_t get_spectrum_max_hold_decay() const
+ {
+ return _prop_epsilon.get();
+ }
+
+ void set_waterfall_predivision(
+ const fosphor_waterfall_predivision_ratio waterfall_predivision)
+ {
+ set_property<int>(
+ PROP_KEY_WF_PREDIVISION_RATIO, static_cast<int>(waterfall_predivision));
+ }
+
+ fosphor_waterfall_predivision_ratio get_waterfall_predivision() const
+ {
+ return static_cast<fosphor_waterfall_predivision_ratio>(
+ _prop_wf_prediv_ratio.get());
+ }
+
+ void set_waterfall_mode(const fosphor_waterfall_mode waterfall_mode)
+ {
+ set_property<int>(PROP_KEY_WF_MODE, static_cast<int>(waterfall_mode));
+ }
+
+ fosphor_waterfall_mode get_waterfall_mode() const
+ {
+ return static_cast<fosphor_waterfall_mode>(_prop_wf_mode.get());
+ }
+
+ void set_waterfall_decimation(const uint16_t waterfall_decimation)
+ {
+ set_property<int>(PROP_KEY_WF_DECIMATION, waterfall_decimation);
+ }
+
+ uint16_t get_waterfall_decimation() const
+ {
+ return _prop_wf_decim.get();
+ }
+
+ /**************************************************************************
+ * Initialization
+ *************************************************************************/
+private:
+ void _register_props()
+ {
+ // register user properties
+ register_property(&_prop_enable_histogram, [this]() { _program_enables(); });
+ register_property(&_prop_enable_waterfall, [this]() { _program_enables(); });
+ register_property(&_prop_clear_history,
+ [this]() { this->regs().poke32(REG_CLEAR_ADDR, RESET_HISTORY_MASK); });
+ register_property(
+ &_prop_enable_dither, [this]() { _program_randomness_enables(); });
+ register_property(
+ &_prop_enable_noise, [this]() { _program_randomness_enables(); });
+ register_property(&_prop_hist_decimation, [this]() {
+ int decim = _prop_hist_decimation.get();
+ if (decim < 2 || decim > 1024) {
+ throw uhd::value_error(
+ "Histogram decimation value must be in [2, 1024]");
+ }
+ this->regs().poke32(REG_DECIM_ADDR, uint32_t(decim - 2));
+ });
+ register_property(&_prop_offset, [this]() {
+ int offset = _prop_offset.get();
+ if (offset < 0 || offset > 65535) {
+ throw uhd::value_error("Offset value must be in [0, 65535]");
+ }
+ this->regs().poke32(REG_OFFSET_ADDR, uint32_t(offset));
+ });
+ register_property(&_prop_scale, [this]() {
+ int scale = _prop_scale.get();
+ if (scale < 0 || scale > 65535) {
+ throw uhd::value_error("Scale value must be in [0, 65535]");
+ }
+ this->regs().poke32(REG_SCALE_ADDR, uint32_t(scale));
+ });
+ register_property(&_prop_trise, [this]() {
+ int trise = _prop_trise.get();
+ if (trise < 0 || trise > 65535) {
+ throw uhd::value_error("Rise rate value must be in [0, 65535]");
+ }
+ this->regs().poke32(REG_TRISE_ADDR, uint32_t(trise));
+ });
+ register_property(&_prop_tdecay, [this]() {
+ int tdecay = _prop_tdecay.get();
+ if (tdecay < 0 || tdecay > 65535) {
+ throw uhd::value_error("Decay rate value must be in [0, 65535]");
+ }
+ this->regs().poke32(REG_TDECAY_ADDR, uint32_t(tdecay));
+ });
+ register_property(&_prop_alpha, [this]() {
+ int alpha = _prop_alpha.get();
+ if (alpha < 0 || alpha > 65535) {
+ throw uhd::value_error("Alpha value must be in [0, 65535]");
+ }
+ this->regs().poke32(REG_ALPHA_ADDR, uint32_t(alpha));
+ });
+ register_property(&_prop_epsilon, [this]() {
+ int epsilon = _prop_epsilon.get();
+ if (epsilon < 0 || epsilon > 65535) {
+ throw uhd::value_error("Max hold decay rate must be in [0, 65535]");
+ }
+ this->regs().poke32(REG_EPSILON_ADDR, uint32_t(epsilon));
+ });
+ register_property(&_prop_wf_prediv_ratio, [this]() {
+ int prediv_ratio = _prop_wf_prediv_ratio.get();
+ if (prediv_ratio
+ < static_cast<int>(fosphor_waterfall_predivision_ratio::RATIO_1_1)
+ || prediv_ratio > static_cast<int>(
+ fosphor_waterfall_predivision_ratio::RATIO_1_256)) {
+ throw uhd::value_error(
+ "Waterfall predivision ratio value must be in [0, 3]");
+ }
+ _program_waterfall_mode();
+ });
+ register_property(&_prop_wf_mode, [this]() {
+ int wf_mode = _prop_wf_mode.get();
+ if (wf_mode < static_cast<int>(fosphor_waterfall_mode::MAX_HOLD)
+ || wf_mode > static_cast<int>(fosphor_waterfall_mode::AVERAGE)) {
+ throw uhd::value_error("Waterfall mode value must be 0 or 1");
+ }
+ _program_waterfall_mode();
+ });
+ register_property(&_prop_wf_decim, [this]() {
+ int wf_decim = _prop_wf_decim.get();
+ if (wf_decim < 2 || wf_decim > 257) {
+ throw uhd::value_error(
+ "Waterfall decimation value must be in [2, 257]");
+ }
+ this->regs().poke32(REG_WF_DECIM_ADDR, uint32_t(wf_decim - 2));
+ });
+
+ // register edge properties
+ register_property(&_prop_type_in);
+ register_property(&_prop_type_out_histogram);
+ register_property(&_prop_type_out_wf);
+
+ // add resolvers for type
+ add_property_resolver({&_prop_type_in}, {&_prop_type_in}, [this]() {
+ _prop_type_in.set(IO_TYPE_SC16);
+ });
+ add_property_resolver({&_prop_type_out_histogram},
+ {&_prop_type_out_histogram},
+ [this]() { _prop_type_out_histogram.set(IO_TYPE_U8); });
+ add_property_resolver({&_prop_type_out_wf}, {&_prop_type_out_wf}, [this]() {
+ _prop_type_out_wf.set(IO_TYPE_U8);
+ });
+ }
+
+ void _program_enables()
+ {
+ uint32_t reg_value = this->regs().peek32(REG_ENABLE_ADDR)
+ & ~(HISTOGRAM_ENABLE_MASK | WATERFALL_ENABLE_MASK);
+ uint32_t histogram_enable_bit =
+ (_prop_enable_histogram.get()) ? HISTOGRAM_ENABLE_MASK : 0;
+ uint32_t waterfall_enable_bit =
+ (_prop_enable_waterfall.get()) ? WATERFALL_ENABLE_MASK : 0;
+ this->regs().poke32(
+ REG_ENABLE_ADDR, reg_value | histogram_enable_bit | waterfall_enable_bit);
+ }
+
+ void _program_randomness_enables()
+ {
+ uint32_t reg_value = this->regs().peek32(REG_RANDOM_ADDR)
+ & ~(DITHER_ENABLE_MASK | NOISE_ENABLE_MASK);
+ uint32_t dither_enable_bit = (_prop_enable_dither.get()) ? DITHER_ENABLE_MASK : 0;
+ uint32_t noise_enable_bit = (_prop_enable_noise.get()) ? NOISE_ENABLE_MASK : 0;
+ this->regs().poke32(
+ REG_RANDOM_ADDR, reg_value | dither_enable_bit | noise_enable_bit);
+ }
+
+ void _program_waterfall_mode()
+ {
+ uint32_t reg_value = this->regs().peek32(REG_WF_CTRL_ADDR)
+ & ~(WATERFALL_MODE_MASK | PREDIV_RATIO_MASK);
+ int prediv_ratio = _prop_wf_prediv_ratio.get();
+ int wf_mode = _prop_wf_mode.get();
+ uint32_t wf_mode_bits = (wf_mode << WATERFALL_MODE_BIT) | prediv_ratio;
+ this->regs().poke32(REG_WF_CTRL_ADDR, reg_value | wf_mode_bits);
+ }
+
+ /**************************************************************************
+ * Attributes
+ *************************************************************************/
+ property_t<std::string> _prop_type_in = property_t<std::string>{
+ PROP_KEY_TYPE, IO_TYPE_SC16, {res_source_info::INPUT_EDGE}};
+ property_t<std::string> _prop_type_out_histogram = property_t<std::string>{
+ PROP_KEY_TYPE, IO_TYPE_U8, {res_source_info::OUTPUT_EDGE, HISTOGRAM_PORT}};
+ property_t<std::string> _prop_type_out_wf = property_t<std::string>{
+ PROP_KEY_TYPE, IO_TYPE_U8, {res_source_info::OUTPUT_EDGE, WATERFALL_PORT}};
+
+ property_t<bool> _prop_enable_histogram =
+ property_t<bool>{PROP_KEY_ENABLE_HISTOGRAM, true, {res_source_info::USER}};
+ property_t<bool> _prop_enable_waterfall =
+ property_t<bool>{PROP_KEY_ENABLE_WATERFALL, true, {res_source_info::USER}};
+ property_t<bool> _prop_clear_history =
+ property_t<bool>{PROP_KEY_CLEAR_HISTORY, false, {res_source_info::USER}};
+ property_t<bool> _prop_enable_dither =
+ property_t<bool>{PROP_KEY_ENABLE_DITHER, true, {res_source_info::USER}};
+ property_t<bool> _prop_enable_noise =
+ property_t<bool>{PROP_KEY_ENABLE_NOISE, true, {res_source_info::USER}};
+ property_t<int> _prop_hist_decimation =
+ property_t<int>{PROP_KEY_HIST_DECIMATION, 2, {res_source_info::USER}};
+ property_t<int> _prop_offset =
+ property_t<int>{PROP_KEY_OFFSET, 0, {res_source_info::USER}};
+ property_t<int> _prop_scale =
+ property_t<int>{PROP_KEY_SCALE, 256, {res_source_info::USER}};
+ property_t<int> _prop_trise =
+ property_t<int>{PROP_KEY_RISE_TIME, 4096, {res_source_info::USER}};
+ property_t<int> _prop_tdecay =
+ property_t<int>{PROP_KEY_DECAY_TIME, 16384, {res_source_info::USER}};
+ property_t<int> _prop_alpha =
+ property_t<int>{PROP_KEY_ALPHA, 65280, {res_source_info::USER}};
+ property_t<int> _prop_epsilon =
+ property_t<int>{PROP_KEY_EPSILON, 1, {res_source_info::USER}};
+ property_t<int> _prop_wf_prediv_ratio = property_t<int>{PROP_KEY_WF_PREDIVISION_RATIO,
+ static_cast<int>(fosphor_waterfall_predivision_ratio::RATIO_1_1),
+ {res_source_info::USER}};
+ property_t<int> _prop_wf_mode = property_t<int>{PROP_KEY_WF_MODE,
+ static_cast<int>(fosphor_waterfall_mode::MAX_HOLD),
+ {res_source_info::USER}};
+ property_t<int> _prop_wf_decim =
+ property_t<int>{PROP_KEY_WF_DECIMATION, 8, {res_source_info::USER}};
+};
+
+UHD_RFNOC_BLOCK_REGISTER_DIRECT(
+ fosphor_block_control, FOSPHOR_BLOCK, "Fosphor", CLOCK_KEY_GRAPH, "bus_clk")