原理分析
分频电路是将给定clk时钟信号频率降低为div_clk的电路,假设原时钟的频率为M Hz,分频后的时钟为N Hz,那么就称该分频电路为M/N分频。
如果M/N是奇数,实现该功能的电路就是奇数分频电路;
如果M/N是偶数,实现该功能的电路就是偶数分频电路。
频率和周期的对应关系:
由于频率f=1/T,因此二分频电路即M/N=(1/T1)/(1/T2) = T2/T1 = 2 ,即一个二分频后的时钟周期是原时钟周期的两倍。
同理,一个N分频后的时钟周期是原时钟的M/N倍。
偶数分频举例:
以两分频电路为例,由于周期为原先的两倍,那么只需要在clk每个上升沿到来时,div_clk翻转,就可以了。
以四分频电路为例,由于周期为原先的四倍,那么需要在clk每两个周期div_clk翻转一次。需要使用计数器来数clk过了几个周期。
同理,一个K分频电路,K为偶数,那么由于周期为原先的K倍,那么需要在clk每K/2个周期div_clk翻转一次。
奇数分频举例:
以三分频电路为例,周期为原先的三倍。
如果对占空比(高电平占整个周期的比例)没有要求,我们可以令out_clk在clk的一个周期为高,两个周期为低,如此反复。
如果占空比为一半,那么就是每1.5个周期翻转一次,无法通过检测上升和下降沿来翻转,那么应该怎么做?
如图所示,我们的目标是每1.5个周期翻转一次,那么可以这样做,得到一个占空比为50%的奇数分频。
- 构造out_clk1和out_clk2信号,高电平都是1个周期,低电平都是2个周期,两个信号的区别是相差半个周期的相位。
- assign out_clk = out_clk1 | out_clk2
- out_clk就是分频的结果
同理,任意奇数分频都可以用类似的思路实现。因为3 5 7 9 11…总是可以分解成3=1+2,5=2+3,7=3+4,9=4+5…
实现和仿真
偶数分频的电路RTL代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
| `timescale 1ns / 1ps
module divide_even ( rst_n, clk, div_clk, ); input rst_n; input clk; output reg div_clk; reg [5:0] cnt; parameter DIVIDE_NUM = 4; always@(posedge clk or negedge rst_n) begin if( ~rst_n ) begin cnt <= 0; div_clk <= 0; end else if( cnt == DIVIDE_NUM / 2 - 1 ) begin div_clk <= ~div_clk; cnt <= 0; end else begin cnt <= cnt + 1; end end endmodule
|
偶数分频的电路Testbench代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| `timescale 1ns / 1ps
module tb_divide_even(); reg clk; reg rst_n; wire div_clk; initial begin clk = 0; rst_n = 0; #48; rst_n = 1; #203; $stop; end divide_even divide_even_u0 ( .clk(clk), .rst_n(rst_n), .div_clk(div_clk) ); always #5 clk = ~clk; endmodule
|
偶数分频的电路仿真波形
占空比为50%的三分频电路RTL代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59
| `timescale 1ns / 1ps
module divide_3( clk, rst_n, div_clk ); input clk; input rst_n; output div_clk; reg out_clk1; reg out_clk2; reg [1:0] cnt1; reg [1:0] cnt2; always@(posedge clk or negedge rst_n) begin if(~rst_n) begin cnt1 <= 2'b0; out_clk1 <= 0; end else if(out_clk1 == 1) begin cnt1 <= 2'b0; out_clk1 <= ~out_clk1; end else if(out_clk1 == 0) begin if(cnt1 == 1) begin cnt1 <= 2'b0; out_clk1 <= ~out_clk1; end else begin cnt1 <= cnt1 + 1; end end end always@(negedge clk or negedge rst_n) begin if(~rst_n) begin cnt2 <= 2'b0; out_clk2 <= 0; end else if(out_clk2 == 1) begin cnt2 <= 2'b0; out_clk2 <= ~out_clk2; end else if(out_clk2 == 0) begin if(cnt2 == 1) begin cnt2 <= 2'b0; out_clk2 <= ~out_clk2; end else begin cnt2 <= cnt2 + 1; end end end assign div_clk = out_clk1 | out_clk2;
endmodule
|
占空比为50%的三分频电路Testbench代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
| `timescale 1ns / 1ps
module tb_divide_3(); reg clk,rst_n; wire div_clk; initial begin rst_n = 0; clk = 0; #48; rst_n = 1; #202; $stop; end divide_3 divide_3_u0 ( .clk(clk), .rst_n(rst_n), .div_clk(div_clk) ); always #5 clk = ~clk; endmodule
|
占空比为50%的三分频电路仿真波形
参考资料
- 正点原子逻辑设计指南
- B站 FPGA探索者 牛客Verilog刷题 奇数分频