AVR 定时器使用范例

发布者:sumig最新更新时间:2016-07-18 来源: eefocus关键字:AVR  定时器  使用范例 手机看文章 扫描二维码
随时随地手机看文章
/***********************************************

****      AVR 定时器使用范例                   ***
****                                        ***
**** 作者:  HJJourAVR                        ***
**** 编译器:WINAVR20050214                   ***
****                                         ***
****   www.OurAVR.com  2005.9.22           ***
***********************************************/
/*
本程序简单的示范了如何使用ATMEGA16的定时器
 AVR定时器的要点介绍
 T0工作于CTC模式,输出1KHz/2KHz 50%占空比的方波
 T1工作于快速PWM模式兼输入捕捉
 T2工作于相位修正PWM模式,输出490Hz的8bit PWM波
      
出于简化程序考虑,各种数据没有对外输出,学习时建议使用JTAG ICE硬件仿真器
对于定时器,AVRstudio的软件仿真是不准确的。
*/

#include 
#include 
#include 
//时钟定为8MHz,F_CPU=8000000

//管脚定义
#define ICPKEY   6  //ICP1  PD6 按键模拟ICP输入
#define PWM0   3  //OC0   PB3
#define PWM1A   5  //OC1A  PD5
#define PWM1B   4  //OC1B  PD4
#define PWM2   7  //OC2   PD7

//宏定义
#define PWM1A_ON()  PORTD|= (1<
#define PWM1A_OFF()  PORTD&=~(1<<>

 

//全局变量
volatile unsigned int ICP_Time;  //记录ICP输入捕捉事件的发生时刻
volatile unsigned char T2PWM;  //设置T2的PWM值
volatile unsigned char T0OCR;  //设置T0的时间值

//仿真时在watch窗口,监控这些变量。

 

void timer0_init(void) //CTC模式输出1KHz/2KHz方波
{
 OCR0  = T0OCR;  //设定TOP值
 //TOP=8000000/(2*64*1000)-1=61.5  选61  1.008KHz(0.992mS)
 //TOP=8000000/(2*64*2000)-1=30.25 选30  2.016KHz(0.496ms)
 TCCR0 = (1<<<<
 //64分频,CTC模式,OC0取反输出方波
}

void timer1_init(void)
{

 OCR1A  = 39062; //设定TOP值.时间5S(0.2Hz)
 //TOP=8000000/(1024*0.2)=39062.5
 OCR1B  = 15624; //设定OC1B的PWM值 约2秒钟 40%
 TCCR1A = (1<<< TCCR1B = (0<<<< //1024分频,WGM1=15 快速PWM模式,TOP=OCRnA,ICP下降沿触发,OC1B正向PWM输出,OC1A为普通IO
}

SIGNAL(SIG_INPUT_CAPTURE1) //输入捕捉中断
{
   ICP_Time=ICR1;  //读取ICP输入捕捉事件的发生时刻
}

SIGNAL(SIG_OUTPUT_COMPARE1A)//T1输出比较A匹配中断
{
 
 //在WGM1=15 快速PWM模式下,TOP=39062等同于5S左右的定时中断
 T2PWM+=10;
 OCR2=T2PWM;   //修改T2的PWM值
 if (T0OCR==61)
  T0OCR=30;  //改成1KHz
 else 
     T0OCR=61;  //改成2KHz
 OCR0=T0OCR;   //修改T0的时间值
 
}

void timer2_init(void)//相位修正PWM模式
{
 OCR2  = T2PWM;  //设定PWM值(最大值固定为255,8bit)
 TCCR2 = (0<<<<
 //32分频,相位修正PWM模式,PWM频率为490Hz,OC2正向PWM输出
 //fPWM=fclk_IO/(2*N*TOP)=8000000/(2*32*255)=490Hz
}


int main(void)
{

    //上电默认DDRx=0x00,PORTx=0x00 输入,无上拉电阻
    PORTA =0xFF;         //不用的管脚使能内部上拉电阻。
    PORTC =0xFF;
    PORTB =~ (1<
    DDRB  =  (1<
    PORTD =~((1<<
    DDRD  =  (1<<
    
    
    T2PWM=0x80;
    T0OCR=30;
    ICP_Time=0x0000;
    
 timer0_init();
 timer1_init();
 timer2_init();

 TIMSK = (1<<

    sei();           //使能全局中断
    while (1)
    {
     if (ICP_Time>15624)
      PWM1A_ON();   //如果数值大于15624(约2秒),OC1A输出高电平
     else 
      PWM1A_OFF();  //否则输出低电平
    }
}

/*
程序运行效果
     引脚OC0(每5秒钟切换)交替输出1KHz和2KHz的50%占空比方波,接到无源蜂鸣器上,能听到不同频率的声音
     引脚OC1B输出0.2Hz的40%占空比的PWM波,精度39061级(略大于15bit)
     引脚OC2输出490Hz的PWM波,精度8bit,每5秒钟PWM值增大10级,对应的LED亮度将会随之变化)
     
     ICP由引脚ICP1上的按键触发,ICP_Time将会记录下时间发生的时刻(相对于T1定时器的本次计数开始时间),
     如果数值大于15624(约2秒),OC1A输出高电平,否则输出低电平(刚好跟OC1B反相)

     如果使用AVR-51实验板作本实验,注意输出电平和LED的关系。还有蜂鸣器的声音较大,耳朵比较难受)
*/

 /*
附录 AVR定时器的要点介绍
 (大部分摘自 M16中文手册,未能一一测试)

  M16的T1 16位定时器一共有15种工作模式,其他2个8位定时器(T0/T2)相对简单,除了T2有异步工作模式用于RTC应用外
                       (可以利用溢出中断和比较匹配中断作定时功能)
  分5种工作类型
  1  普通模式 WGM1=0
    跟51的普通模式差不多,有TOV1溢出中断,发生于TOP时
    1 采用内部计数时钟     用于 ICP捕捉输入场合---测量脉宽/红外解码
        (捕捉输入功能可以工作在多种模式下,而不单单只是普通模式)
    2 采用外部计数脉冲输入  用于 计数,测频
    其他的应用,采用其他模式更为方便,不需要像51般费神
    
  2 CTC模式 [比较匹配时清零定时器模式] WGM1=4,12
   跟51的自动重载模式差不多
   1 用于输出50%占空比的方波信号
   2 用于产生准确的连续定时信号
   WGM1=4时, 最大值由OCR1A设定,TOP时产生OCF1A比较匹配中断
   WGM1=12时,最大值由ICF1设定, TOP时产生ICF1输入捕捉中断
   注:WGM=15时,也能实现从OC1A输出方波,而且具备双缓冲功能
   计算公式: fOCn=fclk_IO/(2*N*(1+TOP))
     变量N 代表预分频因子(1、8、32,64、256,1024)。   
     
  3 快速PWM模式 WGM1=5,6,7,14,15 
    单斜波计数,用于输出高频率的PWM信号(比双斜波的高一倍频率)
    都有TOV1溢出中断,发生于TOP时
    比较匹配后可以产生OCF1x比较匹配中断.
    WGM1=5时, 最大值为0x00FF, 8位分辨率
    WGM1=6时, 最大值为0x01FF, 9位分辨率
    WGM1=7时, 最大值为0x03FF,10位分辨率 
   WGM1=14时,最大值由ICF1设定, TOP时产生ICF1输入捕捉中断 (单缓冲)
   WGM1=15时,最大值由OCR1A设定,TOP时产生OCF1A比较匹配中断(双缓冲,但OC1A将没有PWM能力,最多只能输出方波)
   改变TOP值时必须保证新的TOP值不小于所有比较寄存器的数值
    注意,即使OCR1A/B设为0x0000,也会输出一个定时器时钟周期的窄脉冲,而不是一直为低电平
    计算公式:fPWM=fclk_IO/(N*(1+TOP))

  4 相位修正PWM模式 WGM1=1,2,3,10,11 
    双斜波计数,用于输出高精度的,相位准确的,对称的PWM信号
    都有TOV1溢出中断,但发生在BOOTOM时
    比较匹配后可以产生OCF1x比较匹配中断.
    WGM1=1时, 最大值为0x00FF, 8位分辨率
    WGM1=2时, 最大值为0x01FF, 9位分辨率
    WGM1=3时, 最大值为0x03FF,10位分辨率 
   WGM1=10时,最大值由ICF1设定, TOP时产生ICF1输入捕捉中断 (单缓冲)
   WGM1=11时,最大值由OCR1A设定,TOP时产生OCF1A比较匹配中断(双缓冲,但OC1A将没有PWM能力,最多只能输出方波)
    改变TOP值时必须保证新的TOP值不小于所有比较寄存器的数值
    可以输出0%~100%占空比的PWM信号
    若要在T/C 运行时改变TOP 值,最好用相位与频率修正模式代替相位修正模式。若TOP保持不变,那么这两种工作模式实际没有区别
    计算公式:fPWM=fclk_IO/(2*N*TOP)

  5 相位与频率修正PWM模式 WGM1=8,9 
    双斜波计数,用于输出高精度的、相位与频率都准确的PWM波形
    都有TOV1溢出中断,但发生在BOOTOM时
    比较匹配后可以产生OCF1x比较匹配中断.
   WGM1=8时,最大值由ICF1设定, TOP时产生ICF1输入捕捉中断 (单缓冲)
   WGM1=9时,最大值由OCR1A设定,TOP时产生OCF1A比较匹配中断(双缓冲,但OC1A将没有PWM能力,最多只能输出方波)
    相频修正修正PWM 模式与相位修正PWM 模式的主要区别在于OCR1x 寄存器的更新时间
    改变TOP值时必须保证新的TOP值不小于所有比较寄存器的数值
    可以输出0%~100%占空比的PWM信号
    使用固定TOP 值时最好使用ICR1 寄存器定义TOP。这样OCR1A 就可以用于在OC1A输出PWM 波。
    但是,如果PWM 基频不断变化(通过改变TOP值), OCR1A的双缓冲特性使其更适合于这个应用。
    计算公式:fPWM=fclk_IO/(2*N*TOP)

T/C 的时钟源
 T/C 的时钟源可以有多种选择,由CS12:0控制,分别用于高速(低分频)/长时间(高分频)/外部计数场合
 一个16位定时器,在8MHz系统时钟驱动下,可以实现uS级的高速定时和长达8秒的超长定时,这可是标准51的弱点

 CS12 CS11 CS10 说明
     0    0    0  无时钟源     (T/C 停止)
     0    0    1  clkIO/1    ( 无预分频) 
     0    1    0  clkIO/8    ( 来自预分频器)
     0    1    1  clkIO/64   ( 来自预分频器)
     1    0    0  clkIO/256  ( 来自预分频器) 
     1    0    1  clkIO/1024 ( 来自预分频器)
     1    1    0  外部T1 引脚,下降沿驱动
     1    1    1  外部T1 引脚,上升沿驱动
  分频器复位
    在高预分频应用时, 通过复位预分频器来同步T/C 与程序运行,可以减少误差。
 但是必须注意另一个T/C是否也在使用这一预分频器,因为预分频器复位将会影响所有与其连接的T/C。
  
  外部时钟源
 由于使用了引脚同步逻辑,建议外部时钟的最高频率不要大于fclk_IO/2.5。
 外部时钟源不送入预分频器
 选择使用外部时钟源后,即使T1引脚被定义为输出,其T1引脚上的逻辑信号电平变化仍然会驱动T/C1 计数,这个特性允许用户通过软件来控制计数。
 
输入捕捉单元
 T/C 的输入捕捉单元可用来捕获外部事件,并为其赋予时间标记以说明此时间的发生时刻。
 外部事件发生的触发信号由引脚ICP1 输入,也可通过模拟比较器单元来实现。
 时间标记可用来计算频率、占空比及信号的其它特征,以及为事件创建日志。
 
 输入捕捉单元可以工作在多种工作模式下
 (使用ICR1定义TOP的(WGM1=12,14,10,8)波形产生模式时,ICP1与输入捕捉功能脱开,从而输入捕捉功能被禁用。)
 在任何输入捕捉工作模式下都不推荐在操作过程中改变TOP值
 
 当引脚ICP1 上的逻辑电平( 事件) 发生了变化,或模拟比较器输出ACO 电平发生了变化,并且这个电平变化为边沿检测器所证实,输入捕捉即被激发:
 16位的TCNT1 数据被拷贝到输入捕捉寄存器ICR1,同时输入捕捉标志位ICF1 置位。
 如果此时ICIE1 = 1,输入捕捉标志将产生输入捕捉中断。
 中断执行时ICF1 自动清零,或者也可通过软件在其对应的I/O 位置写入逻辑"1” 清零。
 
 注意,改变触发源有可能造成一次输入捕捉。因此在改变触发源后必须对输入捕捉标志执行一次清零操作以避免出现错误的结果
 
 除去使用ICR1定义TOP的波形产生模式外, T/C中的噪声抑制器与边沿检测器总是使能的。
 (其实就是永远使能??)
 使能噪声抑制器后,在边沿检测器前会加入额外的逻辑电路并引入4个系统时钟周期的延迟.
 噪声抑制器使用的是系统时钟,因而不受预分频器的影响
 
 使用输入捕捉中断时,中断程序应尽可能早的读取ICR1 寄存器
 如果处理器在下一次事件出现之前没有读取ICR1 的数据, ICR1 就会被新值覆盖,从而无法得到正确的捕捉结果。
 
 测量外部信号的占空比时要求每次捕捉后都要改变触发沿。
 因此读取ICR1 后必须尽快改变敏感的信号边沿。改变边沿后,ICF1 必须由软件清零( 在对应的I/O 位置写"1”)。
 若仅需测量频率,且使用了中断发生,则不需对ICF1 进行软件清零。
 
输出比较单元
 16位比较器持续比较TCNT1与OCR1x的内容,一旦发现它们相等,比较器立即产生一个匹配信号。
 然后OCF1x 在下一个定时器时钟置位。
 如果此时OCIE1x = 1, OCF1x 置位将引发输出比较中断。
 (就是说输出比较可以工作在所有工作模式下,但PWM模式下更好用,功能更强)
 
 输出比较单元A(OCR1A) 的一个特质是定义T/C 的TOP 值( 即计数器的分辨率)。
 TOP 值还用来定义通过波形发生器产生的波形的周期。
 
 由于在任意模式下写TCNT1 都将在下一个定时器时钟周期里阻止比较匹配,在使用输出比较时改变TCNT1就会有风险,不管T/C是否在运行
 这个特性可以用来将OCR1x初始化为与TCNT1 相同的数值而不触发中断。

  强制输出比较(FOC)
 工作于非PWM 模式时,可以通过对强制输出比较位FOC1x 写”1” 的方式来产生比较匹配。
 强制比较匹配不会置位 OCF1x 标志,也不会重载/ 清零定时器,
 但是OC1x 引脚将被更新,好象真的发生了比较匹配一样(COMx1:0 决定OC1x 是置位、清零,还是交替变化)。
 
比较匹配输出单元
 比较匹配模式控制位COM1x1:0 具有双重功能。
 1 波形发生器利用COM1x1:0 来确定下一次比较匹配发生时的输出比较OC1x 状态;
 2 COM1x1:0 还控制OC1x 引脚输出的来源。
 只要COM1x1:0 不全为零,波形发生器的输出比较功能就会重载OC1x 的通用I/O 口功能。
 但是OC1x 引脚的方向仍旧受控于数据方向寄存器 (DDR)。
 从OC1x 引脚输出有效信号之前必须通过数据方向寄存器的DDR_OC1x 将此引脚设置为输出。
 
 波形发生器利用COM1x1:0 的方法在普通模式、CTC 模式和PWM 模式下有所区别。
 对于所有的模式,设置COM1x1:0=0 表明比较匹配发生时波形发生器不会操作OC1x寄存器

访问16位寄存器
 写16 位寄存器时,应先写入该寄存器的高位字节.
  usigned int k;
  k=0x1234;
  TCNT1H=(unsigned char)(k>>8);
  TCNT1L=(unsigned char) k;
 而读16 位寄存器时应先读取该寄存器的低位字节.
  usigned int k;
     k=TCNT1L;
     k+=(unsigned int)(TCNT1H<<8);
 使用“C” 语言时,编译器会自动处理16位操作.
  usigned int k;
  k=0x1234;
  TCNT=k;
  k=TCNT1;
*/

关键字:AVR  定时器  使用范例 引用地址:AVR 定时器使用范例

上一篇:AVR定时器工作原理
下一篇:AVR学习心得(三)

推荐阅读最新更新时间:2024-03-16 15:00

MSP430单片机定时器
在MSP430系列单片机中带有功能强大的定时器资源,这定时器在单片机应用系统中起到重要的作用。 在F11X,F11X1中是不带定时器B资源的。430的定时模块:看门狗定时器,定时器A,定时器B。定时器A主要资源特点有16位定时计数器,其计数模式有4种。多种计数时钟信号供输入的捕获/比较功能寄存器和8种输出模式的3个可配置输出单片。 定时器资源功能说明 (1)看门狗定时器(WDT):主要用于程序在生错误时用作单片机系统复位重起的。另外,也可作为一个基本定时器使用。 (2)定时器A:作基本定时器使用,结合捕获/比较功能模块可实现时序控制,可编程波形信号发生输出。可作串口波特率 (3)定时器B:作基本定时器使用,与定时器A基本
[单片机]
S6B0724芯片的LCD128*64的AVR128驱动程序
以下是用S6B0724做驱动芯片的128*64的AVR128驱动程序,已经测试通过。请大家放心使用,呵呵! /*************************************************************/ #i nclude avr/io.h /*************************************************************/ /* 宏定义替换 */ /*************************************************************/ #define uchar unsigned char //宏定义替换 #def
[单片机]
STM32 SysTick定时器的1秒钟设计与实现
单片机源程序如下: #include stm32f10x.h #include Delay.h #include smg.h //定义0~9十个数字的字型码表 uint16_t table ={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f}; uint16_t wei ={0x0fe,0x0fd,0x0fb,0x0f7,0x0ef,0x0df,0xff,0xff}; //位码; uint8_t i; int main(void) { SMG_Init(); while(1) { for(i=1;i 7;i++)
[单片机]
STM32 SysTick<font color='red'>定时器</font>的1秒钟设计与实现
STM32学习笔记:通用定时器基本设定
STM32 包含11个定时器,其中TIM2~TIM5为通用定时器。 通用定时器挂载在低速外设总线APB1上,其时钟来源于输入为APB1的一个倍频器,只要APB1的时钟分频数不为1,TIMx的时钟频率就会为APB1时钟频率的2倍,即72MHz 。 配置通用定时器需要4步(以配置TIM2为例): (1)配置系统时钟 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);倍 开启TIM2与GPIOA的时钟,其中TIM2挂载在APB1上,GPIOA挂载在APB2上;
[单片机]
STM32F407 纯寄存器操作定时器,PWM,ADC(专治花里胡哨)
定时器 真多啊,14个定时器,三种。 通用定时器 注意: 需要手动清除中断标志 TIM3- ARR = 5000-1; //自动重载寄存器,10Khz 的计数频率,计数 5K 次为 500ms TIM3- PSC = 8400-1; //预分频器 两个寄存器的计算 PSC预分频器 //这里时钟选择为 APB1 的 2 倍,而 APB1 为 42M,所以TIM3的时钟频率为84M PSC=8400 f=84M/8400=10000Hz=10KHz t=1/f=0.1ms 每次技术时间0.1ms ARR=5000-1; //自动重载寄存器 计数5000次,共500ms //定时器溢出时间计算方法:Tou
[单片机]
STM32F407 纯寄存器操作<font color='red'>定时器</font>,PWM,ADC(专治花里胡哨)
无需更换电池的倒计时数字定时器
  只需 MCU、LCD、32kHz 晶振以及一个电阻即可构成一部基本的倒计时定时器,而且只需一颗普通的钮扣电池就能实现连续超过 10 年的工作时间。有两大关键性设计可最小化功耗并尽可能延长电池使用寿命,一是认真选择电池,二是充分利用 MCU 的低功耗模式。   我们之所以选择 CR2032 锂离子钮扣电池,主要是因为它占地较小,可以支持便携式应用,放电曲线较平,可直接驱动LCD,无需补偿,而且漏电极低,能够实现更长的工作时间。典型的 CR2032 额定容量为 ~200mA,工作时间可维持数小时。为了实现连续工作 10 年的设计目标,平均系统电流消耗应不得超过 2.28μA,电流消耗的计算方式为电池能量值除以应用的工作寿命,如
[单片机]
基于AVR单片机实现超声波测距的程序编写
#include #defineucharunsignedchar #defineuintunsignedint ucharint_flag; uintqian,bai,shi,ge; uinti; floatdistance; //AlphanumericLCDModulefuncTIons #asm .equ__lcd_port=0x18;PORTB #endasm #include //ExternalInterrupt0servicerouTIne interrupt voidext_int0_isr(void) {#asm( sei ) if(int_flag) { distance=i*17.3; } elsei
[单片机]
基于<font color='red'>AVR</font>单片机实现超声波测距的程序编写
基于AVR的LED数字大屏幕的设计与实现
1引言   LED显示屏是八十年代后期在全球迅速发展起来的新型信息显示媒体,应用越来越广泛。ATMEGA16L单片机是一款基于AVR RISC 的、低功耗CMOS 8 位单片机,具有16K字节的自编程FLASH, 512字节的EEPROM, 1K字节的SRAM, 通过AVR单片机来控制LED的显示,结构简单,应用灵活,并且易于扩展。 2 系统设计原理 2.1 编码设计   编码设计有两个过程:字库提取以及字模编码转换。字库的提取涉及到的两个字库文件,一个为ASC16,专门存放ASCII编码;另一个为HZK16,专门存放汉字字库编码。要提取字库,首先要对它进行初始化,即打开字库。然后根据一个字符提取它相应的编码。首先取出字符的内
[电源管理]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved