# Digital System Design with HDL (I) Lecture 7

Dr. Ming Xu

Dept of Electrical & Electronic Engineering

XJTLU

### In This Session

- Functions and Tasks
- Verilog for Combinational Circuits
  - Multiplexers
  - Decoders
  - Encoders
  - Comparators

1

# **Functions**

- · Functions should have at least one input.
- The function name serves as the *only* output variable.
   No other output or inout is allowed.
- · Functions are only defined within a module.

### **Functions**

- A function is a part of the code that may be re-used and thus separated from the other parts in a module.
- Syntax:

function [ size\_or\_type ] function\_name;

input declarations

local variable declarations

procedural statements

### endfunction

size\_or\_type (optional) is the returned type or range as
[msb:lsb]. The default is a 1-bit value.

### **Functions**

Example: a 16-to-1 multiplexer



```
module mux f (W, S16, f);
input [0:15] W;
 input [3:0] $16;
 output reg f;
 reg [0:3] M;
 function mux4to1:
  input [0:3] W;
  input [1:0] S:
  if (S = 0) \max_{0 \le t \le 1} W[0];
  else if (S = = 1) \max_{i=1}^{n} 4 to_{i} = W[1];
  else if (S = 2) \text{ mux4to1} = W[2];
  else if (S = = 3) \text{ mux4to1} = W[3]:
 endfunction
 always @(W, S16)
 begin
  M[0] = mux4to1(W[0:3], S16[1:0]);
  M[1] = mux4to1(W[4:7], S16[1:0]);
  M[2] = mux4to1(W[8:11], S16[1:0]);
  M[3] = mux4to1(W[12:15], S16[1:0]);
  f = mux4to1(M[0:3], S16[3:2]);
 end
                                       5
endmodule
```

### **Tasks**

- A task is a part of the code that may be re-used and thus separated from the other parts in a module.
- A task may have any number of inputs, outputs or inouts.
- · A task does not return any value by its name.
- · A task is like a subroutine.

6

### **Tasks**

### Syntax:

endtask

task task\_name; input declarations output declarations local variable declarations procedural statements

### **Tasks**

Example: a 16-to-1 multiplexer



```
module mux t (W, S16, f);
input [0:15] W;
input [3:0] $16;
 output reg f;
 reg [0:3] M;
 task mux4to1;
 input [0:3] W;
 input [1:0] S;
 output Result;
 begin
  if(S = 0) Result = W[0];
   else if (S = = 1) Result = W[1];
   else if (S = 2) Result = W[2];
   else if (S = 3) Result = W[3];
 end
 endtask
 always @(W, S16)
 begin
 mux4to1(W[0:3], S16[1:0], M[0]);
 mux4to1(W[4:7], S16[1:0], M[1]);
 mux4to1(W[8:11], S16[1:0], M[2]);
 mux4to1(W[12:15], S16[1:0], M[3]);
 mux4to1(M[0:3], S16[3:2], f);
 end
                                   8
endmodule
```

### Combinational circuits



- · Outputs only depend on inputs
- Could include: gates, multiplexers, encoders, decoders, code converters, comparators ...
- · Verilog description: gates, logic expression, behaviour

## Multiplexers

### Using conditional operators





```
\label{eq:module} \begin{split} & \textbf{module} \ mux2to1 \ (w0, w1, s, f); \\ & \textbf{input} \ w0, w1, s; \\ & \textbf{output} \ f; \\ & \textbf{reg} \ f; \\ & \textbf{always} \ @(w0 \ or \ w1 \ or \ s) \\ & f = s \ ? \ w1 : w0; \\ \\ & \textbf{endmodule} \end{split}
```

Conditional operator in procedural statement

in continuous assignment

1

# Multiplexers

### Using nesting conditional operators



```
\label{eq:module} \begin{split} & \textbf{module} \ \text{mux4to1} \ (w0, w1, w2, w3, S, f); \\ & \textbf{input} \ w0, w1, w2, w3; \\ & \textbf{input} \ [1:0] \ S; \\ & \textbf{output} \ f; \\ & \textbf{assign} \ f = S[1] \ ? \ (S[0] \ ? \ w3: w2): (S[0] \ ? \ w1: w0); \\ & \textbf{endmodule} \end{split}
```

# Multiplexers

### Using If-else statements

```
\label{eq:module} \begin{split} \textbf{module} & \ \text{mux2to1} \ (w0,w1,s,f); \\ & \ \textbf{input} \ w0,w1,s; \\ & \ \textbf{output} \ f; \\ & \ \textbf{reg} \ f; \\ \\ & \ \textbf{always} \ @ (w0 \ \text{or} \ w1 \ \text{or} \ s) \\ & \ \textbf{if} \ (s==0) \\ & \ f=w0; \\ & \ \textbf{else} \\ & \ f=w1; \\ \\ & \ \textbf{endmodule} \end{split}
```

### Using case statements

# Multiplexers

### Hierarchical 16-to-1 Multiplexer

```
module mux16to1 (W, S16, f);
    input [0:15] W;
    input [3:0] S16;
    output f;
    wire [0:3] M;

mux4to1 Mux1 (W[0:3], S16[1:0], M[0]);
    mux4to1 Mux2 (W[4:7], S16[1:0], M[1]);
    mux4to1 Mux3 (W[8:11], S16[1:0], M[2]);
    mux4to1 Mux4 (W[12:15], S16[1:0], M[3]);
    mux4to1 Mux5 (M[0:3], S16[3:2], f);

endmodule
```

13

### **Decoders**

### Decoder using case



Concatenate operator

Default for En=0

```
module dec2to4 (W, Y, En);
input [1:0] W;
input En;
output [0:3] Y;
reg [0:3] Y;

always @(W or En)
case ({En, W})
3'b100: Y = 4'b1000;
3'b110: Y = 4'b0100;
3'b111: Y = 4'b0001;
→ default: Y = 4'b0000;
endcase
```

### **Decoders**

### Decoder using if-else, case



```
module dec2to4 (W, Y, En):
    input [1:0] W;
    input En;
    output [0:3] Y;
    reg [0:3] Y;
    always @(W or En)
    begin
       if (En == 0)
        Y = 4'b0000:
       else
         case (W)
           0: Y = 4'b1000;
           1: Y = 4'b0100;
           2: Y = 4'b0010;
           3: Y = 4'b0001;
         endcase
    end
                                   15
endmodule
```

### **Decoders**

### Decoder using for loops



```
\label{eq:module} \begin{split} & \textbf{module} \ dec2to4 \ (W, Y, En); \\ & \textbf{input} \ [1:0] \ W; \\ & \textbf{input} \ En; \\ & \textbf{output reg} \ [0:3] \ Y; \\ & \textbf{integer} \ k; \\ & \textbf{always} \ @(W, En) \\ & \textbf{for} \ (k = 0; k <= 3; k = k+1) \\ & \textbf{if} \ ((W == k) \ \&\& \ (En == 1)) \\ & Y[k] = 1; \\ & \textbf{else} \\ & Y[k] = 0; \\ & \textbf{endmodule} \end{split}
```

### **Decoders**

### Hierarchical code for 4-16 decoder



```
module dec4to16 (W, Y, En);
input [3:0] W;
input En;
output [0:15] Y;
wire [0:3] M;

dec2to4 Dec1 (W[3:2], M[0:3], En);
dec2to4 Dec2 (W[1:0], Y[0:3], M[0]);
dec2to4 Dec3 (W[1:0], Y[4:7], M[1]);
dec2to4 Dec4 (W[1:0], Y[8:11], M[2]);
dec2to4 Dec5 (W[1:0], Y[12:15], M[3]);

endmodule
```

17

### **Decoders**

### BCD-to-7-segment decoder



| <i>w</i> <sub>3</sub> | <i>w</i> <sub>2</sub> | <i>w</i> <sub>1</sub> | $w_0$ | а | b | С | d | е | f | g |
|-----------------------|-----------------------|-----------------------|-------|---|---|---|---|---|---|---|
| 0                     | 0                     | 0                     | 0     | 1 | 1 | 1 | 1 | 1 | 1 | 0 |
| 0                     | 0                     | 0                     | 1     | 0 | 1 | 1 | 0 | 0 | 0 | 0 |
| 0                     | 0                     | 1                     | 0     | 1 | 1 | 0 | 1 | 1 | 0 | 1 |
| 0                     | 0                     | 1                     | 1     | 1 | 1 | 1 | 1 | 0 | 0 | 1 |
| 0                     | 1                     | 0                     | 0     | 0 | 1 | 1 | 0 | 0 | 1 | 1 |
| 0                     | 1                     | 0                     | 1     | 1 | 0 | 1 | 1 | 0 | 1 | 1 |
| 0                     | 1                     | 1                     | 0     | 1 | 0 | 1 | 1 | 1 | 1 | 1 |
| 0                     | 1                     | 1                     | 1     | 1 | 1 | 1 | 0 | 0 | 0 | 0 |
| 1                     | 0                     | 0                     | 0     | 1 | 1 | 1 | 1 | 1 | 1 | 1 |
| 1                     | 0                     | 0                     | 1     | 1 | 1 | 1 | 1 | 0 | 1 | 1 |
|                       |                       |                       |       |   |   |   |   |   |   |   |

```
module seg7 (bcd, leds);
   input [3:0] bcd;
   output [1:7] leds;
   reg [1:7] leds;
   always @(bcd)
      case (bcd) //abcdefg
        0: leds = 7'b11111110;
        1: leds = 7'b0110000;
        2: leds = 7'b1101101;
        3: leds = 7'b1111001;
        4: leds = 7'b0110011:
        5: leds = 7'b1011011:
        6: leds = 7'b10111111;
        7: leds = 7'b1110000:
        8: leds = 7'b11111111;
        9: leds = 7'b1111011:
        default: leds = 7'bx;
      endcase
endmodule
                                 18
```

### **Encoders**



```
module encoder (Code, Data);
output[2:0] Code;
input[7:0]Data;
reg[2:0] Code;
always@ (Data)
  case (Data)
    8'b00000001 : Code = 3'b000;
    8'b00000010 : Code = 3'b001;
    8'b00000100 : Code = 3'b010;
    8'b00001000 : Code = 3'b011;
    8'b00010000 : Code = 3'b100;
    8'b00100000 : Code = 3'b101;
    8'b01000000 : Code = 3'b110:
    8'b10000000 : Code = 3'b111;
    default Code = 3'bx;
  endcase
endmodule
                                    19
```

### **Encoders**

Priority encoder using casex

| $W_3$ | $W_2$ | $w_1$ | $w_0$ | <i>y</i> <sub>1</sub> | <i>y</i> <sub>0</sub> | Z |
|-------|-------|-------|-------|-----------------------|-----------------------|---|
| 0     | 0     | 0     | 0     | d                     | d                     | 0 |
| 0     | 0     | 0     | 1     | 0                     | 0                     | 1 |
| 0     | 0     | 1     | X     | 0                     | 1                     | 1 |
| 0     | 1     | Χ     | Х     | 1                     | 0                     | 1 |
| 1     | Χ     | Χ     | Χ     | 1                     | 1                     | 1 |

```
module priority (W, Y, z);
   input [3:0] W;
   output [1:0] Y;
   output z;
   reg [1:0] Y;
   reg z;
   always @(W)
   begin
      z = 1;
      casex(W)
        4'b1xxx: Y = 3;
        4'b01xx: Y = 2:
        4'b001x: Y = 1;
        4'b0001: Y = 0;
        default: begin
                   z = 0;
                   Y = 2'bx:
                end
      endcase
   end
                                20
endmodule
```

# **Encoders**

Priority encoder using for loops

| <i>w</i> <sub>3</sub> | $W_2$ | $w_1$ | $w_0$ | J | 4 | <i>y</i> <sub>0</sub> | Z |
|-----------------------|-------|-------|-------|---|---|-----------------------|---|
| 0                     | 0     | 0     | 0     |   | d | d                     | 0 |
| 0                     | 0     | 0     | 1     |   | 0 | 0                     | 1 |
| 0                     | 0     | 1     | Х     |   | 0 | 1                     | 1 |
| 0                     | 1     | Χ     | Х     |   | 1 | 0                     | 1 |
| 1                     | Х     | Х     | Х     |   | 1 | 1                     | 1 |

```
module priority (W, Y, z);
   input [3:0] W;
   output reg [1:0] Y;
   output reg z;
   integer k;
   always @(W)
   begin
      Y = 2'bx;
      z = 0;
      for (k = 0; k < 4; k = k+1)
        if (W[k])
        begin
           \mathbf{Y} = \mathbf{k};
           z = 1;
        end
   end
endmodule
                               21
```

# Comparators



```
module compare (A, B, AeqB, AgtB, AltB);
   input [3:0] A, B;
   output reg AeqB, AgtB, AltB;
   always @(A, B)
   begin
      AeqB = 0;
      AgtB = 0;
      AltB = 0;
      if(A == B)
       AeqB = 1;
      else if (A > B)
       AgtB = 1;
      else
       AltB = 1;
   end
endmodule
```

22