在FPGA中使用for循环一定浪费资源吗?
渐渐地,发现自己已经习惯于发现细节,喜欢打破常规,真的非常喜欢这种feel。
相信很多人在书上或者博文上都有提出“在FPGA中使用for语句是很占用资源的”的观点,特权同学也不例外。那么,这种观点正确吗?我的答案是:对一半,错一半。在某些情况下,使用for循环也许真的挺占用资源的。但我并不想去探讨这种情况。而是谈谈在另外一些情况下使用for语句的好处。
第一个好处:有时使用for循环不但不会浪费多余的资源,而且可以减少代码量,从而提高编码效率;第二个好处是:方便模块的移植。下面举个移位寄存器的简单例子来说明就一目了然了。假设设计一个深度为16、位宽为8的移位寄存器。
1、基于非for语句的移位寄存器电路设计如下:
module shift_register
#(
parameter DATA_WIDTH = 8
)
(
input clk,
input [DATA_WIDTH-1:0] din,
output [DATA_WIDTH-1:0] dout
);
//---------------------------------------------
reg [DATA_WIDTH-1:0] mem [0:15];
always @(posedge clk)
begin
mem[0 ] <= din;
mem[1 ] <= mem[0 ];
mem[2 ] <= mem[1 ];
mem[3 ] <= mem[2 ];
mem[4 ] <= mem[3 ];
mem[5 ] <= mem[4 ];
mem[6 ] <= mem[5 ];
mem[7 ] <= mem[6 ];
mem[8 ] <= mem[7 ];
mem[9 ] <= mem[8 ];
mem[10] <= mem[9 ];
mem[11] <= mem[10];
mem[12] <= mem[11];
mem[13] <= mem[12];
mem[14] <= mem[13];
mem[15] <= mem[14];
end assign dout = mem[15]; endmodule
综合后资源消耗为:

2、基于for语句的移位寄存器电路设计如下:
module shift_register_for
#(
parameter DATA_WIDTH = 8,
parameter SHIFT_LEVEL = 16
)
(
input clk,
input [DATA_WIDTH-1:0] din,
output [DATA_WIDTH-1:0] dout
);
//-------------------------------------------------------
reg [DATA_WIDTH-1:0] mem [0:SHIFT_LEVEL-1];
always @(posedge clk)
begin : shift_reg
integer i;
for(i = 0; i < SHIFT_LEVEL-1; i = i + 1)
mem[i+1] <= mem[i];
mem[0] <= din;
end assign dout = mem[SHIFT_LEVEL-1]; endmodule
综合后资源消耗为:

通过对比,两者消耗资源一样,但使用for语句可大大减少代码量。在这里,我问大家一个问题,如果我想要一个深度为40、位宽为8的移位寄存器,怎么办?对基于for语句的移位寄存器很容易实现,只要将SHIFT_LEVEL=16改为SHIFT_LEVEL=40就可以了;而对于没有使用for语句的移位寄存器并没有快捷的修改方法,只能乖乖地按部就班了,如下所示:
module shift_register
#(
parameter DATA_WIDTH = 8
)
(
input clk,
input [DATA_WIDTH-1:0] din,
output [DATA_WIDTH-1:0] dout
);
//--------------------------
reg [DATA_WIDTH-1:0] mem [0:15];
always @(posedge clk)
begin
mem[0 ] <= din;
mem[1 ] <= mem[0 ];
mem[2 ] <= mem[1 ];
mem[3 ] <= mem[2 ];
mem[4 ] <= mem[3 ];
mem[5 ] <= mem[4 ];
mem[6 ] <= mem[5 ];
mem[7 ] <= mem[6 ];
mem[8 ] <= mem[7 ];
mem[9 ] <= mem[8 ];
mem[10] <= mem[9 ];
mem[11] <= mem[10];
mem[12] <= mem[11];
mem[13] <= mem[12];
mem[14] <= mem[13];
mem[15] <= mem[14];
mem[16] <= mem[15];
mem[17] <= mem[16];
mem[18] <= mem[17];
mem[19] <= mem[18];
mem[20] <= mem[19];
mem[21] <= mem[20];
mem[22] <= mem[21];
mem[23] <= mem[22];
mem[24] <= mem[23];
mem[25] <= mem[24];
mem[26] <= mem[25];
mem[27] <= mem[26];
mem[28] <= mem[27];
mem[29] <= mem[28];
mem[30] <= mem[39];
mem[31] <= mem[30];
mem[32] <= mem[31];
mem[33] <= mem[32];
mem[34] <= mem[33];
mem[35] <= mem[34];
mem[36] <= mem[35];
mem[37] <= mem[36];
mem[38] <= mem[37];
mem[39] <= mem[38];
end assign dout = mem[39]; endmodule
由此,使用for语句的移位寄存器很方便移植。
总结:在一些情况下,适当使用for语句不但可以节省设计的时间,还有利于设计的移植。
转载自:http://blog.chinaaet.com/crazybird/p/5100000274
在FPGA中使用for循环一定浪费资源吗?的更多相关文章
- 【转载】FPGA 中的latch 锁存器
以下这篇文章讲述了锁存器的一些概念和注意事项.原文标题及链接: FPGA 中的latch 锁存器 - 快乐至永远上的博客 - 与非博客 - 与网 http://www.eefocus.com/liuy ...
- [转]如何在 JS 代码中消灭 for 循环
一,用好 filter,map,和其它 ES6 新增的高阶遍历函数 二,理解和熟练使用 reduce 三,用递归代替循环(可以break!) 四,使用高阶函数遍历数组时可能遇到的陷阱 五,死磕到底,T ...
- JavaScript中的事件循环
JavaScript是单线程单并发语言 单线程:主程序只有一个线程,即同一时间片段内其只能执行单个任务. 引发的问题: 单线程,意味着任务都需要排队,前一个任务结束,才会执行后一个任务.若前一个任务耗 ...
- Oracle中三种循环(For、While、Loop)
1.ORACLE中的GOTO用法 DECLARE x number; BEGIN x := 9; <<repeat_loop>> --循环点 x := x - 1; DBMS_ ...
- FPGA中的delay与latency
delay和latency都有延迟的意义,在FPGA中二者又有具体的区别. latency出现在时序逻辑电路中,表示数据从输入到输出有效经过的时间,通常以时钟周期为单位. delay出现在组合逻辑电路 ...
- FPGA中的INOUT接口和高阻态
除了输入输出端口,FPGA中还有另一种端口叫做inout端口.如果需要进行全双工通信,是需要两条信道的,也就是说需要使用两个FPGA管脚和外部器件连接.但是,有时候半双工通信就能满足我们的要求,理论上 ...
- FPGA中的时序分析(四)
常用约束语句说明 关于Fmax 上述是实现Fmax的计算公式,clock skew delay的计算如下图, 就是两个时钟的差值.到头来,影响Fmax的值的大小就是组合逻辑,而Fmax是针对 ...
- TMsgThread, TCommThread -- 在delphi线程中实现消息循环
http://delphi.cjcsoft.net//viewthread.php?tid=635 在delphi线程中实现消息循环 在delphi线程中实现消息循环 Delphi的TThread类使 ...
- 深入了解JavaScript中的for循环
在ECMAScript5中,有三种for循环,分别是: 简单for循环 for-in forEach 在ES6中,新增了一种循环 for-of 简单for循环 const arr = [1, 2, 3 ...
随机推荐
- 在自己的服务器上部署 GitLab 社区版
GitLab 简介 因为我的个人网站 restran.net 已经启用,博客园的内容已经不再更新.这篇文章是在 Gitlab 7.4 的环境下配置的,相关内容可能已经过时. 后续做了一次迁移,将 Gi ...
- 算法笔记_001:斐波那契数的多种解法(Java)
本篇文章解决的问题来源于算法设计与分析课程的课堂作业,主要是运用多种方法来计算斐波那契数.具体问题及解法如下: 一.问题1: 问题描述:利用迭代算法寻找不超过编程环境能够支持的最大整数的斐波那契数是第 ...
- sed备忘
sed 是一种在线编辑器,它一次处理一行内容.处理时,把当前处理的行存储在临时缓冲区中,称为“模式空间”(pattern space),接着用sed命令处理缓冲区中的内容,处理完成后,把缓冲区的内容送 ...
- Linux密钥认证错误解决
问题描述: Xshell用key认证登录,提示所选的用户密钥未在远程主机上注册 问题解决: 查看日志/var/log/secure,基本上都是用户根目录的权限问题 根据日志提示: Authentica ...
- cygwin完全安装步骤方法(组图)
我们可以到Cygwin的官方网站下载Cygwin的安装程序,地址是: http://www.cygwin.com/ 或者直接使用下载连接来下载安装程序,下载连接是: http://www.cygwin ...
- 每一个软件开发人员绝对必须掌握的关于 Unicode 和字符集的最基础的知识
2013-02-05 14:18 48人阅读 评论(0) 收藏 举报 关键字: Unicode, Character Set, 字符集, UTF-8, ANSI, ASCII, UTF-7 ...
- IT常用设备及服务器安全公告或漏洞地址pa安全预警微软安全公告 HP 安全公告AIX 安全公告REDHAT安全公告ORACLE 安全公告F5 安全公告 Cisco 安全公告Juniper 安全公告 VMware 安全公告IOS 安全公告
IT常用设备及服务器安全公告或漏洞地址: 微软安全公告 https://technet.microsoft.com/en-us/library/security/MS14-085 HP 安全公告 ht ...
- 【java设计模式】之 责任链(chain of resposibility)模式
责任链模式,顾名思义,就是一条链.这个链到底是怎么运行的呢?它主要是将能够处理同一类请求的对象连成一条链,所提交的请求沿着链传递,链上的对象逐个判断是否有能力处理该请求,如果能则处理,如果不能则传递给 ...
- pthread到Win32thread
一.什么是线程. 线程(thread)是为了提高系统内程序的并发(concurrency)执行程度而提出来的概念,它是比进程更小的能够独立运行的基本单位.在引入线程的系统中,线程是处理器调度(sche ...
- Google的创新九原则(转)
原文url:http://www.365xiaoxi.com/All/News/2013-11-22/6432.html 想知道是什么让Google成为生产力与创造力的圣杯?当然不是喝山景城脚下的神水 ...