

# **VL405 Design for Testability**

Lab Assignment 3

## **Design and Fault Analysis using VHDL**

**Student Name :** Ayan Garg

**Roll Number :** B23484

## Contents

|                                                                                     |          |
|-------------------------------------------------------------------------------------|----------|
| <b>1 Objective</b>                                                                  | <b>2</b> |
| <b>2 Theory</b>                                                                     | <b>2</b> |
| 2.1 Stuck-At Fault Model . . . . .                                                  | 2        |
| 2.2 Test Pattern Generation . . . . .                                               | 2        |
| 2.3 Gate-Specific Fault Detection Requirements . . . . .                            | 3        |
| 2.3.1 Input $z$ Stuck-at Faults . . . . .                                           | 3        |
| 2.3.2 Input $w$ Stuck-at Faults . . . . .                                           | 4        |
| 2.3.3 Output $D_3$ Stuck-at Faults . . . . .                                        | 5        |
| 2.3.4 Verification of the $4 \rightarrow 16$ Decoder with Fault Injection . . . . . | 6        |
| <b>3 VHDL Implementation</b>                                                        | <b>6</b> |
| 3.1 $3 \times 8$ Decoder with Enable . . . . .                                      | 6        |
| 3.2 Top-level $4 \times 16$ Decoder with Fault Injection . . . . .                  | 6        |
| 3.3 MUX for Fault Injection . . . . .                                               | 8        |
| <b>4 Simulation Results</b>                                                         | <b>8</b> |
| <b>5 Conclusion</b>                                                                 | <b>9</b> |

## 1 Objective

The objective of this experiment is to design and implement even parity generator and checker circuits using VHDL, and to analyze their behavior under stuck-at fault conditions (SA0 and SA1). The experiment aims to verify the correctness of the parity-based error detection mechanism, study the limitations of fault detection, and understand the practical applications of parity codes in digital system reliability.

For this lab, we will assume that there are no stuck-at faults at the inputs of any gate.

## 2 Theory

### 2.1 Stuck-At Fault Model

The stuck-at fault model is the most widely used fault model in digital testing. It represents a logical failure in which a signal line (input, output, or internal net) of a circuit is assumed to be permanently fixed at a logic level, irrespective of the actual inputs applied to the circuit. Such faults generally occur due to manufacturing defects like short-circuits, open-circuits, or wear-out effects in integrated circuits.

- **Stuck-at-0 (SA0):** The line is permanently tied to logic ‘0’.
- **Stuck-at-1 (SA1):** The line is permanently tied to logic ‘1’.

This abstraction does not refer to a physical short to ground or VDD but instead to the logical behavior of the faulty circuit. For instance, if a line is modeled as SA0, then whenever the circuit attempts to drive that line to ‘1’, the actual response will still be ‘0’, leading to a logic error.

The stuck-at fault model is preferred because:

1. It simplifies fault analysis while still covering a wide range of practical defects.
2. Tests designed for single stuck-at faults often detect multiple other fault types.
3. It allows systematic test generation using Automatic Test Pattern Generation (ATPG).

### 2.2 Test Pattern Generation

To detect stuck-at faults in the 4-to-16 decoder, we must apply test vectors that **differentiate the fault-free circuit from the faulty circuit**. The general principle is:

- **SA0 detection:** Apply a test vector that forces the fault-free node/output to be logic ‘1’. If the faulty circuit produces a ‘0’ instead, the fault is detected.
- **SA1 detection:** Apply a test vector that forces the fault-free node/output to be logic ‘0’. If the faulty circuit produces a ‘1’ instead, the fault is detected.

For the given assignment, we consider faults on input lines  $w$ ,  $z$ , and output line  $D3$ . The following test vectors are sufficient to detect the respective stuck-at faults:

- **Input  $z$ :**

- $z$  stuck-at-0: Apply  $wxyz = 1101$ .  
Fault-free output:  $D_{13} = 1$ .  
Faulty output:  $D_{12} = 1$ .
- $z$  stuck-at-1: Apply  $wxyz = 1100$ .  
Fault-free output:  $D_{12} = 1$ .  
Faulty output:  $D_{13} = 1$ .

- **Input  $w$ :**

- $w$  stuck-at-0: Apply  $wxyz = 1100$ .  
Fault-free output:  $D_{12} = 1$ .  
Faulty output:  $D_4 = 1$ .
- $w$  stuck-at-1: Apply  $wxyz = 0100$ .  
Fault-free output:  $D_4 = 1$ .  
Faulty output:  $D_{12} = 1$ .

- **Output  $D3$ :**

- $D3$  stuck-at-0: Apply  $wxyz = 0011$ .  
Fault-free output:  $D_3 = 1$ .  
Faulty output:  $D_3 = 0$ .
- $D3$  stuck-at-1: Apply  $wxyz = 0000$ .  
Fault-free output:  $D_0 = 1, D_3 = 0$ .  
Faulty output:  $D_0 = 1, D_3 = 1$ .

### Summary of Test Vectors:

Table 1: Test Vectors for Stuck-at Fault Detection

| Fault site | Test Vector (wxyz) | Fault-free output  | Faulty output      |
|------------|--------------------|--------------------|--------------------|
| $z$ SA0    | 1101               | $D_{13}$           | $D_{12}$           |
| $z$ SA1    | 1100               | $D_{12}$           | $D_{13}$           |
| $w$ SA0    | 1100               | $D_{12}$           | $D_4$              |
| $w$ SA1    | 0100               | $D_4$              | $D_{12}$           |
| $D3$ SA0   | 0011               | $D_3 = 1$          | $D_3 = 0$          |
| $D3$ SA1   | 0000               | $D_0 = 1, D_3 = 0$ | $D_0 = 1, D_3 = 1$ |

## 2.3 Gate-Specific Fault Detection Requirements

### 2.3.1 Input $z$ Stuck-at Faults

The input  $z$  drives both  $3 \rightarrow 8$  decoders in the hierarchical  $4 \rightarrow 16$  design. If  $z$  is stuck, the effective decoded index shifts by  $\pm 1$  within each active half of the decoder.

- **SA0 at  $z$ :** Whenever  $z = 1$ , the circuit instead behaves as if  $z = 0$ . Thus, the selected output line is shifted down by one. Example: For  $wxyz = 1101$  the correct output is  $D_{13}$ , but under SA0 on  $z$  the output becomes  $D_{12}$ .

- **SA1 at  $z$ :** Whenever  $z = 0$ , the circuit behaves as if  $z = 1$ . Thus, the selected output line is shifted up by one. Example: For  $wxyz = 1100$  the correct output is  $D_{12}$ , but under SA1 on  $z$  the output becomes  $D_{13}$ .

Table 2: Example Test Vectors for Stuck-at Fault Detection in 4→16 Decoder

| Fault     | Test Vector (wxyz) | Faulty vs. Fault-Free Output         |
|-----------|--------------------|--------------------------------------|
| $z$ SA0   | 1101               | $D_{13} \rightarrow D_{12}$          |
| $z$ SA1   | 1100               | $D_{12} \rightarrow D_{13}$          |
| $w$ SA0   | 1100               | $D_{12} \rightarrow D_4$             |
| $w$ SA1   | 0100               | $D_4 \rightarrow D_{12}$             |
| $D_3$ SA0 | 0011               | $D_3 = 1 \rightarrow$ None (all low) |
| $D_3$ SA1 | 0000               | $D_0 = 1 \rightarrow (D_0, D_3)$     |

### 2.3.2 Input $w$ Stuck-at Faults

The input  $w$  selects between the lower ( $D_0$ – $D_7$ ) and upper ( $D_8$ – $D_{15}$ ) halves. If  $w$  is stuck, the activation swaps between halves, shifting the decoded index by  $\pm 8$ .

- **SA0 at  $w$ :** When  $w = 1$ , the circuit instead behaves as if  $w = 0$ . The intended high line from  $D_8$ – $D_{15}$  is replaced by its alias in  $D_0$ – $D_7$ . Example: For  $wxyz = 1100$ , the correct output is  $D_{12}$ , but under SA0 on  $w$  the output becomes  $D_4$ .
- **SA1 at  $w$ :** When  $w = 0$ , the circuit instead behaves as if  $w = 1$ . The intended high line from  $D_0$ – $D_7$  is replaced by its alias in  $D_8$ – $D_{15}$ . Example: For  $wxyz = 0100$ , the correct output is  $D_4$ , but under SA1 on  $w$  the output becomes  $D_{12}$ .

Table 3: Test Vectors for  $w$  Stuck-at-0 (Global)

| Input (wxyz) | Fault-Free Output | Faulty Output ( $w=0$ ) |
|--------------|-------------------|-------------------------|
| 1000         | $D_8$             | $D_0$                   |
| 1001         | $D_9$             | $D_1$                   |
| 1010         | $D_{10}$          | $D_2$                   |
| 1011         | $D_{11}$          | $D_3$                   |
| 1100         | $D_{12}$          | $D_4$                   |
| 1101         | $D_{13}$          | $D_5$                   |
| 1110         | $D_{14}$          | $D_6$                   |
| 1111         | $D_{15}$          | $D_7$                   |

Table 4: Test Vectors for  $w$  Stuck-at-1 (Global)

| Input (wxyz) | Fault-Free Output | Faulty Output (w=1) |
|--------------|-------------------|---------------------|
| 0000         | $D_0$             | $D_8$               |
| 0001         | $D_1$             | $D_9$               |
| 0010         | $D_2$             | $D_{10}$            |
| 0011         | $D_3$             | $D_{11}$            |
| 0100         | $D_4$             | $D_{12}$            |
| 0101         | $D_5$             | $D_{13}$            |
| 0110         | $D_6$             | $D_{14}$            |
| 0111         | $D_7$             | $D_{15}$            |

### 2.3.3 Output $D_3$ Stuck-at Faults

The output line  $D_3$  should be active only for input  $wxyz = 0011$ . Stuck-at faults on  $D_3$  either suppress it or cause it to spuriously assert alongside other outputs.

- **SA0 at  $D_3$ :** When  $wxyz = 0011$ , no line is activated (the one-hot property is lost).
- **SA1 at  $D_3$ :**  $D_3$  is always high. For input 0011, behavior is correct; for all others,  $D_3$  is erroneously high along with the correct line.

Table 5: Test Vectors for  $D_3$  Stuck-at-0

| Input (wxyz) | Fault-Free Output | Faulty Output  |
|--------------|-------------------|----------------|
| 0011         | $D_3$             | None (all low) |

Table 6: Test Vectors for  $D_3$  Stuck-at-1

| Input (wxyz) | Fault-Free Output | Faulty Output |
|--------------|-------------------|---------------|
| 0000         | $D_0$             | $D_0, D_3$    |
| 0001         | $D_1$             | $D_1, D_3$    |
| 0010         | $D_2$             | $D_2, D_3$    |
| 0011         | $D_3$             | $D_3$         |
| 0100         | $D_4$             | $D_4, D_3$    |
| 0101         | $D_5$             | $D_5, D_3$    |
| 0110         | $D_6$             | $D_6, D_3$    |
| 0111         | $D_7$             | $D_7, D_3$    |
| 1000         | $D_8$             | $D_8, D_3$    |
| 1001         | $D_9$             | $D_9, D_3$    |
| 1010         | $D_{10}$          | $D_{10}, D_3$ |
| 1011         | $D_{11}$          | $D_{11}, D_3$ |
| 1100         | $D_{12}$          | $D_{12}, D_3$ |
| 1101         | $D_{13}$          | $D_{13}, D_3$ |
| 1110         | $D_{14}$          | $D_{14}, D_3$ |
| 1111         | $D_{15}$          | $D_{15}, D_3$ |

### 2.3.4 Verification of the 4→16 Decoder with Fault Injection

The design implements a hierarchical 4→16 decoder with fault injection on input lines ( $z$ ,  $w$ ) and output ( $D3$ ). The VHDL architecture allows selection of fault sites using control signals, and the testbench exhaustively applies all 16 input combinations under fault-free and faulty modes.

- In normal mode, exactly one line among D0–D15 is active (one-hot).
- Under input faults ( $z$ ,  $w$ ), aliasing effects occur:  $\pm 1$  shifts (for  $z$ ) or  $\pm 8$  shifts (for  $w$ ).
- Under output fault ( $D3$ ), the one-hot property is broken: either missing activation (SA0) or double activation (SA1).

By comparing the observed output with the expected truth table, all injected faults are detectable. This methodology ensures comprehensive verification of the decoder's fault behavior and validates the fault models described.

## 3 VHDL Implementation

### 3.1 3×8 Decoder with Enable

```

1 library ieee;
2 use ieee.std_logic_1164.all;
3
4 entity decoder_3x8 is
5   port (
6     Ctrl1, Ctrl2, Ctrl3 : in std_logic;
7     Enable : in std_logic;
8     Y : out std_logic_vector (7 downto 0)
9   );
10 end entity decoder_3x8;
11
12 architecture behavioral of decoder_3x8 is
13   signal sel : std_logic_vector(2 downto 0);
14 begin
15   sel <= Ctrl3 & Ctrl2 & Ctrl1;
16
17   process(sel, Enable)
18   begin
19     if Enable = '1' then
20       case sel is
21         when "000" => Y <= "00000001";
22         when "001" => Y <= "00000010";
23         when "010" => Y <= "00000100";
24         when "011" => Y <= "00001000";
25         when "100" => Y <= "00010000";
26         when "101" => Y <= "00100000";
27         when "110" => Y <= "01000000";
28         when "111" => Y <= "10000000";
29         when others => Y <= (others => '0');
30       end case;
31     else
32       Y <= (others => '0');
33     end if;
34   end process;
35 end architecture behavioral;

```

Listing 1: 3-to-8 Decoder with Enable

### 3.2 Top-level 4×16 Decoder with Fault Injection

```

1 library IEEE;
2 use IEEE.STD_LOGIC_1164.ALL;
3
4 entity test is
5   port(
6     sao : in std_logic; -- output stuck-at enable
7     CtrlA : in std_logic_vector(1 downto 0); -- input fault selector
8     CtrlB : in std_logic_vector(2 downto 0); -- fault type

```

```

9      Ctrl0 : in std_logic_vector(4 downto 0); -- output fault control
10
11     x, y, z, w : in std_logic;           -- inputs
12     OutA : out std_logic_vector (15 downto 0) -- outputs
13   );
14 end test;
15
16 architecture Structural of test is
17   signal fx1, fx2, fy1, fy2, fz1, fz2, fw1, fw2 : std_logic;
18   signal fx, fy, fz, fw : std_logic;
19   signal nfw1 : std_logic;
20
21   signal mux_out : std_logic_vector(1 downto 0);
22   signal mux_in_sig : std_logic;
23
24   signal dec_low, dec_high : std_logic_vector(7 downto 0);
25   signal out_int : std_logic_vector(15 downto 0);
26 begin
27
28   ----- Fault input selector -----
29
30   process(x, y, z, w, CtrlA)
31   begin
32     case CtrlA is
33       when "00" => mux_in_sig <= x; fy <= y; fz <= z; fw <= w;
34       when "01" => mux_in_sig <= y; fx <= x; fz <= z; fw <= w;
35       when "10" => mux_in_sig <= z; fx <= x; fy <= y; fw <= w;
36       when others => mux_in_sig <= w; fx <= x; fy <= y; fz <= z;
37     end case;
38   end process;
39
40   uut_mux: entity work.mux_special
41     port map(in_sig => mux_in_sig, sel => CtrlB, out_sig => mux_out);
42
43   -- Assign mux output to faulted inputs
44   process(CtrlA, mux_out, x, y, z, w)
45   begin
46     fx1 <= x; fx2 <= x;
47     fy1 <= y; fy2 <= y;
48     fz1 <= z; fz2 <= z;
49     fw1 <= w; fw2 <= w;
50
51     case CtrlA is
52       when "00" => fx1 <= mux_out(0); fx2 <= mux_out(1);
53       when "01" => fy1 <= mux_out(0); fy2 <= mux_out(1);
54       when "10" => fz1 <= mux_out(0); fz2 <= mux_out(1);
55       when others => fw1 <= mux_out(0); fw2 <= mux_out(1);
56     end case;
57   end process;
58
59   nfw1 <= not fw1;
60
61   ----- 4x16 Decoder using two 3x8 decoders -----
62
63   dec_low_inst: entity work.decoder_3x8
64     port map(Ctrl1 => fx1, Ctrl2 => fy1, Ctrl3 => fz1, Enable => nfw1, Y => dec_low);
65
66   dec_high_inst: entity work.decoder_3x8
67     port map(Ctrl1 => fx2, Ctrl2 => fy2, Ctrl3 => fz2, Enable => fw2, Y => dec_high);
68
69   out_int <= dec_low & dec_high;
70
71
72   -- Output stuck-at fault injection
73
74   process(out_int, sao, Ctrl0)
75   variable temp : std_logic_vector(15 downto 0);
76   begin
77
78     temp := out_int;
79     if sao = '1' then
80       case Ctrl0(4 downto 1) is
81         when "0000" => temp(0) := Ctrl0(0);
82         when "0001" => temp(1) := Ctrl0(0);
83         when "0010" => temp(2) := Ctrl0(0);
84         when "0011" => temp(3) := Ctrl0(0);
85         when "0100" => temp(4) := Ctrl0(0);
86         when "0101" => temp(5) := Ctrl0(0);
87         when "0110" => temp(6) := Ctrl0(0);
88         when "0111" => temp(7) := Ctrl0(0);
89         when "1000" => temp(8) := Ctrl0(0);
90         when "1001" => temp(9) := Ctrl0(0);
91         when "1010" => temp(10) := Ctrl0(0);
92         when "1011" => temp(11) := Ctrl0(0);
93         when "1100" => temp(12) := Ctrl0(0);
94         when "1101" => temp(13) := Ctrl0(0);
95         when "1110" => temp(14) := Ctrl0(0);
96         when "1111" => temp(15) := Ctrl0(0);
97         when others => null;
98       end case;
99     end if;
100    OutA <= temp;
101  end process;
102 end Structural;

```

Listing 2: 4-to-16 Decoder with Fault Injection

### 3.3 MUX for Fault Injection

```

1 library IEEE;
2 use IEEE.STD_LOGIC_1164.ALL;
3
4 entity mux_special is
5 Port (
6     in_sig : in std_logic;
7     sel    : in std_logic_vector(2 downto 0);
8     out_sig : out std_logic_vector(1 downto 0)
9 );
10 end mux_special;
11
12 architecture Behavioral of mux_special is
13 begin
14     process(in_sig, sel)
15     begin
16         case sel is
17             when "000" => out_sig <= in_sig & in_sig; -- pass-through
18             when "001" => out_sig <= in_sig & '0'; -- lower=in, upper=0
19             when "010" => out_sig <= '0' & in_sig; -- lower=0, upper=in
20             when "011" => out_sig <= "00"; -- both stuck-0
21             when "100" => out_sig <= in_sig & '1'; -- lower=in, upper=1
22             when "101" => out_sig <= '1' & in_sig; -- lower=1, upper=in
23             when "110" => out_sig <= "11"; -- both stuck-1
24             when others => out_sig <= in_sig & in_sig;
25         end case;
26     end process;
27 end Behavioral;

```

Listing 3: MUX for Fault Injection

## 4 Simulation Results



Figure 1: Stuck at fault at input Z



Figure 2: Stuck at fault at input W



Figure 3: Stuck at fault at input D3

## 5 Conclusion

In this experiment, a 4-to-16 line decoder was successfully designed and verified using VHDL. The design was implemented hierarchically with two 3-to-8 decoders controlled by the most significant input bit, thereby reducing complexity and improving modularity.

In addition to the normal operation, fault injection capability was introduced through the stuck-at fault model. By integrating a special multiplexer and control signals, both input and output nodes could be forced into stuck-at-0 (SA0) or stuck-at-1 (SA1) conditions. This allowed the decoder to be tested under fault scenarios, demonstrating how manufacturing defects or physical errors may alter circuit behavior.

The truth tables and test vectors confirmed correct operation of the fault-free circuit as well as the ability of the design to simulate and detect faults. Therefore, this experiment not only reinforced the fundamentals of combinational logic design in VHDL but also highlighted the importance of Design-for-Testability (DFT) techniques in ensuring robust and reliable digital systems.