# Interrupts & Interrupt Service Routines (ISRs)

### Marten van Dijk, Syed Kamran Haider

Department of Electrical & Computer Engineering
University of Connecticut
Email: {vandijk, syed.haider}@engr.uconn.edu





## Interrupts

### Lower range of program storage in flash:

#### 11.4 Interrupt Vectors in ATmega328P

Table 11-6. Reset and Interrupt Vectors in ATmega328P

| Table 11-6. | Reset and Interrupt vectors in Armegas |              | 201                                                                     |  |  |  |
|-------------|----------------------------------------|--------------|-------------------------------------------------------------------------|--|--|--|
| VectorNo.   | Program<br>Address <sup>(2)</sup>      | Source       | Interrupt Definition                                                    |  |  |  |
| 1           | 0x0000 <sup>(1)</sup>                  | RESET        | External Pin, Power-on Reset, Brown-out Reset and Watchdog System Reset |  |  |  |
| 2           | 0x0002                                 | INT0         | External Interrupt Request 0                                            |  |  |  |
| 3           | 0x0004                                 | INT1         | External Interrupt Request 1                                            |  |  |  |
| 4           | 0x0006                                 | PCINT0       | Pin Change Interrupt Request 0                                          |  |  |  |
| 5           | 0x0008                                 | PCINT1       | Pin Change Interrupt Request 1                                          |  |  |  |
| 6           | 0x000A                                 | PCINT2       | Pin Change Interrupt Request 2                                          |  |  |  |
| 7           | 0x000C                                 | WDT          | Watchdog Time-out Interrupt                                             |  |  |  |
| 8           | 0x000E                                 | TIMER2 COMPA | Timer/Counter2 Compare Match A                                          |  |  |  |
| 9           | 0x0010                                 | TIMER2 COMPB | Timer/Counter2 Compare Match B                                          |  |  |  |
| 10          | 0x0012                                 | TIMER2 OVF   | Timer/Counter2 Overflow                                                 |  |  |  |
| 11          | 0x0014                                 | TIMER1 CAPT  | Timer/Counter1 Capture Event                                            |  |  |  |
| 12          | 0x0016                                 | TIMER1 COMPA | Timer/Counter1 Compare Match A                                          |  |  |  |
| 13          | 0x0018                                 | TIMER1 COMPB | Timer/Coutner1 Compare Match B                                          |  |  |  |
| 14          | 0x001A                                 | TIMER1 OVF   | Timer/Counter1 Overflow                                                 |  |  |  |
| 15          | 0x001C                                 | TIMER0 COMPA | Timer/Counter0 Compare Match A                                          |  |  |  |
| 16          | 0x001E                                 | TIMER0 COMPB | Timer/Counter0 Compare Match B                                          |  |  |  |
| 17          | 0x0020                                 | TIMER0 OVF   | Timer/Counter0 Overflow                                                 |  |  |  |
| 18          | 0x0022                                 | SPI, STC     | SPI Serial Transfer Complete                                            |  |  |  |
| 19          | 0x0024                                 | USART, RX    | USART Rx Complete                                                       |  |  |  |
| 20          | 0x0026                                 | USART, UDRE  | USART, Data Register Empty                                              |  |  |  |
| 21          | 0x0028                                 | USART, TX    | USART, Tx Complete                                                      |  |  |  |
| 22          | 0x002A                                 | ADC          | ADC Conversion Complete                                                 |  |  |  |

If you want to set the mask bit of an interrupt, i.e., you enable a certain interrupt, then you *must* write a corresponding ISR (interrupt service routine).

The table contains the address of the ISR that you write (upon the HW event that will cause the interrupt, the program counter will jump to the address indicated by the table to execute the programmed ISR).

Program memory has 2<sup>1</sup>6 registers

- → an address has 16 bits, e.g., 0xabcd
- → 0xabcd is stored in two 8-bit registers
- → Interrupt vector table associates interrupt vectors to addresses 0x0000, 0x0002, 0x0004 etc. (by increments of 2)

Table 11-6. Reset and Interrupt Vectors in ATmega328P (Continued)

| VectorNo. | Program<br>Address <sup>(2)</sup> | Source      | Interrupt Definition       |
|-----------|-----------------------------------|-------------|----------------------------|
| 23        | 0x002C                            | EE READY    | EEPROM Ready               |
| 24        | 0x002E                            | ANALOG COMP | Analog Comparator          |
| 25        | 0x0030                            | TWI         | 2-wire Serial Interface    |
| 26        | 0x0032                            | SPM READY   | Store Program Memory Ready |

# Program Layout

### Initialization procedure:

- Set up tables,
- Initialize timers,
- Do bookkeeping before you can put on interrupts
- Turn on the master interrupt bit: This is the I-bit in register SREG, the C-macro sei() does this for you

### 6.3.1 SREG – AVR Status Register

The AVR Status Register - SREG - is defined as:

| Bit           | 7   | 6   | 5   | 4  | 3  | 2  | 1  | 0   | _    |
|---------------|-----|-----|-----|----|----|----|----|-----|------|
| 0x3F (0x5F)   | I   | T   | Н   | S  | V  | N  | Z  | С   | SREG |
| Read/Write    | R/W | R/W | R/W | RW | RW | RW | RW | R/W |      |
| Initial Value | 0   | 0   | 0   | 0  | 0  | 0  | 0  | 0   |      |

#### Bit 7 – I: Global Interrupt Enable

The Global Interrupt Enable bit must be set for the interrupts to be enabled. The individual interrupt enable control is then performed in separate control registers. If the Global Interrupt Enable Register is cleared, none of the interrupts are enabled independent of the individual interrupt enable settings. The I-bit is cleared by hardware after an interrupt has occurred, and is set by the RETI instruction to enable subsequent interrupts. The I-bit can also be set and cleared by the application with the SEI and CLI instructions, as described in the instruction set reference.

# Program Layout

- Main() executes slow background code forever → you never exit main in a MCU
  - Interrupt driven tasks are asynchronously called from main, for example:
    - a HW timer may cause a HW event every 1000 cycles, upon which in the corresponding ISR a SW counter is incremented;
    - upon reaching an a-priori defined maximum value, the background code calls a corresponding procedure which executes some task, and upon returning the SW counter is reset to 0
- ISRs have no parameters, no return value, they save CPU state (and C does this for you); they are called by HW events:
  - E.g., the bit value for position RXC0 in UCSR0A goes to high when receiving a character is completed
  - If the mask bit in UCSROB for position RXCIEO is set [meaning that an interrupt is enabled for the flag ( UCSROA & (1 << RXCO) ) ], then the MCU will jump to the address of the ISR as indicated by the interrupt vector table for source USART, RX.
  - It takes about 75 cycles to go in and out of a ISR; another 32 cycles to safe state of the MCU (32 registers); another 7/8 cycles overhead.

## **AVR** Architecture

Figure 6-1. Block Diagram of the AVR Architecture



## Execution of an ISR



### Execution of an ISR

- 1. Some HW event sets a flag in some register, e.g., (UCSROA & (1<<RXCO)) goes to high

  → If the corresponding interrupt is enabled, e.g., by initially programming UCSROB |=
  (1<<RXCIEO), then this flag is detected by the Interrupt Unit of the MCU which scans the
  flags which correspond to the vector table in round robin fashion
- 2. If the CPU is executing an ISR, then finish the ISR, else finish current instruction
- 3. Push the Program Counter (PC) on the stack
- 4. Clear the I-bit in SREG (after this, none of the interrupts are enabled)
- 5. Clear flag in register UCSROA as in UCSROA  $\&=\sim(1<<RXCO)$
- 6. The CPU jumps to the vector table and clears the corresponding flag
- 7. The CPU jumps to the ISR indicated by the address in the vector table
- 8. The compiler created a binary which saves state, executes your ISR code, restores state, and returns: return from interrupt (RETI)
- 9. RETI enables the I-bit in SREG and re-checks the interrupt flag registers in the vector table (since other HW events may have occurred in the meantime)

## **Problems**

- Example: An ISR with print statement calls the print procedure, which buffers the characters to be printed in HW since printing is slow.
- Now, the HW executes the printing statement in parallel with the rest of the ISR.
- The ISR finishes.
- Before the print statement is finished the ISR is triggered again
- Not even a single character may be printed!!

## **Problems**

In your ISR you may enable the master interrupt bit → this creates a nested interrupt
 → not recommended

- Memory of one event deep: e.g.,
  - MCU handles a first flag of (UCSROA & (1 << RXCO))</li>
  - After clearing this flag, the same HW event happens again which will again set the interrupt flag vector for ( UCSROA & (1 << RXCO) ) (which will be handled after the current interrupt)</li>
  - But more interrupts for ( UCSROA & (1 << RXCO) ) are forgotten while handling the current interrupt (first flag)!!
  - You need to write efficient ISR code to avoid missing HW events, which may cause your application to be unreliable.

# Interrupt Service Routine (ISR)

- http://www.atmel.com/webdoc/AVRLibcReferenceManual/group avr interrupts. html
- #include <avr/interrupt.h>

| USART_RXC_vect | SIG_USART_RECV,<br>SIG_UART_RECV   | USART, Rx<br>Complete | ATmega16, ATmega32, ATmega8                                                                                                                                                                                 |
|----------------|------------------------------------|-----------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| USART_RX_vect  | SIG_USART_RECV,<br>SIG_UART_RECV   | USART, Rx<br>Complete | AT90PWM3, AT90PWM2, AT90PWM1, ATmega168P, ATmega3250, ATmega325 P, ATmega328P, ATmega3290, ATmega3290P, ATmega48P, ATmega6450, ATmega6490, ATmega8535, ATmega88P, ATmega168, ATmega48, ATmega88, ATtiny2313 |
| USART_TXC_vect | SIG_USART_TRANS,<br>SIG_UART_TRANS | USART, Tx<br>Complete | ATmega16, ATmega32, ATmega323, ATmega8                                                                                                                                                                      |
| USART_TX_vect  | SIG_USART_TRANS,<br>SIG_UART_TRANS | USART, Tx<br>Complete | AT90PWM3, AT90PWM2, AT90PWM1, ATmega168, ATmega328P, ATD ega48P, ATmega8535, ATmega88P, ATmega168, ATmega48, ATmega88, ATtiny2313                                                                           |

We need to program ISR(USART\_RX\_vect)

## ISR(USART\_\_RX\_\_vect)

fscanf uses:

```
int uart_getchar(FILE *stream)
{
    ...
    while (!(UCSR0A & (1<<RXCO)));
    c = UDR0;
    ...
    uart_putchar(c, stream);
    ...
}</pre>
```

- During the while loop other tasks need to wait  $\rightarrow$  fscanf's implementation is blocking
- Need non-blocking code: write a ISR which waits until the character is there

## ISR(USART\_\_RX\_\_vect)

```
// To make sure that the program does not need to wait
// we write our own get_string procedure.
// This requires a circular buffer, index, and a ready flag.
#define r buffer size 50
char r_buffer[r_buffer_size];
int r_index;
volatile char r_ready;
// After getstr(), the USART receive interrupt is enabled.
// The interrupt loads r_buffer, when done r_ready is set to 1.
void getstr(void)
  r_ready = 0; // clear ready flag
  r_{index} = 0; // clear buffer
  UCSROB | = (1 << RXCIEO);
```

The volatile keyword warns the compiler that the declared variable can change at any time without warning and that the compiler shouldn't optimize it away no matter how static it seems

# ISR(USART\_RX\_vect)

- The while loop represents task-based programming, which we repeat throughout the course: While a string is being inputted by the user, other tasks (e.g. Task2) can be executed in parallel
  - No stalling → Efficient execution
  - Modularity as a Computer System Design Principle
- The getstr() can be called in any subroutine, not only in the main while loop
- Inside getstr() r\_ready and r\_index are reset to 0 → Task\_InterpretReadBuffer() may also call getstr() at the end of its code; Here we show the reset explicitly in the main while loop.
- It is often more natural to merge Task() and ResetCond(), especially if the reset should happen at the start of a task rather than at the end

```
int main(void)
  // Initializations etc.
  sei(); // Enable global interrupt
  getstr();
  etc.
  while(1)
    if (r_ready == 1) {Task_InterpretReadBuffer(); getstr();}
    if Condition2 { Task2(...); ResetCond2; }
    etc.
  return 0;
```

## ISR(USART\_RX\_vect)

```
ISR(USART_RX_vect)
 char r char = UDR0;
 // Echo character back out over the system such that a human user
 //can see this
 UDR0 = r_char;
 if (r_{char} = 'r') // compare to the enter character
   if (r_char == 127) // compare to the backspace character
                      // (using '\b' instead of 127 does not work!)
      putchar(' ');  // erase character on screen
      putchar('\b'); // backspace
      --r_index; // erase previously read character in r_buffer
   else
```

```
else
      r\_buffer[r\_index] = r\_char;
      if (r_index < r_buffer_size-1) {r_index++;}</pre>
      else \{r_{index} = 0;\}
  else
    putchar('\n');
                     // new line
    r_buffer[r_index]=0; //strings are terminated
                               // with a 0
    r_ready = 1;
    UCSROB ^=(1<<RXCIEO); // turn off receive
interrupt
```

Implements a simple line editor; we can add more line editing commands from the original uart\_getcha $\chi$ (...)!