From 174d61a64775b4d6891e46aa4ae1b26530046870 Mon Sep 17 00:00:00 2001 From: Shupei Fan Date: Sun, 8 Dec 2024 06:57:28 +0000 Subject: [PATCH] [verbatim] add DpiAxiWrite/DpiAxiRead --- t1rocketemu/vsrc/VerbatimModule.sv | 367 ++++++++++++++++++++++++++++- 1 file changed, 357 insertions(+), 10 deletions(-) diff --git a/t1rocketemu/vsrc/VerbatimModule.sv b/t1rocketemu/vsrc/VerbatimModule.sv index 2cf3d542e..690225224 100644 --- a/t1rocketemu/vsrc/VerbatimModule.sv +++ b/t1rocketemu/vsrc/VerbatimModule.sv @@ -1,3 +1,21 @@ +package t1_common; + typedef byte int8_t; + typedef shortint int16_t; + typedef int int32_t; + typedef longint int64_t; + typedef byte unsigned uint8_t; + typedef shortint unsigned uint16_t; + typedef int unsigned uint32_t; + typedef longint unsigned uint64_t; + + typedef uint32_t axi_tag_t; + import "DPI-C" function axi_tag_t t1_axi_get_id( + string tag, + uint32_t width, + string direction + ); +endpackage + module VerbatimModule #( parameter integer T1_DLEN, parameter integer T1_VLEN, @@ -8,6 +26,7 @@ module VerbatimModule #( output reg initFlag, input wire idle ); + import t1_common::*; // This module contains everything we can not represent in Chisel currently, // including clock gen, plusarg parsing, sim control, etc @@ -21,18 +40,18 @@ module VerbatimModule #( // +t1_dump_start (optional T) cycle when dump starts, by default it's simulation start, for debug only // +t1_dump_end (optional T) cycle when dump ends, by default is's simulation end, for debug only - longint unsigned cycle = 0; - longint unsigned quit_cycle = 0; - longint unsigned global_timeout = 0; - longint unsigned timeout_after_quit = 10000; - longint unsigned dpi_timeout = 1000000; + uint64_t cycle = 0; + uint64_t quit_cycle = 0; + uint64_t global_timeout = 0; + uint64_t timeout_after_quit = 10000; + uint64_t dpi_timeout = 1000000; string elf_file; `ifdef T1_ENABLE_TRACE - longint unsigned dump_start = 0; - longint unsigned dump_end = 0; + uint64_t dump_start = 0; + uint64_t dump_end = 0; string wave_path; - function void dump_wave(input string file); + function void dump_wave(string file); `ifdef VCS $fsdbDumpfile(file); @@ -66,7 +85,7 @@ module VerbatimModule #( ); import "DPI-C" context function void t1_cosim_set_timeout(longint unsigned timeout); import "DPI-C" context function void t1_cosim_final(); - import "DPI-C" context function byte unsigned t1_cosim_watchdog(); + import "DPI-C" context function uint8_t t1_cosim_watchdog(); initial begin clock = 1'b0; @@ -115,7 +134,7 @@ module VerbatimModule #( end else begin // do not call watchdog if cosim already quit - automatic byte unsigned st = t1_cosim_watchdog(); + automatic uint8_t st = t1_cosim_watchdog(); if (st == 255) begin quit_cycle = cycle; if (idle) begin @@ -152,3 +171,331 @@ module VerbatimModule #( initial #(100) reset = 1'b0; initial #(20) initFlag = 1'b0; endmodule + +module VerifAxiWrite( + input bit clock, + bit reset, + bit awfire, + bit [7:0] awlen, + bit wfire, + bit wlast, + bit bfire +); + // check transaction level state machine + // + // TODO: current we enforce AW comes before W (or the same cycle) + + int pending_trans[$]; + int pending_beat; + int pending_ack; + +no_fire_at_reset: + assert property (@(posedge clock) reset |-> !awfire && !wfire && !bfire); + +no_pending_at_reset: + assert property (@(posedge clock) reset |-> pending_trans.size() == 0 && pending_beat == 0); + + always @(posedge clock) begin + if (awfire) begin + pending_trans.push_back(int'(awlen) + 1); + end + + if (wfire) begin + if (pending_beat == 0) begin + assert_wfire: assert (pending_trans.size() > 0); + + pending_beat = pending_trans.pop_front(); + end + + assert_wlast: assert (wlast == (pending_beat == 1)); + + pending_beat -= 1; + if (pending_beat == 0) begin + pending_ack += 1; + end + end + + if (bfire) begin + assert_bfire: assert (pending_ack > 0); + pending_ack -= 1; + end + end +endmodule + +module DpiAxiWrite #( + parameter string TAG, + parameter integer WIDTH +)( + input logic clock, + logic reset, + + input logic awvalid, + logic [31:0] awaddr, + logic [7:0] awlen, + logic [2:0] awsize, + output logic awready, + + input logic wvalid, + input logic [WIDTH-1:0] wdata, + input logic [WIDTH/8-1:0] wstrobe, + input logic wlast, + output logic wready, + + output logic bvalid, + logic [1:0] bresp, + input logic bready +); + import t1_common::*; + + if (!(WIDTH inside {8, 16, 32, 64, 128, 256, 512, 1024})) + $fatal(1, "invalid AXI width"); + + // delay to avoid race with init + axi_tag_t TAG_ID; + initial #1 TAG_ID = t1_axi_get_id(TAG, WIDTH, "W"); + + import "DPI-C" function bit t1_axi_aw_ready(axi_tag_t tag); + import "DPI-C" function bit t1_axi_aw_transfer( + axi_tag_t tag, + uint32_t awaddr, + uint8_t awlen, + uint8_t awsize + ); + import "DPI-C" function bit t1_axi_w_ready(axi_tag_t tag); + import "DPI-C" function bit t1_axi_w_transfer( + axi_tag_t tag, + bit wlast, + uint8_t[] wdata, + bit[] wstrobe + ); + import "DPI-C" function bit t1_axi_b_valid( + axi_tag_t tag, + bit is_transfer, + output uint8_t bresp + ); + + wire awfire = awvalid && awready; + wire wfire = wvalid && wready; + wire bfire = bvalid && bready; + + // AW channel + always @(posedge clock) begin + if (reset) begin + awready <= 1'b0; + end + else begin + if (awready) begin + if (awvalid) begin + awready <= t1_axi_aw_transfer( + TAG_ID, + awaddr, + awlen, + awsize + ); + end + end else begin + awready <= t1_axi_aw_ready(TAG_ID); + end + end + end + + // W channel + always @(posedge clock) begin + if (reset) begin + wready <= 1'b0; + end + else begin + if (wready) begin + if (wvalid) begin + wready <= t1_axi_w_transfer( + TAG_ID, + wlast, + wdata, + wstrobe + ); + end + end else begin + wready <= t1_axi_w_ready(TAG_ID); + end + end + end + + // B channel + uint8_t bresp_; + always @(posedge clock) begin + if (reset) begin + bvalid <= 1'b0; + bresp <= 0; + end + else begin + if (bfire) bvalid <= 1'b0; + + if (!bvalid || bfire) begin + if (t1_axi_b_ready( + TAG_ID, + bvalid, + bresp_, + )) begin + bvalid <= 1'b1; + bresp <= bresp_; + end + end + end + end + + VerifAxiWrite verif( + clock, + reset, + awfire, + awlen, + wfire, + wlast, + bfire + ); +endmodule + +module VerifAxiRead( + input bit clock, + bit reset, + bit arfire, + bit [7:0] arlen, + bit rfire, + bit rlast +); + // check transaction level state machine + + int pending_trans[$]; + int pending_beat; + +no_fire_at_reset: + assert property (@(posedge clock) reset |-> !arfire && !rfire); + +no_pending_at_reset: + assert property (@(posedge clock) reset |-> pending_trans.size() == 0 && pending_beat == 0); + + always @(posedge clock) begin + if (arfire) begin + pending_trans.push_back(int'(arlen) + 1); + end + + if (rfire) begin + if (pending_beat == 0) begin + assert_rfire: assert (pending_trans.size() > 0); + + pending_beat = pending_trans.pop_front(); + end + + assert_rlast: assert (rlast == (pending_beat == 1)); + + pending_beat -= 1; + end + end +endmodule + +module DpiAxiRead #( + parameter string TAG, + parameter integer WIDTH +)( + input logic clock, + logic reset, + + input logic arvalid, + logic [31:0] araddr, + logic [7:0] arlen, + logic [2:0] arsize, + output logic arready, + + output logic rvalid, + logic [WIDTH-1:0] rdata, + logic [1:0] rresp, + logic rlast, + input logic rready +); + import t1_common::*; + + if (!(WIDTH inside {8, 16, 32, 64, 128, 256, 512, 1024})) + $fatal(1, "invalid AXI width"); + + // delay to avoid race with init + axi_tag_t TAG_ID; + initial #1 TAG_ID = t1_axi_get_id(TAG, WIDTH, "R"); + + import "DPI-C" function bit t1_axi_ar_ready(axi_tag_t tag); + import "DPI-C" function bit t1_axi_ar_transfer( + axi_tag_t tag, + uint32_t araddr, + uint8_t arlen, + uint8_t arsize + ); + import "DPI-C" function bit t1_axi_r_valid( + axi_tag_t tag, + bit is_transfer, + output uint8_t rresp, + output bit rlast, + output uint8_t[] rdata + ); + + wire arfire = arvalid && arready; + wire rfire = rvalid && rready; + + // AR channel + always @(posedge clock) begin + if (reset) begin + arready <= 1'b0; + end + else begin + if (arready) begin + if (arvalid) begin + arready <= t1_axi_ar_transfer( + TAG_ID, + araddr, + arlen, + arsize + ); + end + end else begin + arready <= t1_axi_ar_ready(TAG_ID); + end + end + end + + // B channel + uint8_t rresp_; + bit rlast_; + uint8_t rdata_[WIDTH/8]; + always @(posedge clock) begin + if (reset) begin + rvalid <= 1'b0; + rresp <= 0; + rlast <= 1'b0; + rdata <= '0; + end + else begin + if (rfire) rvalid <= 1'b0; + + if (!rvalid || rfire) begin + if (t1_axi_b_ready( + TAG_ID, + rvalid, + rresp_, + rlast_, + rdata_ + )) begin + rvalid <= 1'b1; + rresp <= rresp_; + rlast <= rlast_; + rdata <= rdata_; + end + end + end + end + + VerifAxiRead verif( + .clock, + .reset, + .arfire, + .arlen, + .rfire, + .rlast + ); +endmodule