找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2539|回复: 0
收起左侧

51单片机+MPU6050陀螺仪程序

[复制链接]
ID:811102 发表于 2020-10-8 15:10 | 显示全部楼层 |阅读模式
陀螺仪角度  可用x,y轴测量角度
单片机源程序如下:
  1. //****************************************
  2. // Update to MPU6050 by shinetop
  3. // MCU: STC89C52
  4. // 2012.3.1
  5. // 功能: 显示加速度计和陀螺仪的10位原始数据
  6. //****************************************
  7. // GY-52 MPU3050 IIC测试程序
  8. // 使用单片机STC89C51
  9. // 晶振:11.0592M
  10. // 显示:LCD1602
  11. // 编译环境 Keil uVision2
  12. // 参考宏晶网站24c04通信程序
  13. //****************************************
  14. #include <REG52.H>        
  15. #include <math.h>    //Keil library  
  16. #include <stdio.h>   //Keil library        
  17. #include <INTRINS.H>
  18. typedef unsigned char  uchar;
  19. typedef unsigned short ushort;
  20. typedef unsigned int   uint;
  21. //****************************************
  22. // 定义51单片机端口
  23. //****************************************
  24. #define DataPort P0                //LCD1602数据端口
  25. sbit    SCL=P1^0;                        //IIC时钟引脚定义
  26. sbit    SDA=P1^1;                        //IIC数据引脚定义
  27. sbit    LCM_RS=P2^0;                //LCD1602命令端口               
  28. sbit    LCM_RW=P2^1;                //LCD1602命令端口               
  29. sbit    LCM_EN=P2^2;                //LCD1602命令端口
  30. //****************************************
  31. // 定义MPU6050内部地址
  32. //****************************************
  33. #define        SMPLRT_DIV                0x19        //陀螺仪采样率,典型值:0x07(125Hz)
  34. #define        CONFIG                        0x1A        //低通滤波频率,典型值:0x06(5Hz)
  35. #define        GYRO_CONFIG                0x1B        //陀螺仪自检及测量范围,典型值:0x18(不自检,2000deg/s)
  36. #define        ACCEL_CONFIG        0x1C        //加速计自检、测量范围及高通滤波频率,典型值:0x01(不自检,2G,5Hz)
  37. #define        ACCEL_XOUT_H        0x3B
  38. #define        ACCEL_XOUT_L        0x3C
  39. #define        ACCEL_YOUT_H        0x3D
  40. #define        ACCEL_YOUT_L        0x3E
  41. #define        ACCEL_ZOUT_H        0x3F
  42. #define        ACCEL_ZOUT_L        0x40
  43. #define        TEMP_OUT_H                0x41
  44. #define        TEMP_OUT_L                0x42
  45. #define        GYRO_XOUT_H                0x43
  46. #define        GYRO_XOUT_L                0x44        
  47. #define        GYRO_YOUT_H                0x45
  48. #define        GYRO_YOUT_L                0x46
  49. #define        GYRO_ZOUT_H                0x47
  50. #define        GYRO_ZOUT_L                0x48
  51. #define        PWR_MGMT_1                0x6B        //电源管理,典型值:0x00(正常启用)
  52. #define        WHO_AM_I                        0x75        //IIC地址寄存器(默认数值0x68,只读)
  53. #define        SlaveAddress        0xD0        //IIC写入时的地址字节数据,+1为读取
  54. //****************************************
  55. //定义类型及变量
  56. //****************************************
  57. uchar dis[4];                                                        //显示数字(-511至512)的字符数组
  58. int        dis_data;                                                //变量
  59. //int        Temperature,Temp_h,Temp_l;        //温度及高低位数据
  60. //****************************************
  61. //函数声明
  62. //****************************************
  63. void  delay(unsigned int k);                                                                                //延时
  64. //LCD相关函数
  65. void  InitLcd();                                                                                                                //初始化lcd1602
  66. void  lcd_printf(uchar *s,int temp_data);
  67. void  WriteDataLCM(uchar dataW);                                                                        //LCD数据
  68. void  WriteCommandLCM(uchar CMD,uchar Attribc);                                //LCD指令
  69. void  DisplayOneChar(uchar X,uchar Y,uchar DData);                        //显示一个字符
  70. void  DisplayListChar(uchar X,uchar Y,uchar *DData,L);        //显示字符串
  71. //MPU6050操作函数
  72. void  InitMPU6050();                                                                                                        //初始化MPU6050
  73. void  Delay5us();
  74. void  I2C_Start();
  75. void  I2C_Stop();
  76. void  I2C_SendACK(bit ack);
  77. bit   I2C_RecvACK();
  78. void  I2C_SendByte(uchar dat);
  79. uchar I2C_RecvByte();
  80. void  I2C_ReadPage();
  81. void  I2C_WritePage();
  82. void  display_ACCEL_x();
  83. void  display_ACCEL_y();
  84. void  display_ACCEL_z();
  85. uchar Single_ReadI2C(uchar REG_Address);                                                //读取I2C数据
  86. void  Single_WriteI2C(uchar REG_Address,uchar REG_data);        //向I2C写入数据
  87. //****************************************
  88. //整数转字符串
  89. //****************************************
  90. void lcd_printf(uchar *s,int temp_data)
  91. {
  92.         if(temp_data<0)
  93.         {
  94.                 temp_data=-temp_data;
  95.                 *s='-';
  96.         }
  97.         else *s=' ';
  98.         *++s =temp_data/100+0x30;
  99.         temp_data=temp_data%100;     //取余运算
  100.         *++s =temp_data/10+0x30;
  101.         temp_data=temp_data%10;      //取余运算
  102.         *++s =temp_data+0x30;         
  103. }
  104. //****************************************
  105. //延时
  106. //****************************************
  107. void delay(unsigned int k)        
  108. {                                                
  109.         unsigned int i,j;                                
  110.         for(i=0;i<k;i++)
  111.         {                        
  112.                 for(j=0;j<121;j++);
  113.         }                                                
  114. }
  115. //****************************************
  116. //LCD1602初始化
  117. //****************************************
  118. void InitLcd()                                
  119. {                        
  120.         WriteCommandLCM(0x38,1);        
  121.         WriteCommandLCM(0x08,1);        
  122.         WriteCommandLCM(0x01,1);        
  123.         WriteCommandLCM(0x06,1);        
  124.         WriteCommandLCM(0x0c,1);
  125.         DisplayOneChar(0,0,'A');
  126.         DisplayOneChar(0,1,'G');
  127. }                        
  128. //****************************************
  129. //LCD1602写允许
  130. //****************************************
  131. void WaitForEnable(void)        
  132. {                                       
  133.         DataPort=0xff;               
  134.         LCM_RS=0;LCM_RW=1;_nop_();
  135.         LCM_EN=1;_nop_();_nop_();
  136.         while(DataPort&0x80);        
  137.         LCM_EN=0;                                
  138. }                                       
  139. //****************************************
  140. //LCD1602写入命令
  141. //****************************************
  142. void WriteCommandLCM(uchar CMD,uchar Attribc)
  143. {                                       
  144.         if(Attribc)WaitForEnable();        
  145.         LCM_RS=0;LCM_RW=0;_nop_();
  146.         DataPort=CMD;_nop_();        
  147.         LCM_EN=1;_nop_();_nop_();LCM_EN=0;
  148. }                                       
  149. //****************************************
  150. //LCD1602写入数据
  151. //****************************************
  152. void WriteDataLCM(uchar dataW)
  153. {                                       
  154.         WaitForEnable();               
  155.         LCM_RS=1;LCM_RW=0;_nop_();
  156.         DataPort=dataW;_nop_();        
  157.         LCM_EN=1;_nop_();_nop_();LCM_EN=0;
  158. }               
  159. //****************************************
  160. //LCD1602写入一个字符
  161. //****************************************
  162. void DisplayOneChar(uchar X,uchar Y,uchar DData)
  163. {                                                
  164.         Y&=1;                                                
  165.         X&=15;                                                
  166.         if(Y)X|=0x40;                                       
  167.         X|=0x80;                        
  168.         WriteCommandLCM(X,0);               
  169.         WriteDataLCM(DData);               
  170. }                                                
  171. //****************************************
  172. //LCD1602显示字符串
  173. //****************************************
  174. void DisplayListChar(uchar X,uchar Y,uchar *DData,L)
  175. {
  176.         uchar ListLength=0;
  177.         Y&=0x1;               
  178.         X&=0xF;               
  179.         while(L--)            
  180.         {                       
  181.                 DisplayOneChar(X,Y,DData[ListLength]);
  182.                 ListLength++;  
  183.                 X++;                        
  184.         }   
  185. }
  186. //**************************************
  187. //延时5微秒(STC90C52RC@12M)
  188. //不同的工作环境,需要调整此函数
  189. //当改用1T的MCU时,请调整此延时函数
  190. //**************************************
  191. void Delay5us()
  192. {
  193.         _nop_();_nop_();_nop_();_nop_();
  194.         _nop_();_nop_();_nop_();_nop_();
  195.         _nop_();_nop_();_nop_();_nop_();
  196.         _nop_();_nop_();_nop_();_nop_();
  197.         _nop_();_nop_();_nop_();_nop_();
  198.         _nop_();_nop_();_nop_();_nop_();
  199. }
  200. //**************************************
  201. //I2C起始信号
  202. //**************************************
  203. void I2C_Start()
  204. {
  205.     SDA = 1;                    //拉高数据线
  206.     SCL = 1;                    //拉高时钟线
  207.     Delay5us();                 //延时
  208.     SDA = 0;                    //产生下降沿
  209.     Delay5us();                 //延时
  210.     SCL = 0;                    //拉低时钟线
  211. }
  212. //**************************************
  213. //I2C停止信号
  214. //**************************************
  215. void I2C_Stop()
  216. {
  217.     SDA = 0;                    //拉低数据线
  218.     SCL = 1;                    //拉高时钟线
  219.     Delay5us();                 //延时
  220.     SDA = 1;                    //产生上升沿
  221.     Delay5us();                 //延时
  222. }
  223. //**************************************
  224. //I2C发送应答信号
  225. //入口参数:ack (0:ACK 1:NAK)
  226. //**************************************
  227. void I2C_SendACK(bit ack)
  228. {
  229.     SDA = ack;                  //写应答信号
  230.     SCL = 1;                    //拉高时钟线
  231.     Delay5us();                 //延时
  232.     SCL = 0;                    //拉低时钟线
  233.     Delay5us();                 //延时
  234. }
  235. //**************************************
  236. //I2C接收应答信号
  237. //**************************************
  238. bit I2C_RecvACK()
  239. {
  240.     SCL = 1;                    //拉高时钟线
  241.     Delay5us();                 //延时
  242.     CY = SDA;                   //读应答信号
  243.     SCL = 0;                    //拉低时钟线
  244.     Delay5us();                 //延时
  245.     return CY;
  246. }
  247. //**************************************
  248. //向I2C总线发送一个字节数据
  249. //**************************************
  250. void I2C_SendByte(uchar dat)
  251. {
  252.     uchar i;
  253.     for (i=0; i<8; i++)         //8位计数器
  254.     {
  255.         dat <<= 1;              //移出数据的最高位
  256.         SDA = CY;               //送数据口
  257.         SCL = 1;                //拉高时钟线
  258.         Delay5us();             //延时
  259.         SCL = 0;                //拉低时钟线
  260.         Delay5us();             //延时
  261.     }
  262.     I2C_RecvACK();
  263. }
  264. //**************************************
  265. //从I2C总线接收一个字节数据
  266. //**************************************
  267. uchar I2C_RecvByte()
  268. {
  269.     uchar i;
  270.     uchar dat = 0;
  271.     SDA = 1;                    //使能内部上拉,准备读取数据,
  272.     for (i=0; i<8; i++)         //8位计数器
  273.     {
  274.         dat <<= 1;
  275.         SCL = 1;                //拉高时钟线
  276.         Delay5us();             //延时
  277.         dat |= SDA;             //读数据               
  278.         SCL = 0;                //拉低时钟线
  279.         Delay5us();             //延时
  280.     }
  281.     return dat;
  282. }
  283. //**************************************
  284. //向I2C设备写入一个字节数据
  285. //**************************************
  286. void Single_WriteI2C(uchar REG_Address,uchar REG_data)
  287. {
  288.     I2C_Start();                  //起始信号
  289.     I2C_SendByte(SlaveAddress);   //发送设备地址+写信号
  290.     I2C_SendByte(REG_Address);    //内部寄存器地址,
  291.     I2C_SendByte(REG_data);       //内部寄存器数据,
  292.     I2C_Stop();                   //发送停止信号
  293. }
  294. //**************************************
  295. //从I2C设备读取一个字节数据
  296. //**************************************
  297. uchar Single_ReadI2C(uchar REG_Address)
  298. {
  299.         uchar REG_data;
  300.         I2C_Start();                   //起始信号
  301.         I2C_SendByte(SlaveAddress);    //发送设备地址+写信号
  302.         I2C_SendByte(REG_Address);     //发送存储单元地址,从0开始        
  303.         I2C_Start();                   //起始信号
  304.         I2C_SendByte(SlaveAddress+1);  //发送设备地址+读信号
  305.         REG_data=I2C_RecvByte();       //读出寄存器数据
  306.         I2C_SendACK(1);                //接收应答信号
  307.         I2C_Stop();                    //停止信号
  308.         return REG_data;
  309. }
  310. //**************************************
  311. //初始化MPU6050
  312. //**************************************
  313. void InitMPU6050()
  314. {
  315.         Single_WriteI2C(PWR_MGMT_1, 0x00);        //解除休眠状态
  316.         Single_WriteI2C(SMPLRT_DIV, 0x07);
  317.         Single_WriteI2C(CONFIG, 0x06);
  318.         Single_WriteI2C(GYRO_CONFIG, 0x18);
  319.         Single_WriteI2C(ACCEL_CONFIG, 0x01);
  320. }
  321. //**************************************
  322. //合成数据
  323. //**************************************
  324. int GetData(uchar REG_Address)
  325. {
  326.         char H,L;
  327.         H=Single_ReadI2C(REG_Address);
  328.         L=Single_ReadI2C(REG_Address+1);
  329.         return (H<<8)+L;   //合成数据
  330. }
  331. //**************************************
  332. //在1602上显示10位数据
  333. //**************************************
  334. void Display10BitData(int value,uchar x,uchar y)
  335. {
  336.         value/=64;                                                        //转换为10位数据
  337.         lcd_printf(dis, value);                        //转换数据显示
  338.         DisplayListChar(x,y,dis,4);        //启始列,行,显示数组,显示长度
  339. }
  340. //**************************************
  341. //显示温度
  342. //**************************************
  343. //void display_temp()
  344. //{
  345. //        Temp_h=Single_ReadI2C(TEMP_OUT_H); //读取温度
  346. //        Temp_l=Single_ReadI2C(TEMP_OUT_L); //读取温度
  347. //        Temperature=Temp_h<<8|Temp_l;     //合成温度
  348. //        Temperature = 35+ ((double) (Temperature + 13200)) / 280; // 计算出温度
  349. //        lcd_printf(dis,Temperature);     //转换数据显示
  350. //        DisplayListChar(11,1,dis,4);     //启始列,行,显示数组,显示位数
  351. //}
  352. //*********************************************************
  353. //主程序
  354. //*********************************************************
  355. void main()
  356. {
  357.         delay(500);                //上电延时               
  358.         InitLcd();                //液晶初始化
  359.         InitMPU6050();        //初始化MPU6050
  360.         delay(150);
  361.         while(1)
  362.         {
  363.                 Display10BitData(GetData(ACCEL_XOUT_H),2,0);        //显示X轴加速度
  364.                 Display10BitData(GetData(ACCEL_YOUT_H),7,0);        //显示Y轴加速度
  365.                 Display10BitData(GetData(ACCEL_ZOUT_H),12,0);        //显示Z轴加速度
  366.                 Display10BitData(GetData(GYRO_XOUT_H),2,1);        //显示X轴角速度
  367.                 Display10BitData(GetData(GYRO_YOUT_H),7,1);        //显示Y轴角速度
  368.                 Display10BitData(GetData(GYRO_ZOUT_H),12,1);        //显示Z轴角速度
  369.                 delay(500);
  370.         }
  371. }
复制代码

全部资料51hei下载地址:
陀螺仪1602显示.rar (37.86 KB, 下载次数: 82)
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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