一、 MPU6050简介
3 \7 ]9 V1 k: J. `8 qMPU6050集成了3轴加速度和3轴陀螺仪,是一款不错的传感器模块,可以用在很多方面,比如:四轴飞控、空中鼠标、两轮平衡车、GPS定位方面、游戏机、3D遥控器、平板设备等等,此模块给我们提供了强大的数据供应,由于将加速度和陀螺仪集成到了一起,免去了组合这两个模块时之间的轴差问题,减少了包装问题,这两年很火,成了DIY制作者的最爱。
# e N M" u* I+ ]5 v% p2 B% E2 ?6 _- N, J
二、 细节问题) @9 P$ l9 v/ M; |) L
在淘宝上买MPU6050时,要注意一下几点:; r7 A1 ]. @* n' y+ k
9 y+ P! I" Z" K7 y+ C2 O
1.查看卖家给出的介绍信息,是否刻意夸大,结合自己所学的知识进行判断。比如:当时在某个论坛上看到某位大神将的,tb上说他们的模块采用高性能的微处理器和先进的动力学解算与卡尔曼动态滤波算法,能够快速求解出模块当前的实时运动姿态。细心的你就会发现它的这种处理器根本就不能提供这种需求,也就是会所计算不出来。2.不要让模块受到碰撞,否则会影响他的性能。
, l `$ `/ V, \! B0 x% ]& P$ H. J
三、相关技术
* C' |% w' b6 H5 B3 qMPU6050数据是用IIC进行读取的,So必须学会IIC。类似于USB协议,不过和USB比起来可以说是小巫见大巫。4 J7 _+ X& f8 }6 n3 c
7 J% Q F3 m$ Z7 ?
1.IIC技术概述
& y) M% ]* a/ Z2 [1 E3 bIIC 即Inter-Integrated Circuit(集成电路总线),这种总线类型是 由飞利浦半导体公司在八十年代初设计出来的两线式串行总线
% D( J% E' t% G: j9 W特点:接口线少、器件封装形式小、通信速率较高5 @) {4 z' s% W" B; |/ ^; ]
IIC总线只有两根双向信号线,如下图所示:
4 R7 m1 T5 h; ~; [* C, `% R M) T! z! Q* z9 `! j+ @+ I
6 u* e; I+ m5 H$ P
- M& c {) u. h* W2.IIC数据传输2 U0 n) E& p$ ?7 B- \
数据有效位的定义:
9 w9 a5 Q G4 V `6 a IIC总线在进行数据传输时,时钟信号为高电平期间,数据线上的数据必须保持稳定,只有在时钟线上的信号为低电平期间,数据线的高电平或低电平状态才允许变化
6 V* I; ^/ X+ {起始和停止条件:: m1 r8 i8 f Z% s; z! ?* F
当SCL线是高电平时,SDA线从高电平向低电平切换,这个情况视为起始条件。
, c0 a' I) S7 }# ?, J, X当SCL线是高电平时,SDA线有低电平向高电平切换,表示停止条件
: f' ^: T ?1 a i8 v# w, ^: a数据传输格式:
5 `+ {" E% Z, k, [- t" v8 F7 K' J6 R. R d$ s+ E w7 ~2 C
$ m/ X6 L, Z7 R3 G9 f9 h- r% l
0 t* m% S& [+ S. U! \: ?
3.IIC总线寻址
" p) B/ p7 j; k+ ?! Z- [: z6 M; NIIC总线规定:从机地址有第一个字节的7位组成: I9 y2 H/ S B' D5 L6 H" D
(想要插入表格,可是在这里不会用了,直接从world中截张图算了)
6 Z1 X6 j' G9 A' N/ K* ]. p, n/ M" [% [& B5 D
2 z% s4 Q% ]. M# |/ h6 F- ]* \' ?4 l! \4 g1 {7 F
4.IIC总线编号
0 G, M( A0 U2 J" b; t( h( K* M2 @从机的地址有固定部分和可编程部分组成。在一个系统中可能希望接入多个相同的从机,从机地址中科编程部分决定了可接入总线该类期间的最大数目
# B }6 z. [. E/ X- Y) a4 Q
- p) f' N' x/ d) t2 E+ D例如:一个从机的7位寻址中有4为是固定的,3位是可编程的,那个这时仅能寻址8个同类期间。3 _' N6 n+ T) `2 A
7 j M- o3 B3 o
5.STM32F103中的IIC B% }; I! n% S$ `: i1 v, s
这里主要看下载STM32中,IIC是怎么个分布:
/ s+ O# Z4 J3 ]( l) p* Y1 n$ r2 b2 u5 c8 y/ ?4 P4 i
) O0 s6 }1 \) @! V9 o
$ v: A: { p K/ _* W9 }
四、操作步骤& M3 j$ P: q( a, L2 V" H3 d7 _
1.熟悉MPU6050管脚
) Q: P1 K" {2 ?6 y0 v6 `3 B* [: o. K; c7 M# `% @; l
6 j1 L! {# d! [! F a( d! b" S) v8 X. z2 n1 X2 X- v
上图是我自己的模块,由于连上了线,管脚不是很清楚,可以看着一张,管脚和清楚的。
* x* j+ v# `0 M6 }( q5 D/ O( E& s$ O8 R8 A2 b5 {
|9 V' W! u1 z) A; S. g
$ i% _, W$ U1 d* p2.管脚功能介绍: o {8 {) X# i: P! W7 a
3 j+ q4 [7 ?; V7 J$ e
+ e2 x8 N# S: u8 W
) s( e8 q4 n% F# E7 V6 v& N3 @这里已经很详细的介绍了各个管脚的功能,在心里就有个大概了解,在后面编写代码中就会明白好多,更多细节可以查看手册的!0 |5 `3 ~# y, E( @6 z- ?
3 p/ x q5 y- u' x6 i0 Q
2 |; r& v+ |. d9 `6 [
五、实现代码9 Y+ g& S+ y5 G7 N
当时开始做这个东西的时候参考了原子、野火的各个例子,再次感谢你们。7 M9 B8 \3 m9 D
2 \% r2 H- o8 O
代码块
* S/ M. E' | ~& v9 s' A" {MPU初始化:# G+ X! G; u! U/ P: s6 r0 }
2 T& A+ s4 D5 {% Y- void MPU_Init()
* Z6 ?7 i$ l/ t! P" t - {& H. M3 l5 |3 |2 }& ~- h7 O# I# }
- ANBT_I2C_Configuration(); //IIC初始化
/ q0 I& R9 S( U/ X0 a! s9 w7 W - delay_ms(30);3 o9 g9 x* m+ m2 x
- AnBT_DMP_MPU6050_Init(); //MPU6050 的DMP初始化
- `+ h9 U% T6 ^ - }
复制代码 8 W" o- p( C" l, ?8 b- i: e
下边是读取陀螺仪的数据函数:
; R2 }& B+ @6 H5 `# g: D7 j* I. T1 D, d4 v# s/ s' m
- void Read_Gyro_data(short *rxbuf)/ Q0 c' Q6 j0 ^& _
- {
* A9 ~: O0 g3 h( q1 b. j - unsigned long sensor_timestamp;, G" V7 N: O# v( T" R2 |5 z2 @( x" w( s
- unsigned char i = 0;
, r) W1 Z9 N+ h+ z. \* e - short gyro[3], accel[3], sensors;//陀螺仪存放数组,加速度存放数组,返回状态量
; Q% O% X( z8 y - unsigned char more;# m1 y4 a0 r$ c3 h
- long quat[4]; //四元数存放数组) {, T4 t. d( y0 G, c& ^: G" Z
, \, y# u; e3 y; z4 O- dmp_read_fifo(gyro, accel, quat, &sensor_timestamp, &sensors,&more); ^/ O- r `1 X1 P' K
- if(sensors & INV_WXYZ_QUAT)
6 y% ]6 x, ?( Y* \ - {
. y3 D4 ]& X1 h! s7 d - rxbuf[0] = gyro[0];
) f t; k% H$ W/ w7 h% M9 ?: I; T0 J - rxbuf[1] = gyro[1];( x- b" G8 k! i% n8 Q! I0 E
- rxbuf[2] = gyro[2];
! ?* s$ w D/ Z) o- y! L' H# O2 o
* Z# e) s- x6 e( _- printf("\r\nRead_Gyro_data gyro:\r\n");
7 p5 N3 p/ N% ]$ o( M! g - for (i = 0; i < 3; i++)
+ c" \% U, p3 ?, b - {
4 L$ E2 m3 X! G g, M' H. m - printf(" %d ",gyro<i>); X% }; [9 [3 p0 a* J, ?
- }( @3 @9 H/ L2 r$ K, s
- printf("\r\n");2 j' b8 W% q- {. t4 J$ d
- }
" [! x, F7 |& s: T, E - }</i>
复制代码 6 w! F8 f, {$ S! a2 @; ]* v
其实参考了圆点博士的开源代码的,看了下匿名四轴下位机协议,当初试了下自己获取的数据是否正确,很不错的,下面是匿名的协议:
$ E- I# o& W0 c: s& c/ _7 G( g6 ]- I3 m. O: p
- /*
' V5 g$ A) I# q1 f* [% r! b0 Q - * 函数名:Data_Send_Status* h! m: U# Y+ K! ~1 [, a
- * 描述 :数据发送 传感器 的状态
4 m! T9 c7 `5 E6 v! i* C - 根据匿名四轴最新上位机编写的显示姿态的程序
$ P6 z( I& e5 K" {# C - * 输入 :Pitch:俯仰角
m1 U9 F. f% Q+ T n n* @ - Roll :横滚角/ L: b4 g; R8 y; ]
- Yaw :航向角7 c+ n* l, d+ v* }' Z- \
- gyro :陀螺仪1 R1 H1 J% R. {+ @0 R/ O
- accel:加速度
% O) ^% H. v1 q5 p' h" A - * 输出 :4 I& O% E. _9 r/ @
- * 调用 :7 s& Y1 A7 y% q/ D9 ]3 v
- */
W; `% d9 M q7 C- H7 |/ G: q% l - void Data_Send_Status(float Pitch,float Roll,float Yaw,int16_t *gyro,int16_t *accel)! M% O& F/ j1 a& v# z
- {
5 Y# ?. e2 h$ q - unsigned char i = 0;
" c$ q; D, V" ]# h0 ]/ g# n - unsigned char j = 0;
1 G* w9 ` z) p7 X7 \; T' A - unsigned char _cnt = 0,sum = 0;1 x& q# D) n: v; D* ^6 ?' e
- unsigned int _temp;
$ l9 Z O. P& u" B
2 R0 i1 ~) v- A p. |- u8 data_to_send[12] = {0}; //发送数组,初始化为0
& O/ v" @; z3 C/ n$ s4 `+ L; \ - 2 V4 o+ \" ]0 p
- data_to_send[_cnt++] = 0xAA; // 帧头 170$ E! @: W; g, Y) Q+ f
- data_to_send[_cnt++] = 0xAA; // 1707 [ B5 N) i# ]$ K& k; ]. m% }# s) w
- data_to_send[_cnt++] = 0x01; // 功能字 1
' f, W( N! f$ W3 M - data_to_send[_cnt++] = 0; // 长度 08 O1 D" _$ Q1 j0 D
- ( K& R/ T- B t, }) x* u- o
- //横滚角
0 l* N- P6 Z4 u0 U* z - _temp = (int)(Roll * 100);
; m; b3 r: ] y# ?. o - data_to_send[_cnt++] = BYTE1(_temp); // 高 8 位
' q" J/ w. f* W2 i6 C - data_to_send[_cnt++] = BYTE0(_temp); // 低8位
5 O7 k# a3 f. b$ I; p y2 A - 4 Z& B) y3 |$ h4 Q. {3 ^
- //俯仰角+ [' I- f2 z' v- O. |# L2 k, A- G
- _temp = 0 - (int)(Pitch * 100);% S: Q+ E+ I' W! x$ ]6 p. |
- data_to_send[_cnt++] = BYTE1(_temp);* P S$ E' Y* B8 m
- data_to_send[_cnt++] = BYTE0(_temp);- X' H4 @3 v( Q& g% Y3 _" k6 T
$ q1 a+ J7 U( e+ m. {2 h9 ]' A+ `- //航向角
, F+ a& x3 Q \ - _temp = (int)(Yaw * 100);- [2 A( [) ]: L; c7 O
- data_to_send[_cnt++] = BYTE1(_temp);
- ?0 f2 }( @) x- ~7 ?8 a - data_to_send[_cnt++] = BYTE0(_temp);9 V |/ E% H5 f* m: Z
* o' _$ R1 j7 `- n( \, E# `- data_to_send[3] = _cnt - 4; //数据长度
9 S1 j6 ~7 x" P7 {
* @9 Y1 r+ y/ t3 K8 w; g/ E- //和校验6 [: e* b* u# y6 U
- for(i = 0;i < _cnt;i++)
, C+ h/ o4 p& i* J - <i> sum += data_to_send;. `9 u* I/ c( p
- data_to_send[_cnt++] = sum;5 A* ?( w* I" Z, S3 v! ]9 t
: t1 n' U% ^, x( N, M* ]- data_to_send[_cnt++] = '\0';
+ U+ V& ]" w+ x6 m
( M/ i( T4 G+ j7 [, d- //NRF_Tx_Dat(data_to_send); //发送到NRF缓冲区" m( L. \. Y/ ]+ [- J
- //printf("\r\nData_Send_Status:\r\n");
" ^4 c+ P- @6 S; T; _* q - // for (j = 0; j < 12;j++)
$ B- f5 U2 P5 U8 ^ - // {
3 ^' R8 d; N1 @7 P; U - // printf(" %d ",data_to_send[j]);4 ?% w' }& f; E I' [
- // }3 M# V' q3 R7 J: n# P4 |
- // printf("\r\n");
; S4 [2 M2 x6 I# A% R4 ] - // //串口发送数据
% t# |7 ~' A9 D. c - for(i=0;i<_cnt;i++)( v/ Y0 g; `" @4 q0 I% G$ f
- AnBT_Uart1_Send_Char(data_to_send);
; x$ f+ Y, Y$ w u I$ M% }. I' `5 L* N - " Q- D6 w5 ` E' ^0 {7 z; @
- }
- b1 H' |$ u* s2 `7 u* I
" x( K1 g$ E! W9 j1 ^! S- /*
7 u7 q5 G7 F6 m. b' K: q8 F! n - * 函数名:Send_Data3 X( E4 L/ ?$ h8 M
- * 描述 :用于发送传感器的数据
1 a/ M: }: q3 N8 d. M( G' a - * 输入 :Gyro:陀螺仪存放字符指针7 b5 j" y5 `$ [' O8 u( j& j) T1 k- R
- Accel:加速度存放指针
7 N" v8 F% w# `1 _" B$ M9 j) Y - * 输出 :2 Z9 X0 v3 X E: a. y6 q" \' ?
- * 调用 :- r& P* S# L6 u0 z
- */
+ O) r0 d# S4 U$ K- x2 Q - void Send_Data(int16_t *Gyro,int16_t *Accel)
- Z8 V( T2 x" b" x9 _) a - {+ y; c; s# h. L% B x" y
- unsigned char j = 0;$ L+ Q* M& R$ z. `
- unsigned char i = 0;
' @: ~: d8 p! C3 r- N: i% v - unsigned char _cnt = 0;$ `$ L1 C. ^ [
- unsigned char sum = 0;
4 S7 K# o- ~* ?/ e! u - u8 status;& M. [! b$ {" b- f% j) y
- // unsigned int _temp;4 O3 D9 A2 C0 Y( [
- /*; D' {# _5 v5 Q. L# N' K2 {8 n2 L, F$ F
- 匿名数据协议:) D( W2 f$ q4 N1 e* ~' @0 M
- 帧头、功能字、长度(len)、数据、校验和(Sum) 共32字节& {: O" t# N) h9 ~. Q
- 接收端在接收到数据的时候进行对应的解析即可! @5 u) e& ]: Q/ d2 U; R
- */
) |9 I3 T( X( a( h/ T - u8 data_to_send[12];; `" z4 ]# n7 i, _$ r
7 ]5 z; p' ]) x' g. _- data_to_send[_cnt++] = 0xAA; //帧头 AAAA) i( i9 Y8 r) \- d% _" ~* [
- data_to_send[_cnt++] = 0xAA; & a' t+ H4 u/ Q/ C
- data_to_send[_cnt++] = 0x02; //功能字$ `5 @7 v2 w4 g" D6 K" g
- data_to_send[_cnt++] = 0; //长度* u F8 ^: l; f* e6 I! S0 s
- ' ~/ S" Y0 x' m$ g1 [% [
- ' S* J5 T# {+ f6 @; Y; K3 B, ?9 K! a
- data_to_send[_cnt++] = BYTE1(Accel[0]);
- m' t) ] N9 Y- @ - data_to_send[_cnt++] = BYTE0(Accel[0]);# M- ?- n* [, l* j- p$ f
- data_to_send[_cnt++] = BYTE1(Accel[1]);
9 }8 U6 b' \6 [. h. x - data_to_send[_cnt++] = BYTE0(Accel[1]);+ I4 @1 U. h i3 p
- data_to_send[_cnt++] = BYTE1(Accel[2]);
, d) U+ q7 o7 W - data_to_send[_cnt++] = BYTE0(Accel[2]);
' I: k( ^4 | F1 B - 7 `+ @. H3 }! Y5 n" A# u
- data_to_send[_cnt++] = BYTE1(Gyro[0]);
7 r- j5 r& V' r4 {" w$ f3 V3 a - data_to_send[_cnt++] = BYTE0(Gyro[0]);
2 X3 m7 j: O$ N$ @3 ^9 U; e* S6 Z2 l$ t - data_to_send[_cnt++] = BYTE1(Gyro[1]);
* `* g( R7 [( W - data_to_send[_cnt++] = BYTE0(Gyro[1]);
+ P# T* b8 k1 ]; I9 } - data_to_send[_cnt++] = BYTE1(Gyro[2]);% G9 a! E5 @9 q$ _! H: }2 [
- data_to_send[_cnt++] = BYTE0(Gyro[2]);
$ b6 d: q) ?: u \. E | - data_to_send[_cnt++] = 0;
$ _; F/ a1 A7 O. R" a R$ [ - data_to_send[_cnt++] = 0;
0 R6 G# I0 v" C! Y9 n' I/ e - data_to_send[_cnt++] = 0;
j8 R2 y3 g, Q) B
* ^; Y, N$ s( ?9 r- data_to_send[3] = _cnt - 4;
) G( z% \, Q$ F - for(i = 0;i < _cnt;i++)
O2 c$ l2 _0 b: ?- ^. K( o - sum += data_to_send;
# R* Y1 g! A9 y B; R7 \% y$ K - 5 {- {1 f3 Z. c$ L. m j
- data_to_send[_cnt++] = sum;
5 r) Q' t% b2 i, P- d7 y4 i - * }2 l2 e' f: q
- data_to_send[_cnt++] = '\0';
& K* A: N5 L" f1 g/ f - 6 u+ C& m! G, |/ S* r! {
- // printf("\r\nSend_Data:\r\n");
3 L7 n- F! p* f) i - // for (j = 0; j < 12;j++)
% l5 X" P( g' z: z0 ~! b; X - // {' X3 b: G9 L' E
- // printf(" %d ",data_to_send[j]);/ Q# t2 E- @! l' x! l; P
- // }) |( v R( ^ J6 ^
- // printf("\r\n");
! Z$ K# ~& L9 `0 C" k& c - // delay_ms(1000);
9 F( |% U# E: w+ Z5 h - // status = NRF_Tx_Dat(data_to_send);
& r5 U, I; i( L' {! i: l+ O& I& v - // 0 |( o$ q( Z! l5 W
- // /*判断发送状态*/ f8 P5 j4 z! n# I$ `8 ^# M! t
- // switch(status)
' c/ R5 }% E/ L, p4 w& C - // {1 ]1 s/ m2 Y# x
- // case MAX_RT:" I; d1 Q( S8 w2 `
- // printf("\r\n 主机端 没接收到应答信号,发送次数超过限定值,发送失败。 \r\n");
- I( W+ \* Z/ e - // break;
; c6 C1 L; S# _8 o) ^% \ - ! I- R9 A+ ?% X4 a5 q0 d, @" ]
- // case ERROR:' d! I) o" O' g
- // printf("\r\n 未知原因导致发送失败。 \r\n");: W7 P k! J k$ M
- // break;% g$ I/ r& U( f9 g( x" Z* {5 z
- - d; ]% R. B6 Y
- // case TX_DS:
* l/ q" U8 @; P! ?1 p - // printf("\r\n 主机端 接收到 从机端 的应答信号,发送成功! \r\n"); % y" J$ Q( b' A7 L5 ?% r7 P
- // break; 1 d7 }5 T$ A3 l( v9 z
- // }. q2 v% c" s1 Q+ p6 `1 y
- //
4 D- M' f4 w. R6 v( `; [1 B( y - // delay_ms(1000);( o! `+ O7 H- E( s
! i4 S# y' }% o1 y( n O& Q1 W' i- // //串口发送数据+ |; b7 K% g) j% W1 ~# J6 x \7 K( l( V
- for(i = 0;i <_cnt;i++)
& M7 ?4 H9 B. c$ C/ z8 O - AnBT_Uart1_Send_Char(data_to_send);6 H* ~6 r* j) ~$ ~0 I
- }</i>
复制代码 2 W! V( F6 V7 l+ u; O: D8 n7 t u
在主函数中调用相应的函数即可获取相关数据,做一个简单的测试。5 I6 f/ ~" X+ w5 ?* d6 B/ _
一下是我从串口中获取的数据:' k( m! b% u: U6 b7 R* l/ C, C
% H: G' l, h# U& v5 U% T3 ^( f( ]+ R' z# m5 g, l
以及其他数据:: O1 a6 J7 h7 _' T& z- E; V
3 v5 }/ j g% z2 S* f* H7 j! d
U5 W& c2 g+ f' O5 q, W! z- U. W$ W
/ ^. Q4 x/ ?+ H) Q因为代码中使用了圆点博士的开源代码,而且都是现成的,就不往出贴了……4 K+ e( Q- O. U* Z
以上只是做了个简单介绍,以及获取到了数据,但是并没有说明怎样处理数据,这是很关键的也是最重要的,本次无线飞鼠的数据不好处理,我做到现在数据也不是很稳定,任然有一定幅度的变化。
) l% U# J Q; e; c+ r0 \. R& _# U6 ] i3 i/ l
# u" j- I! s6 N$ B- b0 _' w: E C0 i* e7 h- X4 w; S
' T% I/ g- e5 H8 J$ D, k( F |