

## INTRODUCTION

→ What is Verilog?

- \* It is a Hardware Description language.
- \* It is used to design & describe digital systems such as ALU's, Timers, Counters & microprocessors.
- \* Designs which are described in HDL are independent of circuit fabrication technology.
- \* IEEE Standard HDL.
- \* Verilog is used for Both synthesis & simulation.
- \* The first HDL was ISP (In-System Programming), invented by C. Gordon Bell and Alan Newell
- \* System Verilog is an extended version of verilog.

→ History :-

- \* It was designed & first implemented by Pradhyo Gopal & Phil Moorby at Gateway Design Automation in 1984-1985.
- \* 1989 - Cadence Design System purchased gateway Automation.
- \* 1990 - Open Verilog International formed.
- \* 1995 - IEEE standard 1364 adopted.
- \* It is similar in Syntax to the C programming language.
- \* Verilog allows different levels of abstraction to be mixed in the same model.
- \* Verilog is case sensitive language & most of the syntax is adopted from C language.
- \* Extension used by Verilog files is ".v"
- \* Verilog have some advance features like Timing checks, User Define Primitive (UDP) & Programming language Interface (PLI).
- \* Verilog is synthesizable as it is concurrent language whereas JAVA, C are sequential language.

### → Comments :-

\* There are two ways to provide comment

- Single line comment : //

- Block comment : /\* ... \*/

Ex) // example of single line comment

/\* example of block comment statement 1

Statement 2 \*/

### → Whitespace :-

\* Blank spaces (\b), Tabs (\t) & Newlines (\n) constitute whitespace in Verilog.

\* White spaces are ignored by Verilog except when it separates two identifiers.

\* Whitespace are not ignored in strings.

### → Design flow of IC :-

\* Design Specification



\* Implementation



\* Behavioral Description



\* Layout Verification



\* RTL description (IP)



\* PAR floor planning



\* Functional Verification(VF)



\* Gate level netlist

### → Lexical Elements :-

i) Identifiers

ii) Comments

iii) Data Types

iv) Keywords

v) Number Representation

vi) Operators.

### i) Identifiers :-

- \* Identifiers are names given to objects that can be referred in a design.
- \* An identifier can contain alphabets (a-z, A-Z), digits (0-9) and special characters (-, \$).
- \* An identifier must begin with alphabet or underscore.
- + Identifiers are case sensitive.
- \* There are identifiers which allow use of any printable ASCII character. They are called as escaped identifier. (\@app, \#apple).
- \* An escaped identifier starts with a \ and ends with a white space.
- \* An identifier can't be a reserved keyword of verilog library.
- \* Always use lowercase letters as they are relevant to the object.

Valid

Invalid

|                     |                   |
|---------------------|-------------------|
| eg) integer my_var; | integer \$my_var; |
| integer my\$-var;   | integer 5my-var;  |
| integer my-57var;   | integer 78;       |

### ii) Keywords :-

- \* In verilog few identifiers are reserved to define language constructs, they are called as keywords.
- \* All keywords are in lower case.

|           |          |          |              |
|-----------|----------|----------|--------------|
| always    | default  | include  | wait         |
| assign    | disable  | initial  | time         |
| and       | edge     | input    | realtime     |
| automatic | for      | integer  | reg          |
| begin     | forever  | modul    | wire         |
| case      | fork     | output   | while        |
| casex     | function | signed   | primitive    |
| cases     | generate | unsigned | endprimitive |

### iii) Number Specification :-

- \* Verilog simulator by default treat the number in decimal format.
- \* The numbers can be represented in Binary, Octal & Hexadecimal formats too.
- \* There are two types of number specifications:
  - Signed numbers
  - Unsigned numbers

#### \* Signed Number :-

Syntax  $\Rightarrow$  `<size>'<base><number>`

- Size: decimal value specifying number of bits to represent number.
- Base: base represents the format of the number. (`dec0`, `hexH`, `octO`, `binB`).
- Number: specifying number in chosen base format.

Eg) `4'b1011;`

`a = 32'b19;` Syntax error

`4'd113`

`4'hB;`

#### \* Unsigned Number :-

- Numbers are written without size specification
- Default no. of bits depends upon the machine & simulator, must be at least 32. (`a=100`; // `a` is `32'd100`.)
- Decimal is the default base format if not specified.

Eg) `a integer a = 2500` // `a` obtains the decimal value = 2500

`integer b = 'h ABCD` // `a` takes 32 Bit value = `32'h0000-ABCD`.

#### \* Negative Number:-

- To represent a negative number, add minus sign '-' before the size specification.

- Negative numbers are stored in 2's complement in memory.

Eg) -8'd 7; // 8 bit negative number stored as 2's complement of 7.

- If declared variable has larger size than specified value

$$a \rightarrow 8\text{bit} \Rightarrow a = 8'b1010; \quad \boxed{00001010}$$

- If declared variable has smaller size than specified value

$$a \rightarrow 4\text{bit}$$

$$a = 4'b1011 - 100;$$

$$\boxed{1100}$$

(-) used for separating & simplicity which is ignored by the compiler.

- X or Z values :-

- X is used to represent unknown values.
- Z is used to represent high impedance.
- If the MSB bit of the number is X or Z, then X or Z is padded respectively to fill the remaining most significant bits.
- If the MSB bit of the number is 1 or 0, then 0 is padded to fill the remaining most significant bits.

- or ? :-

- is allowed anywhere in a number except at first character.
- ? is an alternative of Z in Verilog.

Eg) 12'h14x // 12 bit hex no with 4 MSB Bits X

11'bz32 // 11 bit bit vector with 5 MSB Bits Z

5'h110 // 5 Bit Binary no with 2 MSB Bits 0

16'b1011\_0100\_1100\_1010 // 16 Bit Binary no

10'h9? // 10 Bit hex no with ~~unspecified bits~~ padded zeros  
10'bzzzzzzz

iv) Data Types :- Mainly 2 types Nets & Registers.

\* Verilog supports following data types :

- |             |            |            |
|-------------|------------|------------|
| • Value set | • Vectors  | • Time     |
| • Nets      | • Integers | • Arrays   |
| • Registers | • Real     | • Strings. |

\* Value Set :- 4 State

- 0 - logic zero
- 1 - logic one
- X - Unknown logic value
- Z - High impedance value
- X & Z are case insensitive.

\* \* Nets :-

- Generally used to create connection between multiple elements of a module or to provide connection between various instances in top module.
- In net type signals, there is no storage capacity & requires continuous driving.
- Nets generally declared as "wire", "word", "wor", "supply 1", "supply 0" etc. (tri, trior, triand, tri0, tri1, trireg). (Unidirectional)
- If there is no value on net then by default it will take Z.

Ex) wire temp1; // 1 bit net

wire [3:0] temp2; // 4 bit net

\* Supply is used for strengthening in switch level modelling.

\* Nets are generally inputs, outputs in case of gate level & data flow structural modelling.

\* Except for the trireg, they don't store values.

## \* Register :-

- Used when storage is required i.e. they can hold the value until value is replaced.
  - Registers don't require a constant driver.
  - Default value is X.
  - Reg type signal assigned only inside always or initial statements.
- ```

ex) reg temp1;           // 11-bit unsigned reg
    reg [3:0] temp2;      // 4-bit unsigned reg
    reg signed [11:0] temp3; // 12-bit signed reg.

reg integer, word, time, realtime etc.
In case of Behavioural model its always output only.
  
```

## \* Vectors :-

- It is used to represent a group of Bits.
  - int or reg data types can be declared as vectors.
- ```

ex) wire temp1;           // 11 scalar type : int
    wire [0:7] temp2;      // 8-bit vector type : int
    reg [3:0] temp3;       // 4-bit vector type : reg
  
```

### \* Selecting part of vector

- `temp2[5];` // Selecting 5<sup>th</sup> bit of temp2
  - `temp2[3:6];` // Selecting 3<sup>rd</sup> to 6<sup>th</sup> bit of temp2
  - `temp3[0:2];` // Invalid because order is different w.r.t declaration
  - `wire [3:0] temp2;`      `wire [0:3] temp3;`
- |   |   |   |   |
|---|---|---|---|
| 3 | 2 | 1 | 0 |
| z | z | z | z |
- |   |   |   |   |
|---|---|---|---|
| 0 | 1 | 2 | 3 |
| z | z | z | z |
- No. of can be accessed in the name order of their declaration

- We can make a vector of datatypes which are by default of one bit.

### \* Integer :-

- A type of register used to store integer values.
- The default size depends upon the host machine but should be atleast 32 bits.
- Default value :  $\times$
- It is used to store signed quantities.
- Syntax  $\Rightarrow$  integer variable-name;

Ex) integer int1, int2; // signed int1, int2

integer [3:0] int3; // invalid syntax, sign not allowed  
(can't be vector).

### \* Real :-

- A type of register used to store real values.
- Values can be stored in decimal or scientific notation.
- Default value : 0 (0.0)
- Syntax  $\Rightarrow$  real variable-name;

Ex) real a, b; // assigned inside initial or always block

a = 3.14; // Assign a floating value 3.14

b = 5e12; // Scientific notation ( $5 \times 10^{12}$ )

### \* Time & Realtime :-

- Time :- A type of register used to store simulation time.
- Size depends upon implementation but should be atleast 64 bits.
- Realtime :- Store the time as floating point quantity.
- \$time is system function which returns current simulation time.
- Default value : 0

Syntax : time variable-name;

Ex) time a, b; // time variable

a = \$time; // to be used inside the initial & always block.

## \* Arrays:-

- Used to store the same type of data together.
- Array declaration of reg or wire can be scalar or vector.
- A multidimensional array can be declared by mentioning the dimension identifier name.
- In verilog arrays are allowed for 'reg', 'wire', 'integer' & 'real' data type.
- Syntax  $\Rightarrow \langle \text{data-type} \rangle \text{ array-name } \langle \text{array-size} \rangle;$

### \* Vector

`reg [3:0] a;`



### \* array

`reg a [3:0];`

32bit



### \* Declaration $\Rightarrow$

$\rightarrow \text{integer num } [0:15];$  // num is scalar type integer array stores 16 values each 32 bit.

$\rightarrow \text{reg } [7:0] \text{ data } [31:0];$  // data is 8 bit vector of type reg size of array is 32

$\bullet \text{integer a } [10];$  compact declaration (0---9)

$\bullet \text{integer a } [9:0];$  verbose declaration  
 $a [0:9];$  preferred

(rows column (RC))

$\rightarrow \text{wire temp } [0:10] [0:5];$  // temp is 2-D not type array, rows = 11  
columns = 6.

$\rightarrow \text{reg } [3:0] \text{ mem } [0:7] [0:7];$  // mem is 2-D reg type array, rows = 8  
(8x8)  
cols = 8, each 4 bit wide.

### • Assigning & Accessing the array :-

- `data[5];` // Accessing 8 bit data stored at index 5
- <sup>10</sup> → `data[10][2];` // 2nd Bit of data at location 10
- <sup>20</sup> → `num[3][2];` // Accessing the 3<sup>rd</sup> row & 2<sup>nd</sup> column data of num
- `num[3] = 45;` // Assign integer value 45 at index = 3 of num
- `num = 0;` // Illegal + all elements can't be assigned without indexing
- `data[20] = 8'hAA;` // Assign hexadecimal value AA at index 20 of data.
- `num[3][3] = 4'b1011;` // Assign 1011 at row=3 & column 3 of num

with arr[0:2][0:3][0:4];

|  |   |   |      |      |   |   |   |
|--|---|---|------|------|---|---|---|
|  | R | C | SubC | 0    | 1 | 2 | 3 |
|  |   |   |      | 1011 |   |   |   |
|  |   |   |      |      |   |   |   |

• arr[0][0][0];

### \* Strings :-

- rug data type is used to store string data
- Each character requires 8 Bits for its storage
- If rug size is more than size of string, <sup>Blank space</sup> are padded on left.
- If rug size is less, the leftover Bits of strings are truncated.
- Syntax  $\Rightarrow$  rug [size of string] string name ;
- Declaration  $\Rightarrow$
- `rug [8+9:1] str1 = "FutureWiz";` // To store FutureWiz 9 Bytes are required  
 $\text{str1} = "FutureWiz"$
- `rug [8+3:1] str2 = "Futurewiz";` // str2 stores only 3 Bytes, others are truncated,  $\text{str2} = "Wiz"$
- `rug [8*12:1] str3 = "Futurewiz";` // str3 can store 12 Bytes, extra Bytes are padded with Blank space/zero  
 $\text{str3} = "...Futurewiz..."$

• 8 bit  $\rightarrow$  ASCII value

| Name      | Size default                                                                            | Default value       | Note                                                                                                         |
|-----------|-----------------------------------------------------------------------------------------|---------------------|--------------------------------------------------------------------------------------------------------------|
| wire      | 1-Bit                                                                                   | Z                   | <ul style="list-style-type: none"> <li>Can be multi Bit or vector also</li> <li>Unsigned, 4-state</li> </ul> |
| Reg       | 1-Bit                                                                                   | X                   | <ul style="list-style-type: none"> <li>Can be multi-bit or vector also</li> <li>Unsigned, 4-state</li> </ul> |
| *         | wire & reg can be signed also $\Rightarrow$ wire signed [3:0] a;<br>reg signed [3:0] a; |                     |                                                                                                              |
| Integer   | 32 bit                                                                                  | XX...X              | <ul style="list-style-type: none"> <li>Can't be a vector</li> <li>Signed, 4 state</li> </ul>                 |
| Real      | 64 bit                                                                                  | 0 (0xd)<br>0.0 (%f) | Signed                                                                                                       |
| time      | 64 bit                                                                                  | xxxx...x            | Unsigned                                                                                                     |
| real time | 64 bit                                                                                  | 00....0             | Signed                                                                                                       |

## ⇒ Fundamental Blocks of Verilog :-

Module Declaration



Port Declaration



Declaration of Variables  
(reg, wire)



Instantiation of lower  
level module

Endmodule



Tasks & function



Procedural Blocks &  
Assignments



Structural Statements  
(gate, assign)

### → Module :-

- \* A "module" is the basic building block in verilog.
- \* A module can be an element or a collection of lower-level design blocks.
- \* A module provides the necessary functionality to the higher-level block through its port interface.
- \* Module declaration comes on top of every verilog program.
- \* Module is declared by the keyword module.
- \* A corresponding keyword endmodule must appear at the end of the module definition.
- \* Each module must have a module-name, which is the identifier for the module, and a port list, which describes the input & output terminals of the module.
- \* Syntax ⇒
  - `module module_name (port definition);`  
 // Body of module  
`endmodule`
  - `module module_name;`  
 // Module may have empty port list  
`endmodule`

## → Ports :-

- \* Ports are the pins or signal which act as input or output of a specified module.
- \* Port list is an important component of verilog module.
- \* Ports provide a means for a module to communicate with the external world through input & output.
- \* There are 3 types of port available in Verilog :
  - Input // module can only receive info using its input ports
  - Output // module can only send values using its output ports
  - Inout // module can either send or receive values using its inout ports.
- \* \* All ports declared as one of the above is assumed to be a wire by default, otherwise it is necessary to declare it again.
- \* Declaring multiple ports with same name is illegal.



- internally must always be type net, externally the inputs can be connected to variable reg or int type



- internally can be type net or reg, externally the output must be connected to a variable net type.



- inout port must be of type net both internally as well as externally.

\* Syntax  $\Rightarrow$  <port-direction> <port-type> port-name;

Ex) input a, b;

output [2:0] y;

output reg y;

inout j;

- input a; // wire by default
  - output b; // wire by default
  - output reg c; // output of type reg
  - output integer e; // output of type integer
  - input wire g; // input of type wire, default is wire
  - input [31:0] h; // vector input, wire by default
  - input integer i; // invalid as input must be a net type
  - reg for input & inout is not allowed.
- ↳ Alternative → Input signed [31:0] i;

### → Parameters :-

- \* Parameter is basically used for defining constants in verilog code
- \* It can be a binary, integer, real or string.
- \* Syntax =>

Parameter variable\_name = value;

#### \* Example :

```
module adder(a,b,sum);
parameter N=4;
input [N-1:0] a;
input [N-1:0] b;
output [N-1:0] sum;
assign sum=a+b;
endmodule
```

- \* Parameter is a constant value which can't be changed. But using "dynamically" we can change the parameter value in the testbench. (overriding).
  - \* module adder #(Parameter WIDTH=8)(); // Parameterized module
- ```
input [WIDTH-1:0] a,b,
output [WIDTH-1:0] sum;
assign sum=a+b;
endmodule
```

## Modelling Styles

=> levels of Abstraction :-

- \* In verilog HDL a model can be defined using various levels of abstraction :

i) Behavioral Level

ii) Register Transfer Level (Data flow, structural)

iii) Gate Level (Gate Level Simulation)

iv) Switch Level

ii) Gate Level Modelling :-

- \* In this level of abstraction the system modelling is done at the gate level, i.e. the properties of the gates.

\* For this style of modelling verilog provides in-built primitives.

- \* There are three classes for gate level modelling :-

- `buf`

- `Buffer & Inverter`

- `Tri-state buf`

\*\* In gate level modelling inputs + outputs are both net/wire type.

\* Syntax  $\Rightarrow$  primitive-name instance-name (optional) [instance range] (output port(s), input port(s), control port);

• instance range when we want to call the same primitive multiple times.

• `buf` :-

- The first part is output, other parts are input.

- This primitive has only one output & any number of inputs

- The following gates are available:

- “and” “or” “nand” “nor” “xor” “xnor”

- Max input values can go upto 16.

- Syntax  $\Rightarrow$  primitive-name instance-name (output port, input1, input2...);

- Ex  $\Rightarrow$  and (Y,A,B) / and a1(Y,A,B);

- or (Y,A,B) / or o1(Y,A,B);

| A | B | AND | OR | NAND | NOR | XOR               | XNOR |
|---|---|-----|----|------|-----|-------------------|------|
| 0 | 0 | 0   | 0  | 1    | 1   | 0                 | 1    |
| 0 | 1 | 0   | 1  | 1    | 0   | 1                 | 0    |
| 0 | x | 0   | x  | 1    | x   | x                 | x    |
| 0 | z | 0   | x  | 1    | x   | x                 | x    |
| 1 | 0 | 0   | 1  | 1    | 0   | -                 | 0    |
| 1 | 1 | 1   | 1  | 0    | 0   | 1                 | 0    |
| 1 | x | x   | 1  | x    | 0   | x                 | x    |
| 1 | z | x   | 1  | x    | 0   | wired OR fix      | x    |
| x | 0 | 0   | x  | 1    | x   | x                 | x    |
| x | 1 | x   | 1  | x    | 0   | overdriven OR fix | x    |
| x | x | x   | x  | x    | x   | x                 | x    |
| x | z | x   | x  | x    | x   | x                 | x    |
| z | 0 | 0   | x  | 1    | x   | x                 | x    |
| z | 1 | x   | 1  | x    | 0   | x                 | x    |
| z | x | x   | x  | x    | x   | x                 | x    |
| z | z | x   | x  | x    | x   | x                 | x    |

### • Buffer and Inverter :-

- The last port is input, other ports are output.
- These primitives can have any number of o/p & only one i/p.
- The following gates are available  
 "buf"    "not"

Ex:    buf (o1, i1p1);  
 buf (o1, o2, i1p1);  
 not (o1, i1p1);  
 not (o2, o2, i1p1);

| A | Buf | Not |
|---|-----|-----|
| 0 | 0   | 1   |
| 1 | 1   | 0   |
| x | x   | x   |
| z | x   | x   |

### • Tri-state logic :-

- There are buf/not gates with control input
- The last port is control input, middle port is input & first port is output.
- The following gates are available

"bufif1"    "notif1"    "bufif0"    ~~bufif1~~    "notif0"

Ex :    bufif0 (o1, i1p, ctrl);  
 notif0 (o1, i1p, ctrl);

| A | ctrl | Y |
|---|------|---|
| 0 | 0    | 0 |
| 1 | 0    | 1 |
| x | 1    | z |

\* Properties of Built Primitives :-

- They have one scalar output & multiple scalar inputs. The output of the gate is evaluated as soon as the input changes.
- Since they are predefined in verilog, they don't require module definition.
- Primitives are instantiated like module.
- If you are using the same primitive more than one time then label is must for each primitive for better readability.
- Array instantiation is also possible with primitives.  
 and a0 (out[0], in[0], in[0]); // A[0]  
 and a1 (out[1], in[1], in[1]); // A[1]  
 and a[1:0] (out, in[0], in[1]);

→ Write a gate level code for Half adder



RTL :-

\* module HA(a,b,s,c);

  input a,b;

  output s,c;

  xor s(xl(a,b));

  and c(a,(c,a,b));

endmodule



Test Bench :-

\* module HA\_test;

  reg a,b;

  wire s,c;

  FA FA(a,b,s,c);

  initial begin

    a=0;b=0;

    #2 a=0;b=1;

    #2 a=1;b=0;

    #2 a=1;b=1;

//VCD file

  initial begin

    \$dumpfile("dump.vcd");

    \$dumpvars(0,a,b,s,c);

  end

endmodule

  initial begin

    \$monitor('a=%b,b=%b,s=%b,c=%b',a,b,s,c);

  end

## → Testbench :-

- \* Generating waveforms could be a complex task as, instead of testing few specific cases, we sometimes need to test a set of cases.
- In all such cases, we write stimulus files for our design known as test bench files.
- \* Used to verify the functionality of the design
- \* It involves four major steps in generating stimulus
  - i) Driving the values on DUT
  - ii) Capturing or monitoring DUT o/p's
  - iii) Matching the expected with obtained results



- \* All testbenches are implemented in Behavioural modelling & the testbench need not be synthesizable
- \* Write module without ports.
  - Inputs are reg & outputs are wire
  - DUT instantiation
  - initial begin
    - ≡ Generation of stimulus of
    - ≡ different construction of DUT i/p's
  - \$delsig if required
  - end
  - initial begin
    - capturing the results & printing
    - \$monitor ( ) ; it.
  - end
  - initial begin
- To print the waveforms  
VCD → Value Change Dump
- \$dumpfile ("dump.vcd");
- \$dumpvars (o, list of variables);
- end
- whenever the entries of o/p's & i/p's are changed these details are kept into "VCD" file
- endmodel;

→ module instantiation :-

- \* Complex systems can be designed by combining sub-systems.
- \* So the modules describing the sub-system will be called by the top module.
- \* Two approaches :- i) Port connection by ordered list. (Positional mapping)  
ii) Port connection by name.
- \* Syntax  $\Rightarrow$  module-name inst-name ( port mapping );  
mandatory

i) Port connection by ordered list :-

- We use ordered list inside the parent module to instantiate the lower modules.
- We ~~know~~ must know the exact order of ports in lower module to make the right connections.

ii) Port connection by name :-

- A better way is to link by calling the names of each port one by one.
- In this approach we use the dot (.) symbol to indicate the port mapping.
- The parent port is written after the dot & then inside the brackets, we write the lower module port name.

Ex:- Sub1 do (.a(in1), .b(in2), .c(out1));

- \* Ports that are not connected with any wire in instantiated module attain the High - impedance value.

→ Write a gate level code for Full adder

|                        |           |                                                                                                               |
|------------------------|-----------|---------------------------------------------------------------------------------------------------------------|
| $a$<br>$b$<br>$c_{in}$ | <b>FA</b> | $sum = A \oplus B \oplus c_{in}$<br>$carry = AB + BC_{in} + AC_{in}$<br>$\Rightarrow AB + (A \oplus B)c_{in}$ |
|------------------------|-----------|---------------------------------------------------------------------------------------------------------------|



RTL :-

Testbench :-

\* module FA(a,b,cin,s,c);

\* module FA\_test;

input a,b,cin;

reg a,b,cin;

output s,c;

and wire s,c;

wire w1,w2,w3;  
 $xor\ x1(w1,a,b);$

| FA dut(a,b,cin,s,c);  
 initial begin

a=0; b=0; cin=0;

$xor\ x2(s,w1,cin);$

#2 a=0; b=0; cin=1;

and a1(w2,w1,cin);

#2 a=0; b=1; cin=0;

and a2(w3,a,b);

#2 a=0; b=1; cin=1;

or o1(c,w2,w3);

#2 a=1; b=0; cin=0;

endmodule.

#2 a=1; b=0; cin=1;

#2 a=1; b=1; cin=0;

#2 a=1; b=1; cin=1;

end

initial begin

\$monitor ("a=%b, b=%b, cin=%b,  
 s=%b, c=%b", a,b,cin,s,c);

end

initial begin

\$dumpfile ("dump.vcd");

\$dumpvars(0,a,b,cin,s,c);

end

endmodule



### iii) Register Transfer Level Modelling :-

- \* It is a design abstraction that describes how data moves between registers and what operations are performed on that data.
- \* In this we have two types :- Data flow modelling
  - Structural modelling.

#### → Data flow modelling :-

- \* Implementation of logic circuit is done with the help of operators.
- \* We use "assign" keyword i.e. continuous assignment.
- \* In data flow modelling inputs & outputs both are wire type.
- \* Data flow modelling is used to implement combinational circuits.

#### → Continuous Assignment :-

- \* Used to assign values to a net. (lhs)
  - \* This assignment are always active, i.e. any change on RHS side leads to assignment of the target. (hence no triggering required)
  - \* An expression can be made up of a net, register or a function call (rhs)
  - \* Target has to be of type net.
  - \* Syntax  $\Rightarrow$  assign target = expression or const value;
- Ex  $\Rightarrow$  input a,b;  
          output c;  
          assign c = a & b;



\* module my\_design (A,B,C,P,E);  
    input A,B,C,P;  
    output E;  
    assign E = ((A & B) ^ (C & P));  
endmodule.

- \* How assignment operator works?
  - \* Every assignment operator works in two phases
    - Evaluation of RHS
    - Assign the evaluated value in LHS

\* wire x,y,a;

assign a = x + y;      Multiple driving  $\Rightarrow$  Allowable when target is  
assign a = x | y;      condition      wire a not allowed if  
                              reg ...

$$\underline{x = 1} \quad y = 0$$

$$x = 0 \quad y = 0$$

0 | P => x

0 / P = 2 C

→ Write a data flow code for 2:4 decoder



## Testbench

\* module Dec2to4(A,B);

input [1:0] A;

Output [3:0] B;

assign  $B[0] = (\sim A[1]) \wedge (\sim A[0])$ ;

assign  $B[i] = (\sim A[i]) \wedge (A[0])$ ;

assign  $B[2] = (A[1])^4 \sim A[0]$ ;

assign  $B[3] = (A[1]) + (A[0])$ ;

endmodule

\* modul die-test;

sug [ɪ:ə] A;

WIR [3:0] B

Dec 2 to 4  $d(A, B)$

Initial Sigin

Bitte erläutern.

$$A[1] = 0; A[0] = 1;$$

#2  $A[1] = 0$ ;  $A[0] = 1$

$$\#2 \quad A\Sigma_3 = 1; \quad A\Sigma_0 = 0;$$

#2  $A[\{j\}] = 1$ ;  $A[\{j\}] = 1$

en

end modul

## OPERATORS

- i) Arithmetic Operators
- ii) Logical Operators
- iii) Shift Operators
- iv) Concatenation Operators
- v) Replication Operators
- vi) Reduction Operator
- vii) Conditional Operator
- viii) Relational Operator
- ix) Equality Operator
- x) Bitwise Operator

### ii) Arithmetic Operators :-

| Operator | Expression | Description    |
|----------|------------|----------------|
| +        | a + b      | Addition       |
| -        | a - b      | Subtraction    |
| *        | a * b      | Multiplication |
| /        | a / b      | Division       |
| **       | a ** b     | Exponent       |
| %        | a % b      | Modulus        |

### \* Key Points :-

- If any operand has  $\infty$  or  $\text{NaN}$  value then result of entire expression would be  $\infty$ .
- If the second operand of division or modulus is zero, the result will be  $\infty$ .
- If the second operand of exponent operator is zero, then result  $\rightarrow 1$ .
- If either operand of exponent operator is real, then result will also be real.

### \* Modulus (Takes sign of numerator).

$$5 \% 3 \Rightarrow 2$$

$$-5 \% 3 \Rightarrow -2$$

$$5 \% -3 \Rightarrow 2$$

$$-5 \% -3 \Rightarrow -2$$

→ Arithmetical Operations:

\* module arithop;

begin [15:0] A, B;

initial begin

A = 8'd25;

B = 8'd15;

\$display ("Addition of %d + %d = %d", A, B, A+B);

\$display ("Subtraction of %d - %d = %d", A, B, A-B);

\$display ("Multiplication of %d \* %d = %d", A, B, A\*B);

\$display ("Division of %d / %d = %d", A, B, A/B);

\$display ("Modulus of %d % %d = %d", A, B, A% B);

\$display ("Square of %d & %d = %d", A, A \*\* 2);

end

endmodule.

O/p =>

Addition of 25 + 15 = 40

Subtraction of 25 - 15 = 10

Multiplication of 25 \* 15 = 375

Division of 25 / 15 = 1

Modulus of 25 % 15 = 10

Square of 25 = 625

Ans

- \* \$monitor → will print when ever there is a change } printing task
- \* \$display → will print only once.

### iii) Logical Operators :-

- \* They are binary operators

| Operator | Expression | Description |
|----------|------------|-------------|
| !        | ! A        | logical NOT |
| &&       | A && B     | logical AND |
|          | A    B     | logical OR  |

#### \* Key Points :-

- It returns single bit 1, 0 or x.
- If operand is non zero, it is equivalent to logic 1.
- If operand is zero, it is equivalent to logic 0.
- If operand is neither zero nor non zero, it is equivalent to x.
- It is treated as false condition by simulators. ( $\text{if}(x) \rightarrow \text{false}$ )

| A | B | && |   |
|---|---|----|---|
| 0 | 0 | 0  | 0 |
| 0 | 1 | 0  | 1 |
| 1 | 0 | 0  | 1 |
| 1 | 1 | 1  | 1 |
| 0 | x | 0  | x |
| x | 0 | 0  | x |
| 1 | x | x  | 1 |
| x | 1 | x  | 1 |
| x | x | x  | x |

$$! \Rightarrow 0 \rightarrow 1$$

$$1 \rightarrow 0$$

$$x \rightarrow x$$

$$\star A = 4'd10; B = 4'd15;$$

$$A \&\& B \Rightarrow 1$$

$$A = 4'd0; B = 4'd15;$$

$$A \&\& B \Rightarrow 0$$

#### \* Operand      logic 0/1/x ?

$$\bullet A = 4'dx; B = 4'd15;$$

$$4'b1100$$

$$\text{logic } 1$$

$$A \&\& B \Rightarrow x$$

$$4'b1x100$$

$$\text{logic } 1$$

$$\bullet A = 4'd10; B = 4'd15;$$

$$4'b0x20$$

$$\text{logic } x$$

$$A || B = 1$$

$$4'b0000$$

$$\text{logic } 0$$

$$\bullet A = 4'd0; B = 4'd15;$$

$$A || B = 1$$

$$\bullet A = 4'dx; B = 4'd0;$$

$$A || B = 1$$

$$\bullet A = 0 \quad !B = 1$$

### iii) Shift Operators:-

\* They are binary Operators (means 2 operands are needed).

| Operator | Expression | Description            |
|----------|------------|------------------------|
| >>       | A >> B     | Logical Right Shift    |
| <<       | A << B     | Logical Left Shift     |
| >>>      | A >>> B    | Arithmetic Right Shift |
| <<<      | A <<< B    | Arithmetic Left Shift  |

#### \* Key Points

- Logical shift, operand 1 must be unsigned.
- Arithmetic shift, operand 2 must be signed.
- In logical shift, the vacant position is filled by zero.
- Arithmetic right shift pads MSB bits.
- Arithmetic left shift works same as logical left shift.
- Shift by 1'b x or 1'b z result is x. (operand 2)

\* •  $A = 8'b0000_0001$

$A << 1 \Rightarrow 00000010$

$A << 3 \Rightarrow 00001000$

$A = 8'b1000_0000$  \* (it should be signed)

$A >>> 1 \Rightarrow 11000000$

$A >>> 2 \Rightarrow 11100000$

$A <<< 1 \Rightarrow 00000000$

\* modifiable b;

: 32 Bit + dual

reg [3:0] a;

integer b;

initial begin

a = 4'b0011;

b = 3>>> a;

b = 3<<< a;

begin

Right  
// Right shift

-----+ 0101010

Left  
// Left shift

-----+ 11000

#### iv) Concatenation Operators :-

- \* Concatenation operators can accept any number of operands.
- \* This operation provides mechanism to append multiple operands.
- \* Syntax  $\Rightarrow \{ \text{op}_1, \text{op}_2, \dots, \text{op}_n \}$ ;

#### \* Example :

• module concat;

    begin [3:0] A,B,C;

        initial begin

            A=4'b0001; B=4'b1001; C=4'b1111;

            \$display ("%b", {A,B}); // Result  $\Rightarrow$  00011001

            \$display ("%b", {A,B,C}); // Result  $\Rightarrow$  00011001111

            \$display ("%b", {A,4'b0011, B[2],C}); // Result  $\Rightarrow$  0001001101111

        end

    endmodule

<sup>32 bit default</sup>

•  $\{ 4'b0100, 1, B, A[3], B[3:2] \}$

0100 - 1 - 1001 - 0 - 10

#### v) Replication Operator :-

- \* It can accept any number of operands.
- \* It is used to replicate an operand specified no. of times.
- \* Syntax  $\Rightarrow \{ \text{replicate-time} \{ \text{operand} \} \}$ ;
- \* Examples:

$\{ 2 \{ A \} \} \Rightarrow 00010001$

$\{ A[0], \{ 2 \{ B \} \} \} \Rightarrow 1_0001_1001$

$\{ A, 4'b0011, \{ 2 \{ C \} \} \} \Rightarrow 0001_0011_1111_1111$

### vii) Reduction Operator :-

- \* They are unary operators
- \* It performs bit wise operation on a single vector
- \* Reduction operators return 1 bit result.

| Operator                     | Expansion       | Description             | O/P when A = 0101 |
|------------------------------|-----------------|-------------------------|-------------------|
| &                            | $\& A$          | Reduc <sup>n</sup> AND  | 0                 |
| $\sim \&$                    | $\sim \& A$     | Reduc <sup>n</sup> NAND | 1                 |
| 1                            | $1A$            | Reduc <sup>n</sup> OR   | 1                 |
| $\sim 1$                     | $\sim 1A$       | Reduc <sup>n</sup> NOR  | 0                 |
| $\wedge$                     | $\wedge A$      | Reduc <sup>n</sup> XOR  | 1                 |
| $\sim \wedge$ or $\sim \sim$ | $\sim \wedge A$ | Reduc <sup>n</sup> XNOR | 0                 |

### viii) Conditional operators :-

- \* They are ternary operators
- \* If condition evaluates to true, then true-exp is returned  
else false-exp is returned.
- \* Nesting of conditional operator is also allowed.
- \* Syntax  $\Rightarrow$  Condition ? true-exp : false-exp ;
- \* Example

Assign y2 = (sd == 2) ? a : b; // if sd = 2, then return a, else b

Assign y1 = sel ? I1 : I0; // 2:1 mux using ternary operator  
- If sel = x/z

y1 = x

Assign y3 = sel ? (sd0 ? I3 : I2) : (sd0 ? I1 : I0); // 4:1 mux

### viii) Relational Operator :-

- \* They are binary operators.
- \* It returns logic 1, if expression is true & return logic 0 if expression is false.
- \* It return X, if any operand contains X or Z.

| Operator | Expression             | Description        | o/p |
|----------|------------------------|--------------------|-----|
| >        | A(1001)>B(1100)        | Greater than       | 0   |
| <        | A(1001)<B(1100)        | Less than          | 1   |
| $\geq$   | A(1100) $\geq$ B(1100) | Greater than equal | 1   |
| $\leq$   | A(100x) $\leq$ B(1100) | Less than equal    | X   |

### ix) Equality Operator :-

- \* They are binary operator
- \* logical equality ( $=$ ,  $!=$ ) returns 0, 1 or X
- \* can equality ( $==$ ,  $!==$ ) returns 0 or 1
- \* logical equality returns X if either operands has X or Z data.
- \* can equality compares all bits including X & Z.
- \* Verilog provides following equality operators.

| Operator | Expression | Description    | o/p A=1001, B=1100 | o/p A=100X, B=100X |
|----------|------------|----------------|--------------------|--------------------|
| $=$      | $A == B$   | Equality       | 0                  | X                  |
| $!=$     | $A != B$   | Inequality     | 1                  | X                  |
| $== =$   | $A == = B$ | can equality   | 0                  | 1                  |
| $!= =$   | $A != = B$ | can Inequality | 1                  | 0                  |

### x) Bitwise Operator:

- \* They are binary operators
- \* It performs bit by bit operation on two operands
- \* If one operand is shorter than other, zeros are padded to match the length of larger operand.
- \* Return sign is equal to that of larger operand.

| Operator                 | Expression     | Description    | $A = 1001, B = 1100$ | $A = 100x, B = 100x$ |
|--------------------------|----------------|----------------|----------------------|----------------------|
| $\sim$                   | $\sim A$       | Bitwise Negate | 0110                 | 011x                 |
| $\&$                     | $A \& B$       | Bitwise AND    | 1000                 | 100x                 |
| 1                        | $A   B$        | Bitwise OR     | 1101                 | 100x                 |
| $\wedge$                 | $A \wedge B$   | Bitwise XOR    | 0101                 | 000x                 |
| $\sim\sim$ or $\sim\sim$ | $A \sim\sim B$ | Bitwise XNOR   | 1010                 | 111x                 |

- \* In XOR if either of the i/p is 1/2 then o/p is x.

\*  $A = 1011$

•  $A \& 1'b0 \Rightarrow 0$

•  $A \& 1'b1 \Rightarrow 0000$

•  $\& A \Rightarrow 0$

$\Rightarrow$  Hierarchy in Verilog / Structural Modelling :-

- \* Verilog provides a concept of code reusability
- \* This reusability concept is implemented by taking instances of one code & use that instance in another code.
- \* With the help of this reusability concept a hierarchical structure can be created either top-to-bottom or bottom-to-top.
- \* This reusability concept reduces the length of design also provides the modularity in code.
- \* This kind of modelling is known as structural style of modelling.

-o Write structural code for Full adder using Half adder



```

* module HA(a,b,s,c);
  input a,b;
  output s,c;
  assign s = a'b;
  assign c = a'b;
endmodule
  
```

```

* module FA(a,b,cin,s,c);
  input a,b,cin;
  output s,c;
  wire w1,w2,w3;
  HA h1(.a(a),.b(b),.s(w1),.c(w2));
  HA h2(.a(w1),.b(cin),.s(s),.c(w3));
  assign c = w2|w3;
endmodule
  
```

## ⇒ Behavioral Modelling :-

- \* It defines the behaviour of digital circuit. It doesn't describe how the circuit will be implemented into actual hardware.
- \* This type of modelling is like writing a algorithm.
- \* Describes the behaviour at higher level.
- \* It uses higher level constructs such as: always, for, while, function, task etc.
- \* Mostly used in RTL verification or writing a testbench.
- \* As the code written inside the procedural block are executed sequentially, hence avoided in designs where speed is one of the factor.
- \* Can be used to describe combinational as well as sequential logic circuits.

→ Write a Behavioral Code to implement full adder module FA(A,B,Cin,Sum,Cout);

```
input A,B,Cin;  
output Sum,Cout;  
output reg Sum,Cout;  
always @(*) begin  
    {Cout,Sum} = A + B + Cin;  
end  
endmodule
```



## ⇒ Procedural Blocks :-

- \* It represents that section of verilog where statements are executed in a sequential manner.
- \* There can be multiple procedure blocks, each block starts at 0 simulation time.
- \* In these blocks output must be of type reg.
- \* Nesting of procedural blocks is not allowed.
- \* Code inside the procedural block is executed continuously unless it's stopped by certain timing control (#, @ or wait).
- \* Assignment statement inside procedural blocks are : Blocking or Non Blocking.
- \* Verilog offers two types of procedural statements :- initial block
  - always block.

## → Initial Block :-

- \* Initial block is used for one time execution during the whole simulation.
- \* There may be any number of initial block inside a module.
- \* If there are more than one initial block, every block starts to execute at 0 simulation time.
- \* If there are multiple statements within an initial block, they must be grouped using begin & end keyword.
- \* Initial block are typically used for initialization, monitoring, providing test inputs & other operations which must be executed only once during the entire simulation.
- \* Syntax ⇒ initial begin

// procedural assign

// delay statement;

end

- \* begin & end required only if there are more than one statement inside the initial block.
  - \* Timing control is used to provide delay in execution of assignment by specified time.
  - \* '#' followed by time value to provide delay.
  - \*\* initial block is non-synthesizable.
- \* module gate (input a,b, output reg y1,y2);  
initial begin  
    y1 = a & b; } → both assignments will execute  
    y2 = a | b; at '0' simulation time.  
end  
endmodule

\* module tb;  
reg b,c,out;  
reg a=1; // reg can be initialized during declaration.  
initial begin  
    a=0; b=0; // a & b will get value '0' at 0ns  
    #5 b=1; // b will get value '1' at 5ns  
    #13 a=1; // a will get value '1' at 18ns  
end  
endmodule

## ⇒ Always Block :-

- \* Always Block executes continuously during simulation just like a looping fashion.
- \* This statement is used to model block of hardware that requires continuous execution.
- \* If there are multiple always blocks, then all blocks start at 0 time & executes simultaneously.
- \* If there are multiple statements within an always block, they should be grouped using begin and end keyword.
- \* Syntax ⇒ always @ (sensitivity list)

```

begin
    // timing control
    // procedural assignment
end

```

- always Block can be controlled with help of sensitivity list
- If sensitivity list is not provided, then always Block runs continuously.
- Always Block is created whenever event occurs on any of the variable present in sensitivity list.
- @ is event control directive :- which holds the execution of statement until the event occur on any of the variable listed in sensitivity list.
- Begin & end required only if there are more than one statement.
- \* Statement inside the always Block are executed in sequential manner.
- \* Timing control is used to provide delay in the execution of a assignment by specified time.
- \* Always Block is synthesizable provided the coding guidelines for synthesis are applied.
- \* Always Block can be used for both combinational & sequential logic design.

- Always @ (a,b)
  - begin
  - #2 a=0; b=0; execute
  - end
  - #2 a=0; b=1; execute
  
- always @ (a)
  - begin
  - #2 a=0; b=0; execute
  - end
  - #2 a=0; b=1; Not execute (no change)

→ Always Block for Combinational Circuit :-

- \* As in combinational circuits, the output depends only on the input signals, hence all the input signals must be provided in the sensitivity list.
- \* Hence, the statements written inside the always block will be executed when event occurs on any signal present in sensitivity list.
- \* reg variable is used inside the always block (target).
- \* Primarily blocking assignments ('=' ) are used.

Ex) module xor (out, a, b);

    input a, b;

    output reg out;

    always @ (a, b) // always @ (a or b) // always @ (\*) (not used in program)  
 out = a ^ b;

↳ includes all the signals

    endmodule

→ Always Block for Sequential Circuits :-

- \* To implement synchronous logic in RTL coding, non-blocking assignments ('<=') are primarily used.

Ex) module dff (input clk, d, output reg q);

    always @ (posedge / negedge clk) begin

        q <= d;

    end

endmodule

→ Key Points of always & initial Block :-

- \* Both always & initial block are procedural blocks
- \* All the statements inside these blocks are executed sequentially
- \* If more than one statements, then they must be grouped using begin and end.
- \* Multiple number of always & initial blocks are allowed in one module.
- \* If module has both initial & always block, then execution of initial block will start before always block.
- \* Initial block executes only once at t=0 while always block execution starts at t=0 and repeats its execution thereafter.
- \* Initial block is not synthesizable, hence used for testbench or simulation not for designs.
- \* Always block is synthesizable provided the RTL guidelines for synthesizable code is incorporated.
- \* always @ (posedge clk or rst) logically incorrect.

→ Write a code which makes initial block work or function like a always block & vice versa.

```
• module initial_as_always;
  reg clk = 0;
  initial begin
    forever #5 clk = ~clk;
  end
endmodule
```

```
• module always_as_init;
  reg done = 0;
  always begin
    if (!done) begin
      $display ("RT=%0dt", $time);
      done = 1;
    end
  end
endmodule
```

\* This runs only once by using disable always-as-init;  
acting like initial.

## $\Rightarrow$ Clock Generator :-

- \* Write Verilog module to generate clock.

module clock\_gen;

reg clk;

initial

clk = 1'bo;

always

#5 clk = ~clk;

endmodule

(continuous running)

module clock\_gen;

reg clk;

initial begin

clk = 0;

#100 \$finish;

end

always

#5 clk = ~clk;

endmodule

- \* Write Verilog module to generate clock of specified frequency

module clock\_gen;

reg clk;

real freq = 100; // freq in KHz

real tp = 1 / (freq \* 1e3) \* 1e9; // convert to ns

initial begin

clk = 1'bo;

#(tp \* 4) \$finish;

end

always begin

#(tp / 2) clk = ~clk;

end

endmodule

\* Write Verilog module to generate clock of specified freq & Duty cycle  
module clock\_gen;

begin

real freq = 100; // freq in KHz

real tp = 1/(freq \* 1e3) \* 1e9; // convert to ns

real duty\_cycle = 50; // in percentage

real on\_time = duty\_cycle \* tp / 100;

initial begin

clk = 1'b0; // initialize the clock with 0

#(tp \* 4) \$finish

end

always begin

#(on\_time) clk = 1'b1;

#(on\_time) clk = 1'b0;

end

endmodule.



## Order of Execution

\* We have 5 regions

i) Preponed region :- Initialization

ii) Active region

iii) Inactive region

iv) NBA region :- RHS update of NBA.

v) Postponed region :- \$monitor & \$strm

ii) Active Region:-

• Blocking assignments

• \$display & \$write

• Continuous assignments

• RHS of Non Blocking assignments

iii) Inactive Region:-

• RHS update for NBA

• #0 Blocking Assignments

$a = b$  :- executed first

#0  $a = b$  :- This later

## System Tasks

- \* In verilog HDL some system task are available by which we can perform specific task & make any simulation more understandable.
- \* By using system task one can perform following task:
  - Displaying & monitoring of input & output
  - Control of simulation
  - Timing checks (setup & hold violation)
- \* Name of system task begins with '\$' followed by name of the task
- \* All system task are not synthesizable
- \* System tasks can be used in code as well as in test bench.
- \* They are always used in procedural blocks

## → Display System tasks :-

- \* The display system tasks are primarily used to display the information or debug messages to track the simulation flow.
- \* Vivado provides following display system tasks:
  - \$display
  - \$write
  - \$monitor
  - \$read

### \* \$display :-

- \* Display strings, variables, & expressions immediately & append newline at the end of the message.
- \* Syntax is like printf in C.
- \* \$display inserts a newline by default.
- & 

```
$display ("HelloWorld");
```

  

```
$display ("Value of count is %d", counter);
```

  

```
$display (a,b,c);
```

  

```
$display ($time); // print current time
```
- \* \$display uses various format specifiers to display the values
  - %d or %D - Print value in decimal
  - %b or %B - Print value in Binary
  - %h or %H - Print value in Hexadecimal
  - %o or %O - Print value in Octal
  - %c or %C - Prints ASCII character
  - %s or %S - Prints String
  - %f or %F - Prints Real number in decimal format
  - %t or %T - Prints current simulation time

### \* \$Write :-

- Display Strings, variables, & expressions without appending the newline at the end of the message.
- Its functionality is similar to \$display except it doesn't add a new line automatically.

Ex) module display\_task;

reg [3:0] a;

initial begin

a=4'b1010;

\$display ("display task");

\$display ("Count value = %b", a);

end

endmodule

O/P =>

O/P => display task

Count value = 1010

Write task Count value = 1010

### \* \$monitor :-

- \$monitor continuously monitors list of variables & executes whenever any of the arguments changes its value.
- \$monitor automatically adds a new line.
- Only one \$monitor can be active at a time.
- If there are multiple \$monitor in the code, the new \$monitor will override any previous \$monitor in effect.
- "\$monitor on" & "\$monitor off" are used to enable & disable monitoring respectively.
- Monitoring is by default ON
- Syntax => \$monitor ("values are a=%d & b=%d", a, b);

Ex module monitor test;

```
reg [3:0] a,b;
```

```
initial begin
```

```
    a=4'hA;
```

```
    b=4'h9;
```

```
$monitor(`$Time=%t,a=%o,b=%o`,$time,a,b);
```

```
#5 a=4'h1;b=4'h1;
```

```
#2 a=4'h2;b=4'h2;
```

```
end
```

```
endmodule
```

Output ->

ST = 0, b = 9

ST = 5, b = 1

ST = 7, b = 2

Sim Time = 7, a = 2, b = 2

### \* \$ strobe :-

- It is similar to \$display
- It automatically adds a new line
- Displays data at the end of current simulation time
- Syntax => \$strobe(`\$strobe task=a=%o b=%o`,\$time,a,b);

Ex) module strobe\_task

O/P ->

```
reg [3:0] a,b;
```

```
initial begin
```

```
    a=4'hA;b=4'h9;
```

```
#5 a=4'h1;
```

```
$display(`$Display:$Time=%t,a=%o,b=%o`,$time,a,b);
```

```
$strobe(`$strobe:$Time=%t,a=%o,b=%o`,$time,a,b);
```

```
b=4'h1;
```

```
end
```

```
endmodule
```

Display : Sim Time = 5, a = 1, b = 9

Strobe : Sim Time = 5, a = 1, b = 1



```
• module tb;
```

```
  begin [2:0] a=3'b000, b=3'b000;
```

```
    initial begin
```

```
      a=3'b101;
```

```
      $display ("display1: Sim time=%t, a=%b, b=%b" $time, a, b);
```

```
      $strchr ("strchr1: Sim time=%t, a=%b, b=%b" $time, a, b);  
      b=3'b000; #10 $strchr ("strchr2: ..."); b=3'b100;
```

```
      $display ("display2: Sim time=%t, a=%b, b=%b" $time, a, b);
```

```
      a=3'b111;
```

```
    end
```

⇒

```
endmodule
```

display1 : Sim time=0 , a=101 b=000

Strchr1 : Sim time=0 , a=101 b=011

display2 :

display2 : Sim time=10 , a=101 b=100

Strchr2 : Sim time=10 , a=111 b=100

## → IPMI System task :-

- \* Verilog offers following functions to access current simulation time
  - \$time - \$realtime.
  - \$stime
- \$time - Returns 64-bit integer representing the current simulation time in terms of timescale unit.
- \$stime - Returns 32-bit Unsigned integer representing the current simulation time in terms of timescale unit.
- \$realtime - Returns real number representing the current simulation time in terms of timescale unit.

## → Simulation control tasks:-

\* Verilog provides following simulation control system tasks.

- \$stop - is used to suspend simulation

- \$finish - is used to exit simulation

- \$ownt - is used to run simulation to time 0.

### .. Syntax

\$stop or \$stop( $N$ );

\$finish or \$finish( $N$ );

\$ownt

### Description

$N=0$  Prints nothing

$N=1$  Prints simulation time & location

$N=2$  Prints simulation time, location, memory & CPU static and during simulation.

- Default  $N=1$

## Blocking & Non Blocking Assignments

→ Continuous Assignment :-

- Continuous assignments are used to assign values to a net.
- Any change on RHS side leads to assignment of the target.
- An expression can be made up of a net, register or a function call.
- Target has to be of net type.
- Syntax  $\Rightarrow$  assign target = expression;

Ex) module my\_design;

    input a,b;                          // a & b are of type net (default).

    output c;                          // c : net type

    assign c = a + b;                 // Continuous assignment

endmodule

→ Procedural Assignments :-

- Procedural assignments are used for assigning the values to reg, integer, real, or time variable in a sequential manner.
- The assignments are always used inside the procedures such as: initial, always, tasks & functions.
- Output must be of reg type.
- The output will hold the value until next assignment to same variable.
- In Verilog there are two types of procedural assignments available
  - i) Blocking ( $=$ )
  - ii) Non Blocking ( $<=$ )

i) Blocking :- ( $=$ )

- Blocking statements are basically preferred for combinational design.
- During simulation it blocks the execution of statements until the assignment occurs.
- Blocking assignments are executed one after other in procedural block.
- In blocking statements, assignments occur at the very same time.

- Race condition may occur.

- Syntax  $\Rightarrow$  register-type-variable = expression;

Ex} module blk1-test;

\$reg [1:0] a=2'b01, b=2'b10, c=2'b11;

O/P  $\Rightarrow$

initial begin

a=00

a=2'b00; b=a; c=b;

b=00

\$display ("a=%b1h, b=%b1h, c=%b1h", a,b,c);

c=00

end

endmodule

### iii) Non-Blocking assignments :- ( $<=$ )

- Non-blocking statements don't blocks the execution of other statements
- In case of non-blocking, the assignment is scheduled to occur at the end of current simulation time or at the end of procedural block
- Verilog recommends to use non-blocking statements to model a sequential circuit.
- Syntax  $\Rightarrow$  register-type-variable <= expression;

Ex}

a<=2'b00; b<=a; c<=b;

O/P

a=01

..

b=10

..

c=11

Q} module ex1;

integer a;

initial begin

#5 a=6;

// #5 a<=6;

\$strobe(\$time, "strobe", a);

\$display (\$time, "display", a);

a=12;

// a <= 12;

end

endmodule

O/P  $\Rightarrow$  (=)

O/P  $\Rightarrow$  ( $<=$ )

5 display 6

5 display X

5 strobe 12

5 strobe 12



Date :

Page No.:

48

Q) module ex2;

integer a,b,c,d;

initial begin

a=6; \$display(\$time,a,b,c,d);

// a <= 6;

b=12; \$display(\$time,a,b,c,d);

// b <= 12;

end

initial begin

c=20; \$display(\$time,a,b,c,d);

// c <= 20;

d=25; \$display(\$time,a,b,c,d);

// d <= 25;

end

endmodule

o/p  $\Rightarrow$  (=)

0 6 x x x

0 6 12 x x

0 6 12 20 x

0 6 12 20 25

o/b  $\Rightarrow$  (c =)

0 x x x x

0 x x x x x

0 x x x x x

0 x x x x x

\$display(ex2) for the structure

Q) a=6;

Q) a <= 6;

#5 b=12;

#5 b <= 12;

#5 c=20

#5 c <= 20;

#5 d=25

#5 d <= 25;

;;,

o/b  $\Rightarrow$

0 6 x x x

5 6 12 x x

5 6 12 20 x

10 6 12 20 25

o/p  $\Rightarrow$

0 x x x x x

5 6 x x x x

5 6 x x x x

10 6 12 20 x

Q) \$Structure (ex2) r/w for the structure

$a = 6;$

$a <= 6;$

#5  $b = 12;$

#5  $b <= 12;$

#5  $c = 20;$

#5  $c <= 20;$

#5  $d = 25;$

#5  $d <= 25;$

\$time a b c d

0 6 x x x

5 6 12 20 x

5 6 12 20 x

16 6 12 20 25

\$time a b c d

0 6 x x x

5 6 12 20 x

5 6 12 20 x

10 6 12 20 25

Q) Integer a,b,c,d;

initial begin

\$monitor (\$time, a, b, c, d);

\$time a b c d

0 5 x x x

$a <= 5;$

3 5 6 x x

#3  $b = 6;$

5 5 6 10 x

begin

10 5 6 10 6

#2  $c <= 10;$

15 3 6 10 6

#5  $d <= 6;$

end

#5  $a = 3;$

end

## Conditional Statements

- \* Conditional statements are statements in which assignments occurs when certain condition will be true or false.
- \* 'if-else' are used as conditional statements.
- \* Nesting of conditional statements is also allowed.
- \* Expression is considered as true if it evaluates to a non zero value.
- \* Expression is considered as false if it evaluates to a zero or ambiguous (x) value.
- \* Syntax : if without else

    if (<expression>)

        begin

            true - statements;

        end

Ex : D latch

always @ (enable, d)

    if (enable)

        begin

            q = d;

    end .

- \* If 'else' part is not written it will call the previous value (latch inferring).

- \* Syntax : if with else

    begin module test (a,b,c);

        if (<expression>) begin

            input a, b;

            true - statements;

            output c;

        end

        always @ (a,b) begin

            if (a == b)

                c = 0;

        end

        else

            c = 1;

    endmodule

- \* Nested if - else :-

    if (<condition>) begin

        end

        true - statement;

    else begin

        el if (<condition2>) begin

            default - statements;

        true - statement;

    end .

\*\* Difference between if - else & ternary operator

- In if-else → Condition - 1 True Statement

Condition - '0' or '1' False Statement

- In ternary → Condition - 1 True Statement

Condition - 0 False Statement

Condition - 1/2 Matching the true & false statements

Ex) assign y = (1'b x)? 8'b 1010\_0100 : 8'b 0101\_0101;

$$y = \begin{matrix} x & x & x & x \\ & & & 010x \end{matrix}$$

$$\begin{array}{r} 1010 \quad 0100 \\ 0101 \quad 0101 \\ \hline xxx \quad 010x \end{array}$$

Ex) if (1'b x) y = 8'b 1010\_0100;

else y = 8'b 0101\_0101;

$$y = \begin{matrix} 8'b 01010101 \\ // \end{matrix}$$

### Multi-way Branching

- \* Verilog provides case statement which tests whether expression matches one of the multiple alternatives.
- \* The expression is compared to the alternatives in the order they are written.
- \* Statements corresponding to first alternative that matches the expression are executed.
- \* In this if no match is found, default statement is executed.
- \* default statement is optional.
- \* default statement can be written in any where, it is always the last alternative expression is to be compared to.
- \* Nesting of case statement is allowed.
- \* begin and end keyword is optional if there is only one statement to be executed for given alternative then begin & end need not to be written.

- \* The case statement is different from if-else in following ways :
  - Expression given in if-else is more general while a single expression is matched with multiple item in case statement.
  - Case provides the definitive results when  $x\bar{z}$  are present in the expression.

### \* Syntax :

`case (<expression>)`

alter 1: begin

multiple statements 1; end

alter 2: begin

single statement 2;

default : begin

default statements; end

end case

\* Ex: 4x1 mux

always @(\*)

case (sel)

2'b00 : out = a;

2'b01 : out = b;

2'b10 : out = c;

2'b11 : out = d;

default : \$display ("invalid statement");

endcase

- \* In 'case' latch inference will be occurring when all the alternatives are not evaluated & there is no default statement also.

### → Case :

In a normal case statement, valid inputs are 0, 1, x & z

The expression & alternatives are compared bit with bit including x & z.

### → Case Z :

In caseZ, valid inputs are 0, 1 & x.

It treats 'Z' or '?' present in expression or alternative as don't care

### → Case X :

In caseX, valid inputs are 0 & 1

It treats x, z or ? present in the expression or alternative as don't care.

Ex) always @ (\*)

case (sel)

2'b0x : out = a;

2'b0? : out = b;

2'b10 : out = c;

2'b11 : out = d;

default : out = 0;

endcase

always

// casez (sel)

// casez (sel)

O/P  $\Rightarrow y$  sel = 00 case: out = b

sel = 01 case: out = a

sel = 02 case: out = b

sel = xx case: out = 0

casez: out = b

casez: out = a

casez: out = a

casez: out = 0

casez

casez : out = a

\* Return case :

case (1)

s[1] : Statement;

s[0] : Statement;

default : Statement;

endcase



\* Parallel case : It only checks only one case at a time

case (s)

2'b00 : y = 10;

only one alternative is written in a case

endcase

writing multiple case will make it parallel case.

## $\Rightarrow$ Race Condition :-

- \* In Verilog sometimes more than one assignments are scheduled for execution at same time.
- \* In these situations non-deterministic condition occurs, that is known as race around condition.
- \* In race around conditions order of execution dependent upon simulator.
- \* To avoid these situations one can use #0 modeling.

Ex) module race-around;

reg [1:0] a = 2'b01, b = 2'b10;

initial a = b;

initial b = a;

endmodule

• Definitely both a & b will be same.

• Depends upon which initial block executes first that depends upon simulation.

st  $\rightarrow$  a = 2'b10 & b = 2'b10

2<sup>nd</sup>  $\rightarrow$  a = 2'b01 & b = 2'b01

module race-around-sol;

reg [1:0] a = 2'b01, b = 2'b10;

a = 2'b10 & b = 2'b01 (swap)

initial a <= b;

initial b <= a;

endmodule

## Loops IN VERILOG

- \* Loops in verilog are used to execute the same piece of code a number of times.
- \* For iteration functionality verilog basically provides four different loops: for, while, repeat, forever.
- \* Loops can have delay statements.
- \* We can combine more than one statements using begin-end block in an looping instruction.
- \* Looping statements should be used within a procedural block i.e inside the initial block or always block.
- \* Synthesized loop: for, repeat (to design HW we require fixed no. of iterations).

### → for loop:-

- \* for loop is most widely used loop in verilog to replicate the hardware.
- \* for loop executes till the condition is true.
- \* Syntax :

```
for (<initialization>; <condition-check>; <update-in-condition>)
begin
    statement 1;
    statement 2;
end
```

Ex module for-test;

O/P =>

```
integer i;
initial begin
    for (i=1; i<=8; i=i+1)
        begin
            $display ("The for loop count = %d", i);
        end
    end
endmodule
```

|                        |     |
|------------------------|-----|
| The for loop count = 1 | = 2 |
|                        | = 3 |
|                        | = 4 |
|                        | = 5 |
|                        | = 6 |
|                        | = 7 |
|                        | = 8 |

Q/P

• `int [2;0] a;``for(a=0; a<=7; a=a+1)``$ display("The value of a is %d", a);`0  
1  
2  
3  
4  
5  
6  
7  
:

Infinite loop.

e) infinite loop

- \* In verilog each part of the for loop is optional, but semicolon(;) must still be present to separate them.

i) `for(; i<5; i=i+1)`

- If not initialized, it may use a default or undefined value, leading to unpredictable behaviour.

ii) `for(i=0; ; i=i+1)`

- Treated as an infinite loop because there is no condition to stop.

iii) `for(i=0; i<5;) ;`

- No automatic change in i, so if not manually updated inside the loop, this may lead to infinite loop.

iv) `for(;;) ;` // infinite loop

- Equivalent to `while(1)`

## → While loop :-

- \* While loops are used in verilog often to run some code for an intermediate amount of time
- \* While loop executes until the condition is not true.
- \* Syntax  $\Rightarrow$  while (condition)

begin

multiple statements;

end

- \* While loops are not synthesizable, as when synthesis tool tries to convert while loop in gates & registers it must know the exact number of times loop will run. In case of while loop it is unknown.
- \* Very useful in testbenches & verification.

```
Ex) module whl_tst;
    integer count = 0;
    initial begin
        while(count != 7)
            #1 count = count + 1;
        $display ("The count value is = %d", count);
    end
endmodule
```

O/P  $\Rightarrow$  The count value is = 7

```
module clk_gen;
    reg clk;
    initial begin
        clk = 1'b0;
        while(1)
            #5 clk = ~clk;
    end
endmodule
```

O/P  $\Rightarrow$  It will generate clock continuously which flip its value after every 5 time units.

\* If 'begin' & 'end' was given multiple statements would have displayed (7)

→ Repeat loop :-

\* repeat loop executes the loop for a fixed number of times

\* repeat (<no. of time>)

begin

Statement 1;

Statement n;

end

Ex) module repeat\_test;

O/P :-

integer i=0;

The value of i is = 10

initial begin

output(0)

# i=i+1;

\$display ("The value of i is = %d",i);

end

endmodule

→ forever loop :-

\* forever is an infinite loop that executes without any condition

\* forever is equivalent to until(1)

\* The forever loop doesn't contain any expression, executes forever until "finis" task is encountered.

\* forever can be exited by use of disable statement

\* primarily used in test bench

\* Ex:- module clk\_gen;

reg clk;

initial begin

forever #5 clk = ~clk;

end

endmodule

\* Syntax :-

forever

begin

multiple statements;

end

Q Write a code which makes for loop work like a while loop vice versa.

• module for-with-while :

```
integer i;  
initial begin  
    i = 0;  
    while (i < 5) begin  
        $display ("while loop : i=%0d", i);  
        i = i + 1;  
    end  
end  
endmodule
```

• module while-with-for :

```
integer i = 0;  
initial begin  
    for (; i < 5;) begin  
        $display ("For loop : i=%0d", i);  
        i = i + 1;  
    end  
end  
endmodule
```



## Generate

=> Generate Statement :-

- \* Generate statement is used to generate replication of hardware which is provided in generate statement.
- \* It works in looping manner that's why also known as generate loop.
- \* All generate instantiations (hardware to be replicated) are written inside generate & endgenerate keywords.
- \* Generate instantiations can be combinations of any of the following:
  - module instances
  - User defined primitives
  - gate primitives
  - continuous assignments
- \* Verilog permits declaration of nets, registers & events in general scope.
- \* Declaration of parameters, local parameters, input, output, inout port & specify block is not allowed.

\* There are three types of generate statements

- i) Generate loop
- ii) Generate case
- iii) Generate condition

- \* "genvar" is the keyword used to declare variables that will be used only in evaluation of generate block.
- \* Value of genvar variable can only be modified by generate loop.
- \* Nesting of generate loop is allowed.
- \* Block identifiers are required for generate loop in most cases.

\* Syntax :-

• generate - for :-

genvar i;

generate

for (i=0;i<8;i=i+1)

begin : label

S1;  
S2;

end

endgenerate

• generate - conditional

generate

if (condition)

≡

else

≡

endgenerate

• generate - case

generate

case (variable) begin

c1;

c2;

default: statement;

endcase

end

endgenerate

→ 4-bit adder using generate for

\* // Full adder

```
module FA(input a,b,lin,output sum,carry);
assign {carry,sum} = a+b+{lin};
endmodule
```

• module adder (input [3:0] op1,op2, output cout, output [3:0] sum);

wire [4:0] carry;

assign carry[0] = 1'b0;

assign cout = carry[4];

genvar i;

generate for (i=0; i<4; i=i+1)

begin : ripple

FA fa (op1[i], op2[i], carry[i], sum[i], carry[i+1]);

end

endgenerate

endmodule

\* module adder #(parameter WIDTH=4) (input [WIDTH-1:0] op1,op2,

output cout, output [WIDTH-1:0] sum);

wire [WIDTH:0] carry;

assign carry[0] = 1'b0;

assign cout = carry[WIDTH];

genvar i;

generate for (i=0; i<WIDTH; i=i+1)

begin : ripple

assign {carry[i+1],sum[i]} = op1[i]+op2[i]+carry[i];

end

endgenerate

endmodule

→ Generate Conditional Example :-

\* module andor #(parameter WIDTH=4)(input [WIDTH-1:0] op1, op2,  
 output [WIDTH-1:0] result);  
 parameter logics = "and1";  
 genvar i;  
 generator  
 if (logics == "and1")  
 for (i=0; i<WIDTH; i=i+1)  
 begin : andlogic  
 assign result[i] = op1[i] & op2[i]; end  
 else  
 for (i=0; i<WIDTH; i=i+1)  
 begin : orlogic  
 assign result[i] = op1[i] | op2[i]; end  
 endgenerate  
 endmodule.

→ Generate Case Example :-

\* module adder (sum, carry, a, b, c);  
 parameter N=3;  
 input [N:0] a, b; input c;  
 output (carry) output [N:0] sum;  
 generate  
 case (c)  
 1: adder\_1bit a1(sum, carry, a, b, c);  
 2: adder\_2bit a2(sum, carry, a, b, c);  
 default: adder\_custom a3 #(N)  
 (sum, carry, a, b, c);

endcase

endgenerate

endmodule

## Procedural Block Statements

- \* Block statements are used to group two or more statements together so that they belong to single logic element.
- \* Procedural Block statements basically of two types:
  - begin - end
  - fork - join
- \* Begin-end is known as sequential block as all the statements inside this are executed in sequential manner
- \* fork-join is known as parallel block as all the statements inside this are executed at same simulation time manner.

### → Begin - end

- \* It is a sequential block
- \* Begin-end is synthesizable statement
- \* Delay statements are allowed
- \* Nesting is allowed
- \* Syntax => initial / always

```
begin  
  S1;  
  S2;  
  S3;  
end
```

Ex) integer a=0, b=0, c=0, d=0, e=0;

initial begin

a=5;                    // a=5    0

#3 b=7;                // b=7    3

#5 c=4;                // c=4    8

#8 d=10;              // d=10   16

#2 e=9;                // e=9    18

end

→ fork - join :-

- \* It is a concurrent block
- \* fork - join is non-synthesizable statement
- \* Delay statements are allowed
- \* Nesting is allowed.
- \* Syntax → initial / always

fork

S1;

S2;

Sn;

cjoin

e.g) integer a=0, b=0, c=0, d=0, e=0;

initial fork

a=5; // a=5 0

#3 b=7; // b=7 3

#5 c=4; // c=4 5

#8 d=10; // d=10 8

#2 e=9; // e=9 2

join

\*\* If we are not providing any delay then the execution of fork - join  
↳ begin-end is same.

\* module test:

| Time | a | b | c | d | e |
|------|---|---|---|---|---|
|------|---|---|---|---|---|

integer a=0, b=0, c=0, d=0, e=0;

|   |   |   |   |   |
|---|---|---|---|---|
| 0 | 5 | 0 | 0 | 0 |
|---|---|---|---|---|

initial begin

|   |   |   |   |   |
|---|---|---|---|---|
| 5 | 5 | 7 | 0 | 0 |
|---|---|---|---|---|

\$monitor(\$time, a, b, c, d, e);

|    |   |   |   |   |
|----|---|---|---|---|
| 11 | 5 | 7 | 4 | 0 |
|----|---|---|---|---|

a=5; #3 b=7; #6 c=4;

|    |   |   |   |   |
|----|---|---|---|---|
| 12 | 9 | 7 | 4 | 0 |
|----|---|---|---|---|

fork

|    |   |   |   |   |
|----|---|---|---|---|
| 15 | 9 | 7 | 4 | 3 |
|----|---|---|---|---|

#1 a=9; #4 d=3; #5 c=8;

|    |   |   |   |   |
|----|---|---|---|---|
| 16 | 9 | 7 | 8 | 3 |
|----|---|---|---|---|

join

|    |   |   |   |    |
|----|---|---|---|----|
| 24 | 9 | 7 | 8 | 10 |
|----|---|---|---|----|

#8 d=10; #2 e=9;

|    |   |   |   |    |
|----|---|---|---|----|
| 26 | 9 | 7 | 8 | 10 |
|----|---|---|---|----|

end endmodule

\* module tb;

integer a=0, b=0, c=0, d=0, e=0;

initial fork

a=2; #6 b=7; #3 c=4;

begin

\$monitor (\$time, a, b, c, d, e);

#1 b=9; #4 d=3; #5 c=1;

end

#9 d=10; #2 e=9;

join

endmodule

|  | TPme | a | b | c | d  | e |
|--|------|---|---|---|----|---|
|  | 0    | 2 | 0 | 0 | 0  | 0 |
|  | 1    | 2 | 9 | 0 | 0  | 0 |
|  | 2    | 2 | 9 | 0 | 0  | 9 |
|  | 3    | 2 | 9 | 4 | 0  | 9 |
|  | 5    | 2 | 9 | 4 | 3  | 9 |
|  | 6    | 2 | 7 | 4 | 3  | 9 |
|  | 9    | 2 | 7 | 4 | 10 | 9 |
|  | 10   | 2 | 7 | 4 | 10 | 9 |

### Named Block

- \* Verilog allows giving names to block statements. These blocks are called named blocks.
- \* Local variables can be declared for named blocks.
- \* Variables in a named block can be accessed by using hierarchical name referencing.
- \* It is possible to stop execution ( disable ) of named block from within the same block or different block.
- \* Syntax =>

initial

begin : Blk1

integer count;

// count variable back to Blk1

statements;

end

initial

begin : Blk2

reg a; // a variable local to Blk2

statements;

end

\* module tb;

o/p

initial

sim time

count

begin : block 1

0

0

integer count;

5

1

count = 0;

10

2

\$monitor(\$time, count);

15

3

forever begin

20

4

# \$count = count + 1;

25

5

if (count == 8)

30

6

    \$display(block 1);

35

7

→ printing values

end end

40

8

endmodule

\* module tb;

Time a b Count

initial begin : block 2

0

2

0

b = 2;

1

3

1

begin : block 1

2

3

2

forever

3

3

3

# \$count = count + 1;

4

2

4

end

5

7

5

end

6

7

6

initial begin

7

7

7

\$monitor(\$time, a, b, count);

8

2

8

a = 3; # \$a = 7; # \$a = 2;

9

2

9

# \$2 disable block 2, block 1;

10

2

10

# \$3 a = 9; (block 1 will be disabled)

14

9

2

end

endmodule

## Functions & Tasks

- \* Verilog lets you define sub-programs using tasks & functions.
- \* They are used to improve the readability & to exploit re-usability code.
- \* Definition of task & function must be in a module.
- \* Task or functions can only contain sequential statements.
- \* Functions can accept any number of inputs, do some processing on inputs & return only one output.
- \* Tasks is more general, that can calculate multiple result values & return them in output or inout type arguments.
- \* Tasks may contain @, posedge & other simulation time consuming elements.

=> Functions :-

- \* Declaration of functions started with keyword "function" & ended with "endfunction".
- \* Function always return a single value.
- \* They can't have output & inout ports.
- \* A function must have at least one input argument. It may contain more than one input.
- \* Delays, Event & timing control statements are not supported by functions.
- \* Functions always executes in zero simulation time.
- \* When a function is declared, a register with the same name is implicitly (by default) declared in verilog.
- \* A value is returned back by assigning appropriate value to the implicit register (function name).
- \* Functions can't have any triggers.
- \* The function is invoked by specifying functions name followed by input arguments.
- \* A function can be called from a function, task, port declaration, continuous assignment & procedural assignment.



\* A function can invoke other functions but it can't invoke other tasks.

\* Syntax  $\Rightarrow$

• Function [return type] function-name;

    input declarations;

    // local registers declaration;

    begin

        sequential statements;

    end

endfunction

• function [return type] function-name (input declarations);

    // local register's declarations;

    begin

        sequential statements;

    end

endfunction.

\* Return type & all inputs are 1-bit reg by default

\* Return variable name is same as function-name

\* Only blocking sequential statements are allowed

\* Functions have access to variables declared outside its scope

$\rightarrow$  How to call function ??

• assign mt\_name = function-name (input\_list);

• reg\_name1 = function-name (input\_list);

• reg\_name2 <= function-name (input\_list);

\* Function always executes in Active region.

Q Write a function which accepts 8 bit input din & returns parity.  
Use for loop instead of reduction operator.

function parity (input [7:0] din);

integer i;

reg temp;

begin

temp = 0;

for(i=0; i<=7; i=i+1)

begin

temp = temp ^ din[i];

parity = temp;

end

endfunction

module parity-call(din, dout);

input [7:0] din;

output dout;

assign dout = parity(din);

endmodule

Q Write a function to return maximum out of three integers a,b,c.

function integer maximum;

input signed [31:0] a,b,c;

begin

if(a > b & a > c)

maximum = a;

else if(b > c)

maximum = b;

else

maximum = c;

end

endfunction

module max-val (a,b,c,max-val);

input signed [31:0] a,b,c;

output integer max-val;

assign max-val = maximum(a,b,c);

endmodule.

## → Automatic Function :-

- \* A function call allocates static memory to variables used in its operations.
- \* If the same function is called again, same set of allocated memories are used for its operation.
- \* In case of recursive or re-entrant functions (function called within a function) using static memories may give wrong result.
- \* "automatic" keyword is added after function keyword to target a recursive function.
- \* An automatic function allocates a new set of memory to the variables every time a function is called.

Q Write a function which accepts positive integer & returns factorial of that number

function [63:0] fact (input [31:0] din);

begin

if (din)

fact = fact(din - 1) \* din;

\* module fact; fact(din, dout);

input [31:0] din;

else

output reg [63:0] dout;

fact = 1;

always @ (\*)

end

dout = fact(din);

endfunction

endmodule

## ➤ Static function Problem



- \* To solve this issue we will add automatic keyword because we intend to design a recursive function.
- \* Function automatic [63:0] fact (input [31:0] din);



- \* Since automatic keyword is added to the function, for every function call a new set of memory is created & the result is returned to location where the function call occurs.

- \* Verilog supports some other type of functions :
  - Signed function
  - Constant function

- function signed my\_func(input list);

begin

sequential statements;

end

endfunction

- \* Function [31:0] log2(input [31:0] data);

reg [31:0] temp;

begin

temp = 0;

while (data > 1) begin

data = data >> 1;

temp = temp + 1;

end

log2 = temp;

end

endfunction

## ⇒ Tasks :-

- \* Tasks are declared with keyword task and endtask.
- \* Tasks can pass multiple values through output & inout ports.
- \* A task may have zero or more arguments of type input, output or inout.
- \* Delay, event & timing control statements are allowed inside tasks.
- \* Tasks may execute in non-zero simulation time.
- \* A task can be called from a task & procedural assignment.
- \* A task can invoke other tasks & functions.
- \* A task is invoked by specifying task name followed by input, output or inout arguments.
- \* The argument should be specified in the same order they are declared.
- \* Tasks can be disabled by "disable" keyword.

\* task task\_name;

    input, output or inout declaration;

    local registers declaration;

    begin

        // begin-end if multiple statements

        [Timing Control]

        sequential statements;

    end

endtask

• task task\_name ( Port Declaration);

- \* By default the size of ports & local arguments is 1 bit reg.
- \* Both blocking & non-blocking sequential statements are allowed.
- \* Call → task-name (input, output or inout list);

Ex)

```

task increment;
    inout integer a;
    a=a+1;
endtask

modul task_inout;
    integer x=0;
    // Task duration
    initial begin
        increment(x); // Task call
        $display ("x=%d",x);
        x=6;
        increment(x);
        $display ("x=%d",x);
    end endmodule

```

O/p =>

x=1

x=7

#### → Static & Automatic Tasks :-

- \* Static task is the default task in verilog.
- \* In static tasks all the member variables will be shared across different invocation of same task.
- \* Automatic Task : "automatic" keyword is placed after task to make task automatic.
- \* In automatic tasks, for each invocations of same task a new memory is allocated to its member variables.

Ex) modul task\_demo;

task display();

integer x=0;

x = x+2;

\$display ("x=%d",x);

endtask

initial display();

initial display();

endmodule

O/p => x=2

x=4

modul task\_demo;

task automatic display();

integer x=0;

x = x+2;

\$display ("x=%d",x);

endtask

initial display();

initial display();

endmodule

O/p => x=2

x=2

## → Global Task :-

- \* The task that are declared outside the module are known as global tasks.
- \* The scope this tasks is global, thus can be called within any module.

```

• task display();
  $display ("Welcome to FutureWiz");
endtask

• module task_demo2;
initial begin
  $display ("The task_demo2:");
  display();
end
endmodule

```

```

• module task_demo1;
initial begin
  $display ("The task_demo1:");
  display();
end
endmodule

```

O/P => The task\_demo1

welcome to FutureWiz

The task\_demo2

Welcome to FutureWiz

## → How to disable a Task :-

- \* Any task can be disabled using the keyword "disable".

Ex) module task\_disable\_demo;

task display();

begin: Message\_display1

\$display("At Sim Time = %t Welcome to FutureWiz", \$time);

# 50 \$display("At Sim Time = %t This is situated in Noida", \$time);

end

begin: Message\_display2

```

#10 $display(`At sim time = %t FutureWig is a VLSI training institute`,$time);
#10 $display(`At sim time = %t it is the venture of Tranchip`,$time);
end endtask

initial display();
initial begin
  #40 disable display;
  $display(`Welcome to FutureWig`);
end endmodule
  
```

O/p => At sim time = 0 Welcome to FutureWig

At sim time = 50 FutureWig is a VLSI training institute

At sim time = 60 It is the venture of Tranchip

```

initial display();
initial begin
  #40 disable display;
  $display(`Welcome to FutureWig`);
end endmodule
  
```

O/p => At sim time = 0 Welcome to FutureWig

At sim time = 50 This is situated in Noida

At sim time = 60 FutureWig is a VLSI training institute

At sim time = 70 It is the venture of Tranchip

### → Difference Between Task & function :-

#### Function

#### Task

- Functions are used to return only single value.
- Task can provide multiple values.
- Function must have at least one input argument.
- Task may have zero or more arguments of type input, output or inout.
- Timing control statements are not allowed.
- Timing control statements are allowed.
- Always executes at zero simulation time.
- May or may not execute at zero simulation time.

- Only blocking assignments are allowed
- Both blocking & non-blocking assignments are allowed
- Functions can invoke other function
- Tasks can invoke other tasks as well as functions.
- Only

## => Parameters :-

- \* Parameters are used to declare constants in verilog.
- \* These constants can be overridden during compilation time (tb)
- \* "dyparam" keyword is used to override a parameter.
- \* Syntax => dyparam instanciable-parameter\_name = value;

Ex) module custom-adder(a,b, Cin, sum, carry);

parameter size=2;

input [size-1:0] a, b;

input Cin;

output [size-1:0] sum;

output carry;

assign {carry, sum} = a + b + Cin;

endmodule

• module adder\_test;

dyparam m0.size=4;

reg Cin; reg[m0.size-1:0] a, b;

wire [m0.size-1:0] sum; wire carry;

Custom-adder m0(a, b, Cin, sum, carry);

Initial begin output(5) begin

a = \$random; b = \$random; c = \$random;

#10; \$display(a, b, Cin, sum, carry);

end end endmodule

## DELAY'S

- \* Delays specify time in which assigned values propagate through nets or from inputs to outputs of ports or from one port to another port.
- \* Aim of any designer may be to produce a circuit that functions correctly, it is equally important that the circuit also conforms to any timing constraints required of it.
- \* Delay & timing control statement are not synthesizable.
- \* Delays can be modelled in a variety of ways, depending on the overall design approach which are mentioned below:
  - Gate level modelling
  - Data flow modelling
  - Behavioral modelling

### Delays

- Gate level modelling
- Gate delays
- ↓
- Data flow modelling
- Net duration delay
- Regular assignment delay
- Implicit continuous assignment
- ↓
- Behavioral modelling
- Regular delay/Inter control
- Intra-Assignment delay
- Better assignment delay

### ⇒ Gate delays :-

- \* They delays basically provided with in-built gate primitives, continuous assignment & user define primitives.
- \* Gate delays can be classified as following :
  - Rise delay
  - Fall delay
  - Turn-off delay

- \* Run delay encountered only when output changes from any state  $(0, x, z)$  to 1.
- \* Fall delay encountered only when output changes from any state  $(1, x, z)$  to 0.
- \* Turn-off delay encountered only when output changes from any state  $(0, 1, x)$  to 2.
- \* Unknown delay : In case when output goes from  $(1, 0, 2)$  to  $x$  then minimum out of run, fall & turn-off delay will be assigned.

#### \* Syntax $\Rightarrow$

primitive-name / assign / wdp # (run, fall, turn-off) label (o/p, i/p1, i/p2);

- Ex :-
- and # (2,3,4) uo(y,a,b);      // RT = 2 , FT = 3 , TOT = 4
  - and # (2,3) uo(y,a,b);      // RT = 2 , FT = 3 , TOT = 2 (min)
  - and # (2) uo(y,a,b);      // RT = 2 = FT = TOT
  - assign # (3) out1 = a & b;      // 3 value will be assigned to all delays
  - assign # (2,1) out2 = a ^ b;      // RT = 2 , FT = 1 , TOT = 1

- \* If no delay is specified then delay value is '0'.

```

• module my_design (input x,y, output o1,o2);
  and #(1) a1(o1,x,y);
  bufjo #(2) b1(o2,x,y);
endmodule

module tb;
  reg a,b; wire o1,o2;
  my_design dut(a,b,o1,o2)
initial begin
  a=0; b=0;

```

```

$monitor ("T=%0d a=%0d b=%0d out1=%0d out2=%0d", $time, a, b, out1, out2);
#10 a<=1; #10 b<=1; #10 a<=0; #10 b<=0;
end
endmodule
  
```

| O/p => | T  | a | b | out1 | out2 |
|--------|----|---|---|------|------|
|        | 0  | 0 | 0 | z    | z    |
|        | 1  | 0 | 0 | 0    | z    |
|        | 2  | 0 | 0 | 0    | 0    |
|        | 10 | 1 | 0 | 0    | 0    |
|        | 12 | 1 | 0 | 0    | 1    |
|        | 26 | 1 | 1 | 0    | 1    |
|        | 21 | 1 | 1 | 1    | 1    |
|        | 22 | 1 | 1 | 1    | z    |
|        | 30 | 0 | 1 | 1    | z    |
|        | 31 | 0 | 1 | 0    | z    |
|        | 40 | 0 | 0 | 0    | z    |
|        | 42 | 0 | 0 | 0    | 0    |

→ min, typ & max delays:

- \* Delays are neither the same in different parts of the fabricated chip nor the same for different temperatures & other variations.
- \* Every digital gate & transistor cell has a minimum, typical & maximum delay specified based on process node & is typically provided by libraries from fabrication foundry.
- \* For rise, fall, & turn-off delays, the three values min, typ & max can be specified.
  - min: minimum, typ: typical & max: maximum.
- \* This is another level of delay control in verilog. Only one of the min, typ & max values can be used in the entire simulation run.
- \* It is specified at the start of the simulation & depends on the simulator used.
- The min value is the minimum delay value that the gate is expected to have.
- The typ value is the typical delay value that the gate is expected to have (default).
- The max value is the maximum delay value the gate is expected to have.

\* Syntax  $\Rightarrow$

```
primitive_name/assign/udp #(min:typ:max, min:typ:max, min:typ:max)
  label (output, input1, i/p2)          |  rise   | Fall   | Turnoff
```

`end` and  $\#(2:3:4)$  `vo`(out, ip1, ip2); // for all transition rise delay = 2

$$Typ = 3 \text{ & } Max = 4$$

and  $\#(2:3:4, 3:5:7)$  `vi`(out, ip1, ip2); // for Rise delay - Min = 2 T = 3 Max = 4

$$\text{Fall delay} = \text{Min} = 3 T = 5 \text{ Max} = 7$$

`bufif0 #(2:3:4, 3:4:6, 4:3:7) vo`(out, ip1, trq1); RD - Min = 2 T = 3 Max = 4

$$FD - Min = 3 T = 4 Max = 6$$

$$TO - Min = 4 T = 5 Max = 7$$

$\Rightarrow$  Data flow delays / Net delays :-

$\rightarrow$  Net declaration delay :-

- \* In this delay assignment, we assign the delay value at the time of declaration.

Ex) wire #10 c; // At the time of declaration delay is assigned  
 assign c = a & b;

$\rightarrow$  Implicit continuous assignment delay :-

- \* Delay value is assigned at the time of implicit declaration.

Ex) wire #10 c = a & b;

$\Rightarrow$  Behavioural modelling delays :-

$\rightarrow$  Regular / Enter assignment delay :-

- \* In regular / enter assignment delay, the execution of statement is delayed by a specified time.

- \* First delay is performed & then register is updated with the value at current time.

- \* Any event on the sensitivity list is ignored till all the delays are executed.

- \* Both blocking & non-blocking shows the similar behaviour with regular delay.

\* Syntax :

# delay-value output = expression ; or

# delay-value output <- expression

Ex) module delay (input signed [31:0] a, output integer b);

always @ (a)

# 3 b = a; // # 3 b <- a; will produce some results

endmodule

→ Intra Assignment Delay :-

- \* In intra assignment delay control, the assignment of statement is delayed by a specified time.
- \* Current values are read & assigned to the register after specified delay.
- \* Any event on the sensitivity list is ignored till all the delays are executed for blocking statements.
- \* Blocking & non-blocking statements don't show same behaviour with this type of delay.
- \* Syntax :-

Output = #delay-value expression; or

Output <= #delay-value expression;

Ex) module delay (input signed [31:0] a, output integer b);

begin

#3 b=a; // #3 b <= a;

O/p =>

endmodule

Simtime = 0, a=10, b=x

module delay\_tb;

Simtime = 3, a=10, b=10

reg signed [31:0] a;

Simtime = 4, a=15, b=10

wire integer b;

Simtime = 7 a=15, b=15

delay d1(a,b);

initial begin

a=10;

\$monitor ("Simtime=%0t, a=%0d, b=%0d", \$time, a, b);

#4 a=15;

end

endmodule

### ~~Assignment Statement~~

### Description

- $\#5 z = x + y;$ 
  - $x + y$  is evaluated after 5 time units & assigned to  $z$
  - $x + y$  changes within 5 time units, ignored
  - If input pulse is shorter than 5 time units then also changes is ignored.
- $\#5 z \leftarrow x + y;$ 
  - same as  $\#5 z = x + y$
- $z = \#5 x + y;$ 
  - $x + y$  evaluated immediately but assigned to  $z$  after 5 time unit
  - If  $x + y$  changes its value within 5 time unit, produces no effect on  $z$ .
- $z \leftarrow \#5 x + y;$ 
  - $x + y$  evaluated immediately & scheduled assignment to  $z$  after 5 time unit.
  - Further changes in  $x + y$  are taken in consideration
  - This statement behaves according to hardware behaviour.

S2 module tb;

begin a=10, c=15 b=5;

initial begin

#5 a=5; // Simulation = 5 a=5

\$display(a);

b = #2 //;

\$display(b); // Simulation = 7 b=11

c = #5 20;

\$display(c); // Simulation = 12 c=15

\$display(c); // Simulation = 12 c=20

#5 c = 30;

\$display(c); // Simulation = 17 c=20

\$display(c); // Simulation = 17 c=30

end

A) integer  $a=5, b=7, c=9;$

initial begin

#5  $b=c-a;$

#5  $c=a*2;$

#5  $a=b+c;$

\$display(a,b,c); 14 4 10

end

endmodule

## Switch Level Modeling

- \* Verilog supports another level of abstraction in which MOS circuits can be designed known as switch level modelling.
- \* The switch level of modelling shows the abstraction level between the digital logic & analog transistor levels of abstraction.
- \* At switch level, transistor works only in two regions (0,1) But verilog uses a 4 value logic value system. so verilog switch input and output signals can take any of the four (0,1, z, x) logic values.
- \* Switch level of modeling is rarely used by RTL designers now a days, because of the higher level of complexity at switch level.

→ Basics of MOS switches:-

- \* Metal oxide semiconductor field effect transistor (MOSFET) is generally known as MOS.
- \* MOSFET's are voltage controlled switches.
- \* Two type of MOSFET's :
  - NMOS (N-channel MOSFET)
  - PMOS (P-channel MOSFET)
- \* It has three connection points : a source, a drain, & a gate.



• Combination of both nmos & pmos switches is known as cmos technology.



• CMOS stands for Complementary metal oxide semiconductor.



|     |            |   |
|-----|------------|---|
| A = | -          | Y |
|     | 1          | 0 |
|     | A → D0 → Y |   |

→ Primitives of switch level modelling :-

- \* For switch level design verilog supports switch primitives.
- \* These primitives can be used like the normal gate primitives.
- \* These switch primitives are not synthesizable.
- \* Verilog provides following switch level primitives.

- pmos                      • tran
- nmos                      • tranijo
- cmos                      • tranif1



\* Syntax →

- nmos (out, in, control);                      • pmos (out, in, control);
- control = 0 out = 2                              control = 0 out = in
- control = 1 out = in                              control = 1 out = 2



Pen control are usually complement of each other

→ NAND & NOR gate :-



## → Bidirectional switches :-

\* Verilog provides following Bidirectional switches:

- tran
- tranif0
- tranif1
- tran (inout1, inout2)



tranif1 (inout1, inout2, control)

tranif0 (inout1, inout2, control)

| control | tranif0         | tranif1         |
|---------|-----------------|-----------------|
| 0       | inout2 = inout1 | inout2 = 2      |
| 1       | inout2 = z      | inout2 = inout1 |

## → Resistive Switches :-

\* Resistive switches have higher source to drain impedance than regular switches & reduce the strength of signals passing through them.

\* nmos, pmos, cmos & bidirectional switches can be modeled as resistor drivers in verilog.

\* Resistive switches are declared with keyword having an "r" prefixed to the corresponding regular switch keyword.

\* Verilog provides following resistive switches:

- rnmos
- rpmos
- rtran
- rtranif0
- rtranif1

\* Same syntax as regular switches.

## → Signal strengths in switch level modeling:-

\* Verilog has 4 driving strengths, 3 - capacitor / storage strengths & high impedance.

\* Syntax  $\Rightarrow$

(Strength 0, Strength 1) or (Strength 1, Strength 0) or capacitor - strength

\* Strength 0 : Strength when the net drives the value 0.

\* Strength 1 : Strength when the net drives the value 1.

- \* Capacitive strength is only for the trieng nut only.
- \* The default strength is strong drive.
- \* For pulleys & pulldown gates, the default strength is pull drive.
- \* For trieng the default strength is medium capacitive.
- \* For supply nuts, the default strength is supply drive.

|        | Strength | Strength 0 | Strength 1           | Strength type | Degre |
|--------|----------|------------|----------------------|---------------|-------|
| Supply | supply 0 | supply 1   | Driving              | Highest       |       |
| Strong | strong 0 | strong 1   | Driving              |               | ↑     |
| Pull   | pull 0   | pull 1     | Driving              |               |       |
| large  | large 0  | large 1    | Storage/Capacitive   |               |       |
| weak   | weak 0   | weak 1     | Driving              |               |       |
| medium | medium 0 | medium 1   | (capacitive/strong)  |               |       |
| Small  | small 0  | small 1    | Capacitive (Storage) |               |       |
| High-Z | High Z   | High 1     | High Z               | lowest        |       |



Q) Write a Switch level code to implement NAND & NOR gate.

\* //NAND

module nand\_switch(output out, input a, b);

wire n;

supply1 Vdd;

supply0 Gnd;

pmos p1(out, Vdd, a); //series connection

pmos p2(out, Vdd, b);

nmos n1a(n1, Gnd, a);

//Parallel connection

nmos n1b(out, n1, b);

endmodule

\* //NOR

module nor\_switch(output out, input a, b);

wire p1\_out;

supply1 Vdd;

supply0 Gnd;

pmos p1(out, Vdd, a);

//parallel connection

pmos p2(out, Vdd, b);

nmos n1(p1\_out, Gnd, a);

//series connection

nmos n2(out, p1\_out, b);

endmodule

## User Defined Primitives

- \* User defined primitives define new primitives, which functions exactly the same as built-in primitives in verilog.
- \* Instances of these new UDPs can then be used in exactly the same manner as the gate primitives to represent the circuit being modeled.
- \* Evaluation of these UDPs is accelerated by the verilog-XL algorithm.
- \* This technique can reduce the amount of memory that a description needs & can improve simulation performance.
- \* UDP definitions are independent of a module. They are at same level as that of module definition in syntax hierarchy.
- \* UDP don't contain any module or primitive instantiation inside it. They are instantiated the same way other modules & gate level primitives are instantiated.

→ Properties :-

- \* One of the advanced concepts of verilog
- \* Non Synthesizable
- \* Single output many input
- \* Consume very less memory
- \* I/O's must be scalar type only.
- \* 'Z' value is not supported by UDPs

→ Types of UDPs :-

- \* 2 categories:-
  - Combinational UDPs
  - Sequential UDPs

→ Combinational UDPs:-

- \* Output has to be first port in the port list.
- \* Output state is function of current input states. whenever input

state changes, VDP is evaluated & output state is set to a value as indicated by entry in the state table for current input state.

\* The output is 'X' for a combinational of input state not present in the state table.

\* primitive primitive-name (output, input, input, ...)

    output terminal declaration;

    input terminal declaration;

    table // state table entries

        table-entry;

        table-entry;

        ;

    endtable

endprimitive

→ State Table :-

- \* State table is specified with help of 'table' & 'endtable' keyword.
- \* Each entry in the state table of a combinational VDP has following syntax  $\Rightarrow$  input<sub>1</sub>-val input<sub>2</sub>-val ... input<sub>N-val</sub> : output-val;
- \* Input values in the state table should be specified in the same order as they appear in port (terminal) list.
- \* ":" is used to separate in inputs & output
- \* ";" is used to end a state table entry.
- \* All possible combinations of inputs where output produces a known value, must be explicitly covered.
- \* Combinations of the inputs that are not explicitly specified will drive the output to the unknown value 'X'.
- \* "?" is used to represent don't care condition  
    ? covers 0, 1, X values.
- \* There should not be any conflicting entries in the state table.

\* State task for OR operation is written below:

\* table

11 a b : c; primitive my\_OR(y,A,B);

0 0 : 0;

Output X;

0 1 : 1;

Input A,B;

1 0 : 1;

table

1 1 : 1;

0 0 : 0;

1 x : 1;

0 1 ? : 1;

x 1 : 1;

? 1 : 1;

endtable

endtable

endprimitive

AND:-

\* primitive my\_AND(y,A,B);

Output X;

Input A,B;

table

0 0 : 0;

1 1 : 1;

0 1 : 0;

endtable

endprimitive

| A | B | y |
|---|---|---|
| 0 | 0 | 0 |
| 0 | 1 | 0 |
| 1 | 0 | 0 |
| 1 | 1 | 1 |
| 0 | X | 0 |
| X | 0 | 0 |

XOR:-

0 0 : 0;

\* XNOR

0 0 : 1;

0 1 : 1;

0 1 : 0;

1 0 : 1;

1 0 : 0;

1 1 : 0;

1 1 : 1;

### → Sequential UDPs :-

- \* Output has to be the first port & has to be declared as "sig".
- \* An initial statement can be used to initialize output of sequential UDP.
- \* The input specification of state table entries can be in terms of input levels or edge transitions.
- \* All possible combinations of input must be specified to avoid unknown output values.
- \* primitive primitive\_name (output, input, input, ...);  
    output sig output-terminal;  
    input input-terminal;  
    initial output-terminal = logic-value;  
    table // State table entries  
        table-entry;  
        table-entry;  
    endtable  
endprimitive

- \* Each entry in the state table of a sequential UDP has following Syntax:

    input-value : previous-output-state : output-value;

- \* Output-value is evaluated based on inputs & previous-output-state.
- \* This value is specified in output-value.
- \* "—" is used to represent no change in state.

• primitive d\_latch (q, d, clear, en);  
    output q;  
    sig q;  
    input d, clear, en;  
    initial q=0;

table

// d char en : q: q<sup>+</sup>

? 1 ? : ? : 0; // char=1, op=0

? 0 0 : ? : -; // char=0, en=0 op= latches

0 0 1 : ? : 0; // char=0, en=1 op=d

1 0 \* : ? : 1; // char=0, en=1 op=d

endtable

endprimitive

\* primitive T-latch (q, T, char, en);

Output q;

reg q;

input T, char, en;

initial q=0;

table

// T char en : q: q<sup>+</sup>

? 1 ? : ? : 0;

? 0 0 : ? : -;

0 0 1 : 0 : 0;

1 0 1 : 0 : 1;

0 0 1 : 1 : 1;

1 0 1 : 1 : 0;

endtable

endprimitive.

## \* Sequential State Table :-

### State Table Symbols

|        | Definition                                                                               |
|--------|------------------------------------------------------------------------------------------|
| 0      | • logic 0 on input or output                                                             |
| 1      | • logic 1 on input or output                                                             |
| x or X | • unknown on input or output                                                             |
| -      | • no change on output (may be only to use with sequential UPDs)                          |
| ?      | • don't care if an input is 0, 1, or x                                                   |
| b or B | • don't care if an input is 0 or 1                                                       |
| (vw)   | • input transition from logic v to logic w<br>eg: (01) represents transition from 0 to 1 |
| r or R | • rising input transition : same as (01)                                                 |
| f or F | • falling input transition : same as (10)                                                |

- \* for edge sensitive UDP transition are denoted by writing old value followed by a new value inside ( ).

ex)  $(01) \rightarrow 0 \text{ to } 1$        $(02) \rightarrow 0 \text{ to } 0, 1 \text{ or } x$   
 $(10) \rightarrow 1 \text{ to } 0$        $(?1) \rightarrow 0, 1 \text{ or } x \text{ to } 1$

- \* Only one edge specifications are allowed per state entry in a state table.
- \* cover all possible combinations of transitions & levels in the state table for which output has known value.

- \* // State table for positive edge triggered d-ff with clear table

D clear clk : q :  $q^+$ ;

? 1 ? : ? = 0; // clear output

? (10) ? : ? : -; // Hold data at negedge clear

0 0 (01) : ? : 0; // capture data at posedge clk

D char (lk) : q : q';

1 0 (01) : ? : 01; // capture data at posedge clk

0 0 (x1) : ? : 0; // capture data at posedge clk

1 0 (x1) : ? : 1; // capture data at posedge clk

? 0 (1?) : ? : -; // ignore negedge of clk

? 0 (x0) : ? : -; // ignore negedge of clk

? 0 0 : ? : -; // ignore data when clock is steady

? 0 (0x) : ? : -; // ignore clock going to unknown state

endtable

## Compiler Directives

- \* Verilog supports some special keywords for controlling the compilation these keywords are known as compiler directives
- \* Compiler directive begin with "```" [ Back-quot ].
- \* Directive can be declared with the syntax as : 'name of directive'
- \* Compiler directive can be declared anywhere in the verilog code but it's recommended that few of them should be used outside the module
- \* Compiler directives are very useful commands in verilog, they are effective from the point they are declared to the point they are overridden by other directive.
- \* Compiler directive helps in conditional compilation as well as in conditional execution.
- \* Compiler directives apply globally to the design & also can save significant compilation time.
- \* Verilog supports following compiler directives :
  - `include
  - `timescale
  - `define & `undef
  - `ifdly, `else, `endif

→ 'include' :-

- \* Verilog models can be organized into different files & then compiled together as one unit. One of the useful features in this regard is the 'Include' compiler directive.
- \* The compiler assumes that content of included source file appears in place of 'include' compiler directive.
- \* 'include' can be used to include globally used function, tasks & macros without enclosing them inside module & endmodule
- \* Syntax => 'include "file-name.v"' // relative path  
 'include "C:/fields/soc/file-name"' // absolute path  
 module xyz (ip & op);  
 // statements  
 endmodule

→ 'define & 'unify':-

- \* 'define' is the second most used directive in verilog & very useful one. It's just like parameter, this one is used to add a MACRO to your design.
- \* The 'unify' compiler directive lets you remove definitions of text macros created by the 'define' compiler directive.
- \* 'define' can be defined with arguments, so every 'define' call can be followed by actual parameters. The compiler recognizes a 'define' by its name preceded by accent grave (`) character.
- \* Syntax =>

'define SIZE 8

'define xor\_b (x,y)(x+!y)|(!x&y)

// Thus text macros can be used as follow :

module abc();

reg [ 'SIZE -1 : 0 ] data\_out;

c = 'xor\_b(a,b);

'unify SIZE

---

endmodule

→ 'timescale':-

- \* This directive specifies time unit & time precision for simulation of modules that follow it.
- \* The time unit specifies the unit of measurement for times & delays.
- \* The time precision value should be equal or smaller than the time unit value (ps, ns, us).
- \* The time precision specifies how delay values should be rounded during the simulation.
- \* Both time precision & time unit should be made up of an integer number & a character string that represent a unit of time.

- \*\* The valid numbers are : 1, 10 , & 100 . The valid characters are : s, ms, us, ns, ps & fs.
- \* Actual time units & time precision values can be displayed by the "\$printtimescale" system task.
- \* Syntax =>
  - 'timescale time-unit unit / time-precision unit
  - time-unit is the unit of measurement for time values, such as simulation time & delay values.
  - time-precision is the minimum value (least count) for time. All time values are rounded off to the time-precision values.
  - \$printtimescale system task is used to display the time unit & precision for a module.

Ex) 'timescale 1ns/1ps  
 module timescale;  
 integer a;  
 initial begin a=1; #5.1 a=2, #7.23 a=3;  
 #1.514 a=4;  
 end  
 initial \$monitor(a, \$realtime); endmodule

|        | a | \$realtime | \$time |
|--------|---|------------|--------|
| o/p => | 1 | 0          | 0      |
|        | 2 | 5.1        | 5      |
|        | 3 | 12.33      | 12     |
|        | 4 | 13.84      | 14     |

1ns → 1000ps

'timescale 1ns/1ns ✓

'timescale 1ps/1ns ✗

'timescale 100ps/1000ns ✓

'timescale 10ns/110ps ✗

- timescale ins/ins

# 0.3 // 0 ns  $0.3 \times 1\text{ns} \Rightarrow 0.3\text{ns} \rightarrow 0\text{ns}$

# 2.3567 // 2 ns  $2.3567 \times 1\text{ns} \Rightarrow 2.3567\text{ns} \rightarrow 2\text{ns}$

# 0.5 // 1 ns  $0.5 \times 1\text{ns} \Rightarrow 0.5\text{ns} \rightarrow 1\text{ns}$

- timescale ins/ps

# 2.35 // 2.35 ns  $2.35 \times 1\text{ns} \Rightarrow 2.35\text{ns}$

// 2 (\$time)  $2.35 \times 1000 \Rightarrow 2350\text{ps}$  (real)

$2350\text{ps} \Rightarrow 2.35\text{ns}$  (real time)

# 0.3 // 0.3 ns  $0.3 \times 1\text{ns} \Rightarrow 0.3\text{ns} \times 100 \Rightarrow 300\text{ps}$

// 0 (\$time)  $300\text{ps} \Rightarrow 0.3\text{ns}$  (real time)

- \$time

# 0.2  $\Rightarrow$  integer  $\times$  time unit & precision unit (0)

- \$real \$time

# 0.2  $\Rightarrow 0.2 \times$  time unit & precision unit  $\Rightarrow$  integer (200ps)

# 0.7325 // 0.733 ns  $0.7325 \times 1\text{ns} \Rightarrow 732.5\text{ns} \times 1000$

// 1 ns (\$time)  $732.5\text{ps} \Rightarrow 733\text{ps}$

$733 \times 1000 \Rightarrow 0.733$

- % of returns value is the lowest unit mentioned.

- timescale ins/100ps

# 2.3567 // 2.4 ns  $2.3567 \times 1\text{ns} \Rightarrow 2.3567\text{ns}$

// 2 (\$time)  $2.3567 \times 10 \Rightarrow 23.567\text{ps}$

$24\text{ps} \Rightarrow 2.4\text{ns}$   
10aps

→ Conditional Compilations :-

- \* In verilog few compiler directives are available to provide the conditional compilation. They allow the specified part of program text that will be compiled only if a specified condition is true.
- \* Nothing of these compiler directive is allowed
- \* The available compiler directives for conditional compilation are :
- `ifdef      • `else      • `endif
- `ifndef      • `elsif
- \* `ifdef & `ifndef is always closed by a corresponding `endif.

Ex) `define AND

module logic\_gate (input a,b, output c);

`ifdef AND

assign c=a&b;

`elsif OR

or(c,a,b);

`else

assign c=a^b;

`endif

endmodule

• `define upcount

module counter (input clk, output integer count=0);

always @ (posedge clk)

`ifndef upcount

Count <= Count + 1;

`else

Count <= Count - 1;

`endif

endmodule

→ Conditional execution :-

- \* Conditional execution is simply control the execution of code at run time.
- \* \$test \$plusargs & \$value \$plusargs system function is used for conditional execution.

\* Syntax :-

\$test \$plusargs ("string")

\$value \$plusargs ("user\_string", value)

\* \$test \$plusargs :-

- This system function is used when a value for the argument is not required.
- It searches the list of plusargs for a user defined string.
- The function will return non zero value if the supplied plusargs matches its all characters in the provided string, else it will return zero.
- module sample;

initial begin

if (\$test \$plusargs ("HELLO")) \$display ("Hello executed");

if (\$test \$plusargs ("HE")) \$display ("Hi executed");

if (\$test \$plusargs ("HELL")) \$display ("Help executed");

end

endmodule

• Testing of \$test \$plusargs

Vim - novopt sample + HELL

→ He executed

→ Hd executed

### \* \$value \$plusargs:

- This system function also searches the list of plusargs like \$list \$plusargs but it can take the value for a user defined string too.
- The function will return non-zero value if the supplied plusargs matches its all characters in the provided string, else it will return zero.

### modul sample:

```
integer dec; reg [8*6:1] str;
```

```
initial begin
```

```
if ($value $plusargs ("HELLO=%d", dec))
```

```
$display ("Hello executed dec=%d", dec);
```

```
if ($value $plusargs ("HE=%s", str))
```

```
$display ("He executed str=%s", str); end
```

```
endmodule
```

### \* Testing of \$value \$plusargs

Vxim - novpt sample + HELLO=9

→ Hello executed dec = 9

## Timing Checks Tasks

- \* Verilog has certain constraints to perform common timing checks.
- \* Timing checks tasks are for verification of timing properties of designs & for reporting timing violations.  
Ex: setup violation, hold violation etc.
- \* System timing checks only be used in specify blocks.
- \* All timing checks begin with \$ sign they are not categorized as system tasks.
- \* No system task is allowed inside specify block.
- \* Timing check performs the following operations.
  - Records occurrence time of a data or reference event
  - Waits for occurrence of second data or reference event.
  - Compares the elapsed time to the specified limit.
  - Reports violation in design if any.
- \* Timing check tasks are invoked every time critical events occur within given time limits.

| Argument        | Description                                                                                                              | Type                                       |
|-----------------|--------------------------------------------------------------------------------------------------------------------------|--------------------------------------------|
| reference_event | The transition at a control signal that establishes the reference time for tracking timing violations on the data_event. | read-only input that is scalar or vector.  |
| data_event      | The signal change that initiates the timing check & is monitored for violations.                                         | read/write input that is scalar or vector. |
| limit           | A time limit used to detect timing violations on the data_event.                                                         | constant expression or parameter.          |

\* Verilog supports following timing checks:

- \$setup
- \$hold
- \$setuphold
- \$recover
- \$removal
- \$rcrem
- \$width
- \$period
- \$nochange
- \$skew
- \$timeskew
- \$fullskew

\* \$setup - checks setup time violation

- \$setup (data-event, reference-event, limit);
- (time of reference-event) - (time of data event) < limit

- Data event is a data signal

- Reference event is a clock signal

- limit is setup time.

Ex) module setup (din, clk, q);

    input din, clk;

    output q;

    always@ (posedge clk) q <= din;

    • specify

        \$setup (din, posedge clk, tsetup);

        (data1 => q) = delay;

    \$setup (din, posedge clk, tsetup);

    • endspecify

endmodule

\* \$hold - check the hold timing violation

- \$hold (reference-event, data-event, limit);

- (time of data event) - (time of reference event) < limit.

- Data event is data signal, reference event is clock signal

- limit is hold time.

Ex) \$hold (posedge clk, din, thold);

\* \$rsthold - \$rsthold (refence-count, data-count, setup-limit, hold-limit);

Ex) \$uparam tsetup = 6, thold = 7, delay = 10;  
 $(\text{din} \Rightarrow \text{q}) = \text{delay};$

\$rsthold (posedge clk, din, tsetup, thold);

\* \$recovery - checks the violation for control signal like clear, rreset, set etc.

- \$recovery (reference-count, data-count, limit)

- Reference is a clear, rreset or set

- data → clock signal

- limit → recovery time

Ex) module recovery (in1, out1);

input in1;

output out1;

assign out1 = in1 ? 'b1 : 'b2;

specify

\$uparam trcovery = 10;

\$recovery (posedge in1, out1, trcovery);

endspecify

endmodule



\$recovery (posedge clear, posedge clk, 3)

\* \$removal - checks the violations for control signals like clear, reset  
 - \$removal (reference-count, data-count, limit);

ex) module DFF (din, CLK, CLR, dout);

    input din, CLR, CLK;

    output reg dout;

    always@ (posedge CLK, posedge CLR)

        if (CLR) dout <= 0; else dout <= din;

    specify

        \$removal (negedge CLR, posedge CLK, 3);

    endspecify

endmodule



\$removal (posedge clear, posedge CLK, 3);

\* \$recover - recovery & removal

- \$recover (ref-count, data-count, recovery-limit, removal-limit);

\* \$skew - it is the time delta between actual & expected arrival time of a signal.

- it checks that skew is not more than the specified limit w.r.t a signal

Syntax → \$skew (ref-count, data-count, limit);



\* \$period - checks whether reference signal time period less than timing check limit

Syntax  $\Rightarrow \$\text{period}(\text{ref\_event}, \text{limit});$

- ref.event is edge triggered event on a signal

- limit is time period of the signal

$\$ \text{period}(\text{posedge clk}, 10)$



\* \$width - checks whether reference signal has width less than the specified check limit.

Syntax  $\Rightarrow (\text{or } \$\text{width}(\text{reference\_event}, \text{limit}))$

- ref.event is edge triggered event on a signal

- limit is pulse width value of the signal

$\$width(\text{posedge clk}, 5);$



## Random Number Generator

- \* There are 3 approaches for verification
  - Directed verification ( $a=0; b=0; c=0; \dots$ )
  - Randomized Verification (\$random)
  - Constraint randomized verification ( $i \in \$random \% 4$ )
- \* Directed verification is a time consuming process
- \* Randomized verification will generate random value & we can't specify
- \* Therefore constraint randomized verification is widely used.
- \* Random numbers are used to provide random test inputs which helps in finding hidden bugs in the design.
- \* "\$random" is system task used to generate random numbers.
- \* Syntax : `$random [(seed_value)]`;
  - seed\_value is used to ensure that same random number sequence is generated each time it runs.
  - seed parameter can be reg, integer or time
- \* \$random returns 32bit signed integer

Ex) module random\_gen;

```

    integer seq1, seq2, seq3, seq4, seq5;
    always begin
        seq1 = $random;           // Generate a 32bit signed integer
        seq2 = $unsigned($random); // Generate unsigned integer
        seq3 = $random % 50;      // Generate signed integer in range -49 to 49
        seq4 = $unsigned($random) % 100; // Generate unsigned (0-99)
        seq5 = {$random[3] % 35};   // 3-bit unsigned (0-34)
        #10;
    end
endmodule

```

- To generate N(0, 5 to 15)

$$a = 5 + \{ \$random \% 11 \};$$

- -20 to 0

$$a = -20 + \{ \$random \% 21 \};$$

- -20 to 5

$$a = -20 + \{ \$random \% 26 \};$$

## File Operations

- \* This is one of the good features that was added to Verilog 2001.
  - \* Using this feature external files can be called in test-bench for reading in routers & storing results of simulations for further analysis.
  - \* The Verilog has system tasks or functions that can open a file, save the values to file, read values from file & load values to other variables, close the file.
  - \* The file operations mainly provide following features:
    - Reading characters from file from a fixed location
    - Reading lines from a file
    - Writing lines into a file.
  
  - \* System tasks:
 

|                                     |                                                                          |
|-------------------------------------|--------------------------------------------------------------------------|
| • \$fopen(file-name);               | • \$fmonitor(arguments);                                                 |
| • \$fclose(file-name);              | • \$readmemh("file", memory-identifier {begin address [, end-address]}); |
| • \$fdisplay(arguments, file-name); | • \$readmemb("file", memory-identifier {begin address [, end-address]}); |
| • \$fwrite(arguments);              | • \$readmemh("file", memory-identifier {begin address [, end-address]}); |
| • \$fread(arguments);               |                                                                          |

$\rightarrow \$open^*$

- \* \$open system task is used to open a file
  - \* Syntax  $\Rightarrow$  file handle = \$fopen("name of the file.txt", "mode");
  - \* \$fopen returns a 32 bit value called multichannel descriptor.
  - \* Only one bit of a multichannel descriptor is set to 1.
  - \* Each bit of multichannel descriptor is used to represent different channel, except Bit 0 (reserved).
  - \* Standard output has a multichannel descriptor with Bit 0 set to 1 (default condition).
  - \* Standard output is also called as channel 0, which is open by default.
  - \* Each successive \$fopen call opens a new channel & returns 32-bit multichannel descriptor with next MSB bit set to 1.
  - \* Multichannel descriptor offer advantage of writing multiple files at once time.

### \* File Opening modes:-

- "r" - open the file in read mode.
- "w" - create the file for writing or over write the existing file.
- "a" - Open the file for appending.
- "rt" - Open for Both reading & writing.
- "w+" - Truncate or create the file.
- "a+" - Append or Create new file for updating at EOF.

`end module file test;`

Initial begin

```
fp = $fopen ("test file.txt","r");
$fclose (fp);
```

• open the file in read mode, store the file pointer in fp.

end endmodule

### → Writing to file :-

- \$fdisplay - Similar function as \$display, it writes to a file.
- \$fmonitor - Similar function as \$monitor, it writes to a file.
- \$fwrite - Similar function as \$write, it writes to a file.
- \$fstrwr - Similar function as \$strwr, it writes to a file.

\* All the above mentioned functions by default print the value on decimal, but they have three more versions to print the values in different radix.

- \$displayb - Binary
- \$displayo - Octal
- \$displayh - Hexadecimal

\* Syntax :- `$fmonitor (file descriptor , p1,p2 ... pn);`

- file descriptor specifies which file are to be written.
- p1, p2, ..., pn can be variables, constants or strings.

- \$f monitor (file1, "Value of a=%d + b=%d", a, b);
- \$f monitor ((file2 | file2), "Value of c=%d + b=%d", c, b); (Both files)
- field1 = file2 | 1; \$f display (field1, "Hello");
- field2 = file2 | file3; \$f display (field2, "world");

Ex) module fil(a,b,c);

    input a,b;

    output c;

    integer f1;

    assign c=a+b;

    initial begin

        f1=\$fopen ("file1.txt");

        \$fmonitor (f1,a,b,c);

    end endmodule

• module hhh;

    integer file1,file2,file3;

    integer field1,field2,field3,field4,field5

    initial begin

        file1=\$fopen ("file1.txt");

        // file1 = 32'h0000\_0002 (8bit 1st)

        file2=\$fopen ("file2.txt");

        // file2 = 32'h0000\_0004 (8bit 2nd)

        file3=\$fopen ("file3.txt");

        // file3 = 32'h0000\_0008 (8bit 3rd)

    end

    initial begin

        field1= file1 | 1; \$display (field1,"Hello");

        // 211 => 32'h0000\_0003

        field2= file2 | file3; \$display (field2,"World");

        // writes to file1.txt & Transcript

        field3= 8; \$display (field3,"or");

        // 4 => 32'h0000\_0004

        field4= file1 | file2 | 1; \$display (field4,"you");

        // writes to file3

        field5= file3 + 1; \$display (field5,"Hello again");

        // 21411 => 32'h0000\_0007

    end endmodule

        // writes to file1.txt, file2.txt  
& transcript.

## ⇒ Memory Initialization :-

- \* Verilog provides a system task to initialize memory from a file.
- \* "\$readmem" & "\$readmemh" are system tasks used to initialize memory.
- \* Syntax =>
  - \$readmem("file-name", memory\_name);
  - \$readmemh("filename", memory\_name, start\_address);
  - \$readmemh("filename", memory\_name, start\_addr, end\_address);
- \* \$readmem & \$readmemh are used if the values present in file are in binary or hexadimal format respectively.
- \* In initialization file, data is separated by whitespace.
- \* Data can contain X & Z.
- \* Optionally address can be specified with @ address. Here address has to be in hexadimal format.
- \* If there are multiple entries for same address, the last entry will be considered.
- \* If there is conflict between (start\_addr, end\_addr) & address provided in initialization file task will report warning.

### • module example:-

```
integer i;
        Data          Address
begin [7:0] mem [0:7];
    end
```

```
initial $readmem ("file.txt", mem);
```

```
initial for (i=0; i<=7; i=i+1)
```

```
$display ("Value at location %d is =%b", i, mem[i]);
```

```
endmodule
```

// whitespace are used to separate data

// Data specified in binary (\$readmem)

1001-0110 11L-0

1011-0100 11L-1

1010-0001 11L-2

1111-0010 11L-3

0111-1110 11L-4

1101-1110 11L-5

1001-1101 11L-6

0101-0111 11L-7

## VCD System Tasks

- \* The VCD (Value chang. Dump) file format is specified in the IEEE 1364 standard.
- \* The VCD file is an ASCII file containing header information, variable definitions, & variable value changes.
- \* VCD is commonly used in Verilog designs, & is controlled by VCD system task call in the Verilog source code.
- \* The VCD files store information about value changes during simulation for nets & registers.
- \* The value change dumping function is more efficient than the \$monitor task both in performance & in storage space.
- \* The files saved in the VCD format contain several sections that describe:
  - The file creation date & time.
  - Version number, \$timescale directive & signals specification
  - \$dumpvars task which defines which variables to dump into a file
- \* Following syntaxes are used for VCD system tasks :

|                                        |               |
|----------------------------------------|---------------|
| - \$dumpfile (filename)                | - \$dumpon    |
| - \$dumpvars                           | - \$dumpall   |
| - \$dumpvars (lure, list of variables) | - \$dumplimit |
| - \$dumpobj                            | - \$dumpflush |
- \* \$dumpfile - Used to create a VCD file.
- \* \$dumpvars - Used to specify variables to be dumped.  
lure → how many levels of hierarchy of modules has to be dumped to the VCD file.
  - 1 → will dump all variables in current specified module, not module instances inside current specified module.
  - 0 → will dump all variables in current specified module & modules instantiated inside current specified module.
  - 2 → will dump current module & one level up.



- \$dumpvars(1, TB); // A & B
  - \$dumpvars(1, TB.u3); // C & D
  - \$dumpvars(2, TB); // A B C D
  - \$dumpvars(0, TB); // All variables
  - \$dumpvars(1, TB.u3, A); // A, C & D
  - \$dumpvars(1, TB.u3.u1, TB.u3); // E1 & C,D

→ Keywords inside VC D file :-

- \* \$dumpall - is used to write current value of all selected variables.
  - \* \$dumpon \$dumpoff - is used to resume or stop dumping process.  
Syntax =>
    - \$dumpall ; // write current values of all variables
    - \$dumpoff ; // stop dumping
    - \$dumpon ; // resume dumping
  - \* \$dumpflush - is used to empty VCD Buffer & ensure all data in buffer is written to vcd file.
    - It is used to update the dump file while simulation is running.
    - \$dumpflush ;
  - \* \$comment - Represents comment inside a vcd file.
    - \$comment // This is single line comment \$end
  - \* \$timescale - Specifies time scale which was used during simulation
    - \$timescale time-number time-unit \$end
  - \* \$date - specifies the date on which vcd file was created
    - \$date date-in-text format \$end.
  - \* \$upscope - indicates change of scope to the next higher level in design hierarchy.
    - \$upscope \$end

→ QuartusSim VCD Features :-

- \* creating VCD file without using verilog system tasks:
  - vsim -nowinopt 1stbench // Start TB simulation
  - vcd file vcd-filename // Create VCD file
  - vcd add var1 var2 // Add variables to be dumped
  - add wave \* // Add wave window
  - run time-value // Run simulation
- \* Converting a VCD file to a waveform file
  - vcd2wlf vcd-filename.wlf-filename.wlf

## Procedural Continuous Assignment

- \* A procedural continuous assignment assigns a value to a register continuously for specified duration.
- \* A procedural continuous assignment overrides any other assignment.
- \* A procedural continuous assignment is not the same as a continuous assignment.
- \* Procedural continuous assignment are declared inside procedural blocks.
- \* Used only for simulation.
- \* Verilog provides two types of procedural continuous assignments:
  - Assign & Deassign
  - Force & Release

→ Assign & Deassign :-

- \* Assign & deassign works according to their name i.e assign simply overrides regular procedural assignments & deassign re-enables regular procedural assignments.
- \*\* It works only for register data type.
- \* After deassign last value remains on the register until a new procedural assignment changes it.

```
Ex) module dff(input din, output q);
    always@(posedge clk)
        q <= din;
    always @ (q)
        if (q == 1'b0) assign q = 1'b1;
        else deassign q; // Next assignment at posedge clk.
    endmodule
```

## → Force & Release :-

- \* The keyword "force" & "release" have the same effect as "assign" & "deassign" i.e force simply overrides regular procedural assignments & release re-enables regular procedural assignments.
- \* force & release can be used for nets, registers, bit - or part slot of a net (not register), or a concatenation.
- \* Assignments in priority order
  - force
  - assign (procedural continuous)
  - Procedural / Continuous assignments

Ex) Main module

```
module dff (din, clk, q);
  input din, clk;
  output reg q;
  always @ (posedge clk)
    q <- din;
endmodule
```

Test bench

```
module dff_tb;
  reg din, clk=1;
  wire q;
  dff u0 (din, clk, q);
  initial begin
    force u0.q=0;
    #10 release u0.q;
    forever #10 din=$random;
  end endmodule
```

## Modelling delays

- \* Functional verification of hardware only tells whether the functionality of design is correct or not, but in real hardware, logic elements & path have delays associated with them.
- \* Therefore, it is also required that design meets timing requirements.
- \* Verilog provides the following modelling delays:
  - Distributed delays - are specified for individual logic elements
  - dumped delays - are specified on per module basis
  - Pin to Pin delays - are assigned individually to paths from each input to output.

→ Distributed Delays:-



module dist\_delay (g,a,b,c,d);  
 Output y;  
 input a,b,c,d;  
 wire e,f;

// delay is applied to each gate  
 and #4 a1(e,a,b);  
 and #3 a2(f,c,d);  
 or #6 a3(y,e,f);  
 endmodule

→ Dumped delays:- They are specified as a single delay on the output gate of the module.

- \* The dumped delay takes into consideration the maximum delay from an input to output.



and a1(e,a,b);  
 and a2(f,c,d);  
 or #10 a3(y,e,f);  
 endmodule

→ Difference between Distributed & lumped delays:-

Distributed delays

- Difficult to implement

- Delays are accurate

- All path may or may not have same amount of delay

Lumped delays

- Not easy to implement

- Delays are less accurate

- All path have same amount of delay.

→ Pin to Pin delays:-

- \* In pin-to-pin delay model, delays are assigned individually to paths from each input to output.

- \* Although pin-to-pin delays may look very detailed, but they are easier for designers, as it requires no knowledge of internal model.

- \* The model may be coded in Behavioral statements, datflow, gates or mixed.



- \* These delays are also known as path delays

path a-e-y, delay = 9

path b-e-y, delay = 9

path c-f-y, delay = 11

path d-f-y, delay = 11

- \* For bigger circuits this delay model is easier to implement as compare to distributed delay model.

- \* Designer needs to know the I/O pins of the module rather than the internals of the module so easier to model, even though it is very difficult.

- \* In Verilog, path delays are specified using "Specify blocks".

→ Specify Block:-

- \* The specify block describes paths across the module & assigns delays to those paths.

- \* It performs timing checks like setup & hold times.

- \* The specify block is declared inside the module declaration & bounded by the keyword specify & endspecify.

\* Syntax  $\Rightarrow$  module `xyz(a,b,c);`

    input a,b; output c;

    always begin

        end

    begin

        endsign

    endmodule

\* There are two ways to describe pin to pin delays :

- Parallel connection

- Full connection

\* Parallel connection :-

- Syntax : (source pin  $\Rightarrow$  destination pin) = delay value;

- Each set of source connects to corresponding set of destination.

- Parallel connection can be created between source & destination only if they are of same size.

- They only connect one source  $\to$  one destination.



module parallel (a,b,c,d,y);

    input a,b,c,d; output y, wire e,f;

    and u1 (e,a,b); and u2 (f,c,d); or (y,e,f);

    begin

        (a  $\Rightarrow$  y) = 9; (b  $\Rightarrow$  y) = 9; (c  $\Rightarrow$  y) = 11;

        (d  $\Rightarrow$  y) = 11;

    endbegin

endmodule

### \* Full Connection :-

- Syntax : (source\\_pin  $\star\gt;$  destination\\_pin) = delay\\_value;
- Each bit of source connects to every bit of destination
- Source & destination need not be of same size
- They can be used to describe path between :
  - a vector & a scalar
  - vector of different sizes
  - multiple sources or multiple destinations

Ex:-



Specify

$$(x, y \star\gt; z) = 7;$$

$$\text{if } (x \Rightarrow z) = 7; (y \Rightarrow z) = 7;$$

$$(A \star\gt; B) = 3;$$

$$\text{if } (A[0] \Rightarrow B) = 3; (A[1] \Rightarrow B) = 3;$$

endspecify endmodule.

### → Specparam :-

- "specparam" keyword is used to declare special parameters inside specify blocks.

They are used to specify pin to pin delay values & then these parameters are used instead of hardcoded delay values.

These parameters remain local to a specify block

Syntax  $\Rightarrow$  specparam parameter-name = delay-value;  
 module design (input a, b, c, output out);  
 assign out = (a & b) | c;

Specify

$$\text{specparam aout} = 2, bout = 4, cout = 3;$$

$$(a \Rightarrow out) = aout; (b \Rightarrow out) = bout; (c \Rightarrow out) = cout;$$

endspecify endmodule

\* Min, Max & Typical delay can also be specified for pin to pin delays.

\* Syntax  $\Rightarrow$  Min : Typical : Max

Ex  $\Rightarrow$

```
specparam min = 1:2:3, fall = 2:4:5, turnoff = 3:5:7;
(a=>b) = (min, fall, turnoff);
```

- module design (input a,b, output out);

assign out = a + b;

specify

$\text{if } (a) (a \Rightarrow \text{out}) = 4;$

$\text{if } (a \& b) (a \Rightarrow \text{out}) = 4;$

$\text{if } (\sim a) (a \Rightarrow \text{out}) = 3;$

$\text{if } (\sim a \& b) (a \Rightarrow \text{out}) = 3;$

endspecify endmodule

- module design (input a,b,c, output out);

assign out = a ^ b;

specify

$\text{if } \{a,b\} == 2'b00 (c \Rightarrow \text{out}) = 4;$

$\text{if } \{a,b\} != 2'b00 (c \Rightarrow \text{out}) = 1;$

endspecify endmodule

- module dff (input clk,din, output reg dout);

always @ (posedges clk)

dout <= din;

specify

(posedge clk => (dout : din)) = (1,2);

endspecify

endmodule

## Finite State Machine

→ 10.1 Overlapping memory :-



\* modularity memory reg\_101\_OV (in\_rq, clk, rst, dit\_out);

input in\_rq, clk, rst;

output reg dit\_out;

reg [1:0] ps, ns; // Declaration for state variables

// State Encoding

parameter idle = 2'b00;

parameter S1 = 2'b01;

parameter S10 = 2'b10;

// Implement State Register

always @ (posedge clk or negedge rst) begin

if (!rst)

ps <= idle;

else ps <= ns;

end

// always Block for NSD and OD

always @ (in\_rq, ps) begin

case (ps)

idle : if (in\_rq) begin

ns = S1; dit\_out = 0; end

else begin

ns = idle; dit\_out = 0; end

S1 : if (in\_rq) begin

ns = S1; dit\_out = 0; end

else begin

ns = S10; dit\_out = 0; end

```

S10 : if(in_rq) begin
    ns=51; dt_out=1; end
else begin
    ns=0; dt_out=0; end
default: begin
    ns=0; dt_out=0; end
endcase
end
endmodule
    
```

### \* //TESTBENCH

```

module rq_101_maly_ov_test;
    reg in_rq, clk, rst;
    wire dt_out;

    maly_rq_101_ov dut(in_rq, clk, rst, dt_out);

    initial begin
        clk=0; rst=0;
        #12 rst=1;
        in_rq=1;
        #10 in_rq=0;
        #10 in_rq=1;
        #10 in_rq=0;
        #10 in_rq=1;
        #20 $finish;
    end
    always #5 clk=~clk;
endmodule
    
```

→ 101 overlapping states :-



\* module moore - seq 101DV (in\_sq, clk, rst, dt\_out);

input in\_sq, clk, rst;

Output seq dt\_out;

seq [1:0] ps, ns;

parameter idle = 2'b00;

parameter S1 = 2'b01;

parameter S10 = 2'b10;

parameter S101 = 2'b11;

always @ (posedge clk) begin

if (!rst) ps <= idle;

else ps <= ns; end

always @ (in\_sq, ps) begin

case (ps)

idle : if (in\_sq) begin

ns = S1; dt\_out = 0; end

else begin

ns = idle; dt\_out = 0; end

S1 : if (in\_sq) begin

ns = S10; dt\_out = 0; end

else begin

ns = S10; dt\_out = 0; end

S10 : if (in\_sq) begin

ns = S101; dt\_out = 0; end

else begin

ns = idle; dt\_out = 0; end

S101 : if (in\_sq) begin

ns = S1; dt\_out = 1; end

else begin

ns = S10; dt\_out = 1; end

default : begin

ns = idle; dt\_out = 0; end

endmodule