-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMPU6050.cpp
135 lines (101 loc) · 3.39 KB
/
MPU6050.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
#include "MPU6050.h"
// Roughly based on Pico MPU6050 app note
MPU6050::MPU6050() {
// Make the I2C pins available to picotool
bi_decl(bi_2pins_with_func(MPU6050_SDA_PIN, MPU6050_SCL_PIN, GPIO_FUNC_I2C));
}
void MPU6050::reset() {
i2c_init(MPU6050_I2C_INSTANCE, 400 * 1000);
gpio_set_function(MPU6050_SDA_PIN, GPIO_FUNC_I2C);
gpio_set_function(MPU6050_SCL_PIN, GPIO_FUNC_I2C);
gpio_pull_up(MPU6050_SDA_PIN);
gpio_pull_up(MPU6050_SCL_PIN);
// This function brings the entire MPU6050 into a known state
this->ax = 0;
this->ay = 0;
this->az = 0;
this->gx = 0;
this->gy = 0;
this->gz = 0;
this->temp = NAN;
// Exit sleep mode
write_reg8(0x6B, 0x00);
// TODO: switch clock to gyroscope after a second or so
// Signal Path Reset
write_reg8(0x68, 0x07);
// Accelerometer Configuration
// Currently fixed at full-scale = +/- 2g
write_reg8(0x1C, 0x00);
// Filter config
// DLPF_CFG = 2
// -> Accelerometer Bandwidth=94Hz, Delay=3ms
// Lower bandwidth would probably also be fine, since we only
// measure tilt and the simulation has a lot of inertia as well
write_reg8(0x1A, 0x02);
}
void MPU6050::update() {
updateAccelerometer();
updateGyroscope();
updateTemperature();
}
void MPU6050::updateAccelerometer() {
// Read raw registers
int16_t ra_x = read_reg16(0x3B);
int16_t ra_y = read_reg16(0x3D);
int16_t ra_z = read_reg16(0x3F);
// TODO: implement calibration and self-test functionality
// Convert to floating-point to ease calculations
this->ax = ra_x*MPU6050_ACCEL_MULT_2G;
this->ay = ra_y*MPU6050_ACCEL_MULT_2G;
this->az = ra_z*MPU6050_ACCEL_MULT_2G;
// Calculate magnitude of vector
float mag = sqrt(ax*ax+ay*ay+az*az);
// Just in case we are in zero-g...
if (mag == 0) {
mag = 1;
}
this->axn = ax/mag;
this->ayn = ay/mag;
this->azn = az/mag;
}
void MPU6050::updateGyroscope() {
int16_t ga_x = read_reg16(0x43);
int16_t ga_y = read_reg16(0x45);
int16_t ga_z = read_reg16(0x47);
// TODO: implement calibration and self-test functionality
this->gx = ga_x*MPU6050_GYRO_MULT_250DPS;
this->gy = ga_y*MPU6050_GYRO_MULT_250DPS;
this->gz = ga_z*MPU6050_GYRO_MULT_250DPS;
}
void MPU6050::updateTemperature() {
int16_t traw = read_reg16(0x41);
// Calculation based on Pico SDK example
double temp_measured = (traw / 340.0) + 36.53;
if (isnan(temp)) {
temp = temp_measured;
} else {
temp = (temp * MPU6050_TEMP_SMOOTH) + (temp_measured * (1 - MPU6050_TEMP_SMOOTH));
}
}
void MPU6050::write_reg8(uint8_t reg, uint8_t data) {
uint8_t buf[2];
buf[0] = reg;
buf[1] = data;
i2c_write_blocking(MPU6050_I2C_INSTANCE, MPU6050_ADDR, buf, 2, false);
}
void MPU6050::write_reg16(uint8_t reg, uint16_t data) {
// Not yet implemented
// Probably just write both bytes at once, like read_reg16
}
uint8_t MPU6050::read_reg8(uint8_t reg) {
uint8_t out;
i2c_write_blocking(MPU6050_I2C_INSTANCE, MPU6050_ADDR, ®, 1, true);
i2c_read_blocking(MPU6050_I2C_INSTANCE, MPU6050_ADDR, &out, 1, false);
return out;
}
uint16_t MPU6050::read_reg16(uint8_t reg) {
uint8_t buf[2];
i2c_write_blocking(MPU6050_I2C_INSTANCE, MPU6050_ADDR, ®, 1, true);
i2c_read_blocking(MPU6050_I2C_INSTANCE, MPU6050_ADDR, buf, 2, false);
return buf[0] << 8 | buf[1];
}