# DIC-Serielle Kommunikation mit einem $\mu C$

## Fabio Plunser

## 27. Januar 2021



# Inhaltsverzeichnis

| 1 | Aufgabenstellung                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              |
|---|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 2 | RS485       2.1 Generelles       2.2 Wie funktioniert es?       2.2 Wie wird der MAX485 angeschlossen                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |
| 3 | Einteilung — Was man wissen muss                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              |
| 4 | Clock 4.1 Clock-Code                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |
| 5 | GPIO       8         5.1 GPIO-Erklärung       8         5.2 GPIO-Code       1                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| 6 | USART126.1 Baudrate-Berechnung-Register-Setzen126.2 Weitere USART einstellungen146.3 USART-Initiallisierung/Konfiguration—Code146.4 USART-InterruptHandler14                                                                                                                                                                                                                                                                                                                                                                                                                                                                  |
| 7 | ADC         7.1       Erklärung       16         7.2       Register       16         7.2.1       ADC_ISR — ADC Interrupts       16         7.2.2       ADC_IER — ADC enable Interrupts       17         7.2.3       ADC_CR — ADC kalibrieren, aktivieren, starten       17         7.2.4       ADC_CFGR2 — Clock Einstellung       18         7.2.5       ADC_CHSELR — ADC Input Channel       18         7.2.6       ADC_DR — 16 Bit Ergebniss der Umnwandlung       19         7.3       Konfiguration       19         7.4       ADC-Initiallisierungs-Code       20         7.5       ADC-Interrupt-Handler-Code       20 |
| 8 | NVIC         25           8.1 Wir wird er verwendet         25           8.2 NVIC-Code         26                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                             |
| 9 | Gesamtes-Programm         9.1 Headerfile       25         9.2 Main       25                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                   |

| <b>10 Anh</b> 10.1                  | nang<br>Verlinkungen         | <b>34</b> 34 |
|-------------------------------------|------------------------------|--------------|
| Abbi                                | ldungsverzeichnis            |              |
| 1                                   | RS485-Diagramm               | 2            |
| 2                                   | MAX485-Arduino-Anschluss-BSP | 3            |
| 3                                   | STM32F030F4P6-Pinout         | 4            |
| 4                                   | Clock-Tree                   | 6            |
| 5                                   | System-Architektur           | 8            |
| 6                                   | GPIO-UART-PIN-Table          | 9            |
| 7                                   | GPIO-AF                      | 10           |
| 8                                   | GPIO-AF-Register             | 11           |
| 9                                   | USART-Aufbaug                | 13           |
| 10                                  | USART_BRR                    | 13           |
| 11                                  | ADC-ISR-Register             | 16           |
| 12                                  | ADC-IER-Register             | 17           |
| 13                                  | ADC-CR-Register              | 17           |
| 14                                  | ADC-CFGR2-Register           | 18           |
| 15                                  | ADC-CHSELR-Register          | 18           |
| 16                                  | ADC-DR-Register              | 19           |
| 17                                  | NVIC-Vector-Table            | 22           |
| 18                                  | Ausschnit-Assembler-File     | 23           |
| 19                                  | NVIC-ISER-Register           | 24           |
| $\operatorname{Cod}_{\mathfrak{C}}$ |                              |              |
| 1                                   | init-CLOCK                   | 7            |
| 2                                   | GPIO-Code                    | 12           |
| 3                                   | Init-UART                    | 14           |
| 4                                   | Init-UART                    | 14           |
| 5                                   | ADC-Initiallisierungs-Code   | 20           |
| 6                                   | ADC-Interrupt-Handler-Code   | 21           |
| 7                                   | NVIC-enable-interrupts       | 24           |
| 8                                   | Headerfile                   |              |
| 9                                   | Gesamter-Code                |              |



## 1 Aufgabenstellung

Die Aufgabe ist es auf dem STM32F030F4 Chip die UART3 so zu programmieren, dass sie den Wert eines eingebauten ADC per interrupt ausgibt.

Die richtige Aufgabe ist es die STM32F030F4 UART3 zu programmieren dass sie wenn sie ein zeichen erhält 10 Bytes zurückschickt.

Da dieses spezielle Package des STM32 keine UART3 besitzt ist die Aufgabenstellung so nicht möglich somit wird einfach die vorhandene UART1 Schnittstelle verwendet.

Der STM UART Ausgang wird mit einem MAX485 auf RS485 übersetzt.

UART einstellungen:

Baudrate: 38400 mit ODD parity

Dieser Arbeitsauftrag kann in dieser GIT-Repo verfolgt werden: https://github.com/FabioPlunser/DIC-Lezuo

PlunserFabio Page 1 of 34



## 2 RS485

#### 2.1 Generelles

Ist ein Industriestandard der eine asynchrone serielle Datenübertragung ermöglicht.

Der Standard verwendet ein symmetrisches Leitungspaar, dass für eine höhere elektromagnetische Resistenz sorgt.

#### 2.2 Wie funktioniert es?

Betriebsspannung 5V oder 3.3V

Der empfänger wertet die die Differenz beider Leitungen aus und kann Pegel ab  $\pm 200mV$  erkennen.

Senderpegel können von  $\pm 1.5 Vbis \pm 6 V$ 

Logik:

Wenn 
$$U_+ - U_- < -0.3V = MARK = OFF = Logisch 1$$

Wenn 
$$U_+ - U_- > +0.3V = \text{SPACE} = \text{ON} = \text{Logisch } 0$$



Abbildung 1: RS485-Diagramm

PlunserFabio Page 2 of 34



## 2.3 Wie wird der MAX485 angeschlossen

Unten ist ein Beispiel mit einem Arduino UNO

Anschluss:

 $\mathrm{DI} \to \mathrm{TX}$ 

 $\mathrm{RO} \to \mathrm{RX}$ 

DE, RE auf einen GPIO Pin. Da wenn DE + RE = 1  $\rightarrow$  Daten können nur gesendet werden. Wenn DE + RE = 0  $\rightarrow$  Daten können nur empfangen werden.



Abbildung 2: MAX485-Arduino-Anschluss-BSP

PlunserFabio Page 3 of 34



Ebenso muss dann der MAX485 and dem STM angeschlossen werden. Das untere Bild ist das Demo Board des STM32F030F4, eines der wenigen Boards, dass man zum Testen des Programmes für den STM Chip kaufen kann.



Abbildung 3: STM32F030F4P6-Pinout

PlunserFabio Page 4 of 34



# 3 Einteilung — Was man wissen muss

- Clock aufbau / einstellen
- GPIO Pins register finden / GPIO Pins einstellen
- UART Register suchen / UART aktivieren und einstellen
- Verstehen wie NVIC funktioniert

PlunserFabio Page 5 of 34



## 4 Clock

Die Clock wird am einfachsten, passend aus der gewollten Baudrate für die USART ausgewählt. Damit die Baudrate richtig berechnet wird.

Die HSI Clock von 8MHz ist standard mäßig aktiviert.

Weiterhin werden im APB2 Bus die Clocks für die USART1 und für den ADC und im AHB Bus für die GPIO Ports aktiviert. Die weiteren Clock einstellungen wie z.B. für die USART sind in der jeweilign Initiallisierungs Funktion enthalten.



Abbildung 4: Clock-Tree

PlunserFabio Page 6 of 34



#### 4.1 Clock-Code

```
int init_CLOCK()
    uint32_t reg_content;
    uint32_t* rcc_ahbenr = RCC_AHBENR;
    uint32_t* rcc_apb2enr = RCC_APB2ENR;
    uint32_t* rcc_cfgr = RCC_CFGR;
    //GPIO A port clock enable
    reg_content = *rcc_ahbenr;
    reg_content |= 0x00020000;
    *rcc_ahbenr = reg_content;
    //USART1 clock enable
    reg_content = *rcc_apb2enr;
    reg_content \mid= 0x00004200;
    *rcc_apb2enr = reg_content;
    //set AHB Clock to not divided so same clock as sysclock
    reg_content = *rcc_cfgr;
    reg_content |= 0x00000000;
    *rcc_cfgr = reg_content;
```

Listing 1: init-CLOCK

PlunserFabio Page 7 of 34



#### 5 GPIO

#### 5.1 GPIO-Erklärung

Wie man bei der Abbildung 5 sehen kann, sind die GPIOs am BUS AHB2 verbunden. Da das verwendete Paket nur GPIO A verwendet müssen dementsprechend die GPIO-A Register richtig konfiguriert werden

Das Register um die GPIOs als output einzustellen ist GPIOA\_MODER im Addressraum 0x4800 0000 (des AHB2 Buses) mit dem Address offset: 0x00

Um Pins für die UART verwenden zu können müssen diese Pins noch als "alternate functions" konfiguriert werden im Register **GPIOA\_AFRH mit Address offset: 0x24**. Welche Pin Nummer man Programmieren muss, sieht man in der Abbildung 6, man benötigt PA9 (Pin:17) und PA10 (Pin:18), da diese als alternate function die USART\_1 TX und RX hinterlegt haben. Wie das alternate function Register konfiguriert werden muss sieht man in den Abbildungen 7 und 8

Weiterhin müssen noch zwei GPIO Pins Für die DE und RE Pins des MAX485 als output konfiguriert werden. Ich verwend PA7 und PA8.

Ebenfalls wird ein GPIO Pin als analog konfiguriert um diesen als ADC Eingang zu verwenden.



Abbildung 5: System-Architektur

PlunserFabio Page 8 of 34



|        | Pin nu | mber   |         |                                       |          |               |       | Pin fun                                                                                                                                                              | ctions                             |
|--------|--------|--------|---------|---------------------------------------|----------|---------------|-------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------------------------------|
| LQFP64 | LQFP48 | LQFP32 | TSSOP20 | Pin name<br>(function after<br>reset) | Pin type | I/O structure | Notes | Alternate functions                                                                                                                                                  | Additional functions               |
| 34     | 26     | -      | -       | PB13                                  | I/O      | FT            | -     | SPI1_SCK <sup>(2)</sup> ,<br>SPI2_SCK <sup>(3)(5)</sup> ,<br>I2C2_SCL <sup>(5)</sup> ,<br>TIM1_CH1N,<br>USART3_CTS <sup>(5)</sup>                                    | -                                  |
| 35     | 27     | •      | •       | PB14                                  | I/O      | FT            | -     | SPI1_MISO <sup>(2)</sup> ,<br>SPI2_MISO <sup>(3)(5)</sup> ,<br>I2C2_SDA <sup>(5)</sup> ,<br>TIM1_CH2N,<br>TIM15_CH1 <sup>(3)(5)</sup> ,<br>USART3_RTS <sup>(5)</sup> | -                                  |
| 36     | 28     | -      | •       | PB15                                  | I/O      | FT            | -     | SPI1_MOSI <sup>(2)</sup> ,<br>SPI2_MOSI <sup>(3)(5)</sup> ,<br>TIM1_CH3N,<br>TIM15_CH1N <sup>(3)(5)</sup> ,<br>TIM15_CH2 <sup>(3)(5)</sup>                           | RTC_REFIN,<br>WKUP7 <sup>(5)</sup> |
| 37     | -      | -      | -       | PC6                                   | I/O      | FT            | -     | TIM3_CH1                                                                                                                                                             | -                                  |
| 38     | -      | -      | -       | PC7                                   | I/O      | FT            | -     | TIM3_CH2                                                                                                                                                             | -                                  |
| 39     | -      | -      | -       | PC8                                   | I/O      | FT            | -     | TIM3_CH3                                                                                                                                                             | -                                  |
| 40     | -      | -      | -       | PC9                                   | I/O      | FT            | -     | TIM3_CH4                                                                                                                                                             | -                                  |
| 41     | 29     | 18     | -       | PA8                                   | I/O      | FT            | -     | USART1_CK,<br>TIM1_CH1,<br>EVENTOUT,<br>MCO                                                                                                                          | -                                  |
| 42     | 30     | 19     | 17      | PA9                                   | I/O      | FT            |       | USART1 TX,<br>TIM1_CH2,<br>TIM15_BKIN <sup>(3)(5)</sup><br>I2C1_SCL <sup>(2)(5)</sup>                                                                                | -                                  |
| 43     | 31     | 20     | 18      | PA10                                  | I/O      | FT            | -     | USART1_RX,<br>TIM1_CH3,<br>TIM17_BKIN<br>I2C1_SDA <sup>(2)(5)</sup>                                                                                                  | -                                  |
| 44     | 32     | 21     | •       | PA11                                  | I/O      | FT            | -     | USART1_CTS,<br>TIM1_CH4,<br>EVENTOUT,<br>I2C2_SCL <sup>(5)</sup>                                                                                                     | -                                  |
| 45     | 33     | 22     | •       | PA12                                  | I/O      | FT            | -     | USART1_RTS,<br>TIM1_ETR,<br>EVENTOUT,<br>I2C2_SDA <sup>(5)</sup>                                                                                                     | -                                  |

Abbildung 6: GPIO-UART-PIN-Table

PlunserFabio Page 9 of 34



Table 12. Alternate functions selected through GPIOA\_AFR registers for port A

| Pin name | AF0                          | AF1                          | AF2       | AF3      | AF4                        | AF5                       | AF6      |
|----------|------------------------------|------------------------------|-----------|----------|----------------------------|---------------------------|----------|
| PA0      |                              | USART1_CTS <sup>(2)</sup>    |           |          | USART4_TX <sup>(1)</sup>   |                           |          |
| PAU      | -                            | USART2_CTS <sup>(1)(3)</sup> | -         | -        | USART4_TX                  | -                         | -        |
| DA4      | EVENTOUT                     | USART1_RTS <sup>(2)</sup>    |           |          | USART4 RX <sup>(1)</sup>   | TIM15 CH1N <sup>(1)</sup> |          |
| PA1      | EVENTOUT                     | USART2_RTS <sup>(1)(3)</sup> | -         | -        | USART4_RX**                | TIM15_CH1N**              | -        |
| PA2      | TIM15_CH1 <sup>(1)(3)</sup>  | USART1_TX <sup>(2)</sup>     |           |          |                            |                           |          |
| PAZ      | TIM15_CH1(*/\**/             | USART2_TX <sup>(1)(3)</sup>  | -         | -        | -                          | -                         | -        |
| 54.0     | TILLE 0110(1)(3)             | USART1_RX <sup>(2)</sup>     |           |          |                            |                           |          |
| PA3      | TIM15_CH2 <sup>(1)(3)</sup>  | USART2_RX <sup>(1)(3)</sup>  | -         | -        | -                          | -                         | -        |
| PA4      | CDIA NICC                    | USART1_CK <sup>(2)</sup>     |           |          | TIMAA CUA                  | USART6 TX <sup>(1)</sup>  |          |
| PA4      | SPI1_NSS                     | USART2_CK <sup>(1)(3)</sup>  | -         | -        | TIM14_CH1                  | USARTO_IX                 | -        |
| PA5      | SPI1_SCK                     | -                            | -         | -        | -                          | USART6_RX <sup>(1)</sup>  | -        |
| PA6      | SPI1_MISO                    | TIM3_CH1                     | TIM1_BKIN | -        | USART3_CTS <sup>(1)</sup>  | TIM16_CH1                 | EVENTOUT |
| PA7      | SPI1_MOSI                    | TIM3_CH2                     | TIM1_CH1N | -        | TIM14_CH1                  | TIM17_CH1                 | EVENTOUT |
| PA8      | MCO                          | USART1_CK                    | TIM1_CH1  | EVENTOUT | -                          | -                         | -        |
| PA9      | TIM15_BKIN <sup>(1)(3)</sup> | USART1_TX                    | TIM1_CH2  | -        | I2C1_SCL <sup>(1)(2)</sup> | MCO <sup>(1)</sup>        | -        |
| PA10     | TIM17_BKIN                   | USART1_RX                    | TIM1_CH3  | -        | I2C1_SDA <sup>(1)(2)</sup> | -                         | -        |
| PA11     | EVENTOUT                     | USART1_CTS                   | TIM1_CH4  | -        | -                          | SCL                       | -        |

Abbildung 7: GPIO-AF

Page 10 of 34 PlunserFabio



# 8.4.10 GPIO alternate function high register (GPIOx\_AFRH) (x = A..D, F)

Address offset: 0x24

Reset value: 0x0000 0000

| 31 | 30                        | 29       | 28 | 27 | 26    | 25      | 24    | 23      | 22    | 21          | 20 | 19 | 18    | 17       | 16 |
|----|---------------------------|----------|----|----|-------|---------|-------|---------|-------|-------------|----|----|-------|----------|----|
|    | AFSEL                     | .15[3:0] |    |    | AFSEL | 14[3:0] |       |         | AFSEL | 13[3:0]     |    |    | AFSEL | .12[3:0] |    |
| rw | rw                        | rw       | rw | rw | rw    | rw      | rw    | rw      | rw    | rw          | rw | rw | rw    | rw       | rw |
| 15 | 14                        | 13       | 12 | 11 | 10    | 9       | 8     | 7       | 6     | 5           | 4  | 3  | 2     | 1        | 0  |
|    | AFSEL11[3:0] AFSEL10[3:0] |          |    |    |       |         | AFSEI | _9[3:0] |       | AFSEL8[3:0] |    |    |       |          |    |
| rw | rw                        | rw       | rw | rw | rw    | rw      | rw    | rw      | rw    | rw          | rw | rw | rw    | rw       | rw |

Bits 31:0 **AFSELy[3:0]:** Alternate function selection for port x pin y (y = 8..15)

These bits are written by software to configure alternate function I/Os

AFSELy selection:

1000: Reserved 0000: AF0 0001: AF1 1001: Reserved 0010: AF2 1010: Reserved 0011: AF3 1011: Reserved 1100: Reserved 0100: AF4 0101: AF5 1101: Reserved 1110: Reserved 0110: AF6 0111: AF7 1111: Reserved

Abbildung 8: GPIO-AF-Register

PlunserFabio Page 11 of 34



#### 5.2 GPIO-Code

```
int init_GPIO()
    //Set GPIO pins PA9 and PA10 alternate function for USART TX and RX,
    //set PA7 and PA6 output for DE and RE of MAX485
    //PAO ADC_INO so set it to analog and then in DAC register set PAO as
       ADC_INO
    uint32_t reg_content;
    uint32_t* gpioa_moder = GPIOA_MODER;
    uint32_t* gpioa_afrh = GPIOA_AFRH;
    uint32_t* gpioa_odr = GPIOA_ODR;
    reg_content = *gpioa_moder;
    reg_content |= 0x00285003;
    *gpioa_moder = reg_content;
    //Set GPIO PA9 as AF=TX and PA10 as AF = RX
    reg_content = *gpioa_afrh;
    reg_content \mid = 0x00000110;
    *gpioa_afrh = reg_content;
    //Ensure that output gpios are 0 for MAX, to read all the time
    //and only send, when needing to send
    reg_content = *gpioa_odr;
    reg content \mid = 0x000000000;
    *gpioa_odr = reg_content;
```

Listing 2: GPIO-Code

## 6 USART

USART Addressraum: **0x4001 3800 - 0x4001 3BFF** Für die USART muss, die Clock, die Bautrate, Parity, Wordlenght und die Interrupts eingestellt werden.

## 6.1 Baudrate-Berechnung-Register-Setzen

Um die Baudrate einstellen zu können muss in das Register **USART\_BRR** die richtige Hexadezimal Zahl geschrieben werden.

```
Berechnung: \frac{Clock}{Baudrate}, \frac{8MHz}{38400} = 208, 3_{dezimal} bzw. D0<sub>hex</sub>
Zurückgerechnet 208*38400 = 7.98MHz. \frac{8MHz}{208} = Baudrate von 38461.
```

Da der Systemclock gleich der HSI clock ist, kann dafür im Register RCC\_CFGR3 bei der Addresse offset: 0x30 die USART1SW entweder mit 01 für Systemclock oder mit 11 für HSI

PlunserFabio Page 12 of 34



clock beschrieben werden. Zusätzlich muss im Register  $\mathbf{USART\_BRR}$  mit Address offset:  $\mathbf{0x0C}$  D0 geschrieben Werden.



Abbildung 9: USART-Aufbaug

#### 23.7.4 Baud rate register (USART\_BRR)

This register can only be written when the USART is disabled (UE=0). It may be automatically updated by hardware in auto baud rate detection mode.

Address offset: 0x0C Reset value: 0x0000

| 31   | 30   | 29   | 28   | 27   | 26   | 25   | 24   | 23     | 22   | 21   | 20   | 19   | 18   | 17   | 16   |
|------|------|------|------|------|------|------|------|--------|------|------|------|------|------|------|------|
| Res.   | Res. | Res. | Res. | Res. | Res. | Res. | Res. |
|      |      |      |      |      |      |      |      |        |      |      |      |      |      |      |      |
| 15   | 14   | 13   | 12   | 11   | 10   | 9    | 8    | 7      | 6    | 5    | 4    | 3    | 2    | 1    | 0    |
|      |      |      |      |      |      |      | BRR  | [15:0] |      |      |      |      |      |      |      |
| rw     | rw   | rw   | rw   | rw   | rw   | rw   | rw   |

Bits 31:16 Reserved, must be kept at reset value.

Bits 15:4 BRR[15:4]

BRR[15:4] = USARTDIV[15:4]

Bits 3:0 BRR[3:0]

When OVER8 = 0, BRR[3:0] = USARTDIV[3:0]. When OVER8 = 1: BRR[2:0] = USARTDIV[3:0] shifted 1 bit to the right. BRR[3] must be kept cleared.

Abbildung 10: USART\_BRR

PlunserFabio Page 13 of 34



## 6.2 Weitere USART einstellungen

Die Eistellungen der USART müssen getroffen werden, bevor sie aktiviert wird. Laut angabe wird noch eine ODD Parity verwendet diese kann in dem Register **USART\_CR1** auf Bit9: festgelegt werden. Weiterhin müssen dort unter Bit3 und Bit2, TX und RX aktivieren.

## 6.3 USART-Initiallisierung/Konfiguration—Code

```
int init_UART()
    uint32_t reg_content;
    uint32_t* rcc_cfr3 = RCC_CFGR3;
    uint32_t* usart1_brr = USART1_BRR;
    uint32_t* usart1_cr1 = USART1_CR1;
    //Use SYSCLK for USART and use Baudrate 38400
    reg_content = *rcc_cfr3;
    reg content \mid = 0x00000001;
    *rcc_cfr3 = reg_content;
    reg_content = *usart1_brr;
    reg_content = 0x00000D00;
    *usart1_brr = reg_content;
    //Wordlenght, Parity control enable, Parity selection, interrupt enable,
        Transmission complete interrupt enable, RXNE interrupt enable
    reg_content = *usart1_cr1;
    reg_content |= 0x000006EC;
    *usart1_cr1 = reg_content;
    reg_content = *usart1_cr1;
    reg_content |= 0x00000001; //enable UART
    *usart1_cr1 = reg_content;
}
```

Listing 3: Init-UART

## 6.4 USART-InterruptHandler

```
void USART1_IRQHandler(void)
{
    uint32_t* usart1_isr = USART1_ISR;
    uint32_t usart1_isr_rxne = USART1_ISR_RXNE;
    uint32_t usart1_isr_tc = USART1_ISR_TC;
    uint32_t* usart1_tc = USART1_ICR;
    uint32_t* usart1_tc = USART1_ICR_TCCF;
    uint32_t* usart1_td = USART1_TDR;
    uint32_t* usart1_tdr = USART1_TDR;
    uint32_t* gpioa_odr = GPIOA_ODR;
    uint32_t reg_content;
```

PlunserFabio Page 14 of 34



```
if ((*usart1_isr & usart1_isr_tc) == usart1_isr_tc)
        if (send == sizeof(USART_write_data))
            //set MAX to listen
            reg_content = *gpioa_odr;
            reg_content |= 0x00000000;
            *gpioa_odr = reg_content;
            send=0;
            *usart1_tc |= *usart1_tc_tcce; // Clear transfer complete flag *
        }
        else
        {
            // clear transfer complete flag and fill TDR with a new char
            *usart1_tdr = USART_write_data[send++];
        }
    }
    //when something comes at usart, begin to write the ADC Data
    if ((*usart1_isr & usart1_isr_rxne) == usart1_isr_rxne)
    {
        USART READ = *((char *)USART1 RDR);
        //Set PA7 and PA6 high for max to send data
        reg_content = *gpioa_odr;
        reg_content |= 0x000000000;
        *gpioa_odr = reg_content;
        *usart1_tdr = USART_write_data[0];
   }
}
```

Listing 4: Init-UART

PlunserFabio Page 15 of 34



## 7 ADC

## 7.1 Erklärung

Der STM hat einen ADC eingebaut, dieser kann bei den größeren Packages sogar als Temperatursensor verwendet werden. Alle GPIOs können als ADC Eingang verwendet werden solange sie als analog GPIO konfiguriert werden.

## 7.2 Register

Die Register für den ADC beginnen bei 0x4001 2400 bis 0x4001 27FF. Für die Konfiguration werden folgende Register benötigt:

#### 7.2.1 ADC\_ISR — ADC Interrupts

| 31         | 30         | 29         | 28         | 27         | 26         | 25        | 24        | 23       | 22        | 21        | 20       | 19         | 18   | 17         | 16         |
|------------|------------|------------|------------|------------|------------|-----------|-----------|----------|-----------|-----------|----------|------------|------|------------|------------|
|            | Res.       | Res.       | Res.       | Res.       | Res.       | Res.      | Res.      | Res.     | Res.      | Res.      | Res.     | Res.       | Res. | Res.       | Res.       |
|            |            |            |            |            |            |           |           |          |           |           |          |            |      |            |            |
|            |            |            |            |            |            |           |           |          |           |           |          |            |      |            |            |
| 15         | 14         | 13         | 12         | 11         | 10         | 9         | 8         | 7        | 6         | 5         | 4        | 3          | 2    | 1          | 0          |
| 15<br>Res. | 14<br>Res. | 13<br>Res. | 12<br>Res. | 11<br>Res. | 10<br>Res. | 9<br>Res. | 8<br>Res. | 7<br>AWD | 6<br>Res. | 5<br>Res. | 4<br>OVR | 3<br>EOSEQ | _    | 1<br>EOSMP | 0<br>ADRDY |

Abbildung 11: ADC-ISR-Register

- EOC (End of conversion):
  - 0: Channel conversion not complete (or the flag event was already acknowledged and cleared by software)
  - 1: Channel conversion complete
- ADRDY (ADC ready): 0: ADC not yet ready to start conversion (or the flag event was already acknowledged and cleared by software)
  - 1: ADC is ready to start conversion

PlunserFabio Page 16 of 34



#### 7.2.2 ADC\_IER — ADC enable Interrupts

| 31   | 30   | 29   | 28   | 27   | 26   | 25        | 24   | 23             | 22   | 21   | 20         | 19               | 18         | 17               | 16               |
|------|------|------|------|------|------|-----------|------|----------------|------|------|------------|------------------|------------|------------------|------------------|
| Res.      | Res. | Res.           | Res. | Res. | Res.       | Res.             | Res.       | Res.             | Res.             |
|      |      |      |      |      |      |           |      |                |      |      |            |                  |            |                  |                  |
| 15   | - 44 |      |      | •    | •    | •         |      | •              |      |      |            |                  |            |                  |                  |
| 10   | 14   | 13   | 12   | 11   | 10   | 9         | 8    | 7              | 6    | 5    | 4          | 3                | 2          | 1                | 0                |
| Res. | Res. | Res. | Res. | Res. | Res. | 9<br>Res. | Res. | 7<br>AWD<br>IE | Res. | Res. | 4<br>OVRIE | 3<br>EOSEQ<br>IE | 2<br>EOCIE | 1<br>EOSMP<br>IE | 0<br>ADRDY<br>IE |

Abbildung 12: ADC-IER-Register

#### Benötigt wird:

- EOCIE(End of conversion interrupt enable):
  - 0: EOC interrupt disabled
  - 1: EOC interrupt enabled. An interrupt is generated when the EOC bit is set.
- ADRDYIE (ADC ready interrupt enable): 0: ADRDY interrupt disabled.
  - 1: ADRDY interrupt enabled. An interrupt is generated when the ADRDY bit is set.

#### 7.2.3 ADC\_CR — ADC kalibrieren, aktivieren, starten

| 31        | 30   | 29   | 28         | 27   | 26         | 25        | 24   | 23   | 22   | 21   | 20         | 19   | 18               | 17         | 16        |
|-----------|------|------|------------|------|------------|-----------|------|------|------|------|------------|------|------------------|------------|-----------|
| AD<br>CAL | Res. | Res. | Res.       | Res. | Res.       | Res.      | Res. | Res. | Res. | Res. | Res.       | Res. | Res.             | Res.       | Res.      |
| rs        |      |      |            |      |            |           |      |      |      |      |            |      |                  |            |           |
| 15        |      |      |            | •    | •          |           |      |      |      |      |            |      |                  |            |           |
| 15        | 14   | 13   | 12         | 11   | 10         | 9         | 8    | 7    | 6    | 5    | 4          | 3    | 2                | 1          | 0         |
| Res.      | Res. | Res. | 12<br>Res. | Res. | 10<br>Res. | 9<br>Res. | Res. | Res. | Res. | Res. | 4<br>ADSTP | Res. | 2<br>ADSTA<br>RT | 1<br>ADDIS | 0<br>ADEN |

Abbildung 13: ADC-CR-Register

#### Benötigt wird:

- ADCAL (ADC calibration):
  - 0: Calibration complete
  - 1: Write 1 to calibrate the ADC. Read at 1 means that a calibration is in progress.
- ADEN (ADC enable command)
  - 0: ADC is disabled (OFF state)
  - 1: Write 1 to enable the ADC.
- ADSTART (ADC start conversion command)
  - 0: No ADC conversion is ongoing.
  - 1: Write 1 to start the ADC. Read 1 means that the ADC is operating and may be converting

PlunserFabio Page 17 of 34



#### 7.2.4 ADC\_CFGR2 — Clock Einstellung

| 31   | 30      | 29   | 28   | 27   | 26   | 25   | 24   | 23   | 22   | 21   | 20   | 19   | 18   | 17   | 16   |
|------|---------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|
| СКМО | DE[1:0] | Res. |
| rw   | rw      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| 15   | 14      | 13   | 12   | 11   | 10   | 9    | 8    | 7    | 6    | 5    | 4    | 3    | 2    | 1    | 0    |
| Res. | Res.    | Res. | Res. | Res. | Res. | Res. | Res. | Res. | Res. | Res. | Res. | Res. | Res. | Res. | Res. |
|      |         |      |      |      |      |      |      |      |      |      |      |      |      |      |      |

Abbildung 14: ADC-CFGR2-Register

#### Benötigt wird:

• CKMODE[1:0] (ADC clock mode):

00: ADCCLK (Asynchronous clock mode), generated at product level (refer to RCC section)

01: PCLK/2 (Synchronous clock mode)

10: PCLK/4 (Synchronous clock mode)

11: Reserved

## 7.2.5 $ADC\_CHSELR$ — ADC Input Channel

| 31          | 30          | 29   | 28                | 27   | 26   | 25   | 24   | 23              | 22   | 21   | 20         | 19   | 18              | 17              | 16              |
|-------------|-------------|------|-------------------|------|------|------|------|-----------------|------|------|------------|------|-----------------|-----------------|-----------------|
| Res.        | Res.        | Res. | Res.              | Res. | Res. | Res. | Res. | Res.            | Res. | Res. | Res.       | Res. | Res.            | CHSEL<br>17     | CHSEL<br>16     |
|             |             |      |                   |      |      |      |      |                 |      |      |            |      |                 | rw              | rw              |
| 15          |             |      |                   |      |      |      |      |                 |      |      |            |      |                 |                 |                 |
| 15          | 14          | 13   | 12                | 11   | 10   | 9    | 8    | 7               | 6    | 5    | 4          | 3    | 2               | 1               | 0               |
| CHSEL<br>15 | CHSEL<br>14 | 1    | 12<br>CHSEL<br>12 |      |      |      |      | 7<br>CHSEL<br>7 | _    | _    | CHSEL<br>4 |      | 2<br>CHSEL<br>2 | 1<br>CHSEL<br>1 | 0<br>CHSEL<br>0 |

Abbildung 15: ADC-CHSELR-Register

#### Benötigt wird:

• CHSELx: Channel-x selection

0: Input Channel-x is not selected for conversion

1: Input Channel-x is selected for conversion

PlunserFabio Page 18 of 34



#### 7.2.6 ADC\_DR — 16 Bit Ergebniss der Umnwandlung



Abbildung 16: ADC-DR-Register

#### Benötigt wird:

• DATA[15:0]: Converted data

## 7.3 Konfiguration

Um den ADC zu konfigurieren, wird zuerst der ADC Clock eingestellt  $\rightarrow$  der ADC kalibriert  $\rightarrow$  ADC aktiviert  $\rightarrow$  GPIO als ADC Input einstellen.

Wenn die Kalibrierung fertig ist, ist das ADCAL Bit 0. Dann kann der ADC und Interrupts aktiviert werden und eingestellt werden welcher GPIO Pin als Input verwendet wird. Weiterhin wird der ADC im vortlaufenden (continuous) Modus betrieben, sodass im EOC Interrupt die Umwandlung nicht wieder gestartet werden muss.

Danach wenn der ADC bereit ist wird im Interrupt Register die ADC\_ADRDY (ADC Ready) flag gesetzt, dieses wird im ADC-Interrupt-Handler abgefragt und dann eine Umwandlung gestartet. Wenn eine Umwandlung fertig ist wird im Interrupt Register das ADC\_TC (Transmission complete) flag gesetzt. Im ADC-Interrupt-Hanlder wird dann aus dem ADC\_DR Register das Ergebniss der Umwandlung weitergeben um dies dann an der USART senden zu können.

PlunserFabio Page 19 of 34



#### 7.4 ADC-Initiallisierungs-Code

```
int init_ADC()
{
    uint32_t reg_content;
    uint32_t* adc_chselr = ADC_CHSELR;
    uint32_t* adc_isr = ADC_ISR;
    uint32_t* adc_cr = ADC_CR;
    uint32 t* adc cfgr1 = ADC CFGR1;
    uint32_t adc_cr_adcal = ADC_CR_ADCAL;
    uint32 t* adc ier = ADC IER;
    uint32_t* adc_cfgr2 = ADC_CFGR2;
    uint32_t adc_isr_adrdy = ADC_ISR_ADRDY;
    //set ADC clock to PCLK so thats is synchronous with sysclock
    reg_content = *adc_cfgr2;
    reg_content |= 0x40000000;
    *adc_cfgr2 = reg_content;
    //before starting callibrate ADC
    reg_content = *adc_cr;
    reg_content |= adc_cr_adcal;
    *adc_cr = reg_content;
    //if calibration is complete enable ADC
    //enable Interrupts
    //set correct channel
    //when ADC is ready, a ad ready interrupt occurs and the interrupt
    //will then start the ADC convertion
    while((*adc_cr & adc_cr_adcal) == adc_cr_adcal); // wait till
       callibration is complete
    //enable ADC and ensure ADSTART=0 for further configuration and enable
       ADC
    reg_content = *adc_cr;
    reg_content |= 0x00000001;
    *adc_cr = reg_content;
    //set ADC to continues convertion so, that convertion doesn't to start
       again in Interrupt
    reg_content = *adc_cr;
    reg_content |= 0x0000200;
    *adc_cr = reg_content;
    //enable Interrupts of ADC
    reg_content = *adc_ier;
    reg_content |= 0x00000005;
    *adc_ier = reg_content;
    //Set ADC Channel to channel O because PAO is ADC_INO
    reg_content = *adc_chselr;
    reg_content |= 0x00000001;
    *adc_chselr = reg_content;
}
```

Listing 5: ADC-Initiallisierungs-Code

PlunserFabio Page 20 of 34



## 7.5 ADC-Interrupt-Handler-Code

```
void ADC1_IRQHandler(void)
{
    uint32_t* adc_isr = ADC_ISR;
    uint32_t* adc_cr = ADC_CR;
    uint32_t adc_isr_eoc = ADC_ISR_EOC;
    uint32_t adc_isr_adrdy = ADC_ISR_ADRDY;
    uint32_t* adc_dr = ADC_DR;
    uint32 t adc dr data = ADC DR DATA;
    uint32_t* gpioa_odr = GPIOA_ODR;
    uint32_t* usart1_tdr = USART1_TDR;
    uint32_t adc_cr_adstart = ADC_CR_ADSTART;
    uint16_t ADC_Value;
    uint32_t reg_content;
    //if ADC Ready start convertion
    if ((*adc_isr & adc_isr_adrdy) == adc_isr_adrdy)
    {
        reg_content = *adc_cr;
        reg_content |= adc_cr_adstart;
        *adc_cr = reg_content;
    }
    //if convertion complete send ADC value
    if((*adc_isr & adc_isr_eoc) == adc_isr_eoc)
    {
        send=0;
        ADC_Value = (*adc_dr & adc_dr_data);
        //Set PA7 and PA6 high for max to send data
        reg_content = *gpioa_odr;
        reg_content |= 0x000000C0;
        *gpioa_odr = reg_content;
        sprintf(USART_write_data, "%d", ADC_Value);
        //write something into the USART Buffer for USART
        //interrupt where rest of adc value gets put into the buffer
        *usart1_tdr = 'n';
   }
}
```

Listing 6: ADC-Interrupt-Handler-Code

PlunserFabio Page 21 of 34



## 8 NVIC

#### 8.1 Wir wird er verwendet

Beim Programmieren des STM muss der Vector Table beschrieben werden mit einem Assembler File und Linker Skript, diese werden von der Cube IDE erstellt.

Im Vector Table sind alle Interrupts mit dem entsprechenden Funktionnamen für das Programm hinterlegt. Im NVIC\_ISER Register können die Interrupts durch ihre Interrupt Position aktiviert werden.

Die IRQ\_Handler Positionen werden beim programmieren des uC in den Flash geschrieben.

Table 22 Vector table (continued)

|          |          |                  | Table 32                | . Vector table (continued)                            |             |
|----------|----------|------------------|-------------------------|-------------------------------------------------------|-------------|
| Position | Priority | Type of priority | Acronym                 | Description                                           | Address     |
| 3        | 10       | settable         | FLASH                   | Flash global interrupt                                | 0x0000 004C |
| 4        | 11       | settable         | RCC                     | RCC global interrupts                                 | 0x0000 0050 |
| 5        | 12       | settable         | EXTIO_1                 | EXTI Line[1:0] interrupts                             | 0x0000 0054 |
| 6        | 13       | settable         | EXTI2_3                 | EXTI Line[3:2] interrupts                             | 0x0000 0058 |
| 7        | 14       | settable         | EXTI4_15                | EXTI Line[15:4] interrupts                            | 0x0000 005C |
| 8        |          |                  | Reserved                |                                                       | 0x0000 0060 |
| 9        | 16       | settable         | DMA_CH1                 | DMA channel 1 interrupt                               | 0x0000 0064 |
| 10       | 17       | settable         | DMA_CH2_3               | DMA channel 2 and 3 interrupts                        | 0x0000 0068 |
| 11       | 18       | settable         | DMA_CH4_5               | DMA channel 4 and 5 interrupts                        | 0x0000 006C |
| 12       | 19       | settable         | ADC                     | ADC interrupts                                        | 0x0000 0070 |
| 13       | 20       | settable         | TIM1_BRK_UP_<br>TRG_COM | TIM1 break, update, trigger and commutation interrupt | 0x0000 0074 |
| 14       | 21       | settable         | TIM1_CC                 | TIM1 capture compare interrupt                        | 0x0000 0078 |
| 15       |          |                  | Reserved                |                                                       | 0x0000 007C |
| 16       | 23       | settable         | TIM3                    | TIM3 global interrupt                                 | 0x0000 0080 |
| 17       | 24       | settable         | TIM6                    | TIM6 global interrupt                                 | 0x0000 0084 |
| 18       |          |                  | Reserved                |                                                       | 0x0000 0084 |
| 19       |          |                  | Reserved                |                                                       | 0x0000 0088 |
| 19       | 26       | settable         | TIM14                   | TIM14 global interrupt                                | 0x0000 008C |
| 20       | 27       | settable         | TIM15                   | TIM15 global interrupt                                | 0x0000 0090 |
| 21       | 28       | settable         | TIM16                   | TIM16 global interrupt                                | 0x0000 0094 |
| 22       | 29       | settable         | TIM17                   | TIM17 global interrupt                                | 0x0000 0098 |
| 23       | 30       | settable         | I2C1                    | I <sup>2</sup> C1 global interrupt                    | 0x0000 009C |
| 24       | 31       | settable         | I2C2                    | I <sup>2</sup> C2 global interrupt                    | 0x0000 00A0 |
| 25       | 32       | settable         | SPI1                    | SPI1 global interrupt                                 | 0x0000 00A4 |
| 26       | 33       | settable         | SPI2                    | SPI2 global interrupt                                 | 0x0000 00A8 |
| 27       | 34       | settable         | USART1                  | USART1 global interrupt                               | 0x0000 00AC |
| 28       | 35       | settable         | USART2                  | USART2 global interrupt                               | 0x0000 00B0 |
| 29       | 36       | settable         | USART3_4_5_6            | USART3, USART4, USART5, USART6 global interrupts      | 0x0000 00B4 |
| 30       |          |                  | Reserved                |                                                       | 0x0000 00B8 |
| 31       | 38       | settable         | USB                     | USB global interrupt (combined with EXTI line 18)     | 0x0000 00BC |

Abbildung 17: NVIC-Vector-Table

PlunserFabio Page 22 of 34



```
g_pfnVectors:
         .word
                estack
126
                Reset Handler
         .word
         .word
                NMI Handler
                HardFault_Handler
         .word
129
         .word
         .word
                0
         .word
                0
                0
         .word
         .word
                0
134
                0
         .word
         .word
         .word
                SVC Handler
         .word
                0
138
         .word
         .word
                PendSV Handler
         .word
                SysTick Handler
141
                WWDG_IRQHandler
                                                    /* Window WatchDog
         .word
142
                                                    /* Reserved
         .word
143
                                                    /* RTC through the EXTI line
                                                                                      */
                RTC IRQHandler
         .word
                FLASH IRQHandler
                                                    /* FLASH
                                                                                      */
         .word
                                                    /* RCC
                                                                                      */
         .word RCC IRQHandler
                                                                                      */
146
         .word
                EXTI0 1 IRQHandler
                                                    /* EXTI Line 0 and 1
                                                    /* EXTI Line 2 and 3
147
                EXTI2 3 IRQHandler
                                                                                      */
         .word
                                                                                      */
                EXTI4_15_IRQHandler
                                                    /* EXTI Line 4 to 15
         .word
                                                    /* Reserved
         .word
                0
150
                DMA1 Channell IROHandler
                                                    /* DMA1 Channel 1
                                                                                      */
         .word
151
                DMA1 Channel2 3 IRQHandler
                                                    /* DMA1 Channel 2 and Channel 3 */
         .word
                                                    /* DMA1 Channel 4 and Channel 5 */
         .word
                DMA1 Channel4 5 IRQHandler
                                                    /* ADC1
                ADC1_IRQHandler
         .word
                                                    /* TIM1 Break, Update, Trigger and Commutation */
154
                TIM1_BRK_UP_TRG_COM_IRQHandler
         .word
                                                    /* TIM1 Capture Compare
                                                                                      */
                TIM1_CC_IRQHandler
         .word
                                                                                      */
                0
                                                    /* Reserved
         .word
                                                    /* TIM3
                                                                                      */
         .word
                TIM3_IRQHandler
                                                    /* Reserved
                                                                                      */
         .word
                                                                                      */
159
                                                    /* Reserved
         .word
                                                    /* TIM14
                                                                                      */
         .word
                TIM14 IRQHandler
                                                    /* Reserved
                                                                                      */
         .word
                Ø
                                                                                      */
                TIM16_IRQHandler
                                                    /* TIM16
         .word
163
                TIM17_IRQHandler
                                                    /* TIM17
                                                                                      */
         .word
                                                    /* I2C1
                                                                                      */
         .word
                I2C1 IRQHandler
         .word
                                                       Reserved
                a
                                                    /* SPI1
                                                                                      */
         .word
                SPI1_IRQHandler
                                                                                      */
167
                                                    /* Reserved
         .word
                                                                                      */
                                                    /* USART1
         .word
                USART1_IRQHandler
                                                       Reserved
                                                                                      */
         .word
                0
                                                                                      */
         .word
                0
                                                       Reserved
                0
                                                    /* Reserved
         .word
                                                    /* Reserved
         .word
```

Abbildung 18: Ausschnit-Assembler-File

PlunserFabio Page 23 of 34



## 4.2.2 Interrupt set-enable register (ISER)

Address offset: 0x00

Reset value: 0x0000 0000

The ISER register enables interrupts, and shows which interrupts are enabled

| 31 | 30            | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 |
|----|---------------|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
|    | SETENA[31:16] |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rs | rs            | rs | rs | rs | rs | rs | rs | rs | rs | rs | rs | rs | rs | rs | rs |
| 15 | 14            | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |
|    | SETENA[15:0]  |    |    |    |    |    |    |    |    |    |    |    |    |    |    |
| rs | rs            | rs | rs | rs | rs | rs | rs | rs | rs | rs | rs | rs | rs | rs | rs |

Bits 31:0 SETENA: Interrupt set-enable bits.

#### Write:

0: No effect

1: Enable interrupt

#### Read:

0: Interrupt disabled

1: Interrupt enabled.

If a pending interrupt is enabled, the NVIC activates the interrupt based on its priority. If an interrupt is not enabled, asserting its interrupt signal changes the interrupt state to pending, but the NVIC never activates the interrupt, regardless of its priority.

Abbildung 19: NVIC-ISER-Register

## 8.2 NVIC-Code

```
void NVIC_enable_interrupts(void)
{
    uint32_t* nvic_iser = NVIC_ISER;
    uint32_t reg_content;
    reg_content = *nvic_iser;
    reg_content |= 0x08001000;
    *nvic_iser = reg_content;
}
```

Listing 7: NVIC-enable-interrupts

PlunserFabio Page 24 of 34



## 9 Gesamtes-Programm

#### 9.1 Headerfile

```
#ifndef __header_H
#define __header_H
//boundary addresses at page 37 - 41
General boundaries
#define PERIPHALS
                        ((uint32_t*)0x4000000)
                                                   //Peripherals
#define APBPERIPHALS
                       PERIPHALS
                                                   //APBPeriphals
                       (PERIPHALS + 0x00020000)
#define AHBPERIPHALS
#define AHB2PERIPHALS
                       (PERIPHALS + 0x08000000)
USART1
#define USART1 BASE (APBPERIPHALS+0x00013800) //USART 1 Base Address
//Base + Address offset + Comment + Page in reference Manual
#define USART1 CR1
                   (USART1 BASE+0x00) //USART Control Register 1
   at Page 625
#define USART1_CR2
                       (USART1_BASE+0x04)
                                            //USART Control Register 2
   at Page 628
#define USART1_CR3
                        (USART1 BASE+0x08)
                                            //USART Control Register 3
   at Page 630
#define USART1_BRR
                        (USART1_BASE+0x0C)
                                            //USART Baudrate Register
   at page 632
#define USART1_RQR
                        (USART1_BASE+0x18)
                                            //USART Request register
#define USART1 ISR
                       (USART1 BASE+0x1c)
                                            //USART Interrupt and status
  register -- at Page 635
#define USART1_ISR_TXE ((uint32_t)0x00000080)
#define USART1_ISR_TC
                        ((uint32_t)0x00000040)
#define USART1_ISR_RXNE ((uint32_t)0x00000020)
#define USART1_ICR
                        (USART1_BASE+0x20) //USART Interrupt and flag
   Clear register -- at Page 638
#define USART1_ICR_TCCF ((uint32_t) 0x00000040) //USART transmission
   complete clear flag -- at Page 639
#define USART1_RDR
                        (USART1_BASE+0x24) //USART Receive Data register
     -- at Page 639
#define USART1_TDR
                        (USART1_BASE+0x28) //USART Transmit Data register
     -- at Page 640
/*
RCC
//From boundary addresses at page 39
#define RCC (AHBPERIPHALS+0x00001000)
//All found from RCC Register Map on Page 125
```

PlunserFabio Page 25 of 34



```
#define RCC_CR
                       (RCC+0x00)
                                          //RCC Control register -- at
   Page 99
#define RCC_CFGR
                      (RCC+0x04)
                                          //RCC Clock configuration
   register -- at Page 101
#define RCC_CIR
                     (RCC+0x08)
                                          //RCC Clock interrupt register
  -- at Page 104
#define RCC_APB2RSTR (RCC+0x0C)
                                         //RCC APB peripheral reset
   register 2 -- at Page 106
#define RCC_APB1RSTR (RCC+0x010)
                                         //RCC APB peripheral reset
   register 1 -- at Page 108
#define RCC_AHBENR
                     (RCC+0x14)
                                          //RCC AHB peripheral clock
   enable register -- at Page 111
#define RCC_APB2ENR (RCC+0x18)
                                          //RCC APB peripheral clock
   enable register 2-- at Page 112
                                          //RCC APB peripheral clock
#define RCC_APB1ENR
                     (RCC+0x1C)
   enable register 1-- at Page 114
#define RCC_BDCR
                      (RCC+0x20)
                                          //RCC RTC domain control
   register -- at Page 117
#define RCC_CSR
                     (RCC+0x24)
                                          //RCC Control/status register --
   at Page 119
#define RCC_AHBRSTR (RCC+0x28)
                                         //RCC AHB peripheral reset
   register -- at Page 120
#define RCC_CFGR2
                                         //RCC Clock configuration
                      (RCC+0x2C)
   register 2 -- at Page 122
#define RCC_CFGR3 (RCC+0x30)
                                         //RCC Clock configuration
   register 3 -- at Page 123
#define RCC_CR2
                                         //RCC Clock control register 2
                     (RCC+0x34)
  -- at Page 123
/*
GPIOA
*/
#define GPIO_A (AHB2PERIPHALS + 0x00000000)
//Page 142
#define GPIOA_MODER (GPIO_A+0x00) //GPIO port moder register -- at
   Page 136
#define GPIOA_OTYPER
                       (GPIO A+0x04)
                                          //GPIO port output type register
   -- at Page 136
#define GPIOA_OSPEEDR
                       (GPIO_A+0x08)
                                          //GPIO port output speed
   register -- at Page 137
#define GPIOA_PUPDR
                                          //GPIO port pull-up/pull-down
                     (GPIO A+0x0C)
  register -- at Page 137
#define GPIOA_IDR
                      (GPIO_A+0x10)
                                          //GPIO port input data register
   -- at Page 138
                                          //GPIO port output data register
#define GPIOA_ODR
                       (GPIO_A+0x14)
   -- at Page 138
#define GPIOA_BSRR
                       (GPIO_A+0x18)
                                          //GPIO port bis set/reset
   register -- at Page 138
                                          //GPIO port configuration lock
#define GPIOA_LCKR
                     (GPIO_A+0x1c)
  register -- at Page 139
                      (GPIO_A+0x20) //GPIO alternate function low
#define GPIOA_AFRL
   register -- at Page 140
```

PlunserFabio Page 26 of 34



```
#define GPIOA_AFRH
                        (GPIO_A+0x024)
                                            //GPIO alternate function high
   register -- at Page 141
#define GPIOA_BRR
                        (GPIO_A+0x28)
                                            //GPIO port bit reset register
   -- at Page 141
/*
ADC
*/
//boundary address at page 40
#define ADC (APBPERIPHALS+0x00012400)
//Page 220
#define ADC_ISR
                       (ADC+0x00)
                                                     //ADC interrupt and
   status register -- at Page 207
#define ADC_ISR_AWD
                                                      //Analog watchdog flag
                        ((uint32_t)0x00000080)
#define ADC_ISR_OVR
                        ((uint32_t)0x00000010)
                                                      //Overrun flag
#define ADC_ISR_EOSEQ
                        ((uint32_t)0x00000008)
                                                     //End of Sequence flag
#define ADC_ISR_EOC
                        ((uint32_t)0x00000004)
                                                     //End of Conversion
                                                      //End of sampling flag
#define ADC_ISR_EOSMP
                        ((uint32_t)0x00000002)
#define ADC_ISR_ADRDY
                        ((uint32_t)0x00000001)
                                                      //ADC Ready
                                      //ADC interrupt enable register --
#define ADC_IER
                        (ADC+0x04)
   at Page 208
#define ADC CR
                        (ADC+0x08)
                                        //ADC control register -- at Page
   210
#define ADC_CR_ADCAL
                        ((uint32_t)0x80000000) //ADC Calibration
#define ADC_CR_ADSTP
                        ((uint32_t)0x00000010) //ADC stop of conversion
   command
#define ADC_CR_ADSTART
                        ((uint32_t)0x00000004) //ADC start of conversion
#define ADC_CR_ADDIS
                        ((uint32_t)0x00000002) //ADC disable command
#define ADC_CR_ADEN
                        ((uint32_t)0x00000001) //ADC enable control
#define ADC_CFGR1
                        (ADC+0xOC)
                                        //ADC configuration register 1 -- at
    Page 212
#define ADC_CFGR2
                        (ADC+0x10)
                                        //ADC configuration register 2 -- at
    Page 216
#define ADC_SMPR
                        (ADC+0x14)
                                        //ADC sampling time register -- at
  Page 216
                                        //ADC watchdog threshold register --
#define ADC_TR
                        (ADC+0x20)
   at Page 217
#define ADC_CHSELR
                        (ADC+0x28)
                                        //ADC channel selection register --
   at Page 218
#define ADC_DR
                        (ADC+0x40)
                                        //ADC data register -- at Page 218
#define ADC_DR_DATA
                                                     //ADC Data Mask
                        ((uint32_t)0x0000FFFF)
#define ADC_CCR
                        (ADC + 0 \times 308)
                                       //ADC common configuration register
   -- at Page 219
//Interrupt ant Page 171
#define ADC_IRQn 12
                              //Address 0x0000 0070
```

PlunserFabio Page 27 of 34



Listing 8: Headerfile

PlunserFabio Page 28 of 34



#### 9.2 Main

```
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/time.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdint.h>
#include "header.h"
char USART_READ;
uint32_t ADC_Value;
//Prototypes
char USART_write_data[32];
int send=0;
//ADC Interrupt Handler
void ADC1_IRQHandler(void)
    uint32_t* adc_isr = ADC_ISR;
    uint32_t* adc_cr = ADC_CR;
    uint32_t adc_isr_eoc = ADC_ISR_EOC;
    uint32 t adc isr adrdy = ADC ISR ADRDY;
    uint32_t* adc_dr = ADC_DR;
    uint32_t adc_dr_data = ADC_DR_DATA;
    uint32_t* gpioa_odr = GPIOA_ODR;
    uint32_t* usart1_tdr = USART1_TDR;
    uint32_t adc_cr_adstart = ADC_CR_ADSTART;
    uint16_t ADC_Value;
    uint32_t reg_content;
    //if ADC Ready start convertion
    if((*adc_isr & adc_isr_adrdy) == adc_isr_adrdy)
    {
        reg_content = *adc_cr;
        reg_content |= adc_cr_adstart;
        *adc_cr = reg_content;
    }
    //if convertion complete send ADC value
    if((*adc_isr & adc_isr_eoc) == adc_isr_eoc)
        send=0;
        ADC_Value = (*adc_dr & adc_dr_data);
        sprintf(USART_write_data, "%d", ADC_Value);
    }
```

PlunserFabio Page 29 of 34



```
}
//Enable Interrupts in NVIC
void NVIC_enable_interrupts(void)
    uint32_t* nvic_iser = NVIC_ISER;
    uint32_t reg_content;
    reg_content = *nvic_iser;
   reg_content |= 0x08001000;
    *nvic_iser = reg_content;
}
//USART Interrupt Handler
void USART1_IRQHandler(void)
    uint32_t* usart1_isr = USART1_ISR;
    uint32_t usart1_isr_rxne = USART1_ISR_RXNE;
    uint32_t usart1_isr_tc = USART1_ISR_TC;
    uint32_t* usart1_tc = USART1_ICR;
    uint32_t* usart1_tc_tcce = USART1_ICR_TCCF;
    uint32_t* usart1_tdr = USART1_TDR;
    uint32_t* gpioa_odr = GPIOA_ODR;
    uint32_t reg_content;
    if ((*usart1_isr & usart1_isr_tc) == usart1_isr_tc)
    {
        if (send == sizeof(USART_write_data))
        {
            //set MAX to listen
            reg_content = *gpioa_odr;
            reg_content |= 0x00000000;
            *gpioa_odr = reg_content;
            send=0;
            *usart1_tc |= *usart1_tc_tcce; // Clear transfer complete flag *
        }
        else
            // clear transfer complete flag and fill TDR with a new char
            *usart1_tdr = USART_write_data[send++];
        }
    //when something comes at usart, begin to write the ADC Data
    if ((*usart1_isr & usart1_isr_rxne) == usart1_isr_rxne)
    {
        USART_READ = *((char *)USART1_RDR);
        //Set PA7 and PA6 high for max to send data
        reg_content = *gpioa_odr;
        reg_content |= 0x000000000;
        *gpioa_odr = reg_content;
        *usart1_tdr = USART_write_data[0];
```

PlunserFabio Page 30 of 34



```
}
//Initialize ADC
int init_ADC()
{
    uint32_t reg_content;
    uint32_t* adc_chselr = ADC_CHSELR;
    uint32_t* adc_isr = ADC_ISR;
    uint32_t* adc_cr = ADC_CR;
    uint32_t* adc_cfgr1 = ADC_CFGR1;
    uint32_t adc_cr_adcal = ADC_CR_ADCAL;
    uint32_t* adc_ier = ADC_IER;
    uint32_t* adc_cfgr2 = ADC_CFGR2;
    uint32_t adc_isr_adrdy = ADC_ISR_ADRDY;
    //set ADC clock to PCLK so thats is synchronous with sysclock
    reg_content = *adc_cfgr2;
    reg_content |= 0x40000000;
    *adc_cfgr2 = reg_content;
    //before starting callibrate ADC
    reg_content = *adc_cr;
    reg_content |= adc_cr_adcal;
    *adc_cr = reg_content;
    //if calibration is complete enable ADC
    //enable Interrupts
    //set correct channel
    //when ADC is ready, a ad ready interrupt occurs and the interrupt
       handler
    //will then start the ADC convertion
    while((*adc_cr & adc_cr_adcal) == adc_cr_adcal); // wait till
       callibration is complete
    //enable ADC and ensure ADSTART=0 for further configuration and enable
       ADC
    reg_content = *adc_cr;
    reg_content |= 0x00000001;
    *adc_cr = reg_content;
    //set ADC to continues convertion so, that convertion doesn't to start
       again in Interrupt
    reg_content = *adc_cr;
    reg_content \mid = 0x0000200;
    *adc_cr = reg_content;
    //enable Interrupts of ADC
    reg_content = *adc_ier;
    reg_content |= 0x00000005;
    *adc_ier = reg_content;
```

PlunserFabio Page 31 of 34



```
//Set ADC Channel to channel O because PAO is ADC_INO
    reg_content = *adc_chselr;
    reg_content |= 0x00000001;
    *adc_chselr = reg_content;
}
//Initialize needed clocks
int init CLOCK()
    uint32_t reg_content;
    uint32_t* rcc_ahbenr = RCC_AHBENR;
    uint32_t* rcc_apb2enr = RCC_APB2ENR;
    uint32_t* rcc_cfgr = RCC_CFGR;
    //GPIO A port clock enable
    reg_content = *rcc_ahbenr;
    reg_content |= 0x00020000;
    *rcc_ahbenr = reg_content;
    //USART1 clock enable
    reg content = *rcc apb2enr;
    reg_content |= 0x00004000;
    *rcc_apb2enr = reg_content;
    //set AHB Clock to not divided so same clock as sysclock
    reg_content = *rcc_cfgr;
   reg_content |= 0x00000000;
    *rcc_cfgr = reg_content;
}
//Initialize GPIOs
int init_GPIO()
    //Set GPIO pins PA9 and PA10 alternate function for USART TX and RX,
    //set PA7 and PA6 output for DE and RE of MAX485
    //PAO ADC_INO so set it to analog and then in DAC register set PAO as
       ADC_INO
    uint32_t reg_content;
    uint32_t* gpioa_moder = GPIOA_MODER;
    uint32_t* gpioa_afrh = GPIOA_AFRH;
    uint32_t* gpioa_odr = GPIOA_ODR;
    //set above mentioned Pins
    reg_content = *gpioa_moder;
    reg_content |= 0x00285003;
    *gpioa_moder = reg_content;
    //Set GPIO PA9 as AF=TX and PA10 as AF = RX
    reg_content = *gpioa_afrh;
    reg_content |= 0x00000110;
    *gpioa_afrh = reg_content;
    //Set GPIO PA9 as AF=TX and PA10 as AF = RX
```

PlunserFabio Page 32 of 34



```
reg_content = *gpioa_afrh;
    reg_content |= 0x00000110;
    *gpioa_afrh = reg_content;
    //Ensure that output gpios are 0 for MAX, to read all the time
    //and only send, when needing to send
    reg_content = *gpioa_odr;
    reg_content |= 0x00000000;
    *gpioa_odr = reg_content;
}
//Initialize USART
int init_UART()
    uint32_t reg_content;
    uint32_t* rcc_cfr3 = RCC_CFGR3;
    uint32_t* usart1_brr = USART1_BRR;
    uint32_t* usart1_cr1 = USART1_CR1;
    //Use SYSCLK for USART and use Baudrate 38400
    reg_content = *rcc_cfr3;
    reg_content |= 0x00000001;
    *rcc_cfr3 = reg_content;
    reg_content = *usart1_brr;
    reg_content = 0x00000D00;
    *usart1_brr = reg_content;
    //Wordlenght, Parity control enable, Parity selection, interrupt enable,
        Transmission complete interrupt enable, RXNE interrupt enable
    reg_content = *usart1_cr1;
    reg_content |= 0x000006EC;
    *usart1_cr1 = reg_content;
    reg_content = *usart1_cr1;
    reg_content |= 0x00000001; //enable UART
    *usart1_cr1 = reg_content;
}
int main()
   init_CLOCK();
    init_GPIO();
    init_UART();
    init_ADC();
   NVIC_enable_interrupts();
    for(;;);
}
```

Listing 9: Gesamter-Code

PlunserFabio Page 33 of 34



## 10 Anhang

## 10.1 Verlinkungen

Abbildung: 0

http://www.mathe-mit-methode.com/schlaufuchs\_web/elektrotechnik/mikrocontroller\_lernmaterial/microcontroller\_allgemein/mikrocontroller\_ext\_hardware/mikrocontroller\_uart\_bild\_001.html

Abbildung: 1

https://de.wikipedia.org/wiki/EIA-485

Abbildung: 3

68a216485b59.jpg von aryeguetta

PlunserFabio Page 34 of 34