blob: 0cd59b0b15cebe4b0a51d2bf17c908e8c23b9e93 [file] [log] [blame]
Matt Venn08cd6eb2020-11-16 12:01:14 +01001`default_nettype none
Tim Edwardscd64af52020-08-07 11:11:58 -04002// (True) digital PLL
3//
4// Output goes to a trimmable ring oscillator (see documentation).
5// Ring oscillator should be trimmable to above and below maximum
6// ranges of the input.
7//
8// Input "osc" comes from a fixed clock source (e.g., crystal oscillator
9// output).
10//
11// Input "div" is the target number of clock cycles per oscillator cycle.
12// e.g., if div == 8 then this is an 8X PLL.
13//
14// Clock "clock" is the PLL output being trimmed.
15// (NOTE: To be done: Pass-through enable)
16//
17// Algorithm:
18//
19// 1) Trim is done by thermometer code. Reset to the highest value
20// in case the fastest rate clock is too fast for the logic.
21//
22// 2) Count the number of contiguous 1s and 0s in "osc"
23// periods of the master clock. If the count maxes out, it does
24// not roll over.
25//
26// 3) Add the two counts together.
27//
28// 4) If the sum is less than div, then the clock is too slow, so
29// decrease the trim code. If the sum is greater than div, the
30// clock is too fast, so increase the trim code. If the sum
31// is equal to div, the the trim code does not change.
32//
33
34module digital_pll_controller(reset, clock, osc, div, trim);
35 input reset;
36 input clock;
37 input osc;
38 input [4:0] div;
39 output [25:0] trim; // Use ring_osc2x13, with 26 trim bits
40
41 wire [25:0] trim;
42 reg [2:0] oscbuf;
43 reg [2:0] prep;
44
45 reg [4:0] count0;
46 reg [4:0] count1;
47 reg [6:0] tval; // Includes 2 bits fractional
48 wire [4:0] tint; // Integer part of the above
49
50 wire [5:0] sum;
51
52 assign sum = count0 + count1;
53
54 // Integer to thermometer code (maybe there's an algorithmic way?)
55 assign tint = tval[6:2];
56 // |<--second-->|<-- first-->|
57 assign trim = (tint == 5'd0) ? 26'b0000000000000_0000000000000 :
shalanfd13eb52020-08-21 16:48:07 +020058 (tint == 5'd1) ? 26'b0000000000000_0000000000001 :
59 (tint == 5'd2) ? 26'b0000000000000_0000001000001 :
60 (tint == 5'd3) ? 26'b0000000000000_0010001000001 :
61 (tint == 5'd4) ? 26'b0000000000000_0010001001001 :
62 (tint == 5'd5) ? 26'b0000000000000_0010101001001 :
63 (tint == 5'd6) ? 26'b0000000000000_1010101001001 :
64 (tint == 5'd7) ? 26'b0000000000000_1010101101001 :
65 (tint == 5'd8) ? 26'b0000000000000_1010101101101 :
66 (tint == 5'd9) ? 26'b0000000000000_1011101101101 :
67 (tint == 5'd10) ? 26'b0000000000000_1011101111101 :
68 (tint == 5'd11) ? 26'b0000000000000_1111101111101 :
69 (tint == 5'd12) ? 26'b0000000000000_1111101111111 :
70 (tint == 5'd13) ? 26'b0000000000000_1111111111111 :
71 (tint == 5'd14) ? 26'b0000000000001_1111111111111 :
72 (tint == 5'd15) ? 26'b0000001000001_1111111111111 :
73 (tint == 5'd16) ? 26'b0010001000001_1111111111111 :
74 (tint == 5'd17) ? 26'b0010001001001_1111111111111 :
75 (tint == 5'd18) ? 26'b0010101001001_1111111111111 :
76 (tint == 5'd19) ? 26'b1010101001001_1111111111111 :
77 (tint == 5'd20) ? 26'b1010101101001_1111111111111 :
78 (tint == 5'd21) ? 26'b1010101101101_1111111111111 :
79 (tint == 5'd22) ? 26'b1011101101101_1111111111111 :
80 (tint == 5'd23) ? 26'b1011101111101_1111111111111 :
81 (tint == 5'd24) ? 26'b1111101111101_1111111111111 :
82 (tint == 5'd25) ? 26'b1111101111111_1111111111111 :
83 26'b1111111111111_1111111111111;
Tim Edwardscd64af52020-08-07 11:11:58 -040084
85 always @(posedge clock or posedge reset) begin
shalanfd13eb52020-08-21 16:48:07 +020086 if (reset == 1'b1) begin
87 tval <= 7'd0; // Note: trim[0] must be zero for startup to work.
88 oscbuf <= 3'd0;
89 prep <= 3'd0;
90 count0 <= 5'd0;
91 count1 <= 5'd0;
Tim Edwardscd64af52020-08-07 11:11:58 -040092
shalanfd13eb52020-08-21 16:48:07 +020093 end else begin
94 oscbuf <= {oscbuf[1:0], osc};
Tim Edwardscd64af52020-08-07 11:11:58 -040095
shalanfd13eb52020-08-21 16:48:07 +020096 if (oscbuf[2] != oscbuf[1]) begin
97 count1 <= count0;
98 count0 <= 5'b00001;
99 prep <= {prep[1:0], 1'b1};
Tim Edwardscd64af52020-08-07 11:11:58 -0400100
shalanfd13eb52020-08-21 16:48:07 +0200101 if (prep == 3'b111) begin
102 if (sum > div) begin
Tim Edwardsbb3cd692020-10-09 22:00:23 -0400103 if (tval < 127) begin
104 tval <= tval + 1;
105 end
shalanfd13eb52020-08-21 16:48:07 +0200106 end else if (sum < div) begin
Tim Edwardsbb3cd692020-10-09 22:00:23 -0400107 if (tval > 0) begin
108 tval <= tval - 1;
109 end
shalanfd13eb52020-08-21 16:48:07 +0200110 end
111 end
112 end else begin
113 if (count0 != 5'b11111) begin
114 count0 <= count0 + 1;
115 end
116 end
117 end
Tim Edwardscd64af52020-08-07 11:11:58 -0400118 end
119
120endmodule // digital_pll_controller
Tim Edwards581068f2020-11-19 12:45:25 -0500121`default_nettype wire