找回密码
 立即注册

QQ登录

只需一步,快速开始

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

单片机mpu6050三轴加速器源程序

[复制链接]
跳转到指定楼层
楼主
程序绝对可用的


单片机源程序如下:
  1. //****************************************
  2. // 功能: 显示加速度计和陀螺仪的10位原始数据
  3. //****************************************
  4. // GY-52 MPU3050 IIC测试程序

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

所有资料51hei提供下载:
mpu6050三轴加速器程序.zip (37.62 KB, 下载次数: 22)



评分

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

查看全部评分

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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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