

2. Registers

Gisselquist Technology, LLC

Daniel E. Gisselquist, Ph.D.





#### **Lesson Overview**



- Lesson Overview
- Registers
- Combinatorial
- Latches
- Flip Flops
- Blocking
- All in Parallel
- Feedback
- Blinky
- Broken Blinky
- Verilator
- **Parameters**
- Sim Result
- Trace Generation
- **GTKWave**
- Strobe
- PPS-I
- PPS-II
- Stretch
- Too Slow
- Dimmer
- Exercises

- What is a register (reg)?
- How do things change with time?
- Discover the system clock

#### **Objectives**

- Learn how to create combinatorial logic with registers
- Learn to create clocked (synchronous) logic
- Understand that registers can "remember" things
- Understand where your System Clock comes from
- Timing Checks, and why they are important



### Registers



Lesson Overview 

→ Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

Why use registers?

- Wires have no memory.
- Only registers can hold state (data)

Two basic types, both set with an always

Combinatorial: Like wires

```
always @(*)
A = B;
```

This form can be easier to read when the logic becomes complex

2. Synchronous: Only changes values on a clock

```
always @(posedge i_clk)
    A <= B;</pre>
```



### **Combinatorial Regs**



```
Lesson Overview
Registers
```

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

```
always @(*)
A = 9'h87;
```

- Registers can only be assigned in always blocks.
- Always blocks may consist of one statement, or
- Many statements between a begin and end pair



### **Combinatorial Regs**

```
W
```

```
Lesson Overview
Registers
```

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

**Exercises** 

#### This block

- Looks like software
- Acts like you would expect in a simulator
- Takes no time at all in hardware
   The hardware acts as if all statements were done at once

Only use "=" in a combinatorial always block



#### Latches



Lesson Overview Registers

Combinatorial

▶ Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

What happens here?

```
wire
input
                         i_S;
                 [7:0] i_V;
input
        wire
                 [7:0] o_R;
output
        reg
always @(*)
if (i_S)
        o_R = i_V:
```

This is called a latch

- It requires memory
- May do one thing in simulation, another in hardware
- Most FPGA's don't support latches
- Can have subtle timing problems in hardware

Avoid using latches!



### Last Assignment Wins



Lesson Overview Registers

Combinatorial

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

What happens here?

```
always @(*)
begin

o_R = 0;
if (i_S)

o_R = i_V;
end
```

No latch is inferred

- This is a very useful pattern!
- o\_R now has a default value
   This prevents a latch from being inferred
- No memory is required
- The last assignment gives o\_R its final value



### Flip Flops



```
Lesson Overview
Registers
Combinatorial
Latches
> Flip Flops
Blocking
All in Parallel
```

Blinky Broken Blinky

Feedback

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

**Exercises** 

- Any registers set within an always @(posedge i\_clk) block will transitions to their new values on the next clock edge only
  - Only a bonafide clock edge should be used
  - Do not transition on anything you create in logic
- ${\scriptscriptstyle \square}$  Note that we are using <= for assignment
  - This is a non-blocking assignment
  - Most, if not all, clocked register should be set with <=



## **Blocking**



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

➢ Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

This is a non-blocking assignment

```
always @(posedge i_clk)
    A <= A + 1'b1;</pre>
```

Blocking assignment

```
always @(posedge i_clk)

A = A + 1'b1;
```

- A blocking assignment's value may be referenced again before the clock edge
  - Creates the appearance of time passing within the block
  - It may also cause simulation-hardware mismatch
  - Use with caution
- In this case, both generate the same logic



## Non-Blocking



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

➢ Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

What value will be given for A?

- Assume it starts at zero
- What will it be after one clock tick?

- The assignment only takes place on the clock edge
- Last assignment wins
- A is set to 1, then 2 on the next clock, 3 on the clock after,
   etc.



## **Blocking**



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

➢ Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

Now what value will be given for A?

- Assume it starts at zero
- What will it be after one clock tick?

- Again, the assignment only takes place on the clock edge
- It appears as though it took several steps
- □ A is set to 6



## **Blocking**



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

➢ Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

What if something depends upon A in another block?

 ${\scriptscriptstyle\perp}$  Assume A=0 before the clock tick

- This result is simulation dependent!
- B may be set to 0, or it may be set to 6

Don't do this! Use <= within an always @(posedge i\_clk)



### Non-Blocking



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

➢ Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

Now what will B be set to?

 ${\scriptscriptstyle\perp}$  Assume A=0 before the clock tick

- A will be set to 1, and B will be set to 0
- On the next clock, A will be set to 2 and B to 1, etc.

Now simulation matches hardware



#### All in Parallel



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

- A design may contain multiple always blocks
- The hardware will execute all at once
- The simulator will execute one at a time

Rules: When using the simulator, ...

- Make sure your design can be synthesized
- Make sure it fits within your chosen device
  - This is not a simulator task
  - Requires using the synthesizer periodically
- Make sure it maintains an appropriate clock rate
  - We'll get to timing checks in a moment



#### **Feedback**



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

➢ Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

- Wires in a loop created circular logic
- Clocked registers in a loop creates feedback

Feedback is used commonly in control systems



### **Blinky**



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

➢ Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

Let's make an LED blink!

```
module blinky(i_clk, o_led);
    input wire i_clk;
    output wire o_led;

reg [26:0] counter;
    initial counter = 0;
    always @(posedge i_clk)
        counter <= counter + 1'b1;

assign o_led = counter[26];
endmodule</pre>
```

Feel free to synthesize and try this

- The LED should blink at a steady rate
- Rate is determined by the 26 above



### **Broken Blinky**



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

➢ Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

#### Here's a common beginner mistake

#### Don't make this mistake

- Notice that counter is only 1-bit
- This will blink at half the i\_clk frequency
- Result is typically way too fast to see any changes
- LED may glow dimly
- Need to slow it down



#### Verilator



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

∨ Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

**Exercises** 

Simulating our design (blinky) now requires a clock:

- We'll need to toggle the clock input for anything to happen
- This operation is so common, it deserves its own function, tick ()



#### **Verilator**



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

➢ Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

**Exercises** 

We can now simplify our main loop a touch

```
int main(int argc, char **argv) {
    int last_led;
    // .... Setup
    last_led = tb->o_led;
    for (int k=0; k<(1<<20); k++) {
         // Toggle the clock
         tick(tb);
         // Now let's print the LEDs value
         // anytime it changes
         if (last_led != tb->o_led) {
              printf("k_{\sqcup} = {\sqcup} \%7d, {\sqcup}", k);
              printf("led_{\sqcup}=_{\sqcup}%d\n", tb->o_led);
         } last_led = tb->o_led;
    }
```



#### Verilator



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

∨ Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

Can we simulate this? Not easily

- $\Box$  Counting to  $2^{27}$  may take seconds in hardware, but ...
- It's extreme slow in simulation.
- Let's speed blinky up—just for simulation
- We can do this by adjusting the width of the counter

We'll use a parameter to do this

```
\begin{array}{lll} \textbf{parameter} & \texttt{WIDTH}\!=\!27; \\ \textbf{reg} & \texttt{[WIDTH}\!-\!1\!:\!0] & \texttt{counter}; \\ // & \dots & \\ \textbf{assign} & \texttt{o\_led} = \texttt{counter}[\texttt{WIDTH}\!-\!1]; \end{array}
```



#### **Parameters**



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

▶ Parameters

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

Parameters are very powerful! They allow us to

- Reconfigure a design, after it's been written
- Examples:
  - ZipCPU cache sizes can be adjusted by parameters
  - Internal memory sizes, implement the divide instruction or not, specify the type of multiply
  - Default serial port speed, number of GPIO pins supported by a GPIO controller, and more

Verilator argument -GWIDTH=12 sets the WIDTH parameter to 12

```
% verilator -Wall -GWIDTH=12 -cc blinky.v
```



### Sim Result

```
W
```

```
Lesson Overview
Registers
Combinatorial
Latches
Flip Flops
Blocking
All in Parallel
Feedback
Blinky
Broken Blinky
Verilator
Parameters
Sim Result
Trace Generation
GTKWave
Strobe
PPS-I
PPS-II
Stretch
Too Slow
```

Dimmer Exercises

```
./blinky
       2047,
            led
       4095,
             led = 0
      6143, led = 1
       8191, led = 0
      10239, led
      12287, led
      14335, led
      16383, led = 0
      18431, led = 1
      20479, led = 0
k
  .... (Lines skipped for brevity)
```



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace

□ Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

This is easy. For more complex designs, we'll need a trace

That means writing to a trace file on every clock
 Steps

1. Add the --trace option to the Verilator command line

```
% verilator -Wall --trace -GWIDTH=12 \
    -cc blinky.v
```

2. Create a trace from your .cpp file





Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace

□ Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

Create the trace file within C++

```
// ...
int main(int argc, char **argv) {
        // ...
        // Generate a trace
        Verilated::traceEverOn(true);
        VerilatedVcdC* tfp = new VerilatedVcdC;
        tb->trace(tfp, 99);
        tfp ->open("blinkytrace.vcd");
        // ...
        for (int k=0; k<(1<<20); k++) {
                tick(++tickcount, tb, tfp);
                // ...
```





Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace

□ Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

3. Write trace data on every clock

```
tick(int tickcount, Vblinky *tb,
void
                           VerilatedVcdC* tfp) {
        tb->eval();
         if (tfp) // dump 2ns before the tick
                  tfp ->dump(tickcount * 10 - 2);
         tb \rightarrow i_c lk = 1;
        tb->eval();
         if (tfp) // Tick every 10ns
                  tfp ->dump(tickcount * 10);
         tb \rightarrow i_c lk = 0;
         tb->eval():
         if (tfp) { // Trailing edge dump
                  tfp ->dump(tickcount * 10 + 5);
                  tfp -> flush();
```



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace

□ Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

You'll need to add verilated\_vcd\_c.cpp to your g++ build command in order to support generating a trace as well

```
% export VINC=/usr/share/verilator/include
% g++ -I${VINC} -I obj_dir
${VINC}/verilated.cpp
${VINC}/verilated_vcd_c.cpp blinky.cpp
obj_dir/Vblinky__ALL.a -o blinky
```

Now, running blinky will generate a trace

```
% ./blinky # ...
```

You can view it with GTKwave

```
% gtkwave blinkytrace.vcd
```

# G GTKWave

Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

□ GTKWave

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises



This is how logic debugging is done

- The simulator trace shows you every register's value
- ...at every clock tick
- You can zoom in to find any bugs



#### **Strobe**



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

How is this design different from blinky?

```
module strobe(i_clk, o_led);
    input wire i_clk;
    output wire o_led;

reg [26:0] counter;

always @(posedge i_clk)
    counter <= counter + 1'b1;

assign o_led = &counter[26:24];
endmodule</pre>
```





Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

Exercises

Can we get an LED to blink once per second?

When  $CLOCK_RATE_HZ/2$  ticks have passed, the LED will toggle

- This structure is known as an integer clock divider
- It offers an exact division





Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

▶ PPS-II

Stretch

Too Slow

Dimmer

**Exercises** 

Can we get an LED to blink once per second?

```
parameter CLOCK_RATE_HZ = 100_000_000;
parameter [31:0] INCREMENT
            = (1 < < 30)/(CLOCK_RATE_HZ/4);
input
       wire i clk;
output
       wire    o_led;
        [31:0] counter;
reg
initial counter = 0;
always @(posedge i_clk)
        counter <= counter + INCREMENT:</pre>
assign o_led = counter[31];
```

W

Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

▶ PPS-II

Stretch

Too Slow

Dimmer

**Exercises** 

- After CLOCK\_RATE\_HZ clock edges, the counter will roll over
- The divide by four above, on both numerator and denominator, is just to keep this within 32-bit arithmetic

$$INCREMENT = \frac{2^{32}}{CLOCK\_RATE\_HZ}$$

- This is called a fractional clock divider
  - The division isn't exact
  - It's often good enough



#### Stretch

```
₩
```

```
Lesson Overview
 Registers
  Combinatorial
  Latches
Flip Flops
 Blocking
 All in Parallel
  Feedback
 Blinky
 Broken Blinky
 Verilator
 Parameters
 Sim Result
 Trace Generation
  GTKWave
 Strobe
  PPS-I
  PPS-II

    Stretch
    ■
    Stretch
    ■
    Stretch
    ■
    The stretch
    The stretch
    ■
    The stretch
    The stretch
    ■
    The stretch
    The stretch
 Too Slow
  Dimmer
```

Exercises

```
module stretch(i_clk, i_event, o_led);
       input wire i_clk, i_event;
       output wire o_led;
           [26:0] counter;
       reg
       always @(posedge i_clk)
        if (i_event)
               counter \leq 0:
        else if (! (&counter))
               counter <= counter + 1:
        assign o_led = !counter[26];
endmodule
```

FPGA signals are often too fast to see



### Stretch

```
W
```

```
Lesson Overview
Registers
 Combinatorial
 Latches
Flip Flops
Blocking
 All in Parallel
 Feedback
 Blinky
Broken Blinky
Verilator
 Parameters
 Sim Result
 Trace Generation
 GTKWave
 Strobe
 PPS-I
 PPS-II

    Stretch
    ■
    Stretch
    ■
    Stretch
    ■
    The stretch
    The stretch
    ■
    The stretch
    The stretch
    ■
    The stretch
    The stretch
Too Slow
```

Dimmer

Exercises

FPGA signals are often too fast to see

- This slows them down to eye speed
- Only works for a single event though
- Multiple events would overlap, and be no longer distinct



#### **Too Slow**

```
W
```

```
Lesson Overview
Registers
Combinatorial
Latches
Flip Flops
Blocking
All in Parallel
```

Blinky Broken Blinky

Feedback

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Dimmer

**Exercises** 

```
module tooslow(i_clk, o_led);
        input wire i_clk;
        output wire
                        o_led;
                                 NBITS = 1024;
        parameter
                [NBITS - 1:0]
        reg
                                 counter;
        always @(posedge i_clk)
                counter \le counter + 1;
        assign o_{led} = counter[NBITS - 1];
endmodule
```

This is guaranteed to fail a timing check

- It's now time to learn how to check timing
- This design should fail, for reasonable clock speeds



### **Too Slow**

Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II Stretch

➤ Too Slow

100 310

Dimmer

Exercises

Follow your chip vendor's instructions to do a timing check

- Use your system clock frequency
  - For now, that's the clock frequency coming into your board
  - We'll adjust it later
- Make sure this design fails
  - The carry chain takes time to propagate
  - Extra long carry chains take extra long
  - If the propagation doesn't complete before the next clock
     ... your design will fail (like this one)
- From now on, always check timing for a design
  - Before loading it onto a board
  - Every now and then while simulating



#### **Dimmer**



Lesson Overview Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

➢ Dimmer

Exercises

Can you tell me what this will do?

```
module dimmer(i_clk, o_led);
       input wire i_clk;
       output wire
                       o_led;
               [27:0]
                      counter:
       reg
       always @(posedge i_clk)
               counter <= counter + 1;
        assign o_led = (counter[7:0]
                        < counter[27:20]);
endmodule
```



#### **Exercises**



Lesson Overview

Registers

Combinatorial

Latches

Flip Flops

Blocking

All in Parallel

Feedback

Blinky

Broken Blinky

Verilator

**Parameters** 

Sim Result

Trace Generation

**GTKWave** 

Strobe

PPS-I

PPS-II

Stretch

Too Slow

Dimmer

- Implement blinky on your hardware
- Implement one of the two PPS designs
  - Using a stopwatch, verify the blink rate of 1Hz
  - Make the blinks shorter, but at the same frequency
- Verify that the 1024 bit tooslow counter will fail timing
- Implement the dimmer