在使用UVM搭建環(huán)境時(shí),遇到問(wèn)題時(shí),調(diào)試方式有千千萬(wàn)萬(wàn),但很有必要了解下UVM庫(kù)提供了哪些內(nèi)建的調(diào)試手段,可以少走彎路,大大提升效率,而不是瘋狂加各種打印消息。
UVM庫(kù)給各個(gè)主要機(jī)制都提供了易于使用的內(nèi)置調(diào)試方法,來(lái)輔助定位使用UVM環(huán)境遇到的問(wèn)題。
一、調(diào)試config_db機(jī)制問(wèn)題
UVM庫(kù)內(nèi)實(shí)現(xiàn)了一個(gè)資源庫(kù),它用于存儲(chǔ)配置信息,TB里各個(gè)組件可以根據(jù)情況使用config_db往里面存或者取各種類(lèi)型數(shù)據(jù)。config_db存(set())和取(get())的關(guān)鍵在于字符串匹配,為此UVM庫(kù)提供了一些功能來(lái)幫助調(diào)試這些字符串匹配。
1. 使用+UVM_CONFIG_DB_TRACE和+UVM_RESOURCE_DB_TRACE命令行參數(shù)
UVM庫(kù)在Command Line上提供了+UVM_CONFIG_DB_TRACE和+UVM_RESOURCE_DB_TRACE命令行參數(shù),當(dāng)運(yùn)行仿真命令時(shí),如果帶上上述的參數(shù),那么在log中會(huì)打印出對(duì)資源庫(kù)的存和取的信息。+UVM_CONFIG_DB_TRACE用于uvm_config_db進(jìn)行的存取,+UVM_RESOURCE_DB_TRACE用于uvm_resource_db進(jìn)行的存取。比如我們?cè)赒uestasim工具的vsim命令后加上+UVM_CONFIG_DB_TRACE,然后有以下的uvm_config_db的set()和get()調(diào)用:
// In the TB env:
uvm_config_db #(int)::set(this, "*", "var", 666);
// In the TB driver:
int get_value;
if ( !uvm_config_db #(int)::get(this, "*", "var", get_value) ) begin
`uvm_fatal(get_type_name(), "var is missing in config_db")
end else begin
`uvm_info(get_type_name(), $sformatf("get var from env"), UVM_LOW)
end
那么在log中 可以找到以下打印信息:
UVM_INFO …/uvm-1.2/src/base/uvm_resource_db.svh(121) @ 0.000ns: reporter
// db類(lèi)型 匹配字符串 數(shù)據(jù)類(lèi)型 路徑 數(shù)據(jù)值
[CFGDB/SET] Configuration 'uvm_test_top.env.*.var' (type int) set by uvm_test_top.env = (int) 666
UVM_INFO …/uvm-1.2/src/base/uvm_resource_db.svh(121) @ 0.000ns: reporter [CFGDB/GET] Configuration
'uvm_test_top.env.d_agent.drv_h.*.var' (type int) read by uvm_test_top.env.d_agent.drv_h = (int) 666
從log信息可以看出,UVM會(huì)把對(duì)資源庫(kù)的set()和get()的數(shù)據(jù)類(lèi)型,數(shù)據(jù)值、存取路徑、存取類(lèi)型和匹配字符都打印出來(lái),這樣就很方便我們?nèi)ザㄎ籾vm_config_db的匹配問(wèn)題了。
2. 調(diào)用UVM component內(nèi)置函數(shù)
在uvm_component內(nèi)部提供了print_config()內(nèi)建函數(shù),使用它可以打印出當(dāng)前uvm_component范圍可見(jiàn)的所有config_db操作內(nèi)容。如果參數(shù)recurse為1,會(huì)把所有子components的可見(jiàn)的config_db操作內(nèi)容也遞歸調(diào)用打印出來(lái)。如果audit為1,會(huì)把調(diào)用config_db進(jìn)行操作的時(shí)間、次數(shù)和操作者路徑也打印出來(lái)。print_config()的函數(shù)定義如下:
function void uvm_component::print_config(bit recurse = 0, audit = 0);
假如我們?cè)谥袄拥腡B driver里調(diào)用:
print_config(.recurse(0), .audit(1));
那么將會(huì)有以下log輸出:
# var [/^uvm_test_top\\.env\\..*$/] : (int) 666
# UVM_INFO .../uvm-1.2/src/base/uvm_resource.svh(564) @ 0.000ns: reporter [UVM/RESOURCE/ACCESSOR]
uvm_test_top.env reads: 0 @ 0.000ns writes: 1 @ 0.000ns
# uvm_test_top.env.d_agent.drv_h reads: 1 @ 0.000ns writes: 0 @ 0.000ns
它會(huì)把TB driver上config_db操作的字符串匹配、數(shù)據(jù)類(lèi)型和數(shù)據(jù)值都打印出來(lái),另外,由于我們指定audit為1,因此也會(huì)把config_db操作的時(shí)間、次數(shù)和操作者路徑打印出來(lái)了。這個(gè)一個(gè)很強(qiáng)大的debug功能。
建議可以在end_of_elaboration_phase里去調(diào)用這個(gè)函數(shù),因?yàn)檫@時(shí)候config_db操作基本都已經(jīng)完成了。
3. dump整個(gè)資源庫(kù)
如果遇到奇怪的訪問(wèn)資源庫(kù)問(wèn)題無(wú)法解決,另一種暴力debug方式就是將整個(gè)資源庫(kù)都打印出來(lái)。UVM提供了uvm_config_db #()::dump()函數(shù),可以將當(dāng)前資源庫(kù)的信息都打印出來(lái),其中可以指定任何類(lèi)型,主要是因?yàn)閐ump()是個(gè)static的函數(shù),提供任何類(lèi)型最終訪問(wèn)的dump()函數(shù)是同一個(gè),打印出的資源庫(kù)信息也是一樣的。
比如我們?nèi)栽赥B driver里調(diào)用:
uvm_config_db #(bit)::dump();
在log里增加的信息將有:
# UVM_INFO .../uvm-1.2/src/base/uvm_resource.svh(1347) @ 0.000ns: reporter
[UVM/RESOURCE/DUMP]
# === resource pool ===
...
# var [/^uvm_test_top\\.env\\..*$/] : (int) 666
...
# UVM_INFO .../uvm-1.2/src/base/uvm_resource.svh(1354) @ 0.000ns: reporter
[UVM/RESOURCE/DUMP] === end of resource pool ===
也是建議可以在end_of_elaboration_phase里去調(diào)用這個(gè)函數(shù),因?yàn)檫@時(shí)候config_db操作基本都已經(jīng)完成了。
結(jié)合上述的三個(gè)方法,可以說(shuō)100%的config_db相關(guān)的問(wèn)題都可以搞定了。
二、調(diào)試objection機(jī)制問(wèn)題
Objection用于控制消耗時(shí)間的uvm_phase在何時(shí)結(jié)束,TB中raise和drop的objection次數(shù)要一樣,但如果在多個(gè)地方進(jìn)行raise或drop的話,遇到objection沒(méi)有啟動(dòng)或無(wú)法結(jié)束時(shí),就比較難調(diào)試了。因此,UVM庫(kù)提供了用于跟蹤objection raise和drop的命令行參數(shù)+UVM_OBJECTION_TRACE。
比如我們?cè)赒uestasim工具的vsim命令后加上+UVM_OBJECTION_TRACE。那么log里將增加以下類(lèi)似信息。
# UVM_INFO @ 0.000ns: run [OBJTN_TRC] Object uvm_test_top raised 1 objection(s)
(START basetest): count=1 total=1
# UVM_INFO @ 0.000ns: run [OBJTN_TRC] Object uvm_top added 1 objection(s) to its
total (raised from source object uvm_test_top, START basetest): count=0 total=1
...
# UVM_INFO @ 14190.000ns: run [OBJTN_TRC] Object uvm_test_top dropped 1 objection(s)
(END basetest): count=0 total=0
# UVM_INFO @ 14190.000ns: run [OBJTN_TRC] Object uvm_test_top all_dropped 1 objection(s)
(END basetest): count=0 total=0
# UVM_INFO @ 14190.000ns: run [OBJTN_TRC] Object uvm_top subtracted 1 objection(s)
from its total (dropped from source object uvm_test_top, END basetest): count=0 total=0
# UVM_INFO @ 14190.000ns: run [OBJTN_TRC] Object uvm_top subtracted 1 objection(s)
from its total (all_dropped from source object uvm_test_top, END basetest): count=0 total=0
三、調(diào)試phase機(jī)制問(wèn)題
為了幫助用戶查看各個(gè)uvm_phase在何時(shí)開(kāi)始和結(jié)束,UVM庫(kù)提供了+UVM_PHASE_TRACE命令行參數(shù)。
比如我們?cè)赒uestasim工具的vsim命令后加上+UVM_PHASE_TRACE。那么log里將增加以下類(lèi)似信息。
# UVM_INFO .../uvm-1.2/src/base/uvm_phase.svh(1620) @ 0.000ns: reporter [PH/TRC/DONE]
Phase 'common.connect' (id=37) Completed phase
# UVM_INFO .../uvm-1.2/src/base/uvm_phase.svh(1655) @ 0.000ns: reporter [PH/TRC/SCHEDULED]
Phase 'common.end_of_elaboration' (id=40) Scheduled from phase common.connect
# UVM_INFO .../uvm-1.2/src/base/uvm_phase.svh(1345) @ 0.000ns: reporter [PH/TRC/STRT]
Phase 'common.end_of_elaboration' (id=40) Starting phase
# UVM_INFO .../uvm-1.2/src/base/uvm_phase.svh(1620) @ 0.000ns: reporter [PH/TRC/DONE]
Phase 'common.end_of_elaboration' (id=40) Completed phase
四、調(diào)試factory機(jī)制問(wèn)題
UVM庫(kù)的factory機(jī)制用于創(chuàng)建對(duì)象,它是1個(gè)singleton對(duì)象,我們可以通過(guò)調(diào)用uvm_factory::get()獲得它的句柄。當(dāng)我們對(duì)factory機(jī)制創(chuàng)建的對(duì)象有疑問(wèn)時(shí),可以使用factory機(jī)制提供的函數(shù)去調(diào)試有誰(shuí)注冊(cè)了factory,factory override機(jī)制覆蓋了誰(shuí),最終factory為給定類(lèi)型返回什么對(duì)象。Factory機(jī)制提供了3個(gè)函數(shù)去輔助debug。
1. print()
這個(gè)函數(shù)會(huì)根據(jù)參數(shù)all_types的不同,打印出當(dāng)前factory中注冊(cè)的類(lèi)型、實(shí)例覆蓋和類(lèi)型覆蓋。它的定義為:
function void print (int all_types=1);
比如我們?nèi)栽赥B driver中使用以下代碼:
uvm_factory f = uvm_factory::get();
f.print();
那么輸出log將增加以下類(lèi)似信息:
#### Factory Configuration (*)
#
# No instance overrides are registered with this factory
#
# Requested Type Override Type
# -------------- -------------
# seq_base seq1
#
# All types registered with the factory: 288 total
# Type Name
# ---------
# ...
# (*) Types with no associated type name will be printed as < unknown >
從log中可以很清楚的看出,factory注冊(cè)了多少類(lèi)型,類(lèi)型之間的override關(guān)系,instance之間的override關(guān)系,基本上factory的問(wèn)題看這個(gè)信息都可以搞定了。
2. debug_create_by_type()和debug_create_by_name()
這兩個(gè)函數(shù)對(duì)factory的搜索算法類(lèi)似于create_ by_type()和create _by_type(),但它們不創(chuàng)建新對(duì)象。相反,它們提供了關(guān)于將返回的對(duì)象類(lèi)型的詳細(xì)信息,和列出了override相關(guān)信息。具體傳遞參數(shù)用法,大家可以查詢UVM手冊(cè)。
總結(jié)上面的三個(gè)方法,不管有沒(méi)有factory問(wèn)題,推薦統(tǒng)一都在TB base testcase的end_of_elaboration_phase里調(diào)用factory的print()函數(shù),方便大家查詢。
五、調(diào)試TLM 問(wèn)題
UVM中的組件是通過(guò)TLM ports/exports/imps連接在一起的。UVM提供了兩個(gè)函數(shù)都可以在port/export/imp上使用,可以幫助用戶理解哪些對(duì)象連接在一起的。這兩個(gè)函數(shù)是get_connected_to()和get_provided_to(),這兩個(gè)函數(shù)返回的是uvm_port_component_base類(lèi)型的關(guān)聯(lián)數(shù)組。TLM ports通常是fanout類(lèi)型的,所以它通常會(huì)使用get_connected_to(),TLM exports/imps通常是fanin類(lèi)型的,所以它一般會(huì)使用get_provided_to()。
在IEEE 1800.2中,增加了debug_connected_to() 和debug_provided_to(),它們的功能與上述兩個(gè)函數(shù)其實(shí)一樣,只不過(guò)它們返回的是可視化文本消息,方便用戶查看。個(gè)人比較推薦使用這兩個(gè)函數(shù)。
這四個(gè)函數(shù)的定義如下:
function void get_connected_to (ref uvm_port_list list);
function void get_provided_to (ref uvm_port_list list);
function void debug_connected_to (int level=0, int max_level=-1);
function void debug_provided_to (int level=0, int max_level=-1);
這些函數(shù)需要在end_of_elaboration_phase里或之后調(diào)用,由于這時(shí)候TLM的port連接才完成了。
六、調(diào)試callback問(wèn)題
Callback允許標(biāo)準(zhǔn)對(duì)象的外部對(duì)象上調(diào)用函數(shù)和任務(wù),來(lái)擴(kuò)展額外的功能。如果在UVM TB中使用callback功能,可以調(diào)用uvm_typed_callbacks#(type T=uvm_object)里的display()函數(shù)打印出當(dāng)前注冊(cè)的所有callback。display()函數(shù)定義如下:
static function void display( T obj = null )
這個(gè)函數(shù)也是需要在end_of_elaboration_phase里調(diào)用,而且它是靜態(tài)類(lèi)型的,可以使用uvm_callbacks(xxx)::display()方式使用。
UVM也給callback的調(diào)試增加了+define+UVM_CB_TRACE_ON編譯選項(xiàng),當(dāng)編譯帶上UVM_CB_TRACE_ON宏時(shí),在log會(huì)也會(huì)打印出callback的跟蹤信息。
七、其它調(diào)試方式**
1.打印UVM層次結(jié)構(gòu)
在UVM環(huán)境搭建后之后,我們可以通過(guò)print_topology()函數(shù)將UVM層次結(jié)構(gòu)打印出來(lái)。
比如我們?cè)赥B里以下任一種方法代碼:
// 方法1:
uvm_top.print_topology(); // 需要UVM hierarchy建立之后調(diào)用
// 方法2:
uvm_top.enable_print_topology = 1; // 在end_of_elaboration phase之前調(diào)用
在log中會(huì)出現(xiàn)以” [UVMTOP] UVM testbench topology:”開(kāi)頭的打印信息,里面詳細(xì)列出了當(dāng)前UVM結(jié)構(gòu)。
2. uvm_info打印控制
在UVM中,可以指定verbosity來(lái)有選擇性的打印出uvm_info里的消息。UVM提供了全局式和分布式的控制方法。
全局式:這種控制方法是使用+UVM_VERBOSITY命令行參數(shù)來(lái)完成的。
分布式:這種控制方法是使用每個(gè)組件自帶的verbosity設(shè)置方法完成的,通過(guò)使用+uvm_set_verbosity命令行參數(shù)。當(dāng)然也可以直接在組件里使用set_report_verbosity_level()等方法設(shè)置的。
-
UVM
+關(guān)注
關(guān)注
0文章
182瀏覽量
19197 -
TLM
+關(guān)注
關(guān)注
1文章
32瀏覽量
24766
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論