# Lab Workbook

SS 2022

# Industrial Hardware Verification

Jakob Lechner Markus Ferringer

Part 1: VHDL & FLI & OSVVM

# Contents

| T | Inst | allation                                  |
|---|------|-------------------------------------------|
|   | 1.1  | ModelSim/QuestaSim                        |
|   | 1.2  | C Compiler                                |
|   | 1.3  | Source Code                               |
|   | 1.4  | Compilation                               |
|   | 1.5  | Editor                                    |
| 2 | VH   | DL                                        |
|   | 2.1  | Exercise 1: Attributes                    |
|   |      | 2.1.1 Task 1 [1 Point]                    |
|   |      | 2.1.2 Task 2 [1 Point]                    |
|   | 2.2  | Exercise 2: Transactions                  |
|   |      | 2.2.1 Task 1 [1 Point]                    |
|   |      | 2.2.2 Task 2 [1 Point]                    |
|   | 2.3  | Exercise 3: Delay Modes                   |
|   |      | 2.3.1 Task 1 [2 Points]                   |
|   |      | 2.3.2 Task 2 [1 Point]                    |
|   | 2.4  | Exercise 4: Advanced Data Types, Generics |
|   |      | 2.4.1 Task 1 [4 Points]                   |
|   |      | 2.4.2 Task 2 [3 Points]                   |
| 3 | osv  | VVM                                       |
|   | 3.1  | Exercise 5: OSVVM Verification Unit       |
|   |      | 3.1.1 Task 1 [4 Points]                   |
|   |      | 3.1.2 Task 2 [3 Points]                   |
|   | 3.2  | Exercise 6: Constrained Random Testing    |
|   |      | 3.2.1 Task 1 [4 Points]                   |
| 4 | FLI  | 14                                        |
| _ | 4.1  | Exercise 7: FLI                           |
|   |      | 4.1.1 Task 1 [3 Points]                   |

# Lab Mode

- You have to elaborate these exercises on your own. This lab part is **not** a team effort.
- We provide stubs containing basic templates for you which are to be extended with your implementations
- Questions are to be ansered in the respective source-files of the corresponding exercise.

  Just add appropriate comments at the end
- If you provide additional files (eg, images, diagrams, tables as part of your answers to questions) please make a note in the source files so that we don't miss anything
- Additional files must be located in the folder corresponding to the exercise
- I strongly recommend to read through an entire task before starting with the implementation (including the questions section). You will sometimes find hints later that might simplify things
- Once done with all exercises, please execute ./make.sh clean in each folder, zip everything (ex1 ex7) and submit it via TUWEL

## Installation

This chapter explains how to install the necessary software and get the lab exercises. You can do most of the work on your own personal computer. Only when special simulator features (such as Code Coverage or PSL) are needed, you need to work on the lab machines in order to use the respective licenses.

Both Windows and Linux should work, although Linux will be less effort to set up. While it is possible to use Microsoft Windows (eg., using CygWin or WSL), these instructions focus on Linux-based systems like Ubuntu.

You don't need to install any software on the Lab Workstations, everything you need is preinstalled.

## 1.1 ModelSim/QuestaSim

On the lab workstations, everything is set up. Should you want to work with your personal equipment, you need to download the simulator first:

- 1. Download ModelSim-Intel FPGA Starter Edition. Sounds easy? Well...
- 2. https://www.intel.com/content/www/us/en/software/programmable/quartus-prime/model-sim.html
- 3. You need to register. It's free, but it's a requirement.
- 4. Once registered, sign in; Unfortunately, the site is not well structured...
- 5. Click on your Avatar (top right) and choose "Intel FPGA Design Software Download Center"
- 6. On the left, under "Additional Software" choose "Intel FPGA Simulation Tools" and then "Questa-Intel FPGA Starter Edition"
- 7. From the list, choose the one for your platform. Use the latest version, eg "Intel Quartus Prime Pro Edition Design Software Version 21.4 for Windows"
- 8. Now click "Individual Files" as the complete download is over 70GB
- 9. Download "Questa-Intel FPGA Edition (includes Starter Edition)" and "Questa-Intel FPGA Edition (includes Starter Edition) Part 2". It's still over 10GB.
- 10. Install the software

The free version has a couple of restrictions, so for some exercises you have to use the lab workstations.

- Reduced performance (which will not be an issue, though)
- No support for code coverage  $\rightarrow$  use lab workstations
- No support for PSL  $\rightarrow$  use lab workstations

### 1.2 C Compiler

For the FLI examples you need a C compiler. Under Linux, gcc is just perfect. Under Windows, I used mingw32. Please notice that you need a 32 bit compiler (or one that can comile in 32 bit mode using a switch like -m32). It might also be necessary to install 32 bit support first, which can be done with

- (Ubuntu): sudo apt-get install libc6-dev-i386
- (CentOS): sudo yum install glibc-devel.i686

**Remark**: At the time of writing (30.03.2022) the Lab Workstations have not installed 32 bit support. The lab admin is informed, and it should be available soon. I will keep you informed via the Forum. Thoses exercises can also be run on you personal equipment, so lab access is not a requirement.

#### 1.3 Source Code

We prepared a framework for you which contains

- OSVVM (in a separate folder)
- Exercises (each in a separate folder)
- A suitable make.sh (one for each exercise)
- Stubs for all the exercises (containing basic templates where you can fill in your solution)
- The entire source-code needed for all exercises

You can download the execises via TUWEL. It will be available after the respective lectures.

Remark for Windows Users: The make.sh files are symbolic links, which are not supported in Windows. If your make files don't work or are empty, try to copy the original make.sh from the root-folder into the respective exercise folder.

### 1.4 Compilation

Each exercise is located in a separate folder. There is a make.sh file which supports the following targets. Please notice that this is not a regular Makefile, but rather a shell script named make.sh. I prefer this over Makefiles.

- ./make.sh clean: Cleans up temporary files and compiled libraries
- ./make.sh sim: Compiles the exercise's code, then executes the simulation. When called the very first time (or after a make.sh clean), OSVVM is compiled.
- ./make.sh gui: Opens the ModelSim/QuestaSim GUI and sources the OSVVM scripts, so you can use them right away

You need to provide the path to your / the lab's ModelSim/QuestaSim binary. In the root folder (the one that contains OSVVM and all exercises) there is a file called settings.make. Open it and edit the MODELSIM\_PATH variable according to your setup. If ModelSim is available through your PATH variable, just keep the default assignment MODELSIM\_PATH=vsim.

For the FLI example, you also need to provide the path to ModelSim's include folder. Just edit the compile.sh script accordingly. Depending on your setup, you might also need to change the C compiler used. There is a compile\_win.bat for Windows, but you need to update the paths as well.

#### 1.5 Editor

Of course, you can use any editor you like. However, it will be more efficient (and way less frustrating) if you are using a good editor which is capable of understanding VHDL, and which offers at least some degree of support (like parameter help, tooltips, code completion, etc.). You have to deal with a lot of packages and libraries, and 100s of functions, procedures, and datatypes. You should use all the support you get from a good IDE.

As I have written my own VHDL editor (an extension for Visual Studio Code), I will of course recommend this IDE to you :-) VS Code is Open Source and thus for free, and I will provide you with a license for my extension (it will be included in the lab exercises, you don't have to worry about it).

- Download and install VS Code from https://code.visualstudio.com/
- Open VS Code (code in the terminal)
- On the left side, open the Extensions view
- In the search field, enter VHDL for professionals (V4P)
- Select VHDL for Professionals from the results, and click the install button
- If requested, restart VS Code
- Open file workspace.code-workspace which is located in the root folder of the provided zip file

All (VHDL-based) exercises come with a preconfigured VS Code workspace. Just open it in VS Code, and you are set to go. All the OSVVM libraries, functions, types, etc. are at your disposal. The license for V4P will also be part of the workspace.

## $\mathbf{VHDL}$

#### 2.1 Exercise 1: Attributes

### 2.1.1 Task 1 [1 Point]

Write a function ColorToString(pos: integer) return string with the following properties:

- The function shall take an integer parameter which specifies the "index" of the enumvalue in type ColorT
- If the index is invalid, the function shall call Alert() with an appropriate error message and return ''OutOfRange''
- If the index is valid, the string representation of the color at position pos shall be returned
- The function shall work without modification even if colors are added or removed to/from ColorT
- In the main process stimuli\_p, add some calls to ColorToString() and see if it works
- Example:

```
1 architecture beh of ent is
2  type ColorT is (Red, Green, Blue);
3  ...
4 begin
5  — Expected output: ''green''
6  report ColorToString(1) severity note;
7 end architecture;
```

- Answer the following questions (directly in the source file as comment):
  - What is the result of ColorT'leftof(Red), ColorT'succ(Yellow), ColorT'val(5)?
  - How does the simulator handle those situations?

### 2.1.2 Task 2 [1 Point]

Write a function ColorsToList() return string

- The function shall return a comma-separated list of all enumeration fields that are part of the ColorT type
- Example:

```
architecture beh of ent is
type ColorT is (Red, Green, Blue);
...
begin
Expected output: "AllEnums: red, green, blue"
report "AllEnums: " & ColorsToList() severity note;
end architecture;
```

#### • Hints

- Use dynamic strings: Define a string access type (don't use textio.line)
- Make sure the function handles one-element enums
- The function shall work without modifications even if the ColorT type is changed
- Answer the following questions (directly in the source file as comment):
  - During implementation, did you run into segmentation faults? Why?
  - Does your implementation contain memory leaks? If so, explain why.
  - If there are memory leaks, can you imagine a way to avoid them?
  - Especially for this simple exercise, can you imagine a way to find out if the implementation contains memory leaks?

#### 2.2 Exercise 2: Transactions

#### 2.2.1 Task 1 [1 Point]

- We provide a very simple state machine that is controlled by process stimuli\_p. Every 20 clock cycles, the state is advanced.
- Take a look at process state\_p: Depending on the state, signal output is modified in different ways
- Implement a process event\_count\_p that increments eventCounter every time output changes
- Implement a process trans\_count\_p that increments transCounter every time a transaction occurs on output
- Answer the following questions (directly in the source file as comment):
  - Explain the difference between the two processes and how they are triggered
  - Is there a difference between the assignment statements of the states Idle, NotAffected, Keep?

### 2.2.2 Task 2 [1 Point]

- Enhance process stimuli\_p such that the testbench self-checks the counter values
- After each call to WaitForClock() in stimuli\_p, add AffirmIfEqual(actual, expected, message) statements. Just hardcode the expected values.
- Open up the GUI (./make.sh gui), and execute build project.pro
- Now add all the signals to the waveform viewer. Save the waveform (filename wave.do)
- Restart the simulation with build project.pro
- Inspect the results in the waveform viewer. If there is a wave.do it is automatically opened.
- Answer the following questions (directly in the source file as comment):
  - How can you make output'transaction visible in the waveform viewer?

- Can you imagine scenarios where sig'transaction is actually useful? Why would we want events if there is no change of data?

- Why is stimuli\_p reacting on the falling edge, while state\_p triggers on the rising edge?

### 2.3 Exercise 3: Delay Modes



#### 2.3.1 Task 1 [2 Points]

Consider the electrical transmission line in the figure above.

- Model the physical transmission line in VHDL
  - Output is open collector with a  $1k\Omega$  pull-up resistor
  - The transmission line itself has an impedance of  $50\Omega$  and a parasitic capacitance of 50nF at the receiver
  - Assume that the delay from TX to RX is 23ns
  - Assume that RX recognizes  $V_{rx} \ge 0.632 V_{cc}$  as logical 1 (1 $\tau$  as indicated)
  - Assume that RX recognizes  $V_{rx} \leq 0.368V_{cc}$  as logical 0 (1 $\tau$  as indicated)
  - If  $V_{rx}$  is between these two thresholds, the internal logic value does not change
  - The transistor can be modeled as ideal switch
  - You can assume that signal transitions are far enough separated so that they don't interfere with each other
- Answer the following questions (directly in the source file as comment):
  - Given this specification, what is the maximum frequency that can safely be transmitted?
  - In the analog world...
    - \* how do typical transitions on the wire look like? (just add a photo or something adequate of a hand-written diagram)
    - \* how does an eye diagram look like? (just add a photo or something adequate of a hand-written diagram)
  - What happens in your simulation if the maximum frequency is exceeded?
  - What would happen on an actual transmission line if the frequency was exceeded?

#### 2.3.2 Task 2 [1 Point]



Now model an optical transmission line with the following properties

- Fiber length is 120m (speed of light is around 200000km/s in the optical medium)
- The laser's propagation time from electrical input to optical output is 32ns for both rising and falling edges
- After a transition on the input, the laser ignores all glitches on the input for 20 ns (see waveform above)
  - You can assume that "regular" input transitions are separated by at least 20 ns
  - You can assume that glitches fully occur (and disappear again) within those 20 ns (subimage (a) of the waveform)
  - You don't need to handle glitches that are exactly on the border (subimage (b) of the waveform)
- On the receive side, the optical detector has the same characteristics as the transmitter
- As the transmission line is quite long, you cannot longer assume that only one signal is transmitted at a time (i.e., a new transition can be sent before the last one has arrived)
- Hint: You will need an intermediate signal to combine inertial and transport delays
- Answer the following questions (directly in the source file as comment):
  - What is the rejection time of inerial if you don't specify it explicitly?
  - Can the rejection time be greater that the corresponding inertial delay?
  - If modeled correctly, any glitches on the input shorter than 20 ns are ignored. But what happens if a **single** signal transition happens within the rejection time and the signal then remains constant afterwards? Like so:

tx <= '0' after 0 ns, '1' after 100 ns, '0' after 110 ns (Your solution does **not** need to solve this problem!)

## 2.4 Exercise 4: Advanced Data Types, Generics

## 2.4.1 Task 1 [4 Points]

Your task is to implement a (singly) linked list in VHDL

- The list shall be implemented as protected type inside package linked\_list.
- The linked list shall be located in a generic package, which takes two generic parameters
  - type ElementT: The type of the items that are stored in the list
  - function ToStringF(item: ElementT) return string: A function that converts an item to a string representation

• This generic data type denotes the type of the elements that can be stored inside the list

- Implement the following functions/procedures:
  - function Count return integer: Returns the number of elements in the linked list
  - procedure AddLast(item: ElementT): Adds an element to the end of the list
  - procedure AddFirst(item: ElementT): Adds an element to the front of the list
  - function GetFirst return ElementT: Returns the first element of the list
  - function GetLast return ElementT: Returns the last element of the list
  - function GetAt(index: integer) return ElementT: Return the i-th item of the list. The first item (head of list) has index 0.
  - procedure RemoveAt(index: integer): Remove the i-th item from the list. The first item (head of list) has index 0.
  - procedure AddAfter(index: integer; item: ElemetT): Adds the item after the i-th item in the list. The first item (head of list) has index 0.
  - function Dump return string: Returns a comma separated list of string representations of all items in the list (in the stored order, of course)
- In case of invalid operations (eg, get a value from an empty list), create a log entry by calling Alert(...).
- In case of invalid operations, if there is a mandatory return value, just return VHDL's default for the respective data type

#### 2.4.2 Task 2 [3 Points]

Your task is to implement a testbench which tests all the linked list (process stimuli\_p)

- Instantiate the generic list, use PrimeRecT as generic type and implement the ToString function for this data type
- Test all public subroutines that are part of the public interface
- Make sure that Statement Coverage is at 100% (for the linked-list protected type)
- You can use the GUI (Tools Coverage Report...) to generate annotated source files
- Attach the annotated source code (coverage report) to your submission

## OSVVM

#### 3.1 Exercise 5: OSVVM Verification Unit

### 3.1.1 Task 1 [4 Points]

Your task is to implement a Verification Unit for the Avalon Memory Mapped (AVMM) Bus. Your are implementing a master device, which can in turn control DUTs with memory mapped slave interfaces. Notice that this VU is already in preparation for the Verification Project, where you will need an AVMM master component.



- THERE IS A BUG IN THE PROVIDED TEMPLATE: In avmm\_vu.vhd, inside the dispatcher\_loop, replace WaitForTransaction(transio); with
  - WaitForTransaction(Clk => clk\_i, Rdy => trans\_io.Rdy, Ack => trans\_io.Ack);
- The read and write timing is shown in the image. This timing is fixed, and need not be configurable. The shown waveform corresponds to writeWaitTime=0 and readWaitTime=2
- It is not necessary to support simultaneous read and write access.
- The interface uses byte enables.
- It does not use WaitRequest, and ReadDataValid (the latter is implicit due to timing). You don't have to support bursts. You don't have to support pipelined reads / writes (i.e., just one read/write at a time, until the entire sequence is finished)
- ullet There is a template package file which contains definitions of the public interface of the VU
  - Implement the subroutine bodies of the respective header definitions
  - Use the AddressBusTransactionRecT data structure, it is well suited for this task

- There are a couple of predefined OSVVM subroutines (eg, send) which are already
  performing lots of the necessary steps. Use them, or review them to adapt you
  own implementation
- You might want to take a look at OSVVM's UART and AXI implementation for reference
- The command AvmmReadModifyWrite has an argument write mask. It is of the same length as the data vector, and a '1' on a specific location means that this bit is written to the register according to the supplied data. A '0' means the bit's old value is preserved. Out of the bitmask, the byte-enables can easily be derived.
- There is the main VU's source file which already contains a basic skelton of the VU
  - For simplicity, both reading and writing shall be blocking so there is no need to implement FIFOs to buffer incoming or outgoing data transactions
  - Implement the VU. As there are only blocking operations, it is simplest to just use one process which does all the work (read data, write data)

### 3.1.2 Task 2 [3 Points]

Your task is to implement a basic (directed) testbench for your VU

- Test all the functions of the public interface
  - Use an address-width of 5 bits, and a data-width of 16 bits
  - Use a scoreboard. For each call to one of the public functions, create the respective expected waveforms on the avalon bus and store then in the scoreboard.
  - Write a process that monitors the bus; put the observed waveforms in the scoreboard for checking.
  - For this exercise, it is good enough to hardcode the waveforms. Usually, you should of course implement a reference model.
- Reach 100% statement coverage for the main VU design, and the public package

## 3.2 Exercise 6: Constrained Random Testing

### 3.2.1 Task 1 [4 Points]

| Addr | 7    | 6  | 5 | 4 | 3   | 2 | 1   | 0 |
|------|------|----|---|---|-----|---|-----|---|
| 0    | ΙE   | IF | - |   |     |   | RST |   |
| 1    | NAME |    |   |   | CHK |   |     |   |
| 2    | -    |    |   |   | ID  |   |     |   |
| 3    | CNT  |    |   |   |     |   |     |   |

| Name | Access | Default  | Remarks                                                                         |
|------|--------|----------|---------------------------------------------------------------------------------|
| RST  | R/W    | 0        | Write 1 to reset all registers to their default. The field auto-clears to zero. |
| IF   | R      | 1        | Auto-clears when read. Then remains 0 until RST                                 |
| IE   | R/W    | 0        |                                                                                 |
| CHK  | R      | 1010     | NAME xor ID                                                                     |
| NAME | R/W    | 0000     |                                                                                 |
| ID   | R      | 1010     |                                                                                 |
| CNT  | R      | 00000000 | Increments by one with each clock cycle. Overflows to 0 when maximum is reached |

Your task is to use constrained random testing in combination with functional coverage to test a given DUT.

- The DUT is a simple register interface
  - There is no internal state other than the registers directly accessible through the memory mapped interface

- The register description is shown in the table
- Read/Write access timings are the same as for the previous exercise, byteenables are ignored.
- Use the VU implemented in the previous exercise to save time. Alternatively, you can implement all the bit-banging again, neglecting the advantages of code reuse:-)
- It is not necessary to test the bus protocol itself. Limit your testing to sending valid read/write requests.
- Use OSVVM's functional coverage capabilities to specify cover points for the defined register fields
  - Eg, all addresses have been read from
  - Eg, all writable fields have been written to
  - Eg, all writeable fields have been read from after they have been written to, etc.

**–** ...

- The DUT has some undocumented "features/bugs"—, which should not be detectable with simple directed tests.
- Can you find them such "features/bugs" using constrained random testing? It's not mandatory to find those problems for solving this task. If you find something, you can make a note somewhere in your source code.
- Please notice that you don't have to modify the DUT. You can just ignore the insert your code here comments!

## FLI

#### 4.1 Exercise 7: FLI

#### 4.1.1 Task 1 [3 Points]

Your task is to implement a Mandelbrot Set renderer in VHDL and C, and compare the performance of both variants

- We prepared a basic framework for you to easily compile C code and use it from within vhdl
  - Use compile.sh to compile for linux. It should work right way
  - Use compile\_win.bat to compile for windows. You need to update the path to your compiler. Might make problems...
- Calculating the Mandelbrot set is quite easy. Google is your friend.
- In short, given a complex number c, calculate  $z_{n+1} = z_n^2 + c$  with  $z_0 = 0$ . If  $z_n$  evaluates to  $(|z_n| > 2.0)$ , the point is not part of the set.
- We are counting and "colorizing" n, i.e. the number of iterations we need until the point escapes. We also need to define an upper limit for n for which we consider the point to belong to the set. We define it to be 200.
- Implement the following functions / procedures in C
  - function GetTimeCreturn stringprocedure GetTimeC(hour, minute, second: out integer): Write a C function to get the current time (including seconds) as string. We need this for performance measurement.
  - procedure OneStepC(zr, zi, cr, ci: real; or, oi: out real): Write a function that performs one step of the mandelbrot calculation  $o = z^2 + c$
  - function IterateC(x, y: real) return integer: Calculate the Mandelbrot escape value for the given (complex) point (x, y). Don't do more than 200 iterations. Stop if the magnitude of the complex vector exceeds 2.0
  - Take a look at the data type mapping table in the slides to learn how to pass certain data types back and forth via FLI
- Define the subroutine defintions in VHDL and mark them as foreign.
- Implement the same functions and procedures in VHDL. You need to give them distinct names because we want to use both VHDL and C functions at the same time. Use the ieee.math\_complex.Complex data type.
- Run the following simulations and record the execution times:
  - Everything VHDL: Image calls Iterate calls OneStep
  - Image calls Iterate calls OneStepC

CHAPTER 4. FLI

- Image calls IterateC (calls OneStepC directly in C)