diff options
author | Martin Braun <martin.braun@ettus.com> | 2020-01-23 16:10:22 -0800 |
---|---|---|
committer | Martin Braun <martin.braun@ettus.com> | 2020-01-28 09:35:36 -0800 |
commit | bafa9d95453387814ef25e6b6256ba8db2df612f (patch) | |
tree | 39ba24b5b67072d354775272e687796bb511848d /fpga/usrp3/lib/rfnoc/sim | |
parent | 3075b981503002df3115d5f1d0b97d2619ba30f2 (diff) | |
download | uhd-bafa9d95453387814ef25e6b6256ba8db2df612f.tar.gz uhd-bafa9d95453387814ef25e6b6256ba8db2df612f.tar.bz2 uhd-bafa9d95453387814ef25e6b6256ba8db2df612f.zip |
Merge FPGA repository back into UHD repository
The FPGA codebase was removed from the UHD repository in 2014 to reduce
the size of the repository. However, over the last half-decade, the
split between the repositories has proven more burdensome than it has
been helpful. By merging the FPGA code back, it will be possible to
create atomic commits that touch both FPGA and UHD codebases. Continuous
integration testing is also simplified by merging the repositories,
because it was previously difficult to automatically derive the correct
UHD branch when testing a feature branch on the FPGA repository.
This commit also updates the license files and paths therein.
We are therefore merging the repositories again. Future development for
FPGA code will happen in the same repository as the UHD host code and
MPM code.
== Original Codebase and Rebasing ==
The original FPGA repository will be hosted for the foreseeable future
at its original local location: https://github.com/EttusResearch/fpga/
It can be used for bisecting, reference, and a more detailed history.
The final commit from said repository to be merged here is
05003794e2da61cabf64dd278c45685a7abad7ec. This commit is tagged as
v4.0.0.0-pre-uhd-merge.
If you have changes in the FPGA repository that you want to rebase onto
the UHD repository, simply run the following commands:
- Create a directory to store patches (this should be an empty
directory):
mkdir ~/patches
- Now make sure that your FPGA codebase is based on the same state as
the code that was merged:
cd src/fpga # Or wherever your FPGA code is stored
git rebase v4.0.0.0-pre-uhd-merge
Note: The rebase command may look slightly different depending on what
exactly you're trying to rebase.
- Create a patch set for your changes versus v4.0.0.0-pre-uhd-merge:
git format-patch v4.0.0.0-pre-uhd-merge -o ~/patches
Note: Make sure that only patches are stored in your output directory.
It should otherwise be empty. Make sure that you picked the correct
range of commits, and only commits you wanted to rebase were exported
as patch files.
- Go to the UHD repository and apply the patches:
cd src/uhd # Or wherever your UHD repository is stored
git am --directory fpga ~/patches/*
rm -rf ~/patches # This is for cleanup
== Contributors ==
The following people have contributed mainly to these files (this list
is not complete):
Co-authored-by: Alex Williams <alex.williams@ni.com>
Co-authored-by: Andrej Rode <andrej.rode@ettus.com>
Co-authored-by: Ashish Chaudhari <ashish@ettus.com>
Co-authored-by: Ben Hilburn <ben.hilburn@ettus.com>
Co-authored-by: Ciro Nishiguchi <ciro.nishiguchi@ni.com>
Co-authored-by: Daniel Jepson <daniel.jepson@ni.com>
Co-authored-by: Derek Kozel <derek.kozel@ettus.com>
Co-authored-by: EJ Kreinar <ej@he360.com>
Co-authored-by: Humberto Jimenez <humberto.jimenez@ni.com>
Co-authored-by: Ian Buckley <ian.buckley@gmail.com>
Co-authored-by: Jörg Hofrichter <joerg.hofrichter@ni.com>
Co-authored-by: Jon Kiser <jon.kiser@ni.com>
Co-authored-by: Josh Blum <josh@joshknows.com>
Co-authored-by: Jonathon Pendlum <jonathan.pendlum@ettus.com>
Co-authored-by: Martin Braun <martin.braun@ettus.com>
Co-authored-by: Matt Ettus <matt@ettus.com>
Co-authored-by: Michael West <michael.west@ettus.com>
Co-authored-by: Moritz Fischer <moritz.fischer@ettus.com>
Co-authored-by: Nick Foster <nick@ettus.com>
Co-authored-by: Nicolas Cuervo <nicolas.cuervo@ettus.com>
Co-authored-by: Paul Butler <paul.butler@ni.com>
Co-authored-by: Paul David <paul.david@ettus.com>
Co-authored-by: Ryan Marlow <ryan.marlow@ettus.com>
Co-authored-by: Sugandha Gupta <sugandha.gupta@ettus.com>
Co-authored-by: Sylvain Munaut <tnt@246tNt.com>
Co-authored-by: Trung Tran <trung.tran@ettus.com>
Co-authored-by: Vidush Vishwanath <vidush.vishwanath@ettus.com>
Co-authored-by: Wade Fife <wade.fife@ettus.com>
Diffstat (limited to 'fpga/usrp3/lib/rfnoc/sim')
7 files changed, 2310 insertions, 0 deletions
diff --git a/fpga/usrp3/lib/rfnoc/sim/axis_pyld_ctxt_converter_tb/Makefile b/fpga/usrp3/lib/rfnoc/sim/axis_pyld_ctxt_converter_tb/Makefile new file mode 100644 index 000000000..71d2841f9 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/sim/axis_pyld_ctxt_converter_tb/Makefile @@ -0,0 +1,45 @@ +# +# Copyright 2019 Ettus Research, A National Instruments Company +# +# 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/core/Makefile.srcs +include $(BASE_DIR)/../lib/rfnoc/crossbar/Makefile.srcs + +DESIGN_SRCS += $(abspath \ +$(RFNOC_CORE_SRCS) \ +$(RFNOC_XBAR_SRCS) \ +) + +#------------------------------------------------- +# Testbench Specific +#------------------------------------------------- +SIM_TOP = axis_pyld_ctxt_converter_tb + +SIM_SRCS = \ +$(abspath axis_pyld_ctxt_converter_tb.sv) \ + +# MODELSIM_USER_DO = $(abspath wave.do) + +#------------------------------------------------- +# 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/sim/axis_pyld_ctxt_converter_tb/axis_pyld_ctxt_converter_tb.sv b/fpga/usrp3/lib/rfnoc/sim/axis_pyld_ctxt_converter_tb/axis_pyld_ctxt_converter_tb.sv new file mode 100644 index 000000000..c8b50c15a --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/sim/axis_pyld_ctxt_converter_tb/axis_pyld_ctxt_converter_tb.sv @@ -0,0 +1,465 @@ +// +// Copyright 2019 Ettus Research, A National Instruments Company +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: axis_pyld_ctxt_converter_tb +// + +`default_nettype none + + +module axis_pyld_ctxt_converter_tb; + + // ---------------------------------------- + // Global settings + // ---------------------------------------- + + // Include macros and time declarations for use with PkgTestExec + `include "test_exec.svh" + + import PkgTestExec::*; + import PkgAxiStreamBfm::*; + import PkgChdrUtils::*; + import PkgChdrBfm::*; + + // Parameters + localparam bit VERBOSE = 0; + localparam int CHDR_W = 64; + localparam int MTU = 7; + localparam int MTU_BITS = (1 << MTU) * CHDR_W; + localparam int NINST = 6; + localparam int START_INST = 0; + localparam int STOP_INST = NINST-1; + localparam int NUM_PKTS_PER_TEST = 100; + localparam int FAST_STALL_PROB = 0; + localparam int SLOW_STALL_PROB = 35; + localparam realtime CHDR_CLK_PERIOD = 3.0; + localparam int MAX_PYLD_W = 256; + + typedef struct { + realtime clk_period; + int item_w; + int nipc; + int ctxt_fifo; + int pyld_fifo; + bit prefetch; + } inst_params_t; + + // Module instances to test + localparam inst_params_t INST_PARAMS[0:NINST-1] = { + '{clk_period: 6.0, item_w:64, nipc: 1, ctxt_fifo:5, pyld_fifo:7, prefetch:1}, + '{clk_period:20.0, item_w:32, nipc: 6, ctxt_fifo:5, pyld_fifo:1, prefetch:1}, + '{clk_period: 3.0, item_w:32, nipc: 4, ctxt_fifo:1, pyld_fifo:2, prefetch:0}, + '{clk_period:10.0, item_w:16, nipc: 4, ctxt_fifo:8, pyld_fifo:5, prefetch:1}, + '{clk_period: 3.0, item_w:32, nipc: 2, ctxt_fifo:1, pyld_fifo:7, prefetch:0}, + '{clk_period: 3.0, item_w:8, nipc:13, ctxt_fifo:1, pyld_fifo:7, prefetch:0} + }; + + // ---------------------------------------- + // Interfaces and clocks + // ---------------------------------------- + + // Clocks and resets + bit rfnoc_chdr_clk, rfnoc_chdr_rst; + bit [NINST-1:0] rfnoc_data_clk, rfnoc_data_rst; + // Common CHDR Clock + sim_clock_gen #(CHDR_CLK_PERIOD) chdr_clk_gen_i (rfnoc_chdr_clk, rfnoc_chdr_rst); + + // Flush interface + logic [31:0] r2c_framer_errors[0:NINST-1]; + logic [31:0] r2c_flush_timeout[0:NINST-1], c2r_flush_timeout[0:NINST-1]; + logic [0:0] r2c_flush_en [0:NINST-1], c2r_flush_en [0:NINST-1]; + wire [0:0] r2c_flush_active [0:NINST-1], c2r_flush_active [0:NINST-1]; + wire [0:0] r2c_flush_done [0:NINST-1], c2r_flush_done [0:NINST-1]; + + // CHDR interface + wire [CHDR_W-1:0] chdr_tdata [0:NINST-1]; + wire chdr_tlast [0:NINST-1]; + wire chdr_tvalid[0:NINST-1]; + wire chdr_tready[0:NINST-1]; + + // AXIS interfaces and BFMs + AxiStreamIf #(CHDR_W, 4) r2c_ctxt [0:NINST-1] (); + AxiStreamIf #(CHDR_W, 4) c2r_ctxt [0:NINST-1] (); + AxiStreamIf #(MAX_PYLD_W) r2c_pyld [0:NINST-1] (); + AxiStreamIf #(MAX_PYLD_W) c2r_pyld [0:NINST-1] (); + AxiStreamBfm #(CHDR_W, 4) ctxt_bfm [0:NINST-1] ; + AxiStreamBfm #(MAX_PYLD_W) pyld_bfm [0:NINST-1] ; + + // Instantiate DUTs + genvar inst_i; + generate for (inst_i = 0; inst_i < NINST; inst_i++) begin: inst + + // Assign clocks and resets to ctxt and pyld streams + assign r2c_ctxt[inst_i].clk = rfnoc_data_clk[inst_i]; + assign r2c_ctxt[inst_i].rst = rfnoc_data_rst[inst_i]; + assign c2r_ctxt[inst_i].clk = rfnoc_data_clk[inst_i]; + assign c2r_ctxt[inst_i].rst = rfnoc_data_rst[inst_i]; + + assign r2c_pyld[inst_i].clk = rfnoc_data_clk[inst_i]; + assign r2c_pyld[inst_i].rst = rfnoc_data_rst[inst_i]; + assign c2r_pyld[inst_i].clk = rfnoc_data_clk[inst_i]; + assign c2r_pyld[inst_i].rst = rfnoc_data_rst[inst_i]; + + // Instantiate clock generator + sim_clock_gen #(INST_PARAMS[inst_i].clk_period) dclk_gen ( + rfnoc_data_clk[inst_i], rfnoc_data_rst[inst_i] + ); + + // Instantiate PyldCtxt to Chdr DUT + axis_pyld_ctxt_to_chdr #( + .CHDR_W (CHDR_W), + .ITEM_W (INST_PARAMS[inst_i].item_w), + .NIPC (INST_PARAMS[inst_i].nipc), + .SYNC_CLKS (INST_PARAMS[inst_i].clk_period == CHDR_CLK_PERIOD), + .CONTEXT_FIFO_SIZE (INST_PARAMS[inst_i].ctxt_fifo), + .PAYLOAD_FIFO_SIZE (INST_PARAMS[inst_i].pyld_fifo), + .MTU (MTU), + .CONTEXT_PREFETCH_EN (INST_PARAMS[inst_i].prefetch) + ) r2c_dut ( + .axis_chdr_clk (rfnoc_chdr_clk), + .axis_chdr_rst (rfnoc_chdr_rst), + .axis_data_clk (rfnoc_data_clk[inst_i]), + .axis_data_rst (rfnoc_data_rst[inst_i]), + .m_axis_chdr_tdata (chdr_tdata[inst_i]), + .m_axis_chdr_tlast (chdr_tlast[inst_i]), + .m_axis_chdr_tvalid (chdr_tvalid[inst_i]), + .m_axis_chdr_tready (chdr_tready[inst_i]), + .s_axis_payload_tdata (r2c_pyld[inst_i].slave.tdata[(INST_PARAMS[inst_i].item_w*INST_PARAMS[inst_i].nipc)-1:0]), + .s_axis_payload_tkeep (r2c_pyld[inst_i].slave.tkeep[INST_PARAMS[inst_i].nipc-1:0]), + .s_axis_payload_tlast (r2c_pyld[inst_i].slave.tlast), + .s_axis_payload_tvalid(r2c_pyld[inst_i].slave.tvalid), + .s_axis_payload_tready(r2c_pyld[inst_i].slave.tready), + .s_axis_context_tdata (r2c_ctxt[inst_i].slave.tdata), + .s_axis_context_tuser (r2c_ctxt[inst_i].slave.tuser), + .s_axis_context_tlast (r2c_ctxt[inst_i].slave.tlast), + .s_axis_context_tvalid(r2c_ctxt[inst_i].slave.tvalid), + .s_axis_context_tready(r2c_ctxt[inst_i].slave.tready), + .framer_errors (r2c_framer_errors[inst_i]), + .flush_en (r2c_flush_en[inst_i]), + .flush_timeout (r2c_flush_timeout[inst_i]), + .flush_active (r2c_flush_active[inst_i]), + .flush_done (r2c_flush_done[inst_i]) + ); + + // Instantiate Chdr to PyldCtxt DUT + chdr_to_axis_pyld_ctxt #( + .CHDR_W (CHDR_W), + .ITEM_W (INST_PARAMS[inst_i].item_w), + .NIPC (INST_PARAMS[inst_i].nipc), + .SYNC_CLKS (INST_PARAMS[inst_i].clk_period == CHDR_CLK_PERIOD), + .CONTEXT_FIFO_SIZE (INST_PARAMS[inst_i].ctxt_fifo), + .PAYLOAD_FIFO_SIZE (INST_PARAMS[inst_i].pyld_fifo), + .CONTEXT_PREFETCH_EN (INST_PARAMS[inst_i].prefetch) + ) c2r_dut ( + .axis_chdr_clk (rfnoc_chdr_clk), + .axis_chdr_rst (rfnoc_chdr_rst), + .axis_data_clk (rfnoc_data_clk[inst_i]), + .axis_data_rst (rfnoc_data_rst[inst_i]), + .s_axis_chdr_tdata (chdr_tdata[inst_i]), + .s_axis_chdr_tlast (chdr_tlast[inst_i]), + .s_axis_chdr_tvalid (chdr_tvalid[inst_i]), + .s_axis_chdr_tready (chdr_tready[inst_i]), + .m_axis_payload_tdata (c2r_pyld[inst_i].master.tdata[(INST_PARAMS[inst_i].item_w*INST_PARAMS[inst_i].nipc)-1:0]), + .m_axis_payload_tkeep (c2r_pyld[inst_i].master.tkeep[INST_PARAMS[inst_i].nipc-1:0]), + .m_axis_payload_tlast (c2r_pyld[inst_i].master.tlast), + .m_axis_payload_tvalid(c2r_pyld[inst_i].master.tvalid), + .m_axis_payload_tready(c2r_pyld[inst_i].master.tready), + .m_axis_context_tdata (c2r_ctxt[inst_i].master.tdata), + .m_axis_context_tuser (c2r_ctxt[inst_i].master.tuser), + .m_axis_context_tlast (c2r_ctxt[inst_i].master.tlast), + .m_axis_context_tvalid(c2r_ctxt[inst_i].master.tvalid), + .m_axis_context_tready(c2r_ctxt[inst_i].master.tready), + .flush_en (c2r_flush_en[inst_i]), + .flush_timeout (c2r_flush_timeout[inst_i]), + .flush_active (c2r_flush_active[inst_i]), + .flush_done (c2r_flush_done[inst_i]) + ); + + // Assert Reset and start BFMs + initial begin + dclk_gen.reset(); + r2c_flush_en[inst_i] = 0; + c2r_flush_en[inst_i] = 0; + pyld_bfm[inst_i] = new(r2c_pyld[inst_i], c2r_pyld[inst_i]); + pyld_bfm[inst_i].run(); + ctxt_bfm[inst_i] = new(r2c_ctxt[inst_i], c2r_ctxt[inst_i]); + ctxt_bfm[inst_i].run(); + end + end endgenerate + + function automatic bit pyld_pkts_equal( + ref AxiStreamPacket #(MAX_PYLD_W) exp, + ref AxiStreamPacket #(MAX_PYLD_W) act, + input int item_w, + input int nipc + ); + if (exp.data.size() != act.data.size()) return 0; + if (exp.keep.size() != act.keep.size()) return 0; + for (int i = 0; i < exp.data.size(); i++) begin + // Convert to bit + automatic bit [MAX_PYLD_W-1:0] mask = '0; + automatic bit [MAX_PYLD_W-1:0] data_exp = exp.data[i]; + automatic bit [MAX_PYLD_W-1:0] data_act = act.data[i]; + for (int r = 0; r < nipc; r++) begin + if (exp.keep[i][r] === 1'b1) begin + automatic bit [MAX_PYLD_W-1:0] samp_mask = ((1<<item_w)-1); + mask |= (samp_mask << (r*item_w)); + end + end + if (exp.keep[i] !== act.keep[i]) return 0; + if ((data_exp&mask) !== (data_act&mask)) return 0; + end + return 1; + endfunction + + task automatic send_recv_data_packets( + input int inst, + input inst_params_t params, //We pass this separately to work around Vivado bug + input int num_pkts, + input int mst_stall_prob, + input int slv_stall_prob, + input bit flushing = 0 + ); + int nipc = params.nipc; + int item_w = params.item_w; + bit prefetch = params.prefetch; + + AxiStreamPacket #(MAX_PYLD_W) pyld_pkt_arr[$] = {}; + AxiStreamPacket #(CHDR_W, 4) ctxt_pkt_arr[$] = {}; + + // Set stall probabilities + ctxt_bfm[inst].set_master_stall_prob(mst_stall_prob); + ctxt_bfm[inst].set_slave_stall_prob(slv_stall_prob); + pyld_bfm[inst].set_master_stall_prob(mst_stall_prob); + pyld_bfm[inst].set_slave_stall_prob(slv_stall_prob); + + // Generate a stream of data packets + for (int p = 0; p < num_pkts; p++) begin + int len_lines = $urandom_range((MTU_BITS/(item_w*nipc))-10, 1); + int keep_int = $urandom_range(nipc, 1); + pyld_pkt_arr[p] = new(); + for (int i = 0; i < len_lines; i++) begin + logic [MAX_PYLD_W-1:0] rand_samp; + logic [(MAX_PYLD_W/8)-1:0] keep_val = 'x; + for (int r = 0; r < (((nipc*item_w)+31)/32); r++) + rand_samp[r*32 +: 32] = $urandom(); + pyld_pkt_arr[p].data.push_back(rand_samp); + pyld_pkt_arr[p].user.push_back('x); + for (int r = 0; r < nipc; r++) begin + if (i == len_lines-1) + keep_val[r] = (r < keep_int) ? 1'b1 : 1'b0; + else + keep_val[r] = 1'b1; + end + pyld_pkt_arr[p].keep.push_back(keep_val); + end + end + + // Generate context packet for each data packet + foreach (pyld_pkt_arr[p]) begin + automatic chdr_header_t chdr_hdr; + automatic bit has_time = $urandom_range(1); + automatic int num_mdata = $urandom_range(5); + automatic int num_pyld_lines = pyld_pkt_arr[p].data.size(); + automatic int invalid_samps = 0; + automatic int length; + for (int r = 0; r < nipc; r++) + if (pyld_pkt_arr[p].keep[num_pyld_lines-1][r] === 1'b0) + invalid_samps++; + length = + (CHDR_W/8) + // header + ((has_time && (CHDR_W == 64)) ? (CHDR_W/8) : 0) + // timestamp + (num_mdata * (CHDR_W/8)) + // metadata + (num_pyld_lines * nipc * (item_w/8)) + // payload + (-invalid_samps * (item_w/8)); // payload (back out empty slots) + + chdr_hdr = '{ + vc : $urandom_range(63), + eob : $urandom_range(1), + eov : $urandom_range(1), + pkt_type : has_time ? CHDR_DATA_WITH_TS : CHDR_DATA_NO_TS, + num_mdata : num_mdata, + seq_num : p, + length : length, + dst_epid : $urandom() + }; + + ctxt_pkt_arr[p] = new(); + ctxt_pkt_arr[p].data.push_back(chdr_hdr); + ctxt_pkt_arr[p].user.push_back((has_time && (CHDR_W > 64)) ? CONTEXT_FIELD_HDR_TS : CONTEXT_FIELD_HDR); + ctxt_pkt_arr[p].keep.push_back('x); + if (has_time && (CHDR_W == 64)) begin + ctxt_pkt_arr[p].data.push_back(~p); + ctxt_pkt_arr[p].user.push_back(CONTEXT_FIELD_TS); + ctxt_pkt_arr[p].keep.push_back('x); + end + for (int i = 0; i < num_mdata; i++) begin + ctxt_pkt_arr[p].data.push_back(i); + ctxt_pkt_arr[p].user.push_back(CONTEXT_FIELD_MDATA); + ctxt_pkt_arr[p].keep.push_back('x); + end + end + + // Spin up 4 threads: {RX, TX} x {Context, Payload} + fork + begin: tx_context + timeout_t timeout; + for (int p = 0; p < num_pkts; p++) begin + test.start_timeout(timeout, 50us, "Waiting to send TX context pkt"); + ctxt_bfm[inst].put(ctxt_pkt_arr[p].copy()); + test.end_timeout(timeout); + if (VERBOSE) $display("[INST%0d:TxContext:%0d]\n%s", inst, p, ctxt_pkt_arr[p].sprint()); + end + end + begin: tx_payload + timeout_t timeout; + for (int p = 0; p < num_pkts; p++) begin + test.start_timeout(timeout, 50us, "Waiting to send TX payload pkt"); + pyld_bfm[inst].put(pyld_pkt_arr[p].copy()); + test.end_timeout(timeout); + if (VERBOSE) $display("[INST%0d:TxPayload:%0d]\n%s", inst, p, pyld_pkt_arr[p].sprint()); + end + end + begin: rx_context + if (!flushing) begin + timeout_t timeout; + automatic AxiStreamPacket #(CHDR_W, 4) rx_ctxt_pkt; + for (int p = 0; p < num_pkts; p++) begin + test.start_timeout(timeout, 50us, "Waiting to recv RX context pkt"); + ctxt_bfm[inst].get(rx_ctxt_pkt); + test.end_timeout(timeout); + if (VERBOSE) $display("[INST%0d:RxContext:%0d]\n%s", inst, p, rx_ctxt_pkt.sprint()); + if (VERBOSE) $display("[INST%0d:ExpContext:%0d]\n%s", inst, p, ctxt_pkt_arr[p].sprint()); + `ASSERT_ERROR(ctxt_pkt_arr[p].equal(rx_ctxt_pkt), "RX context packet did not match TX"); + end + end + end + begin: rx_payload + if (!flushing) begin + timeout_t timeout; + automatic AxiStreamPacket #(MAX_PYLD_W) rx_pyld_pkt; + for (int p = 0; p < num_pkts; p++) begin + test.start_timeout(timeout, 50us, "Waiting to recv RX payload pkt"); + pyld_bfm[inst].get(rx_pyld_pkt); + test.end_timeout(timeout); + if (VERBOSE) $display("[INST%0d:RxPayload:%0d]\n%s", inst, p, rx_pyld_pkt.sprint()); + if (VERBOSE) $display("[INST%0d:ExpPayload:%0d]\n%s", inst, p, pyld_pkt_arr[p].sprint()); + `ASSERT_ERROR(pyld_pkts_equal(pyld_pkt_arr[p], rx_pyld_pkt, item_w, nipc), "RX payload packet did not match TX"); + end + end + end + join + endtask + + + // ---------------------------------------- + // Test Process + // ---------------------------------------- + initial begin + + // Shared Variables + // ---------------------------------------- + timeout_t timeout; + string tc_label; + + // Initialize + // ---------------------------------------- + test.start_tb("axis_pyld_ctxt_converter_tb"); + + // Reset + // ---------------------------------------- + chdr_clk_gen_i.reset(); + + test.start_test("Wait for reset"); + test.start_timeout(timeout, 1us, "Waiting for reset"); + while (rfnoc_chdr_rst) @(posedge rfnoc_chdr_clk); + while (|rfnoc_data_rst) @(posedge rfnoc_chdr_clk); + repeat (100) @(posedge rfnoc_chdr_clk); + test.end_timeout(timeout); + `ASSERT_ERROR(!rfnoc_chdr_rst && !(|rfnoc_data_rst), "Reset did not deassert"); + test.end_test(); + + for (int inst_num = START_INST; inst_num <= STOP_INST; inst_num++) begin + $display("-----------------------------------------------------------------------------------------------"); + $display("Testing INST%0d:%p", inst_num, INST_PARAMS[inst_num]); + $display("-----------------------------------------------------------------------------------------------"); + + // Stream Random Data + // ---------------------------------------- + for (int cfg = 0; cfg < 4; cfg++) begin + automatic integer mst_cfg = cfg[0]; + automatic integer slv_cfg = cfg[1]; + $sformat(tc_label, "INST%0d: Stream Random Data (%s Mst, %s Slv)", + inst_num,(mst_cfg?"Slow":"Fast"), (slv_cfg?"Slow":"Fast")); + test.start_test(tc_label); + send_recv_data_packets(inst_num, INST_PARAMS[inst_num], NUM_PKTS_PER_TEST, + mst_cfg ? SLOW_STALL_PROB : FAST_STALL_PROB, + slv_cfg ? SLOW_STALL_PROB : FAST_STALL_PROB + ); + `ASSERT_ERROR(r2c_framer_errors[inst_num] === '0, "Encountered framer errors"); + test.end_test(); + end + + // Flush + // ---------------------------------------- + $sformat(tc_label, "INST%0d: Flush PyldCtxt => CHDR (Idle)", inst_num); + test.start_test(tc_label); + r2c_flush_timeout[inst_num] = $urandom_range(400, 200); + r2c_flush_en[inst_num] = 1'b1; + repeat (100) @(posedge rfnoc_chdr_clk); + `ASSERT_ERROR(r2c_flush_active[inst_num] === 1, "Flushing did not begin on time"); + `ASSERT_ERROR(r2c_flush_done[inst_num] === 0, "Flushing ended prematurely"); + repeat (r2c_flush_timeout[inst_num] + 1) @(posedge rfnoc_chdr_clk); + `ASSERT_ERROR(r2c_flush_done[inst_num] === 1, "Flushing did not end on time"); + r2c_flush_en[inst_num] = 1'b0; + @(posedge rfnoc_chdr_clk); + test.end_test(); + + $sformat(tc_label, "INST%0d: Flush CHDR => PyldCtxt (Idle)", inst_num); + test.start_test(tc_label); + c2r_flush_timeout[inst_num] = $urandom_range(400, 200); + c2r_flush_en[inst_num] = 1'b1; + repeat (100) @(posedge rfnoc_data_clk[inst_num]); + `ASSERT_ERROR(c2r_flush_active[inst_num] === 1, "Flushing did not begin on time"); + `ASSERT_ERROR(c2r_flush_done[inst_num] === 0, "Flushing ended prematurely"); + repeat (c2r_flush_timeout[inst_num] + 1) @(posedge rfnoc_data_clk[inst_num]); + `ASSERT_ERROR(c2r_flush_done[inst_num] === 1, "Flushing did not end on time"); + c2r_flush_en[inst_num] = 1'b0; + @(posedge rfnoc_data_clk[inst_num]); + test.end_test(); + + $sformat(tc_label, "INST%0d: Flush PyldCtxt => CHDR (Streaming)", inst_num); + test.start_test(tc_label); + r2c_flush_timeout[inst_num] = $urandom_range(400, 200); + r2c_flush_en[inst_num] = 1'b1; + repeat (100) @(posedge rfnoc_chdr_clk); + `ASSERT_ERROR(r2c_flush_active[inst_num] === 1, "Flushing did not begin on time"); + `ASSERT_ERROR(r2c_flush_done[inst_num] === 0, "Flushing ended prematurely"); + send_recv_data_packets(inst_num, INST_PARAMS[inst_num], NUM_PKTS_PER_TEST/10, + FAST_STALL_PROB, FAST_STALL_PROB, 1 /*flushing*/ + ); + repeat (NUM_PKTS_PER_TEST/10 * (1<<MTU) * 4) @(posedge rfnoc_chdr_clk); + repeat (r2c_flush_timeout[inst_num] + 1) @(posedge rfnoc_chdr_clk); + `ASSERT_ERROR(r2c_flush_done[inst_num] === 1, "Flushing did not end on time"); + r2c_flush_en[inst_num] = 1'b0; + @(posedge rfnoc_chdr_clk); + test.end_test(); + + $sformat(tc_label, "INST%0d: Stream Data After Flush", inst_num); + test.start_test(tc_label); + send_recv_data_packets(inst_num, INST_PARAMS[inst_num], NUM_PKTS_PER_TEST/10, + FAST_STALL_PROB, FAST_STALL_PROB + ); + `ASSERT_ERROR(r2c_framer_errors[inst_num] === '0, "Encountered framer errors"); + test.end_test(); + end + + // Finish Up + // ---------------------------------------- + // Display final statistics and results + test.end_tb(); + end + +endmodule diff --git a/fpga/usrp3/lib/rfnoc/sim/chdr_stream_endpoint_tb/Makefile b/fpga/usrp3/lib/rfnoc/sim/chdr_stream_endpoint_tb/Makefile new file mode 100644 index 000000000..b2773db02 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/sim/chdr_stream_endpoint_tb/Makefile @@ -0,0 +1,44 @@ +# +# Copyright 2019 Ettus Research, A National Instruments Company +# +# 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/core/Makefile.srcs +include $(BASE_DIR)/../lib/rfnoc/crossbar/Makefile.srcs + +DESIGN_SRCS += $(abspath \ +$(RFNOC_CORE_SRCS) \ +$(RFNOC_XBAR_SRCS) \ +) + +#------------------------------------------------- +# Testbench Specific +#------------------------------------------------- +SIM_TOP = chdr_stream_endpoint_tb + +SIM_SRCS = \ +$(abspath lossy_xport_model.v) \ +$(abspath chdr_stream_endpoint_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/sim/chdr_stream_endpoint_tb/chdr_stream_endpoint_tb.sv b/fpga/usrp3/lib/rfnoc/sim/chdr_stream_endpoint_tb/chdr_stream_endpoint_tb.sv new file mode 100644 index 000000000..0626ee447 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/sim/chdr_stream_endpoint_tb/chdr_stream_endpoint_tb.sv @@ -0,0 +1,1149 @@ +// +// Copyright 2019 Ettus Research, A National Instruments Company +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: chdr_stream_endpoint_tb +// + +`default_nettype none + + +module chdr_stream_endpoint_tb; + + // ---------------------------------------- + // Global settings + // ---------------------------------------- + + // Include macros and time declarations for use with PkgTestExec + `include "test_exec.svh" + + import PkgTestExec::*; + import PkgChdrUtils::*; + import PkgChdrBfm::*; + + // Clocks and resets + bit rfnoc_chdr_clk, rfnoc_chdr_rst; + bit rfnoc_ctrl_clk, rfnoc_ctrl_rst; + sim_clock_gen #(6.0) rfnoc_chdr_clk_gen (rfnoc_chdr_clk, rfnoc_chdr_rst); // 166.6 MHz + sim_clock_gen #(20.0) rfnoc_ctrl_clk_gen (rfnoc_ctrl_clk, rfnoc_ctrl_rst); // 50 MHz + + // Parameters + localparam bit VERBOSE = 0; + localparam int NUM_PKTS_PER_TEST = 200; + localparam int FAST_STALL_PROB = 0; + localparam int SLOW_STALL_PROB = 35; + + localparam int CHDR_W = 64; + localparam int MTU = 7; + localparam [15:0] PROTOVER = {8'd1, 8'd0}; + localparam [15:0] DEV_ID = 16'hBEEF; + localparam [15:0] EPID_TB = 16'h1001; + localparam [15:0] EPID_A = 16'h1002; + localparam [15:0] EPID_B = 16'h1003; + localparam [9:0] PORT_TB = 10'd0; + localparam [9:0] PORT_A = 10'd1; + localparam [9:0] PORT_B = 10'd2; + + // ---------------------------------------- + // DUT (and Crossbar) Instantiations + // ---------------------------------------- + wire [CHDR_W-1:0] c2ae_chdr_tdata , c2ax_chdr_tdata , a2c_chdr_tdata ; + wire c2ae_chdr_tlast , c2ax_chdr_tlast , a2c_chdr_tlast ; + wire c2ae_chdr_tvalid, c2ax_chdr_tvalid, a2c_chdr_tvalid; + wire c2ae_chdr_tready, c2ax_chdr_tready, a2c_chdr_tready; + wire [CHDR_W-1:0] c2be_chdr_tdata , c2bx_chdr_tdata , b2c_chdr_tdata ; + wire c2be_chdr_tlast , c2bx_chdr_tlast , b2c_chdr_tlast ; + wire c2be_chdr_tvalid, c2bx_chdr_tvalid, b2c_chdr_tvalid; + wire c2be_chdr_tready, c2bx_chdr_tready, b2c_chdr_tready; + + wire [31:0] a_ctrl_in_tdata, a_ctrl_out_tdata, b_ctrl_in_tdata, b_ctrl_out_tdata; + wire a_ctrl_loop_tlast , b_ctrl_loop_tlast ; + wire a_ctrl_loop_tvalid, b_ctrl_loop_tvalid; + wire a_ctrl_loop_tready, b_ctrl_loop_tready; + + logic a_signal_data_err, b_signal_data_err; + logic a_lossy_input, b_lossy_input; + logic [7:0] a_seqerr_prob, b_seqerr_prob; + logic [7:0] a_rterr_prob, b_rterr_prob; + + AxiStreamIf #(CHDR_W) m_tb_chdr (rfnoc_chdr_clk, rfnoc_chdr_rst); + AxiStreamIf #(CHDR_W) s_tb_chdr (rfnoc_chdr_clk, rfnoc_chdr_rst); + + AxiStreamIf #(CHDR_W) m_a0_data (rfnoc_chdr_clk, rfnoc_chdr_rst); + AxiStreamIf #(CHDR_W) s_a0_data (rfnoc_chdr_clk, rfnoc_chdr_rst); + AxiStreamIf #(CHDR_W) m_a1_data (rfnoc_chdr_clk, rfnoc_chdr_rst); + AxiStreamIf #(CHDR_W) s_a1_data (rfnoc_chdr_clk, rfnoc_chdr_rst); + AxiStreamIf #(CHDR_W) m_b0_data (rfnoc_chdr_clk, rfnoc_chdr_rst); + AxiStreamIf #(CHDR_W) s_b0_data (rfnoc_chdr_clk, rfnoc_chdr_rst); + AxiStreamIf #(CHDR_W) m_b1_data (rfnoc_chdr_clk, rfnoc_chdr_rst); + AxiStreamIf #(CHDR_W) s_b1_data (rfnoc_chdr_clk, rfnoc_chdr_rst); + + chdr_stream_endpoint #( + .PROTOVER (PROTOVER), + .CHDR_W (CHDR_W), + .AXIS_CTRL_EN (1), + .AXIS_DATA_EN (1), + .INST_NUM (0), + .NUM_DATA_I (2), + .NUM_DATA_O (2), + .CTRL_XBAR_PORT (PORT_A), + .INGRESS_BUFF_SIZE (MTU+1), + .MTU (MTU), + .REPORT_STRM_ERRS (1), + .SIM_SPEEDUP (1) + ) sep_a ( + .rfnoc_chdr_clk (rfnoc_chdr_clk ), + .rfnoc_chdr_rst (rfnoc_chdr_rst ), + .rfnoc_ctrl_clk (rfnoc_ctrl_clk ), + .rfnoc_ctrl_rst (rfnoc_ctrl_rst ), + .device_id (DEV_ID ), + .s_axis_chdr_tdata (c2ae_chdr_tdata ), + .s_axis_chdr_tlast (c2ae_chdr_tlast ), + .s_axis_chdr_tvalid (c2ae_chdr_tvalid ), + .s_axis_chdr_tready (c2ae_chdr_tready ), + .m_axis_chdr_tdata (a2c_chdr_tdata ), + .m_axis_chdr_tlast (a2c_chdr_tlast ), + .m_axis_chdr_tvalid (a2c_chdr_tvalid ), + .m_axis_chdr_tready (a2c_chdr_tready ), + .s_axis_data_tdata ({m_a1_data.slave.tdata , m_a0_data.slave.tdata }), + .s_axis_data_tlast ({m_a1_data.slave.tlast , m_a0_data.slave.tlast }), + .s_axis_data_tvalid ({m_a1_data.slave.tvalid , m_a0_data.slave.tvalid }), + .s_axis_data_tready ({m_a1_data.slave.tready , m_a0_data.slave.tready }), + .m_axis_data_tdata ({s_a1_data.master.tdata , s_a0_data.master.tdata }), + .m_axis_data_tlast ({s_a1_data.master.tlast , s_a0_data.master.tlast }), + .m_axis_data_tvalid ({s_a1_data.master.tvalid, s_a0_data.master.tvalid}), + .m_axis_data_tready ({s_a1_data.master.tready, s_a0_data.master.tready}), + .s_axis_ctrl_tdata (a_ctrl_out_tdata ), + .s_axis_ctrl_tlast (a_ctrl_loop_tlast ), + .s_axis_ctrl_tvalid (a_ctrl_loop_tvalid ), + .s_axis_ctrl_tready (a_ctrl_loop_tready ), + .m_axis_ctrl_tdata (a_ctrl_in_tdata ), + .m_axis_ctrl_tlast (a_ctrl_loop_tlast ), + .m_axis_ctrl_tvalid (a_ctrl_loop_tvalid ), + .m_axis_ctrl_tready (a_ctrl_loop_tready ), + .strm_seq_err_stb ( ), + .strm_data_err_stb ( ), + .strm_route_err_stb ( ), + .signal_data_err (a_signal_data_err ) + ); + + chdr_stream_endpoint #( + .PROTOVER (PROTOVER), + .CHDR_W (CHDR_W), + .AXIS_CTRL_EN (1), + .AXIS_DATA_EN (1), + .INST_NUM (1), + .NUM_DATA_I (2), + .NUM_DATA_O (2), + .CTRL_XBAR_PORT (PORT_B), + .INGRESS_BUFF_SIZE (MTU+1), + .MTU (MTU), + .REPORT_STRM_ERRS (1), + .SIM_SPEEDUP (1) + ) sep_b ( + .rfnoc_chdr_clk (rfnoc_chdr_clk ), + .rfnoc_chdr_rst (rfnoc_chdr_rst ), + .rfnoc_ctrl_clk (rfnoc_ctrl_clk ), + .rfnoc_ctrl_rst (rfnoc_ctrl_rst ), + .device_id (DEV_ID ), + .s_axis_chdr_tdata (c2be_chdr_tdata ), + .s_axis_chdr_tlast (c2be_chdr_tlast ), + .s_axis_chdr_tvalid (c2be_chdr_tvalid ), + .s_axis_chdr_tready (c2be_chdr_tready ), + .m_axis_chdr_tdata (b2c_chdr_tdata ), + .m_axis_chdr_tlast (b2c_chdr_tlast ), + .m_axis_chdr_tvalid (b2c_chdr_tvalid ), + .m_axis_chdr_tready (b2c_chdr_tready ), + .s_axis_data_tdata ({m_b1_data.slave.tdata , m_b0_data.slave.tdata }), + .s_axis_data_tlast ({m_b1_data.slave.tlast , m_b0_data.slave.tlast }), + .s_axis_data_tvalid ({m_b1_data.slave.tvalid , m_b0_data.slave.tvalid }), + .s_axis_data_tready ({m_b1_data.slave.tready , m_b0_data.slave.tready }), + .m_axis_data_tdata ({s_b1_data.master.tdata , s_b0_data.master.tdata }), + .m_axis_data_tlast ({s_b1_data.master.tlast , s_b0_data.master.tlast }), + .m_axis_data_tvalid ({s_b1_data.master.tvalid, s_b0_data.master.tvalid}), + .m_axis_data_tready ({s_b1_data.master.tready, s_b0_data.master.tready}), + .s_axis_ctrl_tdata (b_ctrl_out_tdata ), + .s_axis_ctrl_tlast (b_ctrl_loop_tlast ), + .s_axis_ctrl_tvalid (b_ctrl_loop_tvalid ), + .s_axis_ctrl_tready (b_ctrl_loop_tready ), + .m_axis_ctrl_tdata (b_ctrl_in_tdata ), + .m_axis_ctrl_tlast (b_ctrl_loop_tlast ), + .m_axis_ctrl_tvalid (b_ctrl_loop_tvalid ), + .m_axis_ctrl_tready (b_ctrl_loop_tready ), + .strm_seq_err_stb ( ), + .strm_data_err_stb ( ), + .strm_route_err_stb ( ), + .signal_data_err (b_signal_data_err ) + ); + + chdr_crossbar_nxn #( + .CHDR_W (CHDR_W), + .NPORTS (3), + .DEFAULT_PORT (0), + .MTU (MTU), + .ROUTE_TBL_SIZE (6), + .MUX_ALLOC ("ROUND-ROBIN"), + .OPTIMIZE ("AREA"), + .NPORTS_MGMT (1), + .EXT_RTCFG_PORT (0), + .PROTOVER (PROTOVER) + ) xbar_c ( + .clk (rfnoc_chdr_clk), + .reset (rfnoc_chdr_rst), + .device_id (DEV_ID), + .s_axis_tdata ({b2c_chdr_tdata, a2c_chdr_tdata, m_tb_chdr.slave.tdata }), + .s_axis_tlast ({b2c_chdr_tlast, a2c_chdr_tlast, m_tb_chdr.slave.tlast }), + .s_axis_tvalid ({b2c_chdr_tvalid, a2c_chdr_tvalid, m_tb_chdr.slave.tvalid }), + .s_axis_tready ({b2c_chdr_tready, a2c_chdr_tready, m_tb_chdr.slave.tready }), + .m_axis_tdata ({c2bx_chdr_tdata, c2ax_chdr_tdata, s_tb_chdr.master.tdata }), + .m_axis_tlast ({c2bx_chdr_tlast, c2ax_chdr_tlast, s_tb_chdr.master.tlast }), + .m_axis_tvalid ({c2bx_chdr_tvalid, c2ax_chdr_tvalid, s_tb_chdr.master.tvalid}), + .m_axis_tready ({c2bx_chdr_tready, c2ax_chdr_tready, s_tb_chdr.master.tready}), + .ext_rtcfg_stb ('0), + .ext_rtcfg_addr ('0), + .ext_rtcfg_data ('0), + .ext_rtcfg_ack () + ); + + lossy_xport_model #( .CHDR_W(CHDR_W) ) xport_a ( + .clk (rfnoc_chdr_clk ), + .rst (rfnoc_chdr_rst ), + .s_axis_tdata (c2ax_chdr_tdata ), + .s_axis_tlast (c2ax_chdr_tlast ), + .s_axis_tvalid (c2ax_chdr_tvalid), + .s_axis_tready (c2ax_chdr_tready), + .m_axis_tdata (c2ae_chdr_tdata ), + .m_axis_tlast (c2ae_chdr_tlast ), + .m_axis_tvalid (c2ae_chdr_tvalid), + .m_axis_tready (c2ae_chdr_tready), + .seqerr_prob (a_seqerr_prob ), + .rterr_prob (a_rterr_prob ), + .lossy (a_lossy_input ) + ); + + lossy_xport_model #( .CHDR_W(CHDR_W) ) xport_b ( + .clk (rfnoc_chdr_clk ), + .rst (rfnoc_chdr_rst ), + .s_axis_tdata (c2bx_chdr_tdata ), + .s_axis_tlast (c2bx_chdr_tlast ), + .s_axis_tvalid (c2bx_chdr_tvalid), + .s_axis_tready (c2bx_chdr_tready), + .m_axis_tdata (c2be_chdr_tdata ), + .m_axis_tlast (c2be_chdr_tlast ), + .m_axis_tvalid (c2be_chdr_tvalid), + .m_axis_tready (c2be_chdr_tready), + .seqerr_prob (b_seqerr_prob ), + .rterr_prob (b_rterr_prob ), + .lossy (b_lossy_input ) + ); + + // ---------------------------------------- + // BFMs and Test Models + // ---------------------------------------- + + ChdrBfm #(CHDR_W) a0_data_bfm = new(m_a0_data, s_a0_data); + ChdrBfm #(CHDR_W) b0_data_bfm = new(m_b0_data, s_b0_data); + ChdrBfm #(CHDR_W) a1_data_bfm = new(m_a1_data, s_a1_data); + ChdrBfm #(CHDR_W) b1_data_bfm = new(m_b1_data, s_b1_data); + ChdrBfm #(CHDR_W) tb_chdr_bfm = new(m_tb_chdr, s_tb_chdr); + + // Simple responders for AXIS-Ctrl transactions + reg a_first = 1'b1, b_first = 1'b1; + always @(posedge rfnoc_ctrl_clk) begin + if (rfnoc_ctrl_rst) begin + a_first <= 1'd1; + b_first <= 1'd1; + end else begin + if (a_ctrl_loop_tvalid & a_ctrl_loop_tready) + a_first <= a_ctrl_loop_tlast; + if (b_ctrl_loop_tvalid & b_ctrl_loop_tready) + b_first <= b_ctrl_loop_tlast; + end + end + // Respond with an ACK and the source and destination ports swapped + assign a_ctrl_out_tdata = + a_first ? {1'b1, a_ctrl_in_tdata[30:20], a_ctrl_in_tdata[9:0], a_ctrl_in_tdata[19:10]} : a_ctrl_in_tdata; + assign b_ctrl_out_tdata = + b_first ? {1'b1, b_ctrl_in_tdata[30:20], b_ctrl_in_tdata[9:0], b_ctrl_in_tdata[19:10]} : b_ctrl_in_tdata; + + // ---------------------------------------- + // Test Utilities + // ---------------------------------------- + integer cached_mgmt_seqnum = 0; + integer cached_ctrl_seqnum = 0; + integer cached_data_seqnum = 0; + + task automatic send_recv_mgmt_packet( + input chdr_header_t tx_mgmt_hdr, + input chdr_mgmt_t tx_mgmt_pl, + output chdr_header_t rx_mgmt_hdr, + output chdr_mgmt_t rx_mgmt_pl + ); + automatic timeout_t mgmt_timeout; + automatic ChdrPacket #(CHDR_W) tx_chdr = new(), rx_chdr; + tx_chdr.write_mgmt(tx_mgmt_hdr, tx_mgmt_pl); + test.start_timeout(mgmt_timeout, 2us, "Waiting for management transaction"); + if (VERBOSE) begin $write("Tx"); tx_chdr.print(); end + tb_chdr_bfm.put_chdr(tx_chdr.copy()); + tb_chdr_bfm.get_chdr(rx_chdr); + test.end_timeout(mgmt_timeout); + rx_chdr.read_mgmt(rx_mgmt_hdr, rx_mgmt_pl); + if (VERBOSE) begin $write("Rx"); rx_chdr.print(); end + endtask + + task automatic mgmt_read_err_counts( + input [15:0] dst_epid, + output [31:0] seq_err_count, + output [31:0] route_err_count, + output [31:0] data_err_count + ); + automatic chdr_header_t tx_mgmt_hdr, rx_mgmt_hdr; + automatic chdr_mgmt_t tx_mgmt_pl, rx_mgmt_pl; + automatic chdr_mgmt_op_t exp_mgmt_op; + + // Generic management header + tx_mgmt_pl.header = '{ + default:'0, prot_ver:PROTOVER, chdr_width:translate_chdr_w(CHDR_W), src_epid:EPID_TB + }; + // Read error counts + tx_mgmt_pl.header.num_hops = 3; + tx_mgmt_pl.ops.delete(); + + tx_mgmt_pl.ops[0] = '{ // Hop 1: Crossbar: Nop + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_pl.ops[1] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_SEQ_ERR_CNT}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd3}; + tx_mgmt_pl.ops[2] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_DATA_ERR_CNT}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd2}; + tx_mgmt_pl.ops[3] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_ROUTE_ERR_CNT}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd1}; + tx_mgmt_pl.ops[4] = '{ // Hop 2: Stream Endpoint: Return + op_payload:48'h0, op_code:MGMT_OP_RETURN, ops_pending:8'd0}; + tx_mgmt_pl.ops[5] = '{ // Hop 3: Nop for return + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_hdr = '{ + pkt_type:CHDR_MANAGEMENT, seq_num:cached_mgmt_seqnum++, dst_epid:dst_epid, default:'0}; + + // Send the packet and ensure that error counts are zero + send_recv_mgmt_packet(tx_mgmt_hdr, tx_mgmt_pl, rx_mgmt_hdr, rx_mgmt_pl); + `ASSERT_ERROR(rx_mgmt_pl.header.num_hops == 1, + "Check Errs: Mgmt header was incorrect"); + seq_err_count = 32'hx; + route_err_count = 32'hx; + data_err_count = 32'hx; + for (int i = 1; i <= 3; i++) begin + if (rx_mgmt_pl.ops[i].op_payload[15:0] == sep_a.REG_OSTRM_SEQ_ERR_CNT) + seq_err_count = rx_mgmt_pl.ops[i].op_payload[47:16]; + else if (rx_mgmt_pl.ops[i].op_payload[15:0] == sep_a.REG_OSTRM_DATA_ERR_CNT) + data_err_count = rx_mgmt_pl.ops[i].op_payload[47:16]; + else if (rx_mgmt_pl.ops[i].op_payload[15:0] == sep_a.REG_OSTRM_ROUTE_ERR_CNT) + route_err_count = rx_mgmt_pl.ops[i].op_payload[47:16]; + end + endtask + + task automatic send_recv_ctrl_packets( + input [15:0] dst_epid, + input [15:0] num_pkts, + input [15:0] seq_num_start + ); + for (int n = 0; n < num_pkts; n=n+1) begin + automatic timeout_t ctrl_timeout; + automatic ChdrPacket #(CHDR_W) tx_chdr = new(), rx_chdr, exp_chdr = new(); + automatic chdr_header_t chdr_hdr; + automatic chdr_ctrl_header_t ctrl_hdr, exp_ctrl_hdr; + automatic ctrl_op_word_t ctrl_op; + automatic ctrl_word_t ctrl_data[$]; + automatic chdr_word_t ctrl_ts; + + ctrl_data.delete(); + for (int i = 0; i < $urandom_range(15,1); i++) + ctrl_data[i] = $urandom(); + ctrl_hdr = '{ + default : '0, + src_epid : EPID_TB, + is_ack : 1'b0, + has_time : $urandom_range(1), + seq_num : seq_num_start[5:0], + num_data : ctrl_data.size(), + src_port : $urandom(), + dst_port : $urandom() + }; + ctrl_ts = $urandom(); + ctrl_op = '{ + default : '0, + op_code : ctrl_opcode_t'($urandom_range(9)), + byte_enable : $urandom_range(15), + address : $urandom() + }; + chdr_hdr = '{ + dst_epid : dst_epid, + seq_num : seq_num_start + n[15:0], + pkt_type : CHDR_CONTROL, + default : 0 + }; + tx_chdr.write_ctrl(chdr_hdr, ctrl_hdr, ctrl_op, ctrl_data, ctrl_ts); + + test.start_timeout(ctrl_timeout, 2us, "Waiting for management transaction"); + if (VERBOSE) begin $write("Tx"); tx_chdr.print(); end + tb_chdr_bfm.put_chdr(tx_chdr.copy()); + tb_chdr_bfm.get_chdr(rx_chdr); + test.end_timeout(ctrl_timeout); + if (VERBOSE) begin $write("Rx"); rx_chdr.print(); end + + exp_ctrl_hdr = ctrl_hdr; + exp_ctrl_hdr.dst_port = ctrl_hdr.src_port; + exp_ctrl_hdr.src_port = ctrl_hdr.dst_port; + exp_ctrl_hdr.src_epid = dst_epid; + exp_ctrl_hdr.is_ack = 1'b1; + + exp_chdr.write_ctrl(chdr_hdr, exp_ctrl_hdr, ctrl_op, ctrl_data, ctrl_ts); + exp_chdr.header.dst_epid = EPID_TB; + + + if (VERBOSE) begin $write("ExpRx"); exp_chdr.print(); end + + // Validate contents + `ASSERT_ERROR(exp_chdr.equal(rx_chdr), + "Received CHDR control packet was incorrect"); + end + endtask + + task automatic send_recv_data_packets( + input [15:0] src_epid, + input [15:0] dst_epid, + input [15:0] num_pkts, + input [15:0] seq_num_start, + input bit ignore_seq_route_errs = 0 + ); + // Pick a VC for this run randomly + logic [5:0] vc = $urandom_range(1); + fork + begin: tx_loop + for (int txi = 0; txi < num_pkts; txi=txi+1) begin + automatic timeout_t tx_timeout; + automatic ChdrPacket #(CHDR_W) tx_chdr = new(); + automatic chdr_header_t tx_hdr; + automatic chdr_word_t tx_ts; + automatic chdr_word_t tx_mdata[$]; + automatic chdr_word_t tx_data[$]; + // Fill data in the packet + tx_hdr = '{ + vc : vc, + dst_epid : dst_epid, + seq_num : seq_num_start + txi[15:0], + pkt_type : (txi%4==0) ? CHDR_DATA_WITH_TS : CHDR_DATA_NO_TS, + num_mdata : $urandom_range(5), + default : 0 + }; + tx_ts = txi; + tx_mdata.delete(); + for (int i = 0; i < tx_hdr.num_mdata; i++) + tx_mdata[i] = $urandom(); + tx_data.delete(); + for (int i = 0; i < $urandom_range((1<<MTU)-10); i++) + tx_data[i] = {txi << 16, i[15:0]}; + tx_chdr.write_raw(tx_hdr, tx_data, tx_mdata, tx_ts); + if (VERBOSE) $display("%s%0d:Tx:%0d:",(src_epid == EPID_A)?"A":"B", vc, txi, tx_chdr.sprint()); + // Send the packet + test.start_timeout(tx_timeout, 2us, "Waiting to send data packet"); + if (src_epid == EPID_A) + if (vc == 0) + a0_data_bfm.put_chdr(tx_chdr.copy()); + else + a1_data_bfm.put_chdr(tx_chdr.copy()); + else + if (vc == 0) + b0_data_bfm.put_chdr(tx_chdr.copy()); + else + b1_data_bfm.put_chdr(tx_chdr.copy()); + test.end_timeout(tx_timeout); + end + end + begin: rx_loop + for (int rxi = 0; rxi < num_pkts; rxi=rxi+1) begin + automatic timeout_t rx_timeout; + automatic ChdrPacket #(CHDR_W) rx_chdr; + // Receive a packet + test.start_timeout(rx_timeout, 2us, "Waiting to recv data packet"); + if (dst_epid == EPID_A) + if (vc == 0) + a0_data_bfm.get_chdr(rx_chdr); + else + a1_data_bfm.get_chdr(rx_chdr); + else + if (vc == 0) + b0_data_bfm.get_chdr(rx_chdr); + else + b1_data_bfm.get_chdr(rx_chdr); + test.end_timeout(rx_timeout); + // Validate the packet + if (VERBOSE) $display("%s:Rx%0d:%0d:",(src_epid == EPID_A)?"A":"B", vc, rxi, rx_chdr.sprint()); + `ASSERT_ERROR(ignore_seq_route_errs || rx_chdr.header.dst_epid == dst_epid, "Data Pkt: dst_epid was incorrect"); + `ASSERT_ERROR(ignore_seq_route_errs || (rx_chdr.header.seq_num == rxi + seq_num_start), "Data Pkt: seq_num was incorrect"); + if (rx_chdr.header.pkt_type == CHDR_DATA_WITH_TS) + `ASSERT_ERROR(rx_chdr.timestamp == rxi, "Data Pkt: timestamp was incorrect"); + foreach (rx_chdr.data[i]) begin + `ASSERT_ERROR(rx_chdr.data[i] == {rxi << 16, i[15:0]}, "Data Pkt: payload was incorrect"); + end + end + end + join + endtask + + task automatic set_unidir_stall_prob( + input [15:0] src_epid, + input [15:0] dst_epid, + int src_stall_prob, + int dst_stall_prob + ); + if (src_epid == EPID_A) begin + a0_data_bfm.set_master_stall_prob(src_stall_prob); + a1_data_bfm.set_master_stall_prob(src_stall_prob); + b0_data_bfm.set_slave_stall_prob (dst_stall_prob); + b1_data_bfm.set_slave_stall_prob (dst_stall_prob); + end else begin + b0_data_bfm.set_master_stall_prob(src_stall_prob); + b1_data_bfm.set_master_stall_prob(src_stall_prob); + a0_data_bfm.set_slave_stall_prob (dst_stall_prob); + a1_data_bfm.set_slave_stall_prob (dst_stall_prob); + end + endtask + + task automatic set_bidir_stall_prob( + int src_stall_prob, + int dst_stall_prob + ); + set_unidir_stall_prob(EPID_A, EPID_B, src_stall_prob, dst_stall_prob); + set_unidir_stall_prob(EPID_B, EPID_A, src_stall_prob, dst_stall_prob); + endtask + + // ---------------------------------------- + // Test Process + // ---------------------------------------- + initial begin + + // Shared Variables + // ---------------------------------------- + timeout_t timeout; + string tc_label; + bit stop_responder = 0; + logic [31:0] seq_err_count; + logic [31:0] route_err_count; + logic [31:0] data_err_count; + + a_signal_data_err = 0; + b_signal_data_err = 0; + a_seqerr_prob = 0; + a_rterr_prob = 0; + a_lossy_input = 0; + b_seqerr_prob = 0; + b_rterr_prob = 0; + b_lossy_input = 0; + + // Initialize + // ---------------------------------------- + test.start_tb("chdr_stream_endpoint_tb"); + + // Start the BFMs + a0_data_bfm.run(); + b0_data_bfm.run(); + a1_data_bfm.run(); + b1_data_bfm.run(); + tb_chdr_bfm.run(); + + tb_chdr_bfm.set_master_stall_prob(0); + tb_chdr_bfm.set_slave_stall_prob(0); + + // Reset + // ---------------------------------------- + rfnoc_ctrl_clk_gen.reset(); + rfnoc_chdr_clk_gen.reset(); + + test.start_test("Wait for reset"); + test.start_timeout(timeout, 1us, "Waiting for reset"); + while (rfnoc_ctrl_rst) @(posedge rfnoc_ctrl_clk); + while (rfnoc_chdr_rst) @(posedge rfnoc_chdr_clk); + test.end_timeout(timeout); + `ASSERT_ERROR(!rfnoc_chdr_rst && !rfnoc_ctrl_rst, "Reset did not deassert"); + test.end_test(); + + // Discover Topology + // ---------------------------------------- + test.start_test("Discover Topology"); + begin + automatic chdr_header_t tx_mgmt_hdr, rx_mgmt_hdr; + automatic chdr_mgmt_t tx_mgmt_pl, rx_mgmt_pl; + automatic chdr_mgmt_op_t exp_mgmt_op; + + // *Status* We know nothing about the network. Need to discover stuff. + + // Generic management header + tx_mgmt_pl.header = '{ + default:'0, prot_ver:PROTOVER, chdr_width:translate_chdr_w(CHDR_W), src_epid:EPID_TB + }; + // Send a node info request to the crossbar + tx_mgmt_pl.header.num_hops = 2; + tx_mgmt_pl.ops.delete(); + tx_mgmt_pl.ops[0] = '{ // Hop 1: Send node info + op_payload:48'h0, op_code:MGMT_OP_INFO_REQ, ops_pending:8'd1}; + tx_mgmt_pl.ops[1] = '{ // Hop 1: Return + op_payload:48'h0, op_code:MGMT_OP_RETURN, ops_pending:8'd0}; + tx_mgmt_pl.ops[2] = '{ // Hop 2: Nop for return + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_hdr = '{ + pkt_type:CHDR_MANAGEMENT, seq_num:cached_mgmt_seqnum++, dst_epid:16'h0, default:'0}; + + // Send the packet and check the response + send_recv_mgmt_packet(tx_mgmt_hdr, tx_mgmt_pl, rx_mgmt_hdr, rx_mgmt_pl); + `ASSERT_ERROR(rx_mgmt_pl.header.num_hops == 1, + "Discover XB: Mgmt header was incorrect"); + exp_mgmt_op = '{op_payload:{2'h0, 8'd1/*ports_mgmt*/, 8'd3 /*ports*/, 10'd0 /*inst*/, 4'd1 /*type*/, DEV_ID}, + op_code:MGMT_OP_INFO_RESP, ops_pending:8'd0}; + `ASSERT_ERROR(rx_mgmt_pl.ops[1] == exp_mgmt_op, + "Discover XB: Mgmt response ops were incorrect"); + + // *Status* We just discovered a crossbar with 3 ports! + + // Configure the crossbar routing table with our (TB) address + // then send node info request on the other two ports + tx_mgmt_pl.header.num_hops = 3; + tx_mgmt_pl.ops.delete(); + tx_mgmt_pl.ops[0] = '{ // Hop 1: Crossbar: Config router to return packet to dest + op_payload:{22'h0, PORT_TB, EPID_TB}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd1}; + tx_mgmt_pl.ops[1] = '{ // Hop 1: Crossbar: Config router + op_payload:{38'h0, PORT_A}, op_code:MGMT_OP_SEL_DEST, ops_pending:8'd0}; + tx_mgmt_pl.ops[2] = '{ // Hop 2: Stream Endpoint: Send node info + op_payload:48'h0, op_code:MGMT_OP_INFO_REQ, ops_pending:8'd1}; + tx_mgmt_pl.ops[3] = '{ // Hop 2: Return + op_payload:48'h0, op_code:MGMT_OP_RETURN, ops_pending:8'd0}; + tx_mgmt_pl.ops[4] = '{ // Hop 3: TB: Nop for return + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_hdr = '{ + pkt_type:CHDR_MANAGEMENT, seq_num:cached_mgmt_seqnum++, dst_epid:16'h0, default:'0}; + + // Send the packet and check the response + send_recv_mgmt_packet(tx_mgmt_hdr, tx_mgmt_pl, rx_mgmt_hdr, rx_mgmt_pl); + `ASSERT_ERROR(rx_mgmt_pl.header.num_hops == 1, + "Discover SEP A: Mgmt header was incorrect"); + exp_mgmt_op = '{op_payload:{{4'd1, 6'd2, 6'd2, 2'b11} /*ext_info*/, 10'd0 /*inst*/, 4'd2 /*type*/, DEV_ID}, + op_code:MGMT_OP_INFO_RESP, ops_pending:8'd0}; + `ASSERT_ERROR(rx_mgmt_pl.ops[1] == exp_mgmt_op, + "Discover SEP A: Mgmt response ops were incorrect"); + + // *Status* We just discovered a stream endpoint on crossbar port 1 + + // Send node info request on the last port + tx_mgmt_pl.header.num_hops = 3; + tx_mgmt_pl.ops.delete(); + tx_mgmt_pl.ops[0] = '{ // Hop 1: Crossbar: Config router + op_payload:{38'h0, PORT_B}, op_code:MGMT_OP_SEL_DEST, ops_pending:8'd0}; + tx_mgmt_pl.ops[1] = '{ // Hop 2: Stream Endpoint: Send node info + op_payload:48'h0, op_code:MGMT_OP_INFO_REQ, ops_pending:8'd1}; + tx_mgmt_pl.ops[2] = '{ // Hop 2: Return + op_payload:48'h0, op_code:MGMT_OP_RETURN, ops_pending:8'd0}; + tx_mgmt_pl.ops[3] = '{ // Hop 3: TB: Nop for return + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_hdr = '{ + pkt_type:CHDR_MANAGEMENT, seq_num:cached_mgmt_seqnum++, dst_epid:16'h0, default:'0}; + + // Send the packet and check the response + send_recv_mgmt_packet(tx_mgmt_hdr, tx_mgmt_pl, rx_mgmt_hdr, rx_mgmt_pl); + `ASSERT_ERROR(rx_mgmt_pl.header.num_hops == 1, + "Discover SEP B: Mgmt header was incorrect"); + exp_mgmt_op = '{op_payload:{{4'd1, 6'd2, 6'd2, 2'b11} /*ext_info*/, 10'd1 /*inst*/, 4'd2 /*type*/, DEV_ID}, + op_code:MGMT_OP_INFO_RESP, ops_pending:8'd0}; + `ASSERT_ERROR(rx_mgmt_pl.ops[1] == exp_mgmt_op, + "Discover SEP B: Mgmt response ops were incorrect"); + + // *Status* We just discovered a stream endpoint on crossbar port 2 + end + test.end_test(); + + // Configure Routes to Stream Endpoints A and B + // ---------------------------------------- + test.start_test("Configure Routes"); + begin + automatic chdr_header_t tx_mgmt_hdr, rx_mgmt_hdr; + automatic chdr_mgmt_t tx_mgmt_pl, rx_mgmt_pl; + automatic chdr_mgmt_op_t exp_mgmt_op; + + // Generic management header + tx_mgmt_pl.header = '{ + default:'0, prot_ver:PROTOVER, chdr_width:translate_chdr_w(CHDR_W), src_epid:EPID_TB + }; + // Send a node info request to the crossbar + tx_mgmt_pl.header.num_hops = 2; + tx_mgmt_pl.ops.delete(); + + tx_mgmt_pl.ops[0] = '{ // Hop 1: Crossbar: Config path to EP A + op_payload:{22'h0, PORT_A, EPID_A}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd2}; + tx_mgmt_pl.ops[1] = '{ // Hop 1: Crossbar: Config path to EP B + op_payload:{22'h0, PORT_B, EPID_B}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd1}; + tx_mgmt_pl.ops[2] = '{ // Hop 1: Request node info to make the packet come back + op_payload:48'h0, op_code:MGMT_OP_RETURN, ops_pending:8'd0}; + tx_mgmt_pl.ops[3] = '{ // Hop 2: Nop for return + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_hdr = '{ + pkt_type:CHDR_MANAGEMENT, seq_num:cached_mgmt_seqnum++, dst_epid:16'h0, default:'0}; + + // Send the packet and check the response + send_recv_mgmt_packet(tx_mgmt_hdr, tx_mgmt_pl, rx_mgmt_hdr, rx_mgmt_pl); + `ASSERT_ERROR(rx_mgmt_pl.header.num_hops == 1, + "Config Routes: Mgmt header was incorrect"); + exp_mgmt_op = '{op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + `ASSERT_ERROR(rx_mgmt_pl.ops[0] == exp_mgmt_op, + "Config Routes: Mgmt response ops were incorrect"); + end + test.end_test(); + + // Configure Stream Endpoints + // ---------------------------------------- + test.start_test("Configure Stream Endpoints"); + begin + automatic chdr_header_t tx_mgmt_hdr, rx_mgmt_hdr; + automatic chdr_mgmt_t tx_mgmt_pl, rx_mgmt_pl; + automatic chdr_mgmt_op_t exp_mgmt_op; + + logic [15:0] epids[2] = {EPID_A, EPID_B}; + foreach (epids[i]) begin + // Generic management header + tx_mgmt_pl.header = '{ + default:'0, prot_ver:PROTOVER, chdr_width:translate_chdr_w(CHDR_W), src_epid:EPID_TB + }; + // Send a node info request to the crossbar + tx_mgmt_pl.header.num_hops = 3; + tx_mgmt_pl.ops.delete(); + + tx_mgmt_pl.ops[0] = '{ // Hop 1: Crossbar: Nop + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_pl.ops[1] = '{ // Hop 2: Reset + op_payload:{32'b111, sep_a.REG_RESET_AND_FLUSH}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd4}; + tx_mgmt_pl.ops[2] = '{ // Hop 2: Write EPID + op_payload:{16'h0, epids[i], sep_a.REG_EPID_SELF}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd3}; + tx_mgmt_pl.ops[3] = '{ // Hop 2: Read EPID + op_payload:{32'h0, sep_a.REG_EPID_SELF}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd2}; + tx_mgmt_pl.ops[4] = '{ // Hop 2: Read EPID + op_payload:{32'h0, sep_a.REG_OSTRM_CTRL_STATUS}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd1}; + tx_mgmt_pl.ops[5] = '{ // Hop 2: Stream Endpoint: Return + op_payload:48'h0, op_code:MGMT_OP_RETURN, ops_pending:8'd0}; + tx_mgmt_pl.ops[6] = '{ // Hop 3: Nop for return + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_hdr = '{ + pkt_type:CHDR_MANAGEMENT, seq_num:cached_mgmt_seqnum++, dst_epid:epids[i], default:'0}; + + // Send the packet and check the response + send_recv_mgmt_packet(tx_mgmt_hdr, tx_mgmt_pl, rx_mgmt_hdr, rx_mgmt_pl); + `ASSERT_ERROR(rx_mgmt_pl.header.num_hops == 1, + "Config SEP: Mgmt header was incorrect"); + exp_mgmt_op = '{op_payload:{16'h0, epids[i], sep_a.REG_EPID_SELF}, + op_code:MGMT_OP_CFG_RD_RESP, ops_pending:8'd1}; + `ASSERT_ERROR(rx_mgmt_pl.ops[1] == exp_mgmt_op, + "Config SEP: Mgmt response ops were incorrect"); + exp_mgmt_op = '{op_payload:{32'h0, sep_a.REG_OSTRM_CTRL_STATUS}, + op_code:MGMT_OP_CFG_RD_RESP, ops_pending:8'd0}; + `ASSERT_ERROR(rx_mgmt_pl.ops[2] == exp_mgmt_op, + "Config SEP: Mgmt response ops were incorrect"); + end + end + test.end_test(); + + // Setup a stream between Endpoint A and B + // ---------------------------------------- + test.start_test("Setup bidirectional stream between endpoints A and B"); + begin + automatic chdr_header_t tx_mgmt_hdr, rx_mgmt_hdr; + automatic chdr_mgmt_t tx_mgmt_pl, rx_mgmt_pl; + automatic chdr_mgmt_op_t exp_mgmt_op; + + logic [15:0] epids[2] = {EPID_A, EPID_B}; + foreach (epids[i]) begin + // Generic management header + tx_mgmt_pl.header = '{ + default:'0, prot_ver:PROTOVER, chdr_width:translate_chdr_w(CHDR_W), src_epid:EPID_TB + }; + // Configure FC on streams + tx_mgmt_pl.header.num_hops = 3; + tx_mgmt_pl.ops.delete(); + + tx_mgmt_pl.ops[0] = '{ // Hop 1: Crossbar: Nop + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_pl.ops[1] = '{ // Hop 2: Write destination EPID + op_payload:{16'h0, epids[1-i], sep_a.REG_OSTRM_DST_EPID}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd7}; + tx_mgmt_pl.ops[2] = '{ // Hop 2: Configure flow ack control freq + op_payload:{32'd50, sep_a.REG_OSTRM_FC_FREQ_BYTES_LO}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd6}; + tx_mgmt_pl.ops[3] = '{ // Hop 2: Configure flow ack control freq + op_payload:{32'd0, sep_a.REG_OSTRM_FC_FREQ_BYTES_HI}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd5}; + tx_mgmt_pl.ops[4] = '{ // Hop 2: Configure flow ack control freq + op_payload:{32'd1000, sep_a.REG_OSTRM_FC_FREQ_PKTS}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd4}; + tx_mgmt_pl.ops[5] = '{ // Hop 2: Configure flow headroom + op_payload:{32'd0, sep_a.REG_OSTRM_FC_HEADROOM}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd3}; + tx_mgmt_pl.ops[6] = '{ // Hop 2: Configure word swapping + op_payload:{32'h44, sep_a.REG_ISTRM_CTRL_STATUS}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd2}; // Swap 32-bit words, endianness + tx_mgmt_pl.ops[7] = '{ // Hop 2: Configure lossy and start config + op_payload:{32'h47, sep_a.REG_OSTRM_CTRL_STATUS}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd1}; // Swap 32-bit words, endianness, lossy and reset + tx_mgmt_pl.ops[8] = '{ // Hop 2: Stream Endpoint: Return + op_payload:48'h0, op_code:MGMT_OP_RETURN, ops_pending:8'd0}; + tx_mgmt_pl.ops[9] = '{ // Hop 3: Nop for return + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_hdr = '{ + pkt_type:CHDR_MANAGEMENT, seq_num:cached_mgmt_seqnum++, dst_epid:epids[i], default:'0}; + + // Send the packet and check the response + send_recv_mgmt_packet(tx_mgmt_hdr, tx_mgmt_pl, rx_mgmt_hdr, rx_mgmt_pl); + + // Wait for some time for node to flush and reset + // Typically we would poll in SW but we just wait to keep the code simple + repeat (256) @(posedge rfnoc_chdr_clk); + + // Read back FC status + tx_mgmt_pl.header.num_hops = 3; + tx_mgmt_pl.ops.delete(); + + tx_mgmt_pl.ops[0] = '{ // Hop 1: Crossbar: Nop + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_pl.ops[1] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_CTRL_STATUS}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd7}; + tx_mgmt_pl.ops[2] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_BUFF_CAP_BYTES_LO}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd6}; + tx_mgmt_pl.ops[3] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_BUFF_CAP_BYTES_HI}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd5}; + tx_mgmt_pl.ops[4] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_BUFF_CAP_PKTS}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd4}; + tx_mgmt_pl.ops[5] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_SEQ_ERR_CNT}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd3}; + tx_mgmt_pl.ops[6] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_DATA_ERR_CNT}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd2}; + tx_mgmt_pl.ops[7] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_ROUTE_ERR_CNT}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd1}; + tx_mgmt_pl.ops[8] = '{ // Hop 2: Stream Endpoint: Return + op_payload:48'h0, op_code:MGMT_OP_RETURN, ops_pending:8'd0}; + tx_mgmt_pl.ops[9] = '{ // Hop 3: Nop for return + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_hdr = '{ + pkt_type:CHDR_MANAGEMENT, seq_num:cached_mgmt_seqnum++, dst_epid:epids[i], default:'0}; + + // Send the packet and check the response + send_recv_mgmt_packet(tx_mgmt_hdr, tx_mgmt_pl, rx_mgmt_hdr, rx_mgmt_pl); + `ASSERT_ERROR(rx_mgmt_pl.header.num_hops == 1, + "Config SEP: Mgmt header was incorrect"); + exp_mgmt_op = '{op_payload:{32'h80000006, sep_a.REG_OSTRM_CTRL_STATUS}, // FC on, no errors and lossy + op_code:MGMT_OP_CFG_RD_RESP, ops_pending:8'd6}; + `ASSERT_ERROR(rx_mgmt_pl.ops[1] == exp_mgmt_op, "Config SEP: Mgmt response was incorrect"); + exp_mgmt_op = '{op_payload:{((1<<(MTU+1))*(CHDR_W/8)-1), sep_a.REG_OSTRM_BUFF_CAP_BYTES_LO}, + op_code:MGMT_OP_CFG_RD_RESP, ops_pending:8'd5}; + `ASSERT_ERROR(rx_mgmt_pl.ops[2] == exp_mgmt_op, "Config SEP: Mgmt response was incorrect"); + exp_mgmt_op = '{op_payload:{32'h0, sep_a.REG_OSTRM_BUFF_CAP_BYTES_HI}, + op_code:MGMT_OP_CFG_RD_RESP, ops_pending:8'd4}; + `ASSERT_ERROR(rx_mgmt_pl.ops[3] == exp_mgmt_op, "Config SEP: Mgmt response was incorrect"); + exp_mgmt_op = '{op_payload:{32'h00ffffff, sep_a.REG_OSTRM_BUFF_CAP_PKTS}, + op_code:MGMT_OP_CFG_RD_RESP, ops_pending:8'd3}; + `ASSERT_ERROR(rx_mgmt_pl.ops[4] == exp_mgmt_op, "Config SEP: Mgmt response was incorrect"); + exp_mgmt_op = '{op_payload:{32'h0, sep_a.REG_OSTRM_SEQ_ERR_CNT}, + op_code:MGMT_OP_CFG_RD_RESP, ops_pending:8'd2}; + `ASSERT_ERROR(rx_mgmt_pl.ops[5] == exp_mgmt_op, "Config SEP: Mgmt response was incorrect"); + exp_mgmt_op = '{op_payload:{32'h0, sep_a.REG_OSTRM_DATA_ERR_CNT}, + op_code:MGMT_OP_CFG_RD_RESP, ops_pending:8'd1}; + `ASSERT_ERROR(rx_mgmt_pl.ops[6] == exp_mgmt_op, "Config SEP: Mgmt response was incorrect"); + exp_mgmt_op = '{op_payload:{32'h0, sep_a.REG_OSTRM_ROUTE_ERR_CNT}, + op_code:MGMT_OP_CFG_RD_RESP, ops_pending:8'd0}; + `ASSERT_ERROR(rx_mgmt_pl.ops[7] == exp_mgmt_op, "Config SEP: Mgmt response was incorrect"); + end + end + test.end_test(); + + // Control transactions to Endpoint A + // ---------------------------------------- + cached_ctrl_seqnum = 0; + for (int cfg = 0; cfg < 2; cfg++) begin + $sformat(tc_label, "Control Xact to A (%s)", (cfg?"Slow":"Fast")); + test.start_test(tc_label); + begin + tb_chdr_bfm.set_master_stall_prob(cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + tb_chdr_bfm.set_slave_stall_prob(cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + send_recv_ctrl_packets(EPID_A, NUM_PKTS_PER_TEST, cached_ctrl_seqnum); + end + test.end_test(); + cached_ctrl_seqnum += NUM_PKTS_PER_TEST; + end + + // Control transactions to Endpoint B + // ---------------------------------------- + cached_ctrl_seqnum = 0; + for (int cfg = 0; cfg < 2; cfg++) begin + $sformat(tc_label, "Control Xact to B (%s)", (cfg?"Slow":"Fast")); + test.start_test(tc_label); + begin + tb_chdr_bfm.set_master_stall_prob(cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + tb_chdr_bfm.set_slave_stall_prob(cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + send_recv_ctrl_packets(EPID_B, NUM_PKTS_PER_TEST, cached_ctrl_seqnum); + end + test.end_test(); + cached_ctrl_seqnum += NUM_PKTS_PER_TEST; + end + + // Stream data from A to B + // ---------------------------------------- + cached_data_seqnum = 0; + for (int cfg = 0; cfg < 4; cfg++) begin + automatic logic mst_cfg = cfg[0]; + automatic logic slv_cfg = cfg[1]; + $sformat(tc_label, "Stream Data from A to B (%s Mst, %s Slv)", + (mst_cfg?"Slow":"Fast"), (slv_cfg?"Slow":"Fast")); + test.start_test(tc_label); + begin + set_unidir_stall_prob(EPID_A, EPID_B, + mst_cfg?SLOW_STALL_PROB:FAST_STALL_PROB, + slv_cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + send_recv_data_packets(EPID_A, EPID_B, NUM_PKTS_PER_TEST, cached_data_seqnum); + end + test.end_test(); + cached_data_seqnum += NUM_PKTS_PER_TEST; + end + + // Stream data from B to A + // ---------------------------------------- + cached_data_seqnum = 0; + for (int cfg = 0; cfg < 4; cfg++) begin + automatic logic mst_cfg = cfg[0]; + automatic logic slv_cfg = cfg[1]; + $sformat(tc_label, "Stream Data from B to A (%s Mst, %s Slv)", + (mst_cfg?"Slow":"Fast"), (slv_cfg?"Slow":"Fast")); + test.start_test(tc_label); + begin + set_unidir_stall_prob(EPID_B, EPID_A, + mst_cfg?SLOW_STALL_PROB:FAST_STALL_PROB, + slv_cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + send_recv_data_packets(EPID_B, EPID_A, NUM_PKTS_PER_TEST, cached_data_seqnum); + end + test.end_test(); + cached_data_seqnum += NUM_PKTS_PER_TEST; + end + + // Stream data between A <=> B simultaneously + // ---------------------------------------- + for (int cfg = 0; cfg < 4; cfg++) begin + automatic logic mst_cfg = cfg[0]; + automatic logic slv_cfg = cfg[1]; + $sformat(tc_label, "Stream Data between A <=> B simultaneously (%s Mst, %s Slv)", + (mst_cfg?"Slow":"Fast"), (slv_cfg?"Slow":"Fast")); + test.start_test(tc_label); + begin + set_bidir_stall_prob( + mst_cfg?SLOW_STALL_PROB:FAST_STALL_PROB, + slv_cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + fork + send_recv_data_packets(EPID_B, EPID_A, NUM_PKTS_PER_TEST, cached_data_seqnum); + send_recv_data_packets(EPID_A, EPID_B, NUM_PKTS_PER_TEST, cached_data_seqnum); + join + end + test.end_test(); + cached_data_seqnum += NUM_PKTS_PER_TEST; + end + + // Stream data and control between A <=> B simultaneously + // ---------------------------------------- + for (int cfg = 0; cfg < 4; cfg++) begin + automatic logic mst_cfg = cfg[0]; + automatic logic slv_cfg = cfg[1]; + $sformat(tc_label, "Stream Data and Control between A <=> B (%s Mst, %s Slv)", + (mst_cfg?"Slow":"Fast"), (slv_cfg?"Slow":"Fast")); + test.start_test(tc_label); + begin + tb_chdr_bfm.set_master_stall_prob(mst_cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + tb_chdr_bfm.set_slave_stall_prob(slv_cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + set_bidir_stall_prob( + mst_cfg?SLOW_STALL_PROB:FAST_STALL_PROB, + slv_cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + fork + send_recv_data_packets(EPID_B, EPID_A, NUM_PKTS_PER_TEST/2, cached_data_seqnum); + send_recv_data_packets(EPID_A, EPID_B, NUM_PKTS_PER_TEST/2, cached_data_seqnum); + send_recv_ctrl_packets(EPID_A, NUM_PKTS_PER_TEST, cached_ctrl_seqnum); + join + cached_data_seqnum += NUM_PKTS_PER_TEST/2; + fork + send_recv_data_packets(EPID_B, EPID_A, NUM_PKTS_PER_TEST/2, cached_data_seqnum); + send_recv_data_packets(EPID_A, EPID_B, NUM_PKTS_PER_TEST/2, cached_data_seqnum); + send_recv_ctrl_packets(EPID_B, NUM_PKTS_PER_TEST, cached_ctrl_seqnum); + join + cached_data_seqnum += NUM_PKTS_PER_TEST/2; + cached_ctrl_seqnum += NUM_PKTS_PER_TEST; + end + test.end_test(); + end + + // Check zero sequence errors after streaming + // ---------------------------------------- + test.start_test("Check zero sequence errors after streaming"); + begin + logic [15:0] epids[2] = {EPID_A, EPID_B}; + foreach (epids[i]) begin + mgmt_read_err_counts(epids[i], seq_err_count, route_err_count, data_err_count); + `ASSERT_ERROR(seq_err_count == 32'd0, "Check NoErrs: Incorrect seq error count"); + `ASSERT_ERROR(route_err_count == 32'd0, "Check NoErrs: Incorrect route error count"); + `ASSERT_ERROR(data_err_count == 32'd0, "Check NoErrs: Incorrect data error count"); + end + end + test.end_test(); + + // Force sequence error + // ---------------------------------------- + test.start_test("Force sequence error"); + begin + // First sequence error + send_recv_data_packets(EPID_A, EPID_B, 1, cached_data_seqnum++, 1); + b_seqerr_prob = 100; // Simulate a dropped packet + send_recv_data_packets(EPID_A, EPID_B, 1, cached_data_seqnum++, 1); + b_seqerr_prob = 0; + repeat (100) @(posedge rfnoc_chdr_clk); // Wait for sequence error to reach the upstream port + mgmt_read_err_counts(EPID_A, seq_err_count, route_err_count, data_err_count); + `ASSERT_ERROR(seq_err_count == 32'd1, "Force SeqErr: Incorrect seq error count"); + `ASSERT_ERROR(route_err_count == 32'd0, "Force SeqErr: Incorrect route error count"); + `ASSERT_ERROR(data_err_count == 32'd0, "Force SeqErr: Incorrect data error count"); + + // Second and third sequence error + send_recv_data_packets(EPID_A, EPID_B, 1, cached_data_seqnum++, 1); + b_seqerr_prob = 100; // Simulate another dropped packet + send_recv_data_packets(EPID_A, EPID_B, 1, cached_data_seqnum++, 1); + b_seqerr_prob = 0; + repeat (100) @(posedge rfnoc_chdr_clk); // Wait for sequence error to reach the upstream port + mgmt_read_err_counts(EPID_A, seq_err_count, route_err_count, data_err_count); + `ASSERT_ERROR(seq_err_count > 32'd1, "Force SeqErr: Incorrect seq error count"); + `ASSERT_ERROR(route_err_count == 32'd0, "Force SeqErr: Incorrect route error count"); + `ASSERT_ERROR(data_err_count == 32'd0, "Force SeqErr: Incorrect data error count"); + end + test.end_test(); + + // Force routing error + // ---------------------------------------- + test.start_test("Force routing error"); + begin + logic [31:0] old_route_err_count; + // First sequence error + send_recv_data_packets(EPID_B, EPID_A, 1, cached_data_seqnum++, 1); + a_rterr_prob = 100; // Simulate a routing error + send_recv_data_packets(EPID_B, EPID_A, 1, cached_data_seqnum++, 1); + a_rterr_prob = 0; + repeat (100) @(posedge rfnoc_chdr_clk); // Wait for sequence error to reach the upstream port + mgmt_read_err_counts(EPID_B, seq_err_count, route_err_count, data_err_count); + `ASSERT_ERROR(seq_err_count == 32'd0, "Force RouteErr 1: Incorrect seq error count"); + `ASSERT_ERROR(route_err_count > 32'd0, "Force RouteErr 1: Incorrect route error count"); + `ASSERT_ERROR(data_err_count == 32'd0, "Force RouteErr 1: Incorrect data error count"); + old_route_err_count = route_err_count; + + // Second routing error + send_recv_data_packets(EPID_B, EPID_A, 1, cached_data_seqnum++, 1); + a_rterr_prob = 100; // Simulate a routing error + send_recv_data_packets(EPID_B, EPID_A, 1, cached_data_seqnum++, 1); + a_rterr_prob = 0; + repeat (100) @(posedge rfnoc_chdr_clk); // Wait for sequence error to reach the upstream port + mgmt_read_err_counts(EPID_B, seq_err_count, route_err_count, data_err_count); + `ASSERT_ERROR(seq_err_count == 32'd0, "Force RouteErr 2: Incorrect seq error count"); + `ASSERT_ERROR(route_err_count > old_route_err_count, "Force RouteErr 2: Incorrect route error count"); + `ASSERT_ERROR(data_err_count == 32'd0, "Force RouteErr 2: Incorrect data error count"); + end + test.end_test(); + + // Setup a stream between Endpoint A and B + // ---------------------------------------- + test.start_test("Reconfigure flow control (reset state)"); + begin + automatic chdr_header_t tx_mgmt_hdr, rx_mgmt_hdr; + automatic chdr_mgmt_t tx_mgmt_pl, rx_mgmt_pl; + automatic chdr_mgmt_op_t exp_mgmt_op; + + logic [15:0] epids[2] = {EPID_A, EPID_B}; + foreach (epids[i]) begin + // Generic management header + tx_mgmt_pl.header = '{ + default:'0, prot_ver:PROTOVER, chdr_width:translate_chdr_w(CHDR_W), src_epid:EPID_TB + }; + // Configure FC on streams + tx_mgmt_pl.header.num_hops = 3; + tx_mgmt_pl.ops.delete(); + + tx_mgmt_pl.ops[0] = '{ // Hop 1: Crossbar: Nop + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_pl.ops[1] = '{ // Hop 2: Disable swapping + op_payload:{32'd0, sep_a.REG_ISTRM_CTRL_STATUS}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd2}; + tx_mgmt_pl.ops[2] = '{ // Hop 2: Configure lossy and start config + op_payload:{32'd3, sep_a.REG_OSTRM_CTRL_STATUS}, op_code:MGMT_OP_CFG_WR_REQ, ops_pending:8'd1}; + tx_mgmt_pl.ops[3] = '{ // Hop 2: Stream Endpoint: Return + op_payload:48'h0, op_code:MGMT_OP_RETURN, ops_pending:8'd0}; + tx_mgmt_pl.ops[4] = '{ // Hop 3: Nop for return + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_hdr = '{ + pkt_type:CHDR_MANAGEMENT, seq_num:cached_mgmt_seqnum++, dst_epid:epids[i], default:'0}; + + // Send the packet and check the response + send_recv_mgmt_packet(tx_mgmt_hdr, tx_mgmt_pl, rx_mgmt_hdr, rx_mgmt_pl); + + // Wait for some time for node to flush and reset + // Typically we would poll in SW but we just wait to keep the code simple + repeat (256) @(posedge rfnoc_chdr_clk); + + // Read back FC status + tx_mgmt_pl.header.num_hops = 3; + tx_mgmt_pl.ops.delete(); + + tx_mgmt_pl.ops[0] = '{ // Hop 1: Crossbar: Nop + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_pl.ops[1] = '{ // Hop 2: Read status + op_payload:{32'h0, sep_a.REG_OSTRM_CTRL_STATUS}, op_code:MGMT_OP_CFG_RD_REQ, ops_pending:8'd1}; + tx_mgmt_pl.ops[2] = '{ // Hop 2: Stream Endpoint: Return + op_payload:48'h0, op_code:MGMT_OP_RETURN, ops_pending:8'd0}; + tx_mgmt_pl.ops[3] = '{ // Hop 3: Nop for return + op_payload:48'h0, op_code:MGMT_OP_NOP, ops_pending:8'd0}; + tx_mgmt_hdr = '{ + pkt_type:CHDR_MANAGEMENT, seq_num:cached_mgmt_seqnum++, dst_epid:epids[i], default:'0}; + + // Send the packet and check the response + send_recv_mgmt_packet(tx_mgmt_hdr, tx_mgmt_pl, rx_mgmt_hdr, rx_mgmt_pl); + `ASSERT_ERROR(rx_mgmt_pl.header.num_hops == 1, + "Config SEP: Mgmt header was incorrect"); + exp_mgmt_op = '{op_payload:{32'h80000002, sep_a.REG_OSTRM_CTRL_STATUS}, // FC on, no errors and lossy + op_code:MGMT_OP_CFG_RD_RESP, ops_pending:8'd0}; + `ASSERT_ERROR(rx_mgmt_pl.ops[1] == exp_mgmt_op, "Config SEP: Mgmt response was incorrect"); + end + end + test.end_test(); + + // Check zero errors after reinit + // ---------------------------------------- + test.start_test("Check zero errors after reinit"); + begin + logic [15:0] epids[2] = {EPID_A, EPID_B}; + foreach (epids[i]) begin + mgmt_read_err_counts(epids[i], seq_err_count, route_err_count, data_err_count); + `ASSERT_ERROR(seq_err_count == 32'd0, "Check NoErrs: Incorrect seq error count"); + `ASSERT_ERROR(route_err_count == 32'd0, "Check NoErrs: Incorrect route error count"); + `ASSERT_ERROR(data_err_count == 32'd0, "Check NoErrs: Incorrect data error count"); + end + end + test.end_test(); + + // Stream data between A <=> B simultaneously + // ---------------------------------------- + test.start_test("Stream Data between A <=> B with a lossy link"); + begin + cached_data_seqnum = 0; + set_bidir_stall_prob(FAST_STALL_PROB, SLOW_STALL_PROB); + a_lossy_input = 1; + b_lossy_input = 1; + fork + send_recv_data_packets(EPID_B, EPID_A, NUM_PKTS_PER_TEST * 10, cached_data_seqnum); + send_recv_data_packets(EPID_A, EPID_B, NUM_PKTS_PER_TEST * 10, cached_data_seqnum); + join + a_lossy_input = 0; + b_lossy_input = 0; + end + test.end_test(); + cached_data_seqnum += NUM_PKTS_PER_TEST*10; + + // Finish Up + // ---------------------------------------- + // Display final statistics and results + test.end_tb(); + end + +endmodule diff --git a/fpga/usrp3/lib/rfnoc/sim/chdr_stream_endpoint_tb/lossy_xport_model.v b/fpga/usrp3/lib/rfnoc/sim/chdr_stream_endpoint_tb/lossy_xport_model.v new file mode 100644 index 000000000..a93b4b305 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/sim/chdr_stream_endpoint_tb/lossy_xport_model.v @@ -0,0 +1,66 @@ +// +// Copyright 2019 Ettus Research, A National Instruments Company +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: lossy_xport_model +// + +module lossy_xport_model #( + parameter CHDR_W = 256 +)( + input wire clk, + input wire rst, + input wire [CHDR_W-1:0] s_axis_tdata, + input wire s_axis_tlast, + input wire s_axis_tvalid, + output wire s_axis_tready, + output wire [CHDR_W-1:0] m_axis_tdata, + output wire m_axis_tlast, + output wire m_axis_tvalid, + input wire m_axis_tready, + input wire [7:0] seqerr_prob, + input wire [7:0] rterr_prob, + input wire lossy +); + wire [CHDR_W-1:0] tmp_tdata; + wire tmp_tlast; + wire tmp_tvalid; + wire tmp_tready; + + reg pkt_header = 1'b1; + always @(posedge clk) begin + if (rst) begin + pkt_header <= 1'b1; + end else if (s_axis_tvalid && s_axis_tready) begin + pkt_header <= s_axis_tlast; + end + end + wire pkt_stb = (s_axis_tvalid && s_axis_tready && s_axis_tlast); + + reg force_seq_err, force_route_err; + always @(pkt_stb or seqerr_prob) begin + force_seq_err = ($urandom_range(99) < seqerr_prob); + end + always @(pkt_stb or rterr_prob) begin + force_route_err = ($urandom_range(99) < rterr_prob); + end + + wire [15:0] new_seq_num = s_axis_tdata[47:32] + 16'd1; //Increment SeqNum + wire [15:0] new_dst_epid = ~s_axis_tdata[15:0]; //Invert DstEPID + + assign tmp_tdata = !pkt_header ? s_axis_tdata : ( + force_seq_err ? {s_axis_tdata[CHDR_W-1:48], new_seq_num, s_axis_tdata[31:0]} : ( + force_route_err ? {s_axis_tdata[CHDR_W-1:16], new_dst_epid} : s_axis_tdata)); + assign tmp_tlast = s_axis_tlast; + assign tmp_tvalid = s_axis_tvalid; + assign s_axis_tready = lossy || tmp_tready; + + axi_fifo #(.WIDTH(CHDR_W+1), .SIZE(1)) out_fifo ( + .clk(clk), .reset(rst), .clear(1'b0), + .i_tdata({tmp_tlast, tmp_tdata}), .i_tvalid(tmp_tvalid), .i_tready(tmp_tready), + .o_tdata({m_axis_tlast, m_axis_tdata}), .o_tvalid(m_axis_tvalid), .o_tready(m_axis_tready), + .space(), .occupied() + ); + +endmodule
\ No newline at end of file diff --git a/fpga/usrp3/lib/rfnoc/sim/ctrlport_endpoint_tb/Makefile b/fpga/usrp3/lib/rfnoc/sim/ctrlport_endpoint_tb/Makefile new file mode 100644 index 000000000..f1f064547 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/sim/ctrlport_endpoint_tb/Makefile @@ -0,0 +1,39 @@ +# +# Copyright 2019 Ettus Research, A National Instruments Company +# +# 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 + +DESIGN_SRCS += $(abspath \ +$(RFNOC_CORE_SRCS) \ +) + +#------------------------------------------------- +# Testbench Specific +#------------------------------------------------- +SIM_TOP = ctrlport_endpoint_tb + +SIM_SRCS = \ +$(abspath ctrlport_endpoint_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/sim/ctrlport_endpoint_tb/ctrlport_endpoint_tb.sv b/fpga/usrp3/lib/rfnoc/sim/ctrlport_endpoint_tb/ctrlport_endpoint_tb.sv new file mode 100644 index 000000000..492e48829 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/sim/ctrlport_endpoint_tb/ctrlport_endpoint_tb.sv @@ -0,0 +1,502 @@ +// +// Copyright 2019 Ettus Research, A National Instruments Company +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: ctrlport_endpoint_tb +// + +`default_nettype none + + +module ctrlport_endpoint_tb; + + // Include macros and time declarations for use with PkgTestExec + `include "test_exec.svh" + + import PkgTestExec::*; + import PkgChdrUtils::*; + import PkgAxisCtrlBfm::*; + + // Parameters + localparam [9:0] THIS_PORTID = 10'h17; + localparam [15:0] THIS_EPID = 16'hDEAD; + + localparam integer NUM_XACT_PER_TEST = 300; + localparam integer FAST_STALL_PROB = 0; + localparam integer SLOW_STALL_PROB = 50; + localparam bit VERBOSE = 0; + + // Clock and Reset Definition + bit rfnoc_ctrl_clk, rfnoc_ctrl_rst; + bit ctrlport_clk, ctrlport_rst; + + sim_clock_gen #(6.0) rfnoc_ctrl_clk_gen (rfnoc_ctrl_clk, rfnoc_ctrl_rst); // 166.6 MHz + sim_clock_gen #(20.0) ctrlport_clk_gen (ctrlport_clk, ctrlport_rst); // 50 MHz + + // ---------------------------------------- + // Instantiate DUT + // ---------------------------------------- + AxiStreamIf #(32) m_ctrl (rfnoc_ctrl_clk, rfnoc_ctrl_rst); + AxiStreamIf #(32) s_ctrl (rfnoc_ctrl_clk, rfnoc_ctrl_rst); + AxisCtrlBfm axis_ctrl_bfm; + + wire [31:0] axis_mst_tdata, axis_slv_tdata , axis_req_tdata , axis_resp_tdata ; + wire axis_mst_tlast, axis_slv_tlast , axis_req_tlast , axis_resp_tlast ; + wire axis_mst_tvalid, axis_slv_tvalid, axis_req_tvalid, axis_resp_tvalid; + wire axis_mst_tready, axis_slv_tready, axis_req_tready, axis_resp_tready; + + wire cp_slv_req_wr; + wire cp_slv_req_rd; + wire [19:0] cp_slv_req_addr; + wire [31:0] cp_slv_req_data; + wire [3:0] cp_slv_req_byte_en; + wire cp_slv_req_has_time; + wire [63:0] cp_slv_req_time; + reg cp_slv_resp_ack; + reg [1:0] cp_slv_resp_status; + reg [31:0] cp_slv_resp_data; + + logic cp_mst_req_wr; + logic cp_mst_req_rd; + logic [19:0] cp_mst_req_addr; + logic [9:0] cp_mst_req_portid; + logic [15:0] cp_mst_req_rem_epid; + logic [9:0] cp_mst_req_rem_portid; + logic [31:0] cp_mst_req_data; + logic [3:0] cp_mst_req_byte_en; + logic cp_mst_req_has_time; + logic [63:0] cp_mst_req_time; + wire cp_mst_resp_ack; + wire [1:0] cp_mst_resp_status; + wire [31:0] cp_mst_resp_data; + + ctrlport_endpoint #( + .THIS_PORTID (THIS_PORTID), + .SYNC_CLKS (0), + .AXIS_CTRL_MST_EN (1), + .AXIS_CTRL_SLV_EN (1), + .SLAVE_FIFO_SIZE (5) + ) dut ( + .rfnoc_ctrl_clk (rfnoc_ctrl_clk ), + .rfnoc_ctrl_rst (rfnoc_ctrl_rst ), + .ctrlport_clk (ctrlport_clk ), + .ctrlport_rst (ctrlport_rst ), + .s_rfnoc_ctrl_tdata (axis_mst_tdata ), + .s_rfnoc_ctrl_tlast (axis_mst_tlast ), + .s_rfnoc_ctrl_tvalid (axis_mst_tvalid ), + .s_rfnoc_ctrl_tready (axis_mst_tready ), + .m_rfnoc_ctrl_tdata (axis_slv_tdata ), + .m_rfnoc_ctrl_tlast (axis_slv_tlast ), + .m_rfnoc_ctrl_tvalid (axis_slv_tvalid ), + .m_rfnoc_ctrl_tready (axis_slv_tready ), + .m_ctrlport_req_wr (cp_slv_req_wr ), + .m_ctrlport_req_rd (cp_slv_req_rd ), + .m_ctrlport_req_addr (cp_slv_req_addr ), + .m_ctrlport_req_data (cp_slv_req_data ), + .m_ctrlport_req_byte_en (cp_slv_req_byte_en ), + .m_ctrlport_req_has_time (cp_slv_req_has_time ), + .m_ctrlport_req_time (cp_slv_req_time ), + .m_ctrlport_resp_ack (cp_slv_resp_ack ), + .m_ctrlport_resp_status (cp_slv_resp_status ), + .m_ctrlport_resp_data (cp_slv_resp_data ), + .s_ctrlport_req_wr (cp_mst_req_wr ), + .s_ctrlport_req_rd (cp_mst_req_rd ), + .s_ctrlport_req_addr (cp_mst_req_addr ), + .s_ctrlport_req_portid (cp_mst_req_portid ), + .s_ctrlport_req_rem_epid (cp_mst_req_rem_epid ), + .s_ctrlport_req_rem_portid(cp_mst_req_rem_portid), + .s_ctrlport_req_data (cp_mst_req_data ), + .s_ctrlport_req_byte_en (cp_mst_req_byte_en ), + .s_ctrlport_req_has_time (cp_mst_req_has_time ), + .s_ctrlport_req_time (cp_mst_req_time ), + .s_ctrlport_resp_ack (cp_mst_resp_ack ), + .s_ctrlport_resp_status (cp_mst_resp_status ), + .s_ctrlport_resp_data (cp_mst_resp_data ) + ); + + // ---------------------------------------- + // Test Helpers + // ---------------------------------------- + + // Add a MUX and DEMUX on the ctrlport logic to loop responses + // back into the endpoint and to allow external access from the + // master and slave BFM. + axi_mux #( + .WIDTH(32), .SIZE(2), .PRIO(0), .PRE_FIFO_SIZE(0), .POST_FIFO_SIZE(0) + ) mux_i ( + .clk(rfnoc_ctrl_clk), .reset(rfnoc_ctrl_rst), .clear(1'b0), + .i_tdata ({m_ctrl.slave.tdata , axis_resp_tdata }), + .i_tlast ({m_ctrl.slave.tlast , axis_resp_tlast }), + .i_tvalid({m_ctrl.slave.tvalid, axis_resp_tvalid}), + .i_tready({m_ctrl.slave.tready, axis_resp_tready}), + .o_tdata (axis_mst_tdata ), + .o_tlast (axis_mst_tlast ), + .o_tvalid(axis_mst_tvalid), + .o_tready(axis_mst_tready) + ); + + wire [31:0] in_hdr; + axi_demux #( + .WIDTH(32), .SIZE(2), .PRE_FIFO_SIZE(0), .POST_FIFO_SIZE(0) + ) demux_i ( + .clk(rfnoc_ctrl_clk), .reset(rfnoc_ctrl_rst), .clear(1'b0), + .header(in_hdr), .dest(in_hdr[31]), + .i_tdata (axis_slv_tdata ), + .i_tlast (axis_slv_tlast ), + .i_tvalid(axis_slv_tvalid), + .i_tready(axis_slv_tready), + .o_tdata ({s_ctrl.master.tdata , axis_req_tdata }), + .o_tlast ({s_ctrl.master.tlast , axis_req_tlast }), + .o_tvalid({s_ctrl.master.tvalid, axis_req_tvalid}), + .o_tready({s_ctrl.master.tready, axis_req_tready}) + ); + + // -------------------------- + // [Dummy Control Port Slave] + // Slave Model: + // - Respond in 1 clock cycle + // - Status = Upper 2 bits of the address + // - Response Data = 0xFEED and Negated bottom 16 bits of addr + always @(posedge ctrlport_clk) begin + if (ctrlport_rst) begin + cp_slv_resp_ack <= 1'b0; + end else begin + cp_slv_resp_ack <= cp_slv_req_wr | cp_slv_req_rd; + cp_slv_resp_status <= cp_slv_req_addr[19:18]; + cp_slv_resp_data <= {16'hFEED, ~cp_slv_req_addr[15:0]}; + end + end + // -------------------------- + + // ---------------------------- + // [Dummy AXIS-Ctrl Port Slave] + // Slave Model: + // - Response = Request but with the ACK bit set + // - Status = Upper 2 bits of the address + // - Response Data = Request Data + reg [4:0] line_num = 5'd0; + reg pkt_has_time = 1'b0; + wire pkt_hdr_line = (line_num == 5'd0); + wire pkt_op_line = pkt_has_time ? (line_num == 5'd4) : (line_num == 5'd2); + always @(posedge rfnoc_ctrl_clk) begin + if (rfnoc_ctrl_rst) begin + line_num <= 5'd0; + pkt_has_time <= 1'b0; + end else if (axis_req_tvalid & axis_resp_tready) begin + if (pkt_hdr_line) + pkt_has_time <= axis_req_tdata[30]; + line_num <= axis_req_tlast ? 5'd0 : (line_num + 1); + end + end + assign axis_resp_tdata = + pkt_hdr_line ? {1'b1, axis_req_tdata[30:0]} : ( + pkt_op_line ? {axis_req_tdata[19:18], axis_req_tdata[29:0]} : + axis_req_tdata); + assign axis_resp_tlast = axis_req_tlast; + assign axis_resp_tvalid = axis_req_tvalid; + assign axis_req_tready = axis_resp_tready; + // ---------------------------- + + // Task to send a ctrlport request and receive a response + task ctrlport_transact( + input wr, + input rd, + input [19:0] addr, + input [9:0] portid, + input [15:0] rem_epid, + input [9:0] rem_portid, + input [31:0] data, + input [3:0] byte_en, + input has_time, + input [63:0] timestamp, + output [1:0] resp_status, + output [31:0] resp_data + ); + if (rd | wr) begin + cp_mst_req_wr <= wr; + cp_mst_req_rd <= rd; + cp_mst_req_addr <= addr; + cp_mst_req_portid <= portid; + cp_mst_req_rem_epid <= rem_epid; + cp_mst_req_rem_portid <= rem_portid; + cp_mst_req_data <= data; + cp_mst_req_byte_en <= byte_en; + cp_mst_req_has_time <= has_time; + cp_mst_req_time <= timestamp; + @(posedge ctrlport_clk); + cp_mst_req_wr <= 0; + cp_mst_req_rd <= 0; + while (~cp_mst_resp_ack) @(posedge ctrlport_clk); + resp_status = cp_mst_resp_status; + resp_data = cp_mst_resp_data; + + // Validate contents + if (VERBOSE) begin + $display("%s(addr=%0x, data=%0x, portid=%0x, has_time=%0b) = %0x (Status = %0d)", + (rd&wr)?"WRRD":(rd?"RD":"WR"), addr, data, portid, has_time, resp_data, resp_status); + end + `ASSERT_ERROR(cp_mst_resp_status == addr[19:18], + "Received Ctrlport response had the wrong status"); + `ASSERT_ERROR(cp_mst_resp_data == data, + "Received Ctrlport response had the wrong data"); + end + endtask + + // Task to send a AxisCtrl request and receive a response + logic [5:0] cached_seq_num = 0; + task axis_ctrl_transact( + input [3:0] opcode, + input [19:0] addr, + input [9:0] portid, + input [15:0] rem_epid, + input [9:0] rem_portid, + input [31:0] data[$], + input [3:0] byte_en, + input has_time, + input [63:0] timestamp, + output [1:0] resp_status, + output [31:0] resp_data + ); + automatic AxisCtrlPacket tx_pkt, rx_pkt = null, exp_pkt = null; + automatic axis_ctrl_header_t header; + automatic ctrl_op_word_t op_word; + automatic ctrl_status_t exp_status; + automatic ctrl_word_t exp_data0; + + // Opcode specific logic + case (ctrl_opcode_t'(opcode)) + CTRL_OP_SLEEP: begin + // data[0] = cycles of sleep so limit its value + if (data.size() > 0) data[0][31:5] = 0; + exp_status = CTRL_STS_OKAY; + exp_data0 = data[0]; + end + CTRL_OP_WRITE_READ: begin + exp_status = ctrl_status_t'(addr[19:18]); + exp_data0 = {16'hFEED, ~addr[15:0]}; + end + CTRL_OP_WRITE: begin + exp_status = ctrl_status_t'(addr[19:18]); + exp_data0 = data[0]; + end + CTRL_OP_READ: begin + exp_status = ctrl_status_t'(addr[19:18]); + exp_data0 = {16'hFEED, ~addr[15:0]}; + end + default: begin + exp_status = CTRL_STS_CMDERR; + exp_data0 = data[0]; + end + endcase + + // Build TX packet + tx_pkt = new(); + header = '{ + default : '0, + rem_dst_port : rem_portid, + rem_dst_epid : rem_epid, + is_ack : 1'b0, + has_time : has_time, + seq_num : cached_seq_num, + num_data : data.size(), + src_port : THIS_PORTID, + dst_port : portid + }; + op_word = '{ + default : '0, + op_code : ctrl_opcode_t'(opcode), + byte_enable : byte_en, + address : addr + }; + tx_pkt.write_ctrl(header, op_word, data, timestamp); + + // Build expected packet (NULL if data vector is empty) + if (data.size() > 0) begin + exp_pkt = tx_pkt.copy(); + exp_pkt.header.is_ack = 1'b1; + exp_pkt.op_word.status = exp_status; + exp_pkt.data[0] = exp_data0; + end + + if (VERBOSE) $display("*******************"); + fork + // Send the packet + begin + axis_ctrl_bfm.put_ctrl(tx_pkt.copy()); + if (VERBOSE) begin $display("[TRANSMITTED]"); tx_pkt.print(); end + end + // Wait for response only if we are expecting one + if (exp_pkt != null) begin + axis_ctrl_bfm.get_ctrl(rx_pkt); + if (VERBOSE) begin $display("[RECEIVED]"); rx_pkt.print(); end + end + join + cached_seq_num = cached_seq_num + 1; + + // Validate contents + if (exp_pkt != null) begin + if (VERBOSE) begin $display("[EXPECTED]"); exp_pkt.print(); end + `ASSERT_ERROR(exp_pkt.equal(rx_pkt), + "Received AXIS-Ctrl packet was incorrect"); + end + endtask + + // ---------------------------------------- + // Test Process + // ---------------------------------------- + initial begin + // Shared Variables + // ---------------------------------------- + timeout_t timeout; + string tc_label; + logic [31:0] data_vtr[$]; + logic [1:0] resp_status; + logic [31:0] resp_data; + + // Initialize + // ---------------------------------------- + test.start_tb("ctrlport_endpoint_tb"); + + // Start the BFMs + axis_ctrl_bfm = new(m_ctrl, s_ctrl); + axis_ctrl_bfm.run(); + + // Reset + // ---------------------------------------- + rfnoc_ctrl_clk_gen.reset(); + ctrlport_clk_gen.reset(); + + test.start_test("Wait for reset"); + test.start_timeout(timeout, 1us, "Waiting for reset"); + while (rfnoc_ctrl_rst) @(posedge rfnoc_ctrl_clk); + while (ctrlport_rst) @(posedge ctrlport_clk); + test.end_timeout(timeout); + `ASSERT_ERROR(!ctrlport_rst && !rfnoc_ctrl_rst, "Reset did not deassert"); + test.end_test(); + + // AXIS-Ctrl Slave Test + // ---------------------------------------- + // Send AXIS-Ctrl packets to the DUT and expect AXIS-Ctrl + // responses. There is a ctrlport slave implemented above + for (int cfg = 0; cfg < 4; cfg++) begin + automatic logic mst_cfg = cfg[0]; + automatic logic slv_cfg = cfg[1]; + $sformat(tc_label, + "AXIS-Ctrl Slave (%s Master, %s Slave)", + (mst_cfg?"Slow":"Fast"), (slv_cfg?"Slow":"Fast")); + test.start_test(tc_label); + begin + // Set bus stall probabilities based on configuration + axis_ctrl_bfm.set_master_stall_prob(mst_cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + axis_ctrl_bfm.set_slave_stall_prob(slv_cfg?SLOW_STALL_PROB:FAST_STALL_PROB); + // Test multiple transactions + for (int n = 0; n < NUM_XACT_PER_TEST; n++) begin + // Generate random data for the payload + // It is illegal in the protocol to have a zero + // data length but we test it here to ensure no lockups + data_vtr.delete(); + for (int i = 0; i < $urandom_range(15); i++) + data_vtr[i] = $urandom(); + // Perform transaction + test.start_timeout(timeout, 10us, "Waiting for AXIS-Ctrl transaction"); + axis_ctrl_transact( + $urandom_range(5), // opcode + $urandom(), // addr + THIS_PORTID, // portid + $urandom(), $urandom(), // rem_epid, rem_portid + data_vtr, + $urandom_range(15), // byte_en + $urandom_range(1), // has_time + {$urandom(), $urandom()}, // timestamp + resp_status, + resp_data + ); + test.end_timeout(timeout); + end + end + test.end_test(); + end + + // AXIS-Ctrl Master Test + // ---------------------------------------- + // Send Ctrlport packets to the DUT and expect Ctrlport + // responses. There is a AXIS-Ctrl slave implemented above + test.start_test("AXIS-Ctrl Master"); + begin + // Test multiple transactions + for (int n = 0; n < NUM_XACT_PER_TEST * 4; n++) begin + test.start_timeout(timeout, 10us, "Waiting for Ctrlport transaction"); + ctrlport_transact( + $urandom_range(1), $urandom_range(1), // wr and rd + $urandom(), // addr + THIS_PORTID, // portid + $urandom(), $urandom(), // rem_epid, rem_portid + $urandom(), // data + $urandom_range(15), // byte_en + $urandom_range(1), // has_time + {$urandom(), $urandom()}, // timestamp + resp_status, + resp_data + ); + + test.end_timeout(timeout); + end + end + test.end_test(); + + // AXIS-Ctrl Master+Slave Test + // ---------------------------------------- + test.start_test("AXIS-Ctrl Master + Slave Simultaneously"); + begin + axis_ctrl_bfm.set_master_stall_prob(FAST_STALL_PROB); + axis_ctrl_bfm.set_slave_stall_prob(FAST_STALL_PROB); + test.start_timeout(timeout, 10us * NUM_XACT_PER_TEST, "Waiting for test case"); + fork + for (int n = 0; n < NUM_XACT_PER_TEST; n++) begin + // Generate random data for the payload + // It is illegal in the protocol to have a zero + // data length but we test it here to ensure no lockups + data_vtr.delete(); + for (int i = 0; i < $urandom_range(15); i++) + data_vtr[i] = $urandom(); + // Perform transaction + axis_ctrl_transact( + $urandom_range(5), // opcode + $urandom(), // addr + THIS_PORTID, // portid + $urandom(), $urandom(), // rem_epid, rem_portid + data_vtr, + $urandom_range(15), // byte_en + $urandom_range(1), // has_time + {$urandom(), $urandom()}, // timestamp + resp_status, + resp_data + ); + end + for (int n = 0; n < NUM_XACT_PER_TEST; n++) begin + ctrlport_transact( + $urandom_range(1), $urandom_range(1), // wr and rd + $urandom(), // addr + THIS_PORTID, // portid + $urandom(), $urandom(), // rem_epid, rem_portid + $urandom(), // data + $urandom_range(15), // byte_en + $urandom_range(1), // has_time + {$urandom(), $urandom()}, // timestamp + resp_status, + resp_data + ); + end + join + test.end_timeout(timeout); + end + test.end_test(); + + // Finish Up + // ---------------------------------------- + // Display final statistics and results + test.end_tb(); + end + +endmodule |