

# **Outline**

- ✓ Section 1- Introduction to Verification
- ✓ Section 2- Pattern
- ✓ Section 3- Testbench
- ✓ Section 4- Environment

## **Outline**

- ✓ Section 1- Introduction to Verification
- √ Section 2- Pattern
- ✓ Section 3- Testbench
- ✓ Section 4- Environment

# What is Verification?

#### Verification

- To demonstrate the functional correctness of a design
- To make sure that you are implementing what you want
- To ensure the result of some transformation is as expected

#### ✓ Verification vs. Testing

Testing: verified the design is manufacturing correctly



## What is Verification?

#### ✓ Verification == Bug Hunting

 A process in which a design is verified against a given design specification before tape out

#### Verification includes:

- Functionality (Main goal !!)
- Performance
- Power
- Security
- Safety

#### ✓ How to perform the verification?

- Simulation of RTL design model (Lab03)
- Assertions and Functional Coverage (Lab10)
- Formal verification (Lab11)
- Static and dynamic timing checks (Lab07)
- Power-aware simulations (Lab08)
- Emulation/FPGA prototyping





### Introduction to Verification

### Stages of verification

- Preliminary verification -> Specification (ex. Output = 0 after reset)
- Broad-spectrum verification -> Test pattern (ex. Random test)
- Corner-case verification -> Special test pattern (ex. Boundary)

#### Steps of verification

- Generate stimulus (the inputs that drive the design)
- Apply stimulus to DUT (Design Under Test)
- Capture the response
- Check for correctness
- Measure progress against overall verification goal





# Stages of verification

#### ✓ Which one to use?





# Stages of verification

✓ Time consuming vs. coverage





# Stages of verification

#### ✓ Two kinds of strategies

- Directed Testing -> Check what you know
- Random Testing -> Find what you don't know









# The Verilog Design Environment

- √ TESTBED.v (00\_TESTBED)
  - Connecting testbench and design modules
  - Dump waveform
- ✓ DESIGN.v (01\_RTL)
  - Design under test (DUT)
- ✓ PATTERN.v (00\_TESTBED)
  - Pattern
  - Test program





# **Outline**

- ✓ Section 1- Introduction to Verification
- ✓ Section 2- Pattern
- ✓ Section 3- Testbench
- ✓ Section 4- Environment

#### **Pattern**

# ✓ A simple example

```
define CYCLE TIME 5.0
module PATTERN (
 clk, rst n, in valid, a, b, out valid, sum, overflow
output reg clk, rst n, in valid, a, b;
input out valid, sum, overflow;
real CYCLE = `CYCLE TIME;
parameter PATNUM = 100;
integer in_read, out_read, i_pat, i, latency, out_cnt;
reg [7:0] data a, data b, data sum;
reg data overflow;
initial clk = 0:
always #(CYCLE/2.0) clk = ~clk;
initial begin
 in read = $fopen("../00 TESTBED/in.txt", "r");
 out file = $fopen("../00 TESTBED/out.txt", "w");
  reset signal task;
  for (i pat = 0; i pat < PATNUM; i pat = i pat+1) begin
   input task;
   wait out valid task;
   check ans task;
   $display("PASS PATTERN NO.%4d", i_pat);
 $fclose(in read);
 $fclose(out file);
  YOU PASS task:
```

```
task reset signal task; begin
 rst n = 1'b1;
 in valid = 1'b0;
 a = 1'bx; b = 1'bx;
 force clk = 1'b0;
 \#(0.5); rst n = 1'b0;
 #(2);
 if((out valid !== 0)||(sum !== 0)||(overflow !== 0)) begin
   $display("Output should be 0 after RESET at %4t",$time);
   $finish;
 end
 \#(10); rst n = 1'b1;
 #(3); release clk;
end
endtask
task input task; begin
  repeat($urandom_range(3, 5)) @(negedge clk);
  in valid = 1'b1;
 if ($feof(in read)) begin
   data a = $urandom;
   data b = $urandom;
  end else begin
   $fscanf(in read, "%d %d", data a, data b);
 for (i = 0; i < 8; i = i+1) begin
   a = data a[i];
   b = data b[i];
   @(negedge clk);
 in valid = 1'b0;
 a = bx;
 b = bx:
end endtask
```

```
task wait out valid task; begin
  latency = 0;
 while(out_valid !== 1'b1) begin
   if(latency == 100) begin
     YOU_FAIL_task;
     $display("Latency are over 100 cycles at %8t", $time);
   latency = latency + 1;
   @(negedge clk);
  end
and andtack
task check ans task; begin
  {data_overflow, data_sum} = data_a + data_b;
  out cnt = 0;
 while (out valid === 1'b1) begin
    if (out cnt +1 > 8) begin
     YOU FAIL task;
     $display("The output count are over 8 cycles at %8t",$time);
   else begin
     if ((data sum[out cnt] !== sum) || (data overflow !== overflow)) begin
       YOU FAIL task:
       $display("The output is wrong at %8t", $time);
       $finish:
     end
    $fdisplay(out file, "Data A: %0d, Data B: %0d, Sum: %0d, Overflow: %0d",
   data a, data b, sum, overflow);
   out cnt = out cnt + 1;
   @(negedge clk);
  if (out cnt < 8) begin
   YOU FAIL task;
   $display("The output count are less than 8 cycles at %8t", $time);
   $finish:
  end
end endtask
task YOU PASS task; begin
   $display ("Congratulations!");
   $display ("Your execution cycles = %5d cycles", total_latency);
   $display ("Your clock period = %.1f ns", CYCLE);
   $display ("Total Latency = %.1f ns", total_latency*CYCLE);
   $finish;
end endtask
task YOU FAIL task; begin
  $display ("Fail!");
end endtask
```

endmodule

#### **Pattern**

#### Port declaration

```
module PATTERN (
   clk, rst_n, in_valid, a, b, out_valid, sum, overflow
);
output reg clk, rst_n, in_valid, a, b;
input out_valid, sum, overflow;
```

```
real CYCLE = `CYCLE_TIME;
parameter PATNUM = 100;
integer in_read, out_read, i_pat, i, latency, out_cnt;
reg [7:0] data_a, data_b, data_sum;
reg data_overflow;
```

#### Data type declaration

```
initial begin
  in_read = $fopen("../00_TESTBED/in.txt", "r");
  out_file = $fopen("../00_TESTBED/out.txt", "w");
  reset_signal_task;
  for (i_pat = 0; i_pat < PATNUM; i_pat = i_pat+1) begin
    input_task;
  wait_out_valid_task;
  check_ans_task;
  $display("PASS_PATTERN_NO.%4d", i_pat);
end

$fclose(in_read);
$fclose(out_file);
YOU_PASS_task;
end</pre>
```

```
task reset_signal_task; begin
    rst_n = 1'b1;
    in_valid = 1'b0;
    a = 1'bx; b = 1'bx;
    force clk = 1'b0;
    #(0.5); rst_n = 1'b0;
    #(2);
    if((out_valid !== 0)||(sum !== 0)||(overflow !== 0)) begin
        $display("Output should be 0 after RESET at %4t",$time);
        $finish;
    end
    #(10): rst n = 1'b1:
```

#### **Generate stimulus**

```
task input task; begin
 repeat($urandom_range(3, 5)) @(negedge clk);
 in valid = 1'b1;
 if ($feof(in read)) begin
   data a = $urandom;
   data b = $urandom;
 end else begin
   $fscanf(in read, "%d %d", data a, data b);
  for (i = 0; i < 8; i = i+1) begin
   a = data a[i];
   b = data b[i];
   @(negedge clk);
 in valid = 1'b0;
 a = bx;
 b = bx:
 nd endtask
```

```
task wait_out_valid_task; begin
  latency = 0;
  while(out_valid !== 1'b1) begin
   if(latency == 100) begin
    YOU_FAIL_task;
   $display("Latency are over 100 cycles at %8t",$time);
   $finish;
```

#### **Check result**

```
task check ans task; begin
 {data_overflow, data_sum} = data_a + data_b;
 out cnt = 0;
 while (out valid === 1'b1) begin
   if (out_cnt + 1 > 8) begin
     YOU FAIL task;
     $display("The output count are over 8 cycles at %8t",$time);
     $finish;
   end
   else begin
     if ((data sum[out cnt] !== sum) || (data overflow !== overflow)) begin
       $display("The output is wrong at %8t",$time);
       $finish:
     end
   $fdisplay(out file, "Data A: %0d, Data B: %0d, Sum: %0d, Overflow: %0d"
   data_a, data_b, sum, overflow);
   out cnt = out cnt + 1;
   @(negedge clk);
 if (out cnt < 8) begin
   YOU FAIL task;
   $display("The output count are less than 8 cycles at %8t",$time);
   $finish:
 end
end endtask
```

endmodule



## **Procedural Blocks**

#### ✓ Start the simulation

```
initial begin
  in read = $fopen("../00 TESTBED/in.txt", "r");
  out file = $fopen("../00 TESTBED/out.txt", "w");
  reset signal task;
  for (i pat = 0; i pat < PATNUM; i pat = i pat+1) begin
    input task;
   wait out valid task;
    check ans task;
    $display("PASS PATTERN NO.%4d", i pat);
  end
 $fclose(in read);
 $fclose(out file);
 YOU PASS task;
end
```

# **Procedural Blocks**

- ✓ All procedural blocks will be executed concurrently.
- ✓ Initial Blocks
  - Only be executed once

```
initial
begin
statement...
end
```

- ✓ Always Blocks
  - Will be executed if the condition is met

```
always@(condition_expression)
begin
statement...
end
```

# **Procedural Blocks**

#### ✓ A simple example

```
module Test (OUT, A, B, SEL);
         output A,B,SEL;
                           Port declaration
         input OUT;
         initial
         begin
                     A=0;B=0;SEL=0;
Delay and timing
                      #10
                                 A=0;B=1;SEL=1;
                                 A=1;B=0;
                      #10
                                 SEL=0;
                      #10
                                            End simulation!
                      #10
                                 $finish;
         end
         endmodule
```

- ➤ If simulation never stop, check ...
- 1. No "\$finish" in pattern
- 2. Have combinational loop in design
- 3. Have loop in pattern







# **Tasks**

#### ✓ Reset Task

```
define CYCLE TIME 5.0
module PATTERN (
 clk, rst_n, in_valid, a, b, out_valid, sum, overflow
output reg clk, rst_n, in_valid, a, b;
input out_valid, sum, overflow;
real CYCLE = `CYCLE_TIME;
parameter PATNUM = 100;
integer in_read, out_read, i_pat, i, latency, out_cnt;
reg [7:0] data a, data b, data sum;
reg data overflow;
initial clk = 0:
always #(CYCLE/2.0) clk = ~clk;
initial begin
 in_read = $fopen("../00_TESTBED/in.tkt", "r");
  out file = $fonen("../00 TESTBED/out txt", "w");
  reset signal task;
  ror (1_pat = 0; 1_pat < PATNUM; i_pat = i_pat+1) begin
    input task;
   wait out valid task;
   check_ans_task;
   $display("PASS PATTERN NO.%4d", i_pat);
 $fclose(in_read);
 $fclose(out file);
 YOU PASS task;
```

Case inequality

```
task input task; begin
 repeat($urandom range(3, 5)) @(negedge clk);
 in valid = 1'b1;
 if ($feof(in read)) begin
   data a = $urandom;
   data b = $urandom;
 end else begin
   $fscanf(in_read, "%d %d", data_a, data_b);
 end
 for (i = 0; i < 8; i = i+1) begin
   a = data a[i];
   b = data b[i];
   @(negedge clk);
 end
 in valid = 1'b0;
 a = bx;
 b = bx:
end endtask
```



# **Asynchronous Reset and Clock**

#### Asynchronous reset:

Reset signal will reset all registers on the falling edge of reset signal.

#### Clock signal

- Clock signal should be forced to 0 before reset signal is given.
- Using always procedure to produce a duty cycle 50% clock signal
- Example:

```
reg clk, rst_n;
real CYCLE = 2.5;
initial clk = 0;
always #(CYCLE/2.0) clk = ~clk;
initial begin
    rst_n = 1'b1;
    force clk =0;
    #(0.5);    rst_n= 1'b0;
    #(10);    rst_n= 1'b1;
    #(3);    release clk;
end
```

### **Tasks**

task reset signal task; begin

rst n = 1'b1;

# ✓ Input data

```
define CYCLE TIME 5.0
module PATTERN (
 clk, rst n, in valid, a, b, out valid, sum, overflow
output reg clk, rst_n, in_valid, a, b;
input out_valid, sum, overflow;
real CYCLE = `CYCLE_TIME;
parameter PATNUM = 100;
integer in_read, out_read, i_pat, i, latency, out_cnt;
reg [7:0] data a, data b, data sum;
reg data overflow;
initial clk = 0:
always #(CYCLE/2.0) clk = ~clk;
initial begin
 in_read = $fopen("../00_TESTBED/in.txt", "r");
 out file = $fopen("../00 TESTBED/out.txt", "w");
  reset_signal_task;
  for (i pat = 0: i pat < PATNUM; i_pat = i_pat+1) begin
   input task;
   wait out valid task;
   check ans task;
   $display("PASS PATTERN NO.%4d", i_pat);
 $fclose(in_read);
 $fclose(out file);
 YOU PASS task;
```

```
in_valid = 1'b0;
  a = 1'bx; b = 1'bx;
  force clk = 1'b0;
 \#(0.5); rst n = 1'b0;
  #(2);
 if((out valid !== 0)||(sum !== 0)||(overflow !== 0)) begin
   $display("Output should be 0 after RESET at %4t", $time);
   $finish;
  end
  \#(10); rst n = 1'b1;
  #(3); release clk;
end
endtask
task input task; begin
  repeat($urandom range(3, 5)) @(negedge clk);
 in valid = 1'b1;
 if ($feof(in_read)) begin
    data a = $urandom;
    data b = $urandom;
  end else begin
    $fscanf(in read, "%d %d", data a, data b);
  end
  for (i = 0; i < 8; i = i+1) begin
    a = data a[i];
   b = data b[i];
   @(negedge clk)
  end
  in valid = 1'b0;
  a = bx;
  b = bx;
end endtask
```

 Input data change at negedge clk



#### **Pattern**

## Check output data

```
define CYCLE TIME 5.0
module PATTERN (
 clk, rst n, in valid, a, b, out valid, sum, overflow
output reg clk, rst n, in valid, a, b;
input out valid, sum, overflow;
real CYCLE = `CYCLE TIME;
parameter PATNUM = 100;
integer in_read, out_read, i_pat, i, latency, out_cnt;
reg [7:0] data a, data b, data sum;
reg data overflow;
initial clk = 0:
always #(CYCLE/2.0) clk = ~clk;
initial begin
 in read = $fopen("../00 TESTBED/in.txt", "r");
 out file = $fopen("../00 TESTBED/out.txt", "w");
 reset signal task;
 for (i_pat = 0; i_pat < PATNUM; i_pat = i_pat+1) begin
   input task:
   wait out valid task:
  check ans task:
   $display("PASS PATTERN NO.%4d", i_pat);
 $fclose(in read);
 $fclose(out file);
 YOU PASS task;
```

```
task check ans task; begin
 {data overflow, data sum} = data a + data b;
 out cnt = 0:
 while (out valid === 1'b1) begin
   if (out cnt + 1 > 8) begin
     YOU FAIL task;
     $display("The output count are over 8 cycles at %8t",$time);
     $finish:
   end
   else begin
     if ((data sum[out cnt] !== sum) || (data overflow !== overflow)) begin
       YOU FAIL task;
       $display("The output is wrong at %8t",$time);
       $finish;
     end
   $fdisplay(out_file, "Data A: %0d, Data B: %0d, Sum: %0d, Overflow: %0d",
   data a, data b, sum, overflow);
   out cnt = out cnt + 1;
                                 Check output data at
  @(negedge clk)
                                 negedge clk
 if (out_cnt < 8) begin
   YOU_FAIL_task;
   $display("The output count are less than 8 cycles at %8t",$time);
   $finish;
 end
end endtask
```



# **Input Delay**

#### Consider the input interface

end

- Input signals should be synchronous to either positive clock edge or negative clock edge with specified input delays to avoid timing violation
- Assign input delays by absolute delay value

```
CLK _____
```

Assign input delays by relative delay value (relative to clock period)

@(posedge CLK) **#INDLY** IN1 = 0;

Example:





# **Tasks**

## ✓ Simplified Syntax

```
task identifier;
    parameter_declaration;
    input_decleration;
    output_decleration;
    inout_declaration;
    register_declaration;

    begin
        statement;
    ....
    end
endtask
```

## **Tasks**

#### ✓ An example of using a task

 Task can take, drive and source global variables, when no local variables are used.

```
Global →
                                                      Global →
             temp_in = 30;
                                                                   temp_in = 30;
             convert(temp in, temp out);
                                                                   convert;
              $display("%d ,%d",temp_in,temp_out);
                                                                   $display("%d,%d",temp_in,temp_out);
             task convert;
                                                                   task convert;
             input [7:0] temp_in;
                                                                   begin
             output [7:0] temp out;
                                                      Global
                                                                      temp_in = 20;
             begin
                                                                     temp_out = (9/5)*(temp_in+32);
 Local
                temp_in = 20;
                                                                   end
                temp_out = (9/5)*(temp_in+32);
                                                                   endtask
             end
              endtask
```

```
temp_in: 30 temp_out: 52
```

```
temp_in: 20 temp_out: 52
```



## Task and Function

✓ To break up a task into smaller, more manageable ones, and encapsulate reusable code, you can either divide your code into modules, or you can use tasks and functions.

#### ✓ Task

 A task is typically used to perform debugging operations, or to behaviorally describe hardware.

#### ✓ Function

 A function is typically used to perform a computation, or to represent combinational logics.

# **Functions**

## ✓ Simplified Syntax

```
function type_or_range identifier;
    parameter_declaration;
    input_decleration;
    register_declaration;

    begin
        statement;
    ....
    end
endfunction
```

# **Functions**

#### ✓ An example of using a function

 Although the function cannot contain timing, you can call it from a procedural block that does.

```
always@(posedge CLK)
sum = add(a,b);
...

function [7:0] add;
input [7:0] a;
input [7:0] b;
begin
add = a + b;
end
endfunction
...
```

# **Tasks and Functions**

#### ✓ Task

- Can have timing controls (#delay, @, wait).
- Tasks may execute in non-zero simulation time
- Can have port arguments (input, output, and inout) or none.
- Does not return a value.
- Can enable task or function.
- Not synthesizable
- Can call it from a procedural block

#### ✓ Function

- Can't have timing controls.
- Always execute in 0 simulation time
- Has only input arguments and no output port
- Returns a single value through the function name.
- Can enable function but can't enable task.
- Synthesizable
- Can call it from a procedural block



```
task input task; begin
  repeat($urandom range(3, 5)) @(negedge clk);
  in valid = 1'b1;
  if ($feof(in read)) begin
   data a = $urandom;

    Broad-spectrum verification

    data b = $urandom;
  end else begin
   $fscanf(in read, "%d %d", data_a, data_b);
  end
                                       Corner-case verification
  for (i = 0; i < 8; i = i+1) begin
   a = data a[i];
   b = data b[i];
   @(negedge clk);
  end
  in valid = 1'b0;
  a = bx;
  b = bx:
end endtask
```



### ✓ Using Verilog random system task

- \$random(seed);
  - Return 32-bit signed value
  - Seed is optional
- \$urandom(seed);
  - Return 32-bit unsigned value
  - Seed is optional
- \$urandom\_range(int unsigned MAX, int unsigned MIN=0);
  - Return value inside range

### Using high level language with file IO

- Generate random stimulus from MATLAB or Python etc. and output the stimulus into files.
- Read the files in pattern.v



### ✓ A simple example

```
integer SEED, number;
SEED = 123;
number = $random(SEED) % 7;
```

```
integer SEED;
reg[3:0] number;
SEED = 123;
number = $random(SEED);
number = number % 7;
```

```
integer SEED, number;
SEED = 123;
number = $urandom(SEED) % 7;
```

```
integer SEED;
reg[3:0] number;
SEED = 123;
number = $random(SEED) % 7;
```

```
integer SEED;
reg[3:0] number;
SEED = 123;
number = $random(SEED) % 'd7;
```





#### ✓ A simple example

Produce random number in 0~6

#### May be negative



#### **Correct**



#### Correct



#### The value may not in range 0~6

```
integer SEED;
reg[3:0] number;
SEED = 123;
number = $random(SEED) % 7;
unsigned signed (signed operation)
```

#### ✓ A simple example

Produce random number in 0~6

#### 

The key point is to use unsigned operation!

# File I/O

#### ✓ Open file

- \$fopen opens the specified file and returns a 32-bit descriptor.
  - file\_descriptor = \$fopen("file\_name", "type");
    - ◆ file\_descriptor: bit 32 always be set (=1), remaining bits hold a small number indicating what file is opened.
    - type: "r", open for read; "w", open for write

```
file = $fopen("test.txt","w");
file2 = $fopen("test2.txt","w");
file3 = $fopen("test3.txt","w");
```

- multi\_channel\_descriptor = \$fopen("file\_name");
  - ◆ Multi\_channel\_descriptor: bit 32 always be clear (=0), bit 0 represent standard output, each remaining bit represents a single output channel.

```
file = $fopen("test.txt");
file2 = $fopen("test2.txt");
file3 = $fopen("test3.txt");
```



# File I/O

#### ✓ Close file

- \$fclose system task closes the channels specified in the multichannel descriptor
  - \$fclose(<multichannel\_descriptor>);
  - The \$fopen task will reuse channels that have been closed

# File Input

#### ✓ Read data from specific file

- \$fgetc reading a byte at a time
  - c = \$fgetc(<descriptor>);
- \$fgets reading a line at a time
  - i = \$fgets(string, <descriptor>);
- \$fscanf reading formatted data
  - i = \$fscanf(<descriptor>," text", signal,signal,...);

```
opa=101,opb=1010
opa=3,opb=12
```

```
rc = $fscanf(file_input,"opa=%d,opb=%d",opa,opb);
```

- \$readmemb, \$readmemh
  - \$readmemb("file\_name", memory\_name [ , start\_address [ , end\_address ]] );
  - \$readmemh("file\_name", memory\_name [ , start\_address [ , end\_address ]] );

```
$readmemh ("IN.txt",in);
$readmemh ("OUT.txt",out);
```



# File Input

#### ✓ A simple example for \$readmemb

```
@002
11111111 01010101
00000000 10101010
@006
1111zzzz 00001111
```

```
reg [7:0] meme[0:7];
$readmemb("test.txt",meme);
```

```
@002
11111111_01010101
00000000_10101010
@006
1111zzzz_00001111
```

```
reg [15:0] meme[0:7];
$readmemb("test.txt",meme);
```

# File Input

### ✓ A special example for \$readmemb

```
reg [7:0] meme[0:7];
$readmemb("IN.txt", meme);
```

```
1 @002
2 11111111 01010101
3 00000000 10101010
4 11110000 00001111
5 @004
6 1z1z1z1z 0z0z0z0z
```

```
1 @002
2 11111111 01010101
3 00000000 10101010
4 11110000 00001111
5 zzzz0000 zzzz1111
```



# File Output

### ✓ Display tasks that writes to specific files

- \$fdisplay, \$fwrite, \$fstrobe, \$fmonitor
  - \$fdisplay (<descriptor>,["format\_specifiers",] <argument\_list>);
  - \$fwrite (<descriptor>,["format\_specifiers",] <argument\_list>);
  - \$fstrobe (<descriptor>,["format\_specifiers",] <argument\_list>);
  - \$fmonitor (<descriptor>,["format\_specifiers",] <argument\_list>);

```
$fdisplay(file_output,"%d + %d = %d",opa,opb,sum);
```

|             | Active Events |          |
|-------------|---------------|----------|
| w/ newline  | fdisplay      | fstrobe  |
| w/o newline | fwrite        | fmonitor |

## **Stratified Event Queue**

### Stratified Event Queue of Verilog





### **Stratified Event Queue**

```
1 module nb schedule1;
    3 reg a, b;
    4 integer fp;
    6 initial begin
      fp = $fopen("log.txt","w");
      a = 0:
      b = 0:
      #1:
   11
      a = 0:
   12 b = 1:
   13
      a <= b:
      b <= a:
   15
   16
       $monitor("%0dns :\$monitor: a=%b b=%b" , $stime, a, b);
   17
      $display("%0dns :\$display: a=%b b=%b" , $stime, a, b);
   18
       $strobe ("%0dns :\$strobe : a=%b b=%b\n", $stime, a, b);
        $fwrite(fp, "%0dns :\$fwrite : a=%b b=%b\n", $stime, a, b);
   20 #0 $display("%0dns :#0 : a=%b b=%b" , $stime, a, b);
   21
   22
   23 #1 $monitor("%0dns:\$monitor: a=%b b=%b" , $stime, a, b);
   24 $display("%0dns:\$display: a=%b b=%b" , $stime, a, b);
      $strobe ("%0dns :\$strobe : a=%b b=%b\n", $stime, a, b);
       $fwrite(fp, "%0dns :\$fwrite : a=%b b=%b\n", $stime, a, b);
   27 #0 $display("%0dns :#0 : a=%b b=%b" , $stime, a, b);
   28
   29
      $fclose(fp);
   31 end
   32
   33 initial begin
   34 $fsdbDumpfile("nb_schedule1.fsdb");
35 $fsdbDumpvars(0, nb schedule1);
   36 end
```



#### Result

# File Output

### ✓ Display tasks that writes to specific files

- All these four output system tasks support multiple default bases
  - \$fdisplay /\$fdisplayb /\$fdisplayo
  - \$fwrite /\$fwriteb /\$fwriteo
  - \$fstrobe /\$fstrobeb /\$fstrobeo
  - \$fmonitor /\$fmonitorb /\$fmonitoro

```
reg [3:0] num;
initial begin
    num = 15;
    $fdisplay (f_out, "Num: %d", num);
    $fdisplayh(f_out, "Num: %d", num);
    $fdisplayb(f_out, "Num: %d", num);
    $fdisplayo(f_out, "Num: %d", num);
    $fdisplay (f_out, "Num: ", num);
    $fdisplay (f_out, "Num: ", num);
    $fdisplayh(f_out, "Num: ", num);
    $fdisplayb(f_out, "Num: ", num);
    $fdisplayo(f_out, "Num: ", num);
    $fdisplayo(f_out, "Num: ", num);
end
```

```
Num: 15
Num: 15
Num: 15
Num: 15
=======
Num: 15
Num: f
Num: 1111
Num: 17
```



### File I/O

### ✓ Complete file I/O example

```
define CYCLE_TIME 5.0

module PATTERN (
    clk, rst_n, in_valid, a, b, out_valid, sum, overflow
);

output reg clk, rst_n, in_valid, a, b;
input out_valid, sum, overflow;

real CYCLE = `CYCLE_TIME;
parameter PATNUM = 100;
integer in_read, out_read, i_pat, i, latency, out_cnt;
reg [7:0] data_a, data_b, data_sum;
reg data_overflow;
```

### **Open file**

```
in_read = $fopen("../00_TESTBED/in.txt", "r");
out_file = $fopen("../00_TESTBED/out.txt", "w");
reset_signal_task;
for (i_pat = 0; i_pat < PATNUM; i_pat = i_pat+1) begin
    input_task;
    wait_out_valid_task;
    check_ans_task;
    $display("PASS PATTERN NO.%4d", i_pat);
and
$fclose(in_read);
$fclose(out_file);</pre>
```

#### Close file

```
task reset signal task; begin
  rst n = 1'b1;
  in valid = 1'b0;
  a = 1'bx; b = 1'bx;
  force clk = 1'b0;
  \#(0.5); rst n = 1'b0;
  #(2);
  if((out valid !== 0)||(sum !== 0)||(overflow !== 0)) begin
   $display("Output should be 0 after RESET at %4t",$time);
  end
  \#(10); rst n = 1'b1;
 #(3); release clk;
end
endtask
task input task; begin
  repeat($urandom_range(3, 5)) @(negedge clk);
  in valid = 1'b1;
  if ($feof(in read)) begin
   data a = $urandom;
    data b = $urandom;
  end else begin
   $fscanf(in_read, "%d %d", data_a, data_b)
    Read file
    @(negedge clk);
  in valid = 1'b0;
  a = bx;
 b = bx:
end endtask
```

```
task wait out valid task; begin
 latency = 0;
  while(out_valid !== 1'b1) begin
   if(latency == 100) begin
     YOU_FAIL_task;
     $display("Latency are over 100 cycles at %8t", $time);
    latency = latency + 1;
   @(negedge clk);
and andtack
task check_ans_task; begin
  {data_overflow, data_sum} = data_a + data_b;
  out cnt = 0;
  while (out valid === 1'b1) begin
    if (out_cnt + 1 > 8) begin
     YOU FAIL task;
     $display("The output count are over 8 cycles at %8t",$time);
    else begin
    if ((data_sum[out_cnt] !== sum) || (data_overflow !== overflow)) begin
                                          ",$time);
    Write file
   $fdisplay(out file, "Data A: %0d, Data B: %0d, Sum: %0d, Overflow: %0d"
   data_a, data_b, sum, overflow);
   out cnt = out cnt + 1;
   @(negedge clk);
  if (out cnt < 8) begin
   YOU FAIL task;
   $display("The output count are less than 8 cycles at %8t",$time);
   $finish:
  end
end endtask
task YOU PASS task; begin
   $display ("Congratulations!");
   $display ("Your execution cycles = %5d cycles", total_latency);
   $display ("Your clock period = %.1f ns", CYCLE);
   $display ("Total Latency = %.1f ns", total_latency*CYCLE);
   $finish;
end endtask
task YOU FAIL task; begin
  $display ("Fail!");
end endtask
endmodule
```

## **Display Information**

✓ There are mainly four kinds of instruction to display information.

```
- $display (["format_specifiers",] <argument_list>);
```

- \$write (["format\_specifiers",] <argument\_list>);
- \$strobe (["format\_specifiers",] <argument\_list>);
- \$monitor(["format\_specifiers",] <argument\_list>);

|             | Active Events | <b>Monitor Events</b> |  |
|-------------|---------------|-----------------------|--|
| w/ newline  | display       | strobe                |  |
| w/o newline | write         | monitor               |  |

# Display Information (cont.)

✓ The following escape sequences are used for display special characters

| \n | New line character | \" | " character                               |
|----|--------------------|----|-------------------------------------------|
| \t | Tab character      | \o | A character specified in 1-3 octal digits |
| 11 | \ character        | %% | Percent character                         |

✓ The following table shows the escape sequences used for format specifications

| specifier | Display format      | specifier | Display format                                  |
|-----------|---------------------|-----------|-------------------------------------------------|
| %h or %H  | Hexadecimal         | %m or %M  | Hierarchical name                               |
| %d or %D  | Decimal             | %s or %S  | String                                          |
| %o or %O  | Octal               | %t or %T  | Current time                                    |
| %b or %B  | Binary              | %e or %E  | real number in exponential                      |
| %c or %C  | ASCII character     | %f or %F  | Real number in decimal                          |
| %v or %V  | Net signal strength | %p or %P  | Array <only for="" system="" verilog=""></only> |



# **Display Information**

#### ✓ More detail

- http://verilog.renerta.com/source/vrg00013.htm
- http://www.cnblogs.com/oomusou/archive/2011/06/25/verilog\_strobe.html



## **Outline**

- ✓ Section 1- Introduction to Verification
- √ Section 2- Pattern
- ✓ Section 3- Testbench
- ✓ Section 4- Environment

### **Testbench**

- Encapsulate DESIGN.v and PATTERN.v to be a top verification file
- Key element
  - Timescale
  - Dump Waveform
  - Port Connection



```
timescale 1ns/10ps
 include "PATTERN.v"
 ifdef RTL
  `include "ADD.v"
 endif
 ifdef GATE
  `include "ADD SYN.v"
module TESTBED:
wire clk, rst n, in valid, a, b;
wire out valid, sum, overflow;
initial begin
  `ifdef RTL
   $fsdbDumpfile("ADD.fsdb");
   $fsdbDumpvars(0,"+mda");
   endif
  `ifdef GATE
   $sdf annotate("ADD SYN.sdf", u ADD);
   $fsdbDumpfile("ADD SYN.fsdb");
   $fsdbDumpvars(0,"+mda");
   endif
```

```
ADD u ADD(
    .clk(clk),
    .rst_n(rst_n),
    .in_valid(in_valid),
    .a(a),
    .b(b),
    .out valid(out valid),
    .sum(sum),
    .overflow(overflow)
   );
PATTERN u PATTERN(
    .clk(clk),
    .rst_n(rst_n),
    .in_valid(in_valid),
    .a(a),
    .b(b),
    .out valid(out valid),
    .sum(sum),
    .overflow(overflow)
```

### Timescale

#### √ 'timescale

- A compiler directive
- Specifies the unit of measurement for time and the degree of precision of the time
- Syntax:

`timescale <time\_unit> / <time\_precision>

- time\_unit specifies the unit of measurement for times and delays
- time\_precision specifies the degree of precision
- The time\_precision must be at least as precise as the time\_unit
- Valid integers are 1, 10, and 100
- Valid character strings are s, ms, us, ns, ps, and fs

### **Timescale**

#### ✓ A simple example:

CYCLE/2.0 = 1.25 ns
Precision requirement: 0.01ns < 100ps
Precision loss !! The CYCLE/2.0 will
become 1.3ns

| 'timescale Unit/Precision | Delay  | Time Delay |
|---------------------------|--------|------------|
| 'timescale 10ns/1ns       | #5     | 50ns       |
| 'timescale 10ns/1ns       | #5.738 | 57ns       |
| 'timescale 10ns/10ns      | #5.5   | 60ns       |
| 'timescale 10ns/100ps     | #5.738 | 57.4ns     |

 Note: if you use memory in your design, you should set timescale according to the timescale specified by memory file



## **Timescale**

#### ✓ A simple example:

```
`timescale 1ns/<del>100ps</del> → 10ps (avoid precision loss)
module TEST;

parameter CYCLE = 2.5;

reg CLK;

initial CLK = 1;

always #(CYCLE/2.0) CLK = ~CLK;
endmodule
```

CYCLE/2.0 = 1.25 ns
Precision requirement: 0.01ns < 100ps
Precision loss!! The CYCLE/2.0 will
become 1.3ns

| 'timescale Unit/Precision | Delay  | Time Delay |
|---------------------------|--------|------------|
| 'timescale 10ns/1ns       | #5     | 50ns       |
| 'timescale 10ns/1ns       | #5.738 | 57ns       |
| 'timescale 10ns/10ns      | #5.5   | 60ns       |
| 'timescale 10ns/100ps     | #5.738 | 57.4ns     |

 Note: if you use memory in your design, you should set timescale according to the timescale specified by memory file



- ✓ There are many different waveform file formats.
  - Value Change Dump (.VCD)
    - Included in Verilog HDL IEEE Standard
  - Wave Log File (.wlf)
    - Mentor Graphics Modelsim
  - SHM (.shm)
    - Cadence NC Verilog / Simvision
  - VPD (.vpd)
    - Synopsys VCS
  - Fast Signal DataBase (.fsdb)
    - Spring Soft (Merged with Synopsys) Debussy/Verdi

#### Command often used

- \$fsdbDumpfile(fsdb\_name[,limit\_size])
  - fsdb\_name: assign waveform file name
  - (Optional) limit\_size: assign the limitation of file size
- \$fsdbDumpvars([depth, instance][,"option"])
  - depth: level of waveform to be dumped
  - instance: module to be dumped
  - "option": other additional specification, ex: "+mda"
- \$sdf\_annotate("sdf\_file"[,instance][,config\_file][,log\_file][,mtm\_spec][,scale\_factors][,scale\_type])

#### ✓ Parameters of \$fsdbDumpvars()

- Depth
  - 0: all signals in all scopes
  - 1: all signals in current scope (scope of TESTBED.v)
  - 2: all signals in the current scope and all scopes one level below
  - n: all signals in the current scope and all scopes n-1 levels below

#### Option

- "+IO\_only": only IO port signals will be dumped.
- "+Reg\_only": only reg type signals will be dumped.
- "+all": dump all signals including the memory, MDA, packed array, structure, union and packed structure signals in all scopes specified in \$fsdbDumpvars.
- "+mda": dump all memory and MDA(multiple dimensional array) signals in all scopes specified in \$fsdbDumpvars.
- For further information, please refer http://www.eetop.cn/blog/html/55/1518355-433686.html

#### **✓** A simple example:

- Used in RTL simulation or gate-level simulation
- Dump wave form in fsdb format for viewing in nWave
- Include timing information in the simulation



#### ✓ A simple example:

- Used in RTL simulation or gate-level simulation
- Dump wave form in fsdb format for viewing in nWave
- Include timing information in the simulation

```
initial begin
            `ifdef RTL
                        $fsdbDumpfile("Design.fsdb");
                        $fsdbDumpvars(0,"+mda");
            `endif
            `ifdef GATE
                        $fsdbDumpfile("Design_SYN.fsdb");
                        $fsdbDumpvars(0,"+mda");
                        $sdf_annotate("CORE_SYN.sdf", dut);
            `endif
end
                                  module dut (OUT, A, B, SEL);
                                   endmodule
   sdf
                  sdf: standard delay format
```



## **Port Connection**

- ✓ The input and output is reverse between design.v and pattern.v.
- ✓ A simple example:

```
TESTBED.v
//input signals
wire clk, rst_n, in;
//output signals
                                                                       cĺk
wire out;
                                                                      rst n
                                                                                PATTERN.v
                                                 DESIGN.v
DESIGN U_DESIGN(
            .in(in)
                                                                       in
                                                                                  (module)
                                                  (module)
            .rst_n(rst_n)
                                                                       out
            .clk(clk)
            .out(out)
PATTERN U_PATTERN(
            .in(in)
            .rst_n(rst_n)
            .clk(clk)
            .out(out)
endmodule
```



## **Outline**

- ✓ Section 1- Introduction to Verification
- √ Section 2- Pattern
- ✓ Section 3- Testbench
- ✓ Section 4- Environment

## **Simulation Environment**

- ✓ 00\_TESTBED
  - Pattern and testbench location.
- √ 01\_RTL
  - RTL code functionality simulation
- ✓ 02\_SYN
  - Circuit synthesize
- **✓** 03\_GATE
  - Gate level simulation
- **✓** 04\_MEM
- **✓** 05\_APR
- **✓** 06\_POST



# 00\_TESTBED

### ✓ 00\_TESTBED:

- TESTBED.v
- PATTERN.v
- filelist.f: files to run simulation.



## 01\_RTL

#### **✓** 01\_RTL:

- DESIGN.v
- O1\_run\_vcs\_rtl: run simulation using VCS powered by Synopsys.
- 02\_irun\_rtl: run simulation using irun powered by Cadence.
- 03\_xrun\_rtl: run simulation using xrun powered by Cadence.
- O4\_verdi: run debug tool using Verdi powered by Synopsys.
- O5\_nWave: view waveform using nWave powered by Synopsys.
- 08 check: check the simulation result.
- 09\_clean\_up: clean the temporary files during simulation.
- Link:
  - PATTERN.v & TESTBED.v
  - filelist.f
- We use VCS to simulate when demo in this semester.

## **02\_SYN**

#### √ 02\_SYN:

- syn.tcl
- O1\_run\_dc\_shell: synthesize using Design Compiler shell
- O2\_run\_design\_vision: synthesize using Design Vision (GUI version)
- O3\_read\_dv\_rtl: read design (Verilog code) into Design Vision
- O4\_read\_ddc: read design (Design Data Checkpoint (DDC) files) into Design Vision.
- 08\_check: check the synthesis result.
- 09\_clean\_up: clean the temporary files during synthesis.
- Generate file:
  - DESIGN\_SYN.v & DESIGN\_SYN.sdf

## 03\_GATE

#### **✓** 03\_GATE:

- 01\_run\_vcs\_rtl: run simulation using VCS powered by Synopsys.
- 02\_irun\_rtl: run simulation using irun powered by Cadence.
- O3\_xrun\_rtl: run simulation using xrun powered by Cadence.
- O4\_verdi: run debug tool using Verdi powered by Synopsys.
- O5\_nWave: view waveform using nWave powered by Synopsys.
- 08\_check: check the simulation result.
- O9\_clean\_up: clean the temporary files during simulation.
- Link:
  - DESIGN\_SYN.v & DESIGN\_SYN.sdf
  - PATTERN.v & TESTBED.v
  - filelist.f

## **Supplemental Instructions**

### ✓ Check quota

quota

```
[iclab180@ee01 ~]$ quota
Disk quotas for user iclab180 (uid 16975):
Filesystem blocks quota limit grace files quota limit grace
raid:/RAID2 73428 15728640 20971520 1134 0 0
```

### ✓ Pattern encryption

- ncprotect -autoprotect PATTERN.v
  - For Cadence tool: irun, xrun
- vcs +autoprotect PATTERN.v
  - Encrypt the whole content except module name.
  - For Synopsys tool: vcs
- vcs +auto2protect PATTERN.v
  - Encrypt the whole content except module name and port declaration.
  - For Synopsys tool: vcs
- Make sure there are no PATTERN.vp in the directory before encrypt.

