路科验证UVM入门与进阶详解实验3

news2025/1/17 3:00:07

在UVM——lab3中,将SV环境移植到UVM的重点内容如下:

  1. TLM的单向、多向通信端口使用
  2. TLM的通信管道
  3. UVM的回调类型 uvm_callback
  4. UVM的一些仿真控制函数
  5. TLM单向、多向通信

1、何为事务级、何为事务级通信(TLM)?有什么用?

事务级:一个transaction就是把具有某一特定功能的一组信息封装在一起而形成的一个类。

事务级通信:在TLM中,使用事务对不同模块之间的通信进行建模;即使DUT的实际接口由信号级别的活动表示,但大多数验证任务(如生成激励,功能检查,收集覆盖率数据等)都可以在事务级别上更好地完成,只要使它们独立于实际信号级别的细节即可 。

作用:

  • TLM端口是提高数据通信抽象级的必要条件,同时使得组件之间的通信频率下降,提高了整体的仿真效率。失去标准化的TLM通信,那么就算UVM环境构建好,也无法完成高效的仿真任务。
  • 通信函数可以定制化:比如可以定制get()、peek()、put()的内容和参数,比mailbox通信更加灵活。
  • 降低组件之间的依赖性;降低组件和环境之间的耦合度。比如,也就是initiator一端不用去关心谁会和它连接,和它连接的有几个端口,只需要正常声明、例化端口即可,方法名用put/get/peek就行,不需要加后缀。而这些需要修改的部分在target一端进行即可。
  • 标准化的TLM通信为UVM和其他验证方法混合使用提供了统一对话的基础。

2、何为单向、多向?

单向:从initiator到target的数据流是单向的,或者说initiator和target智能搬移producer和consumer中的一个角色。

多向:多向通信仍是两个组件之间的通信,指如果initiator与target之间的相同TLM端口数目超过1个的情况。比如下图组件1和组件2分别有两个uvm_blocking_put_port和uvm_blocking_put_imp端口。虽然我们可以给端口例化不同的名字,连接也可以通过不同名字来索引,但是端口对应的task名都是put,会产生冲突。

通过端口宏定义就可以解决上述问题,即让不同的端口对应不同命的任务来解决方法名冲突问题。

下面代码为示例:

//解决target一端方法名冲突的几个要点
//端口宏声明
`uvm_blocking_put_imp_deck(_p1)
`uvm_blocking_put_imp_deck(_p2)

//端口声明
uvm_blocking_put_imp_p1 #(type, comp2) bp_imp_p1;
uvm_blocking_put_imp_p2 #(type, comp2) bp_imp_p2;

//端口例化
function new(string name, uvm_component parent);
  super.new(name, parent);
  bt_imp_p1 = new("bt_imp_p1", this);
  bt_imp_p2 = new("bt_imp_p2", this);
……
endfunction

//不同名put任务的定义 
task put_p1(itrans t);
	……
	key.put();
endtask

task put_p2(itrans t);
	……
	key.put();
endtask

注意:只有端口类型为imp(target)的才需要通过端口宏声明来解决方法名冲突的问题,而port不需要关注对方名字,直接用put就行。这是因为initiator一端不用去关心谁将会和它连接,有几个端口,只要正常地声明、例化端口即可,方法就用put\get\peek等就行,不用加后缀。而这些需要修改的部分只在target端进行即可。

假设我们在initiator这边的端口也将方法名设置为put_p1等有后缀的,那么一来会报错,二来下次你连接的target只要一个端口,那么你到时又需要修改。这就是耦合度高了,不利于维护。

TLM通信步骤:

  • 搞清楚控制流和数据流——分辨initiator(通信发起端)和target(通信接收方),producer(数据流出方)和consumer(数据流入方);
  • 在target中实现TLM通信方法,以方便initiator调用target中的通信方法实现数据传输;
  • 在两个对象中创建TLM端口;
  • 在更高层次中将两个对象的端口进行连接。一般做法及注意事项如下:

1、声明port和export端口,只需要指定transaction参数类型。而声明imp端口,则需要同时指定transaction参数类型以及所在的component类型。因为端口只是作为通信的管道,最终的操作还是要落实到component上,所以需要指定所在的component类型;

2、在initiator端例化port,在中间层次例化export,在target端例化imp;

实验部分:

之前的monitor到checker的通信,以及checker与refmod的通信都是通过mailbox以及在上层进行句柄传递实现的。接下来我们将采用TLM端口进行通信。
图1所示,我们将在monitor、checker和refmod上添加若干TLM的通信端口,完成下列要求:

1、将monitor中的用来与checker中的mailbox通信的mon_mb句柄替换为对应的uvm_blocking_put_port类型。

以chnl_monitor为例。

	// mailbox #(mon_data_t) mon_mb;
    uvm_blocking_put_port #(mon_data_t) mon_bp_port;

2、在checker中声明与monitor通信的import端口类型,以及reference model通信的import端口类型。由于checker与多个monitor以及reference model通信(数据从多个组件流入checker),是典型的多方向通信类型,因此需要使用多端口通信的宏声明方法。在使用了宏声明端口类型之后,再在checker中声明其句柄,并且完成例化。

	//宏声明
  `uvm_blocking_put_imp_decl(_chnl0)
  `uvm_blocking_put_imp_decl(_chnl1)
  `uvm_blocking_put_imp_decl(_chnl2)
  `uvm_blocking_put_imp_decl(_fmt)  
  `uvm_blocking_put_imp_decl(_reg)

  `uvm_blocking_get_peek_imp_decl(_chnl0)
  `uvm_blocking_get_peek_imp_decl(_chnl1)
  `uvm_blocking_get_peek_imp_decl(_chnl2)
  
  `uvm_blocking_get_imp_decl(_reg)
	//句柄声明
   uvm_blocking_put_imp_chnl0 #(mon_data_t, this) chnl0_bp_imp;
   uvm_blocking_put_imp_chnl1 #(mon_data_t, this) chnl1_bp_imp;
   uvm_blocking_put_imp_chnl2 #(mon_data_t, this) chnl2_bp_imp;
   uvm_blocking_put_imp_fmt #(fmt_trans, this) fmt_bp_imp;
   uvm_blocking_put_imp_reg #(reg_trans, this) reg_bp_imp;

   uvm_blocking_get_peek_imp_chnl0 #(mon_data_t, this) chnl0_bgpk_imp;
   uvm_blocking_get_peek_imp_chnl1 #(mon_data_t, this) chnl1_bgpk_imp;
   uvm_blocking_get_peek_imp_chnl2 #(mon_data_t, this) chnl2_bgpk_imp;

   uvm_blocking_get_imp_reg #(reg_trans, this) reg_bg_imp;
	
	//例化
	chnl0_bp_imp = new("chnl0_bp_imp", this);
    chnl1_bp_imp = new("chnl1_bp_imp", this);
    chnl2_bp_imp = new("chnl2_bp_imp", this);
    fmt_bp_imp = new("fmt_bp_imp", this);
    reg_bp_imp = new("reg_bp_imp", this);

    chnl0_bgpk_imp = new("chnl0_bgpk_imp", this);
    chnl1_bgpk_imp = new("chnl1_bgpk_imp", this);
    chnl2_bgpk_imp = new("chnl2_bgpk_imp", this);
    
    reg_bg_imp = new("reg_bg_imp", this);	

3、根据声明的import端口类型,实现其对应的方法。


//put
	task put_chnl0(mon_data_t t);
      chnl_mbs[0].put(t);
    endtask
    task put_chnl1(mon_data_t t);
      chnl_mbs[1].put(t);
    endtask
    task put_chnl2(mon_data_t t);
      chnl_mbs[2].put(t);
    endtask
    task put_fmt(fmt_trans t);
      fmt_mb.put(t);
    endtask
    task put_reg(reg_trans t);
      reg_mb.put(t);
    endtask  	
//peek、get
	task peek_chnl0(output mon_data_t t);
      chnl_mbs[0].put(t);
    endtask
    task peek_chnl1(output mon_data_t t);
      chnl_mbs[1].put(t);
    endtask
    task peek_chnl2(output mon_data_t t);
      chnl_mbs[2].put(t);
    endtask
    task get_chnl0(output mon_data_t t);
      chnl_mbs[0].get(t);
    endtask
    task get_chnl1(output mon_data_t t);
      chnl_mbs[1].get(t);
    endtask
    task get_chnl2(output mon_data_t t);
      chnl_mbs[2].get(t);
    endtask
    task get_reg(output reg_trans t);
      reg_mb.get(t);
    endtask

4、根据图1,继续在mcdf_refmod中声明用来与mcdf_checker中的import连接的端口,并且完成例化。同时注意其端口类型的区别。

	//声明端口
    // mailbox #(reg_trans) reg_mb;
    // mailbox #(mon_data_t) in_mbs[3];
    uvm_blocking_get_port #(reg_trans) reg_bg_port;
    uvm_blocking_get_peek_port #(mon_data_t) in_bgpk_ports[3];

	//例化
	 reg_bg_port = new("reg_bg_port", this);
     foreach(in_bgpk_ports[i]) in_bgpk_ports[i] = new(sformatf("in_bgpk_ports[%0d]",i),this);

5、在mcdf_env的connect_phase()阶段,完成monitor的TLM port 与mcdf_checker TLM import的连接

      chnl_agts[0].monitor.mon_bp_port.connect(chker.chnl0_bp_imp);
      chnl_agts[1].monitor.mon_bp_port.connect(chker.chnl1_bp_imp);
      chnl_agts[2].monitor.mon_bp_port.connect(chker.chnl2_bp_imp);
      reg_agt.monitor.mon_bp_port.connect(chker.reg_bg_imp);
      fmt_agt.monitor.mon_bp_port.connect(chker.fmt_bp_imp);

注意:只有initiator才能调用connect函数,target则作为connect的参数。**此处initiator是monitor,它们要将数据put到checker里,所以是monitor来调用connect函数,而checker作为参数。

6、在mcdf_checker的connect_phase阶段,完成mcdf_refmod的TLM port与mcdf_checker的TLM import的连接。

	  // foreach(this.refmod.in_mbs[i]) begin
      //   this.refmod.in_mbs[i] = this.chnl_mbs[i];
      //   this.exp_mbs[i] = this.refmod.out_mbs[i];
      // end
      this.refmod.in_bgpk_ports[0].connect(this.chnl0_bgpk_imp);
      this.refmod.in_bgpk_ports[1].connect(this.chnl1_bgpk_imp);
      this.refmod.in_bgpk_ports[2].connect(this.chnl2_bgpk_imp);

      this.refmod.reg_bg_port.connect(this.reg_bp_imp);   

同样,此处的initiator是refmod,他们要将数据get过来,所以是refmod来调用connect函数。

TLM通信管道

完成上述试验后感觉到工作量增加了不少,比如需要自己去实现get、put、peek方法。有没有既可以使用TLM端口,又不用自己去实现的方法呢?有,那就是通信管道!!!

TLM通信管道的作用?

可以不再target一端实现传输方法,也可以享受到TLM的好处

几个TLM组件和端口可以帮助用户免除这些烦恼:

  • TLM FIFO
  • analysis port
  • analysis TLM FIFO
  • request&response 通信管道

TLM_FIFO

FIFO的本质是一块缓存加两个IMP。

引入原因:

  • consumer在没有分析transaction时,希望先存储到本地FIFO中,稍后使用
  • TLM FIFO uvm_tlm_fifo类是一个新组件,继承于uvm_componnet类,预先内置了多个端口以及实现了多个对应方法,用户无需自己实现这些方法,同时又能够享受TLM端口的好处。(注意,uvm_tlm_fifo只是声明了这些端口,当我们要使用的时候还是要例化的)

uvm_tlm_fifo组件内部实现

  • 内置了一个没有尺寸限制的mailbox#(T),用来存储数据类型T。而uvm_tlm_fifo的多个端口对应的方法均是利用该mailbox实现了数据读写;
  • 也就是说,功能类似于mailbox,不同之处在于其提供了各种端口供用户使用

 

常用端口

  • put_export:用户可以通过该端口调用put、try_put、can_put
  • put_ap:调用了put方法写入的数据同时也会通过该端口的write函数送出
  • get_peek_export:用户可通过该端口调用get、try_get、can_get、peek、try_peek、can_peek
  • get_ap:调用了get和peek方法读出的数据也会通过该端口的write函数送出

analysis端口

作用:

    如果数据源端发生变化需要通知和它关联的多个组件,就可以用到analysis端口,可以满足一端到多端的通信

分类:

    uvm_analysis_port、uvm_analysis_export、uvm_analysis_imp

analysis port类型端口连接实例

说明:

  • 这是从一个initiator端到多个target端的方式。analysis port采用的是“push"模式,也就是从initiator端调用多个target端的write函数实现数据传输;
  • 在initiator调用write函数时,实际上是通过循环的方式将所有连接的target内置的write函数进行调用。由于函数是立即返回的,所以无论连接多少个target,initiator端调用write函数总是可以立即返回
  • 和单一端口函数调用不同的是,即使没有和target相连,调用write函数时也不会发生错误

analysis_port/analysis_export和port/export端口的区别
1、可连接imp数量

  • analysis_port、analysis_export更像是一个广播,可以一对多

2、方法区别

  • analysis_port、analysis_export没有阻塞、非阻塞的概念,因为本身就是广播,不必等待相连端口的响应(感觉也可以理解为非阻塞,因为马上返回)
  • analysis_port、analysis_export只有write一种操作,而port、export端口有put/get/peek等操作
     

analysis TLM FIFO

  • uvm_tlm_analysis_fifo继承于uvm_tlm_fifo,既有面向单一TLM端口的数据缓存特性,又可以实现一对多通信

 如果想轻松实现一端到多端的数据传输,可以如下插入多个uvm_tlm_analysis_fifo

  • 将initiator的analysis port连接到tlm_analysis_fifo的get_export端口,这样数据可以从initiator发起,写入到各个tlm_analysis_fifo的缓存中
  • 将多个target的get_port连接到tlm_analysis_fifo的get_export,注意保持端口类型的匹配,这样从target一侧只需要调用get方法就可以得到先前存储在tlm_analysis_fifo中的数据

实验

1、将原本在mcdf_refmod中的out_mb替换为uvm_tlm_fifo类型,并且完成例化,以及对应的变量名替换;

//TODO-2.1 replace the out_mbs[3] with uvm_tlm_fifo type 
   // mailbox #(fmt_trans) out_mbs[3];
   uvm_tlm_fifo #(fmt_trans) out_tlm_fifos[3];
//TODO-2.1 instantiate the TLM fifos
    // foreach(this.out_mbs[i]) this.out_mbs[i] = new();
    foreach(out_tlm_fifos[i]) out_tlm_fifos[i] = new($sformatf("out_tlm_fifos[%0d]",i),this);     
//TODO-2.1 replace the out_mbs[3] with uvm_tlm_fifo type 
    // this.out_mbs[id].put(ot);
    this.out_tlm_fifos[id].put(ot);

2、将原本在mcdf_checker中的exp_mbs[3]的邮箱句柄数组,替换为uvm_blocking_get_port类型句柄数组,并做相应的例化以及变量名替换;

//TODO-2.2 replace exp_mbs[3] with TLM uvm_blocking_get_port type
    // mailbox #(fmt_trans) exp_mbs[3];
    uvm_blocking_get_port #(fmt_trans) exp_bg_ports[3];
//TODO-2.2 instantiate the TLM blocking_get ports
    foreach(exp_bg_ports[i]) exp_bg_ports[i] = new($sformatf("exp_bg_ports[%0d]",i),this);    
//TODO-2.2 replace the exp_mbs with the TLM ports
    // this.exp_mbs[mont.ch_id].get(expt);
    this.exp_bg_ports[mont.ch_id].get(expt);

3、在mcdf_checker中,完成在mcdf_checker中的TLM port端口到mcdf_refmod中的uvm_tlm_fifo自带的blocking_get_export端口的连接。

//TODO-2.3 connect the TLM blocking_get ports to the blocking_get
      //exports of the reference model
   foreach(exp_bg_ports[i]) begin
     exp_bg_ports[i].connect(refmod.out_tlm_fifos[i].blocking_get_export);
   end

此处的blocking_get_export我们不需要去声明以及例化,uvm_tlm_fifo中预先内置了。

UVM的回调类型uvm_callback

1、callback机制必要性?

  • callback机制的最大用处就是提高验证平台的可重用性。 很多情况下, 验证人员期望在一个项目中开发的验证平台能够用于另外一个项目。 但是, 通常来说, 完全的重用是比较难实现的, 两个不同的项目之间或多或少会有一些差异。 如果把两个项目不同的地方使用callback函数来做, 而把相同的地方写成一个完整的env, 这样重用时, 只要改变相关的callback函数,env可完全的重用。
  • 例如,对于VIP( Verification Intellectual Property) 来说, 一个很容易预测到的需求是在driver中, 在发送transaction之前, 用户可能会针对transaction做某些动作, 因此应该提供一个pre_tran的接口, 如用户A可能在pre_tran中将要发送内容的最后4个字节设置为发送的包的序号, 这样在包出现比对错误时, 可以快速地定位, B用户可能在整个包发送之前先在线路上发送几个特殊的字节, C用户可能将整个包的长度截去一部分, D用户……总之不同的用户会有不同的需求。 正是callback机制的存在, 满足了这种需求, 扩大了VIP的应用范围。
  • 工厂机制的override是为了覆盖,而callback是为了延伸。

2、为什么要专门定义一个uvm_callback类?

以前用户自定义了回调函数不就够了,为啥还要专门定义一个uvm_callback类?
为了使函数回调拥有顺序和继承性(是通过两个相关类uvm_callback_iter和uvm_callback#(T,CB)来实现的)。

  • 所谓的顺序指的是,当你添加了多个cb,那就按顺序执行。也就是下面先执行m_cb1,再执行m_cb2
uvm_callbacks #(comp1)::add(c1,m_cb1); 
uvm_callbacks #(comp1)::add(c1,m_cb2);
  •  所谓的继承性指的是,如果cb2继承于cb1,那么将其与comp1绑定时只需要绑定父类cb1就行了、(`uvm_register_cb(comp1,cb1)),不需要绑定子类cb2(不需要写`uvm_register_cb(comp1,cb1))

预留一个callback任务/函数接口的三步骤

1、定义callback类

class cb1 extends uvm_callback;
    `uvm_object_utils(cb1)
    function new(string name = "cb1");
      super.new(name);
    endfunction
    virtual function void do_trans(edata d);//此处do_trans指具体的回调函数
      d.data = 200;
      `uvm_info("CB", $sformatf("cb1 executed with data %0d", d.data), UVM_LOW)
    endfunction
  endclass

2、绑定及插入callback
绑定及插入cb(在要预留callback函数/任务接口的类中调用uvm_register_cb进行绑定,并在调用callback函数/任务接口的函数/任务中, 使用uvm_do_callbacks宏来插入cb

`uvm_register_cb(comp1,cb1)

 注意,comp1和cb1都是类名,而不是实例名。

`uvm_do_callbacks(comp1,cb1,do_trans(d))

这里uvm_do_callbacks的三个参数意义如下:

  • 第一个参数comp1是调用回调函数do_trans的类的名字
  • 第二个参数cb1是指回调函数do_trans在哪个uvm_callback类里
  • 第三个参数do_trans是要调用的回调函数,在调用时要顺便给出do_trans的参数,此处为d

3、添加callback

`uvm_callbacks #(comp1)::add(c1,m_cb1)

注意,c1和m_cb1都是实例名。

实验

接下来,我们将练习umv_callback的定义、连接和使用方式。由此,我们可以将原由的mcdf_data_consistence_basic_test和mcdf_full_random_test的类实现方式(即类继承方式)修改为回调函数的实现方式,完成类的复用除了可以使用继承,还可以使用回调函数。

1、请在给定的uvm_callback代码中,预先定义需要的几个虚方法。

class cb_mcdf_base extends uvm_callback;
    `uvm_object_utils(cb_mcdf_base)
    mcdf_base_test test;
    function new (string name = "cb_mcdf_base");
      super.new(name);
    endfunction

    virtual task cb_do_reg();
      // User to define the content
    endtask

    virtual task cb_do_formatter();
      // User to define the content
    endtask

    virtual task cb_do_data();
      // User to define the content
    endtask
endclass

回调类中需要包含之后要用到的回调函数。

2、请使用callback对应的宏,完成目标uvm_test类型与目标uvm_callback类型的关联;

`uvm_register_cb(mcdf_base_test,cb_mcdf_base)

3、请继续在目标uvm_test类型中指定的方法中,完成uvm_callback的方法回调指定;

virtual task do_reg();
      //TODO-3.3 Use callback macro to link the callback method
      `uvm_do_callbacks(mcdf_base_test, cb_mcdf_base, cb_do_reg())
    endtask

    // do external formatter down stream slave configuration
    virtual task do_formatter();
      //TODO-3.3 Use callback macro to link the callback method
      `uvm_do_callbacks(mcdf_base_test, cb_mcdf_base, cb_do_formatter())
    endtask

    // do data transition from 3 channel slaves
    virtual task do_data();
      //TODO-3.3 Use callback macro to link the callback method
      `uvm_do_callbacks(mcdf_base_test, cb_mcdf_base, cb_do_data())
    endtask

此处要调用回调函数的类为mcdf_base_test,包含回调函数的callback类为cb_mcdf_base,要调用的回调函数分别为cb_do_reg()、cb_do_formatter()、cb_do_data()。

4、请分别完成uvm_callback和对应test类的定义;

a、cb_mcdf_data_consistence_basic 和 cb_mcdf_data_consistence_basic_test

此处cb_mcdf_data_consistence_basic 加上 cb_mcdf_data_consistence_basic_test,其作用就相当于mcdf_data_consistence_basic_test。
 

class cb_mcdf_data_consistence_basic extends cb_mcdf_base;
    `uvm_object_utils(cb_mcdf_data_consistence_basic)
    function new (string name = "cb_mcdf_data_consistence_basic");
      super.new(name);
    endfunction
    task cb_do_reg();
      //user to adapt contents from mcdf_data_consistence_basic_test
      bit[31:0] wr_val, rd_val;
      super.cb_do_reg();
      // slv0 with len=8,  prio=0, en=1
      wr_val = (1<<3)+(0<<1)+1;
      test.write_reg(`SLV0_RW_ADDR, wr_val);
      test.read_reg(`SLV0_RW_ADDR, rd_val);
      void'(test.diff_value(wr_val, rd_val, "SLV0_WR_REG"));

      // slv1 with len=16, prio=1, en=1
      wr_val = (2<<3)+(1<<1)+1;
      test.write_reg(`SLV1_RW_ADDR, wr_val);
      test.read_reg(`SLV1_RW_ADDR, rd_val);
      void'(test.diff_value(wr_val, rd_val, "SLV1_WR_REG"));

      // slv2 with len=32, prio=2, en=1
      wr_val = (3<<3)+(2<<1)+1;
      test.write_reg(`SLV2_RW_ADDR, wr_val);
      test.read_reg(`SLV2_RW_ADDR, rd_val);
      void'(test.diff_value(wr_val, rd_val, "SLV2_WR_REG"));

      // send IDLE command
      test.idle_reg();
    endtask

    task cb_do_formatter();
      //user to adapt contents from mcdf_data_consistence_basic_test
      super.cb_do_formatter();
      void'(test.fmt_gen.randomize() with {fifo == LONG_FIFO; bandwidth == HIGH_WIDTH;});
      test.fmt_gen.start();
    endtask

    task cb_do_data();
      //user to adapt contents from mcdf_data_consistence_basic_test
      super.cb_do_data();
      void'(test.chnl_gens[0].randomize() with {ntrans==100; ch_id==0; data_nidles==0; pkt_nidles==1; data_size==8; });
      void'(test.chnl_gens[1].randomize() with {ntrans==100; ch_id==1; data_nidles==1; pkt_nidles==4; data_size==16;});
      void'(test.chnl_gens[2].randomize() with {ntrans==100; ch_id==2; data_nidles==2; pkt_nidles==8; data_size==32;});
      fork
        test.chnl_gens[0].start();
        test.chnl_gens[1].start();
        test.chnl_gens[2].start();
      join
      #10us; // wait until all data haven been transfered through MCDF
    
    endtask
  endclass: cb_mcdf_data_consistence_basic

  //TODO-3.4 define cb_mcdf_data_consistence_basic_test
class cb_mcdf_data_consistence_basic_test extends mcdf_base_test;
    // declare uvm_callback member
    `uvm_component_utils(cb_mcdf_data_consistence_basic_test)

    function new(string name = "cb_mcdf_data_consistence_basic_test", uvm_component parent);
      super.new(name, parent);
    endfunction

    function void build_phase(uvm_phase phase);
      super.build_phase(phase);
      // instantiate uvm_callback and add it 
      cb = cb_mcdf_data_consistence_basic::type_id::create("cb"); //创建回调函数
      uvm_callbacks#(mcdf_base_test)::add(this,cb);//添加回调函数。注意mcdf_base_test是父类
    endfunction

    function void connect_phase(uvm_phase phase);
      super.connect_phase(phase);
      // connect test handle to uvm_callback member
      cb.test = this;
    endfunction
  endclass: cb_mcdf_data_consistence_basic_test

可以发现,cb_mcdf_data_consistence_basic_test 中并没有具体实现do_reg()、do_formatter()、do_data(),具体实现的内容放在了cb_mcdf_data_consistence_basic;
我们要做的是,采用下列语句将要调用回调函数的组件类以及包含回调函数的回调类进行add,

uvm_callbacks#(mcdf_base_test)::add(this,cb);

那么运行test时,就会调用cb_mcdf_data_consistence_basic这个回调类里的回调函数。

b、cb_mcdf_full_random 和 cb_mcdf_full_random_test

类似,此处略。

UVM的仿真控制函数

  • end_of_elaboration_phase位于build_phase和connect_phase之后,用于微调测试环境

实验

我们也可以回顾实验1对uvm_root类的应用,学习更多关于uvm_root类的方法,请按照以下实验要求实现代码:

  1. 请在 mcdf_base_test 类中添加新的 phase 函数 end_of_elaboration_phase().同时利用uvm_root类来将信息的冗余度设置为UVM_HIGH ,以此来允许更多低级别的信息打印出来。另外,请设置uvm_root::set_report_max_quit_count()函数来设置仿真退出的最大数值,即如果uvm_error数量超过其指定值,那么仿真会退出。该变量默认数值为-1, 表示仿真不会伴随uvm_error退出。
  2. 请利用uvm_root::set_timeout()设置仿真的最大时间长度,同时由于此功能的生效,可以清除原有方法do_watchdog()的定义和调用。

1、2实验要求代码如下:

function void end_of_elaboration_phase(uvm_phase phase);
      super.end_of_elaboration_phase(phase);
      uvm_root::get().set_report_verbosity_level_hier(UVM_HIGH);
      uvm_root::get().set_report_max_quit_count(1);
      uvm_root::get().set_timeout(10ms);
    endfunction

原先仿真的退出逻辑如下:do_watchdog的作用是,超过某个时间后如果仿真数据还没发送完,也直接结束。也就是,在fork… join_any块中,如果在规定时间内,数据正常发送完,那就正常退出。如果规定时间到了还没发送完,那么do_watchdog()完成,也会直接跳出当前的fork… join_any块,从而调用drop_objection结束仿真。

      fork
        this.do_data();
        this.do_watchdog();//此处用join_any,也就是达到指定时间数据还没发送完,就drop_objection
      join_any

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/415484.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

禁用linux系统默认的22端口号

前提条件&#xff1a;在云平台上开放一个代替22的端口。 1、需查看服务器的防火墙是否开启&#xff0c; 命令&#xff1a;systemctl status firewalld.service 如果防火墙的状态为开启&#xff0c;在运行&#xff0c;需要先在防火墙增加端口 1.命令&#xff1a;firewall-cm…

【消息队列】Kafka如何实现高性能IO

我们直到Kafka是一个自称高性能的消息队列引擎&#xff0c;一般来说对于中间件的设计需要从计算、存储、网络三方面进行下手&#xff0c;而消息从产生到消费&#xff0c;也会经历多个流程&#xff0c;比如在生产者端采用异步\同步方式发送&#xff0c;采用高效的压缩算法&#…

JavaWeb03 Cookie和Session

一个网站怎么证明你来过&#xff1f; 1.首次访问时服务器给客户端一个cookie&#xff0c;下次客户端再次访问会自动携带cookie&#xff0c;注意cookie可以是多个 2.首次访问时服务器登记了客户端一系列信息&#xff0c;下次客户端再进行访问时服务器自动匹配此客户端是否访问…

Win10搭建我的世界Minecraft服务器「内网穿透远程联机」

文章目录1. Java环境搭建2.安装我的世界Minecraft服务3. 启动我的世界服务4.局域网测试连接我的世界服务器5. 安装cpolar内网穿透6. 创建隧道映射内网端口7. 测试公网远程联机8. 配置固定TCP端口地址8.1 保留一个固定tcp地址8.2 配置固定tcp地址9. 使用固定公网地址远程联机转载…

Mac远程控制工具有哪些

适用于Mac的远程控制工具有很多&#xff0c;这里我们给大家列举五个常用软件。 1、Apple Remote Desktop 苹果自带远程桌面正如其名称所承诺的那样。作为 Apple 出品的应用程序&#xff0c;您可以想象它的配置和上手是多么容易。从 App Store 下载 Apple Remote Desktop 后&a…

ThreadPoolExecutor获取原始异常

ThreadPoolExecutor作用 ThreadPoolTaskExecutor是Spring框架提供的一个线程池实现&#xff0c;它是基于Java的ThreadPoolExecutor实现的。ThreadPoolTaskExecutor可以管理线程池中的线程&#xff0c;以满足多线程并发执行任务的需要。 FutureTask作用 FutureTask的主要作用…

自动化篇 | 13 | app自动化:airtest

1 airtes简介 1.1 参考地址 http://airtest.netease.com/ # airtest官网 https://airtest.doc.io.netease.com/ # airtest操作方法 https://airtest.doc.io.netease.com/IDEdocs/faq/1_code_examples/ # 代码示例2 airtest架构 Airtest框架 3 airtest使用 3.1 打开界…

【C语言】猜数字游戏的实现

简单不先于复杂&#xff0c;而是在复杂之后。 目录 1.思路&#xff08;代码会附在最后面&#xff09; 2.随机数函数rand 1.思路&#xff08;代码会附在最后面&#xff09; 首先实现一个猜数字游戏我们先要有一个思路&#xff1a;1.产生一个随机数 2.猜数字 3.可能猜大了 4.可能…

java简单一写一个springboot入门案例

首先 我们先打开idea编辑器 然后 选择File New>Project… 项目的话 我们选择 Spring lnitializr 右边的路径要改一下 然后 选择好路径之后 一定要要在后面加上 \项目名 例如 我这里 选择了 D盘下的学习案例下的springbootDom 项目名叫 threshold 那就这样配置 java版本选…

探索【PhySO】

目录0、 初步报错情况1、230、 初步报错情况 根据原链接&#xff0c;已确定安装成功&#xff0c;运行demo时出现报错&#xff0c;不能正常运行 报错&#xff1a; .Dummy epoch time 238.366736 ms .Dummy epoch time (w duplicate elimination) 180.852427 ms (found 15/1…

【结构型模式】代理模式

文章目录优秀借鉴1、简介2、结构3、实现方式3.1、案例引入3.2、静态代理3.3、JDK动态代理3.4、CGLIB动态代理4、区别对比4.1、静态代理和动态代理4.2、JDK动态代理和CGLIB动态代理5、代理模式优缺点6、应用场景优秀借鉴 设计模式&#xff08;四&#xff09;——搞懂什么是代理…

线段树的懒标记

上次看的那个视频讲线段树的时候压根没讲懒标记&#xff0c;然后我今天去写题目直接被薄纱&#xff01;都是70分&#xff0c;剩下3个节点tml&#xff01;&#xff01;&#xff01; 懒标记 我们在修改一些区间的时候&#xff0c;按照我昨天来学的来修改要改到最下面的叶节点去…

Visual Studio如何将UTF-8字符串输出到控制台

解决c语言使用libcurl库时控制台输出中文出现乱码的问题。 字符编码问题一直以来都是Windows系统的诟病&#xff0c;而Visual Studio也是饱受此诟病。由于历史原因&#xff0c;Windows系统对各个不同的国家地区可能采用不同的code page。由于早先Unicode并未发展成熟&#xff0…

Java设计模式之工厂模式

什么是工厂模式 工厂模式就是将创建对象的具体过程和使用过程分开&#xff0c;这样能够使代码更加灵活。 工厂模式主要分为三类&#xff1a; 简单工厂模式工厂方法模式抽象工厂模式 比如在没有工厂的时候&#xff0c;用户需要一台奥迪车&#xff0c;那么就需要用户去创建一台…

【java 多线程】并发设计模式-两阶段终止模式(对interrupt的理解)

&#x1f4cb; 个人简介 &#x1f496; 作者简介&#xff1a;大家好&#xff0c;我是阿牛&#xff0c;全栈领域优质创作者。&#x1f61c;&#x1f4dd; 个人主页&#xff1a;馆主阿牛&#x1f525;&#x1f389; 支持我&#xff1a;点赞&#x1f44d;收藏⭐️留言&#x1f4d…

惊艳!阿里出品“Java全栈进阶宝典”,广度与深度齐头并进

前言 据有关数据统计&#xff0c;目前来看&#xff0c;大大小小的招聘网站上面&#xff0c;Java岗的招聘量仍然是最多的&#xff0c;基本是其他语言的3倍以上&#xff0c;由于目前Java所处的统治级地位&#xff0c;单就数量来看&#xff0c;Java语言实现的系统是海量的&#x…

linux安装MongoDB

环境说明 系统CentOS&#xff1a;CentOS7 mongodb版本 4.2.24 下载 官网下载地址 Download MongoDB Community Server | MongoDB 某云盘 链接&#xff1a;https://pan.baidu.com/s/1G4AC3h5rvz9WM3fx4gJzbA 提取码&#xff1a;ojkl 上传解压 在根目录下创建opt文件夹…

【数据分析之道-基础知识(九)】推导式

文章目录专栏导读1、前言2、列表推导式3、集合推导式4、字典推导式5、元组推导式专栏导读 ✍ 作者简介&#xff1a;i阿极&#xff0c;CSDN Python领域新星创作者&#xff0c;专注于分享python领域知识。 ✍ 本文录入于《数据分析之道》&#xff0c;本专栏针对大学生、初级数据分…

C语言学习1--------Visual Studio集成开发环境的搭建

C语言学习1--------Visual Studio集成开发环境的搭建适合初学者适用集成开发环境下载 Visual Studio 2019安装 Visual Studio 2019安装工作负载为C自定义安装位置激活 Visual Studio适合初学者适用集成开发环境 建议初学者适用最新的——Visual Studio 2019为集成开发环境。 部…

【数据库原理 • 五】数据库安全性与完整性

前言 数据库技术是计算机科学技术中发展最快&#xff0c;应用最广的技术之一&#xff0c;它是专门研究如何科学的组织和存储数据&#xff0c;如何高效地获取和处理数据的技术。它已成为各行各业存储数据、管理信息、共享资源和决策支持的最先进&#xff0c;最常用的技术。 当前…