问:既然是启动多个seq,那么启动顺序是什么?
答:对于transaction来说,存在优先级的概念,通常来说,优先级越高越容易被选中。当使用uvm_do或者uvm_do_with宏时,产生的transaction的优先级是默认的优先级,即-1。可以通过uvm_do_pri及uvm_do_pri_with改变所产生的transaction的优先级。如:`uvm_do_pri(m_trans, 100)或者`uvm_do_pri_with(m_trans, 200, {m_trans.pload.size < 500;})。uvm_do_pri与uvm_do_pri_with的第二个参数是优先级,这个数值必须是一个大于等于-1的整数。数字越大,优先级越高。
如:
class sequence0 extends uvm_sequence #(my_transaction);
…
virtual task body();
…
repeat (5) begin
`uvm_do_pri(m_trans, 100)
`uvm_info("sequence0", "send one transaction", UVM_MEDIUM)
end
#100;
…
endtask
…
endclass
class sequence1 extends uvm_sequence #(my_transaction);
…
virtual task body();
…
repeat (5) begin
`uvm_do_pri_with(m_trans, 200, {m_trans.pload.size < 500;})
`uvm_info("sequence1", "send one transaction", UVM_MEDIUM)
end
…
endtask
…
endclass
但是,如果仅仅设置以上优先级,并不能按照预想的顺序产生transaction!
因为,这里面还涉及到sequencer的仲裁算法:
1、SEQ_ARB_FIFO(默认算法,严格遵循先入先出的顺序,而不会考虑优先级)
2、SEQ_ARB_WEIGHTED(加权的仲裁)
3、SEQ_ARB_RANDOM(完全随机选择)
4、SEQ_ARB_STRICT_FIFO(严格按照优先级的,当有多个同一优先级的sequence时,按照先入先出的顺序选择
)
5、SEQ_ARB_STRICT_RANDOM(严格按照优先级的,当有多个同一优先级的sequence时,随机从最高优先级中选择
)
6、SEQ_ARB_USER(是用户可以自定义一种新的仲裁算法)
因此,还必须设置seqr的仲裁算法:env.i_agt.sqr.set_arbitration(SEQ_ARB_STRICT_FIFO)或者env.i_agt.sqr.set_arbitration(SEQ_ARB_STRICT_RANDOM)。如:
task my_case0::main_phase(uvm_phase phase);
…
env.i_agt.sqr.set_arbitration(SEQ_ARB_STRICT_FIFO);
fork
seq0.start(env.i_agt.sqr);
seq1.start(env.i_agt.sqr);
join
endtask
经过如上的设置后,会发现直到
sequence1
发送完
transaction
后,
sequence0
才开始发送。
除transaction有优先级外,sequence也有优先级的概念。可以在sequence启动时指定其优先级:
task my_case0::main_phase(uvm_phase phase);
…
env.i_agt.sqr.set_arbitration(SEQ_ARB_STRICT_FIFO);
fork
seq0.start(env.i_agt.sqr, null, 100);
seq1.start(env.i_agt.sqr, null, 200);
join
endtask
注:start
任务的第一个参数是
sequencer
,第二个参数是
parent sequence
,可以设置为
null
,第三个参数是优先级,如果不指定则此值为-1
,它同样不能设置为一个小于
-1的数字。
运行上述代码,会发现 sequence1中的transaction完全发送完后才发送sequence0中transaction。所以,对sequence设置优先级的本质即设置其内产生的transaction的优先级。