always 語句
always 語句是重復(fù)執(zhí)行的。always 語句塊從 0 時刻開始執(zhí)行其中的行為語句;當(dāng)執(zhí)行完最后一條語句后,便再次執(zhí)行語句塊中的第一條語句,如此循環(huán)反復(fù)。多用于仿真時鐘的產(chǎn)生,信號行為的檢測等。
always語句的格式:
always @(posedge clk) begin
temp = a ;
a = b ;
b = temp ;
end
()中可以是*,表示:每當(dāng)任何輸入發(fā)生變化時執(zhí)行begin和end之間的語句。也可以是posedge clk,表示:時鐘上升沿時執(zhí)行begin和end之間的語句。assign語句和always @(*)創(chuàng)建相同的組合邏輯,比如下面代碼中的兩個輸出波形是一致的。
module top_module(
input a,
input b,
output wire out_assign,
output reg out_alwaysblock
);
assign out_assign=a&b;
always @(*) begin
out_alwaysblock=a&b;
end
endmodule
連續(xù)性賦值,過程性賦值 :連續(xù)性賦值總是處于激活狀態(tài),任何操作數(shù)的改變都會影響表達(dá)式的結(jié)果;過程賦值只有在語句執(zhí)行的時候,才會起作用。
阻塞賦值與非阻塞賦值 :是過程賦值的兩種類型。
阻塞賦值 :阻塞賦值屬于順序執(zhí)行,下一條語句執(zhí)行前,當(dāng)前語句一定會執(zhí)行完畢。使用等號 = 作為賦值符。initial語句用阻塞賦值。
非阻塞賦值 :并行執(zhí)行,即下一條語句的執(zhí)行和當(dāng)前語句的執(zhí)行是同時進(jìn)行的,它不會阻塞位于同一個語句塊中后面語句的執(zhí)行。使用 <= 作為賦值符。
使用非阻塞賦值是為了避免競爭冒險,那么實際使用中只需要記?。篴lways時序邏輯塊中多用非阻塞賦值<=(后面的時序邏輯電路會另外介紹),always組合邏輯塊中多用阻塞賦值=; 在仿真電路時,initial 塊中一般多用阻塞賦值=。 例如下面代碼中,用assign語句,always組合語句,always時鐘語句實現(xiàn)異或:
module top_module(
input clk,
input a,
input b,
output wire out_assign,
output reg out_always_comb,
output reg out_always_ff );
assign out_assign=a^b;
always @(*) begin
out_always_comb=a^b;
end
always @(posedge clk) begin
out_always_ff<=a^b;
end
endmodule
if 語句格式
if (condition1) true_statement1 ;
else if (condition2) true_statement2 ;
else if (condition3) true_statement3 ;
else default_statement ;
如果只有兩種情況,就只有if和else。下面用上一節(jié)中MUX2_1做練習(xí),上一節(jié)中采用了assign的三元運(yùn)算符語句,這里使用always if語句實現(xiàn)2選1的功能:
module mux2_1
(
input wire in1,
input wire in2,
input wire sel,
output reg out
);
always@(*) begin
if(sel == 1'b1)//當(dāng)“if...else...”中只有一個變量時不需要加“begin...end".但是為了不遺漏,還是加上
out = in1; //always塊中的組合邏輯關(guān)系時使用阻塞賦值“=”進(jìn)行賦值
else
out = in2;
end
endmodule
3-8譯碼器的實現(xiàn)
從真值表看譯碼器功能:A2A1A0=000-111共8種輸入,對應(yīng)輸出Y的下標(biāo),對應(yīng)下標(biāo)的輸出為1(高電平),其他輸出為0(低電平)。3-8譯碼器的作用:按照之前點亮LED燈的思路,如果一個I/O端口控制一個LED燈,那么I/O端口很有可能不夠用,這時候聰明人就想出用3-8譯碼器的方式,3個I/O口就可以控制8個LED。
設(shè)計規(guī)劃
這個示例中,繪制波形圖如圖所示。
編寫代碼
module decoder3_8
(
input wire in1 ,
input wire in2 ,
input wire in3 ,
output reg [7:0] out
);
always@(*) begin
//使用"{}"位拼接符將3個1bit數(shù)據(jù)按照順序拼成一個3bit數(shù)據(jù)
if({in1, in2, in3} == 3'b000)
out = 8'b0000_0001;
else if({in1, in2, in3} == 3'b001)
out = 8'b0000_0010;
else if({in1, in2, in3} == 3'b010)
out = 8'b0000_0100;
else if({in1, in2, in3} == 3'b011)
out = 8'b0000_1000;
else if({in1, in2, in3} == 3'b100)
out = 8'b0001_0000;
else if({in1, in2, in3} == 3'b101)
out = 8'b0010_0000;
else if({in1, in2, in3} == 3'b110)
out = 8'b0100_0000;
else if({in1, in2, in3} == 3'b111)
out = 8'b1000_0000;
else
out = 8'b0000_0001;
end
endmodule
代碼中使用always組合邏輯塊。由于if對應(yīng)的執(zhí)行語句只有一句,可以不使用begin,end但是由于怕以后復(fù)雜代碼會遺漏,還是選擇加上。always組合邏輯使用阻塞語句,用=進(jìn)行賦值。
上面RTL代碼綜合出的RTL視圖:
編寫testbench
`timescale 1ns/1ns
module tb_decoder3_8();
//reg define
reg in1;
reg in2;
reg in3;
wire [7:0] out;
//初始化輸入信號
initial begin
in1 <= 1'b0;
in2 <= 1'b0;
in3 <= 1'b0;
end
always #10 in1 <= {$random} % 2;
always #10 in2 <= {$random} % 2;
always #10 in3 <= {$random} % 2;
//-------------decoder3_8_inst----------------
decoder3_8 decoder3_8_ins
(
.in1(in1),
.in2(in2),
.in3(in3),
.out(out)
);
endmodule
經(jīng)過兩個實例后,testbench已經(jīng)超級熟悉了吧。
對比波形
看看波型:
驗證了波形的正確性
-
FPGA
+關(guān)注
關(guān)注
1630文章
21766瀏覽量
604598 -
FPGA設(shè)計
+關(guān)注
關(guān)注
9文章
428瀏覽量
26560 -
仿真
+關(guān)注
關(guān)注
50文章
4108瀏覽量
133779 -
時鐘
+關(guān)注
關(guān)注
11文章
1738瀏覽量
131623 -
譯碼器
+關(guān)注
關(guān)注
4文章
311瀏覽量
50394
發(fā)布評論請先 登錄
相關(guān)推薦
評論