# Topic 9: Act, Timers and Interrupts

CAB202. Topic 9 Feras Dayoub

The slides were prepared by Dr. Luis Mejias



### **Outline**

- Debouncing
- Timers
- Interrupts
- Live Coding!

# **Debouncing**

- A switch is a mechanical component, and as a result there is often mechanical noise caused by contact bouncing.
- Contact bouncing is a physical problem that arises when the switch's contacts come together. This connection is not instantaneous and consequently noise is generated.
- This noise causes bouncing, which means that the output from the switch bounces between logical high (i.e. the high voltage) and logical low (i.e. the low voltage which is often ground).
- The results is that a single press appear like multiple presses causing the pin we are interested in reading going rapidly and repeatedly between a circuit's high voltage state and its low voltage state.
- Fortunately this can be overcome through switch debouncing, which is typically performed in software implementations (can also be performed at the hardware level).
- In both software and hardware there are multiple ways of dealing with the problem, as is explained on the second page of this link: <a href="http://www.ganssle.com/debouncing.htm">http://www.ganssle.com/debouncing.htm</a>. The hardware provided for this unit does not have software bouncing in the physical circuitry and consequently debouncing will have to be implemented with software. Switch debouncing is an extremely important component of reading switch measurements in embedded systems.

# Pull-up resistor example





# **Debouncing example**

Code for topic 9 provides versions of a program with/without debouncing implemented.

```
- Folder debouncing. In general, a way of implementing debouncing could be as follows
while (1)
     debounce(); // this function will check whether the pin of the port
            // where the switch is connected change state, and keep track of it
            // with a counter
     if(button down){
        //do something
    _delay_ms(10); // delay execution
Example: debouncing.c
```



#### **Timers are essential**

 The timer systems on the AVR series of Microcontrollers are very essential

 They have a myriad of uses ranging from simple delay intervals right up to complex PWM generation.

#### Timers and the main core

- The AVR timers are very useful as they can run asynchronous to the main AVR core.
- This is a fancy way of saying that the timers are separate circuits on the AVR chip which can run independently of the main program, interacting via the control and count registers, and the timer interrupts.
- Timers can be configured to produce outputs directly to predetermined pins, reducing the processing load on the AVR core.

#### **Timers and clocks**

- Like all digital systems, the timer requires a clock in order to function.
- As each clock pulse increments the timer's counter by one, the timer measures intervals in periods of one on the input frequency.
- This means the smallest amount of time the timer can measure is one period of the incoming clock signal.

#### **Four Timers**

- Our microcontroller has four timer, Timer 0, Timer 1, Timer 3, and Timer 4
- Each timer is associated to a counter and a clock signal.
- The counter is incremented by 1 in every period of the timer's clock signal
- The clock signal can come from
  - The internal system clock
  - An external clock signal

# Timer's registers

 Timers store their values into internal 8 or 16 bit registers, depending on the size of the timer being used.

 These registers can only store a finite number of values, resulting in the need to manage the timer (via prescaling, software extension, etc) so that the interval to be measured fits within the range of the chosen timer.

#### **Timers: Overflow**

- What happens when the range of the timer is exceeded.
  - Does the AVR explode? Does the application crash? Does the timer automatically stop?
- In the event of the timer register exceeding its capacity, it will automatically roll around back to zero and keep counting.
- When this occurs, we say that the timer has "overflowed".

#### **Timers: Overflow**

 When an overflow occurs, a bit is set in one of the timer status registers to indicate to the main application that the event has occurred.

 There is also a corresponding bit which can enable an interrupt to be fired up each time the timer resets back to zero.

# Timers How often does the timer overflow?

- Clock speed (8MHz)
- Prescaler (1, 8, 64, 256, 1024)
- Counter size (8 or 16 bit)
- Timer 0: Normal timer mode, Prescaler of 1024, 8 Bit timer
- Timer Speed
  - = 1 / (Clock Speed/Prescaler)
  - = 1 / (8000000/1024)
  - = 128 micro seconds (sometime also called timer resolution)
- Timer Overflow Speed
  - = (Timer Speed \* 256)
  - = 32768 micro seconds

#### **Timers**

- Four timers (the atmega32U4 doesn't have a timer 2)
  - Timers 1 and 3 are 16 bits
  - Timer 0, is a 8 bits
  - Timer 4, is a 10 bit
- Let's focus on Timer 0, its register are
  - TCCR0A
  - TCCR0B
  - TCNT0
  - OCR0A
  - OCR0B
  - TIMSK0
  - TIFR0
- They can be used set the behavior of the timer, override direction of I/O pins, clock, prescaler, and also to set specific triggers

# Timer Registers: TCCR0B setting timer pre-scaler



#### Bits 2:0 – CS02:0: Clock Select

The three Clock Select bits select the clock source to be used by the Timer/Counter.

Table 13-9. Clock Select Bit Description

| CS02 | CS01 | CS00 | Description                                             |
|------|------|------|---------------------------------------------------------|
| 0    | 0    | 0    | No clock source (Timer/Counter stopped)                 |
| 0    | 0    | 1    | clk <sub>I/O</sub> /(No prescaling)                     |
| 0    | 1    | 0    | clk <sub>I/O</sub> /8 (From prescaler)                  |
| 0    | 1    | 1    | clk <sub>I/O</sub> /64 (From prescaler)                 |
| 1    | 0    | 0    | clk <sub>I/O</sub> /256 (From prescaler)                |
| 1    | 0    | 1    | clk <sub>I/O</sub> /1024 (From prescaler)               |
| 1    | 1    | 0    | External clock source on T0 pin. Clock on falling edge. |
| 1    | 1    | 1    | External clock source on T0 pin. Clock on rising edge.  |

# Timers setting a timer

```
void TimerInit(void)
   //Set to Normal Timer Mode using TCCR0B
   TCCR0B \&= \sim ((1 << WGM02));
   //Set Prescaler using TCCR0B, using Clock Speed find timer speed = 1/(Clock
Speed/Prescaler)
   //Prescaler = 1024
   //Timer Speed = 128 microseconds
   //Timer Overflow Speed = 32640 micro seconds (Timer Speed * 255) - (255 since 8-bit timer)
   TCCR0B = (1 < CS02) (1 < CS00);
   TCCR0B &= \sim((1<<CS01));
Example: simpleTimer.c
```

### **Interrupts**

- In most microcontrollers, there is something called interrupt. This interrupt can be fired up whenever certain condition is met.
- Now whenever an interrupt is fired, the AVR stops and saves its execution of the main routine, attends to the interrupt call (by executing a special routine, called the Interrupt Service Routine, ISR) and once it is done with it, returns to the main routine and continues executing it.

# **Polling**

- In all of the solutions to the pracs throughout this semester, the methods you've have been using are what is known as polling.
- A polling method is one which constantly loops over the same code that checks each iteration to see if a particular state has changed (i.e. performing a desired action by continuously checking if a button has been pressed).
- In other words, when we are polling we are waiting for a specific event. This action is considered blocking due to the fact that no other code can be run until this condition is met.
- Consequently, this blocking code can cause problems in larger embedded systems where you have multiple inputs and outputs. It is in scenarios like these where the distinct advantage of interrupts is apparent.

CRICOS No. 000213J

# **Polling vs. Interrupts**

```
while (1)
    check_device_status();
    if(service_required)
        service_routine();
    normal_execution();
```



### Halts, Run, Return

- Interrupts are a crucial part of writing code for embedded systems. In performing the interrupt, the microcontroller goes through the following three steps:
  - 1. Halts the current process (noting where it is in this process)
  - 2. Runs the interrupt code until it completes
  - Returns back to the original process and continue from where it was before the interrupt

## Interrupt service routine

• The interrupt code is called the interrupt service routine (ISR). The syntax for code that creates an ISR is similar to that of a function (except it does not need a declaration – only an implementation). An example of an ISR is shown below for the USART receive complete interrupt event (you must include avr/interrupt.h to run any interrupt related code):

```
ISR(USART_RXC_vect) {
    // Code to be executed within the interrupt routine
}
```

- You can think of the ISR as basically an isolated section of code which will get called anytime an event occurs.
- As already discussed, there are many different types of interrupts. Each type has as
  its own interrupt vector. In the above example, the interrupt vector is the
  USART\_RXC\_vect part. It is this part of the ISR declaration code that would need to
  be changed for a different interrupt event.

CRICOS No. 000213J

## **Interrupts**

- There are three important conditions that must be met for the ISR to be called and executed correctly:
  - 1. The enable bit for global interrupts must be set. This allows the microcontroller to process interrupts via ISRs when set, and prevents them from running when cleared. It defaults to being cleared on power up, so we need to set it by using the sei() utility function. Conversely, you can clear it by using the cli() utility function.
  - 2. The individual interrupt source's enable bit must explicitly be set. Each hardware interrupt source has its own separate interrupt enable bit (this resides in the related peripheral's control registers).
  - 3. The condition for the interrupt must be met. For example, a character must have been received through USART for the USART receive complete (USART\_RXC) interrupt to be executed.

CRICOS No. 000213J

Table 9-1. Reset and Interrupt Vectors

| Vector<br>No. | Program<br>Address <sup>(2)</sup>           | Source                         | Interrupt Definition                                                              |  |  |  |  |  |
|---------------|---------------------------------------------|--------------------------------|-----------------------------------------------------------------------------------|--|--|--|--|--|
| 1             | \$0000 <sup>(1)</sup>                       | RESET                          | External Pin, Power-on Reset, Brown-out Reset, Watchdog Reset, and JTAG AVR Reset |  |  |  |  |  |
| 2             | \$0002                                      | INT0                           | External Interrupt Request 0                                                      |  |  |  |  |  |
| 3             | \$0004                                      | INT1                           | External Interrupt Request 1                                                      |  |  |  |  |  |
| 4             | \$0006                                      | INT2                           | External Interrupt Request 2                                                      |  |  |  |  |  |
| 5             | \$0008                                      | INT3                           | External Interrupt Request 3                                                      |  |  |  |  |  |
| 6             | \$000A                                      | Reserved                       | Reserved Reserved                                                                 |  |  |  |  |  |
| 7             | \$000C                                      | Reserved                       |                                                                                   |  |  |  |  |  |
| 8             | \$000E                                      | INT6                           | External Interrupt Request 6                                                      |  |  |  |  |  |
| 9             | \$0010                                      | Reserved                       | Reserved                                                                          |  |  |  |  |  |
| 10            | \$0012                                      | Pin Change Interrupt Request 0 |                                                                                   |  |  |  |  |  |
| 11            | \$0014                                      | USB General                    | USB General Interrupt request                                                     |  |  |  |  |  |
| 12            | \$0016                                      | USB Endpoint                   | USB Endpoint Interrupt request                                                    |  |  |  |  |  |
| 13            | \$0018                                      | WDT                            | Watchdog Time-out Interrupt                                                       |  |  |  |  |  |
| 14            | \$001A                                      | Reserved                       | Reserved                                                                          |  |  |  |  |  |
| 15            | \$001C                                      | Reserved                       | Reserved                                                                          |  |  |  |  |  |
| 16            | \$001E                                      | Reserved                       | Reserved                                                                          |  |  |  |  |  |
| 17            | \$0020                                      | TIMER1 CAPT                    | Timer/Counter1 Capture Event                                                      |  |  |  |  |  |
| 18            | \$0022                                      | TIMER1 COMPA                   | Timer/Counter1 Compare Match A                                                    |  |  |  |  |  |
| 19            | \$0024                                      | TIMER1 COMPB                   | Timer/Counter1 Compare Match B                                                    |  |  |  |  |  |
| 20            | \$0026                                      | TIMER1 COMPC                   | Timer/Counter1 Compare Match C                                                    |  |  |  |  |  |
| 21            | \$0028 TIMER1 OVF Timer/Counter1 Overflow   |                                | Timer/Counter1 Overflow                                                           |  |  |  |  |  |
| 22            | \$002A                                      | TIMER0 COMPA                   | Timer/Counter0 Compare Match A                                                    |  |  |  |  |  |
| 23            | \$002C                                      | TIMER0 COMPB                   | Timer/Counter0 Compare match B                                                    |  |  |  |  |  |
| 24            | \$002E                                      | TIMER0 OVF                     | Timer/Counter0 Overflow                                                           |  |  |  |  |  |
| 25            | \$0030                                      | SPI (STC)                      | SPI Serial Transfer Complete                                                      |  |  |  |  |  |
| 26            | 26 \$0032 USART1 RX<br>27 \$0034 USART1 UDF |                                | USART1 Rx Complete                                                                |  |  |  |  |  |
| 27            |                                             |                                | USART1 Data Register Empty                                                        |  |  |  |  |  |
| 28            | \$0036                                      | USART1TX                       | USART1 Tx Complete                                                                |  |  |  |  |  |
| 29            | \$0038                                      | ANALOG COMP                    | Analog Comparator                                                                 |  |  |  |  |  |

1 reset interrupt 5 external interrupts 1 Pin Change interrupt 8 timer interrupts serial port interrupts

# Interrupts Problem

### Trigger an interrupt when a timer overflows

#### Steps:

- 1. Set up the timer with correct prescaler.
- 2. Turn on interrupts
- 3. Write the Interrupt Service Routine that is called when the timer overflows. Data shared between the ISR and your main program **must be both volatile and global in scope** in the C language. i. e

volatile int overflow\_count;

# Set up timer 0 overflow interrupt

|           |        |              | Entornal interrupt (request)                                                                                                         |  |  |  |  |  |
|-----------|--------|--------------|--------------------------------------------------------------------------------------------------------------------------------------|--|--|--|--|--|
| 9         | \$0010 | Reserved     | Reserved                                                                                                                             |  |  |  |  |  |
| 10        | \$0012 | PCINT0       | Pin Change Interrupt Request 0  USB General Interrupt request  USB Endpoint Interrupt request  Watchdog Time-out Interrupt  Reserved |  |  |  |  |  |
| 11        | \$0014 | USB General  |                                                                                                                                      |  |  |  |  |  |
| 12        | \$0016 | USB Endpoint |                                                                                                                                      |  |  |  |  |  |
| 13        | \$0018 | WDT          |                                                                                                                                      |  |  |  |  |  |
| 14        | \$001A | Reserved     |                                                                                                                                      |  |  |  |  |  |
| 15        | \$001C | Reserved     | Reserved                                                                                                                             |  |  |  |  |  |
| 16        | \$001E | Reserved     | Reserved                                                                                                                             |  |  |  |  |  |
| 17        | \$0020 | TIMER1 CAPT  | Timer/Counter1 Capture Event                                                                                                         |  |  |  |  |  |
| 18 \$0022 |        | TIMER1 COMPA | Timer/Counter1 Compare Match A                                                                                                       |  |  |  |  |  |
| 19        | \$0024 | TIMER1 COMPB | Timer/Counter1 Compare Match B                                                                                                       |  |  |  |  |  |
| 20        | \$0026 | TIMER1 COMPC | Timer/Counter1 Compare Match C                                                                                                       |  |  |  |  |  |
| 21        | \$0028 | TIMER1 OVF   | Timer/Counter1 Overflow                                                                                                              |  |  |  |  |  |
| 22        | \$002A | TIMERO COMPA | Timer/Counter0 Compare Match A                                                                                                       |  |  |  |  |  |
| 23        | \$002C | TIMER0 COMPB | Timer/Counter0 Compare match B                                                                                                       |  |  |  |  |  |
| 24        | \$002E | TIMER0 OVF   | Timer/Counter0 Overflow                                                                                                              |  |  |  |  |  |
| 25        | \$0030 | SPI (STC)    | SPI Serial Transfer Complete                                                                                                         |  |  |  |  |  |
| 26        | \$0032 | USART1 RX    | USART1 Rx Complete                                                                                                                   |  |  |  |  |  |
| 27        | \$0034 | USART1 UDRE  | USART1 Data Register Empty                                                                                                           |  |  |  |  |  |
| 28        | \$0036 | USART1TX     | USART1 Tx Complete                                                                                                                   |  |  |  |  |  |
| 29        | \$0038 | ANALOG COMP  | Analog Comparator                                                                                                                    |  |  |  |  |  |
|           |        | •            |                                                                                                                                      |  |  |  |  |  |

# **Set up Overflow Interrupt Enable**

#### 13.8.6 Timer/Counter Interrupt Mask Register – TIMSK0

| Bit           | 7 | 6 | 5 | 4 | 3 | 2      | 1      | 0     | _      |
|---------------|---|---|---|---|---|--------|--------|-------|--------|
|               | - | - | - | - | _ | OCIE0B | OCIE0A | TOIE0 | TIMSK0 |
| Read/Write    | R | R | R | R | R | R/W    | R/W    | R/W   |        |
| Initial Value | 0 | 0 | 0 | 0 | 0 | 0      | 0      | 0     |        |

• Bits 7..3, 0 - Res: Reserved Bits

These bits are reserved bits and will always read as zero.

 $TIMSK0 = (1 \ll TOIE0);$ 

#### • Bit 2 - OCIE0B: Timer/Counter Output Compare Match B Interrupt Enable

When the OCIE0B bit is written to one, and the I-bit in the Status Register is set, the Timer/Counter Compare Match B interrupt is enabled. The corresponding interrupt is executed if a Compare Match in Timer/Counter occurs, i.e., when the OCF0B bit is set in the Timer/Counter Interrupt Flag Register - TIFR0.

#### Bit 1 – OCIE0A: Timer/Counter0 Output Compare Match A Interrupt Enable

When the OCIE0A bit is written to one, and the I-bit in the Status Register is set, the Timer/Counter0 Compare Match A interrupt is enabled. The corresponding interrupt is executed if a Compare Match in Timer/Counter0 occurs, i.e., when the OCF0A bit is set in the Timer/Counter 0 Interrupt Flag Register – TIFR0.

#### • Bit 0 – TOIE0: Timer/Counter0 Overflow Interrupt Enable

When the TOIE0 bit is written to one, and the I-bit in the Status Register is set, the Timer/Counter0 Overflow interrupt is enabled. The corresponding interrupt is executed if an overflow in Timer/Counter0 occurs, i.e., when the TOV0 bit is set in the Timer/Counter 0 Interrupt Flag Register - TIFR0.

# Set up timer 0 and overflow interrupt

```
void InterruptInit(void)
  //Set to Normal Timer Mode using TCCR0B
  TCCR0B &= \sim((1<<WGM02));
  //Set Prescaler using TCCR0B, using Clock Speed find timer speed = 1/(ClockSpeed/Prescaler)
  //Prescaler = 1024
  //Timer Speed = 128 micro seconds
  //Timer Overflow Speed = 32640 micro seconds (Timer Speed * 255) - (255 since 8-bit timer)
  TCCR0B = (1 < CS02) (1 < CS00);
  TCCR0B &= \sim((1<<CS01));
  //enable overflow interrupt
  TIMSK0 = (1 \ll TOIE0);
  //Ensure to enable global interrupts as well.
 sei();
Example: TimerInterrupt.c
```

## **Summary**

- Debouncing
  - Quite important when working with buttons
  - Can be implemented using interrupts
- Timers and Interrupts
  - Needed to generate events or execute part of your program with good time predictability.
- Example code on Blackboard