SlideShare a Scribd company logo
Arduino 101: Timers and Interrupts

 by RobotFreak
 Collected by 69 users
Arduino, timer, pwm, counter, timers, interrupt
Tip/walkthrough
By RobotFreak @ August 7, 2011
How to use timer and interrupts on Arduino boards.
 Programming language: C++, Arduino
Attachment Size
Arduino101-Timers.zip 32.9 KB
ArduinoTimer101.zip 2.74 KB
Update 07/24/2013 04/24/2013
 Example 3 has been updated to work with Arduino v1.x.
 Added ArduinoTimer101.zip examples source code for Arduino v1.x .
This tutorial shows the use of timers and interrupts for Arduino boards. As Arduino
programmer you will have used timers and interrupts without knowledge, bcause all the low
level hardware stuff is hidden by the Arduino API. Many Arduino functions uses timers, for
example the time functions: delay(), millis() and micros() and delayMicroseconds(). The
PWM functions analogWrite() uses timers, as the tone() and the noTone() function does.
Even the Servo libraryuses timers and interrupts.
What is a timer?
A timer or to be more precise a timer / counter is a piece of hardware builtin the Arduino
controller (other controllers have timer hardware, too). It is like a clock, and can be used to
measure time events.
The timer can be programmed by some special registers. You can configure the prescaler
for the timer, or the mode of operation and many other things.
The controller of the Arduino is the Atmel AVR ATmega168 or the ATmega328. These chips
are pin compatible and only differ in the size of internal memory. Both have 3 timers, called
timer0, timer1 and timer2. Timer0 and timer2 are 8bit timer, where timer1 is a 16bit timer.
The most important difference between 8bit and 16bit timer is the timer resolution. 8bits
means 256 values where 16bit means 65536 values for higher resolution.
The controller for the Arduino Mega series is the Atmel AVR ATmega1280 or the
ATmega2560. Also identical only differs in memory size. These controllers have 6 timers.
Timer 0, timer1 and timer2 are identical to the ATmega168/328. The timer3, timer4 and
timer5 are all 16bit timers, similar to timer1.
All timers depends on the system clock of your Arduino system. Normally the system clock is
16MHz, but for the Arduino Pro 3,3V it is 8Mhz. So be careful when writing your own timer
functions.
The timer hardware can be configured with some special timer registers. In the Arduino
firmware all timers were configured to a 1kHz frequency and interrupts are gerally enabled.
Timer0:
Timer0 is a 8bit timer.
In the Arduino world timer0 is been used for the timer functions,
like delay(), millis() and micros(). If you change timer0 registers, this may influence the
Arduino timer function. So you should know what you are doing.
Timer1:
Timer1 is a 16bit timer.
In the Arduino world the Servo library uses timer1 on Arduino Uno (timer5 on Arduino Mega).
Timer2:
Timer2 is a 8bit timer like timer0.
In the Arduino work the tone() function uses timer2.
Timer3, Timer4, Timer5:
Timer 3,4,5 are only available on Arduino Mega boards. These timers are all 16bit timers.
Timer Register
You can change the Timer behaviour through the timer register. The most important timer
registers are:
TCCRx - Timer/Counter Control Register. The prescaler can be configured here.
TCNTx - Timer/Counter Register. The actual timer value is stored here.
OCRx - Output Compare Register
ICRx - Input Capture Register (only for 16bit timer)
TIMSKx - Timer/Counter Interrupt Mask Register. To enable/disable timer interrupts.
TIFRx - Timer/Counter Interrupt Flag Register. Indicates a pending timer interrupt.
Clock select and timer frequency
Different clock sources can be selected for each timer independently. To calculate the timer frequency
(for example 2Hz using timer1) you will need:
1. CPU frequency 16Mhz for Arduino
2. maximum timer counter value (256 for 8bit, 65536 for 16bit timer)
3. Divide CPU frequency through the choosen prescaler (16000000 / 256 = 62500)
4. Divide result through the desired frequency (62500 / 2Hz = 31250)
5. Verify the result against the maximum timer counter value (31250 < 65536 success) if fail,
choose bigger prescaler.
Timer modes
Timers can be configured in different modes.
PWM mode. Pulth width modulation mode. the OCxy outputs are used to generate PWM signals
CTC mode. Clear timer on compare match. When the timer counter reaches the compare match
register, the timer will be cleared
What is an interrupt?
The program running on a controller is normally running sequentially instruction by
instruction. An interrupt is an external event that interrupts the running program and runs a
special interrupt service routine (ISR). After the ISR has been finished, the running program
is continued with the next instruction. Instruction means a single machine instruction, not a
line of C or C++ code.
Before an pending interrupt will be able to call a ISR the following conditions must be true:
 Interrupts must be generally enabled
 the according Interrupt mask must be enabled
Interrupts can generally enabled / disabled with the function interrupts() / noInterrupts(). By
default in the Arduino firmware interrupts are enabled. Interrupt masks are enabled /
disabled by setting / clearing bits in the Interrupt mask register (TIMSKx).
When an interrupt occurs, a flag in the interrupt flag register (TIFRx) is been set. This
interrupt will be automatically cleared when entering the ISR or by manually clearing the bit
in the interrupt flag register.
The Arduino functions attachInterrupt() and detachInterrupt() can only be used for external
interrupt pins. These are different interrupt sources, not discussed here.
Timer interrupts
A timer can generate different types of interrupts. The register and bit definitions can be
found in the processor data sheet (Atmega328 or Atmega2560) and in the I/O definition
header file (iomx8.h for Arduino, iomxx0_1.h for Arduino Mega in the
hardware/tools/avr/include/avr folder). The suffix x stands for the timer number (0..5), the
suffix y stands for the output number (A,B,C), for example TIMSK1 (timer1 interrupt mask
register) or OCR2A (timer2 output compare register A).
Timer Overflow:
Timer overflow means the timer has reached is limit value. When a timer overflow interrupt
occurs, the timer overflow bit TOVx will be set in the interrupt flag register TIFRx. When the
timer overflow interrupt enable bit TOIEx in the interrupt mask register TIMSKx is set, the
timer overflow interrupt service routine ISR(TIMERx_OVF_vect) will be called.
Output Compare Match:
When a output compare match interrupt occurs, the OCFxy flag will be set in the interrupt
flag register TIFRx . When the output compare interrupt enable bit OCIExy in the interrupt
mask register TIMSKx is set, the output compare match interrupt service
ISR(TIMERx_COMPy_vect) routine will be called.
Timer Input Capture:
When a timer input capture interrupt occurs, the input capture flag bit ICFx will be set in the
interrupt flag register TIFRx. When the input capture interrupt enable bit ICIEx in the
interrupt mask register TIMSKx is set, the timer input capture interrupt service routine
ISR(TIMERx_CAPT_vect) will be called.
PWM and timer
There is fixed relation between the timers and the PWM capable outputs. When you look in
the data sheet or the pinout of the processor these PWM capable pins have names like
OCRxA, OCRxB or OCRxC (where x means the timer number 0..5). The PWM functionality
is often shared with other pin functionality.
The Arduino has 3Timers and 6 PWM output pins. The relation between timers and PWM
outputs is:
Pins 5 and 6: controlled by timer0
Pins 9 and 10: controlled by timer1
Pins 11 and 3: controlled by timer2
On the Arduino Mega we have 6 timers and 15 PWMoutputs:
Pins 4 and 13: controlled by timer0
Pins 11 and 12: controlled by timer1
Pins 9 and10: controlled by timer2
Pin 2, 3 and 5: controlled by timer 3
Pin 6, 7 and 8: controlled by timer 4
Pin 46, 45 and 44:: controlled by timer 5
Usefull 3rd party libraries
Some 3rd party libraries exists, that uses timers:
 Ken Shirrifs IR library. Using timer2. Send and receive any kind of IR remote signals
like RC5, RC6, Sony
 Timer1 and Timer3 library. Using timer1 or tiner3. The easy way to write your own
timer interupt service routines.
I have ported these libraries to different timers for Arduino Mega. All ported libraries can be
found in the attached file.
Pitfalls
There exists some pitfalls you may encounter, when programming your Arduino and make
use of functions or libraries that uses timers.
 Servo Library uses Timer1. You can’t use PWM on Pin 9, 10 when you use the Servo
Library on an Arduino. For Arduino Mega it is a bit more difficult. The timer needed
depends on the number of servos. Each timer can handle 12 servos. For the first 12
servos timer 5 will be used (loosing PWM on Pin 44,45,46). For 24 Servos timer 5
and 1 will be used (loosing PWM on Pin 11,12,44,45,46).. For 36 servos timer 5, 1
and 3 will be used (loosing PWM on Pin 2,3,5,11,12,44,45,46).. For 48 servos all
16bit timers 5,1,3 and 4 will be used (loosing all PWM pins).
 Pin 11 has shared functionality PWM and MOSI. MOSI is needed for the SPI
interface, You can’t use PWMon Pin 11 and the SPI interface at the same time on
Arduino. On the Arduino Mega the SPI pins are on different pins.
 tone() function uses at least timer2. You can’t use PWM on Pin 3,11 when you use
the tone() function an Arduino and Pin 9,10 on Arduino Mega.
Examples
Time for some examples.
Blinking LED with compare match interrupt
The first example uses the timer1 in CTC mode and the compare match interrupt to toggle a
LED. The timer is configured for a frequency of 2Hz. The LED is toggled in the interrupt
service routine.
/* Arduino 101: timer and interrupts
1: Timer1 compare match interrupt example
more infos: http://www.letmakerobots.com/node/28278
created by RobotFreak
*/
#define ledPin 13
void setup()
{
pinMode(ledPin, OUTPUT);
// initialize timer1
noInterrupts(); // disable all interrupts
TCCR1A = 0;
TCCR1B = 0;
TCNT1 = 0;
OCR1A = 31250; // compare match register 16MHz/256/2Hz
TCCR1B |= (1 << WGM12); // CTC mode
TCCR1B |= (1 << CS12); // 256 prescaler
TIMSK1 |= (1 << OCIE1A); // enable timer compare interrupt
interrupts(); // enable all interrupts
}
ISR(TIMER1_COMPA_vect) // timer compare interrupt service routine
{
digitalWrite(ledPin, digitalRead(ledPin) ^ 1); // toggle LED pin
}
void loop()
{
// your program here...
}
Blinking LED with timer overflow interrupt
same example like before but now we use the timer overflow interrupt. In this case timer1 is
running in normal mode.
The timer must be preloaded every time in the interrupt service routine.
/*
* Arduino 101: timer and interrupts
* 2: Timer1 overflow interrupt example
* more infos: http://www.letmakerobots.com/node/28278
* created by RobotFreak
*/
#define ledPin 13
void setup()
{
pinMode(ledPin, OUTPUT);
// initialize timer1
noInterrupts(); // disable all interrupts
TCCR1A = 0;
TCCR1B = 0;
TCNT1 = 34286; // preload timer 65536-16MHz/256/2Hz
TCCR1B |= (1 << CS12); // 256 prescaler
TIMSK1 |= (1 << TOIE1); // enable timer overflow interrupt
interrupts(); // enable all interrupts
}
ISR(TIMER1_OVF_vect) // interrupt service routine that wraps a user
defined function supplied by attachInterrupt
{
TCNT1 = 34286; // preload timer
digitalWrite(ledPin, digitalRead(ledPin) ^ 1);
}
void loop()
{
// your program here...
}
Reading quadrature encoders with a timer
The next example is part of my Ardubot project. It uses timer2 and the compare match
interrupt to read the encoder inputs. Timer2 is initialized by default to a frequency of 1kHz
(1ms period). In the interrupt service routine the state of all encoder pins is read and a state
machine is used to eliminate false readings. Using the timer interrupt is much easier to
handle than using 4 input change interrupts.
The signals of a quadrature encoder is a 2bit Gray code. Only 1 bit is changing from state to state. A
state machine is perfect to check the signal and count the encoder ticks. The timer must be fast
enough, to recognize each state change. For the Pololu wheel encoders used here, the 1ms timer is
fast enough.
The following example has been modified to work with Arduino V1.x
/*
/*
* Arduino 101: timer and interrupts
* 3: Timer2 compare interrupt example. Quadrature Encoder
* more infos: http://www.letmakerobots.com/node/28278
* created by RobotFreak
*
* Credits:
* based on code from Peter Dannegger
* http://www.mikrocontroller.net/articles/Drehgeber
*/
#if ARDUINO >= 100
#include "Arduino.h"
#else
#include "WConstants.h"
#endif
// Encoder Pins
#define encLtA 2
#define encLtB 3
#define encRtA 11
#define encRtB 12
#define ledPin 13
#define LT_PHASE_A digitalRead(encLtA)
#define LT_PHASE_B digitalRead(encLtB)
#define RT_PHASE_A digitalRead(encRtA)
#define RT_PHASE_B digitalRead(encRtB)
static volatile int8_t encDeltaLt, encDeltaRt;
static int8_t lastLt, lastRt;
int encLt, encRt;
ISR( TIMER2_COMPA_vect )
{
int8_t val, diff;
digitalWrite(ledPin, HIGH); // toggle LED pin
val = 0;
if( LT_PHASE_A )
val = 3;
if( LT_PHASE_B )
val ^= 1; // convert gray to binary
diff = lastLt - val; // difference last - new
if( diff & 1 ){ // bit 0 = value (1)
lastLt = val; // store new as next last
encDeltaLt += (diff & 2) - 1; // bit 1 = direction (+/-)
}
val = 0;
if( RT_PHASE_A )
val = 3;
if( RT_PHASE_B )
val ^= 1; // convert gray to binary
diff = lastRt - val; // difference last - new
if( diff & 1 ){ // bit 0 = value (1)
lastRt = val; // store new as next last
encDeltaRt += (diff & 2) - 1; // bit 1 = direction (+/-)
}
digitalWrite(ledPin, LOW); // toggle LED pin
}
void QuadratureEncoderInit(void)
{
int8_t val;
cli();
TIMSK2 |= (1<<OCIE2A);
sei();
pinMode(encLtA, INPUT);
pinMode(encRtA, INPUT);
pinMode(encLtB, INPUT);
pinMode(encRtB, INPUT);
val=0;
if (LT_PHASE_A)
val = 3;
if (LT_PHASE_B)
val ^= 1;
lastLt = val;
encDeltaLt = 0;
val=0;
if (RT_PHASE_A)
val = 3;
if (RT_PHASE_B)
val ^= 1;
lastRt = val;
encDeltaRt = 0;
encLt = 0;
encRt = 0;
}
int8_t QuadratureEncoderReadLt( void ) // read single step
encoders
{
int8_t val;
cli();
val = encDeltaLt;
encDeltaLt = 0;
sei();
return val; // counts since last call
}
int8_t QuadratureEncoderReadRt( void ) // read single step
encoders
{
int8_t val;
cli();
val = encDeltaRt;
encDeltaRt = 0;
sei();
return val; // counts since last call
}
void setup()
{
Serial.begin(38400);
pinMode(ledPin, OUTPUT);
QuadratureEncoderInit();
}
void loop()
{
encLt += QuadratureEncoderReadLt();
encRt += QuadratureEncoderReadRt();
Serial.print("Lt: ");
Serial.print(encLt, DEC);
Serial.print(" Rt: ");
Serial.println(encRt, DEC);
delay(1000);
}

More Related Content

PDF
Using Timer2 in Microchip MCUs
PDF
Handling Interrupts in Microchip MCUs
PDF
Exercises with timers and UART
PDF
Itsp documentation quadcopter flight controller based on kalman filters
PDF
UART MCU
PPT
8 interrupt 8051
PDF
Practical reverse engineering and exploit development for AVR-based Embedded ...
Using Timer2 in Microchip MCUs
Handling Interrupts in Microchip MCUs
Exercises with timers and UART
Itsp documentation quadcopter flight controller based on kalman filters
UART MCU
8 interrupt 8051
Practical reverse engineering and exploit development for AVR-based Embedded ...

What's hot (20)

PDF
Interrupt
PPT
8051 Inturrpt
PDF
Advanced view of atmega microcontroller projects list at mega32 avr
PPT
how to generate sms
PDF
Introduction to Arduino
PPTX
PIC18 TIMER PROGRAMMING IN ASSEMBLY AND C
PPTX
37471656 interrupts
PDF
Embedded system course projects - Arduino Course
PPT
W10: Interrupts
PDF
PDF
The IO concept
PPTX
Contiki os timer tutorial
PDF
Meta88full
PPT
Contiki introduction I.
PPTX
Distance measurement using Ultrasonic sensor on Arduino Uno
PPTX
Rig nitc [autosaved] (copy)
DOCX
How to write timings and delays
PDF
Arduino arduino boardnano
PDF
Advanced view of atmega microcontroller projects list at mega32 avr
DOCX
All about ir arduino - cool
Interrupt
8051 Inturrpt
Advanced view of atmega microcontroller projects list at mega32 avr
how to generate sms
Introduction to Arduino
PIC18 TIMER PROGRAMMING IN ASSEMBLY AND C
37471656 interrupts
Embedded system course projects - Arduino Course
W10: Interrupts
The IO concept
Contiki os timer tutorial
Meta88full
Contiki introduction I.
Distance measurement using Ultrasonic sensor on Arduino Uno
Rig nitc [autosaved] (copy)
How to write timings and delays
Arduino arduino boardnano
Advanced view of atmega microcontroller projects list at mega32 avr
All about ir arduino - cool
Ad

Similar to Arduino 101 (20)

PPTX
timer counter (1).pptx
PDF
Avr timers
PDF
Lecture7
PPTX
Timer & Interrupt Atmega16
PPTX
Arduino intro.pptx
PPTX
AVRTIMER.pptx
PDF
timers.pdf
PDF
AVR_Course_Day7 timers counters and interrupt programming
PPTX
Embedded systems and robotics by scmandota
PPT
UNCC-IESLecture13 - Timers and Event Counters.ppt
PPTX
Unit 3 timer and counter and there application .pptx
PPTX
Timers done by Priyanga KR
PPTX
8051 MICROCONTROLLER TIMER AND ITS APPLICATIONS
PDF
Arduino reference
PPTX
8051 Timers and Counters
PDF
Lecture6
PPTX
5th unit embedded system and iot design timer and controller
PPTX
Arduino.pptx
PPTX
KTU_Microprocessor and Microcontrollers_Module2
PDF
AVR arduino dasar
timer counter (1).pptx
Avr timers
Lecture7
Timer & Interrupt Atmega16
Arduino intro.pptx
AVRTIMER.pptx
timers.pdf
AVR_Course_Day7 timers counters and interrupt programming
Embedded systems and robotics by scmandota
UNCC-IESLecture13 - Timers and Event Counters.ppt
Unit 3 timer and counter and there application .pptx
Timers done by Priyanga KR
8051 MICROCONTROLLER TIMER AND ITS APPLICATIONS
Arduino reference
8051 Timers and Counters
Lecture6
5th unit embedded system and iot design timer and controller
Arduino.pptx
KTU_Microprocessor and Microcontrollers_Module2
AVR arduino dasar
Ad

More from josnihmurni2907 (20)

DOCX
Twin wheeler modified for arduino simplified serial protocol to sabertooth v22
DOCX
Twin wheeler modified for arduino simplified serial protocol to sabertooth v21
DOCX
Twin wheeler modified for arduino simplified serial protocol to sabertooth v2
DOCX
Simple timer code for arduino
DOCX
Michael kontopoulos
DOCX
Michael kontopoulo1s
DOCX
Define ba1
DOCX
Define ba
DOCX
Call and message using arduino and gsm module
DOCX
PDF
140813560 nota-kimia-tingkatan-4
DOC
1.funtions (1)
PDF
132944997 rancangan-tahunan-mm-t5-2013
DOCX
Agihan kertas 1 dan kertas 2
DOCX
Ciri pemimpin
DOCX
Cara menghilangkan rasa ngantuk di kelas
DOCX
Bercakap mengenai nikmat allah s
DOCX
Bandar purba dalam tasik di china
DOCX
12 kaum yang telah allah swt binasakan
PPTX
Smkts 2015 p1
Twin wheeler modified for arduino simplified serial protocol to sabertooth v22
Twin wheeler modified for arduino simplified serial protocol to sabertooth v21
Twin wheeler modified for arduino simplified serial protocol to sabertooth v2
Simple timer code for arduino
Michael kontopoulos
Michael kontopoulo1s
Define ba1
Define ba
Call and message using arduino and gsm module
140813560 nota-kimia-tingkatan-4
1.funtions (1)
132944997 rancangan-tahunan-mm-t5-2013
Agihan kertas 1 dan kertas 2
Ciri pemimpin
Cara menghilangkan rasa ngantuk di kelas
Bercakap mengenai nikmat allah s
Bandar purba dalam tasik di china
12 kaum yang telah allah swt binasakan
Smkts 2015 p1

Recently uploaded (20)

PPTX
Pharmacology of Heart Failure /Pharmacotherapy of CHF
PPTX
master seminar digital applications in india
PPTX
IMMUNITY IMMUNITY refers to protection against infection, and the immune syst...
PDF
FourierSeries-QuestionsWithAnswers(Part-A).pdf
PDF
Anesthesia in Laparoscopic Surgery in India
PDF
TR - Agricultural Crops Production NC III.pdf
PDF
Mark Klimek Lecture Notes_240423 revision books _173037.pdf
PDF
Physiotherapy_for_Respiratory_and_Cardiac_Problems WEBBER.pdf
PPTX
Microbial diseases, their pathogenesis and prophylaxis
PDF
Classroom Observation Tools for Teachers
PPTX
Renaissance Architecture: A Journey from Faith to Humanism
PDF
RMMM.pdf make it easy to upload and study
PDF
Origin of periodic table-Mendeleev’s Periodic-Modern Periodic table
PDF
grade 11-chemistry_fetena_net_5883.pdf teacher guide for all student
PPTX
PPT- ENG7_QUARTER1_LESSON1_WEEK1. IMAGERY -DESCRIPTIONS pptx.pptx
PPTX
The Healthy Child – Unit II | Child Health Nursing I | B.Sc Nursing 5th Semester
PDF
The Lost Whites of Pakistan by Jahanzaib Mughal.pdf
PPTX
PPH.pptx obstetrics and gynecology in nursing
PDF
Saundersa Comprehensive Review for the NCLEX-RN Examination.pdf
PDF
O7-L3 Supply Chain Operations - ICLT Program
Pharmacology of Heart Failure /Pharmacotherapy of CHF
master seminar digital applications in india
IMMUNITY IMMUNITY refers to protection against infection, and the immune syst...
FourierSeries-QuestionsWithAnswers(Part-A).pdf
Anesthesia in Laparoscopic Surgery in India
TR - Agricultural Crops Production NC III.pdf
Mark Klimek Lecture Notes_240423 revision books _173037.pdf
Physiotherapy_for_Respiratory_and_Cardiac_Problems WEBBER.pdf
Microbial diseases, their pathogenesis and prophylaxis
Classroom Observation Tools for Teachers
Renaissance Architecture: A Journey from Faith to Humanism
RMMM.pdf make it easy to upload and study
Origin of periodic table-Mendeleev’s Periodic-Modern Periodic table
grade 11-chemistry_fetena_net_5883.pdf teacher guide for all student
PPT- ENG7_QUARTER1_LESSON1_WEEK1. IMAGERY -DESCRIPTIONS pptx.pptx
The Healthy Child – Unit II | Child Health Nursing I | B.Sc Nursing 5th Semester
The Lost Whites of Pakistan by Jahanzaib Mughal.pdf
PPH.pptx obstetrics and gynecology in nursing
Saundersa Comprehensive Review for the NCLEX-RN Examination.pdf
O7-L3 Supply Chain Operations - ICLT Program

Arduino 101

  • 1. Arduino 101: Timers and Interrupts   by RobotFreak  Collected by 69 users Arduino, timer, pwm, counter, timers, interrupt Tip/walkthrough By RobotFreak @ August 7, 2011 How to use timer and interrupts on Arduino boards.  Programming language: C++, Arduino Attachment Size Arduino101-Timers.zip 32.9 KB ArduinoTimer101.zip 2.74 KB Update 07/24/2013 04/24/2013  Example 3 has been updated to work with Arduino v1.x.  Added ArduinoTimer101.zip examples source code for Arduino v1.x . This tutorial shows the use of timers and interrupts for Arduino boards. As Arduino programmer you will have used timers and interrupts without knowledge, bcause all the low level hardware stuff is hidden by the Arduino API. Many Arduino functions uses timers, for example the time functions: delay(), millis() and micros() and delayMicroseconds(). The PWM functions analogWrite() uses timers, as the tone() and the noTone() function does. Even the Servo libraryuses timers and interrupts. What is a timer? A timer or to be more precise a timer / counter is a piece of hardware builtin the Arduino controller (other controllers have timer hardware, too). It is like a clock, and can be used to measure time events. The timer can be programmed by some special registers. You can configure the prescaler for the timer, or the mode of operation and many other things. The controller of the Arduino is the Atmel AVR ATmega168 or the ATmega328. These chips are pin compatible and only differ in the size of internal memory. Both have 3 timers, called timer0, timer1 and timer2. Timer0 and timer2 are 8bit timer, where timer1 is a 16bit timer. The most important difference between 8bit and 16bit timer is the timer resolution. 8bits means 256 values where 16bit means 65536 values for higher resolution. The controller for the Arduino Mega series is the Atmel AVR ATmega1280 or the
  • 2. ATmega2560. Also identical only differs in memory size. These controllers have 6 timers. Timer 0, timer1 and timer2 are identical to the ATmega168/328. The timer3, timer4 and timer5 are all 16bit timers, similar to timer1. All timers depends on the system clock of your Arduino system. Normally the system clock is 16MHz, but for the Arduino Pro 3,3V it is 8Mhz. So be careful when writing your own timer functions. The timer hardware can be configured with some special timer registers. In the Arduino firmware all timers were configured to a 1kHz frequency and interrupts are gerally enabled. Timer0: Timer0 is a 8bit timer. In the Arduino world timer0 is been used for the timer functions, like delay(), millis() and micros(). If you change timer0 registers, this may influence the Arduino timer function. So you should know what you are doing. Timer1: Timer1 is a 16bit timer. In the Arduino world the Servo library uses timer1 on Arduino Uno (timer5 on Arduino Mega). Timer2: Timer2 is a 8bit timer like timer0. In the Arduino work the tone() function uses timer2. Timer3, Timer4, Timer5: Timer 3,4,5 are only available on Arduino Mega boards. These timers are all 16bit timers. Timer Register You can change the Timer behaviour through the timer register. The most important timer registers are: TCCRx - Timer/Counter Control Register. The prescaler can be configured here. TCNTx - Timer/Counter Register. The actual timer value is stored here. OCRx - Output Compare Register ICRx - Input Capture Register (only for 16bit timer) TIMSKx - Timer/Counter Interrupt Mask Register. To enable/disable timer interrupts. TIFRx - Timer/Counter Interrupt Flag Register. Indicates a pending timer interrupt. Clock select and timer frequency Different clock sources can be selected for each timer independently. To calculate the timer frequency (for example 2Hz using timer1) you will need:
  • 3. 1. CPU frequency 16Mhz for Arduino 2. maximum timer counter value (256 for 8bit, 65536 for 16bit timer) 3. Divide CPU frequency through the choosen prescaler (16000000 / 256 = 62500) 4. Divide result through the desired frequency (62500 / 2Hz = 31250) 5. Verify the result against the maximum timer counter value (31250 < 65536 success) if fail, choose bigger prescaler. Timer modes Timers can be configured in different modes. PWM mode. Pulth width modulation mode. the OCxy outputs are used to generate PWM signals CTC mode. Clear timer on compare match. When the timer counter reaches the compare match register, the timer will be cleared What is an interrupt? The program running on a controller is normally running sequentially instruction by instruction. An interrupt is an external event that interrupts the running program and runs a special interrupt service routine (ISR). After the ISR has been finished, the running program is continued with the next instruction. Instruction means a single machine instruction, not a line of C or C++ code. Before an pending interrupt will be able to call a ISR the following conditions must be true:
  • 4.  Interrupts must be generally enabled  the according Interrupt mask must be enabled Interrupts can generally enabled / disabled with the function interrupts() / noInterrupts(). By default in the Arduino firmware interrupts are enabled. Interrupt masks are enabled / disabled by setting / clearing bits in the Interrupt mask register (TIMSKx). When an interrupt occurs, a flag in the interrupt flag register (TIFRx) is been set. This interrupt will be automatically cleared when entering the ISR or by manually clearing the bit in the interrupt flag register. The Arduino functions attachInterrupt() and detachInterrupt() can only be used for external interrupt pins. These are different interrupt sources, not discussed here. Timer interrupts A timer can generate different types of interrupts. The register and bit definitions can be found in the processor data sheet (Atmega328 or Atmega2560) and in the I/O definition header file (iomx8.h for Arduino, iomxx0_1.h for Arduino Mega in the hardware/tools/avr/include/avr folder). The suffix x stands for the timer number (0..5), the suffix y stands for the output number (A,B,C), for example TIMSK1 (timer1 interrupt mask register) or OCR2A (timer2 output compare register A). Timer Overflow: Timer overflow means the timer has reached is limit value. When a timer overflow interrupt occurs, the timer overflow bit TOVx will be set in the interrupt flag register TIFRx. When the timer overflow interrupt enable bit TOIEx in the interrupt mask register TIMSKx is set, the timer overflow interrupt service routine ISR(TIMERx_OVF_vect) will be called. Output Compare Match: When a output compare match interrupt occurs, the OCFxy flag will be set in the interrupt flag register TIFRx . When the output compare interrupt enable bit OCIExy in the interrupt mask register TIMSKx is set, the output compare match interrupt service ISR(TIMERx_COMPy_vect) routine will be called. Timer Input Capture: When a timer input capture interrupt occurs, the input capture flag bit ICFx will be set in the interrupt flag register TIFRx. When the input capture interrupt enable bit ICIEx in the interrupt mask register TIMSKx is set, the timer input capture interrupt service routine ISR(TIMERx_CAPT_vect) will be called. PWM and timer There is fixed relation between the timers and the PWM capable outputs. When you look in the data sheet or the pinout of the processor these PWM capable pins have names like OCRxA, OCRxB or OCRxC (where x means the timer number 0..5). The PWM functionality is often shared with other pin functionality. The Arduino has 3Timers and 6 PWM output pins. The relation between timers and PWM outputs is: Pins 5 and 6: controlled by timer0 Pins 9 and 10: controlled by timer1 Pins 11 and 3: controlled by timer2 On the Arduino Mega we have 6 timers and 15 PWMoutputs: Pins 4 and 13: controlled by timer0 Pins 11 and 12: controlled by timer1 Pins 9 and10: controlled by timer2 Pin 2, 3 and 5: controlled by timer 3
  • 5. Pin 6, 7 and 8: controlled by timer 4 Pin 46, 45 and 44:: controlled by timer 5 Usefull 3rd party libraries Some 3rd party libraries exists, that uses timers:  Ken Shirrifs IR library. Using timer2. Send and receive any kind of IR remote signals like RC5, RC6, Sony  Timer1 and Timer3 library. Using timer1 or tiner3. The easy way to write your own timer interupt service routines. I have ported these libraries to different timers for Arduino Mega. All ported libraries can be found in the attached file. Pitfalls There exists some pitfalls you may encounter, when programming your Arduino and make use of functions or libraries that uses timers.  Servo Library uses Timer1. You can’t use PWM on Pin 9, 10 when you use the Servo Library on an Arduino. For Arduino Mega it is a bit more difficult. The timer needed depends on the number of servos. Each timer can handle 12 servos. For the first 12 servos timer 5 will be used (loosing PWM on Pin 44,45,46). For 24 Servos timer 5 and 1 will be used (loosing PWM on Pin 11,12,44,45,46).. For 36 servos timer 5, 1 and 3 will be used (loosing PWM on Pin 2,3,5,11,12,44,45,46).. For 48 servos all 16bit timers 5,1,3 and 4 will be used (loosing all PWM pins).  Pin 11 has shared functionality PWM and MOSI. MOSI is needed for the SPI interface, You can’t use PWMon Pin 11 and the SPI interface at the same time on Arduino. On the Arduino Mega the SPI pins are on different pins.  tone() function uses at least timer2. You can’t use PWM on Pin 3,11 when you use the tone() function an Arduino and Pin 9,10 on Arduino Mega. Examples Time for some examples. Blinking LED with compare match interrupt The first example uses the timer1 in CTC mode and the compare match interrupt to toggle a LED. The timer is configured for a frequency of 2Hz. The LED is toggled in the interrupt service routine.
  • 6. /* Arduino 101: timer and interrupts 1: Timer1 compare match interrupt example more infos: http://www.letmakerobots.com/node/28278 created by RobotFreak */ #define ledPin 13 void setup() { pinMode(ledPin, OUTPUT); // initialize timer1 noInterrupts(); // disable all interrupts TCCR1A = 0; TCCR1B = 0; TCNT1 = 0; OCR1A = 31250; // compare match register 16MHz/256/2Hz TCCR1B |= (1 << WGM12); // CTC mode TCCR1B |= (1 << CS12); // 256 prescaler TIMSK1 |= (1 << OCIE1A); // enable timer compare interrupt interrupts(); // enable all interrupts
  • 7. } ISR(TIMER1_COMPA_vect) // timer compare interrupt service routine { digitalWrite(ledPin, digitalRead(ledPin) ^ 1); // toggle LED pin } void loop() { // your program here... } Blinking LED with timer overflow interrupt same example like before but now we use the timer overflow interrupt. In this case timer1 is running in normal mode. The timer must be preloaded every time in the interrupt service routine. /* * Arduino 101: timer and interrupts * 2: Timer1 overflow interrupt example * more infos: http://www.letmakerobots.com/node/28278 * created by RobotFreak */ #define ledPin 13 void setup() { pinMode(ledPin, OUTPUT);
  • 8. // initialize timer1 noInterrupts(); // disable all interrupts TCCR1A = 0; TCCR1B = 0; TCNT1 = 34286; // preload timer 65536-16MHz/256/2Hz TCCR1B |= (1 << CS12); // 256 prescaler TIMSK1 |= (1 << TOIE1); // enable timer overflow interrupt interrupts(); // enable all interrupts } ISR(TIMER1_OVF_vect) // interrupt service routine that wraps a user defined function supplied by attachInterrupt { TCNT1 = 34286; // preload timer digitalWrite(ledPin, digitalRead(ledPin) ^ 1); } void loop() { // your program here... } Reading quadrature encoders with a timer The next example is part of my Ardubot project. It uses timer2 and the compare match interrupt to read the encoder inputs. Timer2 is initialized by default to a frequency of 1kHz (1ms period). In the interrupt service routine the state of all encoder pins is read and a state machine is used to eliminate false readings. Using the timer interrupt is much easier to handle than using 4 input change interrupts.
  • 9. The signals of a quadrature encoder is a 2bit Gray code. Only 1 bit is changing from state to state. A state machine is perfect to check the signal and count the encoder ticks. The timer must be fast enough, to recognize each state change. For the Pololu wheel encoders used here, the 1ms timer is fast enough. The following example has been modified to work with Arduino V1.x /* /* * Arduino 101: timer and interrupts * 3: Timer2 compare interrupt example. Quadrature Encoder * more infos: http://www.letmakerobots.com/node/28278 * created by RobotFreak * * Credits: * based on code from Peter Dannegger * http://www.mikrocontroller.net/articles/Drehgeber */ #if ARDUINO >= 100 #include "Arduino.h" #else
  • 10. #include "WConstants.h" #endif // Encoder Pins #define encLtA 2 #define encLtB 3 #define encRtA 11 #define encRtB 12 #define ledPin 13 #define LT_PHASE_A digitalRead(encLtA) #define LT_PHASE_B digitalRead(encLtB) #define RT_PHASE_A digitalRead(encRtA) #define RT_PHASE_B digitalRead(encRtB) static volatile int8_t encDeltaLt, encDeltaRt; static int8_t lastLt, lastRt; int encLt, encRt; ISR( TIMER2_COMPA_vect ) { int8_t val, diff; digitalWrite(ledPin, HIGH); // toggle LED pin val = 0; if( LT_PHASE_A ) val = 3; if( LT_PHASE_B ) val ^= 1; // convert gray to binary diff = lastLt - val; // difference last - new
  • 11. if( diff & 1 ){ // bit 0 = value (1) lastLt = val; // store new as next last encDeltaLt += (diff & 2) - 1; // bit 1 = direction (+/-) } val = 0; if( RT_PHASE_A ) val = 3; if( RT_PHASE_B ) val ^= 1; // convert gray to binary diff = lastRt - val; // difference last - new if( diff & 1 ){ // bit 0 = value (1) lastRt = val; // store new as next last encDeltaRt += (diff & 2) - 1; // bit 1 = direction (+/-) } digitalWrite(ledPin, LOW); // toggle LED pin } void QuadratureEncoderInit(void) { int8_t val; cli(); TIMSK2 |= (1<<OCIE2A); sei(); pinMode(encLtA, INPUT); pinMode(encRtA, INPUT); pinMode(encLtB, INPUT); pinMode(encRtB, INPUT);
  • 12. val=0; if (LT_PHASE_A) val = 3; if (LT_PHASE_B) val ^= 1; lastLt = val; encDeltaLt = 0; val=0; if (RT_PHASE_A) val = 3; if (RT_PHASE_B) val ^= 1; lastRt = val; encDeltaRt = 0; encLt = 0; encRt = 0; } int8_t QuadratureEncoderReadLt( void ) // read single step encoders { int8_t val; cli(); val = encDeltaLt; encDeltaLt = 0; sei();
  • 13. return val; // counts since last call } int8_t QuadratureEncoderReadRt( void ) // read single step encoders { int8_t val; cli(); val = encDeltaRt; encDeltaRt = 0; sei(); return val; // counts since last call } void setup() { Serial.begin(38400); pinMode(ledPin, OUTPUT); QuadratureEncoderInit(); } void loop() { encLt += QuadratureEncoderReadLt(); encRt += QuadratureEncoderReadRt(); Serial.print("Lt: "); Serial.print(encLt, DEC); Serial.print(" Rt: ");