历史上的今天
今天是:2024年11月13日(星期三)
2019年11月13日 | PIC16F684 对输入脉冲进行计数和脉冲宽度测量
2019-11-13 来源:51hei
使用cnnt 对脉冲计数,TMR1对上升沿计时,TMR2对下降沿计时,TMR1设定为100MS的定时器,TMR2 设定为40MS的定时器。
脉冲周期 为50HZ到300HZ的变化,脉冲宽度为2MS到20MS的变化,检测脉冲的个数以及宽度。
按下RA0(UP)电机正转,开始对RA2外部脉冲计数判断宽度。cnnt=380时。停止电机转动
按下RA5(DN)电机反转,开始对RA2外部脉冲计数判断宽度。cnnt==0时。 停止电机转动
程序里边用了RA1对是否进入中断进行判断。目前程序检测不到RA1的变化。是否我的设计架构方案不对。
#include
__CONFIG(0x3004);//(0X30C4);
#define T0_40MS 100 // 定义 TMR0 延时10MS 的时间常数
#define T1_100MS 40536
#define DN RA5 //Motor rise 2 DN
#define UP RA0 //Motor decline 13 UP
#define MOTA RC5 //Motor Forward 5
#define MOTB RC4 //Motor Reversal 6
bit rise; //正反转标志位
bit tmrl; //正反转标志位
bit tmrh; //正反转标志位
unsigned int num; //脉冲总个数
unsigned int cnnt; //脉冲计数
unsigned int timerl; //TMR1脉冲上升沿时间
unsigned int timerh; //TMR1脉冲下降沿时间
unsigned int cnnt_timer;//TMR1脉冲总时间
unsigned int cnnt_tmr2; //TMR2脉冲总时间
unsigned int signal_key();
unsigned int getkey() ;
void Delay_MS(unsigned int t);
void initial()
{
TRISA=0x3D;//3D;RA2=1,
TRISC=0X0E;
OPTION=0x87;//RA2内部中断时源
INTCON=0xF0;//外围中断允许
CMCON0 = 0X07;
ANSEL = 0;
// TMR0=T0_40MS;//启动TMR0自带振荡器,分频比为1:256
T1CON=0X22;//2启动TMR1自带振荡器,分频比为1:4关闭TRM1ON=0;
TMR1H=T1_100MS>>8;//0X80; //设初值
TMR1L==T1_100MS;//0X00;
TMR1IE=1; //TMR1中断允许
T2CON=0x63; //TMR2 预分频系数为1:16 ,后分频系数为1:13,开始工作 关闭TRM2ON=0;
PR2=239; //TMR2的溢出值,当 TMR2 为此值+1时溢出
TMR2IE=1; //TMR1中断允许 50MS
cnnt=0;
cnnt_timer=0;
num=10;
}
void Delay_MS(unsigned int t)
{
unsigned int a,b;
for(a=t;a>0;a--)
for(b=110;b>0;b--);
}
void Forward()
{
rise=1;
MOTB=1;
MOTA=0;
signal_key();
}
void Reversal()
{
rise=0;
MOTB=0;
MOTA=1;
signal_key();
}
void Stop()
{
MOTA=0;
MOTB=0;
}
unsigned int getkey()
{
if(!UP)
{
while(!UP)
{
if(cnnt==num){Stop();break;}
Forward();
}
}
if(!DN)
{
while(!DN)
{
if(cnnt==0){Stop();break;}
Reversal();
}
}
while(DN&&UP)
{
tmrh=0;
tmrl=0;
break;
}
}
unsigned int signal_key()
{
//---------------------- 信号下降沿检测 --------------------------//
while(!RA2) //再次确认信号,没有按下信号则退出
{
tmrh=0;
if(tmrl) break;
tmrl=1;
RA1=0;
INTF=0;
TMR1ON=0;
TMR2ON=1;
if(TMR2IF==1)//检测是否50MS错误信号,停止运行
{
TMR2ON=0;
TMR2IF=0;
PR2=239; //设初值
Stop(); //停止电机运行
return;
}
}
//---------------------- 信号上升沿检测 --------------------------//
while(RA2)
{
tmrl=0;
if(tmrh) break;
tmrh=1;
TMR2ON=0;
INTF=1;
if(TMR1IF==1)//检测是否100MS错误信号,停止运行
{
TMR1ON=0;
TMR1IF=0;
TMR1H=T1_100MS>>8;//0X80; //设初值
TMR1L=T1_100MS;//0X00;
Stop(); //停止电机运行
return;
}
break;
}
//---------------------- 取出上升沿脉冲信号时间 --------------------------//
if(TMR1ON==0)
{
timerh=TMR1H;
timerl=TMR1L;
cnnt_timer = timerh<<8;
cnnt_timer += timerl;
TMR1IF=0;
TMR1H=T1_100MS>>8;//0X80; //设初值
TMR1L=T1_100MS;//0X00;
}
//---------------------- 取出下降沿脉冲信号时间 --------------------------//
if(TMR2ON==0)
{
cnnt_tmr2 = TMR2;
TMR2IF=0;
PR2=239; //设初值
}
}
void interrupt ISR(void)
{
if(INTF==1) //脉冲开始计数,同时开启TMR1上升降计时
{
INTF=0;
RA1=1;
if(rise) cnnt++;
else cnnt--;
TMR1ON=1;
}
if( TMR1IF==1)
{
TMR1IF=0;
TMR1H=T1_100MS>>8;//0X80; //设初值
TMR1L=T1_100MS;//0X00;
}
if( TMR2IF==1)
{
TMR2IF=0;
PR2=239; //设初值
}
}
void main()
{
initial();
while (1)
{
getkey();
}
}
下一篇:超声波PIC单片机C程序
史海拾趣
|
《程序员》杂志的编辑约我写一篇命题作文,想了几天都无从下手不知道写什么才好。在这篇文章里,我不打算将创业的艰辛与喜悦重新回忆一遍,我确实不想去误导大家,因为我所处的年代是一个物质缺乏的年代,成功相对来说要容易得多。每个人的成长 ...… 查看全部问答> |
|
这是一个现实数字和英文字母的函数,我觉得他给你字模显示出的字太小,所以想自己重新做字模。我的一个字大小是32字节,不知道应该怎么改程序啊? void PrintASCII(WORD x,WORD y, BOOL asciicode) // 坐标X为像素列块0-79列块,3像素点数据/列 ...… 查看全部问答> |
|
如何在eVC下用__try来捕获文件异常?是用序列化的方式来读取文件数据的.提供的SDK只能用__try. try catch不可以用,在微软上下载了ritti.exe,但是对我这个wince的SDK没有作用. 我用的是工控机… 查看全部问答> |
|
. 酒店VOD多媒体信息服务网络系统 VOD技术就是让人们随时随地按需点播及互传多媒体信息的网络技术。建立宽带高速交互式VOD网络系统是网络社会发展的最高目标,是实现地球村理想的必要条件。 “酒店VOD及多媒体信息服务网络系统”是用于客人查询及 ...… 查看全部问答> |
|
熟悉430的做过无线电子作品的,有兴趣参加利尔达全国物联网应用设计大赛的 请进 不知道大家是否知道利尔达举办的 利尔达杯 首届全国物联网应用设计大赛? 具体大家可以看这个链接,奖品什么的还是很优厚的,而且还有挺丰富的开发板芯片等免费申请或者优惠价格购买的优惠政策。 http://www.lierda.com/topic/iot ...… 查看全部问答> |
|
小弟现在遇到一点小问题,就是我想在单片机开发板上用一根usb线供电和下载程序到单片机上,我们看到的是用ch340和max232芯片完成的,我想问的是不是max232是多余的,有没有大虾发个电路图来看看。… 查看全部问答> |




