找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3897|回复: 5
打印 上一主题 下一主题
收起左侧

蓝牙串口 和 超声波测距,中断使用有冲突 这个怎么解决

[复制链接]
跳转到指定楼层
楼主
ID:312541 发表于 2018-7-2 10:03 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
#include<reg52.h>
#include <intrins.h>
#define uchar unsigned char
#define uint unsigned int


sbit ina=P1^0;
sbit IN1=P1^1;
sbit IN2=P1^2;

sbit inb=P1^5;
sbit IN3=P1^3;
sbit IN4=P1^4;


sbit inc=P0^0;
sbit IN5=P0^1;
sbit IN6=P0^2;

sbit ind=P0^5;
sbit IN7=P0^3;
sbit IN8=P0^4;

         sbit Sevro_moto_pwm  = P2^7;           //接舵机信号端输入PWM信号调节速度
     sbit  ECHO=  P0^7;                                   //超声波接口定义
          sbit  TRIG= P0^6;                                   //超声波接口定义
                                                          
#define left_go     {inc=1,IN5=1,IN6=0,inb=1,IN3=1,IN4=0;}//P1:0-3控制左边   前
#define left_back   {inc=1,IN5=0,IN6=1,inb=1,IN3=0,IN4=1;}                                 //后
#define left_stop   {inc=0,IN5=0,IN6=0,inb=0,IN3=0,IN4=0;}                                 //停止
#define right_go    {ina=1,IN1=1,IN2=0,ind=1,IN7=1,IN8=0;}//P1:4-7控制右边
#define right_back  {ina=1,IN1=0,IN2=1,ind=1,IN7=0,IN8=1;}                         //
#define right_stop  {ina=0,IN1=0,IN2=0,ind=0,IN7=0,IN8=0;}



unsigned char pwm_val_left  = 0;//变量定义
        unsigned char push_val_left =14;//舵机归中,产生约,1.5MS 信号
    unsigned long S=0;
        unsigned long S1=0;
        unsigned long S2=0;
        unsigned long S3=0;
        unsigned long S4=0;
        unsigned int  time=0;                    //时间变量
        unsigned int  timer=0;                        //延时基准变量
        unsigned char timer1=0;                        //扫描时间变量
       
        void run();
void backrun();
void leftrun();
void rightrun();
void stoprun();
void init();
void init();
uchar flag,com;


/************************************************************************/       
//延时函数       
   void delay(unsigned int k)
{   
     unsigned int x,y;
         for(x=0;x<k;x++)
           for(y=0;y<2000;y++);
}
       
       
        void run()           //前进
{
        left_go;
        right_go;
}
void backrun()                //后退
{
        left_back;
        right_back;
}
void leftrun()                //左转
{
        left_back;
        right_go;
}
void rightrun()           //右转
{
        left_go;
        right_back;
}
void stoprun()                //停止
{
        left_stop;
        right_stop;
}       


  void  StartModule()                       //启动测距信号
   {
          TRIG=1;                                        
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          TRIG=0;
   }
                 /***************************************************/
          void Conut(void)                   //计算距离
        {
          while(!ECHO);                       //当RX为零时等待
          TR0=1;                               //开启计数
          while(ECHO);                           //当RX为1计数并等待
          TR0=0;                                   //关闭计数
          time=TH0*256+TL0;                   //读取脉宽长度
          TH0=0;
          TL0=0;
          S=(time*1.7)/100;        //算出来是CM
          
        }
/************************************************************************/
void  COMM( void )                      
  {
               
                 
                  push_val_left=5;          //舵机向左转90度
                  timer=0;
                  while(timer<=4000); //延时400MS让舵机转到其位置
                  StartModule();          //启动超声波测距
          Conut();                           //计算距离
                  S2=S;  

                  push_val_left=23;          //舵机向右转90度
                  timer=0;
                  while(timer<=4000); //延时400MS让舵机转到其位置
                  StartModule();          //启动超声波测距
          Conut();                          //计算距离
                  S4=S;        
       

                  push_val_left=14;          //舵机归中
                  timer=0;
                  while(timer<=4000); //延时400MS让舵机转到其位置
                  StartModule();          //启动超声波测距
          Conut();                          //计算距离
                  S1=S;        

          if((S2<20)||(S4<20)) //只要左右各有距离小于,20CM小车后退
                  {
                  backrun();                   //后退
                  timer=0;
                  while(timer<=4000);
                  }
                  
                  if(S2>S4)                 
                     {
                                rightrun();          //车的左边比车的右边距离小        右转       
                        timer=0;
                        while(timer<=4000);
                     }                                     
                       else
                     {
                       leftrun();                //车的左边比车的右边距离大        左转
                       timer=0;
                       while(timer<=4000);
                     }
                                            

}
/*调节push_val_left的值改变电机转速,占空比            */
                void pwm_Servomoto(void)
{  

    if(pwm_val_left<=push_val_left)
               Sevro_moto_pwm=1;
        else
               Sevro_moto_pwm=0;
        if(pwm_val_left>=200)
        pwm_val_left=0;

}

//*TIMER1中断服务子函数产生PWM信号*/
        void time1()interrupt 3   using 2
{       
     TH1=(65536-100)/256;          //100US定时
         TL1=(65536-100)%256;
         timer++;                                  //定时器100US为准。在这个基础上延时
         pwm_val_left++;
         pwm_Servomoto();

         
}

void csb() //
{
if(timer>=1000)          //100MS检测启动检测一次
           {
               timer=0;
                   StartModule(); //启动检测
           Conut();                  //计算距离
           if(S<20)                  //距离小于20CM
                   {
                   stoprun();          //小车停止
                   COMM();                   //方向函数
                   }
                   else
                   if(S>30)                  //距离大于,30CM往前走
                   run();
           }
}




void init()         //串口
{
   TMOD=0x20;
        TH0=0xfd;
        TL0=0xfd;
        TR0=1;
        REN=1;
        SM0=0;
        SM1=1;
        EA=1;
        ES=1;
        flag=0;
}

   void ser() interrupt 4
{
        RI=0;
        com=SBUF;
        flag=1;
}
/************************************************************************/


/*********************************************************************/                 
/*--主函数--*/
        void main(void)
{

         //IP=0x10;
        TMOD =0X10;
        TH1=(65536-100)/256;          //100US定时
        TL1=(65536-100)%256;
       
        TR1= 1;
        ET1= 1;
       
        EA = 1;

        delay(100);
    push_val_left=14;          //舵机归中
                init();                  

        while(1)                       /*无限循环*/
        {
                if(flag==1)
                {
                        switch(com)
                        {
                                case 'A':
                                        run();
                                        break;
                                case 'B':
                                        backrun();
                                        break;
                                  case 'C':
                                        leftrun();
                                        break;
                                case 'D':
                                        rightrun();
                                        break;
                                case 'E':
                                        stoprun();
                                        break;
                                        case '1':
                                         TH0=0;TL0=0;ET0= 1; csb();break;
                                                break;


                                default:
                                        break;
                        }

                }
                RI=1;
        }
         

        }
       
               


分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏1 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:164602 发表于 2018-7-2 14:04 | 只看该作者
不知道你用的是什么单片机?!
如果是一般的,如STC89系列,单片机只有三个定时器,而你的程序要四个(超声波两个、串口一个、PWM一个)。没关系,修改一下程序,用其它方式定时发射超声波就好了。这样就刚好够。
你的程序是100ms超声波检测一次,其实,超声波检测时间间隔在40ms之间是没有问题的。所以,自己看看程序在超声波检测后的语句,大致要多少时间,是可以完成的。

评分

参与人数 1黑币 +70 收起 理由
admin + 70 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

板凳
ID:312541 发表于 2018-7-2 15:56 | 只看该作者
HC6800-ES-V2.0 发表于 2018-7-2 14:04
不知道你用的是什么单片机?!
如果是一般的,如STC89系列,单片机只有三个定时器,而你的程序要四个(超声 ...

你好我用的是at89c52  我这里只用了超声波用定时了两个的器吗 不是只用一个T0来启动测距信号吗
用其它方式定时发射超声波就好   用T2可行吗
回复

使用道具 举报

地板
ID:164602 发表于 2018-7-3 08:53 | 只看该作者
121522121 发表于 2018-7-2 15:56
你好我用的是at89c52  我这里只用了超声波用定时了两个的器吗 不是只用一个T0来启动测距信号吗
用其它 ...

看清楚你的程序了。
你的T0复用了——波特率和超声波测距,这样当然是不行的嘛。
你的T1也复用了——既是PWM波控制,又是发超声波控制,完全可行。

所以,超声波测距用T2,应该就没问题了。
回复

使用道具 举报

5#
ID:312541 发表于 2018-7-4 12:25 | 只看该作者
HC6800-ES-V2.0 发表于 2018-7-3 08:53
看清楚你的程序了。
你的T0复用了——波特率和超声波测距,这样当然是不行的嘛。
你的T1也复用了——既 ...

谢谢 现在我能结合在一起了但  进去超声波功能 就出不来了 只在超声波功能  蓝牙控制不了这是为什么

评分

参与人数 1黑币 +1 收起 理由
Chen666666 + 1 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

6#
ID:364810 发表于 2018-7-4 14:41 | 只看该作者
设置好中断优先级,让串口中断优先进行就可以了,也可以在串口中断时关闭超声波的定时器中断
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表