2015-02-03 07:10:56 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Eistec AB
|
2019-04-04 11:05:26 +02:00
|
|
|
* 2019 Otto-von-Guericke-Universität Magdeburg
|
2015-02-03 07:10:56 +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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2019-04-01 16:50:28 +02:00
|
|
|
* @ingroup drivers_ina2xx
|
2015-02-03 07:10:56 +01:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
2019-04-01 16:50:28 +02:00
|
|
|
* @brief Device driver implementation for Texas Instruments INA2XX High
|
2015-02-03 07:10:56 +01:00
|
|
|
* or Low Side, Bi-Directional CURRENT/POWER MONITOR with Two-Wire
|
|
|
|
* Interface
|
|
|
|
*
|
2015-09-20 13:47:39 +02:00
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
2019-04-04 11:05:26 +02:00
|
|
|
* @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
|
2015-02-03 07:10:56 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2019-04-04 11:05:26 +02:00
|
|
|
#include <errno.h>
|
2015-02-03 07:10:56 +01:00
|
|
|
#include <stdint.h>
|
|
|
|
|
2019-04-01 16:50:28 +02:00
|
|
|
#include "ina2xx.h"
|
2019-04-04 11:05:26 +02:00
|
|
|
#include "ina2xx_defines.h"
|
2015-02-03 07:10:56 +01:00
|
|
|
#include "periph/i2c.h"
|
|
|
|
#include "byteorder.h"
|
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2015-02-03 07:10:56 +01:00
|
|
|
#include "debug.h"
|
|
|
|
|
2019-04-04 11:05:26 +02:00
|
|
|
/*
|
|
|
|
* The value in the current register is obtained by:
|
|
|
|
*
|
|
|
|
* I = (V_shunt * C) / 4096
|
|
|
|
*
|
|
|
|
* Where V_shunt is the value in the shunt voltage register and C is the
|
|
|
|
* value programmed (upon driver initialization) into the calibration register
|
|
|
|
*/
|
|
|
|
#define CURRENT_QUOTIENT (4096LU)
|
|
|
|
|
2019-04-01 16:50:28 +02:00
|
|
|
/** @brief Read one 16 bit register from a INA2XX device and swap byte order, if necessary. */
|
|
|
|
static int ina2xx_read_reg(const ina2xx_t *dev, uint8_t reg, uint16_t *out)
|
2015-02-03 07:10:56 +01:00
|
|
|
{
|
|
|
|
union {
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t c[2];
|
2015-02-03 07:10:56 +01:00
|
|
|
uint16_t u16;
|
2019-04-04 11:05:26 +02:00
|
|
|
} tmp;
|
2015-02-03 07:10:56 +01:00
|
|
|
int status = 0;
|
|
|
|
|
2020-02-10 10:32:27 +01:00
|
|
|
i2c_acquire(dev->params.i2c);
|
2019-04-04 11:05:26 +02:00
|
|
|
status = i2c_read_regs(dev->params.i2c, dev->params.addr, reg, &tmp.c[0],
|
|
|
|
2, 0);
|
2020-02-10 10:32:27 +01:00
|
|
|
i2c_release(dev->params.i2c);
|
2015-02-03 07:10:56 +01:00
|
|
|
|
2018-06-02 00:28:20 +02:00
|
|
|
if (status < 0) {
|
|
|
|
return status;
|
2015-02-03 07:10:56 +01:00
|
|
|
}
|
|
|
|
|
2017-04-13 11:35:35 +02:00
|
|
|
*out = ntohs(tmp.u16);
|
2015-02-03 07:10:56 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-04-01 16:50:28 +02:00
|
|
|
/** @brief Write one 16 bit register to a INA2XX device and swap byte order, if necessary. */
|
|
|
|
static int ina2xx_write_reg(const ina2xx_t *dev, uint8_t reg, uint16_t in)
|
2015-02-03 07:10:56 +01:00
|
|
|
{
|
|
|
|
union {
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t c[2];
|
2015-02-03 07:10:56 +01:00
|
|
|
uint16_t u16;
|
2019-04-04 11:05:26 +02:00
|
|
|
} tmp;
|
2015-02-03 07:10:56 +01:00
|
|
|
int status = 0;
|
|
|
|
|
2017-04-13 11:35:35 +02:00
|
|
|
tmp.u16 = htons(in);
|
2015-02-03 07:10:56 +01:00
|
|
|
|
2020-02-10 10:32:27 +01:00
|
|
|
i2c_acquire(dev->params.i2c);
|
2019-04-04 11:05:26 +02:00
|
|
|
status = i2c_write_regs(dev->params.i2c, dev->params.addr, reg, &tmp.c[0],
|
|
|
|
2, 0);
|
2020-02-10 10:32:27 +01:00
|
|
|
i2c_release(dev->params.i2c);
|
2015-02-03 07:10:56 +01:00
|
|
|
|
2018-06-02 00:28:20 +02:00
|
|
|
if (status < 0) {
|
|
|
|
return status;
|
2015-02-03 07:10:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-04-04 11:05:26 +02:00
|
|
|
int ina2xx_init(ina2xx_t *dev, const ina2xx_params_t *params)
|
2015-02-03 07:10:56 +01:00
|
|
|
{
|
2019-04-04 11:05:26 +02:00
|
|
|
uint16_t config;
|
|
|
|
int status;
|
|
|
|
if (!dev || !params) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2015-02-03 07:10:56 +01:00
|
|
|
|
2019-04-04 11:05:26 +02:00
|
|
|
dev->params = *params;
|
|
|
|
/* Reset device */
|
|
|
|
status = ina2xx_write_reg(dev, INA2XX_REG_CONFIGURATION, INA2XX_RESET);
|
|
|
|
if (status < 0) {
|
2020-02-10 10:32:27 +01:00
|
|
|
DEBUG("[ina2xx]: Sending reset (write reg) failed with %d\n", status);
|
2019-04-04 11:05:26 +02:00
|
|
|
return status;
|
|
|
|
}
|
2015-02-03 07:10:56 +01:00
|
|
|
|
2019-04-04 11:05:26 +02:00
|
|
|
/* Check if default config is preset after reset */
|
|
|
|
status = ina2xx_read_reg(dev, INA2XX_REG_CONFIGURATION, &config);
|
|
|
|
if (status < 0) {
|
2020-02-10 10:32:27 +01:00
|
|
|
DEBUG("[ina2xx]: Verifying device (read reg) failed with %d\n", status);
|
2019-04-04 11:05:26 +02:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config != INA2XX_DEFCONFIG) {
|
|
|
|
DEBUG("[ina2xx]: Reset did't restore default config. Wrong device?\n");
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = ina2xx_write_reg(dev, INA2XX_REG_CONFIGURATION, params->config);
|
|
|
|
if (status < 0) {
|
2020-02-10 10:32:27 +01:00
|
|
|
DEBUG("[ina2xx]: Setting configuration (write reg) with %d\n", status);
|
2019-04-04 11:05:26 +02:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set calibration register based on the shunt resistor.
|
|
|
|
* ==> Current will be in mA
|
|
|
|
* Multiply by 100
|
|
|
|
* ==> Current will be in mA
|
|
|
|
*/
|
|
|
|
uint32_t calib = (100 * CURRENT_QUOTIENT) / params->rshunt_mohm;
|
|
|
|
/* Divide by 2^i_range to reduce the resolution by factor 2^i_range */
|
|
|
|
calib >>= params->i_range;
|
|
|
|
|
|
|
|
if (calib > UINT16_MAX) {
|
|
|
|
return -ERANGE;
|
|
|
|
}
|
|
|
|
|
2020-02-10 10:32:27 +01:00
|
|
|
status = ina2xx_write_reg(dev, INA2XX_REG_CALIBRATION, (uint16_t)calib);
|
|
|
|
if (status < 0) {
|
|
|
|
DEBUG("[ina2xx]: Setting calibration (write reg) with %d\n", status);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
2015-02-03 07:10:56 +01:00
|
|
|
}
|
|
|
|
|
2019-04-01 16:50:28 +02:00
|
|
|
int ina2xx_read_shunt(const ina2xx_t *dev, int16_t *voltage)
|
2015-02-03 07:10:56 +01:00
|
|
|
{
|
2019-04-01 16:50:28 +02:00
|
|
|
return ina2xx_read_reg(dev, INA2XX_REG_SHUNT_VOLTAGE, (uint16_t *)voltage);
|
2015-02-03 07:10:56 +01:00
|
|
|
}
|
|
|
|
|
2019-04-04 11:05:26 +02:00
|
|
|
int ina2xx_read_bus(const ina2xx_t *dev, uint16_t *voltage)
|
2015-02-03 07:10:56 +01:00
|
|
|
{
|
2019-04-04 11:05:26 +02:00
|
|
|
uint16_t tmp;
|
|
|
|
int status = ina2xx_read_reg(dev, INA2XX_REG_BUS_VOLTAGE, &tmp);
|
|
|
|
if (status < 0) {
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The voltage given by bits 15-3 in steps of 4 mV
|
|
|
|
* ==> Take bits 15-3, shift 3 bits right, multiply by 4
|
|
|
|
* ==> Same as: Take bits 15-3, shift 3 bits right, shift 2 bits left
|
|
|
|
* ==> Same as: Take bits 15-3, shift 1 bit right
|
|
|
|
*/
|
|
|
|
*voltage = (tmp & ~0x7) >> 1;
|
|
|
|
|
|
|
|
if (tmp & INA2XX_VBUS_OVF) {
|
|
|
|
return -EDOM;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (tmp & INA2XX_VBUS_CNVR) ? 1 : 0;
|
2015-02-03 07:10:56 +01:00
|
|
|
}
|
|
|
|
|
2019-04-04 11:05:26 +02:00
|
|
|
int ina2xx_read_current(const ina2xx_t *dev, int32_t *current)
|
2015-02-03 07:10:56 +01:00
|
|
|
{
|
2019-04-04 11:05:26 +02:00
|
|
|
int16_t tmp;
|
|
|
|
int status = ina2xx_read_reg(dev, INA2XX_REG_CURRENT, (uint16_t *)&tmp);
|
|
|
|
if (status < 0) {
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The calibration is chosen according to the selected value in
|
|
|
|
* dev->params.i_range, so that tmp * 2^i_range gives us the
|
|
|
|
* current in E-05 A. We can thus simple use a left shift to convert
|
|
|
|
* the current into E-05 A.
|
|
|
|
*/
|
|
|
|
*current = ((int32_t)tmp) << dev->params.i_range;
|
|
|
|
return 0;
|
2015-02-03 07:10:56 +01:00
|
|
|
}
|
|
|
|
|
2019-04-04 11:05:26 +02:00
|
|
|
int ina2xx_read_power(const ina2xx_t *dev, uint32_t *power)
|
2015-02-03 07:10:56 +01:00
|
|
|
{
|
2019-04-04 11:05:26 +02:00
|
|
|
int status;
|
|
|
|
uint16_t tmp;
|
|
|
|
status = ina2xx_read_reg(dev, INA2XX_REG_POWER, &tmp);
|
|
|
|
if (status < 0) {
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The resolution of the raw power value depends only on the resolution
|
|
|
|
* of the raw current value, as the bus voltage has a resolution of 4 mV
|
|
|
|
* regardless of configuration and calibration values. The product of
|
|
|
|
* bus voltage and raw current value is divided by 5000, this results in
|
|
|
|
* the following resolutions:
|
|
|
|
*
|
|
|
|
* Res current | res power
|
|
|
|
* 0.01 mA | 0.2 mW
|
|
|
|
* 0.02 mA | 0.4 mW
|
|
|
|
* 0.04 mA | 0.8 mW
|
|
|
|
* ...
|
|
|
|
*
|
|
|
|
* ==> multiply by 2^(1 + i_range) to get power in E-04W
|
|
|
|
*/
|
|
|
|
*power = ((uint32_t)tmp) << (1 + dev->params.i_range);
|
|
|
|
return 0;
|
2015-02-03 07:10:56 +01:00
|
|
|
}
|