

### EECS 373 Design of Microprocessor-Based Systems

"PLEASE FEEL FREE TO INTERRUPT

IF YOU HAVE A QUESTION."

### Thomas Schmid University of Michigan

### Lecture 7: Interrupts, ARM NVIC September 28, 2010





### Minute Quiz...

### **Recap of the last lecture**



- Why is Reset Vector +1?
  - It's an ARM specific thing. The least significant bit in jump instructions indicates the type of instruction at that location (0: for ARM, 1: for Thumb). Since the Cortex-M3 can only execute Thumb2, this will always

### The SAT instruction





### Saturating at 32-bit signed value to a 16-bit



SSAT.W <Rd>, #<immed>, <Rn>, {,<shift>}
SSAT.W R1, #16, R0

| Input (R0) | Output (R1)    | Q Bit     |  |
|------------|----------------|-----------|--|
| 0x00020000 | 0x00007FFF     | Set       |  |
| 0x00008000 | 0x00007FFF Set |           |  |
| 0x00007FFF | 0x00007FFF     | Unchanged |  |
| 0x0000000  | 0x0000000      | Unchanged |  |
| 0xFFFF8000 | 0xFFFF8000     | Unchanged |  |
| 0xFFFF8001 | 0xFFFF8000     | Set       |  |
| 0xFFFE0000 | 0xFFFF8000     | Set       |  |

From: The Definitive Guide to the ARM Cortex-M3



### Interrupts

### **Generalization of Interrupts**

MICHIGAN

- Merriam-Webster: "to break the uniformity or continuity of"
- Informs a program of some external events
- Breaks execution flow
- Where do interrupts come from?
- How do we save state for later continuation?
- How can we ignore interrupts?
- How can we prioritize interrupts?
- How can we share interrupts?



### How does an embedded system boot?

### **The Reset Interrupt**



### 1. No power

### 2. System is held in RESET as long as VCC15 < 0.8V

- a. In reset: registers forced to default
- b. RC-Osc begins to oscillate
- c. MSS\_CCC drives RC-Osc/4 into FSCK
- d. PORESET\_N is held low
- 3. Once VCC15GOOD, PORESET\_N goes high
  - a. MSS reads from eNVM address 0x0 and 0x4



### The Reset Interrupt (2)





• The Reset Interrupt is Non-Maskable!





- On the Cortex-M3
  - Source: Software, Peripheral
  - Controller: Nested Vectored Interrupt Controller (NVIC)
  - MPU: Cortex-M3 Core

### **Sources of Interrupts**





### **Types of Interrupts**



- Physical interrupts
  - Level-triggered
  - Edge-triggered (positive, negative)
  - Hybrid
    - Look for edges, but signal must stay for a while
    - Often used for non-maskable interrupts to avoid glitches
- Non-maskable interrupts
- Interrupt priorities
- Software interrupts

## The Nested Vectored Interrupt Controller (NVIC) on the Cortex-M3



- Control registers are memory mapped
- Contains control logic for interrupt processing
- Also contains MPU, SYSTICK Timer, and Debug
- 15 internal interrupts (defined by ARM)
- Supports up to 240 external interrupts (vendor specific)
- Accessed at 0xE000E000 on any Cortex-M3!
- Register definitions can be found at:
  - ARM Cortex-M3 Technical Reference Manual v2.1, Chapter 6
  - The Definitive Guide to the ARM Cortex-M3

### System Exceptions NVIC Interrupts 1-15



| Exception | Exception Type  | Priority     | Description                                                                                                                                                                                                    |
|-----------|-----------------|--------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Number    |                 |              |                                                                                                                                                                                                                |
| 1         | Reset           | −3 (Highest) | Reset                                                                                                                                                                                                          |
| 2         | NMI             | -2           | Nonmaskable interrupt (external NMI input)                                                                                                                                                                     |
| 3         | Hard Fault      | -1           | All fault conditions, if the corresponding fault handler is not enabled                                                                                                                                        |
| 4         | MemManage Fault | Programmable | Memory management fault; MPU violation or access to illegal locations                                                                                                                                          |
| 5         | Bus Fault       | Programmable | Bus error; occurs when AHB interface receives an<br>error response from a bus slave (also called <i>prefetch</i><br><i>abort</i> if it is an instruction fetch or <i>data abort</i> if it is a<br>data access) |
| 6         | Usage Fault     | Programmable | Exceptions due to program error or trying to access<br>coprocessor (the Cortex-M3 does not support a<br>coprocessor)                                                                                           |
| 7–10      | Reserved        | NA           | _                                                                                                                                                                                                              |
| 11        | SVCall          | Programmable | System Service call                                                                                                                                                                                            |
| 12        | Debug Monitor   | Programmable | Debug monitor (breakpoints, watchpoints, or external debug requests)                                                                                                                                           |
| 13        | Reserved        | NA           | _                                                                                                                                                                                                              |
| 14        | PendSV          | Programmable | Pendable request for system device                                                                                                                                                                             |
| 15        | SYSTICK         | Programmable | System Tick Timer                                                                                                                                                                                              |

From: The Definitive Guide to the ARM Cortex-M3

### **Actel SmartFusion Interrupts**



#### Table 1-5 • SmartFusion Interrupt Sources

| Cortex-M3 NVIC Input | IRQ Label          | IRQ Source       |
|----------------------|--------------------|------------------|
| NMI                  | WDOGTIMEOUT_IRQ    | WATCHDOG         |
| INTISR[0]            | WDOGWAKEUP_IRQ     | WATCHDOG         |
| INTISR[1]            | BROWNOUT1_5V_IRQ   | VR/PSM           |
| INTISR[2]            | BROWNOUT3_3V_IRQ   | VR/PSM           |
| INTISR[3]            | RTCMATCHEVENT_IRQ  | RTC              |
| INTISR[4]            | PU_N_IRQ           | RTC              |
| INTISR[5]            | EMAC_IRQ           | Ethernet MAC     |
| INTISR[6]            | M3_IAP_IRQ         | IAP              |
| INTISR[7]            | ENVM_0_IRQ         | ENVM Controller  |
| INTISR[8]            | ENVM_1_IRQ         | ENVM Controller  |
| INTISR[9]            | DMA_IRQ            | Peripheral DMA   |
| INTISR[10]           | UART_0_IRQ         | UART_0           |
| INTISR[11]           | UART_1_IRQ         | UART_1           |
| INTISR[12]           | SPI_0_IRQ          | SPI_0            |
| INTISR[13]           | SPI_1_IRQ          | SPI_1            |
| INTISR[14]           | I2C_0_IRQ          | I2C_0            |
| INTISR[15]           | I2C_0_SMBALERT_IRQ | I2C_0            |
| INTISR[16]           | I2C_0_SMBSUS_IRQ   | I2C_0            |
| INTISR[17]           | I2C_1_IRQ          | I2C_1            |
| INTISR[18]           | I2C_1_SMBALERT_IRQ | I2C_1            |
| INTISR[19]           | I2C_1_SMBSUS_IRQ   | I2C_1            |
| INTISR[20]           | TIMER_1_IRQ        | TIMER            |
| INTISR[21]           | TIMER_2_IRQ        | TIMER            |
| INTISR[22]           | PLLLOCK_IRQ        | MSS_CCC          |
| INTISR[23]           | PLLLOCKLOST_IRQ    | MSS_CCC          |
| INTISR[24]           | ABM_ERROR_IRQ      | AHB BUS MATRIX   |
| INTISR[25]           | Reserved           | Reserved         |
| INTISR[26]           | Reserved           | Reserved         |
| INTISR[27]           | Reserved           | Reserved         |
| INTISR[28]           | Reserved           | Reserved         |
| INTISR[29]           | Reserved           | Reserved         |
| INTISR[30]           | Reserved           | Reserved         |
| INTISR[31]           | FAB_IRQ            | FABRIC INTERFACE |
| INTISR[32]           | GPIO_0_IRQ         | GPIO             |
| INTISR[33]           | GPIO_1_IRQ         | GPIO             |
| INTISR[34]           | GPIO_2_IRQ         | GPIO             |

| -          |                        |     |
|------------|------------------------|-----|
| INTISR[64] | ACE_PC0_FLAG0_IRQ      | ACE |
| INTISR[65] | ACE_PC0_FLAG1_IRQ      | ACE |
| INTISR[66] | ACE_PC0_FLAG2_IRQ      | ACE |
| INTISR[67] | ACE_PC0_FLAG3_IRQ      | ACE |
| INTISR[68] | ACE_PC1_FLAG0_IRQ      | ACE |
| INTISR[69] | ACE_PC1_FLAG1_IRQ      | ACE |
| INTISR[70] | ACE_PC1_FLAG2_IRQ      | ACE |
| INTISR[71] | ACE_PC1_FLAG3_IRQ      | ACE |
| INTISR[72] | ACE_PC2_FLAG0_IRQ      | ACE |
| INTISR[73] | ACE_PC2_FLAG1_IRQ      | ACE |
| INTISR[74] | ACE_PC2_FLAG2_IRQ      | ACE |
| INTISR[75] | ACE_PC2_FLAG3_IRQ      | ACE |
| INTISR[76] | ACE_ADC0_DATAVALID_IRQ | ACE |
| INTISR[77] | ACE_ADC1_DATAVALID_IRQ | ACE |
| INTISR[78] | ACE_ADC2_DATAVALID_IRQ | ACE |
| INTISR[79] | ACE_ADC0_CALDONE_IRQ   | ACE |
| INTISR[80] | ACE_ADC1_CALDONE_IRQ   | ACE |
| INTISR[81] | ACE_ADC2_CALDONE_IRQ   | ACE |
| INTISR[82] | ACE_ADC0_CALSTART_IRQ  | ACE |
| INTISR[83] | ACE_ADC1_CALSTART_IRQ  | ACE |
| INTISR[84] | ACE_ADC2_CALSTART_IRQ  | ACE |
| INTISR[85] | ACE_COMP0_FALL_IRQ     | ACE |
| INTISR[86] | ACE_COMP1_FALL_IRQ     | ACE |
| INTISR[87] | ACE_COMP2_FALL_IRQ     | ACE |
| INTISR[88] | ACE_COMP3_FALL_IRQ     | ACE |
| INTISR[89] | ACE_COMP4_FALL_IRQ     | ACE |
| INTISR[90] | ACE_COMP5_FALL_IRQ     | ACE |
| INTISR[91] | ACE_COMP6_FALL_IRQ     | ACE |
| INTISR[92] | ACE_COMP7_FALL_IRQ     | ACE |
| INTISR[93] | ACE_COMP8_FALL_IRQ     | ACE |
| INTISR[94] | ACE_COMP9_FALL_IRQ     | ACE |
| INTISR[95] | ACE_COMP10_FALL_IRQ    | ACE |

#### 54 more ACE specific interrupts

#### GPIO\_3\_IRQ to GPIO\_31\_IRQ cut

### **Pending Interrupts**





### Pending Interrupts (2)





 Software clears pending status while PRIMASK/ FAULTMASK is 1

From: The Definitive Guide to the ARM Cortex-M3

### Active Status set during handler execution





### **Interrupt Request not Cleared**



Interrupt request stays active



### **Multiple Interrupt Pulses**





Interrupt \_\_\_\_\_ Active Status



Processor Mode

### **New Interrupt Request after Pending Cleared**





### **Configuring the NVIC**



### Interrupt Set Enable and Clear Enable 0xE000E100-0xE000E11C, 0xE000E180-0xE000E19C

| 0xE000E100 | SETENA0 | R/W | 0 | Enable for external interrupt #0–31                |
|------------|---------|-----|---|----------------------------------------------------|
|            |         |     |   | <pre>bit[0] for interrupt #0 (exception #16)</pre> |
|            |         |     |   | bit[1] for interrupt #1 (exception #17)            |
|            |         |     |   | •••                                                |
|            |         |     |   | bit[31] for interrupt #31 (exception #47)          |
|            |         |     |   | Write 1 to set bit to 1; write 0 has no effect     |
|            |         |     |   | Read value indicates the current status            |
| 0xE000E180 | CLRENA0 | R/W | 0 | Clear enable for external interrupt #0–31          |
|            |         |     |   | bit[0] for interrupt #0                            |
|            |         |     |   | bit[1] for interrupt #1                            |
|            |         |     |   |                                                    |
|            |         |     |   | bit[31] for interrupt #31                          |
|            |         |     |   | Write 1 to clear bit to 0; write 0 has no effect   |
|            |         |     |   | Read value indicates the current enable status     |

### Configuring the NVIC (2)



### Set Pending & Clear Pending 0xE000E200-0xE000E21C, 0xE000E280-0xE000E29C

| 0xE000E200 | SETPEND0 | R/W | 0 | Pending for external interrupt #0-31             |
|------------|----------|-----|---|--------------------------------------------------|
|            |          |     |   | bit[0] for interrupt #0 (exception #16)          |
|            |          |     |   | bit[1] for interrupt #1 (exception #17)          |
|            |          |     |   |                                                  |
|            |          |     |   | bit[31] for interrupt #31 (exception #47)        |
|            |          |     |   | Write 1 to set bit to 1; write 0 has no effect   |
|            |          |     |   | Read value indicates the current status          |
|            |          | 1   | 1 |                                                  |
| 0xE000E280 | CLRPEND0 | R/W | 0 | Clear pending for external interrupt #0–31       |
|            |          |     |   | bit[0] for interrupt #0 (exception #16)          |
|            |          |     |   | bit[1] for interrupt #1 (exception #17)          |
|            |          |     |   |                                                  |
|            |          |     |   | bit[31] for interrupt #31 (exception #47)        |
|            |          |     |   | Write 1 to clear bit to 0; write 0 has no effect |
|            |          |     |   | Read value indicates the current pending status  |

### Configuring the NVIC (3)



Interrupt Active Status Register
 - 0xE000E300-0xE000E31C

| Address    | Name    | Туре | Reset Value                                 | Description                                 |
|------------|---------|------|---------------------------------------------|---------------------------------------------|
| 0xE000E300 | ACTIVE0 | R    | 0 Active status for external interrupt #0-3 |                                             |
|            |         |      |                                             | bit[0] for interrupt #0                     |
|            |         |      |                                             | bit[1] for interrupt #1                     |
|            |         |      |                                             | •••                                         |
|            |         |      |                                             | bit[31] for interrupt #31                   |
| 0xE000E304 | ACTIVE1 | R    | 0                                           | Active status for external interrupt #32–63 |
|            | _       | _    | _                                           | _                                           |

### **Interrupt Priority**



- What do we do if several interrupts arrive at the same time?
- NVIC allows to set priorities for (almost) every interrupt
- 3 fixed highest priorities, up to 256 programmable priorities
  - 128 preemption levels
  - Not all priorities have to be implemented by a vendor!

| Bit 7       | Bit 6 | Bit 5                         | Bit 4 | Bit 3 | Bit 2 | Bit 1 | Bit 0 |
|-------------|-------|-------------------------------|-------|-------|-------|-------|-------|
| Implemented |       | Not implemented, read as zero |       |       |       |       |       |

- SmartFusion has 32 priority levels, i.e., 0x00, 0x08, ..., 0xF8
- Higher priority interrupts can pre-empt lower priorities
- Priority can be sub-divided into priority groups
  - splits priority register into two halves, preempt priority and subpriority
  - preempt priority: indicates if an interrupt can preempt another
  - subpriority: used if two interrupts of same group arrive concurrently

### **Interrupt Priority (2)**



### • Interrupt Priority Level Registers

- 0xE000E400-0xE000E4EF

| Name   | Туре                | Reset Value                                                         | Description                                                                                                       |
|--------|---------------------|---------------------------------------------------------------------|-------------------------------------------------------------------------------------------------------------------|
| PRI_0  | R/W                 | 0 (8-bit)                                                           | Priority-level external interrupt #0                                                                              |
| PRI_1  | R/W                 | 0 (8-bit)                                                           | Priority-level external interrupt #1                                                                              |
| -      | _                   | _                                                                   | _                                                                                                                 |
| PRI_31 | R/W                 | 0 (8-bit)                                                           | Priority-level external interrupt #31                                                                             |
| -      | _                   | _                                                                   | -                                                                                                                 |
|        | PRI_0<br>PRI_1<br>- | PRI_0         R/W           PRI_1         R/W           -         - | PRI_0         R/W         0 (8-bit)           PRI_1         R/W         0 (8-bit)           -         -         - |

### **Preemption Priority and Subpriority**



| Priority Group | Preempt Priority Field | Subpriority Field |
|----------------|------------------------|-------------------|
| 0              | Bit [7:1]              | Bit [0]           |
| 1              | Bit [7:2]              | Bit [1:0]         |
| 2              | Bit [7:3]              | Bit [2:0]         |
| 3              | Bit [7:4]              | Bit [3:0]         |
| 4              | Bit [7:5]              | Bit [4:0]         |
| 5              | Bit [7:6]              | Bit [5:0]         |
| 6              | Bit [7]                | Bit [6:0]         |
| 7              | None                   | Bit [7:0]         |

#### Application Interrupt and Reset Control Register (Address 0xE000ED0C)

| Bits  | Name          | Туре | Reset<br>Value | Description                                                                                                                                                                 |
|-------|---------------|------|----------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 31:16 | VECTKEY       | R/W  | -              | Access key; 0x05FA must be written to this field to write<br>to this register, otherwise the write will be ignored; the<br>read-back value of the upper half word is 0xFA05 |
| 15    | ENDIANNESS    | R    | _              | Indicates endianness for data: 1 for big endian (BE8)<br>and 0 for little endian; this can only change after a reset                                                        |
| 10:8  | PRIGROUP      | R/W  | 0              | Priority group                                                                                                                                                              |
| 2     | SYSRESETREQ   | W    | _              | Requests chip control logic to generate a reset                                                                                                                             |
| 1     | VECTCLRACTIVE | W    | _              | Clears all active state information for exceptions;<br>typically used in debug or OS to allow system to recover<br>from system error (Reset is safer)                       |
| 0     | VECTRESET     | W    | -              | Resets the Cortex-M3 processor (except debug logic),<br>but this will not reset circuits outside the processor                                                              |

# Exercise: How many preemption priorities and subpriority levels do we get on the Smart Fusion if we set Priority Group to 5?



| -3<br>-2<br>-1 | Reset<br>NMI<br>Hard Fault | Preempt levels<br>with priority<br>group set to 5 | Subpriority<br>levels |
|----------------|----------------------------|---------------------------------------------------|-----------------------|
| 0x00           | Programmable<br>Exceptions |                                                   |                       |

### PRIMASK, FAULTMASK, and BASEPRI



- What if we quickly want to disable all interrupts?
- Write 1 into PRIMASK to disable all interrupt except NMI
  - MOV R0, #1
  - MSR PRIMASK, RO
- Write 0 into PRIMASK to enable all interrupts
- FAULTMASK is the same as PRIMASK, but also blocks hard fault (priority -1)
- What if we want to disable all interrupts below a certain priority?
- Write priority into BASEPRI
  - MOV R0, #0x60
  - MSR BASEPRI, RO



### What exactly is an interrupt handler?

### Vector Table



- Upon an interrupt, the Cortex-M3 needs to know the address of the interrupt handler (function pointer)
- After powerup, vector table is located at 0x0000000

| Address   | Exception Number | Value (Word Size)                            |
|-----------|------------------|----------------------------------------------|
| 0x0000000 | _                | MSP initial value                            |
| 0x0000004 | 1                | Reset vector (program counter initial value) |
| 0x0000008 | 2                | NMI handler starting address                 |
| 0x000000C | 3                | Hard fault handler starting address          |
| •••       | •••              | Other handler starting address               |

• Can be relocated to change interrupt handlers at runtime (vector table offset register)

### Vector Table in SoftConsole



### Located in startup\_a2fxxxm3.s

| 22  |          |                    |
|-----|----------|--------------------|
| 23g | pfnVecto | rs <mark>:</mark>  |
| 24  | .word    | estack             |
| 25  | .word    | Reset_Handler      |
| 26  | .word    | NMI_Handler        |
| 27  | .word    | HardFault_Handler  |
| 28  | .word    | MemManage_Handler  |
| 29  | .word    | BusFault_Handler   |
| 30  | .word    | UsageFault_Handler |
| 31  | .word    | 0                  |
| 32  | .word    | 0                  |
| 00  |          | 2                  |

| 15/* |             |                             |
|------|-------------|-----------------------------|
| 16 * | Vector tabl | e                           |
| 17 * | /           |                             |
| 18   | .global     | g_pfnVectors                |
| 19   | .section    | .isr_vector, "a", %progbits |
| 20   | .type       | g_pfnVectors, %object       |
| 21   | .size       | g_pfnVectors,g_pfnVectors   |
| 22   |             |                             |

### Put at 0x0000000 in linker script

```
42
    SECTIONS
43
   {
44
     .text :
45
     {
     CREATE OBJECT SYMBOLS
46
       text load = LOADADDR(.text);
47
48
       text start = .;
       *(.isr_vector)
49
```

### **Interrupt Handlers**



```
281 * NMI Handler
282 */
   .weak NMI Handler
283
284
   .type NMI Handler, %function
285NMI Handler:
286
    в.
287
289 * HardFault Handler
290 */
291
    .weak HardFault Handler
    .type HardFault Handler, %function
292
293HardFault Handler:
294
    в.
295
```

### Interrupt Handler in GNU C



• We can overwrite the predefined interrupt handlers

```
attribute__((__interrupt__)) void Timer1_IRQHandler()
{
    MSS_TIM1_disable_irq();
    MSS_TIM1_clear_irq();
    ...
    NVIC_ClearPendingIRQ( Timer1_IRQn );
}
```

```
int main()
{
    MSS_TIM1_enable_irq();
    NVIC_EnableIRQ( Timer1_IRQn );
    ...
    while(1){}
}
```

### **Interrupt Service Routines**



- 1. Automatic saving of registers upon exception
  - PC, PSR, R0-R3, R12, LR pushed on the stack
- 2. While bus busy, fetch exception vector
- 3. Update SP to new location
- 4. Update IPSR (low part of PSR) with new exception number
- 5. Set PC to vector handler
- 6. Update LR to special value EXC\_RETURN
- Several other NVIC registers get updated
- Latency: as short as 12 cycles

### **Return from ISR**



### • 3 ways to return from an ISR

| Return Instruction         | Description                                                                                                                                                                                                                                                                               |
|----------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| BX <reg></reg>             | If the EXC_RETURN value is still in LR, we can use the <i>BX LR</i> instruction to perform the interrupt return.                                                                                                                                                                          |
| POP {PC}, or<br>POP {, PC} | Very often the value of LR is pushed to the stack after entering the exception<br>handler. We can use the POP instruction, either a single POP or multiple POPs, to<br>put the EXC_RETURN value to the program counter. This will cause the processor<br>to perform the interrupt return. |
| LDR, or LDM                | It is possible to produce an interrupt return using the LDR instruction with PC as the destination register.                                                                                                                                                                              |

- Unstack and reset SP
- Update NVIC registers

### **Nested Interrupts**



- Built into the Cortex-M3 (not every MCU has this)
- Make sure main stack is large enough!
- Two methods:
  - Tail Chaining
  - Late Arrival (preemption)

### Tail Chaining





- If first interrupt has same or higher priority
- Skip stacking/unstacking for efficiency

### Late Arrival (Preemption)





 Main stack must be able to hold maximum number of preemptions!

### **Different Concepts of Interrupt Sharing**



- Number of potential interrupts usually larger than interrupt lines availability on Core
- One peripheral often only has one interrupt
- Different types of events are stored in a status register

|                                      | I    | <u>.                                    </u> |   |                                                                                                                                                             |                                                                                                                                                                                                                                                                                                           |
|--------------------------------------|------|----------------------------------------------|---|-------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| • Example, UART<br>- IIR, 0x40000008 | bits | identification                               | R | 0b0001                                                                                                                                                      | 0b0110 = Highest priority. Receiver line status interrupt due<br>to overrun error, parity error, framing error or break<br>interrupt. Reading the Line Status Register resets this<br>interrupt.                                                                                                          |
|                                      |      |                                              |   |                                                                                                                                                             | 0b0100 = Second priority. Receive data available interrupt<br>modem status interrupt. Reading the Receiver Buffer<br>Register (RBR) or the FIFO drops below the trigger level<br>resets this interrupt.                                                                                                   |
|                                      |      |                                              |   |                                                                                                                                                             | 0b1100 = Second priority. Character timeout indication<br>interrupt occurs when no characters have been read from the<br>RX FIFO during the last four character times and there was at<br>least one character in it during this time. Reading the Receive<br>Buffer Register (RBR) resets this interrupt. |
|                                      |      |                                              |   | 0b0010 = Third priority. Transmit Holding Register Empty interrupt. Reading the IIR or writing to the Transmit Holding Register (THR) resets the interrupt. |                                                                                                                                                                                                                                                                                                           |
|                                      |      |                                              |   |                                                                                                                                                             | 0b0000 = Fourth priority. Modem status interrupt due to<br>Clear to Send, Data Set Ready, Ring Indicator, or Data Carrier<br>Detect being asserted. Reading the Modem Status Register<br>resets this interrupt.                                                                                           |
|                                      |      |                                              |   |                                                                                                                                                             | This register is read only; writing has no effect. Also see Table 15-9.                                                                                                                                                                                                                                   |

### ISR Sharing, i.e., Callbacks in C



- There is only one interrupt handler
- Functions have to "subscribe" for events
- Callbacks

}

- Driver provides function to register a function pointer
- Driver stores function pointers in list
- Upon interrupt, each registered function gets called

### **Common Problems and Pit-Falls**



- Too many interrupts
  - Your core can't keep up with handling interrupts
- Concurrency issues
  - One interrupt handler modifies global variables
  - Can be avoided using atomic sections protected through PRIMASK
- Lost interrupts
  - It can happen that an interrupt doesn't get treated by the Core
  - State machine and peripheral has to be aware of this possibility
  - Danger for deadlocks

### Summary



- Overwrite default Interrupt Handler
- Initialization
  - Enable interrupt in NVIC
  - Enable interrupt in Peripheral
- Upon Interrupt
  - Clear interrupt in Peripheral
  - Clear pending bit in NVIC
  - Potentially disable interrupts temporarely