-
Notifications
You must be signed in to change notification settings - Fork 335
/
Copy pathSensorFusion.ino
134 lines (102 loc) · 3.15 KB
/
SensorFusion.ino
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
/* SensorFusion: Sensor fusion on Arduino using TinyEKF.
*
* Copyright (C) 2015 Simon D. Levy
*
* MIT License
*/
// These must be defined before including TinyEKF.h
#define EKF_N 2 // pressure, temperature
#define EKF_M 3 // baro pressure, baro temperature, LM35 temperature
static const uint8_t LM35_PIN = 0;
#include <tinyekf.h>
#include <SFE_BMP180.h>
#include <Wire.h>
static const float EPS = 1e-4;
static const float Q[EKF_N*EKF_N] = {
EPS, 0,
0, EPS
};
static const float R[EKF_M*EKF_M] = {
EPS, 0, 0,
0, EPS, 0,
0, 0, EPS
};
// So process model Jacobian is identity matrix
static const float F[EKF_N*EKF_N] = {
1, 0,
0, 1
};
static const float H[EKF_M*EKF_N] = {
1, 0,
0, 1,
0, 1
};
static ekf_t _ekf;
static SFE_BMP180 baro;
// Adapted from https://github.com/sparkfun/BMP180_Breakout
static void getBaroReadings(double & T, double & P)
{
char status = baro.startTemperature();
if (status != 0) {
delay(status);
status = baro.getTemperature(T);
if (status != 0) {
status = baro.startPressure(3);
if (status != 0) {
delay(status);
status = baro.getPressure(P,T);
if (status == 0)
Serial.println("error retrieving pressure measurement");
}
else Serial.println("error starting pressure measurement");
}
else Serial.println("error retrieving temperature measurement");
}
else Serial.println("error starting temperature measurement");
}
void setup()
{
// Use identity matrix as initiali covariance matrix
const float Pdiag[EKF_N] = {1, 1};
ekf_initialize(&_ekf, Pdiag);
Serial.begin(115200);
// Start reading from baro
baro.begin();
// Set up to read from LM35
analogReference(INTERNAL);
}
void loop()
{
double baroTemperature, baroPressure;
getBaroReadings(baroTemperature, baroPressure);
// Read temperature from LM35
const float lm35Temperature = analogRead(LM35_PIN) / 9.31;
// Set the observation vector z
const float z[EKF_M] = {baroPressure, baroTemperature, lm35Temperature};
// Process model is f(x) = x
const float fx[EKF_N] = { _ekf.x[0], _ekf.x[1] };
// Run the prediction step of the DKF
ekf_predict(&_ekf, fx, F, Q);
// Measurement function simplifies the relationship between state
// and sensor readings for convenience. A more realistic
// measurement function would distinguish between state value and
// measured value; e.g.:
// hx[0] = pow(this->x[0], 1.03);
// hx[1] = 1.005 * this->x[1];
// hx[2] = .9987 * this->x[1] + .001;
const float hx[EKF_M] = {_ekf.x[0], _ekf.x[1], _ekf.x[1] };
// Run the update step
ekf_update(&_ekf, z, hx, H, R);
// Report measured and predicte/fused values
Serial.print("BMP180Press:");
Serial.print(z[0]);
Serial.print(" ");
Serial.print(" BMP180Temp:");
Serial.print(z[1]);
Serial.print(" LM35Temp:");
Serial.print(z[2]);
Serial.print(" EKFPress:");
Serial.print(_ekf.x[0]);
Serial.print(" EKFTemp:");
Serial.println(_ekf.x[1]);
}