找回密码
 注册
关于网站域名变更的通知
查看: 745|回复: 1
打印 上一主题 下一主题

基于9轴惯性运动传感器的三阶卡尔曼滤波器算法

[复制链接]

该用户从未签到

跳转到指定楼层
1#
发表于 2019-2-21 12:30 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式

EDA365欢迎您登录!

您需要 登录 才可以下载或查看,没有帐号?注册

x
基于9轴惯性运动传感器的三阶卡尔曼滤波器算法
0 l9 t/ x! O& c8 G) @/ Z
4 T2 F0 U% z+ j0 b
最近在玩九轴的惯性传感器,很是有挑战性.九轴说的是三轴的加速度计、三轴的陀螺仪以及三轴的磁场传感器。但是只是单纯的测出九个轴的数据没什么用,关键是要能够融合这九轴数据得出我们想要的结果。这里就运用三阶卡尔曼滤波算法来融合这九轴运动数据为三轴的角度。运用这三个角度可以用来做自平衡车或者四轴飞行器.
一、卡尔曼算法理解
其实如果不去考虑kalman算法是怎么来的,我们只需要知道有下面几个式子就可以了,具体意思可以看上面的wikipedia链接
二 卡尔曼滤波算法的实现
这里我的算法是运行在avr单片机上的,所以采用的是c语言写的。下面的代码是要放到avr的定时器中断测试刷新的。用示波器测试了一下,这个算法在16M晶振下的运行时间需要0.35ms,而数据采集需要3ms左右,所以选定定时器时间为8ms.之前也写过一阶的kalman算法,运用在自平衡车上,这边是三阶的,主要是矩阵运算.
//kalman.c
1 V- A8 R& r% B" q
float dtTimer   = 0.008;
2 L' s  N3 X8 L7 l5 K4 O0 z) f
float xk[9] = {0,0,0,0,0,0,0,0,0};
/ M  k8 S2 @2 a; V% [7 R% I7 V
float pk[9] = {1,0,0,0,1,0,0,0,0};

: |4 [9 C- c9 y) S* b
float I[9]  = {1,0,0,0,1,0,0,0,1};

0 k3 R) ]9 F6 C3 Z
float R[9]  = {0.5,0,0,0,0.5,0,0,0,0.01};
$ R$ \$ t/ e+ N+ Q* m
float Q[9] = {0.005,0,0,0,0.005,0,0,0,0.001};

: J# a& t5 F* u4 `5 W- V- e4 h+ t; c) O( M
void matrix_add(float* mata,float* matb,float* matc){
3 t% V! D4 u  T8 M; Z
    uint8_t i,j;
; r( j7 {- e+ H$ I  w2 Q& J
    for (i=0; i<3; i++){

- p1 g8 P6 Y# A. H% l( J: m5 g
       for (j=0; j<3; j++){
7 V4 b- C- `) S3 z4 I
          matc[i*3+j] = mata[i*3+j] + matb[i*3+j];
' _3 s+ a; G( q- M' m
       }

1 a$ F) N! X* U# @
    }

  A- H2 y; k4 R' p9 B9 M) R
}
7 e1 N& r  e' i5 `" `  g* k" ^

' L! E* f& M+ ]1 ?0 ?: n
void matrix_sub(float* mata,float* matb,float* matc){

+ v% R' ]; O  v0 f# |  M
    uint8_t i,j;

* Q# ]: S0 R' K# r' t  I" |
    for (i=0; i<3; i++){

; N# R1 P5 ~% ~/ E# A
       for (j=0; j<3; j++){
, }) X) x1 j* ?" v
          matc[i*3+j] = mata[i*3+j] - matb[i*3+j];

+ }" Y- g7 i1 e2 g
       }

% M6 ?3 p: R9 J9 E4 r
    }
* F' O2 S7 [6 G! \: v; x
}

% p( p( I0 ~0 H5 y. I0 ~+ f/ t- Q  K  a' ]
void matrix_multi(float* mata,float* matb,float* matc){
* k  F, y( ^5 H9 [! Q
  uint8_t i,j,m;

' A" m& O( J$ b& N% W& a
  for (i=0; i<3; i++)
2 t9 j8 u! b. D) s
  {
7 h! x5 H- o- Z' |+ u+ G6 `6 t
    for (j=0; j<3; j++)
! b+ J& l7 U( q- C0 l. Q
   {

- {. G, V, I5 e4 j6 E) Q
      matc[i*3+j]=0.0;

; u$ N) A0 \, f. d
      for (m=0; m<3; m++)
1 B  Q; n4 s0 Y& s$ l( w* l$ `
     {

2 x, @9 s& v5 Q$ Q: D/ f, e9 x9 a6 Y
        matc[i*3+j] += mata[i*3+m] *  matb[m*3+j];
; b8 \) _/ P5 U1 |! v
      }
; {2 m% L/ z5 [  u
    }

' v4 s. b  O: h$ E% O0 W
}

' E' r, v/ C! ^( }2 q  R9 E* W
}
9 |/ y6 v7 I8 c# g* r
5 @' k3 x/ p$ `' a% g
void KalmanFilter(float* am_angle_mat,float* gyro_angle_mat){

* r0 I) z$ O5 n% ~  G4 E
uint8_t i,j;
( y  E( p7 A3 o0 I2 g: ~' J
float yk[9];

/ L& B) e+ s' y5 Z' J
float pk_new[9];
9 d3 K7 j& G3 E
float K[9];

, M6 T( w! G! G
float KxYk[9];
& C) s/ `& p. B4 c: y# e
float I_K[9];

9 v2 m( w0 @$ q$ f& \" @* u
float S[9];
+ c& ]5 ?6 c6 {" G, U7 x. B
float S_invert[9];
( }: I# }' I5 t
float sdet;
5 S' E& W& U7 v( v; c6 {8 U

7 B/ u4 `, P( ?( B  J/ I
//xk = xk + uk

5 H/ }7 e- s0 z
matrix_add(xk,gyro_angle_mat,xk);

% A7 Q' O' g2 M" B
//pk = pk + Q

+ M2 X: R, G: f- g+ B
matrix_add(pk,Q,pk);
7 x8 J& R1 G) x( m
//yk =  xnew - xk

2 o. j( @" |% A, E
matrix_sub(am_angle_mat,xk,yk);

: X  d; w3 w: @* J  H
//S=Pk + R
7 b! ]8 ]& c/ D4 ~
matrix_add(pk,R,S);
! P( c: r' K1 D" c0 R2 K6 `2 ^
//S求逆invert

4 ~& f: t  A" s* M* r/ Z* }
sdet = S[0] * S[4] * S[8]
4 y1 H  m9 l2 f" ~: {$ m
          + S[1] * S[5] * S[6]

+ S/ G6 h( w& t' J" o2 a) A' |. X  c
          + S[2] * S[3] * S[7]
* R4 U8 V# d$ {' t1 D+ X
          - S[2] * S[4] * S[6]

# D& U& V& ~+ m$ w7 w2 F
          - S[5] * S[7] * S[0]
- w4 E' T3 O$ A' }  A
          - S[8] * S[1] * S[3];

: _5 q/ h3 l4 u$ J, i0 i. a' c) z; A) |1 g  I
S_invert[0] = (S[4] * S[8] - S[5] *  S[7])/sdet;

7 S1 I8 d% _+ s& ]
S_invert[1] = (S[2] * S[7] - S[1] *  S[8])/sdet;

$ {+ b" D( a/ j
S_invert[2] = (S[1] * S[7] - S[4] *  S[6])/sdet;
4 N0 ?3 v. o" {0 A1 i+ @& E6 N# H
" P5 O2 J- D4 U6 l$ T" y1 F2 _) A
S_invert[3] = (S[5] * S[6] - S[3] *  S[8])/sdet;

6 f& L; e- K& p! d$ L. L
S_invert[4] = (S[0] * S[8] - S[2] *  S[6])/sdet;

( O/ A( u2 c4 b7 N0 H' _
S_invert[5] = (S[2] * S[3] - S[0] *  S[5])/sdet;

, X# o0 A* O  Y: D) B( X5 p* M+ c
- p  U2 G; z% U  M. s  O
S_invert[6] = (S[3] * S[7] - S[4] *  S[6])/sdet;

  C, ^, n) F; `- n
S_invert[7] = (S[1] * S[6] - S[0] *  S[7])/sdet;

) A$ p5 u* `# @4 s, j6 c7 J
S_invert[8] = (S[0] * S[4] - S[1] *  S[3])/sdet;

& C8 G( d6 P0 I3 u( e2 h
//K = Pk * S_invert
8 G  l- z! |' D2 L) @* ^  y! V
matrix_multi(pk,S_invert,K);

4 h  l% w& w9 I
//KxYk = K * Yk

6 T6 z7 l! u  F6 H! \
matrix_multi(K,yk,KxYk);

; H' }$ w$ y  u. {
//xk = xk + K * yk
" a* u! ^& h8 ]- L
matrix_add(xk,KxYk,xk);

+ s& p- M+ o5 P2 K
//pk = (I - K)*(pk)

$ Q/ \% @- L+ z. ?1 A& W4 D. M
matrix_sub(I,K,I_K);

6 @) }! M* u9 D6 e
matrix_multi(I_K,pk,pk_new);
- m- |+ a7 I: a$ d5 B
//update pk
9 ?, N: g; F! `$ K( X5 z7 t
//pk = pk_new;
8 _% E8 j* @- ^4 }- S1 j
for (i=0; i<3; i++){

9 N: m8 F0 \7 ]0 E2 [
    for (j=0; j<3; j++){

* B2 ]! f' Y( d4 ]
        pk[i*3+j] = pk_new[i*3+j];
1 g1 `  S3 j5 b! L2 W& ]9 j
    }
$ t' `- g6 V, R; L& [3 u* [) V. D" x* t
  }
# P& d3 H2 @0 _/ I" o. `* x- K
}
' o6 \% C' b; z6 f; X9 q$ x. [
8 m* f- h4 v# b7 s: j
三 运用卡尔曼滤波器
这里的kalman滤波器是离散数字滤波器,需要迭代运算。这里把算法放到avr的定时器中断里面执行,进行递归运算.
//isr.c
9 Y+ S; r* ~6 F
#include "kalman.h"

6 h6 \: K. b" |3 g
float mpu_9dof_values[9]={0};
3 K+ j8 h' }6 e+ f
float am_angle[3];

9 q8 I6 Z0 g4 `/ Z) i% `& V* y5 ?
float gyro_angle[3];
, R: y' h2 @/ H5 h7 ~
float am_angle_mat[9]={0,0,0,0,0,0,0,0,0};

! e8 I9 f: A4 j4 N  |6 P) {  m
float gyro_angle_mat[9]={0,0,0,0,0,0,0,0,0};
2 ^7 L  e& |; X0 N

5 N7 i+ r2 o' l! \# Y% W
//8MS

: Q0 A% M0 }) ]
ISR(TIMER0_OVF_vect){
  i, B" ~3 i( U; r
//设置计数开始的初始值

0 [9 d" t; y" z3 n9 D
TCNT0 = 130 ;  //8ms

: z( K- o5 B1 U' Z3 d
sei();

  G7 q; s1 W4 {% q
//采集九轴数据
  \. f; X% Y! h
//mpu_9dof_values 单位为g与度/s
. n* }' n; E# N" o* N9 G
//加速度计和陀螺仪

' `- q  l& \3 n/ e
mpu_getValue6(&mpu_9dof_values[0],&mpu_9dof_values[1],&mpu_9dof_values[2],&mpu_9dof_values[3],&mpu_hmc_values[4],&mpu_hmc_values[5]);

2 ]+ a* V1 u# [7 K$ C; `/ ~
//磁场传感器
7 W: G7 Z0 P9 j8 {
compass_mgetValues(&mpu_9dof_values[6],&mpu_9dof_values[7],&mpu_9dof_values[8]);
# j9 y; @. z9 H, \% k  u6 Y9 ]

% K, b7 M3 L/ F& W, L
accel_compass2angle(&mpu_9dof_values[0],&mpu_9dof_values[6],am_angle);
  o6 D& d. N, N! K& n2 f
gyro2angle(&mpu_9dof_values[3],gyro_angle);
9 R5 ]' T" p/ P  ~4 {
+ j3 n# k! n( z
am_angle_mat[0] = am_angle[0];

! Z$ T4 J3 z# d( ?  X+ ]2 [
am_angle_mat[4] = am_angle[1];
; i. o* g% n1 q1 e6 p& Z8 r5 L0 k
am_angle_mat[8] = am_angle[2];

7 ~# {( C; A( O' s6 H' i2 L2 J
4 \% I6 o$ d# _, H1 I: v4 B
gyro_angle_mat[0] = gyro_angle[1];

8 A5 M" z# D2 U/ o
gyro_angle_mat[4] = - gyro_angle[0];

9 B' ]+ r, `0 G! _* X9 ]) A3 v
gyro_angle_mat[8] = - gyro_angle[2];

3 m; q0 ]" ~+ y+ s2 W1 a6 e' ^& N% v; ]# g* u6 ^2 S
//卡尔曼
) ^. O" F6 X! q: B+ j
KalmanFilter(am_angle_mat,gyro_angle_mat);
) e- r8 P% V2 b+ v& T) ^

7 w. a8 Y; R& Z. L" x0 N& C7 X) c& R1 O
//输出三轴角度
+ U5 ]6 \# K+ e) j' V% t& N( ]
//xk[0] xk[4] xk[8]

4 I: @, I  P. @" P$ h  U+ b
}

4 @% m0 t8 u+ S" u

5 i' L" I) m& |2 ~/ H$ h
实测可以准确的输出三轴的角度,为了获得更好的响应速度和跟踪精度还需调整参数.
5 T$ y9 T1 _8 P

该用户从未签到

2#
发表于 2019-2-21 16:18 | 只看该作者
研究一下,谢谢分享
您需要登录后才可以回帖 登录 | 注册

本版积分规则

关闭

推荐内容上一条 /1 下一条

EDA365公众号

关于我们|手机版|EDA365电子论坛网 ( 粤ICP备18020198号-1 )

GMT+8, 2025-11-4 16:45 , Processed in 0.125000 second(s), 23 queries , Gzip On.

深圳市墨知创新科技有限公司

地址:深圳市南山区科技生态园2栋A座805 电话:19926409050

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