找回密码
 立即注册

QQ登录

只需一步,快速开始

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

怎样在这个超声波避障程序里添加pwm函数 想实现控制小车的速度

[复制链接]
跳转到指定楼层
楼主
ID:449368 发表于 2018-12-21 20:42 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
本帖最后由 裤衩青年 于 2018-12-21 20:50 编辑

/****************************************************************************
         硬件连接
        ****************************************************************************/

        #include<AT89x51.H>
        #include <intrins.h>

        #define Sevro_moto_pwm     P2_7           //接舵机信号端输入PWM信号调节速度

        #define  ECHO  P2_4                                   //超声波接口定义
        #define  TRIG  P2_5                                   //超声波接口定义
   

    #define Left_moto_go      {P1_0=1,P1_1=0;}                   //左电机向前走
        #define Left_moto_back    {P1_0=0,P1_1=1;}                         //左电机向后转
        #define Left_moto_Stop    {P1_0=0,P1_1=0;}                 //左电机停转                     
        #define Right_moto_go     {P1_2=1,P1_3=0;}                      //右电机向前走
        #define Right_moto_back   {P1_2=0,P1_3=1;}                     //右电机向后走
        #define Right_moto_Stop   {P1_2=0,P1_3=0;}                    //右电机停转   

   

        unsigned char const discode[] ={ 0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0xBF,0xff/*-*/};
        unsigned char const positon[3]={ 0xfe,0xfd,0xfb};
        unsigned char disbuff[4]          ={ 0,0,0,0,};
    unsigned char posit=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 delay(unsigned int k)          //延时函数
{   
     unsigned int x,y;
           for(x=0;x<k;x++)
             for(y=0;y<2000;y++);
}
/************************************************************************/
   /* void Display(void)                                  //扫描数码管
        {
         if(posit==0)
         {P0=(discode[disbuff[posit]])&0x7f;}//产生点
         else
         {P0=discode[disbuff[posit]];}

          if(posit==0)
         { P2_1=0;P2_2=1;P2_3=1;}
          if(posit==1)
          {P2_1=1;P2_2=0;P2_3=1;}
          if(posit==2)
          {P2_1=1;P2_2=1;P2_3=0;}
          if(++posit>=3)
          posit=0;
        }*/
/************************************************************************/
     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
          disbuff[0]=S%1000/100;   //更新显示
          disbuff[1]=S%1000%100/10;
          disbuff[2]=S%1000%10 %10;
        }
/************************************************************************/
//前速前进
     void  run(void)
{                                   
         Left_moto_go ;     //左电机往前走
         Right_moto_go ;    //右电机往前走
}
/************************************************************************/
//前速后退
     void  backrun(void)
{
         Left_moto_back ;   //左电机往后走
         Right_moto_back ;  //右电机往后走
}
/************************************************************************/
//左转
     void  leftrun(void)
{
         Left_moto_back ;   //左电机往后走
          Right_moto_go ;   //右电机往前走
}
/************************************************************************/
//右转
     void  rightrun(void)
{
         Left_moto_go ;     //左电机往前走
         Right_moto_back ;  //右电机往后走
}
/************************************************************************/
//STOP
     void  stoprun(void)
{
         Left_moto_Stop ;   //左电机停走
         Right_moto_Stop ;  //右电机停走
}
/************************************************************************/
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);
                     }
                                             

}

/************************************************************************/
/*                    PWM调制电机转速                                   */
/************************************************************************/
/*                    左电机调速                                        */
/*调节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();

         timer1++;                                 //2MS扫一次数码管
         if(timer1>=20)
         {
         timer1=0;
         Display();        
         }
}
/***************************************************/
///*TIMER0中断服务子函数产生PWM信号*/
         void timer0()interrupt 1   using 0
{        
           
}

/***************************************************/

        void main(void)
{

        TMOD=0X11;
        TH1=(65536-100)/256;          //100US定时
        TL1=(65536-100)%256;
        TH0=0;
        TL0=0;  
        TR1= 1;
        ET1= 1;
        ET0= 1;
        EA = 1;

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


        while(1)                       /*无限循环*/
        {

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

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

使用道具 举报

沙发
ID:449368 发表于 2018-12-21 20:52 | 只看该作者
在线等大佬   急求 解决方法
回复

使用道具 举报

板凳
ID:169944 发表于 2018-12-22 11:01 | 只看该作者
在中断里边加就行了,避障方式也没说,而且你写的直接给电机高低电平,也没用pwm,重新写个函数,对电机的控制引脚高低电平用pwm控制,来控制速度
回复

使用道具 举报

地板
ID:169944 发表于 2018-12-22 11:22 | 只看该作者
counter++;
if(counter >= 255) counter = 0;
    if(counter >= PWM_0) Left_moto_go; else Left_moto_Stop
    if(counter >= PWM_1) Right_moto_go; else Right_moto_Stop
把这个放中断里,修改PWM0和PWM1的值来改变速度,直行就用这个就行,你写的那个函数是全速跑的,理论上,可以通过改变PWm的值实现直行,停止,转弯,这个需要你实际测。
   
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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