forked from DavidEGrayson/minimu9-ahrs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMinIMU9.cpp
91 lines (78 loc) · 2.57 KB
/
MinIMU9.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
#include "vector.h"
#include "MinIMU9.h"
#include "exceptions.h"
#include <stdio.h>
#include <iostream>
#include <fstream>
#include <wordexp.h>
MinIMU9::MinIMU9(const char * i2cDeviceName) :
compass(i2cDeviceName), gyro(i2cDeviceName)
{
}
void MinIMU9::enable()
{
compass.enable();
gyro.enable();
}
void MinIMU9::loadCalibration()
{
wordexp_t expansion_result;
wordexp("~/.minimu9-ahrs-cal", &expansion_result, 0);
std::ifstream file(expansion_result.we_wordv[0]);
if (file.fail())
{
throw posix_error("Failed to open calibration file ~/.minimu9-ahrs-cal.");
}
file >> mag_min(0) >> mag_max(0) >> mag_min(1) >> mag_max(1) >> mag_min(2) >> mag_max(2);
if (file.fail() || file.bad())
{
throw std::runtime_error("Failed to parse calibration file ~/.minimu9-ahrs-cal.");
}
}
void MinIMU9::measureOffsets()
{
// LSM303 accelerometer: 8 g sensitivity. 3.8 mg/digit; 1 g = 256.
// TODO: unify this with the other place in the code where we scale accelerometer readings.
gyro_offset = vector::Zero();
const int sampleCount = 32;
for(int i = 0; i < sampleCount; i++)
{
gyro.read();
gyro_offset += vector_from_ints(&gyro.g);
usleep(20*1000);
}
gyro_offset /= sampleCount;
}
vector MinIMU9::readMag()
{
compass.readMag();
IMU::raw_m = int_vector_from_ints(&compass.m);
vector v;
v(0) = (float)(compass.m[0] - mag_min(0)) / (mag_max(0) - mag_min(0)) * 2 - 1;
v(1) = (float)(compass.m[1] - mag_min(1)) / (mag_max(1) - mag_min(1)) * 2 - 1;
v(2) = (float)(compass.m[2] - mag_min(2)) / (mag_max(2) - mag_min(2)) * 2 - 1;
return v;
}
vector MinIMU9::readAcc()
{
// Info about linear acceleration sensitivity from datasheets:
// LSM303DLM: at FS = 8 g, 3.9 mg/digit (12-bit reading)
// LSM303DLHC: at FS = 8 g, 4 mg/digit (12-bit reading probably an approximation)
// LSM303DLH: at FS = 8 g, 3.9 mg/digit (12-bit reading)
// LSM303D: at FS = 8 g, 0.244 mg/LSB (16-bit reading)
const float accel_scale = 0.000244;
compass.readAcc();
IMU::raw_a = int_vector_from_ints(&compass.a);
return vector_from_ints(&compass.a) * accel_scale;
}
vector MinIMU9::readGyro()
{
// Info about sensitivity from datasheets:
// L3G4200D: at FS = 2000 dps, 70 mdps/digit
// L3GD20: at FS = 2000 dps, 70 mdps/digit
// L3GD20H: at FS = 2000 dps, 70 mdps/digit
const float gyro_scale = 0.07 * 3.14159265 / 180;
gyro.read();
IMU::raw_g = int_vector_from_ints(&gyro.g);
return ( vector_from_ints(&gyro.g) - gyro_offset ) * gyro_scale;
}