#### CT2 Praktikum

## **General Purpose Input/Output (GPIO)**

## 1 Einleitung

GPIOs erlauben eine universelle Verwendung von Ein- und Ausgabepins. Dazu müssen Sie vor dem Zugriff konfiguriert werden. Lesen und schreiben von Werten geschieht in der Regel mittels Register (Abbildung 1). Die Datei reg\_stm32f4xx.h definiert Strukturen und Makros für diese Registerzugriffe.



Abbildung 1: Zugriff auf GPIO mittels Register

In diesem Praktikum realisieren Sie die Konfiguration und die Ein- und Ausgabe mittels Registerzugriff für die GPIO der verwendeten Prozessorfamilie STM32F4xx.

#### 2 Lernziele

- Sie können GPIO mit Hilfe von technischen Dokumentationen initialisieren.
- Sie sind in der Lage GPIO zur Ein- und Ausgabe zu verwenden.
- Ihr Wissen zu den verschiedenen Konfigurationsoptionen ist gefestigt.
- Sie können die Strukturen des untersten Hardware Abstraction Layers (HAL), d.h. die Basisadressen, Macros und structs in reg\_stmf4xx.h, zur Konfiguration von Registern in eigenen Programmen einsetzen.

#### 3 Material

- 1x CT Board
- 3x Verbindungsdrähte (female-female)
- 2x Schutzkappen

## 4 Vorbereitungsfragen

Bereiten Sie sich mit der Beantwortung der folgenden Fragen auf das Praktikum vor. Öffnen Sie das vorbereitete Projekt in Keil uVision und 'builden' Sie es. Analysieren Sie den gegebenen Source Code.

| 1. | Ein Registerzu | ugriff | f erfo | lgt beisp | oielsw | eise üb | er GI | PIOB- | >MODE | R. Da | bei gibt G | <b>РІОВ</b> die |
|----|----------------|--------|--------|-----------|--------|---------|-------|-------|-------|-------|------------|-----------------|
|    | Basisadresse   | an     | und    | MODER     | den    | Offset. | Wo    | sind  | diese | zwei  | Symbole    | definiert       |
|    | (Dateiname ur  | nd Z   | eile). |           |        |         |       |       |       |       |            |                 |
|    |                |        |        |           |        |         |       |       |       |       |            |                 |
|    |                |        |        |           |        |         |       |       |       |       |            |                 |

Sie können gegebene Definitionen direkt von Keil suchen lassen. Klicken Sie dazu mit der rechten Maustaste auf den Text und wählen Sie "Go to Definition of…" aus. **Voraussetzung ist**, dass Sie mindestens einmal einen Build durchgeführt haben.

2. Was macht das folgende Macro?

```
#define GPIOB ((reg_gpio_t *) 0x40020400)
```

\_\_\_\_\_

3. Bitte beschreiben Sie in ein bis zwei Sätzen, was struct reg\_gpio\_t enthält?

4. Erklären Sie, wie das C-Statement

GPIOB->MODER = 0x00000280;

funktioniert. Wie erfolgt der Zugriff auf das Register?

## 5 Applikation und Testaufbau mit den Verbindungsdrähten

GPIO A und B sind direkt auf die Ports P5 und P6 des CT Boards herausgeführt. Die Pinbelegung für GPIO B / Port P6 ist in Abbildung 2 dargestellt, das Pin-Mapping von GPIO A auf P5 ist identisch. Details finden sich im CT-Board Wiki / GPIO (https://ennis.zhaw.ch).



Abbildung 2: Pin-Mapping GPIO A und B auf Ports P5 und P6

In diesem Versuch werden Sie die Position der drei DIP-Schalter S10..S8 einlesen, die Werte über GPIO PB2..PB0 ausgeben und über GPIO PA2..PA0 wieder einlesen. Die eingelesenen Informationen werden Sie auf LED18..LED16 anzeigen. Zusätzlich verwenden wir LED10..LED8 und LED2..LED0 zu Debugging Zwecken. Abbildung 3 zeigt den Signalfluss.

Hierzu müssen die GPIO Ports entsprechend als Inputs und Outputs konfiguriert werden.



Abbildung 3: Signalfluss

Stellen Sie noch keine Verbindung zwischen den Pins auf dem CT-Board her!!!

#### Eingelesene Werte GPIOA Verbindungen über isolierte □C2 □C3 □C4 222822 Drähte LED19..16 **GPIOA** Schutzkappen für den Schutz der (Input) JT1 restlichen Pins R1002 R1001 **GPIOB** LED11..B LED3..0 (Output) **DIP-Switches zur** Eingelesene DIP-Switch Positionen Zurückgelesene Bedienung Ausgabe GPIOB

## Abbildung 4 zeigt die verwendeten Elemente des CT-Boards.

Abbildung 4: Im Versuch verwendete Elemente des CT-Boards

## 6 Aufgaben

Im Folgenden implementieren Sie schrittweise einen einfachen Registerzugriff für die GPIO der Microcontrollerfamilie STM32F4xx.

Ergänzen Sie den vorgegebenen Code an den dafür gekennzeichneten Stellen. Verwenden Sie die Konstanten, Macros und structs aus reg stm32f4xx.h, um auf die Register zuzugreifen.

Um die Eigenschaften verschiedener GPIO-Konfigurationen zu untersuchen, konfigurieren wir die Ein- und Ausgänge unterschiedlich:

| GPIO | Mode/Type         | Pull-up/Pull-down | Speed        | Observe on |
|------|-------------------|-------------------|--------------|------------|
| PB0  | Push-Pull Output  | NO Pull-up/-down  | Low Speed    | LED0       |
| PB1  | Open Drain Output | NO Pull-up/-down  | Medium Speed | LED1       |
| PB2  | Open Drain Output | Pull-up           | High Speed   | LED2       |
|      |                   |                   |              |            |
| PA0  | Input             | Pull-down         | -            | LED16      |
| PA1  | Input             | Pull-up           | -            | LED17      |
| PA2  | Input             | NO Pull-up/-down  | -            | LED18      |

## 6.1 Konfiguration der Eingänge: PA2..PA0

Überlegen Sie, welche Werte für die Konfiguration der Eingänge in die Konfigurationsregister von Port GPIO A geschrieben werden müssen.

Füllen Sie dazu die untenstehende Tabelle aus (Sie haben in der Vorlesungs-Übungsaufgabe zur GPIO Konfiguration zwei unterschiedliche Arten kennen gelernt, wie die Masken für das Löschen und Setzen der Bits definiert werden können).

|                        | Port<br>Register<br>Name | Register Bits (Nr) | Bit Werte<br>(Binär) | Clear/Set Maske<br>(shift, invert) | Clear/Set Maske<br>(absolut, hex) |
|------------------------|--------------------------|--------------------|----------------------|------------------------------------|-----------------------------------|
|                        | PA0                      | Input              | _                    |                                    |                                   |
|                        | MODER                    | 10                 | 00                   | clr: ~(0x03 << 0)                  | clr: 0xFFFFFFC                    |
|                        | WOBER                    | 10                 | 00                   | set: 0x00 << 0                     | set: 0x00                         |
| Direction /            | PA1                      | Input              |                      |                                    |                                   |
| Mode                   |                          |                    |                      | clr:                               | clr:                              |
| Mode                   |                          |                    |                      | set:                               | set:                              |
|                        | PA2                      | Input              |                      |                                    |                                   |
|                        |                          |                    |                      | clr:                               | clr:                              |
|                        |                          |                    |                      | set:                               | set:                              |
|                        | PA0                      | Pull-down          | _                    |                                    |                                   |
|                        |                          |                    |                      | clr:                               | clr:                              |
|                        |                          |                    |                      | set:                               | set:                              |
| Dull up /              | PA1                      | Pull-up            |                      |                                    |                                   |
| Pull-up /<br>Pull-down |                          |                    |                      | clr:                               | clr:                              |
| i dii dowii            |                          |                    |                      | set:                               | set:                              |
|                        | PA2                      | No Pull-up/down    |                      |                                    |                                   |
|                        |                          |                    |                      | clr:                               | clr:                              |
|                        |                          |                    |                      | set:                               | set:                              |

Konfigurieren Sie die GPIOs im vorgegebenen Programmrahmen unter init\_GPIOA() und lesen Sie in der Endlosschleife im Hauptprogramm die Werte von PA2..PA0 ein. Zeigen Sie diese auf LED18..LED16 an. Verwenden Sie hierzu struct CT\_LED->BYTE

Da der Elementzugriff im HAL über Pointer (\*) erfolgt, muss anstatt des Punktes (.) ein Pfeil (->) verwendet werden.

Beispiel: pointer->element entspricht (\*pointer).element

a) Bevor Sie das Programm ausführen: Welche LEDs erwarten sie bei der Ausführung hell, welche dunkel?

Hell: Dunkel: Undefiniert:

- b) Testen Sie Ihr Programm auf dem CT-Board. Entspricht das Ergebnis Ihren Erwartungen?
- c) Streichen Sie mit dem Finger über die Pins von P5 und beobachten Sie das Verhalten der LEDs. Wie erklären Sie sich dieses Verhalten?

Sie können in der Keil uVision Entwicklungsumgebung die Werte der verschiedenen GPIO Register anschauen, indem Sie im Debugger Menü View -> System Viewer -> GPIO -> GPIOA auswählen und dann im Debug-Modus durch ihr Programm steppen. Im Beispiel ist für PA0 Pull-down gesetzt. 1-ZHAW\12-CT2\04-Praktika\P\_04\_GPI Core Peripherals k - μVision GPIOA View Project Flash Debug Peri ndow Help RNG ✓ Status Bar **罪 //≟ //½ / 2** DCMI Property Value Toolbars MODER1 0x00 💬 - 💹 - 🔣 MODER0 0x00 Project Window DBG ⊕ OTYPER Books Window DMΔ ▶ PDR |= (0x0 GPIOB\_OSPEE... PUPDR {} Functions Window RCC PUPDR15 0x01 (). Templates Window ✓ GPIOA GPIO PUPDR14 0x02 Source Browser Window SYSCFG ✓ GPIOB PUPDR13 0x01 Build Output Window GPIOC PUPDR12 0x00 X Error List Window PUPDR11 0×00 SDIO PUPDR10 0×00 Find In Files Window ADC **GPIOE** PUPDR9 0x00 USART **GPIOF** Command Window PUPDR8 DAC GPIOG PUPDR7 0x00 Disassembly Window PUPDR6 PWR 0x00 GPIOH Symbols Window PUPDR5 0x00 I2C GPIOI Registers Window PUPDR4 0x00 IWDG Call Stack Window PUPDR3 0x00 WWDG GPIOK PUPDR2 0x00 Watch Windows BLE(); RTC PHIDDR1 0×00 Memory Windows BLE(); PUPDRO TIM Serial Windows ± IDR Ethernet ▶ GPIOA specif Analysis Windows MODER CRC [Bits 31..0] RW (@ 0x40020000) GPIO port mode register Trace PEEDR = 0x00USB\_OTG\_FS  $PDR = 0 \times 6400$ ▶ YPER = 0x000 CAN Toolbox Window GPIOB GPIOA RL = 0x00000NVIC RH = 0x00000Periodic Window Update FLASH Achtung: Das Menu erscheint erst im Debugger.

## 6.2 Konfiguration der Ausgänge: PB2..PB0

Implementieren Sie die notwendigen Registerzugriffe für eine Output-Konfiguration. Gleich wie bei der Konfiguration der Inputs, müssen auch hier die entsprechenden Registerbits zuerst gelöscht und anschliessend richtig beschrieben werden.

|             | Port<br>Register<br>Name | Register Bits (Nr)  | Bit<br>Werte<br>(Binär) | Clear/Set Maske<br>(shift, invert) | Clear/Set Maske<br>(absolut, hex) |
|-------------|--------------------------|---------------------|-------------------------|------------------------------------|-----------------------------------|
|             | PB0                      | Gen. Purpose Output |                         |                                    |                                   |
|             |                          |                     |                         | clr:                               | clr:                              |
|             |                          |                     |                         | set:                               | set:                              |
| Direction / | PB1                      | Gen. Purpose Output |                         |                                    |                                   |
| Mode        |                          |                     |                         | clr:                               | clr:                              |
|             | 22.0                     |                     |                         | set:                               | set:                              |
|             | PB2                      | Gen. Purpose Output |                         |                                    | Ι.                                |
|             |                          |                     |                         | clr:                               | clr:                              |
|             |                          |                     |                         | set:                               | set:                              |
|             | PB0                      | Push-Pull           |                         |                                    | T                                 |
|             |                          |                     |                         | clr:                               | clr:                              |
|             |                          |                     |                         | set:                               | set:                              |
| _           | PB1                      | Open Drain          |                         | 1.                                 |                                   |
| Type        |                          |                     |                         | clr:                               | clr:                              |
|             | DDG                      | 0 0 :               |                         | set:                               | set:                              |
|             | PB2                      | Open Drain          |                         |                                    | Τ,                                |
|             |                          |                     |                         | clr:                               | clr:                              |
|             |                          |                     |                         | set:                               | set:                              |
|             | PB0                      | No Pull-up/-down    |                         | 1.                                 | Ι.                                |
|             |                          |                     |                         | clr:                               | clr:                              |
|             | <b>DD</b> 4              | N 5 11 / 1          |                         | set:                               | set:                              |
| Pull-up /   | PB1                      | No Pull-up/-down    |                         |                                    | Ι.                                |
| Pull-down   |                          |                     |                         | clr:                               | clr:                              |
|             | DDO                      | Dellara             |                         | set:                               | set:                              |
|             | PB2                      | Pull-up             |                         | olr                                | olr                               |
|             |                          |                     |                         | clr:                               | clr:                              |
|             |                          |                     |                         | set:                               | set:                              |
|             | PB0                      | Low                 |                         | -1-                                | .1                                |
|             |                          |                     |                         | clr:                               | clr:                              |
|             | DD4                      | NA o elitare        |                         | set:                               | set:                              |
| Cnaad       | PB1                      | Medium              |                         | ole                                | olm                               |
| Speed       |                          |                     |                         | clr:                               | clr:                              |
|             | DD2                      | Lligh               |                         | set:                               | set:                              |
|             | PB2                      | High                |                         | clr:                               | clr:                              |
|             |                          |                     |                         | clr:                               | clr:                              |
|             |                          |                     |                         | set:                               | set:                              |

Achten Sie auf **OTYPER** (Output Type Register). Es hat eine andere Bitbreite als die übrigen Register.

Konfigurieren Sie PB2..PB0 als Output wie in der Tabelle vorbereitet in der Funktion init\_GPIOB() .

#### 6.3 Ausgabe auf PB2..PB0

Erweitern Sie die Implementierung in der Endlosscheife im Hauptprogramm so, dass Sie die Werte der DIP Switches S10..S8 einlesen und sowohl auf LED10..LED8 ausgeben als auch in das **ODR** (Output Data Register) schreiben.

Lesen Sie den ausgegebenen Wert aus dem ODR wieder zurück und geben Sie ihn auf LED2..LED0 aus.

LED2..LED0 sollten nun die Position von S10..S8 anzeigen, gleich wie LED10..LED8.

#### 6.4 Vervollständigen des Aufbaus

Trennen Sie während des Anbringens der Drähte das CT Board von der Versorgungsspannung.

Bei gröberen Fehlern kann Ihr CT-Board Board Schaden nehmen und zerstört werden. (siehe auch Kapitel 8.2).

Verbinden Sie die jeweils entsprechenden Pins über die Verbindungsdrähte, also PA0 mit PB0, PA1 mit PB1 und PA2 mit PB2.

Schalten Sie die DIP Switches S10..S8 und prüfen Sie, ob die Übertragung über die Verbindungsdrähte funktioniert. Stimmen die Anzeigen auf allen LEDs miteinander überein?

## 6.5 Untersuchung verschiedener I/O Kombinationen

Durch Umstecken der Verbindungsdrähte können Sie nun unterschiedlich konfigurierte Ausund Eingänge miteinander verschalten. Prüfen Sie alle Kombinationen und tragen die Ergebnisse in die folgende Tabelle ein. Zur Stabilitätsprüfung können Sie wieder die «Fingerprobe» vornehmen (wenn ihr Finger schmal genug ist, dass sie die Pins von P5 von unten berühren können). Für die Kombination Push-Pull / Pull-Down sind die Ergebnisse bereits eingetragen.

|         |                                             | Inputs |   |       |   |    |                        |    |                        |
|---------|---------------------------------------------|--------|---|-------|---|----|------------------------|----|------------------------|
|         |                                             |        |   | / LED |   | Р  | A1 / LED17:<br>Pull-Up | P/ | A2 / LED18:<br>No Pull |
|         | PB0 / S8:<br>Push-Pull,<br>NO pull-up/down  | 0:     | 0 | 1:    | 1 | 0: | 1:                     | 0: | 1:                     |
| Outputs | PB1 / S9:<br>Open Drain,<br>NO pull-up/down | 0:     |   | 1:    |   | 0: | 1:                     | 0: | 1:                     |
|         | PB2 / S10:<br>Open Drain,<br>pull-up        | 0:     |   | 1:    |   | 0: | 1:                     | 0: | 1:                     |

Gibt es Kombinationen, die Sie kritisch sehen?

# 7 Bewertung

| Bewertungskriterien                                                                                          | Gewichtung |
|--------------------------------------------------------------------------------------------------------------|------------|
| Die Funktionen für die GPIO Inputs wurden gemäss Aufgabe implementiert und die Fragen können erklärt werden. | 1/4        |
| Die Output-Konfiguration ist implementiert und getestet.                                                     | 1/4        |
| Die Funktionen für die GPIO Outputs wurden gemäss Anforderungen getestet.                                    | 1/4        |
| Die Untersuchung wurde komplett durch geführt und auffällige Ergebnisse können erklärt werden.               | 1/4        |

## 8 Anhang

#### 8.1 Rücksetzen der Debugger Pins(Software Methode)

Ganz am Anfang des Programmes werden die Taster 0 bis 3 abgefragt, ob diese gedrückt werden. Sollte jemand aus Versehen die Debugger Pins umkonfiguriert haben, so kann kurz nach einem Reset einer der Taster 0 bis 3 gedrückt werden, um das Programm zu stoppen bevor der Studentencode ausgeführt wird. Dies verhindert, dass die fehlerhafte Konfiguration ausgeführt wird und man kann einen funktionierenden Code Flashen.

Sollte am Anfang auf dem CT-Board LCD anfangs nicht "Stop: any Tx button" stehen, so handelt es sich um ein veraltetes Projekt und die Hardware Methode in 8.2 muss ausgeführt werden.

#### 8.2 Rücksetzen der Debugger Pins(Hardware Methode)

Nur verwenden, falls die GPIO Pins des Debuggers durch einen fehlerhaften Praktikumscode falsch konfiguriert wurden und kein Code mehr aufs Board geladen werden kann.

Bei GPIOA und GPIOB sind nur die Pins 11-0 direkt abgreifbar. Die Pins 15-12 werden intern vom Discovery Board verwendet für z.B. Verbindung zum Debugger.

Falls die Pins 15-12 um konfiguriert werden, kann nicht mehr über die Debug-Schnittstelle auf den Microcontroller zugegriffen werden => Microcontroller nicht mehr umprogrammierbar. Um diesen Zustand zu verlassen, muss der Microcontroller anders booten (aus dem System Memory anstatt dem Flash Memory). Danach ist der Zugriff über die Debug-Schnittstelle wieder möglich.

Jetzt müssen die Registerzugriffe angepasst werden, die die Umkonfiguration der Pins 15-12 zufolge hatte, ansonsten ist die Debug-Schnittstelle beim nächsten Reset wieder nicht verfügbar!



Abbildung 5: Notwendige Verbindung, um aus dem System Memory zu booten