基于 STM32 的硬件 I2C 读取 MPU6050 数据

发布者:WhisperingWaves最新更新时间:2018-06-02 来源: eefocus关键字:STM32  硬件  I2C  MPU6050 手机看文章 扫描二维码
随时随地手机看文章

MPU6050其实就是一个 I2C 器件,里面有很多寄存器(但是我们用到的只有几个),我们通过读写寄存器来操作这个芯片。所以首要问题就是 STM32 和 MPU6050 的 I2C 通信。

1、配置 STM32 (用I2C1:PB6——SCL;PB7——SDA)

      1)时钟 RCC 

            RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);

            RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1 , ENABLE);

       2)GPIO 配置

            GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;

            GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

            GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD;       //两个引脚都加 4.7K 上拉电阻

            GPIO_Init(GPIOB, &GPIO_InitStructure); 

       3)I2C 配置


           void I2C_Configuration(void)

          {

                 I2C_InitTypeDef  I2C_InitStructure;


                 I2C_InitStructure.I2C_Mode = I2C_Mode_I2C;

                 I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2;

                 I2C_InitStructure.I2C_OwnAddress1 =0xc0; //  STM32 的自身地址,不与从器件相同即可

                 I2C_InitStructure.I2C_Ack = I2C_Ack_Enable;

                 I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;

                 I2C_InitStructure.I2C_ClockSpeed = 100000;  


                 I2C_Init(I2C1, &I2C_InitStructure);

                 I2C_Cmd(I2C1, ENABLE);

           }


至此,STM32 已配置完毕,其实不是那么难。


2、初始化 MPU6050

          void MPU6050_Initialize()             //初始化过程 ,其实就是写 5个寄存器

         {        

                 MPU6050_I2C_ByteWrite(0xd0,0x00,MPU6050_RA_PWR_MGMT_1);      // reg107, 唤醒,8M内部时钟源

                 MPU6050_I2C_ByteWrite(0xd0,0x07,MPU6050_RA_SMPLRT_DIV);         //采用频率 1000

                 MPU6050_I2C_ByteWrite(0xd0,0x06,MPU6050_RA_CONFIG);                 

                 MPU6050_I2C_ByteWrite(0xd0,0x01,MPU6050_RA_ACCEL_CONFIG);     //加速度量程 2g

                 MPU6050_I2C_ByteWrite(0xd0,0x18,MPU6050_RA_GYRO_CONFIG);          //角速度量程 2000度/s

         }


注:0xD0 表示 MPU6050 的地址。我们知道 I2C从器件(在此当然是指 MPU6050)有 8 位的地址,前 7 位由 WHO AM I 确定,第 8 位由 AD0 的电平决定。WHO AM I 默认值是 0x68H(1101000B),AD0 接低电平,所以 MPU6050 的 I2C 地址是 0xD0H(11010000B)。



3、I2C 核心程序 ( 读/写)


1)写入寄存器

一次写操作分为几个步骤: 发送开始信号 -> 起始成功?(可能描述的不太准确) -> 发送 MPU6050 地址、状态(写)-> 写地址成功? -> 发送 MPU6050内部某个待写寄存器地址 -> 发送成功? -> 发送要写入的内容 -> 发送成功? -> 发送结束信号

总结:先写 MPU6050 地址,再写 寄存器地址,最后写 内容,且每次都要验证(应该和应答信号有关)。这就像寄快递一样,先写市县地址,再写街道地址,最后写门牌号。


再看下面的程序就会更容易理解一些。


void MPU6050_I2C_ByteWrite(u8 slaveAddr, u8 pBuffer, u8 writeAddr)

{

  /* Send START condition */

  I2C_GenerateSTART(I2C1, ENABLE);          //发送开始信号

  /* Test on EV5 and clear it */

  while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT));


  /* Send MPU6050 address for write */

  I2C_Send7bitAddress(I2C1, slaveAddr, I2C_Direction_Transmitter);          // 发送 MPU6050 地址、状态(写)

  /* Test on EV6 and clear it */

  while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED));


  /* Send the MPU6050's internal address to write to */

  I2C_SendData(I2C1, writeAddr);                   //发送 MPU6050内部某个待写寄存器地址

  /* Test on EV8 and clear it */

  while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));


  /* Send the byte to be written */

  I2C_SendData(I2C1, pBuffer);                     //发送要写入的内容

  /* Test on EV8 and clear it */

  while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));


  /* Send STOP condition */

  I2C_GenerateSTOP(I2C1, ENABLE);          //发送结束信号

}




2)读寄存器

因为 加速度值和角速度值都在寄存器里,所以必须读它才能获得数据。

附上程序



如:MPU6050_I2C_BufferRead(0xd0, receive_data, MPU6050_RA_WHO_AM_I, 1);   //读 WHO_AM_I 寄存器的值(0x68)

如:

       void MPU6050_GetRawAccelGyro(s16* AccelGyro)        //读加速度值 和 角速度值

      {

               u8 tmpBuffer[14],i; 

               MPU6050_I2C_BufferRead(0xd0, tmpBuffer, MPU6050_RA_ACCEL_XOUT_H, 14); 

               /* Get acceleration */

               for(i=0; i<3; i++)                              

              AccelGyro=((s16)((u16)tmpBuffer[2*i] << 8) + tmpBuffer[2*i+1]);

     /* Get Angular rate */

              for(i=4; i<7; i++)                                             //在此跳过温度寄存器,不需要温度值

              AccelGyro[i-1]=((s16)((u16)tmpBuffer[2*i] << 8) + tmpBuffer[2*i+1]);      

       }


注:

#define MPU6050_RA_ACCEL_XOUT_H     0x3B

#define MPU6050_RA_ACCEL_XOUT_L     0x3C

#define MPU6050_RA_ACCEL_YOUT_H     0x3D

#define MPU6050_RA_ACCEL_YOUT_L     0x3E

#define MPU6050_RA_ACCEL_ZOUT_H     0x3F

#define MPU6050_RA_ACCEL_ZOUT_L     0x40

#define MPU6050_RA_TEMP_OUT_H        0x41

#define MPU6050_RA_TEMP_OUT_L        0x42

#define MPU6050_RA_GYRO_XOUT_H      0x43

#define MPU6050_RA_GYRO_XOUT_L      0x44

#define MPU6050_RA_GYRO_YOUT_H      0x45

#define MPU6050_RA_GYRO_YOUT_L      0x46

#define MPU6050_RA_GYRO_ZOUT_H      0x47

#define MPU6050_RA_GYRO_ZOUT_L      0x48



I2C 读核心程序:

void MPU6050_I2C_BufferRead(u8 slaveAddr, u8* pBuffer, u8 readAddr, u16 NumByteToRead)

{

  /* While the bus is busy */  

  while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BUSY));


  /* Send START condition */

  I2C_GenerateSTART(I2C1, ENABLE);

  /* Test on EV5 and clear it */

  while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT));


  /* Send MPU6050 address for write */

  I2C_Send7bitAddress(I2C1, slaveAddr, I2C_Direction_Transmitter); 

  /* Test on EV6 and clear it */

  while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED));


  /* Clear EV6 by setting again the PE bit */

  I2C_Cmd(I2C1, ENABLE);

  /* Send the MPU6050's internal address to write to */

  I2C_SendData(I2C1, readAddr);

  /* Test on EV8 and clear it */

  while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));


/* Send STRAT condition a second time */

  I2C_GenerateSTART(I2C1, ENABLE);

  /* Test on EV5 and clear it */

  while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT));


  /* Send MPU6050 address for read */

  I2C_Send7bitAddress(I2C1, slaveAddr, I2C_Direction_Receiver);

  /* Test on EV6 and clear it */

  while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED));


  /* While there is data to be read */

  while(NumByteToRead)

  {

    if(NumByteToRead == 1)

    {

      /* Disable Acknowledgement */

      I2C_AcknowledgeConfig(I2C1, DISABLE);


      /* Send STOP Condition */

      I2C_GenerateSTOP(I2C1, ENABLE);

    }


    /* Test on EV7 and clear it */

    if(I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_RECEIVED))

    {

      /* Read a byte from the MPU6050 */

      *pBuffer = I2C_ReceiveData(I2C1);


      /* Point to the next location where the byte read will be saved */

      pBuffer++;


      /* Decrement the read bytes counter */

      NumByteToRead--;

    }

  }


  /* Enable Acknowledgement to be ready for another reception */

  I2C_AcknowledgeConfig(I2C1, ENABLE);

}




最后,把六个值 printf 出来就行。

关键字:STM32  硬件  I2C  MPU6050 引用地址:基于 STM32 的硬件 I2C 读取 MPU6050 数据

上一篇:关于STM32的硬件IIC使用问题解决方案
下一篇:浅谈 STM32 硬件I2C的使用 (中断方式 无DMA 无最高优先级)

推荐阅读最新更新时间:2024-03-16 16:04

基于FFmpeg的H.264视频硬件编解码在S3C6410处理器上的实现
  目前,智能手机、PDA和平板电脑等越来越多的嵌入式设备支持高清视频采集和播放功能,高清视频的采集或播放功能正广泛用于游戏设备、监控设备、视频会议设备和数字网络电视等嵌入式系统中。这些功能的实现建立在高性能视频硬件编解码技术基础之上。本文阐述了基于FFMpeg的H.264视频硬件编解码在S3C6410处理器上的实现方法,为数字娱乐、视频监控和视频通信系统开发过程中的高清视频硬件编解码的实现提供参考。   FFmpeg 是一个开源免费跨平台的视频和音频流方案,属于自由软件。它包含非常先进的音频/视频编解码库libavcodec,提供了录制、转换以及流化音视频的完整解决方案。FFmpeg支持MPEG4、FLV等40多种编码,以及
[单片机]
基于FFmpeg的H.264视频<font color='red'>硬件</font>编解码在S3C6410处理器上的实现
STM32定时器中断应用实例
本文将介绍:通过STM32cube mx配置工程文件,实现定时器中断的功能; 实例中使用的是STM32F407芯片 ①STM32cube mx配置工程文件 ②代码实现 ①STM32cube mx配置工程文件 Step1:使能RCC、选择外部晶振 Step2:配置时钟树 Step3:使能TIM6、配置TIM6 ,1秒产生一次定时器中断 Step4:配置中断的优先级分组、中断优先级 Step4:使能串口、用于调试使用 Step5:生成工程文件 ②代码实现 Step1:编写串口输出重定向函数,用于调试使用;记得添加头文件stdio.h Step2:开启定时器中断 Step3:自定义定时器中断服务函数 先在
[单片机]
<font color='red'>STM32</font>定时器中断应用实例
基于STM32的FSMC接口驱动TFT彩屏的设计方案
  TFT-LCD技术是微电子技术和LCD技术巧妙结合的高新技术。随着人们对图像清晰度、刷新率、保真度的要求越来越高,TFT-LCD的应用范围越来越广。本文提出了一种能直接驱动数字液晶屏的设计方案,方案先介绍了TFT数字彩屏的工作原理,利用STM32处理器的FSMC接口设计的硬件电路和软件程序均能对显示控制芯片进行有效的控制。在实际应用中显示清晰流畅,并且CPU有足够的时间来处理用户程序。该方案能成功应用在电脑横机的人机界面显示中,且其硬件电路结构简单、控制方式灵活、对于其他型号的接口芯片也能提供参考。   0 引言   随着电子产品的不断更新,各种显示界面的开发越来越多,由于TFT彩屏的性价比高,因而被广泛用在各种电子设备上
[单片机]
基于<font color='red'>STM32</font>的FSMC接口驱动TFT彩屏的设计方案
STM32中断优先级库的使用方式
为什么中断要有优先级 何为占先式优先级(pre-emption priority) 高占先式优先级的中断事件会打断当前的主程序/中断程序运行 抢断式优先响应,俗称中断嵌套。 何为副优先级(subpriority) 在占先式优先级相同的情况下,高副优先级的中断优先被响应; 在占先式优先级相同的情况下,如果有低副优先级中断正在执行, 高副优先级的中断要等待已被响应的低副优先级中断执行结束后才 能得到响应 非抢断式响应(不能嵌套)。 判断中断是否会被响应的依据 首先是占先式优先级,其次是副优先级; 占先式优先级决定是否会有中断嵌套; Reset、NMI、Hard Fault 优先级为负(高于普通中断优先级)且不可调整。
[单片机]
stm32 灵活静态存储控制器(FSMC)(NORFLASH\PSRAM)
Flexible static memory controller(FSMC) 今天在处理TFT彩屏的时候突然发现有人用FSMC控制器来处理,然后就认真的研究了下FSMC; 可见他分为4个块,三个类型,我们可以根据自己的需要来选择;这次我就直说FSMC 的Block 1; 首先,基地址BASE_ADDR = 6000 0000;至于片选,datasheet上也说了,我们可以通过控制HADDR(27,26)来选择操作; 然后还有今天一直困扰我的问题,我要选择A16,我用的是16位数据,他的数据地址为6000 0000 + 2^16*2 = 6002 0000;我一直都在疑惑:明明是A16,为什么是第17位被置1,后来
[单片机]
<font color='red'>stm32</font> 灵活静态存储控制器(FSMC)(NORFLASH\PSRAM)
【ARM9嵌入式系统硬件设计指南】产品电源设计过程
6.5产品电源设计过程 6.5.1系统级电源框图设计 系统级的电源设计,主要从整个系统的功能、产品的应用场合、开发周期、性价比等整体出发,综合评估系统前级、板内功能模块、对外供电接口、通讯接口、数据采集接口等对电源的需求及相应需达到的防护等级。在实际的工程应用中,市场机会稍纵即逝,往往留给产品的开发周期都很短,从产品的可靠性出发,系统级的电源设计,通常都尽量采用成熟的模块化电源来设计,降低风险,使产品尽早面世,占领市场。 以如图6.10所示的工控系统应用为例进行说明,系统较为复杂,包括了多种信号的数据采集、CAN、RS-485等现场总线、板内要求双电源供电的运放、电机驱动等多种功能。每个功能模块,对电源的要求都有
[单片机]
【ARM9嵌入式系统<font color='red'>硬件</font>设计指南】产品电源设计过程
STM32 SPI硬件模式
反复试验,发现SPI_NSS引脚的自动硬件控制与想象的不同,无论是否外加上拉,只要一使能SPI,SPI_Cmd(SPI1, ENABLE); SPI_NSS引脚就一直处于低电平,直到SPI_Cmd(SPI1, DISABLE);这个需要用程序来控制。 而用过其他芯片则是发送完成自动会拉高,这点是要注意的 我说的就是做主机的时候 SPI_SSOutputCmd(SPIx,ENABLE) 在soft模式时这句话有必要吗?我的理解是当hard模式,需要multimaster的时候,才应该要开启这个output功能,这点从我的截图上可以看出。 我觉得,这里只要把SPI_InitStructure.SPI_NSS = SPI_NSS_So
[单片机]
<font color='red'>STM32</font> SPI<font color='red'>硬件</font>模式
PLC系统改造节约硬件投资
摘要:为了不增加改造成本,对不同部分设备的PLC控制软件进行了深入的研究,发现CC09-RB056的MOBY-i程序把读写的结果存放在一个数据块DB580内部,而程序的其他部分都从数据块获取车型信息,于是我们把研究方向放在该数据块上。在CC09-RB056工位,若当前实际的车型为B12,则对MOBY-i的读写结果进行修改,最终解决了这个问题。   我公司计划推出一款B级新车B12,由于工艺线路的特殊性,此车型需要在四厂涂装车间喷涂,然后经过二厂涂装车间的喷蜡线进入二厂总装车间装配。具体工艺布局如图1所示:B12经过RB003~RB001(图中红色部分)至CC09-TC/RB048(图中黑色部分),然后经过CC09-RB050、
[工业控制]
PLC系统改造节约<font color='red'>硬件</font>投资
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
  • ARM裸机篇--按键中断
    先看看GPOI的输入实验:按键电路图:GPF1管教的功能:EINT1要使用GPF1作为EINT1的功能时,只要将GPFCON的3:2位配置成10就可以了!GPF1先配 ...
  • 网上下的--ARM入门笔记
    简单的介绍打今天起菜鸟的ARM笔记算是开张了,也算给我的这些笔记找个存的地方。为什么要发布出来?也许是大家感兴趣的,其实这些笔记之所 ...
  • 学习ARM开发(23)
    三个任务准备与运行结果下来看看创建任务和任运的栈空间怎么样的,以及运行输出。Made in china by UCSDN(caijunsheng)Lichee 1 0 0 ...
  • 学习ARM开发(22)
    关闭中断与打开中断中断是一种高效的对话机制,但有时并不想程序运行的过程中中断运行,比如正在打印东西,但程序突然中断了,又让另外一个 ...
  • 学习ARM开发(21)
    先要声明任务指针,因为后面需要使用。 任务指针 volatile TASK_TCB* volatile g_pCurrentTask = NULL;volatile TASK_TCB* vol ...
  • 学习ARM开发(20)
  • 学习ARM开发(19)
  • 学习ARM开发(14)
  • 学习ARM开发(15)
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved