# An introduction into Verilog simulation

# Christian Krieg

October 16, 2023

Welcome to this year's *Digital Integrated Circuits* course! Throughout this course, we will have to solve some assignments that will teach us fundamental concepts of integrated circuit design.

We will demonstrate digital integrated circuit design on a step-by-step simple example of a digital counter. As the name of this course suggests, our ultimate goal is to implement this counter as a digital integrated circuit. In the lab, we map this counter to a field-programmable gate array (FPGA) architecture, and run it on the board shown in Figure 1. The counter's functionality is executed on the FPGA. We will be able to control execution by pushing the buttons, and to visualize the counter value using the light-emitting diodes (LEDs). Because operating a hardware counter with buttons and LEDs may be inconvenient, we also will interface the hardware counter over a universal asynchronous receiver/transmitter (UART), so that we may be able to further process the counter value on a host computer. But this will all happen during the lab week at the end of the exercises.



Figure 1: The icebreaker board, which will run our counter

Before we can bring our counter on hardware, it is important that we learn how to design digital circuits using state-of-the-art (and beyond) methods and tools, and to get a feeling for the results of our design decisions. We will draw attention on the results of the synthesis steps, which we will verify with several tools along the digital design flow.

Generally speaking, we perform the following steps:

- 1. Specify the counter's behavior
- 2. Model (implement) the counter in a hardware description language (HDL)
- 3. Verify the counter's implementation
- 4. Synthesize and optimize the counter's HDL model + verify
- 5. Map the counter to the target (FPGA) technology + verify
- 6. Generate an FPGA bitstream + verify
- 7. Configure the FPGA with the bitstream + verify
- 8. Run the counter on hardware

So, let's get started! Our first task is to specify a synchronous counter that runs in two modes, based on a *mode* switch. The two modes are called *up* and *down*. In *up* mode, the counter increments the counter value by 4 at the rising edge of the clock signal, and in *down* mode, it decrements the counter value by 10 at the rising edge of the clock signal. The counter should implement synchronous reset behavior, such that it is set to an initial value of 17 upon reset. The counter should never exceed a value of 269, and should never go below a value of -263 (the counter value sticks near the



Figure 2: Basic concept of testbench and DUT

minimum/maximum value, until the counting direction changes). The counter value should never be -47 (the counter should jump over this value by one increment/decrement).

# **Testbenches: A quick tutorial**

Before we can bring our counter on hardware, we first have to specify its behavior; then, we have to model it using an HDL; and finally, we have to verify that the counter correctly models its specification. There are various methods available to verify the behavior of HDL models. Today, we will learn about a widely used, very popular verification method: *simulation*. In simulation, a *simulator* takes as input an HDL model of a digital design, along with code that tests the functional correctness of this model. Such testing code is commonly referred to as a *testbench*, and is typically implemented in a separate file to maintain modularity. A testbench tests whether the HDL model, which is commonly known as design under test (DUT), or unit under test (UUT), conforms to its specification. A testbench does not have inputs or outputs. Its only purpose is to provide inputs to a DUT, and to check the DUT's outputs against expected values. Figure 2 shows a typical visualization of a testbench that instantiates and initializes a DUT, provides inputs to the DUT, reads outputs from the DUT, and checks whether the output values conform to the specification. Ideally, a testbench may provide log messages to help debugging the design.

The two major HDLs, VHDL and Verilog, both support simulation. These languages provide constructs to generate input valuations, and functionality to interact with a user and/or simulator (e.g., printing on a screen, reading a file with input data, etc.). The main purpose of an HDL is to model hardware, which means that the final output is a netlist. Although simulation features of HDLs may be very practical to examine a design, they cannot be represented as a netlist—they are not *synthesizable*. For instance, there is no netlist representation for routines like "read a file from the file system", or "print a message to the screen". HDL language subsets that may be used to model synthesizable hardware are called *synthesizable subsets*. A DUT may only contain synthesizable code. Simulation code may only be used in a testbench.

In the following, we develop a testbench step-by-step that may be used for verification. We use Verilog to implement this testbench.

#### **Design specification**

Before we start developing a testbench, we first need to define the reference we are testing against: the design's *specification*. We take the informal, human-language description of the counter from above, and formulate properties that constrain the behavior of our counter. We name the counter's input/output signals as follows:

Table 1: Signal names for the counter

| Signal name | Description   | Width |
|-------------|---------------|-------|
| cnt         | Counter value | 10    |
| mode        | Counter mode  | 1     |
| clk         | Clock         | 1     |
| rst         | Reset         | 1     |

If the mode signal is '0', the counter is in down mode, and in up mode otherwise. We specify the counter's behavioral properties as follows:

1. The counter value always is in the specified range [-263, 269], and it never takes the *invalid* value -47

- 2. The counter increment is always 4, except when the invalid value is jumped over (then, it's twice as big)
- 3. The counter decrement is always 10, except when the invalid value is jumped over (then, it's twice as big)
- 4. The increment/decrement can be different after the first clock cycle, when rst goes from '1' to '0'
- 5. The increment/decrement can be different when the counter value remains near the minimum/maximum value (i.e., the increment/decrement is 0)

Our challenge is now to develop a testbench which verifies that the counter behaves as specified above.

The following examples assume parameters as given in Table 3.

Table 2: Parameters for the example counter

| Parameter | Value |
|-----------|-------|
| MIN       | -287  |
| MAX       | 309   |
| INV       | 75    |
| INIT      | 13    |
| INC       | 4     |
| DEC       | 10    |
|           |       |

### A very simple testbench

By convention, each Verilog *module* (e.g., *counter*, like the DUT we are testing) is implemented in its own Verilog file (e.g., *counter.v*), and each Verilog module is tested by its own testbench, which gets the same filename like its DUT, along with a prefix or suffix that identifies it as a testbench, for instance "tb\_", "test\_", etc. So, if our counter is implemented in a Verilog module *counter*, written into a file *counter.v*, its associated testbench module is named *tb\_counter*, written into a file *tb\_counter.v*.

Listing 1 shows a very simple testbench written in Verilog. Line 1 gives a physical meaning to simulation time, and tells the simulator how to interpret time. The first value specifies the time unit, and the second value specifies the granularity of time, which is important for the simulator to correctly round time values. The testbench itself is implemented in a Verilog module (Lines 3 to 42), without input or output ports that connect to the outside world. While not having own inputs or outputs, the testbench defines registers to store input values provided to the DUT in Lines 6 to 8, and a wire that carries the DUT's output value in Line 11. Lines 14 to 19 show the DUT's instantiation, along with a port map that connect the input/output signals of the DUT with the previously defined input/output signals of the testbench. Line 22 shows an *always* block that generates values for the clock signal by toggling its value every 5 time steps. Lines 25 to 38 show an *initial* block that assigns values to input signals after different amounts of time steps, given after the "hash" operator "#". This way, a reset sequence is implemented in Lines 28 to 29, and the *mode* input is alternated in Lines 32 to 35 to change the DUT's counting direction (up/down) The \$finish statement in Line 36 tells the simulator to end the simulation at this point. Finally, Line 40 implements a directive to print the counter value (which is the DUT's output value) every time it changes.

Listing 1: A very simple Verilog testbench tb counter-1.v

```
'timescale 1ns/1ns
   module tbcounter:
      // Define inputs
      reg clk=0;
      reg rst=0;
      reg mode=0;
10
      // Define outputs
wire signed [9:0] cnt;
11
13
      // Instantiate design under test (DUT)
14
      counter dut (
         .clk(clk),
15
16
         .rst(rst)
17
         .mode(mode),
18
         .cnt(cnt)
19
20
21
22
      // Generate clock
      always #5 clk = !clk;
23
      // Stimulate the DUT (vary input values)
```

```
// Reset the DUT
          #10 \text{ rst} = 0;
29
30
31
          // Operate the DUT
32
          #17 mode = 1;
33
          #22 \text{ mode} = 0;
          #12 mode = 1;
34
35
          #100 \text{ mode} = 0;
36
37
          $finish;
38
      initial $monitor("At time %0t, value = %h (%0d)", $time, cnt, cnt);
41
42
   endmodule
```

We simulate the DUT along with the testbench given in Listing 1. There are several Verilog simulators available, both commercial and open-source. For the following examples, we use the free and open-source Verilog simulator *Icarus Verilog*. Listing 2 invokes *Icarus Verilog*, and provides log output from the simulation.

Running the simulation happens in two steps: First, we call the *Icarus Verilog* compiler *iverilog* in Line 1, which translates the HDL model into an executable program, with the following options and arguments:

Table 3: Options/Arguments to iverilog

| Option/argument                    | Description                                        |
|------------------------------------|----------------------------------------------------|
| -l 'yosys-configdatdir/simcells.v' | Provide Yosys cell library                         |
| -o counter                         | Define the target name                             |
| counter.v                          | The name of the file that implements the DUT       |
| tbcounter-1.v                      | The name of the file that implements the testbench |

When *iverilog* successfully compiles the executable program, we call the *Icarus Verilog vvp runtime engine* to execute the simulation in Line 2. The simulator's debug output is shown in Lines 3 to 19, providing counter values both in hexadecimal and decimal form. We notice the counter counting up and down, recognize the initial value (13), the increment (4), and the decrement (10). In Line 20, the simulator reports the end of simulation.

Listing 2: Simulating the testbench given in Listing 1

```
$ iverilog -l 'yosys-config --datdir/simcells.v' -o counter counter.v tbcounter-1.v
  $ vvp counter
  At time 0, value = 00d (13)
  At time 15, value = 003
  At time 25, value = 3f9
                           (-3)
  At time 35, value = 3fd
  At time 45, value = 001
                           (1)
                      3f7
  At time 55, value
  At time 65, value
                       3ed
  At time 75, value = 3f1
  At time 85, value = 3f5
                           (-11)
              value = 3f9(-7)
  At time 95.
  At time 105, value = 3fd
  At time
          115, value
                            (1)
  At time
          125, value
                        005
16
  At time
          135,
               value
                        009
                            (9)
                            (13)
  At time 145, value
                        00d
  At time 155, value
                      =
                        011
                            (17)
                        015
  At time
                            (21)
          165, value
  tbcounter-1.v:36: $finish called at 166 (1ns)
```

While we get an idea what happens inside our counter, we are fairly limited with this simple testbench:

- 1. The testbench only reports, but does not verify
- 2. The coverage may be very limited
- 3. We see the simulation results printed to standard output, but it is nowhere recorded for later use

In the following, we gradually extend our testbench to address our requirements.

#### Recording simulation results

As noted before, we do not have any opportunity to analyze simulation results at a later time. This is why we add another *initial* block to our testbench in Lines 6 to 9, which invokes routines to dump simulation data.

Listing 3: Adding support for dumping simulation data (tb\_counter-2.v)

```
timescale 1ns/1ns
   module tbcounter;
        / Specify dump file and the scope to be dumped
      initial begin
    $dumpfile("counter.vcd");
8
9
         $dumpvars(0, tbcounter);
10
11
      // Define inputs
12
      reg clk=0;
13
      reg rst=0;
14
      reg mode=0;
15
      // Define outputs
16
17
      wire signed [9:0] cnt;
18
19
      // Instantiate design under test (DUT)
20
      counter dut (
21
          .clk(clk),
22
          .rst(rst).
23
          .mode(mode),
24
          .cnt(cnt)
25
26
27
      // Generate clock
28
      always #5 clk = !clk;
29
30
      // Stimulate the design (vary input values)
31
      initial begin
32
33
34
          // Reset the DUT
         #5 \text{ rst} = 1;
35
         #10 \text{ rst} = 0;
36
37
          // Operate the DUT
         #17 mode = 1;
#22 mode = 0;
38
39
40
         #12 \mod = 1:
41
42
          #100 \text{ mode} = 0;
          $finish;
43
44
45
      initial $monitor("At time %0t, value = %h (%0d)", $time, cnt, cnt);
46
   endmodule
```

When we run the simulation again, the simulator reports about opening a file for dumping simulation results (Line 3 in Listing 4).

Listing 4: Simulating the testbench given in Listing 3

```
$ iverilog -l 'yosys-config --datdir/simcells.v' -o counter counter.v tbcounter-2.v
  $ vvp counter
  VCD info: dumpfile counter.vcd opened for output.
  At time 0, value = 00d (13)
  At time 15, value = 003 (3)
  At time 25, value = 3f9(-7)
  At time 35, value = 3fd
                           (-3)
  At time 45, value = 001
  At time 55, value = 3f7(-9)
10
  At time 65, value = 3ed (-19)
  At time 75, value = 3f1 (-15)
  At time 85, value = 3f5 (-11)
  At time 95, value = 3f9 (-7)
  At time 105, value = 3fd(-3)
15
  At time 115, value = 001 (1)
  At time 125, value = 005 (5)
16
  At time 135, value = 009
                           (9)
18
  At time 145, value = 00d
                            (13)
          155, value
                       011
  At time
20
  At time 165, value = 015 (21)
  tbcounter-2.v:36: $finish called at 166 (1ns)
```

With that file, we are now able to visualize simulation results using a waveform diagram:



Also, we can interactively inspect the waveforms in a waveform analyzer, such as GTKWave (Figure 3):



Figure 3: Interactive waveform analysis with GTKWave

By just adding four lines of code to our testbench, we are much more powerful in using the simulation results.

# **Checking values**

Let us now add another line of code to our testbench, which enables us to check the value of the counter. In Lines 46 and 49 in Listing 5 we add two *assert* statements, which test the counter value at the actual time step. From Line 20 in Listing 3 we know that the counter value in the last time step is  $0 \times 15$ . In the first *assert* statement in Line 46 in Listing 5, we check the counter value for the correct value, which silently passes in Listing 6. In the second *assert* statement in Line 49 in Listing 5, we deliberately check against an incorrect value, which errors out in Lines 21 and 22 in Listing 6.

Listing 5: Checking values of DUT output (*tb\_counter-3.v*)

```
timescale 1ns/1ns
   module tbcounter;
5
      // Specify dump file and the scope to be dumped
      initial begin
    $dumpfile("counter.vcd");
7
8
9
         $dumpvars(0, tbcounter);
      end
10
11
      // Define inputs
      reg clk=0;
12
      reg rst=0;
13
14
15
      reg mode=0;
      // Define outputs
16
17
      wire signed [9:0] cnt;
19
      // Instantiate design under test (DUT)
```

```
counter dut (
21
         .clk(clk),
22
         .rst(rst)
23
         .mode(mode)
24
         .cnt(cnt)
25
26
27
      // Generate clock
28
      always #5 clk = !clk;
29
30
      // Stimulate the design (vary input values)
31
      initial begin
32
33
         // Reset the DUT
         #5 rst = 1;
34
         #10 \text{ rst} = 0;
35
36
37
         // Operate the DUT
         #17 mode = 1;
38
39
         #22 \text{ mode} = 0;
40
         #12 mode = 1;
41
         #100 \text{ mode} = 0;
42
43
         // Verify the counter
44
45
         // Bare inline assertion with passing test case
46
         assert(cnt == 10'h015);
47
48
49
         // Bare inline assertion with failing test case
         assert(cnt == 10'h014);
50
51
         $finish;
52
53
      end
54
55
      initial $monitor("At time %0t, value = %h (%0d)", $time, cnt, cnt);
56
   endmodule
```

When running the simulation, we have to call *iverilog* with support for the *assert* statement. We add the -g2012 option to its command line, which selects IEEE1800-2012 language support.

Listing 6: Simulating the testbench given in Listing 5

```
$ iverilog -g2012 -1 'yosys-config --datdir/simcells.v' -o counter counter.v tbcounter-3.v
    vvp counter
  VCD info: dumpfile counter.vcd opened for output.
  At time 0, value = 00d (13)
  At time 15, value = 003 (3)
                           (-7)
(-3)
  At time 25, value = 3f9
  At time 35, value = 3fd
  At time 45, value = 001
                           (1)
                       3f7
  At time 55, value =
  At time
          65, value =
                       3ed
           75,
                       3f1
  At time
              value
  At time 85, value = 3f5
                           (-11)
13
  At time 95.
              value = 3f9(-7)
14
  At time 105, value =
                        3fd
                             (-3)
          115, value =
15
  At time
                        001
                            (1)
16
  At time 125, value
                        005
                            (5)
           135, value
  At time
                        009
  At time 145, value
                        00d
                            (13)
                             (17)
19
  At time 155, value
                        011
20
  At time 165, value
                      = 015
                            (21)
  ERROR: tbcounter-3.v:49:
       Time: 166 Scope: tbcounter
  tbcounter-3.v:51: $finish called at 166 (1ns)
```

The error message given in Line 21 in Listing 6 leads us to the failing assertion in the file  $tb\_counter$ -3.v, Line 49, which tells us that the counter value is not 0x14 at the respective time step. This is valuable debug information.

#### Checking first properties of our counter

We now have the tools available to check some properties of our counter, which must hold at every time step (or, *at every clock cycle*). We therefore add an *always* block in Lines 50 to 54 to our testbench given in Listing 7, holding assertions that check some of the counter's properties. The assertion in Line 51 checks at every clock cycle the counter value being not smaller than the specified minimum value; the assertion in Line 52 checks the counter value being not larger than the specified maximum value; the assertion in Line 53 checks the counter value being not equal to the specified invalid value.

Listing 7: Checking first properties of our counter (*tb\_counter-4.v*)

```
'timescale 1ns/1ns
```

```
module tbcounter;
5
6
7
      // Specify dump file and the scope to be dumped
      initial begin
          $dumpfile("counter.vcd");
          $dumpvars(0, tbcounter);
10
11
      // Define inputs
      reg clk=0;
12
13
      reg rst=0;
14
      reg mode=0;
15
      // Define outputs
wire signed [9:0] cnt;
16
17
18
19
      // Instantiate design under test (DUT)
20
      counter dut (
21
         .clk(clk),
22
          .rst(rst)
23
          .mode(mode)
24
          .cnt(cnt)
25
26
27
      // Generate clock
28
      always #5 clk = !clk;
29
30
      // Stimulate the design (vary input values)
31
      initial begin
32
33
          // Reset the DUT
34
35
         #10 \text{ rst} = 0;
36
37
          // Operate the DUT
38
         #17 \text{ mode} = 1;
39
         #22 \mod e = 0;
40
          #12 mode = 1;
41
         #100 \text{ mode} = 0;
42
43
         $finish:
44
45
46
47
      initial $monitor("At time %0t, value = %h (%0d)", $time, cnt, cnt);
48
      // Verify properties of the counter
49
50
      always @(posedge clk) begin
         assert (cnt ¿= -287);
assert (cnt ¡= 309);
51
52
53
          assert (cnt != 75);
54
      end
55
   endmodule
```

When running the simulation as shown in Listing 8, we notice that the DUT, our counter, does not violate any of the properties we check. However, looking at the counter values, we notice that the counter value does not even go near any of the values checked for in the assertions, which leaves the expressiveness of the simulation results in a questionable state.

Listing 8: Simulating the testbench given in Listing 7

```
\ iverilog -g2012 -l 'yosys-config --datdir/simcells.v' -o counter counter.v tbcounter-4.v
  $ vvp counter
  VCD info: dumpfile counter.vcd opened for output.
  At time 0, value = 00d (13)
  At time 15, value = 003(3)
  At time 25, value = 3f9
  At time 35, value = 3fd(-3)
  At time 45, value = 001 (1)
  At time 55, value = 3f7
                           (-9)
  At time 65, value = 3ed (-19)
  At time 75, value = 3f1 (-15)
  At time 85, value = 3f5 (-11)
13
  At time 95, value = 3f9(-7)
  At time 105, value = 3fd (-3)
14
15
  At time 115, value = 001 (1)
          125, value
                      = 005
16
  At time
                            (5)
  At time 135, value
                        009
  At time 145, value
                     = 00d
                            (13)
19
  At time 155, value
                     = 011
  At time 165, value = 015 (21)
20
  tbcounter-4.v:43: $finish called at 166 (1ns)
```

# Adding test cases

As we noticed earlier, the expressiveness of a simulation strongly depends on the values we check. In order to increase the expressiveness of simulation results generated by our testbench, the counter value must become close to the minimum, the maximum, and the invalid value. The only input values we have control over is the simulation time and the *mode* signal. We therefore increase simulation time in Listing 9, Lines 39 and 40, and vary the *mode* signal such that the counter value travels to the minimum and maximum values. In order to keep verification time low, we first count up, because the counter's increment is nearly half the value of its decrement, so we like to start counting up from near the reset value rather than counting up from the minimum value.

Listing 9: Increasing the number of test cases (*tb\_counter-5.v*)

```
'timescale 1ns/1ns
   module tbcounter;
      // Specify dump file and the scope to be dumped
6
7
      initial begin
         $dumpfile("counter.vcd");
         $dumpvars(0, tbcounter);
10
11
      // Define inputs
12
      reg clk=0;
13
      reg rst=0:
14
      reg mode=0;
15
16
      // Define outputs
17
      wire signed [9:0] cnt;
18
19
      // Instantiate design under test (DUT)
20
      counter dut (
21
         .clk(clk),
22
         .rst(rst)
23
         .mode(mode)
24
         .cnt(cnt)
25
26
27
      // Generate clock
28
      always #5 clk = !clk;
29
      // Stimulate the design (vary input values)
30
31
      initial begin
32
33
         // Reset the DUT
34
         #5 rst = 1;
35
         #10 \text{ rst} = 0;
36
37
         // Operate the DUT
38
         #17 mode = 1;
         #800 \text{ mode} = 0;
39
40
         #800 \text{ mode} = 1;
41
         #100 mode = 0;
42
         $finish;
43
44
45
46
47
      initial $monitor("At time %0t, value = %h (%0d)", $time, cnt, cnt);
48
49
      // Verify properties of the counter
50
      always @(posedge clk) begin
         assert (cnt ¿= -287);
assert (cnt ¡= 309);
51
52
53
         assert (cnt != 75);
54
      end
55
   endmodule
```

When simulating the testbench given in Listing 9, we learn from the simulation output given in Listing 10, that the counter reaches its maximum value at time 815 (Line 85), and that it comes near its minimum value at time 1415 (Line 144). In Lines 26, 27, 108 and 109 the counter value is close to its invalid value, but they do not match. We assume that it may be challenging to manually find a sequence of *mode* changes such that the counter value would match its invalid value, in order to verify the counter's correct behavior for that case.

Listing 10: Simulating the testbench given in Listing 9

```
$\frac{1}{\$\ \text{iverilog -g2012 -l 'yosys-config --datdir/simcells.v' -o counter counter.v tbcounter-5.v} \]

$\frac{1}{\$\ \text{vvp counter}} \]

$\text{VCD info: dumpfile counter.vcd opened for output.} \]

$\text{At time 0, value = 00d (13)} \]

$\text{At time 15, value = 003 (3)} \]

$\text{At time 25, value = 3f9 (-7)} \]

$\text{At time 35, value = 3fd (-3)} \]
```

```
8 At time 45, value = 001 (1)
  At time 55, value = 005
10
  At time 65, value = 009
                             (9)
11
  At time 75, value = 00d
                             (13)
  At time 85, value = 011
                             (17)
12
  At time 95, value = 015
13
                            (21)
  At time 105, value = 019
14
                              (25)
                              (29<sup>°</sup>)
  At time 115, value = 01d
16
  At time 125, value = 021
                              (33)
17
  At time 135, value = 025
                              (37)
  At time 145, value = 029
18
                              (41)
  At time 155, value = 02d
19
                              (45)
20
  At time 165, value
                              (49)
21
  At time 175, value = 035
                              (53)
22
  At time 185, value = 039
                              (57)
23
  At time 195, value = 03d
                              (61)
  At time 205, value = 041
24
                              (65)
  At time 215, value
25
                       = 045
                              (69)
  At time 225, value = 049
                              (73)
26
27
  At time 235, value
                       = 04d
                              (77)
28
  At time 245, value
                       = 051
                              (81)
29
  At time 255, value = 055
                              (85)
  At time 265, value = 059
30
                              (89)
  At time 275, value = 05d
                              (93)
31
32
  At time 285, value
                         061
                              (97)
  At time 295, value
                         065
                              (101)
33
                              (105)
  At time 305, value
                         069
34
35
  At time 315, value
                       = 06d
                              (109)
36
  At time 325, value = 071
                              (113)
37
           335, value = 075
                              (117)
  At time
38
  At time 345, value =
                         079
                              (121)
                         07d
                              (125)
39
  At time 355, value
40
  At time 365, value
                       = 081
                              (129)
41
  At time 375, value
                         085
                              (133)
42
  At time 385, value
                       = 089
                              (137)
  At time 395, value = 08d
43
                              (141)
  At time 405, value
                         091
                              (145)
44
45
  At time 415, value
                         095
                              (149)
46
  At time 425, value
                         099
                              (153)
47
  At time 435, value = 09d
                              (157)
                              (161)
48
  At time 445, value = 0a1
49
  At time 455, value = 0a5
                              (165)
                         0a9
  At time 465, value =
                              (169)
50
51
  At time 475, value
                         0ad
                              (173)
52
  At time 485, value
                         0b1
                              (177)
53
  At time 495, value
                         0b5
                              (181)
54
  At time 505, value = 0b9
                              (185)
  At time 515, value
                              (189)
                         0bd
55
           525, value
                              (193)
  At time
                         0c1
56
57
  At time 535, value =
                         0c5
                              (197)
58
  At time 545, value
                              (201)
59
  At time 555, value
                         0cd
                              (205)
60
  At time 565, value = 0d1
                              (209)
  At time 575, value = 0d5
61
                              (213)
                         0d9
62
  At time 585, value =
                              (217)
           595, value
                         0dd
                              (221)
  At time
63
  At time 605, value
                         0e1
                              (225)
64
  At time 615, value =
                              (229)
65
                         0e5
66
  At time 625, value = 0e9
                              (233)
67
  At time 635, value = 0ed
                              (237)
  At time 645, value = 0f1
                              (241)
68
  At time 655, value =
                         0f5
                              (245)
69
70
  At time 665, value
                              (249)
71
  At time 675, value
                       = 0 fd
                              (253)
72
  At time 685, value
                       = 101
                              (257)
73
  At time 695, value = 105
                              (261)
  At time 705, value = 109
74
                              (265)
75
           715, value = 10d
  At time
                              (269)
76
  At time 725, value = 111
                              (273)
77
  At time 735, value
                              (277)
78
  At time
           745, value = 119
                              (281)
  At time 755, value = 11d
79
                              (285)
  At time 765, value = 121
80
                              (289)
                              (293)
81
  At time
           775, value = 125
           785, value
                              (297)
82
  At time
                       = 129
           795, value
83
  At time
                       = 12d
                              (301)
84
  At time 805, value = 131
                              (305)
85
  At time 815, value = 135
                              (309)
                              (299)
  At time 835, value = 12b
86
87
  At time 845, value = 121
                              (289)
  At time 855, value
                              (279)
88
                         117
  At time 865, value
                              (269)
90
  At time 875, value
                       = 103
                              (259)
91
  At time 885, value
                       = 0f9
                              (249)
92
  At time 895, value = 0ef
                              (239)
  At time 905, value =
                         0e5
                              (229)
93
  At time
           915, value
                         0db
                              (219)
  At time 925, value =
                         0d1
                              (209)
                              (199)
  At time 935, value
                         0c7
97
  At time 945, value = 0bd
                              (189)
98 At time 955, value = 0b3 (179)
```

```
At time 965, value =
             975,
100
                             09f
                                  (159)
    At time
                   value
             985,
101
   At time
                   value
                             095
                                  (149)
102
    At time
             995
                   value
                             08b
                                  (139)
             1005.
                              081
103
    At time
                    value
                                   (129)
             1015,
                              077
                                   (119)
104
    At time
                    value
                    value
105
    Αt
       time
             1025,
                                   (109)
    Αt
       time
             1035,
                    value
                                   (99)
                    value
107
    Αt
       time
             1045
                              059
                                   (89)
108
    At
       time
             1055
                    value
                              04f
                                   (79)
109
    At time
             1065.
                    value
                              045
                                   (69)
110
             1075.
                    value
                              03b
                                   (59)
   Αt
       time
111
   Αt
       time
             1085,
                    value
    Αt
             1095,
                    value
                              027
                                   (39)
                                   (29)
             1105,
113
   Αt
       time
                    value
                              01d
114
    At.
       time
             1115
                    value
                              013
                                   (19)
115
   At time
             1125.
                    value
                              009
             1135,
                              3ff
116
    Αt
       time
                    value
                                   (-1)
117
   Αt
       time
             1145,
                    value
                              3f5
                                   (-11)
    Αt
             1155.
                                   (-21)
119
    Αt
       time
             1165,
                    value
                                   (-31)
120
   Αt
       time
             1175
                    value
                              3d7
                                    (-41)
121
    At time
             1185
                    value
                              3cd
                                   (-51)
             1195.
122
    Αt
       time
                    value
                              3c3
                                   (-61)
             1205,
                              3b9
                                   (-71)
123
    Αt
       time
                    value
124
    Αt
       time
             1215,
                    value
                              3af
                                     81)
                                    -91)
             1225,
                     value
                              39b
126
    At time
             1235
                    value
                                   (-101)
127
    At.
       time
             1245.
                    value
                              391
                                   (-111)
             1255.
                              387
128
    At time
                    value
                                   (-121)
129
             1265.
                    value
                              37d
                                   (-131)
   At time
130
   Αt
             1275,
       time
                    value
    Αt
       time
             1285,
                     value
                              369
                                     151
132
   Αt
       time
             1295,
                    value
                              35f
                                   (-161)
133
    Αt
       time
             1305.
                    value
                              355
                                   (-171)
134
   Αt
       time
             1315.
                    value
                              34b
                                   (-181)
             1325,
                              341
                                   (-191
135
    At time
                    value
                              337
             1335,
                                   (-201)
136
   Αt
       time
                    value
       time
             1345,
                              32d
                                     -211)
             1355,
138
    Αt
       time
                    value
                              323
                                     -221)
139
   Αt
       time
             1365
                    value
                              319
                                     -231
140
    At time
             1375.
                    value
                              30f
                                   (-241)
141
   Αt
       time
             1385,
                    value
                              305
                                   (-251)
142
   Αt
       time
             1395.
                    value
                              2fb
                                   (-261)
143
    Αt
       time
             1405,
                     value
                                     281)
    At
       time
             1415,
                    value
                              2e7
145
    Αt
       time
             1635
                    value
                              2eb
                                     -277
146
   At.
       time
             1645.
                    value
                              2ef
                                   (-273)
                              2f3
147
   At time
             1655.
                                   (-269)
                    value
                              2f7
148
                    value
   At time
             1665,
                                     -265)
    Αt
       time
             1675,
                    value
                                     261
150
    Αt
       time
             1685,
                    value
                              2ff
151
   At time
             1695,
                    value
                              303
                                     -253
152
    At time
             1705,
                    value
                              307
                                   (-249)
153
   At time
             1715.
                    value
                            =
                              30b
                                   (-245)
                              30f
                                   (-241)
             1725,
154
    At time
                    value
    tbcounter-5.v:43: $finish called at 1732 (1ns)
```

#### Considering the past

By now, we have a considerable testbench which enables us to check some properties of our counter. However, to check the correctness of its increment and decrement, we need to know the counter's value of the previous clock cycle. Also, due to simulation semantics, the counter value only updates at the end of a (simulation) time step in non-blocking statements (which may be assumed in a synchronous design). Therefore, upon reset, the counter is assigned its initial value at the very end of the time step in which the reset occurs, which makes it impossible to test correct reset behavior by comparing the counter value to its initial value when the reset signal is 1. Therfore, we need to observe the reset signal's value of the previous clock cycle. Likewise, we need to know the *mode* signal's value of the previous clock cycle in order to determine from the current counter value if it was correctly incremented or decremented.

The solution is fairly simple: We need to delay signals *rst*, *mode*, and *cnt* by one clock cycle, and store their delayed versions in signals *past\_rst*, *past\_mode*, and *past\_cnt*, as shown in Listing 11, Lines 50 to 57. Having available these past values allows us to check more properties of our counter in Lines 64 to 78.

Listing 11: Checking more properties of our counter, considering past values (tb\_counter-6.v)

```
'timescale 1ns/1ns

module tbcounter;

// Specify dump file and the scope to be dumped
initial begin
stumpfile("counter.vcd");
timescale 1ns/1ns

module tbcounter;

// Specify dump file and the scope to be dumped
initial begin
stumpfile("counter.vcd");
the state of the scope to be dumped
the scope to be dumped
initial begin
the scope to be dumped
the scope to be du
```

```
10
11
       // Define inputs
12
       reg c1k=0;
13
       reg rst=0;
14
       reg mode=0;
15
      // Define outputs
wire signed [9:0] cnt;
16
17
18
19
       // Instantiate design under test (DUT)
20
       counter dut (
21
          .clk(clk),
22
          .rst(rst),
23
           .mode(mode),
24
25
           .cnt(cnt)
26
27
       // Generate clock
28
       always #5 clk = !clk;
29
30
       // Stimulate the design (vary input values)
31
       initial begin
32
33
           // Reset the DUT
          #5 rst = 1;
#10 rst = 0;
34
35
36
37
38
          // Operate the DUT
          #17 \text{ mode} = 1;
39
          #800 \text{ mode} = 0;
40
          #800 \text{ mode} = 1;
41
          #100 \text{ mode} = 0;
42
43
44
          $finish;
45
46
47
       initial $monitor("At time %0t, value = %h (%0d)", $time, cnt, cnt);
48
       // Store past values of signals rst, mode, and cnt
49
      reg signed [9:0] pastcnt = 'x;
reg pastmode = 'x;
50
      reg pastmode = 'x
reg pastrst = 'x;
51
52
53
       always @(posedge clk) begin
          pastcnt ;= cnt;
pastrst ;= rst;
54
55
      pastmode ;= rst;
end
56
57
58
59
       // Verify properties of the counter
      always @(posedge clk) begin
assert (cnt ¿= -287);
assert (cnt ¡= 309);
assert (cnt != 75);
if(pastrst) begin
60
61
62
63
64
65
              assert(cnt == 13);
          end else if (!rst) begin
67
              if (pastmode == 1) begin
68
69
                  // Add the remaining assertions here to check the correct
70
                  // increment
71
72
              end else begin // pastmode == 0
73
74
75
76
                  // Add the remaining assertions here to check the correct
                  // decrement
77
              end
78
          end
79
       end
80
   endmodule
```

Now, let's assume that our counter implements a reset value different than specified, for instance 14 instead of 13. Then, the simulation of Listing 11 results in the output given in Listing 12, with an error message at the very top (Lines 5 and 6), which may be easily overlooked. It would be desirable if the simulation would stop, and the simulator would exit with an exit code different than 0 in case an assertion fails.

Listing 12: Simulating the testbench given in Listing 11

```
$ iverilog -g2012 -1 'yosys-config --datdir/simcells.v' -o counter counter.v tbcounter-6.v

$ vvp counter

VCD info: dumpfile counter.vcd opened for output.
4 At time 0, value = 00e (14)

ERROR: tbcounter-6.v:72:
    Time: 15 Scope: tbcounter

At time 15, value = 004 (4)
```

```
8 At time 25, value = 3fa (-6)
   At time 35, value = 3fe
                             (-2)
10
  At time 45, value = 002
                              (2)
11
   At time 55, value = 006
                             (6)
  At time 65, value = 00a
12
                             (10)
  At time 75, value = 00e
13
  At time 85, value = 012
14
   At time 95, value = 016 (22)
  At time 105, value = 01a (26)
16
17
  At time 115, value = 01e
                              (30)
18
  At time 125, value = 022
                              (34)
  At time 135, value = 026
19
                              (38)
20
  At time 145, value
                              (42)
21
   At time 155, value = 02e
                               (46)
22
  At time 165, value = 032 (50)
23
   At time 175, value = 036
                              (54)
  At time 185, value = 03a
                              (58)
24
   At time 195, value = 03e
25
                               (62)
  At time 205, value = 042
                              (66)
26
27
   At time 215, value = 046
                               (70)
28
   At time 225, value = 04a
                               (74)
29
  At time 235, value = 04e
                               (78)
  At time 245, value = 052
30
                              (82)
  At time 255, value = 056
                              (86)
31
32
  At time 265, value =
                          05a
                               (90)
   At time 275, value
                       = 05e
                               (94)
33
   At time 285, value
                          062
                              (98)
34
35
  At time 295, value = 066
                              (102)
36
  At time 305, value =
                          06a (106)
  At time 315, value = 06e
37
                              (110)
38
  At time 325, value = 072
                              (114)
   At time 335, value
                          076
39
                               (118)
40
  At time 345, value
                       = 07a
                              (122)
41
  At time 355, value = 07e
                              (126)
42
  At time 365, value
                       = 082
                              (130)
  At time 375, value = 086
At time 385, value = 08a
43
                              (134)
                              (138)
44
45
  At time 395, value
                          08e
                               (142)
46
   At time 405, value
                          092
                               (146)
47
   At time 415, value = 096
                               (150)
48
  At time 425, value = 09a
                              (154)
                              (158)
  At time 435, value = 09e
49
  At time 445, value = 0a2
                              (162)
50
  At time 455, value
51
                          0a6
                               (166)
52
   At time 465, value
                          0aa
                              (170)
53
  At time 475, value
                          0ae
                               (174)
54
  At time 485, value = 0b2
                               (178)
  At time 495, value = At time 505, value =
                               (182)
                          0b6
55
56
                          0ba
                              (186)
57
  At time 515, value =
                          0be
                              (190)
                              (194)
58
  At time 525, value
59
  At time 535, value
                          0c6
                               (198)
60
  At time 545, value = 0ca
                              (202)
61
  At time 555, value = 0ce
                              (206)
                              (210)
  At time 565, value = 0d2
At time 575, value = 0d6
62
                              (214)
63
  At time 585, value =
                          0da
                              (218)
64
  At time 595, value = 0de
                              (222)
                               (226)
66
   At time 605, value = 0e2
67
  At time 615, value = 0e6
                               (230)
  At time 625, value = 0ea
                              (234)
68
  At time 635, value =
                          0ee
                               (238)
69
70
   At time 645, value
                              (242)
71
   At time 655, value = 0f6
                               (246)
72
  At time 665, value = 0 fa
                              (250)
73
   At time 675, value = 0fe
                               (254)
  At time 685, value = 102
At time 695, value = 106
74
                              (258)
75
                              (262)
76
  At time 705, value = 10a (266)
77
   At time 715, value
                              (270)
78
  At time 725, value = 112
                              (274)
  At time 735, value = 116 (278)
79
   At time 745, value = 11a
80
                              (282)
  At time 755, value = 11e
At time 765, value = 122
                              (286)
81
82
                              (290)
  At time 775, value = 126
83
                              (294)
84
  At time 785, value = 12a
                              (298)
   At time 795, value = 12e
                              (302)
85
  At time 805, value = 132 (306)
86
  At time 1035, value = 128 (296)
87
  At time 1045, value = 11e
                               (286)
88
   At time 1055, value = 114
                               (276)
90
   At time 1065, value = 10a
                               (266)
                               (256)
91
   At time 1075, value = 100
92
   At time 1085, value = 0f6
                                (246)
   At time 1095, value = 0ec
93
                                (236)
   At time 1105, value = 0e2
                                (226)
  At time 1115, value = 0d8
                                (216)
   At time 1125, value = 0ce
97
  At time 1135, value = 0c4
                               (196)
98 At time 1145, value = 0ba (186)
```

```
At time 1155, value =
100
       time
             1165,
                    value
                             0a6
   Αt
             1175,
101
   At time
                    value
                             090
                                   (156)
102
   At time
             1185.
                    value
                             092
                                   (146)
             1195.
                             088
103
   At time
                    value
                                   (136)
104
   At time
             1205,
                    value
                                   (126)
105
   Αt
      time
             1215,
                    value
                                   (116)
   Αt
       time
             1225,
                    value
107
   Αt
       time
             1235
                    value
                             060
                                   (96)
108
   At.
       time
             1245.
                    value
                             056
                                   (86)
109
   At time
             1255.
                    value
                             04c
                                   (76)
110
   Αt
      time
             1265.
                    value
                             042
                                   (66)
                    value
111
   Αt
      time
             1275,
                                   (56)
   Αt
       time
             1285,
                    value
                             02e
113
   Αt
      time
             1295,
                    value
                             024
                                   (26)
114
   At.
       time
             1305.
                    value
                             01a
115
   At time
             1315.
                    value
                             010
                                   (16)
   Αt
      time
             1325,
                    value
                             006
116
                                   (6)
117
   Αt
       time
             1335,
                    value
                             3fc
       time
             1345.
                    value
             1355,
119
   Αt
       time
                    value
120
   Αt
       time
             1365.
                    value
                             3de
                                   (-34)
121
   At time
             1375
                    value
                             3d4
                                   (-44)
             1385,
                                   (-54)
122
   Αt
      time
                    value
                             3ca
123
             1395,
                    value
   Αt
      time
                             3c0
                                   (-64)
   Αt
       time
             1405,
                    value
   Αt
       time
             1415,
                    value
126
   At time
             1425
                    value
                             3a2
                                   (-94)
127
   At.
       time
             1435.
                    value
                             398
                                   (-104)
                             38e
128
             1445.
   At time
                    value
                                   (-114)
129
             1455
                    value
                             384
                                   (-124)
   At time
130
   Αt
      time
             1465,
                    value
                                   (-134)
   Αt
                             370
131
       time
             1475,
                    value
                                   (-144)
                                   (-154)
132
   Αt
      time
             1485.
                    value
                             366
133
   Αt
      time
             1495.
                    value
                             35c
                                   (-164)
134
   Αt
       time
             1505.
                    value
                             352
                                   (-174)
             1515,
                             348
                                   (-184)
135
   Αt
      time
                    value
                              33e
136
             1525,
                    value
   Αt
       time
   Αt
             1535,
                             334
                                   (-204)
       time
                    value
138
   Αt
       time
             1545,
                    value
                             32a
139
   Αt
      time
             1555
                    value
                             320
                                   (-224)
140
   At time
             1565.
                    value
                             316
                                   (-234)
141
   Αt
      time
             1575,
                    value
                             30c
                                   (-244)
142
   Αt
      time
             1585,
                    value
                                   (-254)
   Αt
       time
             1595,
                    value
                             2f8
             1605,
                    value
                                     274)
   Αt
       time
145
   Αt
      time
             1615
                    value
                             2e4
                                   (-284)
146
   At.
       time
             2035.
                    value
                             2e8
                                   (-280)
             2045.
                             2ec
147
   At time
                    value
                                   (-276)
             2055,
                             2f0
148
   At time
                    value
                                   (-272)
   Αt
      time
             2065,
                    value
150
   Αt
       time
             2075.
                    value
                             2f8
             2085,
151
   At time
                    value
                             2fc
                                   (-260)
             2095,
152
   At time
                    value
                             300
                                   (-256)
153
   At time
             2105.
                    value
                             304
                                   (-252)
             2115,
                             308
                                   (-248)
154
   At time
                    value
            2125,
                             30c
                                   (-244)
   At time
                    value
   tbcounter-6.v:50: $finish
                                  called at 2132 (1ns)
```

#### Change error behavior

It is highly desirable to notice errors that occur during simulation. Because the default behavior of *Icarus Verilog* is to continue simulation after an assertion fails, we define the macro *myassert* that implements the desired behavior to exit the simulation in Listing 13, Lines 3 to 8. Calls to the *assert* statement are replaced by calls to the *myassert* macro in Lines 69 to 71 and 73.

We choose a *macro* over a *task* in order to preserve expressive error messages with line numbers referring to the failed assertion (a macro is expanded where it is called). When encapsulating *myassert* in a task, an error message would refer to the line number in which *assert* is called from within the task, making it potentially harder to debug the DUT.

Listing 13: Changing simulation behavior in case of an error (tb\_counter-7.v)

```
initial begin
          $dumpfile("counter.vcd");
15
16
          $dumpvars(0, tbcounter);
17
18
19
      // Define inputs
20
      reg c1k=0;
      reg rst=0;
21
22
      reg mode=0;
23
24
      // Define outputs
25
      wire signed [9:0] cnt;
26
27
      // Instantiate design under test (DUT)
28
      counter dut (
29
          .clk(clk)
30
          .rst(rst)
31
          .mode(mode),
32
          .cnt(cnt)
33
34
35
      // Generate clock
36
37
      always #5 clk = !clk;
38
      // Stimulate the design (vary input values)
39
      initial begin
40
41
          // Reset the DUT
42
43
          #5 rst = 1;
#10 rst = 0;
44
45
          // Operate the DUT
46
          #17 \text{ mode} = 1;
          \#800 \mod = 0;
47
          #800 mode = 1;
#100 mode = 0;
48
49
50
51
          $finish;
52
53
54
55
      initial $monitor("At time %0t, value = %h (%0d)", $time, cnt, cnt);
56
57
      // Store past values of signals rst, mode, and cnt
      reg signed [9:0] pastcnt = reg pastmode = 'x; reg pastrst = 'x;
58
59
60
      always @(posedge clk) begin
61
         pastcnt ;= cnt;
pastrst ;= rst;
62
63
64
          pastmode ;= mode;
65
66
      // Verify properties of the counter
67
      'myassert (cnt := 309);
'myassert (cnt != 75);
68
69
70
71
72
73
74
          if(pastrst) begin
          'myassert(cnt == 13);
end else if (!rst) begin
75
             if (pastmode == 1) begin
76
77
                 // Add the remaining assertions here to check the correct
78
79
80
             end else begin // pastmode == 0
81
82
                 // Add the remaining assertions here to check the correct
83
                 // decrement
84
85
             end
86
          end
87
      end
88
```

Simulating the testbench given in Listing 13, assuming the DUT to implement incorrect reset behavior, results in the output given in Listing 14, which is way more readable than the output given in Listing 12 for the same simulation. Also, the generated debug output given in Lines 5 and 6 is more expressive than in Lines 5 and 6 of Listing 12.

Listing 14: Simulating the testbench given in Listing 11

```
$ iverilog -g2012 -1 'yosys-config --datdir/simcells.v' -o counter counter.v tbcounter-6.v

$ vvp counter

VCD info: dumpfile counter.vcd opened for output.

4 At time 0, value = 00e (14)

ERROR: tbcounter-7.v:75: FAIL: assert(cnt == 13) in tbcounter-7.v:72

Time: 15 Scope: tbcounter
```

Table 4: Functionality of buggy counters

| Target | Functionality                                                   |
|--------|-----------------------------------------------------------------|
| 0      | correctly implemented counter                                   |
| 1      | cnt may be greater than maximum                                 |
| 2      | cnt may be less than minimum                                    |
| 3      | cnt does not jump over invalid                                  |
| 4      | increment is 1 larger than specified                            |
| 5      | cnt jumps (up) over invalid to value 1 larger than specified    |
| 7      | decrement is 1 less than specified                              |
| 8      | cnt jumps (down) over invalid to value 1 smaller than specified |
| 10     | initial value is 1 larger than specified                        |

```
7 At time 15, value = 004 (4)
```

# Summary

We developed a pretty powerful testbench, ready to be extended and used to verify the conformance of the counter's behavior to its specification.

#### Your task

Your task is to complete the testbench given in Listing 13 such that it checks the correct behavior of the counter's increment, decrement and invalid value.

The goal of the testbench is that you can use it to verify the implementation of your counter. As you have not yet implemented your counter, we provide you with a sample implementation *counter.v*, which you can use to develop your testbench.

In order to enable you debugging your assertions, we provide you with a set of test counters, almost all of them violating properties specified for our counter. When you simulate your testbench along with each of the test counters, an assertion should fail and the simulation should exit (except for *counter-0.v*). This means that your assertions cover that error, which is good for your testbench.

Along with the set of test counters, we provide a *Makefile* to increase usability, and to speed up the verification process. The targets defined in the Makefile are numbers 0, 1, 2, 3, 4, 5, 7, 8, 10. Table 4 summarizes the buggy counters' functionality.

It is fairly straight-forward to execute the simulations of buggy counters using your testbench. In the command-line terminal, change to the directory of your Makefile, and run the command given in Listing 15.

Listing 15: Simulate test counter 10

1 make 10

# Submission details

To submit your testbench, please attach your completed version of *tb\_counter.v* to your submission e-mail.