Revised the clocking scheme in several ways: (1) Removed the output
clock divider from the PLL to the clocking module; (2) changed the
clock divider from a power-of-2 divider to an integer-N divider;
(3) added an enable to the PLL separate from the bypass, so that the
PLL can be started and have time to settle before being switched in.
(4) Made some attempts at glitch-free clock switching when changing
to and from the PLL, and when changing output divider values.
diff --git a/verilog/rtl/digital_pll.v b/verilog/rtl/digital_pll.v
index d4fe6ff..77cce42 100644
--- a/verilog/rtl/digital_pll.v
+++ b/verilog/rtl/digital_pll.v
@@ -9,35 +9,31 @@
vdd,
vss,
`endif
- resetb, extclk_sel, osc, clockc, clockp, clockd, div, sel, dco, ext_trim);
+ resetb, enable, osc, clockp, div, dco, ext_trim);
`ifdef LVS
input vdd;
input vss;
`endif
- input resetb; // Sense negative reset
- input extclk_sel; // External clock select (acts as 2nd reset)
- input osc; // Input oscillator to match
- input [4:0] div; // PLL feedback division ratio
- input [2:0] sel; // Core clock select
- input dco; // Run in DCO mode
+ input resetb; // Sense negative reset
+ input enable; // Enable PLL
+ input osc; // Input oscillator to match
+ input [4:0] div; // PLL feedback division ratio
+ input dco; // Run in DCO mode
input [25:0] ext_trim; // External trim for DCO mode
- output clockc; // Selected core clock output
output [1:0] clockp; // Two 90 degree clock phases
- output [3:0] clockd; // Divided clock (2, 4, 8, 16)
- wire [25:0] itrim; // Internally generated trim bits
- wire [25:0] otrim; // Trim bits applied to the ring oscillator
- wire [3:0] nint; // Internal divided down clocks
- wire resetbb; // Internal buffered negative sense reset
- wire creset; // Controller reset
- wire ireset; // Internal reset (external reset OR extclk_sel)
+ wire [25:0] itrim; // Internally generated trim bits
+ wire [25:0] otrim; // Trim bits applied to the ring oscillator
+ wire creset; // Controller reset
+ wire ireset; // Internal reset (external reset OR disable)
- assign ireset = ~resetb | extclk_sel;
+ assign ireset = ~resetb | ~enable;
// In DCO mode: Hold controller in reset and apply external trim value
+
assign itrim = (dco == 1'b0) ? otrim : ext_trim;
assign creset = (dco == 1'b0) ? ireset : 1'b1;
@@ -55,53 +51,4 @@
.trim(otrim)
);
- // Select core clock output
- assign clockc = (sel == 3'b000) ? clockp[0] :
- (sel == 3'b001) ? clockd[0] :
- (sel == 3'b010) ? clockd[1] :
- (sel == 3'b011) ? clockd[2] :
- clockd[3];
-
- // Derive internal negative-sense reset from the input negative-sense reset
-
- sky130_fd_sc_hd__buf_8 irbb (
- .A(resetb),
- .X(resetbb)
- );
-
- // Create divided down clocks. The inverted output only comes
- // with digital standard cells with inverted resets, so the
- // reset has to be inverted as well.
-
- sky130_fd_sc_hd__dfrbp_1 idiv2 (
- .CLK(clockp[1]),
- .D(clockd[0]),
- .Q(nint[0]),
- .Q_N(clockd[0]),
- .RESET_B(resetbb)
- );
-
- sky130_fd_sc_hd__dfrbp_1 idiv4 (
- .CLK(clockd[0]),
- .D(clockd[1]),
- .Q(nint[1]),
- .Q_N(clockd[1]),
- .RESET_B(resetbb)
- );
-
- sky130_fd_sc_hd__dfrbp_1 idiv8 (
- .CLK(clockd[1]),
- .D(clockd[2]),
- .Q(nint[2]),
- .Q_N(clockd[2]),
- .RESET_B(resetbb)
- );
-
- sky130_fd_sc_hd__dfrbp_1 idiv16 (
- .CLK(clockd[2]),
- .D(clockd[3]),
- .Q(nint[3]),
- .Q_N(clockd[3]),
- .RESET_B(resetbb)
- );
endmodule