概述
转载网址:http://blog.sina.com.cn/s/blog_6592e7700100x68n.html#cmt_3255991
图1
module yibufifo(rdata,rempty,rrep,rclk,rrst_n,wdata,wfull,wrep,wclk,wrst_n);
input wclk,wrep,wrst_n;
input rclk,rrep,rrst_n;
input[7:0]wdata;
output[7:0]rdata;
output rempty,wfull;
wire wclk,wrep,wrst_n;
wire rclk,rrep,rrst_n;
wire [7:0]wdata;
wire [7:0]rdata;
wire [3:0]wptr,rptr;
wire [3:0]waddr,raddr;
wire aempty_n,afull_n;
ram i1(.wdata(wdata),//读写存储模块
async_cmp i2(.aempty_n(aempty_n),//异步比较读写指针产生异步空满标志
rptr_empty2 i3(.rempty(rempty),//根据rclk产生读指针rptr和空标志rempty
wptr_full2 i4(.wfull(wfull),//根据wclk产生写指针wptr和满标志wfull
endmodule
顶层模块图:
读写地址线一般有多位,如果在不同的时钟域内直接同步二进制码的地址指针,则有可能产生亚稳态。例如,读指针从011变化到100时,所有位都要变化,读指针的每一位在读时钟的作用下,跳变不一致,即产生毛刺。如果写时钟恰好在读指针的变化时刻采样,得到的采样信号可能是000~111中的任何一个,从而导致空/满信号判断错误。由实践可知,同步多个异步输入信号出现亚稳态的概率远远大于同步一个异步信号的概率[3]。解决这一问题的有效方法是采用格雷码。格雷码的主要特点是相邻的两个编码之间只有一位变化。图2是格雷码产生的逻辑框图。在读使能或写使能信号有效、并且空/满标志无效的情况下,读写指针开始累加,进行FIFO读或写操作。二进制码与格雷码的转换是一个“异或”运算:gnext=(bnext>>1)^bnext。格雷码gnext 经寄存器输出格雷码指针ptr。这种方法采用了两组寄存器,虽然面积较大,但是有助于提高系统的工作频率。
图2
always @ (posedge rclk or negedge rrst_n)
begin
end
assign rbnext=!rempty?rbin+rrep:rbin;
assign rgnext=(rbnext>>1)^rbnext;//二进制码转换成格雷码
begin
end
assign wbnext=!wfull?wbin+wrep:wbin;
assign wgnext=(wbnext>>1)^wbnext;//二进制码转换成格雷码
2.2 空/满标志产生逻辑
图4
wire close_empty_n=~((~(wptr[n]^rptr[n-1]) & (wptr[n-1]^rptr[n])) |~wrst);
always @(posedge high or negedge dirset or negedge dirrst)
assign aempty=~((wptr==rptr) && !direction);
assign afull=~((wptr==rptr) && direction);
always @(posedge rclk or negedge aempty)
always @(posedge wclk or negedge afull)
设计中FIFO空/满标志的设置是保守的,即FIFO空/满标志的置位是立即有效的,而其失效则是在一段时间之后。例如一旦读指针追上写指针,就会立即声明一个低电平有效的异步空信号aempty。此信号会立即把图6所示的set触发器置位,使触发器输出为1,即向外部输出同步的空信号rempty,并且保证了FIFO一旦为空,读指针就不增加,避免了FIFO的读溢出。当写地址增加时,表明FIFO已经非空,空标志aempty由低变高,此时 可以进行安全的读操作。aempty信号的失效与写时钟同步。空信号rempty是在读时钟域中同步aempty信号得到的。由于同步器使用了两个触发器,因此空信号rempty的失效要经过至少两个时钟周期的延迟。所以,空信号的声明是及时的,而空信号的失效是保守的。也就是说,虽然FIFO已经非空了,但是空信号rempty要经过几个周期的延迟才能变为无效。满信号也有类似的情况。
虽然空/满标志的设置是保守的,但这并不影响FIFO功能的正确性,经验证保守的空/满标志能够满足FIFO的设计要求。
具体的各个模块程序如下:
//异步FIFO缓存16*8数据,即数据宽度为8,深度为 16//
module ram(wclk,wrep,wdata,waddr,raddr,rdata);
input wclk,wrep;//读时钟,读使能
input [7:0]wdata;//写数据8位
input[3:0]waddr,raddr;//读写地址
output[7:0]rdata;
reg[7:0]fifomem[0:3];
//du
assign
//xie
always@(posedge wclk)
begin
end
endmodule
ram模块图如下:
异步比较读写指针产生异步空满标志async_cmp模块
//异步FIFO缓存16*8数据,即数据宽度为8,深度为 16//
module async_cmp(aempty_n,afull_n,wptr,rptr,wrst_n);//异步比较产生空满信号
//parameter addrsize=4;
//parameter N=addrsize-1;//地址位宽
input wrst_n;//写复位
input [3:0]wptr,rptr;//读写指针
output aempty_n,afull_n;//空满标志
reg direction;// 用一个寄存器direction来寄存当前是否接近满空的状态
wire close_full_direction_n; //快接近满了,低电平有效
wire close_empty_direction_n;//快接近空了,低电平有效
assign close_full_direction_n=~((wptr[3]^rptr[3-1])&~(wptr[3-1]^rptr[3]));
assign close_empty_direction_n=~((~(wptr[3]^rptr[3-1])&(wptr[3-1]^rptr[3]))|~wrst_n);
always@( negedge close_full_direction_n or negedge close_empty_direction_n)//低电平有效,用来判断direction
begin
end
assign aempty_n=~((wptr==rptr)&&!direction);//几乎空信号产生
assign afull_n=~((wptr==rptr)&& direction);//几乎满信号产生
endmodule
async_cmp模块图如下:
根据rclk产生读指针rptr和空标志rempty
//异步FIFO缓存16*8数据,即数据宽度为8,深度为 16//
module rptr_empty2(rempty,rptr,aempty_n,rrep,rclk,rrst_n);
input rrep,rclk,rrst_n;//读时钟,读使能,读复位
input aempty_n;//空标志
output[3:0]rptr;//读指针
output rempty;//读空标志
reg[3:0]rptr,rbin;//读指针,读地址二进制
reg rempty,rempty2;
wire[3:0]rgnext,rbnext;//读格雷码地址,读二进制地址
always@(posedge rclk or negedge rrst_n)
begin
end
assign rbnext=!rempty?rbin+rrep:rbin;
assign rgnext=(rbnext>>1)^rbnext;//二进制码转换成格雷码
always@(posedge rclk or negedge aempty_n)//读空标志的产生。用格雷码表示
begin
end
endmodule
rptr_empty2模块图:
根据wclk产生写指针wptr和满标志wfull,wptr_full2模块
//异步FIFO缓存16*8数据,即数据宽度为8,深度为 16//
module wptr_full2(wfull,wptr,afull_n,wrep,wclk,wrst_n);
input wrep,wclk,wrst_n;//写时钟,写使能,写复位
input afull_n;//满标志
output[3:0]wptr;//写指针
output wfull;//写满标志
reg[3:0]wptr,wbin;//写指针,写地址二进制
reg wfull,wfull2;
wire[3:0]wgnext,wbnext;//写格雷码地址,写二进制地址
always@(posedge wclk or negedge wrst_n)
begin
end
assign wbnext=!wfull?wbin+wrep:wbin;
assign wgnext=(wbnext>>1)^wbnext;//二进制码转换成格雷码
always@(posedge wclk or negedge afull_n)//写满标志的产生。用格雷码表示
begin
end
endmodule
wptr_full2模块图如下:
测试程序:
//parameter DSIZE = 8;
//parameter ASIZE = 4;
wire [7:0] rdata;
wire wfull;
wire rempty;
reg [7:0] wdata;
reg wrep, wclk, wrst_n;
reg rrep, rclk, rrst_n;
initial begin
end
always #5 wclk=!wclk;//10ns
always #10 rclk=!rclk;//20ns
always @(negedge wclk)
yibufifo i1 (
endmodule
在modelsim中的仿真结果如下:
最后
以上就是甜美白羊为你收集整理的异步FIFO设计与实现的全部内容,希望文章能够帮你解决异步FIFO设计与实现所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
发表评论 取消回复