0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Verilog设计增加延时的仿真技术

电子设计 来源:csdn 作者:bleauchat 2020-11-25 15:29 次阅读

作者:bleauchat

在设计仿真激励文件时,为了满足和外部芯片接口的时序要求,经常会用到延时赋值语句,由于不同的延时赋值语句在仿真过程中行为不同,会产生不同的激励输出,如果不认真区分不同表达式引起的差异,就可能产生错误的激励,无法保证仿真结果的正确,本文就是区分各种延时赋值语句的差异,并给出比较结果。

1:阻塞式左延时赋值语句

举例说明如下:

module adder_t1 (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
reg co;
reg [3:0] sum;
always @(a or b or ci)
#12 {co, sum} = a + b + ci; // 在15ns时a发生变化,启动该块,但是等到27ns时才执行后面的语句,所以是最新的结果
endmodule

分析:上面例子是希望在输入信号变化后12ns再更新输出结果,假设在15ns时a发生变化,在27ns时,结果将被更新,但是如果在15ns到24ns这一段时间,a,b,ci又发生了变化,在27ns时,结果将按照最新的a,b,ci进行计算并被更新;

如果将程序修改成如下格式,仿真的结果不变:

module adder_t7a (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
reg co;
reg [3:0] sum;
reg [4:0] tmp;
always @(a or b or ci)
begin
#12 tmp = a + b + ci; // 在15ns a发生变化时,启动该always块,但是12ns后即第27ns时才执行 
 //tmp = a + b + ci tmp才被赋值,因此赋值的是最近的a,b,ci变化的值
{co, sum} = tmp;
end
endmodule

如果将程序做如下修改:

module adder_t7b (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
reg co;
reg [3:0] sum;
reg [4:0] tmp;
always @(a or b or ci)
begin
tmp = a + b + ci;  // 在15ns a发生变化时,启动该always块,执行该语句,获得当前tem ,然后就开始执行下一句 #12
#12 {co, sum} = tmp; // 所以tmp的值是15ns的值,再过12ns即27ns时赋值给输出。因此,中间的变换被忽视
end
//15~27之间执行该语句块,只有执行完该语句块之后才是开始执行下一个语句块
endmodule

仿真的结果如下图所示:从15ns到27ns之间的变化被忽视;

结论:阻塞式赋值语句是一句一句执行的,一句没有执行完,下一句绝不会执行。正因为如此,在此例中在延时12个ns的时间里,不作任何处理,tmp值保持不变(2’b10),而且对敏感变量的变化不作反应。不要将延时放在阻塞式赋值语句的左侧,这是一种不好的代码设计方式;

2:阻塞式右延时赋值语句

看下面的例子:

module adder_t6 (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
reg co;
reg [3:0] sum;
always @(a or b or ci)
{co, sum} = #12 a + b + ci; // 先计算a + b + ci的值,过12ns后赋值为输出;赋值语句是从右往左执行
endmodule

它的仿真结果同adder_t7b,赋值语句是从右往左执行,信号从15ns到27ns之间的变化被忽视;即 同:

tmp=a+b+ci;
 
#12{co,sum}=tmp;

下面两个例子的仿真结果和adder_t6相同:

module adder_t11a (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
reg co;
reg [3:0] sum;
reg [4:0] tmp;
always @(a or b or ci)
begin
tmp = #12 a + b + ci;
{co, sum} = tmp;
end
endmodule
module adder_t11b (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
reg co;
reg [3:0] sum;
reg [4:0] tmp;
always @(a or b or ci)
begin
tmp = a + b + ci;
{co, sum} = #12 tmp;
end
endmodule

结论:不要将延时放在阻塞式赋值语句的右侧,这是一种不好的代码设计方式;

3:非阻塞式左延时赋值语句

看例子:

module adder_t2 (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
reg co;
reg [3:0] sum;
always @(a or b or ci)
#12 {co, sum} <= a + b + ci;
endmodule

它的仿真结果同adder_t1,在27ns时,结果将按照最新的a,b,ci进行计算并被更新;由于首先要执行#12,然后才能执行非阻塞赋值,所以能对信号进行实时跟踪!

结论:不要将延时放在非阻塞式赋值语句的左侧,这是一种不好的代码设计方式;

4:非阻塞式右延时赋值语句

看例子:

module adder_t3 (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
reg co;
reg [3:0] sum;
always @(a or b or ci)
{co, sum} <= #12 a + b + ci;
endmodule

非阻塞赋值,不会阻止下一条语句的执行,输出结果能随时跟踪输入信号的变化;

结论:使用非阻塞式右延时赋值语句可以,输出结果能够跟随输入的变化,建议使用;

5:非阻塞式右延时多重赋值语句

看例子:

module adder_t9c (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
reg co;
reg [3:0] sum;
reg [4:0] tmp;
always @(a or b or ci or tmp)
begin
tmp <= #12 a + b + ci;
{co, sum} <= tmp;
end
endmodule

非阻塞赋值,不会阻止下一条语句的执行,输出结果能随时跟踪输入信号的变化;这与下面的例子结果是相同的:

module adder_t9d (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
reg co;
reg [3:0] sum;
reg [4:0] tmp;
always @(a or b or ci or tmp)
begin
tmp <= a + b + ci;
{co, sum} <= #12 tmp;
end
endmodule

6:连续赋值语句

看例子:

module adder_t4 (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
assign #12 {co, sum} = a + b + ci;
endmodule
module adder_t5 (co, sum, a, b, ci);
output co;
output [3:0] sum;
input [3:0] a, b;
input ci;
assign {co, sum} = #12 a + b + ci;
endmodule

输出结果能随时跟踪输入信号的变化;

结论:非阻塞语句和连续赋值语句无论怎么添加延时语句,其输出都会随着输入的变化而跟踪变化!这里要特别注意阻塞语句的延时添加,一般不在阻塞赋值语句中添加延时!

编辑:hfy


声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • Verilog
    +关注

    关注

    28

    文章

    1340

    浏览量

    109905
收藏 人收藏

    评论

    相关推荐

    基于MATLAB/Simulink的系统仿真技术与应用

    基于MATLAB_Simulink的系统仿真技术与应用. 本书首先介绍了MATLAB语言的程序设计的基本内容,在此基础上系统介绍了系统仿真所必要的数值计算方法及MATLAB实现
    发表于 06-19 13:15

    实时控制与仿真技术应用

    基于LabVIEW的实时控制与仿真技术应用研究
    发表于 04-21 19:08

    Cadence PSpice仿真技术的优势有哪些

    Designer提供PCB设计人员的仿真技术是把电路仿真环境与PCB布局布线设计环境完全集成在一起,构成一个完整的统一集成环境。设计师通过集成模拟和事件驱动数字仿真,可以在不牺牲精度的情况下提高
    发表于 07-07 09:47

    ADAS研发与测试的模拟仿真技术

    ADAS高级辅助驾驶的ADAS研发与测试的模拟仿真技术,大家可以看看,文档介绍的特别丰富
    发表于 09-02 18:06

    仿真技术在汽车电子设计中的应用有哪些?

    什么是SABER软件仿真技术?汽车电子仿真技术的应用是什么?
    发表于 05-18 06:42

    系统仿真技术与应用

    基于MATLAB_Simulink的系统仿真技术与应用.  本书首先介绍了MATLAB语言的程序设计的基本内容,在此基础上系统介绍了系统仿真所必要的
    发表于 06-19 13:40 37次下载

    基于MATLAB Simulink的系统仿真技术与应用 教材

    基于MATLAB Simulink的系统仿真技术与应用 教材   目录第一章 系统仿真技术与应用
    发表于 04-29 14:51 0次下载
    基于MATLAB Simulink的系统<b class='flag-5'>仿真技术</b>与应用 教材

    Cadence PCB设计仿真技术

    Cadence PCB设计仿真技术 Cadence PCB设计仿真技术提供了一个全功能的模拟仿真器,并支持数字元件帮助解决几乎所有的设计挑战,从高频系统到低功耗I
    发表于 04-29 08:41 4489次阅读
    Cadence PCB设计<b class='flag-5'>仿真技术</b>

    CST仿真技术交流

    CST仿真技术交流。
    发表于 03-11 09:46 17次下载

    电路设计与仿真技术

    电路设计与仿真技术电子类资料材料,有兴趣的同学可以下载学习
    发表于 04-29 18:24 33次下载

    模拟仿真技术是什么

    则是在人类认识自然界客观规律的历程中一直被有效地使用着。由于计算机技术的发展,仿真技术逐步自成体系,成为继数学推理、科学实验之后人类认识自然界客观规律的第三类基本方法,而且正在发展成为人类认识、改造
    发表于 05-19 17:13 1.6w次阅读

    EE-68: JTAG 仿真技术参考

    EE-68: JTAG 仿真技术参考
    发表于 03-21 02:06 2次下载
    EE-68: JTAG <b class='flag-5'>仿真技术</b>参考

    电源仿真技术

    电源仿真技术资料分享。
    发表于 04-19 17:11 10次下载

    DDR设计和仿真技术详解

    DDR2设计和仿真技术详解。
    发表于 10-24 15:10 2次下载

    pcb仿真技术有什么用

    pcb仿真技术有什么用
    的头像 发表于 11-28 15:22 1045次阅读