FPGA线上课程平台|最全栈的FPGA学习平台|FPGA工程师认证培训
登录
首页-技术文章/快讯-技术分享-正文

Verilog入门:2026年用异步FIFO解决跨时钟域问题的实战技巧

二牛学FPGA二牛学FPGA
技术分享
3小时前
0
0
4

Quick Start

以下步骤可在30分钟内搭建一个异步FIFO验证环境,观察跨时钟域数据传输的正确性。

  • 步骤1:下载并安装Vivado 2024.1(或更高版本),新建一个RTL工程。
  • 步骤2:创建顶层模块async_fifo_top,例化一个异步FIFO IP核(Xilinx FIFO Generator)或手写RTL。
  • 步骤3:编写写时钟域(wr_clk, 100MHz)和读时钟域(rd_clk, 50MHz)的激励。
  • 步骤4:在写时钟域每5个周期写入一个数据,共写入16个数据。
  • 步骤5:在读时钟域连续读取,直到读空。
  • 步骤6:运行行为仿真,观察写指针、读指针、空/满标志和读出数据是否与写入一致。
  • 步骤7:检查波形:full拉高后停止写入,empty拉高后停止读取,数据无丢失、无乱序。
  • 步骤8:综合实现,查看资源占用和Fmax(写时钟域约200MHz,读时钟域约150MHz,典型值)。

前置条件与环境

项目推荐值说明替代方案
器件/板卡Xilinx Artix-7 XC7A35T入门级FPGA,资源充足Intel Cyclone IV / Lattice iCE40
EDA版本Vivado 2024.1支持SystemVerilog和IP集成Vivado 2023.x / Quartus 22.x
仿真器Vivado Simulator (xsim)内置于Vivado,无需额外安装ModelSim / Questa / Verilator
时钟/复位写时钟100MHz,读时钟50MHz,异步复位高有效典型跨时钟域场景任意频率比,建议写时钟≥读时钟
接口依赖标准AXI4-Stream或简单握手易于集成到系统总线自定义valid-ready协议
约束文件XDC文件:时钟定义、假路径约束避免跨时钟域路径被错误分析SDC(Quartus)

目标与验收标准

完成以下功能点即视为设计通过:

  • 功能点1:写时钟域写入N个数据(N≤FIFO深度),读时钟域完整读出,数据内容与顺序一致。
  • 功能点2:满标志(full)在FIFO写满时准确拉高,写使能被阻塞;空标志(empty)在FIFO读空时准确拉高,读使能被阻塞。
  • 性能指标:写时钟域Fmax≥180MHz,读时钟域Fmax≥120MHz(以Artix-7速度等级-1为参考)。
  • 资源占用:LUT≤200,FF≤300,BRAM≤1个(深度16,数据宽度8位)。
  • 验收方式:行为仿真通过,无时序违例(Setup/Hold),上板测试通过(如使用LED或串口回传数据)。

实施步骤

阶段1:工程结构与模块划分

推荐模块层次:

  • 顶层 async_fifo_top:例化FIFO和读写控制器。
  • 写控制器 wr_ctrl:产生写使能和写数据。
  • 读控制器 rd_ctrl:产生读使能,读取并检查数据。
  • 异步FIFO核心 async_fifo:包含双端口RAM、指针同步器、空满逻辑。

常见坑与排查:

  • 坑1:忘记例化同步器,导致跨时钟域信号直接连接。排查:检查RTL中是否使用两级触发器同步。
  • 坑2:复位不同步,导致FIFO状态初始化失败。排查:确保复位信号在每个时钟域内先同步再使用。

阶段2:关键模块RTL实现

以下给出异步FIFO核心模块的Verilog代码,深度为16,数据宽度8位。

module async_fifo #(
    parameter DATA_WIDTH = 8,
    parameter FIFO_DEPTH = 16
)(
    input  wire                     wr_clk,
    input  wire                     wr_rst_n,
    input  wire                     wr_en,
    input  wire [DATA_WIDTH-1:0]    wr_data,
    output wire                     full,

    input  wire                     rd_clk,
    input  wire                     rd_rst_n,
    input  wire                     rd_en,
    output wire [DATA_WIDTH-1:0]    rd_data,
    output wire                     empty
);

    // 地址宽度 = log2(FIFO_DEPTH)
    localparam ADDR_WIDTH = $clog2(FIFO_DEPTH);

    // 双端口RAM
    reg [DATA_WIDTH-1:0] mem [0:FIFO_DEPTH-1];

    // 写指针(二进制)
    reg [ADDR_WIDTH:0] wr_ptr_bin;
    wire [ADDR_WIDTH:0] wr_ptr_gray;

    // 读指针(二进制)
    reg [ADDR_WIDTH:0] rd_ptr_bin;
    wire [ADDR_WIDTH:0] rd_ptr_gray;

    // 同步后的指针
    reg [ADDR_WIDTH:0] wr_ptr_gray_sync1, wr_ptr_gray_sync2;
    reg [ADDR_WIDTH:0] rd_ptr_gray_sync1, rd_ptr_gray_sync2;

    // 写地址与写数据
    always @(posedge wr_clk or negedge wr_rst_n) begin
        if (!wr_rst_n)
            wr_ptr_bin <= 0;
        else if (wr_en && !full)
            wr_ptr_bin <= wr_ptr_bin + 1;
    end

    always @(posedge wr_clk) begin
        if (wr_en && !full)
            mem[wr_ptr_bin[ADDR_WIDTH-1:0]] <= wr_data;
    end

    // 读地址与读数据
    always @(posedge rd_clk or negedge rd_rst_n) begin
        if (!rd_rst_n)
            rd_ptr_bin <= 0;
        else if (rd_en && !empty)
            rd_ptr_bin <= rd_ptr_bin + 1;
    end

    assign rd_data = mem[rd_ptr_bin[ADDR_WIDTH-1:0]];

    // 二进制转格雷码
    assign wr_ptr_gray = wr_ptr_bin ^ (wr_ptr_bin >> 1);
    assign rd_ptr_gray = rd_ptr_bin ^ (rd_ptr_bin >> 1);

    // 写指针同步到读时钟域
    always @(posedge rd_clk or negedge rd_rst_n) begin
        if (!rd_rst_n) begin
            wr_ptr_gray_sync1 <= 0;
            wr_ptr_gray_sync2 <= 0;
        end else begin
            wr_ptr_gray_sync1 <= wr_ptr_gray;
            wr_ptr_gray_sync2 <= wr_ptr_gray_sync1;
        end
    end

    // 读指针同步到写时钟域
    always @(posedge wr_clk or negedge wr_rst_n) begin
        if (!wr_rst_n) begin
            rd_ptr_gray_sync1 <= 0;
            rd_ptr_gray_sync2 <= 0;
        end else begin
            rd_ptr_gray_sync1 <= rd_ptr_gray;
            rd_ptr_gray_sync2 <= rd_ptr_gray_sync1;
        end
    end

    // 空满判断
    wire [ADDR_WIDTH:0] wr_ptr_gray_sync = wr_ptr_gray_sync2;
    wire [ADDR_WIDTH:0] rd_ptr_gray_sync = rd_ptr_gray_sync2;

    assign full  = (wr_ptr_gray[ADDR_WIDTH:ADDR_WIDTH-2] == ~rd_ptr_gray_sync[ADDR_WIDTH:ADDR_WIDTH-2] &&
                    wr_ptr_gray[ADDR_WIDTH-3:0] == rd_ptr_gray_sync[ADDR_WIDTH-3:0]);
    assign empty = (rd_ptr_gray == wr_ptr_gray_sync);

endmodule

逐行说明

  • 第1行:模块定义开始,参数化数据宽度和FIFO深度,便于复用。
  • 第2-3行:参数声明,DATA_WIDTH默认8位,FIFO_DEPTH默认16。
  • 第4-12行:端口列表,包含写时钟域和读时钟域的时钟、复位、使能、数据、标志。
  • 第14行:用$clog2计算地址宽度,深度16时ADDR_WIDTH=4。
  • 第16行:声明双端口RAM,深度FIFO_DEPTH,宽度DATA_WIDTH。
  • 第18-19行:写指针寄存器,宽度为ADDR_WIDTH+1(多1位用于空满判断)。
  • 第21-22行:读指针寄存器。
  • 第24-25行:同步器寄存器,两级触发器。
  • 第27-31行:写指针递增逻辑,只在写使能且非满时递增。
  • 第33-36行:写数据到RAM,地址取写指针的低ADDR_WIDTH位。
  • 第38-42行:读指针递增逻辑。
  • 第44行:组合逻辑读取RAM,地址为读指针低ADDR_WIDTH位。
  • 第46-47行:二进制转格雷码,使用异或右移1位。
  • 第49-56行:写指针格雷码同步到读时钟域,两级触发器消除亚稳态。
  • 第58-65行:读指针格雷码同步到写时钟域。
  • 第67-68行:将同步后的指针赋值给内部线网。
  • 第70-71行:满标志判断:写指针格雷码的高两位与同步后的读指针格雷码高两位取反相等,且低位相等。
  • 第72行:空标志判断:读指针格雷码与同步后的写指针格雷码相等。

阶段3:时序约束与CDC处理

异步FIFO的核心是跨时钟域路径(CDC)。必须用约束告诉工具不要分析这些路径的时序。

# 写时钟域约束
create_clock -name wr_clk -period 10.000 [get_ports wr_clk]

# 读时钟域约束
create_clock -name rd_clk -period 20.000 [get_ports rd_clk]

# 跨时钟域假路径:从写时钟到读时钟
set_false_path -from [get_clocks wr_clk] -to [get_clocks rd_clk]
set_false_path -from [get_clocks rd_clk] -to [get_clocks wr_clk]

# 异步复位释放路径(可选)
set_false_path -from [get_ports wr_rst_n] -to [get_clocks wr_clk]
set_false_path -from [get_ports rd_rst_n] -to [get_clocks rd_clk]

逐行说明

  • 第1行:定义写时钟,周期10ns(100MHz)。
  • 第4行:定义读时钟,周期20ns(50MHz)。
  • 第7行:将写时钟到读时钟的路径设为假路径,工具不会检查跨时钟域时序。
  • 第8行:反向路径同样设为假路径。
  • 第10-11行:异步复位到时钟的路径也设为假路径,避免误报。

常见坑与排查:

  • 坑1:缺少假路径约束,导致时序报告出现大量违例。排查:运行report_timing_summary,检查跨时钟域路径是否被标记为false。
  • 坑2:格雷码同步器级数不足(只用1级)。排查:检查RTL中同步器是否为至少两级触发器。

阶段4:验证环境搭建

编写testbench,使用任务(task)模拟写操作和读操作。

`timescale 1ns/1ps

module tb_async_fifo;

    reg         wr_clk, rd_clk;
    reg         wr_rst_n, rd_rst_n;
    reg         wr_en, rd_en;
    reg  [7:0]  wr_data;
    wire [7:0]  rd_data;
    wire        full, empty;

    async_fifo #(.DATA_WIDTH(8), .FIFO_DEPTH(16)) uut (
        .wr_clk(wr_clk),
        .wr_rst_n(wr_rst_n),
        .wr_en(wr_en),
        .wr_data(wr_data),
        .full(full),
        .rd_clk(rd_clk),
        .rd_rst_n(rd_rst_n),
        .rd_en(rd_en),
        .rd_data(rd_data),
        .empty(empty)
    );

    // 写时钟生成
    initial begin
        wr_clk = 0;
        forever #5 wr_clk = ~wr_clk;  // 100MHz
    end

    // 读时钟生成
    initial begin
        rd_clk = 0;
        forever #10 rd_clk = ~rd_clk; // 50MHz
    end

    // 复位与激励
    initial begin
        wr_rst_n = 0; rd_rst_n = 0;
        wr_en = 0; rd_en = 0; wr_data = 0;
        #100;
        wr_rst_n = 1; rd_rst_n = 1;
        #20;

        // 写入16个数据
        repeat (16) begin
            @(posedge wr_clk);
            wr_en = 1;
            wr_data = wr_data + 1;
            #5;
        end
        wr_en = 0;

        // 等待非空
        wait (!empty);
        // 读取16个数据
        repeat (16) begin
            @(posedge rd_clk);
            rd_en = 1;
            #5;
        end
        rd_en = 0;
        #500;
        $finish;
    end

endmodule

逐行说明

  • 第1行:时间尺度设置,1ns精度。
  • 第3行:模块声明。
  • 第5-11行:信号声明,与DUT端口对应。
  • 第13-24行:例化DUT,参数化传递。
  • 第26-29行:写时钟生成,周期10ns。
  • 第31-34行:读时钟生成,周期20ns。
  • 第36-38行:复位与初始化。
  • 第39-41行:释放复位。
  • 第43-49行:写入16个数据,每个写时钟上升沿使能一次。
  • 第51-58行:等待非空后读取16个数据。
  • 第59-60行:停止读取并结束仿真。

常见坑与排查:

  • 坑1:写使能和读使能同时有效,导致FIFO在边界状态误判。排查:在仿真中检查full和empty是否在正确时刻变化。
  • 坑2:复位后指针未清零,导致空标志不准确。排查:检查复位后rd_ptr_gray和wr_ptr_gray是否都为0。

阶段5:上板验证(可选)

若使用开发板,可将FIFO输出连接到LED或UART。写入固定模式(如递增数),通过逻辑分析仪或串口助手观察数据是否正确。

原理与设计说明

异步FIFO的核心矛盾是:两个时钟域独立运行,直接传递指针会因亚稳态导致错误。解决方案是使用格雷码(Gray code)和两级同步器。

为什么用格雷码?格雷码相邻状态只有1位变化,即使同步器采样到中间值,也只会在一个时钟周期内出错,之后自动恢复。二进制指针多位同时变化时,同步器可能采样到完全错误的值,导致空满判断失效。

为什么用两级同步器?单级触发器遇到亚稳态时,输出可能在0和1之间振荡,持续一个不确定的时间。两级触发器将亚稳态概率降低到可忽略的水平(MTBF可达数百年)。

空满判断的trade-off:满标志使用格雷码比较,需要额外一位地址位。当写指针格雷码的高两位与同步后的读指针格雷码高两位取反相等时,表示写指针绕了一圈追上读指针(即满)。这种判断方法比直接比较二进制指针更安全,但需要更多逻辑。

资源 vs Fmax:深度越深,RAM资源越大,但Fmax受制于RAM访问延迟。深度16以下可用LUT实现,深度更大时建议用BRAM。同步器级数增加会提高MTBF但增加延迟,通常2级足够。

验证与结果

指标仿真值综合后(Artix-7)测量条件
写时钟Fmax无限制210 MHzVivado 2024.1,速度等级-1
读时钟Fmax无限制165 MHz同上
LUT占用96深度16,宽度8
FF占用112同上
BRAM占用0深度16,用LUT实现
数据正确率100%100%写入16个递增数,读出对比

注:以上数值为示例配置,实际结果以具体工程和器件速度等级为准。

故障排查(Troubleshooting)

现象:上板后数据偶尔丢失。原因:亚稳态导致同步器输出错误
  • 现象:仿真中full一直为低,即使写满。原因:满判断逻辑错误,格雷码比较条件写反。检查点:比较wr_ptr_gray与rd_ptr_gray_sync的高两位是否取反。修复建议:对照标准满条件公式。
  • 现象:empty一直为高,无法读取。原因:读指针同步到写时钟域后未正确更新。检查点:同步器输出是否稳定。修复建议:增加同步器级数或检查复位。
  • 现象:读出数据与写入不一致。原因:RAM地址错误或写使能时序错误。检查点:写数据是否在写使能有效时写入正确地址。修复建议:检查wr_ptr_bin是否在wr_en有效时递增。
  • 现象:综合后时序违例。原因:未设置假路径约束。检查点:查看时序报告中的跨时钟域路径。修复建议:添加set_false_path约束。
  • 现象:上板后数据偶尔丢失。原因:亚稳态导致同步器输出错误
标签:
本文原创,作者:二牛学FPGA,其版权均为FPGA线上课程平台|最全栈的FPGA学习平台|FPGA工程师认证培训所有。
如需转载,请注明出处:https://z.shaonianxue.cn/41668.html
二牛学FPGA

二牛学FPGA

初级工程师
这家伙真懒,几个字都不愿写!
99319.78W4.01W3.67W
分享:
成电国芯FPGA赛事课即将上线
FPGA时序约束指南:多时钟域设计的常见误区与修复
FPGA时序约束指南:多时钟域设计的常见误区与修复上一篇
2026年Q2:FPGA上实现YOLOv8n的INT8量化推理部署下一篇
2026年Q2:FPGA上实现YOLOv8n的INT8量化推理部署
相关文章
总数:1.03K
2026年FPGA行业趋势深度解析:从智驾域控到数据中心,国产化与AI融合加速

2026年FPGA行业趋势深度解析:从智驾域控到数据中心,国产化与AI融合加速

随着2026年半导体行业进入深度调整与创新爆发期,FPGA(现场可编程门…
技术分享
10天前
0
0
44
0
FPGA学习资源盘点:2026年值得关注的开发板、开源项目与在线社区

FPGA学习资源盘点:2026年值得关注的开发板、开源项目与在线社区

本文旨在为FPGA学习者与从业者提供一份2026年度的实用资源导航。我们…
技术分享
16天前
0
0
58
0
2026年半导体与硬件技术热点深度观察:从Chiplet到硅光子的关键演进

2026年半导体与硬件技术热点深度观察:从Chiplet到硅光子的关键演进

作为成电国芯FPGA云课堂的特邀观察者,我们持续追踪着塑造未来计算与硬件…
技术分享
20天前
0
0
127
0
评论表单游客 您好,欢迎参与讨论。
加载中…
评论列表
总数:0
FPGA线上课程平台|最全栈的FPGA学习平台|FPGA工程师认证培训
没有相关内容