Library for driving the MMA8452 accelerometer over I2C

Dependents:   MMA8452_Test MMA8452_Demo Dualing_Tanks IMU-Controlled_MP3_Player ... more

Here is a simple example:

#include "mbed.h"
#include "MMA8452.h"

int main() {
   Serial pc(USBTX,USBRX);
   pc.baud(115200);
   double x = 0, y = 0, z = 0;

   MMA8452 acc(p28, p27, 40000);
   acc.setBitDepth(MMA8452::BIT_DEPTH_12);
   acc.setDynamicRange(MMA8452::DYNAMIC_RANGE_4G);
   acc.setDataRate(MMA8452::RATE_100);
   
   while(1) {
      if(!acc.isXYZReady()) {
         wait(0.01);
         continue;
      }
      acc.readXYZGravity(&x,&y,&z);
      pc.printf("Gravities: %lf %lf %lf\r\n",x,y,z);
   }
}

An easy way to test that this actually works is to run the loop above and hold the MMA8452 parallel to the ground along the respective axis (and upsidedown in each axis). You will see 1G on the respective axis and 0G on the others.

Committer:
ashleymills
Date:
Wed Mar 05 17:04:04 2014 +0000
Revision:
18:27d839e6dc0e
Parent:
17:6e4232c421c0
Child:
19:4d6cd7140a71
Debugging enabled by default for convenience

Who changed what in which revision?

UserRevisionLine numberNew contents of line
ashleymills 15:7620a11149b8 1 #pragma once
ashleymills 15:7620a11149b8 2
ashleymills 14:0602b45ca70f 3 // Authors: Ashley Mills, Nicholas Herriot
nherriot 0:bcf2aa85d7f9 4 /* Copyright (c) 2013 Vodafone, MIT License
nherriot 0:bcf2aa85d7f9 5 *
nherriot 0:bcf2aa85d7f9 6 * Permission is hereby granted, free of charge, to any person obtaining a copy of this software
nherriot 0:bcf2aa85d7f9 7 * and associated documentation files (the "Software"), to deal in the Software without restriction,
nherriot 0:bcf2aa85d7f9 8 * including without limitation the rights to use, copy, modify, merge, publish, distribute,
nherriot 0:bcf2aa85d7f9 9 * sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is
nherriot 0:bcf2aa85d7f9 10 * furnished to do so, subject to the following conditions:
nherriot 0:bcf2aa85d7f9 11 *
nherriot 0:bcf2aa85d7f9 12 * The above copyright notice and this permission notice shall be included in all copies or
nherriot 0:bcf2aa85d7f9 13 * substantial portions of the Software.
nherriot 0:bcf2aa85d7f9 14 *
nherriot 0:bcf2aa85d7f9 15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
nherriot 0:bcf2aa85d7f9 16 * BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
nherriot 0:bcf2aa85d7f9 17 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
nherriot 0:bcf2aa85d7f9 18 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
nherriot 0:bcf2aa85d7f9 19 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
nherriot 0:bcf2aa85d7f9 20 */
nherriot 0:bcf2aa85d7f9 21
nherriot 0:bcf2aa85d7f9 22 // the SparkFun breakout board defaults to 1, set to 0 if SA0 jumper on the bottom of the board is set
nherriot 0:bcf2aa85d7f9 23 // see the Table 10. I2C Device Address Sequence in Freescale MMA8452Q pdf
nherriot 0:bcf2aa85d7f9 24
ashleymills 15:7620a11149b8 25 #include "mbed.h"
nherriot 0:bcf2aa85d7f9 26
ashleymills 18:27d839e6dc0e 27 #define MMA8452_DEBUG 1
ashleymills 18:27d839e6dc0e 28
nherriot 0:bcf2aa85d7f9 29 // More info on MCU Master address can be found on section 5.10.1 of http://www.freescale.com/webapp/sps/site/prod_summary.jsp?code=MMA8452Q
nherriot 0:bcf2aa85d7f9 30 #define SA0 1
nherriot 0:bcf2aa85d7f9 31 #if SA0
ashleymills 11:dfd1e0afcb7b 32 #define MMA8452_ADDRESS 0x3A // 0x1D<<1 // SA0 is high, 0x1C if low -
nherriot 0:bcf2aa85d7f9 33 #else
ashleymills 11:dfd1e0afcb7b 34 #define MMA8452_ADDRESS 0x38 // 0x1C<<1
nherriot 0:bcf2aa85d7f9 35 #endif
nherriot 0:bcf2aa85d7f9 36
nherriot 0:bcf2aa85d7f9 37 // Register descriptions found in section 6 of pdf
ashleymills 12:172540ff6b8b 38 #define MMA8452_STATUS 0x00 // Type 'read' : Status of the data registers
ashleymills 12:172540ff6b8b 39 #define MMA8452_OUT_X_MSB 0x01 // Type 'read' : x axis - MSB of 2 byte sample
ashleymills 12:172540ff6b8b 40 #define MMA8452_OUT_X_LSB 0x02 // Type 'read' : x axis - LSB of 2 byte sample
ashleymills 12:172540ff6b8b 41 #define MMA8452_OUT_Y_MSB 0x03 // Type 'read' : y axis - MSB of 2 byte sample
ashleymills 12:172540ff6b8b 42 #define MMA8452_OUT_Y_LSB 0x04 // Type 'read' : y axis - LSB of 2 byte sample
ashleymills 12:172540ff6b8b 43 #define MMA8452_OUT_Z_MSB 0x05 // Type 'read' : z axis - MSB of 2 byte sample
ashleymills 12:172540ff6b8b 44 #define MMA8452_OUT_Z_LSB 0x06 // Type 'read' : z axis - LSB of 2 byte sample
nherriot 0:bcf2aa85d7f9 45
ashleymills 12:172540ff6b8b 46 // register definitions
ashleymills 12:172540ff6b8b 47 #define MMA8452_XYZ_DATA_CFG 0x0E
ashleymills 12:172540ff6b8b 48
ashleymills 12:172540ff6b8b 49 #define MMA8452_SYSMOD 0x0B // Type 'read' : This tells you if device is active, sleep or standy 0x00=STANDBY 0x01=WAKE 0x02=SLEEP
ashleymills 12:172540ff6b8b 50 #define MMA8452_WHO_AM_I 0x0D // Type 'read' : This should return the device id of 0x2A
nherriot 0:bcf2aa85d7f9 51
ashleymills 12:172540ff6b8b 52 #define MMA8452_PL_STATUS 0x10 // Type 'read' : This shows portrait landscape mode orientation
ashleymills 12:172540ff6b8b 53 #define MMA8452_PL_CFG 0x11 // Type 'read/write' : This allows portrait landscape configuration
ashleymills 12:172540ff6b8b 54 #define MMA8452_PL_COUNT 0x12 // Type 'read' : This is the portraint landscape debounce counter
ashleymills 12:172540ff6b8b 55 #define MMA8452_PL_BF_ZCOMP 0x13 // Type 'read' :
ashleymills 12:172540ff6b8b 56 #define MMA8452_PL_THS_REG 0x14 // Type 'read' :
nherriot 0:bcf2aa85d7f9 57
ashleymills 12:172540ff6b8b 58 #define MMA8452_FF_MT_CFG 0X15 // Type 'read/write' : Freefaul motion functional block configuration
ashleymills 12:172540ff6b8b 59 #define MMA8452_FF_MT_SRC 0X16 // Type 'read' : Freefaul motion event source register
ashleymills 12:172540ff6b8b 60 #define MMA8452_FF_MT_THS 0X17 // Type 'read' : Freefaul motion threshold register
ashleymills 12:172540ff6b8b 61 #define MMA8452_FF_COUNT 0X18 // Type 'read' : Freefaul motion debouce counter
nherriot 0:bcf2aa85d7f9 62
ashleymills 12:172540ff6b8b 63 #define MMA8452_ASLP_COUNT 0x29 // Type 'read/write' : Counter settings for auto sleep
ashleymills 12:172540ff6b8b 64 #define MMA8452_CTRL_REG_1 0x2A // Type 'read/write' :
ashleymills 12:172540ff6b8b 65 #define MMA8452_CTRL_REG_2 0x2B // Type 'read/write' :
ashleymills 12:172540ff6b8b 66 #define MMA8452_CTRL_REG_3 0x2C // Type 'read/write' :
ashleymills 12:172540ff6b8b 67 #define MMA8452_CTRL_REG_4 0x2D // Type 'read/write' :
ashleymills 12:172540ff6b8b 68 #define MMA8452_CTRL_REG_5 0x2E // Type 'read/write' :
nherriot 0:bcf2aa85d7f9 69
nherriot 3:ffb0b1650ca2 70 // Defined in table 13 of the Freescale PDF
ashleymills 12:172540ff6b8b 71 /// xxx these all need to have better names
nherriot 3:ffb0b1650ca2 72 #define STANDBY 0x00 // State value returned after a SYSMOD request, it can be in state STANDBY, WAKE or SLEEP
nherriot 3:ffb0b1650ca2 73 #define WAKE 0x01 // State value returned after a SYSMOD request, it can be in state STANDBY, WAKE or SLEEP
nherriot 3:ffb0b1650ca2 74 #define SLEEP 0x02 // State value returned after a SYSMOD request, it can be in state STANDBY, WAKE or SLEEP
nherriot 3:ffb0b1650ca2 75 #define ACTIVE 0x01 // Stage value returned and set in Control Register 1, it can be STANDBY=00, or ACTIVE=01
nherriot 0:bcf2aa85d7f9 76
nherriot 0:bcf2aa85d7f9 77 #define TILT_STATUS 0x03 // Tilt Status (Read only)
nherriot 0:bcf2aa85d7f9 78 #define SRST_STATUS 0x04 // Sample Rate Status Register (Read only)
nherriot 0:bcf2aa85d7f9 79 #define SPCNT_STATUS 0x05 // Sleep Count Register (Read/Write)
nherriot 0:bcf2aa85d7f9 80 #define INTSU_STATUS 0x06 // Interrupt Setup Register
nherriot 0:bcf2aa85d7f9 81 #define MODE_STATUS 0x07 // Mode Register (Read/Write)
nherriot 0:bcf2aa85d7f9 82 #define SR_STATUS 0x08 // Auto-Wake and Active Mode Portrait/Landscape Samples per Seconds Register (Read/Write)
nherriot 0:bcf2aa85d7f9 83 #define PDET_STATUS 0x09 // Tap/Pulse Detection Register (Read/Write)
nherriot 0:bcf2aa85d7f9 84 #define PD_STATUS 0xA // Tap/Pulse Debounce Count Register (Read/Write)
nherriot 0:bcf2aa85d7f9 85
ashleymills 12:172540ff6b8b 86 // masks for enabling/disabling standby
ashleymills 11:dfd1e0afcb7b 87 #define MMA8452_ACTIVE_MASK 0x01
ashleymills 11:dfd1e0afcb7b 88 #define MMA8452_STANDBY_MASK 0xFE
ashleymills 11:dfd1e0afcb7b 89
ashleymills 12:172540ff6b8b 90 // mask for dynamic range reading and writing
ashleymills 11:dfd1e0afcb7b 91 #define MMA8452_DYNAMIC_RANGE_MASK 0xFC
ashleymills 11:dfd1e0afcb7b 92
ashleymills 12:172540ff6b8b 93 // mask and shift for data rate reading and writing
ashleymills 11:dfd1e0afcb7b 94 #define MMA8452_DATA_RATE_MASK 0xC7
ashleymills 11:dfd1e0afcb7b 95 #define MMA8452_DATA_RATE_MASK_SHIFT 0x03
ashleymills 11:dfd1e0afcb7b 96
ashleymills 12:172540ff6b8b 97 // mask and shift for general reading and writing
ashleymills 11:dfd1e0afcb7b 98 #define MMA8452_WRITE_MASK 0xFE
ashleymills 11:dfd1e0afcb7b 99 #define MMA8452_READ_MASK 0x01
ashleymills 11:dfd1e0afcb7b 100
ashleymills 12:172540ff6b8b 101 // mask and shift for bit depth reading and writing
ashleymills 11:dfd1e0afcb7b 102 #define MMA8452_BIT_DEPTH_MASK 0xFD
ashleymills 11:dfd1e0afcb7b 103 #define MMA8452_BIT_DEPTH_MASK_SHIFT 0x01
ashleymills 12:172540ff6b8b 104
ashleymills 12:172540ff6b8b 105 // status masks and shifts
ashleymills 12:172540ff6b8b 106 #define MMA8452_STATUS_ZYXDR_MASK 0x08
ashleymills 16:d6dde2318edc 107 #define MMA8452_STATUS_ZDR_MASK 0x04
ashleymills 16:d6dde2318edc 108 #define MMA8452_STATUS_YDR_MASK 0x02
ashleymills 16:d6dde2318edc 109 #define MMA8452_STATUS_XDR_MASK 0x01
ashleymills 11:dfd1e0afcb7b 110
ashleymills 10:ca9ba7ad4e94 111 class MMA8452
nherriot 0:bcf2aa85d7f9 112 {
nherriot 0:bcf2aa85d7f9 113 public:
ashleymills 10:ca9ba7ad4e94 114
ashleymills 11:dfd1e0afcb7b 115 enum DynamicRange {
ashleymills 11:dfd1e0afcb7b 116 DYNAMIC_RANGE_2G=0x00,
ashleymills 11:dfd1e0afcb7b 117 DYNAMIC_RANGE_4G,
ashleymills 11:dfd1e0afcb7b 118 DYNAMIC_RANGE_8G,
ashleymills 11:dfd1e0afcb7b 119 DYNAMIC_RANGE_UNKNOWN
ashleymills 10:ca9ba7ad4e94 120 };
nherriot 0:bcf2aa85d7f9 121
ashleymills 10:ca9ba7ad4e94 122 enum BitDepth {
ashleymills 11:dfd1e0afcb7b 123 BIT_DEPTH_12=0x00,
ashleymills 12:172540ff6b8b 124 BIT_DEPTH_8, // 1 sets fast read mode, hence the inversion
ashleymills 12:172540ff6b8b 125 BIT_DEPTH_UNKNOWN
ashleymills 10:ca9ba7ad4e94 126 };
ashleymills 11:dfd1e0afcb7b 127
ashleymills 11:dfd1e0afcb7b 128 enum DataRateHz {
ashleymills 11:dfd1e0afcb7b 129 RATE_800=0x00,
ashleymills 11:dfd1e0afcb7b 130 RATE_400,
ashleymills 11:dfd1e0afcb7b 131 RATE_200,
ashleymills 11:dfd1e0afcb7b 132 RATE_100,
ashleymills 11:dfd1e0afcb7b 133 RATE_50,
ashleymills 11:dfd1e0afcb7b 134 RATE_12_5,
ashleymills 11:dfd1e0afcb7b 135 RATE_6_25,
ashleymills 11:dfd1e0afcb7b 136 RATE_1_563,
ashleymills 11:dfd1e0afcb7b 137 RATE_UNKNOWN
ashleymills 11:dfd1e0afcb7b 138 };
ashleymills 11:dfd1e0afcb7b 139
nherriot 0:bcf2aa85d7f9 140
ashleymills 15:7620a11149b8 141 /**
ashleymills 15:7620a11149b8 142 * Create an accelerometer object connected to the specified I2C pins.
nherriot 0:bcf2aa85d7f9 143 *
nherriot 0:bcf2aa85d7f9 144 * @param sda I2C data port
ashleymills 15:7620a11149b8 145 * @param scl I2C clock port
ashleymills 15:7620a11149b8 146 * @param frequency
nherriot 0:bcf2aa85d7f9 147 *
nherriot 0:bcf2aa85d7f9 148 */
ashleymills 10:ca9ba7ad4e94 149 MMA8452(PinName sda, PinName scl, int frequency);
nherriot 0:bcf2aa85d7f9 150
ashleymills 15:7620a11149b8 151 /// Destructor
ashleymills 10:ca9ba7ad4e94 152 ~MMA8452();
nherriot 3:ffb0b1650ca2 153
ashleymills 15:7620a11149b8 154 /**
ashleymills 15:7620a11149b8 155 * Puts the MMA8452 in active mode.
ashleymills 15:7620a11149b8 156 * @return 0 on success, 1 on failure.
ashleymills 15:7620a11149b8 157 */
nherriot 0:bcf2aa85d7f9 158 int activate();
nherriot 3:ffb0b1650ca2 159
ashleymills 15:7620a11149b8 160 /**
ashleymills 15:7620a11149b8 161 * Puts the MMA8452 in standby.
ashleymills 15:7620a11149b8 162 * @return 0 on success, 1 on failure.
ashleymills 15:7620a11149b8 163 */
ashleymills 15:7620a11149b8 164 int standby();
nherriot 0:bcf2aa85d7f9 165
nherriot 0:bcf2aa85d7f9 166 /** Initialization of device MMA8452 (required)
nherriot 0:bcf2aa85d7f9 167 */
nherriot 0:bcf2aa85d7f9 168 void init();
nherriot 0:bcf2aa85d7f9 169
ashleymills 15:7620a11149b8 170 /**
ashleymills 15:7620a11149b8 171 * Read the device ID from the accelerometer (should be 0x2a)
nherriot 1:ef026bf28798 172 *
ashleymills 15:7620a11149b8 173 * @param pointer to store the ID
ashleymills 15:7620a11149b8 174 * @return 0 on success, 1 on failure.
nherriot 1:ef026bf28798 175 */
ashleymills 11:dfd1e0afcb7b 176 int getDeviceID(char* dst);
nherriot 0:bcf2aa85d7f9 177
ashleymills 12:172540ff6b8b 178 int getStatus(char* dst);
nherriot 1:ef026bf28798 179
ashleymills 12:172540ff6b8b 180 /**
ashleymills 12:172540ff6b8b 181 * Read the x,y, and z registers of the MMA8452.
nherriot 1:ef026bf28798 182 *
ashleymills 12:172540ff6b8b 183 * @param dst The destination buffer. Note that this needs to be 3 bytes for
ashleymills 12:172540ff6b8b 184 * BIT_DEPTH_8 and 6 bytes for BIT_DEPTH_12. It is upto the caller to ensure this.
ashleymills 12:172540ff6b8b 185 * @return 0 for success, and 1 for failure
nherriot 1:ef026bf28798 186 */
ashleymills 15:7620a11149b8 187 int readXYZRaw(char *dst);
ashleymills 15:7620a11149b8 188
ashleymills 15:7620a11149b8 189 /**
ashleymills 15:7620a11149b8 190 * Read the x,y, and z counts of the MMA7452.
ashleymills 15:7620a11149b8 191 */
ashleymills 13:4bd8b4cd479d 192 int readXYZCounts(int *x, int *y, int *z);
ashleymills 13:4bd8b4cd479d 193 int readXYZGravity(double *x, double *y, double *z);
ashleymills 12:172540ff6b8b 194
ashleymills 13:4bd8b4cd479d 195 int isXReady();
ashleymills 13:4bd8b4cd479d 196 int isYReady();
ashleymills 13:4bd8b4cd479d 197 int isZReady();
ashleymills 12:172540ff6b8b 198 int isXYZReady();
nherriot 0:bcf2aa85d7f9 199
ashleymills 15:7620a11149b8 200 /**
ashleymills 15:7620a11149b8 201 * Read from specified MMA8452 register.
ashleymills 12:172540ff6b8b 202 *
ashleymills 12:172540ff6b8b 203 * @param addr The internal registeraddress of the MMA8452
ashleymills 12:172540ff6b8b 204 * @return The value of the register
ashleymills 12:172540ff6b8b 205 */
ashleymills 11:dfd1e0afcb7b 206 int readRegister(char addr, char *dst);
ashleymills 11:dfd1e0afcb7b 207
ashleymills 11:dfd1e0afcb7b 208 int readRegister(char addr, char *dst, int nbytes);
nherriot 0:bcf2aa85d7f9 209
ashleymills 12:172540ff6b8b 210 /**
ashleymills 12:172540ff6b8b 211 * Write to the specified MMA8452 register.
nherriot 0:bcf2aa85d7f9 212 *
ashleymills 12:172540ff6b8b 213 * @param addr The internal register address
ashleymills 12:172540ff6b8b 214 * @param data Data to write
nherriot 0:bcf2aa85d7f9 215 */
ashleymills 11:dfd1e0afcb7b 216 int writeRegister(char addr, char data);
ashleymills 12:172540ff6b8b 217
ashleymills 12:172540ff6b8b 218 /**
ashleymills 12:172540ff6b8b 219 * Write a data buffer to the specified MMA8452 register.
ashleymills 12:172540ff6b8b 220 *
ashleymills 12:172540ff6b8b 221 * @param addr The internal register address
ashleymills 12:172540ff6b8b 222 * @param data Data buffer to write
ashleymills 12:172540ff6b8b 223 * @param nbytes The length of the data buffer to write
ashleymills 12:172540ff6b8b 224 */
ashleymills 11:dfd1e0afcb7b 225 int writeRegister(char addr, char *data, int nbytes);
ashleymills 11:dfd1e0afcb7b 226
ashleymills 11:dfd1e0afcb7b 227 int logicalANDRegister(char addr, char mask);
ashleymills 11:dfd1e0afcb7b 228 int logicalORRegister(char addr, char mask);
ashleymills 11:dfd1e0afcb7b 229 int logicalXORRegister(char addr, char mask);
nherriot 0:bcf2aa85d7f9 230
ashleymills 11:dfd1e0afcb7b 231 int setDynamicRange(DynamicRange range, int toggleActivation=1);
ashleymills 11:dfd1e0afcb7b 232 int setBitDepth(BitDepth depth, int toggleActivation=1);
ashleymills 11:dfd1e0afcb7b 233 int setDataRate(DataRateHz dataRate, int toggleActivation=1);
ashleymills 15:7620a11149b8 234
ashleymills 11:dfd1e0afcb7b 235 DynamicRange getDynamicRange();
ashleymills 11:dfd1e0afcb7b 236 DataRateHz getDataRate();
ashleymills 12:172540ff6b8b 237 BitDepth getBitDepth();
ashleymills 11:dfd1e0afcb7b 238
ashleymills 17:6e4232c421c0 239 #ifdef MMA8452_DEBUG
ashleymills 11:dfd1e0afcb7b 240 void debugRegister(char reg);
ashleymills 17:6e4232c421c0 241 #endif
nherriot 0:bcf2aa85d7f9 242
nherriot 0:bcf2aa85d7f9 243 private:
ashleymills 10:ca9ba7ad4e94 244 int readRaw(char src, char *dst, int len);
ashleymills 11:dfd1e0afcb7b 245 int maskAndApplyRegister(char reg, char mask, char value, int toggleActivation);
ashleymills 13:4bd8b4cd479d 246
ashleymills 13:4bd8b4cd479d 247 int twelveBitToSigned(char *buf);
ashleymills 13:4bd8b4cd479d 248 int eightBitToSigned(char *buf);
ashleymills 13:4bd8b4cd479d 249 double convertCountToGravity(int count, int countsPerG);
ashleymills 16:d6dde2318edc 250 char getMaskedRegister(int reg, char mask);
ashleymills 8:89272163f395 251
ashleymills 11:dfd1e0afcb7b 252 I2C _i2c;
ashleymills 11:dfd1e0afcb7b 253 int _frequency;
ashleymills 6:f6bde04bf8be 254 int _readAddress;
ashleymills 6:f6bde04bf8be 255 int _writeAddress;
ashleymills 12:172540ff6b8b 256
ashleymills 12:172540ff6b8b 257 BitDepth _bitDepth;
ashleymills 13:4bd8b4cd479d 258 DynamicRange _dynamicRange;
nherriot 0:bcf2aa85d7f9 259
ashleymills 15:7620a11149b8 260 };