



# Laboratoire 5 : IP AXI4-lite avec I/O de la FPGA

Département : Technologies de l'Information et de la Communication (TIC)

Unité d'enseignement : System on chip on FPGA (SOCF)

Auteurs: Pierrick Muller

Professeur : Alberto Dassati, Etienne Messerli Assistant : Sébastien Masle, Sydney Hauke

Classe: SOCF-1-A Date: 8 mai 2020



#### Table des matières

| 1        | Introduction                  |                                                            |    |  |  |
|----------|-------------------------------|------------------------------------------------------------|----|--|--|
|          | 1.1                           | Objectifs                                                  | 2  |  |  |
|          | 1.2                           | Spécifications sans les interruptions                      | 2  |  |  |
| <b>2</b> | Analyse                       |                                                            |    |  |  |
|          | 2.1                           | Partie 1                                                   | 3  |  |  |
|          | 2.2                           | Partie 2                                                   | 3  |  |  |
|          | 2.3                           | Partie 3                                                   | 3  |  |  |
|          | 2.4                           | Adress Map finale                                          | 4  |  |  |
| 3        | Réalisation et implémentation |                                                            |    |  |  |
|          | 3.1                           | Ajout des PIOs                                             | 5  |  |  |
|          | 3.2                           | Activation des interruptions dans Qsys                     | 7  |  |  |
|          | 3.3                           | Programme pour les features sans interruption              |    |  |  |
|          | 3.4                           | Activation des registres du GIC et de la FPGA dans le code | 8  |  |  |
|          | 3.5                           | Routine d'interruption                                     | 9  |  |  |
| 4        | Sim                           | nulation et tests                                          | 10 |  |  |
| 5        | Cor                           | nclusion                                                   | 13 |  |  |
| 6        | Sign                          | natures                                                    | 13 |  |  |

### 1 Introduction

# 1.1 Objectifs

Ce laboratoire a pour but de réaliser une IP avec une interface AXI4-lite et connectée sur le bus Lightweight HPS-to-FPGA. Cette IP doit permettre d'accéder à des I/O câblées sur la partie FPGA via des registres. Vous devrez analyser le fonctionnement du bus AXI4-lite afin de concevoir une IP personnalisée pour les besoins du laboratoire.

# 1.2 Spécifications sans les interruptions

L'objectif est d'interfacer à l'aide d'une IPAXI4-litetous les I/O disponibles sur la FPGA, sans utiliser des composants PIO, soit les boutons (KEYs), les switchs (SW), les LEDs et les afficheurs 7 segments.

Votre IP AXI4-lite comprendra une constante 32 bits à l'offset 0x0 ainsi qu'un registre de test R/W à l'offset 0x4. Les offsets sont relatifs à l'adresse de base donnée à l'instance de l'IP dans Qsys.

#### Spécifications du programme :

Le but est d'allumer les LEDs selon l'état des boutons (KEYs) et interrupteurs

(switch) disponibles. Les afficheurs 7 segments sont dépendants de la constante définie dans l'IP. La spécification du fonctionnement est la suivante :

- Appui sur KEY0 : l'états des switches est copiés sur les LEDs. Les afficheurs HEX5 à HEX0 affichent en hexadécimal les bits 23 à 0 de la constante définie dans l'IP.
- Appui sur KEY1 : l'états inversesdes switches est copiés sur les LEDs. Les afficheurs HEX5 àHEX0 affichent en hexadécimal l'inversedes bits 23 à 0 de la constante définie dans l'IP
- Appui sur KEY2 : l'affichage des LEDset des afficheurs 7 segments subit une rotation à droite. Rotation d'un bit pour les LEDs, rotation d'un afficheur complet pour les afficheurs 7 segments.
- Appui sur KEY3 : l'affichage des LEDs et des afficheurs 7 segments subit une rotation à gauche. Rotation d'un bit pour les LEDs, rotation d'un afficheur complet pour les afficheurs 7 segments.

Dans une seconde partie, l'appui sur les boutons KEY2 ou KEY3 devra être géré à l'aide d'interruption vers le HPS (2ème partie).

— Votre design doit permettre degénérer une interruption lors de l'activation (détection de flanc) d'un des 4 boutons. Vous devez prévoir les accès et les flags nécessaires pour gérer l'interruption. Il doit être possible d'activer/masquer l'interruption pour chaque bouton.

# 2 Analyse

#### 2.1 Partie 1

Cette partie était principalement une marche à suivre. Il n'y avait pas réellement d'analyse à effectuer sur cette partie du laboratoire. Le but était de suivre la marche à suivre et de comprendre les actions que nous réalisions.

#### 2.2 Partie 2

Cette partie nous demandais d'ajouter la gestion des leds et des afficheurs 7 segments en fonction des switchs et sans gérer pour le moment les interruptions (Key2 et Key3). Ce qui est intéressant dans cette partie, c'est l'ajout des Keys et des afficheurs 7 segments en tant que PIO, en se servant de l'expérience reçu lors de la réalisation de la première partie.

#### 2.3 Partie 3

Cette partie demandait un peu plus d'analyses que les deux parties précédentes. Il n'y avait pas de PIO à ajouter, par contre la gestion des interruptions devait être mise en place pour chacun des différents éléments du système. Ainsi, il falait prendre en compte le CPU, le GIC et les PIO.

Pour ce qui concerne le CPU, la configuration des vecteurs d'interruptions était fournie, mais la routine d'interruption devait être implémentée.

Concernant le GIC, je me suis inspiré des exemples de codes fournis dans le document "Gic\_Altera\_Manuel\_short.pdf", et j'ai dû comprendre comment configurer le GIC. Voici les registres du GIC qui ont dû être configurés ou qui ont été utilisés lors de la routine d'interruption :

- ICCICR : Permet d'activer la transmission depuis le CPU interface jusqu'au CPU lui correspondant. C'est un enable.
- ICCPMR : Permet de set la priorité minimale pour qu'une interruption soit transmise au CPU.
- ICCIAR: Lors d'une interruption, contient l'ID de l'interruption en question.
- ICCEOIR : Permet au processeur de clear l'interrupt en écrivant l'ID correspondant à l'intérieur
- ICDDCR : Permet d'activer le distributor. C'est un enable.
- ICDISER: Permet d'activer une interruption (unmask)
- ICDIPTR : Permet de définir vers quel CPU doit être envoyé l'interruption.

A cela s'ajoutait deux registre, KEYS\_INTERRUPT\_ENABLE et KEY\_INTERRUPT\_REGISTER, qui permettait d'activer les interruptions des keys dans la FPGA et de récupérer ainsi que nettoyer lesdites interruptions.

Nous parlerons des valeurs qui ont été attribué à chaque registre dans la partie implémentation. Les adresses des registres sont basés sur les calculs fournis dans la documentation en fonction du numéro d'interruption 72, qui correspond au numéro d'interruption 0 de la FPGA (Nous verrons dans la partie implémentation d'où vient ce numéro).

# 2.4 Adress Map finale



FIGURE 1 – Laboratoire 2 : Address map

Le choix des offsets ci-dessus vient à la base d'un problème de compréhension. J'avais choisi ces offsets en fonction des adresses correspondantes dans le memory layout fournit dans le document "DE1-SoC\_Computer\_ARM.pdf". Si cela n'apporte pas de problème en tant que tel, il faut tout de même noter que j'aurais pu en choisir d'autres, en faisant en sorte qu'ils soient contiguës par exemple.

# 3 Réalisation et implémentation

J'ai choisi de ne pas présenter l'implémentation faite partie par partie, mais de présenter l'implémentation finale car elle contient les éléments correspondant à chacune des parties.

#### 3.1 Ajout des PIOs

Voici ce que l'on peut voir quand on ouvre le projet Qsys qui a été créé:



FIGURE 2 – Laboratoire 2 : PIOs

On peut voir que 4 parallel I/O (pio) ont été ajoutés, un pour chacun des éléments de notre système que l'on souhaitait gérer.

La configuration de ces pio est la suivante :

- pio leds: une longueur de 10 (Pour les 10 leds) et set en output
- pio\_switch: une longueur de 10 (Pour les 10 switchs) et set en input
- pio\_HEX : une longueur de 32 (8 bits par afficheur, avec 2 bits non utilisé, donc 4\*8 = 32) et set en output
- pio\_KEY: un longueur de 4(Pour les 4 boutons), set en input, avec l'edge capture en rising (Pour récupérer l'appui sur le bouton) et avec les irq actives(irq de type EDGE, afin de se basé sur l'état de l'edge capture pour créer une interruption)

Le câblage avec les autres éléments consistait à brancher toutes les clocks sur la clock du hps, ainsi que les resets (sur le reset du hps). De plus, pour le cas des KEYs, l'interruption est branchée sur la première "banque" d'interruption de la fpga qui va sur le hps (f2h\_irq0).

La génération du VHDL s'est passé sans problème, et j'ai dû ajouter les pio que j'avais créés au fichier DE1\_SoC\_top.vhd. Pour ce faire, j'ai utilisé qsys et "Show instantiation template" :



Figure 3 – Laboratoire 2: Instantiation template

et j'ai ensuite modifié le fichier DE1\_SoC\_top.vhd afin de permettre l'utilisation des pios. J'ai copié les lignes que j'avais trouvées dans l'instanciation template et je les ai mises dans l'import du composant qsys\_system dans le fichier DE1\_Soc\_top. De plus, j'ai dû mapper les entrées et sorties du composant dans le même fichier :

```
1 System : component qsys_system
2 port map (
   -- FPGA Side
   -- Global signals
9
   -- HPS Side
10
   -- DDR3 SDRAM
13 clk clk
                                             => CLOCK 50 i,
                                             => LEDR_o,
14 pio_leds_external_connection_export
pio_switch_external_connection_export => SW_I,
                                             => temp_hex_s,
  pio_hex_external_connection_export
17 pio_key_external_connection_export
                                             => KEY_i,
18 memory_mem_a => HPS_DDR3_ADDR_o,
memory_mem_ba => HPS_DDR3_BA_o,
memory_mem_ck => HPS_DDR3_CK_P_o,
memory_mem_ck_n => HPS_DDR3_CK_N_o,
memory_mem_cke => HPS_DDR3_CKE_o,
memory_mem_cs_n => HPS_DDR3_CS_N_o,
24 memory_mem_ras_n => HPS_DDR3_RAS_N_o,
memory_mem_reset_n => HPS_DDR3_RESET_N_o,
  memory_mem_dq => HPS_DDR3_DQ_io,
  memory_mem_dqs
                        => HPS DDR3 DQS P io,
```

```
30 memory_mem_dqs_n => HPS_DDR3_DQS_N_io,
31 memory_mem_odt => HPS_DDR3_ODT_o,
32 memory_mem_dm => HPS_DDR3_DM_o,
33 memory_oct_rzqin => HPS_DDR3_RZQ_i
34 );
35
36 HEXO_o <= temp_hex_s(6 downto 0);
37 HEX1_o <= temp_hex_s(14 downto 8);
38 HEX2_o <= temp_hex_s(22 downto 16);
39 HEX3_o <= temp_hex_s(30 downto 24);</pre>
```

Comme on peut le voir ci-dessus, j'ai de plus dû créer un signal temporaire permettant de dispatcher la sortie du pio HEX entre les différentes sorties HEX qui étaient présentes dans le fichier.

#### 3.2 Activation des interruptions dans Qsys

Dans Qsys, plusieurs configurations devaient être effectuées afin de pouvoir utiliser les interruptions. Voici une liste des démarches effectuées :

- Il faut activer dans la configuration de hps\_0 , sous "Fpga inteface" -> "interrupts", l'option "Enable FPGA-to-HPS interrupts"
- le/les PIO qui doit bénéficier des interruptions doit être linké sur l'un des deux champs "f2h\_irq0/1".
- Dans la colonne IRQ aussi, l'interruption du pio doit être linkée sur l'un des deux champs vu plus haut. Dans notre cas, l'interruption porte le numéro 0 et est linkée sur "f2h\_irq0", ce qui implique que le numéro de l'interruption sera le numéro 0 de la fpga (L'image ci-dessous montre le numéro de l'interruption, qui est donc 72)



FIGURE 4 – Laboratoire 2 : Numéro de l'interruption

Voici les modifications qui étaient nécessaire dans Qsys afin d'activer les interruptions (En plus de celle déja configurée lors de la création des pios)

# 3.3 Programme pour les features sans interruption

Le fonctionnement du programme pour les parties 1 et 2 du laboratoire a été entièrement commenté directement dans le code.

```
while(1)
while(1)
figure while(1)
while(1)
figure while(1
```

```
LEDS = 0x0;
7
            HEX3\_0 = \sim 0 \times 0;
            // On reporte l'état des switchs sur les leds
            LEDS = SWITCHS;
11
12
            // On effectue les manipulations demandées pour ...
13
                l'afficheur 7 segements pour KEY1
            \text{HEX3 0} = \sim (0 \times 0 \mid (\text{temp} \mid (\text{LEDS \& } 0 \times 200)) >> 9] << 24) \mid \dots
14
                 (temp[(LEDS \& 0x100) >> 8] << 16) | (temp[(LEDS \& ...
                0xF0) >> 4 ] << 8) | (temp[(LEDS & 0xF)]));
15
16
        // Si on appuie sur le bouton KEY2
17
       else if(!(KEYS & 0x2))
19
            // On eteind les leds et l'afficheur 7 segements
20
            LEDS = 0x0;
21
            HEX3\_0 = \sim 0 \times 0;
23
            // On reporte l'état des switchs sur les leds
24
            LEDS = \simSWITCHS;
25
26
            // On effectue les manipulations demandées pour ...
                l'afficheur 7 segements pour KEY2
            \text{HEX3}_0 = \sim (0x0 \mid \text{(temp[(LEDS & 0x200) >> 9]} << 24) \mid \dots
                 (temp[(LEDS \& 0x100) >> 8] << 16) | (temp[(LEDS \& ...
                0xF0) >> 4 ] << 8) | (temp[(LEDS & 0xF)]));
        }
29
30 }
```

# 3.4 Activation des registres du GIC et de la FPGA dans le code

Nous allons nous intéresser ici aux valeurs attribuées aux registres dans la partie de code ci-dessous (Les commentaires expliquent le choix des valeurs) :

#### 3.5 Routine d'interruption

Voici la routine d'interruption qui a été mise en place pour ce laboratoire :

```
1 // Define the IRQ exception handler
void __attribute__ ((interrupt)) __cs3_isr_irq(void)
       // On lit le registre de l'interface CPU pour savoir quel ...
          périphérique a causé l'interruption
       int interrupt_ID = ICCIAR;
       int hex_val, press;
                                              // On récupère le ...
       press = KEYS_INTERRUPT_REGISTER;
          bouton qui à causer l'interruption
       KEYS_INTERRUPT_REGISTER = press;
                                              // On nettoie ...
          l'interruption dans le registre des interruptions pour ...
          les KEYS
10
       // Si l'interruption à été causer par un bouton
11
       if(interrupt_ID == 72)
12
13
           if (press & 0x4)
                                   // KEY2
           {
15
               // On deplace les leds vers la droite ainsi que les ...
16
                  valeurs des afficheurs 7 segments
               LEDS = (LEDS >> 1) | ((LEDS & (0x1 )) << 9);
17
               hex_val = HEX3_0;
18
               HEX3\_0 = \sim 0x0;
19
               \text{HEX3}_0 = (0x0 \mid ((hex_val & (0x7F)) << 24) \mid ...
20
                   ((hex_val & (0x7F << 24)) >> 8) | ((hex_val &
                   (0x7F \ll 16)) \gg 8) \mid ((hex_val & (0x7F \ll 8)) \gg 8));
           }
21
22
           else if (press & 0x8)
                                    // KEY3
```

```
24
                // On deplace les leds vers la gauche ainsi que les ...
25
                    valeurs des afficheurs 7 segments
                LEDS = (LEDS << 1) | ((LEDS & (0x1 << 9))>>9);
27
                hex_val = HEX3_0;
                HEX3\_0 = \sim 0x0;
28
                \text{HEX3}_0 = (0x0 \mid ((hex_val & (0x7F << 16)) << 8) \mid ...
29
                    ((\text{hex val & }(0x7F << 8)) << 8) | ((\text{hex val & ...})
                    (0x7F)) << 8) | ((hex val & (0x7F << 24))>>24));
            }
30
31
32
33
       // On nettoie l'interruption dans le registre de ...
34
           interruptions pour le processeur
       ICCEOIR = interrupt_ID;
       return;
36
37
```

#### 4 Simulation et tests

Il n'y avait pas de simulation à faire dans ce laboratoire. Cependant, des tests ont été effectués tout au long du laboratoire. De plus, le contrôle du fonctionnement de l'application avait été validé auprès de l'assistant avant la période de confinement. Je note tout de même ici les tests qui ont été effectués par moi-même afin de m'assurer du bon fonctionnement de l'application :

J'ai donc commencé par le plus simple. Tous les switchs down et un appui sur key 0 suivi d'un appui sur key 1:



Figure 5 – Laboratoire 2 : Appui key 0 avec switch down



FIGURE 6 – Laboratoire 2 : Appui key 1 avec switch down J'ai ensuite continué en alternant un switch down et un switch up en commençant par activer le switch 0, et j'ai appuyé sur key 0 puis sur key 1 :



FIGURE 7 – Laboratoire 2 : Appui key 0 avec switch up/down



FIGURE 8 – Laboratoire 2: Appui key 1 avec switch up/down Les valeurs correspondaient au fonctionnement demandé par la donnée. Toujours avec cet état de switch, j'ai essayé les interruptions en appuyant deux fois sur key 3 puis une fois sur key 2



Figure 9 – Laboratoire 2 : Appui 2 fois sur key 3



Figure 10 – Laboratoire 2 : Appui 1 fois sur key 2

Tout à fonctionner sans problème. J'ai testé la modification des switchs lorsque j'appuie sur key 2 ou key 3 et conformément à la donnée, les leds et l'afficheur 7 segment n'était pas impacté.

## 5 Conclusion

Ce laboratoire nous a permis de nous familiariser avec les principes de PIO et d'interactions entre FPGA et HPS. De plus, il nous a permis de nous assurer que la machine virtuelle mise en place pour le cours SOCF pour toute la durée de la période de travail à la maison fonctionne bien.

# 6 Signatures

Yverdon-les-Bains le 8 mai 2020

#### Pierrick Muller

# Table des figures

| 1 | Laboratoire 2 : Address map                     |
|---|-------------------------------------------------|
| 2 | Laboratoire 2: PIOs                             |
| 3 | Laboratoire 2: Instantiation template           |
| 4 | Laboratoire 2 : Numéro de l'interruption        |
| 5 | Laboratoire 2 : Appui key 0 avec switch down    |
| 6 | Laboratoire 2 : Appui key 1 avec switch down    |
| 7 | Laboratoire 2 : Appui key 0 avec switch up/down |
| 8 | Laboratoire 2 : Appui key 1 avec switch up/down |

| 9  | Laboratoire 2 : Appui 2 fois sur key 3 | 12 |
|----|----------------------------------------|----|
| 10 | Laboratoire 2 : Appui 1 fois sur key 2 | 13 |