利用定时器/计数器1的CTC模式精确定时

 

  在单片机的应用中我们经常遇到精确定时处理的任务,如果延时要求不是严格可以使用下面这种粗略的延时函数:

 //延时一毫秒
 void DelayMs(uint16_t)
 {
   uint16_t i;
   for(i=0;i<t;i++)
      _delay_loop_2(CPU_FRQ * 4);//CPU_FRQ为系统时钟频率以MHz为单位
 }

 
  这样延时的误差是显而易见的!那么如何实现精确延时呢? 答案通常是使用定时器/计数器,那么如何个使用定时器计数器才能产生精确延时的
中断呢?看看下面这个示例:

//系统时钟1MHz
#include <avr/io.h>
#include <avr/interrupt.h>

#define FLASH_LED PORTB^=_BV(PB0)

//T/C0中断例程,计数250时溢出,所以
ISR(TIMER0_OVF_vect)
{
  // 产生中断周期应为 T = 250 * 1024 / 1MHz = 0.256s
  TCNT0=256-250;//中断中改变计数值以便人6开始计数到256时正好溢出
  FLASH_LED;
}

int main(void)
{
 DDRB|=_BV(PB0);

 TCNT0=0; // T/C0开始值
 TCCR0=_BV(CS02)|_BV(CS00); // 预分频 ck/1024 ,计数允许
 sei(); //总中断标志置位

 while(1);//等待中断
}

  上面程序看似没有问题,实际上仍存在很多误差的,因为中断中改变计数值前CPU需要执行多条指令以保护当前执行的环境.如果说中断中不需要改变计数寄存器,即计数器正好计到256溢出时满足延时要求,那么这才是没有误差的延时.

   AVR器件中有的计数器具有比较匹配自动清零功能,例如ATMEAG8的TIMER1 ,TIMER2,ATMEGA16的TIMER0,TIMER1,TIMER2等等.它们可以很灵活的产生各种精确的延时中断.如下示例

/********************************************
Timer1 CTC 模式精确定时示例
文件名:main.c
编译:WinAVR-20070525

硬件环境:CA-M8X
时钟:外部4MHz
打开的开关如下
S6(1,2) - 外部4MHz晶振
S1(7) - PB0接LED

芯艺设计室 2004-2007 版权所有
转载请保留本注释在内的全部内容
WEB: http://www.chipart.cn
Email: changfutong@sina.com

本程序按0.5秒改变一次接PB0口的LED
状态来说明T/C1 CTC模式精确定时
********************************************/
#include <avr/io.h>
#include <avr/interrupt.h>

#define FLASH_R_LED PORTB^=_BV(PB0)

//系统时钟:4000000Hz 分频:64
//CTC TOP值:0x7A12
//误差:0.000000个系统周期

//Timer1初始化
void Timer1Init(void)
{
 TCCR1A=0;
 TCCR1B=_BV(WGM12)|_BV(CS10)|_BV(CS11);
 OCR1A=0x7A12;
 TIMSK|=_BV(OCIE1A);
}

//Timer1 CTC中断,此中断发生周期是:500毫秒
ISR(TIMER1_COMPA_vect)
{
 FLASH_R_LED;
}

int main(void)
{
 DDRB|=_BV(PB0);
 Timer1Init();
 sei();
 while(1);
}

 

上面示例中当TCNT1计数到OCR1A一样时由硬件自动清为0,并产生中断每周期中没有软件的延时,十分精确.

下面这个小软件可自动生成定时器计数器1产生延时的AVR-GCC源代码

 

 本软件的可执行程序可从CA-M8X示例程序-中级示例2的第5个示例中找到!

 

芯艺设计室    蒙ICP备06005492号

Copyright© 2004-2010 ChipArt Design House All Rights Reserved