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
相关推荐
通信小呆呆8 小时前
单端口RAM、伪双端口RAM、真双端口RAM:功能详解与应用选型指南
fpga开发
s090713621 小时前
【FPGA实战】基于Verilog的MCP2515 CAN控制器SPI驱动详解 | 附完整代码
fpga开发·硬件设计·can通信·mcp2515
szxinmai主板定制专家21 小时前
基于 ARM+FPGA 数据机床实时工业控制设计--以雕刻机为例
arm开发·人工智能·嵌入式硬件·fpga开发
XMAIPC_Robot1 天前
基于RK3588 ARM+FPGA电火花数控机床控制系统设计,兼顾ethercat软硬件实时
linux·arm开发·人工智能·嵌入式硬件·fpga开发
XMAIPC_Robot1 天前
基于 ARM+FPGA 数据机床控制系统设计--以雕刻机为例
arm开发·fpga开发
GateWorld1 天前
LCD显示技术完全指南:原理·制造·驱动·FPGA实现之点屏一
fpga开发·lcd显示·fpga点亮屏幕·minilvds·fpga点屏
風清掦2 天前
【STM32学习笔记-14】WDG看门狗 - 14.2 WWDG窗口看门狗
笔记·stm32·单片机·嵌入式硬件·学习·fpga开发
尤老师FPGA2 天前
HDMI数据的接收发送实验(十二)
fpga开发
坏孩子的诺亚方舟2 天前
FPGA神经网络数学基础0
人工智能·神经网络·线性代数·fpga开发
熠速2 天前
PolarBox高性能实时仿真系统
arm开发·fpga开发·嵌入式实时数据库·硬件在环半实物仿真