

#### 10. Adding a FIFO

Gisselquist Technology, LLC

Daniel E. Gisselquist, Ph.D.



#### **G** Lesson Overview

▷ Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read

Conclusion

Serial ports can easily get overloaded with information

- What if the receiver is faster than the transmitter?
  - Perhaps you are bridging two separate serial channels, and each channel has a different baud rate
- If the serial port feeds a CPU, the CPU might not be able to keep up

Let's build a FIFO to address these problems!

#### Objectives

Know how to build and verify a FIFO

### G Design Goal

Lesson Overview ▷ Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface **FIFO** Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

Let's build a design to buffer a line before transmit

- The design will first read a line of data
- Den Then write it out sometime later, either ...
  - After the design receives a newline, or alternatively
  - After the buffer fills
- We'll use a FIFO to hold the intermediate data

#### **G** What is a FIFO?



#### **G** What is a FIFO?



#### **G** What is a FIFO?



### **G** Basic FIFO Design

Lesson Overview Design Goal What is a FIFO? **Basic FIFO**  $\triangleright$  Design method **FIFO** Interface **FIFO** Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion



Let's spend a moment looking at the I/O ports of a FIFO

## **G** Basic FIFO Design

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design  $\triangleright$  method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion



On any i\_wr, we'll write i\_data to the FIFO On any i\_rd, we'll return o\_data from the FIFO

Not quite ...

- What if there's nothing in the FIFO, should the read succeed?
- What if the FIFO is full, should a write succeed?

### G FIFO Interface

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method  $\triangleright$  FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion



On any i\_wr && !o\_full, we'll write i\_data to memory On any i\_rd && !o\_empty, we'll read and return o\_data from memory

We can simplify this by defining:

assign w\_wr = i\_wr && !o\_full; assign w\_rd = i\_rd && !o\_empty;

# GT FIFO

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method  $\triangleright$  FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion



On any w\_wr, we'll write i\_data to our internal memory On any w\_rd, we'll read and return o\_rdata from memory

#### This is how we'll handle overflows

- It should work much like our i\_wb\_stb && !o\_wb\_stall lesson
- The surrounding context must handle any over- or underflows

### G FIFO Memory

```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
▷ FIFO Memory
Addresses
Formal Verification
FIFO Verification
Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
Sim Trace
Fixing the read
Conclusion
```

The two memory accesses constrain much of our logic

Writes to the FIFO memory

## G FIFO Memory

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface ▷ FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

Our memories constrain much of our logic

- Writes to the FIFO memory
- Reads from the FIFO memory

```
// Maintain a read pointer
initial rd_addr = 0;
always @(posedge i_clk)
if (w_rd) // Increment on any read
    rd_addr <= rd_addr + 1;
// Return the value from the FIFO
// found at the read address
always @(*)
```

```
o_data <= fifo_mem[rd_addr];</pre>
```

Did you notice that this read violates our memory rules? We'll need to come back to this in a bit.

#### **G** Address pointers

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory  $\triangleright$  Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

There's a trick to the addresses ...

- $\square$  For a memory of  $2^N$  elements . . .
- $\hfill\square$  With an N bit array index

... you can only use  $2^N - 1$  elements

 Remember, you need to be able to tell the difference between empty (wr\_addr == rd\_addr) and full

With an N+1 bit array index, you can use all  $2^N$  elements

```
parameter BW = 8; // Bits per element
parameter LGFLEN = 8; // Memory size of 2^8
// Define the memory
reg [(BW-1):0] mem [0:(1<<LGFLEN)-1];
// Give the pointers one extra bit
reg [LGFLEN:0] wr_addr, rd_addr;</pre>
```

#### **G** Address pointers

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory  $\triangleright$  Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

• With an N+1 bit array index, you can use all  $2^N$  elements

| reg [ | (BW-1):0] | mem $[0:(1 < < LGFLEN) - 1];$ |
|-------|-----------|-------------------------------|
| reg [ | LGFLEN:0] | wr_addr, rd_addr;             |

The number of items in the FIFO is the address difference

```
always @(*)
        o_fill = wr_addr - rd_addr;
```

We can now calculate our empty and full outputs

#### G Formal Checks

| Lesson Overview     |  |  |
|---------------------|--|--|
| Design Goal         |  |  |
| What is a FIFO?     |  |  |
| Basic FIFO Design   |  |  |
| method              |  |  |
| FIFO Interface      |  |  |
| FIFO Memory         |  |  |
| ▷ Addresses         |  |  |
| Formal Verification |  |  |
| FIFO Verification   |  |  |
| Cover               |  |  |
| Cover Lesson        |  |  |
| Line Capturer       |  |  |
| Using the FIFO      |  |  |
| Rx + FIFO           |  |  |
| Verifying the FSM   |  |  |
| Simulation          |  |  |
| Random Delay        |  |  |
| Sim Trace           |  |  |
| Fixing the read     |  |  |
| Conclusion          |  |  |
|                     |  |  |

You should get in the habit of writing formal properties as you write your code

Can you think of any appropriate properties from just these definitions?

#### G Formal Checks

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface **FIFO** Memory  $\triangleright$  Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

You should get in the habit of writing formal properties as you write your code

Can you think of any appropriate properties from just these definitions?

Example: our fill can never exceed  $2^N$  elements, so let's keep the solver from trying such a fill

```
always @(*)
    assert(o_fill <= { 1'b1,
        {(LGFLEN){1'b0}} });</pre>
```

#### G Formal Checks

```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
FIFO Memory
\triangleright Addresses
Formal Verification
FIFO Verification
Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
Sim Trace
Fixing the read
Conclusion
```

You should get in the habit of writing formal properties as you write your code

- We might want to adjust our calculations of o\_fill, o\_empty, and o\_full later
- Writing an assertion now might help make sure any rewrite later doesn't fundamentally change anything

Hint: One of the exercises in this lesson is to rewrite this logic

## G Not there yet

```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
FIFO Memory
\triangleright Addresses
Formal Verification
FIFO Verification
Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
Sim Trace
Fixing the read
Conclusion
```

Our design isn't there yet

- We broke our memory rules
  - Our design will only work on some architectures

- This will work fine on any Xilinx FPGA
- This won't map to block RAM on an iCE40
- Our logic all depends upon w\_wr and w\_rd
  - These values depend upon o\_full and o\_empty
  - These depend upon an N bit subtract and comparison
  - This will limit the total speed of our design

Let's come back to these issues after we go through simulation

#### **G** Formal Verification

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface **FIFO** Memory Addresses Formal  $\triangleright$  Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

#### Review: Memory verification

- Let the solver pick a random address
- Follow the value at that address
  - Verify the design works as intended

for that address only

FIFO's offer a slight twist off of this strategy





```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
FIFO Memory
Addresses
Formal Verification
    FIFO
\triangleright Verification
Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
Sim Trace
Fixing the read
Conclusion
```

```
We'll need two consecutive addresses
```

We'll also need two arbitrary values

Fixing the read Conclusion

```
Here's our basic state transitions:
Lesson Overview
Design Goal
                 always @(posedge i_clk)
What is a FIFO?
Basic FIFO Design
                 case(f_state)
method
                 2'h0: // This is the IDLE state
FIFO Interface
FIFO Memory
                            Addresses
                            // Our process starts when we write our
Formal Verification
                            // first value to the FIFO, at the
  FIFO
\triangleright Verification
                            // first of our chosen two addresses
Cover
                            if (w_wr && (wr_addr == f_first_addr)
Cover Lesson
Line Capturer
                                       && (i_data == f_first_data))
Using the FIFO
                            f_state <= 2'h1;
Rx + FIFO
Verifying the FSM
                 // ...
Simulation
                 endcase
Random Delay
Sim Trace
```

#### **FIFO** Verification

Here's our basic state transitions: Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method 2'h0: // ... FIFO Interface FIFO Memory Addresses Formal Verification FIFO  $\triangleright$  Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion // ... endcase

```
always @(posedge i_clk)
case(f_state)
2'h1: // If we read the first value out at
        // this stage, then abort our check
        if (w_rd && rd_addr == f_first_addr)
                f_state <= 2'h0;
        else if (w_wr)
        // Otherwise if we write the second
        // value then move to the next state.
        // If it 's the wrong value then abort
        // the check
        f_state <= (i_data == f_second_data)</pre>
                ? 2'h2 : 2'h0;
```

Conclusion

```
Here's our basic state transitions:
Lesson Overview
Design Goal
                  always @(posedge i_clk)
What is a FIFO?
Basic FIFO Design
                  case(f_state)
method
                  2'h0: // ...
FIFO Interface
                  2'h1: // ...
FIFO Memory
Addresses
                  2'h2: // Wait until we read the first value
Formal Verification
                             // back out of the FIFO
  FIFO
\triangleright Verification
                             if (i_rd && rd_addr == f_first_addr)
Cover
                                         // Then move forward by
Cover Lesson
Line Capturer
                                         // one state
Using the FIFO
                                         f_state <= 2'h3;
Rx + FIFO
Verifying the FSM
                  // ...
Simulation
                  endcase
Random Delay
Sim Trace
Fixing the read
```

Sim Trace

Fixing the read Conclusion

```
Here's our basic state transitions:
Lesson Overview
Design Goal
                   always @(posedge i_clk)
What is a FIFO?
Basic FIFO Design
                   case(f_state)
method
                  2'h0: // ...
FIFO Interface
                  2'h1: // ...
FIFO Memory
Addresses
                   2'h2: // ...
Formal Verification
                   2'h3: // Finally, return to idle when the
   FIFO
\triangleright Verification
                               // last item is read
Cover
                               if (i_rd) f_state <= 2'h0;</pre>
Cover Lesson
Line Capturer
                   endcase
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
```

#### Basic proof:

- If we are in state one,
  - The first address must point to something within the FIFO
    - The memory at that location must be our special value
  - The write address must point to the second special address
- If we are in state two,
  - Both the first and second addresses must point to valid locations within the FIFO
  - The values at both locations must match our special values

...

This is actually harder than it sounds

Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification FIFO  $\triangleright$  Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read

Conclusion

Lesson Overview

#### **G** FIFO Verification





It must be greater or equal to the read address It must be less than the write address

#### **G** FIFO Verification

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification FIFO  $\triangleright$  Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion



It must be greater or equal to the read address
 It must be less than the write address

This is actually harder than it sounds

• The pointers can wrap!

### **G** FIFO Verification

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface **FIFO** Memory Addresses Formal Verification FIFO  $\triangleright$  Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

How shall we determine if our special address is within the FIFO?



The pointers can wrap!

What then does greater or less than mean?

#### Solution:

Unwrap all the pointers by subtracting the read pointer

```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
FIFO Memory
Addresses
Formal Verification
    FIFO
\triangleright Verification
Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
Sim Trace
Fixing the read
```

Conclusion

```
To determine if f_first_addr is within the active addresses of the FIFO:
```

f\_distance\_to\_first can now be checked against the FIFO fill, to determine if the special address references a valid location within the FIFO

```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
FIFO Memory
Addresses
Formal Verification
    FIFO
\triangleright Verification
Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
Sim Trace
Fixing the read
```

Conclusion

To determine if f\_first\_addr is within the active addresses of the FIFO:

We'll need to do this to check both addresses

### **G** First state

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification FIFO  $\triangleright$  Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

Now we can create assertions for the first state

Don't forget, we must be waiting at the second address to write the second piece of data!

```
assert(wr_addr == f_second_addr);
```

#### end

We now need to repeat this for the other two states

#### G Second state

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification FIFO  $\triangleright$  Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

Here's the second state:

```
always Q(*)
if (f_state == 2'b10)
begin
        // First value must be in the FIFO
        assert(f_first_addr_in_fifo);
        assert(fifo_mem[f_first_addr]
                == f_first_data);
        // Second value too!
        assert(f_second_addr_in_fifo);
        assert(fifo_mem[f_second_addr]
                f_second_data);
        if (rd_addr == f_first_addr)
                assert(o_data == f_first_data);
end
```

31 / 61

### G Last state

end

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification FIFO  $\triangleright$  Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

```
Here's the third and last state
always Q(*)
if (f_state == 2'b11)
begin
        // Only the second value need be
        // in the FIFO
        assert(f_second_addr_in_fifo);
        assert(fifo_mem[f_second_addr]
                == f_second_data);
        // The output data must match our
        // second value until the next w_rd
        assert(o_data == f_second_data);
```

### **G** SymbiYosys

|                                                                                                                                                                                                                                                                                                                                                           | Symbirosys                                                |
|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------|
| Lesson Overview<br>Design Goal<br>What is a FIFO?<br>Basic FIFO Design<br>method<br>FIFO Interface<br>FIFO Memory<br>Addresses<br>Formal Verification<br>FIFO<br>▷ Verification<br>Cover<br>Cover Lesson<br>Line Capturer<br>Using the FIFO<br>Rx + FIFO<br>Verifying the FSM<br>Simulation<br>Random Delay<br>Sim Trace<br>Fixing the read<br>Conclusion | You should now be able to prove this design via induction |
|                                                                                                                                                                                                                                                                                                                                                           |                                                           |

## **G** SymbiYosys

| Lesson Overview     |  |
|---------------------|--|
| Design Goal         |  |
| What is a FIFO?     |  |
| Basic FIFO Design   |  |
| method              |  |
| FIFO Interface      |  |
| FIFO Memory         |  |
| Addresses           |  |
| Formal Verification |  |
| FIFO                |  |
| Verification        |  |
| Cover               |  |
| Cover Lesson        |  |
| Line Capturer       |  |
| Using the FIFO      |  |
| Rx + FIFO           |  |
| Verifying the FSM   |  |
| Simulation          |  |
| Random Delay        |  |
| Sim Trace           |  |
| Fixing the read     |  |
| Conclusion          |  |
|                     |  |

You should now be able to prove this design via induction

Does it pass?

# G Cover

```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
FIFO Memory
Addresses
Formal Verification
FIFO Verification
\triangleright Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
Sim Trace
Fixing the read
Conclusion
```

You should also create some cover properties. Here are some of mine:

```
initial f_was_full = 0;
always @(posedge i_clk)
if (o_full)
     f_was_full <= 1;</pre>
```

Of course, these will only pass in a reasonable time if the memory size is small

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover  $\triangleright$  Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

I was once burned by going through all the motions of formal verification, only to have the design fail in simulation

- The design was a data cache
- I had verified both interfaces
  - All the bus properties were met
  - The CPU could depend upon the resulting interface
- I covered the return from a request
  - The cache went to the bus to get the requested value
  - The values returned by the bus were properly placed into the cache
  - The core then returned the right value

The resulting code failed in practice

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover  $\triangleright$  Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

I was once burned by going through all the motions of formal verification, only to have the design fail in simulation

- The design was a data cache
- I had verified both interfaces
  - All the bus properties were met
  - The CPU could depend upon the resulting interface
- I covered the return from a request
  - The cache went to the bus to get the requested value
  - The values returned by the bus were properly placed into the cache
  - The core then returned the right value

The resulting code failed in practice

What went wrong?

#### Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface **FIFO** Memory Addresses Formal Verification **FIFO** Verification Cover $\triangleright$ Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace

Fixing the read

Conclusion

The problem?

- The cache never raised its ready line to indicate it was ready for the next request
- Once it became busy, it never returned to idle
- The CPU froze on its second memory access

```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
FIFO Memory
Addresses
Formal Verification
FIFO Verification
Cover
\triangleright Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
Sim Trace
Fixing the read
Conclusion
```

#### The problem?

- The cache never raised its ready line to indicate it was ready for the next request
- Once it became busy, it never returned to idle
- The CPU froze on its second memory access

Ever since this painful lesson, I've made a point to cover the return to an idle state following the covered state

 That's the reason for the f\_empty check in the cover statement below

## G Line Capturer

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method **FIFO** Interface **FIFO** Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson ▷ Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

We now have a FIFO, what shall we do with it?

Let's build a line capturer



## **G**<sup>T</sup> Using the FIFO

We now have a FIFO, what shall we do with it?

- Let's build a line capturer
- 1. When it receives a character from the serial port, it places it into the FIFO
- 2. Once either the line has reached (or past 80) characters, we'll dump the FIFO to the serial port transmitter
- 3. Likewise, on any new line, we'll dump the FIFO to the serial port transmitter.
  - Think of this like an old fashioned printer:
  - Once the print head starts moving from left to right, it moves across the page at a constant speed
  - You don't want to start the head moving until a whole line is available

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO  $\triangleright$  Rx + FIFO Verifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

#### Step one: the receiver must feed the FIFO

```
// Serial port Receiver
rxuart #(.CLOCKS_PER_BAUD(CLOCKS_PER_BAUD))
receiver(i_clk, i_uart_rx, rx_stb,
rx_data);
```

```
// Our synchronous FIFO
// Fed directly from the receiver
sfifo #(.BW(8), .LGFLEN(8))
fifo(i_clk, rx_stb, rx_data, fifo_full,
fifo_fill,
fifo_rd, tx_data, fifo_empty);
```

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO  $\triangleright$  Rx + FIFO Verifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

Step two: Build a basic FSM to control the FIFO reads

We'll use run\_tx to say we are currently transmitting
 line\_count captures the number of items left to write

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO  $\triangleright$  Rx + FIFO Verifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

Step two: Build a basic FSM to control the FIFO reads

We'll use run\_tx to say we are currently transmitting
 line\_count captures the number of items left to write

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO  $\triangleright$  Rx + FIFO Verifying the FSM Simulation Random Delay Sim Trace Fixing the read Conclusion

Step two: Build a basic FSM to control the FIFO reads

We'll use run\_tx to say we are currently transmitting
 line\_count captures the number of items left to write

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface **FIFO** Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO  $\triangleright$  Rx + FIFO Verifying the FSM Simulation Random Delay

Sim Trace

Fixing the read Conclusion

Last steps:

Read from the FIFO any time we send to the transmitter

```
always @(*)
    fifo_rd = (tx_stb && !tx_busy);
```

Activate the transmitter anytime run\_tx is true

always @(\*)
 tx\_stb = (run\_tx && !fifo\_empty);

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO  $\triangleright$  Rx + FIFO Verifying the FSM Simulation Random Delay Sim Trace Fixing the read

Conclusion

Last steps:

Read from the FIFO any time we send to the transmitter

always @(\*)
 fifo\_rd = (tx\_stb && !tx\_busy);

Activate the transmitter anytime run\_tx is true

always @(\*)
 tx\_stb = (run\_tx && !fifo\_empty);

Q: Can the check for whether the FIFO is empty be removed?

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO  $\triangleright$  Rx + FIFO Verifying the FSM Simulation Random Delay Sim Trace Fixing the read

Conclusion

Last steps:

Read from the FIFO any time we send to the transmitter

```
always @(*)
    fifo_rd = (tx_stb && !tx_busy);
```

Activate the transmitter anytime run\_tx is true

always @(\*)
 tx\_stb = (run\_tx && !fifo\_empty);

Q: Can the check for whether the FIFO is empty be removed? Q: How would you know? Would a formal check help?

## **G** Verifying the FSM

```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
FIFO Memory
Addresses
Formal Verification
FIFO Verification
Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
   Verifying the
▷ FSM
Simulation
Random Delay
Sim Trace
Fixing the read
Conclusion
```

You should be able to formally verify that this works

- Don't reverify the FIFO
- $\hfill\square$  Consider letting the solver pick the output of the FIFO

Focus on the FIFO flags

What properties would you use to verify this FSM design?

- Don't forget to abstract the serial ports
- You may need to assume the receiver is slower than the transmitter

## **G** Simulation

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM  $\triangleright$  Simulation Random Delay Sim Trace Fixing the read

Conclusion

At this point, you know all that is needed to build a simulation

- We have a serial port transmitter, and simulation
- We have a serial port receiver, and simulation
  - We've learned to interact with our design over an O/S pipe that communicates with a child process running the Verilator based simulation
- Indeed, the simulation should look very similar to the one from our last lesson

What more might we need?

## **G** Simulation

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM  $\triangleright$  Simulation Random Delay Sim Trace Fixing the read Conclusion

At this point, you know all that is needed to build a simulation

- We have a serial port transmitter, and simulation
- We have a serial port receiver, and simulation
  - We've learned to interact with our design over an O/S pipe that communicates with a child process running the Verilator based simulation
- Indeed, the simulation should look very similar to the one from our last lesson

What more might we need?

There's one problem: the simulation trace reveals that ...

The last simulation doesn't really exercise our design
 Let's fix this!

### **G** Random Delay

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation ▷ Random Delay Sim Trace Fixing the read Conclusion

Here's the key: let's add a random delay between incoming bytes

- We'll use m\_delay to capture this delay
- We'll drain it to zero before sending a new character

```
int UARTSIM::operator()(const int i_tx) {
    // ...
    if (m_tx_state == TXIDLE)
        if (m_delay > 0) {
            // Wait for a delay to complete
            // before checking for a new
            // data byte
                m_delay--;
        } else {
            // Continue as before and ask
            // the O/S for new data byte
            // the O/S for new data byte
```

## G Random Delay

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation ▷ Random Delay Sim Trace Fixing the read Conclusion

Here's the key: let's add a random delay between incoming bytes

- We'll use m\_delay to capture this delay
- We'll create a random delay at the end of every transmitted character

# **GT** Simulation

|                                                                                                                                                                                                                                                                                                                                                        | JIIIUIALIOII                                 |  |
|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------------|--|
| Lesson Overview<br>Design Goal<br>What is a FIFO?<br>Basic FIFO Design<br>method<br>FIFO Interface<br>FIFO Memory<br>Addresses<br>Formal Verification<br>FIFO Verification<br>Cover<br>Cover Lesson<br>Line Capturer<br>Using the FIFO<br>Rx + FIFO<br>Verifying the FSM<br>Simulation<br>▷ Random Delay<br>Sim Trace<br>Fixing the read<br>Conclusion | You should now be able to run the simulation |  |
|                                                                                                                                                                                                                                                                                                                                                        |                                              |  |

## G Sim Trace

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface **FIFO** Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay ▷ Sim Trace Fixing the read Conclusion

#### Here's the trace I got from running the simulation



Notice the pseudorandom incoming byte stream, and
 The very bursty transmit stream

This was exactly what we wanted!

## **G** Fixing the read

```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
FIFO Memory
Addresses
Formal Verification
FIFO Verification
Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
Sim Trace
\triangleright Fixing the read
Conclusion
```

What we've built is a first word fall through FIFO

- That won't work on an iCE40
  - Because all block RAM reads on an iCE40 *must* be registered
  - The hardware doesn't exist on an iCE40 to do otherwise
- This was our problem code

```
always @(*)
        o_data <= fifo_mem[rd_addr];</pre>
```

```
How shall we fix this?
```

## **G** Fixing the read

#### Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface **FIFO** Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace $\triangleright$ Fixing the read

Conclusion

#### Solution one: bypass the memory

- $\hfill\square$  On a write, store the value in memory and in a register
- On the next clock, offer the register value as a result
- We'll also need to adjust the read address
  - The memory read address needs to be the address it will be on the *next clock*

Let's see what this would look like

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace  $\triangleright$  Fixing the read Conclusion

Our basic logic will be to capture two memory values, and select between them

□ The first is placed in a register

```
always @(posedge i_clk)
    bypass_data <= i_data;</pre>
```

The second is read from the memory

```
always @(*)
rd_next = rd_addr[LGFLEN-1:0] + 1;
```

```
always @(posedge i_clk)
    rd_data <= mem[(w_rd)?rd_next
        : rd_addr[LGFLEN-1:0]];</pre>
```

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace  $\triangleright$  Fixing the read Conclusion

Our basic logic will be to capture two memory values, and select between them

- The first is placed in a register
- The second is read from the memory
- Then we select between them

always @(\*)
 o\_data = (bypass\_valid) ? bypass\_data
 : rd\_data;

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace  $\triangleright$  Fixing the read Conclusion

```
The trick is the selector, bypass_valid, that tells us which value to return
```

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace  $\triangleright$  Fixing the read Conclusion

```
The trick is the selector, bypass_valid, that tells us which value to return
```

```
initial bypass_valid = 0;
always @(posedge i_clk)
begin
        bypass_valid \leq 1'b0;
        if (!i_wr)
                bypass_valid <= 1'b0;
        else if (o_empty ||(i_rd&&(o_fill == 1)))
                // Otherwise if we read, and the
                // memory is now empty, use the
                // register value
                bypass_valid <= 1'b1;
        // Remember, the last assignment wins
end
```

You can use formal methods to prove the result is the same

## **G** Fixing the read

```
Lesson Overview
Design Goal
What is a FIFO?
Basic FIFO Design
method
FIFO Interface
FIFO Memory
Addresses
Formal Verification
FIFO Verification
Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
Simulation
Random Delay
Sim Trace
\triangleright Fixing the read
Conclusion
```

Solution two: Work with the registered output

We could just use the registered data

```
always @(posedge i_clk)
    o_data <= mem[(w_rd)?rd_next
        : rd_addr[LGFLEN-1:0]];</pre>
```

The biggest problem would be our empty FIFO logic

- It would need to be delayed by one clock

## **G** Exercise #1

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method **FIFO** Interface **FIFO** Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace  $\triangleright$  Fixing the read Conclusion

Pick a solution to our memory problem and

- Formally verify it
- Prove that it still meets the two write/two read criteria of a FIFO

## **GTExercise** #2

```
Lesson Overview
Design Goal
                          What is a FIFO?
Basic FIFO Design
method
FIFO Interface
                          FIFO Memory
Addresses
Formal Verification
FIFO Verification
Cover
Cover Lesson
Line Capturer
Using the FIFO
Rx + FIFO
Verifying the FSM
                          Simulation
Random Delay
Sim Trace
\triangleright Fixing the read
Conclusion
```

All of our o\_fill and o\_full logic is combinatorial

- This will prevent keep us from using our FIFO in a high speed design
- Solution: Rewrite this logic so that it's registered

```
always @(posedge i_clk)
    o_fill <= // Your logic here</pre>
```

```
always @(posedge i_clk)
    o_full <= // Your logic here</pre>
```

Use the formal solver to formally prove that it still meets the properties required of o\_fill and o\_full

## **GT** Exercise

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface **FIFO** Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace  $\triangleright$  Fixing the read Conclusion

#### Are you ready? Let's build this!

- Create this design, and place it on your FPGA
- You've already formally verified and simulated it, right?
  - So ... it should work on the hardware the first time, right?

(Guilty admission: Mine still didn't work the first time ...)

## **G** Questions

Lesson Overview Design Goal What is a FIFO? Basic FIFO Design method FIFO Interface FIFO Memory Addresses Formal Verification **FIFO** Verification Cover Cover Lesson Line Capturer Using the FIFO Rx + FIFOVerifying the FSM Simulation Random Delay Sim Trace  $\triangleright$  Fixing the read Conclusion

- Many serial port implementations use RTS and CTS signals
  - How might you use the FIFO level to stop an upstream transmitter when the FIFO was (nearly) full? Don't forget these things don't stop on a dime.
  - How might a downstream receiver signal to this design to stop transmitting, since its FIFO was (nearly) full?
- Most packet format end with a CRC
  - How would you modify this design to add and check a CRC?
- Many packet formats have either a fixed length, or a length specifier
  - How would a variable packet length change things?

## G Conclusion

| Lesson Overview             |
|-----------------------------|
| Design Goal                 |
| What is a FIFO?             |
| Basic FIFO Design           |
| method                      |
| FIFO Interface              |
| FIFO Memory                 |
| Addresses                   |
| Formal Verification         |
| FIFO Verification           |
| Cover                       |
| Cover Lesson                |
| Line Capturer               |
| Using the FIFO              |
| Rx + FIFO                   |
| Verifying the FSM           |
| Simulation                  |
| Random Delay                |
| Sim Trace                   |
| Fixing the read             |
| $\triangleright$ Conclusion |

#### What did we learn this lesson?

- What a FIFO is, and why you might use one
- How to formally verify a FIFO
- Some of the problems associated with reading the data from a FIFO on different pieces of hardware
- How to eliminate combinatorial logic, while making sure that the design functionality doesn't change