aboutsummaryrefslogtreecommitdiffstats
path: root/fpga/usrp3/sim
diff options
context:
space:
mode:
authorAndrew Moch <Andrew.Moch@ni.com>2020-06-22 17:13:36 +0100
committerWade Fife <wade.fife@ettus.com>2020-06-25 14:44:04 -0500
commitc3bca6c87700054c96320de119a58f6a688dbd5a (patch)
tree5caa6aca23b6ea84335ea22398698aa1ea3569ea /fpga/usrp3/sim
parent8661f46df66329511ccb3cf083830e19d65ea402 (diff)
downloaduhd-c3bca6c87700054c96320de119a58f6a688dbd5a.tar.gz
uhd-c3bca6c87700054c96320de119a58f6a688dbd5a.tar.bz2
uhd-c3bca6c87700054c96320de119a58f6a688dbd5a.zip
fpga: lib: Add synthesizable AXI4-Stream SV components
Components are connected together with AxiStreamIfc. Some features include: (1) Add bytes to the start of a packet (2) Remove bytes from a packet (3) Wrappers for some older components a. fifo - buffer but imediately pass a packet b. packet_gate - buffer and hold till end of packet c. width_conv - cross clock domains and change width of axi bus The AxiStreamIf was moved from PkgAxiStreamBfm to its own file. It can be used to connect to ports with continuous assignment. AxiStreamPacketIf must be used procedurally but allows the following new methods: - reached_packet_byte - notify when tdata contains a paritcular byte - get_packet_byte/get_packet_field - extract a byte or field from axi - put_packet_byte/put_packet_field - overwrite a byte or field onto axi
Diffstat (limited to 'fpga/usrp3/sim')
-rw-r--r--fpga/usrp3/sim/rfnoc/Makefile.srcs2
-rw-r--r--fpga/usrp3/sim/rfnoc/PkgAxiStreamBfm.sv239
-rw-r--r--fpga/usrp3/sim/rfnoc/PkgAxisCtrlBfm.sv18
-rw-r--r--fpga/usrp3/sim/rfnoc/PkgEthernet.sv809
4 files changed, 990 insertions, 78 deletions
diff --git a/fpga/usrp3/sim/rfnoc/Makefile.srcs b/fpga/usrp3/sim/rfnoc/Makefile.srcs
index d1b59c8e2..31196678f 100644
--- a/fpga/usrp3/sim/rfnoc/Makefile.srcs
+++ b/fpga/usrp3/sim/rfnoc/Makefile.srcs
@@ -11,6 +11,7 @@
SIM_RFNOC_SRCS = $(abspath $(addprefix $(BASE_DIR)/../lib/, \
axi4lite_sv/PkgAxiLite.sv \
axi4lite_sv/AxiLiteIf.sv \
+axi4s_sv/AxiStreamIf.sv \
))
##################################################
@@ -23,6 +24,7 @@ test_exec.svh \
sim_clock_gen.sv \
PkgAxiStreamBfm.sv \
PkgAxiLiteBfm.sv \
+PkgEthernet.sv \
PkgChdrData.sv \
PkgChdrUtils.sv \
PkgChdrBfm.sv \
diff --git a/fpga/usrp3/sim/rfnoc/PkgAxiStreamBfm.sv b/fpga/usrp3/sim/rfnoc/PkgAxiStreamBfm.sv
index 9e2530350..fd23e501c 100644
--- a/fpga/usrp3/sim/rfnoc/PkgAxiStreamBfm.sv
+++ b/fpga/usrp3/sim/rfnoc/PkgAxiStreamBfm.sv
@@ -1,61 +1,79 @@
//
-// Copyright 2019 Ettus Research, A National Instruments Company
+// Copyright 2020 Ettus Research, A National Instruments Brand
//
// SPDX-License-Identifier: LGPL-3.0-or-later
//
-// Module: PkgAxiStream
+// Module: PkgAxiStreamBfm
//
-// Description: Package for a bi-directional AXI Stream bus functional model
-// (BFM). This consists of the AxiStreamIf interface, and the
-// AxiStreamPacket and AxiStreamBfm classes.
+// Description: Package for a bi-directional AXI Stream bus functional model
+// (BFM). This consists of the AxiStreamPacket and AxiStreamBfm classes.
+// It's based on the AxiStreamIf in lib/axi4s_svPkgAXI4S.sv
//
-
//-----------------------------------------------------------------------------
-// Unidirectional AXI4-Stream interface
+// AXI-Stream BFM Package
//-----------------------------------------------------------------------------
-interface AxiStreamIf #(
- parameter int DATA_WIDTH = 64,
- parameter int USER_WIDTH = 1
-) (
- input logic clk,
- input logic rst = 1'b0
-);
-
- // Signals that make up a unidirectional AXI-Stream interface
- logic tready;
- logic tvalid;
- logic [DATA_WIDTH-1:0] tdata;
- logic [USER_WIDTH-1:0] tuser;
- logic [DATA_WIDTH/8-1:0] tkeep;
- logic tlast;
-
- // View from the master side
- modport master (
- input clk, rst,
- output tvalid, tdata, tuser, tkeep, tlast,
- input tready
- );
-
- // View from the slave side
- modport slave (
- input clk, rst,
- input tvalid, tdata, tuser, tkeep, tlast,
- output tready
- );
-
-endinterface : AxiStreamIf
+package PkgAxiStreamBfm;
+ //---------------------------------------------------------------------------
+ // Raw packets - packet of just bytes
+ //---------------------------------------------------------------------------
+ // Ethernet in particular defies normal word boundaries, so underneath treating
+ // it as the most fundamental quantity - Bytes.
+ typedef byte raw_pkt_t[$]; // Byte Queue
+
+ // Push a packet with random data onto to the AXI Stream bus
+ // Args:
+ // - num_bytes: number of random bytes to add to the raw packet.
+ // - pkt: packet with rand data
+ task automatic get_rand_raw_pkt (
+ input int num_bytes,
+ output raw_pkt_t pkt);
+ begin
+ repeat(num_bytes) begin
+ pkt.push_back($urandom);
+ end
+ end
+ endtask
+
+ // Push a packet with a ramp on to the AXI Stream bus
+ // Args:
+ // - num_samps: Packet size in bytes *8
+ // - ramp_start: Start value for the ramp
+ // - ramp_inc: Increment per clock cycle
+ task automatic get_ramp_raw_pkt (
+ input int num_samps,
+ input logic [63:0] ramp_start,
+ input logic [63:0] ramp_inc,
+ input int SWIDTH=64,
+ output raw_pkt_t pkt);
+ begin
+ logic[63:0] word;
+ automatic integer counter = 0;
+ repeat(num_samps) begin
+ word = ramp_start+(counter*ramp_inc);
+ for (int i=0; i < SWIDTH ; i+=8) begin
+ pkt.push_back(word[i +: 8]);
+ end
+ counter = counter + 1;
+ end
+ end
+ endtask
-//-----------------------------------------------------------------------------
-// AXI-Stream BFM Package
-//-----------------------------------------------------------------------------
+ // Comparison Functions
+ function automatic bit raw_pkt_compare(input raw_pkt_t a, input raw_pkt_t b);
-package PkgAxiStreamBfm;
+ bit queue_match;
+ queue_match = 1;
+ // check each element of the queue and clear queue_match if they don't match.
+ // workaround for vivado bug - could be a==b
+ foreach(a[i]) queue_match = queue_match && a[i] == b[i];
+ return ((a.size() == b.size()) && queue_match);
+ endfunction
//---------------------------------------------------------------------------
// AXI Stream Packet Class
@@ -73,7 +91,7 @@ package PkgAxiStreamBfm;
typedef AxiStreamPacket #(DATA_WIDTH, USER_WIDTH) AxisPacket_t;
-
+ bit verbose=0;
//------------
// Properties
//------------
@@ -105,6 +123,12 @@ package PkgAxiStreamBfm;
keep = {};
endfunction;
+ // Delete a word from the current packet
+ function void delete(int i);
+ data.delete(i);
+ user.delete(i);
+ keep.delete(i);
+ endfunction;
// Return true if this packet equals that of the argument
virtual function bit equal(AxisPacket_t packet);
@@ -117,21 +141,30 @@ package PkgAxiStreamBfm;
foreach (data[i]) begin
data_a = data[i];
data_b = packet.data[i];
- if (data_a !== data_b) return 0;
+ if (data_a !== data_b) begin
+ if (verbose) $display("AxisPacket data mismatch a[%2d]=%X b[%2d]=%X",i,data_a,i,data_b);
+ return 0;
+ end
end
if (user.size() != packet.user.size()) return 0;
foreach (data[i]) begin
user_a = user[i];
user_b = packet.user[i];
- if (user_a !== user_b) return 0;
+ if (user_a !== user_b) begin
+ if (verbose) $display("AxisPacket user mismatch a[%2d]=%X b[%2d]=%X",i,user_a,i,user_b);
+ return 0;
+ end
end
if (keep.size() != packet.keep.size()) return 0;
foreach (keep[i]) begin
keep_a = keep[i];
keep_b = packet.keep[i];
- if (keep_a !== keep_b) return 0;
+ if (keep_a !== keep_b) begin
+ if (verbose) $display("AxisPacket keep mismatch a[%2d]=%X b[%2d]=%X",i,user_a,i,user_b);
+ return 0;
+ end
end
return 1;
@@ -141,10 +174,22 @@ package PkgAxiStreamBfm;
// Format the contents of the packet into a string
function string sprint();
string str = "";
+ string data_str = "";
if (data.size() == user.size() && data.size() == keep.size()) begin
str = { str, "data, user, keep:\n" };
foreach (data[i]) begin
- str = { str, $sformatf("%5d> %X %X %b\n", i, data[i], user[i], keep[i]) };
+ data_str = "";
+ if (DATA_WIDTH > 64) begin
+ for (int b=0; b < DATA_WIDTH; b +=64) begin
+ data_str = { data_str, $sformatf("%3d: %X",b,data[i][b+:64])};
+ if (b+64 < DATA_WIDTH) begin
+ data_str = { data_str, $sformatf("\n ")};
+ end
+ end
+ end else begin
+ data_str = { data_str, $sformatf("%X",data[i]) };
+ end
+ str = { str, $sformatf("%5d> %s %X %X \n", i, data_str, user[i], keep[i]) };
end
end else begin
str = { str, "data:\n" };
@@ -169,6 +214,45 @@ package PkgAxiStreamBfm;
$display(sprint());
endfunction : print
+ // Add an array of bytes (little endian)
+ function void push_bytes(raw_pkt_t raw, input user_t user = '0);
+ data_t word;
+ keep_t my_keep;
+ while (raw.size() > 0) begin
+ // fill tkeep
+ // SIZE = TKEEP / 0 = 0000, 1 = 0001, 2 = 0011, etc
+ my_keep = '1;
+ if (raw.size <= DATA_WIDTH/8) begin
+ foreach (my_keep[i]) my_keep[i] = i < (raw.size);
+ end
+ // fill the word with raw data from bottom up
+ word = '0;
+ for (int i = 0; i < DATA_WIDTH/8 ; i++) begin
+ if (my_keep[i]) word[i*8 +: 8] = raw.pop_front();
+ end
+ this.data.push_back(word);
+ this.keep.push_back(my_keep);
+ this.user.push_back(user);
+ end
+ endfunction
+
+ // Dump data contents as an array of bytes. (little endian)
+ function raw_pkt_t dump_bytes();
+ data_t word;
+ keep_t my_keep;
+ raw_pkt_t raw;
+ assert (data.size == keep.size) else
+ $fatal("data and keep have different sizes!");
+ foreach (data[i]) begin
+ my_keep = this.keep[i];
+ word = this.data[i];
+ for (int j = 0; j < DATA_WIDTH/8 ; j++) begin
+ if (my_keep[j]) raw.push_back(word[j*8 +: 8]);
+ end;
+ end
+ return raw;
+ endfunction
+
endclass : AxiStreamPacket;
@@ -178,8 +262,13 @@ package PkgAxiStreamBfm;
//---------------------------------------------------------------------------
class AxiStreamBfm #(
- parameter int DATA_WIDTH = 64,
- parameter int USER_WIDTH = 1
+ int DATA_WIDTH = 64,
+ int USER_WIDTH = 1,
+ int MAX_PACKET_BYTES = 0,
+ bit TDATA = 1,
+ bit TUSER = 1,
+ bit TKEEP = 1,
+ bit TLAST = 1
);
//------------------
@@ -205,13 +294,16 @@ package PkgAxiStreamBfm;
local const AxisPacket_t::keep_t IDLE_KEEP = {(DATA_WIDTH/8){1'bX}};
// Virtual interfaces for master and slave connections to DUT
- local virtual AxiStreamIf #(DATA_WIDTH, USER_WIDTH).master master;
- local virtual AxiStreamIf #(DATA_WIDTH, USER_WIDTH).slave slave;
+ local virtual AxiStreamIf #(DATA_WIDTH,USER_WIDTH,MAX_PACKET_BYTES,
+ TDATA,TUSER,TKEEP,TLAST).master master;
+ local virtual AxiStreamIf #(DATA_WIDTH,USER_WIDTH,MAX_PACKET_BYTES,
+ TDATA,TUSER,TKEEP,TLAST).slave slave;
// NOTE: We should not need these flags if Vivado would be OK with null check
// without throwing unnecessary null-ptr deref exceptions.
local bit master_en;
local bit slave_en;
+ bit slave_tready_init = 0;
// Queues to store the bus transactions
mailbox #(AxisPacket_t) tx_packets;
mailbox #(AxisPacket_t) rx_packets;
@@ -220,6 +312,8 @@ package PkgAxiStreamBfm;
protected int master_stall_prob = DEF_STALL_PROB;
protected int slave_stall_prob = DEF_STALL_PROB;
+ // Number of clocks betwen packets
+ int inter_packet_gap = 0;
//---------
// Methods
@@ -231,11 +325,13 @@ package PkgAxiStreamBfm;
endfunction : packets_equal
- // Class constructor. This must be given an interface for the master
+ // Class constructor. This must be given an interface for the master
// connection and an interface for the slave connection.
function new(
- virtual AxiStreamIf #(DATA_WIDTH, USER_WIDTH).master master,
- virtual AxiStreamIf #(DATA_WIDTH, USER_WIDTH).slave slave
+ virtual AxiStreamIf #(DATA_WIDTH,USER_WIDTH,MAX_PACKET_BYTES,
+ TDATA,TUSER,TKEEP,TLAST).master master,
+ virtual AxiStreamIf #(DATA_WIDTH,USER_WIDTH,MAX_PACKET_BYTES,
+ TDATA,TUSER,TKEEP,TLAST).slave slave
);
this.master_en = (master != null);
this.slave_en = (slave != null);
@@ -253,7 +349,7 @@ package PkgAxiStreamBfm;
endtask : put
- // Attempt to queue the provided packet for transmission. Return 1 if
+ // Attempt to queue the provided packet for transmission. Return 1 if
// successful, return 0 if the queue is full.
function bit try_put(AxisPacket_t packet);
assert (master_en) else $fatal(1, "Cannot use TX operations for a null master");
@@ -268,7 +364,7 @@ package PkgAxiStreamBfm;
endtask : get
- // Get the next packet if there's one available and return 1. Return 0 if
+ // Get the next packet if there's one available and return 1. Return 0 if
// there's no packet available.
function bit try_get(output AxisPacket_t packet);
assert (slave_en) else $fatal(1, "Cannot use RX operations for a null slave");
@@ -276,7 +372,7 @@ package PkgAxiStreamBfm;
endfunction : try_get
- // Get the next packet when it becomes available (wait if necessary), but
+ // Get the next packet when it becomes available (wait if necessary), but
// don't remove it from the receive queue.
task peek(output AxisPacket_t packet);
assert (slave_en) else $fatal(1, "Cannot use RX operations for a null slave");
@@ -284,8 +380,8 @@ package PkgAxiStreamBfm;
endtask : peek
- // Get the next packet if there's one available and return 1, but don't
- // remove it from the receive queue. Return 0 if there's no packet
+ // Get the next packet if there's one available and return 1, but don't
+ // remove it from the receive queue. Return 0 if there's no packet
// available.
function bit try_peek(output AxisPacket_t packet);
assert (slave_en) else $fatal(1, "Cannot use RX operations for a null slave");
@@ -300,8 +396,8 @@ package PkgAxiStreamBfm;
endfunction
- // Wait until num packets have started transmission (i.e., until num
- // packets have been dequeued). Set num = -1 to wait until all currently
+ // Wait until num packets have started transmission (i.e., until num
+ // packets have been dequeued). Set num = -1 to wait until all currently
// queued packets have started transmission.
task wait_send(int num = -1);
int end_num;
@@ -318,7 +414,7 @@ package PkgAxiStreamBfm;
endtask : wait_send
- // Wait until num packets have completed transmission. Set num = -1 to wait
+ // Wait until num packets have completed transmission. Set num = -1 to wait
// for all currently queued packets to complete transmission.
task wait_complete(int num = -1);
int end_num;
@@ -340,7 +436,7 @@ package PkgAxiStreamBfm;
endtask : wait_complete
- // Set the probability (as a percentage, 0 to 100) of the master interface
+ // Set the probability (as a percentage, 0 to 100) of the master interface
// stalling due to lack of data to send.
function void set_master_stall_prob(int stall_probability = DEF_STALL_PROB);
assert(stall_probability >= 0 && stall_probability <= 100) else begin
@@ -350,7 +446,7 @@ package PkgAxiStreamBfm;
endfunction
- // Set the probability (as a percentage, 0 to 100) of the slave interface
+ // Set the probability (as a percentage, 0 to 100) of the slave interface
// stalling due to lack of buffer space.
function void set_slave_stall_prob(int stall_probability = DEF_STALL_PROB);
assert(stall_probability >= 0 && stall_probability <= 100) else begin
@@ -360,14 +456,14 @@ package PkgAxiStreamBfm;
endfunction
- // Get the probability (as a percentage, 0 to 100) of the master interface
+ // Get the probability (as a percentage, 0 to 100) of the master interface
// stalling due to lack of data to send.
function int get_master_stall_prob(int stall_probability = DEF_STALL_PROB);
return master_stall_prob;
endfunction
- // Get the probability (as a percentage, 0 to 100) of the slave interface
+ // Get the probability (as a percentage, 0 to 100) of the slave interface
// stalling due to lack of buffer space.
function int get_slave_stall_prob(int stall_probability = DEF_STALL_PROB);
return slave_stall_prob;
@@ -397,8 +493,10 @@ package PkgAxiStreamBfm;
master.tlast <= 0;
forever begin
- @(posedge master.clk);
- if (master.rst) continue;
+ repeat(inter_packet_gap) begin
+ @(posedge master.clk);
+ if (master.rst) continue;
+ end
if (tx_packets.try_get(packet)) begin
foreach (packet.data[i]) begin
@@ -433,6 +531,9 @@ package PkgAxiStreamBfm;
master.tuser <= IDLE_USER;
master.tkeep <= IDLE_KEEP;
master.tlast <= 0;
+ end else begin
+ @(posedge master.clk);
+ if (master.rst) continue;
end
end
endtask : master_body
@@ -445,7 +546,7 @@ package PkgAxiStreamBfm;
local task slave_body();
AxisPacket_t packet = new();
- slave.tready <= 0;
+ slave.tready <= slave_tready_init;
forever begin
@(posedge slave.clk);
diff --git a/fpga/usrp3/sim/rfnoc/PkgAxisCtrlBfm.sv b/fpga/usrp3/sim/rfnoc/PkgAxisCtrlBfm.sv
index 148ef5488..cc4c9ac18 100644
--- a/fpga/usrp3/sim/rfnoc/PkgAxisCtrlBfm.sv
+++ b/fpga/usrp3/sim/rfnoc/PkgAxisCtrlBfm.sv
@@ -5,7 +5,7 @@
//
// Module: PkgAxisCtrlBfm
//
-// Description: Package for an AXIS-Ctrl bus functional model (BFM), which
+// Description: Package for an AXIS-Ctrl bus functional model (BFM), which
// consists primarily of the AxisCtrlPacket and AxisCtrlBfm classes.
//
@@ -129,7 +129,7 @@ package PkgAxisCtrlBfm;
// AXIS-Ctrl BFM Methods
//---------------------------------------------------------------------------
- // Class constructor. This must be given an interface for the master
+ // Class constructor. This must be given an interface for the master
// connection and an interface for the slave connection.
function AxisCtrlBfm::new (
virtual AxiStreamIf #(32).master master,
@@ -147,7 +147,7 @@ package PkgAxisCtrlBfm;
endtask : put_ctrl
- // Attempt to queue the provided packet for transmission. Return 1 if
+ // Attempt to queue the provided packet for transmission. Return 1 if
// successful, return 0 if the queue is full.
task AxisCtrlBfm::try_put_ctrl(AxisCtrlPacket ctrl_packet);
AxisPacket_t axis_packet;
@@ -163,7 +163,7 @@ package PkgAxisCtrlBfm;
endtask : get_ctrl
- // Get the next packet if there's one available and return 1. Return 0 if
+ // Get the next packet if there's one available and return 1. Return 0 if
// there's no packet available.
function bit AxisCtrlBfm::try_get_ctrl(output AxisCtrlPacket ctrl_packet);
AxisPacket_t axis_packet;
@@ -173,7 +173,7 @@ package PkgAxisCtrlBfm;
endfunction : try_get_ctrl
- // Convert an AXIS-Ctrl packet data structure to an AXI-Stream packet data
+ // Convert an AXIS-Ctrl packet data structure to an AXI-Stream packet data
// structure.
function AxisCtrlBfm::AxisPacket_t AxisCtrlBfm::axis_ctrl_to_axis(AxisCtrlPacket ctrl_packet);
AxisPacket_t axis_packet = new();
@@ -184,7 +184,7 @@ package PkgAxisCtrlBfm;
axis_packet.data.push_back(ctrl_packet.header[63:32]);
// Insert timestamp if has_time is set (words 2 and 3)
- if (ctrl_packet.header.has_time) begin
+ if (ctrl_packet.header.has_time) begin
axis_packet.data.push_back(ctrl_packet.timestamp[31: 0]);
axis_packet.data.push_back(ctrl_packet.timestamp[63:32]);
end
@@ -201,12 +201,12 @@ package PkgAxisCtrlBfm;
endfunction : axis_ctrl_to_axis
- // Convert an AXI-Stream packet data structure to an AXIS-Ctrl packet data
+ // Convert an AXI-Stream packet data structure to an AXIS-Ctrl packet data
// structure.
function AxisCtrlPacket AxisCtrlBfm::axis_to_axis_ctrl(AxisPacket_t axis_packet);
AxisCtrlPacket ctrl_packet = new();
int i; // Use an index instead of pop_front() to workaround a ModelSim bug
-
+
// Grab words 0 and 1 (header)
ctrl_packet.header[31: 0] = axis_packet.data[0];
ctrl_packet.header[63:32] = axis_packet.data[1];
@@ -224,7 +224,7 @@ package PkgAxisCtrlBfm;
// Grab data
ctrl_packet.data = axis_packet.data[(i+1):$];
-
+
return ctrl_packet;
endfunction : axis_to_axis_ctrl
diff --git a/fpga/usrp3/sim/rfnoc/PkgEthernet.sv b/fpga/usrp3/sim/rfnoc/PkgEthernet.sv
new file mode 100644
index 000000000..a971b5c7e
--- /dev/null
+++ b/fpga/usrp3/sim/rfnoc/PkgEthernet.sv
@@ -0,0 +1,809 @@
+//
+// Copyright 2020 Ettus Research, A National Instruments Brand
+//
+// SPDX-License-Identifier: LGPL-3.0-or-later
+//
+// Module: PkgEthernet
+//
+// Description: This package defines the data types used to represent ETHER,
+// IPV4, and UDP. It's based on a queue of bytes representation named
+// raw_pkt_t.
+//
+
+package PkgEthernet;
+
+ import PkgAxiStreamBfm::*;
+ export PkgAxiStreamBfm::*;
+
+ //************************************************************//
+ //////////////////// ETHER PACKETS /////////////////////////////
+ //************************************************************//
+
+ // Ether type - subset of possibilities. Add more as needed.
+ typedef enum logic [15:0] {
+ // Byte order 1 0
+ IPV4=16'h0800, IPV6=16'h86_DD, VLAN_TAGGED=16'h8100,
+ DOUBLE_VLAN_TAGGED=16'h9100, ROCE=16'h8915
+ } ether_type_t;
+
+ // Some MAC addresses
+ // Byte order 5 4 3 2 1 0
+ localparam logic [47:0] DEF_DEST_MAC_ADDR = 48'h5E_35_EB_71_46_F7;
+ localparam logic [47:0] DEF_SRC_MAC_ADDR = 48'h98_03_9B_8E_09_B9;
+ localparam logic [47:0] DEF_BRIDGE_MAC_ADDR = 48'hBB_BB_BB_BB_BB_BB;
+
+ // Ether Header
+ typedef struct {
+ logic [47:0] dest_mac = DEF_DEST_MAC_ADDR;
+ logic [47:0] src_mac = DEF_SRC_MAC_ADDR;
+ ether_type_t ether_type = IPV4;
+ } eth_hdr_t;
+
+ // Ethernet Packet, Header + Payload
+ typedef struct {
+ eth_hdr_t hdr;
+ raw_pkt_t payload;
+ logic [31:0] fcs;
+ int ipg = 12; // interpacket gap
+ } eth_pkt_t;
+
+ // Break an eth_pkt into a queue of bytes
+ function automatic raw_pkt_t flatten_eth_pkt(input eth_pkt_t pkt);
+ raw_pkt_t pay;
+
+ pay.push_back(pkt.hdr.dest_mac[47:40]);
+ pay.push_back(pkt.hdr.dest_mac[39:32]);
+ pay.push_back(pkt.hdr.dest_mac[31:24]);
+ pay.push_back(pkt.hdr.dest_mac[23:16]);
+ pay.push_back(pkt.hdr.dest_mac[15:8]);
+ pay.push_back(pkt.hdr.dest_mac[7:0]);
+ pay.push_back(pkt.hdr.src_mac[47:40]);
+ pay.push_back(pkt.hdr.src_mac[39:32]);
+ pay.push_back(pkt.hdr.src_mac[31:24]);
+ pay.push_back(pkt.hdr.src_mac[23:16]);
+ pay.push_back(pkt.hdr.src_mac[15:8]);
+ pay.push_back(pkt.hdr.src_mac[7:0]);
+ pay.push_back(pkt.hdr.ether_type[15:8]);
+ pay.push_back(pkt.hdr.ether_type[7:0]);
+ pay = {pay,pkt.payload};
+
+ return pay;
+
+ endfunction
+
+ // Break a queue of bytes into a eth_pkt
+ function automatic eth_pkt_t unflatten_eth_pkt(input raw_pkt_t pay);
+ eth_pkt_t pkt;
+
+ pkt.hdr.dest_mac[47:40] = pay.pop_front();
+ pkt.hdr.dest_mac[39:32] = pay.pop_front();
+ pkt.hdr.dest_mac[31:24] = pay.pop_front();
+ pkt.hdr.dest_mac[23:16] = pay.pop_front();
+ pkt.hdr.dest_mac[15:8] = pay.pop_front();
+ pkt.hdr.dest_mac[7:0] = pay.pop_front();
+ pkt.hdr.src_mac[47:40] = pay.pop_front();
+ pkt.hdr.src_mac[39:32] = pay.pop_front();
+ pkt.hdr.src_mac[31:24] = pay.pop_front();
+ pkt.hdr.src_mac[23:16] = pay.pop_front();
+ pkt.hdr.src_mac[15:8] = pay.pop_front();
+ pkt.hdr.src_mac[7:0] = pay.pop_front();
+ pkt.hdr.ether_type[15:8] = pay.pop_front();
+ pkt.hdr.ether_type[7:0] = pay.pop_front();
+ pkt.payload = pay;
+
+ return pkt;
+
+ endfunction
+
+ function automatic logic eth_pkt_compare(input eth_pkt_t a, input eth_pkt_t b);
+
+ return ((a.hdr.dest_mac == b.hdr.dest_mac) &&
+ (a.hdr.src_mac == b.hdr.src_mac) &&
+ (a.hdr.ether_type == b.hdr.ether_type) &&
+ raw_pkt_compare(a.payload,b.payload));
+
+ endfunction
+
+ //************************************************************//
+ //////////////////// IPV4 PACKETS //////////////////////////////
+ //************************************************************//
+
+ // IP Protocol - subset of possibilities. add more as needed
+ typedef enum logic [7:0] {
+ UDP=8'd17, TCP=8'd6, ICMP=8'd1, IGMP=8'd2, ENCAP=8'd41
+ } ip_protocol_t;
+
+ // follow normal convention of an IP address
+ function automatic logic [31:0] ip(logic [7:0] a,b,c,d);
+ return {a,b,c,d};
+ endfunction
+
+
+ localparam logic [31:0] DEF_DEST_IP_ADDR = ip(192,168,10,2);
+ localparam logic [31:0] DEF_SRC_IP_ADDR = ip(192,168,10,1);
+ localparam logic [31:0] DEF_BRIDGE_IP_ADDR = 32'h33_33_33_33;
+
+ // IPv4 Header
+ typedef struct {
+ logic [3:0] header_length = 4'd5;
+ logic [3:0] version = 4'd4;
+ logic [5:0] dscp = 6'b0000_00;
+ logic [1:0] ecn = 2'b00;
+ logic [15:0] length = 16'hXXXX; //flag for (fill it in please)
+ logic [15:0] identification = 16'h462E;
+ logic rsv_zero = 1'b0;
+ logic dont_frag = 1'b1;
+ logic more_frag = 1'b0;
+ logic [12:0] frag_offset = 16'd0;
+ logic [7:0] time_to_live = 16'd64;
+ ip_protocol_t protocol = UDP;
+ logic [15:0] checksum = 16'hXXXX; //flag for (fill it in please)
+ logic [31:0] src_ip = DEF_SRC_IP_ADDR;
+ logic [31:0] dest_ip = DEF_DEST_IP_ADDR;
+ } ipv4_hdr_t;
+
+ // IP Packet, Header + Payload
+ typedef struct {
+ ipv4_hdr_t hdr;
+ raw_pkt_t payload;
+ } ipv4_pkt_t;
+
+ // The checksum for an IP header is the sum of all the 16 bit words that
+ // make up the header with the checksum set to zero. Add back the carry over
+ // from bits [31:16] then invert.
+ // See https://en.wikipedia.org/wiki/IPv4_header_checksum
+ function automatic logic [15:0] calc_ipv4_checksum(input raw_pkt_t pkt);
+
+ // This is a bit oversized, but it's not costing anything.
+ // 10 max sized words can at most add logbase2 of 10 bits.
+ logic [31:0] checksum;
+
+ checksum = 0;
+ // Iterate over 16 bit chunks reading from a byte addressed memory.
+ // There are 20 bytes in an ipv4 header.
+ for (int i = 0 ; i < 20 ; i+=2 ) begin
+ // BIG endian network ordering... so weird
+ checksum += {pkt[i],pkt[i+1]};
+ end
+ checksum += checksum[31:16];
+ checksum = ~checksum;
+
+ return checksum[15:0];
+
+ endfunction
+
+ // Break an eth_pkt into a queue of bytes
+ function automatic raw_pkt_t flatten_ipv4_pkt(input ipv4_pkt_t pkt);
+ raw_pkt_t pay;
+
+ logic [15:0] length;
+ logic [15:0] checksum;
+ logic [2:0] flags;
+
+ // If header or length is not set to default value then use the value in
+ // the packet.
+ if ($isunknown(pkt.hdr.length))
+ length = pkt.payload.size()+20; // 20 because length includes IP header length.
+ else
+ length = pkt.hdr.length;
+
+ flags = {pkt.hdr.more_frag,pkt.hdr.dont_frag,pkt.hdr.rsv_zero};
+ // Start off with checksum as 0
+ checksum = 0;
+
+ // 20 byte IP header
+ pay.push_back({pkt.hdr.version,pkt.hdr.header_length}); // byte 0
+ pay.push_back({pkt.hdr.dscp,pkt.hdr.ecn}); // byte 1
+ pay.push_back(length[15:8]); // byte 2
+ pay.push_back(length[7:0]); // byte 3
+ pay.push_back(pkt.hdr.identification[15:8]); // byte 4
+ pay.push_back(pkt.hdr.identification[7:0]); // byte 5
+ pay.push_back({flags,pkt.hdr.frag_offset[12:8]}); // byte 6
+ pay.push_back(pkt.hdr.frag_offset[7:0]); // byte 7
+ pay.push_back(pkt.hdr.time_to_live); // byte 8
+ pay.push_back(pkt.hdr.protocol); // byte 9
+ pay.push_back(checksum[15:8]); // byte 10
+ pay.push_back(checksum[7:0]); // byte 11
+ pay.push_back(pkt.hdr.src_ip[31:24]); // byte 12
+ pay.push_back(pkt.hdr.src_ip[23:16]); // byte 13
+ pay.push_back(pkt.hdr.src_ip[15:8]); // byte 14
+ pay.push_back(pkt.hdr.src_ip[7:0]); // byte 15
+ pay.push_back(pkt.hdr.dest_ip[31:24]); // byte 16
+ pay.push_back(pkt.hdr.dest_ip[23:16]); // byte 17
+ pay.push_back(pkt.hdr.dest_ip[15:8]); // byte 18
+ pay.push_back(pkt.hdr.dest_ip[7:0]); // byte 19
+ pay = {pay,pkt.payload};
+
+ if ($isunknown(pkt.hdr.checksum))
+ checksum = calc_ipv4_checksum(pay);
+ else
+ checksum = pkt.hdr.checksum;
+ // replace the checksum (bytes 11:10
+ pay[10] = checksum[15:8];
+ pay[11] = checksum[7:0];
+
+ return pay;
+ endfunction
+
+ // Break a queue of bytes into a ip_pkt
+ function automatic ipv4_pkt_t unflatten_ipv4_pkt(input raw_pkt_t pay);
+ ipv4_pkt_t pkt;
+
+ // 20 byte IP header
+ {pkt.hdr.version,
+ pkt.hdr.header_length} = pay.pop_front(); // byte 0
+ {pkt.hdr.dscp,pkt.hdr.ecn} = pay.pop_front(); // byte 1
+ pkt.hdr.length[15:8] = pay.pop_front(); // byte 2
+ pkt.hdr.length[7:0] = pay.pop_front(); // byte 3
+ pkt.hdr.identification[15:8] = pay.pop_front(); // byte 4
+ pkt.hdr.identification[7:0] = pay.pop_front(); // byte 5
+ {pkt.hdr.more_frag,
+ pkt.hdr.dont_frag,
+ pkt.hdr.rsv_zero,
+ pkt.hdr.frag_offset[12:8]} = pay.pop_front(); // byte 6
+ pkt.hdr.frag_offset[7:0] = pay.pop_front(); // byte 7
+ pkt.hdr.time_to_live = pay.pop_front(); // byte 8
+ pkt.hdr.protocol = ip_protocol_t'(pay.pop_front()); // byte 9
+ pkt.hdr.checksum[15:8] = pay.pop_front(); // byte 10
+ pkt.hdr.checksum[7:0] = pay.pop_front(); // byte 11
+ pkt.hdr.src_ip[31:24] = pay.pop_front(); // byte 12
+ pkt.hdr.src_ip[23:16] = pay.pop_front(); // byte 13
+ pkt.hdr.src_ip[15:8] = pay.pop_front(); // byte 14
+ pkt.hdr.src_ip[7:0] = pay.pop_front(); // byte 15
+ pkt.hdr.dest_ip[31:24] = pay.pop_front(); // byte 16
+ pkt.hdr.dest_ip[23:16] = pay.pop_front(); // byte 17
+ pkt.hdr.dest_ip[15:8] = pay.pop_front(); // byte 18
+ pkt.hdr.dest_ip[7:0] = pay.pop_front(); // byte 19
+ pkt.payload = pay;
+
+ return pkt;
+
+ endfunction
+
+ function automatic logic ipv4_pkt_compare(input ipv4_pkt_t a, input ipv4_pkt_t b);
+
+ return ((a.hdr.header_length == b.hdr.header_length) &&
+ (a.hdr.version == b.hdr.version) &&
+ (a.hdr.dscp == b.hdr.dscp) &&
+ (a.hdr.ecn == b.hdr.ecn) &&
+ (a.hdr.length == b.hdr.length) &&
+ (a.hdr.identification == b.hdr.identification) &&
+ (a.hdr.rsv_zero == b.hdr.rsv_zero) &&
+ (a.hdr.dont_frag == b.hdr.dont_frag) &&
+ (a.hdr.more_frag == b.hdr.more_frag) &&
+ (a.hdr.frag_offset == b.hdr.frag_offset) &&
+ (a.hdr.time_to_live == b.hdr.time_to_live) &&
+ (a.hdr.protocol == b.hdr.protocol) &&
+ (a.hdr.checksum == b.hdr.checksum) &&
+ (a.hdr.src_ip == b.hdr.src_ip) &&
+ (a.hdr.dest_ip == b.hdr.dest_ip) &&
+ raw_pkt_compare(a.payload,b.payload));
+
+ endfunction
+
+ //************************************************************//
+ //////////////////// UDP PACKETS ///////////////////////////////
+ //************************************************************//
+
+ localparam logic [15:0] DEF_SRC_UDP_PORT = 16'd49748;
+ localparam logic [15:0] DEF_DEST_UDP_PORT = 16'd49153;
+ localparam logic [15:0] DEF_BRIDGE_UDP_PORT = 16'h66_55;
+
+ // UDP Header
+ typedef struct {
+ logic [15:0] src_port = DEF_SRC_UDP_PORT;
+ logic [15:0] dest_port = DEF_DEST_UDP_PORT;
+ logic [15:0] length = 16'hXXXX; //flag for (fill it in please)
+ logic [15:0] checksum = 16'hXXXX; //flag for (fill it in please)
+ } udp_hdr_t;
+
+ // UDP Packet, Header + Payload
+ typedef struct {
+ udp_hdr_t hdr;
+ raw_pkt_t payload;
+ } udp_pkt_t;
+
+ function automatic logic [15:0] calc_udp_checksum(
+ input logic [31:0] src_ip,
+ input logic [31:0] dest_ip,
+ input raw_pkt_t pkt);
+
+ logic [31:0] checksum;
+ raw_pkt_t virtual_header;
+
+ // UDP checksum is calculated over a virtual header that is added to
+ // the front of the packet.
+ virtual_header.push_back(src_ip[31:24]); // byte 0
+ virtual_header.push_back(src_ip[23:16]); // byte 1
+ virtual_header.push_back(src_ip[15:8]); // byte 2
+ virtual_header.push_back(src_ip[7:0]); // byte 3
+ virtual_header.push_back(dest_ip[31:24]); // byte 4
+ virtual_header.push_back(dest_ip[23:16]); // byte 5
+ virtual_header.push_back(dest_ip[15:8]); // byte 6
+ virtual_header.push_back(dest_ip[7:0]); // byte 7
+ virtual_header.push_back(0); // byte 8
+ virtual_header.push_back(UDP); // byte 9 UDP (Protocol enum) x11
+ virtual_header.push_back(0); // byte 10
+ virtual_header.push_back(pkt[6]); // byte 11 Length
+ virtual_header.push_back(pkt[7]); // byte 12 Length
+
+ pkt = {virtual_header,pkt}; // add virtual header in front
+
+ checksum = 0;
+ // Iterate over 16 bit chunks reading from an array of bytes
+ // need to traverse the virtual header / udp header / udp data
+ for (int i = 0 ; i < pkt.size ; i+=2 ) begin
+ // BIG endian network ordering... so weird
+ checksum += {pkt[i],pkt[i+1]};
+ end
+ checksum += checksum[31:16];
+ checksum = ~checksum;
+
+ return checksum[15:0];
+
+ endfunction
+
+ // Break a udp_pkt into a queue of bytes
+ function automatic raw_pkt_t flatten_udp_pkt(
+ input logic [31:0] src_ip,
+ input logic [31:0] dest_ip,
+ input udp_pkt_t pkt);
+ raw_pkt_t pay;
+
+ logic [15:0] length;
+ logic [15:0] checksum;
+
+ // If header or length is not set to default value then use the value in
+ // the packet.
+ if ($isunknown(pkt.hdr.length))
+ length = pkt.payload.size()+8; // 8 because length includes UDP header length.
+ else
+ length = pkt.hdr.length;
+
+ //temporary checksum
+ checksum = 0;
+
+ pay.push_back(pkt.hdr.src_port[15:8]); // byte 0
+ pay.push_back(pkt.hdr.src_port[7:0]); // byte 1
+ pay.push_back(pkt.hdr.dest_port[15:8]); // byte 2
+ pay.push_back(pkt.hdr.dest_port[7:0]); // byte 3
+ pay.push_back(length[15:8]); // byte 4
+ pay.push_back(length[7:0]); // byte 5
+ pay.push_back(checksum[15:8]); // byte 6
+ pay.push_back(checksum[7:0]); // byte 7
+ pay = {pay,pkt.payload};
+
+ if ($isunknown(pkt.hdr.checksum))
+ checksum = calc_udp_checksum(src_ip,dest_ip,pay);
+ else
+ checksum = pkt.hdr.checksum;
+
+ pay[6] = checksum[15:8];
+ pay[7] = checksum[7:0];
+
+ return pay;
+
+ endfunction
+
+ // Break a queue of bytes into a udp_pkt
+ function automatic udp_pkt_t unflatten_udp_pkt(input raw_pkt_t pay);
+ udp_pkt_t pkt;
+
+ pkt.hdr.src_port[15:8] = pay.pop_front();
+ pkt.hdr.src_port[7:0] = pay.pop_front();
+ pkt.hdr.dest_port[15:8] = pay.pop_front();
+ pkt.hdr.dest_port[7:0] = pay.pop_front();
+ pkt.hdr.length[15:8] = pay.pop_front();
+ pkt.hdr.length[7:0] = pay.pop_front();
+ pkt.hdr.checksum[15:8] = pay.pop_front();
+ pkt.hdr.checksum[7:0] = pay.pop_front();
+ pkt.payload = pay;
+
+ return pkt;
+
+ endfunction
+
+ function automatic logic udp_pkt_compare(input udp_pkt_t a, input udp_pkt_t b);
+
+ return ((a.hdr.src_port == b.hdr.src_port) &&
+ (a.hdr.dest_port == b.hdr.dest_port) &&
+ (a.hdr.length == b.hdr.length) &&
+ raw_pkt_compare(a.payload,b.payload));
+
+ endfunction
+
+ typedef enum int {
+ NO_PREAMBLE=0, NORMAL_PREAMBLE=1, ZERO_PREAMBLE=2
+ } preamble_t;
+
+ // Build up a raw UDP packet.
+ // Args:
+ // - pkt: Packet data (queue)
+ // - stream: Stream to use (Optional)
+ function automatic raw_pkt_t build_udp_pkt (
+ input eth_hdr_t eth_hdr,
+ input ipv4_hdr_t ipv4_hdr,
+ input udp_hdr_t udp_hdr,
+ input raw_pkt_t pay,
+ input int preamble = NO_PREAMBLE);
+
+ automatic udp_pkt_t udp_pkt;
+ automatic ipv4_pkt_t ipv4_pkt;
+ automatic eth_pkt_t eth_pkt;
+ automatic raw_pkt_t raw_pkt;
+
+ udp_pkt.hdr = udp_hdr;
+ udp_pkt.payload = pay;
+ ipv4_pkt.hdr = ipv4_hdr;
+ ipv4_pkt.payload = flatten_udp_pkt(ipv4_hdr.src_ip,ipv4_hdr.dest_ip,udp_pkt);
+ eth_pkt.hdr = eth_hdr;
+ eth_pkt.payload = flatten_ipv4_pkt(ipv4_pkt);
+ raw_pkt = flatten_eth_pkt(eth_pkt);
+ if (preamble==NORMAL_PREAMBLE) begin
+ raw_pkt.push_front(8'hAB);
+ raw_pkt.push_front(8'hAA);
+ raw_pkt.push_front(8'hAA);
+ raw_pkt.push_front(8'hAA);
+ raw_pkt.push_front(8'hAA);
+ raw_pkt.push_front(8'hAA);
+ end else if (preamble==ZERO_PREAMBLE) begin
+ raw_pkt.push_front(8'h00);
+ raw_pkt.push_front(8'h00);
+ raw_pkt.push_front(8'h00);
+ raw_pkt.push_front(8'h00);
+ raw_pkt.push_front(8'h00);
+ raw_pkt.push_front(8'h00);
+ end
+ return raw_pkt;
+
+ endfunction
+
+ // Wait for a packet to finish on the bus
+ // and decode it
+ task automatic decode_udp_pkt (
+ input raw_pkt_t raw_pkt,
+ output eth_hdr_t eth_hdr,
+ output ipv4_hdr_t ipv4_hdr,
+ output udp_hdr_t udp_hdr,
+ output raw_pkt_t payload);
+
+ eth_pkt_t eth_pkt;
+ ipv4_pkt_t ipv4_pkt;
+ udp_pkt_t udp_pkt;
+
+ eth_pkt = unflatten_eth_pkt(raw_pkt);
+ ipv4_pkt = unflatten_ipv4_pkt(eth_pkt.payload);
+ udp_pkt = unflatten_udp_pkt(ipv4_pkt.payload);
+
+ eth_hdr = eth_pkt.hdr;
+ ipv4_hdr = ipv4_pkt.hdr;
+ udp_hdr = udp_pkt.hdr;
+ payload = udp_pkt.payload;
+
+ endtask
+
+ //---------------------------------------------------------------------------
+ // XPORT Stream Packet Class
+ //---------------------------------------------------------------------------
+ // Extensions to the AxiStreamPacket used in the XPORT code
+ class XportStreamPacket #(
+ int DATA_WIDTH = 64
+ ) extends AxiStreamPacket #(DATA_WIDTH, $clog2((DATA_WIDTH/8)+1));
+
+ typedef XportStreamPacket #(DATA_WIDTH) XportPacket_t;
+ localparam UWIDTH = $clog2((DATA_WIDTH/8)+1);
+ // Class constructor.
+ function new ();
+ super.new();
+ endfunction : new
+
+ // Return a handle to a copy of this transaction
+ function XportPacket_t copy();
+ XportPacket_t temp;
+ temp = new();
+ temp.data = this.data;
+ temp.user = this.user;
+ temp.keep = this.keep;
+ return temp;
+ endfunction
+
+ // bring in data from an AxisPacket
+ function void import_axis(AxisPacket_t axi_pkt);
+ this.data = axi_pkt.data;
+ this.user = axi_pkt.user;
+ this.keep = axi_pkt.keep;
+ endfunction : import_axis
+
+ // take the tuser signal's and use them to set
+ // tkeep signals
+ task automatic tuser_to_tkeep(logic PRESERVE_TUSER=1);
+ keep_t last_tkeep;
+ user_t last_tuser;
+ logic [$clog2(DATA_WIDTH/8)-1:0] last_bytes;
+ last_tuser = this.user[$];
+ last_bytes = last_tuser[$clog2(DATA_WIDTH/8)-1:0];
+ // set all the tuser values to zero
+ // set all the tuser values to zero
+ foreach (this.user[i]) begin
+ this.keep[i] = '1;
+ if(!PRESERVE_TUSER)
+ this.user[i] = 0;
+ end
+ // figure out the value of tkeep for the last word
+ if (last_bytes == 0) last_tkeep = '1;
+ // check if there is an X
+ else if ($isunknown(last_tuser)) last_tkeep = '1;
+ else begin
+ last_tkeep = 0;
+ foreach (last_tkeep[i]) begin
+ last_tkeep[i] = i < last_bytes;
+ end
+ end
+ this.keep[$] = last_tkeep;
+ // set data bytes where the value isn't used to zero
+ foreach (last_tkeep[i]) begin
+ if (last_tkeep[i] == 0) this.data[$][i*8 +: 8] = 0;
+ end
+ endtask : tuser_to_tkeep
+
+ // take the tkeep signal's and use them to set
+ // width in the user signals
+ task automatic tkeep_to_tuser(int ERROR_PROB=0);
+ keep_t last_tkeep;
+ user_t last_tuser;
+
+ last_tkeep = this.keep[$];
+ // set all the tuser values to zero
+ foreach (this.user[i]) begin
+ this.user[i] = 0;
+ this.user[i][UWIDTH-1] = $urandom_range(99) < ERROR_PROB;
+ end
+ // check if there is an X
+ if ($isunknown(last_tkeep)) last_tuser = '0;
+ else begin
+ last_tuser = 0;
+ foreach (last_tkeep[i]) begin
+ if (last_tkeep[i]==1'b1) begin
+ last_tuser = last_tuser+1;
+ end
+ end
+ end
+ // full word is 0. MSB set is error
+ if (last_tuser == DATA_WIDTH/8)last_tuser = 0;
+ this.user[$] = last_tuser;
+ this.user[$][UWIDTH-1] = $urandom_range(99) < ERROR_PROB;
+
+ // set data bytes where the value isn't used to zero
+ foreach (last_tkeep[i]) begin
+ if (last_tkeep[i] == 0) this.data[$][i*8 +: 8] = 0;
+ end
+
+ endtask : tkeep_to_tuser
+
+ function automatic logic has_error();
+ logic error;
+
+ error = 0;
+ foreach (this.user[i]) begin
+ //catch if error was set
+ error = error || this.user[i][UWIDTH-1];
+ end
+ return error;
+ endfunction : has_error
+
+ function automatic int byte_length();
+ int bytes;
+ int last_bytes;
+ bytes = (this.data.size()-1)*DATA_WIDTH/8;
+ last_bytes = this.user[$][UWIDTH-2:0];
+ if (last_bytes == 0)
+ bytes+=DATA_WIDTH/8;
+ else
+ bytes+=last_bytes;
+
+ return bytes;
+
+ endfunction : byte_length
+
+ function automatic void clear_error();
+
+ foreach (this.user[i]) begin
+ this.user[i][UWIDTH-1] = 0;
+ end
+
+ endfunction : clear_error
+
+ function automatic void clear_keep();
+
+ foreach (this.keep[i]) begin
+ this.keep[i] = 0;
+ end
+
+ endfunction : clear_keep
+
+ function automatic void clear_user();
+
+ foreach (this.user[i]) begin
+ this.user[i] = 0;
+ end
+
+ endfunction : clear_user
+
+ task automatic set_error();
+ foreach (this.user[i]) begin
+ this.user[i][UWIDTH-1] = 1;
+ end
+ endtask : set_error
+
+ ///// compare_w_error
+ // Check that this packet has expected error bit in tuser
+ // Keep is not compared
+ // If COMPARE_ERROR_PACKETS is 0
+ // Don't check packet contents
+ // If COMPARE_ERROR_PACKETS is 1
+ // Check that this packet matches the expected packet
+ function automatic logic compare_w_error(
+ XportPacket_t expected,
+ int COMPARE_ERROR_PACKETS=1
+ );
+
+ automatic XportPacket_t actual_copy = this.copy();
+ automatic XportPacket_t expected_copy = expected.copy();
+
+ logic exp_error=0;
+ logic act_error=0;
+ logic error_condition;
+
+ exp_error = expected.has_error();
+ act_error = this.has_error();
+
+ actual_copy.clear_error();
+ expected_copy.clear_error();
+ actual_copy.clear_keep();
+ expected_copy.clear_keep();
+
+ error_condition = (!expected_copy.equal(actual_copy) &&
+ (!exp_error || COMPARE_ERROR_PACKETS)) ||
+ act_error != exp_error;
+ if (error_condition) begin
+ $display("Expected");
+ expected.print();
+ $display("Actual");
+ this.print();
+ if (!expected_copy.equal(actual_copy))
+ $display("ERROR :: packet mismatch");
+ if (act_error != exp_error)
+ $display("ERROR :: error mismatch");
+ end
+
+ return error_condition;
+
+ endfunction : compare_w_error
+
+ ///// compare_w_sof
+ // Check that this packet has expected sof bit in tuser
+ // Keep is not compared
+ // Check that this packet matches the expected packet
+ function automatic logic compare_w_sof(XportPacket_t expected);
+
+ automatic XportPacket_t actual_copy = this.copy();
+ automatic XportPacket_t expected_copy = expected.copy();
+
+ logic sof_error=0;
+ foreach (this.user[i]) begin
+ if (i==0) begin
+ // set if top bit of user isn't set on the first word.
+ sof_error = !this.user[i][UWIDTH-1];
+ end else begin
+ // set if top bit of user is set on any other word.
+ sof_error = this.user[i][UWIDTH-1] || sof_error;
+ end
+ end
+
+ // error bit doubles for SOF
+ actual_copy.clear_error();
+ expected_copy.clear_error();
+ actual_copy.clear_keep();
+ expected_copy.clear_keep();
+
+ // set SOF in expected
+ expected_copy.user[0][UWIDTH-1] = 0;
+
+ if (!expected_copy.equal(actual_copy) ||
+ sof_error) begin
+ $display("Expected");
+ expected_copy.print();
+ $display("Actual");
+ this.print();
+ if (!expected_copy.equal(actual_copy))
+ $display("ERROR :: packet mismatch");
+ if (sof_error)
+ $display("ERROR :: sof mismatch");
+ end
+
+ return !expected_copy.equal(actual_copy) ||
+ sof_error;
+
+ endfunction : compare_w_sof
+
+ ///// compare_w_pad
+ // Check that this packet has expected sof bit in tuser
+ // Keep is not compared
+ // Check that this packet matches the expected packet
+ // if DUMB_ORIGINAL_WAY
+ // User is not compared
+ // else
+ // Check that this packets tuser matches the expected packet
+ function automatic logic compare_w_pad(
+ XportPacket_t expected,
+ logic DUMB_ORIGINAL_WAY=0
+ );
+
+ automatic XportPacket_t actual_copy = this.copy();
+ automatic XportPacket_t expected_copy = expected.copy();
+
+ // not using MSB as error here.
+ actual_copy.clear_error();
+ expected_copy.clear_error();
+ actual_copy.clear_keep();
+ expected_copy.clear_keep();
+ // Add pad bytes to user
+ if (DUMB_ORIGINAL_WAY) begin
+ // I can't figure out how to calculate the expected on the
+ // original so I'm just copying the actual
+ foreach (expected_copy.user[i]) begin
+ expected_copy.user[i] = actual_copy.user[i];
+ end
+ end
+
+ if (!expected_copy.equal(actual_copy)) begin
+ $display("Expected");
+ expected_copy.print();
+ $display("Actual");
+ this.print();
+ if (!expected_copy.equal(actual_copy))
+ $display("ERROR :: packet mismatch");
+ end
+
+ return !expected_copy.equal(actual_copy);
+
+ endfunction : compare_w_pad
+
+ ///// compare_no_user
+ // Check that this packet has expected sof bit in tuser
+ // Keep is not compared
+ // Check that this packet matches the expected packet
+ // User is not compared
+ function automatic logic compare_no_user(XportPacket_t expected);
+
+ automatic XportPacket_t actual_copy = this.copy();
+ automatic XportPacket_t expected_copy = expected.copy();
+
+ // not using MSB as error here.
+ actual_copy.clear_error();
+ expected_copy.clear_error();
+ actual_copy.clear_keep();
+ expected_copy.clear_keep();
+
+ // Add pad bytes to user
+ foreach (expected_copy.user[i]) begin
+ expected_copy.user[i] = 0;
+ actual_copy.user[i] = 0;
+ end
+
+ if (!expected_copy.equal(actual_copy)) begin
+ $display("Expected");
+ expected.print();
+ $display("Actual");
+ this.print();
+ if (!expected_copy.equal(actual_copy))
+ $display("ERROR :: packet mismatch");
+ end
+
+ return !expected_copy.equal(actual_copy);
+
+ endfunction : compare_no_user
+
+ endclass : XportStreamPacket;
+
+endpackage : PkgEthernet