新聞中心

EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > Verilog HDL基礎(chǔ)知識(shí)4之阻塞賦值 & 非阻塞賦值

Verilog HDL基礎(chǔ)知識(shí)4之阻塞賦值 & 非阻塞賦值

作者: 時(shí)間:2024-02-20 來(lái)源:電子森林 收藏

串行塊語(yǔ)句中的語(yǔ)句按順序執(zhí)行,它不會(huì)阻塞其后并行塊中語(yǔ)句的執(zhí)行。語(yǔ)句使用“=”作為賦值符。

本文引用地址:http://www.butianyuan.cn/article/202402/455504.htm
  例子 阻塞賦值語(yǔ)句  reg x, y, z;
 reg [15:0] reg_a, reg_b;
 integer count;   // 所有行為語(yǔ)句必須放在 initial 或 always 塊內(nèi)部
 initial
 begin
         x = 0; y = 1; z = 1; // 標(biāo)量賦值
         count = 0; // 整形變量賦值
         reg_a = 16'b0; reg_b = reg_a; // 向量的初始化           
         #15 reg_a[2] = 1'b1; // 帶延遲的位選賦值
         #10 reg_b[15:13] = {x, y, z} // 把拼接操作的結(jié)果賦值給向量的部分位(域) 
         count = count + 1; // 給整形變量賦值(遞增)
  end 

在例子中,只有在語(yǔ)句x=0執(zhí)行完成后,才會(huì)執(zhí)行y=1,而語(yǔ)句count=count+1按順序在最后執(zhí)行。由于阻塞賦值語(yǔ)句是按順序執(zhí)行的,因此如果在一個(gè)begin-end塊中使用了阻塞賦值語(yǔ)句,那么這個(gè)塊語(yǔ)句表現(xiàn)的是串行行為。例子中,begin-end塊中各條語(yǔ)句執(zhí)行的仿真時(shí)間為:

1.x=0到regb = rega之間的語(yǔ)句在仿真0時(shí)刻執(zhí)行;

2.語(yǔ)句rega[2]=0在仿真時(shí)刻15執(zhí)行; * 語(yǔ)句regb[15:13]={x,y,z}在仿真時(shí)刻25執(zhí)行;

3.語(yǔ)句count=count+1在仿真時(shí)刻25執(zhí)行;

4.由于前面的語(yǔ)句分別包含了15和10個(gè)時(shí)間單位的延遲,因此語(yǔ)句count=count+1將在第25個(gè)單位時(shí)刻執(zhí)行。

注意,在對(duì)寄存器類型變量進(jìn)行過(guò)程賦值時(shí),如果賦值符兩側(cè)的位寬不相等,則采用以下原則:

1.如果右側(cè)表達(dá)式的位寬較寬,則將保留從最低位開(kāi)始的右側(cè)值,把超過(guò)左側(cè)位寬的高位丟棄;

2.如果左側(cè)位寬大于右側(cè)位寬,則不足的高位補(bǔ)0;


語(yǔ)句允許賦值調(diào)度,但它不會(huì)阻塞位于同一個(gè)順序塊中其后語(yǔ)句的執(zhí)行。使用“?”作為賦值符。讀者會(huì)注意到,它與“小于等于”關(guān)系操作符是同一個(gè)符號(hào),但在表達(dá)式中它被解釋為關(guān)系操作符,而在非阻塞賦值的環(huán)境下被解釋成非阻塞賦值。為了說(shuō)明非阻塞賦值的意義以及阻塞賦值的區(qū)別,讓我們來(lái)考慮將阻塞賦值例子中的部分阻塞賦值改為非阻塞賦值后的結(jié)果,修改后語(yǔ)句如下:

  例 非阻塞賦值語(yǔ)句  reg x, y, z;
 reg [15:0] reg_a, reg_b;
 integer count;   // 所有的行為語(yǔ)句必須寫(xiě)在initial 和 always塊內(nèi)
 initial
 begin
         x = 0; y = 1; z = 1; // 標(biāo)量賦值;
         count = 0; // 整形變量賦值
         reg_a = 16'b0; reg_b = reg_a; // 向量的初始化 
         reg_a[2] <= #15  1'b1; // 帶延遲的位選賦值
         reg_b[15:13] <= #10 {x, y, z} // 把拼接操作的結(jié)果賦值給向量的部分位(域) 
         count <= count + 1; // 給整形變量賦值(遞增)
  end 

在這個(gè)例子中,從x=0到regb=rega之間的語(yǔ)句是在仿真0時(shí)刻順序執(zhí)行的,之后的三條非阻塞賦值語(yǔ)句在regb=rega執(zhí)行完成后并發(fā)執(zhí)行。

1.rega[2]=0被調(diào)度到15個(gè)時(shí)間單位之后執(zhí)行,即仿真時(shí)刻為15; * regb[15:13]={x,y,z}被調(diào)度到10個(gè)時(shí)間單位之后執(zhí)行,即仿真時(shí)刻為10;

2.count=count+1被調(diào)度到無(wú)任何延遲執(zhí)行,即仿真時(shí)刻為0。

從上面的分析中可以看到,仿真器將非阻塞賦值調(diào)度到相應(yīng)的仿真時(shí)刻,然后繼續(xù)執(zhí)行后面的語(yǔ)句,而不是停下來(lái)等待賦值的完成。一般情況下,非阻塞賦值是在當(dāng)前仿真時(shí)刻的最后一個(gè)時(shí)間步,即阻塞賦值完成之后才執(zhí)行的。
在上面的例子中,我們把阻塞和非阻塞賦值語(yǔ)句混合在一起使用,目的是想更清楚地比較和說(shuō)明它們的行為。需要提醒大家注意的是,不要在同一個(gè)always塊中混合使用阻塞和非阻塞賦值語(yǔ)句。


非阻塞賦值語(yǔ)句的應(yīng)用

描述了非阻塞賦值的行為之后,理解究竟為什么要在硬件設(shè)計(jì)中使用非阻塞賦值是很重要的。非阻塞賦值可以被用來(lái)為常見(jiàn)的硬件電路行為建立模型,例如當(dāng)某一事件發(fā)生后,多個(gè)數(shù)據(jù)并發(fā)傳輸?shù)男袨?。在下面的例子中,?dāng)時(shí)鐘信號(hào)的上升沿到來(lái)之后,執(zhí)行三個(gè)數(shù)據(jù)的并發(fā)傳輸:

  always @(posedge clock)
 begin
    reg1 <= #1 in1;
    reg2 <= @(negedge clock) in2 ^ in3;
    reg3 <= #1 reg1; // reg1 的“舊值”
 end 

每當(dāng)一個(gè)時(shí)鐘上升沿到來(lái)時(shí),其中的非阻塞賦值語(yǔ)句按下面的順序執(zhí)行:

1.在每個(gè)時(shí)鐘上升沿到來(lái)時(shí)讀取操作數(shù)變量in1, in2, in3和reg1,計(jì)算右側(cè)表達(dá)式的值,該值由仿真器臨時(shí)保存;

2.對(duì)左值的賦值由仿真器調(diào)度到相應(yīng)的仿真時(shí)刻,延遲時(shí)間由語(yǔ)句中內(nèi)嵌的延遲值確定。在本例中,對(duì)reg1的賦值需要等一個(gè)時(shí)間單位,對(duì)reg2的賦值需要等到時(shí)鐘信號(hào)下降沿到來(lái)的時(shí)刻,對(duì)reg3的賦值需要等待一個(gè)時(shí)間單位;

3.每個(gè)賦值操作在被調(diào)度的仿真時(shí)刻完成。注意,對(duì)左側(cè)變量的賦值使用的是由仿真器保存的表達(dá)式“舊值”,因此賦值完成的實(shí)際順序并不重要。在本例中,對(duì)reg3賦值使用的是reg1 的“舊值”,而不是在此之前對(duì)reg1賦予的新值,reg1的“舊值”是在賦值事件調(diào)度時(shí)由仿真器保存的。

由上面的分析可見(jiàn),reg1,reg2和reg3的最終值與賦值完成的順序無(wú)關(guān),體現(xiàn)了非阻塞賦值并行的特點(diǎn)。
為了進(jìn)一步理解阻塞和非阻塞賦值,讓我們來(lái)看以下例子。這個(gè)例子的目的是在每個(gè)時(shí)鐘上升延處交換a和b這兩個(gè)寄存器變量的值,其中使用了兩個(gè)always語(yǔ)句。

  例 使用非阻塞賦值來(lái)避免競(jìng)爭(zhēng)  // 說(shuō)明1:使用阻塞語(yǔ)句的兩個(gè)并行的always塊
 always @(posedge clock)
            a = b;   always @(posedge clock)
            b = a;   // 說(shuō)明2:使用非阻塞語(yǔ)句的兩個(gè)并行的always塊
 always @(posedge clock)
            a <= b;   always @(posedge clock)
            b <= a; 

在第一種描述中我們使用了阻塞賦值,這樣就產(chǎn)生了競(jìng)爭(zhēng)的情況:a=b和b=a,具體執(zhí)行順序的先后取決于所使用的仿真器。因此這段代碼達(dá)不到交換a和b值的目的,而是使得兩者具有相同的值(在時(shí)鐘上升沿到來(lái)之前a或b的值),具體是哪一個(gè)值與使用的仿真器有關(guān)。

在第二種描述中,我們通過(guò)使用非阻塞賦值語(yǔ)句來(lái)避免競(jìng)爭(zhēng):在每個(gè)時(shí)鐘上升沿到來(lái)的時(shí)候,仿真器讀取每個(gè)操作數(shù)的值,進(jìn)而計(jì)算表達(dá)式的值并保存在臨時(shí)變量中;當(dāng)賦值的時(shí)候,仿真器將保存的值賦予非阻塞賦值語(yǔ)句的左側(cè)變量。這樣就將讀和寫(xiě)分開(kāi)來(lái)了,達(dá)到了交換數(shù)據(jù)的目的,并且不受語(yǔ)句執(zhí)行順序的影響。以下例子介紹了如何使用阻塞賦值實(shí)現(xiàn)說(shuō)明2中使用非阻塞語(yǔ)句才能實(shí)現(xiàn)的數(shù)值交換。

  例 使用阻塞賦值來(lái)達(dá)到非阻塞賦值的目的  // 使用臨時(shí)變量和阻塞賦值來(lái)模仿非阻塞賦值的行為
 always @(posedge clock)
 begin
     // 讀操作
     // 把右側(cè)表達(dá)式的值放在臨時(shí)變量中
     temp_a = a;
     temp_b = b;
     // 寫(xiě)操作
     // 把臨時(shí)變量的值放到左側(cè)變量中
     a = temp_b;
     b = temp_a;
  end 

在數(shù)字電路設(shè)計(jì)中,如果某事件發(fā)生后將產(chǎn)生多個(gè)數(shù)據(jù)的并發(fā)傳輸,我們強(qiáng)烈建議讀者使用非阻塞賦值來(lái)描述這種情形。如果我們使用阻塞賦值來(lái)描述這種情形,由于最終結(jié)果依賴于語(yǔ)句的具體執(zhí)行順序,有可能引起競(jìng)爭(zhēng)風(fēng)險(xiǎn);而非阻塞賦值語(yǔ)句的執(zhí)行結(jié)果是與執(zhí)行順序無(wú)關(guān)的,因此它能夠準(zhǔn)確地描述這種情況。非阻塞賦值的典型應(yīng)用包括流水線建模和多個(gè)互斥(mutually exclusive)數(shù)據(jù)傳輸?shù)慕?。使用非阻塞賦值所帶來(lái)的問(wèn)題是,它會(huì)引起仿真速度的下降以及內(nèi)存使用量的增加。



評(píng)論


相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉