

# Embedded Systems

## Lab 7. Interrupts

In this lab, we will study the Zynq SoC's interrupt system structure. The Zynq SoC uses a Generic Interrupt Controller (GIC) to process interrupts. The procedures needed to implement an interrupt-driven system are explained in details. An interrupt-driven application is developed with detailed interrupt handlers (ISRs) and interrupt setup functions.





Lab7 Block Diagram



Lab7 Board Interface

The GIC is circled in red in the figure below:



Zynq Architecture- GIC circled in red

The GIC handles interrupts from the following sources:

- Software-generated interrupts – There are 16 such interrupts for each processor. They can interrupt one or both of the Zynq SoC's ARM® Cortex™-A9 processor cores.
- Shared peripheral interrupts – Numbering 60 in total, these interrupts can come from the I/O peripherals, or to and from the programmable logic (PL) side of the device. They are shared between the Zynq SoC's two CPUs.

- Private peripheral interrupts – The five interrupts in this category are private to each CPU—for example CPU timer, CPU watchdog timer and dedicated PL-to-CPU interrupt.



Interrupt System Structure for the Zynq

When an interrupt occurs within the Zynq SoC, the concerned processor will take the following actions:

1. The interrupt is shown as pending.
2. The processor stops executing the current program.
3. The processor saves the state of the program in the stack to allow processing to continue once it has handled the interrupt.
4. The processor executes the interrupt service routine (ISR), which defines how the interrupt is to be handled.
5. The processor resumes operation of the interrupted program after restoring it from the stack.

It is important to keep in mind that interrupts are asynchronous events, it is possible for multiple interrupts to occur at the same time. To address this issue, the processor prioritizes interrupts such that it can service the highest-priority interrupt pending first. With this being said, it should be clear by now why one needs to write two functions to handle an interrupt: First, an interrupt service routine (ISR) (also known as the interrupt handler) to define the actions that will take place when the interrupt occurs, and second, an interrupt setup function to configure the interrupt settings (priority, level sensitivity for hardware interrupt, etc..).

For this lab, the hardware setup contains a GPIO controller and the ZedboardOLED controller, both connected to the PS through AXI interconnect as illustrated in the block diagram above. The PS configuration is going to be set up to accept interrupts from the PL. On the PL side of the chip, the GPIO will be configured to support interrupts and generate an interrupt signal every time a push button is pressed. The OLED will act as a terminal or a monitor, it is going to help us in observing the flow of execution. The main program is a simple infinite loop that displays the English alphabet sequentially (A-Z) and once it reaches the letter "Z" it rolls back to "A". The main program will be halted, and the interrupt handler (ISR) will be executed every time an interrupt occurs (i.e. a push button is pressed). In the ISR, the GPIO's data register will be read to figure out which button was pressed out of the five push buttons (BTNC, BTND, BTNR, BTNL, BTNU), then this button will be displayed on the OLED using the `print_message()` function of the ZedboardOLED driver. Upon completion serving the interrupt, the main program will resume at a letter exactly after the letter it was halted at. For instance, if we pressed a button when the letter "D" is being displayed on the OLED, the main program should resume at letter "E". The next figure shows a simplified flowchart of the application's behavior (Typical interrupt driven application behavior).



## Objectives

1. Implement a demonstrative interrupt-driven embedded system.
2. Explore the generic interrupt controller of the Zynq SoC.
3. Learn to differentiate between the different sources of interrupts within the Zynq SoC.
4. Expand the functionality of the GPIO controller to support interrupts.
5. Write an interrupt handler (aka ISR) and an interrupt setup function that utilizes the generic interrupt controller (SCUGIC) driver, the PS and the GPIO interrupt related functions.
6. Learn how to increase the stack and heap memory allocated for the program's execution.

## Procedures:

### **A-Create a project in Vivado to target the Zedboard:**

- 1.Launch Vivado Design Suite.
- 2.Click on File->New Project, to open New Project Wizard then click Next.



- 3.In the next window enter “lab7” as the Project Name, specify the directory in which to store the project files “c:/user/”- create the folder “lab7” in your C: drive, which will contain all lab’s files, leave the Create Project Subdirectory option checked.



### Project Type

4. Next, specify the project type, Use the default selection RTL Project and check Do not specify sources at this time. RTL stands for Register Transfer Language, by choosing this option we will have the flexibility to add/modify source files later on as needed.

5. Then, specify the board that you want to test your project on, for all the labs on Zynq Soc Training we will use the Zedboard Zynq Evaluation and Development Kit version D. Click Next, then click Finish.



Default part

The New Project wizard closes and the project you just created opens in Vivado. The board you choose in the wizard has a direct impact on how IP Integrator functions. IP Integrator is board aware and will automatically assign dedicated PS ports to physical pin locations mapped to that specific board. It also applies the correct I/O standard, saving the designer's time.



Vivado Interface

## B-Create ARM processor system with interrupt support:

1. Click on Create Block Design in IP Integrator available on the left top side of Vivado.
2. Type a name for the module and click OK. For this example, use the name: “zynq\_interrupt\_system”.



3. Add the ZYNQ7 Processing System by launching the Add IP wizard

 and typing “zynq”. Choose “ZYNQ7 Processing System”.



4. Double click on the ZYNQ7 Processing System block, to open the Re-Customize IP window. Select **Interrupts** from the Page Navigator on the left-hand side and expand the menu on the right. Since we want to allow interrupts from the programmable logic to the processing system, tick the box to enable **Fabric Interrupts**; fabric is another name for the PL; then tick to enable the shared interrupt port **IRQ\_F2P[15:0]** (read as Interrupt Request\_Fabric to Processing System) as show in the figure below. This means interrupts from the PL can be connected to the interrupt controller within the Zynq PS. Also, make sure to enable UART1 in MIO Configuration (In case we wanted to use a terminal).



Enable Fabric Interrupts

Given that you followed the above directions correctly, the Zynq PS system should look identical to the one shown below:



### ZYNQ7 Processing System with Fabric Shared Interrupt Port Enabled

However, If it is not identical, double click on the block again to re-customize it, and turn ON/OFF the affiliated settings. Notice the shared interrupt port is enabled (enclosed in a red rectangle). This is the port that the PL generated interrupts should be connected to.

Click Run Block Automation in the green information bar. Make sure to **deselect** the option **Apply Board Preset** otherwise all the customization we just did will be lost. Click OK.



Block Automation

Notice that the FIXED\_IO and DDR are now connected to external ports.



### **C-Add a GPIO controller and enable its interrupt port:**

- 1.Add a GPIO controller, by launching the Add IP wizard and typing “**gpio**”.
- 2.Click on Run Connection Automation,select GPIO first and make sure that **btms\_5bits** is selected in the Select Board Part Interface.



Run Connection Automation – GPIO Page

Also make sure that S\_AXI is selected and the Clock Connection is set to Auto.



Run Connection Automation- S\_AXI Page

Click OK to automatically connect the slave interfaces of the GPIO to the master interface of the PS through the AXI Interconnect block, and also to connect its external ports to the five push buttons on the Zedboard.

Click on the Regenerate Layout to arrange the blocks in your design.



Block Diagram After Adding GPIO

3.Double click on the GPIO block to open the Re-customize IP window. Click the IP Configuration tab and enable interrupts by clicking in the box highlighted in figure below, then click OK. This will add an interrupt request port to the GPIO block.



4.Make a connection between the newly created interrupt request port of the GPIO block and the shared interrupt port of the Zynq PS by dragging the mouse's cursor.(hover the mouse until its shape changes to a pencil shape).



Connect GPIO interrupt request port to the shared interrupt port of PS

## **D-Add the ZedboardOLED controller:**

(Add the Zedboard OLED IP to Vivado IP repository)

Before proceeding with this part, make sure you have downloaded the archive file(compressed source code files) of the Zedboard OLED IP core. After the download is complete, create a new folder on your “**C:\**” drive and name it “oled\_lab7”. Extract the archive file inside the newly created folder.

In Vivado, click on the **Project Settings** tab available in the Flow Navigator pane. Select **IP**, then click on the green + symbol in the IP Repositories subview. A browsing window will open up, browse to the location where you extracted the IP core “**C:\oled\_lab7**” , click Select.



After that, select the directory in which the IP for the oled display is located("oled\_lab7").

Now, Vivado detects a new IP in this directory, click Apply then OK.



With this you have added the ZedboardOLED\_v1.0 to the IP repository of the current project, the next step is to add it to the block design and connect it to the Zynq processing system from one side, and to the OLED panel from the other side using external ports.

## E-Connect the ZedboardOLED IP core

1. Launch the Add IP wizard, and type “oled”. The ZedboardOLED\_V1.0 will show up, double click on the IP to add it to the block design.



2. Notice that the Designer Assistance is available now in the green information bar. It will take care of connecting the IP core to the AXI subsystem of the processing system, configuring the clock and reset, assigning base address (0x43C00000) to the IP, and adding the necessary hardware for the interconnect and synchronization. Click on the Run Connection Automation , leave the default settings as is, and click OK.

The block diagram should look something similar to this  
(if not, right click on any free location on the block diagram and select **Regenerate Layout**).



The Designer Assistance has generously done for us the internal connections of the ZedboardOLED IP core. The interface between the ZedboardOLED and the processing system is a well-known protocol (AXI-Lite Slave) for the Designer Assistance to do the connections autonomously, however, it doesn't know much about the external connections of ZedboardOLED to the OLED display panel, and that is why its still unconnected yet.

Now, we have no other alternative than doing these connections ourselves, hover the mouse on the **ZedboardOLED SDIN port** until it changes to a pencil shape, then right click and select **Make External**. Repeat the same process for (**SCLK, DC , RES ,VBAT ,VDD**) ports.



3. Right click on any free location on the block diagram and select Regenerate Layout. This will organize the blocks in the design in a neat way. The final layout might look like the one below:



Now we have finished connecting the IP to the processing system through AXI-interface, next step is to connect the external ports we have previously defined of the IP core to the actual Zynq pins hardwired to the OLED panel(the information of these pins are obtained from Zedboard User Manual).

4.In the Flow Navigator window, select **Add Sources** from the Project Manager section.



The Add Sources dialogue will open. Select **Add or Create Constraints**.



Click Next, then click on the green + symbol and select **Create File**.



In the next window, Select **XDC** as the File type and enter oled\_constraints as the File name.



Click OK, then click Finish in the next window to create the file and close the dialogue.

In the **Sources tab**, expand the **Constraints** group and open the newly created XDC file by double clicking on **oled\_constraints.xdc**



Add the following lines to the constraints file:

```
set_property PACKAGE_PIN U10 [get_ports DC]
set_property PACKAGE_PIN U9 [get_ports RES]
set_property PACKAGE_PIN AB12 [get_ports SCLK]
set_property PACKAGE_PIN AA12 [get_ports SDIN]
set_property PACKAGE_PIN U11 [get_ports VBAT]
set_property PACKAGE_PIN U12 [get_ports VDD]
set_property IOSTANDARD LVCMOS33 [get_ports DC]
set_property IOSTANDARD LVCMOS33 [get_ports RES]
set_property IOSTANDARD LVCMOS33 [get_ports SCLK]
set_property IOSTANDARD LVCMOS33 [get_ports SDIN]
set_property IOSTANDARD LVCMOS33 [get_ports VBAT]
set_property IOSTANDARD LVCMOS33 [get_ports VDD]
```

These physical constraints connects the external ports of the ZedboardOLED IP core to specific pins on the Zynq device. These pins are hardwired to the OLED panel on the Zedboard(Again, the information of these pins are obtained from Zedboard User Manual). Save the constraints file by pressing (Ctrl+S).

5. In the Sources pane, Right click on system.bd and select **Create HDL Wrapper** to create the top level Verilog file from the block diagram. Select **Let Vivado manage wrapper and auto-update** when prompted with the next message. Notice that system\_wrapper.v got created and placed at the top of the design sources hierarchy.



After that, right click on “zynq\_interrupt\_system...” from the Source pane ,choose **Generate Output Products** and click **Generate**.



### **E-Generate Bitstream:**

1. In the Program and Debug section in the Flow Navigator pane, click **Generate Bitstream**. A dialog box will appear to ask you to save the modification you made, click **Save**.



Generating the Bitstream may take a while to complete, depending on the performance of your machine. After the bitstream generation completes select View Reports in the dialog box, and click OK.

### **F-Export hardware design to SDK:**

1. Click File > Export > Export Hardware, make sure you select **Include bitstream**.



2. Select File>**Launch SDK**. This will open up Launch SDK dialog. Leave default options as is and click OK. All the hardware files related to design have now been exported to SDK and we can work on the software.

We are now done with the hardware configuration of the system. You may close Vivado.

## G-Working with SDK:

- 1.In SDK, select File > New > Application Project.
- 2.In the next window, enter the parameters as provided in the snapshot below:



In the next window, select **Empty Application** from the available templates and click Finish. This will compile the BSP and the related drivers.

3. Expand interrupt\_test directory , right click on src directory, New→Source File.



4. In the next window that shows up type “lab7.c” in the source file and click Finish. This will create an empty C file in the src directory.



5. Paste the content of lab7.c in in the editor view of lab7.c of SDK , and click on Save or hit (Ctrl+S) , by doing so both the lab7 application, and its BSP are compiled automatically and the executable .elf file is generated( the default settings of SDK triggers compilation with save). This is the executable file that the ARM processor on the PS side of the Zynq will execute.

There are four functions in application lab7.c : Function **main()** first initializes the peripherals & set directions of gpio, then it calls **IntcInitFunction()**, which in turns initialize interrupt controller, register GPIO interrupt handler( which means associate a function with the interrupt coming from GPIO), and enable GPIO interrupts in the GPIO itself and in the GIC. IntcInitFunction() also calls a sub-function **InterruptSystemSetup()** to register GIC interrupt handler.

`IntcInitFunction()` returns to `main()` which enters an infinite loop (lines 140-145). In the infinite loop it function `print_message()` is called to display the alphabet on the OLED. When an interrupt is triggered ( by pressing a push button in this example) the processor stops executing `main()` and moves its attention to the function associated with the source of interrupt (in our case it's the `BTN_Intr_Handler`) .`BTN_Intr_Handler()` reads the GPIO's data register to figure out which button was pressed out of the five push buttons (BTNC, BTND, BTNR, BTNL, BTNU) and then prints this button on the OLED using the `print_message()` function of the ZedboardOLED driver.

In this lab, the stack will be used extensively to swap in and out different parts of the code ; namely the main function and ISR function; therefore we need to enlarge the memory reserved for the stack just to stay on the safe side. This can be done by right click on the `interrupt_test` directory and select Generate Linker Script. In the Linker script we can choose where our program will reside(DDR memory or RAM ) and also we can configure the head and stack memories allocated for our application.



Change the stack and heap size from 1KB to 10 KB by adding a zero at the end of the 1024 number. The stack and heap memories are used to save information related to function calls and returns, which are happening quite often in our code. Click Generate and select Yes to overwrite existing file.

### **H-Download bitstream and run the application(Hardware Verification)**

1. Select Xilinx Tools→ **Program FPGA** to download the Bitstream (this will take few seconds).



2. Select Xilinx Tools→ **XSDB Console** to open it. In the console, copy-paste the following commands so as to open a terminal window to observe the results of your code:

```
connect -host localhost -port 3121
targets
targets 2
jtagterminal -start
```

3. Select interrupt\_test project directory → Run As → **Launch on Hardware(System Debugger)** to run the oled\_test application on the ARM processor.

Once the application is loaded, you should observe the English alphabet being printed at page 0 of position 0 of the OLED. If a button from the five push buttons of Zedboard is pressed this would call the ISR of the GPIO which displays which button was pressed on the OLED and then it returns back to the main program to resume printing the letter following the one it was interrupted at.

### **ΑΣΚΗΣΕΙΣ:**

- 1) Να τροποποιήσετε τον κώδικα σας, έτσι ώστε να τυπώνεται στην 2ή γραμμή της οθόνης ποιό button πατήθηκε κατά το interrupt(BTN\* WAS PRESSED) κάθε 1 sec. Επίσης το αλφάβητο να τυπώνεται στο 8ό block της 1ής γραμμής της oled οθόνης.
- 2) Να τροποποιήσετε τον κώδικα σας, ούτως ώστε να τυπώνεται στο terminal πόσες φορές εχει πατηθεί το εκάστοτε button κατά τη διάρκεια του interrupt.
- 3) Επιπλέον, με τη χρήση ενός πίνακα και της συνάρτησης sprintf της C, να τυπώνετε και στην 3ή γραμμή της oled οθόνης τον εκάστοτε counter του ερωτήματος 2.

**tip:** Μελετήστε συντόμως το αρχείο “Oled Reference Manual.pdf”, ώστε να κατανοήσετε τη λειτουργία της oled οθόνης του Zedboard.