Yay! More interrupts. Let's see how to set up timer interrupts in an Arduino.
Previously, we looked at using registers directly in Arduino and setting up external interrupts. This time, we configure a timer interrupt to toggle an LED every 0.5 seconds. That's right; we're going to redo blinky the hard way.
The ATmega328P has three timers that continually increment. Note that Arduino uses them for various functions, which means you probably don't want to use those functions if you plan to mess with the timers. On the UNO and RedBoard, you will find:
Name | Size | Possible Interrupts | Uses in Arduino |
---|---|---|---|
Timer0 | 8 bits (0 - 255) |
Compare Match Overflow |
delay(), millis(), micros() analogWrite() pins 5, 6 |
Timer1 | 16 bits (0 - 65,535) |
Compare Match Overflow Input Capture |
Servo functions analogWrite() pins 9, 10 |
Timer2 | 8 bits (0 - 255) |
Compare Match Overflow |
tone() analogWrite() pins 3, 11 |
Each of these timers can be configured for one or more interrupts:
To help make timing longer events easier (especially when some of the timers can only count up to 255), a prescaler can be used. A prescaler, in effect, divides the system clock to make something operate more slowly. For example, here is Timer1 counting up to 15 with no prescaler. You can see how Timer1 increments on every rising edge of the system clock.
If we set a prescaler of 8, we can see how the timer counts up at a much slower pace. The timer only increments once for every 8 counts of the system clock.
Knowing that we have a 16MHz clock on the Arduino UNO and RedBoard, we can calculate what kind of prescaler we need and to what value we need to count to trigger an interrupt every 500ms. Timer1 supports four different prescalers in addition to no prescaler (directly connected to the system clock): 8, 64, 256, 1024. We can find these by looking at the TCCR1B register description in the ATmega328P datasheet. We can multiply the desired delay by the clock speed and divide that answer by the prescaler to get the number of required timer counts. If we use a prescaler of 256, we see that we need 31,250 counts:
We set the prescaler to 256 and load 31,250 into the OCR1A register. Don't forget to set the output compare A interrupt enable bit in TIMSK1 and enable global interrupts! Now, whenever Timer1 reaches 31,250, a TIMER1_COMPA interrupt will occur. Create your ISR to toggle the LED and reset the timer.
// Pins
const int led_pin = PB5;
// Counter and compare values
const uint16_t t1_load = 0;
const uint16_t t1_comp = 31250;
void setup() {
// Set LED pin to be output
DDRB |= (1 << led_pin);
// Reset Timer1 Control Reg A
TCCR1A = 0;
// Set to prescaler of 256
TCCR1B |= (1 << CS12);
TCCR1B &= ~(1 << CS11);
TCCR1B &= ~(1 << CS10);
// Reset Timer1 and set compare value
TCNT1 = t1_load;
OCR1A = t1_comp;
// Enable Timer1 overflow interrupt
TIMSK1 = (1 << OCIE1A);
// Enable global interrupts
sei();
}
void loop() {
delay(500);
}
ISR(TIMER1_COMPA_vect) {
TCNT1 = t1_load;
PORTB ^= (1 << led_pin);
}
Timers are extremely powerful and useful microcontroller functions. They allow you to sample, control hardware or measure signals at very precise intervals.
Outside of the already given Arduino functions (delay(), millis(), servo, tone, etc.), how else have you used timers in your microcontroller projects?