# Lab 2.1 Full System Integration

## Lab 2.0 – Camera acquisition interface (recap)

In lab 2.0 you built 2 pieces of an interface that captures images from a thermal camera and saves the output to a file on your host machine. The pieces you built were:

- *Statistics computation* unit, which is in charge of computing the minimum, maximum and average pixel values in an image.
- *Level adjuster* unit, which is responsible for interpolating the frame's pixel intensities to obtain a much more visible image.

We provided you a full *system* and you just needed to modify 2 small VHDL files for the thermal camera *interface*.

## Lab 2.1 – Full system integration

Up until now, we have always employed a bottom-up approach in the labs, i.e. we always provided you with the *system*, and you were just implementing small pieces of various *interfaces*. Following this approach, the goal of this lab is to learn how the *system* is created, i.e. how all components are assembled together.

### Creating a system manually

In <u>CS-209</u>, you built the full system shown in Figure 1 *manually*. You did this by implementing each system component (processor, memory, LEDs, buttons, timer ...) from scratch, and by connecting them all together through a bus. The problem with this design is that all components are custom-made and cannot be used in another system without significant modifications. Additionally, the design suffers from low performance due to its very simple microarchitecture.

*Manually* creating systems is great for *learning* how computer systems work and interact, since one starts from the basics and builds the system incrementally. However, such systems are unsuitable for production environments where high performance is expected and is the norm.

All systems use a standard set of components (processors, memories, timers ...), so one could theoretically spend a considerable amount of time to create such components for re-use in various projects. However, the amount of time required to implement and debug such designs would greatly surpass the time available for the project you have been hired for! *There must be a way for engineering time to be better spent*.



FIGURE 1. ARCHSOC LAB SYSTEM: CUSTOM-MADE SYSTEM COMPONENTS INTERCONNECTED WITH A SHARED BUS.

#### Creating a system automatically

Solutions exist for the previously described problem, and come in the form of *system integration tools*. Most FPGA & ASIC design tools have their own system integration tool, but they all essentially expose the same functionality.

A system integration tool provides a *library* of *standard components*. Designers can choose any component from the library and use the tool to connect them together. All components don't forcefully have the same interface, but most tools generally take care of this by inserting adapters to convert between the different bus sizes and handle minor timing adjustments required by each component.

Using such a tool, we can connect *standard interfaces* together to build up the *backbone* of the system, and concentrate on implementing *custom hardware* only for our *specialized* task.

We will look at one such tool provided by Altera for use in their FPGAs, Qsys.

#### The Qsys system integration tool

Using a tool is like speaking a language, as you must first learn it before you can use it. The *only* way to learn a tool is to follow the same method all *serious engineers* have previously used. It consists of an ancestral technique that can be summarized by 1 acronym: **RTFM**.

If you are not familiar with this acronym, we highly suggest you look it up on the internet and add it to your skillset. The world is not a kind place, and you will see that RTFM is unfortunately your only friend when you find a job and are all alone in front of your computer.

As such, we are going to train you to become serious engineers. Let's start!

#### Learning Qsys

Download the <u>Quartus Prime Standard Edition Handbook</u> and read the chapters relevant to Qsys. For this first introduction to Qsys, it is enough to only read "CHAPTER 5: CREATING A SYSTEM WITH QSYS" and "CHAPTER 6: CREATING QSYS COMPONENTS".

You don't need to read the full chapters, but reading through the following sections gives you the big picture of what Qsys can do.

- VOLUME 1 CHAPTER 5: CREATING A SYSTEM WITH QSYS (PG 179)
  - INTERFACE SUPPORT IN QSYS (PG 180)
  - Adding IP Cores to the IP Catalog (pg 181)
  - SET UP THE IP INDEX FILE (.IPX) TO SEARCH FOR IP COMPONENTS (PG 184)
  - CREATE A QSYS SYSTEM (PG 185 208)
  - INTEGRATE A QSYS SYSTEM AND THE QUARTUS PRIME SOFTWARE WITH THE .QSYS FILE (PG 233)
  - VIEW THE QSYS HDL EXAMPLE (PG 251)
- VOLUME 1 CHAPTER 6: CREATING QSYS COMPONENTS (PG 362)
  - QSYS COMPONENTS (PG 362)
  - CREATE IP COMPONENTS IN THE QSYS COMPONENT EDITOR (PG 366 368)
  - SPECIFY IP COMPONENT TYPE INFORMATION (PG 368)
  - SPECIFY HDL FILES FOR SYNTHESIS IN THE QSYS COMPONENT EDITOR (PG 373)
  - ANALYZE SYNTHESIS FILES IN THE QSYS COMPONENT EDITOR (PG 374)
  - Add Signals and Interfaces in the Qsys Component Editor (pg 378)

#### Using Qsys

- 1. Open the lab template project in Quartus Prime. Observe that the only files included in the project are the top-level VHDL file and a clock constraint file (\*.sdc).
- 2. Copy your implementations for the various components designed in the previous labs to the appropriate areas in the "hw/hdl" directory.
- 3. Launch Qsys.
- 4. Open the "hw/quartus/soc\_system.qsys" file. This file contains the Qsys system design of Lab 2.0, but where all custom peripherals you developed are missing (PWM, MCP3204, Lepton). The partial design in show in Figure 2 and consists simply of a Nios II processor, an on-chip memory, and a JTAG UART (for printf support).

0x0004\_9030 0x0004\_9037

| Connections                                                                                                                                      | Name               | Description                 | Export                 | Clock    | Base                     | End         | IRQ          |
|--------------------------------------------------------------------------------------------------------------------------------------------------|--------------------|-----------------------------|------------------------|----------|--------------------------|-------------|--------------|
|                                                                                                                                                  | ⊟ clk_0            | Clock Source                |                        |          |                          |             |              |
| D-                                                                                                                                               | clk_in             | Clock Input                 | clk                    | exported |                          |             |              |
| <u>о</u>                                                                                                                                         | clk_in_reset       | Reset Input                 | reset                  |          |                          |             |              |
|                                                                                                                                                  | clk                | Clock Output                | Double-click to export | clk_0    |                          |             |              |
|                                                                                                                                                  |                    | Reset Output                | Double-click to export |          |                          |             |              |
|                                                                                                                                                  | 日唱 nios2_gen2_0    | Nios II Processor           |                        |          |                          |             |              |
| $  \bullet   \longrightarrow$                                                                                                                    | clk                | Clock Input                 | Double-click to export | clk_0    |                          |             |              |
| $   \downarrow \rightarrow$                                                                                                                      | reset              | Reset Input                 | Double-click to export | [clk]    |                          |             |              |
|                                                                                                                                                  |                    | Avalon Memory Mapped Master |                        | [clk]    |                          |             |              |
|                                                                                                                                                  | instruction_master | Avalon Memory Mapped Master | Double-click to export | [clk]    |                          |             |              |
|                                                                                                                                                  | irq                | Interrupt Receiver          | Double-click to export | [clk]    | IRQ 0                    | IRQ 31      | $\leftarrow$ |
|                                                                                                                                                  | debug_reset_req    | Reset Output                | Double-click to export | [clk]    |                          |             |              |
| $    \downarrow \phi \phi \downarrow \downarrow \rightarrow$                                                                                     |                    | Avalon Memory Mapped Slave  | Double-click to export | [clk]    | 🖹 0x0004_8800            | 0x0004_8fff |              |
| ×—<                                                                                                                                              | _                  | Custom Instruction Master   | Double-click to export |          |                          |             |              |
|                                                                                                                                                  | onchip_memory      | On-Chip Memory (RAM or ROM) |                        |          |                          |             |              |
| $  \blacklozenge   + + + + \rightarrow$                                                                                                          | clkl               | Clock Input                 | Double-click to export | clk_0    |                          |             |              |
| $     \bullet \bullet   \longrightarrow$                                                                                                         | sl                 | Avalon Memory Mapped Slave  |                        | [clk1]   | <pre> 0x0002_0000 </pre> | 0x0003_ffff |              |
| $   \blacklozenge   \rightarrow   \rightarrow   \rightarrow   \rightarrow   \rightarrow   \rightarrow   \rightarrow   \rightarrow   \rightarrow$ | resetl             | Reset Input                 | Double-click to export | [clk1]   |                          |             |              |
|                                                                                                                                                  | ⊟ jtag_uart_0      | JTAG UART                   |                        |          |                          |             |              |
| $ \bullet +++\rightarrow$                                                                                                                        | clk                | Clock Input                 | Double-click to export | clk_0    |                          |             |              |
| $  \bullet   + \bullet \rightarrow$                                                                                                              | reset              | Reset Input                 | Double-click to export | [clk]    |                          |             |              |

#### FIGURE 2. PARTIAL QSYS SYSTEM

Double-click to export [clk]

5. Create the Qsys IP component for the PWM interface you used in Lab 1.1.

Interrupt Sender

irq

6. Create the Qsys IP component for the MCP3204 interface you used in Lab 1.2.

avalon\_jtag\_slave Avalon Memory Mapped Slave Double-click to export [clk]

 Create the Qsys IP component for the lepton thermal camera interface that you used in Lab 2.0. Unlike the PWM and MCP3204 interfaces, the Lepton requires you to set the number of *read wait states* to 9, as shown in Figure 3. This is due to the speed of the division which occurs over multiple clock cycles, so a CPU cannot read from the Lepton interface as fast as it can from the PWM and MCP3204 interfaces.

| <ul> <li>Timing</li> </ul> |          |
|----------------------------|----------|
| Setup:                     | 0        |
| Read wait:                 | 9        |
| Write wait:                | 0        |
| Hold:                      | 0        |
| Timing units:              | Cycles 💌 |

FIGURE 3. QSYS LEPTON IP CORE TIMINGS

8. Use the IP cores available in the IP Catalog to create the full system shown in Figure 4 which you used in Lab 2.0 to capture images from the thermal camera.

| Connections                                                                                     | Name               | Description                 | Export                 | Clock    | Base          | End         | IRQ          |
|-------------------------------------------------------------------------------------------------|--------------------|-----------------------------|------------------------|----------|---------------|-------------|--------------|
|                                                                                                 | ⊟ clk_0            | Clock Source                |                        |          |               |             |              |
| P-                                                                                              | clk_in             | Clock Input                 | clk                    | exported |               |             |              |
| · · · · ·                                                                                       | clk in reset       | Reset Input                 | reset                  | -        |               |             |              |
|                                                                                                 | clk                | Clock Output                | Double-click to export | clk 0    |               |             |              |
|                                                                                                 | clk reset          | Reset Output                | Double-click to export | -        |               |             |              |
|                                                                                                 | ⊟ 🖳 nios2 gen2 0   | Nios II Processor           |                        |          |               |             |              |
| $  \bullet   \longrightarrow$                                                                   | clk                | Clock Input                 | Double-click to export | clk 0    |               |             |              |
| $   \bullet \bullet \bullet \bullet \bullet$                                                    | reset              | Reset Input                 | Double-click to export | [clk]    |               |             |              |
|                                                                                                 | data master        | Avalon Memory Mapped Master | Double-click to export | [clk]    |               |             |              |
|                                                                                                 | instruction master | Avalon Memory Mapped Master | Double-click to export | [clk]    |               |             |              |
| $      \rightarrow$                                                                             | irq                | Interrupt Receiver          | Double-click to export | [clk]    | IRQ 0         | IRQ 31      | $\leftarrow$ |
|                                                                                                 | debug_reset_req    | Reset Output                | Double-click to export | [clk]    |               |             |              |
|                                                                                                 | debug mem slave    | Avalon Memory Mapped Slave  | Double-click to export | [clk]    |               | 0x0004 8fff |              |
|                                                                                                 | custom instructi   | Custom Instruction Master   | Double-click to export |          | _             | _           |              |
|                                                                                                 | onchip_memory      | On-Chip Memory (RAM or ROM) |                        |          |               |             |              |
| $  \bullet   + + + + \rightarrow$                                                               | clk1               | Clock Input                 | Double-click to export | clk_0    |               |             |              |
|                                                                                                 | sl                 | Avalon Memory Mapped Slave  | Double-click to export | [clk1]   | 0x0002_0000   | 0x0003_ffff |              |
| $   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   $ | resetl             | Reset Input                 | Double-click to export | [clk1]   | _             | _           |              |
|                                                                                                 | 🗆 jtag_uart_0      | JTAG UART                   |                        |          |               |             |              |
| $  \bullet                                    $                                                 | clk                | Clock Input                 |                        | clk_0    |               |             |              |
| $   \blacklozenge   \leftrightarrow   \rightarrow  $                                            | reset              | Reset Input                 | Double-click to export | [clk]    |               |             |              |
|                                                                                                 | avalon_jtag_slave  | Avalon Memory Mapped Slave  | Double-click to export | [clk]    | ■ 0x0004_9030 | 0x0004_9037 |              |
|                                                                                                 | irq                | Interrupt Sender            | Double-click to export | [clk]    |               |             |              |
|                                                                                                 | ⊟ pwm_0            | pwm                         |                        |          |               |             |              |
| $  \blacklozenge                                    $                                           | clock              | Clock Input                 | Double-click to export | clk_0    |               |             |              |
| $   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   $ | reset              | Reset Input                 | Double-click to export | [clock]  |               |             |              |
|                                                                                                 | avalon_slave_0     | Avalon Memory Mapped Slave  | Double-click to export | [clock]  |               | 0x0004_902f |              |
|                                                                                                 | conduit_end        | Conduit                     | pwm_0_conduit_end      | [clock]  |               |             |              |
|                                                                                                 | ⊡ pwm_1            | pwm                         |                        |          |               |             |              |
| $  \bullet   \rightarrow  $                                                                     | clock              | Clock Input                 |                        | clk_0    |               |             |              |
| $   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   \bullet   $ | reset              | Reset Input                 | Double-click to export | [clock]  |               |             |              |
|                                                                                                 | avalon_slave_0     | Avalon Memory Mapped Slave  | Double-click to export | [clock]  |               | 0x0004_901f |              |
| · · · · · · · · · · · · · · · · · · ·                                                           | conduit_end        | Conduit                     | pwm_1_conduit_end      | [clock]  |               |             |              |
|                                                                                                 | ⊟ mcp3204_0        | mcp3204                     |                        |          |               |             |              |
| • • • • • • • • • • • • • • • • • • • •                                                         | clock              | Clock Input                 |                        | clk_0    |               |             |              |
| $   \bullet   \to \bullet$                                                                      | reset              | Reset Input                 | Double-click to export | [clock]  |               |             |              |
|                                                                                                 | avalon_slave_0     | Avalon Memory Mapped Slave  | Double-click to export | [clock]  |               | 0x0004_900f |              |
| ~~                                                                                              | conduit_end        | Conduit                     | mcp3204_0_conduit_end  | [clock]  |               |             |              |
|                                                                                                 | □ lepton_0         | lepton                      |                        |          |               |             |              |
|                                                                                                 | clock              | Clock Input                 |                        | clk_0    |               |             |              |
|                                                                                                 | reset              | Reset Input                 | Double-click to export | [clock]  |               |             |              |
|                                                                                                 | avalon_slave_0     | Avalon Memory Mapped Slave  | Double-click to export | [clock]  |               | 0x0004_7tff |              |
| - ~ ~                                                                                           | spi                | Conduit                     | lepton_0_spi           | [clock]  |               |             |              |

FIGURE 4. LAB 2.1 SYSTEM TO ASSEMBLE

- 9. Do what is required to correctly instantiate your Qsys system in the top-level design file located at "hw/hdl/DE0\_Nano\_SoC\_PrSoC\_extn\_board\_top\_level.vhd".
- 10. Create a Nios II SBT project for your new system and test if everything works like in Lab 2.0. By everything works, we mean that you can successfully capture a thermal image with the lepton and save it to a file on your host PC.

If you are stuck, remember **RTFM**. We stress that *all* the information you need to successfully implement points 3, 4, 5, and 6 listed above are fully described in the specified pages of the <u>Quartus Prime Standard Edition</u> Handbook!

Don't hesitate to *ask us any questions if something is unclear*. It is essential you understand how all the components in the system are assembled and how they interact.