-
Notifications
You must be signed in to change notification settings - Fork 83
/
Copy pathAHRS.cpp
147 lines (121 loc) · 5.04 KB
/
AHRS.cpp
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
139
140
141
142
143
144
145
/*
* File: AHRS.cpp
* Author: matt
*
* Created on 29 October 2012, 22:38
*/
#include "AHRS.h"
#include "HMC5883L.h"
#include "MS5611.h"
//Values calculated from matlab script MgnCalibration
const double accelZeroX = 0.2238;
const double accelZeroY = 0.1543;
const double accelZeroZ = -0.3633;
const double accelEllipsoid00_ = 0.1007;
const double accelEllipsoid01_ = -0.0007;
const double accelEllipsoid02_ = 0.0002;
const double accelEllipsoid11_ = 0.1020;
const double accelEllipsoid12_ = 0.0005;
const double accelEllipsoid22_ = 0.1003;
//Values calculated from matlab script MgnCalibration
const double magZeroX = 0.0576;
const double magZeroY = -0.0929;
const double magZeroZ = -0.0092;
const double magEllipsoid00_ = 1.8925;
const double magEllipsoid01_ = 0.0399;
const double magEllipsoid02_ = 0.0132;
const double magEllipsoid11_ = 1.8375;
const double magEllipsoid12_ = 0.0474;
const double magEllipsoid22_ = 2.1528;
AHRSClass AHRS;
AHRSClass::AHRSClass() {
}
AHRSClass::AHRSClass(const AHRSClass& orig) {
}
AHRSClass::~AHRSClass() {
}
void AHRSClass::update() {
getSensors_();
calibrateData_();
temperatureCompensate_();
fuse_();
}
void AHRSClass::getSensors_() {
MPU6050.getSensors(&rawData_);
HMC5883L.getField(&rawData_);
MS5611.getPressure(&rawData_.pressure);
}
void AHRSClass::calibrateData_() {
calibratedData.x = (rawData_.x * (9.81 / 4096.0));
calibratedData.y = (rawData_.y * (9.81 / 4096.0));
calibratedData.z = (rawData_.z * (9.81 / 4096.0));
calibratedData.temp = (rawData_.temp + 12412) / 340.0;
calibratedData.p = (rawData_.p / 65.5);
calibratedData.q = (rawData_.q / 65.5);
calibratedData.r = (rawData_.r / 65.5);
calibratedData.magx = rawData_.mag_x / 1090.0;
calibratedData.magy = rawData_.mag_y / 1090.0;
calibratedData.magz = rawData_.mag_z / 1090.0;
calibratedData.pressure = rawData_.pressure; //Pascals
calibratedData.altitude = ((-8.31447 * 288.15) / (9.80665 * 0.0289644)) * log(calibratedData.pressure / 101325);
calibratedData.q = -calibratedData.q;
//Accelerometer scale and bias correction
static double acceltemp[3];
acceltemp[0] = calibratedData.x - accelZeroX;
acceltemp[1] = calibratedData.y - accelZeroY;
acceltemp[2] = calibratedData.z - accelZeroZ;
calibratedData.x = accelEllipsoid00_ * acceltemp[0] + accelEllipsoid01_ * acceltemp[1] + accelEllipsoid02_ * acceltemp[2];
calibratedData.y = accelEllipsoid11_ * acceltemp[1] + accelEllipsoid12_ * acceltemp[2];
calibratedData.z = accelEllipsoid22_ * acceltemp[2];
//Magnetometer scale and bias correction
static double magtemp[3];
magtemp[0] = calibratedData.magx - magZeroX;
magtemp[1] = calibratedData.magy - magZeroY;
magtemp[2] = calibratedData.magz - magZeroZ;
calibratedData.magx = magEllipsoid00_ * magtemp[0] + magEllipsoid01_ * magtemp[1] + magEllipsoid02_ * magtemp[2];
calibratedData.magy = magEllipsoid11_ * magtemp[1] + magEllipsoid12_ * magtemp[2];
calibratedData.magz = magEllipsoid22_ * magtemp[2];
//Altitude LPF
#define LENGTH 20
static int i = 0;
static double mvAvg[LENGTH] = {0};
mvAvg[i] = calibratedData.altitude;
calibratedData.altitude = 0;
for(int k = 0; k < LENGTH; k++) {
calibratedData.altitude += mvAvg[k];
}
calibratedData.altitude /= LENGTH;
i++;
if(i == LENGTH) {
i = 0;
}
//End Altitude LPF
}
void AHRSClass::temperatureCompensate_() {
static double tempPow1 = calibratedData.temp;
static double tempPow2 = pow(calibratedData.temp, 2);
static double tempPow3 = pow(calibratedData.temp, 3);
static double tempPow4 = pow(calibratedData.temp, 4);
//Coefficients calculated from freezetest4, 4th degree polynomial
calibratedData.p -= 8.4877e-9 * tempPow4 + 6.4219e-6 * tempPow3 + 2.5782e-4 * tempPow2 - 0.0041145 * tempPow1 - 1.2974;
calibratedData.q -= 5.863e-8 * tempPow4 - 5.9746e-6 * tempPow3 + 5.1324e-5 * tempPow2 + 0.0079355 * tempPow1 + 0.59859;
calibratedData.r -= 4.4929e-8 * tempPow4 - 1.6137e-7 * tempPow3 + 4.8876e-5 * tempPow2 + 0.021246 * tempPow1 - 2.9723;
//calibratedData.x -= -2.8664e-6 * tempPow2 + 4.9565e-4 * tempPow1; - 0.0011611;
//calibratedData.y -= 1.2728e-6 * tempPow2 + 6.5989e-6 * tempPow1; + 0.025702;
//calibratedData.z -= 1.6966e-5 * tempPow2 - 0.0035421 * tempPow1; + 0.056; //Z axis accel shows huge temperature drift (15% over 40 degrees)
}
void AHRSClass::fuse_() {
if(Timer.dt < 0.03) {
quaternion = EKF.predict(&calibratedData, Timer.dt);
}
quaternion = EKF.update(&calibratedData, Timer.dt);
quaternionToYPR_(&quaternion, &orientation);
}
void AHRSClass::quaternionToYPR_(QuaternionClass* q, s_euler* orientation) {
orientation->pitch = -(180/pi) * atan2(2*(q->w*q->x + q->y*q->z), 1 - 2*(pow(q->x,2)+pow(q->y,2)));
orientation->roll = (180/pi) * asin(2*(q->w*q->y - q->z*q->x));
orientation->yaw = (180/pi) * atan2(2*(q->w*q->z + q->x*q->y), 1 - 2*(pow(q->y,2)+pow(q->z,2)));
}
double AHRSClass::magnitude_(double x, double y, double z) {
return sqrt(pow(x, 2) + pow(y, 2) + pow(z, 2));
}