forked from MatrixPilot/MatrixPilot
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdeadReckoning.c
188 lines (151 loc) · 6.73 KB
/
deadReckoning.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
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
// This file is part of MatrixPilot.
//
// http://code.google.com/p/gentlenav/
//
// Copyright 2009-2011 MatrixPilot Team
// See the AUTHORS.TXT file for a list of authors of MatrixPilot.
//
// MatrixPilot is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// MatrixPilot is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with MatrixPilot. If not, see <http://www.gnu.org/licenses/>.
#include "libDCM.h"
#include "gpsParseCommon.h"
#include "deadReckoning.h"
#include "mathlibNAV.h"
#include "estWind.h"
#include "gpsData.h"
#include "rmat.h"
#include "../libUDB/heartbeat.h"
// heartbeats
#define DR_PERIOD (int16_t)((HEARTBEAT_HZ/GPS_RATE)+4)
// seconds
#define DR_TIMESTEP (1.0/HEARTBEAT_HZ)
// 1.0 in 0.16 format
#define MAX16 (4.0*RMAX)
// seconds
#define DR_TAU 2.5
// seconds * (cm/sec^2 / count) ??? is G always represented as cm/sec^2 ?
// GRAVITYM is 980 cm/sec^2, GRAVITY is 2000 counts
// dx/dt^2 * ACCEL2DELTAV = cm/sec
#define ACCEL2DELTAV ((DR_TIMESTEP*GRAVITYM*MAX16)/GRAVITY)
// seconds; the .01 must convert from cm/sec^2 to m/sec^2
// cm/sec * VELOCITY2LOCATION = meters
#define VELOCITY2LOCATION (DR_TIMESTEP*.01*MAX16*16.0)
// The factor of 16 is so that the gain is more precise.
// There is a subsequent right shift by 4 to cancel the multiply by 16.
// dimensionless
#define DR_FILTER_GAIN (int16_t)(DR_TIMESTEP*MAX16/DR_TAU)
// 1/seconds
#define ONE_OVER_TAU (uint16_t)(MAX16/DR_TAU)
int16_t dead_reckon_clock = DR_PERIOD;
// velocity, as estimated by the IMU: high word is cm/sec
union longww IMUvelocityx = { 0 };
union longww IMUvelocityy = { 0 };
union longww IMUvelocityz = { 0 };
int16_t forward_ground_speed = 0 ;
// location, as estimated by the IMU
// high word is meters, low word is fractional meters
union longww IMUlocationx = { 0 };
union longww IMUlocationy = { 0 };
union longww IMUlocationz = { 0 };
// integral of acceleration
union longww IMUintegralAccelerationx = { 0 };
union longww IMUintegralAccelerationy = { 0 };
union longww IMUintegralAccelerationz = { 0 };
uint16_t air_speed_3DIMU = 0;
int16_t total_energy = 0;
// GPSlocation - IMUlocation: meters
fractional locationErrorEarth[] = { 0, 0, 0 };
// GPSvelocity - IMUvelocity
fractional velocityErrorEarth[] = { 0, 0, 0 };
void dead_reckon(void)
{
int16_t air_speed_x, air_speed_y, air_speed_z;
union longww accum;
union longww energy;
if (dcm_flags._.dead_reckon_enable == 1) // wait for startup of GPS
{
// integrate the accelerometers to update IMU velocity
IMUintegralAccelerationx.WW += __builtin_mulss(((int16_t)(ACCEL2DELTAV)), accelEarth[0]);
IMUintegralAccelerationy.WW += __builtin_mulss(((int16_t)(ACCEL2DELTAV)), accelEarth[1]);
IMUintegralAccelerationz.WW += __builtin_mulss(((int16_t)(ACCEL2DELTAV)), accelEarth[2]);
// integrate IMU velocity to update the IMU location
IMUlocationx.WW += (__builtin_mulss(((int16_t)(VELOCITY2LOCATION)), IMUintegralAccelerationx._.W1)>>4);
IMUlocationy.WW += (__builtin_mulss(((int16_t)(VELOCITY2LOCATION)), IMUintegralAccelerationy._.W1)>>4);
IMUlocationz.WW += (__builtin_mulss(((int16_t)(VELOCITY2LOCATION)), IMUintegralAccelerationz._.W1)>>4);
if (dead_reckon_clock > 0)
// apply drift adjustments only while valid GPS data is in force.
// This is done with a countdown clock that gets reset each time new data comes in.
{
dead_reckon_clock --;
IMUintegralAccelerationx.WW += __builtin_mulss(DR_FILTER_GAIN, velocityErrorEarth[0]);
IMUintegralAccelerationy.WW += __builtin_mulss(DR_FILTER_GAIN, velocityErrorEarth[1]);
IMUintegralAccelerationz.WW += __builtin_mulss(DR_FILTER_GAIN, velocityErrorEarth[2]);
IMUlocationx.WW += __builtin_mulss(DR_FILTER_GAIN, locationErrorEarth[0]);
IMUlocationy.WW += __builtin_mulss(DR_FILTER_GAIN, locationErrorEarth[1]);
IMUlocationz.WW += __builtin_mulss(DR_FILTER_GAIN, locationErrorEarth[2]);
IMUvelocityx.WW = IMUintegralAccelerationx.WW +
__builtin_mulus(ONE_OVER_TAU, 100*locationErrorEarth[0]);
IMUvelocityy.WW = IMUintegralAccelerationy.WW +
__builtin_mulus(ONE_OVER_TAU, 100*locationErrorEarth[1]);
IMUvelocityz.WW = IMUintegralAccelerationz.WW +
__builtin_mulus(ONE_OVER_TAU, 100*locationErrorEarth[2]);
}
else // GPS has gotten disconnected
{
yaw_drift_reset();
dcm_flags._.gps_history_valid = 0; // restart GPS history variables
IMUvelocityx.WW = IMUintegralAccelerationx.WW;
IMUvelocityy.WW = IMUintegralAccelerationy.WW;
IMUvelocityz.WW = IMUintegralAccelerationz.WW;
}
if (gps_nav_valid() && (dcm_flags._.reckon_req == 1))
{
// compute error indications and restart the dead reckoning clock to apply them
dcm_flags._.reckon_req = 0;
dead_reckon_clock = DR_PERIOD;
locationErrorEarth[0] = GPSlocation.x - IMUlocationx._.W1;
locationErrorEarth[1] = GPSlocation.y - IMUlocationy._.W1;
locationErrorEarth[2] = GPSlocation.z - IMUlocationz._.W1;
velocityErrorEarth[0] = GPSvelocity.x - IMUintegralAccelerationx._.W1;
velocityErrorEarth[1] = GPSvelocity.y - IMUintegralAccelerationy._.W1;
velocityErrorEarth[2] = GPSvelocity.z - IMUintegralAccelerationz._.W1;
}
}
else
{
IMUintegralAccelerationx.WW = 0;
IMUintegralAccelerationy.WW = 0;
IMUintegralAccelerationz.WW = 0;
IMUvelocityx.WW = 0;
IMUvelocityy.WW = 0;
IMUvelocityz.WW = 0;
IMUlocationx.WW = 0;
IMUlocationy.WW = 0;
IMUlocationz.WW = 0;
}
air_speed_x = IMUvelocityx._.W1 - estimatedWind[0];
air_speed_y = IMUvelocityy._.W1 - estimatedWind[1];
air_speed_z = IMUvelocityz._.W1 - estimatedWind[2];
accum.WW = ((__builtin_mulss(-IMUintegralAccelerationx._.W1, rmat[1])
+ __builtin_mulss( IMUintegralAccelerationy._.W1, rmat[4])) << 2);
forward_ground_speed = accum._.W1 ;
air_speed_3DIMU = vector3_mag(air_speed_x, air_speed_y, air_speed_z);
accum.WW = __builtin_mulsu(air_speed_x, 37877);
energy.WW = __builtin_mulss(accum._.W1, accum._.W1);
accum.WW = __builtin_mulsu(air_speed_y, 37877);
energy.WW += __builtin_mulss(accum._.W1, accum._.W1);
accum.WW = __builtin_mulsu(air_speed_z, 37877);
energy.WW += __builtin_mulss(accum._.W1, accum._.W1);
energy.WW += IMUlocationz.WW;
total_energy = energy._.W1;
}