# Interrupts

Lecture 18

Josh Brake Harvey Mudd College

### Learning Objectives

By the end of this lecture you will be able to:

- Explain the basic exception model used on ARM Cortex-M4 processors
- Configure interrupts to quickly respond to information from on-board peripherals like GPIO pins and timers

#### Outline

- Interrupts and Exceptions
  - The exception model in ARM Cortex-M4 processors
  - The Nested Vector Interrupt Controller (NVIC)
  - Configuring interrupts on ARM Cortex-M4
- Activity
  - Toggle LED with switch using polling and interrupts

# **ARM Cortex-M4 Exception Model**

#### **Sources of exceptions**



#### FIGURE 7.1

Various sources of exceptions in a typical microcontroller

#### **Interrupt Servicing Sequence**

- 1. Peripheral asserts interrupt request
- 2. Processor suspends currently operating task
- 3. Processor executes an Interrupt Service Routine (ISR) to service the peripheral and optionally clear the interrupt request
- 4. Processor resumes previously suspended task.

#### **Vector Table**

- Table of addresses to functions that are placed starting at address  $0 \times 0000000$ .
- Each memory location contains the address of different exception/interrupt handlers

| Position | Priority | Type of<br>priority | Acronym                 | Description                                   | Addres   |
|----------|----------|---------------------|-------------------------|-----------------------------------------------|----------|
| 11       | 18       | settable            | DMA1_CH1                | DMA1 channel 1 interrupt                      | 0x0000 0 |
| 12       | 19       | settable            | DMA1_CH2                | DMA1 channel 2 interrupt                      | 0x0000 0 |
| 13       | 20       | settable            | DMA1_CH3                | DMA1 channel 3 interrupt                      | 0x0000 0 |
| 14       | 21       | settable            | DMA1_CH4                | DMA1 channel 4 interrupt                      | 0x0000 0 |
| 15       | 22       | settable            | DMA1_CH5                | DMA1 channel 5 interrupt                      | 0x0000 0 |
| 16       | 23       | settable            | DMA1_CH6                | DMA1 channel 6 interrupt                      | 0x0000 0 |
| 17       | 24       | settable            | DMA1_CH7                | DMA1 channel 7 interrupt                      | 0x0000 0 |
| 18       | 25       | settable            | ADC1_2                  | ADC1 and ADC2 <sup>(2)</sup> global interrupt | 0x0000 0 |
| 19       | 26       | settable            | CAN1_TX <sup>(1)</sup>  | CAN1_TX interrupts                            | 0x0000 0 |
| 20       | 27       | settable            | CAN1_RX0 <sup>(1)</sup> | CAN1_RX0 interrupts                           | 0x0000 0 |
| 21       | 28       | settable            | CAN1_RX1 <sup>(1)</sup> | CAN1_RX1 interrupt                            | 0x0000 0 |
| 22       | 29       | settable            | CAN1_SCE <sup>(1)</sup> | CAN1_SCE interrupt                            | 0x0000 0 |
| 23       | 30       | settable            | EXTI9_5                 | EXTI Line[9:5] interrupts                     | 0x0000 0 |
| 24       | 31       | settable            | TIM1_BRK/TIM15          | TIM1 Break/TIM15 global interrupts            | 0x0000 0 |
| 25       | 32       | settable            | TIM1_UP/TIM16           | TIM1 Update/TIM16 global interrupts           | 0x0000 0 |
| 26       | 33       | settable            | TIM1_TRG_COM            | TIM1 trigger and commutation interrupt        | 0x0000 0 |
| 27       | 34       | settable            | TIM1_CC                 | TIM1 capture compare interrupt                | 0x0000 0 |
| 28       | 35       | settable            | TIM2                    | TIM2 global interrupt                         | 0x0000 0 |
| 29       | 36       | settable            | TIM3 <sup>(3)</sup>     | TIM3 global interrupt                         | 0x0000 0 |
| 30       | 37       | settable            | -                       | Reserved                                      | 0x0000 0 |
| 31       | 38       | settable            | I2C1_EV                 | I2C1 event interrupt                          | 0x0000 0 |
| 32       | 39       | settable            | I2C1_ER                 | I2C1 error interrupt                          | 0x0000 0 |
| 33       | 40       | settable            | I2C2_EV <sup>(4)</sup>  | I2C2 event interrupt                          | 0x0000 0 |
| 34       | 41       | settable            | I2C2_ER <sup>(4)</sup>  | I2C2 error interrupt                          | 0x0000 0 |
| 35       | 42       | settable            | SPI1                    | SPI1 global interrupt                         | 0x0000 0 |
| 36       | 43       | settable            | SPI2 <sup>(4)</sup>     | SPI2 global interrupt                         | 0x0000 0 |
| 37       | 44       | settable            | USART1                  | USART1 global interrupt                       | 0x0000 0 |
| 38       | 45       | settable            | USART2                  | USART2 global interrupt                       | 0x0000 0 |
| 39       | 46       | settable            | USART3 <sup>(4)</sup>   | USART3 global interrupt                       | 0x0000 0 |
| 40       | 47       | settable            | EXTI15_10               | EXTI Line[15:10] interrupts                   | 0x0000 0 |
| 41       | 48       | settable            | RTC_ALARM               | RTC alarms through EXTI line 18 interrupts    | 0x0000 0 |
| 42       | 49       | settable            | -                       | Reserved                                      | 0x0000 0 |
| 43       | 50       | settable            | -                       | Reserved                                      | 0x0000 0 |

#### **Nested Vector Interrupt Controller**

- On Cortex-M4 the NVIC supports up to 240 IRQs, a Non-Maskable Interrupt, a SysTick timer interrupt, and a number of system exceptions.
- When handling and IRQ, some of the registers are stored on the stack automatically and are automatically restored. This allows exception handlers to be written as normal C functions.
- Nested refers to the fact that we have different priorities and therefore can handle an interrupt with a higher priority in the middle of handling an interrupt of a lower priority.
- Vector refers to the fact that the interrupt service handlers are addresses pointing to functions.

#### **Interrupt Priorities**

- Interrupt priority levels allow us to define which interrupts can pre-empt others
- Cortex-M processors support three fixed highest-priority levels and up to 256 level of programmable priority.
  - However, the actual number of available levels is chip dependent since implementing all 256 levels can be costly in terms of power and speed.
- Three negative priorities (hard fault, NMI, and reset) can pre-empt any other exceptions

#### **Interrupt Priorities**

| Bit 7       | Bit 6 | Bit 5 | Bit 4           | Bit 3 | Bit 2 | Bit 1 | Bit 0 |  |
|-------------|-------|-------|-----------------|-------|-------|-------|-------|--|
| Implemented |       |       | Not Implemented |       |       |       |       |  |

#### FIGURE 7.2

A priority-level register with 3 bits implemented (8 programmable priority levels)

Figure 7.2 p. 236 The Definitive guide to ARM Cortex-M3 and Cortex-M4 Processors



#### FIGURE 7.4

Available priority levels with 3-bit or 4-bit priority width

Figure 7.4 p. 237 The Definitive guide to ARM Cortex-M3 and Cortex-M4 Processors

#### **Exception Definitions**

| Exception |                     | CMSIS-Core            | CMSIS-Core        |                        |
|-----------|---------------------|-----------------------|-------------------|------------------------|
| Number    | Exception Type      | Enumeration (IRQn)    | Enumeration Value | Exception Handler Name |
| 1         | Reset               | -                     | -                 | Reset_Handler          |
| 2         | NMI                 | NonMaskableInt_IRQn   | -14               | NMI_Handler            |
| 3         | Hard Fault          | HardFault_IRQn        | -13               | HardFault_Handler      |
| 4         | MemManage Fault     | MemoryManagement_IRQn | -12               | MemManage_Handler      |
| 5         | Bus Fault           | BusFault_IRQn         | -11               | BusFault_Handler       |
| 6         | Usage Fault         | UsageFault_IRQn       | -10               | UsageFault_Handler     |
| 11        | SVC                 | SVCall_IRQn           | -5                | SVC_Handler            |
| 12        | Debug Monitor       | DebugMonitor_IRQn     | _4                | DebugMon_Handler       |
| 14        | PendSV              | PendSV_IRQn           | -2                | PendSV_Handler         |
| 15        | SYSTICK             | SysTick_IRQn          | -1                | SysTick_Handler        |
| 16        | Interrupt #0        | (device-specific)     | 0                 | (device-specific)      |
| 17        | Interrupt #1 - #239 | (device-specific)     | 1 to 239          | (device-specific)      |

Table 7.3 p.234 The Definitive guide to ARM Cortex-M3 and Cortex-M4 Processors

#### **Interrupt Setup**

- 1. Enable global interrupts.
- 2. Set the priority level (optional).
- 3. Enable interrupt generation in the peripheral that triggers the interrupt.
- 4. Enable the interrupt in the NVIC.

The name of the ISR needs to match the name used in the vector table so that the linker can place the starting address of the ISR into the vector table correctly.

#### Handling an interrupt



#### FIGURE 7.14

A simple case of interrupt pending and activation behavior

### Relevant Files in Common Microcontroller Software Interface Standard (CMSIS)

core\_cm4.h - Definitions which are global to the Cortex-M4

- e.g., NVIC\_Type which specifies the NVIC registers
- Sidenote: Documentation for this is in the Cortex-M4 user manual, not in the reference manual or datasheet.

cmsis\_gcc.h - Compiler specific definitions

- e.g., the specific directive syntax necessary to force functions to be inline. void \_\_\_\_enable\_irq(void) and void \_\_\_\_disable\_irq(void) are defined in cmsis\_gcc.h
- These are compiler specific and use the cpsie i (enable) and cpsid i disable special assembly instructions.

stm321432xx.h - Device specific configurations.

• e.g., the number of NVIC priority bits

#### **NVIC Memory Location**



Figure 2. Memory map

#### **Core Registers for NVIC**

#### 6.3 NVIC programmers model

Table 6-1 shows the NVIC registers.

| Address                    | Name                       | Туре | Reset      | Description                              |
|----------------------------|----------------------------|------|------------|------------------------------------------|
| 0xE000E004                 | ICTR                       | RO   | -          | Interrupt Controller Type Register, ICTR |
| 0xE000E100 -<br>0xE000E11C | NVIC_ISER0 -<br>NVIC_ISER7 | RW   | 0x00000000 | Interrupt Set-Enable Registers           |
| 0xE000E180 -<br>0E000xE19C | NVIC_ICER0 -<br>NVIC_ICER7 | RW   | 0×00000000 | Interrupt Clear-Enable Registers         |
| 0xE000E200 -<br>0xE000E21C | NVIC_ISPR0 -<br>NVIC_ISPR7 | RW   | 0x00000000 | Interrupt Set-Pending Registers          |
| 0xE000E280 -<br>0xE000E29C | NVIC_ICPR0 -<br>NVIC_ICPR7 | RW   | 0x00000000 | Interrupt Clear-Pending Registers        |
| 0xE000E300 -<br>0xE000E31C | NVIC_IABR0 -<br>NVIC_IABR7 | RO   | 0x00000000 | Interrupt Active Bit Register            |
| 0xE000E400 -<br>0xE000E41F | NVIC_IPR0 -<br>NVIC_IPR59  | RW   | 0x00000000 | Interrupt Priority Register              |

The following sections describe the NVIC registers whose implementation is specific to this processor. Other registers are described in the *ARMv7M Architecture Reference Manual*.

#### core-cm4.h

403 /\*\* 404 \brief Structure type to access the Nested Vectored Interrupt Controller (NVIC). 405 \*/ 406 typedef struct 407 /\*!< Offset: 0x000 (R/W) Interrupt Set Enable Register \*/</pre> 408 IOM uint32\_t ISER[8U]; 409 uint32 t RESERVED0[24U]; 410 \_\_IOM uint32\_t ICER[8U]; /\*!< Offset: 0x080 (R/W) Interrupt Clear Enable Register \*/</pre> 411 uint32\_t RESERVED1[24U]; 412 \_\_IOM uint32\_t ISPR[8U]; /\*!< Offset: 0x100 (R/W) Interrupt Set Pending Register \*/</pre> 413 uint32\_t RESERVED2[24U]; 414 \_\_IOM uint32\_t ICPR[8U]; /\*!< Offset: 0x180 (R/W) Interrupt Clear Pending Register \*/</pre> 415 uint32\_t RESERVED3[24U]; 416 \_\_IOM uint32\_t IABR[8U]; /\*!< Offset: 0x200 (R/W) Interrupt Active bit Register \*/</pre> 417 uint32\_t RESERVED4[56U]; 418 IOM uint8 t IP[240U]; /\*!< Offset: 0x300 (R/W) Interrupt Priority Register (8Bit wide) \*/</pre> 419 uint32\_t RESERVED5[644U]; OM uint32 t STIR; /\*!< Offset: 0xE00 ( /W) Software Trigger Interrupt Register \*/</pre> 420 } NVIC\_Type; 421 400

## cmsis\_gcc.h

| 118 | /* ####################################                                   |
|-----|---------------------------------------------------------------------------|
| 119 | <pre>/** \ingroup CMSIS_Core_FunctionInterface</pre>                      |
| 120 | \defgroup CMSIS_Core_RegAccFunctions CMSIS Core Register Access Functions |
| 121 | @{                                                                        |
| 122 | */                                                                        |
| 123 |                                                                           |
| 124 | /**                                                                       |
| 125 | \brief Enable IRQ Interrupts                                              |
| 126 | \details Enables IRQ interrupts by clearing the I-bit in the CPSR.        |
| 127 | Can only be executed in Privileged modes.                                 |
| 128 | */                                                                        |
| 129 | STATIC_FORCEINLINE voidenable_irq(void)                                   |
| 130 | {                                                                         |
| 131 | <pre>ASM volatile ("cpsie i" : : : "memory");</pre>                       |
| 132 | }                                                                         |
| 133 |                                                                           |
| 134 |                                                                           |
| 135 | /**                                                                       |
| 136 | \brief Disable IRQ Interrupts                                             |
| 137 | \details Disables IRQ interrupts by setting the I-bit in the CPSR.        |
| 138 | Can only be executed in Privileged modes.                                 |
| 139 | */                                                                        |
| 140 | <pre>STATIC_FORCEINLINE voiddisable_irq(void)</pre>                       |
| 141 | {                                                                         |
| 142 | <pre>ASM volatile ("cpsid i" : : : "memory");</pre>                       |
| 143 | }                                                                         |

### stm32l432xx.h

| 1  | /**                         |                 |                                                                          |
|----|-----------------------------|-----------------|--------------------------------------------------------------------------|
| 2  |                             | - Number De     | finition, according to the selected device                               |
| 3  | -                           |                 |                                                                          |
|    |                             | II LGUI ALLOII. | _Section                                                                 |
| 4  | */                          |                 |                                                                          |
| 5  | typedef enum                |                 |                                                                          |
| 6  |                             |                 | N 1                                                                      |
| 7  |                             |                 | Numbers ************************************                             |
| 8  | NonMaskableInt_IRQn         | -               | /*!< 2 Cortex-M4 Non Maskable Interrupt                                  |
| 9  | —                           | -               | /*!< 3 Cortex-M4 Hard Fault Interrupt                                    |
| 10 |                             | -               | /*!< 4 Cortex-M4 Memory Management Interrupt                             |
| 11 | BusFault_IRQn               | = -11,          | /*!< 5 Cortex-M4 Bus Fault Interrupt                                     |
| 12 | UsageFault_IRQn             | = -10,          | /*!< 6 Cortex-M4 Usage Fault Interrupt                                   |
| 13 | SVCall_IRQn                 | = -5,           | /*!< 11 Cortex-M4 SV Call Interrupt                                      |
| 14 | DebugMonitor_IRQn           | = -4,           | /*!< 12 Cortex-M4 Debug Monitor Interrupt                                |
| 15 | PendSV_IRQn                 | = -2,           | /*!< 14 Cortex-M4 Pend SV Interrupt                                      |
| 16 | SysTick_IRQn                | = -1,           | /*!< 15 Cortex-M4 System Tick Interrupt                                  |
| 17 | /***** STM32 specific Inter | rupt Numbe      | rs ************************************                                  |
| 18 | WWDG_IRQn                   | = 0,            | /*!< Window WatchDog Interrupt                                           |
| 19 | PVD_PVM_IRQn                | = 1,            | <pre>/*!&lt; PVD/PVM3/PVM4 through EXTI Line detection Interrupts</pre>  |
| 20 | TAMP_STAMP_IRQn             | = 2,            | <pre>/*!&lt; Tamper and TimeStamp interrupts through the EXTI line</pre> |
| 21 | RTC_WKUP_IRQn               | = 3,            | <pre>/*!&lt; RTC Wakeup interrupt through the EXTI line</pre>            |
| 22 |                             |                 |                                                                          |
| 23 |                             |                 |                                                                          |
| 24 | <pre>} IRQn_Type;</pre>     |                 |                                                                          |
|    |                             |                 |                                                                          |

# **Interrupt Activity**

### **Activity: GPIO Pin Interrupts**

- Download the code from the course Github.
- Build and upload button\_polling.c

### button\_polling.c

```
// button polling.c
 // Josh Brake
 // jbrake@hmc.edu
 // 10/31/22
⊟/*
   This program polls the user button on the Nucleo-L432KC board and has a
    delay within the main loop to simulate the problems with polling for
   catching events.
 */
  #include "main.h"
int main(void) {
      // Enable LED as output
      gpioEnable(GPIO PORT B);
      pinMode(LED PIN, GPIO OUTPUT);
      // Enable button as input
      gpioEnable(GPIO_PORT A);
      pinMode(BUTTON PIN, GPÍÓ INPUT);
      GPIOA->PUPDR |= _VAL2FLD(GPIO_PUPDR_PUPD2, 0b01); // Set PA2 as pull-up
      // Initialize timer
      RCC->APB1ENR1 |= RCC APB1ENR1 TIM2EN;
      initTIM(DELAY TIM);
      int volatile cur button state = digitalRead(BUTTON PIN);
      int volatile led state = 0;
      int volatile prev button state = cur button state;
      while(1){
          prev button state = cur button state;
          cur_button_state = digitalRead(BUTTON_PIN);
if ((prev_button_state == 1) && (cur_button_state == 0)) {
              led state = !led state;
              digitalWrite(LED PIN, led state);
          delay_millis(DELAY_TIM, 200);
      }
 ∟}
```

#### External Interrupt/Event Controller (EXTI)



EXTI0[3:0] bits in the SYSCFG\_EXTICR1 register





#### Results

- 1. Configure EXTI controller
- 2. Define IRQ handler name
- 3. Capture trace on oscilloscope demonstrating latency between button press and LED response for three cases:
  - Polling with 200 ms delay
  - Polling with no delay (comment out delay)
  - Interrupt driven

#### Results: Interrupt vs. Polling with 200 ms Delay





#### **Results: Interrupt vs. Polling with No Delay**



