# Assignment - 2 Verilog Modelling Techniques

Department of Electronics and Communication Engineering Indian Institute of Technology, Roorkee

ECN 104 Digital Logic Design

## 1 Introduction

Verilog supports wide variety of modelling techniques. Different modelling techniques allows hardware description at different level of abstraction, starting from switch level modelling (PMOS/NMOS) and all the way up to behavioural modelling (algorithmic description). Each one of them have their own benefits and use cases. In this assignment we will discuss three of them, namely Gate-level modelling, structural modelling and behavioural modelling.

## 1.1 Gate-level modelling

Gate-level modelling in Verilog is used to describe a circuit only using logic gates. This approach is used to describe critical parts of a design, like adders and multipliers. Using gate-level implementation allows greater control over the design than other techniques. Gate-level modelling is only used for small scale design, due to its complexity other modelling techniques are commonly used to abstract gate level implementation.

### 1.1.1 Gate Primitives in Verilog

Verilog support following gates:

ANDNANDORNORNOT

Gates in Verilog are available as primitives and can be instantiated similar to modules, Listing 1 is an example gate level circuit.

# Listing 1: Example module using Gate-level modelling

```
/* Example using gate-level modelling, gate is a built-in
 1
    * primitive in Verilog. Gates are instantiated in a way
 3
    * similar to modules. Gates can be of single input or
 4
   * multiple input
5
   */
 6
   module gateLevelExample(input1, input2, input3, result);
 7
       input input1, input2, input3;
8
       output [8:0] result;
 9
10
       // Single input gates
                                     // = ~input1
11
       not g0(result[0], input1);
12
       // Two input gates
13
       and g1(result[1], input1, input2);
14
                                              // = input1 & input2
       nand g2(result[2], input1, input2);
                                              // = ~(input1 & input2)
15
                                              // = input1 | input2

// = ~(input1 | input2 )

// = input1 ^ input2
       or g3(result[3], input1, input2);
16
       nor g4(result[4], input1, input2);
17
       xor g5(result[5], input1, input2);
18
                                              // = \sim (input1 \wedge input2)
       xnor g6(result[6], input1, input2);
19
20
```

```
// Gates with more than two input
xnor g7(result[7], input1, input2, input3);
and g8(result[8], input1, input2, input3, input1);
endmodule // gateLevelExample
```

Verilog also supports instantiating gates without a instance name demonstrated in Listing 2.

```
Listing 2: Instantiating unnamed gates
```

```
/*
1
   * Gates in Verilog can be instantiated without a name,
2
   * such instantiation in Verilog is legal.
3
4
5
   module unnamedGate(input1, input2, result);
6
      input input1, input2;
7
      output result;
8
9
      and(result, input1, input2); // Unnamed gate
10 endmodule // unnamedGate
```

#### 1.1.2 Delay specification

All the circuits we have studied so far have no delay associated with them, these are called 0-delay circuits. Real circuits however, always have a delay between their input and output. Verilog allows modelling of delays at various level of abstraction using delay statements.

Syntax for specifying a delay is:

To specify unit of delay we'll write a compiler directive, this is typically written at the begining of the description.

```
`timescale 1ns/1ps
```

Here 1ns is the time unit while 1ps is the time resolution. Which will be explained in later assignment. A complete example using delay statements is given in Listing 3.

Listing 3: Example usage of delays statement to specify propagation delay of logic gates.

```
/* Compiler directive to specify time unit, which will be
1
    * used for assigning propogation delays of logic gates.
3
   `timescale 1ns/1ps
   module delayExample(input1, input2, result);
6
7
      input input1, input2;
8
      output result;
9
10
      // Specify a NAND gate having propogation delay of 2ns
11
      nand #(2) nd1(result, input1, input2);
12 endmodule // delayExample
```

## 1.2 Data-flow Modelling

Data flow modelling is a higher level of abstraction. Describing a circuit using data-flow modelling does not require knowledge of gates level circuit, thus it is easier than gate-level modelling when description of large scale circuits are written. All the examples from Assignment 1 used data flow modelling.

#### 1.2.1 Continuous Assignment

Continuous assignment in Verilog are used for data flow modelling, these assignment starts with assign keyword. Continuous assignment drives value into a net (wire). Following example describes use of continuous assignment:

**Note:** Verilog is concurrent language unlike programming languages such as C,C++ or Java. All the continuous assignments are evaluated at the same time.

#### Listing 4: Example usage of continuous assignment. 1 /\* 2 \* Following module uses continuous assignement to model 3 \* an AND gate 4 5 module continuousAssignment(input1, input2, result); 6 input input1, input2; 7 output result; 8 /\* Use continuous assignment to set result 9 \* This is equalent to: 10 and (result, input1, input2); 11 12 assign result = input1 & input2; 13 endmodule // continuousAssignment 14

Continuous assignment in Verilog can also be done implicitly, which assigning value on declaration of a net (wire). Implicit delcaration of Verilog is down as follows:

```
wire new_wire = input1 & input2;
```

#### 1.2.2 Assignment Delays

Similar to gate-level modelling, Verilog allows specifying delays in assignment to model real circuits. Assignment delay specify the delay between the change of LHS and RHS of a continuous assignment. Listing 5 shows example usage of assignment delay while Fig. 1 shows simulation result of Listing 5.

```
Listing 5: Using assignment delay in Verilog.
    `timescale 1ns/1ps
1
2
3
   module assignmentDelay(input1, input2, result);
4
      input input1, input2;
5
      output result;
6
       // Defines a circuit which will have output value
7
       // input1 | input2 15ns after changing input
8
       assign #15 result = input1 | input2;
9
10
   endmodule // assignmentDelay
```



Figure 1: Simulation result for Listing 5, output changes 15ns after the change in input.

## 2 Behavioral Modelling

Behavioral modelling is a higher level of modelling where circuit description is written as its behaviour, this algorithmic representation of a circuit abstracts the details of gate-level and data flow modelling. Behavioral modelling resembles more to programming languages such as C than circuit description.

## 2.1 reg Element

reg element is used to represent abstract storage device in Verilog. regs can be used to store information (single bit or of arbitrary length using vector). We'll get back to usage of reg and will differentiate it with wire after studying procedural blocks.

#### 2.2 Procedural Blocks

Earlier we read about continuous assignment which allows us to drive value to a net whenever the driver changes, this kind of assignment allows only to describe combinational circuit. To describe sequential circuits Verilog provides procedural blocks, these blocks are used to drive values to net (reg) only inf the condition is met.

#### 2.3 initial Block

Initial blocks in Verilog are used to specify initial values of all the storage elements. When simulation starts simulator doesn't know what values are to be assigned to storage elements. Initial block begins with initial begin and ends with end. Initial blocks gets executed only once when the simulation is started.

**Note:** Only a reg can be assigned values inside an initial block, this is because unlike wire which are used for connection reg stores information and this information is unknown to the simulator at t=0. Structure of an initial block is given in Listing 6.

```
Listing 6: Structure of an initial block.

initial begin

// Specify initial condition

end

...
```

#### 2.4 always@ block

always@ block in Verilog are used for describing a event which should happen only under certain conditions, such as change in value of one of the elements.

Basic structure of an always@ block is given in Listing 7.

```
Listing 7: Structure of an always@ block.

1 ...
2 always @( ...condition... ) begin
3 ...
4 // Description of conditional event
5 ...
6 end
7 ...
```

Listing 8 shows a module which changes output only one the positive edge of input clk.

```
Listing 8: Synchronous logic which changes value of result only at the positive edge of clk.

1 /*
2 Synchronous module which changes output only on positive
3 edge of clk.
4 */
```

```
module alwaysExample(clk, input1, input2, result);
6
       input clk, input1, input2;
7
      output result;
8
9
      // Declare output as reg, this will be explained later
10
      reg result;
11
      // Here condition is *positive edge of clk*, this implies
12
      // following event only takes place when a positive edge
13
14
      // of clk arrives
       always @(posedge clk) begin
15
           // Assign value to reg result using procedural assignment
16
17
           result = input1 & input2;
18
      end
19
   endmodule // alwaysExample
```

## 2.5 Blocking and Non-Blocking Assignment

#### 2.5.1 Blocking Assignments

Blocking assingments are assignments which block the simulation while their value is being calculated. This means that the execution flow will stop at this statement until it is executed. Blocking assignment in Verilog is done using =. Listing 9 demonstrates the functioning of blocking assignment.

```
Listing 9: Swapping bytes using blocking assignment.
   // Following example is for swapping bytes
   // input is of size 16 bits (2 bytes)
   // This example does not work due to the use of blocking
 4
   // assignment.
 5
   // This example is taken from:
6
7
   // http://www.sutherland-hdl.com/papers/1996-CUG-presentation_nonblocking_assigns.pdf
8
9
   module swapBytes;
10
      reg [15:0] temp;
11
12
13
14
      // Can you guess the reason why this block won't swap
15
      // bytes?
16
       always @(...some_condition...) begin
17
           temp[15:8] = temp[7:0];
18
           temp[7:0] = temp[15:8];
19
   endmodule // swapBytes
```

In the above example (Listing 9) statement 18 doesn't gets executed until statement 17 is executed, this is due to the use of blocking assignment.

#### 2.5.2 Non-blocking Assignments

In Listing 9 we saw that blocking assignments cannot be used to swap bytes, this is where non-blocking assignments will come to use. Non-blocking assignments are evaluated in two steps first all the RHS values are calculated at the begining of the procedural block, and then the value is assigned to LHS when the execution reaches particular statement.

Listing 10 shows how non-blocking assignments can be used for swapping bytes.

```
Listing 10: Swapping bytes using non-blocking assignment it works!

1 // Following example is for swapping bytes
2 // using non-blocking assignment, this module
3 // unlike last example uses non-blocking assignment
4 // this allows swapping bytes correctly.
5
```

```
6 // This example is taken from:
7
   // http://www.sutherland-hdl.com/papers/1996-CUG-presentation_nonblocking_assigns.pdf
8
9
   module swapBytes;
10
      reg [15:0] temp;
11
12
13
14
      // Can you guess the reason why this block won't swap
15
      // bytes?
      always @(...some_condition...) begin
16
17
          temp[15:8] \le temp[7:0];
           temp[7:0] \le temp[15:8];
18
19
20 endmodule // swapBytes
```

## 3 Differences between wire and reg and where to use what

## 3.1 Legal use of wire

Wires in Verilog are used to connect two elements, they can be assigned a value or a value can be read from them. However they cannot store it, you'll have to drive them with values (constant, other wires or regs).

- wires are allowed only in continuous assignments (page 3).
- A wire cannot be assigned a value inside a procedural block.
- wire can be used to assign a value to a reg or a wire.

## 3.2 Legal use of reg

regs in Verilog are storage elements, they however do not represent physical registers. Once synthesized they can be represented by a physical register, RAM or ROM.

- regs cannot be assigned a value using continuous assigment.
- A wire can only be assigned a value in a procedural block.
- wire can be used to assign a value to a reg or a wire.

## 3.3 Places where both wire and reg are allowed

#### References

- http://inst.eecs.berkeley.edu/~cs150/fa08/Documents/Always.pdf
- · Aenean in sem ac leo mollis blandit.