# Interrupt Management: Auto-vectoring and Prioritization

# Background

The AT91 is based on the ARM7TDMI<sup>™</sup> microcontroller core. It features the Advanced Interrupt Controller (AIC), an 8-level priority, individually maskable, vectored interrupt controller.

This microcontroller core implements two physically independent sources of interrupt:

- FIQ Fast Interrupt
- IRQ Normal Interrupt

Each of these interrupts has a corresponding vector, at addresses 0x00000018 for the IRQ and 0x0000001C for the FIQ.

The AIC is connected to the NFIQ (Fast Interrupt Request) and the NIRQ (Standard Interrupt Request) inputs of the ARM7TDMI processor.

The processor's NFIQ line can only be asserted by the external fast interrupt request input: FIQ (multiplexed with the PIO P12). Therefore, when an FIQ occurs, it is not necessary to de-multiplex the handler according to the cause of the interrupt (it is assumed that there is no multiplexing added by the external hardware). The FIQ management code can be reached either directly from the vector (0x0000001C), or by using the Fast Interrupt Vector Register (AIC\_FVR) as described in the datasheet of the AT91 products.

The NIRQ line can be asserted by the interrupts generated by the on-chip peripherals and the external interrupt request lines: IRQ0 to IRQ2. Therefore it is necessary to manage a prioritization when several interrupt sources are asserted at once and to de-multiplex the handler according to the source of the interrupt.



AT91 ARM Thumb<sup>®</sup> Microcontrollers

# **Application Note**

Rev. 1168A-09/98





# **Auto-Vectoring**

This feature consists of a set of registers which provide the address of the handler to execute according to the source of an interrupt.

Each interrupt source is associated with a Source Vector Register (AIC\_SVR1 - AIC\_SVR31) which contains the address of the function corresponding to the active interrupt. When the Interrupt Vector Register (AIC\_IVR) is read, it automatically returns the contents of the source vector register corresponding to the active interrupt with the highest priority. Note that AIC\_IVR is located at address 0xFFFF100.

During the boot sequence and before enabling the interrupts, the software must:

- 1. Initialize the source vector registers for each interrupt
- 2. Initialize the IRQ vector at address 0x00000018 with the following code:

ldr pc,[pc,#-0xF20]

When an interrupt occurs, the core performs the following (see the ARM Architectural Reference Manual):

R14\_irq = address of next instruction to be executed + 4

| SPSR_irq = CPSR      |                                    |
|----------------------|------------------------------------|
| CPSR[5:0] = 0b010010 | Interrupt mode                     |
| CPSR[6] = unchanged  | Fast interrupt status is unchanged |
| CPSR[7] = 1          | Normal interrupts disabled         |
| PC = 0x00000018      |                                    |

When the instruction at the address 0x00000018 is executed, the effective address is:

0x00000020 - 0x0F20 = 0xFFFFF100

(0x00000020 is the value of the PC when the instruction at address 0x18 is executed)

This causes the core to load the PC with the value read in AIC\_IVR which returns the value of AIC\_SVR corresponding to the active interrupt. This has the effect of directly jumping to the correct interrupt service routine.

Also note that when the AIC\_IVR is read, the AIC does the following:

- · deasserts the NIRQ line on the core
- · determines which pending interrupt has the highest priority
- pushes the level of this interrupt in its internal hardware stack
- · clears the interrupt if it is configured to be edge triggered

The interrupt level is popped when the End of Interrupt (EOI) is indicated to the AIC by a write in AIC\_EOICR (see "Prioritization" on page 3).

# AT91 ARM Thumb

# Prioritization

The NIRQ line is controlled by an 8-level priority encoder. Each source has a programmable priority level of 7 to 0. Level 7 is the highest priority and level 0 the lowest.

When the AIC receives more than one unmasked interrupt at a time, the interrupt with the highest priority is serviced first. The interrupt management of the interrupt with the lower priority level is therefore delayed.

The AIC manages the prioritization by using an internal stack on which the current interrupt level is automatically pushed when AIC\_IVR is read, and popped when AIC\_EOICR is written (any value). Between these two events, the software can manage the state and the mode of the core in order to re-enable the IRQ line and to allow an interrupt with a higher priority.

When an interrupt is managed by the core, R14\_irq and SPSR\_irq are automatically overwritten without being saved: it is mandatory to save these registers before re-enabling the IRQ line and to restore them before exiting the interrupt management routine. Moreover, if the interrupt treatment performs function calls (Branch with Link), R14\_irq is used. In this case, IRQ can not be re-enabled while the core is in IRQ mode. It is mandatory to first change the mode of the core. In order to keep all exceptions available, the SYSTEM mode must be used. Therefore, the stack used during the interrupt execution is the same as that used out of the interrupt. This must be taken into account in the sizing of the SYSTEM/USER stack.

This is performed as follows:

- 1. Save R14\_irq and SPSR\_irq in the IRQ stack (current)
- 2. Set the mode bits in CPSR with the SYSTEM value (0b11111)
- 3. Re-enable IRQ by clearing bit I in CPSR
- 4. Execute the actions related to the interrupt
- 5. Disable IRQ by clearing bit I in CPSR
- 6. Set the mode bits in CPSR with the USER value (0b10000)
- 7. Restore R14\_irq and SPSR\_irq from the IRQ stack

Note that this sequence is automatically preceded by a read of AIC\_IVR (see "Auto-Vectoring" on page 2) and must be followed by a write in AIC\_EOICR before exiting from the interrupt.



# <u>AÎMEL</u>

# AT91M40400 Implementation

The implementation of the auto-vectoring is done by initializing the IRQ vector at address 0x0000018 with the following instruction:

ldr pc,[pc,#-0xF20]

The implementation of the prioritization is described in the file "irq.mac" which is included in the folder "at91\_include" of the AT91 library (examples of use can be found in the files "irq\_\*.s" of the folder "at91\_lib"). This file includes 2 macros:

- IRQ\_ENTRY which saves the registers and switches to SYSTEM mode
- IRQ\_EXIT which switches back to IRQ mode, restores the registers and exits from the interrupt after acknowledging the current interrupt by writing in AIC\_EOICR.

The standard format of an interrupt handler is:

- 1. Auto-Vectoring: instruction "ldr pc,[pc,#-0xF20]"
- 2. Validate the nested interrupts: macro-definition IRQ\_ENTRY
- 3. Perform interrupt treatment (e.g. for the USART transmitter, write a new byte in the US\_THR)

4. Disable the nested interrupts: macro-definition IRQ\_EXIT

#### **IRQ\_ENTRY Macro Definition**

MACRO IRQ\_ENTRY \$reg ;- Adjust and save LR of current mode in current stack sub r14, r14, #4 sp!, {r14} stmfd ;- Save SPSR and r0 in current stack r14, SPSR mrs stmfd sp!, {r0, r14} ;- Read Modify Write the CPSR to Enable the Core Interrupt ;- and Switch in SYS Mode ( same LR and stack than USR Mode ) mrg r14, CPSR bic r14, r14, #I BIT r14, r14, #ARM\_MODE\_SYS orr CPSR, r14 msr ;- Save used registers and LR\_usr in the System/User Stack stmfd sp!, {r1-r3, \$reg, r12, r14} MEND

The parameter "\$reg" allow the list of the registers used by the interrupt treatment to be pushed on the SYSTEM/USER stack by using the instruction which pushes R14\_User. This list must be the same for the IRQ\_EXIT call. Note that in this application note, all registers defined as "scratched" by APCS (r0, r1, r2, r3, r12) are saved by IRQ\_ENTRY and restored by IRQ\_EXIT.

AT91 ARM Thumb

#### **IRQ\_EXIT Macro Definition**

```
MACRO
      IRQ_EXIT
                  $reg,
;- Restore used registers and LR_usr from the System/User Stack
            ldmfd
                              sp!, {r1-r3, $reg, r12, r14}
;- Read Modify Write the CPSR to disable interrupts
;- and to go back in the mode corresponding to the exception
            mrs
                              r0, CPSR
                              r0, r0, #ARM_MODE_SYS
            bic
                              r0, r0, #I_BIT:OR:ARM_MODE_IRQ
            orr
            msr
                              CPSR, r0
;- Mark the End of Interrupt on the interrupt controller
            ldr
                              r0, = AIC_BASE
            str
                              r0, [r0, #AIC_EOICR]
;- Restore SPSR_irg and r0 from the IRQ stack
            ldmfd
                              sp!, {r0, r14}
                              SPSR, r14
            msr
;- Restore ajusted LR_irq from IRQ stack directly in the PC
            ldmfd
                              sp!, {pc}^
      MEND
```

#### The IRQ Stack

The IRQ stack pointer (R13\_irq) must be initialized at the top (upper address) of a reserved space. The size needed for this stack is 12 bytes (3 words for registers r0, r14 and SPSR) per level used in the application. If all levels are used, the stack space must be 96 bytes.

#### Constants

The constants used in this application note are as follows:

| ARM_MODE_SYS | EQU | 0x1F       |
|--------------|-----|------------|
| I_BIT        | EQU | 0x80       |
| AIC_BASE     | EQU | 0xFFFFF000 |
| AIC_EOICR    | EQU | 0x0130     |





#### **Atmel Headquarters**

#### **Corporate Headquarters**

2325 Orchard Parkway San Jose, CA 95131 TEL (408) 441-0311 FAX (408) 487-2600

#### Europe

Atmel U.K., Ltd. **Coliseum Business Centre Riverside Way** Camberley, Surrey GU15 3YL England TEL (44) 1276-686677 FAX (44) 1276-686697

#### Asia

Atmel Asia, Ltd. Room 1219 Chinachem Golden Plaza 77 Mody Road Tsimshatsui East Kowloon, Hong Kong TEL (852) 27219778 FAX (852) 27221369

#### Japan

Atmel Japan K.K. Tonetsu Shinkawa Bldg., 9F 1-24-8 Shinkawa Chuo-ku, Tokyo 104-0033 Japan TEL (81) 3-3523-3551 FAX (81) 3-3523-7581

### **Atmel Operations**

#### Atmel Colorado Springs

1150 E. Chevenne Mtn. Blvd. Colorado Springs, CO 80906 TEL (719) 576-3300 FAX (719) 540-1759

#### **Atmel Rousset**

Zone Industrielle 13106 Rousset Cedex, France TEL (33) 4 42 53 60 00 FAX (33) 4 42 53 60 01

#### Fax-on-Demand

North America: 1-(800) 292-8635 International: 1-(408) 441-0732

#### e-mail

literature@atmel.com

Web Site http://www.atmel.com

BBS

1-(408) 436-4309

#### © Atmel Corporation 1998.

Atmel Corporation makes no warranty for the use of its products, other than those expressly contained in the Company's standard warranty which is detailed in Atmel's Terms and Conditions located on the Company's website. The Company assumes no responsibility for any errors which may appear in this document, reserves the right to change devices or specifications detailed herein at any time without notice, and does not make any commitment to update the information contained herein. No licenses to patents or other intellectual property of Atmel are granted by the Company in connection with the sale of Atmel products, expressly or by implication. Atmel's products are not authorized for use as critical components in life support devices or systems. Marks bearing <sup>®</sup> and/or <sup>™</sup> are registered trademarks and trademarks of Atmel Corporation. ARM, Thumb and ARM Powered are registered trademarks of ARM Limited.

The ARM7TDMI is a trademark of ARM Ltd.

Terms and product names in this document may be trademarks of others.

