



# Aristotle University of Thessaloniki Computer Science Department

## Semester assignment in NNA-07-08

Digital Electronic Systems in VHDL

Alexandros Korkos alexkork@csd.auth.gr 3870



## Abstract

Digital design of logic circuits is a fundamental part of computer engineering. The design of logic circuits is done using Hardware Description Languages (HDLs). The most common HDLs are Verilog and VHDL. This project focuses on the design of logic circuits using VHDL. The project is divided into two parts. The first part focuses on the design of combinational circuits, while the second part focuses on the design of sequential circuits. All the circuits designed and implemented are fundamental circuits, that are used in the design of more complex circuits.

Besides the design of the circuits, the project also focuses on the simulation and validation of the circuits. All the above are done using free and open source software. The software used for the design and simulation of the circuits are GHDL and GTKWave (see Appendix B). The VHDL codes for the circuits are following the version 2008 of IEEE standard 1076.

## Contents

| Ι  | Sequential circuits                                                       | 4                                      |
|----|---------------------------------------------------------------------------|----------------------------------------|
| 1  | LED Counter                                                               | 4                                      |
| 2  | D flip-flop                                                               | 5                                      |
| 3  | JK flip-flop                                                              | 7                                      |
| 4  | Random Access Memory (RAM)                                                | 9                                      |
| II | Combinatorial circuits                                                    | 11                                     |
| 5  | Multiplexer                                                               | 11                                     |
| 6  | Full adder                                                                | 13                                     |
| 7  | Comparator                                                                | 16                                     |
| 8  | Sequential multiplier                                                     | 17                                     |
| 9  | Encoder                                                                   | 21                                     |
| A  | Supplementary code for LED counter & multiplier           A.1 LED counter | 24<br>24<br>25<br>25<br>25<br>26<br>27 |
| В  | Tools used for digital design and VHDL compilation                        | 29                                     |

## Part I

## Sequential circuits

A sequential circuit, consists of a combinational circuit to which storage elements are connected to form a feedback path. The storage elements are devices capable of storing binary information. The binary information stored in these elements at any given time defines the state of the sequential circuit at that time. The sequential circuit receives binary information from external inputs that, together with the present state of the storage elements, determine the binary value of the outputs.

## 1 LED Counter

The concept of a counter is quite simple: it is a circuit that counts. In this particular case, the counter is a 4-bit counter, which means that it can count from 0 to 15.

A LED counter is a combination of a LED decoder and a counter (see Appendix A.1). The LED Counter takes in the clock and reset inputs from the user and outputs the 7-bit value to be displayed on the LEDs. The LED Counter is a hierarchical design, where the LED Decoder and the Counter are instantiated as components in the LED Counter.

This design is hierarchical; both the LED decoder and the counter are autonomous units connected to each other through another higher-level unit to create the final circuit.

The VHDL code for the LED Counter is shown here listing 1. The LED Counter has a clock and reset input, which are used to control the counter. The counter value is then passed to the LED Decoder, which returns the 7-bit value to be displayed on the LEDs. The LED Counter also has a 4-bit output, which is the counter value.

```
library ieee;
use ieee.std_logic_1164.all;

entity led_counter is
port (
    clk, reset: in std_logic;
    Q_out: out std_logic_vector (6 downto 0)
);
end entity;

architecture behavior of led_counter is
signal counter_value : std_logic_vector (3 downto 0);
begin

LED: entity work.led_decoder port map (clk, counter_value, Q_out);
COUNT: entity work.counter port map (clk, reset, counter_value);
end architecture;
```

**Listing 1:** LED Counter

The test bench for the LED Counter is shown here listing 2. The test bench instantiates the LED Counter and provides the clock and reset inputs. The test bench also displays the counter value on the console. The test bench is run for 100ns, which is enough time for the counter to reach its maximum value and reset back to 0.

```
library ieee;
use ieee.std_logic_1164.all;
use std.env.finish;

entity tb_led_counter is
end entity;

architecture behavior of tb_led_counter is
constant clk_period: time := 10 ns;
```

```
signal clk: std_logic := '0';
        signal reset: std_logic := '0';
12
        signal led: std_logic_vector(6 downto 0) := "00000000";
14
      UUT: entity work.led_counter port map(clk, reset, led);
16
      clk_process : process
      begin
18
        clk ≤ '0';
19
        wait for clk_period/2;
20
        clk ≤ '1';
        wait for clk_period/2;
      end process;
24
      stimulus : process
25
26
        reset ≤ '1':
        wait for 100 ns;
28
        reset ≤ '0';
29
        wait for 100 ns;
                 11';
31
        reset ≤
        wait for 50 ns;
        reset ≤ '0';
33
        wait for 450 ns;
34
35
        finish;
36
      end process;
   end architecture;
37
```

Listing 2: Test bench for LED Counter

The waveform extracted form GTKWave is presented in fig. 1.



Figure 1: Waveform of LED Counter

#### D flip-flop 2

A D flip-flop, is a type of digital storage element used in digital circuits, particularly in sequential logic systems like registers and memories. The output of the D flip-flop follows the value of the D input when the appropriate synchronization pulses are applied.

In this implementation, the D flip-flop is triggered on the rising edge of the clock signal. Its operation is shown in the following truth table.

**Table 1:** D flip-flop truth table

| clk      | D | $\mathbf{Q_{t+1}}$ |
|----------|---|--------------------|
| <b>上</b> | 0 | 0                  |
| 工        | 1 | 1                  |
| other    | X | $Q_t$              |

In the implementation seen in listing 3 the negated Q is not configured, because it was not required by the instructor. Also, the listing 3 uses an asynchronous reset.

```
library ieee;
use ieee.std_logic_1164.all;
```

```
entity d_flip_flop is
     port (
        clk
                      : in std logic:
        rst_n, preset : in std_logic;
        D_in
                      : in std_logic;
8
                      : out std_logic
        Q_out
     );
10
    end entity;
    -- asunchronous reset
    architecture behavior of d_flip_flop is
14
    begin
15
     DUT: process(clk, rst_n, preset)
16
      begin
        if (rst_n = '0') then
18
          Q_out < '0';
19
        elsif (preset = '0') then
20
          Q_out ≤ '1';
        elsif (clk'event and clk = '1') then
22
          Q_out ≤ D_in;
24
        end if;
      end process;
25
   end architecture;
```

**Listing 3:** D flip-flop

For the testing of the circuit the test bench created as shown here listing 4 finishes in 90ns without any errors. Additionally, it uses the asserted instruction, to check instantly the value of the output. The clock period simulated is 50ns long, where each 25ns edge is rising/falling.

```
library ieee;
use ieee.std_logic_1164.all;
    use std.env.finish;
    entity tb_d_flip_flop is
    end entity;
    architecture bench of tb_d_flip_flop is
      signal D_tb_in: std_logic := '0';
      signal clk_tb: std_logic := '0';
10
      signal rst_n_tb: std_logic := '0';
      signal preset_tb: std_logic := '0';
12
      signal Q_tb_out: std_logic;
13
14
      constant clk_period : time := 50 ns;
15
16
      UUT: entity work.d_flip_flop port map (clk_tb, rst_n_tb, preset_tb, D_tb_in, Q_tb_out
18
      → );
19
      clk_process: process
20
21
      begin
        clk_tb ≤ '0';
        wait for clk_period/2;
24
        clk_tb ≤ '1';
        wait for clk_period/2;
25
      end process;
26
      stimulus : process
28
29
      begin
        rst_n_tb ≤ '0';
30
31
        wait for 30 ns;
        assert(Q_tb_out = '0') report "Error in q" severity failure;
32
33
        rst_n_tb
34
        preset_tb ≤ '1';
35
        D_tb_in
                 ≤ '0':
36
```

```
wait for 15 ns;
37
       assert(Q_tb_out = '0') report "Error in q" severity failure;
38
39
40
        rst_n_tb ≤ '1';
       preset_tb ≤ '1';
41
       wait for 15 ns;
42
       assert(Q_tb_out = '0') report "Error in q" severity failure;
44
        rst_n_tb ≤ '1';
       preset_tb ≤ '0';
       wait for 10 ns;
47
       assert(Q_tb_out = '1') report "Error in q" severity failure;
48
49
50
       rst_n_tb ≤ '1';
       preset_tb ≤ '1';
        wait for 20 ns;
52
       assert(Q_tb_out = '0') report "Error in q" severity failure;
55
       finish;
     end process:
56
   end architecture;
```

Listing 4: Testbench for D flip-flop

The waveform of this circuit simulation is shown in the fig. 2 following.



Figure 2: Waveform of D flip-flop

#### 3 JK flip-flop

A JK flip-flop is a type of digital storage element that has the ability to toggle its output between two states. It is a synchronous device, meaning that it only responds to input signals at transitions of a clock signal. The JK flip- flop is a refinement of the SR flip-flop, where S and R inputs are combined into a single input J (set) with the addition of a clock input and a second input K (reset). The output of the JK flip-flop toggles between two states according the inputs.

In this implementation, the JK flip-flop is triggered on the rising edge of the clock signal. Its operation is shown in the following truth table 2.

| clk      | J | K | $\mathbf{Q_{t+1}}$ |
|----------|---|---|--------------------|
|          | 0 | 0 | $Q_t$              |
| 壬        | 0 | 1 | 0                  |
| 壬        | 1 | 0 | 1                  |
| <u>_</u> | 1 | 1 | $\overline{Q_t}$   |
| other    | X | X | $Q_t$              |

**Table 2:** JK flip-flop truth table

In the implementation seen in listing 5 the negated Q output is like in the D flip-flop implementation, not configured. Also, the listing 5 uses an asynchronous reset.

```
library ieee;
use ieee.std_logic_1164.all;
entity jk_flip_flop is
```

```
port (
   5
                                      J_in: in std_logic;
                                      K_in: in std_logic;
                                      rst, clk: in std_logic;
                                      Q_out: inout std_logic
   9
                            );
10
                   end entity;
                   architecture behavior of jk_flip_flop is
13
14
                            DUT: process (rst, clk) is
15
                            begin
16
                                      if rst = '1' then
                                                Q_out ≤ '0';
18
                                      elsif(rising_edge(clk)) then
19
                                                if (J_in = '0' and K_in = '0') then
20
                                                          Q_out ≤ Q_out;
                                                 elsif (J_in = '0' and K_in = '1') then
                                                          Q_out ≤ '0';
23
                                                 elsif (J_in = '1' and K_in = '0') then
24
                                                  Q_{\text{out}} \leqslant \begin{subarray}{l} \begin{su
25
26
                                                          Q_out \leq not (Q_out);
27
28
                                                end if;
29
                                      end if:
                            end process;
 30
                  end behavior;
```

#### Listing 5: JK flip-flop

For the testing of this JK flip-flop, the test bench created as shown here listing 6 finishes in 100ns without any errors. Additionally, it uses the asserted instruction, to check instantly the value of the output. Here the clock period simulated is 10ns long, where each 5ns edge is rising/falling.

```
|library ieee;
   use ieee.std_logic_1164.all;
    use std.env.finish;
    entity tb_jk_flip_flop is
    end entity;
    architecture bench of tb_jk_flip_flop is
8
      signal J_tb_in: std_logic := '0';
      signal K_tb_in: std_logic := '0'
10
      signal rst_n_tb: std_logic := '0';
11
      signal clk_tb: std_logic := '0';
      signal Q_tb_out: std_logic;
14
      constant clk_period : time := 10 ns;
15
16
    begin
      UUT: entity work.jk_flip_flop port map (J_tb_in, K_tb_in, rst_n_tb, clk_tb, Q_tb_out
18
19
20
      clk_process : process
      begin
        clk_tb ≤ '0';
22
        wait for clk_period/2;
        clk_tb ≤ '1';
24
        wait for clk_period/2;
25
      end process;
26
28
      stimulus: process
     begin
29
        J_{tb_in} \leq '1';
30
        K_tb_in ≤ '0';
31
        rst_n_tb ≤ '0';
32
```

```
33
        wait for 20 ns;
        assert(Q_tb_out = '1') report "Error in q" severity failure;
34
35
        J_{tb_in} \leq '0';
        K_tb_in ≤ '0';
37
        rst_n_tb ≤ '0';
38
        wait for 20 ns;
39
        assert(Q_tb_out = '1') report "Error in q" severity failure;
40
        J_{tb_in} \leq 0;
42
        K_tb_in ≤ '1';
43
        rst_n_tb ≤ '0';
        wait for 20 ns;
45
        assert(Q_tb_out = '0') report "Error in q" severity failure;
47
        J_{tb_in} \leq '1';
        K_{tb_in} \leq '1'
        rst_n_tb ≤ '0';
        wait for 20 ns;
        assert(Q_tb_out = '0') report "Error in q" severity failure;
        J_{tb_in} \leq '1';
        K_tb_in ≤ '1';
55
        rst_n_tb ≤ '1';
56
        wait for 20 ns;
        assert(Q_tb_out = '0') report "Error in q" severity failure;
58
59
60
      end process;
61
    end architecture;
62
```

**Listing 6:** Testbench for JK flip-flop

The results of the simulation are shown in the fig. 3 following.



Figure 3: Waveform of JK flip-flop

For the first 5ns, the Q output is Undefined, because all the inputs are set to 0 and the clock is in a falling edge which sets the flip-flop to a no-change state (default value of std logic is 'U').

#### Random Access Memory (RAM) 4

There are two types of memories that are used in digital systems: random access memory (RAM) and read only memory (ROM). A RAM stores new information for later use. The process of storing new information into memory is referred to as a memory write operation. The process of transferring the stored information out of memory is referred to as a memory read operation. RAM can perform both write and read operations.

The RAM implemented, is a 16 to 4. This means, 16 addresses to store 4 bits long data. An enable signal is used, to control the RAM. If enable is true, the system is operational, if not, the system is in an idle state. Besides that, a rw (read = 0, write = 1) is expected, to determine if the system is in a read or write state. Additionally, a vector of an address is expected, to determine which address to read or write. For the data input and output, a vector of 4 bits of input is used, to combine in one. The sensitivity list only includes the clock signal.

```
library ieee;
   use ieee.std_logic_1164.all;
   use ieee.numeric_std.all;
   entity ram is
     port(
        clk, enable, rw: in std_logic;
        address: in std_logic_vector (3 downto 0);
        data: inout std_logic_vector (3 downto 0)
     );
10
    end ram;
    architecture behavior of ram is
      type ram_t is array (0 to 15) of std_logic_vector(3 downto 0);
      signal tmp_ram: ram_t;
16
      process (clk)
17
      begin
18
        if (clk = '1' and clk'event) then
19
          if enable = '1' then
20
            if rw = '1' then
              tmp_ram(to_integer(unsigned(address))) \le data;
22
            elsif rw = '0' then
              data ≤ tmp_ram(to_integer(unsigned(address)));
25
            else
              data \leq (data'range \Rightarrow 'Z');
26
            end if;
          end if:
28
        end if;
29
     end process;
30
   end architecture;
```

#### Listing 7: RAM

The test bench for the RAM is shown in listing 8. The clock period is 1ns and the raise and fall time is 0.25ns.

```
library ieee;
   use ieee.std_logic_1164.all;
   use std.env.finish;
    entity tb_ram is
    end tb_ram;
   architecture behavior of tb_ram is
      signal clk: std_logic;
      signal enable: std_logic;
10
      signal rw: std_logic;
      signal address: std_logic_vector (3 downto 0);
      signal data: std_logic_vector (3 downto 0);
14
      constant clk_period: time := 1 ns;
16
      RAMO: entity work.ram port map (clk, enable, rw, address, data);
18
      clock : process
19
      begin
        for i in 1 to 20 loop
20
          clk < '1';
21
          wait for clk_period/4;
          clk ≤ '0';
          wait for clk_period/4;
24
        end loop;
25
26
      finish;
      end process;
28
      process
29
      begin
30
```

```
wait for clk_period;
31
        enable ≤ '1';
32
        wait for clk_period;
33
        rw < '1';
34
        address < "0011";
data < "1001";
35
36
        wait for clk_period;
38
        address ≤ "0001";
        data ≤ "ZZZZ";
40
        wait for clk_period;
41
        address ≤ "0011";
42
        data ≤ "ZZZZ";
43
        wait for clk_period;
44
        wait;
45
      end process;
    end architecture;
```

Listing 8: Test bench for the RAM

The simulation of the RAM circuit, are presented with the following waveform image.



Figure 4: Waveform of RAM

## Part II

## Combinatorial circuits

## 5 Multiplexer

A multiplexer is a combinational circuit that selects binary information from one of many input lines and directs it to a single output line. The selection of a particular input line is controlled by a set of selection lines. Normally, there are  $2^n$  input lines and n selection lines whose bit combinations determine which input is selected.

Here the n value is 2, so this means that the MUX implemented is a 4 to 1 circuit. The truth table for the multiplexer is shown in table 3.

Table 3: Truth table for the 4 to 1 MUX

| $S_0$ | $\mathbf{S_1}$ | $\mathbf{Q_{out}}$ |
|-------|----------------|--------------------|
| 0     | 0              | A                  |
| 0     | 1              | В                  |
| 1     | 0              | С                  |
| 1     | 1              | D                  |

The circuit design, of a 4 to 1 MUX using digital logic gates, is shown in fig. 5.



Figure 5: Circuit design of MUX 4 to 1

The developed code seen in listing 9, uses a vector to store the input values, and then uses the selection lines, also a vector to select the output value. The sensitivity list includes the selection and the input vectors, that means if the values of those two vectors the output gets recalculated. A small but important detail is that the output gets the value Uninitialized ('X') as a default value in the case statement, this is to avoid latches.

```
library ieee;
    use ieee.std_logic_1164.all;
    entity mux_4to1 is
      port (
        ABCD_in : in std_logic_vector(3 downto 0);
        Sel_in : in std_logic_vector(1 downto 0);
        Q_out
                  : out std_logic
      );
    end entity;
10
    architecture behavior of mux_4to1 is
      DUT: process(ABCD_in, Sel_in)
14
      begin
16
        case Sel_in is
           when "00"
                        \Rightarrow Q_out \leqslant ABCD_in(0);
           when "01"
18
                         \Rightarrow Q_out \leq ABCD_in(1);
           when "10"
                         \Rightarrow Q_out \leq ABCD_in(2);
19
           when "11"
                        \Rightarrow Q_out \leq ABCD_in(3);
20
21
           when others \Rightarrow Q_out \leq 'X';
22
        end case;
      end process;
    end architecture;
```

#### **Listing 9:** MUX 4 to 1

The test bench for the listing 9 is shown in listing 10. To avoid long and repeated case statements, the test bench uses a for loop to iterate through all the possible combinations of the selection, input and output values. These values are stored in a separate file, and then read by the test bench. For this to work, the library STD is used.

```
library ieee;
use ieee.std_logic_1164.all;
use ieee.std_logic_textio.all;
library STD;
use STD.textio.all;
entity tb_mux_4to1 is
end entity:
```

```
architecture bench of tb_mux_4to1 is
11
      component mux_4to1
        port (
14
          ABCD_in: in std_logic_vector(3 downto 0);
          Sel_in: in std_logic_vector(1 downto 0);
16
          Q_out: out std_logic
        );
18
      end component;
19
20
      signal ACBD_tb_in: std_logic_vector(3 downto 0);
      signal Sel_tb_in: std_logic_vector(1 downto 0);
      signal Q_tb_in: std_logic;
    begin
      DUT: mux_4to1 port map(ABCD_in \Rightarrow ACBD_tb_in, Sel_in \Rightarrow Sel_tb_in, Q_out \Rightarrow Q_tb_in);
26
      process
        file Fin : text open read_mode is "mux_4to1_tests.txt";
28
29
30
        variable current_read_line
                                       : line;
        variable current_read_field1 : std_logic_vector(0 to 3);
        variable current_read_field2 : std_logic_vector(0 to 1);
        variable current_read_field3 : std_logic;
35
        while (not endFile(Fin)) loop
          readline(Fin, current_read_line);
          read(current_read_line, current_read_field1);
          read(current_read_line, current_read_field2);
          read(current_read_line, current_read_field3);
          ACBD_tb_in ≤ current_read_field1;
          Sel_tb_in ≤ current_read_field2;
44
          wait for 50 ns;
45
          assert(Q_tb_in = current_read_field3);
47
48
        end loop;
49
        wait;
      end process;
    end architecture;
```

**Listing 10:** Testbench for MUX 4 to 1 The simulation of the listing 9 creates the waveform shown in fig. 6.



Figure 6: Waveform of MUX 4 to 1

Which is the expected result, because the output value is the same as the input value that is selected by the selection lines.

#### 6 Full adder

A full adder is a combinational circuit that forms the arithmetic sum of three bits. It consists of three inputs and two outputs. Two of the input variables, denoted by x and y, represent the two significant bits to be added. The third input, z, represents the carry from the previous stage. Two outputs are necessary because the arithmetic sum of three binary digits ranges in value from 0 to 3, and binary representation of 2 or 3 needs two bits. The two outputs are designated by the symbols S for sum and C for carry. The binary variable S gives the value of the least significant bit of the sum. The binary variable C gives the output carry formed by adding the input carry and the bits of the words. The truth table of the full adder is listed in table 4.

| x | y | $\mathbf{c_{in}}$ | $\mathbf{C_{out}}$ | $\mathbf{S}$ |
|---|---|-------------------|--------------------|--------------|
| 0 | 0 | 0                 | 0                  | 0            |
| 0 | 0 | 1                 | 0                  | 1            |
| 0 | 1 | 0                 | 0                  | 1            |
| 0 | 1 | 1                 | 1                  | 0            |
| 1 | 0 | 0                 | 0                  | 0            |
| 1 | 0 | 1                 | 0                  | 0            |
| 1 | 1 | 0                 | 1                  | 0            |
| 1 | 1 | 1                 | 1                  | 1            |

**Table 4:** Truth table for the 3 bit full adder

The circuit design, of a full adder using digital logic gates, is shown in fig. 7.



Figure 7: Circuit design of a full adder

The VHDL code for the full adder is shown in listing 11. This implementation is quite simple and follows the basic description from the design shown in fig. 7. The sensitivity list includes all the inputs, so if any of them changes the output gets recalculated.

```
library ieee;
    use ieee.std_logic_1164.all;
    entity full_adder is
      port (
         x_in, y_in, c_in : in std_logic;
                             : out std_logic
         s_out, c_out
      );
    end entity;
10
    architecture behavior of full_adder is
    begin
       DUT: process(x_in, y_in, c_in)
14
      begin
         s_out ≤ x_in xor y_in xor c_in;
         c_{out} \leq (x_{in} \text{ and } y_{in}) \text{ or } (x_{in} \text{ and } c_{in}) \text{ or } (y_{in} \text{ and } c_{in});
16
      end process:
    end architecture;
```

#### **Listing 11:** Full adder

For the same reasons as before mentioned, the test bench for the full adder uses a for loop to iterate through all the possible combinations of the inputs and outputs. If an unexpected value is presented, to assert instruction used, will underline the error. The test bench implantation is shown in below.

```
library ieee;
    use ieee.std_logic_1164.all;
    use ieee.std_logic_textio.all;
    library STD;
    use STD.textio.all;
    entity tb_full_adder is
    end entity;
    architecture bench of tb_full_adder is
      component full_adder
        port (
13
          x_in, y_in, c_in : in std_logic;
14
          s_out, c_out
15
                         : out std_logic
16
      end component;
18
      signal x_tb_in
                       : std_logic;
19
                       : std_logic;
      signal y_tb_in
20
21
      signal c_tb_in
                      : std_logic:
      signal s_tb_out : std_logic;
      signal c_tb_out : std_logic;
25
    begin
      DUT: full_adder port map(x_in \Rightarrow x_tb_in, y_in \Rightarrow y_tb_in, c_in \Rightarrow c_tb_in, s_out \Rightarrow
      \rightarrow s_tb_out, c_out \Rightarrow c_tb_out);
27
      process
28
29
        file Fin : text open read_mode is "full_adder_input.txt";
30
        variable current_read_line
31
                                       : line;
32
        variable current_read_field1 : std_logic;
        variable current_read_field2 : std_logic;
33
        variable current_read_field3 : std_logic;
        variable current_read_field4 : std_logic;
35
36
        variable current_read_field5 : std_logic;
37
      begin
38
39
        while (not endFile(Fin)) loop
40
          readline(Fin, current_read_line);
41
          read(current_read_line, current_read_field1);
42
          read(current_read_line, current_read_field2);
43
          read(current_read_line, current_read_field3);
          read(current_read_line, current_read_field4);
45
46
          read(current_read_line, current_read_field5);
47
          x_tb_in ≤ current_read_field1;
48
          y_tb_in ≤ current_read_field2;
49
          c_tb_in < current_read_field3;</pre>
50
51
          wait for 50 ns;
52
53
          assert(c_tb_out = current_read_field4) report "Error in carry" severity failure;
54
          assert(s_tb_out = current_read_field5) report "Error in sum" severity failure;
        end loop;
56
        wait:
57
      end process:
58
```

```
59 | end architecture;
```

**Listing 12:** Test bench for the full adder

#### 7 Comparator

A (magnitude) comparator is a combinational circuit that compares two numbers A and B and determines their relative magnitudes. The outcome of the comparison is specified by a variable which indicates, if the numbers are equal or not.

The VHDL implementation following in listing 13, uses generic value for the size of the numbers, this means that the comparator can be used for any size of numbers.

```
library ieee;
   use ieee.std_logic_1164.all;
   entity comparator is
     generic(N: integer:= 4);
     port (
       x_in, y_in: in std_logic_vector(N-1 downto 0);
       is_eq: out std_logic
     );
   end entity:
10
   architecture behavior of comparator is
     DUT: process(x_in, y_in)
14
15
     begin
       is_eq \leq '1' when (x_in = y_in)
16
       else '0':
18
     end process;
   end architecture;
```

**Listing 13:** Comparator

For the testing of the comparator, the N value was set to 4, i.e. 4 bits numbers. The test bench for the comparator is shown in listing 14.

```
library ieee;
use ieee.std_logic_1164.all;
    use std.env.finish;
    entity tb_comparator is
    end tb_comparator;
    architecture behavior of tb_comparator is
      signal x_tb_in, y_tb_in: std_logic_vector(3 downto 0) := "0000";
      signal is_eq_tb: std_logic;
      constant clk: time := 1 ns;
    DUT: entity work.comparator port map (x_tb_in, y_tb_in, is_eq_tb);
14
      process
15
      begin
         wait for clk;
16
         x_tb_in \left\begin{align*} "0001";
y_tb_in \left\begin{align*} "0000";
18
         wait for clk;
19
         x_{tb}in \leq "0110";
20
         y_tb_in ≤ "0110";
21
         wait for clk;
         wait:
      finish;
24
      end process;
25
    end architecture;
```

#### Listing 14: Test bench for Comparator

The simulation results of the above test bench are summed up in the following waveform image.



Figure 8: Waveform of Comparator

#### Sequential multiplier 8

Multiplication of binary numbers is performed in the same way as multiplication of decimal numbers. The multiplicand is multiplied by each bit of the multiplier, starting from the least significant bit. Each such multiplication forms a partial product. Successive partial products are shifted one position to the left. The final product is obtained from the sum of the partial products.

The algorithm for the multiplication, is summed up by the pseudocode following.

```
Algorithm 1: Sequential multiplier
```

```
Data: A, B same n-bits binary numbers
Result: A \times B
```

 $\mathbf{1} P \leftarrow 0$ 

**2** Load A, B

з while  $B \neq 0$  do

$$\begin{array}{c|c}
\mathbf{4} & \mathbf{if} \ b_0 = 1 \mathbf{then} \\
\mathbf{5} & P \leftarrow P + A
\end{array}$$

left shift A

right shift B

The digital circuit for the sequential multiplier, implements the procedure shown from before and the implementation is a follows bellow.



Figure 9: Circuit design of the sequential multiplier

Both the circuit and VHDL implementations, are developed in such way to calculate the

multiplication of two n-bits numbers and not fixed length numbers.

The FSM following, shows the states that the sequential multiplier goes through to calculate the multiplication of two n-bits numbers. The states are represented by the circles and the arrows represent the transitions between the states. The FSM instructs to store the value of the partial product, in the p register. Finally, the p register is also the output of the sequential multiplier.



Figure 10: Fine state machine

The sequential multiplier system, uses VHDL implementations of n-bits register, n-bits parallel access shift register and adder/subtractor circuits. For space-saving reasons, the VHDL implementations are not shown here, but can be found in the appendix.

```
library IEEE;
   use IEEE.STD_LOGIC_1164.ALL;
   entity multiplier is
     generic (N: INTEGER := 4);
       clk, reset, s: in std_logic;
       dA, dB: in std_logic_vector (N-1 downto 0);
       P: out std_logic_vector (2*N-1 downto 0);
       done: out std_logic
     );
   end multiplier;
12
   architecture Behavioral of multiplier is
14
      component parallel_shift_reg
16
          N: INTEGER := 4;
          DIR: STRING := "LEFT"
18
19
       port (
          clk, reset: in std_logic;
21
          din, E, s_l: in std_logic; --din: shiftin input
22
          D: in std_logic_vector (N-1 downto 0);
          Q: out std_logic_vector (N-1 downto 0);
24
          shift_out: out std_logic
       );
26
      end component;
28
      component add_sub
29
       generic (N: INTEGER := 4);
30
31
          addsub: in std_logic;
32
          x, y: in std_logic_vector (N-1 downto 0);
          s: out std_logic_vector (N-1 downto 0);
34
          overflow, cout: out std_logic
       );
36
     end component;
38
39
      component n_register
       generic (N: INTEGER := 4);
40
```

```
port (
41
            clk, reset: in std_logic;
42
            E, clear: in std_logic; -- clear: Synchronous clear
43
            D: in std_logic_vector (N-1 downto 0);
44
            Q: out std_logic_vector (N-1 downto 0)
45
46
47
       end component;
48
       type state is (S1, S2, S3);
49
        signal y: state;
50
51
        signal L, E, EP, sclrP, z: std_logic;
52
       signal B: std_logic_vector (N-1 downto 0);
       signal A, dAx, dP, Pt: std_logic_vector (2*N -1 downto 0);
54
55
     begin
56
     rA: parallel_shift_reg generic map (N \Rightarrow 2*N, DIR \Rightarrow "LEFT")
57
          port map (
58
59
          clk \Rightarrow clk, reset \Rightarrow reset, din \Rightarrow '0',
          s_l \Rightarrow L, E \Rightarrow E, D \Rightarrow dAx, Q \Rightarrow A
60
61
       dAx (2*N-1 downto N) \leq (others \Rightarrow '0');
62
       dAx (N-1 downto 0) \leq dA;
63
64
     rB: parallel_shift_reg generic map (N ⇒ N, DIR ⇒ "RIGHT")
65
          port map (
66
          clk \Rightarrow clk, reset \Rightarrow reset, din \Rightarrow '0',
67
          \texttt{s\_l} \, \Rightarrow \, \texttt{L, E} \, \Rightarrow \, \texttt{E, D} \, \Rightarrow \, \texttt{dB, Q} \, \Rightarrow \, \texttt{B}
68
       ):
69
70
     -- n-bit NOR gate:
71
       process (B)
72
          variable result_or: std_logic;
       begin
          result_or := '0';
75
          for i in B'range loop -- 'range: iterates through all bits in 'A'
76
           result_or := result_or or B(i);
77
          end loop;
78
79
          z ≤ not (result_or);
       end process;
80
81
     --Adder:
82
     qa: add_sub generic map (N \Rightarrow 2*N)
83
          port map (addsub \Rightarrow '0', x \Rightarrow A, y \Rightarrow Pt, s \Rightarrow dP);
84
85
     --Register P:
86
     rP: n_register generic map (N \Rightarrow 2*N)
87
       port map (
88
          clk \Rightarrow clk, reset \Rightarrow reset, E \Rightarrow EP,
89
            clear \Rightarrow sclrP, D \Rightarrow dP, Q \Rightarrow Pt
90
91
       P ≤ Pt;
92
93
94
     -- FSM:
       Transitions: process (reset, clk, s, z, B(0))
95
96
       if reset = '0' then -- asynchronous signal
97
          y ≤ S1; -- if reset asserted, go to initial state: S1
98
       elsif (clk'event and clk = '1') then
99
          case y is
100
            when S1 \Rightarrow if s = '1' then y \leqslant S2; else y \leqslant S1; end if;
101
            when S2 \Rightarrow if z = '1' then y \leqslant S3; else y \leqslant S2; end if;
102
            when S3 \Rightarrow if s = '1' then y \leq S3; else y \leq S1; end if;
103
          end case;
104
       end if:
105
106
       end process;
107
```

```
Outputs: process (y, s, z, B(0))
108
109
       begin
       -Initialization of output signals
       sclrP \leqslant '0'; EP \leqslant '0'; L \leqslant '0'; E \leqslant '0'; done \leqslant '0';
       case y is
         when S1 \Rightarrow
            sclrP ≤ '1'; EP ≤ '1';
            if s = '1' then
              L \leq '1'; E \leq '1';
            end if;
          when S2 \Rightarrow
118
            E \le '1';
            if z = '0' then
120
               if B(0) = '1' then EP \leq '1'; end if;
            end if:
          when S3 \Rightarrow
            done ≤ '1';
124
       end case;
126
       end process;
     end Behavioral;
```

**Listing 15:** *n*-bits multiplier

Point of interest of the above implementation listing 15 is the finite state machine (FSM) that is used to control the system. The FSM is implemented using a case statement, and the sensitivity list includes the clock signal. It follows exactly the FSM diagram is shown in fig. 10 and the logic presented in that figure.

The test bench for the multiplier is shown in listing 16. It has to be pointed out that, in worst case scenario, the multiplication of two n-bits numbers, needs n+1 clock cycles to complete. For that reason, on each multiplication the test bench waits n+3 clock cycles, to ensure that the multiplication is completed.

```
library ieee;
   use ieee.std_logic_1164.ALL;
   use ieee.std_logic_arith.all;
    use std.env.finish;
    entity tb_multiplier is
     generic (N: INTEGER := 4);
    end tb_multiplier;
    architecture behavior of tb_multiplier is
10
       --Inputs
       signal clk: std_logic := '0';
       signal reset: std_logic := '0';
13
       signal s: std_logic := '0';
       signal dA: std_logic_vector (N-1 downto 0) := (others \Rightarrow '0');
       signal dB: std_logic_vector (N-1 downto 0) := (others \Rightarrow '0');
       --Outputs
18
       signal P: std_logic_vector (2*N - 1 downto 0);
19
20
       signal done: std_logic;
22
       -- Clock period definitions
       constant clock_period: time := 10 ns;
    begin
24
        Instantiate the Unit Under Test (UUT)
      uut: entity work.multiplier port map (clk, reset, s, dA, dB, P, done);
27
      -- Clock process definitions
28
      clock_process: process
29
      begin
      clk ≤ '0';
31
        wait for clock_period/2;
32
      clk ≤ '1';
33
        wait for clock_period/2;
34
```

```
end process;
35
36
        -- Stimulus process
37
38
        stimulus: process
        begin
39
            - hold reset state for 100 ns.
40
          wait for 100 ns; reset ≤ '1';
41
42
          dA \leq conv\_std\_logic\_vector (13, N); -- "1101";
43
          dB ≤ conv_std_logic_vector (11, N); -- "1011";
44
          s \leq '1'; wait for clock_period; s \leq '0';
45
46
          wait for clock_period*(N+3); -- Multiplication takes at most N + 1 cycles
47
48
          \begin{array}{lll} \text{dA} & \leqslant & \text{conv\_std\_logic\_vector} & (9, \ \text{N}); \\ \text{dB} & \leqslant & \text{conv\_std\_logic\_vector} & (15, \ \text{N}); \\ \end{array}
                                                                  -- "1001";
49
                                                                  -- "1111";
50
          s \leq '1'; wait for clock_period; s \leq '0';
51
52
          wait for clock_period*(N+3);
53
          finish;
54
55
        end process;
     end architecture;
56
```

**Listing 16:** Test bench for the n-bits multiplier

The test bench in listing 16, generates the waveform shown in fig. 11. Before the end result is calculated, in the p register the partial multiplication result is stored.



Figure 11: Waveform of the multiplier

#### 9 Encoder

An encoder is a digital circuit that performs the inverse operation of a decoder. An encoder has  $2^n$  input lines and n output lines. The output lines, as an aggregate, generate the binary code corresponding to the input value.

In this implementation, the encoder is a 16 to 4 circuit. The table for how the systems switches between the given values is shown below table 5.

| Input (16-bits)                         | Output (4-bits) |
|-----------------------------------------|-----------------|
| 000000000000000001                      | 0000            |
| 000000000000000000000000000000000000000 | 0001            |
| 00000000000000100                       | 0010            |
| 0000000000001000                        | 0011            |
| 0000000000010000                        | 0100            |
| 0000000000100000                        | 0101            |
| 0000000001000000                        | 0110            |
| 0000000010000000                        | 0111            |
| 0000000100000000                        | 1000            |
| 0000001000000000                        | 1001            |
| 00000100000000000                       | 1010            |
| 00001000000000000                       | 1011            |
| 00010000000000000                       | 1100            |
| 00100000000000000                       | 1101            |
| 01000000000000000                       | 1110            |
| 100000000000000000                      | 1111            |

**Table 5:** Truth table for the 16 to 4 encoder

The implemented encoder, shown in listing 17, reads the input vector and then uses a case statement to select the output value. The sensitivity list only consists of the input vector. A default value of Undefined ('XXXX') is also used to avoid latches.

```
library ieee;
   use ieee.std_logic_1164.all;
   entity encoder_16to4 is
     port (
       D_in: in std_logic_vector(15 downto 0);
       Q_out: out std_logic_vector(3 downto 0)
     ):
   end entity;
   architecture behavior of encoder_16to4 is
     DUT: process(D_in)
14
     begin
       case D in is
         when "0000000000000001" ⇒ Q_out ≤
                                             "0000";
16
                                              "0001";
         18
         "0010"
         when "0000000000000000" \Rightarrow Q_out \leq
                                             "0011"
19
         when "0000000000010000" ⇒ Q_out ≤
                                              "0100";
20
         when "0000000000100000" \Rightarrow \dot{Q}_out
                                              "0101";
21
         when "000000001000000" ⇒ Q_out
                                              "0110"
         when "0000000010000000" ⇒ Q_out
                                              "0111"
         when "0000000100000000" \Rightarrow Q_out
                                             "1000"
24
         when "0000001000000000" \Rightarrow 0_out
         when "0000010000000000" ⇒ Q_out ≤
                                             "1010";
         when "000010000000000" \Rightarrow Q_out \leqslant
                                             "1011"
         when "000100000000000" ⇒ Q_out
                                              "1100"
28
         when "001000000000000" ⇒ Q_out ≤
                                             "1101"
29
         when "010000000000000" ⇒ 0_out ≤
                                             "1110"
30
         when "1000000000000000" ⇒ Q_out ≤ "1111"
31
         when others ⇒ Q_out ≤ "XXXX";
32
       end case;
33
34
     end process;
   end architecture;
```

#### **Listing 17:** Encoder 16 to 4

For the testing of the encoder, a text file was used to store the input and output values and like shown before, the contents of the file were read and compared with the values that the encoder produced.

```
library ieee;
   use ieee.std_logic_1164.all;
   use ieee.std_logic_textio.all;
   library STD;
   use STD.textio.all;
   entity tb_encoder_16to4 is
   end entity;
   architecture behavior of tb_encoder_16to4 is
     signal D_tb_in: std_logic_vector(15 downto 0);
     signal Q_tb_out: std_logic_vector(3 downto 0);
   begin
14
     DUT: entity work.encoder_16to4 port map (D_tb_in, Q_tb_out);
16
     process
       file f_in: text open read_mode is "encoder_16to4_test.txt";
18
       variable current_read_line: line;
19
       variable current_read_field1: std_logic_vector(0 to 15);
20
       variable current_read_field2: std_logic_vector(0 to 3);
21
     begin
       while (not endFile(f_in)) loop
          readline(f_in, current_read_line);
24
          read(current_read_line, current_read_field1);
          read(current_read_line, current_read_field2);
26
27
          D_tb_in ≤ current_read_field1;
28
29
          wait for 50 ns;
30
          assert(Q_tb_out = current_read_field2);
        end loop;
32
33
        wait;
     end process;
34
   end architecture;
35
```

### **Listing 18:** Encoder 16 to 4

The simulation results of the test bench in listing 18, are shown in fig. 12. The easy-tounderstand functionality of the circuit, makes it possible to verify the results by looking in the waveform.



Figure 12: Waveform of the encoder

## Supplementary code for LED counter & multiplier

#### $\mathbf{A.1}$ LED counter

#### A.1.1 LED Decoder

The LED Decoder is a combination of LEDs that should display the value of the decoder at any given moment. The decoder takes in 4-bits at its input and outputs 7 bits. Essentially, the LED will represent the hexadecimal value of the 4-bits as shown in the table below.

| Binary value (4-bits) | LED value |
|-----------------------|-----------|
| 0000                  | 0         |
| 0001                  | 1         |
| 0010                  | 2         |
| 0011                  | 3         |
| 0100                  | 4         |
| 0101                  | 5         |
| 0110                  | 6         |
| 0111                  | 7         |
| 1000                  | 8         |
| 1001                  | 9         |
| 1010                  | A         |
| 1011                  | В         |
| 1100                  | C         |
| 1101                  | D         |
| 1110                  | E         |
| 1111                  | F         |

Table 6: LED Decoder

The VHDL code for the LED Decoder is shown here listing 19.

```
library ieee;
   use ieee.std_logic_1164.all;
   use ieee.std_logic_unsigned.all;
   use ieee.std_logic_arith.ALL;
    entity led_decoder is
     port (
        clk: in std_logic;
        d: in std_logic_vector (3 downto 0);
        s: out std_logic_vector (6 downto 0)
10
     );
   end entity;
12
    architecture behavior of led_decoder is
14
    begin
16
      DUT: process(clk, d)
     begin
        if rising_edge(clk) then
18
          s ≤ "1110111" when d="0000" else
19
          "0010010" when d="0001" else
20
          "1011101" when d="0010" else
21
          "1011011" when d="0011" else
22
          "0111010" when d="0100" else
          "1101011" when d="0101" else
          "1101111" when d="0110" else
25
          "1010010" when d="0111" else
26
          "1111111" when d="1000" else
27
          "1111011" when d="1001" else
```

```
"1111110" when d="1010" else
29
          "0101111" when d="1011" else
30
          "1100101" when d="1100" else
31
          "0011111" when d="1101" else
32
          "1101101" when d="1110" else
33
          "1101100";
34
        end if;
35
      end process;
36
   end architecture;
```

**Listing 19:** LED Decoder

#### A.1.2 Counter

The counter implemented, is a 4-bit counter that counts from 0 to 15. The increment of the counter is a synchronous manner, i.e. the counter value is updated only when the clock signal is high. When the counter reaches its maximum value (15), it resets back to 0 (wrap around).

The VHDL code for the counter is shown here listing 20. The reset input is used to reset the counter back to 0. Besides that, the counter also has a clock input, which is used to increment the counter value. The counter value returned is a 4-bit value, using a vector.

```
library ieee;
    use ieee.std_logic_1164.all:
   use ieee.std_logic_unsigned.all;
    entity counter is
     port (
        clk, reset: in std_logic;
        count: out std_logic_vector (3 downto 0)
     ):
10
    end entity;
    architecture behavior of counter is
    begin
      process(clk, reset)
      begin
15
        if reset = '1' then
16
          count ≤ "0000"
17
        elsif rising_edge(clk) then
18
          if count = "1001" then
19
            count ≤ "0000";
          else
            count ≤ count + 1;
22
          end if;
        end if;
24
25
      end process;
   end architecture;
```

Listing 20: Counter

### Multiplier

### A.2.1 Adder/Subtractor

```
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
entity add_sub is
  generic (N: INTEGER:= 4);
  port(
    addsub: in std_logic; -- addsub = 0 (add), addsub = 1 (sub)
    x, y: in std_logic_vector (N-1 downto 0);
    s: out std_logic_vector (N-1 downto 0);
```

```
overflow: out std_logic;
10
        cout: out std_logic);
11
    end add_sub;
    architecture structure of add_sub is
14
      component full_adder
15
        port(
16
          x_in, y_in, c_in : in std_logic;
18
            s_out, c_out: out std_logic
19
      end component;
20
21
      signal c: std_logic_vector (N downto 0);
22
      signal yx: std_logic_vector (N-1 downto 0);
    begin
24
         c(0) \leq addsub;
25
      cout \leq c(N);
26
      overflow \leq c(N) xor c(N-1);
28
      gi: for i in 0 to N-1 generate
29
30
          yx(i) \leq y(i) xor addsub;
          fi: full_adder port map (
31
            c_i = c(i), x_i = x(i), y_i = yx(i), s_o = s(i),
32
            c_{out} \Rightarrow c(i+1)
33
          );
34
          end generate;
35
    end structure;
```

**Listing 21:** *n*-bits adder/subtractor

### A.2.2 Register

```
library IEEE;
    use IEEE.STD_LOGIC_1164.ALL;
    -- N-bit Register
    -- E = '1', clear = '0' --> Input data 'D' is copied on Q
-- E = '1', clear = '1' --> Q is cleared (0)
    entity n_register is
         generic (N: INTEGER:= 4);
8
      port (
        clk, reset : in std_logic;
10
           E, clear : in std_logic; -- clear: Synchronous clear
                : in std_logic_vector (N-1 downto 0);
                  : out std_logic_vector (N-1 downto 0));
    end n_register;
    architecture Behavioral of n_register is
17
      signal Qt
                   : std_logic_vector (N-1 downto 0);
    begin
18
      process (reset, clk)
19
20
      begin
        if reset = '0' then
           Qt \leq (others \Rightarrow '0');
        elsif (clk'event and clk = '1') then
           if E = '1' then
24
             if clear = '1' then
25
               Qt \leq (others \Rightarrow '0');
26
27
             else
               Qt ≤ D;
28
29
             end if;
           end if;
30
        end if;
31
32
      end process;
      Q \leq Qt;
33
```

```
34 | end Behavioral;
35
```

**Listing 22:** *n*-bits register

### A.2.3 Parallel Shift Register

```
library IEEE;
    use IEEE.STD_LOGIC_1164.ALL;
    -- N-bit Parallel access Shift register:
    -- DIR = "LEFT" --> Shift to the left: from LSB to MSB
    -- DIR = "RIGHT" --> Shift to the right: from MSB to LSB
    -- s_l = 0 --> Shift operation (MSB to LSB)
    -- s_l = 1 -> Parallel load
    entity parallel_shift_reg is
10
        generic (
        N : INTEGER := 4;
          DIR : STRING := "LEFT"
14
      port (
15
        clk, reset : in std_logic;
16
          din, E, s_l : in std_logic; -- din: shiftin input
              : in std_logic_vector (N-1 downto 0);
18
                 : out std_logic_vector (N-1 downto 0);
19
            shift_out : out std_logic
20
21
      );
    end parallel_shift_reg;
    architecture Behavioral of parallel_shift_reg is
24
      signal Qt: std_logic_vector (N-1 downto 0);
27
    a0: assert (DIR = "LEFT" or DIR = "RIGHT")
28
        report "DIR can only be LEFT or RIGHT" severity error;
29
30
      process (reset, clk)
31
32
      begin
        if reset = '0' then
33
          Qt \leq (others \Rightarrow '0');
34
        elsif (clk'event and clk = '1') then
35
          if E = '1' then
36
            if s_l = '1' then
37
              Qt ≤ D;
38
39
            else
              if DIR = "LEFT" then
40
                Qt(0) \leq din;
41
42
                for i in 1 to N-1 loop
                  Qt(i) \leq Qt(i-1);
43
                end loop;
44
              elsif DIR = "RIGHT" then
45
                Qt(N-1) \leq din;
46
                for i in 0 to N-2 loop
47
                  Qt(i) \leq Qt(i+1);
48
49
                end loop;
              end if:
50
            end if;
51
          end if;
52
        end if:
53
54
      end process;
56
57
      Q \leq Qt;
58
      gl: if DIR = "LEFT" generate
```

```
60
               shift_out \leq Qt(N-1);
        end generate;
gr: if DIR = "RIGHT" generate
shift_out \le Qt(0);
61
62
63
end generate;
end Behavioral;
```

Listing 23: n-bits parallel shift register

#### В Tools used for digital design and VHDL compilation

There are a lot of development environments suites for analyzing, compiling, simulating and synthesizing VHDL code. Those products come with a hefty price for licensing and the proprietary. For this project, a vocal point was to use free and open source software.

For the analyzing, compiling and simulating of the VHDL code, the open source software GHDL was used. Unlike some other simulators, GHDL is a compiler: it directly translates a VHDL file to machine code, without using an intermediary language such as C/C++.

If synthesizing was needed, the open source software Yosys is an excellent tool for that.

To validate the behavior of the designs, the test benches simulations were exported as vcd files and then imported into GTKWave, which is an external waveform viewer.

The design of some circuits with logic gates, was done using the open source software Logisim-Evolution. This software is a digital logic designer and simulator. It allows the user to design and simulate digital logic circuits.

Finally, the IDE of choice for this project was VSCodium with the extension of ghdl-ls, that acts as a language server for VHDL. This extension provides syntax highlighting, code completion, linting and other features. Additionally, for each design, a Makefile was created to automate the process of compiling and simulating the VHDL code.

### How to use GHDL and GTKWave

The GHDL framework is available through various package managers. The most common way to install GHDL is by building yourself from source. The instructions for building GHDL from source are available here. For the GTKWave viewer, visit their official website and follow from there the instructions for your operating system.

For each design, a source file of the particular design is needed, which can consists of one or more VHDL files in a hierarchical way. Furthermore, a test bench file is needed to simulate the design. The test bench file is usually named with the prefix tb followed by the name of the design.

For each design the following instructions have to be used:

• Analyze the source file(s):

• Analyze the testbench file(s):

• Generate executable file:

• Run the simulation:

• View the waveform:

Where **design** is the name of the design. If all tests are passed, then the no errors will be shown on the terminal. Alternatively, for each design the Makefile can be used to automate the process.

## References

- [1] M. M. Mano and M. D. Ciletti, Digital Design (4th Edition). USA: Prentice-Hall, Inc., 2006.
- [2] S. Brown and Z. Vranesic, Fundamentals of Digital Logic with VHDL Design with CD-ROM. USA: McGraw-Hill, Inc., 3 ed., 2008.
- "Unit 7 introduction digital design." Llamocca, tosystem https://www.secs.oakland.edu/llamocca/, 2023.
- [4] V. Angelov, "Vhdl vorlesung ss2009." https://www.physi.uni-heidelberg.de/  ${\tt gelov/VHDL/VHDL\_SS09\_Teil05.pdf,\ 2009.}$