

# CHITTAGONG UNIVERSITY OF ENGINEERING & TECHNOLOGY

---

## Department of Electronics And Telecommunication Engineering



## Lab Report

---

**Experiment Name:** Introduction to Verilog Testbenches and Functional Verification

**Experiment No.:** 10

**Course Title:** VLSI Technology Sessional

**Course No.:** ETE 404

**Date of Experiment:** 20-10-2024

**Date of Submission:** 27-10-2024

| Submitted By                                                                              | Submitted To                                                             |
|-------------------------------------------------------------------------------------------|--------------------------------------------------------------------------|
| <b>Name:</b> M.I. Yasir Arafat<br><b>ID:</b> 1908031<br><b>Level:</b> 4<br><b>Term:</b> I | <b>Arif Istiaque Rupom</b><br>Assistant Professor,<br>Dept. of ETE, CUET |

## Objectives:

1. To familiarize with Model-sim.
2. To familiarize with writing test-benches.
3. To verify combinational circuits imposing test vectors.

## Required Software:

1. Model-sim-Altera

## Design Process:

The test-bench codes written in Hardware Description Language (HDL) are run in the Model-sim-Altera software to evaluate the behaviour of Design Under Test (DUT). In this process, four of the components are simulated to check the behaviour with their test benches, which are derived from there truth table. The four components are listed below with their HDL code, test-bench code, simulation waveform, and the transcripts for matching the output.

### 1. 2X1 Multiplexer

After creating the new file name as ‘mux\_2x1’ the code editor window has been opened. The below code has been written in the editor.



The screenshot shows a code editor window titled 'C:/altera/13.0sp1/mux\_2x1.v - Default'. The code is a Verilog module definition for a 2x1 multiplexer. It includes input wires 'a' and 'b', a control wire 'sel', and an output wire 'y'. The logic is defined using an 'assign' statement that selects 'b' if 'sel' is high, and 'a' otherwise. The code is numbered from 1 to 13.

```
1 module mux_2x1 (
2     input wire a,
3     input wire b,
4     input wire sel,
5     output wire y
6 );
7
8     // Multiplexer logic
9     assign y = (sel) ? b : a;
10
11 endmodule
12
13
```

Fig. 1

Then another new file ‘mux\_2x1\_tb’ has been opened from the project panel to write the test-benches which is shown below.

```

M C:/altera/13.0sp1/mux_2x1_tb.v - Default
Ln# | 
1   module mux_2x1_tb;
2
3     // Inputs to the DUT (reg type for testbench)
4     reg a, b, sel;
5
6     // Output from the DUT (wire type for testbench)
7     wire y;
8
9     // Instantiate the DUT
10    mux_2x1 uut (
11      .a(a),
12      .b(b),
13      .sel(sel),
14      .y(y)
15    );
16
17    // Test stimulus
18    initial begin
19
20      // Initialize inputs
21      a = 0;
22      b = 0;
23      sel = 0;
24
25      // Apply test vectors
26      #10 a = 0; b = 0; sel = 0; // Expect y = a = 0
27      #10 a = 0; b = 1; sel = 0; // Expect y = a = 0
28      #10 a = 1; b = 0; sel = 0; // Expect y = a = 1
29      #10 a = 1; b = 1; sel = 0; // Expect y = a = 1
30      #10 a = 0; b = 0; sel = 1; // Expect y = b = 0
31      #10 a = 0; b = 1; sel = 1; // Expect y = b = 1
32      #10 a = 1; b = 0; sel = 1; // Expect y = b = 0
33      #10 a = 1; b = 1; sel = 1; // Expect y = b = 1
34
35      // Finish the simulation
36      #10 $finish;
37    end
38
39    // Monitor the output
40    initial begin
41      $monitor("At time %0t: a = %b, b = %b, sel = %b -> y = %b",
42                  $time, a, b, sel, y);
43    end
44
45  endmodule
46
47

```

Fig. 2:

When the compile all was pressed then the code was compiled, and then we could see the simulation from the 'start simulation'. Moreover, we executed the Wave function to visualize the waveform generated from the HDL code. Then, adding the whole Objects to the wave window we got the outcome. The transcript with the proper test benches is shown below in Figure 3.

```

Transcript :
add wave -position insertpoint \
sim:/mux_2x1_tbsel
add wave -position insertpoint \
sim:/mux_2x1_tb/y
VSIM 6> run
# At time 0: a = 0, b = 0, sel = 0 -> y = 0
run
run
# At time 20: a = 0, b = 1, sel = 0 -> y = 0
run
# At time 30: a = 1, b = 0, sel = 0 -> y = 1
run
# At time 40: a = 1, b = 1, sel = 0 -> y = 1
run
# At time 50: a = 0, b = 0, sel = 1 -> y = 0
run
# At time 60: a = 0, b = 1, sel = 1 -> y = 1
run
# At time 70: a = 1, b = 0, sel = 1 -> y = 0
run
# At time 80: a = 1, b = 1, sel = 1 -> y = 1
# ** Note: $finish : C:/altera/13.0sp1/mux_2x1_tb.v(36)
#   Time: 90 ps  Iteration: 0  Instance: /mux_2x1_tb

```

Fig. 3

Before getting the proper waveform, the timing was set to 10ps. The waveform we got is shown below in figure 4.



Fig.4

To get the view of the wave differently, such as with 0s and 1s, the Literal format has been executed. The binary format waveform is shown in figure 5.



Fig. 5

## 2. Full Adder

A new file named ‘full\_adder’ was added to the working file panel to write the HDL code of full adder and then visualize the waveform created from the code. The code for Full Adder code is shown below in figure 6.

```
C:/altera/13.0sp1/Full_adder.v - Default
Ln# | 1 module full_adder (
2   input a, // First input bit
3   input b, // Second input bit
4   input cin, // Carry input
5   output sum, // Sum output
6   output cout // Carry output
7 );
8
9   // Full adder logic: sum and carry-out
10  assign {cout, sum} = a + b + cin;
11
12 endmodule
13
14
```

```

M C:/altera/13.0sp1/Full_adder_tb.v -Default
Ln# | 
1   module full_adder_tb;
2     reg a, b, cin; // Input signals for the DUT
3     wire sum, cout; // Output signals from the DUT
4     // Instantiate the full adder
5     full_adder uut (
6       .a(a),
7       .b(b),
8       .cin(cin),
9       .sum(sum),
10      .cout(cout)
11    );
12    // Apply test vectors
13    initial begin
14      // Initialize inputs
15      a = 0; b = 0; cin = 0;
16
17      // Test case 1: 0 + 0 + 0 = 0, carry = 0
18      #10 a = 0; b = 0; cin = 0;
19      // Test case 2: 0 + 0 + 1 = 1, carry = 0
20      #10 a = 0; b = 0; cin = 1;
21      // Test case 3: 0 + 1 + 0 = 1, carry = 0
22      #10 a = 0; b = 1; cin = 0;
23      // Test case 4: 0 + 1 + 1 = 0, carry = 1
24      #10 a = 0; b = 1; cin = 1;
25      // Test case 1: 1 + 0 + 0 = 1, carry = 0
26      #10 a = 1; b = 0; cin = 0;
27      // Test case 2: 1 + 0 + 1 = 0, carry = 1
28      #10 a = 1; b = 0; cin = 1;
29      // Test case 3: 1 + 1 + 0 = 0, carry = 1
30      #10 a = 1; b = 1; cin = 0;
31      // Test case 4: 1 + 1 + 1 = 1, carry = 1
32      #10 a = 1; b = 1; cin = 1;
33
34      // Finish simulation
35      #10 $finish;
36    end
37    // Monitor the inputs and outputs
38    initial begin
39      $monitor("At time %0t: a = %b, b = %b, cin = %b -> sum = %b,
40      cout = %b",
41      $time, a, b, cin, sum, cout);
42    end

```

Fig. 6

The testbench code was also run, and then the simulation was done as previously. The waveform we got was matching the transcript with the testbenches. Figure 7 shows the transcript and then the figure 8 shows the waveform which is matching the transcript.

```

Transcript =====
VSIM 2> run
# At time 0: a = 0, b = 0, cin = 0 -> sum = 0,
# cout = 0
run
run
# At time 20: a = 0, b = 0, cin = 1 -> sum = 1,
# cout = 0
run
# At time 30: a = 0, b = 1, cin = 0 -> sum = 1,
# cout = 0
run
# At time 40: a = 0, b = 1, cin = 1 -> sum = 0,
# cout = 1
run
# At time 50: a = 1, b = 0, cin = 0 -> sum = 1,
# cout = 0
run
# At time 60: a = 1, b = 0, cin = 1 -> sum = 0,
# cout = 1
run
# At time 70: a = 1, b = 1, cin = 0 -> sum = 0,
# cout = 1
VSIM 3> run
# At time 80: a = 1, b = 1, cin = 1 -> sum = 1,
# cout = 1
# ** Note: $finish    : C:/altera/13.0sp1/Full_adder_tb.v(35)
#   Time: 90 ps  Iteration: 0  Instance: /full_adder_tb
.

```

Fig, 7:



Fig, 8

### 3. 2X4 Decoder

We also worked with the 2x4 decoder HDL to verify the functionality of the code and the output as waveform with the given testbenches. The verilog code is given below in figure 9 along with the testbenches in figure 10.

```

M C:/altera/13.0sp1/decoder.v (/decoder_2x4_tb/uut) - Default :-----:-----
Ln# | 
1   module decoder_2x4 (
2     input [1:0] in, // 2-bit input
3     output [3:0] out // 4-bit output
4   );
5
6   // Decoder logic
7   assign out = (in == 2'b00) ? 4'b0001 : // When in = 00, out = 0001
8       (in == 2'b01) ? 4'b0010 : // When in = 01, out = 0010
9       (in == 2'b10) ? 4'b0100 : // When in = 10, out = 0100
10      (in == 2'b11) ? 4'b1000 : // When in = 11, out = 1000
11      4'b0000; // Default case
12
13 endmodule
14
15

```

Fig. 9

```

1  module decoder_2x4_tb;
2    reg [1:0] in; // 2-bit input for the DUT
3    wire [3:0] out; // 4-bit output from the DUT
4    // Instantiate the 2-to-4 decoder
5    decoder_2x4 uut (
6      .in(in),
7      .out(out)
8    );
9    // Apply test vectors
10   initial begin
11     // Test case 1: in = 00 -> out = 0001
12     #10 in = 2'b00;
13     // Test case 2: in = 01 -> out = 0010
14     #10 in = 2'b01;
15     // Test case 3: in = 10 -> out = 0100
16     #10 in = 2'b10;
17     // Test case 4: in = 11 -> out = 1000
18     #10 in = 2'b11;
19     // Finish simulation
20     #10 $finish;
21   end
22   // Monitor the inputs and outputs
23   initial begin
24     $monitor("At time %0t: in = %b -> out = %b", $time, in, out);
25   end
26 endmodule
--
```

Fig. 10

After running the HDL code, the simulation part was done. The simulation of the decoder output waveform perfectly matched the transcript. Both are shown below in figure 11 and 12. The binary literal waveform has been showed in the figure 12 for better understanding.

```

VSIM 4> run    -  -
# At time 0: in = xx -> out = xxxx
run
# At time 10: in = 00 -> out = 0001
run
# At time 20: in = 01 -> out = 0010
run
# At time 30: in = 10 -> out = 0100
run
# At time 40: in = 11 -> out = 1000
# ** Note: $finish      : C:/altera/13.0spl/decoder_tb.v(27)
#      Time: 50 ps  Iteration: 0  Instance: /decoder_2x4_tb

```

Fig. 11



Fig. 12

## 4.2 Bit Magnitude Comparator

A comparator used to compare two binary numbers each of two bits is called a 2-bit magnitude comparator. It consists of four inputs and three outputs to generate less than, equal to, and greater than between two binary numbers. The code regarding the basics is provided as HDL code in the editor depicted in figure 13 along with the testbenches in figure 14.



The screenshot shows a code editor window with the title bar "C:/altera/13.0sp1/Magnitude\_com.v". The code is a VHDL module named "comparator\_2bit". It has two input ports, "a" and "b", both of which are 2-bit unsigned integers. It has three output ports: "gt" (greater than), "lt" (less than), and "eq" (equal). The logic is implemented using an always block that checks the values of "a" and "b" and sets the appropriate output signals based on their comparison. The code is color-coded for readability, with keywords in red and identifiers in blue.

```
Ln#  
1  module comparator_2bit (  
2      input [1:0] a, // First 2-bit input  
3      input [1:0] b, // Second 2-bit input  
4      output reg gt, // Output: a > b  
5      output reg lt, // Output: a < b  
6      output reg eq   // Output: a == b  
7  );  
8      // Always block for combinational logic  
9      always @(*) begin  
10         if (a > b) begin  
11             gt = 1;  
12             lt = 0;  
13             eq = 0;  
14         end  
15         else if (a < b) begin  
16             gt = 0;  
17             lt = 1;  
18             eq = 0;  
19         end  
20         else begin  
21             gt = 0;  
22             lt = 0;  
23             eq = 1;  
24         end  
25     end  
26 endmodule
```

Fig. 13

```

M C:/altera/13.0sp1/Magnitude_com_tb.v (/tb_comparator_2bit) - Default
Ln#
1 module tb_comparator_2bit;
2     reg [1:0] a, b; // 2-bit inputs for the DUT
3     wire gt, lt, eq; // Outputs from the DUT
4     // Instantiate the 2-bit comparator
5     comparator_2bit uut (
6         .a(a),
7         .b(b),
8         .gt(gt),
9         .lt(lt),
10        .eq(eq)
11    );
12    // Apply test vectors
13    initial begin
14        // Test case 1: a = 00, b = 01 -> a < b
15        #10 a = 2'b00; b = 2'b01;
16
17        // Test case 2: a = 10, b = 10 -> a == b
18        #10 a = 2'b10; b = 2'b10;
19
20        // Test case 3: a = 11, b = 01 -> a > b
21        #10 a = 2'b11; b = 2'b01;
22
23        // Test case 4: a = 01, b = 10 -> a < b
24        #10 a = 2'b01; b = 2'b10;
25
26        // Finish simulation
27        #10 $finish;
28    end
29
30    // Monitor the inputs and outputs
31    initial begin
32        $monitor("At time %0t: a = %b, b = %b -> gt = %b, lt = %b, eq = %b",
33                    $time, a, b, gt, lt, eq);
34    end
35 endmodule
36
37

```

Fig. 14

The simulation waveform is shown below in figure 15. The transcript which has been run in the terminal is also shown in figure 16.



Fig. 15

```

VSIM 7> run
# At time 0: a = xx, b = xx -> gt = x, lt = x, eq = x
run
# At time 10: a = 00, b = 01 -> gt = 0, lt = 1, eq = 0
run
# At time 20: a = 10, b = 10 -> gt = 0, lt = 0, eq = 1
run
# At time 30: a = 11, b = 01 -> gt = 1, lt = 0, eq = 0
run
# At time 40: a = 01, b = 10 -> gt = 0, lt = 1, eq = 0
# ** Note: $finish : C:/altera/13.0sp1/Magnitude_com_tb.v(27)
#   Time: 50 ps Iteration: 0 Instance: /tb_comparator_2bit

```

Fig. 16

## Result Analysis:

### 1. 2X1 Multiplexer

If we take the waveform from **figure 5** we can see that the simulation of the waveform of the multiplexer is accurate because of the matching with the transcript in **figure 3**. There are 8 combinations of inputs and outputs. For an example,

- At the beginning when A and B both are 0, and the selection pin is 0 the output remains 0
  - But while the A is set to 1 and B is 0, but the selection pin is 0, the output changed to 1.
- This behaviour is just as opposite while we see while selection pin is 1 but the output is 0.

### 2. Full Adder

From **figure 8** we can see the waveform depicting the input and output combination of a full adder. From the truth table of a Full Adder we can match the transcript as well as the waveform. The truth table is:

| A | B | Cin | Sum | Cout |
|---|---|-----|-----|------|
| 0 | 0 | 0   | 0   | 0    |
| 0 | 0 | 1   | 1   | 0    |
| 0 | 1 | 0   | 1   | 0    |
| 0 | 1 | 1   | 0   | 1    |
| 1 | 0 | 0   | 1   | 0    |
| 1 | 0 | 1   | 0   | 1    |
| 1 | 1 | 0   | 0   | 1    |

|   |   |   |   |   |
|---|---|---|---|---|
| 1 | 1 | 1 | 1 | 1 |
|---|---|---|---|---|

From the transcript, the waveform can be matched.

- At the beginning, when A = 0 and B = 0, Cin = 0, the Sum and Cout are 0 -  
When A = 1 and B = 0, Cin = 0 and the Sum = 1 but Cout = 0

### 3. 2X4 Decoder

All of the 4 combinations have been tested with the testbenches along with the simulation waveform.  
The truth table is given as:

| A | B | Y0 | Y1 | Y2 | Y3 |
|---|---|----|----|----|----|
| 0 | 0 | 1  | 0  | 0  | 0  |
| 0 | 1 | 0  | 1  | 0  | 0  |
| 1 | 0 | 0  | 0  | 1  | 0  |
| 1 | 1 | 0  | 0  | 0  | 1  |

To take an example, from the simulation in **figure 12** we can see that,

- When A=0 and B=0 the output binary format shows 0001 in reverse order of the Y0 Y1 Y2 Y3.

### 4. 2 Bit Magnitude Comparator

The truth table for the 2 bit comparator is as shown below.

| A | B | A<B | A=B | A>B |
|---|---|-----|-----|-----|
| 0 | 0 | 0   | 1   | 0   |
| 0 | 1 | 1   | 0   | 0   |
| 1 | 0 | 0   | 0   | 1   |
| 1 | 1 | 0   | 1   | 0   |

From the table, it is evident that whenever one bit is 1, then the greater or less than sign comes to be 1, and while both are equal, the comparator shows that they are equal. From the **figure 15** of the simulation waveform of 2 Bit magnitude comparator, we can observe that,

- From the binary literal, we can see clearly that when A is 00 but B is 01 then the less than part ('lt' in the transcript) is up to 1 (as we are comparing A with B).
- Again, when both A and B are 10 then the last comparison, which is the equal ('eq' in the transcript) is up to 1 and others are 0.

## **Discussion:**

1. The experiment tests the behaviour by generating simulation waveforms and transcripts from the testbenches provided by the HDL codes for multiplexer, full adder, decoder and magnitude comparator.
2. The waveform simulation has been perfectly matched at the end of every simulation with the transcripts that were produced at terminal.