-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmpu.ino
121 lines (115 loc) · 6.27 KB
/
mpu.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
// start I2C communication and send commands to set up the MPU6050.
// A command is set by starting a transmission, writing a byte (written here in hexadecimal) to signal what register should be changed,
// and then sending a new register value
void setupMPU6050() {
Wire.begin(); // setup mpu6050. reference for the mpu6050's registers https://www.invensense.com/wp-content/uploads/2015/02/MPU-6000-Register-Map1.pdf
Wire.setClock(400000L); // send data at a faster clock speed
Wire.beginTransmission(0x68);
Wire.write(0x6B);
Wire.write(0); // wakeup
Wire.endTransmission(true);
Wire.beginTransmission(0x68);
Wire.write(0x1B);
Wire.write(0x10); // gyro range //18=2000//10=1000
Wire.endTransmission(true);
Wire.beginTransmission(0x68);
Wire.write(0x19);
Wire.write(0x02); // clock divider
Wire.endTransmission(true);
Wire.beginTransmission(0x68);
Wire.write(0x1A);
Wire.write(0x00); // buffering
Wire.endTransmission(true); // end setup mpu6050
}
void readMPU6050() {
Wire.beginTransmission(0x68);
Wire.write(0x3B); // location of first byte of data
Wire.endTransmission(false);
Wire.requestFrom(0x68, 14, true); // ask for accel and gyro data bytes
accelerationX = Wire.read() << 8 | Wire.read(); // read two bytes and put them together into a sixteen bit integer value
accelerationY = Wire.read() << 8 | Wire.read();
accelerationZ = Wire.read() << 8 | Wire.read();
Wire.read(); Wire.read(); // throw away temperature, it's annoying they put it in the middle here
rotationX = Wire.read() << 8 | Wire.read();
rotationY = Wire.read() << 8 | Wire.read();
rotationZ = Wire.read() << 8 | Wire.read();
rotationDPS_X = (rotationX - rotationOffsetX) * 1000.00 / 32766; // zero gyro with offset values recorded on startup and convert to degrees per second
rotationDPS_Y = (rotationY - rotationOffsetY) * 1000.00 / 32766;
rotationDPS_Z = (rotationZ - rotationOffsetZ) * 1000.00 / 32766;
if (micros() < lastCalcedMPU6050) { // try to handle micros' long overflow in a harmless way
lastCalcedMPU6050 = micros() - 10000;
}
pitch3 = pitch2;
pitch2 = pitch1;
// complementary filter combines gyro and accelerometer tilt data in a way that takes advantage of short term accuracy of the gyro and long term accuracy of the accelerometer
pitch1 = COMPLEMENTARY_FILTER_CONSTANT * ((pitch1) + rotationDPS_X * (micros() - lastCalcedMPU6050) / 1000000.000) // add rotation rate as measured by the gyro to current pitch - valid in short term
+ (1 - COMPLEMENTARY_FILTER_CONSTANT) * (degrees(atan2(accelerationY, accelerationZ)) - pitchOffset * pitchInverter); // in the long term drift towards the angle of gravity measured by the accelerometer
pitch = pitchInverter * (pitch1 + pitch2 + pitch3) / 3; //averages the last 3 pitch readings and inverts if necessary
if (robotEnabled) { // only adjust pitchOffset when the robot is enabled
pitchOffset = (pitch + pitchOffset) * (1 - PITCH_OFFSET_CHANGE) + pitchOffset * (PITCH_OFFSET_CHANGE); // slowly move pitchOffset towards the current pitch value, the overall average pitch value should be close to the balance point
}
lastCalcedMPU6050 = micros(); // record time of last calculation so we know next time how much time has passed (how much time to integrate rotation rate for)
}
void zeroMPU6050() { // find how much offset each gyro axis has to zero out drift. should be run on startup (when robot is still)
do {
Wire.beginTransmission(0x68);
Wire.write(0x3B);
Wire.endTransmission(false);
Wire.requestFrom(0x68, 14, true);
accelerationX = Wire.read() << 8 | Wire.read(); // same reading code as earlier
accelerationY = Wire.read() << 8 | Wire.read();
accelerationZ = Wire.read() << 8 | Wire.read();
Wire.read(); Wire.read(); // throw away temperature
int16_t lastrotationX = Wire.read() << 8 | Wire.read();
int16_t lastrotationY = Wire.read() << 8 | Wire.read();
int16_t lastrotationZ = Wire.read() << 8 | Wire.read();
rotationOffsetX = 0;
rotationOffsetY = 0;
rotationOffsetZ = 0;
for (int i = 0; i < movementMeasurements; i++) {
Wire.beginTransmission(0x68);
Wire.write(0x3B);
Wire.endTransmission(false);
Wire.requestFrom(0x68, 14, true);
accelerationX = Wire.read() << 8 | Wire.read(); // same reading code as earlier
accelerationY = Wire.read() << 8 | Wire.read();
accelerationZ = Wire.read() << 8 | Wire.read();
Wire.read(); Wire.read(); // throw away temperature
rotationX = Wire.read() << 8 | Wire.read();
rotationY = Wire.read() << 8 | Wire.read();
rotationZ = Wire.read() << 8 | Wire.read();
rotationOffsetX += abs(rotationX - lastrotationX);
rotationOffsetY += abs(rotationY - lastrotationY);
rotationOffsetZ += abs(rotationZ - lastrotationZ);
lastrotationX = rotationX;
lastrotationY = rotationY;
lastrotationZ = rotationZ;
digitalWrite(LED_BUILTIN, i % 2);
delay(15);
}
} while (abs(rotationOffsetX) > movementThreshold * movementMeasurements || abs(rotationOffsetY) > movementThreshold * movementMeasurements || abs(rotationOffsetZ) > movementThreshold * movementMeasurements);
rotationOffsetX = 0;
rotationOffsetY = 0;
rotationOffsetZ = 0;
digitalWrite(LED_BUILTIN, LOW);
for (int i = 0; i < 50; i++) { // run the following code 50 times so we can get many measurements to average into an offset value
Wire.beginTransmission(0x68);
Wire.write(0x3B);
Wire.endTransmission(false);
Wire.requestFrom(0x68, 14, true);
accelerationX = Wire.read() << 8 | Wire.read(); // same reading code as earlier
accelerationY = Wire.read() << 8 | Wire.read();
accelerationZ = Wire.read() << 8 | Wire.read();
Wire.read(); Wire.read(); // throw away temperature
rotationX = Wire.read() << 8 | Wire.read();
rotationY = Wire.read() << 8 | Wire.read();
rotationZ = Wire.read() << 8 | Wire.read();
rotationOffsetX += rotationX; // add all the reads together
rotationOffsetY += rotationY;
rotationOffsetZ += rotationZ;
delay(10 + i / 5); // add some time between reads, changing the delay each time a bit to be less likely to be thrown by a periodic oscillation
}
rotationOffsetX /= 50; // devide by the number of reads that were taken to get an average value
rotationOffsetY /= 50;
rotationOffsetZ /= 50;
}