



# Interrupts, PWM and ADC

Lecture 3



# Interrupts, PWM and ADC

- Interrupts
- Counters
- Timers and Alarms
- About Analog and Digital Signals
- Pulse Width Modulation (PWM)
- Analog to Digital Converters (ADC)



# Exceptions

for the ARM Cortex-M33 processor



# Bibliography

for this section

**Joseph Yiu, *The Definitive Guide to ARM® Cortex®-M23 and Cortex-M33 Processors***

- Chapter 4 - *Architecture*
  - Section 4.5 - *Exceptions and Interrupts*
    - Subsection 4.4.1 - *What are exceptions*
- Chapter 8 - *Exceptions and Interrupts*
  - Section 8.1 - *What are Exceptions and Interrupts*
  - Section 8.2 - *Exception types*<sup>+</sup>



# Processor Exceptions

what happens if something does not work as required





# Standard ARM Cortex-M Exceptions

what happens if something does not work as required





# ARM Cortex-M Interrupts

some hardware device notifies the MCU





# Exceptions and Interrupts in Embassy

- Embassy registers handlers for Exceptions
- Each of the Embassy drivers that you use provides interrupt handlers for the peripheral they control
- Developers have to **bind** interrupts to the driver.

| IRQ | Interrupt Source |
|-----|------------------|-----|------------------|-----|------------------|-----|------------------|-----|------------------|
| 0   | TIMER0_IRQ_0     | 11  | DMA_IRQ_1        | 22  | IO_IRQ_BANK0_NS  | 33  | UART0_IRQ        | 44  | POWMAN_IRQ_POW   |
| 1   | TIMER0_IRQ_1     | 12  | DMA_IRQ_2        | 23  | IO_IRQ_QSPI      | 34  | UART1_IRQ        | 45  | POWMAN_IRQ_TIMER |
| 2   | TIMER0_IRQ_2     | 13  | DMA_IRQ_3        | 24  | IO_IRQ_QSPI_NS   | 35  | ADC_IRQ_FIFO     | 46  | SPAREIRQ_IRQ_0   |

List of some of the RP2350's interrupts

Register the Interrupt

```
bind_interrupts!(struct Irqs {
    ADC_IRQ_FIFO => InterruptHandler;
});
```

Bind it to the driver

```
let mut adc = Adc::new(p.ADC, Irqs, Config::default());
```



# Timers



# Bibliography

for this section

## 1. Joseph Yiu, *The Definitive Guide to ARM® Cortex®-M23 and Cortex-M33 Processors*

- Chapter 11 - *OS support features*
  - Section 11.2 - *SysTick timer*

## 2. Raspberry Pi Ltd, *RP2350 Datasheet*

- Chapter 8 - *Clocks*
  - Chapter 8.1 - *Overview*
    - Subchapter 8.1.1
    - Subchapter 8.1.2
- Chapter 12 - *Peripherals*
  - Chapter 12.8 - *System Timers*



# Clocks for RP235x

all peripherals and the MCU use a clock to execute at certain intervals

| Source                            | Usage                                                       |
|-----------------------------------|-------------------------------------------------------------|
| <i>external crystal</i><br>(XOSC) | a stable frequency is required, for instance when using USB |
| <i>internal ring</i><br>(ROSC)    | low frequency, in between 1.8 - 12 MHz (varies)             |

Embassy initializes the Raspberry Pi Pico with the clock source from the 12 MHz crystal.

```
1 let p = embassy_rp::init(Default::default());
```





# Clocks for STM32U545RE

all peripherals and the MCU use a clock to execute at certain intervals

## Source      Usage

*LSE, LSE32 and HSE*      oscillators based on crystals external crystals

*LSI, HSI, HSI48, MSIS and MSIK*      Internal RC oscillators which can be used as main clock source or as a clock source for peripherals

Embassy initializes the STM32U545RE with the clock source from Multi-Speed Internal oscillator (**MSIS**).

```
1 let p = embassy_stm32::init(Default::default());
```





# Frequency divider

stabilizing the signal and adjusting it

1. divides down the clock signals used for the timer, giving reduced overflow rates
2. allows the timer to be clocked at a user desired rate





# Counter

increments a register at every clock cycle

## Registers      Description

`value`

the current value of the counter

`direction`

set to count UP or DOWN

`reset`

UP: the value at which the counter resets to `0`  
DOWN: the value to which the counter resets after getting to `0`





# SysTick

ARM Cortex-M time counter

The ARM Cortex-M0+ registers start at a base address of `0xe0000000` (defined as `PPB_BASE` in SDK).

| Offset | Name                    | Info                                |
|--------|-------------------------|-------------------------------------|
| 0xe010 | <code>SYST_CSR</code>   | SysTick Control and Status Register |
| 0xe014 | <code>SYST_RVR</code>   | SysTick Reload Value Register       |
| 0xe018 | <code>SYST_CVR</code>   | SysTick Current Value Register      |
| 0xe01c | <code>SYST_CALIB</code> | SysTick Calibration Value Register  |

- decrements the value of `SYST_CVR` every  $\mu\text{s}$
- when `SYST_CVR` becomes `0` :
  - triggers the `SysTick` exception
  - next clock cycle sets the value of `SYST_CVR` to `SYST_RVR`
- `SYST_CALIB` is the value of `SYST_RVR` for a 10ms interval (might not be available)

## SYST\_CSR register

| Bits  | Name      | Description                                                                                                                                                                        | Type | Reset |
|-------|-----------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------|-------|
| 31:17 | Reserved. | -                                                                                                                                                                                  | -    | -     |
| 16    | COUNTFLAG | Returns 1 if timer counted to 0 since last time this was read. Clears on read by application or debugger.                                                                          | RO   | 0x0   |
| 15:3  | Reserved. | -                                                                                                                                                                                  | -    | -     |
| 2     | CLKSOURCE | SysTick clock source. Always reads as one if SYST_CALIB reports NOREF.<br>Selects the SysTick timer clock source:<br>0 = External reference clock.<br>1 = Processor clock.         | RW   | 0x0   |
| 1     | TICKINT   | Enables SysTick exception request:<br>0 = Counting down to zero does not assert the SysTick exception request.<br>1 = Counting down to zero asserts the SysTick exception request. | RW   | 0x0   |
| 0     | ENABLE    | Enable SysTick counter:<br>0 = Counter disabled.<br>1 = Counter enabled.                                                                                                           | RW   | 0x0   |

$$f = \frac{1}{SYST_RVR} * 1,000,000 [Hz]_{SI}$$



# SysTick

ARM Cortex-M peripheral

The ARM Cortex-M0+ registers start at a base address of `0xe0000000` (defined as `PPB_BASE` in SDK).

| Offset | Name                    | Info                                |
|--------|-------------------------|-------------------------------------|
| 0xe010 | <code>SYST_CSR</code>   | SysTick Control and Status Register |
| 0xe014 | <code>SYST_RVR</code>   | SysTick Reload Value Register       |
| 0xe018 | <code>SYST_CVR</code>   | SysTick Current Value Register      |
| 0xe01c | <code>SYST_CALIB</code> | SysTick Calibration Value Register  |

```
1 const SYST_RVR: *mut u32 = 0xe000_e014 as *mut u32;
2 const SYST_CVR: *mut u32 = 0xe000_e018 as *mut u32;
3 const SYST_CSR: *mut u32 = 0xe000_e010 as *mut u32;
4
5 // fire systick every 5 seconds
6 let interval: u32 = 5_000_000;
7 unsafe {
8     write_volatile(SYST_RVR, interval);
9     write_volatile(SYST_CVR, 0);
10    // set fields `ENABLE` and `TICKINT`
11    write_volatile(SYST_CSR, 0b11);
12 }
```

## SYST\_CSR register

| Bits  | Name      | Description                                                                                                                                                                        | Type | Reset |
|-------|-----------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------|-------|
| 31:17 | Reserved. | -                                                                                                                                                                                  | -    | -     |
| 16    | COUNTFLAG | Returns 1 if timer counted to 0 since last time this was read. Clears on read by application or debugger.                                                                          | RO   | 0x0   |
| 15:3  | Reserved. | -                                                                                                                                                                                  | -    | -     |
| 2     | CLKSOURCE | SysTick clock source. Always reads as one if SYST_CALIB reports NOREF.<br>Selects the SysTick timer clock source:<br>0 = External reference clock.<br>1 = Processor clock.         | RW   | 0x0   |
| 1     | TICKINT   | Enables SysTick exception request:<br>0 = Counting down to zero does not assert the SysTick exception request.<br>1 = Counting down to zero asserts the SysTick exception request. | RW   | 0x0   |
| 0     | ENABLE    | Enable SysTick counter:<br>0 = Counter disabled.<br>1 = Counter enabled.                                                                                                           | RW   | 0x0   |

## Register SysTick handler

```
1 #[exception]
2 unsafe fn SysTick() {
3     /* systick fired */
4 }
```



# Alarm

counter that triggers interrupts after a time interval

## Registers      Description

|                        |                                                                                                    |
|------------------------|----------------------------------------------------------------------------------------------------|
| <code>value</code>     | the current value of the counter                                                                   |
| <code>direction</code> | set to count UP or DOWN                                                                            |
| <code>reset</code>     | UP: max value before 0<br>DOWN: value after 0                                                      |
| <code>alarm_x</code>   | when <code>value == alarm_x</code> , triggers an interrupt, <code>x</code> in <code>1 ... n</code> |





# RP2350's Timers

two timers, `TIMER0` and `TIMER1`

- store a 64 bit number (`reset` is  $2^{64-1}$ )
- start with `0` at (the peripheral's) reset
- increment the number every  $\mu\text{s}$
- in practice fully monotonic (cannot over flow)
- allow 4 alarms that trigger interrupts
  - `TIMER0_IRQ_0` and `TIMER1_IRQ_0`
  - `TIMER0_IRQ_1` and `TIMER1_IRQ_1`
  - `TIMER0_IRQ_2` and `TIMER1_IRQ_2`
  - `TIMER0_IRQ_3` and `TIMER1_IRQ_3`
- `alarm_0 ... alarm_3` registers are only 32 bits wide





# RP2350's Timer instance

read the number of elapsed  $\mu\text{s}$  since reset

Reading the time elapsed since restart

```
1 const TIMERLR: *const u32 = 0x400b_000c;
2 const TIMERHR: *const u32 = 0x400b_0008;
3
4 let time: u64 = unsafe {
5     let low = read_volatile(TIMERLR);
6     let high = read_volatile(TIMERHR);
7     high as u64 << 32 | low
8 }
```

The **reading order matters** as reading `TIMELR` latches the value in `TIMEHR` (stops being updated) until `TIMEHR` is read. Works only in **single core**.

The `TIMER0` and `TIMER1` registers start at base addresses of `0x400b0000` and `0x400b8000` respectively (defined as `TIMER0_BASE` and `TIMER1_BASE` in SDK).

| Offset | Name                  | Info                                                                                                                                                                                                                                  |
|--------|-----------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 0x00   | <code>TIMEHW</code>   | Write to bits 63:32 of time<br>always write timelw before timehw                                                                                                                                                                      |
| 0x04   | <code>TIMElw</code>   | Write to bits 31:0 of time<br>writes do not get copied to time until timehw is written                                                                                                                                                |
| 0x08   | <code>TIMEHR</code>   | Read from bits 63:32 of time<br>always read timerl before timehr                                                                                                                                                                      |
| 0x0c   | <code>TIMElr</code>   | Read from bits 31:0 of time                                                                                                                                                                                                           |
| 0x10   | <code>ALARM0</code>   | Arm alarm 0, and configure the time it will fire.<br>Once armed, the alarm fires when <code>TIMER_ALARM0 == TIMELR</code> .<br>The alarm will disarm itself once it fires, and can be disarmed early using the ARMED status register. |
| 0x14   | <code>ALARM1</code>   | Arm alarm 1, and configure the time it will fire.<br>Once armed, the alarm fires when <code>TIMER_ALARM1 == TIMELR</code> .<br>The alarm will disarm itself once it fires, and can be disarmed early using the ARMED status register. |
| 0x18   | <code>ALARM2</code>   | Arm alarm 2, and configure the time it will fire.<br>Once armed, the alarm fires when <code>TIMER_ALARM2 == TIMELR</code> .<br>The alarm will disarm itself once it fires, and can be disarmed early using the ARMED status register. |
| 0x1c   | <code>ALARM3</code>   | Arm alarm 3, and configure the time it will fire.<br>Once armed, the alarm fires when <code>TIMER_ALARM3 == TIMELR</code> .<br>The alarm will disarm itself once it fires, and can be disarmed early using the ARMED status register. |
| 0x20   | <code>ARMED</code>    | Indicates the armed/disarmed status of each alarm.<br>A write to the corresponding ALARMx register arms the alarm.<br>Alarms automatically disarm upon firing, but writing ones here will disarm immediately without waiting to fire. |
| 0x24   | <code>TIMERAWH</code> | Raw read from bits 63:32 of time (no side effects)                                                                                                                                                                                    |
| 0x28   | <code>TIMERAWL</code> | Raw read from bits 31:0 of time (no side effects)                                                                                                                                                                                     |
| 0x2c   | <code>DBGPAUSE</code> | Set bits high to enable pause when the corresponding debug ports are active                                                                                                                                                           |
| 0x30   | <code>PAUSE</code>    | Set high to pause the timer                                                                                                                                                                                                           |
| 0x34   | <code>LOCKED</code>   | Set locked bit to disable write access to timer<br>Once set, cannot be cleared (without a reset)                                                                                                                                      |



# Alarm

triggering an interrupt at an interval

```
1 #[interrupt]
2 unsafe fn TIMER0_IRQ_0() { /* alarm fired */ }

1 const TIMERLR: *const u32 = 0x400b_000c;
2 const ALARM0: *mut u32 = 0x400b_0010;
3 // + 0x2000 is bitwise set
4 const INTE_SET: *mut u32 = 0x400b_0040;
5
6 // set an alarm after 3 seconds
7 let us = 3_0000_0000;
8
9 unsafe {
10     let time = read_volatile(TIMERLR);
11     // use `wrapping_add` as overflowing may panic
12     write_volatile(ALARM0, time.wrapping_add(us));
13     write_volatile(INTE_SET, 1 << 0);
14 }
```

- the alarm can be set only for the lower 32 bits
- maximum 72 minutes (use *RTC* for longer alarms)

The `TIMER0` and `TIMER1` registers start at base addresses of `0x400b0000` and `0x400b8000` respectively (defined as `TIMER0_BASE` and `TIMER1_BASE` in SDK).

| Offset | Name   | Info                                                                                                                                                                                                                                  |
|--------|--------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 0x00   | TIMEHW | Write to bits 63:32 of time<br>always write timelw before timehw                                                                                                                                                                      |
| 0x04   | TIMELW | Write to bits 31:0 of time<br>writes do not get copied to time until timehw is written                                                                                                                                                |
| 0x08   | TIMEHR | Read from bits 63:32 of time<br>always read timelr before timehr                                                                                                                                                                      |
| 0x0c   | TIMELR | Read from bits 31:0 of time                                                                                                                                                                                                           |
| 0x10   | ALARM0 | Arm alarm 0, and configure the time it will fire.<br>Once armed, the alarm fires when <code>TIMER_ALARM0 == TIMELR</code> .<br>The alarm will disarm itself once it fires, and can be disarmed early using the ARMED status register. |
| 0x14   | ALARM1 | Arm alarm 1, and configure the time it will fire.<br>Once armed, the alarm fires when <code>TIMER_ALARM1 == TIMELR</code> .<br>The alarm will disarm itself once it fires, and can be disarmed early using the ARMED status register. |
| 0x18   | ALARM2 | Arm alarm 2, and configure the time it will fire.<br>Once armed, the alarm fires when <code>TIMER_ALARM2 == TIMELR</code> .<br>The alarm will disarm itself once it fires, and can be disarmed early using the ARMED status register. |
| 0x1c   | ALARM3 | Arm alarm 3, and configure the time it will fire.<br>Once armed, the alarm fires when <code>TIMER_ALARM3 == TIMELR</code> .<br>The alarm will disarm itself once it fires, and can be disarmed early using the ARMED status register. |
| Offset | Name   | Info                                                                                                                                                                                                                                  |
| 0x38   | SOURCE | Selects the source for the timer. Defaults to the normal tick configured in the ticks block (typically configured to 1 microsecond). Writing to 1 will ignore the tick and count <code>clk_sys</code> cycles instead.                 |
| 0x3c   | INTR   | Raw Interrupts                                                                                                                                                                                                                        |
| 0x40   | INTE   | Interrupt Enable                                                                                                                                                                                                                      |
| 0x44   | INTF   | Interrupt Force                                                                                                                                                                                                                       |
| 0x48   | INTS   | Interrupt status after masking & forcing                                                                                                                                                                                              |



# STM32U5's Timers

11 timers and 4 low power timers

## Basic Timers

- two basic 16-bit timers

## Timers

- PWM generation
- four 32-bit timers
- three 16-bit timers
- two advanced control 16-bit timers

## Low Power Timers

- four low power 16-bit timers

| Timer type       | Timer                  | Counter resolution | Counter type      | Prescaler factor                | DMA request generation | Capture/compare channels | Complementary outputs |
|------------------|------------------------|--------------------|-------------------|---------------------------------|------------------------|--------------------------|-----------------------|
| Advanced control | TIM1, TIM8             | 16 bits            | Up, down, Up/down | Any integer between 1 and 65536 | Yes                    | 4                        | 3                     |
| General-purpose  | TIM2, TIM3, TIM4, TIM5 | 32 bits            | Up, down, Up/down | Any integer between 1 and 65536 | Yes                    | 4                        | No                    |

| Timer type      | Timer        | Counter resolution | Counter type | Prescaler factor                | DMA request generation | Capture/compare channels | Complementary outputs |
|-----------------|--------------|--------------------|--------------|---------------------------------|------------------------|--------------------------|-----------------------|
| General-purpose | TIM15        | 16 bits            | Up           | Any integer between 1 and 65536 | Yes                    | 2                        | 1                     |
| General-purpose | TIM16, TIM17 | 16 bits            | Up           | Any integer between 1 and 65536 | Yes                    | 1                        | 1                     |
| Basic           | TIM6, TIM7   | 16 bits            | Up           | Any integer between 1 and 65536 | Yes                    | 0                        | No                    |



# Signals

Digital Signals - Recap



# Signals

Analog vs Digital

- *analog signals* are *real signals*
- *digital signals* are a numerical representation of an analog signal (software level)
- hardware usually works with two-level digital signals (hardware level)

Exceptions

- in wireless and in high-speed cable communication things get more complicated

for PCB level / between integrated circuits on the same board / inside the same chip - things are a "a little simpler" - as detailed in the following





# Why use digital in computing?

Signal that we *want* to generate with an output pin



Signal that what we actually generate



Why we still use it? Because after passing through an IC or a gate inside an IC - the signal is "rebuilt" and if the "digital discipline" described in the following is respected - we can preserve the information after numerous "passes". Thus, each element can behave with a large margin for error, yet the final result is correct.



# Noise Margin





# PWM

Pulse Width Modulation



# Bibliography

for this section

## 1. Raspberry Pi Ltd, *RP2350 Datasheet*

- Chapter 12 - *Peripherals*
  - Section 12.5 - *PWM*

## 2. Paul Denisowski, *Understanding PWM*



# PWM

simulates an *analog* signal (using integration)

- generates a square signal
- if integrated (averaged), it looks like an analog signal

---

*frequency* Hz The number of repeats per s

---

*duty\_cycle* % The percentage of the time when the signal is High



$$f = \frac{1}{\text{period}} \left[ \frac{1}{s} = 1 \text{Hz} \right]_{SI}$$

$$\text{duty\_cycle} = \frac{\text{time\_on}}{\text{period}} \%$$



# PWM

generic device

$$f = \begin{cases} \frac{f_{clock}}{divider \times (top+1)} & correction = 0 \\ \frac{f_{clock}}{divider \times 2 \times (top+1)} & correction = 1 \end{cases}$$

$$pin_{a,b} = \begin{cases} 0 & compare_{a,b} \geq value \\ 1 & compare_{a,b} < value \end{cases}$$





# Usage examples

- dimming an LED



- controlling motors
  - controlling the angle of a stepper motor
  - controlling the RPM of a motor





# RP2350's PWM

- generates square signals
- counts the pulse width of input signals
- 8 or 12<sup>[1]</sup> PWM slices, each A and B channels
- each PWM channel is linked to a fixed pin
- some channels are connected to two pins
- may be used as timers ( `IRQ1_INTE` )



| GPIO        | 0  | 1  | 2  | 3  | 4   | 5   | 6   | 7   | 8  | 9  | 10 | 11 | 12  | 13  | 14  | 15  |
|-------------|----|----|----|----|-----|-----|-----|-----|----|----|----|----|-----|-----|-----|-----|
| PWM Channel | 0A | 0B | 1A | 1B | 2A  | 2B  | 3A  | 3B  | 4A | 4B | 5A | 5B | 6A  | 6B  | 7A  | 7B  |
| GPIO        | 16 | 17 | 18 | 19 | 20  | 21  | 22  | 23  | 24 | 25 | 26 | 27 | 28  | 29  | 30  | 31  |
| PWM Channel | 0A | 0B | 1A | 1B | 2A  | 2B  | 3A  | 3B  | 4A | 4B | 5A | 5B | 6A  | 6B  | 7A  | 7B  |
| GPIO        | 32 | 33 | 34 | 35 | 36  | 37  | 38  | 39  | 40 | 41 | 42 | 43 | 44  | 45  | 46  | 47  |
| PWM Channel | 8A | 8B | 9A | 9B | 10A | 10B | 11A | 11B | 8A | 8B | 9A | 9B | 10A | 10B | 11A | 11B |

## Registers

The PWM registers start at a base address of `0x400a8000` (defined as `PWM_BASE` in the SDK).

| Offset | Name                 | Info                                                                                                                                                                            |
|--------|----------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 0x000  | <code>CH0_CSR</code> | Control and status register                                                                                                                                                     |
| 0x004  | <code>CH0_DIV</code> | INT and FRAC form a fixed-point fractional number.<br>Counting rate is system clock frequency divided by this number.<br>Fractional division uses simple 1st-order sigma-delta. |
| 0x008  | <code>CH0_CTR</code> | Direct access to the PWM counter                                                                                                                                                |
| 0x00c  | <code>CH0_CC</code>  | Counter compare values                                                                                                                                                          |
| 0x010  | <code>CH0_TOP</code> | Counter wrap value                                                                                                                                                              |



1. Depends on the RP2350 package ↪



# RP2350's PWM Modes

standard mode



phase-correct mode



$$period = (TOP + 1) \times (PH\_CORRECT + 1) \times \left( DIV\_INT + \frac{DIV\_FRAC}{16} \right) [s]_{SI}$$

$$f = \frac{f_{sys}}{period} [Hz]_{SI}$$



# Example

using Embassy

```
1  use embassy_rp::pwm::{Config, Pwm};  
2  
3  let p = embassy_rp::init(Default::default());  
4  
5  let mut c: Config = Default::default();  
6  c.top = 0x8000;  
7  c.compare_b = 8;  
8  
9  let mut pwm = Pwm::new_output_b(  
10    p.PWM_SLICE4,  
11    p.PIN_25,  
12    c.clone()  
13);  
14  
15 loop {  
16     info!("LED duty cycle: {} / 32768", c.compare_b);  
17     Timer::after_secs(1).await;  
18     c.compare_b += 10;  
19     pwm.set_config(&c);  
20 }
```

```
pub struct Config {  
    /// Inverts the PWM output signal on channel A.  
    pub invert_a: bool,  
    /// Inverts the PWM output signal on channel B.  
    pub invert_b: bool,  
    /// Enables phase-correct mode for PWM operation.  
    pub phase_correct: bool,  
    /// Enables the PWM slice, allowing it to generate an output.  
    pub enable: bool,  
    /// A fractional clock divider, represented as a fixed-point type.  
    /// It has 8 integer bits and 4 fractional bits. It allows precise control  
    /// over the PWM output frequency by gating the PWM counter in software.  
    /// A higher value will result in a slower output frequency.  
    pub divider: fixed::FixedU16<fixed::types::extra::U4>,  
    /// The output on channel A goes high when `compare_a` is greater than or equal to  
    /// the current counter value. A compare of 0 will produce an always low output.  
    pub compare_a: u16,  
    /// The output on channel B goes high when `compare_b` is greater than or equal to  
    /// the current counter value.  
    pub compare_b: u16,  
    /// The point at which the counter wraps, representing the PWM period.  
    /// The counter will either wrap to 0 or reverse direction depending on the  
    /// setting of `phase_correct`. The value must be greater than 0.  
    pub top: u16,
```



# STM32U545RE's PWM

- generates square signals
- counts the pulse width of input signals
- each **timer** (*TIM*) has up to four channels
- each PWM channel is connected to one or more pins
- frequency is determined by the value of the *TIMx\_ARR* register, and the duty cycle by the value of the *TIMx\_CCRy* register.

## Pin Alternate functions

Table 28. Alternate function AF0 to AF7<sup>(1)</sup> (continued)

| Port | AF0                   | AF1                 | AF2                         | AF3                                                   | AF4                            | AF5                                         | AF6                            | AF7           |
|------|-----------------------|---------------------|-----------------------------|-------------------------------------------------------|--------------------------------|---------------------------------------------|--------------------------------|---------------|
|      | CRS/LPTIM1/<br>SYS_AF | LPTIM1/<br>TIM1/2/8 | LPTIM1/2/3/<br>TIM1/2/3/4/5 | ADF1/I2C4/<br>OCTOSPI/<br>SAI1/SPI2/<br>TIM18/<br>USB | DCMI/<br>I2C1/2/3/4/<br>LPTIM3 | DCMI/I2C4/<br>MDF1/<br>OCTOSPI/<br>SPI1/2/3 | I2C3/MDF1/<br>OCTOSPI/<br>SPI3 | USART1/3      |
| PB0  | -                     | TIM1_CH2N           | TIM3_CH3                    | TIM8_CH2N                                             | LPTIM3_CH1                     | SPI1 NSS                                    | -                              | USART3 CK     |
| PB1  | -                     | TIM1_CH3N           | TIM3_CH4                    | TIM8_CH3N                                             | LPTIM3_CH2                     | -                                           | MDF1_SDIO                      | USART3 RTS DE |
| PB2  | -                     | LPTIM1_CH1          | -                           | TIM8_CH4N                                             | I2C3_SMBA                      | SPI1_RDY                                    | MDF1_CKIO                      | -             |

Table 15. ARDUINO® D[7-0] connector (CN9) pinout

| Pin | Pin name          | Signal name | STM32 pin | MCU function |
|-----|-------------------|-------------|-----------|--------------|
| 1   | D7                | IO          | PA8       | I/O          |
| 2   | D6                | PWM         | PB10      | TIM2_CH3     |
| 3   | D5                | PWM         | PB4       | TIM3_CH1     |
| 4   | D4                | IO          | PB5       | I/O          |
| 5   | D3                | PWM         | PB3       | TIM2_CH2     |
| 6   | D2                | IO          | PC8       | I/O          |
| 7   | D1 <sup>(1)</sup> | USART_A_TX  | PA2       | LPUART1      |
| 8   | D0 <sup>(1)</sup> | USART_A_RX  | PA3       | LPUART1      |





# Example

using Embassy

```
1  use embassy_stm32::timer::simple_pwm::PwmPin;
2  use embassy_stm32::timer::simple_pwm::SimplePwm;
3  use embassy_stm32::timer::low_level::CountingMode;
4
5  let p = embassy_stm32::init(Default::default());
6
7  let pin = PwmPin::new(p.PB0, OutputType::PushPull);
8  let mut pwm = SimplePwm::new(
9      p.TIM3,                                // Timer instance
10     None, None, Some(pin), None,           // Pin channel map
11     khz(10),                               // Frequency
12     CountingMode::default()               // Counter config
13 );
14 let mut ch3 = pwm.ch3();
15
16 loop {
17     ch3.set_duty_cycle_fully_off();
18     Timer::after_millis(300).await;
19     ch3.set_duty_cycle_fraction(1, 2);
20     Timer::after_millis(300).await;
21     ch3.set_duty_cycle(ch3.max_duty_cycle() - 1);
```

```
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
pub enum CountingMode {
    #[default]
    /// The timer counts up to the reload value and then
    /// resets back to 0.
    EdgeAlignedUp,
    /// The timer counts down to 0 and then resets back to
    /// the reload value.
    EdgeAlignedDown,
    /// The timer counts up to the reload value and then
    /// counts back to 0.
    /// The output compare interrupt flags of channels
    /// configured in output are set when the counter is
    /// counting down.
    CenterAlignedDownInterrupts,
    /// The timer counts up to the reload value and then
    /// counts back to 0.
    /// The output compare interrupt flags of channels
    /// configured in output are set when the counter is
    /// counting up.
    CenterAlignedUpInterrupts,
    /// The timer counts up to the reload value and then
    /// counts back to 0.
    /// The output compare interrupt flags of channels
    /// configured in output are set when the counter is
    /// counting both up or down.
```



# ADC

Analog to Digital Converter



# Bibliography

for this section

**Raspberry Pi Ltd, RP2040 Datasheet**

- Chapter 12 - *Peripherals*
  - Section 12.4 - *ADC and Temperature Sensor*
    - Subchapter 12.4.2
    - Subchapter 12.4.3
    - Subchapter 12.4.6



# ADC

sampling an analog signal to an array of values

*sampling rate*

Hz

the frequency at which a new sample is read

*resolution*

bits

the number of bits used to store a sampled value



Lower sample rates yield the *aliasing effect*.



# Nyquist–Shannon Sampling Theorem

$$\text{sampling}_f > 2 \times \text{max}_f$$

The **sampling frequency** has to be at least **two times higher** than the **maximum frequency** of the signal to avoid frequency aliasing<sup>[1]</sup>.



- 
1. Aliasing is the overlapping of frequency components. This overlap results in distortion or artifacts when the signal is reconstructed from samples which causes the **reconstructed signal to differ from the original** continuous signal. ↵



# Sampling

how the ADC works

- assumes bit<sub>n-1</sub> of compare\_value is 1
- compares the input signal with a generated analog signal from compare\_value
  - if input is lower, bit<sub>n-1</sub> is 0
  - if input if higher, bit<sub>n-1</sub> is 1
- repeats for bit<sub>n-2</sub>, bit<sub>n-3</sub> ... bit<sub>0</sub>



There are different types of ADCs depending on the architecture. The most common used is SAR (*Successive Approximation Register*) ADC, also integrated in RP2350.



# RP2350's ADC

*channels* 4 or 8<sup>[1]</sup>

*sampling rate* 500 kHz

*resolution* 12 bits

$V_{max}$  3.3 V

- requires a 48 MHz clock signal
- channel 4 or 8<sup>[1:1]</sup> is connected to the internal temperature sensor

$$t = 27 - \frac{(V_{input\_4} - 0.706)}{0.001721} [{}^{\circ}\text{C}]_{SI}$$





# ADC

## in Embassy

```
1  use embassy_rp::adc::{Adc, Channel, Config, InterruptHandler};  
2  
3  bind_interrupts!(struct Irqs {  
4      ADC_IRQ_FIFO => InterruptHandler;  
5  });  
6  
7  let p = embassy_rp::init(Default::default());  
8  let mut adc = Adc::new(p.ADC, Irqs, Config::default());  
9  
10 let mut p26 = Channel::new_pin(p.PIN_26, Pull::None);  
11  
12 loop {  
13     let level = adc.read(&mut p26).await.unwrap();  
14     info!("Pin 26 ADC: {}", level);  
15     let voltage = 3300 * level / 4095;  
16     info!("Pin 26 voltage: {}.{}/V", voltage / 1000, voltage % 1000);  
17     Timer::after_secs(1).await;  
18 }
```



# STM32U545RE's ADC

ADC12 and ADC4

ADC12      ADC4

|                      |          |          |
|----------------------|----------|----------|
| <i>channels</i>      | 20       | 23       |
| <i>sampling rate</i> | 2.5 Msps | 2.5 Msps |
| <i>resolution</i>    | 14 bits  | 12 bits  |
| $V_{max}$            | 3.3 V    | 3.3 V    |

- internal channels connected to
  - temperature sensors ( $V_{SENSE}$ )
  - $V_{BAT}$  monitoring channel
  - internal reference voltage ( $V_{REFERENCE}$ )
  - $V_{CORE}$  and DAC 1 and 2 output channels

## ADC1 connectivity





# ADC - blocking

in Embassy

```
1  use embassy_stm32::adc;
2
3  let mut p = embassy_stm32::init(Default::default());
4
5  let mut adc1 = adc::Adc::new(p.ADC1);
6
7  adc1.set_resolution(adc::Resolution::BITS14);
8  adc1.set_averaging(adc::Averaging::Samples1024);
9  adc1.set_sample_time(adc::SampleTime::CYCLES160_5);
10
11 let measurement = adc1.blocking_read(&p.PA3);
12
13 let max = adc::resolution_to_max_count(adc::Resolution::BITS14);
14 let voltage: f32 = 3.3 * measurement as f32 / max as f32;
```



# ADC - asynchronous

in Embassy

```
1  use embassy_stm32::adc;
2
3  let mut p = embassy_stm32::init(Default::default());
4
5  let mut adc1 = adc::Adc::new(p.ADC1);
6  let mut adc1_pin = p.PA3;
7
8  adc1.set_resolution(adc::Resolution::BITS14);
9  adc1.set_averaging(adc::Averaging::Samples1024);
10 adc1.set_sample_time(adc::SampleTime::CYCLES160_5);
11
12 let mut degraded_channel = adc1_pin.degrade_adc();
13
14 let mut measurements = [0u16; 1];
15 adc1.read(
16     p.GPDMA1_CH0.reborrow(),
17     [(&mut degraded_channel, adc::SampleTime::CYCLES160_5)].into_iter(),
18     &mut measurements,
19 ).await;
20
21 let max = adc::resolution_to_max_count(adc::Resolution::BITS14);
22 let voltage: f32 = 3.3 * measurements[0] as f32 / max as f32;
```



# Conclusion

we talked about

- Exceptions and Interrupts
- Counters
- SysTick
- Timers and Alarms
- PWM
- Analog and Digital
- ADC