2014-11-28 16:26:13 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 PHYTEC Messtechnik GmbH
|
2017-04-22 20:54:39 +02:00
|
|
|
* 2017 HAW Hamburg
|
2014-11-28 16:26:13 +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_mpl3115a2
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Driver for the Freescale MPL3115A2 Sensor.
|
|
|
|
*
|
|
|
|
* @author Johann Fischer <j.fischer@phytec.de>
|
2015-01-28 13:32:53 +01:00
|
|
|
* @author Peter Kietzmann <peter.kietzmann@haw-hamburg.de>
|
2017-04-22 20:54:39 +02:00
|
|
|
* @author Sebastian Meiling <s@mlng.net>
|
2014-11-28 16:26:13 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdbool.h>
|
2017-04-22 20:54:39 +02:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "log.h"
|
2014-11-28 16:26:13 +01:00
|
|
|
#include "periph/i2c.h"
|
2017-04-22 20:54:39 +02:00
|
|
|
|
2014-11-28 16:26:13 +01:00
|
|
|
#include "mpl3115a2.h"
|
|
|
|
#include "mpl3115a2_reg.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
#define I2C_SPEED I2C_SPEED_FAST
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
#define BUS (dev->params.i2c)
|
|
|
|
#define ADDR (dev->params.addr)
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
int mpl3115a2_init(mpl3115a2_t *dev, const mpl3115a2_params_t *params)
|
2014-11-28 16:26:13 +01:00
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
assert(dev);
|
|
|
|
assert(params);
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
/* write device descriptor */
|
|
|
|
memcpy(dev, params, sizeof(mpl3115a2_params_t));
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_acquire(BUS);
|
2014-11-28 16:26:13 +01:00
|
|
|
/* initialize the I2C bus */
|
2017-04-22 20:54:39 +02:00
|
|
|
if (i2c_init_master(BUS, I2C_SPEED) < 0) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
LOG_ERROR("mpl3115a2_init: failed to init I2C!\n");
|
|
|
|
return -MPL3115A2_ERROR_I2C;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
2017-04-22 20:54:39 +02:00
|
|
|
/* test device */
|
|
|
|
if (i2c_read_regs(BUS, ADDR, MPL3115A2_WHO_AM_I, ®, 1) != 1) {
|
|
|
|
/* Release the bus for other threads. */
|
|
|
|
i2c_release(BUS);
|
|
|
|
LOG_ERROR("mpl3115a2_init: I2C error!\n");
|
|
|
|
return -MPL3115A2_ERROR_I2C;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
2017-04-22 20:54:39 +02:00
|
|
|
if (reg != MPL3115A2_ID) {
|
|
|
|
LOG_ERROR("mpl3115a2_init: invalid WHO_AM_I value (0x%02x)!\n", (int)reg);
|
|
|
|
return -MPL3115A2_ERROR_DEV;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
2017-04-22 20:54:39 +02:00
|
|
|
/* set sample rate */
|
|
|
|
reg = MPL3115A2_CTRL_REG1_OS(dev->params.ratio);
|
|
|
|
if (i2c_write_regs(BUS, ADDR, MPL3115A2_CTRL_REG1, ®, 1) != 1) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
LOG_ERROR("mpl3115a2_init: failed to set sample rate!\n");
|
|
|
|
return -MPL3115A2_ERROR_CNF;
|
|
|
|
}
|
|
|
|
/* configure device */
|
|
|
|
reg = MPL3115A2_PT_DATA_CFG_TDEFE |
|
|
|
|
MPL3115A2_PT_DATA_CFG_PDEFE |
|
|
|
|
MPL3115A2_PT_DATA_CFG_DREM;
|
|
|
|
if (i2c_write_regs(BUS, ADDR, MPL3115A2_PT_DATA_CFG, ®, 1) != 1) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
LOG_ERROR("mpl3115a2_init: config failure!\n");
|
|
|
|
return -MPL3115A2_ERROR_CNF;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_release(BUS);
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
return MPL3115A2_OK;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
int mpl3115a2_reset(const mpl3115a2_t *dev)
|
2014-11-28 16:26:13 +01:00
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_acquire(BUS);
|
2014-11-28 16:26:13 +01:00
|
|
|
reg = MPL3115A2_CTRL_REG1_RST;
|
2017-04-22 20:54:39 +02:00
|
|
|
if (i2c_write_regs(BUS, ADDR, MPL3115A2_CTRL_REG1, ®, 1) != 1) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
LOG_ERROR("mpl3115a2_reset: failed!\n");
|
|
|
|
return -MPL3115A2_ERROR_I2C;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_release(BUS);
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
return MPL3115A2_OK;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int mpl3115a2_set_active(const mpl3115a2_t *dev)
|
2014-11-28 16:26:13 +01:00
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_acquire(BUS);
|
|
|
|
if (i2c_read_regs(BUS, ADDR, MPL3115A2_CTRL_REG1, ®, 1) != 1) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
return -MPL3115A2_ERROR_I2C;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
reg |= MPL3115A2_CTRL_REG1_SBYB;
|
2017-04-22 20:54:39 +02:00
|
|
|
if (i2c_write_regs(BUS, ADDR, MPL3115A2_CTRL_REG1, ®, 1) != 1) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
return -MPL3115A2_ERROR_I2C;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_release(BUS);
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
return MPL3115A2_OK;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int mpl3115a2_set_standby(const mpl3115a2_t *dev)
|
2014-11-28 16:26:13 +01:00
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_acquire(BUS);
|
|
|
|
if (i2c_read_regs(BUS, ADDR, MPL3115A2_CTRL_REG1, ®, 1) != 1) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
return -MPL3115A2_ERROR_I2C;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
reg &= ~MPL3115A2_CTRL_REG1_SBYB;
|
2017-04-22 20:54:39 +02:00
|
|
|
if (i2c_write_regs(BUS, ADDR, MPL3115A2_CTRL_REG1, ®, 1) != 1) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
return -MPL3115A2_ERROR_I2C;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_release(BUS);
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
return MPL3115A2_OK;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int mpl3115a2_is_ready(const mpl3115a2_t *dev)
|
2014-11-28 16:26:13 +01:00
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg;
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_acquire(BUS);
|
|
|
|
if (i2c_read_regs(BUS, ADDR, MPL3115A2_STATUS, ®, 1) != 1) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
return -MPL3115A2_ERROR_I2C;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_release(BUS);
|
2014-11-28 16:26:13 +01:00
|
|
|
|
|
|
|
return reg & MPL3115A2_STATUS_PTDR;
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int mpl3115a2_read_pressure(const mpl3115a2_t *dev, uint32_t *pres, uint8_t *status)
|
2014-11-28 16:26:13 +01:00
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t buf[4];
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_acquire(BUS);
|
|
|
|
if (i2c_read_regs(BUS, ADDR, MPL3115A2_STATUS, buf, 4) != 4) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
return -MPL3115A2_ERROR_I2C;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_release(BUS);
|
2014-11-28 16:26:13 +01:00
|
|
|
|
|
|
|
*status = buf[0];
|
|
|
|
|
|
|
|
*pres = ((uint32_t)buf[1] << 16) | ((uint32_t)buf[2] << 8) | buf[3];
|
|
|
|
*pres = *pres / 64;
|
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
return MPL3115A2_OK;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int mpl3115a2_read_temp(const mpl3115a2_t *dev, int16_t *temp)
|
2014-11-28 16:26:13 +01:00
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t buf[2];
|
2014-11-28 16:26:13 +01:00
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_acquire(BUS);
|
|
|
|
if (i2c_read_regs(BUS, ADDR, MPL3115A2_OUT_T_MSB, buf, 2) != 2) {
|
|
|
|
i2c_release(BUS);
|
|
|
|
return -MPL3115A2_ERROR_I2C;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|
2017-04-22 20:54:39 +02:00
|
|
|
i2c_release(BUS);
|
2014-11-28 16:26:13 +01:00
|
|
|
|
|
|
|
*temp = ((int16_t)(((int16_t)buf[0] << 8) | buf[1]) * 10) / 256;
|
|
|
|
|
2017-04-22 20:54:39 +02:00
|
|
|
return MPL3115A2_OK;
|
2014-11-28 16:26:13 +01:00
|
|
|
}
|