# **Lab 3 - Counters and Shifters I Report**

106033233 資工大四 周聖諺 (Sheng-Yen Chou)

2022-03-22

## **Contents**

| Lab 3 - Counters and Shifters I Report                | 3  |
|-------------------------------------------------------|----|
| Lab 3 - Pre 1: 4-bit Synchronous Binary Up Counter    | 3  |
| Design Specification                                  | 3  |
| Design Implementation                                 | 3  |
| Lab 3 - Pre 2: 8-Cascaded Shift Registers             | 5  |
| Design Specification                                  | 5  |
| Design Implementation                                 | 5  |
| Lab 3 - 1: $1/2^{27}$ Frequency Divider               | 7  |
| Design Specification                                  | 7  |
| Design Implementation                                 | 7  |
| Lab 3 - 2: 1Hz Count-for-50M Frequency Divider        | 8  |
| Design Specification                                  | 8  |
| Design Implementation                                 | 9  |
| Lab 3 - 3: 1Hz 4-bit Synchronous Binary Up Counter    | 10 |
| Design Specification                                  | 10 |
| Design Implementation                                 | 11 |
| Lab 3 - 4: 1 Hz 8-Cascaded Shift Registers            | 12 |
| Design Specification                                  | 12 |
| Design Implementation                                 | 13 |
| Lab 3 - 5: 1 Hz 10 Digit Marquee on 7-Segment Display | 14 |
|                                                       | 14 |
| Design Implementation                                 | 15 |

### **Lab 3 - Counters and Shifters I Report**

106033233 資工大四 周聖諺 (Sheng-Yen Chou)

### Lab 3 - Pre 1: 4-bit Synchronous Binary Up Counter

#### **Design Specification**

Source Code

#### **4-bit Synchronous Binary Up Counter**

Input: rst, clk

Output [3:0]q

#### **Design Implementation**

To implement the binary up counter, I use a variable  $q_i$  to count from 0 to 15. Whenever the output of the counter  $q_i$  changes, the variable  $q_i$  in should be changed to  $q_i$ . In addition, when the circuit detects the raise of the clock, the output of the counter will change to the variable  $q_i$ . On the other hand, if the reset switch to 0 or the counter hit the upper limit(15),  $q_i$  will be reset to 0.

```
1 'define BCD_COUNTER_BITS 4
3 module binary_up_counter(
4
       q,
5
       clk,
6
       rst
7
       );
8
       output [`BCD_COUNTER_BITS-1:0]q;
9
10
       input clk;
       input rst;
11
12
       reg [`BCD_COUNTER_BITS-1:0]q;
13
14
       reg [`BCD_COUNTER_BITS-1:0]q_in;
15
16
       always@(q)
17
       begin
            q_in <= q + `BCD_COUNTER_BITS'd1;</pre>
18
```

```
19
        end
20
21
        always@(posedge clk or negedge rst)
22
        begin
23
             if(~rst)
24
             begin
25
                 q <= `BCD_COUNTER_BITS'd0;</pre>
26
27
             else
28
             begin
29
                 q <= q_in;
             end
31
        end
    endmodule
32
```

## Lab 3 - Pre 1: Binary Up Counter



Figure 1: Lab 3-Pre1 Logic Diagram

#### **RTL Simulation**



Figure 2: Lab 2-Pre1 RTL Simulation

#### Lab 3 - Pre 2: 8-Cascaded Shift Registers

#### **Design Specification**

Source Code

#### D Flip Flop

Input d, clk, rst

Output q

#### **8-Cascaded Shift Registers**

Input: rst, clk

Output [7:0]q

#### **Design Implementation**

#### D Flip Flop

I use a variable d to store the input and q as the output of the flip flop. Whenever the clock is raised, the output variable q will be updated with the input variable d. In addition, when the reset is triggered, the output q will be 0.

#### **8-Cascaded Shift Registers**

I use a variable q with 8 entries to store the value. Whenever the clock is raised, it will shift the value in the previous register to the next one and the last one will be shift to the first register.

```
1
   `define BIT_WIDTH 8
2
3 module shifter(
4
       q,
       clk,
5
6
       rst
       );
8
9
       output [`BIT_WIDTH-1:0]q;
10
       input clk;
11
       input rst;
12
13
       reg [`BIT_WIDTH-1:0]q;
14
15
       always@(posedge clk or negedge rst)
16
       begin
```

```
if(~rst)
17
18
            begin
19
                 q <= `BIT_WIDTH'b01010101;</pre>
20
             end
21
            else
22
            begin
23
                 q[0] <= q[7];
24
                 q[1] <= q[0];
25
                 q[2] <= q[1];
                 q[3] <= q[2];
26
                 q[4] <= q[3];
27
28
                 q[5] <= q[4];
29
                 q[6] <= q[5];
                 q[7] <= q[6];
31
             end
32
        end
33 endmodule
```

Lab 3 - Pre 2: 8-Cascaded Shift Register



Figure 3: Lab 3-Pre2 Logic Diagram

#### **RTL Simulation**



Figure 4: Lab 2-Pre2 RTL Simulation

## Lab 3 - 1: $1/2^{27}$ Frequency Divider

#### **Design Specification**

Source Code

Input: rst, clk

Output: clk\_out

#### **Design Implementation**

To implement the frequency divider, I use a variable with 26 bits to count from 0 to  $1^{27}-1$ . and then back to 0 while the variable hits  $1^{27}-1$ .

Actually,  $1/2^{27}$  frequency divider can also be implemented by cascading 26 D-type flip flop. Whenever we cascade one flip flop, the frequency can be divided by 2.

```
`define FREQ_DIV_BITS 1
2
3 module lab3_1(
       clk_out,
4
5
       clk,
6
       rst
7);
8
9
       output clk_out;
       input clk;
10
11
       input rst;
12
13
       reg clk_out;
       reg [`FREQ_DIV_BITS-1:0]count_out;
14
       reg [`FREQ_DIV_BITS:0]count;
15
16
       always@(clk_out or count_out)
17
```

```
18
        begin
            count = {clk_out, count_out} + `FREQ_DIV_BITS'd1;
19
        end
21
22
        always@(posedge clk or negedge rst)
23
        begin
            if(~rst)
24
25
            begin
                 {clk_out, count_out} = `FREQ_DIV_BITS'd0;
26
27
28
            else
29
            begin
                 {clk_out, count_out} = count;
31
            end
32
        end
33
    endmodule
```

#### I/O Pin Assignment

```
I/O clk rst clk_out
LOC W5 V17 U16
```

#### **Block Diagram**

Lab 3 - 1: 1/2^27 Frequency Divider



Figure 5: Lab 3-1 Logic Diagram

#### Lab 3 - 2: 1Hz Count-for-50M Frequency Divider

#### **Design Specification**

Source Code

Input: rst, clk

Output: clk\_out

#### **Design Implementation**

To generate the 1 Hz clock, I use variables counter\_in and counter\_out to count from 0 to 50M. The counter\_in will store the value for the next time step and pass the value to the counter\_out when the clock raises. The reason why we need 50M counting is each counting is triggered only when the clock raises, so the circuit will count 1 more for every twice clock pulses.

```
'define FREQ_DIV_BITS 30
2 //`define FREQ_DIV_COUNT `FREQ_DIV_BITS'd1000000
    define FREQ_DIV_COUNT `FREQ_DIV_BITS'd50000000
4
5 module lab3_2(
6
       clk_out,
7 //
        counter,
       clk,
8
9
       rst
10
        );
11
       input clk;
12
13
       input rst;
14
       output clk_out;
15 //
         output counter;
16
17
        reg clk_in;
        reg clk_out;
18
19
        reg [`FREQ_DIV_BITS-1:0] counter_in;
20
        reg [`FREQ_DIV_BITS-1:0] counter_out;
21
22
        always@(counter_out or clk_out)
23
            if(counter_out < (`FREQ_DIV_COUNT - 1))</pre>
24
            begin
25
                counter_in <= counter_out + `FREQ_DIV_BITS'd1;</pre>
26
                clk_in <= clk_out;</pre>
27
            end
28
            else
29
            begin
                counter_in <= `FREQ_DIV_BITS'd0;</pre>
31
                clk_in <= ~clk_out;</pre>
32
            end
33
34
        always@(posedge clk or negedge rst)
35
            if(~rst)
            begin
```

```
counter_out <= `FREQ_DIV_BITS'd0;</pre>
37
38
                  clk_out <= 1'd0;
39
             end
40
             else
41
             begin
42
                  counter_out <= counter_in;</pre>
43
                  clk_out <= clk_in;</pre>
44
45 endmodule
```

#### I/O Pin Assignment

| I/O | I/O clk |     | clk_out |  |  |
|-----|---------|-----|---------|--|--|
| LOC | W5      | V17 | U16     |  |  |

#### **Block Diagram**

#### Lab 3 - 2: 1 Hz Frequency Divider



Figure 6: Lab 3-2 Logic Diagram

#### Lab 3 - 3: 1Hz 4-bit Synchronous Binary Up Counter

#### **Design Specification**

Source Code

#### **Frequency Divider**

Input: rst, clk

Output: clk\_out

#### **4-bit Synchronous Binary Up Counter**

Input: rst, clk

Output [3:0]q;

#### 1Hz 4-bit Synchronous Binary Up Counter

Input: rst, clk

Output [3:0]q;

#### **Design Implementation**

#### **Frequency Divider**

Same as Lab3-2.

#### **4-bit Synchronous Binary Up Counter**

Same as Lab3-pre1.

#### 1Hz 4-bit Synchronous Binary Up Counter

All we need to do is combine the 1 Hz frequency divider and the 4-bit binary up counter which triggered by the 1 Hz frequency divider.

#### **Verilog Code**

```
1 'define BCD_COUNTER_BITS 4
2 'define RST_HIGH 1'b1
3
4 module lab3_3(
5
     q,
6
      rst,
7
      clk
8
      );
9
      output [`BCD_COUNTER_BITS-1:0]q;
10
      input rst;
11
      input clk;
12
        reg [`BCD_COUNTER_BITS-1:0]q;
13 //
       wire DIV_CLK;
14
15
16
       frequency_divider U0(.clk(clk), .rst(rst), .clk_out(DIV_CLK));
17
       binary_up_counter U1(.clk(DIV_CLK), .rst(rst), .q(q));
18 endmodule
```

#### I/O Pin Assignment

| I/O | clk | rst | q[0] | q[1] | q[2] | q[3] |
|-----|-----|-----|------|------|------|------|
| LOC | W5  | V17 | U16  | E19  | U19  | V19  |

## Lab 3 - 3: 1 Hz Binary Up Counter



Figure 7: Lab 3-3 Logic Diagram

#### Lab 3 - 4: 1 Hz 8-Cascaded Shift Registers

#### **Design Specification**

Source Code

#### **Frequency Divider**

Input: rst, clk

Output: clk\_out

#### **8-Cascaded Shift Registers**

Input: rst, clk

Output [7:0]q

## 1 Hz 8-Cascaded Shift Registers

Input: rst, clk

Output [7:0]q

#### **Design Implementation**

#### **Frequency Divider**

Same as Lab3-2.

#### **8-Cascaded Shift Registers**

Same as Lab3-Pre2.

#### 1 Hz 8-Cascaded Shift Registers

We can achieve this functionality with the 1 Hz frequency divider as the clock trigger of the 8-cascaded shift registers.

#### **Verilog Code**

```
1 'define BIT_WIDTH 8
3 module shift_register(
4
5
       clk,
6
       rst
7
       );
8
9
       output [`BIT_WIDTH-1:0]q;
10
       input clk;
11
       input rst;
12
       wire [`BIT_WIDTH-1:0]q;
13
14
       wire CLK_OUT;
15
       frequency_divider U0(.clk(clk), .rst(rst), .clk_out(CLK_OUT));
16
       shifter U1(.clk(CLK_OUT), .rst(rst), .q(q));
17
18 endmodule
```

#### **Block Diagram**



Lab 3 - 4: 1 Hz 8-Cascaded Shift Register

Figure 8: Lab 3-4 Logic Diagram

#### I/O Pin Assignment

| I/O | clk | rst | q[0] | q[1] | q[2] | q[3] | q[4] | q[5] | q[6] | q[7] |
|-----|-----|-----|------|------|------|------|------|------|------|------|
| LOC | W5  | V17 | U16  | E19  | U19  | V19  | W18  | U15  | U14  | V14  |

## Lab 3 - 5: 1 Hz 10 Digit Marquee on 7-Segment Display

#### **Design Specification**

Source Code

#### **Frequency Divider**

Input: rst, clk

Output: clk\_out

## **10-Cascaded Shift Registers**

Input: rst, clk

Output [9:0]q

## **1Hz 10-Cascaded Shift Registers**

Input: rst, clk

Output [9:0]q

#### 7-Segment Display

```
Output [0:3]d_sel, [7:0]d_out
Input clk, rst, [7:0]d0, [7:0]d1, [7:0]d2, [7:0]d3
```

#### 1 Hz 10 Digit Marquee on 7-Segment Display

```
Output [0:3]d_sel, [7:0]d_out
Input clk, rst
```

#### **Design Implementation**

#### **Frequency Divider**

Same as Lab3-2.

#### **10-Cascaded Shift Registers**

Simply extend the 8-digit shift registers in the Lab3-Pre2 to 10-digit. Each digit contains 8 bits to store the pattern shown on the 7-segment display.

#### **1Hz 10-Cascaded Shift Registers**

Simply extend the 8-digit shift register in Lab3-4 to the 10-digit one.

```
'define SHIFTER WIDTH 10
2
  `define REG_SIZE 8
3
4 module shift_register(
5
       q0, q1, q2, q3, q4, q5, q6, q7, q8, q9,
6
       clk,
7
       rst
8
       );
9
10
      output [`REG_SIZE-1:0]q0;
      output [`REG_SIZE-1:0]q1;
11
      output [`REG_SIZE-1:0]q2;
12
13
     output [`REG_SIZE-1:0]q3;
14
     output [`REG_SIZE-1:0]q4;
15
      output [`REG_SIZE-1:0]q5;
16
       output [`REG_SIZE-1:0]q6;
17
       output [`REG_SIZE-1:0]q7;
18
       output [`REG_SIZE-1:0]q8;
       output [`REG_SIZE-1:0]q9;
19
20
       input clk;
```

```
21
       input rst;
22
23
       wire [`REG_SIZE-1:0]q0;
24
       wire [`REG_SIZE-1:0]q1;
25
       wire [`REG_SIZE-1:0]q2;
       wire [`REG_SIZE-1:0]q3;
27
       wire [`REG_SIZE-1:0]q4;
       wire [`REG_SIZE-1:0]q5;
28
       wire [`REG_SIZE-1:0]q6;
29
       wire [`REG_SIZE-1:0]q7;
30
       wire [`REG_SIZE-1:0]q8;
31
32
       wire [`REG_SIZE-1:0]q9;
       wire CLK_OUT;
34
       frequency_divider U0(.clk(clk), .rst(rst), .clk_out(CLK_OUT));
       shifter U1(.clk(CLK_OUT), .rst(rst), .q0(q0), .q1(q1), .q2(q2), .q3
           (q3), .q4(q4), .q5(q5), .q6(q6), .q7(q7), .q8(q8), .q9(q9));
37 endmodule
```

#### 7-Segment Display

Since we can only control one digit of the 7-segment display each time, I design a module that takes the 4-digit patterns as input and shows the 1 digit on the display when the clock raises. Whenever the clock raises, the module will switch the control d\_sel to different digit and shows the corresponding digit. Take an example, when the first clock raise occur, the module will set  $d_sel = 4' b1110$  and  $d_out = d0$ . As for second clock pulse, the module will output  $d_sel = 4' b1101$  and  $d_out = d1$  and so on.

```
`define DIGIT_N 4
2
   `define SEGMENT_N 8
   'define NONE_BITS `SEGMENT_N'b1111111_0
4 'define EMPTY_BITS `SEGMENT_N'b1111111_1
5
6 module display_7seg(
7
       d_sel,
8
       d_out,
9
       clk,
       rst,
11
       d0,
12
       d1,
13
       d2,
14
       d3
15
       );
16
17
       output [0:`DIGIT_N-1]d_sel;
18
       output [`SEGMENT_N-1:0]d_out;
19
       input clk;
20
       input rst;
       input [`SEGMENT_N-1:0]d0;
21
```

```
22
        input [`SEGMENT_N-1:0]d1;
23
        input [`SEGMENT_N-1:0]d2;
        input [`SEGMENT_N-1:0]d3;
24
25
        reg [0:`DIGIT_N-1]d_sel;
27
        reg [`SEGMENT_N-1:0]d_out;
28
        reg [0:`DIGIT_N-1]d_sel_temp;
29
        reg [`SEGMENT_N-1:0]d_out_temp;
        wire clk_out;
31
        segment7_frequency_divider U0(.clk(clk), .rst(rst), .clk_out(
            clk_out));
        always@(d_sel)
34
        begin
            case((d_sel << 1) | (d_sel >> (`DIGIT_N-1)))
                 DIGIT_N'b1110: d_out_temp <= d0;</pre>
                 `DIGIT_N'b1101: d_out_temp <= d1;
                 `DIGIT_N'b1011: d_out_temp <= d2;
40
                 `DIGIT_N'b0111: d_out_temp <= d3;
41
                 default: d_out_temp <= `NONE_BITS;</pre>
42
            endcase
43
            d_sel_temp <= (d_sel << 1) | (d_sel >> (`DIGIT_N-1));
44
        end
45
46
        always@(posedge clk_out or negedge rst)
47
        begin
            if(~rst)
48
49
            begin
                 d_out <= `EMPTY_BITS;</pre>
                 d_sel <= `DIGIT_N'b1110;</pre>
51
52
            end
53
            else
54
            begin
55
                 d_out <= d_out_temp;</pre>
                 d_sel <= d_sel_temp;</pre>
57
            end
58
        end
   endmodule
```

#### 1 Hz 10 Digit Marquee on 7-Segment Display

To implement the marquee, just simply combine the 1 Hz 10-cascaded shift registers and the 7-segment display. The 1 Hz 10-cascaded shift registers will store the patterns of each time step and shift the digits when the clock raises. Then I inject the digit  $0 \sim 4$  into the 7-segment display module to show the patterns.

```
1 'define DIGIT_N 4
2 'define SEGMENT_N 8
```

```
'define H_BITS `SEGMENT_N'b1001000_1
   'define N_BITS `SEGMENT_N'b1101010_1
'define T_BITS `SEGMENT_N'b1110000_1
6
   'define U_BITS `SEGMENT_N'b1000001_1
7
8
9 module lab3_5(
        d_sel,
10
11
        d_out,
12
        clk,
13
        rst
14
        );
15
        output [`DIGIT_N-1:0]d_sel;
16
        output [`SEGMENT_N-1:0]d_out;
17
        input clk;
18
19
        input rst;
20
        wire [`SEGMENT_N-1:0]Q0;
21
        wire [`SEGMENT_N-1:0]Q1;
22
23
        wire [`SEGMENT_N-1:0]Q2;
24
        wire [`SEGMENT_N-1:0]Q3;
25
        shift_register U0(.clk(clk), .rst(rst), .q0(Q0), .q1(Q1), .q2(Q2),
26
            .q3(Q3));
        display_7seg U1(.clk(clk), .rst(rst), .d0(Q0), .d1(Q1), .d2(Q2), .
27
           d3(Q3), .d_sel(d_sel), .d_out(d_out));
28 endmodule
```

## Lab 3 - 5: 1 Hz 10 Digit Marquee on 7-Segment Display



Figure 9: Lab 3-5 Logic Diagram

## I/O Pin Assignment

| I/O      | clk    | rst   | d_oı     | ut[0] d_c | ut[1] d_o | ut[2] d_out[ | 3] d_out[4] |
|----------|--------|-------|----------|-----------|-----------|--------------|-------------|
| LOC      | W5     | V17   | V7       | U7        | V5        | U5           | V8          |
| d_out[5] | d_out[ | [6] c | I_out[7] | d_sel[0]  | d_sel[1]  | d_sel[2]     | d_sel[3]    |
| U8       | W6     | V     | V7       | U2        | U4        | V4           | W4          |