编写高效的测试设计(
test benches)
原文作者:Mujtaba Hamid
注:
一个设计的测试验证是非常重要的。有效的测试可以助我们快速的完成或改善设计。Testbenches建议编写有效的测试代码来通过软件实现可靠的验证。无意中发现,顺手译为中文,以备将来方便。也贴给没有找到更好中文版本的同道人。
Testbenches本意应该是测试平台更合理,但是在中文中阅读起来很不舒服。所以本文中有时译为“测试设计”,“测试代码”,有时干脆是“测试”。
摘要:
应用笔记为
HDL验证设计的新手,或者是没有丰富的测试设计经验的逻辑设计者而编写。
测试设计是验证
HDL设计的主要手段。本应用笔记为创建或准备和构建有效的测试设计提供准则。它也提供一个为任何设计开发自较验测的测试设计的一个代数方法。
涉及的所有设计文件可以从以下的站点获得:
PC: ftp://ftp.xilinx.com/pub/applications/xapp/xapp199.zip
UNIX: ftp://ftp.xilinx.com/pub/applications/xapp/xapp199.tar.gz
简介:
由于设计的规模越来越大和越来越复杂,数字设计的验证已经成为一个日益困难和繁琐的事。面对挑战,验证工程师们依靠许多的验证工具和方法。对于大的系统,如几百万门的设计,工程师们一般使用一套可靠的验证工具。当然,对于一些小的设计,设计工程师常常发现带有测试的
hdl仿真器就可以做得很好。
测试设计已经成为一个验证高级语言设计
HLL (High-Level Language) 的标准方法。
典型的,测试设计完成以下任务:
实现测试设计;
仿真通过使用模块的测试向量来仿真测试设计;
输出结果到终端或波形窗口以检视;
可选择的将实际结果和预期结果进行比较。
一般测试设计使用工业标准的
VHDL
或
verilog
硬件描述语言来编写。测试设计调用功能设计,然后仿真。复杂的测试设计完成一些附加的功能
----如它们包含逻辑来为设计决定适当的设计激励或比较实际结果和预期结果。
后续的章节说明了一个非常稳定的测试设计的结构,并且提供了一个自较验测例子
----它将自动比较实际结果和测试设计的预期结果。
图
1
说明一个基于以上基本要求的标准的
hdl
验证流程。由于测试设计使用
VHDL
或
verilogHDL
来描述,测试设计的验证过程可以在不同的平台或不同公司的软件工具环境完成。另外,由于
VHDL
或
verilogHDL
是公开的通用标准语言
,
使用
VHDL
或
verilogHDL来描述验证设计可以毫无困难的在将来重用。
图
1
使用测试设计的
HDL
测试验证流程
构建测试设计:
测试设计可以用
VHDL
或
verilogHDL
来描述
.
因为测试设计只用来进行仿真,它们没有那些适应综合中仅应用的
rtl
语言子集的语法约束的限制
.
而是所有的行为结构都可以使用。从而测试设计可以编写的更为通用,使得它们可以更容易维护。
所有的测试设计包含了如表1的基本程序段块。正如上面所提到的,测试设计一般包含更多的附加功能,如在终端上可视的结果和内建的错误检测。
表1
测试设计的基本程序段
下面的例子说明经常使用的测试设计的结构。
产生时钟信号
使用系统时钟来的时序逻辑设计必须产生时钟。重复的时钟信号可以很容易的在
vhdl
或
verilog
源码中实现。以下是
vhdl
和
verilog的时钟发生示例。
VHDL:
-- Declare a clock period constant.
Constant ClockPeriod : TIME := 10 ns;
-- Clock Generation method 1:
Clock <= not Clock after ClockPeriod / 2;
-- Clock Generation method 2:
GENERATE CLOCK: process
begin
wait for (ClockPeriod / 2)
Clock <= ’1’;
wait for (ClockPeriod / 2)
Clock <= ’0’;
end process;
Verilog:
// Declare a clock period constant.
Parameter ClockPeriod = 10;
// Clock Generation method 1:
initial begin
forever Clock = #(ClockPeriod / 2) ~ Clock;
end
// Clock Generation method 2:
initial begin
always #(ClockPeriod / 2) Clock = ~Clock;
end
准备激励信号
为了获得测试设计的验证结果,激励必须在测试设计中提供。在测试设计中使用的并行激励块提供必要的激励。两个方法被考虑:绝对时间激励和相对时间激励。在第一个方法里,仿真变量被详细描述为相对于仿真时间零点。通过比较,相对时间激励提供初始值,然后在重触发激励前等待一个事件。根据设计者的需要,两种方法可以在测试设计中组合使用。
表
2
绝对时间激励
表
2
和表
3
分别以
vhdl
和
verilog
提供了一个绝对时间激励和相对时间激励的源代码。
表
3
相对时间激励
VHDL
进程块和
Verilog初始块与设计文件中的其他的进程块或初始块同时执行。然而,在每一个进程块或初始块中,事件是按照书写的顺序有序的规划的。这说明在仿真时间零点并发的每一个块激励的顺序。多模块应该被用来将复杂的激励顺序分解为有更好的可读性和方便维护的代码。
显示结果
在
verilog
中推荐使用关键字
$display
和
$monitor
显示结果。虽然
vhdl
没有等效的显示指令,它提供了
std_textio
标准文本输入输出程序包。它允许文件的
i/o重定向到显示终端窗口(作为这个技术的示例,参看下面的自较验查验证设计)
下面是
verilog
示例
,它将在终端屏幕上显示一些值。
// pipes the ASCII results to the terminal or text editor
initial begin
$timeformat(-9,1,"ns",12);
$display(" Time Clk Rst Ld SftRg Data Sel");
$monitor("%t %b %b %b %b %b %b", $realtime,
clock, reset, load, shiftreg, data, sel);
end
关键字
$display
在终端屏幕上输出引用的附加的说明文字(
“
。。。
”
)
.
关键字
$monitor
操作不同。因为它的输出是事件驱动的。例中的变量
$realtime
(由用户赋值到当前的仿真时间)用于触发信号列表中值的显示。信号表由变量
$realtime
开始,跟随其他将要显示的信号名(
clock, reset, load
等)。以
%
开始的关键字包含一个格式描述的表,用来控制如何格式化显示信号列表中的每个信号的值。格式列表是位置确定的。每个格式说明有序地与信号列表中的信号顺序相关。比如
%t
说明规定了
$realtime
的值是时间格式。并且第一个
%b
说明符格式化
clock
的值是二进制形式。
verilog
提供附加的格式说明,比如
%h
用于说明十六进制,
%d
说明十进制,
%c
说明显示为八进制。(参见
verilog准则了解完整的关键字及格式描述符)
图
2
说明格式显示结果
图
2
仿真结果返回结果
简单的测试设计
简单的测试设计实例化用户设计,然后提供相应的激励。测试输出被图形化显示在仿真器的波形窗口里或者作为文本发送到用户的终端或者是管道输出文本。
以下是一个简单的用
Verilog实现的设计,它实现了一个移位寄存器的功能。
module shift_reg (clock, reset, load, sel, data, shiftreg);
input clock;
input reset;
input load;
input [1:0] sel;
input [4:0] data;
output [4:0] shiftreg;
reg [4:0] shiftreg;
always @ (posedge clock)
begin
if (reset)
shiftreg = 0;
else if (load)
shiftreg = data;
else
case (sel)
2’b00 : shiftreg = shiftreg;
2’b01 : shiftreg = shiftreg << 1;
2’b10 : shiftreg = shiftreg >> 1;
default : shiftreg = shiftreg;
endcase
end
endmodule
以下是简单的测试设计示例移位寄存器设计的例子
,verilog描述。
module testbench; // declare testbench name
reg clock;
reg load;
reg reset; // declaration of signals
wire [4:0] shiftreg;
reg [4:0] data;
reg [1:0] sel;
// instantiation of the shift_reg design below
shift_reg dut(.clock (clock),
.load (load),
.reset (reset),
.shiftreg (shiftreg),
.data (data),
.sel (sel));
//this process block sets up the free running clock
initial begin
clock = 0;
forever #50 clock = ~clock;
end
initial begin// this process block specifies the stimulus.
reset = 1;
data = 5’b00000;
load = 0;
sel = 2’b00;
#200
reset = 0;
load = 1;
#200
data = 5’b00001;
#100
sel = 2’b01;
load = 0;
#200
sel = 2’b10;
#1000 $stop;
end
initial begin// this process block pipes the ASCII results to the
//terminal or text editor
$timeformat(-9,1,"ns",12);
$display(" Time Clk Rst Ld SftRg Data Sel");
$monitor("%t %b %b %b %b %b %b", $realtime,
clock, reset, load, shiftreg, data, sel);
end
endmodule
以上的测试设计实例化设计,设置时钟,提供激励信号。所有的进程块在仿真时间零点开始。英镑标记(
#
)说明下一个激励作用前的延迟。
$stop
命令使仿真器停止测试仿真(所有测试设计中都应该包含一个停止命令)。最后,
$monitor
语句返回
ascII
格式的结果到屏幕或者管道输出到一个文本编辑器。接后的是一个
vhdl
描述的的测试设计,它实例化设计并提供激励到上述用
verilog描述的移位寄存器.
VHDL 测试设计示例:
library IEEE;
use IEEE.std_logic_1164.all;
entity testbench is
end entity testbench;
architecture test_reg of testbench is
component shift_reg is
port (clock : in std_logic;
reset : in std_logic;
load : in std_logic;
sel : in std_logic_vector(1 downto 0);
data : in std_logic_vector(4 downto 0);
shiftreg : out std_logic_vector(4 downto 0));
end component;
signal clock, reset, load: std_logic;
signal shiftreg, data: std_logic_vector(4 downto 0);
signal sel: std_logic_vector(1 downto 0);
constant ClockPeriod : TIME := 50 ns;
begin
UUT : shift_reg port map (clock => clock, reset => reset,
load => load, data => data,
shiftreg => shiftreg);
process begin
clock <= not clock after (ClockPeriod / 2);
end process;
process begin
reset <= ’1’;
data <= "00000";
load <= ’0’;
set <= "00";
wait for 200 ns;
reset <= ’0’;
load <= ’1’;
wait for 200 ns;
data <= "00001";
wait for 100 ns;
sel <= "01";
load <= ’0’;
wait for 200 ns;
sel <= "10";
wait for 1000 ns;
end process;
end architecture test_reg;
上述
vhdl
测试设计与之前提到的
verilog
测试设计的功能是相似的,如希望用一个命令来返回输出到终端。在
vhdl
中,
std_textio程序包被用于在终端上显示信息,它将被搁到下一节说明。
自动验证
推荐自动实现测试结果的验证,尤其是对于较大的设计来说。自动化减少了检查设计是否正确所要求的时间,也使人可能的犯错最少。
一般有以下几种常用的自动测试验证的方法:
1、数据库比较。首先,要创建一个包含预期输出(一个黄金向量文件)的数据库文件。然后,仿真输出被捕获并与黄金向量文件中参考的向量比较(在
unix
中的
diff
工具可以用来比较
ascii数据文件)。然而,因为从输出到输入文件指针没有提供,是这种方法的一个缺点,使得跟踪一个导致错误输出的原因比较困难。
2、波形比较。波形比较可以自动或是手动的运行。自动的方法使用一个测试比较器来比较黄金波形与测试输出波形。
xilinx
的
hdl bencher
工具可以用于执行一个自动波形比较(关于
hdl bencher的相关信息,请参看
http://www.xilinx.com/products/software/statecad/index.htm
)
3
、自较验测试。一个自较验测试检查预期的结果与运行时间的实际结果,并不是在仿真结束以后。因为有用的错误跟踪信息可以内建在一个测试设计中,用来说明哪些地方设计有误,调试时间可以非常明显地缩短。更多的关于自较验测试的信息在下一节说明。
自较验测试
自较验测试通过在一个测试文档中放置一系列的预期向量表来实现。运行时间时间间隔将这些向量与定义好的实际仿真结果进行比较。如果实际结果与预期结果匹配,仿真成功。如果结果不匹配,测试报告两者的差异。
为同步设计实现自较验测试更简单一些,因为与实现的结果相比较可以在一个时钟沿或任何一个整数倍的时钟周期后。比较的方法基于设计本身的特性。比如一个用于内存
I/O的测试应该检查每一次更新数据时的结果或者从一个内存位置读取。类似的,如果一个设计用了一个显而易见的组合块的数字,在预期结果描述时,组合时延就必须要考虑。
在自较验测试中,预期输出与实际输出在一个特定的运行时间间隔比较以便提供自动的错误检查。这个技术在小到中型的设计中非常好。但是,因为当设计复杂后,可能的输出组合成指数倍的增长,为一个大型设计编写一个自较验测试设计是非常困难和非常费时的。
以下是一个用
verilog
和
vhdl
描述的自较验测试的简单的例子
:
Verilog
例子
下述的设计实例中,预期的结果被详细说明。后面的代码,两种结果被比较,比较的结果被返回终端。如果没有错误,一个
“
end of good simulation
”消息会显示。如果失配发生,根据期望与实际值的失配情况,错误会被相应报告。
‘timescale 1 ns / 1 ps
module test_sc;
reg tbreset, tbstrtstop;
reg tbclk;
wire [6:0] onesout, tensout;
wire [9:0] tbtenthsout;
parameter cycles = 25;
reg [9:0] Data_in_t [0:cycles];
// /
// Instantiation of the Design
// /
stopwatch UUT (.CLK (tbclk), .RESET (tbreset), .STRTSTOP (tbstrtstop),
.ONESOUT (onesout), .TENSOUT (tensout), .TENTHSOUT (tbtenthsout));
wire [4:0] tbonesout, tbtensout;
assign tbtensout = led2hex(tensout);
assign tbonesout = led2hex(onesout);
///
//EXPECTED RESULTS
///
initial begin
Data_in_t[1] =10’b1111111110;
Data_in_t[2] =10’b1111111101;
Data_in_t[3] =10’b1111111011;
Data_in_t[4] =10’b1111110111;
Data_in_t[5] =10’b1111101111;
Data_in_t[6] =10’b1111011111;
Data_in_t[7] =10’b1110111111;
Data_in_t[8] =10’b1101111111;
Data_in_t[9] =10’b1011111111;
Data_in_t[10]=10’b0111111111;
Data_in_t[11]=10’b1111111110;
Data_in_t[12]=10’b1111111110;
Data_in_t[13]=10’b1111111101;
Data_in_t[14]=10’b1111111011;
Data_in_t[15]=10’b1111110111;
Data_in_t[16]=10’b1111101111;
Data_in_t[17]=10’b1111011111;
Data_in_t[18]=10’b1110111111;
Data_in_t[19]=10’b1101111111;
Data_in_t[20]=10’b1011111111;
Data_in_t[21]=10’b0111111111;
Data_in_t[22]=10’b1111111110;
Data_in_t[23]=10’b1111111110;
Data_in_t[24]=10’b1111111101;
Data_in_t[25]=10’b1111111011;
end
reg GSR;
assign glbl.GSR = GSR;
initial begin
GSR = 1;
// ///
// Wait till Global Reset Finished
// ///
#100 GSR = 0;
end
//
// Create the clock
//
initial begin
tbclk = 0;
// Wait till Global Reset Finished, then cycle clock
#100 forever #60 tbclk = ~tbclk;
end
initial begin
// //
// Initialize All Input Ports
// //
tbreset = 1;
tbstrtstop = 1;
// /
// Apply Design Stimulus
// /
#240 tbreset = 0;
tbstrtstop = 0;
#5000 tbstrtstop = 1;
#8125 tbstrtstop = 0;
#500 tbstrtstop = 1;
#875 tbreset = 1;
#375 tbreset = 0;
#700 tbstrtstop = 0;
#550 tbstrtstop = 1;
// /
// simulation must be halted inside an initial statement
// /
// #100000 $stop;
end
integer i,errors;
///
///
// Block below compares the expected vs. actual results
// at every negative clock edge.
///
///
always @ (posedge tbclk)
begin
if (tbstrtstop)
begin
i = 0;
errors = 0;
end
else
begin
for (i = 1; i <= cycles; i = i + 1)
begin
@(negedge tbclk)
// check result at negedge
$display("Time%d ns; TBSTRTSTOP=%b; Reset=%h; Expected
TenthsOut=%b; Actual TenthsOut=%b", $stime, tbstrtstop, tbreset,
Data_in_t, tbtenthsout);
if ( tbtenthsout !== Data_in_t )
begin
$display(" ------ERROR. A mismatch has occurred-----");
errors = errors + 1;
end
end
if (errors == 0)
$display("Simulation finished Successfully.");
else if (errors > 1)
$display("%0d ERROR! See log above for details.",errors);
else
$display("ERROR! See log above for details.");
#100 $stop;
end
end
endmodule
这种简单的自较验测试设计可以转换到任何测试场合----当然,预期的输出值和信号的名字在重用时是需要更改的。如果不需要每个时钟沿检查,需要的话可以修改for-loop结构。
如果仿真成功,下图的信息就会在显示终端上显示:
图3 verilog
VHDL
示例
:
在
VHDL
中,向量文件包含预期的结果。
VHDL
的
textio
程序包用于从向量文件中读取数据,和显示错误信息。这个测试用
VHDL
示例秒表设计.
LIBRARY IEEE;
USE IEEE.std_logic_1164.all;
LIBRARY ieee;
USE IEEE.STD_LOGIC_TEXTIO.ALL;
USE STD.TEXTIO.ALL;
ENTITY testbench IS
END testbench;
ARCHITECTURE testbench_arch OF testbench IS
COMPONENT stopwatch
PORT (
CLK : in STD_LOGIC;
RESET : in STD_LOGIC;
STRTSTOP : in STD_LOGIC;
TENTHSOUT : out STD_LOGIC_VECTOR (9 DOWNTO 0);
Figure 3: Verilog Example Verification
ONESOUT : out STD_LOGIC_VECTOR (6 DOWNTO 0);
TENSOUT : out STD_LOGIC_VECTOR (6 DOWNTO 0)
);
END COMPONENT;
SIGNAL CLK : STD_LOGIC;
SIGNAL RESET : STD_LOGIC;
SIGNAL STRTSTOP : STD_LOGIC;
SIGNAL TENTHSOUT : STD_LOGIC_VECTOR (9 DOWNTO 0);
SIGNAL ONESOUT : STD_LOGIC_VECTOR (6 DOWNTO 0);
SIGNAL TENSOUT : STD_LOGIC_VECTOR (6 DOWNTO 0);
constant ClockPeriod : Time := 60 ns;
FILE RESULTS: TEXT IS OUT "results.txt";
signal i: std_logic;
BEGIN
UUT : stopwatch
PORT MAP (
CLK => CLK,
RESET => RESET,
STRTSTOP => STRTSTOP,
TENTHSOUT => TENTHSOUT,
ONESOUT => ONESOUT,
TENSOUT => TENSOUT
);
stimulus: PROCESS
begin
reset <= ’1’;
strtstop <= ’1’;
wait for 240 ns;
reset <= ’0’;
strtstop <= ’0’;
wait for 5000 ns;
strtstop <= ’1’;
wait for 8125 ns;
strtstop <= ’0’;
wait for 500 ns;
strtstop <= ’1’;
wait for 875 ns;
reset <= ’1’;
wait for 375 ns;
reset <= ’0’;
wait for 700 ns;
strtstop <= ’0’;
wait for 550 ns;
strtstop <= ’1’;
end process stimulus;
clock: process
begin
clk <= ’1’;
wait for 100 ns;
loop
wait for (ClockPeriod / 2);
CLK <= not CLK;
end loop;
end process clock;
check_results : process
variable tmptenthsout: std_logic_vector(9 downto 0);
variable l: line;
variable good_val, good_number, errordet: boolean;
variable r : real;
variable vector_time: time;
variable space: character;
file vector_file: text is in "values.txt";
begin
while not endfile(vector_file) loop
readline(vector_file, l);
read(l, r, good => good_number);
next when not good_number;
vector_time := r * 1 ns;
if (now < vector_time) then
wait for vector_time - now;
end if;
read(l, space);
read(l, tmptenthsout, good_val);
assert good_val REPORT "bad tenthsoutvalue";
wait for 10 ns;
if (tmptenthsout /= tenthsout) then
assert errordet REPORT "vector mismatch";
end if;
end loop;
wait;
end process check_results;
end testbench_arch;
library XilinxCoreLib;
CONFIGURATION stopwatch_cfg OF testbench IS
FOR testbench_arch
FOR ALL : stopwatch use configuration work.cfg_tenths;
END FOR;
END FOR;
END stopwatch_cfg;
以下向量文件用于上述的测试。它包含了预期的仿真值。
-- Vector file containing expected results
0 1111111110
340 1111111110
400 1111111101
460 1111111011
520 1111110111
580 1111101111
640 1111011111
700 1110111111
760 1101111111
820 1011111111
880 0111111111
940 1111111110
1000 1111111110
1060 1111111101
1120 1111111011
1180 1111110111
1240 1111101111
1300 1111011111
1360 1110111111
1420 1101111111
1480 1011111111
1540 0111111111
1600 1111111110
1660 1111111110
1720 1111111101
1780 1111111011
如果错误被检测到,它会显示在一个仿真提示器中显示。图
4
展示在
MTI
脚本窗口的错误显示
图
4
仿真命令错误报告
编辑测试文件的准则
本节提供测试设计的编辑准则。正如计划一个电路设计可以帮助构建更好的电路性能,计划好测试方案可以提高仿真验证的结果。
在编写测试设计前要了解仿真器
虽然通用仿真工具遵从
HDL工业标准,但标准并没有说明多少重要的仿真描述条项。不同的仿真器有不同的功能,兼容能力,和执行性能,形成不同的仿真结果。
--基于事件
vs基于周期的仿真
仿真器使用基于事件或基于周期的仿真方法。基于事件的仿真器,当输入,信号,或是门改变了值,来确定仿真器事件的时间。在一个基于事件的仿真器中,一个延时值可以附加在门电路或是电路网络上来构建最适的时间仿真。基于周期的仿真器面向同步设计。他们最优化组合逻辑,在时钟周期内分析结果。这个功能使得基于周期的仿真器比基于事件的仿真器更快更有效。然而,由于基于周期的仿真器不允许详细的时间说明,他们并不如基于事件的仿真器准确。对于更多的关于两者的差异的信息,参看
http://www.ednmag.com/ednmag/reg/1996/070496/14df4.htm
上的
"数字逻辑仿真:事件驱动,周期驱动,和Home-Brewed"(Digital Logic Simulation: Event-Driven, Cycle-Based, and Home-Brewed),
--确定事件时间
基于事件的仿真器提供商使用不同的运算法则来确定仿真事件。所以,根据仿真器用来确定的运算法则不同,同一个仿真时间的事件被确定为不同的次序(根据在每个事件之间插入的
delta延时)。为避免对运算法则的依赖和确保正确的结果,一个事件驱动测试应该详细描述明确的激励次序。
--
避免使用无限循环
当一个事件添加到基于事件的仿真器,
cpu
和内存的使用就增加了,仿真过程就会变慢。除非是评价测试设计,无限循环不应该使用来作为设计的激励。一般地,时钟被说明为一个内部的无限循环(如
verilog
中的
'forever'循环),但是没有其他信号事件。
--细分激励到逻辑模块
在测试中,所有初始块(
verilog
)或进程块
(VHDL)并列地运行。如果无关的激励被分离到独立的块,测试激励的次序会变得更容易实现和检视。因为每个并行的块相关于零点运行,对于分离的块传递激励更容易。分离激励块的使用使得测试设计建立,维护和升级(参看后面的高级测试技术,及该技术的示例)
--
避免显示并不重要的数据
大型设计的测试可能包含
10
万以上的事件或匿名信号。显示大量的仿真数据会相当地降低仿真的速度。最好只是尝试每整数时钟周期时相应的信号来确保适当的仿真速度。
Xilinx 仿真流程要决
配置语句(
VHDL)
一个
VHDL
配置语句允许一个实体链接到一个面向综合或者仿真的详细结构。在
xilinx core generator vhdl
功能仿真流程中,配置语句被用于一个设计的核心仿真模块的调用(切分)。如果核心仿真模块没有绑定在一个设计,仿真将不能正确的工作。关于配置语句使用的例子,参看前述的
VHDL
的自较验查测试代码。在
xilinx CORE Generator
设计中详细配置语句的使用信息可以在
http://support.xilinx.com/suppor ... als31i.htm#Modelsim
页面中的
Modelsim VHDL仿真教程中找到。
http://support.xilinx.com/suppor ... ls31i.htm#Modelsim.
为仿真初始化内存
RAM块
根据默认值,
Xilinx Virtex?
块
RAMs
在所有数据位置都是初始为零,从零点开始的。对于一个
post-NGDBuild, post-MAP, or Post-PAR
的时序仿真,块
RAMs
初始为用户在约束文件(
ucf
)中指定的值或在输入设计文件到
NGDBuild
时通过
init
属性来指定值
.
对于一个
pre-synthesis
或
post-synthesis (pre-NGDBuild)
的功能仿真,一个配置语句必须用来给
RAM
块提供初值。下面是一个用来初始化
RAM块的配置语句的例子。
library IEEE;
use IEEE.std_logic_1164.all;
use IEEE.std_logic_unsigned.all;
library UNISIM;
use UNISIM.vcomponents.all;
configuration cfg_ex_blkram_tb of ex_blkram_tb is
for tb
for uut : ex_blkram use entity work.ex_blkram(struct);
for struct
for INST_RAMB4_S4 : RAMB4_S4 use entity
unisim.RAMB4_S4(RAMB4_S4_V)
generic map (INIT_00 =>
X"1F1E1D1C1B1A191817161514131211100F0E0D0C0B0A09080706050403020100
",
INIT_01 =>
X"3F3E3D3C3B3A393837363534333231302F2E2D2C2B2A29282726252423222120
",
.
.
.
INIT_0F=>
X"FFFEFDFCFBFAF9F8F7F6F5F4F3F2F1F0EFEEEDECEBEAE9E8E7E6E5E4E3E2E1E0
");
end for;
end for;
end for;
end for;
end cfg_ex_blkram_tb;
高级测试技术
根据任务和过程细分激励模块
在创建一个大的测试设计时,激励将会被分割使得代码清晰而易于修改。任务(
verilog
)或过程(
VHDL
)可以被用来分割信号。在下面的例子中,测试激励用于一个
SDRAM控制器的设计。设计包括重复的激励模块,以便测试设计中通过声明独立的任务分割激励,这些任务稍后被调用来进行独立块的功能的仿真执行。
Verilog
示例
task addr_wr;
input [31 : 0] address;
begin
data_addr_n = 0;
we_rn = 1;
ad = address;
end
endtask
task data_wr;
input [31 : 0] data_in;
begin
data_addr_n = 1;
we_rn = 1;
ad = data_in;
end
endtask
task addr_rd;
input [31 : 0] address;
begin
data_addr_n = 0;
we_rn = 0;
ad = address;
end
endtask
task data_rd;
input [31 : 0] data_in;
begin
data_addr_n = 1;
we_rn = 0;
ad = data_in;
end
endtask
task nop;
begin
data_addr_n = 1;
we_rn = 0;
ad = hi_z;
end
endtask
这些任务指定独立的设计功能元素
----地址可读可写,数据可读可写,或者空操作。一量指定,这些任务可以在激励进程中被调用。如下所示:
Initial begin
nop ; // Nop
#( 86* ‘CYCLE +1); addr_wr (32’h20340400); // Precharge, load
Controller MR
#(‘CYCLE); data_wr (32’h0704a076); // value for Controller MR
#(‘CYCLE); nop ; // Nop
#(5 * ‘CYCLE); addr_wr (32’h38000000); // Auto Refresh
#(‘CYCLE); data_wr (32’h00000000); //
#(‘CYCLE); nop ; // Nop
…
…
end
VHDL例程
以下是相当设计的
VHDL测试文件,分别细分到独立的过程。
Stimulus : process
procedure addr_wr (address: in std_logic_vector(31 downto 0)) is
begin
data_addr_n <= ‘0’;
we_rn <= ‘1’;
ad <= address;
end addr_wr;
procedure data_wr (data_in: in std_logic_vector(31 downto 0 )) is
begin
data_addr_n <= ‘1’;
we_rn <= ‘1’;
ad <= data_in;
end data_wr;
procedure addr_rd (address: in std_logic_vector(31 downto 0)) is
begin
data_addr_n <= ‘0’;
we_rn <= ‘0’;
ad <= address;
end addr_rd;
procedure data_rd (data_in: in std_logic_vector(31 downto 0)) is
begin
data_addr_n <= ‘1’;
we_rn <= ‘0’;
ad <= data_in;
end data_rd;
procedure nop is
begin
data_addr_n <= ‘1’;
we_rn = ‘0’;
ad = ‘Z’;
end nop;
begin
nop ; -- Nop
wait for 200 ns;
addr_wr (16#20340400#); -- Precharge, load Controller MR
wait for 8 ns;
data_wr (16#0704a076#); -- value for Controller MR
wait for 8 ns;
nop ; -- Nop
wait for 40 ns;
addr_wr (16#38000000#); -- Auto Refresh
wait for 8 ns;
data_wr (16#00000000#);
wait for 8 ns;
nop ; -- Nop
..
..
细分激励到独立的任务使得激励很容易实现,也使得代码的可读性更好。
在仿真时控制双向信号
多数设计使用双向信号,在测试设计中必须区别对待双向信号和单向信号。
VHDL
示例
The following is a VHDL bi-directional signal example:
以下是一个
vhdl描述的双向信号示例
Library IEEE;
use IEEE.STD_LOGIC_1164.all;
use IEEE.STD_LOGIC_UNSIGNED.all;
entity bidir_infer is
port (DATA : inout STD_LOGIC_VECTOR(1 downto 0);
READ_WRITE : in STD_LOGIC);
end bidir_infer;
architecture XILINX of bidir_infer is
signal LATCH_OUT : STD_LOGIC_VECTOR(1 downto 0);
begin
process(READ_WRITE, DATA)
begin
if (READ_WRITE = ’1’) then
LATCH_OUT <= DATA;
end if;
end process;
process(READ_WRITE, LATCH_OUT)
begin
if (READ_WRITE = ’0’) then
DATA(0) <= LATCH_OUT(0) and LATCH_OUT(1);
DATA(1) <= LATCH_OUT(0) or LATCH_OUT(1);
else
DATA(0) <= ’Z’;
DATA(1) <= ’Z’;
end if;
end process;
end XILINX;
为访问上例中的双向的
DATA信号,一个测试可以设置如下:
library ieee;
use ieee.std_logic_1164.all;
Entity testbench is
End testbench;
Architecture test_bidir of testbench is
Component bidir_infer
port (DATA : inout STD_LOGIC_VECTOR(1 downto 0);
READ_WRITE : in STD_LOGIC);
end component;
signal read_writet: std_logic;
signal datat, data_top : std_logic_vector(1 downto 0);
begin
datat <= data_top when (Read_writet = ’1’) else (others => ’Z’);
data_top <= datat when (Read_writet = ’0’) else (others => ’Z’);
uut : bidir_infer port map (datat, read_writet);
process begin
read_writet <= ’1’;
data_top <= "10";
wait for 50 ns;
read_writet <= ’0’;
wait;
end process;
end test_bidir;
双向总线由测试台控制,双向总线的值可以通过数据顶层信号来访问。
Verilog示例
以下是
verilog设计的可决断的双向总线示例。
module bidir_infer (DATA, READ_WRITE);
input READ_WRITE ;
inout [1:0] DATA ;
reg [1:0] LATCH_OUT ;
always @ (READ_WRITE or DATA)
begin
if (READ_WRITE == 1)
LATCH_OUT <= DATA;
end
assign DATA = (READ_WRITE == 1) ? 2’bZ : LATCH_OUT;
endmodule
Verilog测试设计可以如下设置:
module test_bidir_ver;
reg read_writet;
reg [1:0] data_in;
wire [1:0] datat, data_out;
bidir_infer uut (datat, read_writet);
assign datat = (read_writet == 1) ? data_in : 2’bZ;
assign data_out = (read_writet == 0) ? datat : 2’bZ;
initial begin
read_writet = 1;
data_in = 11;
#50 read_writet = 0;
end
endmodule
在这些测试设计中,
data_in
信号提供激励到设计中的双向
DATA
数据信号,
data_out
信号读取该
DATA数据信号.
为仿真初始化内存
请参考前段的
xilinx
仿真流程要决
(Xilinx Simulation Flow Tips)
有用的语言结构
Verilog
有用的
Verilog
语言结构,如
$monitor, $display,
及
$time,
在前面的
verilog
测试示例中论述过,这一节说明另外的可以在测试设计中使用的
verilog语句结构。
force/release
force/release语句可以用来跨越进程对一个寄存器或一个电路网络的赋值。这结结构一般用于强制特定的设计的行为。一旦一个强制值释放,这个信号保持它的状态直到新的值被进程赋值。以下是
force/release语句的用法。
module testbench;
..
initial begin
reset = 1;
force DataOut = 101;
#25 reset = 0;
#25 release DataOut;
..
..
end
endmodule
assign/deassign
assign/deassign语句与
force/release
相类似,但是
assign/deassign
只用于设计中的寄存器。他们一般用于设置输入值。就象一个
force
语句,
assign
语句超越进程语句的赋值。以下是一个
assign/deassign语句的用法。
module testbench;
..
..
initial begin
reset = 1;
DataOut = 101;
#25 reset = 0;
release DataOut;
..
..
end
initial begin
#20 assign reset = 1;// this assign statement overrides the earlier
statement #25 reset = 0;
#50 release reset;
endmodule
timescales
timescale指示被用于为测试指定单位时间步。它也影响仿真器的精确度。表示符号为:‘timescale reference_time/precision
Reference_time
是一个用于测量的单位时间。
Precision
决定延时应该达到的精度
,
为仿真设置单位步距。以下是‘
timescale的使用方法。
‘timescale 1 ns / 1 ps
// this sets the reference time to 1 ns and precision to 1 ps.
module testbench;
..
..
initial begin
#5 reset = 1; // 5 unit time steps correspond to 5 * 1ns = 5ns in
simulation time
#10 reset = 0;
..
end
initial begin
$display (“%d , Reset = %b”, $time, reset); // this display
// statement will get executed
// on every simulator step, ie, 1 ps.
end
endmodule
如果仿真使用时延值,仿真就必须运行在一个比最小时延还好的精确度以内(为了归一化时延)。例如,如果
9ps
延时在仿真库中使用,相应仿真的精确度就必须是在
1ps
到
9ps之间可调的范围。
只读储器初始化文件
verilog提供
$readmemb
和
$readmemh
命令来读取
ascii
文件来初始化存储器的内容。这个命令可以在仿真中用来初始化
Xilinx BlockRAM
或者
SelectRAM器件。符号表达如下:
$readmemb (“ ”, design_instance);
MIF是由
crorgenerator
生成的存储器初始化文件(
Memory Initialization File
)。使用者指定
MIF的内容。
VHDL
除了前文曾经叙述过的
vhdl
命令以外(
assert,wait,report
),以下的结构也对
vhdl测试台文件的创建有所帮助。
meminitfile
vhdl提供一个
meminitfile记录用来输入存储模块的内容。以下是它的符号说明:
FILE meminitfile: TEXT IS IN “ ”;
MIF是由
crorgenerator
生成的存储器初始化文件(
Memory Initialization File
)。使用者指定
MIF
的内容。
编码风格准则以下编码准则帮助创建易于阅读和维护的代码
缩进
总是缩进代码来使它易读。推荐使用每个为三或四个英文字符的缩进宽度。
5
个以上字符宽度的缩进常会在右边的页边留下一个空白,当一个字符宽度小于
3个字符时导致过小的缩进。
文件名
总是在源文件名中保持
".v"(verilog)
或
".vhd"(VHDL)文件扩展名。如果这些标准的扩展名被改变了,一些编辑器或文件过滤器就会不能认出这些源文件。
信号命令
使用同样的大小写
----
推荐小写
----
来表示所有的用户信号。
verilog是大小写敏感的,错位的大写可能引起设计综合和仿真失败。并且,一致的信号名称格式风格促使信号名字在源文件中易于定位。使用短的,描述含义的信号名。短的名称更容易输入,而有含义的名称会帮助表明信号的功能。
注释
可以自由地注释测试设计文件代码。注释对于那些要继承或重用代码的人是非常重要的。队此以外,
verilog
和
vhdl
代码语法结构是没有明确含义
----注释代码填补了重要的细节说明,极大地增加了源代码的清晰性和可重用能力
设计结构
为每一个模块或实体保持一个物理文件。独立模块或实体的独立文件使得设计更易于维护。
更多的信息,请参考
hdl
准则的书。许多包含全面的代码编制准则。参考
FPGA设计重用指南,在以下站点可以找到。
http://www.xilinx.com/ipcenter/designreuse/xrfg.htm
结语
Testbenches
提供工程师以可移动,可升级的验证工具。使用混合语言传真器的有效性,设计者可以自由地使用他们选择的语言来验证
vhdl
和
verilog两种设计。高层次行为语言推动了测试设计的发展,测试设计可以用简单的结构并只要求最小数量的源代码。设计得益于自较验测测试,它在仿真过程中自动实现合适的设计的验证。
Xilinx Foundation ise v3.1i
被设计来提供一个无缝的,集成
hdl
设计工作流。
Synplicity
的
Synplify, Synopsys FPGA Express,
和
Xilinx Synthesis Technology (XST),
沿着
Xilinx Foundation
,非常融合地工作在一起来综合代码。
Foundation ISE
被集成与
Modelsim(XE,PE,SE)
一起来仿真我们的设计,与
Xilinx HDL Bencher
集成来自动实现测试台的创建,与
Xilinx StateCad集成来创建状态机的编码。
关于完整的
Foundation ISE及其集成成员套件的信息请连结:
http://www.xilinx.com/xlnx/xil_prodcat_landingpage.jsp.
系统分类: |
电子制造 |
用户分类: |
无分类 |
标签: |
无标签 |
来源: |
整理 |
发表评论 阅读全文(265) | 回复(0)
模拟交通灯控制
发表于 2007-7-15 21:45:40
此程序设计的是模拟交通灯系统的控制,其中状态:ally是东西南北方向都亮黄灯,sny、sng分别为南北方向亮黄、绿灯,同样,ewy和ewg分别为东西方向亮黄、绿灯。
本设计假设南北方向为主干道,通行时间为60秒,东西方向为次干道,通行时间为30秒,在两干道交替通行时有5秒的时间,此时两条干道的黄灯都要闪,在这个时候,原来通行的干道上的车应准备等待,原来在等待的车辆则可准备行驶;
library ieee;
use ieee.std_logic_1164.all;
use ieee.std_logic_unsigned.all;
use ieee.std_logic_arith.all;
entity traffic is
port(clk:in std_logic;
rst:in std_logic;
pout:out std_logic_vector(12 downto 1));
end traffic;
architecture one of traffic is
type state is (ally,sng,sny,ewg,ewy);
signal ct:state:=ally;
signal nt:state;
signal pp :std_logic_vector(12 downto 1);
signal maxtime:bit;
signal mediatime:bit;
signal mintime:bit;
signal mintime1:bit;
signal a:bit;
signal time:integer range 0 to 60;
signal time1:integer range 0 to 30;
signal time2:integer range 0 to 5;
signal time3:integer range 0 to 5;
begin
neg:process(clk,rst) is
begin
if rst='1' then
ct<=ally;
elsif (clk'event and clk='1') then
ct<=nt;
end if;
end process neg;
com:process(ct,clk) is
begin
if rising_edge(clk) then
case ct is
when ally => pp<="100100100100";
nt<=sng;
when sng => pp<="010001010001";mediatime<='0';time1<=time1+1;
if time1=29 then mediatime<='1';end if;
if mediatime='1'then
nt<=sny;
end if;
when sny => mintime<='0';time2<=time2+1;
if a='0' then pp<="110001110001";
else pp<="010001010001";
end if;
if time2=4 then mintime<='1';end if;
if mintime='1' then
nt<=ewg;
end if;
when ewg => pp<="001010001010";maxtime<='0';time<=time+1;
if time="59" then maxtime<='1';end if;
if maxtime='1' then
nt<=ewy;
end if;
when ewy => mintime1<='0';time3<=time3+1;
if a='0' then pp<="001110001110";
else pp<="001010001010";
end if;
if time3=4 then mintime1<='1';end if;
if mintime1='1' then
nt<=sng;
end if;
when others => nt<=ally;
end case;
end if;
end process com;
la:process(ct)
begin
if rising_edge(clk) then
a<=not a;
end if;
end process la;
pout<=pp;
end one;
本人水平十分有限,错误之处难免,望大家指正。
系统分类: |
CPLD/FPGA |
用户分类: |
初学EDA技术 |
标签: |
交通灯控制 |
来源: |
原创 |
发表评论 阅读全文(329) | 回复(0)
基于FPGA的数字钟设计
发表于 2006-12-9 21:15:05
本文所设计的数字钟具有通过reset键对时、 分、 秒调整功能.该设计分为六个部分: 六进制计数器 counter6,十进制计数器 counter10 ,二四进制计数器 counter24, 时钟模块 bclock, LED扫描显示模块 ledctrl。设计使用VHDL 语言, 程序代码如下:
--*****************************************************************
--模块名 : 顶层设计
--文件名: myclock.vhd
--时间:2006年12月9日
--*********************************************************************
library ieee;
use ieee.std_logic_1164.all;
entity myclock is
port(clk1,clk2,reset:in std_logic; --clk1为计数脉冲,clk2为LED扫描脉冲;
hh_set:in std_logic_vector(1 downto 0); --时高位调整;
mh_set,sh_set:in std_logic_vector(2 downto 0); --分,秒高位调整;
hl_set,ml_set,sl_set:in std_logic_vector(3 downto 0); --时,分,秒低位调整;
led_dp:out std_logic; --LED小数点;
sel:out std_logic_vector(2 downto 0); --送三-八译码生成位选信号;
seg:out std_logic_vector(6 downto 0)); --段码;
end myclock;
architecture one of myclock is
component bclock is
port(clk,reset:in std_logic;
hhin:in std_logic_vector(1 downto 0);
mhin,shin:in std_logic_vector(2 downto 0);
hlin, mlin,slin:in std_logic_vector(3 downto 0);
hho: out std_logic_vector(1 downto 0);
mho,sho:out std_logic_vector(2 downto 0);
hlo,mlo,slo:out std_logic_vector(3 downto 0));
end component bclock;
component ledctrl is
port(clk:in std_logic;
hh:in std_logic_vector(1 downto 0);
mh,sh:in std_logic_vector(2 downto 0);
hl,ml,sl:in std_logic_vector(3 downto 0);
dp:out std_logic;
selo:out std_logic_vector(2 downto 0);
sego:out std_logic_vector(6 downto 0));
end component ledctrl;
signal hh1:std_logic_vector(1 downto 0);
signal mh1,sh1:std_logic_vector(2 downto 0);
signal hl1,ml1,sl1:std_logic_vector(3 downto 0);
begin
u1:bclock port map(clk1,reset,hh_set,mh_set,sh_set,hl_set,ml_set,sl_set,hh1,mh1,sh1,hl1,ml1,sl1);
u2:ledctrl port map(clk2,hh1,mh1,sh1,hl1,ml1,sl1,led_dp,sel,seg);
end one;
--*********************************************************************
--模块名 : 十进制数器
--文件名: counter10.vhd
--时间:2006年12月9日
--*********************************************************************
library ieee;
use ieee.std_logic_1164.all;
use ieee.std_logic_unsigned.all;
entity counter10 is
port(clk,reset:in std_logic;
din:in std_logic_vector(3 downto 0);
c:out std_logic;
dout:out std_logic_vector(3 downto 0));
end counter10;
architecture one of counter10 is
signal dd:std_logic_vector(3 downto 0);
signal c1:std_logic;
begin
process(clk,reset) is
begin
if reset='1' then dd<=din;c1<='0';
elsif rising_edge(clk) then
if dd="1001" then dd<="0000";c1<='1';
else dd<=dd+1;c1<='0';
end if;
end if;
end process;
dout<=dd;c<=c1;
end one;
--*****************************************************
--模块名 : 六进制计数器
--文件名: counter6.vhd
--时间:2006年12月9日
--******************************************************
library ieee;
use ieee.std_logic_1164.all;
use ieee.std_logic_unsigned.all;
entity counter6 is
port(clk,reset:in std_logic;
din:in std_logic_vector(2 downto 0);
c:out std_logic;
dout:out std_logic_vector(2 downto 0));
end counter6;
architecture one of counter6 is
signal dd:std_logic_vector(2 downto 0);
signal c1:std_logic;
begin
process(clk,reset) is
begin
if reset='1' then dd<=din;c1<='0';
elsif rising_edge(clk) then
if dd="101" then dd<="000";c1<='1';
else dd<=dd+1;c1<='0';
end if;
end if;
end process;
dout<=dd;c<=c1;
end one;
--*****************************************************
--模块名 : 二十四进制数器
--文件名: counter24.vhd
--时间:2006年12月9日
--******************************************************
library ieee;
use ieee.std_logic_1164.all;
use ieee.std_logic_unsigned.all;
entity counter24 is
port(clk,reset:in std_logic;
dhin:in std_logic_vector(1 downto 0);
dlin:in std_logic_vector(3 downto 0);
dh:out std_logic_vector(1 downto 0);
dl:out std_logic_vector(3 downto 0));
end counter24;
architecture one of counter24 is
signal dl1:std_logic_vector(3 downto 0);
signal dh1:std_logic_vector(1 downto 0);begin
process(clk,reset) is
begin
if reset='1' then dl1<=dlin;dh1<=dhin;
elsif rising_edge(clk) then
if dh1="10"and dl1="0011" then dl1<="0000"; dh1<="00";
elsif dl1="1001" then dl1<="0000";dh1<=dh1+1;
else dl1<=dl1+1;
end if;
end if;
end process;
dh<=dh1;dl<=dl1;
end one;
--*******************************************************
--模块名 : 时钟模块
--文件名: bclock.vhd
--时间:2006年12月9日
--*******************************************************
library ieee;
use ieee.std_logic_1164.all;
use ieee.std_logic_unsigned.all;
entity bclock is
port(clk,reset:in std_logic;
hhin:in std_logic_vector(1 downto 0);
mhin,shin:in std_logic_vector(2 downto 0);
hlin, mlin,slin:in std_logic_vector(3 downto 0);
hho: out std_logic_vector(1 downto 0);
mho,sho:out std_logic_vector(2 downto 0);
hlo,mlo,slo:out std_logic_vector(3 downto 0));
end bclock;
architecture one of bclock is
component counter10 is
port(clk,reset:in std_logic;
din:in std_logic_vector(3 downto 0);
c:out std_logic;
dout:out std_logic_vector(3 downto 0));
end component counter10;
component counter6 is
port(clk,reset:in std_logic;
din:in std_logic_vector(2 downto 0);
c:out std_logic;
dout:out std_logic_vector(2 downto 0));
end component counter6;
component counter24 is
port(clk,reset:in std_logic;
dhin:in std_logic_vector(1 downto 0);
dlin:in std_logic_vector(3 downto 0);
dh:out std_logic_vector(1 downto 0);
dl:out std_logic_vector(3 downto 0));
end component counter24;
signal csl,csh,cml,cmh:std_logic;
begin
u1:counter10 port map(clk,reset,slin,csl,slo);
u2:counter6 port map(csl,reset,shin,csh,sho);
u3:counter10 port map(csh,reset,mlin,cml,mlo);
u4:counter6 port map(cml,reset,mhin,cmh,mho);
u5:counter24 port map(cmh,reset,hhin,hlin,hho,hlo);
end one;
--******************************************************
--模块名 : LED扫描显示模块
--文件名:ledctrl.vhd
--时间:2006年12月9日
--*********************************************************
library ieee;
use ieee.std_logic_1164.all;
use ieee.std_logic_unsigned.all;
use ieee.std_logic_arith.all;
entity ledctrl is
port(clk:in std_logic;
hh:in std_logic_vector(1 downto 0);
mh,sh:in std_logic_vector(2 downto 0);
hl,ml,sl:in std_logic_vector(3 downto 0);
dp:out std_logic;
selo:out std_logic_vector(2 downto 0);
sego:out std_logic_vector(6 downto 0));
end ledctrl;
architecture one of ledctrl is
signal ledon:std_logic_vector(3 downto 0);
signal count:std_logic_vector(2 downto 0);
begin
process(clk) is
begin
if rising_edge(clk) then
if count="101" then count<="000";
else count<=count+1;
end if;
end if;
end process;
selo<=count;
ledon<=sl when count="000" else
'0'&sh when count="001" else
ml when count="010" else
'0'&mh when count="011" else
hl when count="100" else
"00"&hh when count="101" else
"000" ;
dp<='1' when count="010" or count="100" else
'0';
sego<="0111111" when ledon="0000" else
"0000110" when ledon="0001" else
"1011011" when ledon="0010" else
"1001111" when ledon="0011" else
"1100110" when ledon="0100" else
"1101101" when ledon="0101" else
"1111101" when ledon="0110" else
"0000111" when ledon="0111" else
"1111111" when ledon="1000" else
"1101111" when ledon="1001" else
"0000000";
end one;
以上便是全部代码, 错误难免, 敬请指正.
谢谢!!
示例验证