异步FIFO的FPGA实现
本文大部分内容来自Clifford E. Cummings的《Simulation and Synthesis Techniques for Asynchronous FIFO Design》,同时加上一些自己的一些理解,有兴趣的朋友可以阅读原文。
一、FIFO简介
FIFO是英文First In First Out 的缩写,是一种先进先出的数据缓存器,它与普通存储器的区别是没有外部读写地址线,这样使用起来非常简单,但缺点就是只能顺序写入数据,顺序的读出数据,其数据地址由内部读写指针自动加1完成,不能像普通存储器那样可以由地址线决定读取或写入某个指定的地址。
用途1:
异步FIFO读写分别采用相互异步的不同时钟。在现代集成电路芯片中,随着设计规模的不断扩大,一个系统中往往含有数个时钟,多时钟域带来的一个问题就是,如何设计异步时钟之间的接口电路。异步FIFO是这个问题的一种简便、快捷的解决方案,使用异步FIFO可以在两个不同时钟系统之间快速而方便地传输实时数据。
用途2:
对于不同宽度的数据接口也可以用FIFO,例如单片机位8位数据输出,而DSP可能是16位数据输入,在单片机与DSP连接时就可以使用FIFO来达到数据匹配的目的。
二、分类
同步FIFO是指读时钟和写时钟为同一个时钟,在时钟沿来临时同时发生读写操作;
异步FIFO是指读写时钟不一致,读写时钟是互相独立的。
三、FIFO的常见参数
- FIFO的宽度:即FIFO一次读写操作的数据位;
- FIFO的深度:指的是FIFO可以存储多少个N位的数据(如果宽度为N)。
- 满标志:FIFO已满或将要满时由FIFO的状态电路送出的一个信号,以阻止FIFO的写操作继续向FIFO中写数据而造成溢出(overflow)。
- 空标志:FIFO已空或将要空时由FIFO的状态电路送出的一个信号,以阻止FIFO的读操作继续从FIFO中读出数据而造成无效数据的读出(underflow)。
- 读时钟:读操作所遵循的时钟,在每个时钟沿来临时读数据。
- 写时钟:写操作所遵循的时钟,在每个时钟沿来临时写数据。

- 读写指针的工作原理
读指针:总是指向下一个将要被写入的单元,复位时,指向第1个单元(编号为0)。
写指针:总是指向当前要被读出的数据,复位时,指向第1个单元(编号为0)
- FIFO的“空”/“满”检测
FIFO设计的关键:产生可靠的FIFO读写指针和生成FIFO“空”/“满”状态标志。
当读写指针相等时,表明FIFO为空,这种情况发生在复位操作时,或者当读指针读出FIFO中最后一个字后,追赶上了写指针时,如下图所示:

当读写指针再次相等时,表明FIFO为满,这种情况发生在,当写指针转了一圈,折回来(wrapped around)又追上了读指针,如下图:

为了区分到底是满状态还是空状态,可以采用以下方法:
方法1:在指针中添加一个额外的位(extra bit),当写指针增加并越过最后一个FIFO地址时,就将写指针这个未用的MSB加1,其它位回零。对读指针也进行同样的操作。此时,对于深度为2n的FIFO,需要的读/写指针位宽为(n+1)位,如对于深度为8的FIFO,需要采用4bit的计数器,0000~1000、1001~1111,MSB作为折回标志位,而低3位作为地址指针。
如果两个指针的MSB不同,说明写指针比读指针多折回了一次;如r_addr=0000,而w_addr = 1000,为满。
如果两个指针的MSB相同,则说明两个指针折回的次数相等。其余位相等,说明FIFO为空;
3.二进制FIFO指针的考虑
将一个二进制的计数值从一个时钟域同步到另一个时钟域的时候很容易出现问题,因为采用二进制计数器时所有位都可能同时变化,在同一个时钟沿同步多个信号的变化会产生亚稳态问题。而使用格雷码只有一位变化,因此在两个时钟域间同步多个位不会产生问题。所以需要一个二进制到gray码的转换电路,将地址值转换为相应的gray码,然后将该gray码同步到另一个时钟域进行对比,作为空满状态的检测。

4. 使用gray码进行对比,如何判断“空”与“满”
使用gray码解决了一个问题,但同时也带来另一个问题,即在格雷码域如何判断空与满。
对于“空”的判断依然依据二者完全相等(包括MSB);
而对于“满”的判断,如下图,由于gray码除了MSB外,具有镜像对称的特点,当读指针指向7,写指针指向8时,除了MSB,其余位皆相同,不能说它为满。因此不能单纯的只检测最高位了,在gray码上判断为满必须同时满足以下3条:
- wptr和同步过来的rptr的MSB不相等,因为wptr必须比rptr多折回一次。
- wptr与rptr的次高位不相等,如上图位置7和位置15,转化为二进制对应的是0111和1111,MSB不同说明多折回一次,111相同代表同一位置。
- 剩下的其余位完全相等。
<ignore_js_op>
5.总体实现
系统的总体框图如下:

1)顶层模块
- module AsyncFIFO
- #(parameter ASIZE = 4, //地址位宽
- parameter DSIZE = 8) //数据位宽
- (
- input [DSIZE-1:0] wdata,
- input winc, wclk, wrst_n, //写请求信号,写时钟,写复位
- input rinc, rclk, rrst_n, //读请求信号,读时钟,读复位
- output [DSIZE-1:0] rdata,
- output wfull,
- output rempty
- );
- wire [ASIZE-1:0] waddr, raddr;
- wire [ASIZE:0] wptr, rptr, wq2_rptr, rq2_wptr; /************************************************************
- * In order to perform FIFO full and FIFO empty tests using
- * this FIFO style, the read and write pointers must be
- * passed to the opposite clock domain for pointer comparison
- *************************************************************/
- /*在检测“满”或“空”状态之前,需要将指针同步到其它时钟域时,使用格雷码,可以降低同步过程中亚稳态出现的概率*/
- sync_r2w I1_sync_r2w(
- .wq2_rptr(wq2_rptr),
- .rptr(rptr),
- .wclk(wclk),
- .wrst_n(wrst_n));
- sync_w2r I2_sync_w2r (
- .rq2_wptr(rq2_wptr),
- .wptr(wptr),
- .rclk(rclk),
- .rrst_n(rrst_n));
- /*
- * DualRAM
- */
- DualRAM #(DSIZE, ASIZE) I3_DualRAM(
- .rdata(rdata),
- .wdata(wdata),
- .waddr(waddr),
- .raddr(raddr),
- .wclken(winc),
- .wclk(wclk));
- /*
- * 空、满比较逻辑
- */
- rptr_empty #(ASIZE) I4_rptr_empty(
- .rempty(rempty),
- .raddr(raddr),
- .rptr(rptr),
- .rq2_wptr(rq2_wptr),
- .rinc(rinc),
- .rclk(rclk),
- .rrst_n(rrst_n));
- wptr_full #(ASIZE) I5_wptr_full(
- .wfull(wfull),
- .waddr(waddr),
- .wptr(wptr),
- .wq2_rptr(wq2_rptr),
- .winc(winc),
- .wclk(wclk),
- .wrst_n(wrst_n));
- endmodule
复制代码
2)DualRAM模块
- module DualRAM
- #(
- parameter DATA_SIZE = 8, // 数据位宽
- parameter ADDR_SIZE = 4 // 地址位宽
- )
- (
- input wclken,wclk,
- input [ADDR_SIZE-1:0] raddr, //RAM read address
- input [ADDR_SIZE-1:0] waddr, //RAM write address
- input [DATA_SIZE-1:0] wdata, //data input
- output [DATA_SIZE-1:0] rdata //data output
- );
- localparam RAM_DEPTH = 1 << ADDR_SIZE; //RAM深度 = 2^ADDR_WIDTH
- reg [DATA_SIZE-1:0] Mem[RAM_DEPTH-1:0];
- always@(posedge wclk)
- begin
- if(wclken)
- Mem[waddr] <= wdata;
- end
- assign rdata = Mem[raddr];
- endmodule
复制代码
3)同步模块
- module sync_r2w
- #(parameter ADDRSIZE = 4)
- (
- output reg [ADDRSIZE:0] wq2_rptr,
- input [ADDRSIZE:0] rptr,
- input wclk, wrst_n
- );
- reg [ADDRSIZE:0] wq1_rptr;
- always @(posedge wclk or negedge wrst_n)
- if (!wrst_n)
- {wq2_rptr,wq1_rptr} <= 0;
- else
- {wq2_rptr,wq1_rptr} <= {wq1_rptr,rptr};
- endmodule
复制代码
4)同步模块2
- module sync_w2r
- #(parameter ADDRSIZE = 4)
- (
- output reg [ADDRSIZE:0] rq2_wptr,
- input [ADDRSIZE:0] wptr,
- input rclk, rrst_n
- ); reg [ADDRSIZE:0] rq1_wptr;
- always @(posedge rclk or negedge rrst_n)
- if (!rrst_n)
- {rq2_wptr,rq1_wptr} <= 0;
- else
- {rq2_wptr,rq1_wptr} <= {rq1_wptr,wptr};
- endmodule
复制代码
5)空判断逻辑
- module rptr_empty
- #(parameter ADDRSIZE = 4)
- (
- output reg rempty,
- output [ADDRSIZE-1:0] raddr,
- output reg [ADDRSIZE :0] rptr,
- input [ADDRSIZE :0] rq2_wptr,
- input rinc, rclk, rrst_n);
- reg [ADDRSIZE:0] rbin;
- wire [ADDRSIZE:0] rgraynext, rbinnext;
- wire rempty_val;
- //-------------------
- // GRAYSTYLE2 pointer: gray码读地址指针
- //-------------------
- always @(posedge rclk or negedge rrst_n)
- if (!rrst_n)
- begin
- rbin <= 0;
- rptr <= 0;
- end
- else
- begin
- rbin <= rbinnext ;
- rptr <= rgraynext;
- end
- // gray码计数逻辑
- assign rbinnext = !rempty ? (rbin + rinc) : rbin;
- assign rgraynext = (rbinnext>>1) ^ rbinnext; //二进制到gray码的转换
- assign raddr = rbin[ADDRSIZE-1:0];
- //---------------------------------------------------------------
- // FIFO empty when the next rptr == synchronized wptr or on reset
- //---------------------------------------------------------------
- /*
- * 读指针是一个n位的gray码计数器,比FIFO寻址所需的位宽大一位
- * 当读指针和同步过来的写指针完全相等时(包括MSB),说明二者折回次数一致,FIFO为空
- *
- */
- assign rempty_val = (rgraynext == rq2_wptr);
- always @(posedge rclk or negedge rrst_n)
- if (!rrst_n)
- rempty <= 1'b1;
- else
- rempty <= rempty_val;
- endmodule
复制代码
6)满判断逻辑
- module wptr_full
- #(
- parameter ADDRSIZE = 4
- )
- (
- output reg wfull,
- output [ADDRSIZE-1:0] waddr,
- output reg [ADDRSIZE :0] wptr,
- input [ADDRSIZE :0] wq2_rptr,
- input winc, wclk, wrst_n);
- reg [ADDRSIZE:0] wbin;
- wire [ADDRSIZE:0] wgraynext, wbinnext;
- wire wfull_val;
- // GRAYSTYLE2 pointer
- always @(posedge wclk or negedge wrst_n)
- if (!wrst_n)
- begin
- wbin <= 0;
- wptr <= 0;
- end
- else
- begin
- wbin <= wbinnext;
- wptr <= wgraynext;
- end
- //gray 码计数逻辑
- assign wbinnext = !wfull ? wbin + winc : wbin;
- assign wgraynext = (wbinnext>>1) ^ wbinnext;
- assign waddr = wbin[ADDRSIZE-1:0];
- /*由于满标志在写时钟域产生,因此比较安全的做法是将读指针同步到写时钟域*/
- /**/
- //------------------------------------------------------------------
- // Simplified version of the three necessary full-tests:
- // assign wfull_val=((wgnext[ADDRSIZE] !=wq2_rptr[ADDRSIZE] ) &&
- // (wgnext[ADDRSIZE-1] !=wq2_rptr[ADDRSIZE-1]) &&
- // (wgnext[ADDRSIZE-2:0]==wq2_rptr[ADDRSIZE-2:0]));
- //------------------------------------------------------------------
- assign wfull_val = (wgraynext=={~wq2_rptr[ADDRSIZE:ADDRSIZE-1],
- wq2_rptr[ADDRSIZE-2:0]});
- always @(posedge wclk or negedge wrst_n)
- if (!wrst_n)
- wfull <= 1'b0;
- else
- wfull <= wfull_val;
- endmodule
复制代码
P.S : 在quartus中有异步FIFO IP核,为安全起见推荐使用IP核定制FIFO,本文的目的只是作为思路参考。
转载:http://www.openhw.org/module/forum/thread-596561-1-1.html
异步FIFO的FPGA实现的更多相关文章
- 异步FIFO总结
异步FIFO总结 异步FIFO的基本概念 异步FIFO读写分别采用相互异步的不同时钟,使用异步FIFO可以在两个不同时钟系统之间快速而方便地传输实时数据 FIFO的常见参数 FIFO的宽度:即FIFO ...
- 【iCore、iCore2、iBoard例程】【异步FIFO跨时钟域通信(通过ARM 读FPGA FIFO)】
欢迎访问电子工程师学堂,以便了解更多内容:http://www.eeschool.org 一.本实验基于iCore2 完成,通过简单改动,即可用在 iCore 核心板.iBoard 电子学堂上. iC ...
- 基于FPGA的异步FIFO验证
现在开始对上一篇博文介绍的异步FIFO进行功能验证,上一篇博文地址:http://blog.chinaaet.com/crazybird/p/5100000872 .对异步FIFO验证的平台如图1所示 ...
- 基于FPGA的异步FIFO设计
今天要介绍的异步FIFO,可以有不同的读写时钟,即不同的时钟域.由于异步FIFO没有外部地址端口,因此内部采用读写指针并顺序读写,即先写进FIFO的数据先读取(简称先进先出).这里的读写指针是异步的, ...
- 异步fifo的设计(FPGA)
本文首先对异步 FIFO 设计的重点难点进行分析 最后给出详细代码 一.FIFO简单讲解 FIFO的本质是RAM, 先进先出 重要参数:fifo深度(简单来说就是需要存多少个数据) ...
- 异步FIFO空满设计延迟问题
由于设计的时候读写指针用了至少两级寄存器同步,同步会消耗至少两个时钟周期,势必会使得判断空或满有所延迟,这会不会导致设计出错呢? 异步FIFO通过比较读写指针进行满空判断,但是读写指针属于不同的时钟域 ...
- 异步FIFO跨时钟域亚稳态如何解决?
跨时钟域的问题:前一篇已经提到要通过比较读写指针来判断产生读空和写满信号,但是读指针是属于读时钟域的,写指针是属于写时钟域的,而异步FIFO的读写时钟域不同,是异步的,要是将读时钟域的读指针与写时钟域 ...
- 异步fifo with 读控制
之前做LDPC编码器时,学习了一下异步FIFO的相关知识,主要参考了http://www.cnblogs.com/aslmer/p/6114216.html,并在此基础上根据项目需求,添加了一个读控制 ...
- 异步FIFO中空满信号如何产生?
异步FIFO中,空满信号该如何产生呢? 在复位的时候,读指针和写指针相等,读空信号有效(这里所说的指针其实就是读地址.写地址)当读指针赶上写指针的时候,写指针等于读指针意味着最后一个数据被读完,此时读 ...
随机推荐
- .NET:负载平衡的主意事项
允许局域网发现和共享. 设置固定IP. 网站的IP设置为“全部未分配”. 注意:如果停止IIS的话,不会对负载平衡有影响,负载还是会分配停止了的IIS所在在的电脑,只有停止服务器了,负载不会再分配给停 ...
- ABC定制视图导航控制器
ABCustomUINavigationController ABC定制视图导航控制器 Subclass of UINavigationController for overwriting ...
- java.io.FileNotFoundException: ***(Too many open files)
http://yizhilong28.iteye.com/blog/1154098 在linux下跑一个多线程读取文件的程序,待读取文件有数十万个.程序读取过程中抛出如下异常*****(Too man ...
- iOS:上线的基本流程
1.创建唯一标书符App ID 首先打开开发者网站,进入证书页面,我们点击Identifiers下边App IDs选项,然后点击右上角加号,如图: 2.申请发布证书 我们点击Certificates下 ...
- vue刷新当前路由:router-view 复用组件时不刷新的3种解决方案总结
vue-router是Vue.js官方的路由插件,它和vue.js是深度集成的,适合用于构建单页面应用.vue的单页面应用是基于路由和组件的,路由用于设定访问路径,并将路径和组件映射起来.传统的页面应 ...
- 用curl抓取网站数据,仿造IP、防屏蔽终极强悍解决方式
最近在做一些抓取其它网站数据的工作,当然别人不会乖乖免费给你抓数据的,有各种防抓取的方法.不过道高一尺,魔高一丈,通过研究都是有漏洞可以钻的.下面的例子都是用PHP写的,不会用PHP来curl的孩纸先 ...
- POJ2528:Mayor's posters(线段树区间更新+离散化)
Description The citizens of Bytetown, AB, could not stand that the candidates in the mayoral electio ...
- TextView子类的常用属性
TextView常见的子类包括EditText,Button,CheckBox, RadioButton等. 1.EditText EditText继承自TextView,因此TextView所有属性 ...
- css 小问题解决方法整理
1,图片垂直居中: 设置包括图片的div:height=20px:line-height=20px. 设置图片vertical-align:middle 就可以. 2,行内块元素有3px bug,可通 ...
- Discuz常见小问题-如何删除用户
用户-用户管理,直接拉到底部点搜索 切换到最后一页,然后勾选要删除的用户(最后一页就是最新注册的用户) 要勾选两个选项,删除过程比较长,不要中途退出,等待删除完毕 再次搜索用户就少一个了