微波EDA网,见证研发工程师的成长!
首页 > 研发问答 > 嵌入式设计讨论 > FPGA,CPLD和ASIC > 关于altera fpga 时钟切换(clk mux) 的讨论

关于altera fpga 时钟切换(clk mux) 的讨论

时间:10-02 整理:3721RD 点击:

首先我用fpga是 altera cyclone II:  里面有个时钟mux: 一个时钟来自 pll , 一个时钟来自tck (jtag)
// 需要不同时钟的debug_ram。
debug_ram
debug_ram_inst (.address (clk_sw ? jtag_addr : rom_addr),
                                .clken   (jtag_wr|~rom_rd_n            ),
                        .clock   (debug_ram_clk                ),
                        .data    (jtag_data                    ),
                        .wren    (jtag_wr                      ),
                        .q       (rom_data                     )
                                                               );

(1)如果直接用rtl作逻辑选择:assign debug_ram_clk = clk_sw ? tck : clk;  //使用此逻辑综合要报rom_addr hold violation 时序warning(估计是因为做时钟切换clk时钟延时太长,看时序报告rom_addr hold需要2点几纳钞)
  (2) 如果使用quartus ii 里的 ip, altclkctrl :clk_ctrl
u_clk_ctrl(.clkselect( clk_sw            ),
                                                                              .inclk0x  ( clk                  ),
                                                                              .inclk1x  ( tck                 ),
                                                                              .outclk   ( debug_ram_clk)
                                                                                                                  );
        会报如此错:Error: inclk[0] port of Clock Control Block "clk_ctrl:u_clk_ctrl|clk_ctrl_altclkctrl_8af:clk_ctrl_altclkctrl_8af_component|clkctrl1" is driven by PLLLL_inst|altpll:altpll_component|_clk0, but must be driven by a clock pin
                         Error: inclk[1] port of Clock Control Block "clk_ctrl:u_clk_ctrl|clk_ctrl_altclkctrl_8af:clk_ctrl_altclkctrl_8af_component|clkctrl1" is driven by altera_internal_jtag~TCKUTAP, but must be driven by a clock pin
         //不知道为什么会报错,还在查资料中,也请大神赐教。
(3)使用时钟mux代码如下:
module clk_mux#(parameter num_clk = 2      )
               (input  [num_clk-1:0] clk    ,
//input  [num_clk-1:0] clk_rst,
input  [num_clk-1:0] clk_sel,
output clk_o
                                   );
genvar i;
reg  [num_clk-1:0] ena_r0;
reg  [num_clk-1:0] ena_r1;
reg  [num_clk-1:0] ena_r2;
wire [num_clk-1:0] qualified_sel;
// A look-up-table (LUT) can glitch when multiple inputs
// change simultaneously. Use the keep attribute to
// insert a hard logic cell buffer and prevent
// the unrelated clocks from appearing on the same LUT.
wire [num_clk-1:0] gated_clks /* synthesis keep */;
wire [num_clk-1:0] tmp_mask[0:num_clk-1];
generate
  for (i=0; i<num_clk; i=i+1)
  begin : lp0
    assign tmp_mask = {num_clk{1'b1}} ^ (1 << i);
    assign qualified_sel = clk_sel & (~|(ena_r2 & tmp_mask));
    always @(posedge clk) begin
    ena_r0 <= qualified_sel;
    ena_r1 <= ena_r0;
  end
  always @(negedge clk) begin
  ena_r2 <= ena_r1;
  end
  assign gated_clks = clk & ena_r2;
end
endgenerate
// These will not exhibit simultaneous toggle by construction
assign clk_o = |gated_clks;
endmodule
电路如下:



再实例化:clk_mux u_clk_mux({tclk,clk},{clk_sw,~clk_sw},debug_ram_clk); // 则这样就不会报错
(4)还有一种方法是动态配置pll(有些altera fpga支持),可是有一个时钟不是来自pll,故不能使用此方法了
前三种方法,暂时只有(3)不报错





Clock Control Block (ALTCLKCTRL) Megafunction资料上有如此表格:





下有如此一段话:The global clock network allows a clock signal (or other global signals) to reach all parts of the chip with a similar amount of skew. The regional clock network allows a signal to reach one quadrant of the chip (though half of the chip can be reached by driving two quadrants). The external clock-out path represents the clock path from the outputs of the phase-locked loop (PLL) to the dedicated PLL_OUT pins. The ALTCLKCTRL megafunction also provides glitch-free implementation for multiple clock input signals.
好像:cyclone II 只支持(1)global clock network ,(2) Dedicated External Clock Out Path , 但是选择(2)又无法支持时钟的mux,如下图:



完全无语呀,限制真心多

纠正:其实(3)也会报rom_addr相对clk的hold violation的WARNING(原因是之类代码有点问题)

欢迎大家多多讨论,不知道大家一般在fpga中怎么做clk的mux呢。

第一种方法没有问题,也是Synopsys推荐的方法。关键是,你在时序约束里有没把这两个时钟设为不同的clock group?
如果需要动态切换则需要进一步约束,你这里应该用不着。

设了的,我是在altera quartus 中综合,没有使用dc综合,dc综合要求没那么多(asic想怎么走线,就怎么走啥,而fpga布线资源是固定的),不过dc也还是要注意切换时的glitch啥(对了,synosys 的dc会自动考虑切换时的glitch而加保护电路吗,如果使用三目运算符)。


你给我推荐的是时钟gating的文档吧,好像不是时钟切换吧

你看看第五个例子。
作为一个最简单的尝试,你试试在Fitter设置里把自动全局时钟关了,看看还有没有hold time的大违规。


example 5: 是假设TSTCLK 在切换过程中,它是关了的,并且假设clk1和clk2是同一个时钟,故才可以 no glitch switch。(原文:Assume that the test clock is not a free-running clock, and that it is quiet when you switch to/from it.)

时钟无glitch的切换就只需要保证,一个时钟关了之后,另外一个时钟再打开(当然它们的开关一定不能出现glitch):                                        (1)当然它们可以直接或起来(clk_out = clk1|clk2)
                                         (2) 或者在第一个时钟关了之后,切换成第二个时钟(改变clk_sw的值),然后再打开第二个时钟(clk_out =                                    
                                                                                                                                                        clk_sw ? clk1 : clk2)

其实我认为只要不需要切换完成后保证马上可以用,也不用考虑那么多,直接切换完成时钟后,复位不就行了。

其中(3)的时钟mux 代码就是保证在一个时钟完全无glitch的关好后(输出clk_out的或门一个输入),另一个时钟才会打开(输出clk_out的或门另一个输入)

另外我想问的是为什么要把自动全局时钟关掉?

经改了clk_mux代码后, 法(1)和法(3)相比,法(1)时序会更好一些,由于我不关心有没有latch(因为我时钟切换完成再复位),但是时序一直有hold violation

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top