概述
本文主体翻译自C. E. Cummings and S. Design, “Simulation and Synthesis Techniques for Asynchronous FIFO Design 一文,添加了笔者的个人理解与注释,文中蓝色部分为笔者注或意译。前文链接:
异步FIFO设计的仿真与综合技术(5)https://blog.csdn.net/apple_53311083/article/details/132974327?spm=1001.2014.3001.5501
6.0 FIFO设计的RTL代码(RTL code for FIFO Style #1)
本节中列出了FIFO样式#1模型的Verilog RTL代码。
6.1 fifo1.v - FIFO top-level module
顶层的FIFO模块是一个参数化的FIFO设计,所有的子块都使用推荐的执行命名端口连接的方式进行实例化。另一种常见的编码方式是为顶层模块实例化提供与模块名称相同的名称。这样做是为了方便调试,因为如果实例名称与模块名称匹配,那么在层次路径中引用模块名称将是直接的。
module fifo1 #(parameter DSIZE = 8,parameter ASIZE = 4)(output [DSIZE-1:0] rdata,output wfull,output rempty,input [DSIZE-1:0] wdata,input winc, wclk, wrst_n,input rinc, rclk, rrst_n);wire [ASIZE-1:0] waddr, raddr;wire [ASIZE:0] wptr, rptr, wq2_rptr, rq2_wptr;sync_r2w sync_r2w (.wq2_rptr(wq2_rptr), .rptr(rptr),.wclk(wclk), .wrst_n(wrst_n));sync_w2r sync_w2r (.rq2_wptr(rq2_wptr), .wptr(wptr),.rclk(rclk), .rrst_n(rrst_n));fifomem #(DSIZE, ASIZE) fifomem(.rdata(rdata), .wdata(wdata),.waddr(waddr), .raddr(raddr),.wclken(winc), .wfull(wfull),.wclk(wclk));rptr_empty #(ASIZE) rptr_empty(.rempty(rempty),.raddr(raddr),.rptr(rptr), .rq2_wptr(rq2_wptr),.rinc(rinc), .rclk(rclk),.rrst_n(rrst_n));wptr_full #(ASIZE) wptr_full(.wfull(wfull), .waddr(waddr),.wptr(wptr), .wq2_rptr(wq2_rptr),.winc(winc), .wclk(wclk),.wrst_n(wrst_n));endmodule
6.2 fifomem.v - FIFO memory buffer
FIFO存储器缓冲区通常是一个实例化的ASIC或FPGA双端口,同步存储器设备。内存缓冲区也可以使用这个模块中的RTL代码合成到ASIC或FPGA寄存器。关于实例化的供应商RAM与verilog声明的RAM,同步系统设计团队进行了内部分析,发现对于高达256位的大小,与实例化的供应商RAM相比,使用verilog声明的RAM不会损失区域或性能。如果已实例化了供应商RAM,则强烈建议使用已命名的端口连接进行实例化。
module fifomem #(parameter DATASIZE = 8, // Memory data word widthparameter ADDRSIZE = 4) // Number of mem address bits(output [DATASIZE-1:0] rdata,input [DATASIZE-1:0] wdata,input [ADDRSIZE-1:0] waddr, raddr,input wclken, wfull, wclk);`ifdef VENDORRAM// instantiation of a vendor's dual-port RAMvendor_ram mem (.dout(rdata), .din(wdata),.waddr(waddr), .raddr(raddr),.wclken(wclken),.wclken_n(wfull), .clk(wclk));`else// RTL Verilog memory modellocalparam DEPTH = 1<<ADDRSIZE;reg [DATASIZE-1:0] mem [0:DEPTH-1];assign rdata = mem[raddr];always @(posedge wclk)if (wclken && !wfull) mem[waddr] <= wdata;`endif
endmodule
6.3 sync_r2w.v - Read-domain to write-domain synchronizer
这是一个简单的同步器模块,用于通过由FIFO写入时钟时钟的一对寄存器,将n位指针从读取时钟域传递到写入时钟域。请注意,将两个寄存器连接在一起以进行重置和移位的总是块的简单性。同步器always块只有三行代码。所有模块输出都被注册为使用时间预算进行简化合成。该模块的所有输出都与wclk完全同步,而该模块的所有异步输入都来自rclk域,所有信号都使用“r”前缀命名,这使得很容易在所有“r*”信号上设置错误路径,以便简化静态定时分析。
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
6.4 sync_w2r.v - Write-domain to read-domain synchronizer
这是一个简单的同步器模块,用于通过FIFO读时钟时钟的一对寄存器,从写时钟域传递到读时钟域。请注意将两个寄存器连接在一起以进行重置和移动的总是块的简单性。同步器always块只有三行代码。所有模块输出都被注册为使用时间预算进行简化合成。该模块的所有输出都与rclk完全同步,而该模块的所有异步输入都来自wclk域,所有信号都使用“w”前缀命名,这使得很容易在所有“w*”信号上设置错误路径,以便简化静态定时分析。
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
6.5 rptr_empty.v - Read pointer & empty generation logic
此模块包含在读取时钟域内生成的所有FIFO逻辑(同步器除外)。读取指针是一个双n位格雷码计数器。n位指针(rptr)通过sync_r2w模块传递到写时钟域。(n-1)位指针(raddr)用于寻址FIFO缓冲区。当下一个rptr值等于同步的wptr值时,注册FIFO空输出并在下一个上升沿生效。所有模块输出都被注册为使用时间预算进行简化合成。这个模块与rclk完全同步,以简化静态定时分析。
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;//-------------------// GRAYSTYLE2 pointer//-------------------always @(posedge rclk or negedge rrst_n)if (!rrst_n) {rbin, rptr} <= 0;else {rbin, rptr} <= {rbinnext, rgraynext};// Memory read-address pointer (okay to use binary to address memory)assign raddr = rbin[ADDRSIZE-1:0];assign rbinnext = rbin + (rinc & ~rempty);assign rgraynext = (rbinnext>>1) ^ rbinnext;//---------------------------------------------------------------// FIFO empty when the next rptr == synchronized wptr or on reset//---------------------------------------------------------------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.6 wptr_full.v - Write pointer & full generation logic
此模块包含在写入时钟域内生成的所有FIFO逻辑(同步器除外)。写指针是一个双n位格雷码计数器。n位指针(wptr)通过sync_w2r模块传递到读取时钟域。(n-1)位指针(waddr)用于处理FIFO缓冲区。当下一个修改的wgnext值等于同步和修改的wrptr2值(MSBs除外)时,注册FIFO完整输出并在下一个wclk上升沿生效。所有模块输出都被注册为使用时间预算进行简化合成。该模块与wclk完全同步,以简化静态定时分析。
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;// GRAYSTYLE2 pointeralways @(posedge wclk or negedge wrst_n)if (!wrst_n) {wbin, wptr} <= 0;else {wbin, wptr} <= {wbinnext, wgraynext};// Memory write-address pointer (okay to use binary to address memory)assign waddr = wbin[ADDRSIZE-1:0];assign wbinnext = wbin + (winc & ~wfull);assign wgraynext = (wbinnext>>1) ^ wbinnext;//------------------------------------------------------------------// 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