电子说
上一篇文章介绍了SystemVerilog的各种随机化方法,本文将在其基础上引入SystemVerilog的随机约束方法(constraints)。通过使用随机约束,我们可以将随机限制在一定的空间内,有针对性地提高功能覆盖率。
SV随机约束的应用,就像是我们用陈述性(declarative)的语句告诉仿真器我们要的随机数要满足哪些条件,然后仿真器的约束解算器(constraint solver)就会去找到能够满足我们所有描述语句的解,再从这些解中随机选出来一个值作为随机的结果。
约束解算器
约束解算器是SV仿真器重要的一部分,它被专门用来求解约束。这里说的求解,就是指的仿真程序通过某种算法,找到能够满足我们所有约束条件的随机值的过程。如果随机被过约束(over-constraint)了,或者存在随机值的组合情况不能满足约束,则约束解算器就会解算失败。在实际应用中,仿真程序就会报错,然后打印出来告诉我们是哪段约束没有解算成功。
理解约束解算器的重要性,可以想想如果没有约束解算器会怎么样?举一个简单的栗子,现在我们有一个约束条件:变量A的随机值总是小于变量B的随机值。如果没有约束功能,代码可以这么实现:
do begin
A = $urandom;
B = $urandom;
end while (! (A
使用约束语句代码是这样的:
class ictalking;
rand bit [7:0] A, B;
constraint c_ab { A < B; }
endclass
可以看得出来,如果没有约束解算器,我们在描述约束的时候就会变得比较绕,通常会花掉很多时间去重复执行相同的一段代码,有时候甚至会跑了半天都撞不出来一个满足约束的随机值,更别说那些复杂的约束了。有了约束解算器,我们就可以在其框架内加入各种约束语句,它总能帮我们快速找到那个解。
约束关系和控制
约束的解算顺序: 约束的解算顺序可以使用solve-before来控制。约束解算器会优先求解before之前的约束,因此使用solve-before会影响随机数组合的概率分布情况。
class ictalking;
rand bit [7:0] A, B;
constraint c_a { A > B; }
constraint c_order {solve A before B;}
// 顺序约束可以写在同一个约束块中, // 也可以分开写在不同的约束块中(如本例)endclass
硬约束和软约束: 当我们在不同的层次对随机变量附加约束的时候,软约束可以被后面指定的约束给覆盖。典型的应用场景是在UVM的sequence_item(或者叫transaction)定义时,我们可以通过软约束指定默认的随机约束,这样方便我们后面在继承或者例化的时候可以使用更高优先级的约束对其覆盖。
class ictalking;
rand int count;
constraint c_count {
soft count inside {[666:888]}; // 指定软约束需要使用关键字soft
}
endclass
ictalking ict = new();
ict.randomize() with { count inside {[123:456]}; }
约束的控制开关: 默认情况下,所有的约束一写上就默认使能,即约束解算器就会按照这些约束开始算。但SV提供约束条件的控制方法constraint_mode(),可以很方便的控制约束是否启用,以及查询约束的启用状态。
// 继续上面的例子
int con_status;
ictalking ict_obj1 = new();
ictalking ict_obj2 = new();
ict_obj1.c_count.constraint_mode(0); // 不启用ict_obj1中的约束c_count
ict_obj2.c_count.constraint_mode(1); // 启用ict_obj2中的约束c_count
ict_obj1.count.rand_mode(0); // 顺便提一嘴,随机变量类似的可以使用rand_mode开关随机功能
con_status = ict_obj1.c_count.constraint_mode(); // 获得ict_obj1中约束c_count的启用状态
五花八门的约束代码
SV中的约束非常灵活,下面给出一些常用的约束代码,可以作为参考和总结。
rand int temp_var;
constraint c_var_1 {temp_var inside {[2000:2021]}; } // 限定范围
constraint c_var_2 {temp_var inside {2008, 2016, 2019}; } // 限定枚举值
constraint c_var_3 {! (temp_var inside {[1:2007]}); // 反向限定范围
rand bit mode;
rand int count;
constraint c_var_1 { mode == 1 -> count < 2021; } // 使用implication操作符->
constraint c_var_2 { if (mode == 1) {count < 2021;} else {count > 6000;} } // 使用if-else
rand bit mode;
rand int count;
constraint c_var_1 {
mode == READ -> count inside {[2008:2016]};
mode == WRITE -> count inside {[2017:2021]};
mode dist {READ := 4, WRITE := 6}; // mode随机成READ的概率为40%,WRITE为60%
}
initial begin
repeat (100) begin
randcase 2: $display("In 1st branch."); // 在100次循环中,执行分支1的概率是20% 7: $display("In 2nd branch."); // 在100次循环中,执行分支2的概率是70%
1: $display("In 3rd branch."); // 在100次循环中,执行分支3的概率的10%
endcase
end
end
rand int a, b, c;
rand int array[5];
int q[$] = `{200, 53, 656};
constraint c_unique {
unique {a, b, c}; // 该约束要求a和b和c两两之间互不相等
unique {a, b, array}; // 该约束要求a和b和array中的所有值互不相等
unique {array}; // 该约束要求array数组内的5个值互不相等
unique {a, q}; // 该约束要求a随机出来的值不等于q中的任一值
}
rand int q[$];
constraint c_foreach {
q.size() inside {[3:8]};
foreach (q[i]) {
if (i > 0) q[i] > q[i-1]; // 约束q队列的下一个值总比上一个值大
}
}
rand bit qbit[$];
rand int qint[$];
constraint c_qbit {
qbit.size() inside {[4:6]};
(qbit.sum with (int'(item))) == 3; // 将当前元素item转为int类型,并约束所有元素有且只有3个为1
}
constraint c_qint {
qint.size() inside {[5:9]};
(qint.sum with ((item.index < 3) ? item : 0)) == 45; // 约束qint队列的前三个加起来等于45
}
class ictalking;
rand int count;
static constraint c_count {count > 34;}
endclass
module testbench;
initial begin ictalking ict_obj1 = new(); ictalkingict_obj2=new(); ict_obj1.c_count.constraint_mode(0); // 关掉的之后ict_obj2中的c_count也会失效 ...
end
endmodule
参考文献
[1] IEEE Standard Association. "IEEE Standard for SystemVerilog-Unified Hardware Design, Specification, and Verification Language." (2013).
全部0条评论
快来发表一下你的评论吧 !