第三部分 定时器中断培训(09.17)
关于定时器中断,我们简单介绍一下原理就可以了,因为具体介绍已经有很多资料都已经提供了。
80C51单片机内部设有两个16位的可编程定时器/计数器。可编程的意思是指其功能(如工作方式、定时时间、量程、启动方式等)均可由指令来确定和改变。在定时器/计数器中除了有两个16位的计数器之外,还有两个特殊功能寄存器(控制寄存器和方式寄存器)。我们可以看出,16位的定时/计数器分别由两个8位专用寄存器组成,即:T0由TH0和TL0构成;T1由TH1和TL1 构成。其访问地址依次为8AH-8DH。每个寄存器均可单独访问。这些寄存器是用于存放定时或计数初值的。此外,其内部还有一个8位的定时器方式寄存器 TMOD和一个8位的定时控制寄存器TCON。这些寄存器之间是通过内部总线和控制逻辑电路连接起来的。TMOD主要是用于选定定时器的工作方式; TCON主要是用于控制定时器的启动停止,此外TCON还可以保存T0、T1的溢出和中断标志。当定时器工作在计数方式时,外部事件通过引脚T0 (P3.4)和T1(P3.5)输入。定时计数器的原理:
当定时器/计数器为定时工作方式时,计数器的加1信号由振荡器的12分频信号产生,即每过一个机器周期,计数器加1,直至计满溢出为止。显然,定时器的定时时间与系统的振荡频率有关。因一个机器周期等于12个振荡周期,所以计数频率fcount=1/12osc。如果晶振为12MHz,则计数周期为:
T=1/(12×106)Hz×1/12=1μs
这是最短的定时周期。若要延长定时时间,则需要改变定时器的初值,并要适当选择定时器的长度(如8位、13位、16位等)。
当定时器/计数器为计数工作方式时,通过引脚T0和T1对外部信号计数,外部脉冲的下降沿将触发计数。计数器在每个机器周期的S5P2期间采样引脚输入电 平。若一个机器周期采样值为1,下一个机器周期采样值为0,则计数器加1。此后的机器周期S3P1期间,新的计数值装入计数器。所以检测一个由1至0的跳 变需要两个机器周期,故外部事年的最高计数频率为振荡频率的1/24。例如,如果选用12MHz晶振,则最高计数频率为0.5MHz。虽然对外部输入信号 的占空比无特殊要求,但为了确保某给定电平在变化前至少被采样一次,外部计数脉冲的高电平与低电平保持时间均需在一个机器周期以上。
当CPU用软件给定时器设置了某种工作方式之后,定时器就会按设定的工作方式独立运行,不再占用CPU的操作时间,除非定时器计满溢出,才可能中断CPU 当前操作。CPU也可以重新设置定时器工作方式,以改变定时器的操作。由此可见,定时器是单片机中效率高而且工作灵活的部件。
综上所述,我们已知定时器/计数器是一种可编程部件,所以在定时器/计数器开始工作之前,CPU必须将一些命令(称为控制字)写入定时/计数器。将控制字写入定时/计数器的过程叫定时器/计数器初始化。在初始化过程中,要将工作方式控制字写入方式寄存器,工作状态字(或相关位)写入控制寄存器,赋定时/计 数初值。下面我们就提出的控制字的格式及各位的主要功能与大家详细的讲解。
控制寄存器 定时器/计数器T0和T1有2个控制寄存器-TMOD和TCON,它们分别用来设置各个定时器/计数器的工作方式,选择定时或计数功能,控制启动运行,以及作为运行状态的标志等。其中,TCON寄存器中另有4位用于中断系统。
TMOD定时器/计数器方式寄存器
定时器方式控制寄存器TMOD在特殊功能寄存器中,字节地址为89H,无位地址。TMOD的格式如下图所示。
由图可见,TMOD的高4位用于T1,低4使用于T0,4种符号的含义如下:
GATE:门控制位。GATE和软件控制位TR、外部引脚信号INT的状态,共同控制定时器/计数器的打开或关闭。
C/T:定时器/计数器选择位。C/T=1,为计数器方式;C/T=0,为定时器方式。
M1M0:工作方式选择位,定时器/计数器的4种工作方式由M1M0设定。
M1M0
| 工作方式
| 功能描述
|
00
| 工作方式0
| 13位计数器
|
01
| 工作方式1
| 16位计数器
|
10
| 工作方式2
| 自动再装入8位计数器
|
11
| 工作方式3
| 定时器0:分成两个8位计数器;
定时器1:停止计数
|
定时器/计数器方式控制寄存器TMOD不能进行位寻址,只能用字节传送指令设置定时器工作方式,低半字节定义为定时器0,高半字节定义为定时器1。复位时,TMOD所有位均为0。
例:设定定时器1为定时工作方式,要求软件启动定时器1按方式2工作。定时器0为计数方式,要求由软件启动定时器0,按方式1工作。
我们怎么来实现这个要求呢?
大家先看上面TMOD寄存器各位的分布图
第一个问题:控制定时器1工作在定时方式或计数方式是哪个位?通过前面的学习,我们已知道,C/T位(D6)是定时或计数功能选择位,当C/T=0时定时/计数器就为定时工作方式。所以要使定时/计数器1工作在定时器方式就必需使D6为0。
第二个问题:设定定时器1按方式2工作。上表中可以看出,要使定时/计数器1工作在方式2,M0(D4) M1(D5)的值必须是1 0。
第三个问题:设定定时器0为计数方式。与第一个问题一样,定时/计数器0的工作方式选择位也是C/T(D2),当C/T=1时,就工作在计数器方式。
第四个问题:由软件启动定时器0,前面已讲过,当门控位GATE=0时,定时/计数器的启停就由软件控制。
第五个问题:设定定时/计数器工作在方式1,使定时/计数器0工作在方式1,M0(D0) M1(D1)的值必须是0 1。
从上面的分析我们可以知道,只要将TMOD的各位,按规定的要求设置好后,定时器/计灵敏器就会按我们预定的要求工作。我们分析的这个例子最后各位的情况如下:
D7 D6 D5 D4 D3 D2 D1 D0
0 0 1 0 0 1 0 1
二进制数00100101=十六进制数25H。所以执行MOV TMOD,#25H这条指令就可以实现上述要求。
TCON定时器/计数器控制寄存器
TCON在特殊功能寄存器中,字节地址为88H,位地址(由低位到高位)为88H一8FH,由于有位地址,十分便于进行位操作。
TCON的作用是控制定时器的启、停,标志定时器溢出和中断情况。
TCON的格式如下图所示。其中,TFl,TRl,TF0和TR0位用于定时器/计数器;IEl,ITl,IE0和IT0位用于中断系统。
各位定义如下:
TF1:定时器1溢出标志位。当字时器1计满溢出时,由硬件使TF1置“1”,并且申请中断。进入中断服务程序后,由硬件自动清“0”,在查询方式下用软件清“0”。
TR1:定时器1运行控制位。由软件清“0”关闭定时器1。当GATE=1,且INT1为高电平时,TR1置“1”启动定时器1;当GATE=0,TR1置“1”启动定时器1。
TF0:定时器0溢出标志。其功能及操作情况同TF1。
TR0:定时器0运行控制位。其功能及操作情况同TR1。
IE1:外部中断1请求标志。
IT1:外部中断1触发方式选择位。
IE0:外部中断0请求标志。
IT0:外部中断0触发方式选择位。
TCON中低4位与中断有关,我们将在下节课讲中断时再给予讲解。由于TCON是可以位寻址的,因而如果只清溢出或启动定时器工作,可以用位操作命令。例如:执行“CLR TF0”后则清定时器0的溢出;执行“SETB TR1”后可启动定时器1开始工作(当然前面还要设置方式定)。
定时器/计数器的初始化
由于定时器/计数器的功能是由软件编程确定的,所以一般在使用定时/计数器前都要对其进行初始化,使其按设定的功能工作。初始货的步骤一般如下:
1、确定工作方式(即对TMOD赋值);
2、预置定时或计数的初值(可直接将初值写入TH0、TL0或TH1、TL1);
3、根据需要开放定时器/计数器的中断(直接对IE位赋值);
4、启动定时器/计数器(若已规定用软件启动,则可把TR0或TR1置“1”;若已规定由外中断引脚电平启动,则需给外引脚步加启动电平。当实现了启动要求后,定时器即按规定的工作方式和初值开始计数或定时)。
下面介绍一下确定时时/计数器初值的具体方法。
因为在不同工作方式下计数器位数不同,因而最大计数值也不同。
现假设最大计数值为M,那么各方式下的
最大值M值如下:
方式0:M=213=8 192
方式1:M=216=65 536
方式2:M=28=256
方式3:定时器0分成两个8位计数器,所以两个M均为256。
因为定时器/计数器是作“加1”计数,并在计数满溢出时产生中断,因此初值X可以这样计算:
X=M-计数值
下面举例说明初值的确定方法。
例1、选择T1方式0用于定时,在P1.1输出周期为1ms方波,晶振fosc=6MHz。
解:根据题意,只要使P1.1每隔500us取反一次即可得到1ms的方波,因而T1的定时时间为500us,因定时时间不长,取方式0即可。则M1 M0=0;因是定时器方式,所以C/T=0;在此用软件启动T1,所以GATE=0。T0不用,方式字可任意设置,只要不使其进入方式3即可,一般取0, 故TMOD=00H。系统复位后TMOD为0,可不对TMOD重新清0。
下面计算500us定时T1初始值:
机器周期T=12/fosc=12/(6×106)Hz=2μs
设初值为X,则:
(1013-X)×2×10-6s=500×10-6s
X=7942D=1111100000110B=1F06H
因为在作13位计数器用时,TL1的高3位未用,应填写0,TH1占用高8位,所以X的实际填写应为:
X=111100000000110B=F806H
结果:TH1=F8H,TL1=06H
源程序如下:
ORG 2000H
MOV TL1,#06H ;给TL1置初值
MOV TH1,#0F8H ;给TH1置初值
SETB TR1 ;启动T1
LP1:JBC TF1,LP2 ;查询计数溢出否?
AJMP LP1
LP2:MOV TL1,#06H ;重新设置计数初值
MOV TH1,#0F8H
CPL P1.1 ;输出取反
AJMP LP1 ;重复循环
看了上面的介绍,我们现在应该会计算定时器的初值和设置相关的参数了,下面我们先看看硬件如何设计,在protel99se中,我们可以设计成如下led电路图:
我们在这里还是要设计自己的要求:
使用定时器0,定时1s,工作在方式1,使用16位计数方式,让D0在1s后取反闪烁。
现在我们计算一下:在定时器中最大的延时只能做到65多ms,无法做到1s,那么我们只能通过寄存器进行计算,定时器为50ms,累加到1s,然后对D0进行取反操作。
看如下汇编程序:
LED0 BIT P1.0 ;灯的输出信号
LED1 BIT P1.1
MSS EQU 30H ;20=1s
MS1 EQU 31H
;**************************************************
;主程序
;**************************************************
ORG 0000H
NOP
NOP
LJMP MAIN
ORG 00BH ;计数器
NOP
NOP
LJMP SCAN \
ORG 01BH ;计数器
NOP
NOP
LJMP SCAN1
ORG 0100H
MAIN: LCALL DEL_1S
MOV P1, #0FFH
MOV P3, #0FFH
MOV SP, #60H ;设置堆栈
MOV MSS, #00H
MOV TH0, #3CH ;50ms
MOV TL0, #0B0H
MOV TH1, #3CH ;50ms
MOV TL1, #0B0H
MOV TMOD, #11H
SETB EA ;打开定时中断
SETB TR0
SETB ET0
SETB TR1
SETB ET1
MOV P1, #0FFH
AJMP $
NOP
;***********************************************
;定时子程序
;***********************************************
SCAN: MOV TH0, #3CH
MOV TL0, #0B0H
PUSH ACC
INC MSS
MOV A, MSS
CJNE A, #20, SCA_E
MOV MSS, #00
CPL LED0 ;1S到了对LED0取反
AJMP SCA_E
SCA_E: POP ACC
RETI
SCAN1: MOV TH1, #3CH
MOV TL1, #0B0H
PUSH ACC
INC MS1
MOV A, MS1
CJNE A, #20, SCA_E1
MOV MS1, #00
CPL LED1 ;1S到了对LED0取反
AJMP SCA_E
SCA_E1: POP ACC
RETI
END
如下为c51的程序:
//---------------------------------------------------
#include <reg52.h>
//---------------------------------------------------
//重定义I/0引脚名称
sbit LED0=P1^0;
sbit LED1=P1^1;
//---------------------------------------------------
//全局变量及位标志定义
unsigned char MSS;
unsigned char MS1;
//---------------------------------------------------
//固定函数声明
void timer_0();
//定时器中断0
void timer_1();
//定时器中断1
//---------------------------------------------------
void main(){
TH0= 0x3C ;50ms
TL0=0x0B0;
TH1= 0x3C ;50ms
TL1=0x0B0;
EA=1; ;打开定时中断
TR0=1;
ET0=1;
TR1=1;
ET1=1;
TMOD=0X11;
while(1){
{
;
}
}
//-------------------------------------------------------------------
//定时器0中断
void timer_0() interrupt 1 using 2
{
TH0= 0x3C ;50ms
TL0=0x0B0;
MSS=MSS+1;
If(MSS>=20)
{
LED0=!LED0;
}
}
//-------------------------------------------------------------------
//定时器1中断
void timer_1() interrupt 3 using 3
{
TH1= 0x3C ;50ms
TL1=0x0B0;
MS1=MS1+1;
If(MS1>=20)
{
LED1=!LED1;
}
}
以上就是定时器的用法,有问题欢迎交流。