

# EE6205: Hardware Description Language (C-18)

## End-Semester Examination - July 2025

### Model Answers

#### Question 1

##### a) Truth table and circuit diagram

The given RTL model is:

```
1 module unknown1 (input wire a, input wire b, output wire c);
2 assign c = (a & b) | (~a & ~b);
3 endmodule
4
```

This expression corresponds to the logical XNOR operation ( $c = \overline{a \oplus b}$ ).

**Truth Table:**

| a | b | c |
|---|---|---|
| 0 | 0 | 1 |
| 0 | 1 | 0 |
| 1 | 0 | 0 |
| 1 | 1 | 1 |

**Circuit Diagram:**

##### b) Modified module code

###### i) Behavioral modeling

```
1 module unknown1_bhv (
2     input wire a,
3     input wire b,
4     output reg c  );
5
6     always @ (a or b) begin
7         if (a == b)   c = 1'b1;
8         else          c = 1'b0;
9     end
10
11    endmodule
12
```

Listing 1: Behavioral model for unknown1

## ii) Gate-level modeling

```
1 module unknown1_gate (
2     input wire a,
3     input wire b,
4     output wire c
5 );
6     wire not_a, not_b, term1, term2;
7
8     // Gate instantiations
9     not inv_a (not_a, a);
10    not inv_b (not_b, b);
11    and and_1 (term1, not_a, not_b);
12    and and_2 (term2, a, b);
13    or  or_1   (c, term1, term2);
14
15 endmodule
16
```

Listing 2: Gate-level model for unknown1

## c) Testbench analysis

### i) Console display

```
a=0, b=0, c=1
a=0, b=1, c=0
a=1, b=1, c=1
a=1, b=0, c=0
a=0, b=0, c=1
```

### ii) Waveform output

## Question 2

a) ‘initial’ vs ‘always’ and ‘wire’ vs ‘reg’

i) ‘initial’ block vs ‘always’ block

‘initial’ block executes exactly once at the beginning of a simulation (at time 0). Typically used for initialization, stimulus generation in testbenches, and defining single-shot behaviors. Multiple ‘initial’ blocks in a module execute concurrently.

‘always’ block executes throughout the simulation, whenever triggered by events in the sensitivity list. For example, ‘always @(posedge clk)’ executes on every positive clock edge, while ‘always @(\*)’ executes whenever any of the variables read inside the block change. ‘always’ blocks are used to model both computational and sequential logic that is always active.

```
1  module example;
2    reg clk, reset;
3
4    initial begin      // initial block for stimulus (runs once)
5      clk = 0;
6      reset = 1;
7      #10 reset = 0;
8    end
9
10   // always block for clock generation (runs forever)
11   always #5 clk = ~clk;
12
13 endmodule
14
```

Listing 3: Example of initial and always blocks

ii) ‘wire’ vs ‘reg’

A ‘wire’ is a net data type that represents a physical wire in a circuit. A ‘wire’ cannot store a value on its own; it must be continuously driven by something, like the output of a gate or an ‘assign’ statement. It is used to model combinational logic connections.

A ‘reg’, is a variable data type that can store a value between assignments. A ‘reg’ is required for any variable that is a target of an assignment within a procedural block (i.e., ‘initial’ or ‘always’ blocks). It can be used to model both combinational and sequential logic.

```
1  module example2(input a, b, clk, output y_comb, output reg y_seq);
2
3    wire w1; // wire for combinational connection
4
5    // 'wire' is driven by a continuous assignment
6    assign w1 = a & b;
7    assign y_comb = ~w1;
8
9    // 'reg' is used for the output of a procedural block
10   always @ (posedge clk) begin
11     y_seq <= a | b; // y_seq must be a reg
12   end
13
14 endmodule
15
```

Listing 4: Example of wire and reg

## b) Synthesis issues and resolution

The given code snippet is:

```
1  always @ (posedge clk or in1 or in2) begin
2      if (in1) q <= 1'b1;
3      else if (in2) q <= 1'b0;
4  end
5
```

This code has two main synthesis issues:

- Inferred Latch:** The ‘if-else if’ structure does not specify what happens to ‘q‘ when neither ‘in1‘ nor ‘in2‘ is true or both ‘in1‘ and ‘in2‘ are true. In a combinational ‘always‘ block, this would imply that ‘q‘ should hold its previous value, leading to the synthesis of an unintended latch. In this mixed-signal block, the behavior is ambiguous but problematic.
- Mixed-signal sensitivity list:** The list ‘posedge clk or in1 or in2‘ mixes a clock edge trigger with level-sensitive signals. Synthesis tools generally do not support this for creating standard synchronous logic. It is interpreted as a flip-flop that is sensitive to the clock edge AND asynchronously sensitive to changes in ‘in1‘ and ‘in2‘, which is not standard hardware.

**Resolved Code (assuming synchronous logic with active-high enables):** The most likely intention is a synchronous flip-flop where ‘in1‘ and ‘in2‘ act as prioritized enables.

```
1 // Assuming a D-flipflop with prioritized synchronous load signals
2 always @ (posedge clk) begin
3     if (in1) begin
4         q <= 1'b1;
5     end else if (in2) begin
6         q <= 1'b0;
7     end
8     // Note: To avoid an inferred latch, you might want an else case
9     // else q <= q; // Explicitly hold value (optional for FFs)
10    end
11
```

Listing 5: Corrected synchronous logic

## c) 8-bit Full Adder

### i) fulladder8b() by modifying fulladder1b()

```
1 module fulladder_8b (
2     input wire [7:0] a,
3     input wire [7:0] b,
4     input wire       cin, // input carry bit
5     output wire [7:0] s,  // results
6     output wire       cout // resulting carry bit
7 );
8
9     // Verilog handles vector arithmetic directly
10    assign {cout, s} = a + b + cin;
11
12 endmodule
13
```

Listing 6: 8-bit full adder using vector arithmetic

ii) fulladder8bm() by instantiating fulladder1b()

```
1  module fulladder_8b_m (
2    input wire [7:0] a,
3    input wire [7:0] b,
4    input wire      cin,
5    output wire [7:0] s,
6    output wire      cout
7  );
8
9  // Internal wire to chain the carry bits
10 wire [7:0] carry_chain;
11
12 // Instantiate 8 1-bit full adders
13 fulladder_1b fa0 (.a(a[0]), .b(b[0]), .cin(cin),           .s(s[0]), .cout(
14 carry_chain[0]));
15 fulladder_1b fa1 (.a(a[1]), .b(b[1]), .cin(carry_chain[0]), .s(s[1]), .cout(
16 carry_chain[1]));
17 fulladder_1b fa2 (.a(a[2]), .b(b[2]), .cin(carry_chain[1]), .s(s[2]), .cout(
18 carry_chain[2]));
19 fulladder_1b fa3 (.a(a[3]), .b(b[3]), .cin(carry_chain[2]), .s(s[3]), .cout(
20 carry_chain[3]));
21 fulladder_1b fa4 (.a(a[4]), .b(b[4]), .cin(carry_chain[3]), .s(s[4]), .cout(
22 carry_chain[4]));
23 fulladder_1b fa5 (.a(a[5]), .b(b[5]), .cin(carry_chain[4]), .s(s[5]), .cout(
24 carry_chain[5]));
25 fulladder_1b fa6 (.a(a[6]), .b(b[6]), .cin(carry_chain[5]), .s(s[6]), .cout(
26 carry_chain[6]));
27 fulladder_1b fa7 (.a(a[7]), .b(b[7]), .cin(carry_chain[6]), .s(s[7]), .cout(
28 cout)); // Final cout
29
30 endmodule
31
32 // The 1-bit full adder module is assumed to be available
33 module fulladder_1b (
34   input wire a, b, cin,
35   output wire s, cout
36 );
37   assign {cout, s} = a + b + cin;
38 endmodule
```

Listing 7: 8-bit full adder using 1-bit instances

## Question 3

### a) D-Flipflop analysis

#### i) D-Flipflop function and timing diagrams

The module implements a positive-edge triggered D-Flipflop with a synchronous, active-high reset ('reset') and a synchronous, active-high enable ('en'). The reset has higher priority than the enable.

- On the positive edge of 'clk', if 'reset' is high, the output 'q' is synchronously reset to 0.
- If 'reset' is low, and 'en' is high, 'q' takes the value of the input 'd'.
- If both 'reset' and 'en' are low, 'q' holds its previous value.

#### Timing Diagram:



#### ii) Testbench tbdff()

```
1  `timescale 1ns/10ps
2
3  module tb_dff();
4    // Inputs to DUT
5    reg d, clk, en, reset;
6    // Output from DUT
7    wire q;
8
9    // Clock period definition
10   parameter CLK_PERIOD = 10;
11
12  // Instantiate the Device Under Test (DUT)
13  dflipflop uut (
14    .d(d), .clk(clk), .en(en), .reset(reset), .q(q)
15  );
16
17  // Clock generation block
18  initial clk = 1'b0;
19  always #(CLK_PERIOD/2) clk = ~clk;
20
21  // Stimulus generation block
22  initial begin
23    // 1. Initialize and test reset
24    $display("T=%0t: Initializing signals", $time);
25    reset = 1'b1; en = 1'b0; d = 1'b0;
```

```

26 //20; // Wait for a couple of clock cycles
27
28 // 2. De-assert reset, test enable=0 (hold state)
29 $display("T=%0t: De-assert reset, en=0", $time);
30 reset = 1'b0;
31 d = 1'b1;
32 #20;
33
34 // 3. Test enable=1 (data propagation)
35 $display("T=%0t: Assert enable, en=1", $time);
36 en = 1'b1;
37 #10;
38 d = 1'b0;
39 #10;
40 d = 1'b1;
41 #10;
42
43 // 4. Test reset priority over enable
44 $display("T=%0t: Assert reset again", $time);
45 reset = 1'b1;
46 #20;
47
48 $display("T=%0t: Test finished", $time);
49 $finish;
50 end
51
52 // Monitor changes
53 initial begin
54 $monitor("T=%0t | reset=%b en=%b d=%b | q=%b", $time, reset, en, d, q);
55 end
56
57 endmodule
58

```

Listing 8: Testbench for the D-Flipflop

### iii) Asynchronous reset modification

To make the reset asynchronous, it must be added to the sensitivity list as an edge-triggered event (e.g., ‘posedge reset’). The logic inside the ‘always‘ block remains the same.

```

1 always @ (posedge clk or posedge reset) begin
2   if (reset)
3     q <= 1'b0; // This now happens immediately on posedge reset
4   else if (en)
5     q <= d;
6   end
7

```

Listing 9: D-Flipflop with asynchronous reset

## b) LED blinker circuit (1 Hz)

To blink an LED every second from a 50 MHz clock, we need to count 50,000,000 clock cycles.

```

1 module led_blinker (
2   input wire clk_50mhz,
3   input wire reset,
4   output reg led_out
5 );
6   // Counter to divide the clock
7   // Needs to count up to 49,999,999 (50 million cycles)
8   // 2^26 is > 50,000,000, so we need a 26-bit counter.

```

```

9   reg [25:0] counter;
10
11 // The value to count up to
12 parameter MAX_COUNT = 50_000_000 - 1;
13
14 always @(posedge clk_50mhz or posedge reset) begin
15   if (reset) begin
16     counter <= 0;
17     led_out <= 0;
18   end else begin
19     if (counter == MAX_COUNT) begin
20       counter <= 0;
21       led_out <= ~led_out; // Toggle LED every second
22     end else begin
23       counter <= counter + 1;
24     end
25   end
26 end
27
28 endmodule

```

Listing 10: 1 Hz LED blinker from 50 MHz clock

### c) ‘bus()‘ module for Figure Q3(c)

The circuit diagram shows the following connections:

- P[5] = A[0]
- P[4] = A[1] & B[3]
- P[3] = A[3]
- P[2] = A[2] & B[2]
- P[1] = B[1]
- P[0] = B[0]

```

1 module bus (
2   input wire [3:0] A,
3   input wire [3:0] B,
4   output wire [5:0] P
5 );
6   // Using continuous assignments based on the diagram
7   assign P[0] = B[0];
8   assign P[1] = B[1];
9   assign P[2] = A[2] & B[2];
10  assign P[3] = A[3];
11  assign P[4] = A[1] & B[3];
12  assign P[5] = A[0];
13
14 endmodule
15

```

Listing 11: Module for the bus breakout circuit

## Question 4

### a) Timing delays and logic values

#### i) Propagation ( $t_{pd}$ ) and contamination ( $t_{cd}$ ) delay

- **Contamination Delay ( $t_{cd}$ ):** The minimum time from when an input changes until the output *begins* to change. Before  $t_{cd}$ , the output is guaranteed to hold its previous value.
- **Propagation Delay ( $t_{pd}$ ):** The maximum time from when an input changes until the output is guaranteed to have settled to its new, stable value.

#### ii) Circuit delay calculation

*Note: The question states  $t_{pd} = 60\text{ps}$  and  $t_{cd} = 100\text{ps}$ . This is physically impossible as  $t_{pd}$  must be greater than or equal to  $t_{cd}$ . We will assume a typo and use the more conventional values of  $t_{pd} = 100\text{ps}$  and  $t_{cd} = 60\text{ps}$ .*

Circuit with paths highlighted:



- **Critical Path (Longest Delay Path):** The path from inputs A or B through gates G1, G2, and G4 to the output Y. This path traverses 3 gates.
  - Total Propagation Delay ( $t_{pd}$ ) =  $3 \times t_{pd,gate} = 3 \times 100\text{ps} = 300\text{ps}$ .
- **Shortest Path (Shortest Delay Path):** The paths from C (through G2, G4) or D/E (through G3, G4). These paths traverse 2 gates.
  - Total Contamination Delay ( $t_{cd}$ ) =  $2 \times t_{cd,gate} = 2 \times 60\text{ps} = 120\text{ps}$ .

#### iii) ‘X’ and ‘Z’ values

- **X - The Illegal Value:** In simulation, ‘X’ represents an unknown or uninitialized logic value. At the circuit level, it represents a condition of contention, where two or more drivers are attempting to drive a single wire to opposite logic levels (e.g., one driver

outputs '1' and another outputs '0' onto the same wire). This can cause a short circuit and an indeterminate voltage level.

- **Z - The Floating Value:** Represents a high-impedance state. At the circuit level, this means a wire is not being actively driven to either '1' or '0'. It is effectively disconnected. This state is essential for implementing shared buses, where multiple devices can be connected to the same wire, but only one is allowed to drive the bus at any given time (using tri-state buffers).

## b) FPGA architecture

### i) Configurable Logic Block (CLB)

A Configurable Logic Block (CLB) is the fundamental logic resource in an FPGA. A typical CLB consists of a Look-Up Table (LUT), a D-Flipflop (DFF), and multiplexers (MUX) for routing.

#### Block Diagram:

#### Functionality:

- **Look-Up Table (LUT):** A small SRAM that can be programmed to implement any Boolean function of its inputs (typically 4 to 6 inputs). It provides the CLB's combinational logic capability.
- **D-Flipflop (DFF):** A storage element that registers the LUT's output on a clock edge. This allows the CLB to implement sequential logic.
- **Multiplexer (MUX):** Selects the output of the CLB. It can route either the direct combinational output from the LUT or the registered sequential output from the D-Flipflop.

## ii) Other essential FPGA building blocks

1. **Programmable Interconnect:** A network of routing channels and programmable switches that connect the CLBs, I/O blocks, and other elements together.
2. **I/O Blocks (IOBs):** Located at the periphery of the FPGA chip, these blocks connect the internal logic to the external pins of the device and can be configured for various voltage standards.
3. **Block RAM (BRAM):** Dedicated blocks of on-chip static memory, providing large and fast storage resources.
4. **DSP Slices:** Specialized blocks containing hardware multipliers and accumulators, designed to efficiently implement digital signal processing algorithms.

## Question 5

### a) State transition diagram for sequence ‘1101’ (Mealy, Overlapping)

The FSM detects the sequence ‘1101’. As a Mealy machine, the output depends on the current state and the current input. Overlap is allowed (e.g., in ‘1101101’, two sequences are detected).

#### States:

- **S0:** Idle/Reset state. No part of the sequence has been detected.
- **S1:** The last input was ‘1’.
- **S2:** The last two inputs were ‘11’.
- **S3:** The last three inputs were ‘110’.

## b) Verilog code for the FSM

```
1  module sequence_detector (
2    input wire clk,
3    input wire reset, // Active-high asynchronous reset
4    input wire in,
5    output reg out
6  );
7
8  // Define states using parameters
9  parameter S0 = 2'b00; // Idle state
10 parameter S1 = 2'b01; // Seen '1'
11 parameter S2 = 2'b10; // Seen '11'
12 parameter S3 = 2'b11; // Seen '110'
13
14 // State registers
15 reg [1:0] current_state, next_state;
16
17 // Sequential logic for state transitions
18 always @(posedge clk or posedge reset) begin
19   if (reset) begin
20     current_state <= S0;
21   end else begin
22     current_state <= next_state;
23   end
24 end
25
26 // Combinational logic for next state determination
27 always @(current_state or in) begin
28   case (current_state)
29   S0: next_state = in ? S1 : S0;
30   S1: next_state = in ? S2 : S0;
31   S2: next_state = in ? S2 : S3;
32   S3: next_state = in ? S1 : S0;
33   default: next_state = S0;
34   endcase
35 end
36
37 // Combinational logic for Mealy output
38 always @(current_state or in) begin
39   if ((current_state == S3) && (in == 1'b1)) begin
40     out = 1'b1;
41   end else begin
42     out = 1'b0;
43   end
44 end
45 // A more concise way for the output:
46 // assign out = (current_state == S3) && in;
47
48 endmodule
49
```

Listing 12: Verilog code for 1101 sequence detector