本文作为一个入门的指导吧,当时自己刚刚接触UVM 的时候一头雾水,虽然能够看明白代码但是始终没有自己运行起来。

https://github.com/shirainbown/UVM-Step-by-Step/tree/master

接上文的UVM 环境配置,按照这个配置好之后可以按照本文运行第一个UVM验证平台,需要用到的文件和层级关系如下:

├── 2.2.1
│   ├── filelist.f
│   ├── Makefile
│   ├── my_driver.sv
│   └── top_tb.sv
└── dut
    └── dut.sv

书本中有对代码较为详细的说明,这里不再赘述。

// dut.sv
`timescale 1ns/1ps
module dut(clk,
           rst_n, 
           rxd,
           rx_dv,
           txd,
           tx_en);
input clk;
input rst_n;
input[7:0] rxd;
input rx_dv;
output [7:0] txd;
output tx_en;

reg[7:0] txd;
reg tx_en;

always @(posedge clk) begin
   if(!rst_n) begin
      txd <= 8'b0;
      tx_en <= 1'b0;
   end
   else begin
      txd <= rxd;
      tx_en <= rx_dv;
   end
end
endmodule
// top_tb.sv
`timescale 1ns/1ps
`include "uvm_macros.svh"

import uvm_pkg::*;
`include "my_driver.sv"

module top_tb;

reg clk;
reg rst_n;
reg[7:0] rxd;
reg rx_dv;
wire[7:0] txd;
wire tx_en;

dut my_dut(.clk(clk),
           .rst_n(rst_n),
           .rxd(rxd),
           .rx_dv(rx_dv),
           .txd(txd),
           .tx_en(tx_en));

initial begin
   my_driver drv;
   drv = new("drv", null);
   drv.main_phase(null);  // main_phase 类似于主函数
   $finish();
end

initial begin
   clk = 0;
   forever begin
      #100 clk = ~clk;
   end
end

initial begin
   rst_n = 1'b0;
   #1000;
   rst_n = 1'b1;
end
// 下面的代码是为了生成波形
initial begin 
    $fsdbDumpfile("top_tb.fsdb");
    $fsdbDumpvars(0,top_tb);
    $fsdbDumpMDA();
end

endmodule
// my_driver.sv
`ifndef MY_DRIVER__SV
`define MY_DRIVER__SV
class my_driver extends uvm_driver;

   function new(string name = "my_driver", uvm_component parent = null);
      super.new(name, parent);
   endfunction
   extern virtual task main_phase(uvm_phase phase);  //task可以有时序
endclass

task my_driver::main_phase(uvm_phase phase);
   top_tb.rxd <= 8'b0; 
   top_tb.rx_dv <= 1'b0;
   while(!top_tb.rst_n)
      @(posedge top_tb.clk);
   for(int i = 0; i < 256; i++)begin
      @(posedge top_tb.clk);
      top_tb.rxd <= $urandom_range(0, 255);
      top_tb.rx_dv <= 1'b1;
      `uvm_info("my_driver", "data is drived", UVM_LOW)
   end
   @(posedge top_tb.clk);
   top_tb.rx_dv <= 1'b0;
endtask
`endif

解释一下上述代码:

uvm_driver 本身是 uvm_component 类的子类,因此my_driver也是其子类,于是在定义构造函数new时还要指定其父组件。父组件默认为null,因此这部分可以省略,我们可以看到书中后续的transaction的new函数就是省略了这部分。null指定了driver的层级,即没有上层组件,自身就已经是顶层。

extern virtual task main_phase(uvm_phase phase);

extern 表示task定义在类外, virtual表示虚任务,可以被子类重载。 main_phase 是 类需要执行的一种phase,类似main函数。

另外还有一个问题,为什么此处driver可以直接访问到top_tb?你会发现top_tb在driver中完全没有定义,这是因为在这个例子中,driver在top_tb中例化,top_tb是整个程序的顶层,而顶层路径是可以被所有组件访问到的。

vcs		:
		vcs 	\
		-f filelist.f 	\
		-LDFLAGS -Wl,--no-as-needed          \
		-fsdb -full64 -R +vc +v2k -sverilog -debug_all \
		-ntb_opts uvm-1.1	\
		-timescale=1ns/1ps
verdi 	:
		verdi -sv -f filelist.f -ssf *.fsdb &
clean	:
		rm -rf *.log simv *.daidir csrc *.key DVEfiles *.vpd *.conf *.rc *.fsdb verdiLog
# filelist.f
+incdir+$UVM_HOME/src
$UVM_HOME/src/uvm_pkg.sv
../dut/dut.sv
top_tb.sv
my_driver.sv

在Makefile 目录下,运行make vcs :

运行 make verdi得到波形: