# ECEN30 Lab 1 - Introduction to VHDL and FPGAs

Joshua Benfell - 300433229 July 26, 2020

# 1 Objectives

The primary objective of this lab is to learn the basics of VHDL and the process taken to put this code onto an FPGA. For this, the Vivado development environment will be used to develop the circuit descriptions and simulate them before uploading them to the chip. In addition to this, the location and use of example code is covered to make the overall process faster for future tasks. It is important that we meet the objective and become familiar with VHDL and the tool chain as it will be used in future lab experiments.

# 2 Methodology

To start off this lab, a counter project was created in Vivado and the ports were defined as a clock input, a direction input, and a count output which was a bus. It as set as a bus as this meant it would take up multiple bits and allow counting to numbers larger than 1.

This project was initially filled with template code for a counter, which Vivado has a wide selection, so it will be useful to refer back to this for other projects as a starting point. There was no copy to clipboard button anywhere and it didn't offer the option to add this file to your project, so the way to use this code was to highlight it all, copy and paste. From here, the IEEE.NUMERIC\_STD.ALL library was included which allowed us to use the unsigned data type for keeping track of the count as a variable. For outputting this variable to the LEDs that are meant to be turned on, it needs to be typecast as a std\_logic\_vector as this is what the bus is wanting as it's type.



Figure 1: Timing graph from the counter simulation

Before this code can be put onto some hardware it should be tested to ensure it functions as intended. For this a test bench was written that implement the counter entity, maps the signals in that entity to the signals in the test bench and then runs the counters methods by toggling the clock every predefined period and after 200ns, it switches the direction bit

to allow testing of both directions. Figure 1 shows the results of this testing and it can be seen on the line with hexadecimal, that the counter is counting in the desired order.

Now that it's been tested, it had to be implemented in software. This involved selecting which pins were to be used. While this information was provided in the lab script. It won't always be, so how this selection would be done in future projects is by looking at the silk screen on the circuit board to determine which pins we want to use. In the Vivado software package, it is possible to select the IO standard used on the pins. This initially displays "default (LVCMOS18)", which could be taken as default is selected and this is what it is. However, this is not the case and it is actually, this is the default you should select, which is very confusing for this package and something to remember for the future.

The code could finally be uploaded to the FPGA. As this was running at the base clock speed of the chip, it as counting too fast for it to be perceived. All that was visible was 4 always on LEDs at varying brightnesses as the on off time acted as a PWM signal. To fix this, the number of bits that had to be counted was increased to 28 and the 4 MSBs were selected as the output. This was more visible. This didn't get simulated as the time it would have to simulate over would've taken far too long.

The circuit boards have more display options on them than just the LEDs. One of the options is seven segment displays. To use this, a decoder entity for the displays was made. This took all the expected inputs and mapped them to an output combination this is done through a lookup table instead of traditional electronics which will speed up the conversion. The entity is them registered in the counter so that it can be used, and the outputs mapped to the required ports in this component. Once mapping the pins was done, the code could be uploaded, which turned all the displays on. Something to investigate next time is how to do a single display.

Up until this point, the program has been uploaded to volatile memory, which means that when the board was reset, the counter program didn't load in. To store the program on the board so that it can be reused after a power loss, it was uploaded to the non-volatile flash memory.

#### 3 Code

#### 3.1 Counter

```
1 library IEEE;
  use IEEE.STD_LOGIC_1164.ALL;
  use IEEE.NUMERIC.STD.ALL;
3
4
  entity counter is
5
      Port (
6
              clk : in STD_LOGIC;
              direction : in STD_LOGIC;
              count_out : out STD_LOGIC_VECTOR (3 downto 0);
9
              Seg_Out : out STDLOGIC_VECTOR (6 downto 0)
10
11
  end counter;
12
13
  architecture Behavioral of counter is
15
16
   -signal count_int : unsigned(3 downto 0) := (others => '0');
    END PART 1
   - PART 2
```

```
20 signal count_int : unsigned(27 downto 0) := (others => '0'); — make it 28
      bits long
    END PART 2
21
22
  —PART 3
23
  Component ssd_decoder is — This registers the entity ssd_decoder in this
      project.
      Port (
25
              Bin : in STDLOGIC_VECTOR (3 downto 0);
26
              Seg_Out: out STD_LOGIC_VECTOR (6 downto 0)
27
  end Component;
29
  — END PART 3
30
  begin
33
34
  process (clk) — Forces Sequential Operation
36
     if clk='1' and clk' event then
37
        if direction = '1' then — Comparison with char?
            count_int <= count_int + 1; — Push count_int + 1 into count_int
39
40
            count_int \le count_int - 1;
41
42
        end if;
     end if;
43
44 end process;
45
   — PART 1
  -- count_out <= std_logic_vector(count_int); -- Type cast from unsigned to</pre>
      std_logic_vector
48 — END PART 1
49 — PART 2
50 count_out <= std_logic_vector(count_int(27 downto 24)); — Select 4MSBs of
     the 28 bit long thing
    - END PART 2
   - PART 3
53
ssd: ssd_decoder PORT MAP (
      Bin => std_logic_vector(count_int(27 downto 24)),
      Seg_Out \Rightarrow Seg_Out
56
57 );
   - END PART 3
59
60 end Behavioral;
```

code/counter.vhd

#### 3.2 Counter Test Bench

```
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;

entity counter_tb is
— Port ();
end counter_tb;

architecture Behavioral of counter_tb is

Component Counter
```

```
11 port (
       clk: in STD_LOGIC;
       direction : in std_logic;
13
       count_out : out std_logic_vector(3 downto 0)
14
  );
15
  end Component;
16
17
  signal clock : std_logic := '0';
18
signal dir : std_logic := '0';
  signal count : std_logic_vector(3 downto 0) := "0000";
21
  constant clock_period : time := 10ns;
23
  begin
24
25
  uut: counter PORT MAP (
26
       clk \Rightarrow clock,
27
       direction => dir,
       count_out => count
29
  );
30
31
32
  process
  begin
33
       dir <= '1';
34
       wait for 200 ns;
35
       dir <= '0';
36
       wait for 200ns;
37
  end process;
38
39
40
  clock <= not clock after clock_period;</pre>
41
42
43 end Behavioral;
```

code/counter\_tb.vhd

## 3.3 Seven Segment Display Decoder

```
1 library IEEE;
<sup>2</sup> use IEEE.STD_LOGIC_1164.ALL;
3
  entity ssd_decoder is
      Port (Bin : in STDLOGIC_VECTOR (3 downto 0);
              Seg_Out : out STD_LOGIC_VECTOR (6 downto 0));
6
  end ssd_decoder;
8
  architecture Behavioral of ssd_decoder is
9
10
  begin
11
12
  with Bin select
13
      Seg_Out \le "1111001" when "0001",
14
                   "0100100" when "0010",
15
                   "0110000" when "0011"
                                                3
16
                   "0011001" when "0100",
                                                4
17
                   "0010010" when "0101",
18
                  "0000010" when "0110"
19
                  "1111000" when "0111"
                                                7
20
                   "0000000" when "1000"
                                                8
21
                   "0010000" when "1001",
22
```

```
"0001000" when "1010",
23
                   "0000011" when "1011",
24
                   "1000110" when "1100",
                                                   12 (C)
25
                   "0100001" when "1101",
                                                   13 (D)
26
                   "0000110" when "1110",
                                                   14 \ (E)
                   "0001110" when "1111",
                                                   15
                                                      (\mathbf{F})
                   "1000000" when others;
29
30
31
  end Behavioral;
```

code/ssd\_decoder.vhd

## 4 Questions

### 4.1 Q1

The name of the Xilinx chip we are using is the Artix 7 (Xilinx Part Number XC7A100T-1CSG324C) on the Nexys4DDR board by Digilent. This chip has 324 pins.

#### 4.2 Q2

This chip operates above 450MHz internally.

#### 4.3 Q3

If we assume that the our clock is running at 450MHz, as stated by the data sheet on the wiki, the frequency of bit 24 can be found using eq. (1).

$$F = \frac{450 \times 10^6}{2^{24}}$$
= 26.8Hz

## 4.4 Q4

Using a clock frequency of 450MHz again, getting a frequency of 3KHz can be done by reversing eq. (1). This is shown in eq. (2)

$$3000 = \frac{450 \times 10^6}{2^n}$$

$$2^n = \frac{450 \times 10^6}{3000}$$

$$= 150000$$

$$n = \log_2 150000$$

$$= 17.19$$

$$\approx 17$$
(2)

The result of eq. (2) gives a frequency of 3.4KHz.

## 4.5 Q5

It assigns the signal count\_out to the variable count\_int which is configured as an output to some on board LEDs turning on the ones dictated by what's stored in count\_int and now count\_out.

## 4.6 Q6

A test bench is used to provide a simulated environment for testing your code before it goes onto the FPGA. This is useful as it allows us to simulate timings, whether it is possible to load the code onto the FPGA physically. This is all easier in a test bench as you don't need special equipment to see what your hardware is doing, the built in simulations provide an accurate enough representation of the chip and are able to reproduce what you can expect on the hardware. Additionally, you don't need to have an FPGA on hand to run your code.