FPGA基础知识:深度剖析异步复位同步释放

在数字电路设计中,"异步复位,同步释放"是一种被广泛采用的、可靠的复位策略。它巧妙地结合了异步复位和同步复位的优点,有效规避了各自的缺点。

为什么要同步释放?

核心原因是为了避免在复位信号撤销(即"释放")时可能引发的亚稳态问题。
异步复位的优势与风险 :异步复位的好处是响应迅速,一旦复位信号有效,电路会立即进入复位状态,无需等待时钟边沿。然而,其风险恰恰在于"释放"阶段。如果复位信号的撤销(例如从低电平变为高电平)恰好发生在时钟有效边沿的建立时间(Setup Time)或保持时间(Hold Time)窗口内,目标寄存器就可能因违反时序要求而进入亚稳态

亚稳态的危害:处于亚稳态的寄存器输出既不是稳定的高电平也不是稳定的低电平,而是在一个中间状态震荡,并可能在一段时间后随机收敛到0或1。这个不确定的状态会传播到后续的逻辑电路中,导致系统状态混乱、状态机跑飞,甚至整个系统功能失效。

因此,"同步释放"的目的就是将复位信号的撤销动作与系统时钟对齐 ,确保所有寄存器都在一个确定的、安全的时钟边沿退出复位状态,从而彻底消除亚稳态风险。

"异步复位,同步释放"的原理可以概括为:复位生效时异步,复位撤销时同步。

异步生效 (Assert):当复位信号有效时(例如 rst_n 变为低电平),电路会立即被强制复位,这个过程不受时钟控制,保证了复位的即时性。

同步撤销 (De-assert):当复位信号无效时(例如 rst_n 变为高电平),电路并不会立即退出复位。相反,这个"释放"的动作会被目标时钟域的时钟信号采样,经过同步处理后,再驱动内部逻辑退出复位状态。

c 复制代码
module rst_sync (
    input      clk,          // 系统时钟
    input      async_rst_n,  // 外部异步复位信号,低电平有效
    output reg sys_rst_n     // 内部同步复位信号,低电平有效
);
    reg sync1, sync2; // 两级同步寄存器

    // 核心逻辑:异步复位,同步释放
    always @(posedge clk or negedge async_rst_n) begin
        if (!async_rst_n) begin
            // 异步复位路径:当外部复位有效时,两级寄存器立即清零
            sync1 <= 1'b0;
            sync2 <= 1'b0;
        end else begin
            // 同步释放路径:当外部复位无效时,在时钟驱动下传递'1'
            sync1 <= 1'b1;
            sync2 <= sync1;
        end
    end

    // 将第二级寄存器的输出作为最终的同步复位信号
    assign sys_rst_n = sync2;
将上述sys_rst_n连接到一个专用的BUFG(全局时钟缓冲器或复位分发模块)上。
endmodule

在系统设计中,直接将一个复位同步器输出的信号连接到所有模块,确实会导致扇出(Fanout)过大,进而引发驱动能力不足、信号延迟增加和复位偏移(Skew)严重等问题

为了解决上述问题,工业界的标准做法是构建一个分层的复位树。其核心思想是"分而治之",通过多级缓冲来平衡负载,确保复位信号能够均匀、同步地到达系统的每一个角落。

1.使用使用约束文件:你只需要在代码中保持简单的连接,然后在约束文件(.xdc, .sdc)中告诉工具:"这个信号的扇出不能超过 X"。

c 复制代码
// 顶层
wire rst_n_sys;
rst_sync u_sync(.clk(clk), .async_rst_n(rst_n_in), .sys_rst_n(rst_n_sys));

// 直接连接几十个模块,不用管扇出问题
genvar i;
generate
    for (i = 0; i < 50; i = i + 1) begin : gen_modules
        my_module u_mod (
            .clk(clk),
            .rst_n(rst_n_sys) // 工具会自动处理这里的扇出
        );
    end
endgenerate
# 告诉工具,rst_n_sys 的最大扇出不能超过 15
# 工具会自动计算,如果超过了,就会自动插入 Buffer
set_max_fanout 15 [get_nets rst_n_sys]

2.手动分层:如果不想依赖工具自动优化,或者为了代码的可读性,想显式地把复位信号分开,可以采用手动分组的方法。这比写复杂的"树"要简单得多。

c 复制代码
module top_system (
    input clk,
    input rst_n_in
);
    wire rst_n_root;
    
    // 1. 同步释放
    rst_sync u_sync (.clk(clk), .async_rst_n(rst_n_in), .sys_rst_n(rst_n_root));

    // 2. 手动分组:将 50 个模块分成 5 组,每组 10 个
    // 利用简单的 wire 赋值,综合工具通常会将其识别为需要隔离的 nets
    wire rst_n_group_0;
    wire rst_n_group_1;
    wire rst_n_group_2;
    wire rst_n_group_3;
    wire rst_n_group_4;

    // 这里的 assign 在物理综合时,工具可能会插入 Buffer,
    // 或者你可以显式实例化 BUFG/CLKBUF
    assign rst_n_group_0 = rst_n_root;
    assign rst_n_group_1 = rst_n_root;
    assign rst_n_group_2 = rst_n_root;
    assign rst_n_group_3 = rst_n_root;
    assign rst_n_group_4 = rst_n_root;

    // 3. 分配模块
    // 组 0
    my_module u_mod_0 (.rst_n(rst_n_group_0));
    my_module u_mod_1 (.rst_n(rst_n_group_0));
    // ... (组0的其他模块)

    // 组 1
    my_module u_mod_10 (.rst_n(rst_n_group_1));
    // ...
endmodule
相关推荐
坏孩子的诺亚方舟4 天前
FPGA系统架构设计实践15_高云Arora V系列时钟体系
fpga开发·系统架构
FPGA小徐4 天前
入门 CNN 结构全解析|从流程图理论到 FPGA Verilog 硬件实现(含习题带讲解)
fpga开发
FPGA小徐4 天前
FPGA 数字信号处理:并行 FIR 与串行滤波器设计原理、对比与完整 Verilog 实现
fpga开发
Saniffer_SH5 天前
【高清视频】Gen6 服务器还没到,Gen6 SSD 怎么测?Emily 现场演示三种测试环境
人工智能·驱动开发·测试工具·缓存·fpga开发·计算机外设·压力测试
zlinear数据采集卡5 天前
双核架构深度解析:ARM+FPGA如何让数据采集卡实现500Ksps高性能?
arm开发·fpga开发·架构
9527华安5 天前
FPGA实现GTH Transceivers Wizard传输2路视频,基于aurora 8b10b编解码架构,提供4套工程源码和技术支持
fpga开发·gth·aurora 8b10b·transceivers
FPGA小徐6 天前
FPGA 数字信号处理(二):并行 FIR 滤波器的 Verilog 全流程设计与实现
fpga开发
国科安芯6 天前
基于AS32S601ZIT2型抗辐照MCU的商业航天卫星姿态确定与控制系统研究
单片机·嵌入式硬件·安全·fpga开发·架构·risc-v
ALINX技术博客6 天前
【黑金云课堂】FPGA技术教程FPGA基础:I2C 总线通信技术
fpga开发·i2c
Hello-FPGA6 天前
Xilinx KU040 FPGA Camera Link 图像采集
c++·fpga开发