## 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*+ 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 | IRQ | Interrupt Source | IRQ | Interrupt Source | IRQ | Interrupt Source | 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 it to the driver ``` bind_interrupts!(struct Irqs { ADC_IRQ_FIFO => InterruptHandler; }); ``` ``` let mut adc = Adc::new(p.ADC, Irqs, Config::default()); ``` ## **Timers** ### Bibliography for this section #### 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* all peripherals and the MCU use a clock to execute at certain intervals | Source | Usage | |----------------------------|-------------------------------------------------------------| | external crystal<br>(XOSC) | a stable frequency is required, for instance when using USB | | internal ring<br>(ROSC) | low frequency, in between 1.8 - 12<br>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()); ``` ### 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 | Counte increments a re | <b>r</b><br>egister at every clock cycle | Clock | |------------------------|---------------------------------------------------------------------------------------------------------------|-----------------------| | Registers | Description | | | value | the current value of the counter | Counter | | direction | set to count UP or DOWN | Up / Down value CMP | | reset | UP: the value at which the counter resets to 0 DOWN: the value to which the counter resets after getting to 0 | direction reset_value | ### SysTick #### ARM Cortex-M time counter The ARM Cortex-M0+ registers start at a base address of <code>0xe0000000</code> (defined as PPB\_BASE in SDK). | Offset | Name | Info | |--------|------------|-------------------------------------| | 0xe010 | SYST_CSR | SysTick Control and Status Register | | 0xe014 | SYST_RVR | SysTick Reload Value Register | | 0xe018 | SYST_CVR | SysTick Current Value Register | | 0xe01c | SYST_CALIB | SysTick Calibration Value Register | | | | | - decrements the value of SYST\_CVR every µ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 | Туре | 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. Selects the SysTick timer clock source: 0 = External reference clock. 1 = Processor clock. | RW | 0x0 | | 1 | TICKINT | Enables SysTick exception request: 0 = Counting down to zero does not assert the SysTick exception request. 1 = Counting down to zero to asserts the SysTick exception request. | RW | 0x0 | | 0 | ENABLE | Enable SysTick counter: 0 = Counter disabled. 1 = Counter enabled. | RW | 0x0 | $$f = rac{1}{SVST_- RVR} * 1,000,000 [Hz]_{SI}$$ ### SysTick #### ARM Cortex-M peripheral The ARM Cortex-M0+ registers start at a base address of @xe0000000 (defined as PPB\_BASE in SDK). | Offset | Name | Info | |--------|------------|-------------------------------------| | 0xe010 | SYST_CSR | SysTick Control and Status Register | | 0xe014 | SYST_RVR | SysTick Reload Value Register | | 0xe018 | SYST_CVR | SysTick Current Value Register | | 0xe01c | SYST_CALIB | SysTick Calibration Value Register | ``` const SYST_RVR: *mut u32 = 0xe000_e014 as *mut u32; const SYST_CVR: *mut u32 = 0xe000_e018 as *mut u32; const SYST_CSR: *mut u32 = 0xe000_e010 as *mut u32; // fire systick every 5 seconds let interval: u32 = 5_000_000; unsafe { write_volatile(SYST_RVR, interval); write_volatile(SYST_CVR, 0); // set fields `ENABLE` and `TICKINT` write_volatile(SYST_CSR, 0b11); } ``` #### SYST\_CSR register | Bits | Name | Description | Туре | 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. Selects the SysTick timer clock source: 0 = External reference clock. 1 = Processor clock. | RW | 0x0 | | 1 | TICKINT | Enables SysTick exception request: 0 = Counting down to zero does not assert the SysTick exception request. 1 = Counting down to zero to asserts the SysTick exception request. | RW | 0x0 | | 0 | ENABLE | Enable SysTick counter: 0 = Counter disabled. 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 | | |-----------|----------------------------------------------------------------------|--| | value | the current value of the counter | | | direction | set to count UP or DOWN | | | reset | UP: max value before 0 DOWN: value after 0 | | | alarm_x | <pre>when value == alarm_x , triggers an interrupt, x in 1 n</pre> | | #### RP2350's Timers two timers, TIMERO 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$ 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 #### RP2350's Timer instance read the number of elapsed $\mu s$ since reset #### Reading the time elapsed since restart ``` const TIMERLR: *const u32 = 0x400b_000c; const TIMERHR: *const u32 = 0x400b_0008; let time: u64 = unsafe { let low = read_volatile(TIMERLR); let high = read_volatile(TIMERHR); high as u64 << 32 | low }</pre> ``` The **reading order maters** as reading TIMELR latches the value in TIMEHR (stops being updated) until TIMEHR is read. Works only in **single core**. The TIMERO and TIMERI registers start at base addresses of 0x400b0000 and 0x400b8000 respectively (defined as TIMERO\_BASE and TIMER1\_BASE in SDK). | Offset | Name | Info | |--------|----------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 0x00 | TIMEHW | Write to bits 63:32 of time always write timelw before timehw | | 0x04 | TIMELW | Write to bits 31:0 of time writes do not get copied to time until timehw is written | | 80x0 | TIMEHR | Read from bits 63:32 of time always read timelr before timehr | | 0x0c | TIMELR | Read from bits 31:0 of time | | 0x10 | ALARMO | Arm alarm 0, and configure the time it will fire. Once armed, the alarm fires when TIMER_ALARM0 == TIMELR. 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. Once armed, the alarm fires when TIMER_ALARM1 == TIMELR. 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. Once armed, the alarm fires when TIMER_ALARM2 == TIMELR. 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. Once armed, the alarm fires when TIMER_ALARM3 == TIMELR. The alarm will disarm itself once it fires, and can be disarmed early using the ARMED status register. | | 0x20 | ARMED | Indicates the armed/disarmed status of each alarm. A write to the corresponding ALARMx register arms the alarm. Alarms automatically disarm upon firing, but writing ones here will disarm immediately without waiting to fire. | | 0x24 | TIMERAWH | Raw read from bits 63:32 of time (no side effects) | | 0x28 | TIMERAWL | Raw read from bits 31:0 of time (no side effects) | | 0x2c | DBGPAUSE | Set bits high to enable pause when the corresponding debug ports are active | | 0x30 | PAUSE | Set high to pause the timer | | 0x34 | LOCKED | Set locked bit to disable write access to timer Once set, cannot be cleared (without a reset) | #### Alarm triggering an interrupt at an interval ``` #[interrupt] unsafe fn TIMERO_IRQ_0() { /* alarm fired */ } const TIMERLR: *const u32 = 0x400b 000c; const ALARMO: *mut u32 = 0x400b 0010; // + 0x2000 is bitwise set const INTE SET: *mut u32 = 0x400b 0040; // set an alarm after 3 seconds let us = 3 0000 0000; 8 unsafe { let time = read volatile(TIMERLR); 10 // use `wrapping add` as overflowing may panic 11 12 write volatile(ALARM0, time.wrapping add(us)); write volatile(INTE SET, 1 << 0);</pre> 13 14 }; ``` - the alarm can be set only for the lower 32 bits - maximum 72 minutes (use RTC for longer alarms) The TIMERO and TIMER1 registers start at base addresses of 0x400b0000 and 0x400b8000 respectively (defined as TIMERO\_BASE and TIMER1\_BASE in SDK). | Offset | Name | Info | |--------|--------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 0x00 | TIMEHW | Write to bits 63:32 of time always write timelw before timehw | | 0x04 | TIMELW | Write to bits 31:0 of time writes do not get copied to time until timehw is written | | 0x08 | TIMEHR | Read from bits 63:32 of time always read timelr before timehr | | 0x0c | TIMELR | Read from bits 31:0 of time | | 0x10 | ALARMO | Arm alarm 0, and configure the time it will fire. Once armed, the alarm fires when TIMER_ALARM0 == TIMELR. 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. Once armed, the alarm fires when TIMER_ALARM1 == TIMELR. 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. Once armed, the alarm fires when TIMER_ALARM2 == TIMELR. 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. Once armed, the alarm fires when TIMER_ALARM3 == TIMELR. 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 clk_sys cycles instead. | | 0x3c | INTR | Raw Interrupts | | 0x40 | INTE | Interrupt Enable | | 0x44 | INTF | Interrupt Force | | 0x48 | INTS | Interrupt status after masking & forcing | | | - | | # Signals Digital Signals - Recap ### 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 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 deviations that risk to change the logic state if they underpass (for high) or overpass (for low) the acceptable 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= rac{1}{period}\left[ rac{1}{s}=1Hz ight]_{SI}$$ $$duty\_cycle = rac{time\_on}{period}\%$$ #### **PWM** generic device $$f = egin{cases} rac{f_{clock}}{divider imes (top+1)} & correction = 0 \ & & \ rac{f_{clock}}{divider imes 2 imes (top+1)} & correction = 1 \end{cases}$$ $$pin_{a,b} = egin{cases} 0 & compare_{a,b} >= value \ 1 & compare_{a,b} < value \end{cases}$$ PIN OUTPUT A PIN OUTPUT N ### 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[1] 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 ) #### Registers The PWM registers start at a base address of 0x400a8000 (defined as PWM\_BASE in the SDK) | Offset | Name | Info | |--------|---------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 0x000 | CH0_CSR | Control and status register | | 0x004 | CH0_DIV | INT and FRAC form a fixed-point fractional number. Counting rate is system clock frequency divided by this number. Fractional division uses simple 1st-order sigma-delta. | | 0x008 | CH0_CTR | Direct access to the PWM counter | | 0x00c | CH0_CC | Counter compare values | | 0x010 | CH0_TOP | Counter wrap value | 1. Depends on the RP2350 package ← #### RP2350's PWM Modes #### standard mode #### phase-correct mode $$period = (TOP + 1) imes (PH\_CORRECT + 1) imes \left(DIV\_INT + rac{DIV\_FRAC}{16} ight)[s]_{SI}$$ $f = rac{f_{sys}}{period}[Hz]_{SI}$ ### Example using Embassy ``` use embassy rp::pwm::{Config, Pwm}; let p = embassy rp::init(Default::default()); let mut c: Config = Default::default(); c.top = 0 \times 8000; c.compare b = 8; 8 let mut pwm = Pwm::new_output_b( 9 p.PWM CH4, 10 11 p.PIN 25, 12 c.clone() 13 ); 14 15 loop { info!("LED duty cycle: {}/32768", c.compare b); 16 Timer::after secs(1).await; 17 c.compare b += 10; 18 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 out pub enable: bool, /// A fractional clock divider, represented as a fixed-po /// 8 integer bits and 4 fractional bits. It allows preci /// the PWM output frequency by gating the PWM counter in /// A higher value will result in a slower output frequen pub divider: fixed::FixedU16<fixed::types::extra::U4>, /// The output on channel A goes high when `compare a` is /// counter. A compare of 0 will produce an always low ou pub compare a: u16, /// The output on channel B goes high when `compare b` is counter. pub compare b: u16, /// The point at which the counter wraps, representing th /// period. The counter will either wrap to 0 or reverse /// setting of `phase correct`. pub top: u16, ``` ## **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<br>rate | Hz | the frequency at<br>which a new sample<br>is read | |------------------|------|--------------------------------------------------------| | resolution | bits | the number of bits<br>used to store a<br>sampled value | Lower sample rates yield the *aliasing effect*. $sampling_f > 2 imes max_f$ The **sampling frequency** has to be at least **two times higher** than the **maximum frequency** of the signal to avoid frequency aliasing [1]. 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 $_{n-1}$ is 1 - repeats for $bit_{n-2}$ , $bit_{n-3}$ ... $bit_0$ 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[1:1] is connected to the internal temperature sensor $$t = 27 - rac{(V_{input\_4} - 0.706)}{0.001721} [\degree C]_{SI}$$ #### in Embassy ``` use embassy rp::adc::{Adc, Channel, Config, InterruptHandler}; 2 bind interrupts!(struct Irgs { ADC IRQ FIFO => InterruptHandler; }); 6 let p = embassy rp::init(Default::default()); let mut adc = Adc::new(p.ADC, Irgs, 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); let voltage = 3300 * level / 4095; 15 16 info!("Pin 26 voltage: {}.{}V", voltage / 1000, voltage % 1000); Timer::after secs(1).await; 17 18 } ``` #### Conclusion #### we talked about - Exceptions and Interrupts - Counters - SysTick - Timers and Alarms - PWM - Analog and Digital - ADC