-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathmadgwickFilter.c
140 lines (110 loc) · 7.01 KB
/
madgwickFilter.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
//
// madgwickFilter.c
// madgwickFilter
//
// Created by Blake Johnson on 4/28/20.
//
#include "madgwickFilter.h"
struct quaternion q_est = { 1, 0, 0, 0}; // initialize with as unit vector with real component = 1
struct quaternion quat_mult (struct quaternion L, struct quaternion R){
struct quaternion product;
product.q1 = (L.q1 * R.q1) - (L.q2 * R.q2) - (L.q3 * R.q3) - (L.q4 * R.q4);
product.q2 = (L.q1 * R.q2) + (L.q2 * R.q1) + (L.q3 * R.q4) - (L.q4 * R.q3);
product.q3 = (L.q1 * R.q3) - (L.q2 * R.q4) + (L.q3 * R.q1) + (L.q4 * R.q2);
product.q4 = (L.q1 * R.q4) + (L.q2 * R.q3) - (L.q3 * R.q2) + (L.q4 * R.q1);
return product;
}
// The resulting quaternion is a global variable (q_est), so it is not returned or passed by reference/pointer
// Gyroscope Angular Velocity components are in Radians per Second
// Accelerometer componets will be normalized
void imu_filter(float ax, float ay, float az, float gx, float gy, float gz){
//Variables and constants
struct quaternion q_est_prev = q_est;
struct quaternion q_est_dot = {0}; // used as a place holder in equations 42 and 43
//const struct quaternion q_g_ref = {0, 0, 0, 1};// equation (23), reference to field of gravity for gradient descent optimization (not needed because I used eq 25 instead of eq 21
struct quaternion q_a = {0, ax, ay, az}; // equation (24) raw acceleration values, needs to be normalized
float F_g [3] = {0}; // equation(15/21/25) objective function for gravity
float J_g [3][4] = {0}; // jacobian matrix for gravity
struct quaternion gradient = {0};
/* Integrate angluar velocity to obtain position in angles. */
struct quaternion q_w; // equation (10), places gyroscope readings in a quaternion
q_w.q1 = 0; // the real component is zero, which the Madgwick uses to simplfy quat. mult.
q_w.q2 = gx;
q_w.q3 = gy;
q_w.q4 = gz;
quat_scalar(&q_w, 0.5); // equation (12) dq/dt = (1/2)q*w
q_w = quat_mult(q_est_prev, q_w); // equation (12)
/* NOTE
* Page 10 states equation (40) substitutes equation (13) into it. This seems false, as he actually
* substitutes equation (12), q_se_dot_w, not equation (13), q_se_w.
*
* // quat_scalar(&q_w, deltaT); // equation (13) integrates the angles velocity to position
* // quat_add(&q_w, q_w, q_est_prev); // addition part of equation (13)
*/
/* Compute the gradient by multiplying the jacobian matrix by the objective function. This is equation 20.
The Jacobian matrix, J, is a 3x4 matrix of partial derivatives for each quaternion component in the x y z axes
The objective function, F, is a 3x1 matrix for x y and z.
To multiply these together, the inner dimensions must match, so use J'.
I calculated "by hand" the transpose of J, so I will be using "hard coordinates" to get those values from J.
The matrix multiplcation can also be done hard coded to reduce code.
Note: it is possible to compute the objective function with quaternion multiplcation functions, but it does not take into account the many zeros that cancel terms out and is not optimized like the paper shows
*/
quat_Normalization(&q_a); // normalize the acceleration quaternion to be a unit quaternion
//Compute the objective function for gravity, equation(15), simplified to equation (25) due to the 0's in the acceleration reference quaternion
F_g[0] = 2*(q_est_prev.q2 * q_est_prev.q4 - q_est_prev.q1 * q_est_prev.q3) - q_a.q2;
F_g[1] = 2*(q_est_prev.q1 * q_est_prev.q2 + q_est_prev.q3* q_est_prev.q4) - q_a.q3;
F_g[2] = 2*(0.5 - q_est_prev.q2 * q_est_prev.q2 - q_est_prev.q3 * q_est_prev.q3) - q_a.q4;
//Compute the Jacobian matrix, equation (26), for gravity
J_g[0][0] = -2 * q_est_prev.q3;
J_g[0][1] = 2 * q_est_prev.q4;
J_g[0][2] = -2 * q_est_prev.q1;
J_g[0][3] = 2 * q_est_prev.q2;
J_g[1][0] = 2 * q_est_prev.q2;
J_g[1][1] = 2 * q_est_prev.q1;
J_g[1][2] = 2 * q_est_prev.q4;
J_g[1][3] = 2 * q_est_prev.q3;
J_g[2][0] = 0;
J_g[2][1] = -4 * q_est_prev.q2;
J_g[2][2] = -4 * q_est_prev.q3;
J_g[2][3] = 0;
// now computer the gradient, equation (20), gradient = J_g'*F_g
gradient.q1 = J_g[0][0] * F_g[0] + J_g[1][0] * F_g[1] + J_g[2][0] * F_g[2];
gradient.q2 = J_g[0][1] * F_g[0] + J_g[1][1] * F_g[1] + J_g[2][1] * F_g[2];
gradient.q3 = J_g[0][2] * F_g[0] + J_g[1][2] * F_g[1] + J_g[2][2] * F_g[2];
gradient.q4 = J_g[0][3] * F_g[0] + J_g[1][3] * F_g[1] + J_g[2][3] * F_g[2];
// Normalize the gradient, equation (44)
quat_Normalization(&gradient);
/* This is the sensor fusion part of the algorithm.
Combining Gyroscope position angles calculated in the beginning, with the quaternion orienting the accelerometer to gravity created above.
Noticably this new quaternion has not be created yet, I have only calculated the gradient in equation (19).
Madgwick however uses assumptions with the step size and filter gains to optimize the gradient descent,
combining it with the sensor fusion in equations (42-44).
He says the step size has a var alpha, which he assumes to be very large.
This dominates the previous estimation in equation (19) to the point you can ignore it.
Eq. 36 has the filter gain Gamma, which is related to the step size and thus alpha. With alpha being very large,
you can make assumptions to simplify the fusion equatoin of eq.36.
Combining the simplification of the gradient descent equation with the simplification of the fusion equation gets you eq.
41 which can be subdivided into eqs 42-44.
*/
quat_scalar(&gradient, BETA); // multiply normalized gradient by beta
quat_sub(&q_est_dot, q_w, gradient); // subtract above from q_w, the integrated gyro quaternion
quat_scalar(&q_est_dot, DELTA_T);
quat_add(&q_est, q_est_prev, q_est_dot); // Integrate orientation rate to find position
quat_Normalization(&q_est); // normalize the orientation of the estimate
//(shown in diagram, plus always use unit quaternions for orientation)
}
/*
returns as pointers, roll pitch and yaw from the quaternion generated in imu_filter
Assume right hand system
Roll is about the x axis, represented as phi
Pitch is about the y axis, represented as theta
Yaw is about the z axis, represented as psi (trident looking greek symbol)
*/
void eulerAngles(struct quaternion q, float* roll, float* pitch, float* yaw){
*yaw = atan2f((2*q.q2*q.q3 - 2*q.q1*q.q4), (2*q.q1*q.q1 + 2*q.q2*q.q2 -1)); // equation (7)
*pitch = -asinf(2*q.q2*q.q4 + 2*q.q1*q.q3); // equatino (8)
*roll = atan2f((2*q.q3*q.q4 - 2*q.q1*q.q2), (2*q.q1*q.q1 + 2*q.q4*q.q4 -1));
*yaw *= (180.0f / PI);
*pitch *= (180.0f / PI);
*roll *= (180.0f / PI);
}