2017-01-17 18:02:44 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 OTA keys S.A.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2018-06-01 12:09:01 +02:00
|
|
|
* @ingroup drivers_adcxx1c
|
2017-01-17 18:02:44 +01:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief ADCXX1C ADC device driver
|
|
|
|
*
|
|
|
|
* @author Vincent Dupont <vincent@otakeys.com>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "adcxx1c.h"
|
|
|
|
#include "adcxx1c_params.h"
|
|
|
|
#include "adcxx1c_regs.h"
|
|
|
|
|
|
|
|
#include "periph/i2c.h"
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2019-04-08 13:30:48 +02:00
|
|
|
#define DEV (dev->params.i2c)
|
2017-01-17 18:02:44 +01:00
|
|
|
#define ADDR (dev->params.addr)
|
|
|
|
|
|
|
|
/* Configuration register test value
|
|
|
|
* value 0x20: cycle time = Tconvert x 64 */
|
|
|
|
#define CONF_TEST_VALUE (0x20)
|
|
|
|
|
|
|
|
int adcxx1c_init(adcxx1c_t *dev, const adcxx1c_params_t *params)
|
|
|
|
{
|
|
|
|
assert(dev && params);
|
2018-11-29 10:07:45 +01:00
|
|
|
int status;
|
2017-01-17 18:02:44 +01:00
|
|
|
|
|
|
|
dev->params = *params;
|
|
|
|
dev->cb = NULL;
|
|
|
|
|
2019-04-08 13:30:48 +02:00
|
|
|
i2c_acquire(DEV);
|
2017-01-17 18:02:44 +01:00
|
|
|
uint8_t reg = 0;
|
|
|
|
|
|
|
|
/* Test communication write and read configuration register */
|
2019-04-08 13:30:48 +02:00
|
|
|
status = i2c_write_reg(DEV, ADDR, ADCXX1C_CONF_ADDR, CONF_TEST_VALUE, 0);
|
|
|
|
status += i2c_read_reg(DEV, ADDR, ADCXX1C_CONF_ADDR, ®, 0);
|
2017-01-17 18:02:44 +01:00
|
|
|
|
2018-11-29 10:07:45 +01:00
|
|
|
if (status < 0 || reg != CONF_TEST_VALUE) {
|
2019-04-08 13:30:48 +02:00
|
|
|
i2c_release(DEV);
|
2018-11-29 10:07:45 +01:00
|
|
|
DEBUG("[adcxx1c] init - error: unable to communicate with the device "
|
|
|
|
"(reg=%x)\n", reg);
|
2017-01-17 18:02:44 +01:00
|
|
|
return ADCXX1C_NODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
reg = dev->params.cycle << 5;
|
2019-04-08 13:30:48 +02:00
|
|
|
status = i2c_write_reg(DEV, ADDR, ADCXX1C_CONF_ADDR, reg, 0);
|
|
|
|
i2c_release(DEV);
|
2018-11-29 10:07:45 +01:00
|
|
|
if (status < 0) {
|
|
|
|
DEBUG("[adcxx1c] init - error: unable to communicate with the device "
|
|
|
|
"(err=%x)\n", status);
|
|
|
|
return ADCXX1C_NOI2C;
|
|
|
|
}
|
2017-01-17 18:02:44 +01:00
|
|
|
|
2018-11-29 10:07:45 +01:00
|
|
|
return adcxx1c_set_alert_parameters(dev, dev->params.low_limit,
|
|
|
|
dev->params.high_limit,
|
|
|
|
dev->params.hysteresis);
|
2017-01-17 18:02:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int adcxx1c_read_raw(const adcxx1c_t *dev, int16_t *raw)
|
|
|
|
{
|
|
|
|
uint8_t buf[2];
|
|
|
|
int status;
|
|
|
|
|
2019-04-08 13:30:48 +02:00
|
|
|
i2c_acquire(DEV);
|
|
|
|
status = i2c_read_regs(DEV, ADDR, ADCXX1C_CONV_RES_ADDR, buf, 2, 0);
|
|
|
|
i2c_release(DEV);
|
2018-05-28 09:59:58 +02:00
|
|
|
if (status < 0) {
|
2017-01-17 18:02:44 +01:00
|
|
|
return ADCXX1C_NOI2C;
|
|
|
|
}
|
|
|
|
|
|
|
|
*raw = ((buf[0] & 0x0F) << 8 | buf[1]) >> (12 - dev->params.bits);
|
|
|
|
|
|
|
|
return ADCXX1C_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _alert_cb(void *arg)
|
|
|
|
{
|
|
|
|
adcxx1c_t *dev = arg;
|
|
|
|
|
|
|
|
if (dev->cb) {
|
|
|
|
dev->cb(dev->arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int adcxx1c_enable_alert(adcxx1c_t *dev, adcxx1c_cb_t cb, void *arg)
|
|
|
|
{
|
|
|
|
uint8_t reg;
|
2018-11-29 10:07:45 +01:00
|
|
|
int status;
|
2017-01-17 18:02:44 +01:00
|
|
|
|
2019-04-08 13:30:48 +02:00
|
|
|
i2c_acquire(DEV);
|
|
|
|
i2c_read_reg(DEV, ADDR, ADCXX1C_CONF_ADDR, ®, 0);
|
2020-01-17 12:45:13 +01:00
|
|
|
reg |= (gpio_is_valid(dev->params.alert_pin) ? ADCXX1C_CONF_ALERT_PIN_EN : 0)
|
2017-01-17 18:02:44 +01:00
|
|
|
| ADCXX1C_CONF_ALERT_FLAG_EN;
|
2019-04-08 13:30:48 +02:00
|
|
|
status = i2c_write_reg(DEV, ADDR, ADCXX1C_CONF_ADDR, reg, 0);
|
|
|
|
i2c_release(DEV);
|
2018-11-29 10:07:45 +01:00
|
|
|
if (status < 0) {
|
|
|
|
DEBUG("[adcxx1c] enable_alert - error: unable to communicate with the "
|
|
|
|
"device (err=%d)\n", status);
|
|
|
|
return ADCXX1C_NOI2C;
|
|
|
|
}
|
2017-01-17 18:02:44 +01:00
|
|
|
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(dev->params.alert_pin)) {
|
2017-01-17 18:02:44 +01:00
|
|
|
dev->cb = cb;
|
|
|
|
dev->arg = arg;
|
|
|
|
/* alert active low */
|
|
|
|
gpio_init_int(dev->params.alert_pin, GPIO_IN, GPIO_FALLING, _alert_cb, dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ADCXX1C_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int adcxx1c_set_alert_parameters(const adcxx1c_t *dev, int16_t low_limit,
|
|
|
|
int16_t high_limit, int16_t hysteresis)
|
|
|
|
{
|
|
|
|
uint8_t buf[2];
|
2018-11-29 10:07:45 +01:00
|
|
|
int status;
|
2017-01-17 18:02:44 +01:00
|
|
|
|
2019-04-08 13:30:48 +02:00
|
|
|
i2c_acquire(DEV);
|
2017-01-17 18:02:44 +01:00
|
|
|
|
|
|
|
low_limit <<= (12 - dev->params.bits);
|
|
|
|
buf[0] = low_limit >> 8;
|
|
|
|
buf[1] = low_limit & 0xFF;
|
2019-04-08 13:30:48 +02:00
|
|
|
status = i2c_write_regs(DEV, ADDR, ADCXX1C_LOW_LIMIT_ADDR, buf, 2, 0);
|
2018-11-29 10:07:45 +01:00
|
|
|
if (status < 0) {
|
2019-04-08 13:30:48 +02:00
|
|
|
i2c_release(DEV);
|
2018-11-29 10:07:45 +01:00
|
|
|
DEBUG("[adcxx1c] set_alert (low limit) - error: unable to communicate "
|
|
|
|
"with the device (err=%d)\n", status);
|
|
|
|
return ADCXX1C_NOI2C;
|
|
|
|
}
|
2017-01-17 18:02:44 +01:00
|
|
|
|
|
|
|
high_limit <<= (12 - dev->params.bits);
|
|
|
|
buf[0] = high_limit >> 8;
|
|
|
|
buf[1] = high_limit & 0xFF;
|
2019-04-08 13:30:48 +02:00
|
|
|
status = i2c_write_regs(DEV, ADDR, ADCXX1C_HIGH_LIMIT_ADDR, buf, 2, 0);
|
2018-11-29 10:07:45 +01:00
|
|
|
if (status < 0) {
|
2019-04-08 13:30:48 +02:00
|
|
|
i2c_release(DEV);
|
2018-11-29 10:07:45 +01:00
|
|
|
DEBUG("[adcxx1c] set_alert (high limit) - error: unable to communicate "
|
|
|
|
"with the device (err=%d)\n", status);
|
|
|
|
return ADCXX1C_NOI2C;
|
|
|
|
}
|
2017-01-17 18:02:44 +01:00
|
|
|
|
|
|
|
hysteresis <<= (12 - dev->params.bits);
|
|
|
|
buf[0] = hysteresis >> 8;
|
|
|
|
buf[1] = hysteresis & 0xFF;
|
2019-04-08 13:30:48 +02:00
|
|
|
status = i2c_write_regs(DEV, ADDR, ADCXX1C_HYSTERESIS_ADDR, buf, 2, 0);
|
2018-11-29 10:07:45 +01:00
|
|
|
if (status < 0) {
|
2019-04-08 13:30:48 +02:00
|
|
|
i2c_release(DEV);
|
2018-11-29 10:07:45 +01:00
|
|
|
DEBUG("[adcxx1c] set_alert (hysteresis) - error: unable to communicate "
|
|
|
|
"with the device (err=%d)\n", status);
|
|
|
|
return ADCXX1C_NOI2C;
|
|
|
|
}
|
2017-01-17 18:02:44 +01:00
|
|
|
|
2019-04-08 13:30:48 +02:00
|
|
|
i2c_release(DEV);
|
2017-01-17 18:02:44 +01:00
|
|
|
|
|
|
|
return ADCXX1C_OK;
|
|
|
|
}
|
2019-04-16 17:00:28 +02:00
|
|
|
|
|
|
|
int adcxx1c_get_and_clear_alert(const adcxx1c_t *dev)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
uint8_t alert;
|
|
|
|
|
|
|
|
i2c_acquire(DEV);
|
|
|
|
|
|
|
|
status = i2c_read_reg(DEV, ADDR, ADCXX1C_ALERT_STATUS_ADDR, &alert, 0);
|
|
|
|
if (status < 0) {
|
|
|
|
i2c_release(DEV);
|
|
|
|
DEBUG("[adcxx1c] get_and_clear_alert - error: unable to read reg (%d)\n", status);
|
|
|
|
return ADCXX1C_NOI2C;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = i2c_write_reg(DEV, ADDR, ADCXX1C_ALERT_STATUS_ADDR, alert, 0);
|
|
|
|
if (status < 0) {
|
|
|
|
i2c_release(DEV);
|
|
|
|
DEBUG("[adcxx1c] get_and_clear_alert - error: unable to write reg (%d)\n", status);
|
|
|
|
return ADCXX1C_NOI2C;
|
|
|
|
}
|
|
|
|
|
|
|
|
i2c_release(DEV);
|
|
|
|
|
|
|
|
return alert;
|
|
|
|
}
|