MPU9255
MPU9255 Arduino Library
MPU9255.cpp
Go to the documentation of this file.
1 
6 // This file is a part of MPU9255 library.
7 // Copyright (c) 2017-2020 Krzysztof Adamkiewicz <kadamkiewicz835@gmail.com>
8 //
9 // Permission is hereby granted, free of charge, to any person obtaining a copy of
10 // this software and associated documentation files (the “Software”), to deal in the
11 // Software without restriction, including without limitation the rights to use, copy,
12 // modify, merge, publish, distribute, sublicense, and/or sell copies of the Software,
13 // and to permit persons to whom the Software is furnished to do so, subject to the
14 // following conditions: THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND,
15 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
18 // OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
19 
20 #include "MPU9255.h"
21 #include "Arduino.h"
22 
27 uint8_t MPU9255::init()
28 {
29  Wire.begin();//enable I2C interface
30  Hreset();//reset the chip
31  write(MPU_address,CONFIG, 0x03); // set DLPF_CFG to 11
32  write(MPU_address,SMPLRT_DIV, 0x00);// set prescaler sample rate to 0
33  write(MPU_address,GYRO_CONFIG, 0x01);// set gyro to 3.6 KHz bandwidth, and 0.11 ms using FCHOICE bits
34  write(MPU_address,INT_PIN_CFG, 0x02);// enable bypass
35  write(MAG_address, CNTL, 0x16);//set magnetometer to read in mode 2 and enable 16 bit measurements
36 
37  //read magnetometer sensitivity
38  mx_sensitivity = (((read(MAG_address, ASAX)-128)*0.5)/128)+1;
39  my_sensitivity = (((read(MAG_address, ASAY)-128)*0.5)/128)+1;
40  mz_sensitivity = (((read(MAG_address, ASAZ)-128)*0.5)/128)+1;
41 
42  //read factory gyroscope offset
43  GX_offset = ((int16_t)read(MPU_address, XG_OFFSET_H) << 8) | read(MPU_address, XG_OFFSET_L);
44  GY_offset = ((int16_t)read(MPU_address, YG_OFFSET_H) << 8) | read(MPU_address, YG_OFFSET_L);
45  GZ_offset = ((int16_t)read(MPU_address, XG_OFFSET_H) << 8) | read(MPU_address, ZG_OFFSET_L);
46 
47  //Based on http://www.digikey.com/en/pdf/i/invensense/mpu-hardware-offset-registers .
48  //read factory accelerometer offset
49 
50  //read the register values and save them as a 16 bit value
51  AX_offset = (read(MPU_address,XA_OFFSET_H)<<8) | (read(MPU_address,XA_OFFSET_L));
52  AY_offset = (read(MPU_address,YA_OFFSET_H)<<8) | (read(MPU_address,YA_OFFSET_L));
53  AZ_offset = (read(MPU_address,ZA_OFFSET_H)<<8) | (read(MPU_address,ZA_OFFSET_L));
54 
55  //shift offset values to the right to remove the LSB
56  AX_offset = AX_offset>>1;
57  AY_offset = AY_offset>>1;
58  AZ_offset = AZ_offset>>1;
59 
60  return (testIMU() || testMag());//return the output
61 }
62 
68 void MPU9255::set_gyro_offset(axis selected_axis, int16_t offset)
69 {
70  switch(selected_axis)
71  {
72  case X_axis:
73  offset = offset + GX_offset;//add offset to the factory offset
74  write(MPU_address,XG_OFFSET_L,(offset & 0xFF));//write low byte
75  write(MPU_address,XG_OFFSET_H,(offset>>8));//write high byte
76  break;
77 
78  case Y_axis:
79  offset = offset + GY_offset;
80  write(MPU_address,YG_OFFSET_L,(offset & 0xFF));
81  write(MPU_address,YG_OFFSET_H,(offset>>8));
82  break;
83 
84  case Z_axis:
85  offset = offset + GZ_offset;
86  write(MPU_address,ZG_OFFSET_L,(offset & 0xFF));
87  write(MPU_address,ZG_OFFSET_H,(offset>>8));
88  break;
89  }
90 }
91 
97 void MPU9255::set_acc_offset(axis selected_axis, int16_t offset)
98 {
99 
100  switch(selected_axis)
101  {
102  case X_axis:
103  offset = offset + AX_offset;//add offset to the factory offset
104  write(MPU_address,XA_OFFSET_L,(offset & 0xFF)<<1);//write low byte
105  write(MPU_address,XA_OFFSET_H,(offset>>7));//write high byte
106  break;
107 
108  case Y_axis:
109  offset = offset + AY_offset;
110  write(MPU_address,YA_OFFSET_L,(offset & 0xFF)<<1);
111  write(MPU_address,YA_OFFSET_H,(offset>>7));
112  break;
113 
114  case Z_axis:
115  offset = offset + AZ_offset;
116  write(MPU_address,ZA_OFFSET_L,(offset & 0xFF)<<1);
117  write(MPU_address,ZA_OFFSET_H,(offset>>7));
118  break;
119  }
120 }
121 
126 void MPU9255::set_acc_bandwidth(bandwidth selected_bandwidth)
127 {
128  switch(selected_bandwidth)
129  {
130  case acc_1113Hz:
131  write_OR(MPU_address,ACCEL_CONFIG_2,(1<<3));//set accel_fchoice_b to 1
132  break;
133 
134  case acc_460Hz:
135  //set accel_fchoice_b to 0 and A_DLPF_CFG to 0(000)
136  write_AND(MPU_address,ACCEL_CONFIG_2,~((1<<3)|(1<<2)|(1<<1)|(1<<0)));
137  break;
138 
139  case acc_184Hz:
140  write_AND(MPU_address,ACCEL_CONFIG_2,~(1<<3));//set accel_fchoice_b to 0
141  //set A_DLPF_CFG to 1(001)
142  write_AND(MPU_address,ACCEL_CONFIG_2,~((1<<1)|(1<<2)));
143  write_OR(MPU_address,ACCEL_CONFIG_2,(1<<0));
144  break;
145 
146  case acc_92Hz:
147  write_AND(MPU_address,ACCEL_CONFIG_2,~(1<<3));//set accel_fchoice_b to 0
148  //set A_DLPF_CFG to 2(010)
149  write_AND(MPU_address,ACCEL_CONFIG_2,~((1<<0)|(1<<2)));
150  write_OR(MPU_address,ACCEL_CONFIG_2,(1<<1));
151  break;
152 
153  case acc_41Hz:
154  write_AND(MPU_address,ACCEL_CONFIG_2,~(1<<3));//set accel_fchoice_b to 0
155  //set A_DLPF_CFG to 3(011)
156  write_AND(MPU_address,ACCEL_CONFIG_2,~(1<<2));
157  write_OR(MPU_address,ACCEL_CONFIG_2,(1<<0)|(1<<1));
158  break;
159 
160  case acc_20Hz:
161  write_AND(MPU_address,ACCEL_CONFIG_2,~(1<<3));//set accel_fchoice_b to 0
162  //set A_DLPF_CFG to 4(100)
163  write_AND(MPU_address,ACCEL_CONFIG_2,~((1<<0)|(1<<1)));
164  write_OR(MPU_address,ACCEL_CONFIG_2,(1<<2));
165  break;
166 
167  case acc_10Hz:
168  write_AND(MPU_address,ACCEL_CONFIG_2,~(1<<3));//set accel_fchoice_b to 0
169  //set A_DLPF_CFG to 5(101)
170  write_AND(MPU_address,ACCEL_CONFIG_2,~(1<<1));
171  write_OR(MPU_address,ACCEL_CONFIG_2,(1<<0)|(1<<2));
172  break;
173 
174  case acc_5Hz:
175  write_AND(MPU_address,ACCEL_CONFIG_2,~(1<<3));//set accel_fchoice_b to 0
176  //set A_DLPF_CFG to 6(110)
177  write_AND(MPU_address,ACCEL_CONFIG_2,~(1<<0));
178  write_OR(MPU_address,ACCEL_CONFIG_2,(1<<1)|(1<<2));
179  break;
180  }
181 }
182 
187 void MPU9255::set_gyro_bandwidth(bandwidth selected_bandwidth)
188 {
189  switch(selected_bandwidth)
190  {
191  case gyro_8800Hz:
192  write_OR(MPU_address,GYRO_CONFIG,(1<<0));//set Fchoice_b <0> to 1
193  break;
194 
195  case gyro_3600Hz:
196  write_AND(MPU_address,GYRO_CONFIG,~(1<<0));//set Fchoice_b <0> to 0
197  write_OR(MPU_address,GYRO_CONFIG,(1<<1));//set Fchoice_b <1> to 1
198  break;
199 
200  case gyro_250Hz:
201  write_AND(MPU_address,GYRO_CONFIG,~((1<<0)|(1<<1)));//set both Fchoice_b to 0
202  //write 0(000) to DLPF_CFG
203  write_AND(MPU_address,CONFIG,~((1<<0)|(1<<1)|(1<<2)));
204  break;
205 
206  case gyro_184Hz:
207  write_AND(MPU_address,GYRO_CONFIG,~((1<<0)|(1<<1)));//set both Fchoice_b to 0
208  //write 1(001) to DLPF_CFG
209  write_AND(MPU_address,CONFIG,~((1<<1)|(1<<2)));
210  write_OR(MPU_address,CONFIG,(1<<0));
211  break;
212 
213  case gyro_92Hz:
214  write_AND(MPU_address,GYRO_CONFIG,~((1<<0)|(1<<1)));//set both Fchoice_b to 0
215  //write 2(010) to DLPF_CFG
216  write_AND(MPU_address,CONFIG,~((1<<2)|(1<<0)));
217  write_OR(MPU_address,CONFIG,(1<<1));
218  break;
219 
220  case gyro_41Hz:
221  write_AND(MPU_address,GYRO_CONFIG,~((1<<0)|(1<<1)));//set both Fchoice_b to 0
222  //write 3(011) to DLPF_CFG
223  write_AND(MPU_address,CONFIG,~(1<<2));
224  write_OR(MPU_address,CONFIG,(1<<0)|(1<<1));
225  break;
226 
227  case gyro_20Hz:
228  write_AND(MPU_address,GYRO_CONFIG,~((1<<0)|(1<<1)));//set both Fchoice_b to 0
229  //write 4(100) to DLPF_CFG
230  write_AND(MPU_address,CONFIG,~((1<<1)|(1<<0)));
231  write_OR(MPU_address,CONFIG,(1<<2));
232  break;
233 
234  case gyro_10Hz:
235  write_AND(MPU_address,GYRO_CONFIG,~((1<<0)|(1<<1)));//set both Fchoice_b to 0
236  //write 5(101) to DLPF_CFG
237  write_AND(MPU_address,CONFIG,~(1<<1));
238  write_OR(MPU_address,CONFIG,(1<<2)|(1<<0));
239  break;
240 
241  case gyro_5Hz:
242  write_AND(MPU_address,GYRO_CONFIG,~((1<<0)|(1<<1)));//set both Fchoice_b to 0
243  //write 6(110) to DLPF_CFG
244  write_AND(MPU_address,CONFIG,~(1<<0));
245  write_OR(MPU_address,CONFIG,(1<<1)|(1<<2));
246  break;
247  }
248 }
249 
258 uint8_t MPU9255::getScale(uint8_t current_state, scales selected_scale)
259 {
260  if(selected_scale == scale_2g || selected_scale == scale_250dps)
261  {
262  current_state &= ~((1<<3)|(1<<4));
263  }
264 
265  if(selected_scale == scale_4g || selected_scale == scale_500dps)
266  {
267  current_state &= ~(1<<4);
268  current_state |= (1<<3);
269  }
270 
271  if(selected_scale == scale_8g || selected_scale == scale_1000dps)
272  {
273  current_state &= ~(1<<3);
274  current_state |= (1<<4);
275  }
276 
277  if(selected_scale == scale_16g || selected_scale == scale_2000dps)
278  {
279  current_state |= (1<<4)|(1<<3);
280  }
281 
282  return current_state;
283 }
284 
289 void MPU9255::set_acc_scale(scales selected_scale)
290 {
291  uint8_t val = read(MPU_address,ACCEL_CONFIG);//read old register value
292  val = getScale(val,selected_scale);//get new register value
293  write(MPU_address,ACCEL_CONFIG,val);//commit changes
294 }
295 
300 void MPU9255::set_gyro_scale(scales selected_scale)
301 {
302  uint8_t val=read(MPU_address,GYRO_CONFIG);
303  val = getScale(val,selected_scale);
304  write(MPU_address,GYRO_CONFIG,val);
305 }
306 
312 {
313  if(read(MPU_address,WHO_AM_I)==0xFF)
314  {
315  return 1;
316  }
317  return 0;
318 }
319 
325 {
326  if(read(MAG_address,MAG_ID)==0xFF)
327  {
328  return 1;
329  }
330  return 0;
331 }
void set_acc_scale(scales selected_scale)
Set accelerometer scale.
Definition: MPU9255.cpp:289
void set_gyro_bandwidth(bandwidth selected_bandwidth)
Set gyroscope bandwidth.
Definition: MPU9255.cpp:187
uint8_t testMag()
Test if magnetometer is working.
Definition: MPU9255.cpp:324
uint8_t init()
Initialise MPU9255 module.
Definition: MPU9255.cpp:27
void set_acc_bandwidth(bandwidth selected_bandwidth)
Set accelerometer bandwidth.
Definition: MPU9255.cpp:126
void set_acc_offset(axis selected_axis, int16_t offset)
Set accelerometer offset.
Definition: MPU9255.cpp:97
Main header of the library.
void set_gyro_scale(scales selected_scale)
Set gyroscope scale.
Definition: MPU9255.cpp:300
uint8_t testIMU()
Test if IMU (gyroscope and accelerometer) is working.
Definition: MPU9255.cpp:311
void set_gyro_offset(axis selected_axis, int16_t offset)
Set gyroscope offset.
Definition: MPU9255.cpp:68