SystemVerilog小白入门3,UVM的uvm_object初体验

UVM的 uvm_object类的随机化、拷贝、克隆、比较、打印

  • create():UVM工厂模式创建对象,是UVM推荐的对象实例化方式(而非直接new());

  • copy():浅拷贝,仅复制注册字段的(基本类型直接复制,对象句柄仅复制指向);

  • clone():深拷贝,先创建新对象,再将原对象的字段值拷贝到新对象,返回父类uvm_object句柄,需$cast转换;

  • compare():对比两个对象的所有注册字段值,返回布尔值;

  • print():自动化打印所有注册字段的信息(名称、类型、大小、值)。

    // 导入UVM核心包,包含UVM的基础类、方法和宏定义
    import uvm_pkg::*;
    // 包含UVM宏定义文件,必须和uvm_pkg配合使用
    `include "uvm_macros.svh"

    // 定义自定义UVM对象类A,继承自uvm_object(UVM所有数据对象的基类)
    // uvm_object主要用于存储数据,支持copy/clone/compare/print等自动化方法
    class A extends uvm_object;
    // 定义随机整型变量data1(32位有符号整型)
    rand int data1;
    // 定义随机48位无符号比特变量data2(bit类型默认无符号)
    rand bit[47:0] data2;

    复制代码
      // UVM自动化字段注册宏:开始注册类A的字段
      // 注册后才能使用UVM提供的copy/clone/compare/print等自动化方法
      `uvm_object_utils_begin(A)
          // 注册整型字段data1,UVM_ALL_ON表示启用该字段的所有自动化功能
          `uvm_field_int(data1, UVM_ALL_ON)
          // 注册48位比特字段data2,同样启用所有自动化功能
          `uvm_field_int(data2, UVM_ALL_ON)
      // 结束字段注册
      `uvm_object_utils_end
    
      // 类的构造函数(UVM推荐显式定义,调用父类构造函数)
      // name参数用于UVM的层次化打印和调试
      function new(string name = "A");
          super.new(name);
      endfunction

    endclass

    // 测试模块tb:仿真的顶层模块,用于验证类A的功能
    module tb;

    复制代码
      // initial块:SystemVerilog的初始化块,仿真开始时执行(仅执行一次)
      initial begin
          // 声明三个类A的对象句柄(仅声明,未分配内存)
          A a1,a2,a3;
          
          // 1. 创建UVM对象:通过type_id::create()方法(UVM推荐方式)
          // 相比直接new(),create()支持工厂模式,便于后续重载/替换类
          a1 = A::type_id::create("a1"); // 创建对象a1,命名为"a1"
          a2 = A::type_id::create("a1"); // 注意:a2命名也为"a1"(名称重复不影响功能,仅调试时易混淆)
          
          // 2. 带内联约束的随机化:
          // randomize() with {} 为内联约束,优先级高于类内约束(本例类内无约束)
          // assert检查随机化是否成功:成功返回1,失败返回0(触发assert报错)
          assert(a1.randomize() with { data1 == 123; data2<10;});
    
          // 3. copy方法:将a1的所有注册字段值浅拷贝到a2
          // 浅拷贝:对于基本类型(int/bit),直接复制值;对于对象句柄,仅复制句柄(指向同一对象)
          a2.copy(a1);
    
          // 4. clone方法:克隆a1(深拷贝),返回uvm_object类型的句柄
          // $cast:将父类句柄(uvm_object)向下转换为子类句柄(A),确保类型匹配
          $cast(a3,a1.clone());
    
          // 打印a1的数值:UVM_INFO宏(优先级UVM_LOW,默认打印)
          // $sformatf:格式化字符串,输出data1和data2的十进制值
          `uvm_info("INFO", $sformatf("a1: data1=%0d; data2=%0d",a1.data1,a1.data2), UVM_LOW)
          // 打印copy后的a2数值,验证拷贝是否成功
          `uvm_info("COPY", $sformatf("a2: data1=%0d; data2=%0d",a2.data1,a2.data2), UVM_LOW)
          // 打印clone后的a3数值,验证克隆是否成功
          `uvm_info("CLONE", $sformatf("a3: data1=%0d; data2=%0d",a3.data1,a3.data2), UVM_LOW)
    
          // 5. compare方法:比较a2和a1的所有注册字段值是否完全一致
          // 一致返回1,不一致返回0
          if(a2.compare(a1))
              `uvm_info("COMPARE","a2 and a1 are Equal",UVM_LOW)
          else
              `uvm_info("COMPARE","a2 and a1 are Different",UVM_LOW)
    
          // 修改a2的data1值,破坏与a1的一致性
          a2.data1 = 111;
    
          // 再次比较a2和a1,验证修改后是否不一致
          if(a2.compare(a1))
              `uvm_info("COMPARE","a2 and a1 are Equal",UVM_LOW)
          else
              `uvm_info("COMPARE","a2 and a1 are Different",UVM_LOW)
    
          // 6. print方法:打印a1的所有注册字段的详细信息(层次化、类型、值)
          // 依赖`uvm_field_int`注册的字段,自动输出格式规整的信息
          a1.print();
    
      end

    endmodule

关键语法

  • rand:标记变量为随机变量,需调用randomize()才能随机取值;
  • randomize() with {}:内联约束,临时修改/添加随机约束,优先级高于类内约束;
  • $cast:类型转换,确保父类句柄安全转换为子类句柄(避免类型不匹配错误);
  • UVM_INFO:UVM标准打印宏,格式为(标识, 消息, 优先级),便于日志分类和过滤。

上机实操

按照**SystemVerilog小白入门2,QuestaSim初体验** 配置编译允许环境。

相关推荐
FakeOccupational5 小时前
【电路笔记 元器件】存储设备:RAM 静态随机存取存储器(SRAM)芯片+异步 SRAM 的特性+异步 SRAM读写测试(HDL)
笔记·fpga开发
嵌入式×边缘AI:打怪升级日志7 小时前
环境监测传感器从设备程序设计(ADC采集与输出控制)
单片机·嵌入式硬件·fpga开发
dadaobusi8 小时前
verilog,generate语句
fpga开发
码不停蹄Zzz1 天前
GTX DRP动态重配置技术
fpga开发
LeoZY_1 天前
CH347/339W开源项目:集SPI、I2C、JTAG、SWD、UART、GPIO多功能为一体(5)
stm32·mcu·fpga开发·开源·硬件架构·硬件工程
博览鸿蒙1 天前
FPGA 工程师如何提升自己?
fpga开发
FPGA小c鸡1 天前
FPGA Transformer加速完全指南:从模型优化到硬件实现(附实战案例)
深度学习·fpga开发·transformer
Fpga_User1 天前
项目FPGA类型获取(以xilinx为例)
fpga开发
maverick_1112 天前
【Verilog】强基础,if else 语句,以及综合RTL
fpga开发