

### MY FIRST HPS-FPGA







Copyright © 2003-2015 Terasic Inc. All Rights Reserved.





| CHAPTER | 2 OVERVIEW                             |
|---------|----------------------------------------|
| 1.1.    | REQUIRED BACKGROUND                    |
| 1.2.    | System Requirements                    |
| 1.3.    | Altera SoC FPGA                        |
| 1.4.    | Source Code                            |
| CHAPTER | QUARTUS PROJECT                        |
| 2.1.    | MY_FIRST_HPS-FPGA_BASE QUARUTS PROJECT |
| 2.2.    | CREATE A QUARTUS PROJECT               |
| 2.3.    | COMPILE AND PROGRAMMING                |
| CHAPTER | R 3 C PROJECT                          |
| 3.1.    | HPS HEADER FILE                        |
| 3.2.    | MAP PIO_LED ADDRESS                    |
| 3.3.    | LED CONTROL                            |
| 3.4.    | MAIN PROGRAM                           |
| 3.5.    | MAKEFILE AND COMPILE                   |
| 3.6.    | Ехесите тне Demo 17                    |







This tutorial introduces to SoC FPGA Beginners how to use the HPS/ARM to communicate with FPGA. The "My First HPS-FPGA" project demonstrates the implementation details. This project includes one Quartus project and one ARM C Project which demonstrates how HPS/ARM program controls the ten LEDs connected to FPGA.

Before reading this tutorial, developers need to read the documents below:

- DE0-Nano-SoC\_Getting\_Started\_Guide.pdf
- My\_First\_Fpga.pdf
- My\_First\_HPS.pdf

For tutorial purpose, this document asks developers to create a HPS enabled Quartus project based on the project named my\_first\_hps-fpga\_base. However, for the development of a formal HPS enabled project, developers are expected to create a Quartus project based on the DE0-Nano-SoC-GHRD (Golden Hardware Reference Design) project, which is included in the SYSTEM CD.

#### **1.1. Required Background**

This tutorial pre-assumed the developers have the following background knowledge:

- **FPGA RTL Design** 
  - Basic Quartus II operation skill •
  - Basic RTL coding skill
  - Basic Qsys operation skill
  - Knowledge about Altera Memory-Mapped Interface



DEO-Nano-SoC My First HPS **FPGA** 

www.terasic.com May 18, 2015

#### C Program Design

- Basic Altera SoC EDS(Embedded Design Suite) operation skill
- Basic C coding and compiling skill
- Skill to Create a Linux Boot SD-Card for DE0-Nano-SoC with a given image file
- Skill to boot Linux from SD-Card on DE0-Nano-SoC
- Skill to cope files into Linux file system on DE0-Nano-SoC
- Basic Linux command operation skill

# **1.2. System Requirements**

Before starting this tutorial, please note that the following items are required to complete the demonstration project:

- Altera DE0-Nano-SoC FPGA board, includes
  - Mini USB Cable for UART terminal
  - Micros SD-Card, at 4GB
  - Micros SD-Card Card Reader

#### ■ A x86 PC

- Windows 7 Installed
- One USB Port
- Quartus II 14.1 or Later Installed
- Altera SoC EDS 14.1 or Later Installed
- Win32 Disk Imager Installed





# **1.3. Altera SoC FPGA**

In Altera SoC FPGA, the HPS logic and FPGA fabric are connected through the AXI (Advanced eXtensible Interface) bridge. For HPS logic to communicate with FPGA fabric, Altera system integration tool **Qsys** should be used for the system design to add Altera **HPS** component. From the AXI master port of the HPS component, HPS can access those Qsys components whose memory-mapped slave ports are connected to the master port.

The HPS contains the following HPS-FPGA AXI bridges:

#### ■ FPGA-to-HPS Bridge

- HPS-to-FPGA Bridge
- Lightweight HPS-to-FPGA Bridge

**Figure 1-1** shows a block diagram of the AXI bridges in the context of the FPGA fabric and the L3 interconnect to the HPS. Each master (M) and slave (S) interface is shown with its data width(s). The clock domain for each interconnect is noted in parentheses.



#### Figure 1-1 AXI Bridge Block Diagram



DEO-Nano-SoC My First HPS FPGA



The HPS-to-FPGA bridge is mastered by the level 3 (L3) main switch and the lightweight HPS-to-FPGA bridge is mastered by the L3 slave peripheral switch. In the Quartus, HPS-to-FPGA bridge is used for ARM/HSP to control the LEDs which is connected to FPGA.

The FPGA-to-HPS bridge masters the L3 main switch, allowing any master implemented in the FPGA fabric to access most slaves in the HPS. For example, the FPGA-to-HPS bridge can access the accelerator coherency.

All three bridges contains global programmer view GPV register. The GPV register control the behavior of the bridge. It is able to access to the GPV registers of all three bridges through the lightweight HPS-to-FPGA bridge.

### **1.4. Source Code**

The demonstration source codes include a Quartus project and a C project. They are located in the folder:

The Quartus Project is located in the sub-folder "fpga-rtl" and the C project is located in the sub-folder "hps-c". In this tutorial, developer are expected to establish these projects from a new one..







This chapter introduces how the MY First HSP-FPGA Quartus project is created based on the my\_first\_hps-fpga base Quartus project. Base on this Quarturs project, a PIO component for controlling LED is added, a connection between the slave port of the PIO component and the master port of HPS component is established.

# 2.1. my\_first\_hps-fpga\_base Quaruts Project

my\_first\_hps-fpga\_base Quartus project is located in the DE0-Nano-SoC System CD folder: CD-ROM\Demonstration\SOC\_FPGA\my\_first\_hps-fpga\_base

This Quartus project includes all required pin declearing for both HPS and FPGA. Note, the pin declaration of HPS needs to specify pin direction and IO standard. Pin location is not required for the pin declaration of HPS. The golden project also includes basic Qsys system which already includes a HPS component. The HPS component has been well-configured according to hardware design of DE0-Nano-SoC HPS.

Developers can open the Qsys system by opening the Quartus project, and clicking the menu item "Tools $\rightarrow$ Qsys" in Quartus II. When Qsys tool is launched, it will ask user to select a target Qsys system file. In this case, please select the Qsys file "soc system.qsys". Figure 2-1 shows the content of soc\_system.qsys Qsys system. It contains hps\_0 HPS component.







Figure 2-1 hps\_0 HSP Component in Qsys System

**Figure 2-2** shows the lightweight HPS-to-FPGA AXI Master port of the HPS component. Developers can connect this port to any memory-mapped slave port of components which developer wish to access from HPS/ARM.



Figure 2-2 AXI Master Port of HPS component





### **2.2. Create a Quartus Project**

This section will show how to add a PIO component in Qsys and how to connect the PIO component to the HPS component. The PIO component is used to control the ten red LEDs connected to FPGA. First, please copy the **my\_first\_hps-fpga\_base** Quartus project to local disk. Open the project and open the Qsys system file "soc\_system.qsys".

In the Library dialog of Qsys tool, enter 'pio' search key as shown in **Figure 2-3**. When "PIO (Parallel I/O)" appears, select it. Then, click "Add..." to add the PIO component to the system.

| 📑 IP Catalog 🛛                          | - d' 🗆 |
|-----------------------------------------|--------|
| Q pio                                   | × 🔯    |
| Project                                 |        |
| Library<br>- Processors and Peripherals |        |
| Peripherals<br>PIO (Parallel I/O)       |        |
|                                         |        |
|                                         |        |
|                                         |        |
|                                         |        |
|                                         |        |
|                                         |        |
| New Edit                                | 🕂 Add  |

Figure 2-3 Find and Add PIO Component

When PIO dialog appears, please change **Width** to 8, make sure "Output" **Direction** is selected, and change the **Output Port Reset Value** to 0xff as shown in **Figure 2-4**.





| 💐 Parameters 🛛                                     |                                                     |  |  |  |  |
|----------------------------------------------------|-----------------------------------------------------|--|--|--|--|
| <pre>soc_system &gt; pio_led</pre>                 | soc_system > pio_led                                |  |  |  |  |
| PIO (Parallel I/O)<br>altera avalon pio            |                                                     |  |  |  |  |
| Basic Settings<br>Width (1-32 bits):<br>Direction: | 8<br>© Bidir                                        |  |  |  |  |
|                                                    | <ul> <li>Input</li> <li>InOut</li> </ul>            |  |  |  |  |
| Output Port Reset Val                              | () Output<br>Dx000000000000000000000000000000000000 |  |  |  |  |

Figure 2-4 Configure PIO Component

When the PIO component is added into the system, please connect the h2f\_lw\_axi\_master AXI master port to the s1 slave port of the PIO component as shown in **Figure 2-5**. By the way, please change PIO component name to **pio\_led**, change the **Clock Input** to **clk\_0**, export the **Conduit** signal as **pio\_led\_external\_connection**, and connect the **Reset Input** to system reset. Note, the **Base** address of pio\_led PIO component is very important. The ARM program will access the component according to this base address. In this demonstration, the base address is fixed at 0x0000\_0000. The ARM program developer should remember this base address or use a given Linux shell batch file to extract the address information to a header file hps\_0.h. The detail procedure will be described in the next chapter.



Figure 2-5 Create Connection Between HPS and PIO component





In the Qsys tool, click menu item "Generate  $\rightarrow$  HDL Example..." can find the new interface signal **pio\_led\_external\_connection\_export** for the added **pio\_led** PIO component as shown in **Figure 2-6**. Developer can click 'Copy' to copy the content to a clipboard, then paste the **pio\_led\_external\_connection\_export** signal to Quartus top and connect it to the **LEDR** port as shown in **Figure 2-7**. Before closing the Qsys tool, please remember to click the menu item "Generate  $\rightarrow$  Generate..." to generate source code for the system.

| L HDL Example                                                                        | 2                                                                                                                |
|--------------------------------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------|
| You can copy the example HDL below to declare an instance<br>HDL Language: Verilog ▼ | e of soc system.                                                                                                 |
| Example HDL                                                                          |                                                                                                                  |
| .memory_mem_ck_n                                                                     | ( <connected-to-memory_mem_ck_n>),</connected-to-memory_mem_ck_n>                                                |
| .memory_mem_cke                                                                      | ( <connected-to-memory_mem_cke>),</connected-to-memory_mem_cke>                                                  |
| .memory_mem_cs_n                                                                     | ( <connected-to-memory_mem_cs_n>),</connected-to-memory_mem_cs_n>                                                |
| .memory_mem_ras_n                                                                    | ( <connected-to-memory_mem_ras_n>),</connected-to-memory_mem_ras_n>                                              |
| .memory_mem_cas_n                                                                    | ( <connected-to-memory_mem_cas_n>),</connected-to-memory_mem_cas_n>                                              |
| .memory_mem_we_n                                                                     | ( <connected-to-memory_mem_we_n>),</connected-to-memory_mem_we_n>                                                |
| .memory_mem_reset_n                                                                  | ( <connected-to-memory_mem_reset_n>),</connected-to-memory_mem_reset_n>                                          |
| .memory_mem_dq                                                                       | ( <connected-to-memory_mem_dq>),</connected-to-memory_mem_dq>                                                    |
| .memory_mem_dqs                                                                      | ( <connected-to-memory_mem_dqs>),</connected-to-memory_mem_dqs>                                                  |
| .memory_mem_dqs_n                                                                    | ( <connected-to-memory_mem_dqs_n>),</connected-to-memory_mem_dqs_n>                                              |
| .memory_mem_odt                                                                      | ( <connected-to-memory_mem_odt>),</connected-to-memory_mem_odt>                                                  |
| .memory_mem_dm                                                                       | ( <connected-to-memory_mem_dm>),</connected-to-memory_mem_dm>                                                    |
| .memory_oct_rzqin                                                                    | ( <connected-to-memory_oct_rzqin>),</connected-to-memory_oct_rzqin>                                              |
| .reset_reset_n                                                                       | ( <connected-to-reset_reset_n>),</connected-to-reset_reset_n>                                                    |
|                                                                                      | <pre>(<connected-to-pio_led_external_connection_export>)</connected-to-pio_led_external_connection_export></pre> |
| ):                                                                                   |                                                                                                                  |
| •                                                                                    |                                                                                                                  |
|                                                                                      | Copy Close                                                                                                       |

Figure 2-6 pio\_led Interface of System

| E | soc_system u0 (              |                                          |                             |  |  |  |  |
|---|------------------------------|------------------------------------------|-----------------------------|--|--|--|--|
|   | Opio led external connection | Opio led external connection export(LED) |                             |  |  |  |  |
|   | //Clock&Reset                |                                          |                             |  |  |  |  |
|   | .clk_clk                     | (FPGA_CLK1_50 ),                         | // clk.clk                  |  |  |  |  |
|   | .reset_reset_n               | (1'b1 ),                                 | <pre>// reset.reset_n</pre> |  |  |  |  |
|   | //HPS ddr3                   |                                          |                             |  |  |  |  |
|   | .memory_mem_a                | ( HPS_DDR3_ADDR),                        | // memory.mem_a             |  |  |  |  |
|   | .memory_mem_ba               | ( HPS_DDR3_BA),                          | // .mem_ba                  |  |  |  |  |
|   | .memory_mem_ck               | ( HPS_DDR3_CK_P),                        | // .mem_ck                  |  |  |  |  |
|   | .memory_mem_ck_n             | ( HPS_DDR3_CK_N),                        | // .mem_ck_n                |  |  |  |  |
|   | .memory_mem_cke              | ( HPS_DDR3_CKE),                         | // .mem_cke                 |  |  |  |  |
|   | .memory_mem_cs_n             | ( HPS_DDR3_CS_N),                        | // .mem_cs_n                |  |  |  |  |
|   | .memory_mem_ras_n            | ( HPS_DDR3_RAS_N),                       | // .mem_ras_n               |  |  |  |  |
|   | .memory_mem_cas_n            | ( HPS_DDR3_CAS_N),                       | // .mem_cas_n               |  |  |  |  |
|   |                              | / 1100 DDD0 /10 10                       | 11                          |  |  |  |  |

Figure 2-7 Initialize .pio\_led\_external\_commection\_export in u0 soc\_system





#### **2.3. Compile and Programming**

Now, developers can start the compile process by clicking the menu item "Processing  $\rightarrow$  Start Compilation". When the compilation process is completed successfully, **soc\_system.sof** is generated. Developers can use this file to configure FPGA by Quartus Programming through the DE0-Nano-SoC on-board USB-Blaster II.

Because .tcl files of SDRAM DDR3 controller for HPS had been executed in my\_frist\_hps-fpga\_base Quartus project, so developers can skip these projects. If developers' Quartus project is not developed based on the my\_frist\_hps-fpga\_base Quartus project, please remember to execute the .tcl files, as show in **Figure 2-8**, before executing 'Start Compilation'. The TCL Scripts dialog can be launched by clicking the menu item "Tools $\rightarrow$ TCL Scripts...". <qsys\_system\_name>\_**parameters.tcl** and <qsys\_system\_name>\_**pin\_assignments.tcl** tcl files should be executed, where <qsys\_system\_name> is the name of your Qsys system. Run this script to assign constrains tor the SDRAM DDR3 component.



Figure 2-8 TCL file for SDRAM DDR3 of HPS





This chapter introduces how to design an ARM C program to control the pio\_led PIO controller. Altera SoC EDS is used to compile the C project. For ARM program to control the pio\_led PIO component, pio\_led address is required. The Linux built-in driver '/dev/mem' and mmap system-call are used to map the physical base address of **pie led** component to a virtual address which can be directly accessed by Linux application software.

### 3.1. HPS Header File

pio\_led component information is required for ARM C program as the program will attempt to control the component. This section describes how to use a given Linux shell batch file to extract the Qsys HPS information to a header file which will be included in the C program later.

The batch file mentioned above is called as generate\_hps\_qsys\_header.sh. It is located in the same folder as my\_first\_hps-fpga Quartus project. To generate the header file, launch Altera SoC EDS command shell, go to the Quartus project folder, and execute generate\_hps\_qsys\_header.sh by typing './generate hps qys header.sh". Then, press ENTER key,a header file hps\_0.h will be generated. In the header file, the pio\_led base address is represented by a constant PIO\_LED\_BASE as show in Figure 3-1. The pio\_led width is represented by a constant PIO\_LED\_DATA\_WIDTH. These two constants will be used in the C program demonstration code.



```
白/*
 * Macros for device 'pio led', class 'altera avalon pio'
 * The macros are prefixed with 'PIO LED '.
 * The prefix is the slave descriptor.
 . */
 #define PIO LED COMPONENT TYPE altera avalon pio
 #define PIO LED COMPONENT NAME pio led
 #define PIO_LED_BASE 0x0
 #define PIO LED SPAN 32
 #define PIO LED END 0x1f
 #define PIO LED BIT CLEARING EDGE REGISTER 0
 #define PIO_LED_BIT_MODIFYING_OUTPUT_REGISTER 0
 #define PIO LED CAPTURE 0
 #define PIO LED DATA WIDTH 8
 #define PIO LED DO TEST BENCH WIRING 0
 #define PIO_LED_DRIVEN_SIM VALUE 0
 #define PIO LED EDGE TYPE NONE
 #define PIO LED FREQ 5000000
 #define PIO LED HAS IN 0
 #define PIO LED HAS OUT 1
 #define PIO LED HAS TRI 0
 #define PIO LED IRQ TYPE NONE
 #define PIO LED RESET VALUE 255
```

Figure 3-1 pio\_led information defined in hps\_0.h

### **3.2. Map pio\_led Address**

This section will describe how to map the pio\_led physical address into a virtual address which is accessible by an application software. **Figure 3-2** shows the C program to derive the virtual address of **pio\_led** base address. First, **open** system-call is used to open memory device driver "/dev/mem", and then the **mmap** system-call is used to map HPS physical address into a virtual address represented by the void pointer variable **virtual\_base**. Then, the virtual address of **pio\_led** can be calculated by adding the below two offset addresses to **virtual\_base**.

- Offset address of Lightweight HPS-to-FPGA AXI bus relative to HPS base address
- Offset address of Pio\_led relative to Lightweight HPS-to-FPGA AXI bus

The first offset address is 0xff200000 which is defined as a constant ALT\_LWFPGASLVS\_OFST in the header hps.h. The hps.h is a header of Altera SoC EDS. It is located in the folder:

Quartus Installed Folder\embedded\ip\altera\hps\altera\_hps\hwlib\include\socal The second offset address is 0x00000000 which is defined as PIO\_LED\_BASE in the hps\_0.h header file which is generated in above section.

The virtual address of pio\_led is represented by a void pointer variable h2p\_lw\_led\_addr.terasicDEO-Nano-SoC My First HPS13WWW. terasic.comMay 18, 2015



Application program can directly use the pointer variable to access the registers in the controller of **pio led**.

Figure 3-2 pio\_led information defined in hps\_0.h

# 3.3. LED Control

C programmers need to understand the Register Map of the PIO core for **pio\_led** before they can control it. **Figure 3-3** shows the Register Map for the PIO Core. Each register is 32-bit width. For detail information, please refer to the datasheet of PIO Core. For led control, we just need to write output value to the offset 0 register. Because the led on DE0-Nano-SoC is high active, writing a value 0x00000000 to the offset 0 register will turn off all of the eight green LEDs. Writing a value 0x0000000ff to the offset 0 register will turn on all of eight green LEDs. In C program, writing a value 0x000000ff to the offset 0 register of pio\_led can be implemented as:

\*(uint32\_t \*) h2p\_lw\_led\_addr= 0x000000ff;

The state will assign the void pointer to a uint32\_t pointer, so C compiler knows write a 32-bit value 0x000000ff to the virtual address h2p\_lw\_led\_addr.





| 0//   | Register Name        |              | R/W      | Fields                              |                                                                          |                           |             |         |
|-------|----------------------|--------------|----------|-------------------------------------|--------------------------------------------------------------------------|---------------------------|-------------|---------|
| Unset |                      |              |          | (n-1)                               |                                                                          | 2                         | 1           | 0       |
| 0     | data                 | read access  | R        | Data value currently on PIO inputs. |                                                                          |                           |             |         |
|       | data                 | write access | w        | New value to drive on PIO outputs.  |                                                                          |                           |             |         |
| 1     | direction (1)        |              | R/W      | Individu<br>directio                | ual direction control for each I/O<br>n to input; 1 sets the direction t | ) port. A va<br>o output. | alue of O s | ets the |
| 2     | interruptmask (1)    |              | R/W      | IRQ ena<br>interrup                 | able/disable for each input port.<br>Its for the corresponding port.     | Setting a                 | bit to 1 en | ables   |
| 3     | edgecapture (1), (2) |              | R/W      | Edge de                             | etection for each input port.                                            |                           |             |         |
| 4     | outset               |              | w        | Specifie                            | es which bit of the output port to                                       | o set.                    |             |         |
| 5     | outclear             | w            | Specifie | es which output bit to clear.       |                                                                          |                           |             |         |

Figure 3-3 Register Map of PIO Core

#### 3.4. Main Program

In the main program, the LED is controlled to perform LED light sifting operation as shown in **Figure 3-4**. When finishing 60 times of shift cycle, the program will be terminated.

```
loop count = 0;
led mask = 0 \times 01;
led direction = 0; // 0: left to right direction
while( loop count < 60 ) {</pre>
  // control led, add ~ because the led is low-active
  *(uint32_t *)h2p_lw_led_addr = ~led_mask;
  // wait 100ms
  usleep( 100*1000 );
  // update led mask
  if (led direction == 0){
    led mask <<= 1;</pre>
    if (led mask == (OxO1 << (PIO LED DATA WIDTH-1)))
       led_direction = 1;
  }else{
    led_mask >>= 1;
    if (led mask == 0x01) {
      led_direction = 0;
      loop_count++;
    }
  }
} // while
```

#### Figure 3-4 C Program for LED Shift Operation



# **3.5. Makefile and compile**

**Figure 3-5** shows the content of Makefile for this C project. Because the program will include the hps.h provided by Altera SoC EDS, so the Makefile should include the following path:

"\${SOCEDS\_DEST\_ROOT}/ip/altera/hps/altera\_hps/hwlib/include"

In the makefile, ARM cross-compile also be specified.

```
1 #
 2 TARGET = my_first_hps-fpga
 З
 4 #
 5 CROSS COMPILE = arm-linux-gnueabihf-
 6 CFLAGS = -static -g -Wall -I${SOCEDS DEST ROOT}/ip/altera/hps/altera hps/hwlib/include
 7 LDFLAGS = -g -Wall
8 CC = $ (CROSS_COMPILE) gcc
9 ARCH= arm
10
11
12 build: $(TARGET)
13 $(TARGET): main.o
14 $ (CC) $ (LDFLAGS)
                        $^ -0 $0
15 %.O : %.C
16
   $ (CC) $ (CFLAGS) -c $< -o $0
17
18 .PHONY: clean
19 clean:
   rm -f $(TARGET) *.a *.o *~
20
```

Figure 3-5 Makefile content

To compile the project, type "make" as shown in **Figure 3-6**. Then, type "ls" to check the generated ARM execution file "my first hps-fpga".



Figure 3-6 ARM C Project Compilation





### **3.6. Execute the Demo**

To execute the demo, please boot the Linux from the SD-card in DEO-Nano-SoC. Copy the execution file "my\_first\_hps-fpga" to the Linux directory, and type "chmod +x my\_first\_hps-fpga" to add execution attribute to the execute file. Use Quartus Programmer to configure FPGA with the **soc\_system.sof** generated in previous chapter. Then, type "./my\_first\_hps-fpga" to launch the ARM program. The LED on DEO-Nano-SoC will be expected to perform 60 times of LED light shift operation, and then the program is terminated.

For details about booting the Linux from SD-card, please refer to the document: DE0-Nano-SoC \_Getting\_Started\_Guide.pdf

For details about copying files to Linux directory, please refer to the document: My\_First\_HPS.pdf

