当前位置: 首页 > article >正文

【UVM项目实战】异步fifo—uvm项目结构以及uvm环境搭建

本文章同步到我的个人博客网站:ElemenX-King:【UVM项目实战】异步fifo—uvm项目结构以及uvm环境搭建 希望大家能使用此网站来进行浏览效果更佳!!!


目录

  • 一、异步FIFO
    • 1.1 异步FIFO的定义
    • 1.2 亚稳态
    • 1.3 异步FIFO关键技术一
    • 1.3 异步FIFO关键技术二
    • 1.4 FIFO逻辑图
  • 二、UVM结构
    • 2.1 interface
    • 2.2 transaction
    • 2.3 driver
    • 2.3 moniter
    • 2.4 sequencer
    • 2.5 agent
    • 2.6 model
    • 2.7 scoreboard
    • 2.8 env
    • 2.9 base_test
    • 2.10 case_sequence
    • 2.11 top_tb
  • 三、UVM仿真环境的搭建
    • 3.1 Makefile文件的编写
    • 3.2 filelist文件的编写
    • 3.3 另辟蹊径
  • 总结

一、异步FIFO

1.1 异步FIFO的定义

异步时序设计指的是在设计中有两个或以上的时钟, 且时钟之间是同频不同相或不同频率的关系。而异步时序设计的关键就是把数据或控制信号正确地进行跨时钟域传输。
一个异步 FIFO 一般由如下部分组成:

  1. Memory, 作为数据的存储器;
  2. 写逻辑部分,主要负责产生写信号和地址;
  3. 读逻辑部分,主要负责产生读信号和地址;
  4. 地址比较部分,主要负责产生 FIFO 空、满的标志。

跟普通的FIFO相比,异步FIFO实际上多了读写地址的跨时钟域同步的逻辑,以及两个时钟域中读写信号的比较逻辑。

1.2 亚稳态

每一个触发器都有其规定的建立(setup)和保持(hold)时间参数, 在这个时间参数内, 输入信号在时钟的上升沿是不允许发生变的。 如果在信号的建立时间中对其进行采样, 得到的结果将是不可预知的,即亚稳态。
为了避免亚稳态。采用双锁存器可以改善这一问题:
双锁存器避免亚稳态

时钟域B两级同步的寄存器跟时钟域A的输出寄存器之间不能有组合逻辑。组合逻辑电路各个输入信号的不一致性以及组合逻辑内部路径的延时时间不一样,运算后的信号存在毛刺。
注意1

1.3 异步FIFO关键技术一

这里用到了一个很重要的概念“回卷”,通常判断读写的地址是否相同来判断空和满,这里使用回卷技术,在深度为8的fifo中多一位来代表回卷位,当fifo溢出之后,回卷位会被置1,当读时钟和写时钟的回卷位不同而其他位相同时,表示fifo已经满,因为写地址在溢出后的位置,而读时钟在溢出前。
回卷

1.3 异步FIFO关键技术二

将满和将空信号实际上表示更加保守的满和空信号。基本思路是,设定一个间隔值,当读写地址之间的间隔小于或等于该间隔就产生将空或将满信号。
对于异步FIFO而言,由于同步过来的地址信号都是格雷码表示的,我们不能直接用格雷码去判断上述的这个间隔,所以需要先对接受到的格雷码进行解码变为二进制,再和当前时钟域下的另一个地址进行将满和将空的生成。

1.4 FIFO逻辑图

FIFO
下面的四个寄存器一方面是对其时序(但是我不理解为什么时序是对其的)另一方面是防止时钟域带来的亚稳态。


二、UVM结构

该项目的UVM包括以下几个文件:
fifo_if.sv
fifo_case0.sv
base_test.sv
top_tb.sv
fifo_driver.sv
fifo_model.sv
fifo_transaction.sv
fifo_in_monitor.sv
fifo_env.sv
fifo_scoreboard.sv
fifo_chk_rst.sv
fifo_in_sequencer.sv
fifo_in_agent.sv
my_env.sv
fifo_out_monitor.sv
fifo_out_agent.sv
下面是本次UVM的整体框架:
在这里插入图片描述
现在对每一个文件进行解释。

2.1 interface

该文件主要是用于连接DUT的物理信号与UVM的事件信号,起主要是在top_tb中进行定义和连接,然后通过uvm_config_db进行定点发送。

`ifndef FIFO_IF__SV
`define FIFO_IF__SV// 括号里面是时钟信号
interface fifo_if(input wclk, input rclk, input wreset_b, input rreset_b);logic write,read;logic [31 : 0] wdata;logic [31 : 0] rdata;wire  wfull,rempty; // logic/***********时钟约束************/// wdata为inoutclocking ckw @(posedge wclk);input  wfull;inout  write;inout  wdata;endclocking// wdata为inputclocking ckim @(posedge wclk);input wfull;              inout  write;             input  wdata;endclocking// clocking ckom @(posedge rclk);input  rempty;inout  read;input  rdata;endclocking/***********方向约束************/// 普通模式modport DUT(input write,input read,input wdata,output rdata,output wfull,output rempty);// 读取模式modport DRV(clocking ckw,input read,input rdata,input rempty);modport OMON(clocking ckom,input wfull,input write,input wdata);
endinterface //interfacename`endif

clocking主要是定义每个logic信号的方向,同时制定该信号同步的时钟域。括号里信号的作用是控制 clocking 块内的所有信号的同步时序。时钟信号 的上升沿会触发对这些信号的采样或更新
modport的作用是定义了一个接口的访问模式,指定了如何访问时钟块 ckw 和接口中的信号。具体来说,它允许从外部访问 read、rdata 和 rempty 信号,并且会在时钟块 ckw 中进行同步。

2.2 transaction

transaction主要是用于对信号进行打包操作,本项目只有一个输入data_in,因此只需要生成一个随机数据。同时基于约束并注册(这是基本操作)。

`ifndef FIFO_TRANSACTION__SV
`define FIFO_TRANSACTION__SVclass fifo_transaction extends uvm_sequence_item;// 产生的随机数据rand bit[31:0] data_in[];// 约束constraint data_in_c {soft data_in.size inside {[1:300]};}// 将数据加入注册`uvm_object_utils_begin(fifo_transaction)`uvm_field_array_int(data_in,UVM_ALL_ON)`uvm_object_utils_end// 构造函数function new(string name = "fifo_transaction");super.new(name);endfunction
endclass
`endif

2.3 driver

驱动器是UVM中的核心,代码如下所示:

`ifndef FIFO_DRIVER__SV
`define FIFO_DRIVER__SVclass fifo_driver extends uvm_driver#(fifo_transaction);virtual fifo_if vif;//define 功能覆盖率 是否达到 空 满 状态covergroup cov_label;option.per_instance = 1;option.auto_bin_max = 2;coverpoint vif.wfull;coverpoint vif.rempty;endgroup// 注册`uvm_component_utils(fifo_driver)// 构造函数function new(string name = "fifo_driver", uvm_component parent = null);super.new(name, parent);cov_label = new(); // 创建覆盖率endfunction// 初始化函数virtual function void build_phase(uvm_phase phase);super.build_phase(phase);if(!uvm_config_db#(virtual fifo_if)::get(this,"","vif",vif))`uvm_fatal("fifo_driver","virtual interface must be set for vif!!!")endfunctionextern task main_phase(uvm_phase phase);extern task drive_one_pkt(fifo_transaction tr);
endclasstask fifo_driver::main_phase(uvm_phase phase);`uvm_info("fifo_driver","begin!",UVM_LOW)while(1) beginseq_item_port.get_next_item(req); // 开始drive_one_pkt(req);seq_item_port.item_done(); // 结束end
endtasktask fifo_driver::drive_one_pkt(fifo_transaction tr);int data_size,j;data_size = tr.data_in.size();`uvm_info("fifo_driver","begin to drive one pkt",UVM_LOW)for(int i = 0; i < data_size; i++) begin@(vif.ckw); // 表示ckw中的内容发生变化if((!vif.ckw.wfull) && (vif.ckw.write == 1)) begin cov_label.sample(); vif.ckw.wdata <= tr.data_in[i];`uvm_info("fifo_driver",$sformatf("%0d number is sent,number is %0h",j++,vif.ckw.wdata),UVM_LOW) end  else if((!vif.ckw.wfull) && (vif.ckw.write == 0)) beginvif.ckw.write <= 1;i--;end else begin // 满了vif.ckw.write <= 0;i--;endend  
endtask`endif

关于覆盖率这里我们先不做介绍,后续会单独做期来讲解覆盖率

需要注意几点,首先在初始化函数中使用uvm_config_db来获取DUT信号,用于对DUT的信号进行写入与读取。
其次是在main_phase中,使用driver自带的端口seq_item_port来获得一个包(就是刚才说的随机数据),这个端口会在agent中进行连接,来源就是sequencer,然后将包中的信息发送到DUT上。
最后就是在drive_one_pkt中,使用@(vif.ckw);来捕获时钟上升沿,然后通过wfull和write的状态来决定是否发送数据。

可以看到,write和wdata信号均有写入和读取的操作,因此在ckw中设置为inout信号

2.3 moniter

moniter的作用在于接受DUT的信息,有在输入和输出都有一个moniter。

`ifndef FIFO_IN_MONITOR__SV
`define FIFO_IN_MONITOR__SV
class fifo_in_monitor extends uvm_monitor;virtual fifo_if vif;uvm_analysis_port #(fifo_transaction)  ap;// 注册`uvm_component_utils(fifo_in_monitor)// 构造函数function new(string name = "fifo_in_monitor", uvm_component parent = null);super.new(name, parent);endfunction// 初始化函数virtual function void build_phase(uvm_phase phase);super.build_phase(phase);if(!uvm_config_db#(virtual fifo_if)::get(this, "", "vif", vif))`uvm_fatal("fifo_in_monitor", "virtual interface must be set for vif!!!")ap = new("ap", this);endfunctionextern virtual task main_phase(uvm_phase phase);extern task collect_one_pkt(fifo_transaction tr);
endclasstask fifo_in_monitor::main_phase(uvm_phase phase);fifo_transaction tr;repeat(2) begintr = new("tr");#4.001;collect_one_pkt(tr);ap.write(tr);end
endtasktask fifo_in_monitor::collect_one_pkt(fifo_transaction tr);int j,k;`uvm_info("in_monitor","begin to collect one pkt",UVM_LOW) while(1) begin @(vif.ckim);  if((!vif.ckim.wfull) && (vif.ckim.write == 1)) begin // 符合要求tr.data_in[j] = vif.ckim.wdata;`uvm_info("in_monitor",$sformatf("%0d number is received,number is %0h",k++,vif.ckim.wdata),UVM_LOW)if(j == 199)  break;   //seq1 发送200个数据 j++;end end`uvm_info("in_monitor","end collect one pkt",UVM_LOW)//tr.print();
endtask`endif

与driver一样,也要使用uvm_config_db来获取DUT信号。
但是这里定义了一个uvm_analysis_port端口其作用就是把接收到的数据transcation发送出。
这里的#4.001大概率是为了等待刚刚发送的数据发送,这里挖一个坑
后面的collect_one_pkt与前面相同,当符合要求后将wdata读到data_in中,但是这里使用的是ckim,原因如上所示。
输出的out_moniter与输入有两个区别,一个是没有#4.001,第二个就是使用的读时钟块ckom以及传输的是rdata。

2.4 sequencer

sequencer相当于sequence的下手,用于帮sequence传递数据的。

`ifndef FIFO_SEQUENCER__SV
`define FIFO_SEQUENCER__SVclass fifo_sequencer extends uvm_sequencer #(fifo_transaction);`uvm_component_utils(fifo_sequencer)function new(string name = "fifo_sequencer",uvm_component parent = null);super.new(name,parent);endfunctiontask main_phase(uvm_phase phase);`uvm_info("fifo_sequencer","main_phase begin",UVM_LOW)endtask
endclass`endif

代码里没有要讲的,仅仅就定义了一下自己。

2.5 agent

agent的作用仅仅就是将三巨头sequencer、driver和monitor进行连接。

`ifndef FIFO_IN_AGENT__SV
`define FIFO_IN_AGENT__SVclass fifo_in_agent extends uvm_agent ;// 三巨头fifo_sequencer       sqr;fifo_driver          drv;fifo_in_monitor      mon;uvm_analysis_port #(fifo_transaction)  ap; // 指向外面`uvm_component_utils(fifo_in_agent)function new(string name = "fifo_in_agent", uvm_component parent);super.new(name, parent);endfunction extern virtual function void build_phase(uvm_phase phase);extern virtual function void connect_phase(uvm_phase phase);
endclassfunction void fifo_in_agent::build_phase(uvm_phase phase);super.build_phase(phase);if(is_active == UVM_ACTIVE) begindrv = fifo_driver::type_id::create("i_drv",this);sqr = fifo_sequencer::type_id::create("i_sqr",this); endmon = fifo_in_monitor::type_id::create("i_mon",this);
endfunctionfunction void fifo_in_agent::connect_phase(uvm_phase phase);super.connect_phase(phase);if(is_active == UVM_ACTIVE) begindrv.seq_item_port.connect(sqr.seq_item_export);endap = mon.ap; // 调用mon的ap指针
endfunction	
`endif

需要注意的是,这里和moniter一样定义了一个uvm_analysis_port 端口,并且在connect_phase中使用ap = mon.ap;的方式将mon的ap的指针指向了该ap,实现外部直接调用mon.ap。
is_active 是agent的固有方法,用来区分输入和输出。但是本项目中还定义了一个out_agent,这个方法应该冗余了。在out_agent中的mon变为out_mon,其余不变。
在connect_phase连接中将sqr的输出与drv的输入连接,等待sqr提供数据。

2.6 model

model按理来说应该是一个参考,用来实现与DUT相同的操作,但这里仅仅定义了两个端口,从port端口到ap。

`ifndef FIFO_MODEL__SV
`define FIFO_MODEL__SVclass fifo_model extends uvm_component;uvm_blocking_get_port #(fifo_transaction)  port;uvm_analysis_port #(fifo_transaction)  ap;extern function new(string name, uvm_component parent);extern function void build_phase(uvm_phase phase);extern virtual  task main_phase(uvm_phase phase);`uvm_component_utils(fifo_model)
endclass function fifo_model::new(string name, uvm_component parent);super.new(name, parent);
endfunction function void fifo_model::build_phase(uvm_phase phase);super.build_phase(phase);port = new("port", this);ap = new("ap", this);
endfunctiontask fifo_model::main_phase(uvm_phase phase);fifo_transaction tr;fifo_transaction new_tr;super.main_phase(phase);while(1) beginport.get(tr);new_tr = new("new_tr");new_tr.copy(tr);`uvm_info("fifo_model", "get one transaction, copy and print it:", UVM_LOW)new_tr.print();ap.write(new_tr);end
endtask
`endif

2.7 scoreboard

scoreboard用于收集来自agent以及model的数据。

`ifndef FIFO_SCOREBOARD__SV
`define FIFO_SCOREBOARD__SV
class fifo_scoreboard extends uvm_scoreboard;fifo_transaction  expect_queue[$];uvm_blocking_get_port #(fifo_transaction)  exp_port;uvm_blocking_get_port #(fifo_transaction)  act_port;`uvm_component_utils(fifo_scoreboard)extern function new(string name, uvm_component parent = null);extern virtual function void build_phase(uvm_phase phase);extern virtual task main_phase(uvm_phase phase);
endclass function fifo_scoreboard::new(string name, uvm_component parent = null);super.new(name, parent);
endfunction function void fifo_scoreboard::build_phase(uvm_phase phase);super.build_phase(phase);exp_port = new("exp_port", this);act_port = new("act_port", this);
endfunction task fifo_scoreboard::main_phase(uvm_phase phase);fifo_transaction  get_expect,  get_actual, tmp_tran;bit result;super.main_phase(phase);fork while (1) beginexp_port.get(get_expect);expect_queue.push_back(get_expect);endwhile (1) beginact_port.get(get_actual);if(expect_queue.size() > 0) begintmp_tran = expect_queue.pop_front();result = get_actual.compare(tmp_tran);if(result) begin `uvm_info("fifo_scoreboard", "Compare SUCCESSFULLY", UVM_LOW);endelse begin`uvm_error("fifo_scoreboard", "Compare FAILED");$display("the expect pkt is");tmp_tran.print();$display("the actual pkt is");get_actual.print();endendelse begin`uvm_error("fifo_scoreboard", "Received from DUT, while Expect Queue is empty");$display("the unexpected pkt is");get_actual.print();end endjoin
endtask
`endif

这里定义了两个接受端口,分别是exp_port以及act_port,其中exp_port来自fifo_model的输入,表示参考的数据。act_port来自o_agt,表示DUT的输出值。将这两个数值进行比较,从而判断程序是否发生错误。

2.8 env

env是整个UVM最接近顶层的存在,其主要包括三个部分:agent、model以及scoreboard。

`ifndef MY_ENV__SV
`define MY_ENV__SVclass my_env extends uvm_env;fifo_in_agent     i_agt;fifo_out_agent    o_agt;fifo_model        mdl;fifo_scoreboard   scb;uvm_tlm_analysis_fifo #(fifo_transaction) agt_scb_fifo;uvm_tlm_analysis_fifo #(fifo_transaction) agt_mdl_fifo;uvm_tlm_analysis_fifo #(fifo_transaction) mdl_scb_fifo;function new(string name = "my_env", uvm_component parent);super.new(name, parent);endfunctionvirtual function void build_phase(uvm_phase phase);super.build_phase(phase);i_agt = fifo_in_agent::type_id::create("i_agt", this);o_agt = fifo_out_agent::type_id::create("o_agt", this);i_agt.is_active = UVM_ACTIVE;o_agt.is_active = UVM_PASSIVE;mdl = fifo_model::type_id::create("mdl", this);scb = fifo_scoreboard::type_id::create("scb", this);agt_scb_fifo = new("agt_scb_fifo", this);agt_mdl_fifo = new("agt_mdl_fifo", this);mdl_scb_fifo = new("mdl_scb_fifo", this);endfunctionextern virtual function void connect_phase(uvm_phase phase);`uvm_component_utils(my_env)
endclassfunction void my_env::connect_phase(uvm_phase phase);super.connect_phase(phase);i_agt.ap.connect(agt_mdl_fifo.analysis_export);mdl.port.connect(agt_mdl_fifo.blocking_get_export);mdl.ap.connect(mdl_scb_fifo.analysis_export);scb.exp_port.connect(mdl_scb_fifo.blocking_get_export);o_agt.ap.connect(agt_scb_fifo.analysis_export);scb.act_port.connect(agt_scb_fifo.blocking_get_export); 
endfunction`endif

这里面共定义了三个fifo,用来构建三者之间数据的缓冲,具体传输方向如下所示:
fifo_model -> mdl_scb_fifo -> fifo_scoreboard
fifo_out_agent-> agt_scb_fifo-> fifo_scoreboard
fifo_in_agent -> agt_scb_fifo -> fifo_model
从这里看也看到,在fifo_model 中传递的数据,其实是来自于fifo_in_agent ,当然,最为fifo输入和输出的数据确实是相同的。

2.9 base_test

base_test才是UVM真正的顶层,他的主要作用很简单,除了例化一个env之外,还可以进行项目错误的统计工作。

`ifndef BASE_TEST__SV
`define BASE_TEST__SVclass base_test extends uvm_test;my_env         env;function new(string name = "base_test", uvm_component parent = null);super.new(name,parent);endfunctionextern virtual function void build_phase(uvm_phase phase);extern virtual function void report_phase(uvm_phase phase);`uvm_component_utils(base_test)
endclassfunction void base_test::build_phase(uvm_phase phase);super.build_phase(phase);env  =  my_env::type_id::create("env", this);
endfunctionfunction void base_test::report_phase(uvm_phase phase);uvm_report_server server;int err_num;super.report_phase(phase);server = get_report_server();err_num = server.get_severity_count(UVM_ERROR);if (err_num != 0) begin$display("TEST CASE FAILED");endelse begin$display("TEST CASE PASSED");end
endfunction`endif

在report_phase中定义了一个uvm_report_server 用于检测错误,根据错误的数目输出验证通过与否。

2.10 case_sequence

这个相当于独立于整个UVM树之外的一个部分,用于产生激励信号。

`ifndef FIFO_CASE0__SV
`define FIFO_CASE0__SVclass case0_sequence extends uvm_sequence #(fifo_transaction);fifo_transaction trans;`uvm_object_utils(case0_sequence)function new(string name = "case0_sequence");super.new(name);endfunctionvirtual task body();repeat(2) begin`uvm_info("case0_sequence","generate one transaction!",UVM_LOW)`uvm_do_with(trans,{trans.data_in.size == 200;})endendtask
endclassclass test_case0 extends base_test;`uvm_component_utils(test_case0)function new(string name = "test_case0", uvm_component parent = null);super.new(name,parent);endfunctionfunction void build_phase(uvm_phase phase);super.build_phase(phase);endfunctiontask main_phase(uvm_phase phase);case0_sequence seq;phase.raise_objection(this);#4;  seq = case0_sequence::type_id::create("l_seq");`uvm_info("case0_sequence","case1_sequence begin",UVM_LOW)seq.start(env.i_agt.sqr);#3000;phase.drop_objection(this);endtask
endclass`endif

在这里,使用uvm_do_with宏的方法实现将特定的数据传输到sequencer。但这里值得注意的是,在下面定义了一个从属于base_test的类test_case0 ,其也是UVM树的一部分,是从属于base_test的,因此与env的关系很近,但是这里其作用仅仅就是建立一个case0_sequence 并且运行,同时还是用phase.raise_objection(this)phase.drop_objection(this)定义整个sequence事件的开始与结束。在这里指定了env.i_agt.sqr为发射的地点。

2.11 top_tb

这就是最终的tb文件,代码如下,具体就不说了,无非就是例化初始化之类的操作,以及保存波形。

这里挖一个坑,针对这里asyn_fifo_chk_rst还没了解清楚。

`timescale 1ns/1ps
`include "uvm_macros.svh"import asyn_fifo_chk_rst::*;
import uvm_pkg::*;module top_tb;logic wclk,rclk,wreset_b,rreset_b;fifo_if my_if(wclk,rclk,wreset_b,rreset_b);fifo_top DUT
(.wclk		   (wclk),.rclk		   (rclk),.wreset_b	(wreset_b),     //wrst_n -> wreset_b.rreset_b	(rreset_b),	    //rrst_n -> rreset_b.write		(my_if.write),	//winc -> write.read		   (my_if.read),	//rinc -> read.wdata		(my_if.wdata),.wfull		(my_if.wfull),.rempty		(my_if.rempty),.rdata		(my_if.rdata)
);fifo_rst_mon fifo_rst_mon1;
fifo_chk_rst fifo_chk_rst1;
event reset_e_w;
event reset_e_r;initial beginwclk = 0;rclk = 0;wreset_b = 1;rreset_b = 1;#2 wreset_b = 0;rreset_b = 0;my_if.write = 0;my_if.read = 0;#2 wreset_b = 1;rreset_b = 1;
endalways #1 wclk = ~wclk;
always #3 rclk = ~rclk;// 参数传递
initial beginuvm_config_db#(virtual fifo_if)::set(null,"uvm_test_top.env.i_agt.i_drv","vif",my_if);uvm_config_db#(virtual fifo_if)::set(null,"uvm_test_top.env.i_agt.i_mon","vif",my_if);uvm_config_db#(virtual fifo_if)::set(null,"uvm_test_top.env.o_agt.o_mon","vif",my_if);
end// 参考?
initial beginfifo_rst_mon1 = new(reset_e_w,reset_e_r);fifo_chk_rst1 = new(reset_e_w,reset_e_r);fifo_rst_mon1.my_if6 = my_if;fifo_chk_rst1.my_if7 = my_if;forkfifo_rst_mon1.run();fifo_chk_rst1.run();join
endinitial begin$fsdbDumpfile("tb.fsdb");$fsdbDumpvars;$fsdbDumpon;
endinitial beginrun_test("test_case0");
endendmodule

三、UVM仿真环境的搭建

这里我简单说一下,因为在环境搭建的过程中踩了很多坑,在网上的教程很乱,也没有一个完美的答案,这里就介绍一下我搭建UVM的结构图:
结构图
可以看出,我这里建立了三个文件夹,有源码DUT部分,存放仿真文件以及启动文件的sim还有存放tb文件和uvm文件的testbench文件。最重要的就是启动文件filelist和Makefile的编写。

3.1 Makefile文件的编写

先给出我的Makefile文件:

.PHONY:file vcs sim verdi cleanVCS = vcs -full64 -cpp g++-4.8 -cc gcc-4.8 \-LDFLAGS -Wl,--no-as-needed \-f filelist.f \-timescale=1ns/1ps \-R \-debug_acc+all \+define+FSDB \-lca -kdb \-ntb_opts uvm-1.1 \-sverilog \+v2k \|tee vcs.logfile:find ../ -name "*.v" -o -name "*.sv" > file.fvcs:${VCS}sim:./simv -gui |tee sim.log &verdi:verdi -f filelist.f -sv -ssf tb.fsdb &clean:rm -rf csrc verdiLog simv.daidir \novas.* \vc_hdrs.h \simv \*.key \*.fsdb \*.log \inter.vpd \DVEfiles

这里包括了五个部分,分别是file、vcs、sim、verdi以及clean。我分开来介绍:

  • file:
    这个主要是用于生成filelist文件,但也不全是filelist,因为该脚本只能获取所有的.v和.sv文件,在filelist中的编写不仅仅要包含这个,而且还有uvm包的文件,并且这些文件的先后顺序有严格的要求,这个后面讲解。
  • vcs:
    这个主要是启动vcs对所有的文件进行编译,在Makefile中添加了很多附加选项,这些选项都是能够让vcs正常运行的选项,你可以使用vcs help来获取vcs命令手册,里面解释了所有符号的意义以及用法。
# 指定使用 VCS (Verilog Compiler Simulator) 进行仿真编译
VCS = vcs -full64 -cpp g++-4.8 -cc gcc-4.8 \# -full64: 使用 64 位编译选项,适用于 64 位操作系统# -cpp g++-4.8: 指定 C++ 编译器使用 g++ 版本 4.8# -cc gcc-4.8: 指定 C 编译器使用 gcc 版本 4.8-LDFLAGS -Wl,--no-as-needed \# -LDFLAGS: 启用链接器的标志# -Wl,--no-as-needed: 在链接时告诉链接器不要自动移除未使用的库-f filelist.f \# -f filelist.f: 指定仿真源文件列表,filelist.f 是一个包含所有待编译文件的文件列表-timescale=1ns/1ps \# -timescale=1ns/1ps: 设置仿真时间尺度为 1ns(纳秒)/ 1ps(皮秒)-R \# -R: 启用调试和恢复功能-debug_acc+all \# -debug_acc+all: 启用所有的调试访问器,允许调试仿真时查看所有信号和变量+define+FSDB \# +define+FSDB: 定义一个名为 FSDB 的宏,通常用于控制 FSDB 文件(仿真波形文件)的输出-lca -kdb \# -lca: 启用 LCA (Library Coverage Analysis),用于库覆盖分析# -kdb: 启用 KDB (Kernel Debugger),为调试目的启用内核调试功能-ntb_opts uvm-1.1 \# -ntb_opts uvm-1.1: 启用与 UVM 1.1 兼容的 NTB(Native Testbench)选项,适用于 UVM(Universal Verification Methodology)验证环境-sverilog \# -sverilog: 启用 SystemVerilog 编译支持+v2k \# +v2k: 启用 Verilog-2001(V2K)编译选项|tee vcs.log# |tee vcs.log: 使用 tee 命令将 VCS 编译过程的输出同时显示在终端并保存到 vcs.log 文件中
  • sim:用来使用vcs自带的仿真工具生成波形,这里我没用过。
  • verdi:用来查看fsdb的波形文件,这个是我最常用的,通过这个可以很方便的对波形进行追溯,调试起来很容易。
  • clean:用来删除生成了的文件。

以防有人不会用Makefile,说一下他的用法。在Makefile文件所在的目录打开终端,使用make [指令]的方式来运行,Makefile本质来说就是将指令进行了一个打包,我这里运行verdi -f filelist.f -sv -ssf tb.fsdb &make verdi的效果是一样的。

3.2 filelist文件的编写

下面是我针对本项目编写的filelist文件。

+incdir+$UVM_HOME/src
$UVM_HOME/src/uvm_pkg.sv../testbench/fifo_if.sv
../testbench/fifo_transaction.sv
../testbench/fifo_driver.sv
../testbench/fifo_in_sequencer.sv
../testbench/fifo_in_monitor.sv
../testbench/fifo_out_monitor.sv
../testbench/fifo_in_agent.sv
../testbench/fifo_out_agent.sv
../testbench/fifo_model.sv
../testbench/fifo_scoreboard.sv
../testbench/my_env.sv
../testbench/fifo_chk_rst.sv
../testbench/base_test.sv
../testbench/fifo_case0.sv
../testbench/top_tb.sv../DUT/pointer.v
../DUT/sync.v
../DUT/fifo_top.v
../DUT/memory.v
../DUT/comparator.v

可以分为三个部分,首先是前两行的+incdir+$UVM_HOME/src$UVM_HOME/src/uvm_pkg.sv,第一句话表示将$UVM_HOME/src中的所有文件加入到编译列表中,这个文件里基本上是所有uvm库所需要的文件,什么uvm_env呀之类的就是这里定义。然后uvm_pkg.sv相当于是所有文件的核心实现,在tob_tb中,只需要调用import uvm_pkg::*;便可以将所有uvm包含进来。

$UVM_HOME表示的是一个宏定义,在Lunix中主目录的.bashrc中定义,我这里的原话是export UVM_HOME=/home/wxm/uvm_study/uvm-1.1d
后面紧接着是testbench中的文件,注意一定要先写testbench再写dut,以防报错。然后在testbench内部也要按照顺序,从独立到树枝再到树根的顺序,比如献血interface和transaction,再从树枝的dirver开始写到树根base_case。因为在UVM编译的过程中是从上到下的顺序,如果你先编译base_case,编译器会报错说找不到env。最后就是top_tb以及其余的DUT文件。
但是貌似DUT文件没有内部顺序

3.3 另辟蹊径

还有一个别的方法,你只需要写两个文件就可以:一个是top_tb.sv;另一个是fifo_top.v,前提是你需要在这两个文件里面把其余所有的文件include一遍,就像这样:

`include "../testbench/my_driver.sv"
`include "../testbench/my_model.sv"
...

因为你的Makefile和filelist以及终端运行的位置都在sim文件夹,而其他文件在testbench和dut文件夹,因此你在include的时候需要以sim文件夹为根目录,使用../回到上一级,再使用/testbench/XXX.sv来调用这些文件。这样操作就不需要在filelist中调整顺序了。


总结

总而言之,这是一个很好的练习UVM的项目,因为其与《UVM实战》这本书的内容大差不差,很多在结构上都有相似的地方,我希望从这个项目为起点,依次加深我对IC验证这一领域的认识。后面我会对该项目的波形图进行研究,并通过调整UVM代码实现一些不一样的功能。

相关文章:

【UVM项目实战】异步fifo—uvm项目结构以及uvm环境搭建

本文章同步到我的个人博客网站&#xff1a;ElemenX-King&#xff1a;【UVM项目实战】异步fifo—uvm项目结构以及uvm环境搭建 希望大家能使用此网站来进行浏览效果更佳&#xff01;&#xff01;&#xff01; 目录 一、异步FIFO1.1 异步FIFO的定义1.2 亚稳态1.3 异步FIFO关键技术…...

【通关函数的递归】--递归思想的形成与应用

目录 一.递归的概念与思想 1.定义 2.递归的思想 3.递归的限制条件 二.递归举例 1.求n的阶乘 2.顺序打印一个整数的每一位 三.递归与迭代 前言:上篇博文分享了扫雷游戏的实现&#xff0c;这篇文章将会继续分享函数的递归相关知识点&#xff0c;让大家了解并掌握递归的思…...

AI日报 - 2025年04月25日

&#x1f31f; 今日概览(60秒速览) ▎&#x1f916; AGI突破 | OpenAI o3模型展现行动能力&#xff0c;英国发布RepliBench评估AI自主复制风险&#xff0c;DeepMind CEO担忧AGI协调挑战。 模型能力向行动和自主性演进&#xff0c;安全与协调成为焦点。 ▎&#x1f4bc; 商业动向…...

【FAQ】针对于消费级NVIDIA GPU的说明

概述 本文概述 HP Anyware 在配备消费级 NVIDIA GPU 的物理工作站上的关​​键组件、安装说明和重要注意事项。 注意&#xff1a;本文档适用于 NVIDIA 消费级 GPU。NVIDIA Quadro 和 Tesla GPU 也支持 HP Anyware 在公有云、虚拟化或物理工作站环境中运行。请参阅PCoIP Graphi…...

几种查看PyTorch、cuda 和 Python 版本方法

在检查 PyTorch、cuda 和 Python 版本时&#xff0c;除了直接使用 torch.__version__ 和 sys.version&#xff0c;我们还可以通过其他方式实现相同的功能 方法 1&#xff1a;直接访问属性&#xff08;原始代码&#xff09; import torch import sysprint("PyTorch Versi…...

网络安全 | F5 WAF 黑白名单配置实践指南

关注&#xff1a;CodingTechWork 引言 在现代网络安全架构中&#xff0c;F5 Web Application Firewall (WAF) 是保护 Web 应用免受攻击的重要工具。F5 WAF 提供了强大的黑白名单功能&#xff0c;结合 Data Group 和 iRules&#xff0c;可以实现更灵活、更高效的流量控制策略。…...

焊接机排错

焊接机 一、前定位后焊接 两个机台&#xff0c;①极柱定位&#xff0c;相机定位所有极柱点和mark点&#xff1b;②焊接机&#xff0c;相机定位mark点原理&#xff1a;极柱定位在成功定位到所有极柱点和mark点后&#xff0c;可以建立mark点和极柱点的关系。焊接机定位到mark点…...

【AI提示词】艺人顾问

提示说明 专业艺人顾问&#xff0c;专注于为客户提供全面的艺术、娱乐和商业咨询服务&#xff0c;帮助他们在竞争激烈的行业中树立品牌影响力&#xff0c;提升市场竞争力 提示词 # Role: 艺人顾问## Profile - language: 中文 - description: 专业艺人顾问&#xff0c;专注于…...

MyBatis操作数据库---从入门到理解

文章目录 关于MyBatis操作数据库MyBatis⼊⻔&#xff08;使用&#xff09;Mybatis操作数据库的步骤&#xff1a;配置数据库连接字符串使⽤MyBatis完成简单的增删改查操作注解xml 单元测试开启驼峰命名(推荐) 打印日志 关于MyBatis操作数据库 在之前的学习,我们了解到web应⽤程…...

本地缓存大杀器-Caffeine

本地缓存大杀器-Caffeine 一、 背景二、 应用三、 实现原理四、 核心设计五、 总结 一、 背景 1、 本地缓存作为一种高效的缓存方式&#xff0c;能够显著减少对远程数据源的访问&#xff0c;从而快速响应请求。而在众多本地缓存工具中&#xff0c;Caffine 凭借其卓越的性能和丰…...

【HFP】蓝牙语音通话控制深度解析:来电拒接与通话终止协议

目录 一、来电拒接的核心流程与信令交互 1.1 拒接场景的分类与触发条件 1.2 HF 端拒接流程 1.3 AG 端拒接流程 二、通话终止流程&#xff1a;主动断开与异常中断 2.1 终止场景的界定 2.2 HF 端终止流程 2.3 AG 端终止流程 三、信令协议的核心要素&#xff1a;AT 命令与…...

使用QML Tumbler 实现时间日期选择器

目录 引言相关阅读项目结构示例实现与代码解析示例一&#xff1a;时间选择器&#xff08;TimePicker&#xff09;示例二&#xff1a;日期时间选择器&#xff08;DateTimePicker&#xff09; 主窗口整合运行效果总结下载链接 引言 在现代应用程序开发中&#xff0c;时间与日期选…...

智能吸顶灯/摄影补光灯专用!FP7195双通道LED驱动,高效节能省空间 !

一、双路调光技术背景与市场需求 随着LED照明技术的快速发展和智能照明需求的激增&#xff0c;双路调光技术正成为照明行业的重要发展方向。传统单路调光方案只能实现整体亮度的统一调节&#xff0c;而双路调光则能够实现对两个独立通道的精确控制。今天&#xff0c;由我来为大…...

如何解决PyQt从主窗口打开新窗口时出现闪退的问题

在PyQt5中&#xff0c;当从主窗口打开新窗口时&#xff0c;经常会出现闪退现象&#xff0c;这通常是由于对象生命周期管理不当或事件循环错误等所导致。 1. 确保新窗口实例被正确引用 新窗口的实例若未被主窗口引用&#xff0c;可能会被Python的垃圾回收机制销毁。 错误示例&…...

分布式微服务架构,数据库连接池设计策略

在分布式微服务架构中&#xff0c;数据库连接池的设计远比单体应用复杂&#xff0c;涉及资源隔离、连接管理、性能调优和高可用等问题。下面是面向专业软件架构师的系统化分析与策略建议&#xff1a; 一、核心挑战 每个服务独立运行&#xff0c;连接池分散 每个微服务维护自己的…...

YOLOv11改进-双Backbone架构:利用双backbone提高yolo11目标检测的精度

一、引言&#xff1a;为什么我们需要双Backbone&#xff1f; 在目标检测任务中&#xff0c;YOLO系列模型因其高效的端到端检测能力而备受青睐。然而&#xff0c;传统YOLO模型大多采用单一Backbone结构&#xff0c;即利用一个卷积神经网络&#xff08;CNN&#xff09;作为特征提…...

redis经典问题

1.缓存雪崩 指缓存同一时间大面积的失效&#xff0c;所以&#xff0c;后面的请求都会落到数据库上&#xff0c;造成数据库短时间内承受大量请求而崩掉。 解决方案&#xff1a; 1&#xff09;Redis 高可用&#xff0c;主从哨兵&#xff0c;Redis cluster&#xff0c;避免全盘崩…...

《逃离云端束缚,拥抱GPT本地部署》

《逃离云端束缚,拥抱GPT本地部署》 一、GPT 热潮与本地部署的兴起 自 OpenAI 推出 ChatGPT 以来,全球范围内掀起了一股人工智能的热潮,其强大的自然语言处理能力和广泛的应用场景,让人们对人工智能的未来充满了想象。GPT(Generative Pretrained Transformer)作为一种基于…...

头歌之动手学人工智能-机器学习 --- PCA

目录 第1关&#xff1a;维数灾难与降维 第2关&#xff1a;PCA算法流程 任务描述 编程要求 测试说明 第3关&#xff1a;sklearn中的PCA 任务描述 编程要求 测试说明 第1关&#xff1a;维数灾难与降维 第2关&#xff1a;PCA算法流程 任务描述 本关任务&#xff1a;补充…...

研0调研入门

一、Web of Science 使用教程 1. 访问与注册 访问入口&#xff1a;通过高校图书馆官网进入&#xff08;需IP权限&#xff09;&#xff0c;或直接访问 Web of Science官网。注册/登录&#xff1a;若机构已订阅&#xff0c;用学校账号登录&#xff1b;个人用户可申请试用或付费…...

神经网络基础[ANN网络的搭建]

神经网络 人工神经网络&#xff08; Artificial Neural Network&#xff0c; 简写为ANN&#xff09;也简称为神经网络&#xff08;NN&#xff09;&#xff0c;是一种模仿生物神经网络结构和功能的计算模型。各个神经元传递复杂的电信号&#xff0c;树突接收到输入信号&#xf…...

五、web自动化测试01

目录 一、HTML基础1、HTML介绍2、常用标签3、基础案例3.1 前端代码3.2 自动化测试 二、CSS定位1、css介绍2、案例3、代码优化 三、表单自动化1、案例2、元素属性定位 四、后台基础数据自动化1、登录1.1 id与class定位1.2 定位一组元素 2、商品新增 一、HTML基础 可参考学习 链…...

数据库监控 | MongoDB监控全解析

PART 01 MongoDB&#xff1a;灵活、可扩展的文档数据库 MongoDB作为一款开源的NoSQL数据库&#xff0c;凭借其灵活的数据模型&#xff08;基于BSON的文档存储&#xff09;、水平扩展能力&#xff08;分片集群&#xff09;和高可用性&#xff08;副本集架构&#xff09;&#x…...

STM32F407使用ESP8266实现阿里云OTA(中)

文章目录 前言一、程序分析二、程序讲解1. main函数2. Get_Version()函数3. esp_Init()函数4. Check_Updata()函数结语前言 从上一章STM32F407使用ESP8266实现阿里云OTA(上)中我们已经对连接阿里云和从阿里云获取升级包的流程非常的熟悉了。所以本章我们进行STM32的程序开发…...

sql server 与navicat测试后,连接qt

先用Navicat测试和sql的连通性&#xff0c;Navicat和sql连通之后&#xff0c;qt也能和sql连通了。 Navicat和Sqlserver Management 能连上&#xff0c;项目无法连接本地 Navicat 连接SQLServer 数据库 QT国内镜像网站 Navicat连接SqlServer的问题点 Sql Server的基本配置以及使…...

Django 入门实战:从环境搭建到构建你的第一个 Web 应用

Django 入门实战&#xff1a;从环境搭建到构建你的第一个 Web 应用 恭喜你选择 Django 作为你学习 Python Web 开发的起点&#xff01;Django 是一个强大、成熟且功能齐全的框架&#xff0c;非常适合构建中大型的 Web 应用程序。本篇将通过一个简单的例子&#xff0c;带你走完…...

ROS2---时间戳对齐

一、ROS2时间系统架构 时间模型 仿真时间&#xff08;Simulation Time&#xff09;&#xff1a;由/clock话题驱动&#xff0c;适用于离线仿真与调试。真实时间&#xff08;Real Time&#xff09;&#xff1a;基于系统硬件时钟&#xff0c;支持PTP协议&#xff08;IEEE 1588&…...

Sublime Text相关设置

一直知道Sublime Text的自由度很高&#xff0c;但是之前使用从未更改过配置&#xff0c;有一天突然想改改设置试一下&#xff0c;感觉打开了新大陆&#xff0c;特此记录一下 设置默认语法 单击 Tools→Developer→New Snippet 弹出一个窗口&#xff0c;把下面这段代码粘贴进去…...

微信小程序 tabbar底部导航栏

官方文档&#xff1a;https://developers.weixin.qq.com/miniprogram/dev/reference/configuration/app.html#tabBar 一、常规菜单格式 在app.json 文件中配置&#xff0c;其他关键点详见官方文档&#xff0c;后续更新不规则图标的写法...

【Maven】项目管理工具

Maven&#xff1a;一个项目管理工具 前言 传统项目管理存在的问题&#xff1a; 依赖管理混乱 需要自己去网上搜 jar 包&#xff0c;找对版本很痛苦&#xff08;还容易找错&#xff09;某个库依赖另一个库&#xff08;传递依赖&#xff09;&#xff0c;你得自己挨个找齐不小心…...