# Nested Vector Interrupt Controller (NVIC)

# Objective

This tutorial demonstrates how to use interrupts on a processor. In general, you will understand the concept behind interrupts on any processor, but we will use the SJ-One board as an example.

# What is an interrupt?

An interrupt is the hardware capability of a CPU to break the normal flow of software to attend an urgent request.

The science behind interrupts lies in the hardware that allows the CPU to be interrupted. Each peripheral in a microcontroller *may be* able to assert an interrupt to the CPU core, and then the CPU core would jump to the corresponding interrupt service routine (**ISR**) to service the interrupt.

# **ISR** Procedure



Figure 1. Nested Interrupt Processing

The following steps demonstrate what happens when an interrupt occurs:

- Interrupt hardware decodes which peripheral sent interrupt signal and alerts the CPU
- CPU manipulates the PC (program counter) to jump to the ISR
   IMPORTANT: CPU will disable interrupts (or that priority level's interrupts until end of ISR)
- Registers are saved before running the ISR (pushed onto the stack)
- ISR is executed and returns
- Registers are restored (popped from stack)
- Interrupts are re-enabled (or that priority level's interrupt is re-enabled)

On some processors, the savings and restoring of registers is a manual step and the compiler would help you do it. You can google "GCC interrupt attribute" to study this topic further. On SJ-One board, which uses LPC17xx (ARM Cortex M3), this step is automatically taken care of by the CPU hardware.

### Nested Vector Interrupt Controller

Nested Vector Interrupt Controllers or NVIC for short, have two properties:

- Can handle multiple interrupts.
  - The number of interrupts implemented is device dependent.
- A programmable priority level for each interrupt.

- A higher level corresponds to a lower priority, so level 0 is the highest interrupt priority.
- Level and pulse detection of interrupt signals.
- Grouping of priority values into group priority and sub-priority fields.
  - $\circ\,$  This means that interrupts of the same priority are grouped together and do not preempt each other.
  - Each interrupt also has a sub-priority field which is used to figure out the run order of pending interrupts of the same priority.
- Interrupt tail-chaining.
  - This enables back-to-back interrupt processing without the overhead of state saving and restoration between interrupts.
  - $\circ$  This saves us from the step of having to restore and then save the registers again.
- An external N



Figure 2. Multiple Interrupt Processing

### The SW to HW Connection

Now that we understand how the CPU hardware services interrupts, we need to define how we inform the CPU WHERE our ISR function is located at.

### Interrupt Vector Table

This table is nothing but addresses of functions that correspond to the microcontroller interrupts. Specific interrupts use specific "slots" in this table, and we have to populate these spots with our software functions that

service the interrupts.

| Table 2.2 Cortex-M3 Exception Types |                                  |                                                  |                                                                               |  |  |
|-------------------------------------|----------------------------------|--------------------------------------------------|-------------------------------------------------------------------------------|--|--|
| Exception<br>Number                 | Exception Type                   | Priority (Default to<br>0 if Programmable)       | Description                                                                   |  |  |
| 0                                   | NA                               | NA                                               | No exception running                                                          |  |  |
| 1                                   | Reset                            | –3 (Highest)                                     | Reset                                                                         |  |  |
| 2                                   | NMI                              | -2                                               | NMI (external NMI input)                                                      |  |  |
| 3                                   | Hard fault                       | -1                                               | All fault conditions, if the corresponding<br>fault handler is not enabled    |  |  |
| 4                                   | MemManage fault                  | Programmable                                     | Memory management fault; MPU<br>violation or access to illegal locations      |  |  |
| 5                                   | Bus fault                        | Programmable                                     | Bus error (prefetch abort or data abort)                                      |  |  |
| 6                                   | Usage fault                      | Programmable                                     | Program error                                                                 |  |  |
| 7–10                                | Reserved                         | NA                                               | Reserved                                                                      |  |  |
| 11                                  | SVCall                           | Programmable                                     | Supervisor call                                                               |  |  |
| 12                                  | Debug monitor                    | Programmable                                     | Debug monitor (break points,<br>watchpoints, or external debug request)       |  |  |
| 13                                  | Reserved                         | NA                                               | Reserved                                                                      |  |  |
| 14                                  | PendSV                           | Programmable                                     | Pendable request for system service                                           |  |  |
| 15                                  | SYSTICK                          | Programmable                                     | System tick timer                                                             |  |  |
| 16                                  | IRQ #0                           | Programmable                                     | External interrupt #0                                                         |  |  |
| 17                                  | IRQ #1                           | Programmable                                     | External interrupt #1                                                         |  |  |
| <br>255                             | <br>IRQ #239                     | <br>Programmable                                 | <br>External interrupt #239                                                   |  |  |
| 16<br>17<br><br>255                 | IRQ #0<br>IRQ #1<br><br>IRQ #239 | Programmable<br>Programmable<br><br>Programmable | External interrupt #0<br>External interrupt #1<br><br>External interrupt #239 |  |  |

The number of external interrupt inputs is defined by chip manufacturers. A maximum of 240 external interrupt inputs can be supported. In addition, the Cortex-M3 also has an NMI interrupt input. When it is asserted, the NMI-ISR is executed unconditionally.

Figure 3. HW Interrupt Vector Table

# SJOne (LPC17xx) Example

The using a linker script and compiler directives (commands for the compiler), the compiler is able to place the software interrupt vector table at a specific location that the CPU expects the interrupt vector table to be located at. This connects the dots about how the CPU is able to determine WHERE your interrupt service routines are located at. From there on, anytime a specific interrupt occurs, the CPU is able to fetch the address and make the JUMP.

#### /\*\*

- \* CPU interrupt vector table that is loaded at the beginning of the CPU start
- \* location by using the linker script that will place it at the isr\_vector location.
- \* CPU loads the stack pointer and begins execution from Reset vector.

```
*/
extern void (* const g_pfnVectors[])(void);
__attribute__ ((section(".isr_vector")))
void (* const g_pfnVectors[])(void) =
{
   // Core Level - CM3
   & vStackTop, // The initial stack pointer
   isr_reset, // The reset handler
                     // The NMI handler
   isr_nmi,
   isr_hard_fault,
                     // The hard fault handler
   isr_mem_fault, // The MPU fault handler
   isr bus fault, // The bus fault handler
   isr_usage_fault, // The usage fault handler
   0,
                     // Reserved
   0,
                     // Reserved
   0,
                     // Reserved
                      // Reserved
   0,
   vPortSVCHandler,
                     // FreeRTOS SVC-call handler (naked function so needs direct call - not a wrap
   isr_debug_mon, // Debug monitor handler
                      // Reserved
   0,
   xPortPendSVHandler, // FreeRTOS PendSV handler (naked function so needs direct call - not a wrappe
                  // FreeRTOS SysTick handler (we enclose inside a wrapper to track OS overhead)
   isr sys tick,
   // Chip Level - LPC17xx - common ISR that will call the real ISR
   isr_forwarder_routine, // 16, 0x40 - WDT
   isr_forwarder_routine, // 17, 0x44 - TIMER0
   isr_forwarder_routine,
                            // 18, 0x48 - TIMER1
   isr_forwarder_routine,
                             // 19, 0x4c - TIMER2
   isr_forwarder_routine, // 20, 0x50 - TIMER3
   isr_forwarder_routine, // 21, 0x54 - UART0
   isr_forwarder_routine,
                             // 22, 0x58 - UART1
   isr_forwarder_routine,
                             // 23, 0x5c - UART2
   isr_forwarder_routine,
                             // 24, 0x60 - UART3
                          // 25, 0x64 - PWM1
   isr_forwarder_routine,
   isr_forwarder_routine,
                             // 26, 0x68 - I2C0
   isr_forwarder_routine,
                             // 27, 0x6c - I2C1
   isr_forwarder_routine,
                             // 28, 0x70 - I2C2
```

| <pre>isr_forwarder_routine,</pre> | // 29, 0x74 - SPI                                           |
|-----------------------------------|-------------------------------------------------------------|
| <pre>isr_forwarder_routine,</pre> | // 30, 0x78 - SSP0                                          |
| <pre>isr_forwarder_routine,</pre> | // 31, 0x7c - SSP1                                          |
| <pre>isr_forwarder_routine,</pre> | // 32, 0x80 - PLL0 (Main PLL)                               |
| <pre>isr_forwarder_routine,</pre> | // 33, 0x84 - RTC                                           |
| <pre>isr_forwarder_routine,</pre> | // 34, 0x88 - EINT0                                         |
| <pre>isr_forwarder_routine,</pre> | // 35, 0x8c - EINT1                                         |
| <pre>isr_forwarder_routine,</pre> | // 36, 0×90 - EINT2                                         |
| <pre>isr_forwarder_routine,</pre> | // 37, 0x94 - EINT3                                         |
| <pre>isr_forwarder_routine,</pre> | // 38, 0x98 - ADC                                           |
| <pre>isr_forwarder_routine,</pre> | // 39, 0x9c - BOD                                           |
| <pre>isr_forwarder_routine,</pre> | // 40, 0×A0 - USB                                           |
| <pre>isr_forwarder_routine,</pre> | // 41, 0xa4 - CAN                                           |
| <pre>isr_forwarder_routine,</pre> | // 42, 0xa8 - GP DMA                                        |
| <pre>isr_forwarder_routine,</pre> | // 43, 0xac - I2S                                           |
| <pre>isr_forwarder_routine,</pre> | // 44, 0xb0 - Ethernet                                      |
| <pre>isr_forwarder_routine,</pre> | // 45, 0xb4 - RITINT                                        |
| <pre>isr_forwarder_routine,</pre> | // 46, 0xb8 - Motor Control PWM                             |
| <pre>isr_forwarder_routine,</pre> | // 47, 0xbc - Quadrature Encoder                            |
| <pre>isr_forwarder_routine,</pre> | // 48, 0xc0 - PLL1 (USB PLL)                                |
| <pre>isr_forwarder_routine,</pre> | <pre>// 49, 0xc4 - USB Activity interrupt to wakeup</pre>   |
| <pre>isr_forwarder_routine,</pre> | <pre>// 50, 0xc8 - CAN Activity interrupt to wakeup};</pre> |
|                                   |                                                             |

Code Block 1. Software Interrupt Vector Table

**NOTE:** that a vector table is really just a lookup table that hardware utilizes.

### Two Methods to setup an ISR on the SJOne

All of the methods require that you run this function to allow the NVIC to accept a particular interrupt request.

#### ? NVIC\_EnableIRQ(EINT3\_IRQn);

Where the input is the IRQ number. This can be found in the LCP17xx.h file. Search for **enum IRQn**.

### Method 1. Modify IVT

DO NOT DO THIS, unless you really know what you are doing. The ISR forwarder works with FreeRTOS to distinguish CPU utilization between ISRs and tasks.

I highly discourage modifying the **startup.cpp** and modifying the vector tables directly. Its not dynamic is less manageable in that, if you switch projects and the ISR doesn't exist, the compiler will through an error.

#### IVT modify

```
/* You will need to include the header file that holds the ISR for this to work */
#include "my_isr.h"
extern void (* const g_pfnVectors[])(void);
__attribute__ ((section(".isr_vector")))
void (* const g_pfnVectors[])(void) =
{
   // Core Level - CM3
   &_vStackTop, // The initial stack pointer
   isr_reset, // The reset handler
   isr_nmi,
                     // The NMI handler
   isr_hard_fault,
                     // The hard fault handler
   isr_mem_fault,
                     // The MPU fault handler
   isr_bus_fault, // The bus fault handler
   isr_usage_fault, // The usage fault handler
   0,
                     // Reserved
   0,
                     // Reserved
   Θ,
                     // Reserved
   0,
                     // Reserved
   vPortSVCHandler,
                     // FreeRTOS SVC-call handler (naked function so needs direct call - not a wrap
   isr_debug_mon, // Debug monitor handler
                      // Reserved
   0,
   xPortPendSVHandler, // FreeRTOS PendSV handler (naked function so needs direct call - not a wrappe
   isr_sys_tick, // FreeRTOS SysTick handler (we enclose inside a wrapper to track OS overhead)
   // Chip Level - LPC17xx - common ISR that will call the real ISR
   isr_forwarder_routine, // 16, 0x40 - WDT
   isr_forwarder_routine, // 17, 0x44 - TIMER0
```

| <pre>isr_forwarder_routine,</pre> | // 18, 0x48 - TIMER1                                                   |
|-----------------------------------|------------------------------------------------------------------------|
| <pre>isr_forwarder_routine,</pre> | // 19, 0x4c - TIMER2                                                   |
| <pre>isr_forwarder_routine,</pre> | // 20, 0×50 - TIMER3                                                   |
| <pre>isr_forwarder_routine,</pre> | // 21, 0x54 - UART0                                                    |
| <pre>isr_forwarder_routine,</pre> | // 22, 0x58 - UART1                                                    |
| <pre>isr_forwarder_routine,</pre> | // 23, 0x5c - UART2                                                    |
| <pre>isr_forwarder_routine,</pre> | // 24, 0x60 - UART3                                                    |
| <pre>isr_forwarder_routine,</pre> | // 25, 0x64 - PWM1                                                     |
| <pre>isr_forwarder_routine,</pre> | // 26, 0×68 - I2C0                                                     |
| <pre>isr_forwarder_routine,</pre> | // 27, 0x6c - I2C1                                                     |
| <pre>isr_forwarder_routine,</pre> | // 28, 0x70 - I2C2                                                     |
| <pre>isr_forwarder_routine,</pre> | // 29, 0x74 - SPI                                                      |
| <pre>isr_forwarder_routine,</pre> | // 30, 0x78 - SSP0                                                     |
| <pre>isr_forwarder_routine,</pre> | // 31, 0x7c - SSP1                                                     |
| <pre>isr_forwarder_routine,</pre> | // 32, 0x80 - PLLO (Main PLL)                                          |
| <pre>isr_forwarder_routine,</pre> | // 33, 0x84 - RTC                                                      |
| <pre>isr_forwarder_routine,</pre> | // 34, 0x88 - EINTO                                                    |
| <pre>isr_forwarder_routine,</pre> | // 35, 0x8c - EINT1                                                    |
| <pre>isr_forwarder_routine,</pre> | // 36, 0×90 - EINT2                                                    |
| runMyISR,                         | <pre>// 37, 0x94 - EINT3 &lt; NOTICE how I changed the name here</pre> |
| <pre>isr_forwarder_routine,</pre> | // 38, 0×98 - ADC                                                      |
| <pre>isr_forwarder_routine,</pre> | // 39, 0x9c - BOD                                                      |
| <pre>isr_forwarder_routine,</pre> | // 40, 0×A0 - USB                                                      |
| <pre>isr_forwarder_routine,</pre> | // 41, 0xa4 - CAN                                                      |
| <pre>isr_forwarder_routine,</pre> | // 42, 0xa8 - GP DMA                                                   |
| <pre>isr_forwarder_routine,</pre> | // 43, 0xac - I2S                                                      |
| <pre>isr_forwarder_routine,</pre> | // 44, 0xb0 - Ethernet                                                 |
| <pre>isr_forwarder_routine,</pre> | // 45, 0xb4 - RITINT                                                   |
| <pre>isr_forwarder_routine,</pre> | // 46, 0xb8 - Motor Control PWM                                        |
| <pre>isr_forwarder_routine,</pre> | // 47, 0xbc - Quadrature Encoder                                       |
| <pre>isr_forwarder_routine,</pre> | // 48, 0xc0 - PLL1 (USB PLL)                                           |
| <pre>isr_forwarder_routine,</pre> | <pre>// 49, 0xc4 - USB Activity interrupt to wakeup</pre>              |
| <pre>isr_forwarder_routine,</pre> | <pre>// 50, 0xc8 - CAN Activity interrupt to wakeup};</pre>            |
|                                   |                                                                        |

Code Block 3. Weak Function Override Template

### Method 2. ISR Register Function

The **EINT3\_IRQn** symbol is defined in an enumeration in LPC17xx.h. All you need to do is specify the IRQ number and the function you want to act as an ISR. This will then swap out the previous ISR with your function.

This is the best option! Please use this option almost always!

```
/**
* Just your run-of-the-mill function
*/
void myEINT3ISR(void)
{
    doSomething();
    clearInterruptFlag();
}
int main()
{
    /**
     * Find the IRQ number for the interrupt you want to define.
     * In this case, we want to override IRO 0x98 EINT3
     * Then specify a function pointer that will act as your ISR
     */
    isr_register(EINT3_IRQn, myEINT3ISR);
    NVIC_EnableIRQ(EINT3_IRQn);}
```

|  | Code Block | 5. | Weak | Function | Override | Template |
|--|------------|----|------|----------|----------|----------|
|--|------------|----|------|----------|----------|----------|

| PROS                                                                                                                                                                                                                                                                                                                 | CONS                                                                                                                                   |
|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------|
| <ul> <li>Can dynamically change ISR during runtime.</li> <li>Does not disturb core library files in the process of adding/changing ISRs. <ul> <li>Always try to prevent changes to the core libraries.</li> </ul> </li> <li>Does not cause compiler errors.</li> <li>Your ISR cpu utilization is tracked.</li> </ul> | <ul> <li>Must wait until main is called before ISR is registered</li> <li>Interrupt events could happen before main begins.</li> </ul> |

# What to do inside an ISR

Do very little inside an ISR. When you are inside an ISR, the whole system is blocked (other than higher priority

interrupts). If you spend too much time inside the ISR, then you are destroying the real-time operating system principle and everything gets clogged.

With that said, here is the general guideline:

### Short as possible

DO NOT POLL FOR ANYTHING! Try to keep loops as small as possible. Note that printing data over UART can freeze the entire system, including the RTOS for that duration. For instance, printing 4 chars may take 1ms at 38400bps.

### FreeRTOS API calls

If you are using FreeRTOS API, you must use **FromISR** functions only! If a FromISR function does not exist, then don't use that API.

### **Clear Interrupt Sources**

Clear the source of the interrupt. For example, if interrupt was for rising edge of a pin, clear the "rising edge" bit such that you will not re-enter into the same interrupt function.

If you don't do this, your interrupt will get stuck in an infinite ISR call loop. For the Port interrupts, this can be done by writing to the IntClr registers.

# ISR processing inside a FreeRTOS Task

It is a popular scheme to have an ISR quickly exit, and then resume a task or thread to process the event. For example, if we wanted to write a file upon a button press, we don't want to do that inside an ISR because it would take too long and block the system. What we can utilize a **wait on semaphore** design pattern.

What you may argue with the example below is that we do not process the ISR immediately, and therefore delay the processing. But you can tackle this scenario by resuming a HIGHEST priority task. Immediately, after the ISR exits, due to the ISR "yield", FreeRTOS will resume the high priority task immediately rather than servicing another task

```
/* Create the semaphore in main() */
SemaphoreHandle_t button_press_semaphore = NULL;
void myButtonPressISR(void)
{
    long yield = 0;
    xSemaphoreGiveFromISR(button_press_semaphore, &yield);
    portYIELD_FROM_ISR(yield);
```

```
}
void vButtonPressTask(void *pvParameter)
{
    while(1)
    {
        if (xSemaphoreTake(button_press_semaphore, portMAX_DELAY))
        {
           /* Process the interrupt */
        }
    }
}
void main(void)
{
    button_press_semaphore = xSemaphoreCreateBinary();
    /* TODO: Hook up myButtonPressISR() using eint.h */
    /* TODO: Create vButtonPressTask() and start FreeRTOS scheduler */}
```

Code Block 6. Wait on Semaphore ISR design pattern example

### Resources

http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ddi0489b/CACDDJHB.html

Revision #15 Created 2 months ago by Admin Updated 1 day ago by Khalil Estell