久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 5371|回復: 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. }
復制代碼
分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

沙發
ID:151348 發表于 2017-4-2 17:06 | 只看該作者
你試試把PWM調速部分單獨取出來,另外寫一個程序,看看能不能實現調速

評分

參與人數 1黑幣 +20 收起 理由
admin + 20 回帖助人的獎勵!

查看全部評分

回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 色播av| 亚洲精品自在在线观看 | 中文成人在线 | 超碰在线亚洲 | 99精品久久| 久久久久国产一区二区三区四区 | 久久一区二区三区四区五区 | 亚洲高清在线视频 | 国产一区二区自拍 | 亚洲午夜视频 | 成人美女免费网站视频 | 国产午夜精品一区二区三区四区 | 久久久久久高潮国产精品视 | 欧美中文字幕 | 性一交一乱一伦视频免费观看 | 日日操夜夜操视频 | 中国黄色在线视频 | 一久久久 | 欧美极品少妇xxxxⅹ免费视频 | 国产精品爱久久久久久久 | 国产日韩亚洲欧美 | 日韩精品一区二区三区久久 | 91欧美激情一区二区三区成人 | 少妇一级淫片aaaaaaaaa | 久久99精品国产自在现线小黄鸭 | 午夜视频在线观看视频 | 亚洲视频手机在线 | 国产精品亚洲综合 | 二区精品| 国内精品久久久久久 | 精品视频在线观看 | 成人一区二区三区 | 国产精品久久av | 欧美一区二区三区视频在线 | 视频一区在线观看 | 亚洲精品日韩在线 | 欧美一区二区三区在线观看 | 日本不卡一区二区三区 | 亚洲国产精久久久久久久 | 9久久婷婷国产综合精品性色 | 欧美啊v在线观看 |