From eed4988cc266a63370a4332351d02fadedde3a3b Mon Sep 17 00:00:00 2001 From: Andrew Moch Date: Thu, 25 Jun 2020 21:49:00 +0100 Subject: fpga: lib: Add width agnostic version of Ethernet Interface The rnfoc/xport section is refactored in System Verilog to allow the following improvements (1) CPU_W - Sets the size of the c2e and e2c pipes. This can be run at a different clock rate than the main ethernet pipe (2) CHDR_W - Sets the size of the v2e and e2v pipes. This can be run at a different clock rate than the main ethernet pipe (3) ENET_W - Sets the size of the eth_tx and eth_rx pipes. eth_interface_tb runs traffic from e2c,e2v,v2e,c2e simultaneously against the original xport_sv implementation, and against the new implementation with widths of 64/128/512. A chdr_management node info request queries the port info of the node0 in the eth_interface. eth_ifc_synth_test.sv can be compiled with the make xsim target to test out the size of various configurations. --- fpga/usrp3/lib/rfnoc/xport_sv/Makefile.srcs | 16 + .../usrp3/lib/rfnoc/xport_sv/chdr_xport_adapter.sv | 529 ++++++++++ fpga/usrp3/lib/rfnoc/xport_sv/eth_constants.vh | 76 ++ .../lib/rfnoc/xport_sv/eth_interface_tb/.gitignore | 3 + .../lib/rfnoc/xport_sv/eth_interface_tb/Makefile | 81 ++ .../xport_sv/eth_interface_tb/eth_ifc_all_tb.sv | 24 + .../eth_interface_tb/eth_ifc_synth_test.sv | 161 +++ .../rfnoc/xport_sv/eth_interface_tb/eth_ifc_tb.sv | 1084 ++++++++++++++++++++ fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_add_udp.sv | 200 ++++ .../lib/rfnoc/xport_sv/eth_ipv4_chdr_adapter.sv | 473 +++++++++ .../lib/rfnoc/xport_sv/eth_ipv4_chdr_dispatch.sv | 435 ++++++++ .../usrp3/lib/rfnoc/xport_sv/eth_ipv4_interface.sv | 228 ++++ fpga/usrp3/lib/rfnoc/xport_sv/eth_regs.vh | 29 + fpga/usrp3/tools/utils/testbenches.excludes | 1 + 14 files changed, 3340 insertions(+) create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/Makefile.srcs create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/chdr_xport_adapter.sv create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_constants.vh create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/.gitignore create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/Makefile create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_all_tb.sv create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_synth_test.sv create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_tb.sv create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_add_udp.sv create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_chdr_adapter.sv create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_chdr_dispatch.sv create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_interface.sv create mode 100644 fpga/usrp3/lib/rfnoc/xport_sv/eth_regs.vh diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/Makefile.srcs b/fpga/usrp3/lib/rfnoc/xport_sv/Makefile.srcs new file mode 100644 index 000000000..397a86326 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/Makefile.srcs @@ -0,0 +1,16 @@ +# +# Copyright 2020 Ettus Research, a National Instruments Brand +# +# SPDX-License-Identifier: LGPL-3.0-or-later +# + +################################################## +# RFNoC Utility Sources +################################################## +RFNOC_XPORT_SV_SRCS = $(abspath $(addprefix $(BASE_DIR)/../lib/rfnoc/xport_sv/, \ +chdr_xport_adapter.sv \ +eth_ipv4_add_udp.sv \ +eth_ipv4_chdr_adapter.sv \ +eth_ipv4_chdr_dispatch.sv \ +eth_ipv4_interface.sv \ +)) diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/chdr_xport_adapter.sv b/fpga/usrp3/lib/rfnoc/xport_sv/chdr_xport_adapter.sv new file mode 100644 index 000000000..b898c1fe2 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/chdr_xport_adapter.sv @@ -0,0 +1,529 @@ +// +// Copyright 2020 Ettus Research, A National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: chdr_xport_adapter (Ethernet IPV4) +// +// Description: An Xport transport adapter module that does the following: +// - Exposes a configuration port for mgmt packets to configure the node. +// (chdr_mgmt_pkt_handler) +// - Implements a return-address map for packets with metadata other than +// the CHDR. Additional metadata can be passed as a tuser to this module +// which will store it in a map indexed by the SrcEPID in a management +// packet. For all returning packets, the metadata will be looked up in +// the map and attached as the outgoing tuser. (kv_map) +// - Implements a loopback path for node-info discovery (axi_switch/axi_mux) +// - Strip UDP headers and extract mac/ip/udp src addresses +// - Add UDP header for outgoing ethernet traffic +// +// Parameters: +// - PROTOVER: RFNoC protocol version {8'd, 8'd} +// - TBL_SIZE: Log2 of the depth of the routing table +// - NODE_SUBTYPE: The node subtype to return for a node-info discovery +// - NODE_INST: The node type to return for a node-info discovery +// - ALLOW_DISC: Controls if the external transport network should be +// discoverable by management packets from RFNoC side. +// +// Signals: +// - device_id : The ID of the device that has instantiated this module +// - my_* : MAC address, IP address, and UDP port that responds/accepts CHDR traffic +// - eth_rx : The input CHDR stream from the transport +// - eth_tx : The output CHDR stream to transport +// - v2e : The input CHDR stream from the rfnoc infrastructure +// - e2v : The output CHDR stream to the rfnoc infrastructure +// + +`include "../xport/rfnoc_xport_types.vh" + +module chdr_xport_adapter #( + int PREAMBLE_BYTES = 6, + int MAX_PACKET_BYTES = 2**16, + logic [15:0] PROTOVER = {8'd1, 8'd0}, + int TBL_SIZE = 6, + logic [7:0] NODE_SUBTYPE = NODE_SUBTYPE_XPORT_IPV4_CHDR64, + int NODE_INST = 0, + bit ALLOW_DISC = 1 +)( + // Device info + input logic [15:0] device_id, + // Device addresses + input logic [47:0] my_mac, + input logic [31:0] my_ip, + input logic [15:0] my_udp_chdr_port, + + // Ethernet + AxiStreamIf.slave eth_rx, // tUser={*not used*} + AxiStreamIf.master eth_tx, // tUser={1'b0,trailing bytes} + + // CHDR + AxiStreamIf.slave v2e, // tUser={*not used*} + AxiStreamIf.master e2v // tUser={*not used*} +); + + //used to store {udp, ipv4, mac} + localparam USER_META_W = 96; + localparam ENET_USER_W = $clog2(eth_rx.DATA_WIDTH/8)+1; + // --------------------------------------------------- + // RFNoC Includes + // --------------------------------------------------- + `include "../core/rfnoc_chdr_utils.vh" + `include "../core/rfnoc_chdr_internal_utils.vh" + `include "eth_constants.vh" + `include "../../axi4s_sv/axi4s.vh" + + // tUser={None} + AxiStreamPacketIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.TKEEP(0),.TUSER(0), + .MAX_PACKET_BYTES(MAX_PACKET_BYTES)) + ru1(eth_rx.clk,eth_rx.rst);// Packet handler input + // tUser={None} + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.TKEEP(0),.TUSER(0), + .MAX_PACKET_BYTES(MAX_PACKET_BYTES)) + ru2(eth_rx.clk,eth_rx.rst);// Packet handler input + // tUser={udp_src_port,ipv4_src_addr,eth_src_addr} + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.USER_WIDTH(USER_META_W),.TKEEP(0)) + ru3(eth_rx.clk,eth_rx.rst);// Packet handler input + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.USER_WIDTH(USER_META_W),.TKEEP(0)) + ru4(eth_rx.clk,eth_rx.rst);// Packet handler input + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.USER_WIDTH(USER_META_W),.TKEEP(0)) + ph(eth_rx.clk,eth_rx.rst);// Packet handler input + // tUser={udp_src_port,ipv4_src_addr,eth_src_addr} + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.USER_WIDTH(USER_META_W),.TKEEP(0)) + e2d(eth_rx.clk,eth_rx.rst);// Eth => Demux + logic [1:0] e2d_tid; + // tUser={udp_src_port,ipv4_src_addr,eth_src_addr} + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.USER_WIDTH(USER_META_W),.TKEEP(0)) + e2e(eth_rx.clk,eth_rx.rst);// Eth => Eth (loopback) + // tUser={udp_dst_port, ipv4_dst_addr, eth_dst_addr} + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.USER_WIDTH(USER_META_W),.TKEEP(0)) + m2e(eth_rx.clk,eth_rx.rst);// Mux => Eth + logic m2e_tdest; // 1: Return to src, 0: CHDR input + + // --------------------------------------------------- + // Strip UDP and grab {udp_src_port_old, ipv4_src_addr_old, eth_src_addr_old} + // --------------------------------------------------- + + always_comb begin : assign_ru1 + `AXI4S_ASSIGN(ru1,eth_rx) + end + + // Cached fields + logic [47:0] eth_src_addr_new, eth_src_addr_old; + logic [31:0] ipv4_src_addr_new, ipv4_src_addr_old; + logic [15:0] udp_src_port_new, udp_src_port_old; + + // save the fields + always_ff @(posedge eth_rx.clk) begin : field_ff + if (eth_rx.rst) begin + eth_src_addr_old <= '0; + ipv4_src_addr_old <= '0; + udp_src_port_old <= '0; + end else begin + eth_src_addr_old <= eth_src_addr_new; + ipv4_src_addr_old <= ipv4_src_addr_new; + udp_src_port_old <= udp_src_port_new; + end + end + + // get the fields - don't use assign. assign will not activate with changes to eth_rx. + always_comb begin : get_fields + eth_src_addr_new = ru1.get_packet_field48(eth_src_addr_old,SRC_MAC_BYTE,.NETWORK_ORDER(1)); + ipv4_src_addr_new = ru1.get_packet_field32(ipv4_src_addr_old,SRC_IP_BYTE,.NETWORK_ORDER(1)); + udp_src_port_new = ru1.get_packet_field16(udp_src_port_old,SRC_PORT_BYTE,.NETWORK_ORDER(1)); + end + + // Strip the udp header + axi4s_remove_bytes #(.REM_START(0),.REM_END(UDP_END) + ) strip_udp ( + .i(ru1), .o(ru2) + ); + + // start driving the port information + always_comb begin : assign_ru3 + `AXI4S_ASSIGN(ru3,ru2) + ru3.tuser = {udp_src_port_old, ipv4_src_addr_old, eth_src_addr_old}; + end + + + chdr_trim_payload #( + .CHDR_W(eth_rx.DATA_WIDTH), .USER_W(USER_META_W) + ) chdr_trim_i ( + .clk(eth_rx.clk), .rst(eth_rx.rst), + .s_axis_tdata(ru3.tdata), .s_axis_tuser(ru3.tuser), + .s_axis_tlast(ru3.tlast), .s_axis_tvalid(ru3.tvalid), .s_axis_tready(ru3.tready), + .m_axis_tdata(ru4.tdata), .m_axis_tuser(ru4.tuser), + .m_axis_tlast(ru4.tlast), .m_axis_tvalid(ru4.tvalid), .m_axis_tready(ru4.tready) + ); + + // Pay close attention to when ph.tuser swtiches versus when it is needed! + always_comb begin : assign_ph + `AXI4S_ASSIGN(ph,ru4) + end + + // --------------------------------------------------- + // Transport => DEMUX + // --------------------------------------------------- + logic op_stb; + logic [15:0] op_src_epid; + logic [USER_META_W-1:0] op_data; + logic lookup_stb, lookup_done_stb, lookup_result_match; + logic [15:0] lookup_epid; + logic [USER_META_W-1:0] lookup_result_value; + logic [47:0] node_info; + + always_comb node_info = chdr_mgmt_build_node_info( + { 10'h0, NODE_SUBTYPE}, + NODE_INST, NODE_TYPE_TRANSPORT, device_id); + + chdr_mgmt_pkt_handler #( + .PROTOVER(PROTOVER), .CHDR_W(eth_rx.DATA_WIDTH), .USER_W(USER_META_W), .MGMT_ONLY(0) + ) mgmt_ep_i ( + .clk(eth_rx.clk), .rst(eth_rx.rst), + .node_info(node_info), + //ph in + .s_axis_chdr_tdata(ph.tdata), .s_axis_chdr_tlast(ph.tlast), + .s_axis_chdr_tvalid(ph.tvalid), .s_axis_chdr_tready(ph.tready), + .s_axis_chdr_tuser(ph.tuser), + //e2d out + .m_axis_chdr_tdata(e2d.tdata), .m_axis_chdr_tlast(e2d.tlast), + .m_axis_chdr_tdest(/* unused */), .m_axis_chdr_tid(e2d_tid), + .m_axis_chdr_tvalid(e2d.tvalid), .m_axis_chdr_tready(e2d.tready), + //unused ctrlport + .ctrlport_req_wr (/* unused */), + .ctrlport_req_rd (/* unused */), + .ctrlport_req_addr (/* unused */), + .ctrlport_req_data (/* unused */), + .ctrlport_resp_ack (1'b0 /* unused */), + .ctrlport_resp_data (32'b0 /* unused */), + // kv_map lookups + .op_stb(op_stb), + .op_dst_epid(/* unused */), + .op_src_epid(op_src_epid), + .op_data(op_data) + ); + + // Key/Value map. + // Stores the destination address information for UDP + // -- storage is controlled from the chdr_managment_node + // -- lookup is done on each packet passing out + kv_map #( + .KEY_WIDTH(16), .VAL_WIDTH(USER_META_W), .SIZE(TBL_SIZE) + ) kv_map_i ( + .clk(eth_rx.clk), .reset(eth_rx.rst), + .insert_stb(op_stb), .insert_key(op_src_epid), .insert_val(op_data), + .insert_busy(/* Time between op_stb > Insertion time */), + .find_key_stb(lookup_stb), .find_key(lookup_epid), + .find_res_stb(lookup_done_stb), + .find_res_match(lookup_result_match), .find_res_val(lookup_result_value), + .count(/* unused */) + ); + + logic ph_hdr = 1'b1; + always_ff @(posedge eth_rx.clk) begin + if (eth_rx.rst) + ph_hdr <= 1'b1; + else if (ph.tvalid && ph.tready) + ph_hdr <= ph.tlast; + end + + // chdr_mgmt_pkt_handler does not buffer packets and has at least one cycle of delay. + // The tuser caching logic could be more robust. + always_ff @(posedge eth_rx.clk) begin + if (ph.tvalid && ph.tready && ph_hdr) + e2d.tuser <= ph.tuser; + end + + // --------------------------------------------------- + // Optional management filter + // --------------------------------------------------- + // tUser={*not used*} + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.TUSER(0),.TKEEP(0)) + f2m(eth_rx.clk,eth_rx.rst); + + if (ALLOW_DISC) begin : gen_no_mgmt_filter + // Allow all packets to pass through + always_comb begin + f2m.tdata = v2e.tdata; + f2m.tlast = v2e.tlast; + f2m.tvalid = v2e.tvalid; + v2e.tready = f2m.tready; + end + + end else begin : gen_mgmt_filter + // Disallow forwarding of management discovery packets from RFNoC to the + // transport interface for transports that don't support them. + //vhook_nowarn unused_* + logic [eth_rx.DATA_WIDTH-1:0] unused_tdata; + logic unused_tlast, unused_tvalid; + logic [eth_rx.DATA_WIDTH-1:0] s_header; + logic dispose_pkt; + + // We identify discovery packets by the fact that they are management + // packets and that they use the null EPID as the destination. + always_comb dispose_pkt = (chdr_get_pkt_type(s_header[63:0]) == CHDR_PKT_TYPE_MGMT) && + (chdr_get_dst_epid(s_header[63:0]) == NULL_EPID); + + + axi_demux #( + .WIDTH (eth_rx.DATA_WIDTH), + .SIZE (2), + .PRE_FIFO_SIZE (0), + .POST_FIFO_SIZE (1) + ) axi_demux_mgmt_filter_i ( + .clk (eth_rx.clk), + .reset (eth_rx.rst), + .clear (1'b0), + .header (s_header), + .dest (dispose_pkt), + .i_tdata (v2e.tdata), + .i_tlast (v2e.tlast), + .i_tvalid (v2e.tvalid), + .i_tready (v2e.tready), + .o_tdata ({unused_tdata, f2m.tdata}), + .o_tlast ({unused_tlast, f2m.tlast}), + .o_tvalid ({unused_tvalid, f2m.tvalid}), + .o_tready ({1'b1, f2m.tready}) + ); + end + + // --------------------------------------------------- + // MUX and DEMUX for return path + // --------------------------------------------------- + + logic [USER_META_W-1:0] unused_tuser; + + axis_switch #( + .DATA_W(eth_rx.DATA_WIDTH+USER_META_W), .DEST_W(1), .IN_PORTS(1), .OUT_PORTS(2), .PIPELINE(0) + ) rtn_demux_i ( + .clk(eth_rx.clk), .reset(eth_rx.rst), + .s_axis_tdata({e2d.tuser, e2d.tdata}), .s_axis_alloc(1'b0), + .s_axis_tdest(e2d_tid == CHDR_MGMT_RETURN_TO_SRC ? 2'b01 : 2'b00), + .s_axis_tlast(e2d.tlast), .s_axis_tvalid(e2d.tvalid), .s_axis_tready(e2d.tready), + .m_axis_tdata({e2e.tuser, e2e.tdata, unused_tuser, e2v.tdata}), + .m_axis_tdest(/* unused */), + .m_axis_tlast({e2e.tlast, e2v.tlast}), + .m_axis_tvalid({e2e.tvalid, e2v.tvalid}), + .m_axis_tready({e2e.tready, e2v.tready}) + ); + + axi_mux #( + .WIDTH(eth_rx.DATA_WIDTH+USER_META_W+1), .SIZE(2), .PRE_FIFO_SIZE(0), .POST_FIFO_SIZE(0) + ) rtn_mux_i ( + .clk(eth_rx.clk), .reset(eth_rx.rst), .clear(1'b0), + .i_tdata({1'b1, e2e.tuser, e2e.tdata, 1'b0, {USER_META_W{1'b0}}, f2m.tdata}), + .i_tlast({e2e.tlast, f2m.tlast}), + .i_tvalid({e2e.tvalid, f2m.tvalid}), .i_tready({e2e.tready, f2m.tready}), + .o_tdata({m2e_tdest, m2e.tuser, m2e.tdata}), .o_tlast(m2e.tlast), + .o_tvalid(m2e.tvalid), .o_tready(m2e.tready) + ); + + // --------------------------------------------------- + // MUX => Transport + // --------------------------------------------------- + + // In this section we must determine what value to put in tuser. If tdest is + // 1 then tuser is passed through unchanged. If tdest is 0 then the tuser + // value is looked up in the KV map using the EPID in the packet header. + // + // To do this we split the data (tdata, tlast) and the routing information + // (tdest, tuser, and the EPID) into two FIFOs. This allows us to perform a + // routing lookup and decide what to do while we continue to buffer data. + // + // With small packets, multiple routing lookups might be enqueued in the + // lookup_fifo, but we can only do one lookup at a time. Output logic + // controls release of packets from the data FIFO to ensure we only output + // one packet per lookup after the lookup is complete. + + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.TUSER(0),.TKEEP(0)) + data_fifo_o(eth_rx.clk,eth_rx.rst); + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.TUSER(0),.TKEEP(0)) + data_fifo_i(eth_rx.clk,eth_rx.rst); + + AxiStreamIf #(.DATA_WIDTH(1+USER_META_W+16),.TUSER(0),.TKEEP(0)) + lookup_fifo_o(eth_rx.clk,eth_rx.rst); + AxiStreamIf #(.DATA_WIDTH(1+USER_META_W+16),.TUSER(0),.TKEEP(0)) + lookup_fifo_i(eth_rx.clk,eth_rx.rst); + + logic lookup_fifo_tdest; + logic [USER_META_W-1:0] lookup_fifo_tuser; + logic [ 15:0] lookup_fifo_tepid; + logic non_lookup_done_stb; + logic data_fifo_o_hdr = 1'b1; + logic pass_packet; + logic [USER_META_W-1:0] result_tuser; + logic result_tuser_valid; + logic [USER_META_W-1:0] reg_o_tuser; + + // Track when the next m2e word contains is the start of a new packet + logic m2e_hdr = 1'b1; + always_ff @(posedge eth_rx.clk) begin : m2e_hdr_ff + if (eth_rx.rst) + m2e_hdr <= 1'b1; + else if (m2e.tvalid && m2e.tready) + m2e_hdr <= m2e.tlast; + end + + // We can only accept data from the mux when when both the data_fifo and + // lookup_fifo are ready. + always_comb data_fifo_i.tdata = m2e.tdata; + always_comb data_fifo_i.tlast = m2e.tlast; + always_comb data_fifo_i.tvalid = m2e.tvalid && m2e.tready; + always_comb m2e.tready = data_fifo_i.tready && lookup_fifo_i.tready; + + // The data_fifo only takes the packet data (tdata, tlast). We use an + // axi_fifo_short module for the data_fifo because it can tolerate tvalid + // going low before a transfer completes. + axi_fifo_short #( + .WIDTH (1+eth_rx.DATA_WIDTH) + ) data_fifo ( + .clk (eth_rx.clk), + .reset (eth_rx.rst), + .clear (1'b0), + .i_tdata ({data_fifo_i.tlast, data_fifo_i.tdata}), + .i_tvalid (data_fifo_i.tvalid), + .i_tready (data_fifo_i.tready), + .o_tdata ({data_fifo_o.tlast, data_fifo_o.tdata}), + .o_tvalid (data_fifo_o.tvalid), + .o_tready (data_fifo_o.tready), + .space (), + .occupied () + ); + + // The lookup FIFO only takes the header routing info (tdest, tuser, epid). + // We use axi_fifo_short since it can tolerate tvalid going low before a + // transfer completes. + + always_comb lookup_fifo_i.tdata = {m2e_tdest, m2e.tuser, chdr_get_dst_epid(m2e.tdata[63:0])}; + always_comb {lookup_fifo_tdest, lookup_fifo_tuser, lookup_fifo_tepid} = lookup_fifo_o.tdata; + always_comb lookup_fifo_i.tvalid = m2e.tvalid && m2e.tready && m2e_hdr; + + axi_fifo_short #( + .WIDTH (1+USER_META_W+16) + ) lookup_fifo ( + .clk (eth_rx.clk), + .reset (eth_rx.rst), + .clear (1'b0), + .i_tdata (lookup_fifo_i.tdata), + .i_tvalid (lookup_fifo_i.tvalid), + .i_tready (lookup_fifo_i.tready), + .o_tdata (lookup_fifo_o.tdata), + .o_tvalid (lookup_fifo_o.tvalid), + .o_tready (lookup_fifo_o.tready), + .space (), + .occupied () + ); + + // Keep track of when we are busy doing a lookup in the KV map. + logic lookup_busy = 1'b0; + always_ff @(posedge eth_rx.clk) begin : lookup_busy_ff + if (eth_rx.rst) + lookup_busy <= 1'b0; + else begin + if (lookup_stb) + lookup_busy <= 1'b1; + else if (lookup_done_stb) + lookup_busy <= 1'b0; + end + end + + // Determine if we can use the output of the lookup_fifo to do a KV map + // lookup. We only perform a KV map lookup if tdest is 0 and we can only do + // so if the KV map is free and the holding register for the tuser value is + // available. + always_comb lookup_epid = lookup_fifo_tepid; + always_comb lookup_stb = lookup_fifo_o.tvalid && !lookup_busy && + !lookup_fifo_tdest && !result_tuser_valid; + + // Determine if we can use the output of the lookup FIFO directly (no lookup + // is needed). We can only use it if we're not already doing a KV lookup and + // if the holding register for the tuser value is available. + always_comb non_lookup_done_stb = lookup_fifo_o.tvalid && !lookup_busy && + lookup_fifo_tdest && !result_tuser_valid; + + // Pop the routing info off of the lookup_fifo if we've started its lookup + always_comb lookup_fifo_o.tready = lookup_stb || non_lookup_done_stb; + + // Track when the next data_fifo_o word is the start of a new packet + always_ff @(posedge eth_rx.clk) begin : data_fifo_o_hdr_ff + if (eth_rx.rst) + data_fifo_o_hdr <= 1'b1; + else if (data_fifo_o.tvalid && data_fifo_o.tready && pass_packet) + data_fifo_o_hdr <= data_fifo_o.tlast; + end + + // Store the lookup result in a holding register. This can come from the KV + // map or the incoming tuser. + always_ff @(posedge eth_rx.clk) begin : result_tuser_ff + if (eth_rx.rst) begin + result_tuser <= {USER_META_W{1'bX}}; // Don't care + result_tuser_valid <= 1'b0; + end else begin + // The tuser holding register becomes available as soon as we start + // transmitting the corresponding packet. + if (data_fifo_o.tvalid && data_fifo_o.tready && data_fifo_o_hdr && pass_packet) begin + result_tuser_valid <= 1'b0; + end + + // Load the result of the lookup + if (lookup_done_stb) begin + result_tuser <= lookup_result_match ? lookup_result_value : {USER_META_W{1'b0}}; + result_tuser_valid <= 1'b1; + end else if (non_lookup_done_stb) begin + result_tuser <= lookup_fifo_tuser; + result_tuser_valid <= 1'b1; + end + end + end + + // Control when the packet from the data_fifo can be passed through. Put the + // tuser value into a register for the duration of the packet. + always_ff @(posedge eth_rx.clk) begin : pass_packet_ff + if (eth_rx.rst) begin + pass_packet <= 1'b0; + reg_o_tuser <= {USER_META_W{1'bX}}; // Don't care + end else begin + // We're done passing through a packet when tlast goes out + if (data_fifo_o.tvalid && data_fifo_o.tready && data_fifo_o.tlast && pass_packet) begin + pass_packet <= 1'b0; + end + + // We can pass the next packet through when we're at the start of a + // packet and we have the tuser value waiting in the holding register. + if (data_fifo_o_hdr && result_tuser_valid && !pass_packet) begin + reg_o_tuser <= result_tuser; + pass_packet <= 1'b1; + end + end + end + + // Device addresses + logic [15:0] au_udp_dst; + logic [31:0] au_ip_dst; + logic [47:0] au_mac_dst; + + AxiStreamIf #(.DATA_WIDTH(eth_rx.DATA_WIDTH),.TKEEP(0),.TUSER(0)) + au(eth_rx.clk,eth_rx.rst);// Add UDP input + always_comb begin + {au_udp_dst,au_ip_dst,au_mac_dst} = reg_o_tuser; + au.tdata = data_fifo_o.tdata; + au.tlast = data_fifo_o.tlast; + au.tvalid = data_fifo_o.tvalid & pass_packet; + data_fifo_o.tready = au.tready & pass_packet; + end + + // add the UDP header back on before sending to EthTx + eth_ipv4_add_udp #( + .PREAMBLE_BYTES(PREAMBLE_BYTES), + .MAX_PACKET_BYTES(MAX_PACKET_BYTES) + ) add_udp_i ( + .i(au), .o(eth_tx), + .mac_src(my_mac), + .ip_src(my_ip), + .udp_src(my_udp_chdr_port), + .mac_dst(au_mac_dst), + .ip_dst(au_ip_dst), + .udp_dst(au_udp_dst) + ); + + +endmodule : chdr_xport_adapter diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/eth_constants.vh b/fpga/usrp3/lib/rfnoc/xport_sv/eth_constants.vh new file mode 100644 index 000000000..371b46cf5 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/eth_constants.vh @@ -0,0 +1,76 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: eth_constants (Header File) +// +// Description: +// Constants for ethernet + +//--------------------------------------- +// Ethernet constants +//--------------------------------------- +localparam [47:0] ETH_ADDR_BCAST = {48{1'b1}}; +localparam [15:0] ETH_TYPE_IPV4 = 16'h0800; +localparam [7:0] IPV4_PROTO_UDP = 8'h11; +localparam [7:0] IPV4_LEN5 = 8'h45; + +// tUser conventions +localparam BYTES_MSB = ENET_USER_W-2; +localparam ERROR_BIT = ENET_USER_W-1; +localparam SOF_BIT = ENET_USER_W-1; + +//--------------------------------------- +// Ethernet byte positions +//--------------------------------------- +// Bytes 7-0------------------------- +// | DstMAC_HI (16) | Preamble (48) | +// ---------------------------------- +localparam PREAMBLE_BYTE = 0; +localparam PREAMBLE_END = 5; +localparam ETH_HDR_BYTE = 0; +localparam DST_MAC_BYTE = ETH_HDR_BYTE+0; +// Bytes 15-8------------------------- +// | SrcMAC_HI (32) | DstMAC_LO (32) | +// ----------------------------------- +localparam SRC_MAC_BYTE = ETH_HDR_BYTE+6; +// Bytes 23-16--------------------------------------- +// | IPv4_Line0 (32)| EthType (16) | SrcMAC_LO (16) | +// -------------------------------------------------- +localparam ETH_TYPE_BYTE = ETH_HDR_BYTE+12; +localparam ETH_PAYLOAD_BYTE = ETH_HDR_BYTE+14; +// Bytes 31-24-------------------------- +// | IPv4_Line2 (32) | IPv4_Line1 (32) | +// ------------------------------------- +localparam IPV4_HDR_BYTE = ETH_PAYLOAD_BYTE; +localparam IP_VERSION_BYTE = IPV4_HDR_BYTE+0; +localparam IP_DSCP_BYTE = IPV4_HDR_BYTE+1; +localparam IP_LENGTH_BYTE = IPV4_HDR_BYTE+2; +localparam IP_ID_BYTE = IPV4_HDR_BYTE+4; +localparam IP_FRAG_BYTE = IPV4_HDR_BYTE+6; +localparam IP_TTL_BYTE = IPV4_HDR_BYTE+8; +localparam PROTOCOL_BYTE = IPV4_HDR_BYTE+9; +localparam IP_CHECKSUM_BYTE = IPV4_HDR_BYTE+10; +// Bytes 39-32------------------------ +// | IPDstAddr (32) | IPSrcAddr (32) | +// ----------------------------------- +localparam SRC_IP_BYTE = IPV4_HDR_BYTE+12; +localparam DST_IP_BYTE = IPV4_HDR_BYTE+16; +localparam IPV4_PAYLOAD_BYTE = IPV4_HDR_BYTE+20; +// Bytes 48-40------------------------------------------------ +// | Chksum (16) | Length (16) | DstPort (16) | SrcPort (16) | +// ----------------------------------------------------------- +localparam UDP_HDR_BYTE = IPV4_PAYLOAD_BYTE; +localparam SRC_PORT_BYTE = UDP_HDR_BYTE+0; +localparam DST_PORT_BYTE = UDP_HDR_BYTE+2; +localparam UDP_LENGTH_BYTE = UDP_HDR_BYTE+4; +localparam UDP_CHECKSUM_BYTE = UDP_HDR_BYTE+6; +localparam UDP_END = UDP_CHECKSUM_BYTE+1; // last byte in the UDP header + +localparam MIN_PACKET_SIZE_BYTE = 63; +//--------------------------------------- +// CHDR_BYTE_POSITION +//--------------------------------------- +localparam CHDR_HDR_BYTE = 0; +localparam CHDR_LENGTH_BYTE = CHDR_HDR_BYTE+2; diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/.gitignore b/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/.gitignore new file mode 100644 index 000000000..0eb397000 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/.gitignore @@ -0,0 +1,3 @@ +wave.do +*.dcp +*.rpt diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/Makefile b/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/Makefile new file mode 100644 index 000000000..b2cc6908e --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/Makefile @@ -0,0 +1,81 @@ +# +# 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 +#------------------------------------------------- +SYNTH_DUT = eth_ifc_synth_test + +# Include makefiles and sources for the DUT and its dependencies +include $(BASE_DIR)/../lib/control/Makefile.srcs +include $(BASE_DIR)/../lib/axi/Makefile.srcs +include $(BASE_DIR)/../lib/axi4s_sv/Makefile.srcs +include $(BASE_DIR)/../lib/packet_proc/Makefile.srcs +include $(BASE_DIR)/../lib/xge_interface/Makefile.srcs +include $(BASE_DIR)/../lib/fifo/Makefile.srcs +include $(BASE_DIR)/../lib/rfnoc/utils/Makefile.srcs +include $(BASE_DIR)/../lib/rfnoc/xport/Makefile.srcs +include $(BASE_DIR)/../lib/rfnoc/xport_sv/Makefile.srcs +include $(BASE_DIR)/../lib/rfnoc/crossbar/Makefile.srcs +include $(BASE_DIR)/../lib/rfnoc/core/Makefile.srcs + +BUILD_IP_DIR = $(abspath $(BASE_DIR)/x300/build-ip/xc7k410tffg900-2) + +IP_SRC = \ +$(abspath $(BUILD_IP_DIR)/fifo_short_2clk/fifo_short_2clk_sim_netlist.v) +# Xilinx IP wants lots of libraries +MODELSIM_LIBS += secureip unimacro_ver unisims_ver xilinx_vip xpm +MODELSIM_ARGS += glbl -t 1fs + +DESIGN_SRCS = $(abspath \ +eth_ifc_synth_test.sv \ +$(FIFO_SRCS) \ +$(CONTROL_LIB_SRCS) \ +$(AXI_SRCS) \ +$(AXI4S_SV_SRCS) \ +$(XGE_INTERFACE_SRCS) \ +$(PACKET_PROC_SRCS) \ +$(RFNOC_UTIL_SRCS) \ +$(RFNOC_XPORT_SRCS) \ +$(RFNOC_XPORT_SV_SRCS) \ +$(RFNOC_XBAR_SRCS) \ +$(RFNOC_CORE_SRCS) \ +$(IP_SRC) \ +) + +#------------------------------------------------- +# Testbench Specific +#------------------------------------------------- +# Define only one toplevel module +TB_TOP_MODULE ?= eth_ifc_all_tb +#TB_TOP_MODULE ?= eth_ifc_synth_test +SIM_TOP = $(TB_TOP_MODULE) + +SIM_SRCS = \ +$(abspath eth_ifc_tb.sv) \ +$(abspath $(TB_TOP_MODULE).sv) + +# supressing the following worthless reminder. +#* Warning: M:/usrp4-hw/oss-repo/fpga/usrp3/lib/axi4s_sv/axi4s_remove_bytes.sv(228): (vlog-2583) [SVCHK] - +# Extra checking for conflicts with always_comb and always_latch variables is done at vopt time +SVLOG_ARGS = -suppress 2583 -keep_delta +VLOG_ARGS = -keep_delta + +#------------------------------------------------- +# 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/xport_sv/eth_interface_tb/eth_ifc_all_tb.sv b/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_all_tb.sv new file mode 100644 index 000000000..f849bc8af --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_all_tb.sv @@ -0,0 +1,24 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: eth_ifc_all_tb +// +// Description: Testbench for eth_ifc +// + +module eth_ifc_all_tb #( + /* no PARAM */ +)( + /* no IO */ +); + + eth_ifc_tb #(.TEST_NAME("ORIGINAL"),.ENET_W(64),.CPU_W(64),.CHDR_W(64),.SV_ETH_IFC(0)) ORIGINAL (); + + eth_ifc_tb #(.TEST_NAME("64B"),.ENET_W(64),.CPU_W(64),.CHDR_W(64)) ETH64 (); + eth_ifc_tb #(.TEST_NAME("512_64B"),.ENET_W(512),.CPU_W(64),.CHDR_W(64)) ETH512_CHDR64 (); + eth_ifc_tb #(.TEST_NAME("512_128B"),.ENET_W(512),.CPU_W(64),.CHDR_W(128)) ETH512_CHDR128 (); + eth_ifc_tb #(.TEST_NAME("512B"),.ENET_W(512),.CPU_W(64),.CHDR_W(512)) ETH512_CHDR512 (); + +endmodule diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_synth_test.sv b/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_synth_test.sv new file mode 100644 index 000000000..f37a23383 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_synth_test.sv @@ -0,0 +1,161 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// +// Module: eth_ifc_synth_test +// +// Description: Wrapper to test size of 64 bit version +// +// Parameters: -- get descriptions from eth_ipv4_interface +// PROTOVER - +// MTU - Max Packet Size +// NODE_INST - +// RT_TBL_SIZE - +// REG_AWIDTH - +// BASE - +// EWIDTH - Ethernet Width +// CWIDTH - CPU Width +// VWIDTH - CHDR Width + +module eth_ifc_synth_test #( + parameter [15:0] PROTOVER = {8'd1, 8'd0}, + parameter MTU = 10, + parameter NODE_INST = 0, + parameter RT_TBL_SIZE = 6, + parameter REG_AWIDTH = 14, + parameter BASE = 0, + parameter EWIDTH = 512, + parameter CWIDTH = 64, + parameter VWIDTH = 128 +) ( + input logic bus_clk, + input logic bus_rst, + input logic [15:0] device_id, + + // Register port: Write port (domain: clk) + input logic reg_wr_req, + input logic [REG_AWIDTH-1:0] reg_wr_addr, + input logic [31:0] reg_wr_data, + + // Register port: Read port (domain: clk) + input logic reg_rd_req, + input logic [REG_AWIDTH-1:0] reg_rd_addr, + output logic reg_rd_resp, + output logic [31:0] reg_rd_data, + + // Status ports (domain: clk) + output logic [47:0] my_mac, + output logic [31:0] my_ip, + output logic [15:0] my_udp_chdr_port, + + // Ethernet ports + output logic [EWIDTH-1:0] eth_tx_tdata, + output logic [$clog2(EWIDTH/8):0] eth_tx_tuser, + output logic eth_tx_tlast, + output logic eth_tx_tvalid, + input logic eth_tx_tready, + input logic [EWIDTH-1:0] eth_rx_tdata, + input logic [$clog2(EWIDTH/8):0] eth_rx_tuser, + input logic eth_rx_tlast, + input logic eth_rx_tvalid, + output logic eth_rx_tready, + + // CHDR router interface + output logic [VWIDTH-1:0] e2v_tdata, + output logic e2v_tlast, + output logic e2v_tvalid, + input logic e2v_tready, + input logic [VWIDTH-1:0] v2e_tdata, + input logic v2e_tlast, + input logic v2e_tvalid, + output logic v2e_tready, + + // CPU + output logic [CWIDTH-1:0] e2c_tdata, + output logic [$clog2(CWIDTH/8):0] e2c_tuser, + output logic e2c_tlast, + output logic e2c_tvalid, + input logic e2c_tready, + input logic [CWIDTH-1:0] c2e_tdata, + input logic [$clog2(CWIDTH/8):0] c2e_tuser, + input logic c2e_tlast, + input logic c2e_tvalid, + output logic c2e_tready + + ); + + localparam MAX_PACKET_BYTES = 2**16; + localparam ENET_W=EWIDTH; + localparam ENET_USER_W=$clog2(EWIDTH/8)+1; + localparam CHDR_W=VWIDTH; + localparam CHDR_USER_W=$clog2(VWIDTH/8)+1; + localparam CPU_W=CWIDTH; + localparam CPU_USER_W=$clog2(CWIDTH/8)+1; + + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0), + .MAX_PACKET_BYTES(MAX_PACKET_BYTES)) + eth_tx (bus_clk, bus_reset); + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0), + .MAX_PACKET_BYTES(MAX_PACKET_BYTES)) + eth_rx (bus_clk, bus_reset); + + AxiStreamIf #(.DATA_WIDTH(CHDR_W),.USER_WIDTH(CHDR_USER_W),.TKEEP(0),.TUSER(0)) + v2e (bus_clk, bus_reset); + AxiStreamIf #(.DATA_WIDTH(CHDR_W),.USER_WIDTH(CHDR_USER_W),.TKEEP(0),.TUSER(0)) + e2v (bus_clk, bus_reset); + + AxiStreamIf #(.DATA_WIDTH(CPU_W),.USER_WIDTH(CPU_USER_W),.TKEEP(0)) + c2e (bus_clk, bus_reset); + AxiStreamIf #(.DATA_WIDTH(CPU_W),.USER_WIDTH(CPU_USER_W),.TKEEP(0)) + e2c (bus_clk, bus_reset); + + assign eth_tx_tdata = eth_tx.tdata; + assign eth_tx_tuser = eth_tx.tuser; + assign eth_tx_tlast = eth_tx.tlast; + assign eth_tx_tvalid = eth_tx.tvalid; + assign eth_tx.tready = eth_tx_tready; + + assign eth_rx.tdata = eth_rx_tdata; + assign eth_rx.tuser = eth_rx_tuser; + assign eth_rx.tlast = eth_rx_tlast; + assign eth_rx.tvalid = eth_rx_tvalid; + assign eth_rx_tready = eth_rx.tready; + + assign e2v_tdata = e2v.tdata; + assign e2v_tlast = e2v.tlast; + assign e2v_tvalid = e2v.tvalid; + assign e2v.tready = e2v_tready; + + assign v2e.tdata = v2e_tdata; + assign v2e.tlast = v2e_tlast; + assign v2e.tvalid = v2e_tvalid; + assign v2e_tready = v2e.tready; + + assign e2c_tdata = e2c.tdata; + assign e2c_tuser = e2c.tuser; + assign e2c_tlast = e2c.tlast; + assign e2c_tvalid = e2c.tvalid; + assign e2c.tready = e2c_tready; + + assign c2e.tdata = c2e_tdata; + assign c2e.tuser = c2e_tuser; + assign c2e.tlast = c2e_tlast; + assign c2e.tvalid = c2e_tvalid; + assign c2e_tready = c2e.tready; + + eth_ipv4_interface #( + .PROTOVER(PROTOVER), .MTU(MTU), .NODE_INST(NODE_INST), + .REG_AWIDTH(REG_AWIDTH), .RT_TBL_SIZE(RT_TBL_SIZE), + .BASE(BASE), + .DROP_UNKNOWN_MAC(0), + .DROP_MIN_PACKET(0), + .PREAMBLE_BYTES(0), + .ADD_SOF(0), + .SYNC(0), + .ENET_W(EWIDTH),.CPU_W(CWIDTH),.CHDR_W(VWIDTH) + ) eth_interface ( + .* ); + +endmodule diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_tb.sv b/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_tb.sv new file mode 100644 index 000000000..6241da9a5 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/eth_interface_tb/eth_ifc_tb.sv @@ -0,0 +1,1084 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: eth_ifc_tb +// +// Description: Testbench for eth_interface +// + +module eth_ifc_tb #( + parameter TEST_NAME = "eth_ifc_tb", + parameter SV_ETH_IFC =1, + parameter ENET_W =64, + parameter CPU_W =64, + parameter CHDR_W =64 +)( + /* no IO */ +); + // Include macros and time declarations for use with PkgTestExec + `define TEST_EXEC_OBJ test + `include "test_exec.svh" + import PkgAxiStreamBfm::*; + import PkgTestExec::*; + import PkgChdrUtils::*; + import PkgChdrBfm::*; + import PkgEthernet::*; + + //--------------------------------------------------------------------------- + // Local Parameters + //--------------------------------------------------------------------------- + localparam ENET_USER_W = $clog2(ENET_W/8)+1; + localparam CPU_USER_W = $clog2(CPU_W/8)+1; + localparam CHDR_USER_W = $clog2(CHDR_W/8)+1; + localparam REG_AWIDTH = 14; + localparam [15:0] PROTOVER = {8'd1, 8'd0}; + localparam MTU = 10; + localparam NODE_INST = 0; + localparam RT_TBL_SIZE = 6; + localparam REG_BASE_ETH_SWITCH = 14'h1000; + localparam BASE = REG_BASE_ETH_SWITCH; + localparam DROP_MIN_PACKET = 0; + localparam SYNC = (ENET_W==512) ? 0:1; + localparam ETH_PERIOD = (ENET_W==512) ? 3.1:5.0; + // can set PREAMBLE_BYTES to 0 or 6 if SV_ETH_IFC, but otherwise + // it's hardcoded to 6. (0 is normal for 100G)(6 is normal for old Xge) + localparam PREAMBLE_BYTES = SV_ETH_IFC ? 0 : 6; + // Include for register offsets + `include "../eth_regs.vh" + // allows the DUT to push full words and tb does not check tuser/tkeep of packets it's transmitting + localparam IGNORE_EXTRA_DATA = 1; + + //--------------------------------------------------------------------------- + // Clocks + //--------------------------------------------------------------------------- + + bit clk; + bit reset; + + sim_clock_gen #(.PERIOD(5.0), .AUTOSTART(1)) + clk_gen (.clk(clk), .rst(reset)); + sim_clock_gen #(.PERIOD(ETH_PERIOD), .AUTOSTART(1)) + eth_clk_gen (.clk(eth_clk), .rst(eth_reset)); + + //--------------------------------------------------------------------------- + // Bus Functional Models + //--------------------------------------------------------------------------- + TestExec test = new(); + + localparam MAX_PACKET_BYTES = 2**16; + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0), + .MAX_PACKET_BYTES(MAX_PACKET_BYTES)) + eth_tx (eth_clk, eth_reset); + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0), + .MAX_PACKET_BYTES(MAX_PACKET_BYTES)) + eth_rx (eth_clk, eth_reset); + + AxiStreamIf #(.DATA_WIDTH(CHDR_W),.USER_WIDTH(CHDR_USER_W),.TKEEP(0),.TUSER(0)) + v2e (clk, reset); + AxiStreamIf #(.DATA_WIDTH(CHDR_W),.USER_WIDTH(CHDR_USER_W),.TKEEP(0),.TUSER(0)) + e2v (clk, reset); + + AxiStreamIf #(.DATA_WIDTH(CPU_W),.USER_WIDTH(CPU_USER_W),.TKEEP(0)) + c2e (clk, reset); + AxiStreamIf #(.DATA_WIDTH(CPU_W),.USER_WIDTH(CPU_USER_W),.TKEEP(0)) + e2c (clk, reset); + + // Bus functional model for a axi_stream controller + AxiStreamBfm #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0), + .MAX_PACKET_BYTES(MAX_PACKET_BYTES)) eth = + new(.master(eth_rx), .slave(eth_tx)); + AxiStreamBfm #(.DATA_WIDTH(CHDR_W),.USER_WIDTH(CHDR_USER_W),.TKEEP(0),.TUSER(0)) v = + new(.master(v2e), .slave(e2v)); + AxiStreamBfm #(.DATA_WIDTH(CPU_W),.USER_WIDTH(CPU_USER_W),.TKEEP(0)) cpu = + new(.master(c2e), .slave(e2c)); + + //---------------------------------------------------- + // Instantiate DUT + //---------------------------------------------------- + + reg reg_wr_req = 1'b0; + reg [REG_AWIDTH-1:0] reg_wr_addr = 0; + reg [31:0] reg_wr_data = 32'd0; + + reg reg_rd_req = 1'b0; + reg [REG_AWIDTH-1:0] reg_rd_addr = 0; + reg reg_rd_resp; + reg [31:0] reg_rd_data; + + reg [3:0] eth_tx_tuser = 4'd0; + reg [3:0] eth_rx_tuser = 4'd0; + reg [3:0] e2c_tuser = 4'd0; + reg [3:0] c2e_tuser = 4'd0; + + reg [15:0] device_id =16'd0; + + reg [47:0] my_mac; + reg [31:0] my_ip; + reg [15:0] my_udp_chdr_port; + + + if (SV_ETH_IFC) begin : gen_new_dut + + eth_ipv4_interface #( + .PREAMBLE_BYTES(PREAMBLE_BYTES), + .PROTOVER(PROTOVER), .MTU(MTU), .NODE_INST(NODE_INST), + .REG_AWIDTH(REG_AWIDTH), .RT_TBL_SIZE(RT_TBL_SIZE), + .BASE(BASE),.SYNC(SYNC), + .ENET_W(ENET_W),.CPU_W(CPU_W),.CHDR_W(CHDR_W) + ) eth_interface ( + .bus_clk(clk),.bus_rst(reset),.* + ); + end else begin : gen_old_dut + logic [63:0] eth_tx_tdata; + logic [3:0] eth_tx_tuser; + logic eth_tx_tlast; + logic eth_tx_tvalid; + logic eth_tx_tready; + + logic [63:0] eth_rx_tdata; + logic [3:0] eth_rx_tuser; + logic eth_rx_tlast; + logic eth_rx_tvalid; + logic eth_rx_tready; + + logic [63:0] e2v_tdata; + logic e2v_tlast; + logic e2v_tvalid; + logic e2v_tready; + + logic [63:0] v2e_tdata; + logic v2e_tlast; + logic v2e_tvalid; + logic v2e_tready; + + logic [63:0] e2c_tdata; + logic [3:0] e2c_tuser; + logic e2c_tlast; + logic e2c_tvalid; + logic e2c_tready; + + logic [63:0] c2e_tdata; + logic [3:0] c2e_tuser; + logic c2e_tlast; + logic c2e_tvalid; + logic c2e_tready; + + always_comb begin + eth_tx.tdata = eth_tx_tdata; + eth_tx.tuser = eth_tx_tuser; + eth_tx.tlast = eth_tx_tlast; + eth_tx.tvalid = eth_tx_tvalid; + eth_tx_tready = eth_tx.tready; + + eth_rx_tdata = eth_rx.tdata; + eth_rx_tuser = eth_rx.tuser; + eth_rx_tlast = eth_rx.tlast; + eth_rx_tvalid = eth_rx.tvalid; + eth_rx.tready = eth_rx_tready; + + e2v.tdata = e2v_tdata; + e2v.tlast = e2v_tlast; + e2v.tvalid = e2v_tvalid; + e2v_tready = e2v.tready; + + v2e_tdata = v2e.tdata; + v2e_tlast = v2e.tlast; + v2e_tvalid = v2e.tvalid; + v2e.tready = v2e_tready; + + e2c.tdata = e2c_tdata; + e2c.tuser = e2c_tuser; + e2c.tlast = e2c_tlast; + e2c.tvalid = e2c_tvalid; + e2c_tready = e2c.tready; + + c2e_tdata = c2e.tdata; + c2e_tuser = c2e.tuser; + c2e_tlast = c2e.tlast; + c2e_tvalid = c2e.tvalid; + c2e.tready = c2e_tready; + end + + eth_interface #( + .PROTOVER(PROTOVER), .MTU(MTU), .NODE_INST(NODE_INST), + .REG_AWIDTH(REG_AWIDTH), .RT_TBL_SIZE(RT_TBL_SIZE), + .BASE(BASE) + ) eth_interface ( + .*, + .my_udp_port (my_udp_chdr_port) + ); + end + always_comb begin + eth_tx.tkeep = '1; + end + + task automatic reg_wr ( + // Register port: Write port (domain: clk) + input int addr, + input int data); + begin + @(posedge clk); + reg_wr_req = 1'b1; + reg_wr_addr = addr; + reg_wr_data = data; + @(posedge clk); + reg_wr_req = 1'b0; + end + endtask + + task automatic reg_rd_check ( + // Register port: Write port (domain: clk) + input int addr, + input int data); + begin + @(posedge clk); + reg_rd_req = 1'b1; // drive at posedge + reg_rd_addr = addr; + @(negedge clk); // check at negedge + if (SV_ETH_IFC) begin + assert(reg_rd_resp==1'b0) else $error("resp set early"); + end else begin + // The original doesn't initialize reg_rd_resp so it comes back as X on first read + assert(reg_rd_resp==1'b0 || $isunknown(reg_rd_resp)) else $error("resp set early"); + end + @(posedge clk); + reg_rd_req = 1'b0; + @(negedge clk); + assert(data==reg_rd_data) else $error("read data didn't match"); + assert(reg_rd_resp==1'b1) else $error("resp didn't pulse"); + @(posedge clk); + @(negedge clk); + assert(reg_rd_resp==1'b0) else $error("resp set late"); + end + endtask + + + //--------------------------------------------------------------------------- + // Reset + //--------------------------------------------------------------------------- + + task test_reset(); + wait(!reset); + repeat (10) @(posedge clk); + test.end_test(); + endtask : test_reset + + //--------------------------------------------------------------------------- + // Test Registers + //--------------------------------------------------------------------------- + + task test_registers(); + test.start_test({TEST_NAME,"Test/Setup Registers"}, 10us); + // DEF_DEST_MAC/IP/UDP are defined in the + // sim_ethernet_lib.svh, as the destination + // addresses. Using the defaults means + // if I don't change the dest address on + // a packet it will go to the CHDR + reg_wr(REG_MAC_LSB,DEF_DEST_MAC_ADDR[31:0]); + reg_wr(REG_MAC_MSB,DEF_DEST_MAC_ADDR[47:32]); + reg_wr(REG_IP,DEF_DEST_IP_ADDR); + reg_wr(REG_UDP,DEF_DEST_UDP_PORT); + + repeat (3) @(posedge clk); + + `ASSERT_ERROR(my_mac==DEF_DEST_MAC_ADDR, "my mac mismatched!"); + `ASSERT_ERROR(my_ip==DEF_DEST_IP_ADDR, "my ip mismatched!"); + `ASSERT_ERROR(my_udp_chdr_port==DEF_DEST_UDP_PORT,"my udp mismatched!"); + + reg_wr(REG_BRIDGE_ENABLE,1); + reg_wr(REG_BRIDGE_MAC_LSB,DEF_BRIDGE_MAC_ADDR[31:0]); + reg_wr(REG_BRIDGE_MAC_MSB,DEF_BRIDGE_MAC_ADDR[47:32]); + reg_wr(REG_BRIDGE_IP,DEF_BRIDGE_IP_ADDR); + reg_wr(REG_BRIDGE_UDP,DEF_BRIDGE_UDP_PORT); + + repeat (3) @(posedge clk); + `ASSERT_ERROR(my_mac==DEF_BRIDGE_MAC_ADDR, "my mac mismatched!"); + `ASSERT_ERROR(my_ip==DEF_BRIDGE_IP_ADDR, "my ip mismatched!"); + `ASSERT_ERROR(my_udp_chdr_port==DEF_BRIDGE_UDP_PORT,"my udp mismatched!"); + + reg_wr(REG_BRIDGE_ENABLE,0); + + // Readback the values + reg_rd_check(REG_MAC_LSB,DEF_DEST_MAC_ADDR[31:0]); + reg_rd_check(REG_MAC_MSB,DEF_DEST_MAC_ADDR[47:32]); + reg_rd_check(REG_IP,DEF_DEST_IP_ADDR); + reg_rd_check(REG_UDP,DEF_DEST_UDP_PORT); + reg_rd_check(REG_BRIDGE_ENABLE,0); + reg_rd_check(REG_BRIDGE_MAC_LSB,DEF_BRIDGE_MAC_ADDR[31:0]); + reg_rd_check(REG_BRIDGE_MAC_MSB,DEF_BRIDGE_MAC_ADDR[47:32]); + reg_rd_check(REG_BRIDGE_IP,DEF_BRIDGE_IP_ADDR); + reg_rd_check(REG_BRIDGE_UDP,DEF_BRIDGE_UDP_PORT); + + test.end_test(); + endtask : test_registers + + //--------------------------------------------------------------------------- + // Ethernet to CPU test + //--------------------------------------------------------------------------- + typedef ChdrData #(CHDR_W)::chdr_word_t chdr_word_t; + typedef chdr_word_t word_queue_t[$]; + + typedef XportStreamPacket #(ENET_W) EthXportPacket_t; + typedef AxiStreamPacket #(ENET_W,ENET_USER_W) EthAxisPacket_t; + + typedef XportStreamPacket #(CPU_W) CpuXportPacket_t; + typedef AxiStreamPacket #(CPU_W,CPU_USER_W) CpuAxisPacket_t; + + typedef XportStreamPacket #(CHDR_W) ChdrXportPacket_t; + typedef AxiStreamPacket #(CHDR_W,CHDR_USER_W) ChdrAxisPacket_t; + typedef ChdrPacket #(CHDR_W,CHDR_USER_W) ChdrPacket_t; + + task automatic test_ethcpu(int num_samples[$], int ERROR_PROB=2); + TestExec test_e2c = new(); + automatic EthXportPacket_t send[$]; + automatic CpuXportPacket_t expected[$]; + automatic int sample_sum = 0; + + test_e2c.start_test({TEST_NAME,"Ethernet to CPU"}, 60us); + // This path is + // eth_rx -> s_mac(eth_adapter) -> s_mac(eth_dispatch) -> + //// in_reg(AXI_FIFO)(SIZE=1) + // (eth_dispatch) in -> STATMACHINE (Dispatch) + cpu -> + //// out_reg_cpu(AXI_FIFO)(SIZE=1) + // (eth_dispatch) o_cpu -> + //// cpu_out_gate(AXI_GATE)(SIZE=11) + // (eth_dispatch) m_cpu -> (eth_adapter) e2c_chdr -> e2c_fifo + //// cpu_fifo(AXI_FIFO)(SIZE=CPU_FIFO_SIZE) + // (eth_adapater) m_cpu -> e2c + + foreach (num_samples[i]) begin + automatic eth_hdr_t eth_hdr; + automatic ipv4_hdr_t ipv4_hdr; + automatic udp_hdr_t udp_hdr; + automatic raw_pkt_t pay,udp_raw; + automatic int preamble; + + if (PREAMBLE_BYTES == 6) preamble = NORMAL_PREAMBLE; + else if (PREAMBLE_BYTES == 0) preamble = NO_PREAMBLE; + else $fatal("Invalid PREAMBLE_BYTES"); + + expected[i] = new; + send[i] = new; + + udp_hdr.dest_port = 0; //change dest port from default so it goes to cpu + get_ramp_raw_pkt(.num_samps(num_samples[i]),.ramp_start((sample_sum)%256), + .ramp_inc(1),.pkt(pay),.SWIDTH(8)); + sample_sum += num_samples[i]; + udp_raw = build_udp_pkt(eth_hdr,ipv4_hdr,udp_hdr,pay, + .preamble(preamble)); + send[i].push_bytes(udp_raw); + send[i].tkeep_to_tuser(.ERROR_PROB(ERROR_PROB)); + + // rebuild the expected packet for comparison without the preamble + udp_raw = build_udp_pkt(eth_hdr,ipv4_hdr,udp_hdr,pay, + .preamble(NO_PREAMBLE)); + expected[i].push_bytes(udp_raw); + expected[i].tkeep_to_tuser(); + + end + + // iterate in descending order so deleting doesn't shift down + // the packets in future loop iterations. + for (int i= num_samples.size()-1 ; i >= 0; i--) begin + // original only checks for errors in last word. + if (!SV_ETH_IFC && send[i].has_error()) send[i].set_error(); + // If a packet has an error it shouldn't make it through + if (send[i].has_error()) expected.delete(i); // MAC ERROR + else if (DROP_MIN_PACKET) begin + if (send[i].byte_length()-PREAMBLE_BYTES < 64) begin + // short packet rejection feature broken in original + if (SV_ETH_IFC) expected.delete(i); // TOO SHORT + end + end + + end + + fork + begin // tx_thread + foreach(send[i])begin + #1 eth.put(send[i]); + end + end + begin //rx_thread + foreach(expected[i]) begin + automatic CpuAxisPacket_t actual_a; + automatic CpuXportPacket_t actual = new(); + cpu.get(actual_a); + actual.import_axis(actual_a); + actual.tuser_to_tkeep(); +// $display({"****************::%s::**********************\n", +// "**send** \n%s\n**expected**\n%s\n**actual**\n%s \n". +// "%d byte with error %d"}, +// TEST_NAME, +// send[i].sprint(),expected[i].sprint(),actual.sprint(), +// num_samples[i],send[i].has_error()); + + `ASSERT_ERROR(!actual.compare_w_sof(expected[i]),"failed to send packet to e2c"); + end + end + join + + test_e2c.end_test(); + endtask : test_ethcpu + + task automatic wait_for_udp_packets(int udp_dest_port); + automatic EthAxisPacket_t actual_a; + automatic raw_pkt_t rcv_raw,rcv_pay; + automatic udp_hdr_t rcv_udp; + automatic eth_hdr_t rcv_eth; + automatic ipv4_hdr_t rcv_ip; + automatic int try_count = 0; + + do begin + ++try_count; + // check if packet is for our port + #100; + eth.peek(actual_a); + rcv_raw = actual_a.dump_bytes(); + repeat(PREAMBLE_BYTES) rcv_raw.delete(0); // strip preamble + decode_udp_pkt(rcv_raw,rcv_eth,rcv_ip,rcv_udp,rcv_pay); + `ASSERT_ERROR(try_count != 100,"unclaimed packet on c2e"); + end while (rcv_udp.dest_port != udp_dest_port); + + endtask : wait_for_udp_packets + + + task automatic test_cpueth(int num_samples[$]); + TestExec test_c2e = new(); + automatic CpuXportPacket_t send[$]; + automatic EthXportPacket_t expected[$]; + automatic int sample_sum = 0; + + test_c2e.start_test({TEST_NAME,"CPU to Ethernet"}, 60us); + // This path is + // c2e -> (eth_adapter) s_cpu -> + //// (ARM_DEFRAMER)(IF ARM) + // (eth_adapater) c2e -> + //// (ETH_MUX)(SIZE=2) + // (eth_adapater) m_mac -> eth_tx + + foreach (num_samples[i]) begin + automatic eth_hdr_t eth_hdr; + automatic ipv4_hdr_t ipv4_hdr; + automatic udp_hdr_t udp_hdr; + automatic raw_pkt_t pay,udp_raw; + automatic int preamble; + + expected[i] = new; + send[i] = new; + + if (PREAMBLE_BYTES == 6) preamble = ZERO_PREAMBLE; + else if (PREAMBLE_BYTES == 0) preamble = NO_PREAMBLE; + else $fatal("Invalid PREAMBLE_BYTES"); + + get_ramp_raw_pkt(.num_samps(num_samples[i]),.ramp_start((sample_sum)%256), + .ramp_inc(1),.pkt(pay),.SWIDTH(8)); + sample_sum += num_samples[i]; + udp_raw = build_udp_pkt(eth_hdr,ipv4_hdr,udp_hdr,pay, + .preamble(NO_PREAMBLE)); + send[i].push_bytes(udp_raw); + send[i].tkeep_to_tuser(); + + // rebuild the expected packet for comparison with a zero preamble + udp_raw = build_udp_pkt(eth_hdr,ipv4_hdr,udp_hdr,pay, + .preamble(preamble)); + if (SV_ETH_IFC) begin + while (udp_raw.size < 64) begin + udp_raw.push_back(0); + end + end; + expected[i].push_bytes(udp_raw); + expected[i].tkeep_to_tuser(); + end + + fork + begin // tx_thread + foreach(send[i])begin + cpu.put(send[i]); + end + end + begin //rx_thread + foreach(expected[i]) begin + automatic EthAxisPacket_t actual_a; + automatic EthXportPacket_t actual = new(); + automatic raw_pkt_t rcv_raw,rcv_pay; + automatic udp_hdr_t rcv_udp; + automatic eth_hdr_t rcv_eth; + automatic ipv4_hdr_t rcv_ip; + automatic int try_count = 0; + + wait_for_udp_packets(DEF_DEST_UDP_PORT); + eth.get(actual_a); + actual.import_axis(actual_a); + actual.tuser_to_tkeep(); + `ASSERT_ERROR(!actual.compare_w_pad(expected[i],!SV_ETH_IFC),"failed to send packet to c2e"); + end + end + join + test_c2e.end_test(); + + endtask : test_cpueth + //--------------------------------------------------------------------------- + // Ethernet to CHDR test + //--------------------------------------------------------------------------- + + function automatic word_queue_t bytes_to_words(raw_pkt_t pay); + automatic ChdrXportPacket_t axis_pkt = new(); + + axis_pkt.push_bytes(pay); + return axis_pkt.data; + + endfunction : bytes_to_words; + + function automatic raw_pkt_t flatten_chdr(ChdrPacket_t chdr_pkt); + automatic ChdrAxisPacket_t axis_chdr; + automatic ChdrXportPacket_t xport_chdr = new(); + axis_chdr = chdr_pkt.chdr_to_axis(); + foreach (axis_chdr.data[i]) begin + axis_chdr.keep[i] = '1; + axis_chdr.user[i] = '0; + end + xport_chdr.import_axis(axis_chdr); + return xport_chdr.dump_bytes(); + endfunction : flatten_chdr + + function automatic ChdrPacket_t unflatten_chdr(raw_pkt_t chdr_raw); + automatic ChdrXportPacket_t xport_chdr = new(); + automatic ChdrPacket_t chdr_pkt = new(); + xport_chdr.push_bytes(chdr_raw); + foreach (xport_chdr.data[i]) begin + xport_chdr.keep[i] = '1; + xport_chdr.user[i] = '0; + end + chdr_pkt.axis_to_chdr(xport_chdr); + return chdr_pkt; + endfunction : unflatten_chdr + + task automatic test_ethchdr(int num_samples[$], int ERROR_PROB=2); + TestExec test_e2v = new(); + automatic EthXportPacket_t send[$]; + automatic ChdrXportPacket_t expected[$]; + automatic int sample_sum = 0; + + test_e2v.start_test({TEST_NAME,"Ethernet to CHDR"}, 60us); + // This path is + // eth_rx -> s_mac(eth_adapter) -> s_mac(eth_dispatch) -> + //// in_reg(AXI_FIFO)(SIZE=1) + // (eth_dispatch) in -> STATMACHINE (Dispatch) + chdr -> + //// chdr_user_fifo(AXI_FIFO)(SIZE=8) (capture eth header) + //// chdr_out_gate(AXI_GATE)(SIZE=11) + // (eth_dispatch) o_chdr -> + //// chdr_trim(CHDR_TRIM_PAYLOAD) + // (eth_dispatch) m_chdr -> (eth_adapater) e2x_chdr -> (xport_adapter_gen) s_axis_xport + //// xport_in_swap (AXIS_DATA_SWAP) + // (xport_adapter_gen) i_xport -> + //// mgmt_ep(CHDR_MGMT_PKT_HANDLER) + // (xport_adapter_gen) x2d -> + //// rtn_demux(AXI_SWITCH) x2x(loopback) or m_axis_rfnoc + // (xport_adapter_gen) m_axis_rfnoc -> (eth_adapter) e2x_fifo + //// chdr_fifo(AXI_FIFO)(SIZE=MTU) + // (eth_adapater) m_chdr -> e2v + + foreach (num_samples[i]) begin + automatic eth_hdr_t eth_hdr; + automatic ipv4_hdr_t ipv4_hdr; + automatic udp_hdr_t udp_hdr; + automatic raw_pkt_t pay,udp_raw,chdr_raw; + + automatic ChdrPacket_t chdr_pkt = new(); + automatic chdr_header_t chdr_hdr; + automatic chdr_word_t chdr_ts; + automatic chdr_word_t chdr_mdata[$]; + automatic chdr_word_t chdr_data[$]; + automatic int preamble; + + expected[i] = new; + send[i] = new; + + if (PREAMBLE_BYTES == 6) preamble = NORMAL_PREAMBLE; + else if (PREAMBLE_BYTES == 0) preamble = NO_PREAMBLE; + else $fatal("Invalid PREAMBLE_BYTES"); + + // build a payload + get_ramp_raw_pkt(.num_samps(num_samples[i]),.ramp_start((sample_sum)%256), + .ramp_inc(1),.pkt(pay),.SWIDTH(8)); + sample_sum += num_samples[i]; + // Fill data in the chdr packet + chdr_hdr = '{ + vc : 0, + dst_epid : 0, + seq_num : 0, + pkt_type : CHDR_DATA_NO_TS, + num_mdata : 0, + default : 0 + }; + chdr_ts = 0; // no timestamp + chdr_mdata.delete(); // not adding meta data + chdr_data = bytes_to_words(pay); + + chdr_pkt.write_raw(chdr_hdr, chdr_data, chdr_mdata, chdr_ts); + chdr_raw = flatten_chdr(chdr_pkt); + + //build a udp packet + udp_raw = build_udp_pkt(eth_hdr,ipv4_hdr,udp_hdr,chdr_raw, + .preamble(preamble)); + send[i].push_bytes(udp_raw); + send[i].tkeep_to_tuser(.ERROR_PROB(ERROR_PROB)); + + // expect just the chdr packet (UDP stripped) + expected[i].push_bytes(chdr_raw); + expected[i].tkeep_to_tuser(); + + end + + // iterate in descending order so deleting doesn't shift down + // the packets in future loop iterations. + for (int i= num_samples.size()-1 ; i >= 0; i--) begin + // original only checks for errors in last word. + if (!SV_ETH_IFC && send[i].has_error()) send[i].set_error(); + // If a packet has an error it shouldn't make it through + if (send[i].has_error()) expected.delete(i);//MAC ERROR + else if (DROP_MIN_PACKET) begin + if (send[i].byte_length()-PREAMBLE_BYTES < 64) begin + // short packet rejection feature broken in original + if (SV_ETH_IFC) expected.delete(i); // TOO SHORT + end + end + end + + fork + begin // tx_thread + foreach(send[i])begin + #1 eth.put(send[i]); + end + end + begin //rx_thread + foreach(expected[i]) begin + automatic ChdrAxisPacket_t actual_a; + automatic ChdrXportPacket_t actual = new(); + v.get(actual_a); + actual.import_axis(actual_a); + actual.tuser_to_tkeep(); + `ASSERT_ERROR(!actual.compare_no_user(expected[i]),"failed to send packet e2v"); + end + end + join + test_e2v.end_test(); + + endtask : test_ethchdr; + + task automatic test_chdreth(int num_samples[$]); + TestExec test_v2e = new(); + automatic ChdrXportPacket_t send[$]; + automatic EthXportPacket_t expected[$]; + automatic int sample_sum = 0; + + test_v2e.start_test({TEST_NAME,"CHDR to Ethernet"}, 60us); + // This path is + // v2e -> s_chdr(eth_adapter) -> s_axis_rfnoc (xport_adapter_gen) -> + //// axi_demux_mgmt_filter (AXI_DEMUX) (IF ALLOW_DISC) (discards discovery packets) + // (xport_adapter_gen) f2m -> + //// rtn_mux(AXI_MUX) between x2x and f2m + // (xport_adapter_gen) m2x -> + //// data_fifo/lookup_fifo (AXI_FIFO_SHORT) + //// LOOKUP LOGIC (lookup_fifo,data_fifo,results) + // (xport_adapter_gen) o_xport -> + //// xport_out_swap (AXIS_DATA_SWAP) + // (xport_adapter_gen) m_axis_xport -> (eth_adapater) x2e_chdr -> + //// ENET_HDR_LOGIC (frame_state) + // (eth_adapater) frame -> (eth_adapater) x2e_framed + //// (ETH_MUX)(SIZE=2) + // (eth_adapater) m_mac -> eth_tx + + foreach (num_samples[i]) begin + automatic eth_hdr_t eth_hdr; + automatic ipv4_hdr_t ipv4_hdr; + automatic udp_hdr_t udp_hdr; + automatic raw_pkt_t pay,udp_raw,chdr_raw; + + automatic ChdrPacket_t chdr_pkt = new(); + automatic chdr_header_t chdr_hdr; + automatic chdr_word_t chdr_ts; + automatic chdr_word_t chdr_mdata[$]; + automatic chdr_word_t chdr_data[$]; + automatic int preamble; + + expected[i] = new; + send[i] = new; + + if (PREAMBLE_BYTES == 6) preamble = ZERO_PREAMBLE; + else if (PREAMBLE_BYTES == 0) preamble = NO_PREAMBLE; + else $fatal("Invalid PREAMBLE_BYTES"); + + // build a payload + get_ramp_raw_pkt(.num_samps(num_samples[i]),.ramp_start((sample_sum)%256), + .ramp_inc(1),.pkt(pay),.SWIDTH(8)); + sample_sum += num_samples[i]; + + // Fill data in the chdr packet + chdr_hdr = '{ + vc : 0, + dst_epid : 0, + seq_num : 0, + pkt_type : CHDR_DATA_NO_TS, + num_mdata : 0, + default : 0 + }; + chdr_ts = 0; // no timestamp + chdr_mdata.delete(); // not adding meta data + chdr_data = bytes_to_words(pay); + + chdr_pkt.write_raw(chdr_hdr, chdr_data, chdr_mdata, chdr_ts); + chdr_raw = flatten_chdr(chdr_pkt); + + // send the raw chedar packet + send[i].push_bytes(chdr_raw); + send[i].tkeep_to_tuser(); + + //build a udp packet + // modify as the EthInterface does + udp_hdr.src_port = DEF_DEST_UDP_PORT; + udp_hdr.dest_port = 0; // Extract from router lookup results (Default) + udp_hdr.checksum = 0; // Checksum not calculated at this point + ipv4_hdr.src_ip = DEF_DEST_IP_ADDR; + ipv4_hdr.dest_ip = 0; // Extract from router lookup results (Default) + ipv4_hdr.dscp = 0; // hardcoded + ipv4_hdr.dont_frag = 1; // hardcoded + ipv4_hdr.identification = 0; // hardcoded + ipv4_hdr.time_to_live = 8'h10; //hardcoded + eth_hdr.src_mac = DEF_DEST_MAC_ADDR; + eth_hdr.dest_mac = 0; // Extract from router lookup results (Default) + + udp_raw = build_udp_pkt(eth_hdr,ipv4_hdr,udp_hdr,chdr_raw, + .preamble(preamble)); + + // expect udp wrapped chdr + expected[i].push_bytes(udp_raw); + if (IGNORE_EXTRA_DATA) begin + expected[i].clear_user(); // expect all full words! + expected[i].tuser_to_tkeep(); + end else begin + expected[i].tkeep_to_tuser(); + end + end + + fork + begin // tx_thread + foreach(send[i])begin + v.put(send[i]); + end + end + begin //rx_thread + foreach(expected[i]) begin + automatic EthAxisPacket_t actual_a; + automatic EthXportPacket_t actual = new(); + automatic eth_hdr_t eth_hdr; + automatic ipv4_hdr_t ipv4_hdr; + automatic udp_hdr_t udp_hdr; + automatic raw_pkt_t chdr_raw,actual_raw; + automatic ChdrPacket_t chdr_pkt; + automatic integer chdr_len; + localparam UDP_LEN = 8/*udp*/+20/*ipv4*/+14/*eth-no vlan*/; + + wait_for_udp_packets(.udp_dest_port(0)); + eth.get(actual_a); + actual.import_axis(actual_a); + actual.tuser_to_tkeep(); + actual_raw = actual.dump_bytes(); + repeat(PREAMBLE_BYTES) actual_raw.pop_front(); + decode_udp_pkt(actual_raw,eth_hdr,ipv4_hdr,udp_hdr,chdr_raw); + chdr_pkt = unflatten_chdr(chdr_raw); + // fills remainder of packet with zeros + if (IGNORE_EXTRA_DATA) begin + for (int w=chdr_pkt.header.length+UDP_LEN;w i.DATA_WIDTH/8 || chdr_end_early) begin + s4.tlast = s3.tvalid && s3.tlast; // s0metimes packets end early + s4.tuser = 0; + end else if (rem_bytes == i.DATA_WIDTH/8) begin + s4.tlast = s3.tvalid; + s4.tuser = 0; + end else begin + s4.tlast = s3.tvalid; + s4.tuser = rem_bytes; + end + s4.tdata = s3.tdata; + s4.put_packet_field48(mac_dst_reg, PREAMBLE_BYTES+DST_MAC_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field48(mac_src, PREAMBLE_BYTES+SRC_MAC_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field16(ETH_TYPE, PREAMBLE_BYTES+ETH_TYPE_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field16(MISC_IP, PREAMBLE_BYTES+IP_VERSION_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field16(ip_len_reg, PREAMBLE_BYTES+IP_LENGTH_BYTE, .NETWORK_ORDER(1)); +// s4.put_packet_field16(IDENT, PREAMBLE_BYTES+IP_ID_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field16(FLAG_FRAG, PREAMBLE_BYTES+IP_FRAG_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field16(TTL_PROT, PREAMBLE_BYTES+IP_TTL_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field16(iphdr_checksum, PREAMBLE_BYTES+IP_CHECKSUM_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field32(ip_src, PREAMBLE_BYTES+SRC_IP_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field32(ip_dst_reg, PREAMBLE_BYTES+DST_IP_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field16(udp_src, PREAMBLE_BYTES+SRC_PORT_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field16(udp_dst_reg, PREAMBLE_BYTES+DST_PORT_BYTE, .NETWORK_ORDER(1)); + s4.put_packet_field16(udp_len_reg, PREAMBLE_BYTES+UDP_LENGTH_BYTE, .NETWORK_ORDER(1)); +// s4.put_packet_field16(udp_checksum, PREAMBLE_BYTES+UDP_CHECKSUM_BYTE,.NETWORK_ORDER(1)); + + end + + always_comb begin : assign_output + `AXI4S_ASSIGN(o,s4) + end + +endmodule // eth_ipv4_add_udp diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_chdr_adapter.sv b/fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_chdr_adapter.sv new file mode 100644 index 000000000..8505836bf --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_chdr_adapter.sv @@ -0,0 +1,473 @@ +// +// Copyright 2020 Ettus Research, A National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: eth_ipv4_chdr_adapter +// Description: A generic transport adapter module that can be used in +// a variety of transports. It does the following: +// - Exposes a configuration port for mgmt packets to configure the node +// - Implements a return-address map for packets with metadata other than +// the CHDR. Additional metadata can be passed as a tuser to this module +// which will store it in a map indexed by the SrcEPID in a management +// packet. For all returning packets, the metadata will be looked up in +// the map and attached as the outgoing tuser. +// - Implements a loopback path for node-info discovery +// +// Parameters: +// - PROTOVER: RFNoC protocol version {8'd, 8'd} +// - MTU: Log2 of the MTU of the packet in 64-bit words +// - CPU_FIFO_SIZE: Log2 of the FIFO depth (in 64-bit words) for the CPU egress path +// - RT_TBL_SIZE: Log2 of the depth of the return-address routing table +// - NODE_INST: The node type to return for a node-info discovery +// - DROP_UNKNOWN_MAC: Drop packets not addressed to us? +// - DROP_MIN_PACKET: Drop packets smaller than 64 bytes? +// - PREAMBLE_BYTES: Number of bytes of Preamble expected +// - ADD_SOF: Add a SOF indication into the tuser field +// If false use TKEEP instead of USER +// - SYNC: Set if MAC is not the same as bus_clk +// - ENET_W: Width of the link to the Ethernet MAC +// - CPU_W: Width of the CPU interface +// - CHDR_W: Width of the CHDR interface +// +// Signals: +// - device_id : The ID of the device that has instantiated this module +// - eth_rx : The input Ethernet stream from the MAC +// - eth_tx : The output Ethernet stream to the MAC +// - v2e : The input CHDR stream from the rfnoc infrastructure +// - e2v : The output CHDR stream to the rfnoc infrastructure +// - c2e : The input Ethernet stream from the CPU +// - e2c : The output Ethernet stream to the CPU +// - my_mac: The Ethernet (MAC) address of this endpoint +// - my_ip: The IPv4 address of this endpoint +// - my_udp_chdr_port: The UDP port allocated for CHDR traffic on this endpoint +// + +module eth_ipv4_chdr_adapter #( + logic [15:0] PROTOVER = {8'd1, 8'd0}, + int MTU = 10, + int CPU_FIFO_SIZE = MTU, + int RT_TBL_SIZE = 6, + int NODE_INST = 0, + bit DROP_UNKNOWN_MAC = 0, + bit DROP_MIN_PACKET = 0, + int PREAMBLE_BYTES = 6, + bit ADD_SOF = 1, + bit SYNC = 0, + int ENET_W = 64, + int CPU_W = 64, + int CHDR_W = 64 +)( + // Device info + input logic [15:0] device_id, + // Device addresses + input logic [47:0] my_mac, + input logic [31:0] my_ip, + input logic [15:0] my_udp_chdr_port, + + // Ethernet MAC + AxiStreamIf.master eth_tx, // tUser = {1'b0,trailing bytes}; + AxiStreamIf.slave eth_rx, // tUser = {error,trailing bytes}; + // CHDR router interface + AxiStreamIf.master e2v, // tUser = {*not used*}; + AxiStreamIf.slave v2e, // tUser = {*not used*}; + // CPU DMA + AxiStreamIf.master e2c, // tUser = {sof,trailing bytes}; + AxiStreamIf.slave c2e // tUser = {1'b0,trailing bytes}; + +); + + `include "../core/rfnoc_chdr_utils.vh" + `include "../core/rfnoc_chdr_internal_utils.vh" + `include "../../axi4s_sv/axi4s.vh" + + + localparam ENET_USER_W = $clog2(ENET_W/8)+1; + localparam CPU_USER_W = $clog2(CPU_W/8)+1; + localparam CHDR_USER_W = $clog2(CHDR_W/8); + localparam MAX_PACKET_BYTES = 2**16; + localparam DEBUG = 1; + + `include "eth_constants.vh" + + //--------------------------------------- + // E2V and E2C DEMUX + //--------------------------------------- + // tUser = {*not used*} + AxiStreamIf #(.DATA_WIDTH(ENET_W),.TKEEP(0),.TUSER(0)) e2v1(eth_rx.clk,eth_rx.rst); + // tUser = {*not used*} + AxiStreamIf #(.DATA_WIDTH(CHDR_W),.TKEEP(0),.TUSER(0)) e2v2(e2v.clk,e2v.rst); + // tUser = {*not used*} + AxiStreamIf #(.DATA_WIDTH(CHDR_W),.TKEEP(0),.TUSER(0)) e2v4(e2v.clk,e2v.rst); + // tUser = {*not used*} + AxiStreamIf #(.DATA_WIDTH(CHDR_W),.TKEEP(0),.TUSER(0)) e2v5(e2v.clk,e2v.rst); + + // tUser = {1'b0,trailing bytes} + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) e2c1(eth_rx.clk,eth_rx.rst); + // tUser = {sof,trailing bytes} IF ADD_SOF + AxiStreamIf #(.DATA_WIDTH(CPU_W),.USER_WIDTH(CPU_USER_W), + .TKEEP(!ADD_SOF), .TUSER(ADD_SOF)) + e2c2(e2c.clk,e2c.rst); + + logic [47:0] e_my_mac; + logic [31:0] e_my_ip; + logic [15:0] e_my_udp_chdr_port; + // crossing clock boundaries. + // my_mac, my_ip,,my_udp_chdr_port must be written + // prior to traffic, or an inconsistent version will + // exist for a clock period or 2. This would be better + // done with a full handshake. + synchronizer #(.WIDTH(96),.STAGES(1)) + e_info_sync (.clk(eth_rx.clk),.rst(eth_rx.rst), + .in({my_mac,my_ip,my_udp_chdr_port}), + .out({e_my_mac,e_my_ip,e_my_udp_chdr_port})); + + // Ethernet sink. Inspects packet and dispatches + // to the correct port. + eth_ipv4_chdr_dispatch #( + .CPU_FIFO_SIZE(CPU_FIFO_SIZE), + .PREAMBLE_BYTES(PREAMBLE_BYTES), + .MAX_PACKET_BYTES(MAX_PACKET_BYTES), + .DROP_UNKNOWN_MAC(DROP_UNKNOWN_MAC), + .DROP_MIN_PACKET(DROP_MIN_PACKET), + .ENET_W(ENET_W) + ) eth_dispatch_i ( + .eth_rx (eth_rx), + .e2v (e2v1), + .e2c (e2c1), + .my_mac (e_my_mac), + .my_ip (e_my_ip), + .my_udp_chdr_port (e_my_udp_chdr_port) + ); + + //--------------------------------------- + // E2C Path + //--------------------------------------- + if (ENET_W != CPU_W || !SYNC) begin : gen_e2c_width_conv + axi4s_width_conv #(.I_USER_TRAILING_BYTES(1),.O_USER_TRAILING_BYTES(ADD_SOF),.SYNC_CLKS(0)) + e2c_width_conv (.i(e2c1), .o(e2c2)); + end else begin : gen_e2c_width_match + always_comb begin : e2c_assign + `AXI4S_ASSIGN(e2c2,e2c1) + end + end + + if (ADD_SOF) begin : add_sof + logic sof = 1'b1; + + // Add SOF + always_ff @(posedge e2c.clk) begin : cpu3_find_sof + if (e2c.rst) begin + sof <= 1'b1; + end else if (e2c2.tvalid && e2c2.tready) begin + sof <= e2c2.tlast; + end + end + always_comb begin : e2c2_sof_assign + `AXI4S_ASSIGN(e2c,e2c2) + e2c.tuser = {sof,e2c2.tuser[CPU_USER_W-2:0]}; + end + end else begin : no_sof + if (DEBUG) begin + `AXI4S_DEBUG_ASSIGN(e2c,e2c2) + end else begin + always_comb begin : e2c_nodebug_assign + `AXI4S_ASSIGN(e2c,e2c2) + end + end + end + + //--------------------------------------- + // E2V Path + //--------------------------------------- + if (ENET_W != CHDR_W || !SYNC) begin : gen_e2v_width_conv + // assumes full words on input + axi4s_width_conv #(.SYNC_CLKS(0)) + e2v_width_conv (.i(e2v1), .o(e2v2)); + end else begin : gen_e2v_width_match + always_comb begin : e2v_assign + `AXI4S_ASSIGN(e2v2,e2v1) + end + end + + + //--------------------------------------- + // CHDR Transport Adapter + //--------------------------------------- + + // tUser = {*not used*} + AxiStreamIf #(.DATA_WIDTH(CHDR_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) + v2e1D(v2e.clk,v2e.rst); + // tUser = {*not used*} + AxiStreamIf #(.DATA_WIDTH(CHDR_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) + v2e1(v2e.clk,v2e.rst); + // tUser = {*not used*} + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) + v2e2(eth_rx.clk,eth_rx.rst); + // tUser = {*not used*} + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) + v2e3(eth_rx.clk,eth_rx.rst); + + chdr_xport_adapter #( + .PREAMBLE_BYTES (PREAMBLE_BYTES), + .MAX_PACKET_BYTES (MAX_PACKET_BYTES), + .PROTOVER (PROTOVER), + .TBL_SIZE (RT_TBL_SIZE), + .NODE_INST (NODE_INST), + .ALLOW_DISC (1) + ) xport_adapter_gen_i ( + .device_id (device_id), + .my_mac (my_mac), + .my_ip (my_ip), + .my_udp_chdr_port (my_udp_chdr_port), + + .eth_rx (e2v2), // from ethernet + .e2v (e2v4), // to CHDR + // optional loop from ethernet to ethernet to talk to node + .v2e (v2e), // from CHDR + .eth_tx (v2e1D) // to ethernet + ); + + if (DEBUG) begin + `AXI4S_DEBUG_ASSIGN(v2e1,v2e1D) + end else begin + always_comb begin : v2e_nodebug_assign + `AXI4S_ASSIGN(v2e1,v2e1D) + end + end + + // Convert incoming CHDR_W + if (ENET_W != CHDR_W || !SYNC) begin : gen_v2e_width_conv + axi4s_width_conv #(.SYNC_CLKS(0),.I_USER_TRAILING_BYTES(1),.O_USER_TRAILING_BYTES(1)) + v2e_width_conv (.i(v2e1), .o(v2e2)); + end else begin : gen_v2e_width_match + always_comb begin : v2e1_assign + `AXI4S_ASSIGN(v2e2,v2e1) + end + end + + // Adding so packet will be contiguous going out + // The MAC needs bandwidth feeding it to be greater than the line rate + if (ENET_W > CHDR_W || !SYNC) begin : gen_v2e_packet_gate + axi4s_packet_gate #(.SIZE(17-$clog2(ENET_W)), .USE_AS_BUFF(0)) + v2e_gate_i (.clear(1'b0),.error(1'b0),.i(v2e2),.o(v2e3)); + end else begin : gen_v2e_no_packet_gate + always_comb begin : v2e1_assign + `AXI4S_ASSIGN(v2e3,v2e2) + end + end + + //--------------------------------------- + // E2V Output Buffering + //--------------------------------------- + + // The transport should hook up to a crossbar downstream, which + // may backpressure this module because it is in the middle of + // transferring a packet. To ensure that upstream logic is not + // blocked, we instantiate one packet worth of buffering here. + axi4s_fifo #( + .SIZE(MTU) + ) chdr_fifo_i ( + .clear(1'b0),.space(),.occupied(), + .i(e2v4),.o(e2v5) + ); + + if (DEBUG) begin + `AXI4S_DEBUG_ASSIGN(e2v,e2v5) + end else begin + always_comb begin : e2v_direct_assign + `AXI4S_ASSIGN(e2v,e2v5) + end + end + + + + + //--------------------------------------- + // C2E Path + //--------------------------------------- + // tUser = {1'b0,trailing bytes} + AxiStreamIf #(.DATA_WIDTH(c2e.DATA_WIDTH),.USER_WIDTH(c2e.USER_WIDTH), + .TKEEP(c2e.TKEEP),.TUSER(c2e.TUSER)) + c2eD(c2e.clk,c2e.rst); + // tUser = {1'b0,trailing bytes} + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) c2e1(eth_rx.clk,eth_rx.rst); + // tUser = {1'b0,trailing bytes} + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0),.MAX_PACKET_BYTES(MAX_PACKET_BYTES)) c2e2(eth_rx.clk,eth_rx.rst); + // tUser = {1'b0,trailing bytes} + AxiStreamPacketIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0),.MAX_PACKET_BYTES(MAX_PACKET_BYTES)) c2e3(eth_rx.clk,eth_rx.rst); + + + if (DEBUG) begin + `AXI4S_DEBUG_ASSIGN(c2eD,c2e) + end else begin + always_comb begin : c2e_nodebug_assign + `AXI4S_ASSIGN(c2eD,c2e) + end + end + + if (ENET_W != CPU_W || !SYNC) begin : gen_c2e_width_conv + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) c2e1_0(eth_rx.clk,eth_rx.rst); + axi4s_width_conv #(.I_USER_TRAILING_BYTES(c2eD.TUSER),.O_USER_TRAILING_BYTES(1),.SYNC_CLKS(0)) + c2e_width_conv (.i(c2eD), .o(c2e1_0)); + + if (ENET_W > CPU_W || !SYNC) begin : gen_c2e_packet_gate + // Adding so packet will be contiguous going out + // I think the MAC needs bandwdith feeding it to + // be greater than the line rate + axi4s_packet_gate #(.SIZE(17-$clog2(ENET_W)), .USE_AS_BUFF(0)) + c2e_gate_i (.clear(1'b0),.error(1'b0),.i(c2e1_0),.o(c2e1)); + end else begin : gen_c2e_no_packet_gate + always_comb begin : c2e1_assign + `AXI4S_ASSIGN(c2e1,c2e1_0) + end + end + end else begin : gen_c2e_width_match + always_comb begin : c2e1_assign + `AXI4S_ASSIGN(c2e1,c2eD) + end + end + + if (PREAMBLE_BYTES > 0) begin : gen_add_preamble + // Add pad of PREAMBLE_BYTES empty bytes to the ethernet packet going + // from the CPU to the SFP. This padding added before MAC addresses + // aligns the source and destination IP addresses, UDP headers etc. + // Note that the xge_mac_wrapper strips this padding to recreate the ethernet + // packet + axi4s_add_bytes #(.ADD_START(0),.ADD_BYTES(PREAMBLE_BYTES) + ) add_header ( + .i(c2e1), .o(c2e2) + ); + end else begin : gen_no_preamble + always_comb begin : c2e2_assign + `AXI4S_ASSIGN(c2e2,c2e1) + end + end + + localparam FORCE_MIN_PACKET = 1; + if (FORCE_MIN_PACKET) begin : gen_force_min + // add extra zero bytes to the end of a packet if it is less + // than the minimum packet size. + typedef enum logic { + ST_IDLE, + ST_AFTER + } pad_state_t; + pad_state_t pad_state = ST_IDLE; + logic clk_before_minpacket; + logic pad_last; + + always_comb clk_before_minpacket = c2e3.reached_packet_byte(MIN_PACKET_SIZE_BYTE); + + always_ff @(posedge eth_rx.clk) begin : pad_state_ff + if (eth_rx.rst) begin + pad_state <= ST_IDLE; + pad_last <= 0; + end else begin + if (c2e3.tready && c2e3.tvalid && c2e3.tlast) begin + pad_state <= ST_IDLE; + end else if (c2e3.tready && c2e3.tvalid && clk_before_minpacket) begin + pad_state <= ST_AFTER; + end + if (c2e3.tready && c2e3.tvalid && c2e3.tlast) begin + pad_last <= 0; + end else if (c2e3.tready && c2e3.tvalid && c2e2.tlast) begin + pad_last <= 1; + end + + + end + end + + always_comb begin : c2e3_pad + if (pad_state == ST_IDLE) begin + //force to a full word + // preserve SOF if it's there, but force + // trailing bytes to zero (full word) + c2e3.tuser = 0; + c2e3.tuser[ENET_USER_W-1] = c2e2.tuser[ENET_USER_W-1]; + c2e3.tvalid = c2e2.tvalid; + + // force any tdata bytes that we pad with zero + // SW recommended forcing the padding bytes to zero + // but I suspect we could save logic by just allowing + // trash data. + foreach (c2e2.tkeep[i]) begin + if (pad_last || (i >= c2e2.tuser[ENET_USER_W-2:0] && + c2e2.tuser[ENET_USER_W-2:0] != 0)) begin + c2e3.tdata[i*8 +:8] = 0; + end else begin + c2e3.tdata[i*8 +:8] = c2e2.tdata[i*8 +:8]; + end + end + + if (ENET_W < 512) begin + // hold off input if we reach the end early + if (c2e2.tlast) begin + c2e2.tready = clk_before_minpacket && c2e3.tready; + end else begin + c2e2.tready = c2e3.tready; + end + // add tlast at end of idle + c2e3.tlast = clk_before_minpacket && c2e2.tlast; + end else begin + c2e2.tready = c2e3.tready; + c2e3.tlast = c2e2.tlast; + end + + end else begin + `AXI4S_ASSIGN(c2e3,c2e2) + end + + end + end else begin : gen_no_force_min + always_comb begin : c2e3_assign + `AXI4S_ASSIGN(c2e3,c2e2) + end + end + + + //--------------------------------------- + // V2E and C2E MUX + //--------------------------------------- + logic c2e3_tready; + logic eth_tx1_tlast; + logic eth_tx1_tvalid; + logic eth_tx1_tready; + logic [ENET_W-1:0] eth_tx1_tdata; + logic [ENET_USER_W-1:0] eth_tx1_tuser; + always_comb begin + c2e3.tready = c2e3_tready; + end + axi_mux #( + .SIZE(2), .PRIO(0), .WIDTH(ENET_W+ENET_USER_W), .PRE_FIFO_SIZE(0), .POST_FIFO_SIZE(1) + ) eth_mux_i ( + .clk(eth_rx.clk), .reset(eth_rx.rst), .clear(1'b0), + .i_tdata({c2e3.tuser, c2e3.tdata, v2e3.tuser, v2e3.tdata}), .i_tlast({c2e3.tlast, v2e3.tlast}), + .i_tvalid({c2e3.tvalid, v2e3.tvalid}), .i_tready({c2e3_tready, v2e3.tready}), + .o_tdata({eth_tx1_tuser, eth_tx1_tdata}), .o_tlast(eth_tx1_tlast), + .o_tvalid(eth_tx1_tvalid), .o_tready(eth_tx1_tready) + ); + + // Clean up the noisy mux output. I suspect it is annoying + // the xilinx cores that tlast and tuser(tkeep) flop around + // when tvalid isn't true. + always_comb begin : eth_tx_assign + if (eth_tx1_tvalid) begin + eth_tx.tvalid = 1'b1; + eth_tx.tdata = eth_tx1_tdata; + eth_tx.tlast = eth_tx1_tlast; + if (eth_tx1_tlast) begin + eth_tx.tuser = eth_tx1_tuser; + end else begin + eth_tx.tuser = '0; + end + end else begin + eth_tx.tvalid = 1'b0; + eth_tx.tdata = 'X; // use X so synth will optimize + eth_tx.tlast = 0; + eth_tx.tuser = '0; + end + eth_tx1_tready = eth_tx.tready; + end + +endmodule // eth_ipv4_chdr_adapter diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_chdr_dispatch.sv b/fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_chdr_dispatch.sv new file mode 100644 index 000000000..36f887190 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_chdr_dispatch.sv @@ -0,0 +1,435 @@ +// +// Copyright 2020 Ettus Research, A National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: eth_ipv4_chdr_dispatch +// +// Description: +// This module serves as an Ethernet endpoint for CHDR traffic. +// Ethernet frames arrive on the eth_rx port where they are +// inspected and classified as CHDR or !CHDR. A frame contains +// CHDR payload if it is addressed to us (Eth and IP), is a UDP +// packet and the destination port is one of the CHDR ports. +// The UDP payload for CHDR frame is sent out of the e2v +// in addition to source information for Eth, IP and UDP. All +// other traffic address to us (Eth) is sent to the e2c port. +// Traffic not addressed (Eth) to us is dropped(optionally). +// +// Parameters: +// - CPU_FIFO_SIZE: log2 size of CPU RX fifo +// - PREAMBLE_BYTES: Number of bytes in the Preamble +// - DROP_UNKNOWN_MAC: Drop packets not addressed to us? +// - DROP_MIN_PACKET: Drop packets smaller than 64 bytes? +// - ENET_W: Width of AXI bus going to Ethernet Mac +// +// Signals: +// - eth_rx : The input Ethernet stream from the MAC +// tUser={error,trailing bytes} +// - e2v : The output CHDR stream to the rfnoc infrastructure +// - e2c : The output Ethernet stream to the CPU +// +// - my_mac : The Ethernet (MAC) address of this endpoint +// - my_ip : The IPv4 address of this endpoint +// - my_udp_chdr_port : The UDP port allocated for CHDR traffic on this endpoint +// + +module eth_ipv4_chdr_dispatch #( + int CPU_FIFO_SIZE = $clog2(1558), + int PREAMBLE_BYTES = 6, + int MAX_PACKET_BYTES = 2**16-1, + bit DROP_UNKNOWN_MAC = 0, + bit DROP_MIN_PACKET = 0, + int ENET_W = 64 +)( + + // AXI-Stream interfaces + AxiStreamIf.slave eth_rx, // tUser={error,trailing bytes}; + AxiStreamIf.master e2v, // tUser={1'b0,trailing bytes}; + AxiStreamIf.master e2c, // tUser={1'b0,trailing bytes}; + + // Device addresses + input logic [47:0] my_mac, + input logic [31:0] my_ip, + input logic [15:0] my_udp_chdr_port +); + + localparam ENET_USER_W = $clog2(ENET_W/8)+1; + //--------------------------------------- + // Include for byte positions + //--------------------------------------- + `include "eth_constants.vh" + // example macro to handle interface assignment + `define AXI4S_ASSIGN(O,I) \ + ``O.tdata = ``I.tdata;\ + ``O.tuser = ``I.tuser;\ + ``O.tlast = ``I.tlast;\ + ``O.tvalid = ``I.tvalid;\ + ``I.tready = ``O.tready; + + // axi_remov_bytes (PREAMBLE Strip) + // tUser = {error,trailing bytes}; + AxiStreamPacketIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W), + .TKEEP(0),.MAX_PACKET_BYTES(MAX_PACKET_BYTES)) + in0(eth_rx.clk,eth_rx.rst); + // in_reg + // tUser = {error,trailing bytes}; + AxiStreamPacketIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W), + .TKEEP(0),.MAX_PACKET_BYTES(MAX_PACKET_BYTES)) + in1(eth_rx.clk,eth_rx.rst); + // STATEMACHINE + // tUser = {error,trailing bytes}; + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) + in2(eth_rx.clk,eth_rx.rst); + + // CPU_BRANCH + // tUser = {error,trailing bytes}; + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) + cpu0(eth_rx.clk,eth_rx.rst); + // out_reg_cpu + // tUser = {error,trailing bytes}; + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) + cpu1(eth_rx.clk,eth_rx.rst); + // cpu_out_gate - throw away error packets + // tUser = {1'b0,trailing bytes}; + AxiStreamIf #(.DATA_WIDTH(ENET_W),.USER_WIDTH(ENET_USER_W),.TKEEP(0)) + cpu2(eth_rx.clk,eth_rx.rst); + // cpu_out_fifo + // e2c (OUTPUT) + + // CHDR_Branch + // tUser = {error,trailing bytes}; + AxiStreamIf #(.DATA_WIDTH(ENET_W),.TKEEP(0),.TUSER(0)) + chdr0(eth_rx.clk,eth_rx.rst); + // e2v(OUTPUT) + + //--------------------------------------- + // Strip Bytes + //--------------------------------------- + if (PREAMBLE_BYTES > 0) begin : gen_strip_preamble + // Strip the preamble + axi4s_remove_bytes #(.REM_START(0),.REM_END(PREAMBLE_BYTES-1) + ) strip_preamble ( + .i(eth_rx),.o(in0) + ); + end else begin : gen_no_preamble + always_comb begin + `AXI4S_ASSIGN(in0,eth_rx); + end + end + + //--------------------------------------- + // Input pipeline stage + //--------------------------------------- + axi4s_fifo #( + .SIZE(1) + ) in_reg_i ( + .clear(1'b0),.space(),.occupied(), + .i(in0), .o(in1) + ); + + //--------------------------------------- + // Classification state machine + //--------------------------------------- + typedef enum logic [2:0] { + ST_IDLE_ETH_L0 = 3'd0, + ST_FWD_CHDR = 3'd1, + ST_FWD_CPU = 3'd2, + ST_DROP_TERM = 3'd3, + ST_DROP_WAIT = 3'd4 + } dispatch_state_t; + + // State info + dispatch_state_t dispatch_state,next_dispatch_state = ST_IDLE_ETH_L0; + logic cpu_error = 1'b0; + logic chdr_error = 1'b0; + logic chdr0_error = 1'b0; + logic mac_error, mac_error_old = 1'b0; + logic min_packet_error, min_packet_error_old = 1'b0; + logic reached_min_packet; + + // Cached fields + logic [47:0] eth_dst_addr_new, eth_src_addr_new; + logic [31:0] ipv4_dst_addr_new, ipv4_src_addr_new; + logic [15:0] udp_dst_port_new, udp_src_port_new, eth_type_new; + logic [7:0] ip_protocol_new, ip_version_new; + logic [47:0] eth_dst_addr_old, eth_src_addr_old; + logic [31:0] ipv4_dst_addr_old, ipv4_src_addr_old; + logic [15:0] udp_dst_port_old, udp_src_port_old, eth_type_old; + logic [7:0] ip_protocol_old, ip_version_old; + logic reached_min_packet_new, reached_min_packet_old; + logic reached_end_of_udp; + + logic eth_dst_is_broadcast; + logic eth_dst_is_me; + logic udp_dst_is_me; + logic ipv4_dst_is_me; + logic ipv4_protocol_is_udp; + logic eth_type_is_ipv4; + + // save the fields + always_ff @(posedge eth_rx.clk) begin : field_ff + if (eth_rx.rst) begin + + eth_dst_addr_old <= '0; + eth_src_addr_old <= '0; + ip_protocol_old <= '0; + ip_version_old <= '0; + ipv4_src_addr_old <= '0; + ipv4_dst_addr_old <= '0; + udp_src_port_old <= '0; + udp_dst_port_old <= '0; + eth_type_old <= '0; + mac_error_old <= 1'b0; + min_packet_error_old <= 1'b0; + reached_min_packet_old <= 1'b0; + + // Statemachine Decisions + eth_dst_is_broadcast <=1'b0; + eth_dst_is_me <=1'b0; + udp_dst_is_me <=1'b0; + ipv4_dst_is_me <=1'b0; + ipv4_protocol_is_udp <=1'b0; + eth_type_is_ipv4 <=1'b0; + end else begin + eth_dst_addr_old <= eth_dst_addr_new; + eth_src_addr_old <= eth_src_addr_new; + ip_protocol_old <= ip_protocol_new; + ip_version_old <= ip_version_new; + ipv4_src_addr_old <= ipv4_src_addr_new; + ipv4_dst_addr_old <= ipv4_dst_addr_new; + udp_src_port_old <= udp_src_port_new; + udp_dst_port_old <= udp_dst_port_new; + eth_type_old <= eth_type_new; + + if (in0.tvalid && in0.tready) begin + eth_dst_is_broadcast <= eth_dst_addr_new == ETH_ADDR_BCAST; + eth_dst_is_me <= eth_dst_addr_new == my_mac; + udp_dst_is_me <= udp_dst_port_new == my_udp_chdr_port; + ipv4_dst_is_me <= ipv4_dst_addr_new == my_ip; + ipv4_protocol_is_udp <= ip_protocol_new == IPV4_PROTO_UDP; + eth_type_is_ipv4 <= eth_type_new == ETH_TYPE_IPV4; + end + if (in1.tvalid && in1.tready) begin + if (in1.tlast) begin + mac_error_old <= 1'b0; + min_packet_error_old <= 1'b0; + reached_min_packet_old <= 1'b0; + end else begin + if (mac_error) + mac_error_old <= 1'b1; + if(min_packet_error) + min_packet_error_old <= 1'b1; + if(reached_min_packet_new) + reached_min_packet_old <= 1'b1; + end + end + end + end + + // get the fields - don't use assign. assign will not activate with changes to in0. + always_comb begin : get_fields + eth_dst_addr_new = in0.get_packet_field48(eth_dst_addr_old,DST_MAC_BYTE,.NETWORK_ORDER(1)); + eth_src_addr_new = in0.get_packet_field48(eth_src_addr_old,SRC_MAC_BYTE,.NETWORK_ORDER(1)); + ip_version_new = in0.get_packet_byte(ip_version_old,IP_VERSION_BYTE); + ip_protocol_new = in0.get_packet_byte(ip_protocol_old,PROTOCOL_BYTE); + ipv4_src_addr_new = in0.get_packet_field32(ipv4_src_addr_old,SRC_IP_BYTE,.NETWORK_ORDER(1)); + ipv4_dst_addr_new = in0.get_packet_field32(ipv4_dst_addr_old,DST_IP_BYTE,.NETWORK_ORDER(1)); + udp_src_port_new = in0.get_packet_field16(udp_src_port_old,SRC_PORT_BYTE,.NETWORK_ORDER(1)); + udp_dst_port_new = in0.get_packet_field16(udp_dst_port_old,DST_PORT_BYTE,.NETWORK_ORDER(1)); + eth_type_new = in0.get_packet_field16(eth_type_old,ETH_TYPE_BYTE,.NETWORK_ORDER(1)); + end + + + always_comb begin : reached_bytes + reached_min_packet_new = in1.reached_packet_byte(MIN_PACKET_SIZE_BYTE); + reached_end_of_udp = in1.reached_packet_byte(DST_PORT_BYTE+3);// we have enough to decide + end + assign mac_error = in1.tuser[ERROR_BIT] || mac_error_old; + if (DROP_MIN_PACKET) begin + assign reached_min_packet = (reached_min_packet_new && in1.tuser[BYTES_MSB:0] ==0) || reached_min_packet_old; + assign min_packet_error = (in1.tlast && !reached_min_packet) || min_packet_error_old; + end else begin + assign reached_min_packet = 1'b1; + assign min_packet_error = 1'b0; + end + + always_ff @(posedge eth_rx.clk) begin : dispatch_sm_ff + if (eth_rx.rst) begin + dispatch_state <= ST_IDLE_ETH_L0; + end else begin + if (in1.tvalid && in1.tready) begin + if (in1.tlast) + dispatch_state <= ST_IDLE_ETH_L0; + else + dispatch_state <= next_dispatch_state; + end + end + end + + always_comb begin : dispatch_sm_next_state + //defaults + next_dispatch_state = dispatch_state; + `AXI4S_ASSIGN(in2,in1); + in2.tuser[ERROR_BIT] = mac_error || min_packet_error; + cpu_error = 1'b0; + chdr_error = 1'b0; + + // Statemachine always returns to ST_IDLE_ETH_L0 when tlast is set + case (dispatch_state) + ST_IDLE_ETH_L0: begin + cpu_error = 1'b0; + chdr_error = 1'b0; + if (mac_error || min_packet_error) begin + cpu_error = 1'b1; + chdr_error = 1'b1; + next_dispatch_state = ST_DROP_TERM; + end else if (reached_end_of_udp) begin + // all header values are decoded + if (eth_dst_is_broadcast) begin + // If Eth destination is bcast then fwd to CPU + cpu_error = 1'b0; + chdr_error = 1'b1; + next_dispatch_state = ST_FWD_CPU; + end else if (!eth_dst_is_me && DROP_UNKNOWN_MAC) begin + // If Eth destination is not us then drop the packet + cpu_error = 1'b1; + chdr_error = 1'b1; + next_dispatch_state = ST_DROP_TERM; + end else if (udp_dst_is_me && + ipv4_dst_is_me && + //ip_version_new == IPV4_LEN5 && // NEW CHECK --verify if this is ok + ipv4_protocol_is_udp && + eth_type_is_ipv4) begin + // The conditions matches CHDR port + cpu_error = 1'b1; + chdr_error = 1'b0; + next_dispatch_state = ST_FWD_CHDR; + end else begin + // Not the CHDR port. Forward to CPU + cpu_error = 1'b0; + chdr_error = 1'b1; + next_dispatch_state = ST_FWD_CPU; + end + end + end + + // CHDR Payload + ST_FWD_CHDR: begin + cpu_error = 1'b1; + chdr_error = 1'b0; + if (mac_error || min_packet_error) begin + cpu_error = 1'b1; + chdr_error = 1'b1; + next_dispatch_state = ST_DROP_TERM; + end + end + + // NotCHDR Payload: Send to CPU + ST_FWD_CPU: begin + cpu_error = 1'b0; + chdr_error = 1'b1; + if (mac_error || min_packet_error) begin + cpu_error = 1'b1; + chdr_error = 1'b1; + next_dispatch_state = ST_DROP_TERM; + end + end + + // Unwanted Payload: Drop + ST_DROP_TERM: begin + cpu_error = 1'b1; + chdr_error = 1'b1; + in2.tlast = 1'b1; + in2.tvalid = 1'b1; + in1.tready = in2.tready; + next_dispatch_state = ST_DROP_WAIT; + end + + // Unwanted Payload: wait + ST_DROP_WAIT: begin + cpu_error = 1'b0; + chdr_error = 1'b0; + in2.tlast = 1'b0; + in2.tvalid = 1'b0; + in1.tready = 1'b1; + end + + // We should never get here + default: begin + cpu_error = 1'b0; + chdr_error = 1'b0; + in1.tready = 1'b1; + in2.tvalid = 1'b0; + in2.tlast = 1'b0; + next_dispatch_state = ST_IDLE_ETH_L0; + end + endcase + end + + + //--------------------------------------- + // SPLIT + //--------------------------------------- + always_comb begin : cpu0_assign + cpu0.tdata = in2.tdata; + cpu0.tuser = in2.tuser; + cpu0.tlast = in2.tlast; + cpu0.tvalid = in2.tvalid && chdr0.tready; + cpu0.tuser[ERROR_BIT] = in2.tuser[ERROR_BIT] || cpu_error; + + chdr0.tdata = in2.tdata; + chdr0.tuser = in2.tuser; + chdr0.tlast = in2.tlast; + chdr0.tvalid = in2.tvalid && cpu0.tready; + chdr0_error = in2.tuser[ERROR_BIT] || chdr_error; + + in2.tready = cpu0.tready && chdr0.tready; + end + + //--------------------------------------- + // CPU Output processing + //--------------------------------------- + axi4s_fifo #( + .SIZE(1) + ) out_reg_cpu_i ( + .clear(),.space(),.occupied(), + .i(cpu0),.o(cpu1) + ); + + // We cannot make a CHDR/noCHDR routing decision until we are in the middle + // of a packet so we use a packet gate for the CPU path because we can rewind + // the write pointer and drop the packet in case it's destined for the CHDR + // path. + // NOTE: This also rejects packets with FCS failures. + // NOTE: The SIZE of this FIFO must accommodate a 9000 byte jumbo frame + // regardless of the CHDR MTU + // SIZED for 11 bit address when using a 64 bit word -> 16KByte + // SIZED for 8 bit address when using a 512 bit word -> 16KByte + axi4s_packet_gate #( + .SIZE(17-$clog2(ENET_W)), .USE_AS_BUFF(0) + ) cpu_out_gate_i ( + .clear(1'b0), .error(cpu1.tuser[ERROR_BIT]), + .i(cpu1),.o(cpu2) + ); + + // The CPU can be slow to respond (relative to packet wirespeed) so + // extra buffer for packets destined there so it doesn't back up. + axi4s_fifo #( + .SIZE(CPU_FIFO_SIZE) + ) cpu_fifo_i ( + .clear(),.space(),.occupied(), + .i(cpu2),.o(e2c) + ); + + // CHDR DATA GATE + // SIZED for 11 bit address when using a 64 bit word -> 16KByte + // SIZED for 8 bit address when using a 512 bit word -> 16KByte + axi4s_packet_gate #( + .SIZE(17-$clog2(ENET_W)) + ) chdr_out_gate_i ( + .clear(1'b0),.error(chdr0_error), + .i(chdr0),.o(e2v) + ); + +endmodule // eth_ipv4_chdr_dispatch diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_interface.sv b/fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_interface.sv new file mode 100644 index 000000000..0c9f33f8c --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/eth_ipv4_interface.sv @@ -0,0 +1,228 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// Module: eth_ipv4_interface +// +// Description: +// Adapts from internal CHDR to UDP/IPV4 Ethernet packets. +// Packets not specifically addressed to CHDR are routed +// to the CPU +// +// Parameters: +// - PROTOVER: RFNoC protocol version {8'd, 8'd} +// - MTU: Log2 of the MTU of the packet in 64-bit words +// - CPU_FIFO_SIZE: Log2 of the FIFO depth (in 64-bit words) for the CPU egress path +// - RT_TBL_SIZE: Log2 of the depth of the return-address routing table +// - NODE_INST: The node type to return for a node-info discovery +// - DROP_UNKNOWN_MAC: Drop packets not addressed to us? +// - DROP_MIN_PACKET: Drop packets smaller than 64 bytes? +// - PREAMBLE_BYTES: Number of bytes of Preamble expected +// - ADD_SOF: Add a SOF indication into the tuser field +// - SYNC: Set if MAC is not the same as bus_clk +// - ENET_W: Width of the link to the Ethernet MAC +// - CPU_W: Width of the CPU interface +// - CHDR_W: Width of the CHDR interface +// + +module eth_ipv4_interface #( + logic [15:0] PROTOVER = {8'd1, 8'd0}, + int MTU = 10, + int CPU_FIFO_SIZE = MTU, + int NODE_INST = 0, + int RT_TBL_SIZE = 6, + int REG_AWIDTH = 14, + int BASE = 0, + bit DROP_UNKNOWN_MAC = 0, + bit DROP_MIN_PACKET = 0, + int PREAMBLE_BYTES = 6, + bit ADD_SOF = 1, + bit SYNC = 0, + int ENET_W = 64, + int CPU_W = 64, + int CHDR_W = 64 +) ( + input logic bus_clk, + input logic bus_rst, + input logic [15:0] device_id, + + // Register port: Write port (domain: bus_clk) + input logic reg_wr_req, + input logic [REG_AWIDTH-1:0] reg_wr_addr, + input logic [31:0] reg_wr_data, + + // Register port: Read port (domain: bus_clk) + input logic reg_rd_req, + input logic [REG_AWIDTH-1:0] reg_rd_addr, + output logic reg_rd_resp, + output logic [31:0] reg_rd_data, + + // Status ports (domain: bus_clk) + output logic [47:0] my_mac, + output logic [31:0] my_ip, + output logic [15:0] my_udp_chdr_port, + + // Ethernet MAC + AxiStreamIf.master eth_tx, // tUser = {1'b0,trailing bytes}; + AxiStreamIf.slave eth_rx, // tUser = {error,trailing bytes}; + // CHDR router interface + AxiStreamIf.master e2v, // tUser = {*not used*}; + AxiStreamIf.slave v2e, // tUser = {*not used*}; + // CPU DMA + AxiStreamIf.master e2c, // tUser = {sof,trailing bytes}; + AxiStreamIf.slave c2e // tUser = {1'b0,trailing bytes}; + + ); + + localparam [47:0] DEFAULT_MAC_ADDR = {8'h00, 8'h80, 8'h2f, 8'h16, 8'hc5, 8'h2f}; + localparam [31:0] DEFAULT_IP_ADDR = {8'd192, 8'd168, 8'd10, 8'd2}; + localparam [31:0] DEFAULT_UDP_PORT = 16'd49153; + + //--------------------------------------------------------- + // Registers + //--------------------------------------------------------- + // Include for register offsets + `include "eth_regs.vh" + // Allocate one full page for M + // mac_reg: MAC address for the dispatcher module. This value is used to + // determine if the packet is meant for this device and should be consumed. + // + // ip_reg: IP address for the dispatcher module. This value is used to + // determine if the packet is addressed to this device + // + // This module supports two destination ports. + logic [47:0] mac_reg = DEFAULT_MAC_ADDR; + logic [31:0] ip_reg = DEFAULT_IP_ADDR; + logic [15:0] udp_port = DEFAULT_UDP_PORT; + logic [47:0] bridge_mac_reg = DEFAULT_MAC_ADDR; + logic [31:0] bridge_ip_reg = DEFAULT_IP_ADDR; + logic [15:0] bridge_udp_port = DEFAULT_UDP_PORT; + logic bridge_en; + + always_comb begin : bridge_mux + my_mac = bridge_en ? bridge_mac_reg : mac_reg; + my_ip = bridge_en ? bridge_ip_reg : ip_reg; + my_udp_chdr_port = bridge_en ? bridge_udp_port : udp_port; + end + + always_ff @(posedge bus_clk) begin : reg_wr_ff + if (bus_rst) begin + mac_reg <= DEFAULT_MAC_ADDR; + ip_reg <= DEFAULT_IP_ADDR; + udp_port <= DEFAULT_UDP_PORT; + bridge_en <= 1'b0; + bridge_mac_reg <= DEFAULT_MAC_ADDR; + bridge_ip_reg <= DEFAULT_IP_ADDR; + bridge_udp_port <= DEFAULT_UDP_PORT; + end + else begin + if (reg_wr_req) + case (reg_wr_addr) + + REG_MAC_LSB: + mac_reg[31:0] <= reg_wr_data; + + REG_MAC_MSB: + mac_reg[47:32] <= reg_wr_data[15:0]; + + REG_IP: + ip_reg <= reg_wr_data; + + REG_UDP: + udp_port <= reg_wr_data[15:0]; + + REG_BRIDGE_MAC_LSB: + bridge_mac_reg[31:0] <= reg_wr_data; + + REG_BRIDGE_MAC_MSB: + bridge_mac_reg[47:32] <= reg_wr_data[15:0]; + + REG_BRIDGE_IP: + bridge_ip_reg <= reg_wr_data; + + REG_BRIDGE_UDP: + bridge_udp_port <= reg_wr_data[15:0]; + + REG_BRIDGE_ENABLE: + bridge_en <= reg_wr_data[0]; + endcase + end + end + + always_ff @ (posedge bus_clk) begin : reg_rd_ff + if (bus_rst) begin + reg_rd_resp <= 1'b0; + reg_rd_data <= 32'd0; + end + else begin + if (reg_rd_req) begin + // Assert read response one cycle after read request + reg_rd_resp <= 1'b1; + case (reg_rd_addr) + REG_MAC_LSB: + reg_rd_data <= mac_reg[31:0]; + + REG_MAC_MSB: + reg_rd_data <= {16'b0,mac_reg[47:32]}; + + REG_IP: + reg_rd_data <= ip_reg; + + REG_UDP: + reg_rd_data <= {16'b0, udp_port}; + + REG_BRIDGE_MAC_LSB: + reg_rd_data <= bridge_mac_reg[31:0]; + + REG_BRIDGE_MAC_MSB: + reg_rd_data <= {16'b0,bridge_mac_reg[47:32]}; + + REG_BRIDGE_IP: + reg_rd_data <= bridge_ip_reg; + + REG_BRIDGE_UDP: + reg_rd_data <= {16'b0, bridge_udp_port}; + + REG_BRIDGE_ENABLE: + reg_rd_data <= {31'b0,bridge_en}; + + default: + reg_rd_resp <= 1'b0; + endcase + end + // Deassert read response after one clock cycle + if (reg_rd_resp) begin + reg_rd_resp <= 1'b0; + end + end + end + + eth_ipv4_chdr_adapter #( + .PROTOVER (PROTOVER), + .MTU (MTU), + .CPU_FIFO_SIZE (CPU_FIFO_SIZE), + .RT_TBL_SIZE (RT_TBL_SIZE), + .NODE_INST (NODE_INST), + .DROP_UNKNOWN_MAC(DROP_UNKNOWN_MAC), + .DROP_MIN_PACKET (DROP_MIN_PACKET), + .PREAMBLE_BYTES (PREAMBLE_BYTES), + .ADD_SOF (ADD_SOF), + .SYNC (SYNC), + .ENET_W (ENET_W), + .CPU_W (CPU_W), + .CHDR_W (CHDR_W) + ) eth_adapter_i ( + .eth_rx (eth_rx ), + .eth_tx (eth_tx ), + .v2e (v2e ), + .e2v (e2v ), + .c2e (c2e ), + .e2c (e2c ), + .device_id (device_id), + .my_mac (my_mac ), + .my_ip (my_ip ), + .my_udp_chdr_port(my_udp_chdr_port ) + ); + + +endmodule : eth_ipv4_interface diff --git a/fpga/usrp3/lib/rfnoc/xport_sv/eth_regs.vh b/fpga/usrp3/lib/rfnoc/xport_sv/eth_regs.vh new file mode 100644 index 000000000..354a19414 --- /dev/null +++ b/fpga/usrp3/lib/rfnoc/xport_sv/eth_regs.vh @@ -0,0 +1,29 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: eth_constants (Header File) +// +// Description: +// Holds register offsets for the Ethernet Interface +// NOTE: These depend on the following parameters +// Parameters: +// REG_AWIDTH - How wide the register window is in bits +// BASE - Base address added to the offsets here +// REGISTER OFFSETS +// Allocate one full page for MAC +localparam [REG_AWIDTH-1:0] REG_MAC_LSB = BASE + 'h0000; +localparam [REG_AWIDTH-1:0] REG_MAC_MSB = BASE + 'h0004; + +// Source IP address +localparam [REG_AWIDTH-1:0] REG_IP = BASE + 'h1000; +// Source UDP Port +localparam [REG_AWIDTH-1:0] REG_UDP = BASE + 'h1004; + +// Registers for Internal/Bridge Network Mode in CPU +localparam [REG_AWIDTH-1:0] REG_BRIDGE_MAC_LSB = BASE + 'h1010; +localparam [REG_AWIDTH-1:0] REG_BRIDGE_MAC_MSB = BASE + 'h1014; +localparam [REG_AWIDTH-1:0] REG_BRIDGE_IP = BASE + 'h1018; +localparam [REG_AWIDTH-1:0] REG_BRIDGE_UDP = BASE + 'h101c; +localparam [REG_AWIDTH-1:0] REG_BRIDGE_ENABLE = BASE + 'h1020; diff --git a/fpga/usrp3/tools/utils/testbenches.excludes b/fpga/usrp3/tools/utils/testbenches.excludes index 79c3bb557..771d8bff2 100644 --- a/fpga/usrp3/tools/utils/testbenches.excludes +++ b/fpga/usrp3/tools/utils/testbenches.excludes @@ -17,3 +17,4 @@ top/n3xx/dboards/eiscat/radio/noc_block_radio_core_eiscat_tb # These testbenches only work in ModelSim lib/axi4s_sv/axi4s_remove_bytes_tb lib/axi4s_sv/axi4s_add_bytes_tb +lib/rfnoc/xport_sv/eth_interface_tb -- cgit v1.2.3