2014-12-01 14:04:12 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 PHYTEC Messtechnik GmbH
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup drivers_mma8652
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Driver for the Freescale MMA8652 accelerometer.
|
|
|
|
*
|
|
|
|
* @author Johann Fischer <j.fischer@phytec.de>
|
2015-01-28 14:33:34 +01:00
|
|
|
* @author Peter Kietzmann <peter.kietzmann@haw-hamburg.de>
|
2014-12-01 14:04:12 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include "periph/i2c.h"
|
|
|
|
#include "mma8652.h"
|
|
|
|
#include "mma8652_reg.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
#define I2C_SPEED I2C_SPEED_FAST
|
|
|
|
|
|
|
|
int mma8652_test(mma8652_t *dev)
|
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-12-01 14:04:12 +01:00
|
|
|
|
2015-01-28 14:33:34 +01:00
|
|
|
/* Acquire exclusive access to the bus. */
|
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
if (i2c_read_regs(dev->i2c, dev->addr, MMA8652_WHO_AM_I, ®, 1) != 1) {
|
2015-01-28 14:33:34 +01:00
|
|
|
/* Release the bus for other threads. */
|
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
if (reg != MMA8652_ID) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mma8652_init(mma8652_t *dev, i2c_t i2c, uint8_t address, uint8_t dr, uint8_t range)
|
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
/* write device descriptor */
|
|
|
|
dev->i2c = i2c;
|
|
|
|
dev->addr = address;
|
|
|
|
dev->initialized = false;
|
|
|
|
|
|
|
|
if (dr > MMA8652_DATARATE_1HZ56 || range > MMA8652_FS_RANGE_8G) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
/* initialize the I2C bus */
|
|
|
|
if (i2c_init_master(i2c, I2C_SPEED) < 0) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
return -2;
|
|
|
|
}
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
if (mma8652_test(dev)) {
|
|
|
|
return -3;
|
|
|
|
}
|
|
|
|
|
2016-02-12 16:09:59 +01:00
|
|
|
if (mma8652_set_standby(dev) < 0) {
|
|
|
|
return -4;
|
|
|
|
}
|
|
|
|
|
2014-12-01 14:04:12 +01:00
|
|
|
reg = MMA8652_XYZ_DATA_CFG_FS(range);
|
|
|
|
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
if (i2c_write_regs(dev->i2c, dev->addr, MMA8652_XYZ_DATA_CFG, ®, 1) != 1) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2016-02-12 16:09:59 +01:00
|
|
|
return -5;
|
2014-12-01 14:04:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
reg = MMA8652_CTRL_REG1_DR(dr);
|
|
|
|
|
|
|
|
if (i2c_write_regs(dev->i2c, dev->addr, MMA8652_CTRL_REG1, ®, 1) != 1) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2016-02-12 16:09:59 +01:00
|
|
|
return -5;
|
2014-12-01 14:04:12 +01:00
|
|
|
}
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
dev->initialized = true;
|
2016-02-12 16:25:15 +01:00
|
|
|
dev->scale = 1024 >> range;
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mma8652_set_user_offset(mma8652_t *dev, int8_t x, int8_t y, int8_t z)
|
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t buf[3];
|
2014-12-01 14:04:12 +01:00
|
|
|
|
2016-09-30 23:01:46 +02:00
|
|
|
buf[0] = (uint8_t)x;
|
|
|
|
buf[1] = (uint8_t)y;
|
|
|
|
buf[2] = (uint8_t)z;
|
2014-12-01 14:04:12 +01:00
|
|
|
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
if (i2c_write_regs(dev->i2c, dev->addr, MMA8652_OFF_X, buf, 3) != 3) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mma8652_reset(mma8652_t *dev)
|
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
dev->initialized = false;
|
|
|
|
reg = MMA8652_CTRL_REG2_RST;
|
|
|
|
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
if (i2c_write_regs(dev->i2c, dev->addr, MMA8652_CTRL_REG2, ®, 1) != 1) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mma8652_set_active(mma8652_t *dev)
|
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
if (dev->initialized == false) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
if (i2c_read_regs(dev->i2c, dev->addr, MMA8652_CTRL_REG1, ®, 1) != 1) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
reg |= MMA8652_CTRL_REG1_ACTIVE;
|
|
|
|
|
|
|
|
if (i2c_write_regs(dev->i2c, dev->addr, MMA8652_CTRL_REG1, ®, 1) != 1) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mma8652_set_standby(mma8652_t *dev)
|
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-12-01 14:04:12 +01:00
|
|
|
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
if (i2c_read_regs(dev->i2c, dev->addr, MMA8652_CTRL_REG1, ®, 1) != 1) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
reg &= ~MMA8652_CTRL_REG1_ACTIVE;
|
|
|
|
|
|
|
|
if (i2c_write_regs(dev->i2c, dev->addr, MMA8652_CTRL_REG1, ®, 1) != 1) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mma8652_is_ready(mma8652_t *dev)
|
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
if (dev->initialized == false) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
if (i2c_read_regs(dev->i2c, dev->addr, MMA8652_STATUS, ®, 1) != 1) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
return reg & MMA8652_STATUS_ZYXDR;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mma8652_read(mma8652_t *dev, int16_t *x, int16_t *y, int16_t *z, uint8_t *status)
|
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t buf[7];
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
if (dev->initialized == false) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
if (i2c_read_regs(dev->i2c, dev->addr, MMA8652_STATUS, buf, 7) != 7) {
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2015-01-28 14:33:34 +01:00
|
|
|
i2c_release(dev->i2c);
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
*status = buf[0];
|
2016-02-12 16:25:15 +01:00
|
|
|
*x = (int32_t)((int16_t)(((int16_t)buf[1] << 8) | buf[2]) >> 4) * 1000 / dev->scale;
|
|
|
|
*y = (int32_t)((int16_t)(((int16_t)buf[3] << 8) | buf[4]) >> 4) * 1000 / dev->scale;
|
|
|
|
*z = (int32_t)((int16_t)(((int16_t)buf[5] << 8) | buf[6]) >> 4) * 1000 / dev->scale;
|
2014-12-01 14:04:12 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|