ISSDK  1.7
IoT Sensing Software Development Kit
driver_MMA8652.c
Go to the documentation of this file.
1 /*
2  * The Clear BSD License
3  * Copyright (c) 2015 - 2016, Freescale Semiconductor, Inc.
4  * Copyright 2016-2017 NXP
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without modification,
8  * are permitted (subject to the limitations in the disclaimer below) provided
9  * that the following conditions are met:
10  *
11  * o Redistributions of source code must retain the above copyright notice, this list
12  * of conditions and the following disclaimer.
13  *
14  * o Redistributions in binary form must reproduce the above copyright notice, this
15  * list of conditions and the following disclaimer in the documentation and/or
16  * other materials provided with the distribution.
17  *
18  * o Neither the name of the copyright holder nor the names of its
19  * contributors may be used to endorse or promote products derived from this
20  * software without specific prior written permission.
21  *
22  * NO EXPRESS OR IMPLIED LICENSES TO ANY PARTY'S PATENT RIGHTS ARE GRANTED BY THIS LICENSE.
23  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
24  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
25  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
26  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
27  * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
28  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
29  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
30  * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
31  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
32  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
33  */
34 
35 /*! \file driver_MMA8652.c
36  \brief Provides init() and read() functions for the MMA8652 3-axis accel family.
37 
38 1G*/
39 
40 #include "board.h" // generated by Kinetis Expert. Long term - merge sensor_board.h into this file
41 #include "sensor_fusion.h" // Sensor fusion structures and types
42 #include "sensor_drv.h"
43 #include "sensor_io_i2c.h" // Required for registerreadlist_t / registerwritelist_t declarations
44 #include "mma865x.h"
45 #include "drivers.h" // Device specific drivers supplied by NXP (can be replaced with user drivers)
46 #define MMA8652_COUNTSPERG 8192.0
47 #define MMA8652_ACCEL_FIFO_SIZE 32
48 
49 #if F_USING_ACCEL
50 
51 // Command definition to read the WHO_AM_I value.
52 const registerreadlist_t MMA8652_WHO_AM_I_READ[] =
53 {
54  { .readFrom = MMA865x_WHO_AM_I, .numBytes = 1 }, __END_READ_DATA__
55 };
56 
57 // Command definition to read the number of entries in the accel FIFO.
58 const registerreadlist_t MMA8652_F_STATUS_READ[] =
59 {
60  { .readFrom = MMA865x_STATUS, .numBytes = 1 }, __END_READ_DATA__
61 };
62 
63 // Command definition to read the number of entries in the accel FIFO.
64 registerreadlist_t MMA8652_DATA_READ[] =
65 {
66  { .readFrom = MMA865x_OUT_X_MSB, .numBytes = 6 }, __END_READ_DATA__
67 };
68 
69 // Each entry in a RegisterWriteList is composed of: register address, value to write, bit-mask to apply to write (0 enables)
70 const registerwritelist_t MMA8652_Initialization[] =
71 {
72  // write 0000 0000 = 0x00 to CTRL_REG1 to place MMA8652 into standby
73  // [7-1] = 0000 000
74  // [0]: active=0
75  { MMA865x_CTRL_REG1, 0x00, 0x00 },
76 
77  // write 0000 0000 = 0x00 to CTRL_REG1 to place MMA8652 into standby
78  // [7-1] = 0000 000
79  // [0]: active=0
80  { MMA865x_F_SETUP, 0x40, 0x00 },
81 
82  // OK: write 0000 0001 = 0x01 to XYZ_DATA_CFG register to set g range
83  // [7-5]: reserved=000
84  // [4]: HPF_OUT=0
85  // [3-2]: reserved=00
86  // [1-0]: FS=01 for +/-4g: 512 counts / g = 8192 counts / g after 4 bit left shift
87  { MMA865x_XYZ_DATA_CFG, 0x01, 0x00 },
88 
89  // OK: write 0000 0010 = 0x02 to CTRL_REG2 to set MODS bits
90  // [7]: ST=0: self test disabled
91  // [6]: RST=0: reset disabled
92  // [5]: unused
93  // [4-3]: SMODS=00
94  // [2]: SLPE=0: auto sleep disabled
95  // [1-0]: mods=10 for high resolution (maximum over sampling)
96  { MMA865x_CTRL_REG2, 0x02, 0x00 },
97 
98  // write 00XX X001 to CTRL_REG1 to set data rate and exit standby
99  // [7-6]: aslp_rate=00
100  // [5-3]: dr=111 for 1.56Hz data rate giving 0x39
101  // [5-3]: dr=110 for 6.25Hz data rate giving 0x31
102  // [5-3]: dr=101 for 12.5Hz data rate giving 0x29
103  // [5-3]: dr=100 for 50Hz data rate giving 0x21
104  // [5-3]: dr=011 for 100Hz data rate giving 0x19
105  // [5-3]: dr=010 for 200Hz data rate giving 0x11
106  // [5-3]: dr=001 for 400Hz data rate giving 0x09
107  // [5-3]: dr=000 for 800Hz data rate giving 0x01
108  // [2]: unused=0
109  // [1]: f_read=0 for normal 16 bit reads
110  // [0]: active=1 to take the part out of standby and enable sampling
111 #if (ACCEL_ODR_HZ <= 1)
112  { MMA865x_CTRL_REG1, 0x39, 0x00 }, // select 1.5625Hz ODR,
113 #elif (ACCEL_ODR_HZ <= 6)
114  { MMA865x_CTRL_REG1, 0x31, 0x00 }, // select 6.25Hz ODR
115 #elif (ACCEL_ODR_HZ <= 12)
116  { MMA865x_CTRL_REG1, 0x29, 0x00 }, // select 12.5Hz ODR
117 #elif (ACCEL_ODR_HZ <= 50)
118  { MMA865x_CTRL_REG1, 0x21, 0x00 }, // select 50Hz ODR
119 #elif (ACCEL_ODR_HZ <= 100)
120  { MMA865x_CTRL_REG1, 0x19, 0x00 }, // select 100Hz ODR
121 #elif (ACCEL_ODR_HZ <= 200)
122  { MMA865x_CTRL_REG1, 0x11, 0x00 }, // select 200Hz ODR
123 #elif (ACCEL_ODR_HZ <= 400)
124  { MMA865x_CTRL_REG1, 0x09, 0x00 }, // select 400Hz ODR
125 #else
126  { MMA865x_CTRL_REG1, 0x01, 0x00 }, // select 800Hz ODR
127 #endif
129 };
130 
131 // All sensor drivers and initialization functions have the same prototype.
132 // sfg is a pointer to the master "global" sensor fusion structure.
133 // sensor = pointer to linked list element used by the sensor fusion subsystem to specify required sensors
134 
135 // sfg = pointer to top level (generally global) data structure for sensor fusion
136 int8_t MMA8652_Init(struct PhysicalSensor *sensor, SensorFusionGlobals *sfg)
137 {
138  int32_t status;
139  uint8_t reg;
140  status = Register_I2C_Read(sensor->bus_driver, &sensor->deviceInfo, sensor->addr, MMA865x_WHO_AM_I, 1, &reg);
141  if (status==SENSOR_ERROR_NONE) {
142  sfg->Accel.iWhoAmI = reg;
143  if (reg != MMA8652_WHOAMI_VALUE) return(SENSOR_ERROR_INIT);
144  } else {
145  return(status);
146  }
147 
148  status = Sensor_I2C_Write(sensor->bus_driver, &sensor->deviceInfo, sensor->addr, MMA8652_Initialization );
149  // Stash some needed constants in the SF data structure for this sensor
150  sfg->Accel.iCountsPerg = MMA8652_COUNTSPERG;
151  sfg->Accel.fgPerCount = 1.0F / MMA8652_COUNTSPERG;
152  sfg->Accel.fgPerCount = 1.0F / MMA8652_COUNTSPERG;
153 
154  sensor->isInitialized = F_USING_ACCEL;
155  sfg->Accel.isEnabled = true;
156 
157  return (status);
158 }
159 
160 int8_t MMA8652_Read(struct PhysicalSensor *sensor, SensorFusionGlobals *sfg)
161 {
162  uint8_t I2C_Buffer[6 * MMA8652_ACCEL_FIFO_SIZE]; // I2C read buffer
163  int8_t status; // I2C transaction status
164  int8_t j; // scratch
165  uint8_t sensor_fifo_count = 1;
166  int16_t sample[3];
167 
168  if(sensor->isInitialized != F_USING_ACCEL)
169  {
170  return SENSOR_ERROR_INIT;
171  }
172 
173  // read the F_STATUS register (mapped to STATUS) and extract number of measurements available (lower 6 bits)
174  status = Sensor_I2C_Read(sensor->bus_driver, &sensor->deviceInfo, sensor->addr, MMA8652_F_STATUS_READ, I2C_Buffer );
175  if (status==SENSOR_ERROR_NONE) {
176  sensor_fifo_count = I2C_Buffer[0] & 0x3F;
177  // return if there are no measurements in the sensor FIFO.
178  // this will only occur when the FAST_LOOP_HZ equals or exceeds ACCEL_ODR_HZ
179  if (sensor_fifo_count == 0) return SENSOR_ERROR_READ;
180  } else {
181  return(status);
182  }
183 
184  MMA8652_DATA_READ[0].readFrom = MMA865x_OUT_X_MSB;
185  MMA8652_DATA_READ[0].numBytes = 6 * sensor_fifo_count;
186  status = Sensor_I2C_Read(sensor->bus_driver, &sensor->deviceInfo, sensor->addr, MMA8652_DATA_READ, I2C_Buffer );
187 
188  if (status==SENSOR_ERROR_NONE) {
189  for (j = 0; j < sensor_fifo_count; j++)
190  {
191  sample[CHX] = (I2C_Buffer[6 * j ] << 8) | I2C_Buffer[6 * j + 1];
192  sample[CHY] = (I2C_Buffer[6 * j + 2] << 8) | I2C_Buffer[6 * j + 3];
193  sample[CHZ] = (I2C_Buffer[6 * j + 4] << 8) | I2C_Buffer[6 * j + 5];
194  conditionSample(sample); // truncate negative values to -32767
195  addToFifo((union FifoSensor*) &(sfg->Accel), ACCEL_FIFO_SIZE, sample);
196  }
197  }
198 
199  return (status);
200 }
201 
202 
203 // Each entry in a RegisterWriteList is composed of: register address, value to write, bit-mask to apply to write (0 enables)
204 const registerwritelist_t MMA8652_IDLE[] =
205 {
206  // Set ACTIVE = other bits unchanged
207  { MMA865x_CTRL_REG1, 0x00, 0x01 },
209 };
210 
211 // MMA8652_Idle places the sensor into SLEEP mode (wakeup time = 2/ODR+1ms)
212 int8_t MMA8652_Idle(struct PhysicalSensor *sensor, SensorFusionGlobals *sfg)
213 {
214  int32_t status;
215  if(sensor->isInitialized == F_USING_ACCEL) {
216  status = Sensor_I2C_Write(sensor->bus_driver, &sensor->deviceInfo, sensor->addr, MMA8652_IDLE );
217  sensor->isInitialized = 0;
218  sfg->Accel.isEnabled = false;
219  } else {
220  return SENSOR_ERROR_INIT;
221  }
222  return status;
223 }
224 
225 #endif // F_USING_ACCEL
The sensor_drv.h file contains sensor state and error definitions.
#define __END_WRITE_DATA__
Definition: sensor_drv.h:71
void * bus_driver
should be of type (ARM_DRIVER_I2C* for I2C-based sensors, ARM_DRIVER_SPI* for SPI) ...
#define CHY
Used to access Y-channel entries in various data data structures.
Definition: sensor_fusion.h:87
SensorFusionGlobals sfg
int32_t status
The mma865x.h contains the MMA865xFC Digital Accelerometer Sensor register definitions, access macros, and device access functions.
int32_t Sensor_I2C_Read(ARM_DRIVER_I2C *pCommDrv, registerDeviceInfo_t *devInfo, uint16_t slaveAddress, const registerreadlist_t *pReadList, uint8_t *pOutBuffer)
Read register data from a sensor.
#define ACCEL_FIFO_SIZE
FXOS8700 (accel), MMA8652, FXLS8952 all have 32 element FIFO.
#define __END_READ_DATA__
Definition: sensor_drv.h:77
The top level fusion structure.
int8_t MMA8652_Read(struct PhysicalSensor *sensor, SensorFusionGlobals *sfg)
#define MMA8652_COUNTSPERG
#define MMA8652_WHOAMI_VALUE
Definition: mma865x.h:65
uint16_t addr
I2C address if applicable.
void addToFifo(union FifoSensor *sensor, uint16_t maxFifoSize, int16_t sample[3])
addToFifo is called from within sensor driver read functions
The sensor_fusion.h file implements the top level programming interface.
The FifoSensor union allows us to use common pointers for Accel, Mag & Gyro logical sensor structures...
#define CHZ
Used to access Z-channel entries in various data data structures.
Definition: sensor_fusion.h:88
uint16_t isInitialized
Bitfields to indicate sensor is active (use SensorBitFields from build.h)
registerDeviceInfo_t deviceInfo
I2C device context.
Provides function prototypes for driver level interfaces.
int8_t MMA8652_Init(struct PhysicalSensor *sensor, SensorFusionGlobals *sfg)
#define CHX
Used to access X-channel entries in various data data structures.
Definition: sensor_fusion.h:86
void conditionSample(int16_t sample[3])
conditionSample ensures that we never encounter the maximum negative two&#39;s complement value for a 16-...
typedef int32_t(DATA_FORMAT_Append_t))(void *pData
The interface function to append the data on the formated stream.
#define MMA8652_ACCEL_FIFO_SIZE
int32_t Register_I2C_Read(ARM_DRIVER_I2C *pCommDrv, registerDeviceInfo_t *devInfo, uint16_t slaveAddress, uint8_t offset, uint8_t length, uint8_t *pOutBuffer)
The interface function to read a sensor register.
This structure defines the Write command List.
Definition: sensor_drv.h:94
This structure defines the Read command List.
Definition: sensor_drv.h:104
#define F_USING_ACCEL
nominally 0x0001 if an accelerometer is to be used, 0x0000 otherwise
An instance of PhysicalSensor structure type should be allocated for each physical sensors (combo dev...
int8_t MMA8652_Idle(struct PhysicalSensor *sensor, SensorFusionGlobals *sfg)
The sensor_io_i2c.h file declares low-level interface functions for reading and writing sensor regist...
int32_t Sensor_I2C_Write(ARM_DRIVER_I2C *pCommDrv, registerDeviceInfo_t *devInfo, uint16_t slaveAddress, const registerwritelist_t *pRegWriteList)
Write register data to a sensor.
Definition: sensor_io_i2c.c:97