完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
|
|
相关推荐
1个回答
|
|
第一部分 前言
本次单片机基础实验将会通过在Proteus软件中画原理图,然后Keil软件下编写源程序并编译形成可执行文件.hex,下载源程序,进行Protues和Keil的联合仿真运行,最终对单片机仿真有初步的认识。 通过本篇文章定时器/计数器T0、T1的工作方式选择和编程方法。学习Proteus了解定时器/计数器中断服务程序的设计方法 以下是本篇文章正文内容,所设计的案例可供参考 第二部分 定时器/计数器的原理 MCS-51定时器/计数器的结构 MCS-51单片机根据不同型号,其片内定时器/计数器数目不同。8051单片机有两个16位定时器/计数器寄存器:Timer0(T0定时器)与Timer1(T1定时器1)。8052除这两个定时器/计数器外还增加了1个Timer2(T2定时器),而这3个都可设置为定时器或事件计数器。当其作为“定时器”功能时,它是对标准时钟计数,每个时钟周期寄存器自动增1。由于MCS-51单片机的一个机器周期由12个振荡器周期组成,计数速率是振荡器频率的1/12。作为“计数器”功能,寄存器在响应相应的外部输入引脚T0、T1或T2(在8052中)由1至0的转变而增1。不论是“定时器”还是外部事件“计数器”,其工作原理是一样的,即定时器/计数器电路中的内部计数器从某一预定值(此值是可编程的)开始计数,当累计到最大值时产生溢出,并同时会建立一个相应的溢出标志(即中断标志位)。除了“定时器”或“计数器”选择外,定时器0与定时器1有4种工作方式需要选择。在8052单片机中的定时器T2有3种操作方式:“捕获”、“自动重装入”与“波特率生成器”。下面主要以AT89C51单片机为例讲解定时器/计数器的基本结构。 图1 单片机的定时器/计数器结构框图 AT89C51单片机片内定时器/计数器的结构如图1所示。定时器内部实质上是16位加法计数器,其控制电路受软件控制。当用作定时器时,对机器周期计数,每过一个机器周期,计数器加1。由于每个机器周期包含12个振荡信号周期,所以加1计数器的计数频率为振荡器信号频率的1/12。当用作计数器功能时,加1计数器的计数脉冲取自外部输入端T0(P3.4)和T1(P3.5),只要这些引脚上有从“1”到“0”的负跳变,计数器就加1。CPU在每个机器周期的S5P2时刻对外部输入状态进行采样,计数器加1的执行是在检测到跳变后的下一个机器周期的S3P1时刻。由于需要两个机器周期来识别一个从“1”到“0”的负跳变,所以最大计数频率为振荡信号频率的1/24。而外部时钟脉冲持续为0和为1的时间不能少于一个机器周期。 两个可编程的16位定时器 /计数器T0 和 T1 - T0 = TH0(高8位) + TL0(低8位) - T1 = TH1(高8位) + TL1(低8位) 定时器/计数器相关特殊功能寄存器
TMOD用来选择定时器/计数器的工作模式和工作方式,它的字节地址是89H,但该寄存器不能进行位寻址。
[tr]M1M0功能[/tr]
[tr]M1M0功能[/tr]
TCON寄存器的字节地址为88H,可进行位寻址。高4位分别为定时器/计数器的启动控制和溢出中断标志,低4位与外部中断控制有关, 各标志位的功能: 1. IT0—选择外部中断请求0为跳沿触发方式还是电平触发方式: IT0=0,为电平触发方式。 IT0=1,为跳沿触发方式。 可由软件置“1”或清“0”。 2. IE0—外部中断请求0的中断请求标志位。 IE0=0,无中断请求。 IE0=1,外部中断0有中断请求。 当CPU响应该中断,转向中断服务程序时,由硬件清“0”IE0 **3. IT1—外部中断请求1为跳沿触发方式还是电平触发方式,意义与IT0类似。
TR1、TR0 2个位与中断无关,仅与定时器/计数器T1和T0有关。 3.IE 中断允许寄存器:CPU对中断源的开放或屏蔽,由片内的中断允许寄存器IE控制。字节地址为A8H,可位寻址。格式如下: IE中各位的功能如下: 1. EA:中断允许总控制位 EA=0:CPU屏蔽所有的中断请求(CPU关中断) ; EA=1:CPU开放所有中断(CPU开中断) 。 五个中断源的中断请求是否允许,还要由IE中对应的5个中断请求允许控制位的状态来决定。 2. ES:串行口中断允许位 ES=0:禁止串行口中断; ES=1:允许串行口中断。 3. ET1:定时器/计数器T1的溢出中断允许位 ET1=0:禁止T1溢出中断; ET1=1:允许T1溢出中断。 4. EX1:外部中断1中断允许位 EX1=0:禁止外部中断1中断; EX1=1:允许外部中断1中断。 5. ET0:定时器/计数器T0的溢出中断允许位 ET0=0:禁止T0溢出中断; ET0=1:允许T0溢出中断。 6. EX0:外部中断0中断允许位。 EX0=0:禁止外部中断0中断; EX0=1:允许外部中断0中断。 MCS-51复位后,IE清0,所有中断请求被禁止。若使某一个中断源被允许中断,除了IE相应的位被置“1” ,还必须使EA位=1。改变IE的内容,可由位操作指令或字节操作指令来实现。 定时器/计数器工作方式 在MCS-51系列单片机中,定时器/计数器(T0、T1)具有多种工作方式,当选择工作方式不同,定时/计数器的使用方法差别很大。MCS-51单片机片内的定时器/计数器可以通过对特殊功能寄存器TMOD中的控制位 的设置来选择定时器方式或计数器方式;通过对M1 M0两位的设置来选择定时器/计数器的四种工作方式,下面具体讲解定时器/计数器的工作方式。 1. 工作方式0 定时器/计数器0、1在工作方式0时的电路逻辑结构见图9-2。工作方式0(M1 M0 = 0 0)是13位计数结构的工作方式,其计数器由TH的全部8位和TL的低5位构成,TL的高3位不使用。当 =0时,定时器/计数器0、1处于定时工作方式,多路开关接通振荡脉冲的12分频输出,13位计数器依次进行计数。当 =1时,定时器/计数器0、1处于计数工作方式,多路开关接通计数引脚(T0),外部计数脉冲由脚T0输入。当计数脉冲发生负跳变时,计数器加1。 当TL的低5位溢出时,都会向TH进位,而全部13位计数器溢出时,则会向计数器溢出标志位TF0进位。同时,GATA位的状态决定定时器运行控制取决于TR0一个条件还是TR0和/INT0引脚这两个条件。当GATA=1时,由于GATA信号封锁了与门,使引脚/INT0信号无效。而这时候如果TR0=1,则接通模拟开关,使计数器进行加法计数,即定时/计数器工作。而TR0=0,则断开模拟开关,停止计数,定时器/计数器不能工作。当GATA=0时,与门的输出端由TR0和INT0电平的状态确定,此时如果TR0=1,INT0=1与门输出为1,允许定时器/计数器计数,在这种情况下,运行控制由TR0和INT0两个条件共同控制,TR0是确定定时/计数器的运行控制位,由软件置位或清“0”。 在工作方式0下,计数器的计数值X范围为1~8192(2131)。由于MCS-51单片机的T0和T1采用加计数,因此TH0(TH1),TL0(TL1)的初值N=8192-X。如当计数值X=1000,则计数初值N=7192=1C18H,那么TH0(TH1),TL0(TL1)的值分别为0E0H和18H。由于TL0(TL1)为低5为有效,所以该计数初值N不能简单地分成高8位和低8位赋值给TH0(TH1),TL0(TL1)。当计数初值N=7192=1C18H,其二进制编码如下 1C18H=0001 1100 0001 1000B 再将16位的二进制编码去除最高的3位,保留后面13位,并取低5位写入到TL0(TL1),高8位写入到TH0(TH1),具体操作如下 1C18H=0001 1100 0001 1000 = 1110 0000 11000B 其中,11000B是TL0(TL1)对应的低5位,其16进制编码为18H,而1110 0000这8位为TH0(TH1)的内容,其16进制编码为0E0H。 当定时器/计数器工作于方式0且确定了定时时间T1后,其计数初值N的计算公式为 N=8192-(T1×fosc/12) 式中,fosc为系统时钟振荡频率。 假设单片机的晶振选为12MHz,需要用T0进行2ms定时控制,则T0的初值N计算为 N=8192-T1×fosc/12=8192-2×10-3×12×106/12=6192=1830H=0001 1000 0011 0000B 则对应的13位二进制编码为1100 0001 1 0000则TH0=0C1H,TL0=10H 2. 工作方式1 定时器/计数器0、1工作于方式1时,其电路逻辑结构如图所示。 工作方式1(M1 M0=0 1)是16位计数结构的工作方式。方式0和方式1的区别仅在于计数器的位数不同,方式0为13位,而方式1则为16位,由TH0作为高8位,TL0为低8位,有关控制状态字(GATA、TF0、TR0)和方式0相同。 在工作方式1下,计数器的计数值X范围是:1~65536 当定时器/计数器工作于方式1且确定了定时时间T1后,其计数初值N的计算公式为: N=65536-(T1×fosc/12) 则写入到8位寄存器TH0(TH1),TL0(TL1)值分别为 TH0(TH1)=N/256 TL0(TL1)=N%256 3. 工作方式2 当M1 M0=1 0时,定时器/计数器0、1处于工作方式2,此时其电路逻辑结构如图9-4所示。以定时/计数器0为例,定时/计数器1与之完全一致。 4. 工作方式3 当M1 M0 =1 1时,定时器/计数器工作于方式3下。方式3只适用于定时器T0,若将T1置为方式3,则它将停止计数,其效果相当于置TR1=0,即关闭定时器T1。当T0工作在方式3时,TH0和TL0被分成两个相互独立的8位计数器,其电路逻辑结构如图所示。 在工作方式3模式下,TL0既可以作为计数器使用,也可以作为定时器使用,定时器/计数器0的各控制位和引脚信号全归它使用。其功能和操作与方式0或方式1完全相同。但TH0的功能受到限制,只能作为简单的定时器使用,而且由于定时器/计数器0的控制位已被TL0占用,因此只能借用定时器/计数器1的控制位TR1和TF1,也就是以计数溢出去置位TF1,TR1则负责控制TH0定时器的启动和停止。由于TL0既能作定时器也能作计数器使用,而TH0只能作定时器使用而不能作计数器使用,因此在方式3模式下,定时/计数器0可以构成两个定时器或者一个定时器和一个计数器。 第三部分 定时器/计数器编程方法 定时器的初始化编程包括:
设计案例一 基于上述原理可以设计一个60s计时的秒表,完成秒表电路的设计与编程调试。 原理图 定时器T0或T1实现秒表的计时功能,工作模式不限,计时60s后自动从0开始重新计时, INT0中断方式实现秒表的启动和暂停计时; INT1中断方式实现秒表的计时重置。 动态仿真结果 代码 Keil代码如下: #include #define uchar unsigned char #define uint unsigned int ***it duan = P2^0; ***it wei = P2^1; ***it int0=P3^2; ***it int1=P3^3; uchar code LED_D[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71}; uchar code LED_W[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80}; uchar count=0; uchar Time=0; void Delay(uint x);//延迟函数 void Display2(uchar W1, Time);//显示函数 void QP();//清屏函数 //主函数 void main() { QP(); TMOD=0x01;//设置定时器工作方式 TH0=(65536-50000)/256;//计算初值,高8位 TL0=(65536-50000)%256;//计算初值,第8位 ET0=1; EA=1; //开总中断 while(1) { if(int0==0) {Delay(100); if(int0==0){ TR0=!TR0; while(int0==0);}} if(int1==0){ Delay(100); if(int1==0) {Time=0; }} Display2(6,Time); } } void LED_Flash() interrupt 1//中断函数 { EA=0;//关总中断 TH0=(65536-50000)/256;//重装初值 TL0=(65536-50000)%256;//重装初值 if(++count==20) //开始计数 { count=0; Time++; if(Time==61) Time=0; } EA=1; } void Int0() interrupt 0 { if(int0==0) {Delay(100); if(int0==0) { TR0=!TR0; while(int0==0); } } } void Int1() interrupt 2//中断函数 { EA=1; ET0=1; if(int1==0){ Delay(10); if(int1==0) {Time=0; } } } void Display2(uchar W1, Time)//显示函数 { uchar shi,ge; shi = Time/10; ge = Time%10; P0 = LED_D[shi]; duan = 1; duan = 0; P0 = LED_W[W1]; wei = 1; wei = 0; Delay(3); QP(); P0 = LED_D[ge]; duan = 1; duan = 0; P0 = LED_W[W1+1]; wei = 1; wei = 0; Delay(3); QP(); } void QP() { P0=0x00; duan = 1; duan = 0; wei = 1; wei = 0; } void Delay(uint x) //延时 { uchar t; while(x--) for(t=0;t<110;t++); } 设计案例二 利用T0、T1计数器功能,实现按键计数,计数值分别在2个4位数码管上显示。 原理图
通过不同的按键来实现对应的计数功能,利用定时器的计数功能可以使得单片机得到快速响应。 代码 #include #define uchar unsigned char #define uint unsigned int ***it duan = P2^0; ***it wei = P2^1; uchar code LED_D[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71}; uchar code LED_W[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80}; uint count1,count2; void Delay(uint x); void Display2(uchar W1,uint count); void QP(); void main() { QP(); EA=1; //开总中断 TMOD=0x46; //设置定时器工作方式 TH0=TL0=0; //设定初值 TH1=1;TL1=0; ET0=1; EX0=1; IT0=1; TR0=1; TR1=1; while(1) { if(TF1==1) TF1=0; count1 = TL0; count2 = TH1*256+TL1; Display2(1,count1); Display2(5,count2); } } void Flash() interrupt 1 { EA=0; TL1=0; EA=1; } void Clear() interrupt 0 { TL0=0; TL1=0; } void Display2(uchar W1, uint count) { uchar bai,shi,ge; bai = count%1000/100; shi = count%100/10; ge = count%10; P0 = LED_D[bai]; duan = 1; duan = 0; P0 = LED_W[W1]; wei = 1; wei = 0; Delay(3); QP(); P0 = LED_D[shi]; duan = 1; duan = 0; P0 = LED_W[W1+1]; wei = 1; wei = 0; Delay(3); QP(); P0 = LED_D[ge]; duan = 1; duan = 0; P0 = LED_W[W1+2]; wei = 1; wei = 0; Delay(3); QP(); } void QP() { P0=0x00; duan = 1; duan = 0; wei = 1; wei = 0; } void Delay(uint x) { uchar t; while(x--) for(t=0;t<110;t++); } 以上就是本次定时器/计数器的全部内容,希望通过本次的讲解能够加深大家对于定时器/计数器以及中断的理解。文章的内容可能存在部分不足之处,如有错误,请在评论区指出,谢谢。如果大家觉得文章有用,麻烦点赞关注,感谢支持,后续将会推出单片机的进阶设计文章。 |
||
|
||
只有小组成员才能发言,加入小组>>
2253 浏览 0 评论
1287浏览 3评论
741浏览 2评论
410浏览 1评论
1269浏览 0评论
166浏览 0评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-7-21 05:30 , Processed in 1.479148 second(s), Total 81, Slave 61 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号