MPU6050_9Axis_MotionApps41.h 47 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852
  1. // I2Cdev library collection - MPU6050 I2C device class, 9-axis MotionApps 4.1 implementation
  2. // Based on InvenSense MPU-6050 register map document rev. 2.0, 5/19/2011 (RM-MPU-6000A-00)
  3. // 6/18/2012 by Jeff Rowberg <jeff@rowberg.net>
  4. // Updates should (hopefully) always be available at https://github.com/jrowberg/i2cdevlib
  5. //
  6. // Changelog:
  7. // ... - ongoing debug release
  8. /* ============================================
  9. I2Cdev device library code is placed under the MIT license
  10. Copyright (c) 2012 Jeff Rowberg
  11. Permission is hereby granted, free of charge, to any person obtaining a copy
  12. of this software and associated documentation files (the "Software"), to deal
  13. in the Software without restriction, including without limitation the rights
  14. to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  15. copies of the Software, and to permit persons to whom the Software is
  16. furnished to do so, subject to the following conditions:
  17. The above copyright notice and this permission notice shall be included in
  18. all copies or substantial portions of the Software.
  19. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  20. IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  21. FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  22. AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  23. LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  24. OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  25. THE SOFTWARE.
  26. ===============================================
  27. */
  28. #ifndef _MPU6050_9AXIS_MOTIONAPPS41_H_
  29. #define _MPU6050_9AXIS_MOTIONAPPS41_H_
  30. #include "I2Cdev.h"
  31. #include "helper_3dmath.h"
  32. // MotionApps 4.1 DMP implementation, built using the MPU-9150 "MotionFit" board
  33. #define MPU6050_INCLUDE_DMP_MOTIONAPPS41
  34. #include "MPU6050.h"
  35. // Tom Carpenter's conditional PROGMEM code
  36. // http://forum.arduino.cc/index.php?topic=129407.0
  37. #ifndef __arm__
  38. #include <avr/pgmspace.h>
  39. #else
  40. // Teensy 3.0 library conditional PROGMEM code from Paul Stoffregen
  41. #ifndef __PGMSPACE_H_
  42. #define __PGMSPACE_H_ 1
  43. #include <inttypes.h>
  44. #define PROGMEM
  45. #define PGM_P const char *
  46. #define PSTR(str) (str)
  47. #define F(x) x
  48. typedef void prog_void;
  49. typedef char prog_char;
  50. typedef unsigned char prog_uchar;
  51. typedef int8_t prog_int8_t;
  52. typedef uint8_t prog_uint8_t;
  53. typedef int16_t prog_int16_t;
  54. typedef uint16_t prog_uint16_t;
  55. typedef int32_t prog_int32_t;
  56. typedef uint32_t prog_uint32_t;
  57. #define strcpy_P(dest, src) strcpy((dest), (src))
  58. #define strcat_P(dest, src) strcat((dest), (src))
  59. #define strcmp_P(a, b) strcmp((a), (b))
  60. #define pgm_read_byte(addr) (*(const unsigned char *)(addr))
  61. #define pgm_read_word(addr) (*(const unsigned short *)(addr))
  62. #define pgm_read_dword(addr) (*(const unsigned long *)(addr))
  63. #define pgm_read_float(addr) (*(const float *)(addr))
  64. #define pgm_read_byte_near(addr) pgm_read_byte(addr)
  65. #define pgm_read_word_near(addr) pgm_read_word(addr)
  66. #define pgm_read_dword_near(addr) pgm_read_dword(addr)
  67. #define pgm_read_float_near(addr) pgm_read_float(addr)
  68. #define pgm_read_byte_far(addr) pgm_read_byte(addr)
  69. #define pgm_read_word_far(addr) pgm_read_word(addr)
  70. #define pgm_read_dword_far(addr) pgm_read_dword(addr)
  71. #define pgm_read_float_far(addr) pgm_read_float(addr)
  72. #endif
  73. #endif
  74. // NOTE! Enabling DEBUG adds about 3.3kB to the flash program size.
  75. // Debug output is now working even on ATMega328P MCUs (e.g. Arduino Uno)
  76. // after moving string constants to flash memory storage using the F()
  77. // compiler macro (Arduino IDE 1.0+ required).
  78. //#define DEBUG
  79. #ifdef DEBUG
  80. #define DEBUG_PRINT(x) Serial.print(x)
  81. #define DEBUG_PRINTF(x, y) Serial.print(x, y)
  82. #define DEBUG_PRINTLN(x) Serial.println(x)
  83. #define DEBUG_PRINTLNF(x, y) Serial.println(x, y)
  84. #else
  85. #define DEBUG_PRINT(x)
  86. #define DEBUG_PRINTF(x, y)
  87. #define DEBUG_PRINTLN(x)
  88. #define DEBUG_PRINTLNF(x, y)
  89. #endif
  90. #define MPU6050_DMP_CODE_SIZE 1962 // dmpMemory[]
  91. #define MPU6050_DMP_CONFIG_SIZE 232 // dmpConfig[]
  92. #define MPU6050_DMP_UPDATES_SIZE 140 // dmpUpdates[]
  93. /* ================================================================================================ *
  94. | Default MotionApps v4.1 48-byte FIFO packet structure: |
  95. | |
  96. | [QUAT W][ ][QUAT X][ ][QUAT Y][ ][QUAT Z][ ][GYRO X][ ][GYRO Y][ ] |
  97. | 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
  98. | |
  99. | [GYRO Z][ ][MAG X ][MAG Y ][MAG Z ][ACC X ][ ][ACC Y ][ ][ACC Z ][ ][ ] |
  100. | 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 |
  101. * ================================================================================================ */
  102. // this block of memory gets written to the MPU on start-up, and it seems
  103. // to be volatile memory, so it has to be done each time (it only takes ~1
  104. // second though)
  105. const prog_uchar dmpMemory[MPU6050_DMP_CODE_SIZE] PROGMEM = {
  106. // bank 0, 256 bytes
  107. 0xFB, 0x00, 0x00, 0x3E, 0x00, 0x0B, 0x00, 0x36, 0x00, 0x01, 0x00, 0x02, 0x00, 0x03, 0x00, 0x00,
  108. 0x00, 0x65, 0x00, 0x54, 0xFF, 0xEF, 0x00, 0x00, 0xFA, 0x80, 0x00, 0x0B, 0x12, 0x82, 0x00, 0x01,
  109. 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  110. 0x00, 0x28, 0x00, 0x00, 0xFF, 0xFF, 0x45, 0x81, 0xFF, 0xFF, 0xFA, 0x72, 0x00, 0x00, 0x00, 0x00,
  111. 0x00, 0x00, 0x03, 0xE8, 0x00, 0x00, 0x00, 0x01, 0x00, 0x01, 0x7F, 0xFF, 0xFF, 0xFE, 0x80, 0x01,
  112. 0x00, 0x1B, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  113. 0x00, 0x3E, 0x03, 0x30, 0x40, 0x00, 0x00, 0x00, 0x02, 0xCA, 0xE3, 0x09, 0x3E, 0x80, 0x00, 0x00,
  114. 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, 0x60, 0x00, 0x00, 0x00,
  115. 0x41, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x0B, 0x2A, 0x00, 0x00, 0x16, 0x55, 0x00, 0x00, 0x21, 0x82,
  116. 0xFD, 0x87, 0x26, 0x50, 0xFD, 0x80, 0x00, 0x00, 0x00, 0x1F, 0x00, 0x00, 0x00, 0x05, 0x80, 0x00,
  117. 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x03, 0x00, 0x00,
  118. 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, 0x6F, 0x00, 0x02, 0x65, 0x32, 0x00, 0x00, 0x5E, 0xC0,
  119. 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  120. 0xFB, 0x8C, 0x6F, 0x5D, 0xFD, 0x5D, 0x08, 0xD9, 0x00, 0x7C, 0x73, 0x3B, 0x00, 0x6C, 0x12, 0xCC,
  121. 0x32, 0x00, 0x13, 0x9D, 0x32, 0x00, 0xD0, 0xD6, 0x32, 0x00, 0x08, 0x00, 0x40, 0x00, 0x01, 0xF4,
  122. 0xFF, 0xE6, 0x80, 0x79, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0xD0, 0xD6, 0x00, 0x00, 0x27, 0x10,
  123. // bank 1, 256 bytes
  124. 0xFB, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  125. 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00,
  126. 0x00, 0x00, 0xFA, 0x36, 0xFF, 0xBC, 0x30, 0x8E, 0x00, 0x05, 0xFB, 0xF0, 0xFF, 0xD9, 0x5B, 0xC8,
  127. 0xFF, 0xD0, 0x9A, 0xBE, 0x00, 0x00, 0x10, 0xA9, 0xFF, 0xF4, 0x1E, 0xB2, 0x00, 0xCE, 0xBB, 0xF7,
  128. 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x02, 0x02, 0x00, 0x00, 0x0C,
  129. 0xFF, 0xC2, 0x80, 0x00, 0x00, 0x01, 0x80, 0x00, 0x00, 0xCF, 0x80, 0x00, 0x40, 0x00, 0x00, 0x00,
  130. 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00, 0x00, 0x14,
  131. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  132. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  133. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  134. 0x00, 0x00, 0x00, 0x00, 0x03, 0x3F, 0x68, 0xB6, 0x79, 0x35, 0x28, 0xBC, 0xC6, 0x7E, 0xD1, 0x6C,
  135. 0x80, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0xB2, 0x6A, 0x00, 0x00, 0x00, 0x00,
  136. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3F, 0xF0, 0x00, 0x00, 0x00, 0x30,
  137. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  138. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  139. 0x00, 0x00, 0x25, 0x4D, 0x00, 0x2F, 0x70, 0x6D, 0x00, 0x00, 0x05, 0xAE, 0x00, 0x0C, 0x02, 0xD0,
  140. // bank 2, 256 bytes
  141. 0x00, 0x00, 0x00, 0x00, 0x00, 0x65, 0x00, 0x54, 0xFF, 0xEF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  142. 0x00, 0x00, 0x01, 0x00, 0x00, 0x44, 0x00, 0x00, 0x00, 0x00, 0x0C, 0x00, 0x00, 0x00, 0x01, 0x00,
  143. 0x00, 0x00, 0x00, 0x00, 0x00, 0x65, 0x00, 0x00, 0x00, 0x54, 0x00, 0x00, 0xFF, 0xEF, 0x00, 0x00,
  144. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  145. 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  146. 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  147. 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  148. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  149. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  150. 0x00, 0x1B, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  151. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  152. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00,
  153. 0x00, 0x1B, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  154. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  155. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x47, 0x78, 0xA2,
  156. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  157. // bank 3, 256 bytes
  158. 0xD8, 0xDC, 0xF4, 0xD8, 0xB9, 0xAB, 0xF3, 0xF8, 0xFA, 0xF1, 0xBA, 0xA2, 0xDE, 0xB2, 0xB8, 0xB4,
  159. 0xA8, 0x81, 0x98, 0xF7, 0x4A, 0x90, 0x7F, 0x91, 0x6A, 0xF3, 0xF9, 0xDB, 0xA8, 0xF9, 0xB0, 0xBA,
  160. 0xA0, 0x80, 0xF2, 0xCE, 0x81, 0xF3, 0xC2, 0xF1, 0xC1, 0xF2, 0xC3, 0xF3, 0xCC, 0xA2, 0xB2, 0x80,
  161. 0xF1, 0xC6, 0xD8, 0x80, 0xBA, 0xA7, 0xDF, 0xDF, 0xDF, 0xF2, 0xA7, 0xC3, 0xCB, 0xC5, 0xB6, 0xF0,
  162. 0x87, 0xA2, 0x94, 0x24, 0x48, 0x70, 0x3C, 0x95, 0x40, 0x68, 0x34, 0x58, 0x9B, 0x78, 0xA2, 0xF1,
  163. 0x83, 0x92, 0x2D, 0x55, 0x7D, 0xD8, 0xB1, 0xB4, 0xB8, 0xA1, 0xD0, 0x91, 0x80, 0xF2, 0x70, 0xF3,
  164. 0x70, 0xF2, 0x7C, 0x80, 0xA8, 0xF1, 0x01, 0xB0, 0x98, 0x87, 0xD9, 0x43, 0xD8, 0x86, 0xC9, 0x88,
  165. 0xBA, 0xA1, 0xF2, 0x0E, 0xB8, 0x97, 0x80, 0xF1, 0xA9, 0xDF, 0xDF, 0xDF, 0xAA, 0xDF, 0xDF, 0xDF,
  166. 0xF2, 0xAA, 0xC5, 0xCD, 0xC7, 0xA9, 0x0C, 0xC9, 0x2C, 0x97, 0x97, 0x97, 0x97, 0xF1, 0xA9, 0x89,
  167. 0x26, 0x46, 0x66, 0xB0, 0xB4, 0xBA, 0x80, 0xAC, 0xDE, 0xF2, 0xCA, 0xF1, 0xB2, 0x8C, 0x02, 0xA9,
  168. 0xB6, 0x98, 0x00, 0x89, 0x0E, 0x16, 0x1E, 0xB8, 0xA9, 0xB4, 0x99, 0x2C, 0x54, 0x7C, 0xB0, 0x8A,
  169. 0xA8, 0x96, 0x36, 0x56, 0x76, 0xF1, 0xB9, 0xAF, 0xB4, 0xB0, 0x83, 0xC0, 0xB8, 0xA8, 0x97, 0x11,
  170. 0xB1, 0x8F, 0x98, 0xB9, 0xAF, 0xF0, 0x24, 0x08, 0x44, 0x10, 0x64, 0x18, 0xF1, 0xA3, 0x29, 0x55,
  171. 0x7D, 0xAF, 0x83, 0xB5, 0x93, 0xF0, 0x00, 0x28, 0x50, 0xF5, 0xBA, 0xAD, 0x8F, 0x9F, 0x28, 0x54,
  172. 0x7C, 0xB9, 0xF1, 0xA3, 0x86, 0x9F, 0x61, 0xA6, 0xDA, 0xDE, 0xDF, 0xDB, 0xB2, 0xB6, 0x8E, 0x9D,
  173. 0xAE, 0xF5, 0x60, 0x68, 0x70, 0xB1, 0xB5, 0xF1, 0xDA, 0xA6, 0xDF, 0xD9, 0xA6, 0xFA, 0xA3, 0x86,
  174. // bank 4, 256 bytes
  175. 0x96, 0xDB, 0x31, 0xA6, 0xD9, 0xF8, 0xDF, 0xBA, 0xA6, 0x8F, 0xC2, 0xC5, 0xC7, 0xB2, 0x8C, 0xC1,
  176. 0xB8, 0xA2, 0xDF, 0xDF, 0xDF, 0xA3, 0xDF, 0xDF, 0xDF, 0xD8, 0xD8, 0xF1, 0xB8, 0xA8, 0xB2, 0x86,
  177. 0xB4, 0x98, 0x0D, 0x35, 0x5D, 0xB8, 0xAA, 0x98, 0xB0, 0x87, 0x2D, 0x35, 0x3D, 0xB2, 0xB6, 0xBA,
  178. 0xAF, 0x8C, 0x96, 0x19, 0x8F, 0x9F, 0xA7, 0x0E, 0x16, 0x1E, 0xB4, 0x9A, 0xB8, 0xAA, 0x87, 0x2C,
  179. 0x54, 0x7C, 0xB9, 0xA3, 0xDE, 0xDF, 0xDF, 0xA3, 0xB1, 0x80, 0xF2, 0xC4, 0xCD, 0xC9, 0xF1, 0xB8,
  180. 0xA9, 0xB4, 0x99, 0x83, 0x0D, 0x35, 0x5D, 0x89, 0xB9, 0xA3, 0x2D, 0x55, 0x7D, 0xB5, 0x93, 0xA3,
  181. 0x0E, 0x16, 0x1E, 0xA9, 0x2C, 0x54, 0x7C, 0xB8, 0xB4, 0xB0, 0xF1, 0x97, 0x83, 0xA8, 0x11, 0x84,
  182. 0xA5, 0x09, 0x98, 0xA3, 0x83, 0xF0, 0xDA, 0x24, 0x08, 0x44, 0x10, 0x64, 0x18, 0xD8, 0xF1, 0xA5,
  183. 0x29, 0x55, 0x7D, 0xA5, 0x85, 0x95, 0x02, 0x1A, 0x2E, 0x3A, 0x56, 0x5A, 0x40, 0x48, 0xF9, 0xF3,
  184. 0xA3, 0xD9, 0xF8, 0xF0, 0x98, 0x83, 0x24, 0x08, 0x44, 0x10, 0x64, 0x18, 0x97, 0x82, 0xA8, 0xF1,
  185. 0x11, 0xF0, 0x98, 0xA2, 0x24, 0x08, 0x44, 0x10, 0x64, 0x18, 0xDA, 0xF3, 0xDE, 0xD8, 0x83, 0xA5,
  186. 0x94, 0x01, 0xD9, 0xA3, 0x02, 0xF1, 0xA2, 0xC3, 0xC5, 0xC7, 0xD8, 0xF1, 0x84, 0x92, 0xA2, 0x4D,
  187. 0xDA, 0x2A, 0xD8, 0x48, 0x69, 0xD9, 0x2A, 0xD8, 0x68, 0x55, 0xDA, 0x32, 0xD8, 0x50, 0x71, 0xD9,
  188. 0x32, 0xD8, 0x70, 0x5D, 0xDA, 0x3A, 0xD8, 0x58, 0x79, 0xD9, 0x3A, 0xD8, 0x78, 0x93, 0xA3, 0x4D,
  189. 0xDA, 0x2A, 0xD8, 0x48, 0x69, 0xD9, 0x2A, 0xD8, 0x68, 0x55, 0xDA, 0x32, 0xD8, 0x50, 0x71, 0xD9,
  190. 0x32, 0xD8, 0x70, 0x5D, 0xDA, 0x3A, 0xD8, 0x58, 0x79, 0xD9, 0x3A, 0xD8, 0x78, 0xA8, 0x8A, 0x9A,
  191. // bank 5, 256 bytes
  192. 0xF0, 0x28, 0x50, 0x78, 0x9E, 0xF3, 0x88, 0x18, 0xF1, 0x9F, 0x1D, 0x98, 0xA8, 0xD9, 0x08, 0xD8,
  193. 0xC8, 0x9F, 0x12, 0x9E, 0xF3, 0x15, 0xA8, 0xDA, 0x12, 0x10, 0xD8, 0xF1, 0xAF, 0xC8, 0x97, 0x87,
  194. 0x34, 0xB5, 0xB9, 0x94, 0xA4, 0x21, 0xF3, 0xD9, 0x22, 0xD8, 0xF2, 0x2D, 0xF3, 0xD9, 0x2A, 0xD8,
  195. 0xF2, 0x35, 0xF3, 0xD9, 0x32, 0xD8, 0x81, 0xA4, 0x60, 0x60, 0x61, 0xD9, 0x61, 0xD8, 0x6C, 0x68,
  196. 0x69, 0xD9, 0x69, 0xD8, 0x74, 0x70, 0x71, 0xD9, 0x71, 0xD8, 0xB1, 0xA3, 0x84, 0x19, 0x3D, 0x5D,
  197. 0xA3, 0x83, 0x1A, 0x3E, 0x5E, 0x93, 0x10, 0x30, 0x81, 0x10, 0x11, 0xB8, 0xB0, 0xAF, 0x8F, 0x94,
  198. 0xF2, 0xDA, 0x3E, 0xD8, 0xB4, 0x9A, 0xA8, 0x87, 0x29, 0xDA, 0xF8, 0xD8, 0x87, 0x9A, 0x35, 0xDA,
  199. 0xF8, 0xD8, 0x87, 0x9A, 0x3D, 0xDA, 0xF8, 0xD8, 0xB1, 0xB9, 0xA4, 0x98, 0x85, 0x02, 0x2E, 0x56,
  200. 0xA5, 0x81, 0x00, 0x0C, 0x14, 0xA3, 0x97, 0xB0, 0x8A, 0xF1, 0x2D, 0xD9, 0x28, 0xD8, 0x4D, 0xD9,
  201. 0x48, 0xD8, 0x6D, 0xD9, 0x68, 0xD8, 0xB1, 0x84, 0x0D, 0xDA, 0x0E, 0xD8, 0xA3, 0x29, 0x83, 0xDA,
  202. 0x2C, 0x0E, 0xD8, 0xA3, 0x84, 0x49, 0x83, 0xDA, 0x2C, 0x4C, 0x0E, 0xD8, 0xB8, 0xB0, 0x97, 0x86,
  203. 0xA8, 0x31, 0x9B, 0x06, 0x99, 0x07, 0xAB, 0x97, 0x28, 0x88, 0x9B, 0xF0, 0x0C, 0x20, 0x14, 0x40,
  204. 0xB9, 0xA3, 0x8A, 0xC3, 0xC5, 0xC7, 0x9A, 0xA3, 0x28, 0x50, 0x78, 0xF1, 0xB5, 0x93, 0x01, 0xD9,
  205. 0xDF, 0xDF, 0xDF, 0xD8, 0xB8, 0xB4, 0xA8, 0x8C, 0x9C, 0xF0, 0x04, 0x28, 0x51, 0x79, 0x1D, 0x30,
  206. 0x14, 0x38, 0xB2, 0x82, 0xAB, 0xD0, 0x98, 0x2C, 0x50, 0x50, 0x78, 0x78, 0x9B, 0xF1, 0x1A, 0xB0,
  207. 0xF0, 0xB1, 0x83, 0x9C, 0xA8, 0x29, 0x51, 0x79, 0xB0, 0x8B, 0x29, 0x51, 0x79, 0xB1, 0x83, 0x24,
  208. // bank 6, 256 bytes
  209. 0x70, 0x59, 0xB0, 0x8B, 0x20, 0x58, 0x71, 0xB1, 0x83, 0x44, 0x69, 0x38, 0xB0, 0x8B, 0x39, 0x40,
  210. 0x68, 0xB1, 0x83, 0x64, 0x48, 0x31, 0xB0, 0x8B, 0x30, 0x49, 0x60, 0xA5, 0x88, 0x20, 0x09, 0x71,
  211. 0x58, 0x44, 0x68, 0x11, 0x39, 0x64, 0x49, 0x30, 0x19, 0xF1, 0xAC, 0x00, 0x2C, 0x54, 0x7C, 0xF0,
  212. 0x8C, 0xA8, 0x04, 0x28, 0x50, 0x78, 0xF1, 0x88, 0x97, 0x26, 0xA8, 0x59, 0x98, 0xAC, 0x8C, 0x02,
  213. 0x26, 0x46, 0x66, 0xF0, 0x89, 0x9C, 0xA8, 0x29, 0x51, 0x79, 0x24, 0x70, 0x59, 0x44, 0x69, 0x38,
  214. 0x64, 0x48, 0x31, 0xA9, 0x88, 0x09, 0x20, 0x59, 0x70, 0xAB, 0x11, 0x38, 0x40, 0x69, 0xA8, 0x19,
  215. 0x31, 0x48, 0x60, 0x8C, 0xA8, 0x3C, 0x41, 0x5C, 0x20, 0x7C, 0x00, 0xF1, 0x87, 0x98, 0x19, 0x86,
  216. 0xA8, 0x6E, 0x76, 0x7E, 0xA9, 0x99, 0x88, 0x2D, 0x55, 0x7D, 0x9E, 0xB9, 0xA3, 0x8A, 0x22, 0x8A,
  217. 0x6E, 0x8A, 0x56, 0x8A, 0x5E, 0x9F, 0xB1, 0x83, 0x06, 0x26, 0x46, 0x66, 0x0E, 0x2E, 0x4E, 0x6E,
  218. 0x9D, 0xB8, 0xAD, 0x00, 0x2C, 0x54, 0x7C, 0xF2, 0xB1, 0x8C, 0xB4, 0x99, 0xB9, 0xA3, 0x2D, 0x55,
  219. 0x7D, 0x81, 0x91, 0xAC, 0x38, 0xAD, 0x3A, 0xB5, 0x83, 0x91, 0xAC, 0x2D, 0xD9, 0x28, 0xD8, 0x4D,
  220. 0xD9, 0x48, 0xD8, 0x6D, 0xD9, 0x68, 0xD8, 0x8C, 0x9D, 0xAE, 0x29, 0xD9, 0x04, 0xAE, 0xD8, 0x51,
  221. 0xD9, 0x04, 0xAE, 0xD8, 0x79, 0xD9, 0x04, 0xD8, 0x81, 0xF3, 0x9D, 0xAD, 0x00, 0x8D, 0xAE, 0x19,
  222. 0x81, 0xAD, 0xD9, 0x01, 0xD8, 0xF2, 0xAE, 0xDA, 0x26, 0xD8, 0x8E, 0x91, 0x29, 0x83, 0xA7, 0xD9,
  223. 0xAD, 0xAD, 0xAD, 0xAD, 0xF3, 0x2A, 0xD8, 0xD8, 0xF1, 0xB0, 0xAC, 0x89, 0x91, 0x3E, 0x5E, 0x76,
  224. 0xF3, 0xAC, 0x2E, 0x2E, 0xF1, 0xB1, 0x8C, 0x5A, 0x9C, 0xAC, 0x2C, 0x28, 0x28, 0x28, 0x9C, 0xAC,
  225. // bank 7, 170 bytes (remainder)
  226. 0x30, 0x18, 0xA8, 0x98, 0x81, 0x28, 0x34, 0x3C, 0x97, 0x24, 0xA7, 0x28, 0x34, 0x3C, 0x9C, 0x24,
  227. 0xF2, 0xB0, 0x89, 0xAC, 0x91, 0x2C, 0x4C, 0x6C, 0x8A, 0x9B, 0x2D, 0xD9, 0xD8, 0xD8, 0x51, 0xD9,
  228. 0xD8, 0xD8, 0x79, 0xD9, 0xD8, 0xD8, 0xF1, 0x9E, 0x88, 0xA3, 0x31, 0xDA, 0xD8, 0xD8, 0x91, 0x2D,
  229. 0xD9, 0x28, 0xD8, 0x4D, 0xD9, 0x48, 0xD8, 0x6D, 0xD9, 0x68, 0xD8, 0xB1, 0x83, 0x93, 0x35, 0x3D,
  230. 0x80, 0x25, 0xDA, 0xD8, 0xD8, 0x85, 0x69, 0xDA, 0xD8, 0xD8, 0xB4, 0x93, 0x81, 0xA3, 0x28, 0x34,
  231. 0x3C, 0xF3, 0xAB, 0x8B, 0xA3, 0x91, 0xB6, 0x09, 0xB4, 0xD9, 0xAB, 0xDE, 0xB0, 0x87, 0x9C, 0xB9,
  232. 0xA3, 0xDD, 0xF1, 0xA3, 0xA3, 0xA3, 0xA3, 0x95, 0xF1, 0xA3, 0xA3, 0xA3, 0x9D, 0xF1, 0xA3, 0xA3,
  233. 0xA3, 0xA3, 0xF2, 0xA3, 0xB4, 0x90, 0x80, 0xF2, 0xA3, 0xA3, 0xA3, 0xA3, 0xA3, 0xA3, 0xA3, 0xA3,
  234. 0xA3, 0xA3, 0xB2, 0xA3, 0xA3, 0xA3, 0xA3, 0xA3, 0xA3, 0xB0, 0x87, 0xB5, 0x99, 0xF1, 0xA3, 0xA3,
  235. 0xA3, 0x98, 0xF1, 0xA3, 0xA3, 0xA3, 0xA3, 0x97, 0xA3, 0xA3, 0xA3, 0xA3, 0xF3, 0x9B, 0xA3, 0xA3,
  236. 0xDC, 0xB9, 0xA7, 0xF1, 0x26, 0x26, 0x26, 0xD8, 0xD8, 0xFF
  237. };
  238. const prog_uchar dmpConfig[MPU6050_DMP_CONFIG_SIZE] PROGMEM = {
  239. // BANK OFFSET LENGTH [DATA]
  240. 0x02, 0xEC, 0x04, 0x00, 0x47, 0x7D, 0x1A, // ?
  241. 0x03, 0x82, 0x03, 0x4C, 0xCD, 0x6C, // FCFG_1 inv_set_gyro_calibration
  242. 0x03, 0xB2, 0x03, 0x36, 0x56, 0x76, // FCFG_3 inv_set_gyro_calibration
  243. 0x00, 0x68, 0x04, 0x02, 0xCA, 0xE3, 0x09, // D_0_104 inv_set_gyro_calibration
  244. 0x01, 0x0C, 0x04, 0x00, 0x00, 0x00, 0x00, // D_1_152 inv_set_accel_calibration
  245. 0x03, 0x86, 0x03, 0x0C, 0xC9, 0x2C, // FCFG_2 inv_set_accel_calibration
  246. 0x03, 0x90, 0x03, 0x26, 0x46, 0x66, // (continued)...FCFG_2 inv_set_accel_calibration
  247. 0x00, 0x6C, 0x02, 0x40, 0x00, // D_0_108 inv_set_accel_calibration
  248. 0x02, 0x40, 0x04, 0x00, 0x00, 0x00, 0x00, // CPASS_MTX_00 inv_set_compass_calibration
  249. 0x02, 0x44, 0x04, 0x40, 0x00, 0x00, 0x00, // CPASS_MTX_01
  250. 0x02, 0x48, 0x04, 0x00, 0x00, 0x00, 0x00, // CPASS_MTX_02
  251. 0x02, 0x4C, 0x04, 0x40, 0x00, 0x00, 0x00, // CPASS_MTX_10
  252. 0x02, 0x50, 0x04, 0x00, 0x00, 0x00, 0x00, // CPASS_MTX_11
  253. 0x02, 0x54, 0x04, 0x00, 0x00, 0x00, 0x00, // CPASS_MTX_12
  254. 0x02, 0x58, 0x04, 0x00, 0x00, 0x00, 0x00, // CPASS_MTX_20
  255. 0x02, 0x5C, 0x04, 0x00, 0x00, 0x00, 0x00, // CPASS_MTX_21
  256. 0x02, 0xBC, 0x04, 0xC0, 0x00, 0x00, 0x00, // CPASS_MTX_22
  257. 0x01, 0xEC, 0x04, 0x00, 0x00, 0x40, 0x00, // D_1_236 inv_apply_endian_accel
  258. 0x03, 0x86, 0x06, 0x0C, 0xC9, 0x2C, 0x97, 0x97, 0x97, // FCFG_2 inv_set_mpu_sensors
  259. 0x04, 0x22, 0x03, 0x0D, 0x35, 0x5D, // CFG_MOTION_BIAS inv_turn_on_bias_from_no_motion
  260. 0x00, 0xA3, 0x01, 0x00, // ?
  261. 0x04, 0x29, 0x04, 0x87, 0x2D, 0x35, 0x3D, // FCFG_5 inv_set_bias_update
  262. 0x07, 0x62, 0x05, 0xF1, 0x20, 0x28, 0x30, 0x38, // CFG_8 inv_send_quaternion
  263. 0x07, 0x9F, 0x01, 0x30, // CFG_16 inv_set_footer
  264. 0x07, 0x67, 0x01, 0x9A, // CFG_GYRO_SOURCE inv_send_gyro
  265. 0x07, 0x68, 0x04, 0xF1, 0x28, 0x30, 0x38, // CFG_9 inv_send_gyro -> inv_construct3_fifo
  266. 0x07, 0x62, 0x05, 0xF1, 0x20, 0x28, 0x30, 0x38, // ?
  267. 0x02, 0x0C, 0x04, 0x00, 0x00, 0x00, 0x00, // ?
  268. 0x07, 0x83, 0x06, 0xC2, 0xCA, 0xC4, 0xA3, 0xA3, 0xA3, // ?
  269. // SPECIAL 0x01 = enable interrupts
  270. 0x00, 0x00, 0x00, 0x01, // SET INT_ENABLE, SPECIAL INSTRUCTION
  271. 0x07, 0xA7, 0x01, 0xFE, // ?
  272. 0x07, 0x62, 0x05, 0xF1, 0x20, 0x28, 0x30, 0x38, // ?
  273. 0x07, 0x67, 0x01, 0x9A, // ?
  274. 0x07, 0x68, 0x04, 0xF1, 0x28, 0x30, 0x38, // CFG_12 inv_send_accel -> inv_construct3_fifo
  275. 0x07, 0x8D, 0x04, 0xF1, 0x28, 0x30, 0x38, // ??? CFG_12 inv_send_mag -> inv_construct3_fifo
  276. 0x02, 0x16, 0x02, 0x00, 0x03 // D_0_22 inv_set_fifo_rate
  277. // This very last 0x01 WAS a 0x09, which drops the FIFO rate down to 20 Hz. 0x07 is 25 Hz,
  278. // 0x01 is 100Hz. Going faster than 100Hz (0x00=200Hz) tends to result in very noisy data.
  279. // DMP output frequency is calculated easily using this equation: (200Hz / (1 + value))
  280. // It is important to make sure the host processor can keep up with reading and processing
  281. // the FIFO output at the desired rate. Handling FIFO overflow cleanly is also a good idea.
  282. };
  283. const prog_uchar dmpUpdates[MPU6050_DMP_UPDATES_SIZE] PROGMEM = {
  284. 0x01, 0xB2, 0x02, 0xFF, 0xF5,
  285. 0x01, 0x90, 0x04, 0x0A, 0x0D, 0x97, 0xC0,
  286. 0x00, 0xA3, 0x01, 0x00,
  287. 0x04, 0x29, 0x04, 0x87, 0x2D, 0x35, 0x3D,
  288. 0x01, 0x6A, 0x02, 0x06, 0x00,
  289. 0x01, 0x60, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  290. 0x00, 0x60, 0x04, 0x40, 0x00, 0x00, 0x00,
  291. 0x02, 0x60, 0x0C, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  292. 0x01, 0x08, 0x02, 0x01, 0x20,
  293. 0x01, 0x0A, 0x02, 0x00, 0x4E,
  294. 0x01, 0x02, 0x02, 0xFE, 0xB3,
  295. 0x02, 0x6C, 0x04, 0x00, 0x00, 0x00, 0x00, // READ
  296. 0x02, 0x6C, 0x04, 0xFA, 0xFE, 0x00, 0x00,
  297. 0x02, 0x60, 0x0C, 0xFF, 0xFF, 0xCB, 0x4D, 0x00, 0x01, 0x08, 0xC1, 0xFF, 0xFF, 0xBC, 0x2C,
  298. 0x02, 0xF4, 0x04, 0x00, 0x00, 0x00, 0x00,
  299. 0x02, 0xF8, 0x04, 0x00, 0x00, 0x00, 0x00,
  300. 0x02, 0xFC, 0x04, 0x00, 0x00, 0x00, 0x00,
  301. 0x00, 0x60, 0x04, 0x40, 0x00, 0x00, 0x00,
  302. 0x00, 0x60, 0x04, 0x00, 0x40, 0x00, 0x00
  303. };
  304. uint8_t MPU6050::dmpInitialize() {
  305. // reset device
  306. DEBUG_PRINTLN(F("\n\nResetting MPU6050..."));
  307. reset();
  308. delay(30); // wait after reset
  309. // disable sleep mode
  310. DEBUG_PRINTLN(F("Disabling sleep mode..."));
  311. setSleepEnabled(false);
  312. // get MPU product ID
  313. DEBUG_PRINTLN(F("Getting product ID..."));
  314. //uint8_t productID = 0; //getProductID();
  315. DEBUG_PRINT(F("Product ID = "));
  316. DEBUG_PRINT(productID);
  317. // get MPU hardware revision
  318. DEBUG_PRINTLN(F("Selecting user bank 16..."));
  319. setMemoryBank(0x10, true, true);
  320. DEBUG_PRINTLN(F("Selecting memory byte 6..."));
  321. setMemoryStartAddress(0x06);
  322. DEBUG_PRINTLN(F("Checking hardware revision..."));
  323. uint8_t hwRevision = readMemoryByte();
  324. DEBUG_PRINT(F("Revision @ user[16][6] = "));
  325. DEBUG_PRINTLNF(hwRevision, HEX);
  326. DEBUG_PRINTLN(F("Resetting memory bank selection to 0..."));
  327. setMemoryBank(0, false, false);
  328. // check OTP bank valid
  329. DEBUG_PRINTLN(F("Reading OTP bank valid flag..."));
  330. uint8_t otpValid = getOTPBankValid();
  331. DEBUG_PRINT(F("OTP bank is "));
  332. DEBUG_PRINTLN(otpValid ? F("valid!") : F("invalid!"));
  333. // get X/Y/Z gyro offsets
  334. DEBUG_PRINTLN(F("Reading gyro offset values..."));
  335. int8_t xgOffset = getXGyroOffset();
  336. int8_t ygOffset = getYGyroOffset();
  337. int8_t zgOffset = getZGyroOffset();
  338. DEBUG_PRINT(F("X gyro offset = "));
  339. DEBUG_PRINTLN(xgOffset);
  340. DEBUG_PRINT(F("Y gyro offset = "));
  341. DEBUG_PRINTLN(ygOffset);
  342. DEBUG_PRINT(F("Z gyro offset = "));
  343. DEBUG_PRINTLN(zgOffset);
  344. I2Cdev::readByte(devAddr, MPU6050_RA_USER_CTRL, buffer); // ?
  345. DEBUG_PRINTLN(F("Enabling interrupt latch, clear on any read, AUX bypass enabled"));
  346. I2Cdev::writeByte(devAddr, MPU6050_RA_INT_PIN_CFG, 0x32);
  347. // enable MPU AUX I2C bypass mode
  348. //DEBUG_PRINTLN(F("Enabling AUX I2C bypass mode..."));
  349. //setI2CBypassEnabled(true);
  350. DEBUG_PRINTLN(F("Setting magnetometer mode to power-down..."));
  351. //mag -> setMode(0);
  352. I2Cdev::writeByte(0x0E, 0x0A, 0x00);
  353. DEBUG_PRINTLN(F("Setting magnetometer mode to fuse access..."));
  354. //mag -> setMode(0x0F);
  355. I2Cdev::writeByte(0x0E, 0x0A, 0x0F);
  356. DEBUG_PRINTLN(F("Reading mag magnetometer factory calibration..."));
  357. int8_t asax, asay, asaz;
  358. //mag -> getAdjustment(&asax, &asay, &asaz);
  359. I2Cdev::readBytes(0x0E, 0x10, 3, buffer);
  360. asax = (int8_t)buffer[0];
  361. asay = (int8_t)buffer[1];
  362. asaz = (int8_t)buffer[2];
  363. DEBUG_PRINT(F("Adjustment X/Y/Z = "));
  364. DEBUG_PRINT(asax);
  365. DEBUG_PRINT(F(" / "));
  366. DEBUG_PRINT(asay);
  367. DEBUG_PRINT(F(" / "));
  368. DEBUG_PRINTLN(asaz);
  369. DEBUG_PRINTLN(F("Setting magnetometer mode to power-down..."));
  370. //mag -> setMode(0);
  371. I2Cdev::writeByte(0x0E, 0x0A, 0x00);
  372. // load DMP code into memory banks
  373. DEBUG_PRINT(F("Writing DMP code to MPU memory banks ("));
  374. DEBUG_PRINT(MPU6050_DMP_CODE_SIZE);
  375. DEBUG_PRINTLN(F(" bytes)"));
  376. if (writeProgMemoryBlock(dmpMemory, MPU6050_DMP_CODE_SIZE)) {
  377. DEBUG_PRINTLN(F("Success! DMP code written and verified."));
  378. DEBUG_PRINTLN(F("Configuring DMP and related settings..."));
  379. // write DMP configuration
  380. DEBUG_PRINT(F("Writing DMP configuration to MPU memory banks ("));
  381. DEBUG_PRINT(MPU6050_DMP_CONFIG_SIZE);
  382. DEBUG_PRINTLN(F(" bytes in config def)"));
  383. if (writeProgDMPConfigurationSet(dmpConfig, MPU6050_DMP_CONFIG_SIZE)) {
  384. DEBUG_PRINTLN(F("Success! DMP configuration written and verified."));
  385. DEBUG_PRINTLN(F("Setting DMP and FIFO_OFLOW interrupts enabled..."));
  386. setIntEnabled(0x12);
  387. DEBUG_PRINTLN(F("Setting sample rate to 200Hz..."));
  388. setRate(4); // 1khz / (1 + 4) = 200 Hz
  389. DEBUG_PRINTLN(F("Setting clock source to Z Gyro..."));
  390. setClockSource(MPU6050_CLOCK_PLL_ZGYRO);
  391. DEBUG_PRINTLN(F("Setting DLPF bandwidth to 42Hz..."));
  392. setDLPFMode(MPU6050_DLPF_BW_42);
  393. DEBUG_PRINTLN(F("Setting external frame sync to TEMP_OUT_L[0]..."));
  394. setExternalFrameSync(MPU6050_EXT_SYNC_TEMP_OUT_L);
  395. DEBUG_PRINTLN(F("Setting gyro sensitivity to +/- 2000 deg/sec..."));
  396. setFullScaleGyroRange(MPU6050_GYRO_FS_2000);
  397. DEBUG_PRINTLN(F("Setting DMP configuration bytes (function unknown)..."));
  398. setDMPConfig1(0x03);
  399. setDMPConfig2(0x00);
  400. DEBUG_PRINTLN(F("Clearing OTP Bank flag..."));
  401. setOTPBankValid(false);
  402. DEBUG_PRINTLN(F("Setting X/Y/Z gyro offsets to previous values..."));
  403. setXGyroOffset(xgOffset);
  404. setYGyroOffset(ygOffset);
  405. setZGyroOffset(zgOffset);
  406. DEBUG_PRINTLN(F("Setting X/Y/Z gyro user offsets to zero..."));
  407. setXGyroOffsetUser(0);
  408. setYGyroOffsetUser(0);
  409. setZGyroOffsetUser(0);
  410. DEBUG_PRINTLN(F("Writing final memory update 1/19 (function unknown)..."));
  411. uint8_t dmpUpdate[16], j;
  412. uint16_t pos = 0;
  413. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  414. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  415. DEBUG_PRINTLN(F("Writing final memory update 2/19 (function unknown)..."));
  416. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  417. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  418. DEBUG_PRINTLN(F("Resetting FIFO..."));
  419. resetFIFO();
  420. DEBUG_PRINTLN(F("Reading FIFO count..."));
  421. uint8_t fifoCount = getFIFOCount();
  422. DEBUG_PRINT(F("Current FIFO count="));
  423. DEBUG_PRINTLN(fifoCount);
  424. uint8_t fifoBuffer[128];
  425. //getFIFOBytes(fifoBuffer, fifoCount);
  426. DEBUG_PRINTLN(F("Writing final memory update 3/19 (function unknown)..."));
  427. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  428. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  429. DEBUG_PRINTLN(F("Writing final memory update 4/19 (function unknown)..."));
  430. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  431. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  432. DEBUG_PRINTLN(F("Disabling all standby flags..."));
  433. I2Cdev::writeByte(0x68, MPU6050_RA_PWR_MGMT_2, 0x00);
  434. DEBUG_PRINTLN(F("Setting accelerometer sensitivity to +/- 2g..."));
  435. I2Cdev::writeByte(0x68, MPU6050_RA_ACCEL_CONFIG, 0x00);
  436. DEBUG_PRINTLN(F("Setting motion detection threshold to 2..."));
  437. setMotionDetectionThreshold(2);
  438. DEBUG_PRINTLN(F("Setting zero-motion detection threshold to 156..."));
  439. setZeroMotionDetectionThreshold(156);
  440. DEBUG_PRINTLN(F("Setting motion detection duration to 80..."));
  441. setMotionDetectionDuration(80);
  442. DEBUG_PRINTLN(F("Setting zero-motion detection duration to 0..."));
  443. setZeroMotionDetectionDuration(0);
  444. DEBUG_PRINTLN(F("Setting AK8975 to single measurement mode..."));
  445. //mag -> setMode(1);
  446. I2Cdev::writeByte(0x0E, 0x0A, 0x01);
  447. // setup AK8975 (0x0E) as Slave 0 in read mode
  448. DEBUG_PRINTLN(F("Setting up AK8975 read slave 0..."));
  449. I2Cdev::writeByte(0x68, MPU6050_RA_I2C_SLV0_ADDR, 0x8E);
  450. I2Cdev::writeByte(0x68, MPU6050_RA_I2C_SLV0_REG, 0x01);
  451. I2Cdev::writeByte(0x68, MPU6050_RA_I2C_SLV0_CTRL, 0xDA);
  452. // setup AK8975 (0x0E) as Slave 2 in write mode
  453. DEBUG_PRINTLN(F("Setting up AK8975 write slave 2..."));
  454. I2Cdev::writeByte(0x68, MPU6050_RA_I2C_SLV2_ADDR, 0x0E);
  455. I2Cdev::writeByte(0x68, MPU6050_RA_I2C_SLV2_REG, 0x0A);
  456. I2Cdev::writeByte(0x68, MPU6050_RA_I2C_SLV2_CTRL, 0x81);
  457. I2Cdev::writeByte(0x68, MPU6050_RA_I2C_SLV2_DO, 0x01);
  458. // setup I2C timing/delay control
  459. DEBUG_PRINTLN(F("Setting up slave access delay..."));
  460. I2Cdev::writeByte(0x68, MPU6050_RA_I2C_SLV4_CTRL, 0x18);
  461. I2Cdev::writeByte(0x68, MPU6050_RA_I2C_MST_DELAY_CTRL, 0x05);
  462. // enable interrupts
  463. DEBUG_PRINTLN(F("Enabling default interrupt behavior/no bypass..."));
  464. I2Cdev::writeByte(0x68, MPU6050_RA_INT_PIN_CFG, 0x00);
  465. // enable I2C master mode and reset DMP/FIFO
  466. DEBUG_PRINTLN(F("Enabling I2C master mode..."));
  467. I2Cdev::writeByte(0x68, MPU6050_RA_USER_CTRL, 0x20);
  468. DEBUG_PRINTLN(F("Resetting FIFO..."));
  469. I2Cdev::writeByte(0x68, MPU6050_RA_USER_CTRL, 0x24);
  470. DEBUG_PRINTLN(F("Rewriting I2C master mode enabled because...I don't know"));
  471. I2Cdev::writeByte(0x68, MPU6050_RA_USER_CTRL, 0x20);
  472. DEBUG_PRINTLN(F("Enabling and resetting DMP/FIFO..."));
  473. I2Cdev::writeByte(0x68, MPU6050_RA_USER_CTRL, 0xE8);
  474. DEBUG_PRINTLN(F("Writing final memory update 5/19 (function unknown)..."));
  475. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  476. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  477. DEBUG_PRINTLN(F("Writing final memory update 6/19 (function unknown)..."));
  478. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  479. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  480. DEBUG_PRINTLN(F("Writing final memory update 7/19 (function unknown)..."));
  481. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  482. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  483. DEBUG_PRINTLN(F("Writing final memory update 8/19 (function unknown)..."));
  484. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  485. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  486. DEBUG_PRINTLN(F("Writing final memory update 9/19 (function unknown)..."));
  487. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  488. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  489. DEBUG_PRINTLN(F("Writing final memory update 10/19 (function unknown)..."));
  490. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  491. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  492. DEBUG_PRINTLN(F("Writing final memory update 11/19 (function unknown)..."));
  493. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  494. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  495. DEBUG_PRINTLN(F("Reading final memory update 12/19 (function unknown)..."));
  496. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  497. readMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  498. #ifdef DEBUG
  499. DEBUG_PRINT(F("Read bytes: "));
  500. for (j = 0; j < 4; j++) {
  501. DEBUG_PRINTF(dmpUpdate[3 + j], HEX);
  502. DEBUG_PRINT(" ");
  503. }
  504. DEBUG_PRINTLN("");
  505. #endif
  506. DEBUG_PRINTLN(F("Writing final memory update 13/19 (function unknown)..."));
  507. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  508. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  509. DEBUG_PRINTLN(F("Writing final memory update 14/19 (function unknown)..."));
  510. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  511. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  512. DEBUG_PRINTLN(F("Writing final memory update 15/19 (function unknown)..."));
  513. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  514. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  515. DEBUG_PRINTLN(F("Writing final memory update 16/19 (function unknown)..."));
  516. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  517. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  518. DEBUG_PRINTLN(F("Writing final memory update 17/19 (function unknown)..."));
  519. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  520. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  521. DEBUG_PRINTLN(F("Waiting for FIRO count >= 46..."));
  522. while ((fifoCount = getFIFOCount()) < 46);
  523. DEBUG_PRINTLN(F("Reading FIFO..."));
  524. getFIFOBytes(fifoBuffer, min(fifoCount, 128)); // safeguard only 128 bytes
  525. DEBUG_PRINTLN(F("Reading interrupt status..."));
  526. getIntStatus();
  527. DEBUG_PRINTLN(F("Writing final memory update 18/19 (function unknown)..."));
  528. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  529. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  530. DEBUG_PRINTLN(F("Waiting for FIRO count >= 48..."));
  531. while ((fifoCount = getFIFOCount()) < 48);
  532. DEBUG_PRINTLN(F("Reading FIFO..."));
  533. getFIFOBytes(fifoBuffer, min(fifoCount, 128)); // safeguard only 128 bytes
  534. DEBUG_PRINTLN(F("Reading interrupt status..."));
  535. getIntStatus();
  536. DEBUG_PRINTLN(F("Waiting for FIRO count >= 48..."));
  537. while ((fifoCount = getFIFOCount()) < 48);
  538. DEBUG_PRINTLN(F("Reading FIFO..."));
  539. getFIFOBytes(fifoBuffer, min(fifoCount, 128)); // safeguard only 128 bytes
  540. DEBUG_PRINTLN(F("Reading interrupt status..."));
  541. getIntStatus();
  542. DEBUG_PRINTLN(F("Writing final memory update 19/19 (function unknown)..."));
  543. for (j = 0; j < 4 || j < dmpUpdate[2] + 3; j++, pos++) dmpUpdate[j] = pgm_read_byte(&dmpUpdates[pos]);
  544. writeMemoryBlock(dmpUpdate + 3, dmpUpdate[2], dmpUpdate[0], dmpUpdate[1]);
  545. DEBUG_PRINTLN(F("Disabling DMP (you turn it on later)..."));
  546. setDMPEnabled(false);
  547. DEBUG_PRINTLN(F("Setting up internal 48-byte (default) DMP packet buffer..."));
  548. dmpPacketSize = 48;
  549. /*if ((dmpPacketBuffer = (uint8_t *)malloc(42)) == 0) {
  550. return 3; // TODO: proper error code for no memory
  551. }*/
  552. DEBUG_PRINTLN(F("Resetting FIFO and clearing INT status one last time..."));
  553. resetFIFO();
  554. getIntStatus();
  555. } else {
  556. DEBUG_PRINTLN(F("ERROR! DMP configuration verification failed."));
  557. return 2; // configuration block loading failed
  558. }
  559. } else {
  560. DEBUG_PRINTLN(F("ERROR! DMP code verification failed."));
  561. return 1; // main binary block loading failed
  562. }
  563. return 0; // success
  564. }
  565. bool MPU6050::dmpPacketAvailable() {
  566. return getFIFOCount() >= dmpGetFIFOPacketSize();
  567. }
  568. // uint8_t MPU6050::dmpSetFIFORate(uint8_t fifoRate);
  569. // uint8_t MPU6050::dmpGetFIFORate();
  570. // uint8_t MPU6050::dmpGetSampleStepSizeMS();
  571. // uint8_t MPU6050::dmpGetSampleFrequency();
  572. // int32_t MPU6050::dmpDecodeTemperature(int8_t tempReg);
  573. //uint8_t MPU6050::dmpRegisterFIFORateProcess(inv_obj_func func, int16_t priority);
  574. //uint8_t MPU6050::dmpUnregisterFIFORateProcess(inv_obj_func func);
  575. //uint8_t MPU6050::dmpRunFIFORateProcesses();
  576. // uint8_t MPU6050::dmpSendQuaternion(uint_fast16_t accuracy);
  577. // uint8_t MPU6050::dmpSendGyro(uint_fast16_t elements, uint_fast16_t accuracy);
  578. // uint8_t MPU6050::dmpSendAccel(uint_fast16_t elements, uint_fast16_t accuracy);
  579. // uint8_t MPU6050::dmpSendLinearAccel(uint_fast16_t elements, uint_fast16_t accuracy);
  580. // uint8_t MPU6050::dmpSendLinearAccelInWorld(uint_fast16_t elements, uint_fast16_t accuracy);
  581. // uint8_t MPU6050::dmpSendControlData(uint_fast16_t elements, uint_fast16_t accuracy);
  582. // uint8_t MPU6050::dmpSendSensorData(uint_fast16_t elements, uint_fast16_t accuracy);
  583. // uint8_t MPU6050::dmpSendExternalSensorData(uint_fast16_t elements, uint_fast16_t accuracy);
  584. // uint8_t MPU6050::dmpSendGravity(uint_fast16_t elements, uint_fast16_t accuracy);
  585. // uint8_t MPU6050::dmpSendPacketNumber(uint_fast16_t accuracy);
  586. // uint8_t MPU6050::dmpSendQuantizedAccel(uint_fast16_t elements, uint_fast16_t accuracy);
  587. // uint8_t MPU6050::dmpSendEIS(uint_fast16_t elements, uint_fast16_t accuracy);
  588. uint8_t MPU6050::dmpGetAccel(int32_t *data, const uint8_t* packet) {
  589. // TODO: accommodate different arrangements of sent data (ONLY default supported now)
  590. if (packet == 0) packet = dmpPacketBuffer;
  591. data[0] = ((packet[34] << 24) + (packet[35] << 16) + (packet[36] << 8) + packet[37]);
  592. data[1] = ((packet[38] << 24) + (packet[39] << 16) + (packet[40] << 8) + packet[41]);
  593. data[2] = ((packet[42] << 24) + (packet[43] << 16) + (packet[44] << 8) + packet[45]);
  594. return 0;
  595. }
  596. uint8_t MPU6050::dmpGetAccel(int16_t *data, const uint8_t* packet) {
  597. // TODO: accommodate different arrangements of sent data (ONLY default supported now)
  598. if (packet == 0) packet = dmpPacketBuffer;
  599. data[0] = (packet[34] << 8) + packet[35];
  600. data[1] = (packet[38] << 8) + packet[39];
  601. data[2] = (packet[42] << 8) + packet[43];
  602. return 0;
  603. }
  604. uint8_t MPU6050::dmpGetAccel(VectorInt16 *v, const uint8_t* packet) {
  605. // TODO: accommodate different arrangements of sent data (ONLY default supported now)
  606. if (packet == 0) packet = dmpPacketBuffer;
  607. v -> x = (packet[34] << 8) + packet[35];
  608. v -> y = (packet[38] << 8) + packet[39];
  609. v -> z = (packet[42] << 8) + packet[43];
  610. return 0;
  611. }
  612. uint8_t MPU6050::dmpGetQuaternion(int32_t *data, const uint8_t* packet) {
  613. // TODO: accommodate different arrangements of sent data (ONLY default supported now)
  614. if (packet == 0) packet = dmpPacketBuffer;
  615. data[0] = ((packet[0] << 24) + (packet[1] << 16) + (packet[2] << 8) + packet[3]);
  616. data[1] = ((packet[4] << 24) + (packet[5] << 16) + (packet[6] << 8) + packet[7]);
  617. data[2] = ((packet[8] << 24) + (packet[9] << 16) + (packet[10] << 8) + packet[11]);
  618. data[3] = ((packet[12] << 24) + (packet[13] << 16) + (packet[14] << 8) + packet[15]);
  619. return 0;
  620. }
  621. uint8_t MPU6050::dmpGetQuaternion(int16_t *data, const uint8_t* packet) {
  622. // TODO: accommodate different arrangements of sent data (ONLY default supported now)
  623. if (packet == 0) packet = dmpPacketBuffer;
  624. data[0] = ((packet[0] << 8) + packet[1]);
  625. data[1] = ((packet[4] << 8) + packet[5]);
  626. data[2] = ((packet[8] << 8) + packet[9]);
  627. data[3] = ((packet[12] << 8) + packet[13]);
  628. return 0;
  629. }
  630. uint8_t MPU6050::dmpGetQuaternion(Quaternion *q, const uint8_t* packet) {
  631. // TODO: accommodate different arrangements of sent data (ONLY default supported now)
  632. int16_t qI[4];
  633. uint8_t status = dmpGetQuaternion(qI, packet);
  634. if (status == 0) {
  635. q -> w = (float)qI[0] / 16384.0f;
  636. q -> x = (float)qI[1] / 16384.0f;
  637. q -> y = (float)qI[2] / 16384.0f;
  638. q -> z = (float)qI[3] / 16384.0f;
  639. return 0;
  640. }
  641. return status; // int16 return value, indicates error if this line is reached
  642. }
  643. // uint8_t MPU6050::dmpGet6AxisQuaternion(long *data, const uint8_t* packet);
  644. // uint8_t MPU6050::dmpGetRelativeQuaternion(long *data, const uint8_t* packet);
  645. uint8_t MPU6050::dmpGetGyro(int32_t *data, const uint8_t* packet) {
  646. // TODO: accommodate different arrangements of sent data (ONLY default supported now)
  647. if (packet == 0) packet = dmpPacketBuffer;
  648. data[0] = ((packet[16] << 24) + (packet[17] << 16) + (packet[18] << 8) + packet[19]);
  649. data[1] = ((packet[20] << 24) + (packet[21] << 16) + (packet[22] << 8) + packet[23]);
  650. data[2] = ((packet[24] << 24) + (packet[25] << 16) + (packet[26] << 8) + packet[27]);
  651. return 0;
  652. }
  653. uint8_t MPU6050::dmpGetGyro(int16_t *data, const uint8_t* packet) {
  654. // TODO: accommodate different arrangements of sent data (ONLY default supported now)
  655. if (packet == 0) packet = dmpPacketBuffer;
  656. data[0] = (packet[16] << 8) + packet[17];
  657. data[1] = (packet[20] << 8) + packet[21];
  658. data[2] = (packet[24] << 8) + packet[25];
  659. return 0;
  660. }
  661. uint8_t MPU6050::dmpGetMag(int16_t *data, const uint8_t* packet) {
  662. // TODO: accommodate different arrangements of sent data (ONLY default supported now)
  663. if (packet == 0) packet = dmpPacketBuffer;
  664. data[0] = (packet[28] << 8) + packet[29];
  665. data[1] = (packet[30] << 8) + packet[31];
  666. data[2] = (packet[32] << 8) + packet[33];
  667. return 0;
  668. }
  669. // uint8_t MPU6050::dmpSetLinearAccelFilterCoefficient(float coef);
  670. // uint8_t MPU6050::dmpGetLinearAccel(long *data, const uint8_t* packet);
  671. uint8_t MPU6050::dmpGetLinearAccel(VectorInt16 *v, VectorInt16 *vRaw, VectorFloat *gravity) {
  672. // get rid of the gravity component (+1g = +4096 in standard DMP FIFO packet)
  673. v -> x = vRaw -> x - gravity -> x*4096;
  674. v -> y = vRaw -> y - gravity -> y*4096;
  675. v -> z = vRaw -> z - gravity -> z*4096;
  676. return 0;
  677. }
  678. // uint8_t MPU6050::dmpGetLinearAccelInWorld(long *data, const uint8_t* packet);
  679. uint8_t MPU6050::dmpGetLinearAccelInWorld(VectorInt16 *v, VectorInt16 *vReal, Quaternion *q) {
  680. // rotate measured 3D acceleration vector into original state
  681. // frame of reference based on orientation quaternion
  682. memcpy(v, vReal, sizeof(VectorInt16));
  683. v -> rotate(q);
  684. return 0;
  685. }
  686. // uint8_t MPU6050::dmpGetGyroAndAccelSensor(long *data, const uint8_t* packet);
  687. // uint8_t MPU6050::dmpGetGyroSensor(long *data, const uint8_t* packet);
  688. // uint8_t MPU6050::dmpGetControlData(long *data, const uint8_t* packet);
  689. // uint8_t MPU6050::dmpGetTemperature(long *data, const uint8_t* packet);
  690. // uint8_t MPU6050::dmpGetGravity(long *data, const uint8_t* packet);
  691. uint8_t MPU6050::dmpGetGravity(VectorFloat *v, Quaternion *q) {
  692. v -> x = 2 * (q -> x*q -> z - q -> w*q -> y);
  693. v -> y = 2 * (q -> w*q -> x + q -> y*q -> z);
  694. v -> z = q -> w*q -> w - q -> x*q -> x - q -> y*q -> y + q -> z*q -> z;
  695. return 0;
  696. }
  697. // uint8_t MPU6050::dmpGetUnquantizedAccel(long *data, const uint8_t* packet);
  698. // uint8_t MPU6050::dmpGetQuantizedAccel(long *data, const uint8_t* packet);
  699. // uint8_t MPU6050::dmpGetExternalSensorData(long *data, int size, const uint8_t* packet);
  700. // uint8_t MPU6050::dmpGetEIS(long *data, const uint8_t* packet);
  701. uint8_t MPU6050::dmpGetEuler(float *data, Quaternion *q) {
  702. data[0] = atan2(2*q -> x*q -> y - 2*q -> w*q -> z, 2*q -> w*q -> w + 2*q -> x*q -> x - 1); // psi
  703. data[1] = -asin(2*q -> x*q -> z + 2*q -> w*q -> y); // theta
  704. data[2] = atan2(2*q -> y*q -> z - 2*q -> w*q -> x, 2*q -> w*q -> w + 2*q -> z*q -> z - 1); // phi
  705. return 0;
  706. }
  707. uint8_t MPU6050::dmpGetYawPitchRoll(float *data, Quaternion *q, VectorFloat *gravity) {
  708. // yaw: (about Z axis)
  709. data[0] = atan2(2*q -> x*q -> y - 2*q -> w*q -> z, 2*q -> w*q -> w + 2*q -> x*q -> x - 1);
  710. // pitch: (nose up/down, about Y axis)
  711. data[1] = atan(gravity -> x / sqrt(gravity -> y*gravity -> y + gravity -> z*gravity -> z));
  712. // roll: (tilt left/right, about X axis)
  713. data[2] = atan(gravity -> y / sqrt(gravity -> x*gravity -> x + gravity -> z*gravity -> z));
  714. return 0;
  715. }
  716. // uint8_t MPU6050::dmpGetAccelFloat(float *data, const uint8_t* packet);
  717. // uint8_t MPU6050::dmpGetQuaternionFloat(float *data, const uint8_t* packet);
  718. uint8_t MPU6050::dmpProcessFIFOPacket(const unsigned char *dmpData) {
  719. /*for (uint8_t k = 0; k < dmpPacketSize; k++) {
  720. if (dmpData[k] < 0x10) Serial.print("0");
  721. Serial.print(dmpData[k], HEX);
  722. Serial.print(" ");
  723. }
  724. Serial.print("\n");*/
  725. //Serial.println((uint16_t)dmpPacketBuffer);
  726. return 0;
  727. }
  728. uint8_t MPU6050::dmpReadAndProcessFIFOPacket(uint8_t numPackets, uint8_t *processed) {
  729. uint8_t status;
  730. uint8_t buf[dmpPacketSize];
  731. for (uint8_t i = 0; i < numPackets; i++) {
  732. // read packet from FIFO
  733. getFIFOBytes(buf, dmpPacketSize);
  734. // process packet
  735. if ((status = dmpProcessFIFOPacket(buf)) > 0) return status;
  736. // increment external process count variable, if supplied
  737. if (processed != 0) *processed++;
  738. }
  739. return 0;
  740. }
  741. // uint8_t MPU6050::dmpSetFIFOProcessedCallback(void (*func) (void));
  742. // uint8_t MPU6050::dmpInitFIFOParam();
  743. // uint8_t MPU6050::dmpCloseFIFO();
  744. // uint8_t MPU6050::dmpSetGyroDataSource(uint_fast8_t source);
  745. // uint8_t MPU6050::dmpDecodeQuantizedAccel();
  746. // uint32_t MPU6050::dmpGetGyroSumOfSquare();
  747. // uint32_t MPU6050::dmpGetAccelSumOfSquare();
  748. // void MPU6050::dmpOverrideQuaternion(long *q);
  749. uint16_t MPU6050::dmpGetFIFOPacketSize() {
  750. return dmpPacketSize;
  751. }
  752. #endif /* _MPU6050_9AXIS_MOTIONAPPS41_H_ */