目录
定时器
晶振
时钟周期
机械周期
每经过一个机械周期经过多长时间
定时器和计数器的区别
定时器相关寄存器
定时10ms,相关寄存器的配置?
单片机中断
什么是中断
中断源
中断优先级
?中断函数
中断嵌套
中断相关寄存器
PWM信号?
超声波(HC-SR04)测距
感应开盖垃圾桶
思路
代码实现
定时器
晶振
晶体震荡器,又称数字电路的“心脏”,是各种电子产品里面必不可少的频率元器件。数字电路的所有工作都离不开时钟,晶振的好坏、晶振电路设计的好坏,会影响到整个系统的稳定性。
时钟周期
时钟周期也称为振荡周期,定义为时钟频率的倒数。时钟周期是计算机中最基本的、最小的时间单 位。在一个时钟周期内,CPU仅完成一个最基本的动作。时钟周期是一个时间的量。更小的时钟周 期就意味着更高的工作频率。
机械周期
机器周期也称为CPU周期。在计算机中,为了便于管理,常把一条指令的执行过程划分为若干个阶 段(如取指、译码、执行等),每一阶段完成一个基本操作。完成一个基本操作所需要的时间称为 机器周期。一般情况下,一个机器周期由若干个时钟周期组成。
每经过一个机械周期经过多长时间
以晶振频率11.0592MHZ为例,时钟周期为晶振的倒数,即1/1105920000秒?
- 如果是12T模式,机械周期 = 12 X 时钟周期,即12/11059200000秒?= 1.085微秒
- 如果是6T模式,机械周期 = 6?X 时钟周期,即6/11059200000秒?= 0.5425微秒
定时器和计数器的区别
51单片机中的定时器和计数器使用同一个硬件电路,通过修改寄存器的配置来将该硬件电路变成定时器或者计数器。
- 当配置成定时器时,每经过一个机械周期,计数存储器的值加1,C51有两个定时器T0和T1。
- 当配置成计数器时,每来一个负跳变信号(高电平跳到低电平),计数存储器的值加1。
定时器相关寄存器
定时器计时(TH和TL寄存器)
- ?当定时器的TH寄存器和TL寄存器都用起来,即一共有16位,那么该定时器最多数2^16 = 65536下,即大概65536*1.085微秒 =?71毫秒,也就是说定时器T0或者定时器T1最多定时71毫秒。
- 如现在需要使用定时器0定时10毫秒,怎么配置寄存器TH0和TL0,只需要配置TH0 = 0xDC , TL0 = 0x00。
定时器控制寄存器(TCON寄存器)
以定时器0和外部中断0为例
- TF标志位:当定时器0爆表后(即定时结束后),TF标志位,TF0会置1(TF0 =?1),此时会向CPU请求中断,如果中断条件允许的话就执行外部中断0,执行完中断后,TF0会硬件置0(TF = 0),当我们不想它执行中断就可以软件置0,即手动将TF0置0(TF =0)。
- TR标志位:当TR0 =1 时,定时器0才1会允许计数,即开始计时,当TR0 = 0时,不允许定时器0进行计数。
- IE标志位:当IE0 = 1时,会向CPU请求外部中断0,当CPU响应外部中断0后会将IE0硬件置零(IE0 = 0)。
- IT标志位:当IT0?= 1时,低电平触发外部中断0;当IT0?= 0时,下降沿触发外部中断0。
定时器模式寄存器(TMOD寄存器)
以定时器0为例
- GATE标志位:一般为0,GATE = 0时,当TR0 = 0时,定时器0开始计数。
- C/T标志位:一般为0,C/T为0时,让定时器0作为定时器
- M1、M0标志位:一般为0、1,16位定时器,TL0和TH0两个寄存器都使用。
通过定时器0,让蜂鸣器叫一秒,不叫一秒,定时器0爆表后,不执行中断?
#include "reg52.h"
sbit buzzer = P1^2;
void main()
{
int cnt = 0;
buzzer = 1;
TMOD = 0x01; //设置定时器0为16为计时模式
//设置定时器0定时时间为10ms
TH0 = 0xDC;
TL0 = 0x00;
TR0 = 1; //定时器0开始计时
TF0 = 0; //外部中断0标志位置0,不进行定时器0产生的中断
while(1){
//当定时器0爆表时
if(TF0 == 1){
TF0 = 0; //外部中断0标志位置0,不进行定时器0产生的中断
cnt++;
TH0 = 0xDC;
TL0 = 0x00;
if(cnt == 100){
cnt = 0;
buzzer = !buzzer;
}
}
}
}
定时10ms,相关寄存器的配置?
AUXR寄存器?
单片机中断
什么是中断
- 当中央处理机CPU正在处理某件事的时候外界发生了紧急事件请求,要求CPU暂停当前的工作,转而去处理这个紧急事件,处理完以后,再回到原来被中断的地方,继续原来的工作,这样的过程称为中断。
- 实现这种功能的部件称为中断系统。
- 请示CPU中断的请求源称为中断源。
- CPU总是先响应优先级别最高的中断请求。
中断源
中断优先级
?中断函数
中断嵌套
当CPU正在处理一个中断源请求的时候(执行相应的中断服务程序),发生了另外一个优先级比它还高的中断源请求。如果CPU能够暂停对原来中断源的服务程序,转而去处理优先级更高的中断请求源,处理完以后,再回到原低级中断服务程序,这样的过程称为中断嵌套。
中断相关寄存器
中断允许寄存器(IE寄存器)
?辅助中断控制寄存器(XICON寄存器)
?中断优先级控制寄存器高(IPH寄存器)
?中断优先级控制寄存器低(IP寄存器)
?对于XICON寄存器、IP寄存器、IPE寄存器可配置的一些功能
- IPH寄存器的位PX3H和XICON寄存器的位PX3可控制外部中断3的优先级
? - IPH寄存器的位PX2H和XICON寄存器的位PX2可控制外部中断2的优先级
? - IPH寄存器的位PX1H和IP寄存器的位PX1可控制外部中断1的优先级
? - IPH寄存器的位PX0H和IP寄存器的位PX0可控制外部中断0的优先级
? - IPH寄存器的位PSH和IP寄存器的位PS可控制串口1的优先级
? - IPH寄存器的位PTH2和IP寄存器的位PT2可控制定时器2的优先级
? - IPH寄存器的位PTH1和IP寄存器的位PT1可控制定时器1的优先级
? - IPH寄存器的位PTH0和IP寄存器的位PT0可控制定时器0的优先级
?通过定时器0和外部中断0,让蜂鸣器叫一秒,不叫一秒,定时器0爆表后,执行外部中断0
#include "reg52.h"
#include <intrins.h>
sbit buzzer = P1^2;
int cnt = 0;
void timer0Init()
{
TMOD = 0x01; //设置定时器0为16为计时模式
//设置定时器0定时时间为10ms
TH0 = 0xDC;
TL0 = 0x00;
TR0 = 1; //定时器0开始计时
TF0 = 0; //外部中断0标志位置0,不进行定时器0产生的中断
ET0 = 1; //定时器0中断开关
EA = 1; //总中断开关
}
void Delay300ms() //@11.0592MHz
{
unsigned char i, j, k;
_nop_();
i = 3;
j = 26;
k = 223;
do
{
do
{
while (--k);
} while (--j);
} while (--i);
}
void main()
{
buzzer = 1;
timer0Init()
while(1);
}
//定时器0的中断函数
void Time0Handler() interrupt 1
{
cnt++;
TH0 = 0xDC;
TL0 = 0x00;
if(cnt == 100){
buzzer = !buzzer;
cnt = 0;
}
}
PWM信号?
PWM,英文名Pulse Width Modulation,是脉冲宽度调制缩写,它是通过对一系列脉冲的宽度进行调制,等效出所需要的波形(包含形状以及幅值),对模拟信号电平进行数字编码,也就是说通过调节占空比的变化来调节信号、能量等的变化,占空比就是指在一个周期内,信号处于高电平的 时间占据整个信号周期的百分比,例如方波的占空比就是50%。
- PWM是脉冲宽度调制。
- 通过调节占空比来调节信号。
- 一个周期内,高电平占据的时长的百分比称为占空比。
软件模拟pwm波驱动sg90舵机转动不同角度,驱动sg90舵机时,PWM信号的频率不能太高,大概为50HZ,即0.02s(20ms)左右。
- 0.5ms--->舵机转动45°,即占空比为2.5%。、
- 1.0ms--->舵机转动90°,即占空比为5.0%。
- 1.5ms--->舵机转动135°,即占空比为7.5%。
- 2.0ms--->舵机转动45°,即占空比为10.0%。
- 2.5ms--->舵机转动180°,即占空比为12.5%。?
#include "reg52.h"
#include <intrins.h>
sbit sg90 = P1^2;
int angle;
int cnt;
void timer0Init()
{
TMOD = 0x01; //设置定时器0为16为计时模式
//设置定时器0定时时间为0.5ms
TH0 = 0xFE;
TL0 = 0x33;
TR0 = 1; //定时器0开始计时
TF0 = 0; //外部中断0标志位置0,不进行定时器0产生的中断
ET0 = 1; //定时器0中断开关
EA = 1; //总中断开关
}
void Delay300ms() //@11.0592MHz
{
unsigned char i, j, k;
_nop_();
i = 3;
j = 26;
k = 223;
do
{
do
{
while (--k);
} while (--j);
} while (--i);
}
void Delay1000ms() //@11.0592MHz
{
unsigned char i, j, k;
_nop_();
i = 8;
j = 1;
k = 243;
do
{
do
{
while (--k);
} while (--j);
} while (--i);
}
void main()
{
Delay300ms();
timer0Init();
angle = 1; //设置舵机转动角度45°
cnt = 0;
sg90 = 1;
while(1){
angle = 3; //设置舵机转动角度135°
cnt = 0;
Delay1000ms();
angle = 1; //设置舵机转动角度45°
cnt = 0;
Delay1000ms();
}
}
//定时器0的中断函数
void Time0Handler() interrupt 1
{
TH0 = 0xFE;
TL0 = 0x33;
//控制占空比
if(cnt < angle){
sg90 = 1;
}else{
sg90 = 0;
}
cnt++;
if(cnt == 40){ //每个周期为20ms
cnt = 0;
sg90 = 1;
}
}
超声波(HC-SR04)测距
超声波时序图
- 发送超声波:当Trig引脚接收到一个10微秒以上的高电平后开始发送超声波,当开始发送超声波后,Echo引脚会从低电平跳转到高电平。
- 接收超声波:当发出去的超声波返回来并被接收后,Echo引脚会从高电平跳转到低电平。
- 超声波从发出到被接收的时间:Echo持续高电平的时间,当超声波发出去的瞬间启动定时器,超声波被接收的瞬间停止定时器,查看中间经过的时间。
- 测距:距离 = 声音速度(340m/s)* 时间 / 2,除以2是因为超声波经过了两倍距离。
?超声波测距,距离少过10cm时,蜂鸣器叫,距离超过10cm时,蜂鸣器不叫
#include "reg52.h"
#include <intrins.h>
sbit buzzer = P1^4;
sbit Trig = P1^2;
sbit Echo = P1^3;
void Delay15us() //@11.0592MHz
{
unsigned char i;
i = 4;
while (--i);
}
void timer1Init()
{
TMOD = 0x10;
TH1 = 0x00;
TL1 = 0x00;
TF1 = 0;
}
void ultrasonicStart()
{
Trig = 0;
Trig = 1;
Delay15us();
Trig = 0;
}
void main()
{
double time = 0;
double distance = 0;
timer1Init();
while(1){
ultrasonicStart();
while(Echo == 0); //当Echo引脚从低电平跳到高电平时开启定时器1
TR1 = 1;
while(Echo == 1); //当Echo引脚从高电平跳到低电平时关闭定时器1
TR1 = 0;
time = (TH1*256 + TL1) * 1.085; //微秒
/*
定时器16位全用时:
高八位TH寄存器每加次1,计数存储器的值就加256;
低八位TL寄存器每次加1,计数存储器的值就加1;
计数存储器的值每次加1时,就经过了一个机械周期(经过时间1.085微秒)
*/
distance = time * 0.017; //CM
if(distance < 10){
buzzer = 0;
}else{
buzzer = 1;
}
//定时器1清0
TH1 = 0x00;
TL1 = 0x00;
}
}
感应开盖垃圾桶
思路
当检测到有人靠近垃圾桶时,垃圾桶通过sg90舵机开盖,并让蜂鸣器叫一秒,三秒后关盖。当发生垃圾桶震动时,?垃圾桶通过sg90舵机开盖,并让蜂鸣器叫一秒,三秒后关盖。(舵机的软件PWM用定时器0实现,超声波的距离检测用定时器1实现,震动传感器用外部中断1实现)
代码实现
#include "reg52.h"
#include <intrins.h>
sbit Trig = P1^5;
sbit Echo = P1^6;
sbit sg90 = P1^1;
sbit vibrate = P3^2;
sbit buzzer = P3^1;
int angle;
int angleBack;
int cnt = 0;
int markVibrate = 0;
void Delay100ms() //@11.0592MHz
{
unsigned char i, j;
i = 180;
j = 73;
do
{
while (--j);
} while (--i);
}
void Delay500ms() //@11.0592MHz
{
unsigned char i, j, k;
_nop_();
i = 4;
j = 129;
k = 119;
do
{
do
{
while (--k);
} while (--j);
} while (--i);
}
void Delay10us() //@11.0592MHz
{
unsigned char i;
i = 2;
while (--i);
}
void interrupt0Init()
{
EX0 = 1; //外部中断0中断开关
IT0 = 0; //外部中断0为低电平触发
}
void timer0Init()
{
//设置定时器0为16为计时模式
TMOD &=0xF0;
TMOD |=0x01;
//设置定时器0定时时间为0.5ms
TH0 = 0xFE;
TL0 = 0x33;
TR0 = 1; //定时器0开始计时
TF0 = 0; //不执行定时器0爆表时导致的中断
ET0 = 1; //定时器0中断开关
EA = 1; //总中断开关
}
void timer1Init()
{
//设置定时器1为16为计时模式
TMOD &= 0x0F;
TMOD |= 0x10;
TH1 = 0x00;
TL1 = 0x00;
TF1 = 0; //定时器1中断开关
}
void ultrasonicStart()
{
Trig = 0;
Trig = 1;
Delay10us();
Trig = 0;
}
double getDistance()
{
double time = 0;
//定时器1清0
TH1 = 0x00;
TL1 = 0x00;
ultrasonicStart();
while(Echo == 0); //当Echo引脚从低电平跳到高电平时开启定时器1
TR1 = 1;
while(Echo == 1); //当Echo引脚从高电平跳到低电平时关闭定时器1
TR1 = 0;
time = (TH1*256 + TL1) * 1.085; //微秒
return (time * 0.017);
}
void sg90Init()
{
angle = 1; //设置舵机转动角度45°
cnt = 0;
sg90 = 1;
}
//开盖
void openLib()
{
angle = 4; //设置舵机转动角度135°
if(angleBack != angle){
cnt = 0;
buzzer = 0;
Delay500ms();
buzzer = 1;
}
angleBack = angle;
}
//关盖
void closeLib()
{
angle = 1; //设置舵机转动角度45°
if(angleBack != angle){
cnt = 0;
}
angleBack = angle;
Delay100ms();
}
void main()
{
double distance = 0;
timer0Init();
timer1Init();
interrupt0Init();
sg90Init();
while(1){
//超声波测距
distance = getDistance();
if(distance < 10 || markVibrate == 1){
openLib();
markVibrate = 0;
}else{
closeLib();
}
}
}
//定时器0的中断函数
void Time0Handler() interrupt 1
{
TH0 = 0xFE;
TL0 = 0x33;
//控制占空比
if(cnt < angle){
sg90 = 1;
}else{
sg90 = 0;
}
cnt++;
if(cnt == 40){ //每个周期为20ms
cnt = 0;
sg90 = 1;
}
}
void Int0_Routine() interrupt 0
{
markVibrate = 1;
}
|