

Stephanie Doan (604981556)  
Rohan Surve  
CS M152A Lab 6  
24 January 2021

## Lab 2: Clock Design Methodology

# I. Introduction

In this lab, I explore sequential circuits through exercises focusing on counters and clocks based on the system clock. Each of the four clocks are implemented a distinct sub-module called by a top-level module `clock_gen`, which takes the input clock and reset wires and outputs all the registers of the clocks controlled by the four submodules that accomplish the following:

- Divide by  $2^n$  clock, counting by 2, 4, 8, and 16
- Even division clock by 28
- Odd division clock by 5
- Glitchy counter controlled by a strobe counter

# II. Design Tasks

## 1. Design Task: [divide by $2^n$ clock] Assign 4 1-bit wire to each bit from 4-bit counter

In the `clock_div_two` module, I implemented the 4-bit counter as a 4-bit register, with the least significant bit corresponding to a divide-by-2 clock, all the way to the most significant bit corresponding to a divide-by-16 clock. On every positive clock edge, 1 is added to this counter, which resets to `4'b0000` whenever the counter reaches its maximum value. This is essentially a divide by  $2^n$  clock because it acts only on every positive clock edge, taking a whole input clock cycle to flip the least significant bit, then twice as long to flip the second least significant bit, and onwards. On every change to this register, the value of each bit is extracted and stored in output registers `clk_div_2`, `clk_div_4`, `clk_div_8`, and, `clk_div_16`. The figure below shows this 4-bit counter as well as the output register each bit corresponds to.

| count[3]                              | count[2]                             | count[1]                             | count[0]                             |
|---------------------------------------|--------------------------------------|--------------------------------------|--------------------------------------|
| Divide by 16 clock                    | Divide by 8 clock                    | Divide by 4 clock                    | Divide by 2 clock                    |
| Flips bit at 1/16 frequency of clk_in | Flips bit at 1/8 frequency of clk_in | Flips bit at 1/4 frequency of clk_in | Flips bit at 1/2 frequency of clk_in |
| <code>clk_div_16</code>               | <code>clk_div_8</code>               | <code>clk_div_4</code>               | <code>clk_div_2</code>               |

Four-bit register `reg[3:0]` count as divide by  $2^n$  clock

To verify functionality of this 4-bit counter, I visualized the waveforms below showing the values of each output register corresponding to the bits of the counter, as well as the `clk_in`

(clock in). Notice that changes to each output register happens only at the positive edges of `clk_in`.



2. **Verification Task:** [even division clock] Divide by 32 clocks by flipping output clock on every counter overflow

In order to create a divide-by-32 clock, I extend the 4-bit counter in the last task to a 5-bit counter, so that when the least significant 4 bits overflow, the most significant bit flips, which is essentially dividing by 32. Similar to the implementation of the first task, every change to this counter (via an always block) had `clk_div_32` and `clk_div_16` read and store the first and second most significant bits respectively. The waveforms of this are visualized below, with `clk_div_32` flipping at half the frequency of the `clk_div_16` register as expected.



3. **Design Task:** [even division clock] Generate clock 28 times smaller by modifying when counter resets to 0

To extend my module written for the previous task, I created a new 4-bit counter that would reset to 0 whenever it held the value  $4'b1101 = 13$  in decimal. This is essentially resetting every 14 positive clock edges, equivalent to 28 total clock edges. I introduced more conditional logic to my code, explicitly checking the counter value to reset to 0 and flip the `clk_div_28` register.

```
else if (count_28 == 4'b1101)
begin
    count_28 <= 4'b0000;
    clk_div_28 <= ~clk_div_28;
end
```

As seen in the waveform below, `clk_div_8` flips at a frequency in between `clk_div_16` and `clk_div_28` as expected.



Divide-by-28 clock in comparison to divide-by-16 and divide-by-32

#### 4. Verification Task: [odd division clock] Verify waveform of 33% duty cycle clock

Here I create a clock whose signal is active for 1/3 of the time. Using if statements and counters as specified, I create a 2-bit counter that is incremented at every positive clock edge (though this could also be any edge), and checks if the value held is  $2'b10 = 2$  in decimal, essentially resetting to 0 every 3 positive clock edges. Every change to this counter makes the output register `clk_out` read from the most significant bit of the counter, which will read 1 exactly 1/3 of the time, since the counter cycles between holding the values  $2'b00$ ,  $2'b01$ , and  $2'b10$ .

```
module clock_33_duty(
    input wire clk_in,
    input wire rst,
    output reg clk_out
);
    reg[1:0] count = 2'b00;
    always @ (posedge clk_in)
    begin
        if (rst)
            count <= 2'b00;
        else if (count == 2'b10)
            count <= 2'b00;
        else
            count <= count + 1'b1;
    end

    always @ (count)
        clk_out <= count[1];
endmodule
```

The waveform is shown below with clk\_out active for 33% of the time.



## 5. Verification Task: [odd division clock] Rising edge vs falling edge triggered clock

In order to have the output signal change on the rising edge of the input clock, I essentially duplicated my code from the previous task only replacing the always block's sensitivity list

```
always @ (negedge clk_in)
```

so that updates to the counter happens on the rising edge instead. Below are the side by side waveforms of the clock from Task 4 (`clk_out_rising`) and the new one (`clk_out_falling`).



## 6. Verification Task: [odd division clock] Assign wire that takes logical OR of two 33% clocks

In order to take the logical OR of the two clocks from the previous tasks, I added a line to the always block triggered by changes to the counters, to assign to output `clk_out_or` the logical or of the values given to `clk_out_rising` and `clk_out_falling`.

```
clk_out_rising <= count_rising[1];
clk_out_falling <= count_falling[1];
clk_out_or <= count_rising[1] | count_falling[1];
```

The waveforms are shown below, with `clk_out_or` active wherever `clk_out_rising` or `clk_out_falling` is active.



`clk_out_or` as logical or of two 33% clocks

## 7. Design Task: [odd division clock] 50% duty cycle divide-by-5 clock

To implement a divide-by-5 clock, I modify the code written for task 4 to have the always block's sensitivity list include both the positive and negative edge, as the counter needs to be incremented on every edge, as we are now counting by 1s, not 2s. I declare a 3-bit counter `reg [2:0]` to hold 3 bits that count up to  $3'b100 = 4$  in decimal before being reset to 0, essentially counting 5 clock edges.

```
reg[2:0] count = 3'b000;
always @ (posedge clk_in or negedge clk_in)
begin
    if (rst)
        begin
            count <= 3'b000;
            clk_div_5 <= 1'b0;
        end
    else if (count == 3'b100)
        begin
            count <= 3'b000;
            clk_div_5 <= ~clk_div_5;
        end
    else
        count <= count + 1'b1;
end
```

The waveforms are shown below, with `clk_div_5` being flipped every 2.5 clock cycles of `clk_in`. The clock is indeed 50% duty as the output clock signal spends exactly half its time active.



## 8. Verification Task: [glitchy counter] Divide-by-100 clock with 1% duty cycle, 50% duty divide-by-200 clock

To implement the divide-by-100 clock, I initialized a 7-bit counter to store up to  $0b'1100011 = 99$  in decimal, as we would like to count up to 100 clock edges. In an always block that runs on the system clock's positive and negative clock edges, the counter is incremented by 1 until the value held is 99, in which the output `clk_div_100`, originally initialized to 0, is flipped to 1,

then flipped back to 0 in the very next clock edge. As shown in the waveforms below, this clock has 1% duty cycle because it spends 10 ns out of 1000 ns active.

To implement the second clock, I created a second always block also running on the positive and negative edges of the system clock, flipping its value whenever `clk_div_100` is 1 via an if statement. As seen in the waveform below, this output clock, `clk_div_200`, is 50% duty cycle because it spends 1000 ns of 2000 ns active. The clock is at 500KHz because each cycle of it takes 200 clock edges = 100 clock cycles. Since my test bench sets `clk_in` to be flipped every 10 ns, `clk_in` has a 20 ns clock cycle. Therefore, 100 cycles \* (20 ns / cycle) = 2000 ns per clock cycle, and 1 cycle / 2000 ns = 500000 Hz = 500 KHz.



Waveforms of divide-by-100 and divide-by-200 clocks

## 9. Design Task: [glitchy counter] 8-bit counter that goes up by 2 on every positive edge and subtracts 5 by every strobe of divide-by-4 strobe

To implement this counter, I initialized a 2-bit register `reg[2:0]` strobe to count up to  $2^3 = 3$  in decimal for 4 strobes. In an always block that acts on every positive edge of the system clock, I add  $2^2 = 4$  to toggle\_counter, effectively counting up by 2, and add  $1^2 = 1$  to strobe. If strobe holds  $2^2 = 4$  in decimal, then  $3^2 - 4 = 5$  in decimal is subtracted from toggle\_counter and strobe is reset to 0. As shown in the waveform below, the sequence of values held by the counter is  $0 \rightarrow 2 \rightarrow 4 \rightarrow 6 \rightarrow 1 \rightarrow 3 \rightarrow 5 \rightarrow 7 \rightarrow 2 \rightarrow 4 \rightarrow 6 \rightarrow 8 \rightarrow 3 \rightarrow 5 \rightarrow 7 \rightarrow 9 \rightarrow 4 \rightarrow \dots$  as expected.



Waveform for `toggle_counter`

### **III. Clock Generator Module**

With the functionalities and lessons learned from these tasks, I wrote the top level module `clock_gen` to call each of the four submodules, taking `clk_in` and `rst` as inputs set by the written testbench module that is later explained. These are passed to the submodules along with output registers specific to each.

- clock\_div\_two

This submodule has an always block running on the positive edge of `clk_in`, adding 1 to the 4-bit register that is the main counter essentially incremented every 2 edges, or 1 clock cycle. In another always block running on any changes to this counter, the counter's bits are written to the 4 output registers representing the divide-by-[2|4|8|16] clock.

- `clock_div_twenty_eight`

This submodule has an always block also running on the positive edge of `clk_in`, adding 1 to the 4-bit counter, which is reset to 0 when it holds 13, equivalently counting 14 clock cycles or 28 clock edges. Each reset is accompanied by the flipping of the register `clk_div_28`.

- clock div five

This submodule has an always block running on both positive and negative edges of `clk_in`, adding 1 to the 3-bit register that is reset to 0 when it holds 4, equivalently counting 5 clock edges as we are now counting by 1 instead of 2s. Output `clk_div_5` is also flipped on each reset.

- `clock_strobe`

This submodule has an always block running on positive clock edges, adding 2 directly to the `toggle_counter` output register, as well as incrementing the 2-bit register representing the strobe. The `toggle_counter` is decremented by 5 whenever `strobe` reaches 3 which occurs every 4 clock cycles.

The waveforms from the top module is shown below:



In addition, the synthesized RTL schematic is shown below.



RTL Schematic for `clock_gen`

## IV. Testing

The testing procedure was simple, with a short testbench module that stayed constant throughout the development process. It essentially flipped the value of input register `clk_in` every 5 ns, while passing the necessary inputs and outputs to the top level `clock_gen` module.

| clock_gen Project Status (02/06/2021 - 00:01:08) |                                 |                              |                                     |
|--------------------------------------------------|---------------------------------|------------------------------|-------------------------------------|
| <b>Project File:</b>                             | lab2.xise                       | <b>Parser Errors:</b>        | No Errors                           |
| <b>Module Name:</b>                              | clock_gen                       | <b>Implementation State:</b> | Placed and Routed                   |
| <b>Target Device:</b>                            | xc6slx16-3csg324                | <b>• Errors:</b>             | No Errors                           |
| <b>Product Version:</b>                          | ISE 14.7                        | <b>• Warnings:</b>           | No Warnings                         |
| <b>Design Goal:</b>                              | Balanced                        | <b>• Routing Results:</b>    | All Signals Completely Routed       |
| <b>Design Strategy:</b>                          | Xilinx Default (unlocked)       | <b>• Timing Constraints:</b> | All Constraints Met                 |
| <b>Environment:</b>                              | <a href="#">System Settings</a> | <b>• Final Timing Score:</b> | 0 ( <a href="#">Timing Report</a> ) |

| Device Utilization Summary [1]                                  |      |           |                     |
|-----------------------------------------------------------------|------|-----------|---------------------|
| Slice Logic Utilization                                         | Used | Available | Utilization Note(s) |
| Number of Slice Registers                                       | 19   | 18,224    | 1%                  |
| Number used as Flip Flops                                       | 19   |           |                     |
| Number used as Latches                                          | 0    |           |                     |
| Number used as Latch-thrus                                      | 0    |           |                     |
| Number used as AND/OR logics                                    | 0    |           |                     |
| Number of Slice LUTs                                            | 16   | 9,112     | 1%                  |
| Number used as logic                                            | 16   | 9,112     | 1%                  |
| Number using O6 output only                                     | 11   |           |                     |
| Number using O5 output only                                     | 0    |           |                     |
| Number using O5 and O6                                          | 5    |           |                     |
| Number used as ROM                                              | 0    |           |                     |
| Number used as Memory                                           | 0    | 2,176     | 0%                  |
| Number of occupied Slices                                       | 7    | 2,278     | 1%                  |
| Number of MUXCYs used                                           | 0    | 4,556     | 0%                  |
| Number of LUT Flip Flop pairs used                              | 16   |           |                     |
| Number with an unused Flip Flop                                 | 2    | 16        | 12%                 |
| Number with an unused LUT                                       | 0    | 16        | 0%                  |
| Number of fully used LUT-FF pairs                               | 14   | 16        | 87%                 |
| Number of unique control sets                                   | 2    |           |                     |
| Number of slice register sites lost to control set restrictions | 5    | 18,224    | 1%                  |
| Number of bonded IOBs                                           | 16   | 232       | 6%                  |
| Number of RAMB16BWERS                                           | 0    | 32        | 0%                  |
| Number of RAMB8BWERS                                            | 0    | 64        | 0%                  |
| Number of BUFI02/BUFI02_2CLKs                                   | 0    | 32        | 0%                  |
| Number of BUFI02FB/BUFI02FB_2CLKs                               | 0    | 32        | 0%                  |
| Number of BUFG/BUFGMUXs                                         | 1    | 16        | 6%                  |
| Number used as BUFGs                                            | 1    |           |                     |

|                                       |      |     |    |  |
|---------------------------------------|------|-----|----|--|
| Number used as BUFGMUX                | 0    |     |    |  |
| Number of DCM/DCM_CLKGENs             | 0    | 4   | 0% |  |
| Number of ILOGIC2/ISERDES2s           | 0    | 248 | 0% |  |
| Number of IODELAY2/IODRP2/IODRP2_MCBs | 0    | 248 | 0% |  |
| Number of OLOGIC2/OSERDES2s           | 0    | 248 | 0% |  |
| Number of BSCANs                      | 0    | 4   | 0% |  |
| Number of BUFHs                       | 0    | 128 | 0% |  |
| Number of BUFPPLls                    | 0    | 8   | 0% |  |
| Number of BUFPPLL_MCBs                | 0    | 4   | 0% |  |
| Number of DSP48A1s                    | 0    | 32  | 0% |  |
| Number of ICAPs                       | 0    | 1   | 0% |  |
| Number of MCBs                        | 0    | 2   | 0% |  |
| Number of PCILOGICSEs                 | 0    | 2   | 0% |  |
| Number of PLL_ADVs                    | 0    | 2   | 0% |  |
| Number of PMVs                        | 0    | 1   | 0% |  |
| Number of STARTUPs                    | 0    | 1   | 0% |  |
| Number of SUSPEND_SYNCs               | 0    | 1   | 0% |  |
| Average Fanout of Non-Clock Nets      | 4.73 |     |    |  |

| Performance Summary [I]    |                                               |                     |                               |  |  |
|----------------------------|-----------------------------------------------|---------------------|-------------------------------|--|--|
| <b>Final Timing Score:</b> | 0 (Setup: 0, Hold: 0)                         | <b>Pinout Data:</b> | <a href="#">Pinout Report</a> |  |  |
| <b>Routing Results:</b>    | <a href="#">All Signals Completely Routed</a> | <b>Clock Data:</b>  | <a href="#">Clock Report</a>  |  |  |
| <b>Timing Constraints:</b> | <a href="#">All Constraints Met</a>           |                     |                               |  |  |

| Detailed Reports [I]                          |         |                         |        |          |                                 |
|-----------------------------------------------|---------|-------------------------|--------|----------|---------------------------------|
| Report Name                                   | Status  | Generated               | Errors | Warnings | Infos                           |
| <a href="#">Synthesis Report</a>              | Current | Fri Feb 5 20:08:03 2021 | 0      | 0        | <a href="#">4 Infos (0 new)</a> |
| <a href="#">Translation Report</a>            | Current | Sat Feb 6 00:00:39 2021 | 0      | 0        | 0                               |
| <a href="#">Map Report</a>                    | Current | Sat Feb 6 00:00:52 2021 | 0      | 0        | <a href="#">6 Infos (6 new)</a> |
| <a href="#">Place and Route Report</a>        | Current | Sat Feb 6 00:01:02 2021 | 0      | 0        | <a href="#">3 Infos (3 new)</a> |
| Power Report                                  |         |                         |        |          |                                 |
| <a href="#">Post-PAR Static Timing Report</a> | Current | Sat Feb 6 00:01:08 2021 | 0      | 0        | <a href="#">4 Infos (4 new)</a> |
| Bitgen Report                                 |         |                         |        |          |                                 |

| Secondary Reports [I]              |         |                         |
|------------------------------------|---------|-------------------------|
| Report Name                        | Status  | Generated               |
| <a href="#">ISIM Simulator Log</a> | Current | Fri Feb 5 20:37:53 2021 |

**Date Generated:** 02/06/2021 - 00:01:08