

## **DSD LAB FAT**

### **1) MUX Design**

Aim: To design a Verilog model and verify the truth table of the multiplexer using Gate-level modelling and Data flow modelling

Tools Required: ModelSim

Procedure:

- 1) Determine the required number of inputs and outputs and assign a symbol to each
- 2) Derive the truth table
- 3) Obtain the simplified Boolean function for each output as a function of the input variables
- 4) Draw the logic diagram and verify the design

Theory: A Multiplexer selects one of many inputs signals and forwards the selected input to a single output line. It acts as a data selector. The input line chosen is determined by a separate set of select lines.

Program:

Gate-level Modelling:

line 10: and g2(w5,a,w2,I2);

```
2 module mux_41(a,b,I0,I1,I2,I3,F);
3   input (a,b,I0,I1,I2,I3);
4   output F;
5   wire w1, w2, w3, w4, w5, w6;
6   not (w1,a);
7   not (w2,b);
8   and g0(w3,w1,w2,I0);
9   and g1(w4,w1,b,I1);
10  and g3(w6,a,b,I3);
11  or (F,w3,w4,w5,w6);
12 endmodule|
```

Data-flow Modelling:

```
module mux1_4(a,b,i0,i1,i2,i3,f);
  input a,b,i0,i1,i2,i3;
  output f;
  assign f=((~a)&(~b)&(i0)) | ((~a)&(b)&(i1)) | ((a)&(~b)&(i2)) | ((a)&(b)&(i3));
endmodule|
```

Truth Table:

| a | b | F  |
|---|---|----|
| 0 | 0 | I0 |
| 0 | 1 | I1 |
| 1 | 0 | I2 |
| 1 | 1 | I3 |



Conclusion: With the use of gate-level and data flow modelling, the Verilog model for multiplexer were created and simulated successfully. The simulation results matched the theoretical truth tables, indicating the design were correct.

## 2) Decoder

Theory: Decoder is a full subtractor and a combinational circuit that subtracts 2 single-bit binary numbers while also considering a borrow-in bit from a previous stage. It produces a difference bit and a borrow-out bit as its outputs.

Program:

Gate-level Modelling:

```
module decoder(a,b,d0,d1,d2,d3);
input a,b;
output d0,d1,d2,d3;
wire w1,w2;
not g1(w1,a);
not g2(w2,b);
and g3(d0,w1,w2);
and g4(d1,w1,b);
and g5(d2,a,w2);
and g6(d1,a,b);
endmodule
```

Data-flow Modelling:

```
module decoder1(a,b,d0,d1,d2,d3);
input a,b;
output d0,d1,d2,d3;
assign nota=~a;
assign notb=~b;
assign d0=nota&notb;
assign d1=nota&b;
assign d2=a&notb;
assign d3=a&b;
endmodule;
```

Truth Table:

| a | b | d0 | d1 | d2 | d3 |              |
|---|---|----|----|----|----|--------------|
| 0 | 0 | 1  | 0  | 0  | 0  | $d_0 = a'b'$ |
| 0 | 1 | 0  | 1  | 0  | 0  | $d_1 = a'b$  |
| 1 | 0 | 0  | 0  | 1  | 0  | $d_2 = ab'$  |
| 1 | 1 | 0  | 0  | 0  | 1  | $d_3 = ab$   |

### 3) 3-Bit Comparator

Theory: A 3-Bit Comparator is a combinational circuit that compares 2 3-Bit binary numbers ( $A \& B$ ) and produces output indicating their relative magnitudes. The comparator has 3 outputs:  $A > B$ ,  $A = B$ ,  $A < B$ . The circuit checks bit by bit from the most significant bit (MSB) to the least significant bit (LSB) to determine the relationship. Such comparators are commonly used in arithmetic and logic units and control circuits.

Program:

```

2 module comp_3bit(A,B,E,G,L);
3 input [2:0] A,B;
4 output E,G,L;
5 wire x0,w1,x2;
6 assign x[0]=A[0]-~B[0];
7 assign x1=A[1]-~B[1];
8 assign x2=A[2]-~B[2];
9 assign E=x2&x1&x0;
10 assign G=(A[2] & (~B[2])) | (x2 & A[1] & (~B[1])) | (x2 & x1 &(A[0] & (~B[0])));
11 assign L=(B[2] & (~A[2])) | (x2 & (B[1] & (~A[1]))) | (x2 & x1 & (B[0] & (~A[0])));
12 endmodule
13
14 module tb_comp();
15 reg [2:0] A,B;
16 wire E,G,L;
17 comp_3bit mm0(E,G,L,A,B);
18 initial
19 begin
20 A=3'b111; B=3'b111;
21 #100 A=3'b111; B=3'b001;
22 #100 A=3'b001; B=3'b111;
23 end
24 endmodule
25

```

Truth Table:

| A   | B   | E   | G   | L   |
|-----|-----|-----|-----|-----|
| 000 | 000 | 1   | 0   | 0   |
| 000 | 001 | 0   | 0   | 1   |
| 000 | 010 | 0   | 0   | 1   |
| 000 | 011 | 0   | 0   | 1   |
| 000 | 100 | 0   | 0   | 1   |
| --- | --- | --- | --- | --- |
| 111 | 110 | 0   | 1   | 0   |
| 111 | 111 | 1   | 0   | 1   |

#### 4) Up Counter

Theory: Up Counter is a sequential circuit where all flip-flops are connected to the same clock pulse and the output state increments with each pulse.

Program:

```

2 module up_counter(Q,clk,rst);
3 input clk,rst;
4 output [3:0] Q;
5 reg [3:0] Q;
6 always @ (posedge clk or posedge rst)
7 begin
8 if(rst)
9   Q<=4'b0;
10 else
11   Q<=Q+1;
12 end
13 endmodule
14
15 module up_counter_test;
16 wire [3:0] Q;
17 reg clk, rst;
18 up_counter c1(Q,clk,rst);
19 always
20 #5 clk=~clk;
21 initial
22 begin
23 clk=0; rst=1;
24 #10 rst=0;
25 #200 $stop;
26 end
27 endmodule

```



Truth Table:

| clk | Q2 | Q1 | Q0 |
|-----|----|----|----|
| 0   | 0  | 0  | 0  |
| 1   | 0  | 0  | 1  |
| 0   | 0  | 1  | 0  |
| 1   | 0  | 1  | 1  |
| 0   | 1  | 0  | 0  |
| 1   | 1  | 0  | 1  |
| 0   | 1  | 1  | 0  |
| 1   | 1  | 1  | 1  |

|   |   |   |   |
|---|---|---|---|
| 0 | 0 | 0 | 0 |
| 1 | 0 | 0 | 1 |
| 2 | 0 | 1 | 0 |
| 3 | 0 | 1 | 1 |
| 4 | 1 | 0 | 0 |
| 5 | 1 | 0 | 1 |
| 6 | 1 | 1 | 0 |
| 7 | 1 | 1 | 1 |

## 5) Down Counter

Theory: Down Counter is a sequential circuit where all flip-flops are connected to the same clock pulse and the output state decrements with each pulse.

Program:

```

2 module down_counter(Q,clk,rst);
3 input clk,rst;
4 output [3:0] Q;
5 reg [3:0] Q;
6 always @ (posedge clk or posedge rst)
7 begin
8 if(rst)
9   Q<=4'b0;
10 else
11   Q<=Q-1;
12 end
13 endmodule
14
15 module down_counter_test;
16 wire [3:0] Q;
17 reg clk, rst;
18 down_counter c1(Q,clk,rst);
19 always
20 #5 clk=~clk;
21 initial
22 begin
23 clk=0; rst=1;
24 #10 rst=0;
25 #200 $stop;
26 end
27 endmodule

```



Truth Table:

| clk | Q2 | Q1 | Q0 |
|-----|----|----|----|
| 0   | 0  | 0  | 0  |
| 1   | 0  | 0  | 1  |
| 2   | 0  | 1  | 0  |
| 3   | 0  | 1  | 1  |
| 4   | 1  | 0  | 0  |
| 5   | 1  | 0  | 1  |
| 6   | 1  | 1  | 0  |
| 7   | 1  | 1  | 1  |

|   |   |   |   |
|---|---|---|---|
| 0 | 0 | 0 | 0 |
| 1 | 1 | 1 | 1 |
| 2 | 1 | 1 | 0 |
| 3 | 1 | 0 | 1 |
| 4 | 1 | 0 | 0 |
| 5 | 0 | 1 | 1 |
| 6 | 0 | 1 | 0 |
| 7 | 0 | 0 | 1 |

## 6) SISO

Theory: A SISO shift register is a sequential circuit that takes one bit of data in (serially) per clock pulse and outputs one bit of data (serially) per clock pulse.

Program:

```

2 module siso(so,si,clk,rst);
3 input si,clk,rst;
4 output so;
5 reg [3:0] q;
6 always @ (posedge clk or posedge rst)
7 begin
8 if(rst)
9   q<=4'b0;
10 else
11   q<={si,q[3:1]};
12 end
13 assign so=q[0];
14 endmodule
15
16 module siso_test;
17 wire so;
18 reg si,clk,rst;
19 siso s1(so,si,clk,rst);
20 always
21 #5 clk=~clk;
22 initial
23 begin
24 si=0; clk=0; rst=1;
25 #10 rst=0;
26 #10 si=1;
27 #10 si=0;
28 #10 si=0;
29 #10 si=1;
30 #10 si=1;
31 #10 si=0;
32 #50 $stop;
33 end
34 endmodule

```



©Elprocus.com

Truth Table:

| si | clk | rst | so |
|----|-----|-----|----|
|----|-----|-----|----|

|   |   |   |   |
|---|---|---|---|
| 1 | 0 | 0 | 0 |
| 1 | 1 | 0 | 1 |
| 1 | 0 | 0 | 1 |
| 0 | 1 | 0 | 0 |

## 7) PISO

Theory: PISO shift register is a sequential circuit that takes all data bits simultaneously (parallelly) on one clock pulse but outputs are one bit of data (serially) per subsequent clock pulse.

Program:

```

2 module piso(so,in,load,clk,rst);
3 input load,clk,rst;
4 output so;
5 input[3:0] in;
6 reg [3:0] q;
7 always @ (posedge clk or posedge rst)
8 begin
9 if(rst)
10   q<=4'b0;
11 else if(load)
12   q<=in;
13 else
14   q<=[1'b0,q[3:1]];
15 end
16 assign so=~q[0];
17 endmodule
18
19 module piso_test;
20 wire so;
21 reg load,clk,rst;
22 reg [3:0] in;
23 piso p1(so,in,load,clk,rst);
24 always
25 #5 clk=~clk;
26 initial
27 begin
28 load=1; clk=0; rst=1;
29 in=4'b1001;
30 #10 rst=0;
31 #10 load=0;
32 #10 load=1; in=4'b0011;
33 #10 load=0;
34 #50 $stop;
35 end
36 endmodule

```



Truth Table:

|    |     |     |      |
|----|-----|-----|------|
| in | rst | clk | load |
|----|-----|-----|------|

|      |   |   |   |
|------|---|---|---|
| 1001 | 1 | 0 | 1 |
| 1001 | 1 | 1 | 1 |
| 0011 | 0 | 0 | 1 |
| 0011 | 0 | 1 | 0 |

## 8) PIPo

Theory: PIPo shift register is a sequential circuit that takes all data bits simultaneously (parallelly) on one clock pulse and outputs are all stored data bits simultaneously (parallelly).

Program:

```

2 module pipo(q,in,load,clk,rst);
3 input load,clk,rst;
4 output [3:0] q;
5 input[3:0] in;
6 reg [3:0] q;
7 always @ (posedge clk or posedge rst)
8 begin
9 if(rst)
10   q<=4'b0;
11 else if(load)
12   q<=in;
13 else
14   q<=q;
15 end
16 endmodule
17
18 module pipo_test;
19 wire [3:0] q;
20 reg load,clk,rst;
21 reg [3:0] in;
22 pipo p1(q,in,load,clk,rst);
23 always
24 #5 clk=~clk;
25 initial
26 begin
27 load=1; clk=0; rst=1;
28 in=4'b1001;
29 #10 rst=0;
30 #10 load=0;
31 #10 load=1; in=4'b0011;
32 #10 load=0;
33 #50 $stop;
34 end
35 endmodule

```



Truth Table:

| in   | rst | clk | load |
|------|-----|-----|------|
| 1001 | 1   | 0   | 1    |
| 1001 | 1   | 1   | 1    |
| 1001 | 0   | 0   | 0    |
| 0011 | 0   | 1   | 1    |

## 9) Ripple Carry Adder-Subtractor

Theory: A ripple carry adder-subtractor is a combinational circuit that can perform both addition and subtraction on binary numbers. It uses a ripple carry adder along with XOR gates and a control input. When the control input is 0, the circuit performs addition. When the control input is 1, the circuit performs subtraction by taking the 2's complement of one operand (inverting the bits and adding 1). The operation is selected based on the control signal.

Program:

```

1 module rca_add_sub(A, B, CTRL, S, Cout);
2 input [3:0] A, B;
3 input CTRL;
4 output [3:0] S;
5 output Cout;
6 wire [3:0] B_mod;
7 wire c1, c2, c3;
8 assign B_mod = B ^ {4{CTRL}};
9 full_adder fa0(A[0], B_mod[0], CTRL, S[0], c1);
10 full_adder fa1(A[1], B_mod[1], c1, S[1], c2);
11 full_adder fa2(A[2], B_mod[2], c2, S[2], c3);
12 full_adder fa3(A[3], B_mod[3], c3, S[3], Cout);
13 endmodule
14
15 module full_adder(x, y, cin, sum, cout);
16 input x, y, cin;
17 output sum, cout;
18 assign sum = x ^ y ^ cin;
19 assign cout = (x & y) | (cin & (x ^ y));
20 endmodule
21
22 module tb_rca_add_sub;
23 reg [3:0] A, B;
24 reg CTRL;
25 wire [3:0] S;
26 wire Cout;
27 rca_add_sub uut(A, B, CTRL, S, Cout);
28 initial
29 begin
30 $monitor("CTRL=%b A=%b B=%b -> S=%b Cout=%b", CTRL, A, B, S, Cout);
31 CTRL=0; A=4'b0001; B=4'b0000; #10;
32 CTRL=0; A=4'b0010; B=4'b0100; #10;
33 CTRL=0; A=4'b1011; B=4'b0110; #10;
34 CTRL=1; A=4'b0001; B=4'b0000; #10;
35 CTRL=1; A=4'b0010; B=4'b0100; #10;
36 CTRL=1; A=4'b1011; B=4'b0110; #10;
37 CTRL=1; A=4'b0101; B=4'b0011; #10;
38 $stop;
39 end
40 endmodule

```



Truth Table:

| CTRL | A    | B    | S    | Cout |
|------|------|------|------|------|
| 0    | 0001 | 0000 | 0001 | 0    |
| 0    | 0010 | 0100 | 0110 | 0    |
| 0    | 1011 | 0110 | 1101 | 0    |
| 1    | 0001 | 0000 | 1111 | 1    |
| 1    | 0010 | 0100 | 1010 | 1    |
| 1    | 1011 | 0110 | 0000 | 1    |
| 1    | 0101 | 0011 | 1000 | 1    |

```

0      0001  0000  0001  0
0      0010  0100  0110  0
0      1011  0110  0001  1
1      0001  0000  0001  1
1      0010  0100  1110  0
1      1011  0110  0101  1
1      0101  0011  0010  1

```

## 10) SR FF

Theory: An SR flip-flop is a bistable device used to store a single bit, with two inputs: Set (S) and Reset (R). Setting S=1, R=0 stores 1; S=0, R=1 stores 0; both inputs low hold the previous state. S=1, R=1 leads to an invalid state

Program:

```

1  module srff (r, s, clk, rst, q, qbar);
2  input r, s, clk, rst;
3  output q, qbar;
4  reg q;
5  assign qbar = ~q;
6
7  always @ (posedge clk or posedge rst)
8  begin
9    if (rst)
10      q <= 1'b0;
11    else
12    begin
13      case ({r, s})
14        2'b00: q <= q;
15        2'b01: q <= 1'b1;
16        2'b10: q <= 1'b0;
17        2'b11: q <= 1'bz;
18      endcase
19    end
20  end
21 endmodule
22
23 module srff_test;
24 reg r, s, clk, rst;
25 wire q, qbar;
26
27 srff sr1(r, s, clk, rst, q, qbar);
28
29 always #5 clk = ~clk;
30
31 initial
32 begin
33   r = 0; s = 0; clk = 0; rst = 1;
34   #10 rst = 0;
35   #10 r = 0; s = 1;
36   #10 r = 1; s = 0;
37   #10 r = 1; s = 1;
38   #50 $stop;
39 end
40 endmodule
41

```



Truth Table:

| S | R | Q         |
|---|---|-----------|
| 0 | 0 | No change |
| 0 | 1 | 0         |
| 1 | 0 | 1         |
| 1 | 1 | X         |

## 11) JK FF

Theory: A JK flip-flop is a versatile memory circuit using inputs J and K. When J=1 and K=0, it sets; J=0 and K=1, it resets; both low hold the state; both high make the output toggle. It overcomes the invalid state and enables reliable sequential logic.

Program:

```

1 module jkff (j, k, clk, rst, q, qbar);
2 input j, k, clk, rst;
3 output q, qbar;
4 reg q;
5 assign qbar = ~q;
6
7 always @ (posedge clk or posedge rst)
8 begin
9   if (rst)
10     q <= 1'b0;
11   else
12     begin
13       case ({j, k})
14         2'b00: q <= q;
15         2'b01: q <= 1'b0;
16         2'b10: q <= 1'b1;
17         2'b11: q <= ~q;
18       endcase
19     end
20   end
21 endmodule
22
23
24 module jkff_test;
25 reg j, k, clk, rst;
26 wire q, qbar;
27 jkff jk(j, k, clk, rst, q, qbar);
28
29 always #5 clk = ~clk;
30
31 initial
32 begin
33   j = 0; k = 0; clk = 0; rst = 1;
34   #10 rst = 0;
35   #10 j = 0; k = 1;
36   #10 j = 1; k = 0;
37   #10 j = 1; k = 1;
38   #50 $stop;
39 end
40 endmodule

```



Truth Table:

| J | K | Q         |
|---|---|-----------|
| 0 | 0 | No change |
| 0 | 1 | 0         |
| 1 | 0 | 1         |
| 1 | 1 | Toggle    |

## 12) D FF

Theory: A D flip-flop is a memory device with a data input (D) and a clock input. On each clock pulse, it stores the value from D at the output Q. The Q output follows the D input only when triggered by the clock, ensuring stable, synchronized data storage

Program:

```
1 module dff (d, clk, rst, q, qbar);
2 input d, clk, rst;
3 output q, qbar;
4 reg q;
5 assign qbar = ~q;
6
7 always @ (posedge clk or posedge rst)
8 begin
9     if (rst)
10         q <= 1'b0;
11     else
12         q <= d;
13 end
14 endmodule
15
16
17 module dff_test;
18 reg d, clk, rst;
19 wire q, qbar;
20 dff d1(d, clk, rst, q, qbar);
21
22 always #5 clk = ~clk;
23
24 initial
25 begin
26     d = 1; clk = 0; rst = 1;
27     #10 rst = 0;
28     #10 d = 0;
29     #10 d = 1;
30     #50 $stop;
31 end
32 endmodule
```



Truth Table:

| D | Q |
|---|---|
| 0 | 0 |
| 1 | 1 |

### 13) T FF

Theory: A T flip-flop is a bistable circuit used for toggling output. With each clock pulse, if the input T is high, the output Q switches its state from 0 to 1 or from 1 to 0. If T is low, the output holds its state, enabling frequency division and counters.

Program:

```

1 module tff(t, clk, rst, q, qbar);
2 input t, clk, rst;
3 output q, qbar;
4 reg q;
5 assign qbar = ~q;
6
7 always @ (posedge clk or posedge rst)
8 begin
9     if (rst)
10         q <= 1'b0;
11     else if (t)
12         q <= ~q;
13     else
14         q <= q;
15 end
16 endmodule
17
18
19 module tff_test;
20 reg t, clk, rst;
21 wire q, qbar;
22 tff t1(t, clk, rst, q, qbar);
23
24 always #5 clk = ~clk;
25
26 initial
27 begin
28     t = 1; clk = 0; rst = 1;
29     #10 rst = 0;
30     #10 t = 0;
31     #10 t = 1;
32     #50 $stop;
33 end
34 endmodule
35 |

```



Truth Table:

| T | Q         |
|---|-----------|
| 0 | No change |
| 1 | Toggle    |