在同步电路中,输入数据需要与时钟满足setup time和hold time才能进行数据的正常传输,防止亚稳态。同样的道理,对于一个异步复位寄存器来说,同样异步复位信号同样需要和时钟满足recovery time和removal time 才能有效进行复位操作和复位释放操作,防止输出亚稳态。
撤销复位时,恢复到非复位状态的电平必须在时钟有效沿来临之前的一段时间到来,才能保证时钟能有效恢复到非复位状态,此段时间为recovery time,类似于同步时钟的setup time。
复位时,在时钟有效沿来临之后复位信号还需要保持复位状态的一段时间为去除时间removal time(去除时间)。类似同步时钟hold time。
总结:复位信号的操作(无论是复位还是释放)如果在时钟的recovery time和removal time之间的时间窗口内,都会产生亚稳态。
异步复位信号一般会持续相对较长的时间,保证寄存器能复位完成。但是由于复位信号是异步的,我们不知道它会在什么时刻被释放。如果异步复位信号撤销时,不满足recovery time和removal time时,可能会造成亚稳态。
很显然,并不是复位信号撤销时才可能出现亚稳态,复位信号一开始出现时,同样可能会存在亚稳态,只不过异步复位信号持续时间长,即使第一个周期出现了亚稳态(即第一个周期复位未成功),下一个时钟周期也不会再出现亚稳态了。
如果复位信号的变化只持续刚刚一个时钟周期,那么复位信号开始和撤销时都可能会出现亚稳态。
所谓异步复位同步释放(Synchronized Asynchronous Reset),就是在复位信号到来的时候不受时钟信号的同步,而是在复位信号释放的时候受到时钟信号的同步。
电路目的:防止复位信号撤除时产生亚稳态事件。
异步复位:显而易见,reset_n异步复位后,rst_n将拉低,即实现异步复位。
同步释放:这个是关键,看如何实现同步释放,即当复位信号reset_n撤除时,由于双缓冲电路(双寄存器)的作用,rst_n复位信号不会随着reset_n的撤除而撤除。
异步复位同步释放的原理图和代码如下:
//Synchronized Asynchronous Reset
//异步复位、同步释放:只适用于没有PLL的系统复位信号的设置
module sync_async_reset(clock,reset_n,rst_n);
input clock, reset_n;
output rst_n;
reg rst_nr1, rst_nr2;
always @(posedge clock or negedge reset_n) begin
if(!reset_n) begin
rst_nr1 <= 1'b0;
rst_nr2 <= 1'b0; //异步复位
end
else begin
rst_nr1 <= 1'b1;
rst_nr2 <= rst_nr1; //同步释放
end
end
assign rst_n = rst_nr2; //新的系统复位信号rst_n
//信号rst_n作为新的系统复位信号,后续可以用来直接“异步复位”
endmodule // sync_async_reset
当异步复位信号撤销时,用来同步的第二个寄存器输入的数据仍然是0,因此第二个寄存器是不会出现亚稳态的。第一个寄存器虽然可能出现亚稳态,即使其出现了亚稳态,这个亚稳态还需要通过第二个寄存器,这时第二个寄存器就起到了打拍的作用。也就是说,第二级的亚稳态只可能是上一级寄存器传播过来的,但是这个亚稳态经过第二级寄存器后,其大概率已经稳定下来了,就算稳定下来不为1,也就相当与复位信号多持续了一个周期而已。因此我们可以看出,使用异步复位同步释放时,异步复位撤销后是需要额外等待一个时钟周期的,否则会激励可能会被复位信号覆盖掉。
既解决了同步复位的资源消耗问题,又解决了异步复位的亚稳态问题,其根本思想是异步信号同步化;
新的系统复位信号可以直接用来“异步复位”;
只适用于不带PLL的系统。
很多FPGA设计中都会涉及到多个时钟,使用器件内部的PLL或者DLL会使得多个时钟的管理变得更加容易。当多个时钟都是用PLL/DLL产生时,它们的系统复位信号采用如下设计:
①先用FPGA的外部输入时钟clk将FPGA的输入复位信号rst_n做异步复位、同步释放处理(注意采用反逻辑),然后这个复位信号输入PLL,同时将clk也输入PLL。设计的初衷是在PLL输出有效时钟之前,系统的其他部分都保持复位状态。
②PLL的输出locked信号在PLL有效输出前一直时低电平,PLL输出稳定有效之后才会拉高该信号,所以这里就把前面提到的FPGA外部输入复位信号rst_n和这个locked信号相与作为整个系统的复位信号。
③这个复位信号也需要让合适的PLL输出时钟异步复位、同步释放处理一下。也就是说,为了达到可靠稳定的复位信号,该设计中对复位信号进行了两次处理,分别是在PLL输出前和PLL输出后。
源代码和RTL图如下所示:
module sys_ctrl(clk,rst_n,sys_rst_n,clk_25m,clk_100m);
input clk; //FPGA输入时钟信号
input rst_n; //系统硬件复位信号
output sys_rst_n; //系统复位信号,低有效
output clk_25m, clk_100m;
reg rst_r1, rst_r2;
reg sysrst_nr1, sysrst_nr2;
wire pll_rst;
wire locked; //输出信号有效标志,高表示PLL输出有效
wire sysrst_nr0;
wire sys_rst_n; //系统复位信号,低有效
//第一个异步复位同步释放
always @(posedge clk or negedge rst_n)
if(!rst_n) begin
rst_r1 <= 1'b1;
rst_r2 <= 1'b1; //注意这里是“反”逻辑
end
else begin
rst_r1 <= 1'b0;
rst_r2 <= rst_r1;
end
assign pll_rst = rst_r2; //PLL的复位信号
PLL_ctrl uut_PLL_ctrl(
.areset(pll_rst),
.inclk0(clk),
.c0(clk_25m),
.c1(clk_100m),
.locked(locked)
);
assign sysrst_nr0 = rst_n & locked; //"与"逻辑
//第二个异步复位同步释放
//注意这里的同步时钟为合适的PLL输出时钟
always @(posedge clk_100m or negedge sysrst_nr0)
if(!sysrst_nr0) begin
sysrst_nr1 <= 1'b0;
sysrst_nr2 <= 1'b0;
end
else begin
sysrst_nr1 <= 1'b1;
sysrst_nr2 <= sysrst_nr1;
end
//信号sys_rst_n作为新的系统复位信号,后续可用作异步复位
assign sys_rst_n = sysrst_nr2;
endmodule