

# ECEN 449 – Microprocessor System Design



Verilog

## Objectives of this Lecture Unit

- Get a feel for the basics of Verilog
  - The focus of this unit will be along two separate but equally relevant axes
    - We will cover the semantics of Verilog and different modeling styles
    - Also we will cover syntax issues. For this portion, additional resources are also provided on the website.
  - In general Verilog is quite rich, and therefore, there are many ways to achieve the same design goal
    - We will focus on the syntax that is most common, especially from a synthesizability point of view.

Syntax  
(rules to construct correct sentences)  
semantics  
concepts conveying

# Hardware Description Languages (HDLs)

- What is a HDL, why is it useful
- The Verilog HDL
- Modelling a simple circuit in Verilog
  - Gate level (structural)
  - Dataflow (continuous)
  - Procedural (behavioral)
  - Synthesizable Verilog
- Testbenches
- Syntax coverage



## modeling styles (semantics)

③

- synthesizability



# Hardware Description Language (HDLs)

- A HDL is a programming language which is tuned to describe hardware
- HDLs allow us to design and simulate a design at a higher level of abstraction
  - Result = higher designer productivity
- HDLs also have accompanying synthesis tools which allow the designer to obtain an implementation from HDL code.
  - Further improvement in designer productivity
- FPGA based design flows use HDLs heavily!

kloc - s/w

## Common HDLs

- There are mainly two HDLs in use today
  - Verilog HDL
  - VHDL
- VHDL is the somewhat more common
  - Standard developed by US DoD
  - VHDL = (Very High Speed Integrated Circuit) HDL
- We choose Verilog for this class because
  - It is easier to use and teach
  - Resembles “C” and hence easier to learn.
- Which one is “better”?
  - This is the topic of much debate

“RTL  
register  
transfer  
level”

# Verilog HDL

- Verilog constructs are use defined *keywords*
  - Examples: **and**, **or**, **wire**, **input**, **output**
- One important construct is the **module**
  - Modules have inputs and outputs
  - Modules can be built up of Verilog primitives or of user defined submodules.



## A Structural Design - XOR

```
module xor_gate (out, a, b);
    input a, b;
    output out;
    wire abar, bbar, t1, t2;
```

### Built-in gates

```
not invA (abar, a);
not invB (bbar, b);
and and1 (t1, a, bbar);
and and2 (t2, b, abar);
or or1 (out, t1, t2);
```

*modules*

**endmodule**

### Instance name

- Composition of primitive gates to form more complex module



## Another Simple Circuit (in Structural Verilog)



Fig. 3-37 Circuit to Demonstrate HDL

```
module smpl_circuit(A,B,C,x,y);
    input A,B,C;
    output x,y;// blabla ..
    wire e;
    and g1(e,A,B);
    not g2(y, C);
    or  g3(x,e,y);
endmodule
```

*no ;*

## Structural Verilog

- Just specifies primitive gates and wires
  - In other words, the structure of a logical netlist
- Useful if you know exactly what logic you want to generate
  - Not useful for large designs, where we want to specify the design at a higher level of abstraction
    - It is crucial to design at a higher level of abstraction in this case, since structural design would be tedious and error prone
    - In such a case, we will describe the circuit at a high level of abstraction, and let the CAD tools realize the detailed design (by performing the steps of synthesis, mapping, placement+routing, and generation of the netlist (in an FPGA, this is the bitgen file))
    - In special cases, delay or area-critical sub-blocks can be designed in structural manner, while the rest of the logic could be at a higher level of abstraction (typically described in the behavioral fashion).

## Simple Circuit – Comments

- The module starts with module keyword and finishes with endmodule.
- Internal signals are named with wire.
- Comments follow //
- input and output are ports. These are placed at the start of the module definition.
- Each statement ends with a semicolon, except endmodule.

## Adding Delays

- To simulate a circuit's real world behaviour it is important that propagation delays are included.
- The units of time for the simulation can be specified with timescale.
  - Default is 1ns with precision of 100ps
- Component delays are specified as #(delay)  - BUT REMEMBER – these delays will NOT synthesize.
  - Useful only for simulation and verification of your design.

## Simple Circuit with Delay

```
module circuit_with_delay(A,B,C,x,y);
    input A,B,C;
    output x,y;
    wire e;
    and #(30) g1(e,A,B);
    or #(20) g3(x,e,y);
    not #(10) g2(y,C);
endmodule
```



Fig. 3-37 Circuit to Demonstrate HDL

| Time<br>(ns) | Input<br>A B C | Output<br>y e x |
|--------------|----------------|-----------------|
| 0            | 0 0 0          | 1 0 1           |
| 0            | 1 1 1          | 1 0 1           |
| 10           | 1 1 1          | 0 0 1           |
| 20           | 1 1 1          | 0 0 1           |
| 30           | 1 1 1          | 0 1 0           |
| 40           | 1 1 1          | 0 1 0           |
| 50           | 1 1 1          | 0 1 1           |

(12)

## Event driven timing simulation

Given • a logic ckt with delays  
• input events

Find logic value at each node  
as a function of time

---

event : a change in logic  
value of a node  
( $0 \rightarrow 1$ , or  $1 \rightarrow 0$ )

---

(12.1)

Steps

- ✓ ① find logic state of all circuit nodes @  $t=0$
- ✓ ② Create event table
- ✓ ③ populate with input events
- ✓ ④ process events in time order, until no new events



event table

| node | t  | node @<br>$t^-$ | node @<br>$t^+$ | 12.2 |
|------|----|-----------------|-----------------|------|
| ✓ A  | 0  | 0               | 1               |      |
| ✓ B  | 0  | 0               | 1               |      |
| ✓ C  | 0  | 0               | 1               |      |
| ✓ e  | 30 | 0               | X 1             |      |
| ✓ y  | 10 | 1               | X 0             |      |
| ✓ x  | 30 | 1               | X 0             |      |
| ✓ z  | 50 | 0               | 1               |      |

12.3

$$y(10^+) = \overbrace{c(10^+ - 10)} = \\ = \overbrace{c(0^+)}$$

$$e(30^+) = a(30^+ - 30) \text{ AND} \\ b(30^+ - 30)$$

$$e(30^+) = a(0^+) \text{ AND} b(0^+) \\ = 1 \text{ AND} 1 \\ = 1$$

$$\underline{x(t) = e(t-20) + y(t-20)}$$

$$x(30^+) = e(10^+) + \underline{y(10^+)}$$

$$\underline{x(50^+) = e(30^+) + \underline{y(30^+)}}$$

12.4

## Editorial comments

- minimum amount of work (skipped times when no changes occurred)
- Inputs events dont need to be @ time φ necessarily
- Circuit must be acyclic!
  - if cyclic:
    - a) → break cycles
    - b) → stop @ 60/70/...

12.5

- what if I'm not interested in delays, and just want to do logic simulation?

## Structural Model



- Built-in gate primitives:

and, nand, nor, or, xor, xnor, buf, not, bufif0, bufif1,  
notif0, notif1

- Usage:

nand (out, in1, in2); 2-input NAND without delay

and #2 (out, in1, in2, in3); 3-input AND with 2 t.u. delay

not #1 N1(out, in); NOT with 1 t.u. delay and instance name

xor X1(out, in1, in2); 2-input XOR with instance name

(13)

## What is hiZ

- ✓ high-Z
  - ✓ high impedance
  - ✓ tristate
  - ✓ open circuit
  - ✓ floating
- VDD
- }
- synonyms



when  $a \text{ (or) } b = 1$ , switch closed

else .. switch open

| a | b | out |
|---|---|-----|
| 1 | 0 | vdd |
| 0 | 1 | gnd |
| 1 | 1 | NA  |
| 0 | 0 | hiZ |

# Workspace for 'verilog'

Page 7 (row 4, column 2)

13.1



## Dataflow modelling

(continuous)

hyperactive  
monkey

- Another level of abstraction is to model dataflow.
- In dataflow models, signals are continuously assigned values using the assign keyword.
- assign can be used with Boolean expressions.
  - Verilog uses & (and), | (or), ^ (xor) and ~ (not)
- Logic expressions and binary arithmetic are also possible.

target var

assign #10 out = i1 & i2;

- Left hand side must be a net of some kind (scalar or vector), not a register
- Right hand side can be registers, nets.
- Continuous assignments are **always** active. Execution hard to trace
- They are evaluated whenever a right hand side operand changes value
- Delays (inertial) can be added to represent component delays
- LHS evaluates when there is an event on the RHS (therefore independent of ordering of **assign** statements in the code)

deferred

target var

register ≡ remember-er

reg

"variable that  
remembers"

'continuous' assignment is  
updated continuously  
— so no need to "remember"

## Simple Circuit Boolean Expression



Fig. 3-37 Circuit to Demonstrate HDL

$$x = A \cdot B + \bar{C}$$
$$y = \bar{C}$$

## Dataflow Description of Simple Circuit

```
//Circuit specified with Boolean  
equations
```

```
module circuit_bln (x,y,A,B,C);
```

```
    input A,B,C;
```

```
    output x,y;
```

```
    assign x = (A & B) | ~C;
```

```
    assign y = ~C ;
```

```
endmodule
```

, This is not structural

• Order does not  
matter!

## Multiplexor

- Multiplexor is a combinational circuit where an input is chosen by a select signal.
  - Two input mux
  - output =A if select =1
  - output= B if select =0



## Dataflow description of 2-input Mux

- Conditional operator ?: takes three operands:

```
condition? true_expression : false_expression
```

```
module mux2x1_df (A,B,select,OUT);
    input A,B,select;
    output OUT;
    assign OUT = select ? A : B;
endmodule
```

if      then    else

## Behavioural Modelling

(procedural)

parallelism ✓

- Represents circuits at functional and algorithmic level. ✓
- Use procedural statements similar in concept to procedural programming languages (e.g. C, Java),
- Behavioural modelling is mostly used to represent sequential circuits.  
*endmodule*
- We still specify a module in Verilog with inputs and outputs...
  - But inside the module we write code to specify the behavior we want,  
NOT what gates (structure) to connect to make it happen
- Why use behavioral models
  - For high-level specs to drive logic synthesis tools

## Behavioural Modelling

always @ (< trigger >  
condition)

- Behavioural models place procedural statements in a block after the **always** keyword.
- The **always** keyword takes a list of variables which represent a trigger condition. The block of statements is executed whenever the trigger is TRUE.  

- The **target** variables are of type **reg**. This type retains its value until a new value is assigned.
- Behavioral models may also have **initial** blocks.
  - The block *executes only once*
  - By default, starts at time 0
  - Often used for initialization

(20)

1 always @ (a or b)  
must be reg  
 $X = Y;$  "if there is an event  
on a or an event on  
b"

3 always @ (posedge (c))  
 $X = Y;$  

4 always @ (negedge (clk))  
 $X = Y;$  

2 always @ (a)  
 $X = Y;$

## Always Blocks

- Module may have any number of **always** blocks
- Allow us to represent parallelism in hardware.



2D

always @ (<trig. condition>)



- parallelism!

- explicit parallelism

| sw                                                                                                                                   | hw                                                           |
|--------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------|
| <ul style="list-style-type: none"><li>- serial <math>\Rightarrow</math> parallel</li><li>- active, \$\$</li><li>- implicit</li></ul> | <ul style="list-style-type: none"><li>- explicit ✓</li></ul> |

## trick-ish question

# Behavioral Description of an XOR

```
module xorB(X, Y, Z);  
    input X, Y;  
    output Z;  
    reg Z;  
    always @ (X or Y)  
        Z = X ^ Y;  
endmodule
```

# trigger

- Unusual parts of above Verilog

req var → does not imply the variable will have a latch/ff  
however X or Y changes do on it.

~~Cache~~ - “always”  
the following

- “reg” is a  
assignment

- “reg” is only type of behavioral data that can be changed in assignment, so must redeclare z

- Default is single bit data types: X, Y, Z

far data that can be changed in  
Z  
; X, Y, Z  
latch  
or if  
unforced  
on  
always @ (x or y)  
begin w  
z = 0; ✓  
else  
w = 0  
end ✓  
22

## Behavioural description of 2-input Mux



```
module mux2x1_bh(A,B,select,OUT);
    input A,B,select;
    output OUT;
    reg OUT;
    always @ (select or A or B)
        if (select == 1) OUT = A;
        else OUT = B;
endmodule
```

## Behavioral example

- Behavioral edge-triggered DFF implem

```
module dff(Q, D, Clk);  
    output Q;  
    input D, Clk;  
  
    reg O;  
    wire D, Clk;  
  
    always @(posedge Clk)  
        Q = D;  
  
endmodule
```



## Another Behavioral Example



```
always @ (res or posedge clk) begin
    if (res) begin
        Y = 0;
        W = 0;
    end
    else begin
        Y = a & b;
        W = ~c;
    end
end
```

always @ (res or posedge clk) begin  
if (res) begin  
    Y = 0;  
    W = 0;  
end  
else begin  
    Y = a & b;  
    W = ~c;  
end  
end

async reset     } posedge work

## Blocking Assignments



- Represented with an ~~=~~ sign
  - All blocking assignments are executed in sequence

```
module dummy;  
reg x, y, z;  
reg [15:0] reg_a, reg_b;  
integer count;  
initial  
begin  
    x = 0; y = 1; z = 1; // b/o  
    count = 0;  
    reg_a = 16'b0;  
    reg_b = reg_a;  
    reg_a[2] = #15 1;  
    reg_b[15:13] = #10 {x, y, z};  
    count = count + 1;  
end
```

## Non-blocking Assignments

- Represented with a  $\text{<=}$  sign
  - All non-blocking assignments are executed in parallel
  - Try not to mix with blocking assignments

```
module dummy;
    reg x, y, z;
    reg [15:0] reg_a, reg_b;
    integer count;
    initial
    begin
        x = 0; y = 1; z = 1;
        count = 0;
        reg_a[2] <= #15 1;
        reg_b[15:13] <= #10 {x, y, z};
        count = count + 1;
    end
```

(26-27)

always @ (a or b)always @ (a or b)Blocking

- ✓ - software-ish
- ✓ - lines execute in order of appearance
- ✓ - (aka serial)

- "no-timeout" semantics

Non-blocking

- ✓ hardware-ish
- ✓ lines execute in any order.
- ✓ (aka parallel)

- "timeout" semantic

$x=1, y=0$  @ entry

$x=0$   
 $y=0$

$x=0$   
 $y=1$

(27)

always @ ( — )

2      1       $x = a ;$       } B  
3      2       $y = b ;$

1      3       $z \leftarrow x ;$       } NB  
               $w \leftarrow y ;$

✓  
 $x = a ;$   
 $z \leftarrow x ;$   
 $y = b ;$   
 $w \leftarrow y ;$

$$\begin{cases} a=1 \\ b=0 \end{cases}$$

*always @ -*

#1  $x \leq a;$   
#2  $x \leq b;$



27  
*always @ -*

✓ #1  $x = a$  ( $a=1$ )  
✓ #2  $x = b$  ( $b=0$ )



NB = sequential  
B = comb.

## Blocking or Non-blocking???

- Blocking is harder to reason about.
- Also hardware does not work in a blocking (~~sequential~~ serial way)
- So generally you should use non-blocking assignments
  - Easier to synthesize
  - Models parallelism which is inherent in the hardware

(28)

- always @ (posedge clk)

x=a;

y=x;

< B  
matches  
intent >

- $x \leftarrow a$ ;
- $y \leftarrow x$ ;

< NB doesn't  
match  
intent >



## Two kinds of Delays in Verilog

- Inertial Delay – consider the statement

```
assign #4 x = z;
```

- Its delay behavior is called “inertial” delay
- Applicable for gate level primitives and continuous assignments

- Transport delay – consider the statement

```
always @ ( z )  
    y <= #4 z;
```

- Its delay is called “transport” delay
- Applicable in non-blocking assignments



29



~ dataflow  
✓ gatelevel  
procedural

## Delving Deeper

- So far, we saw how some sample circuits are represented in the three styles
- In the next part of this lecture unit, we will talk about
  - ✓ Logic values in Verilog
  - ✓ How to represent hierarchical designs
  - ✓ Testbenches
  - How to represent sequential logic
  - Synthesizability Tips
- Syntax examples (will not go over in class in any detail, this portion of the notes is for your reference)

## Four-Valued Logic

- Verilog Logic Values
  - The underlying data representation allows for any bit to have one of four values
  - $1, 0, x$  (unknown),  $z$  (high impedance)
  - $x$  — one of: 1, 0, z, or in the state of change
  - $z$  — the high impedance output of a tri-state gate.
- What basis do these have in reality?
  - $z$  ... An output is high impedance. Tri-stated outputs are a *real* electrical affect.
  - $x$  ... not a real value. There is no *real* gate that drives an x on to a wire. x is used as a *debugging aid*. x means the simulator can't determine the answer and so maybe you should worry! All values in a simulation start as x.
- Verilog keeps track of more values than these in some situations.

## Four-Valued Logic

- Logic with multi-level logic values
  - Logic with these four values make sense
    - Nand anything with a 0, and you get a 1. This includes having an x or z on the other input. That's the nature of the nand gate
    - Nand two x's and you get an x — makes sense!
  - Note: z treated as an x on input. Their rows and columns are the same
  - If you forget to connect an input ... it will be seen as an z.
  - At the start of simulation, *everything* is an x.

a - 1c  
verilog

|      | Input B |   |   |   |
|------|---------|---|---|---|
| Nand | 0       | 1 | x | z |
| 0    | 1       | 1 | 1 | 1 |
| 1    | 1       | 0 | x | x |
| x    | 1       | x | x | x |
| z    | 1       | x | x | x |



A 4-valued truth table for a Nand gate with two inputs

$$\begin{array}{l}
 x \cdot 1 = 0 \\
 1 \cdot 0 = 1 \\
 x \cdot x = 1 \\
 1 \cdot x = 1 \\
 x \cdot 1 = 1 \\
 1 \cdot 1 = 1 \\
 \Sigma = 0
 \end{array}
 \quad \boxed{= 0} \quad \checkmark$$

|   |   |   |   |
|---|---|---|---|
| A | 1 | 1 | 1 |
| B | 1 | 0 | 1 |
|   | 1 | 0 | 1 |

2-valued NAND

↑ a - 1a - 248

why? - debug easily  
- partition tasks

how

## How to Represent Hierarchy in your Design

- First write the modules for each block of the hierarchy
  - Then wire them up (next page)

```
module B1(a, b, c);  
.....  
endmodule  
  
module B2(a, b, c, d);  
.....  
endmodule  
  
module B3(x, y, z);  
.....  
endmodule
```



## Port Mapping (Connecting things up)

```
module B1(q, w, e, f);  
    input q, w;  
    input [3:0] e;  
    output [1:0] f;  
    ....  
endmodule
```

```
i7  
module top(a,b,c,d);  
    ...  
    B1 b1(w, q, a, c);  
    ....  
endmodule
```

$$w \equiv \text{top } w$$
$$b1.w \equiv B1's \ w$$

Module B1 declared



Module B1 instantiated,  
instance is called myinst  
in this case

## Example (Dataflow, with hierarchy)



✓ ✓ ✓ ✓  
module half\_adder(S, C, A, B);  
output S, C;  
input A, B;

wire S, C, A, B;

assign S = A ^ B;  
assign C = A & B;  
endmodule



## Creating a Full Adder using Half Adder instances



```
✓ module full_adder(sum, cout, in1, in2, cin);
✓ output sum, cout;
✓ input in1, in2, cin;
```

Module name

```
wire sum, cout, in1, in2, cin;
wire I1, I2, I3; C A B
half_adder ha1(I1, I2, in1, in2);
half_adder ha2(sum, I3, I1, cin);
assign cout = I2 || I3;
```

Instance name

(must be unique)

## Hierarchical Names



Remember to use instance names,  
not module names

## Verification

"is what I designed what I wanted"  
"is what I manufactured what I  
designed?"

### testbenches

- Use testbench to verify your design
  - ✓ Special Verilog file for simulating and testing your design
  - ✓ Instantiates the module to be tested
    - Contains code to apply stimulus to the module under test, and monitor the correctness of the response

[no inputs/output]



## Sample Testbench

```
module top_test;
    wire [1:0] t_out;      // Top's signals
    reg [3:0] t_in;
    reg clk;
    top inst(t_out, t_in, clk); // Top's instance
    initial begin          // Generate clock
        clk = 0;
        forever #10 clk = ~clk;
    end
    initial begin          // Generate remaining inputs
        $monitor $time, "%b -> %b", t_in, t_out);
        #5 t_in = 4'b0101;
        #20 t_in = 4'b1110;
        #20 t_in[0] = 1;
        #300 $finish;
    end
endmodule
```

instantiate  
CUT

drive  
input

number  
output



(39)

\$monitor :

→ executed if there is an event  
on any variable in its  
arguments

\$display :

executes when control  
reaches that line

ignore :: structural

## Sequential Logic

```
//Parallel to Serial converter
module ParToSer(LD, X, out, CLK);
    input [3:0] X;
    input LD, CLK;
    output out;
    reg out;
    reg [3:0] Q;
    assign out = Q[0];
    always @ (posedge CLK)
        if (LD) Q=X;
        else Q = Q>>1;
endmodule // mux2

module FF (CLK,Q,D);
    input D, CLK;
    output Q; reg Q;
    always @ (posedge CLK) Q=D;
endmodule // FF
```



- Notes:
  - “always @ (posedge CLK)” forces Q register to be rewritten every simulation cycle.
  - “>>” operator does right shift (shifts in a zero on the left).
  - Shifts on non-reg variables can be done with concatenation:

```
wire [3:0] A, B;
assign B = {1'b0, A[3:1]}
```

# Sequential Logic – another example

```
module mealy (A, CLK, Z);
    input A, CLK;
    output Z;
    reg Z;

    sequential (non-block)
    always @(posedge CLK)
        Pstate = Nstate; // synchronous part
    combination (blocking)
    always @(*)
        begin: COMBINPART
            case (Pstate)
                ST0:
                    if(A)
                        begin
                            Z=1;
                            Nstate = ST3;
                        end
                    else
                        Z=0;
                ST1:
                    if(A)
                        begin
                            Z=0;
                            Nstate = ST2;
                        end
                    else
                        Z=0;
            endcase
        end
endmodule
```



- Notes:

- If we have a state machine updating on a rising clock edge, then we create the always block (triggered on the posedge of clock).
- Also we write the state machine behavior as a case statement.
  - For example if we are in state ST0, and A is 1, then we move to state ST3 in the next clock.
- This kind of code for a state machine is very similar to the state transition diagram based behavior. Hence easy to write.



FYI

- this FSM is completely specified  
(from each state, there is an edge on every input)
- this FSM is deterministic  
(from any state, for every input the FSM goes to a unique state w/ a unique output)

---

seq  
non blk

always @ (posedge CLK)  
PS ← NS;

comb  
blk

always @ (posedge CLK)  
case PS:  
S0: if (A)  
NS = S1;  
Z = 1;

(41.2)

```
else  
    NS = S0;  
    Z = 0;
```

why blocking in comb?

always @ (a)  
1      b =  $\bar{a}$ ;  
2      c =  $\bar{b}$ ;  
3      d =  $\bar{c}$ ;



\* nonblk only 1 executes  
but I wanted 1, 2, 3 to execute

✓ blk if all 3 execute

## Synthesizability Tips

- If you want to synthesize your Verilog code, here are some tips
  - Do not use delays in your code
  - Watch for blocking and non-blocking assignments (next slide)
  - Watch out for complete assignments (2 slides after next)

# Blocking and Non-blocking

Bad: Circuit from blocking assignment.

```
always @(posedge clk)
begin
    b=a;
    a=b;
end
```



Good: Circuit from nonblocking assignment.

```
always @(posedge clk)
begin
    b<=a;
    a<=b;
end
```



## “Complete” Assignments

- If an always block executes, and a variable is *not* assigned
  - Variable keeps its old value (this needs state!!)
  - Hence latch is inserted (inferred memory)
  - This is usually *not* what you want: dangerous for the novice!
- So to avoid this, any variable assigned in an always block should be assigned for any (and every!) execution of the block

*always @ (a or b)  
if (a) z=1;*

*infer memory!*

## Incomplete Triggers

- Leaving out an input trigger usually results in a sequential circuit
- Example: The output of this “and” gate depends on the input history (a latch will be inferred on in2).



```
module and_gate (out, in1, in2);
    input      in1, in2;
    output     out;
    reg       out;

    always @(in1) begin
        out = in1 & in2;
    end

endmodule
```



## Some Verilog Syntax Notes for Your Reference..

- You may find the following slides handy as a partial Verilog reference.
  - It is ***not*** meant to be a complete reference – see the resources on the class website for more detailed references.
  - It is meant to help you with the syntax for common Verilog constructs.

## User Identifiers

- Formed from {[A-Z], [a-z], [0-9], \_, \$}, but ..
- .. can't begin with \$ or [0-9]
  - myidentifier 
  - m\_y\_identifier 
  - 3my\_identifier 
  - \$my\_identifier 
  - \_myidentifier\$ 
- Case sensitivity
  - myid  $\neq$  Myid

## Comments

- `//` The rest of the line is a comment
- `/*` Multiple line  
comment `*/`
- `/*` Nesting `/*` comments `*/` do **NOT** work `*/`

## Numbers in Verilog (i)



- 8'h ax = 1010xxxx
- 12'o 3zx7 = 011zzzxxxx111

## Numbers in Verilog (ii)

- You can insert “\_” for readability
    - `12'b 000_111_010_100`
    - `12'b 000111010100`
    - `12'o 07_24`
  - Bit extension
    - MS bit = 0, x or z  $\Rightarrow$  extend this
      - $4'b x1 = 4'b xx\_x1$
    - MS bit = 1  $\Rightarrow$  zero extension
      - $4'b 1x = 4'b 00\_1x$
- } Represent the same number

## Numbers in Verilog (iii)

- If *size* is omitted it
  - is inferred from the *value* or
  - takes the simulation specific number of bits or
  - takes the machine specific number of bits
- If *radix* is omitted too .. decimal is assumed
  - $15 = <\text{size}>'\text{d } 15$

## Nets (i)

- Can be thought as hardware wires driven by logic
- Equal  $z$  when unconnected
- Various types of nets
  - wire
  - wand (wired-AND)
  - wor (wired-OR)
  - tri (tri-state)
- In following examples: Y is evaluated, *automatically*, every time A or B changes

## Nets (ii)



```
wire Y; // declaration  
assign Y = A & B;
```



```
wand Y; // declaration  
assign Y = A;  
assign Y = B;
```



```
tri Y; // declaration  
assign Y = (dr) ? A : z;
```

|   | A | Y   |
|---|---|-----|
| B | 0 | 0 0 |
| 0 | 1 | 0 1 |

|   | A | Y   |
|---|---|-----|
| B | 0 | 0 1 |
| 0 | 1 | 1 1 |

# Registers

- Variables that store values
- Do not represent real hardware but ..
- .. real hardware can be implemented with registers
- Only one type: `reg`

```
reg A, C; // declaration  
// assignments are always done inside a procedure  
A = 1;  
  
C = A; // C gets the logical value 1  
A = 0; // C is still 1  
C = 0; // C is now 0
```

- Register values are updated explicitly!!

# Vectors

- Represent buses

```
wire [3:0] busA;  
reg [1:4] busB;  
reg [1:0] busC;
```

- Left number is MS bit
- Slice management

$$\text{busC} = \text{busA}[2:1]; \Leftrightarrow \begin{cases} \text{busC}[1] = \text{busA}[2]; \\ \text{busC}[0] = \text{busA}[1]; \end{cases}$$

- Vector assignment (*by position!!*)

$$\text{busB} = \text{busA}; \Leftrightarrow \begin{cases} \text{busB}[1] = \text{busA}[3]; \\ \text{busB}[2] = \text{busA}[2]; \\ \text{busB}[3] = \text{busA}[1]; \\ \text{busB}[4] = \text{busA}[0]; \end{cases}$$

## Integer & Real Data Types

- Declaration

```
integer i, k;  
real r;
```

- Use as registers (inside procedures)

```
i = 1; // assignments occur inside procedure  
r = 2.9;  
k = r; // k is rounded to 3
```

- Integers are not initialized!!
- Reals are initialized to 0.0

## Time Data Type

- Special data type for simulation time measuring

- Declaration

```
time my_time;
```

- Use inside procedure

```
my_time = $time; // get current sim time
```

- Simulation runs at simulation time, not real time

## Arrays (i)

- Syntax

```
integer count[1:5]; // 5 integers
reg var[-15:16]; // 32 1-bit regs
reg [7:0] mem[0:1023]; // 1024 8-bit regs
```

- Accessing array elements

- Entire element: `mem[10] = 8'b 10101010;`
  - Element subfield (needs temp storage):

```
reg [7:0] temp;
.
.
temp = mem[10];
var[6] = temp[2];
```

- Concatenating bits/vectors into a vector

- e.g., sign extend
  - `B[7:0] = {A[3], A[3], A[3], A[3], A[3:0]};`
  - `B[7:0] = {3{A[3]}, A[3:0]};`

- Style: Use `a[7:0] = b[7:0] + c;`  
Not: `a = b + c;` // need to look at declaration

## Arrays (ii)

- Limitation: Cannot access array subfield or entire array at once

```
var[2:9] = ???; // WRONG!!
```

```
var = ???; // WRONG!!
```

- No multi-dimentional arrays

```
reg var[1:10][1:100]; // WRONG!!
```

- Arrays don't work for the Real data type

```
real r[1:10]; // WRONG !!
```

# Strings

- Implemented with regs:

```
reg [8*13:1] string_val; // can hold up to 13 chars  
..  
string_val = "Hello Verilog";  
string_val = "hello"; // MS Bytes are filled with 0  
string_val = "I am overflowed"; // "I " is truncated
```

- Escaped chars:

- \n newline
- \t tab
- %% %
- \\ \
- \\ "

## Logical Operators

- `&&` → logical AND
- `||` → logical OR
- `!` → logical NOT
- Operands evaluated to ONE bit value: `0`, `1` or `x`
- Result is ONE bit value: `0`, `1` or `x`

A = 6;  
B = 0;  
C = x;



A && B → 1 && 0 → 0  
A || !B → 1 || 1 → 1  
C || B → x || 0 → x

but C&&B=0

## Bitwise Operators (i)

- `&` → bitwise AND
  - `|` → bitwise OR
  - `~` → bitwise NOT
  - `^` → bitwise XOR
  - `~~` or `^~` → bitwise XNOR
- 
- Operation on bit by bit basis for bitwise operators.

## Bitwise Operators (ii)



## Reduction Operators

|            |        |
|------------|--------|
| • &        | → AND  |
| •          | → OR   |
| • ^        | → XOR  |
| • ~&       | → NAND |
| • ~        | → NOR  |
| • ~^ or ^~ | → XNOR |

- One multi-bit operand → One single-bit result

```
a = 4'b1001;  
..  
c = |a; // c = 1|0|0|1 = 1
```

- If A = 'b0110, and B = 'b0100 then |B=1, &B=0, ~^A=1
- If C=4'b01x0, then ^C=x.

## Shift Operators

- `>>` → shift right
- `<<` → shift left
- Result is same size as first operand, **always zero filled**

```
a = 4'b1010;  
...  
d = a >> 2;      // d = 0010  
c = a << 1;      // c = 0100
```

## Concatenation Operator

- $\{op1, op2, \dots\}$  → concatenates op1, op2, .. to single number
- Operands must be sized !!

```
reg a;  
reg [2:0] b, c;  
. . .  
a = 1'b 1;  
b = 3'b 010;  
c = 3'b 101;  
catx = {a, b, c};           // catx = 1_010_101  
caty = {b, 2'b11, a};       // caty = 010_11_1  
catz = {b, 1};              // WRONG !!
```

- Replication ..

```
catr = {4{a}, b, 2{c}};    // catr = 1111_010_101101
```

## Relational Operators

- $>$  → greater than
- $<$  → less than
- $\geq$  → greater or equal than
- $\leq$  → less or equal than

- Result is one bit value: 0, 1 or x

1 > 0 → 1

'b1x1 <= 0 → x

10 < z → x

## Equality Operators

- `==` → logical equality
  - `!=` → logical inequality
  - `==>` → case equality
  - `!==` → case inequality
- } Return 0, 1 or  $x$
- } Return 0 or 1

- `4'b 1z0x == 4'b 1z0x` →  $x$
- `4'b 1z0x != 4'b 1z0x` →  $x$
- `4'b 1z0x ==> 4'b 1z0x` → 1
- `4'b 1z0x !==> 4'b 1z0x` → 0

## Conditional Operator

- `cond_expr ? true_expr : false_expr`
- Like a 2-to-1 mux ..



## Arithmetic Operators (i)

- $+, -, *, /, \%$
- If any operand is  $x$  the result is  $x$
- Negative registers:
  - regs can be assigned negative but are treated as unsigned

```
reg [15:0] regA;
```

```
..
```

```
regA = -4'd12;           // stored as  $2^{16}-12 = 65524$ 
```

```
regA/3                  evaluates to 21861
```

## Arithmetic Operators (ii)

- Negative integers:
  - can be assigned negative values
  - different treatment depending on base specification or not

```
reg [15:0] regA;  
  
integer intA;  
  
. . .  
  
intA = -12/3;      // evaluates to -4 (no base spec)  
  
intA = -'d12/3;   // evaluates to 1431655761 (base spec)
```

# Verilog Operators

| Verilog Operator | Name                      | Functional Group |
|------------------|---------------------------|------------------|
| ()               | bit-select or part-select |                  |
| ()               | parenthesis               |                  |
| !                | logical negation          | Logical          |
| ~                | negation                  | Bit-wise         |
| &                | reduction AND             | Reduction        |
|                  | reduction OR              | Reduction        |
| ~&               | reduction NAND            | Reduction        |
| ~                | reduction NOR             | Reduction        |
| ^                | reduction XOR             | Reduction        |
| ~^ or ^~         | reduction XNOR            | Reduction        |
| +                | unary (sign) plus         | Arithmetic       |
| -                | unary (sign) minus        | Arithmetic       |
| {}               | concatenation             | Concatenation    |
| {()}             | replication               | Replication      |
| *                | multiply                  | Arithmetic       |
| /                | divide                    | Arithmetic       |
| %                | modulus                   | Arithmetic       |
| +                | binary plus               | Arithmetic       |
| -                | binary minus              | Arithmetic       |
| <<               | shift left                | Shift            |
| >>               | shift right               | Shift            |

|          |                          |             |
|----------|--------------------------|-------------|
| >        | greater than             | Relational  |
| >=       | greater than or equal to | Relational  |
| <        | less than                | Relational  |
| <=       | less than or equal to    | Relational  |
| ==       | logical equality         | Equality    |
| !=       | logical inequality       | Equality    |
| ==       | case equality            | Equality    |
| !=       | case inequality          | Equality    |
| &        | bit-wise AND             | Bit-wise    |
| ^        | bit-wise XOR             | Bit-wise    |
| ^~ or ~^ | bit-wise XNOR            | Bit-wise    |
|          | bit-wise OR              | Bit-wise    |
| &&       | logical AND              | Logical     |
|          | logical OR               | Logical     |
| :?       | conditional              | Conditional |

# Operator Precedence

|                                 |                    |
|---------------------------------|--------------------|
| <code>+ - ! ~ unary</code>      | highest precedence |
| <code>* / %</code>              |                    |
| <code>+ - (binary)</code>       |                    |
| <code>&lt;&lt; &gt;&gt;</code>  |                    |
| <code>&lt; &lt;= == &gt;</code> |                    |
| <code>== != === !==</code>      |                    |
| <code>&amp; ~ &amp;</code>      |                    |
| <code>^ ^~ ~^</code>            |                    |
| <code>  ~  </code>              |                    |
| <code>&amp;&amp;</code>         |                    |
| <code>  </code>                 |                    |
| <code>? : conditional</code>    | lowest precedence  |

Use parentheses to  
enforce your  
priority

## Verilog Variables

- wire
  - Variable used simply to connect components together
- reg
  - Variable that saves a value as part of a behavioral description
  - Usually corresponds to a wire in the circuit
  - Is *NOT* necessarily a register in the circuit
- usage:
  - Don't confuse reg assignments with the combinational continuous assign statement!
  - Reg should only be used with always blocks (sequential logic, to be presented ...)

# Verilog Module

- Corresponds to a circuit component
  - “Parameter list” is the list of external connections, aka “ports”
  - Ports are declared “input”, “output” or “inout”
    - inout ports used on tri-state buses
  - Port declarations imply that the variables are wires

The diagram shows a Verilog module definition with annotations:

```
module name
  module full_addr (A, B, Cin, S, Cout);
    input A, B, Cin;
    output S, Cout;
    assign {Cout, S} = A + B + Cin;
  endmodule
```

- An arrow points from the text "module name" to the word "module".
- An arrow points from the text "ports" to the port list "(A, B, Cin, S, Cout)".
- An arrow points from the text "inputs/outputs" to the port list "A, B, Cin; S, Cout;".

# Verilog Continuous Assignment

- Assignment is continuously evaluated
- assign corresponds to a connection or a simple component with the described function
- Target is *NEVER* a reg variable
- Dataflow style

```
assign A = x | (Y & ~Z);           ← use of Boolean operators  
                                         (~ for bit-wise, ! for logical negation)
```

```
assign B[3:0] = 4'b01XX;           ← bits can take on four values  
                                         (0, 1, X, Z)
```

```
assign C[15:0] = 4'h00ff;          ← variables can be n-bits wide  
                                         (MSB:LSB)
```

```
assign #3 {Cout, S[3:0]} = A[3:0] + B[3:0] + Cin;
```

↑  
delay of performing computation, only used by simulator, not synthesis

← use of arithmetic operator  
multiple assignment (concatenation)

## Verilog if

- Same as C if statement

```
// Simple 4-1 mux
module mux4 (sel, A, B, C, D, Y);
    input [1:0] sel;      // 2-bit control signal
    input A, B, C, D;
    output Y;
    reg Y;                  // target of assignment

    always @(sel or A or B or C or D)
        if (sel == 2'b00) Y = A;
        else if (sel == 2'b01) Y = B;
        else if (sel == 2'b10) Y = C;
        else if (sel == 2'b11) Y = D;

endmodule
```

## Verilog if

```
// Simple 4-1 mux
module mux4 (sel, A, B, C, D, Y);
    input [1:0] sel;      // 2-bit control signal
    input A, B, C, D;
    output Y;
    reg Y;                  // target of assignment

    always @(sel or A or B or C or D)
        if (sel[0] == 0)
            if (sel[1] == 0) Y = A;
            else              Y = B;
        else
            if (sel[1] == 0) Y = C;
            else              Y = D;
endmodule
```

## Verilog case

- Sequential execution of cases
  - Only first case that matches is executed (no break)
  - Default case can be used

```
// Simple 4-1 mux
module mux4 (sel, A, B, C, D, Y);
    input [1:0] sel;           // 2-bit control signal
    input A, B, C, D;
    output Y;                 // target of assignment

    always @(sel or A or B or C or D)
        case (sel)
            2'b00: Y = A;
            2'b01: Y = B;
            2'b10: Y = C;
            2'b11: Y = D;
        endcase
    endmodule
```

|  
Conditions tested in  
top to bottom order  
↓

## Verilog case

- Without the default case, this example would create a latch for Y
- Assigning X to a variable means synthesis is free to assign any value

```
// Simple binary encoder (input is 1-hot)
module encode (A, Y);
    input [7:0] A;                      // 8-bit input vector
    output [2:0] Y;                     // 3-bit encoded output
    reg     [2:0] Y;                   // target of assignment

    always @(A)
        case (A)
            8'b00000001: Y = 0;
            8'b00000010: Y = 1;
            8'b00000100: Y = 2;
            8'b00001000: Y = 3;
            8'b00010000: Y = 4;
            8'b00100000: Y = 5;
            8'b01000000: Y = 6;
            8'b10000000: Y = 7;
            default:      Y = 3'bx;       // Don't care when input is not 1-hot
        endcase
    endmodule
```

## Verilog case (cont)

- Cases are executed sequentially
  - The following implements a *priority* encoder

```
// Priority encoder
module encode (A, Y);
    input [7:0] A;           // 8-bit input vector
    output [2:0] Y;          // 3-bit encoded output
    reg     [2:0] Y;          // target of assignment

    always @(A)
        case (1'b1)
            A[0]:   Y = 0;
            A[1]:   Y = 1;
            A[2]:   Y = 2;
            A[3]:   Y = 3;
            A[4]:   Y = 4;
            A[5]:   Y = 5;
            A[6]:   Y = 6;
            A[7]:   Y = 7;
            default: Y = 3'bX; // Don't care when input is all 0's
        endcase
    endmodule
```

## Parallel case

- A priority encoder is more expensive than a simple encoder
  - If we know the input is 1-hot, we can tell the synthesis tools
    - “**parallel-case**” pragma says the order of cases does not matter

```
// simple encoder
module encode (A, Y);
    input [7:0] A;           // 8-bit input vector
    output [2:0] Y;          // 3-bit encoded output
    reg   [2:0] Y;           // target of assignment

    always @(A)
        case (1'b1)           // synthesis parallel-case
            A[0]:   Y = 0;
            A[1]:   Y = 1;
            A[2]:   Y = 2;
            A[3]:   Y = 3;
            A[4]:   Y = 4;
            A[5]:   Y = 5;
            A[6]:   Y = 6;
            A[7]:   Y = 7;
            default: Y = 3'bX; // Don't care when input is all 0's
        endcase
    endmodule
```

## Verilog casex

- Like case, but cases can include ‘X’
  - X bits not used when evaluating the cases
  - In other words, you don’t care about those bits!

## casex Example

```
// Priority encoder
module encode (A, valid, Y);
  input [7:0] A;                      // 8-bit input vector
  output [2:0] Y;                     // 3-bit encoded output
  output valid;                      // Asserted when an input is not all 0's
  reg    [2:0] Y;                     // target of assignment
  reg    valid;

  always @(A) begin
    valid = 1;
    casex (A)
      8'bXXXXXXXX1: Y = 0;
      8'bXXXXXX10: Y = 1;
      8'bXXXXX100: Y = 2;
      8'bXXXX1000: Y = 3;
      8'bXXX10000: Y = 4;
      8'bXX100000: Y = 5;
      8'bX1000000: Y = 6;
      8'b10000000: Y = 7;
    default: begin
      valid = 0;
      Y = 3'bX; // Don't care when input is all 0's
    end
    endcase
  end
endmodule
```

## Verilog for

- `for` is similar to C
- `for` statement is executed at compile time (like macro expansion)
  - Useful for parameterized designs.

```
// simple encoder
module encode (A, Y);
    input [7:0] A;                      // 8-bit input vector
    output [2:0] Y;                     // 3-bit encoded output
    reg      [2:0] Y;                   // target of assignment

    integer i;                          // Temporary variables for program only
    reg [7:0] test;

    always @(A) begin
        test = 8b'00000001;
        Y = 3'bX;
        for (i = 0; i < 8; i = i + 1) begin
            if (A == test) Y = N;
            test = test << 1;
        end
    end
endmodule
```

## Verilog while/repeat/forever

- `while` (expression) statement
  - Execute statement while expression is true
- `repeat` (expression) statement
  - Execute statement a fixed number of times
- `forever` statement
  - Execute statement forever

## full-case and parallel-case

- **// synthesis parallel\_case**
  - Tells compiler that ordering of cases is not important
  - That is, cases do not overlap
    - e. g., state machine - can't be in multiple states
  - Gives cheaper implementation
- **// synthesis full\_case**
  - Tells compiler that cases left out can be treated as don't cares
  - Avoids incomplete specification and resulting latches

OFFICE  
HOURS  
DISCUSSION



NOT IN SYLLABUS —



Manifests - antistatic ·  
spark

Layman - static elec car  
Tech zap clip

Tech



<apps>



- Tech News
- Tech News Tube
- Flipboard
- SmartNews
- slashdot
- slashgear

<website>

- interesting engineering
- ee times