2017-03-20 22:52:21 +01:00
|
|
|
/*
|
2018-05-25 13:37:58 +02:00
|
|
|
* Copyright (C) 2017-2018 Mesotic SAS
|
2017-03-20 22:52:21 +01:00
|
|
|
*
|
|
|
|
* 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_adxl345
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Device driver implementation for the ADXL345 accelerometer (i2c only)
|
|
|
|
*
|
|
|
|
* @author Dylan Laduranty <dylan.laduranty@mesotic.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "assert.h"
|
|
|
|
#include "periph/i2c.h"
|
2018-10-05 14:35:10 +02:00
|
|
|
#include "byteorder.h"
|
2017-03-20 22:52:21 +01:00
|
|
|
#include "adxl345.h"
|
|
|
|
#include "adxl345_regs.h"
|
2018-10-01 20:58:51 +02:00
|
|
|
#include "adxl345_params.h"
|
2017-03-20 22:52:21 +01:00
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2018-10-01 20:58:51 +02:00
|
|
|
#define ADXL345_BUS (dev->params->i2c)
|
|
|
|
#define ADXL345_ADDR (dev->params->addr)
|
2017-03-20 22:52:21 +01:00
|
|
|
|
2018-10-15 15:00:04 +02:00
|
|
|
#define ADXL345_PARAM_SCALE_FACTOR (4)
|
|
|
|
|
2017-06-26 16:03:47 +02:00
|
|
|
int adxl345_init(adxl345_t *dev, const adxl345_params_t* params)
|
2017-03-20 22:52:21 +01:00
|
|
|
{
|
|
|
|
uint8_t reg;
|
|
|
|
|
|
|
|
assert(dev && params);
|
|
|
|
|
|
|
|
/* get device descriptor */
|
2017-03-27 22:23:40 +02:00
|
|
|
dev->params = (adxl345_params_t*)params;
|
|
|
|
|
|
|
|
/* get scale_factor from full_res and range parameters */
|
2018-10-01 20:58:51 +02:00
|
|
|
dev->scale_factor = (dev->params->full_res ? ADXL345_PARAM_SCALE_FACTOR : (4 << dev->params->range));
|
2017-03-20 22:52:21 +01:00
|
|
|
|
|
|
|
/* Acquire exclusive access */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_acquire(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
|
|
|
|
/* test if the target device responds */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_read_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_CHIP_ID_REG, ®, 0);
|
|
|
|
if (reg != ADXL345_CHIP_ID) {
|
|
|
|
i2c_release(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
DEBUG("[adxl345] init - error: invalid id value [0x%02x]\n", (int)reg);
|
|
|
|
return ADXL345_NODEV;
|
|
|
|
}
|
|
|
|
/* configure the user offset */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_regs(ADXL345_BUS, ADXL345_ADDR, ADXL345_OFFSET_X, dev->params->offset, 3, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Basic device setup */
|
2018-10-01 20:58:51 +02:00
|
|
|
reg = (dev->params->full_res ? ADXL345_FULL_RES : 0);
|
|
|
|
reg |= dev->params->range;
|
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_DATA_FORMAT, reg, 0);
|
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_BW_RATE, dev->params->rate, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Put device in measure mode */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_POWER_CTL, ADXL345_MEASURE_BIT, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
|
|
|
|
/* Release the bus */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_release(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
|
|
|
|
DEBUG("[adxl345] init: successful\n");
|
|
|
|
|
|
|
|
return ADXL345_OK;
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void adxl345_read(const adxl345_t *dev, adxl345_data_t *data)
|
2017-03-20 22:52:21 +01:00
|
|
|
{
|
2018-10-01 18:04:13 +02:00
|
|
|
int16_t result[3];
|
2017-03-20 22:52:21 +01:00
|
|
|
|
|
|
|
assert(dev && data);
|
|
|
|
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_acquire(ADXL345_BUS);
|
|
|
|
i2c_read_regs(ADXL345_BUS, ADXL345_ADDR, ADXL345_DATA_X0, (void *)result, 6, 0);
|
|
|
|
i2c_release(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
|
2018-10-05 14:35:10 +02:00
|
|
|
/* ADXL345 returns value in little endian, so swap is needed on big endian
|
|
|
|
* platforms. See Analog Devices ADXL345 datasheet page 27. */
|
|
|
|
#if __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__
|
2018-10-01 18:04:13 +02:00
|
|
|
for (int i = 0; i < 3; i++) {
|
2018-10-05 14:35:10 +02:00
|
|
|
result[i] = byteorder_swaps((uint16_t)result[i]);
|
2018-10-01 18:04:13 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
data->x = result[0] * dev->scale_factor;
|
|
|
|
data->y = result[1] * dev->scale_factor;
|
|
|
|
data->z = result[2] * dev->scale_factor;
|
2017-03-20 22:52:21 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void adxl345_set_interrupt(const adxl345_t *dev)
|
2017-03-20 22:52:21 +01:00
|
|
|
{
|
|
|
|
assert(dev);
|
|
|
|
|
|
|
|
DEBUG("[adxl345] Update interruptions configuration\n");
|
|
|
|
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_acquire(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set threshold */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_THRESH_TAP, dev->interrupt.thres_tap, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set Map */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_INT_MAP, dev->interrupt.map, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set Duration */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_TAP_DUR, dev->interrupt.thres_dur, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Enable axes */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_TAP_AXES, dev->interrupt.tap_axes, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set source */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_INT_SOURCE, dev->interrupt.source, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set latent threshold */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_TAP_LAT, dev->interrupt.thres_latent, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set window threshold */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_TAP_WIN, dev->interrupt.thres_window, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set activity threshold */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_THRESH_ACT, dev->interrupt.thres_act, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set inactivity threshold */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_THRESH_INACT, dev->interrupt.thres_inact, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set inactivity time */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_TIME_INACT, dev->interrupt.time_inact, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set free-fall threshold */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_THRESH_FF, dev->interrupt.thres_ff, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set free-fall time */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_TIME_FF, dev->interrupt.time_ff, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Set axis control */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_ACT_INACT_CTL, dev->interrupt.act_inact, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
/* Enable interrupt */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_INT_ENABLE, dev->interrupt.enable, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
|
|
|
|
/* Release the bus */
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_release(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void adxl345_set_measure(const adxl345_t *dev)
|
2017-03-20 22:52:21 +01:00
|
|
|
{
|
|
|
|
uint8_t reg;
|
|
|
|
|
|
|
|
assert(dev);
|
|
|
|
|
|
|
|
DEBUG("[adxl345] set device to measure mode\n");
|
|
|
|
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_acquire(ADXL345_BUS);
|
|
|
|
i2c_read_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_POWER_CTL, ®, 0);
|
|
|
|
reg |= ADXL345_MEASURE_BIT;
|
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_POWER_CTL, reg, 0);
|
|
|
|
i2c_release(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void adxl345_set_standby(const adxl345_t *dev)
|
2017-03-20 22:52:21 +01:00
|
|
|
{
|
|
|
|
uint8_t reg;
|
|
|
|
|
|
|
|
assert(dev);
|
|
|
|
|
|
|
|
DEBUG("[adxl345] set device to standby mode\n");
|
|
|
|
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_acquire(ADXL345_BUS);
|
|
|
|
i2c_read_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_POWER_CTL, ®, 0);
|
|
|
|
reg &= ~ADXL345_MEASURE_BIT;
|
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_POWER_CTL, reg, 0);
|
|
|
|
i2c_release(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void adxl345_set_sleep(const adxl345_t *dev)
|
2017-03-20 22:52:21 +01:00
|
|
|
{
|
|
|
|
uint8_t reg;
|
|
|
|
|
|
|
|
assert(dev);
|
|
|
|
|
|
|
|
DEBUG("[adxl345] set device to sleep mode\n");
|
|
|
|
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_acquire(ADXL345_BUS);
|
|
|
|
i2c_read_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_POWER_CTL, ®, 0);
|
|
|
|
reg |= ADXL345_SLEEP_BIT;
|
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_POWER_CTL, reg, 0);
|
|
|
|
i2c_release(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void adxl345_set_autosleep(const adxl345_t *dev)
|
2017-03-20 22:52:21 +01:00
|
|
|
{
|
|
|
|
uint8_t reg;
|
|
|
|
|
|
|
|
assert(dev);
|
|
|
|
|
|
|
|
DEBUG("[adxl345] set device to autosleep mode\n");
|
|
|
|
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_acquire(ADXL345_BUS);
|
|
|
|
i2c_read_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_POWER_CTL, ®, 0);
|
|
|
|
reg |= ADXL345_AUTOSLEEP_BIT;
|
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_POWER_CTL, reg, 0);
|
|
|
|
i2c_release(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void adxl345_set_bandwidth_rate(const adxl345_t *dev, uint8_t bw_rate)
|
2017-03-20 22:52:21 +01:00
|
|
|
{
|
|
|
|
uint8_t reg;
|
|
|
|
|
|
|
|
assert(dev);
|
|
|
|
|
|
|
|
DEBUG("[adxl345] set device rate to %d Hz\n", (int)bw_rate);
|
|
|
|
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_acquire(ADXL345_BUS);
|
|
|
|
i2c_read_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_BW_RATE, ®, 0);
|
2017-03-20 22:52:21 +01:00
|
|
|
reg |= bw_rate;
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_BW_RATE, reg, 0);
|
|
|
|
i2c_release(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void adxl345_set_fifo_mode(const adxl345_t *dev, uint8_t mode,
|
2017-03-20 22:52:21 +01:00
|
|
|
uint8_t output, uint8_t value)
|
|
|
|
{
|
|
|
|
uint8_t reg;
|
|
|
|
|
|
|
|
assert(dev);
|
|
|
|
|
|
|
|
DEBUG("[adxl345] set fifo mode to %d, output trigger to %d and trigger "
|
|
|
|
"value to :%d\n", (int)mode, (int)output, (int)value);
|
|
|
|
|
2018-10-01 20:58:51 +02:00
|
|
|
i2c_acquire(ADXL345_BUS);
|
|
|
|
reg = ((mode << ADXL345_FIFO_MODE_POS) | (output << ADXL345_FIFO_TRIGGER_POS) | value);
|
|
|
|
i2c_write_reg(ADXL345_BUS, ADXL345_ADDR, ADXL345_FIFO_CTL, reg, 0);
|
|
|
|
i2c_release(ADXL345_BUS);
|
2017-03-20 22:52:21 +01:00
|
|
|
}
|