51单片机-定时器/计数器

时间:2021-06-20 23:33:01

定时器顾名思义就是设定一段时间,这段时间到了之后可以触发中断,在中断中处理我们的任务。定时器还有一个功能就是计数,每次一个出发定时器内部的TH.TL就会加一,如果加满了就会产生溢出中断。那如何控制定时器呢?
第一个是模式寄存器TMOD

gate T/C M1 M0 gate T/C M1 M0
定时器1 定时器2

其中的T/C标志用来选择定时功能还是计数功能。为1表示定时,为0表示计数。

M1 M0 工作模式
0 0 方式0,13位计数/计时器
0 1 方式,1,16位计数/计时器
1 0 方式2,8位自动加载计数/计时器
1 1 方式3,仅适用于T0,定时器0分为两个独立的8位定时器/计数器TH0及TL0,T1在方式3时停止工作

还有一个TCON的寄存器

TF1 TR1 TF0 TR0 IE1 IT1 IE0 IT0
  • TF1:定时器T1溢出标志,可由程序查询和清零,TF1也是中断请求源,当CPU响应T1中断时由硬件清零。
  • TF0:定时器T0溢出标志,可由程序查询和清零,TF0也是中断请求源,当CPU响应T0中断时由硬件清零。
  • TR1:T1充许计数控制位,为1时充许T1计数。
  • TR0:T0充许计数控制位,为1时充许T0计数。
  • IE1:外部中断1请示源(INT1,P3.3)标志。IE1=1,外部中断1正在向CPU请求中断,当CPU响应该中断时由硬件清“0”IE1(边沿触发方式)。
  • IT1:外部中断源1触发方式控制位。IT1=0,外部中断1程控为电平触发方式,当INT1(P3.3)输入低电平时,置位IE1。
  • IE0:外部中断0请示源(INT0,P3.2)标志。IE0=1,外部中断1正在向CPU请求中断,当CPU响应该中断时由硬件清“0”IE0(边沿触发方式)。
  • IT0:外部中断源0触发方式控制位。IT0=0,外部中断1程控为电平触发方式,当INT0(P3.2)输入低电平时,置位IE0。

中断标号可以看一下

中断号 中断源 中断入口地址
0 INT0—外部中断0 0003H
1 T0 — 定时器/计数器0 000BH
2 INT1—外部中断1 0013H
3 T1 —定时器/计数器1 0018H
4 TI/RI 串行口中断 0023H

定时和计数的主要寄存器是THx,TLx。这两个寄存器是用来存储计数值和定时值的。当选择计数功能是每一次脉冲,THTL寄存器的值都会自动加一,当超出选定模式容纳的最大计数值是,TF会硬件置一,产生中断。选择级数功能时,每一个时钟周期都会自动加一,一直加到选定模式的容许的最大值,此时会出发中断。简单说定时器的中断有两种方式,一种是计数值达到最大(定时或计数都可以),会产生溢出中断。另一种是外部脉冲触发中断(只有计数时),此时的中断标志位IT需要软件清除。
那么如何定时呢,比如定时50ms。定时器每一个机器周期自动加一(时钟周期 = 1/晶振频率,机器周期是12个时钟周期)。如果晶振是12M,那么一个机器周期是12 * 1 / 12M = 1us.如果我们定时50ms,那么计数器加了50ms/1us = 50000次。如果THTL初始设置为0,当THTL为50000时我们就可以进行操作了,但是问题是如何捕获THTL=50000这个事件,所以我们让THTL初始值 = 65536-50000 = 15536,这样定时器从15536一直加到65536就可以产生溢出中断,我们可以捕获中断事件,并且进行频率读取。读取频率是设置定时器为计数模式,每一个脉冲THTL加一,只要在定时中断中读取就可以计算出频率。

我用51做的是频率计。定时1设置为计数功能,定时器0设置定时功能,定时为50ms,每50ms进一次中断,20次中断为1s,读取计数器的值,这个值就是频率。

uint16_t fre = 0;
uint8_t extfre=0;
void main()
{   
    TMOD = 0x51;  //0101 0001,定时器1选择计数功能,定时器0选择定时功能,都工作在模式一
    TH0 = (65536-50000) / 256; //设定初始值
  TL0 = (65536-50000) % 256;
  EA = 1;
  ET0 = ET1 = 1;//使能定时器T0,T1 Enable Time
  TR0 = TR1 = 1;//定时器运行 Time Run

    while(1)        
    {
    }
}
void T0_time()interrupt 1    //50ms
{
    static uint8_t count = 0;

    TH0=(65536-50000)/256; //初始化定时器
  TL0=(65536-50000)%256;

    if(20 == ++count)//1s
    {
        count = 0;
        fre = extfre << 16 + TH1 << 8 + TL1;
        TH1 = TL1 =0;//初始化计数器
        extfre = 0;
    }
}
void T1_time()interrupt 3 
{
    if(TF1) 
    {
        extfre++;
    }
}