From c13946f418b0efc0961f341a1da1862349e9df1a Mon Sep 17 00:00:00 2001 From: Thomas Benz Date: Fri, 30 Sep 2022 13:48:55 +0200 Subject: [PATCH] chan_compare: Non-synthesizable module comparing two AXI channels of the same type --- Bender.yml | 1 + CHANGELOG.md | 1 + README.md | 1 + axi.core | 2 + src/axi_chan_compare.sv | 212 ++++++++++++++++++++++++++++++++++++++++ src_files.yml | 2 + 6 files changed, 219 insertions(+) create mode 100644 src/axi_chan_compare.sv diff --git a/Bender.yml b/Bender.yml index f0ffdc11c..952d5139e 100644 --- a/Bender.yml +++ b/Bender.yml @@ -84,6 +84,7 @@ sources: - target: simulation files: + - src/axi_chan_compare.sv - src/axi_dumper.sv - src/axi_sim_mem.sv - src/axi_test.sv diff --git a/CHANGELOG.md b/CHANGELOG.md index c6f5aaed5..ae12b1c2c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,6 +8,7 @@ and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0. ## Unreleased ### Added +- Add `axi_channel_compare.sv`: Non-synthesizable module comparing two AXI channels of the same type ### Changed - `axi_demux`: Replace write FIFO (`w_fifo`) with a write credit counter. diff --git a/README.md b/README.md index 85cf211f1..489eee09b 100644 --- a/README.md +++ b/README.md @@ -67,6 +67,7 @@ In addition to the modules above, which are available in synthesis and simulatio | Name | Description | |------------------------------------------------------|--------------------------------------------------------------------------------------------------------| +| [`axi_chan_compare`](src/axi_chan_compare.sv) | Non-synthesizable module comparing two AXI channels of the same type | | [`axi_chan_logger`](src/axi_test.sv) | Logs the transactions of an AXI4(+ATOPs) port to files. | | [`axi_driver`](src/axi_test.sv) | Low-level driver for AXI4(+ATOPs) that can send and receive individual beats on any channel. | | [`axi_dumper`](src/axi_dumper.sv) | Dumps log to file to be interpreted by `axi_dumper_interpret` script for debugging purposes. | diff --git a/axi.core b/axi.core index cc4e98609..198fd5013 100644 --- a/axi.core +++ b/axi.core @@ -57,6 +57,8 @@ filesets: - src/axi_iw_converter.sv - src/axi_lite_xbar.sv - src/axi_xbar.sv + - src/axi_chan_compare.sv + - src/axi_dumper.sv - src/axi_sim_mem.sv - src/axi_test.sv # Level 5 diff --git a/src/axi_chan_compare.sv b/src/axi_chan_compare.sv new file mode 100644 index 000000000..4de4ea741 --- /dev/null +++ b/src/axi_chan_compare.sv @@ -0,0 +1,212 @@ +// Copyright 2018 ETH Zurich and University of Bologna. +// Copyright and related rights are licensed under the Solderpad Hardware +// License, Version 0.51 (the "License"); you may not use this file except in +// compliance with the License. You may obtain a copy of the License at +// http://solderpad.org/licenses/SHL-0.51. Unless required by applicable law +// or agreed to in writing, software, hardware and materials distributed under +// this License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR +// CONDITIONS OF ANY KIND, either express or implied. See the License for the +// specific language governing permissions and limitations under the License. +// +// Authors: +// - Thomas Benz +// - Paul Scheffler +// - Tim Fischer + +/// Non-synthesizable module comparing two AXI channels of the same type +module axi_chan_compare #( + parameter type aw_chan_t = logic, + parameter type w_chan_t = logic, + parameter type b_chan_t = logic, + parameter type ar_chan_t = logic, + parameter type r_chan_t = logic, + parameter type req_t = logic, + parameter type resp_t = logic +)( + input logic clk_a_i, + input logic clk_b_i, + input req_t axi_a_req, + input resp_t axi_a_res, + input req_t axi_b_req, + input resp_t axi_b_res +); + + function automatic void print_aw ( + input aw_chan_t aw_expected, + input aw_chan_t aw_received + ); + // verilog_lint: waive-start line-length + $display("AW | expected | received "); + $display("--------|------------------------------------------------------------------|-----------------------------------------------------------------"); + $display("id: | %64d | %64d", aw_expected.id, aw_received.id); + $display("addr: | %64x | %64x", aw_expected.addr, aw_received.addr); + $display("len: | %64d | %64d", aw_expected.len, aw_received.len); + $display("size: | %64d | %64d", aw_expected.size, aw_received.size); + $display("burst: | %64d | %64d", aw_expected.burst, aw_received.burst); + $display("lock: | %64d | %64d", aw_expected.lock, aw_received.lock); + $display("cache: | %64d | %64d", aw_expected.cache, aw_received.cache); + $display("prot: | %64d | %64d", aw_expected.prot, aw_received.prot); + $display("qos: | %64d | %64d", aw_expected.qos, aw_received.qos); + $display("region: | %64d | %64d", aw_expected.region, aw_received.region); + $display("user: | %64d | %64d", aw_expected.user, aw_received.user); + $display("atop: | %64d | %64d", aw_expected.atop, aw_received.atop); + $display("--------|------------------------------------------------------------------|-----------------------------------------------------------------"); + // verilog_lint: waive-stop line-length + endfunction + + function automatic void print_ar ( + input ar_chan_t ar_expected, + input ar_chan_t ar_received + ); + // verilog_lint: waive-start line-length + $display("AR | expected | received "); + $display("--------|------------------------------------------------------------------|-----------------------------------------------------------------"); + $display("id: | %64d | %64d", ar_expected.id, ar_received.id); + $display("addr: | %64x | %64x", ar_expected.addr, ar_received.addr); + $display("len: | %64d | %64d", ar_expected.len, ar_received.len); + $display("size: | %64d | %64d", ar_expected.size, ar_received.size); + $display("burst: | %64d | %64d", ar_expected.burst, ar_received.burst); + $display("lock: | %64d | %64d", ar_expected.lock, ar_received.lock); + $display("cache: | %64d | %64d", ar_expected.cache, ar_received.cache); + $display("prot: | %64d | %64d", ar_expected.prot, ar_received.prot); + $display("qos: | %64d | %64d", ar_expected.qos, ar_received.qos); + $display("region: | %64d | %64d", ar_expected.region, ar_received.region); + $display("user: | %64d | %64d", ar_expected.user, ar_received.user); + $display("--------|------------------------------------------------------------------|-----------------------------------------------------------------"); + // verilog_lint: waive-stop line-length + endfunction + + function automatic void print_w ( + input w_chan_t w_expected, + input w_chan_t w_received + ); + // verilog_lint: waive-start line-length + $display("W | expected | received "); + $display("--------|------------------------------------------------------------------|-----------------------------------------------------------------"); + $display("data: | %64x | %64x", w_expected.data, w_received.data); + $display("strb: | %64d | %64d", w_expected.strb, w_received.strb); + $display("last: | %64d | %64d", w_expected.last, w_received.last); + $display("user: | %64d | %64d", w_expected.user, w_received.user); + $display("--------|------------------------------------------------------------------|-----------------------------------------------------------------"); + // verilog_lint: waive-stop line-length + endfunction + + function automatic void print_b ( + input b_chan_t b_expected, + input b_chan_t b_received + ); + // verilog_lint: waive-start line-length + $display("B | expected | received "); + $display("--------|------------------------------------------------------------------|-----------------------------------------------------------------"); + $display("id: | %64d | %64d", b_expected.id, b_received.id); + $display("resp: | %64d | %64d", b_expected.resp, b_received.resp); + $display("user: | %64d | %64d", b_expected.user, b_received.user); + $display("--------|------------------------------------------------------------------|-----------------------------------------------------------------"); + // verilog_lint: waive-stop line-length + endfunction + + function automatic void print_r ( + input r_chan_t r_expected, + input r_chan_t r_received + ); + // verilog_lint: waive-start line-length + $display("R | expected | received "); + $display("--------|------------------------------------------------------------------|-----------------------------------------------------------------"); + $display("id: | %64d | %64d", r_expected.id, r_received.id); + $display("data: | %64x | %64x", r_expected.data, r_received.data); + $display("resp: | %64d | %64d", r_expected.resp, r_received.resp); + $display("last: | %64d | %64d", r_expected.last, r_received.last); + $display("user: | %64d | %64d", r_expected.user, r_received.user); + $display("--------|------------------------------------------------------------------|-----------------------------------------------------------------"); + // verilog_lint: waive-stop line-length + endfunction + + // queues + aw_chan_t aw_queue [$]; + w_chan_t w_queue [$]; + b_chan_t b_queue [$]; + ar_chan_t ar_queue [$]; + r_chan_t r_queue [$]; + + // requests generated at axi A: enqueue elements + always_ff @(posedge clk_a_i) begin : proc_enqueue_a + // aw + if (axi_a_req.aw_valid & axi_a_res.aw_ready) + aw_queue.push_back(axi_a_req.aw); + // w + if (axi_a_req.w_valid & axi_a_res.w_ready) + w_queue.push_back(axi_a_req.w); + // ar + if (axi_a_req.ar_valid & axi_a_res.ar_ready) + ar_queue.push_back(axi_a_req.ar); + end + + // responses generated at axi B: enqueue elements + always_ff @(posedge clk_b_i) begin : proc_enqueue_b + // b + if (axi_b_res.b_valid & axi_b_req.b_ready) + b_queue.push_back(axi_b_res.b); + // r + if (axi_b_res.r_valid & axi_b_req.r_ready) + r_queue.push_back(axi_b_res.r); + end + + // requests arriving at axi B from A: dequeue elements and check + always_ff @(posedge clk_b_i) begin : proc_dequeue_and_check_b + // aw + if (axi_b_req.aw_valid & axi_b_res.aw_ready) begin + automatic aw_chan_t aw; + if (aw_queue.size() == 0) $error("AW queue is empty!"); + aw = aw_queue.pop_front(); // verilog_lint: waive always-ff-non-blocking + if (axi_b_req.aw !== aw) begin + $error("AW mismatch!"); + print_aw(aw, axi_b_req.aw); + end + end + // w + if (axi_b_req.w_valid & axi_b_res.w_ready) begin + automatic w_chan_t w; + if (w_queue.size() == 0) $error("W queue is empty!"); + w = w_queue.pop_front(); // verilog_lint: waive always-ff-non-blocking + if (axi_b_req.w !== w) begin + $error("W mismatch!"); + print_w(w, axi_b_req.w); + end + end + // ar + if (axi_b_req.ar_valid & axi_b_res.ar_ready) begin + automatic ar_chan_t ar; + if (ar_queue.size() == 0) $error("AR queue is empty!"); + ar = ar_queue.pop_front(); // verilog_lint: waive always-ff-non-blocking + if (axi_b_req.ar !== ar) begin + $error("AR mismatch!"); + print_ar(ar, axi_b_req.ar); + end + end + end + + // responses arriving at axi A from B: dequeue elements and check + always_ff @(posedge clk_a_i) begin : proc_dequeue_and_check_a + // b + if (axi_a_res.b_valid & axi_a_req.b_ready) begin + automatic b_chan_t b; + if (b_queue.size() == 0) $error("B queue is empty!"); + b = b_queue.pop_front(); // verilog_lint: waive always-ff-non-blocking + if (axi_a_res.b !== b) begin + $error("B mismatch!"); + print_b(b, axi_a_res.b); + end + end + // r + if (axi_a_res.r_valid & axi_a_req.r_ready) begin + automatic r_chan_t r; + if (r_queue.size() == 0) $error("R queue is empty!"); + r = r_queue.pop_front(); // verilog_lint: waive always-ff-non-blocking + if (axi_a_res.r !== r) begin + $error("R mismatch!"); + print_r(r, axi_a_res.r); + end + end + end + +endmodule : axi_chan_compare diff --git a/src_files.yml b/src_files.yml index 5fc0c4ced..60367172d 100644 --- a/src_files.yml +++ b/src_files.yml @@ -61,6 +61,8 @@ axi: axi_sim: files: + - src/axi_chan_compare.sv + - src/axi_dumper.sv - src/axi_sim_mem.sv - src/axi_test.sv flags: