找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 4999|回复: 1
收起左侧

数字罗盘(GY-271)的51单片机小车

[复制链接]
ID:183949 发表于 2017-4-2 15:14 | 显示全部楼层 |阅读模式
数字罗盘采用IIC通信,小车电机采用定时器中断PWM波调速,但是调速功能不能出现,小车电机一直全速转动。
网上有人说如果使用PWM调速,因为52单片机的模拟IIC与PWM冲突(尚未解决),采取的是当需要改变小车方向时,关PWM中断及定时器。可以在操作完了重新打开。我也加了,但效果还是这样。
下面是代码:
  1. #include "reg52.h"
  2. #include "intrins.h"
  3. #include "math.h"
  4. #include "stdio.h"
  5. typedef   unsigned char u8;
  6. typedef   unsigned int  u16;
  7. #define  Data_Port P0                        //LCD1602数据端口
  8. #define        Slave_Address   0x1a        
  9. sbit        SCL=P1^7;      //IIC时钟线
  10. sbit        SDA=P3^2;      //IIC数据线
  11. sbit    LCD_RS=P3^5;   //LCD1602命令端口               
  12. sbit    LCD_RW=P3^6;   //LCD1602命令端口               
  13. sbit    LCD_EN=P3^4;   //LCD1602使能端口
  14. sbit    LED=P1^2;
  15. sbit zea=P2^1;         //定义接口                                       
  16. sbit zIN1=P2^5;                        //转向轮
  17. sbit zIN2=P2^2;                        //转向轮
  18. sbit zIN3=P2^4;
  19. sbit zIN4=P2^6;
  20. sbit zeb=P2^0;
  21. sbit yea=P1^1;         //z为左边,y为右边                                
  22. sbit yIN1=P1^3;
  23. sbit yIN2=P1^4;
  24. sbit yIN3=P1^6;                        //转向轮
  25. sbit yIN4=P1^5;                        //转向轮
  26. sbit yeb=P1^2;
  27. u16 ynum1=0;               
  28. u16 znum1=0;
  29. u8 ge=0,shi=0,bai=0,qian=0,wan=0;     //显示变量            
  30. u8 BUF[8]=0;                                                  //数据缓冲区
  31. void Delay_5us()
  32. {
  33.         _nop_();_nop_();_nop_();_nop_();
  34.         _nop_();_nop_();_nop_();_nop_();
  35.         _nop_();_nop_();_nop_();_nop_();
  36.         _nop_();_nop_();_nop_();_nop_();
  37.         _nop_();_nop_();_nop_();_nop_();
  38.         _nop_();_nop_();_nop_();_nop_();
  39.         _nop_();_nop_();_nop_();_nop_();
  40.         _nop_();_nop_();_nop_();_nop_();
  41. }
  42. void Delay_ms(u16 t)        
  43. {                                                
  44.         u16 i,j;                                
  45.         for(i=t;i>0;i--)
  46.         for(j=121;j>0;j--);                                                
  47. }


  48. /**************************************
  49. 起始信号
  50. **************************************/
  51. void QMC5883_Start()
  52. {
  53.     SDA = 1;                    //拉高数据线
  54.     SCL = 1;                    //拉高时钟线
  55.     Delay_5us();                 //延时
  56.     SDA = 0;                    //产生下降沿
  57.     Delay_5us();                 //延时
  58.     SCL = 0;                    //拉低时钟线
  59. }


  60. /**************************************
  61. 停止信号
  62. **************************************/
  63. void QMC5883_Stop()
  64. {
  65.     SDA = 0;                    //拉低数据线
  66.     SCL = 1;                    //拉高时钟线
  67.     Delay_5us();                 //延时
  68.     SDA = 1;                    //产生上升沿
  69.     Delay_5us();                 //延时
  70. }


  71. /**************************************
  72. 发送应答信号
  73. 入口参数:ack (0:ACK 1:NAK)
  74. **************************************/
  75. void QMC5883_SendACK(bit ack)
  76. {
  77.     SDA = ack;                  //写应答信号
  78.     SCL = 1;                    //拉高时钟线
  79.     Delay_5us();                 //延时
  80.     SCL = 0;                    //拉低时钟线
  81.     Delay_5us();                 //延时
  82. }


  83. /**************************************
  84. 接收应答信号
  85. **************************************/
  86. bit QMC5883_RecvACK()
  87. {
  88.     SCL = 1;                    //拉高时钟线
  89.     Delay_5us();                //延时
  90.     CY = SDA;                   //进位标志 读应答信号
  91.     SCL = 0;                    //拉低时钟线
  92.     Delay_5us();                //延时
  93.     return CY;
  94. }


  95. /**************************************
  96. 向IIC总线发送一个字节数据
  97. **************************************/
  98. void QMC5883_SendByte(u8 dat)
  99. {
  100.     u8 i;


  101.     for (i=0; i<8; i++)         //8位计数器
  102.     {
  103.         dat <<= 1;              //移出数据的最高位
  104.         SDA = CY;               //送数据口
  105.         SCL = 1;                //拉高时钟线
  106.         Delay_5us();             //延时
  107.         SCL = 0;                //拉低时钟线
  108.         Delay_5us();             //延时
  109.     }
  110.     QMC5883_RecvACK();
  111. }


  112. /**************************************
  113. 从IIC总线接收一个字节数据
  114. **************************************/
  115. u8 QMC5883_RecvByte()
  116. {
  117.     u8 i;
  118.     u8 dat = 0;


  119.     SDA = 1;                    //使能内部上拉,准备读取数据,
  120.     for (i=0; i<8; i++)         //8位计数器
  121.     {
  122.         dat <<= 1;
  123.         SCL = 1;                //拉高时钟线
  124.         Delay_5us();             //延时
  125.         dat |= SDA;             //读数据               
  126.         SCL = 0;                //拉低时钟线
  127.         Delay_5us();             //延时
  128.     }
  129.     return dat;
  130. }


  131. //************************写入单字节数据***************************


  132. void Single_Write_QMC5883(u8 REG_Address,u8 REG_data)
  133. {
  134.     QMC5883_Start();                  //起始信号
  135.     QMC5883_SendByte(Slave_Address);   //发送设备地址+写信号
  136.     QMC5883_SendByte(REG_Address);    //内部寄存器地址,请参考中文pdf
  137.     QMC5883_SendByte(REG_data);       //内部寄存器数据,请参考中文pdf
  138.     QMC5883_Stop();                   //发送停止信号
  139. }


  140. //************************读取单字节数据*************************
  141. //u8 Single_Read_QMC5883(u8 REG_Address)
  142. //{   
  143. //        u8 REG_data;
  144. //        QMC5883_Start();                          //起始信号
  145. //        QMC5883_SendByte(Slave_Address);           //发送设备地址+写信号
  146. //        QMC5883_SendByte(REG_Address);            //发送存储单元地址,从0开始        
  147. //        QMC5883_Start();                          //起始信号
  148. //        QMC5883_SendByte(Slave_Address+1);         //发送设备地址+读信号
  149. //        REG_data=QMC5883_RecvByte();              //读出寄存器数据
  150. //        QMC5883_SendACK(1);   
  151. //        QMC5883_Stop();                           //停止信号
  152. //  return REG_data;
  153. //}


  154. //******************************************************
  155. //连续读出QMC5883内部角度数据,地址范围0x00~0x05
  156. //******************************************************
  157. void Multiple_Read_QMC5883(void)
  158. {   
  159.     u8 i;
  160.     QMC5883_Start();                          //起始信号
  161.     QMC5883_SendByte(Slave_Address);          //发送设备地址+写信号
  162.     QMC5883_SendByte(0x00);                   //发送存储单元地址,从0x00开始        
  163.     QMC5883_Start();                          //起始信号
  164.     QMC5883_SendByte(Slave_Address+1);        //发送设备地址+读信号
  165.          for (i=0; i<6; i++)                      //连续读取6个地址数据,存储中BUF
  166.     {
  167.         BUF[i] = QMC5883_RecvByte();          //BUF[0]存储数据
  168.         if (i == 5)
  169.         {
  170.            QMC5883_SendACK(1);                //最后一个数据需要回非应答NOACK
  171.         }
  172.         else
  173.         {
  174.           QMC5883_SendACK(0);                 //应答ACK
  175.        }
  176.    }
  177.     QMC5883_Stop();                           //停止信号
  178.     Delay_ms(5);
  179. }


  180. //初始化QMC5883,根据需要请参考pdf进行修改****
  181. void Init_QMC5883()
  182. {


  183.         Single_Write_QMC5883(0x09,0x0d);  //控制寄存器配置
  184.         Single_Write_QMC5883(0x0b,0x01);  //设置清除时间寄存器
  185.         Single_Write_QMC5883(0x20,0x40);  //
  186.         Single_Write_QMC5883(0x21,0x01);  //        
  187. }


  188. /***************等待LCD使能****************/
  189. void Wait_For_Enable(void)        
  190. {                                       
  191.         Data_Port=0xff;               
  192.         LCD_RS=0;
  193.         LCD_RW=1;
  194.         _nop_();
  195.         LCD_EN=1;
  196.         _nop_();
  197.         _nop_();
  198.         while(Data_Port&0x80);        
  199.         LCD_EN=0;                                
  200. }                                       
  201. /**************写LCD命令函数*****************/
  202. void Write_Command_LCD(u8 CMD,u8 Attribc)
  203. {                                       
  204.         if(Attribc)Wait_For_Enable();        
  205.         LCD_RS=0;
  206.         LCD_RW=0;
  207.         _nop_();
  208.         Data_Port=CMD;
  209.         _nop_();        
  210.         LCD_EN=1;
  211.         _nop_();
  212.         _nop_();
  213.         LCD_EN=0;
  214. }                                       
  215. /**************写LCD数据寄存器*****************/
  216. void Write_Data_LCD(u8 dataW)
  217. {                                       
  218.         Wait_For_Enable();               
  219.         LCD_RS=1;
  220.         LCD_RW=0;
  221.         _nop_();
  222.         Data_Port=dataW;
  223.         _nop_();        
  224.         LCD_EN=1;
  225.         _nop_();
  226.         _nop_();
  227.         LCD_EN=0;
  228. }               
  229. /****************初始化LCD*******************/
  230. void Init_Lcd()                                
  231. {                        
  232.         Write_Command_LCD(0x38,1);        
  233.         Write_Command_LCD(0x08,1);        
  234.         Write_Command_LCD(0x01,1);        
  235.         Write_Command_LCD(0x06,1);        
  236.         Write_Command_LCD(0x0c,1);
  237. }                        
  238. /****************显示一个字符*******************/
  239. void Display_One_Char(u8 X,u8 Y,u8 DData)
  240. {                                                
  241.         Y&=1;                                                
  242.         X&=15;                                                
  243.         if(Y)X|=0x40;                                       
  244.         X|=0x80;                        
  245.         Write_Command_LCD(X,0);               
  246.         Write_Data_LCD(DData);               
  247. }                                                




  248. //******************************显示在LCD的数据取位********************************//
  249. void Conversion(u16 temp_data)  
  250. {  
  251.     wan=temp_data/10000+0x30 ;
  252.     temp_data=temp_data%10000;   //取余运算
  253.         qian=temp_data/1000+0x30 ;
  254.     temp_data=temp_data%1000;    //取余运算
  255.     bai=temp_data/100+0x30   ;
  256.     temp_data=temp_data%100;     //取余运算
  257.     shi=temp_data/10+0x30    ;
  258.     temp_data=temp_data%10;      //取余运算
  259.     ge=temp_data+0x30;         
  260. }




  261. void fanwei()
  262. {
  263. //        u16 i;
  264.         int x=0,y=0,z=0;
  265.         double angle=0;        
  266.     Delay_ms(200);
  267.         Init_Lcd();
  268.         Init_QMC5883();
  269.         Delay_ms(300);
  270.         Multiple_Read_QMC5883();                                      //连续读取三轴角度数据,存储在BUF中
  271.                 //---------显示XY轴
  272.                 x=BUF[1] << 8 | BUF[0]; //Combine MSB and LSB of X Data output register  最高有效位
  273.                 y=BUF[3] << 8 | BUF[2]; //Combine MSB and LSB of Y Data output register
  274.                 z=BUF[5] << 8 | BUF[4]; //Combine MSB and LSB of Z Data output register
  275.                 angle= atan2((double)y,(double)x) * (180 / 3.14159265) + 180; //计算角度
  276.                 angle*=10;
  277.                 Conversion(angle);       //计算角度数据和显示
  278.                 Display_One_Char(2,0,'A');
  279.                 Display_One_Char(3,0,':');
  280.                 Display_One_Char(4,0,qian);
  281.                 Display_One_Char(5,0,bai);
  282.                 Display_One_Char(6,0,shi);
  283.                 Display_One_Char(7,0,'.');
  284.                 Display_One_Char(8,0,ge);
  285.                 Delay_ms(100);   //延时                     
  286. }
  287. void time0_init()                        //定时器初始化
  288. {
  289.    TMOD=0x01;                                //T0工作方式1
  290.    EA=1;
  291.    ET0=1;
  292.    TR0=1;
  293.    TH0=(65536-917)/256;      //1ms 进入一次中断
  294.    TL0=(65536-917)%256;
  295. }
  296. void time0_stop()                         //关闭中断
  297. {  
  298.   EA=0;
  299.   ET0=0;
  300.   TR0=0;                                                        
  301.   
  302. }
  303. void chang_pwm1(u16 znum3,u16 ynum3)           //zum3,yum3为车速
  304. {      
  305.     if(znum1>znum3)
  306.              {
  307.            zea=0;
  308.            zeb=0;
  309.           }
  310.         else
  311.           {
  312.            zea=1;
  313.            zeb=1;
  314.           }
  315.         if(ynum1>ynum3)
  316.              {
  317.            yea=0;
  318.            yeb=0;
  319.           }
  320.         else
  321.           {
  322.            yea=1;
  323.            yeb=1;
  324.           }
  325.         if(znum1>50)
  326.         {
  327.            znum1=0;
  328.         }                                   
  329.         if(ynum1>50)
  330.         {
  331.            ynum1=0;
  332.         }
  333. }                  
  334. void turn_right()
  335. {      
  336.    chang_pwm1(20,10);                   //右拐               
  337. }
  338. void turn_wei()                                   //变向
  339. {     
  340.      zIN1=0;
  341.      zIN2=1;
  342.          yIN3=1;                                                
  343.      yIN4=0;
  344.          zIN3=0;                                                
  345.      zIN4=0;
  346.          yIN1=0;
  347.      yIN2=0;
  348.          chang_pwm1(10,10);                              
  349. }
  350. void go_straight()
  351. {   
  352.    chang_pwm1(10,10);                   //直行
  353. }  
  354. void init()                        //初始化
  355. {
  356.          yea=1;
  357.      yIN1=1;
  358.      yIN2=0;  
  359.      yIN3=1;                                                
  360.      yIN4=0;
  361.       yeb=1;
  362.           zea=1;         
  363.      zIN1=1;
  364.      zIN2=0;
  365.      zIN3=1;                                                
  366.      zIN4=0;
  367.       zeb=1;
  368. }        
  369. void main()
  370. {
  371.     while(1)
  372.         {
  373.            fanwei();
  374.           if((qian!='2')||((bai!='4')&&(bai!='5')))
  375.           {         
  376.                     init();
  377.                  time0_init();
  378.                  turn_wei();
  379.           }
  380.           else
  381.           {         
  382.              init();
  383.              time0_init();
  384.                  go_straight();            
  385.           }
  386.           time0_stop();        
  387.          }
  388. }
  389. void  T0_time()interrupt 1         //中断程序1
  390. {
  391.         TH0=(65536-917)/256;      //1ms 进入一次中断
  392.     TL0=(65536-917)%256;
  393.     znum1++;
  394.         ynum1++;            
  395. }
复制代码
回复

使用道具 举报

ID:151348 发表于 2017-4-2 17:06 | 显示全部楼层
你试试把PWM调速部分单独取出来,另外写一个程序,看看能不能实现调速

评分

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

查看全部评分

回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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