仿真验证与TESTBENCH编写.ppt
第五章 仿真验证与Testbench编写,5/5/2024,1,5.1 Verilog HDL电路仿真和验证概述,仿真,也叫模拟,是通过使用EDA仿真工具,通过输入测试信号,比对输出信号(波形、文本或者VCD文件)和期望值,来确认是否得到与期望所一致的正确的设计结果,验证设计的正确性。验证是一个证明设计思路如何实现,保证设计在功能上正确的一个过程。验证在Verilog HDL设计的整个流程中分为4个阶段:阶段1:功能验证;阶段2:综合后验证;阶段3:时序验证;阶段4:板级验证。,5/5/2024,2,5.2 Verilog HDL测试程序设计基础,5.2.1 Testbench及其结构在仿真的时候Testbench用来产生测试激励给待验证设计(Design Under Verification,DUV),或者称为待测设计(Design Under Test,DUT)。,Testbench平台结构,测试程序的一般结构,由于Testbench是一个测试平台,信号集成在模块内部,没有输入输出。,在Testbench模块内,例化待测设计的顶层模块,并把测试行为的代码封装在内,直接对待测系统提供测试激励。,例5.2-1 T触发器测试程序示例module Tflipflop_tb;/数据类型声明reg clk,rst_n,T;wire data_out;TFF U1(.data_out(data_out),.T(T),.clk(clk),.rst_n(rst_n);/对被测模块实例化always/产生测试激励#5 clk=clk;Initial begin clk=0;#3 rst_n=0;#5 rst_n=1;T=1;#30 T=0;#20 T=1;endInitial/对输出响应进行收集 begin$monitor($time,T=%b,clk=%b,rst_n=%b,data_out=%b,T,clk,rst_n,data_out);endendmodule,T触发器的仿真波形和部分文本输出结果:,部分文本输出结果:0T=x,clk=0,rst_n=x,data_out=x3T=x,clk=0,rst_n=0,data_out=05T=x,clk=1,rst_n=0,data_out=08T=1,clk=1,rst_n=1,data_out=110T=1,clk=0,rst_n=1,data_out=1,从图中可以清晰地看出Testbench的主要功能:(1)为DUT提供激励信号。(2)正确实例化DUT。(3)将仿真数据显示在终端或者存为文件,也可以显示在波形窗口中以供分析检查。(4)复杂设计可以使用EDA工具,或者通过用户接口自动比较仿真结果与理想值,实现结果的自动检查。,在编写Testbench时需要注意的问题:(1)testbench代码不需要可综合Testbench代码只是硬件行为描述不是硬件设计。(2)行为级描述效率高Verilog HDL语言具备5个描述层次,分别为开关级、门级、RTL级、算法级和系统级。(3)掌握结构化、程式化的描述方式 结构化的描述有利于设计维护,可通过initial、always以及assign语句将不同的测试激励划分开来。一般不要将所有的测试都放在一个语句块中。,5.2.2测试平台举例,DUT的仿真平台,测试平台需要产生时钟信号、复位信号和一系列的仿真向量,观察DUT的响应,确认仿真结果。,(1)组合逻辑电路仿真环境的搭建,module adder1(a,b,ci,so,co);input a,b,ci;output so,co;assignco,so=a+b+ci;endmodule根据全加器的真值表(表5.2-1)编写的全加器测试程序如下:module adder1_tb;wire so,co;reg a,b,ci;adder1 U1(a,b,ci,so,co);/模块例化 initial/测试信号产生 begin a=0;b=0;ci=0;#20 a=0;b=0;ci=1;#20 a=0;b=1;ci=0;#20 a=0;b=1;ci=1;#20 a=1;b=0;ci=0;#20 a=1;b=0;ci=1;#20 a=1;b=1;ci=0;#20 a=1;b=1;ci=1;#200$finish;end endmodule,全加器的输入a、b和ci定义为reg型变量;把输出so和co定义为wire型变量 用模块例化语句“adder1 U1(a,b,ci,so,co);”把全加器设计电路例化到测试仿真环境中;用initial块语句改变输入的变化并生成测试条件,输入的变化语句完全根据全加器的真值表编写 仿真结果:,(2)时序逻辑电路仿真环境的搭建,在于时序逻辑电路仿真环境中,需要考虑时序、定时信息和全局复位、置位等信号要求,并定义这些信号。,用Verilog HDL编写的十进制加法计数器源程序代码是:module cnt10(clk,rst,ena,q,cout);input clk,rst,ena;output 3:0 q;output cout;reg 3:0 q;always(posedge clk or posedge rst)begin if(rst)q=4b0000;else if(ena)begin if(q9)q=q+1;else q=0;end end assign cout=q3 endmodule,Verilog HDL测试程序代码是:module cnt10_tb;reg clk,rst,ena;wire 3:0 q;wire cout;cnt10 U1(clk,rst,ena,q,cout);/模块实例化 always#50 clk=clk;/时钟信号产生 initial begin clk=0;rst=0;ena=1;/控制信号产生#1200 rst=1;#120 rst=0;#2000 ena=0;#200 ena=1;#20000$finish;end endmodule,实例化语句“cnt10 U1(clk,rst,ena,q,cout);”把十进制计数模块例化到仿真环境中;在always中用语句“#50 clk=clk;”产生周期为100(标准时间单位)的时钟方波;用initial块生成复位信号rst和使能控制信号ena的测试条件。测试结果如图:,5.2.3 Verilog HDL仿真结果确认,(1)直接观察波形通过直接观察各信号波形的输出,比较测试值和期望值的大小,来确定仿真结果的正确性。,(2)打印文本输出法,module adder1_tb;wire so,co;reg a,b,ci;adder1 U1(a,b,ci,so,co);/模块例化 initial/测试信号产生 begin a=0;b=0;ci=0;#20 a=0;b=0;ci=1;#20 a=0;b=1;ci=0;#20 a=0;b=1;ci=1;#20 a=1;b=0;ci=0;#20 a=1;b=0;ci=1;#20 a=1;b=1;ci=0;#20 a=1;b=1;ci=1;#200$finish;end$monitor($time,%b%b%b-%b%b,a,b,ci,so,co);endmodule,其输出的结果是:0 0 0 0-0 0 20 0 0 1-1 0 40 0 1 0-1 0 60 0 1 1-0 1 80 1 0 0-1 0,系统任务打印任务:$display,直接输出到标准输出设备;$monitor,监控参数的变化;$fdisplay,输出到文件等,(3)自动检查仿真结果自动检查仿真结果是通过在设计代码中的关键节点添加断言监控器,形成对电路逻辑综合的注释或是对设计特点的说明,以提高设计模块的观察性。(4)使用VCD文件Verilog HDL提供一系列系统任务用于记录信号值变化保存到标准的VCD(Value Change Dump)格式数据库中。VCD文件是一种标准格式的波形记录文件,只记录发生变化的波形。VCD文件将在第5.3.7小节中详细讲述。,5.2.4 Verilog HDL仿真效率,因为要通过串行软件代码完成并行语义的转化,Verilog HDL行为级仿真代码的执行时间比较长。提高Verilog HDL代码的仿真代码执行时间:(1)减小层次结构 仿真代码的层次越少,执行时间就越短。(2)减少门级代码的使用由于门级建模属于结构级建模,建议仿真代码尽量使用行为级语句,建模层次越抽象,执行时间就越短。,(3)仿真精度越高,效率越低计时单位值与计时精度值的差距越大,则模拟时间越长。timescale仿真时间标度将在第5.9.3小节中详细讲述。(4)进程越少,效率越高代码中的语句块越少仿真越快,这是因为仿真器在不同进程之间进行切换也需要时间。(5)减少仿真器的输出显示Verilog HDL语言包含一些系统任务,可以在仿真器的控制台显示窗口输出一些提示信息,但会降低仿真器的执行效率。,5.3与仿真相关的系统任务,5.3.1$display和$write语法格式如下:$display(“”,);$write(“”,);“”通常称为“格式控制”“”则为“信号输出列表”$display自动地在输出后进行换行$write输出特定信息时不自动换行,输出格式说明,由“%”和格式字符组成,其作用是将输出的数据转换成指定的格式输出。常用的几种输出格式如右表。,一些特殊的字符可以通过表中的转换序列来输出。,例5.3-1:$display和$write语句,module disp_tb;reg31:0 rval;pulldown(pd);initialbegin rval=101;$display(t%n123);$display(rval=%h hex%d decimal,rval,rval);$display(rval=%o otal%b binary,rval,rval);$display(rval has%c ascii character value,rval);$display(pd strength value is%v,pd);$display(current scope is%m);$display(%s is ascii value for 101,101);$write(“simulation time is”);$write(“%tn”,$time);endendmodule,其输出结果为:%Srval=00000065 hex 101 decimalrval=00000000145 octal 00000000000000000000000001100101 binaryrval has e ascii character valuepd strength value is StXcurrent scope is dispe is ascii value for 101simulation time is 0,在$display中,输出列表中数据的显示宽度是自动按照输出格式进行调整的,总是用表达式的最大可能值所占的位数来显示表达式的当前值。,5.3.2$monitor和$strobe,$monitor与$stobe都提供了监控和输出参数列表中字符或变量的值的功能(1)$monitor语法格式:$monitor(”,);任务$monitor提供了监控和输出参数列表中的表达式或变量值的功能。每当参数列表中变量或表达式的值发生变化时,整个参数列表中变量或表达式的值都将输出显示。例如:$monitor($time,rxd=%b txd=%b,rxd,txd);注意在上面的语句中,“,代表一个空参数。空参数在输出时显示为空格。,$monitoron和$monitoroff任务的作用是通过打开和关闭监控标志来控制监控任务$monitor的启动和停止,这样使得程序员可以很容易的控制$monitor何时发生。$monitor与$display的不同处在于$monitor往往在initial块中调用,只要不调用$monitoroff,$monitor便不间断地对所设定的信号进行监视。,例5.3-2:$monitor系统任务的应用实例module monitor_tb;integer a,b;initial begin a=2;b=4;forever begin#5 a=a+b;#5 b=a-1;end end initial#40$finish;initial$monitor($time,a=%d,b=%d,a,b);endmodule,输出结果为:0 a=2,b=45 a=6,b=410 a=6,b=515 a=11,b=520 a=11,b=1025 a=21,b=1030 a=21,b=2035 a=41,b=20,(2)$strobe语法格式:$strobe();$strobe(“”,);探测任务用于在某时刻所有时间处理完后,在这个时间步的结尾输出一行格式化的文本。常用的系统任务如下:$strobe:在所有时间处理完后,以十进制格式输出一行格式化的文本;$strobeb:在所有时间处理完后,以二进制格式输出一行格式化的文本;$strobeo:在所有时间处理完后,以八进制格式输出一行格式化的文本;$strobeh:在所有时间处理完后,以十六进制格式输出一行格式化的文本。,$strobe任务在被调用的时刻所有的赋值语句都完成了,才输出相应的文字信息。$strobe任务提供了另一种数据显示机制,可以保证数据只在所有赋值语句被执行完毕后才被显示。,例5.3-3:$strobe系统任务的应用实例module strobe_tb;reg a,b;initial begin a=0;$display(“a by display is:”,a);$strobe(“a by strobe is:”,a);a=1;endinitial begin b=0;$display(“b by display is:”,b);$strobe(“b by strobe is:”,b);,#5;$display(“#5 b by display is:”,b);$display(“#5 b by strobe is:”,b);b=1;endEndmodule显示结果是:a by display is:0 b by display is:x a by strobe is:1 b by strobe is:0#5 b by display is:0#5 b by strobe is:0,5.3.3$time和$realtime,用这两个时间系统函数可以得到当前的仿真时刻,所不同的是,$time函数以64位整数值的形式返回仿真时间,而$realtime函数则以实数型数据返回仿真时间。(1)系统函数$time,例5.3-4:$time系统任务的应用实例timescale 1ns/1nsmodule time_tb;reg ts;parameter delay=2;initial begin#delay ts=1;#delay ts=0;#delay ts=1;#delay ts=0;endinitial$monitor($time,ts=%b,ts);/使用函数$time endmodule,输出结果为:0 ts=x3 ts=15 ts=08 ts=110 ts=0,(2)$realtime系统函数,$realtime返回的时间数字是一个实型数,该数字也是以时间尺度为基准的。,例5.3-5:$realtime系统任务的应用实例timescale1ns/1nsmodule realtime_tb;reg set;parameter p=2;initial begin$monitor($realtime,set=b%,set);/使用函数$realtime#p set=0;#p set=1;endendmodule输出结果为:0 set=x2 set=04 set=1,5.3.4$finish和$stop,系统任务$finish和$stop是用于对仿真过程进行控制,分别表示结束仿真和中断仿真。其语法格式:$finish;$finish(n);$stop;$stop(n);其中,n是$finish和$stop的参数,n可以取0、1或2几个值,分别表示如下含义,如下表所示。,$finish的作用是退出仿真器,返回主操作系统,也就是结束仿真过程。任务$finish可以带参数,根据参数的值输出不同的特征信息。如果不带参数,默认$finish的参数值为1。$stop任务的作用是把EDA工具(例如仿真器)置成暂停模式,在仿真环境下给出一个交互式的命令提示符,将控制权交给用户。这个任务可以带有参数表达式。根据参数值(0,1或2)的不同,输出不同的信息。参数值越大,输出的信息越多。,$finish和$stop实例,例5.3-6:$finish系统任务的应用实例module finish_tb;integer a,b;initial begin a=2;b=4;forever begin#5 a=a+b;#5 b=a-1;end end initial#40$finish;initial begin$monitor($time,a=%d,b=%d,a,b);endendmodule在上例中,程序执行到40个时间单位时退出仿真器。,例5.3-7:$stop系统任务的应用实例module stop_tb;integer a,b;initial begin a=2;b=4;forever begin#5 a=a+b;#5 b=a-1;end end initial#40$stop;initial begin$monitor($time,a=%d,b=%d,a,b);endendmodule在上例中,程序执行到40个时间单位时停止仿真,将EDA仿真器设置为暂停模式。,5.3.5$readmemh和$readmem,在Verilog HDL程序中有两个系统任务$readmemb和$readmemh用来从文件中读取数据到存储器中。这两个系统任务可以在仿真的任何时刻被执行使用,其语法格式共有以下六种:(1)$readmemb(,);(2)$readmemb(,);(3)$readmemb(,);(4)$readmemh(,);(5)$readmemh(,);(6)$readmemh(,);,5/5/2024,32,5/5/2024,33,例$readmemh和$readmemb系统任务的应用实例module read_mem_tb;reg 7:0 memory_b 0:7;reg 31:0 memory_h 0:31;integer i;initial begin/把数据文件init_b.txt读入存储器中的给定地址$readmemb(init_b.txt,memory_b);/把数据文件init_h.txt读入存储器中的给定地址$readmemb(init_h.txt,memory_h);/显示初始化后的存储器内容 for(i=0;i8;i=i+1)begin$display(memory_b%0d=%b,i,memory_bi);$display(memory_h%0h=%h,i,memory_hi);end end endmodule,文件init_b.txt和init_h.txt包含初始化数据。用在数据文件中指定地址。其中,“init_b.txt”指定二进制数据从第二位地址开始写入;而“init_h.txt”指定十六进制数据从地一位地址写入。样本文件如下所示。,5/5/2024,34,“init_b.txt”文件:00211111111 0101010100000000 101010100061111zzzz 00001111“init_h.txt”文件:00100000000000000000000000000000011000000000000000000000000000001110000000000000000000000000000111100000000000000000000000000011111,5.3.6$random,$random是产生随机数的系统函数,每次调用该函数将返回一个32位的随机数,该随机数是一个带符号的整数。语法格式:$random%;这个系统函数提供了一个产生随机数的手段。当函数被调用时返回一个32bit的随机数。它是一个带符号的整形数。$random一般的用法是:$ramdom%b,其中 b0,它给出了一个范围在(-b+1):(b-1)中的随机数。,5/5/2024,35,例$random系统任务的应用实例,5/5/2024,36,timescale 1ns/1nsmodule random_pulse(dout);output 9:0 dout;reg dout;integer delay1,delay2,k;initial begin#10 dout=0;for(k=0;k 100;k=k+1)begin delay1=20*($random%6);/delay1 在0到100ns间变化 delay2=20*(1+$random%3);/delay2 在20到60ns间变化#delay1 dout=1($random%10);/dout的0-9位中随机出现1,并出现的时间在0-100ns间变化#delay2 dout=0;/脉冲的宽度在在20到60ns间变化 end endendmodule,5.4信号时间赋值语句,5/5/2024,37,5.4.1时间延迟的语法说明,延迟语句用于对各条语句的执行时间进行控制,从而快速满足用户的时序要求。Verilog HDL语言中延时控制的语法格式有两类:(1)#行为语句;(2)#;其中,符号“#”是延迟控制的关键字符,可以是直接指定的延迟时间量,并以多少个仿真时间单位的形式给出。在仿真过程中,所有时延都根据时间单位定义。下面是带时延的赋值语句示例。#2 Sum=A B;/#2指定2个时间单位后,将A异或B的值赋值给Sum。,5/5/2024,38,根据时间控制部分在过程赋值语句中出现的位置,可以把过程赋值语句中的时间控制方式分为外部时间控制方式和内部时间控制方式。(1)外部时间控制方式是时间控制出现在整个过程赋值语句的最左端,也就是出现赋值目标变量的左边的时间控制方式,其语法结构如下例所示:#5 a=b;在仿真执行时就相当于如下几条语句的执行:initialbegin#5;a=b;end,5/5/2024,39,(2)内部时间控制方式是过程赋值语句中的时间控制部分还可以出现在“赋值操作符”和“赋值表达式”之间的时间控制方式。其语法结构如下例所示:a=#5b;其中时间控制部分“#5”就出现在赋值操作符“=”和赋值表达式“b”的中间,因此在这条过程赋值语句内带有内部时间控制方式的时间控制。它在执行时就相当于如下几条语句的执行:initial begin temp=b;/先求b的值#5;a=temp;end,5/5/2024,40,5.4.2时间延迟的描述形式,此处时间延迟的描述形式是指延时控制的描述形式,其分为串行延迟控制、并行延迟控制、阻塞式延迟控制和非阻塞式延迟控制四种形式。以实现两组不同波形的信号为例(如图所示q0_out和q1_out),说明四种不同时间延迟的描述形式。,5/5/2024,41,(1)串行延迟控制串行延迟控制是最为常见的信号延迟控制,它是由begin-end过程块加上延迟赋值语句构成,其中延迟赋值语句可以为外部时间控制方式,也可以为内部时间控制方式。在之后也可根据情况来确定是否执行相应的行为语句。在后面有相应的行为语句,则仿真进程遇到这条带有延迟控制的行为语句后并不立即执行行为语句指定的操作,而是要延迟等待到“”所指定的时间量过去后才真正开始执行行为语句指定的操作。,5/5/2024,42,5/5/2024,43,timescale 1ns/1nsmodule serial_delay(q0_out,q1_out);output q0_out,q1_out;reg q0_out,q1_out;initial begin q0_out=1b0;#50 q0_out=1b1;#100q0_out=1b0;#100q0_out=1b1;#50 q0_out=1b0;#100q0_out=1b1;#50 q0_out=1b0;#50 q0_out=1b1;#50 q0_out=1b0;end,initial begin q1_out=1b0;#100q1_out=1b1;#100q1_out=1b0;#50 q1_out=1b1;#100q1_out=1b0;#50 q1_out=1b1;#100q1_out=1b0;#50 q1_out=1b1;#50 q1_out=1b0;endendmodule,例Verilog HDL串行延迟控制方式设计图示信号,(2)并行延迟控制并行延迟控制方式是通过fork-join过程块加上延迟赋值语句构成,其中延迟赋值语句同串行延迟控制方式一样,既可以是外部时间控制方式,也可以是内部时间控制方式。在之后也可根据情况来确定是否执行相应的行为语句。在后面有相应的行为语句,则仿真进程遇到这条带有延迟控制的行为语句后并不立即执行行为语句指定的操作,而是要延迟等待到“”所指定的时间量过去后才真正开始执行行为语句指定的操作。但并行延迟控制方式与串行延迟控制方式不同在于并行延迟控制方式中的多条延迟语句时并行执行的,并不需要等待上一条语句的执行完成才开始执行当前的语句。,5/5/2024,44,5/5/2024,45,timescale 1ns/1nsmodule parallel_delay(q0_out,q1_out);output q0_out,q1_out;reg q0_out,q1_out;initial fork q0_out=1b0;#50 q0_out=1b1;#150q0_out=1b0;#250q0_out=1b1;#300q0_out=1b0;#400q0_out=1b1;#450q0_out=1b0;#500q0_out=1b1;#600q0_out=1b0;join,initial fork q1_out=1b0;#100q1_out=1b1;#200q1_out=1b0;#250q1_out=1b1;#350q1_out=1b0;#400q1_out=1b1;#500q1_out=1b0;#550q1_out=1b1;#600q1_out=1b0;joinendmodule,例Verilog HDL并行延迟控制方式设计图示信号,(3)阻塞式延迟控制以赋值操作符“=”来标识的赋值操作称为“阻塞式过程赋值”,阻塞式过程赋值在之前已经介绍过,在此介绍阻塞式延迟控制。阻塞式延迟控制是在阻塞式过程赋值基础上带有延时控制的情况,例如initialbegina=0;a=#5 1;a=#10 0;a=#15 1;end,5/5/2024,46,各条阻塞式赋值语句将依次得到执行,并且在第一条语句所指定的赋值操作没有完成之前第二条语句不会开始执行。因此在仿真进程开始时刻将“0”值赋给a,此条赋值语句完成之后才开始执行第二条赋值语句;在完成第一条赋值语句之后,延迟5个时间单位将“1”赋值给a;同理第三条赋值语句是在第二条赋值语句完成之后延迟10个时间单位才开始执行,将“0”赋值给a;最后一条赋值语句是在前三条语句都完成的时刻,延迟15个时间单位,将“1”赋值给a。下图给出了上例中信号a的波形。上述两例都采用的是阻塞式赋值语句。,5/5/2024,47,(4)非阻塞式延迟控制以赋值操作符“=”来标识的赋值操作称为“非阻塞式过程赋值”,非阻塞式过程赋值也在之前讲述过,在此主要介绍非阻塞式延迟控制。非阻塞式延迟控制是在非阻塞式过程赋值基础上带有延时控制的情况。如下例所示:initialbegina=0;a=#5 1;a=#10 0;a=#15 1;end,5/5/2024,48,在上例中各条非阻塞式赋值语句均以并行方式执行,虽然执行语句在begin-end串行块中,但其执行方式与并行延迟控制方式一致,在仿真进程开始时刻同时执行四条延迟赋值语句。在仿真进程开始时,将“0”值赋值给a;在离仿真开始时刻5个时间单位时,将“1”值赋值给a;在离仿真开始时刻10个时间单位时,将“0”值赋值给a;最后在离仿真开始时刻15个时间单位时,将“1”值赋值给a。下图给出了上例中信号a的波形。,5/5/2024,49,5/5/2024,50,timescale 1ns/1nsmodule non_blocking_delay(q0_out,q1_out);output q0_out,q1_out;reg q0_out,q1_out;initial begin q0_out=1b0;q0_out=#50 1b1;q0_out=#150 1b0;q0_out=#250 1b1;q0_out=#300 1b0;q0_out=#400 1b1;q0_out=#450 1b0;q0_out=#500 1b1;q0_out=#600 1b0;end,initial begin q1_out=1b0;q1_out=#100 1b1;q1_out=#200 1b0;q1_out=#250 1b1;q1_out=#350 1b0;q1_out=#400 1b1;q1_out=#500 1b0;q1_out=#550 1b1;q1_out=#600 1b0;endendmodule,例Verilog HDL非阻塞延迟控制方式设计,5.4.3边沿触发事件控制,边沿触发事件控制的语法格式可以为如下四种形式:形式1:()行为语句;形式2:();形式3:(ororor)行为语句;形式4:(ororor);,5/5/2024,51,1事件表达式在事件表达式中,可以以三种形式出现:形式1:形式2:posedge形式3:negedge其中,“”可以是任何数据类型的标量或矢量。形式1中,代表触发事件的“”在指定的信号发生逻辑变化时,执行下面的语句,如例所示:(in)out=in;当敏感事件in发生逻辑变化时(包括正跳变和负跳变),执行对应的赋值语句,将in的值赋值给out。,5/5/2024,52,形式2中,代表触发事件的“posedge”在指定的信号发生了正跳变时,执行下面的语句,如下例所示:(posedge in)out=in;当敏感事件in发生正跳变时,执行对应的赋值语句,将in的值赋值给out。形式3中,代表触发事件的“negedge”在指定的信号发生了负跳变时,执行下面的语句,如下例所示:(negedge in)out=in;当敏感事件in发生负跳变时,执行对应的赋值语句,将in的值赋值给out。,5/5/2024,53,在信号发生逻辑变化(正跳变或负跳变)的过程中,信号的值是从0、1、x、z四个值中的一个值变化到另一个值;而信号发生正跳变的过程是信号由低电平向高电平的转变,负跳变是信号由高电平向低电平的转变。表5.4-1为Verilog HDL中规定的正跳变和负跳变。,5/5/2024,54,2边沿触发语法格式形式1:()行为语句;这种语法格式的敏感事件列表内只包含了一个触发事件,只有当这个指定的触发事件发生之后,后面的行为语句才能启动执行。在仿真进程中遇到这种带有事件控制的行为语句时,如果指定的触发事件还没有发生,则仿真进程就会停留在此处等待,直到指定触发事件发生之后再启动执行后面的行为语句,仿真进程继续向下进行。,5/5/2024,55,例5.4-4:时钟脉冲计数器module clk_counter(clk,count_out);input clk;output count_out;reg 3:0 count_out;initial count_out=0;,always(posedge clk)count_out=count_out+1;/在clk的每个正跳变边沿count_out增加1endmodule,形式2:();这种语法格式的敏感事件列表内也只包含了一个触发事件,没有行为语句来指定触发事件发生时要执行的操作。这种格式的事件控制语句的执行过程与延时控制语句中没有行为语句的情况类似,仿真进程在遇到这条事件控制语句后会进入等待状态,直到指定的触发事件发生后才结束等待状态,退出该事件控制语句的执行并开始下一条语句的执行。,5/5/2024,56,5/5/2024,57,module clk_time_mea(clk);input clk;time posedge_time,negedge_time;time high_last_time,low_last_time,last_time;initial begin(posedge clk);/*等待,直到时钟发生正跳变后退出等待状态,继续执行下一条语句*/posedge_time=$time;(negedge clk);/*等待,直到时钟发生负跳变后退出等待状态,继续执行下一条语句*/negedge_time=$time;,(negedge clk);/*等待,直到时钟再次正跳变后退出等待状态,继续执行下一条语句*/last_time=$time-posedge_time;high_last_time=negedge_time-posedge_time;low_last_time=last_time-high_last_time;$display(The clk stay in High level for:%t,high_last_time);$display(The clk stay in Low level for:%t,low_last_time);$display(The clk signal Period is:%t,last_time);endendmodule,例用于测定输入时钟正电平、负电平持续时间以及时钟周期的模块,形式3:(ororor)行为语句;这种语法格式的“敏感事件列表”内指定了由不同“”代表的多个触发事件,这些“”之间要用关键词“or”组合起来。只要这些触发事件中的任何一个得到发生,就启动行为语句的执行。在仿真进程遇到这种格式的边沿触发事件控制语句时如果所有的触发事件都没有发生,则仿真进程就会进入等待状态,直到其中的某一个触发事件发生后才启动执行后面给出的行为语句,仿真进程继续向下进行。,5/5/2024,58,形式4:(ororor);同第三种语法格式一样,这种语法格式内指定了多个触发事件。但是在这种格式中没有行为语句。在这种情况下,该语句的执行过程与第二种语法格式的执行过程类似,仿真进程在遇到这条事件控制语句后会进入等待状态,直到敏感事件列表包含的多个触发事件中的任何一个得到发生后才结束等待状态,退出该事件控制语句并开始执行该事件控制语句后的下一条语句。,5/5/2024,59,例在触发事件发生后退出事件控制语句module display_information_change(a,b);input a,b;wire a,b;always begin,(posedge a or negedge b);/*等待,直到a或b发生变化后退出等待状态,并开始下一条语句的执行*/display(One of a and b changed in time:%t,$time);endendmodule,5.4.4电平敏感事件控制,电平敏感时间控制是另一种事件控制方式,与边沿触发事件控制不同,它是在指定的条件表达式为真时启动需要执行的语句。电平敏感时间控制是用关键词“wait”来表示。电平触发事件控制的语法格式可以为如下两种:形式1:wait(条件表达式)行为语句;形式2:wait(条件表达式);电平敏感事件控制的第一种形式中包含了行为语句,它可以是串行块(begin-end)语句或并行块(