本篇内容为,通过 I2C 配置 GY-91 MPU9250+BMP280 里面的 MPU9250 连接 AK8963 磁感应。两个办法,1)MPU9250 设置 Master Mode 通过 AUX I2C 读写,2)Pass Through Mode 由 Host (单片机或者本文的 BBB)直接用 I2C 读写。本文两种方式都会操作一次。
最初看到淘宝上面有卖 GY91 MPU9250+BMP280 时候,它写着九轴,是九个轴。地球上原来除了 X Y Z 之外,还有四五维空间里面的 6 个轴,原来所谓的九轴,是陀螺仪三轴,加速度三轴,磁感应三轴(指南针),才叫做九轴。MPU9250 是 InvenSense 出品,它本身就是一个模块集成了三个东西,还有个自身的温度计(芯片温度),FIFO,Digital Motion Processor(DMP,就是姿势识别,计步器那些),Interrupt(可惜 INT 脚在 GY91 没有了)。而我买了的 GY91 MPU9250+BMP280,又再集成了 Bosch BMP280 在一起。三个字,比较贵。
其实这个模块产品我是看了很久很久,一直就在购物车里面,最近才舍得拍下。
上一篇只从I2C总线操作 BMP280,http://www.cnblogs.com/leptonation/p/5210331.html,这次分三篇博文,我希望全面攻克这整个模块,用于我将来的遥控直升机项目里。上一篇已经介绍过 I2C 的基本操作,这次内容主要是 Data Sheet 资料和快速的命令行操作方式。
第一部分,是I2C接 MPU9250,操作里面的磁感应。
MPU9250 地址
接线和原理图在上一篇。说明书上说,MPU9250 I2C 从机地址是 110100X,LSB取决于 AD0 是否拉低。从原理图看到是拉低了的,就是 1101000(0x68),见下图:
图片中另一个地址,0x76,是上一篇博文的 BMP280 温度气压传感器地址。
Dump 一下:
看看说明,Device ID 在,WHO AM I 寄存器,地址 0x75:
再看看 Register 描述,只读 value 为 0x71,与上面 Dump 出来结果一致。
但事情还没结束。说明书提到 MPU9250,是磁感应作为一块,加速和陀螺仪为一块,分开两个 die house。看 Register Map 可见,两者是分开描述的,各自有自己的 Device ID。这是因为,磁感应是 Asahi Kasei Microdevices Corporation 生产的 AKM8963 集成进去的,它是连接着 AUX I2C 引脚的,不是我现在接着的总线。GY-91 MPU9250+BMP280 的情况如下面简图:
就是说,还有一个 Device ID,AKM8963的。要它出现,两个办法,第一,利用加速陀螺仪作为 MASTER 让它去读取 AKM8963 的 Device ID Register,或者,Pass-Through Mode,由 Host 接管(本实验的 Host 即 BBB)。两个都操作一次看看。
(1)MPU9250 作为 I2C MASTER 从 AUX I2C 与从机交互
MPU9250 作为 Master 最多只能储存 Slave 的 24 bytes,暂存 AKM 的感应数据,卓卓有余。寄存器是 EXT_SENS_DATA_00 至 23,地址 0x49 至 0x60。多一个从机也是一样,共用的,相关详情、顺序、重置等等的方法,请自己看说明书。
只读的话,关键部位五个:
- I2C Master Mode Enable(MPU 作为主机),
- Slave 0 Enable(启用第一个从机配置),
- Slave 0 I2C Address(从机实际总线地址),
- Slave 0 Register Address (开始读取的地址),
- Slave 0 Length (读取或写入的数据长度,多少个 byte)。
Slave 0 Enable 与 Slave 0 Length
从机连线读取数据的配置,在这里:
Enable 和 读取长度,都在 0x27 I2C_SLV0_CTRL 寄存器内设置。bit 7 为 1 时候启用,我只读一个 byte 所以 <3:0> 设成 1 即可。就是I2C_SLV0_CTRL(0x27) 设成 10000001 (0x81)。
Slave 0 Register Address
起始读取位置设置在 0x26 I2C_SLV0_REG。由于 AKM8963 的 Device ID 在 0x00,I2C_SLV0_REG(0x26) 默认值就是 0x00,不用调,我就只读它。
Slave 0 I2C Address
AKM8963 的从机地址,请参看这里:
http://www.akm.com/akm/en/file/datasheet/AK8963.pdf
文档描述截图中下半部的是加上的 LSB R/W bit, 只看地址不用管它,RW 在 MPU 那边配。7 bit 地址就 0001100 至 0001111 可选,视乎接线,我瞎猜,第一个就中了,0001100 (0x0C),然后回到 MPU9250 Data Sheet 的从机地址配置描述:
RW 在 MSB 的。我只读,那整个 I2C_SLV0_ADDR (0x25) 从机地址配置的 8 bit 值,就是 10001100 (0x8C)。
I2C Master Mode Enable
主机模式配置在这里USER_CTRL register (0x6A) 里面,bit5,I2C_MST_EN:
需要给它值,00100000(0x20)。
总结,用命令行的话,就是这样操作:
i2cset -y 1 0x68 0x25 0x8C
i2cset -y 1 0x68 0x27 0x81
i2cset -y 1 0x68 0x6A 0x20
MPU9250 直接读取从机 AKM 磁感应 Register 效果
再 dump 一次看看效果:
刚才说过,用来放从机读取值的第一个寄存器,EXT_SENS_DATA_00,地址 0x49,现在有值了,它就是从 AKM8963 的 0x00 Device ID 寄存器读取到的值,01001000(0x48),与 AKM 的 Data Sheet 描述一致:
成功。
从 MPU9250 对 AKM8963 的寄存器写入值
然后到 MPU 对 AKM 寄存器写入值。说在前面,说明书的建议是 I2C Pass Through 由 Host 做好 AKM 的配置,然后才使用 AUX I2C,从 SLV0 不断读取。
要 AKM 工作测量,就要设置它的工作模式,CNTL1 Register,地址 0x0A:
我选择连续测量模式,100Hz,16bit Output ,参考上图,0x0A 的值就应该设置为 000 1 0110(0x16),测试而已。知道了要设置哪个寄存器,知道要写入什么值,然后就是如何操作 MPU9250 的 AUX Slave 写入。方法就是把写入的值放在 I2C_SLV0_DO (0x63)。但首先,要配置好 SLV0 先。
指定 SLV0 参数,同上 0x25 指定AKM的 I2C地址,不过这次 bit 7 为零,写入模式。写入的寄存器地址是 0x0A。然后启用 SLV0,虽然是写入,但同样长度是一个byte,所以写入 0x81 到 0x27 寄存器:
i2cset -y 1 0x68 0x25 0x0C
i2cset -y 1 0x68 0x26 0x0A
i2cset -y 1 0x68 0x27 0x81
启用 I2C MASTER 模式(如果你之前还没启用的话):
i2cset -y 1 0x68 0x6A 0x20
最后写入值到 I2C_SLV0_DO,让 MPU 传给 AKM,0x16:
i2cset -y 1 0x68 0x63 0x16
怎么验证是否正确写入了?读出来就知道了,把地址 bit 7 改为 1:
i2cset -y 1 0x68 0x25 0x8C
效果:
上图红框,储存在寄存器 EXT_SENS_DATA_00 (0x49)的值 0x16 ,它就是通过 MPU 读取 AKM 的 0x0A 寄存器的值,同样地是我刚刚通过 MPU 写入的值。这样就确认了写入正确。指南针已经处于连续测量状态,你喜欢的还可以用 AKM 的状态寄存器读取状态,或者做些更有意义的事,看看已测量到的值:
目标数据,在 0x03 至 0x08,共 6 bytes。
i2cset -y 1 0x68 0x25 0x8C
i2cset -y 1 0x68 0x26 0x03
i2cset -y 1 0x68 0x27 0x86
取得的值,单位是 μT,上图中 X 值占 2 bytes,0x49 地址是 LSB,就是说十进位值为 32:
X |
32 μT |
|
Y |
65394 μT |
(注) |
Z |
65452 μT |
(注) |
(注)上表的 Y / Z 是错误的。写这部分时候,我实在非常大意,没注意是二补数,带正负值的。感谢网友RealMagic 提点指出问题,详见评论五楼,和六楼我的回复。
(2)Pass Through Mode,从 Host 用 I2C 总线直接连 AKM
Pass Through Mode 的设置,在 INT Pin / ByPass Register,寄存器地址 0x37,说明如下图:
之前提到这说明书建议,出自这里,Pass-Through mode is useful for configuring the external sensors,or for keeping MPU in a low-power mode ….。先看看怎么做,然后才判断到底应该怎样做。
MASTER 禁用然后 Pass Through 启用后,i2cdetect 就会看到 0x0c 直接出现在总线上,0x0c 就是磁感应的 I2C 地址。
操作一下,连续读取模式,然后 dump 就看到磁感应数据了。
就是普通的 I2C 通讯方式,非常简单。其他的写入值到寄存器等等,与其他 I2C 一样,不写出来了。
为何要用 Master Mode
既然 Pass Through 那么简单,为何不用 Pass Through 而要用 Master mode?因为我希望,生死攸关的数据,用最高速度传给控制器,然后它迅速作出适当反应。SPI 比 I2C 快,倾斜角,对于飞控是生死攸关,所以,我选用 SPI 连。
但是,GY91 这模块没有暴露 AKM 指南针的 SPI 接口给我,引脚的片选只有两个,一个是温度计,一个是 MPU,这情况,用 SPI 之后要接 AKM 我别无选择,只能 MASTER MODE。下一篇,SPI 通讯,读取 Accelerometer/Gyroscope(飞控当前姿态),再通过 Master Mode 用 AUX I2C 定时获取 Magnetometer(飞控航角)。希望在下一篇,事情会变得稍为有趣一些。
我在这群里,欢迎加入交流:
开发板玩家群 578649319
硬件创客 (10105555)