Matt Venn | 08cd6eb | 2020-11-16 12:01:14 +0100 | [diff] [blame] | 1 | `default_nettype none |
Tim Edwards | cd64af5 | 2020-08-07 11:11:58 -0400 | [diff] [blame] | 2 | // (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 | |
| 34 | module 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 : |
shalan | fd13eb5 | 2020-08-21 16:48:07 +0200 | [diff] [blame] | 58 | (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 Edwards | cd64af5 | 2020-08-07 11:11:58 -0400 | [diff] [blame] | 84 | |
| 85 | always @(posedge clock or posedge reset) begin |
shalan | fd13eb5 | 2020-08-21 16:48:07 +0200 | [diff] [blame] | 86 | 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 Edwards | cd64af5 | 2020-08-07 11:11:58 -0400 | [diff] [blame] | 92 | |
shalan | fd13eb5 | 2020-08-21 16:48:07 +0200 | [diff] [blame] | 93 | end else begin |
| 94 | oscbuf <= {oscbuf[1:0], osc}; |
Tim Edwards | cd64af5 | 2020-08-07 11:11:58 -0400 | [diff] [blame] | 95 | |
shalan | fd13eb5 | 2020-08-21 16:48:07 +0200 | [diff] [blame] | 96 | if (oscbuf[2] != oscbuf[1]) begin |
| 97 | count1 <= count0; |
| 98 | count0 <= 5'b00001; |
| 99 | prep <= {prep[1:0], 1'b1}; |
Tim Edwards | cd64af5 | 2020-08-07 11:11:58 -0400 | [diff] [blame] | 100 | |
shalan | fd13eb5 | 2020-08-21 16:48:07 +0200 | [diff] [blame] | 101 | if (prep == 3'b111) begin |
| 102 | if (sum > div) begin |
Tim Edwards | bb3cd69 | 2020-10-09 22:00:23 -0400 | [diff] [blame] | 103 | if (tval < 127) begin |
| 104 | tval <= tval + 1; |
| 105 | end |
shalan | fd13eb5 | 2020-08-21 16:48:07 +0200 | [diff] [blame] | 106 | end else if (sum < div) begin |
Tim Edwards | bb3cd69 | 2020-10-09 22:00:23 -0400 | [diff] [blame] | 107 | if (tval > 0) begin |
| 108 | tval <= tval - 1; |
| 109 | end |
shalan | fd13eb5 | 2020-08-21 16:48:07 +0200 | [diff] [blame] | 110 | end |
| 111 | end |
| 112 | end else begin |
| 113 | if (count0 != 5'b11111) begin |
| 114 | count0 <= count0 + 1; |
| 115 | end |
| 116 | end |
| 117 | end |
Tim Edwards | cd64af5 | 2020-08-07 11:11:58 -0400 | [diff] [blame] | 118 | end |
| 119 | |
| 120 | endmodule // digital_pll_controller |
Tim Edwards | 581068f | 2020-11-19 12:45:25 -0500 | [diff] [blame] | 121 | `default_nettype wire |