1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 09:32:43 +01:00
RIOT/drivers/include/ina3221.h
2021-07-07 08:40:02 +02:00

1108 lines
38 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* Copyright (C) 2019 Otto-von-Guericke-Universität Magdeburg
*
* 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.
*/
/**
* @defgroup drivers_ina3221 INA3221 current/power monitor
* @ingroup drivers_sensors
* @ingroup drivers_saul
* @brief Device driver for Texas Instruments INA3221
* three-channel, high-side current and bus voltage
* monitor
* @{
*
* @file
* @brief Device driver interface for Texas Instruments INA3221
* three-channel, high-side current and bus voltage
* monitor.
*
* The alert pins WRN, CRT, TC, PV are open drain
* output pins, so they should be pulled high with
* internal or external pull-ups. They pull low, if
* the alert condition becomes true. If they become
* pulled low, the configured input pins in
* @ref ina3221_params_t will read a low flank and
* the related interrupt callback will be executed,
* if configured with @ref ina3221_enable_alert.
* WRN condition:
* Any channel´s measured shunt
* voltage drop was higher than the configured value in
* INA3221_REG_CHx_WARN_ALERT_LIMIT,
* CRT condition:
* Any channel´s measured shunt
* voltage drop was higher than the configured value in
* INA3221_REG_CHx_CRIT_ALERT_LIMIT,
* or the sum of all channel´s voltage drops was higher
* than the configured value in
* INA3221_REG_SHUNT_VOLTAGE_SUM_LIMIT. Note that the
* sum of voltage drops only makes sense, if all
* channels are using the same shunt resistance value!
* TC condition:
* After a 1.2-V level is detected on channel 1, if the
* INA3221 does not detect a 1.2-V value or greater on
* the bus voltage measurement following four complete
* cycles of all three channels, the timing control
* TC alert pin pulls low.
* PV condition:
* If any channel´s measured bus voltage was not within
* the configured values in INA3221_REG_PV_LOWER_LIMIT
* and INA3221_REG_PV_UPPER_LIMIT, the PV pin is pulled
* low.
*
* For the number of enabled channels @f$NCH@f$,
* number of samples @f$NS@f$,
* bus voltage conversion time @f$BCT@f$, and
* shunt voltage conversion time @f$SCT@f$,
* the time to complete a full measurement cycle
* can be calculated with:
* @f$ NCH \times NS \times (BCT + SCT) @f$.
*
* @see Datasheet:
* https://www.ti.com/lit/ds/symlink/ina3221.pdf
*
* @author Fabian Hüßler <fabian.huessler@ovgu.de>
*/
#ifndef INA3221_H
#define INA3221_H
#include <assert.h>
#include <errno.h>
#include <stdint.h>
#include <stdbool.h>
#include "kernel_defines.h"
#include "periph/gpio.h"
#include "periph/i2c.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief Number of channels
*/
#define INA3221_NUM_CH (3)
/**
* @brief Number of alerts
*/
#define INA3221_NUM_ALERTS (4)
/**
* @brief INA3221 alert callback for alert pins
* CRT, WRN, TC, PV
*/
typedef void (*ina3221_alert_cb_t)(void *arg);
/**
* @brief I2C device addresses
*/
typedef enum ina3221_dev_addr {
INA3221_ADDR_00 = 0x40, /**< A0 connected to GND */
INA3221_ADDR_01 = 0x41, /**< A0 connected to VS */
INA3221_ADDR_10 = 0x42, /**< A0 connected to SDA */
INA3221_ADDR_11 = 0x43, /**< A0 connected to SCL */
} ina3221_dev_addr_t;
/**
* @brief Channel flags
*/
typedef enum ina3221_channel {
INA3221_CH1 = (1U << 0), /**< Channel 1 */
INA3221_CH2 = (1U << 1), /**< Channel 2 */
INA3221_CH3 = (1U << 2) /**< Channel 3 */
} ina3221_channel_t;
/**
* @brief Alert indices
*/
typedef enum ina3221_alert {
INA3221_ALERT_WRN = 0, /**< Warning alert */
INA3221_ALERT_CRT = 1, /**< Critical alert */
INA3221_ALERT_TC = 2, /**< Timing control */
INA3221_ALERT_PV = 3 /**< Power valid alert */
} ina3221_alert_t;
/**
* @brief Device operation modes
*/
typedef enum ina3221_mode {
INA3221_MODE_POWER_DOWN = 0x0000, /**< Sleep mode */
INA3221_MODE_TRIGGER_SHUNT_ONLY = 0x0001, /**< Trigger shunt voltage measurement only */
INA3221_MODE_TRIGGER_BUS_ONLY = 0x0002, /**< Trigger bus voltage measurement only */
INA3221_MODE_TRIGGER_SHUNT_BUS = 0x0003, /**< Trigger shunt and bus voltage measurement */
INA3221_MODE_POWER_DOWN_ = 0x0004, /**< Also power down (see datasheet) */
INA3221_MODE_CONTINUOUS_SHUNT_ONLY = 0x0005, /**< Continuous shunt voltage measurement only */
INA3221_MODE_CONTINUOUS_BUS_ONLY = 0x0006, /**< Continuous bus voltage measurement only */
INA3221_MODE_CONTINUOUS_SHUNT_BUS = 0x0007 /**< Continuous shunt and bus voltage measurement */
} ina3221_mode_t;
/**
* @brief Device operation mode bit mask
*/
#define INA3221_MODE_MASK (0x0007)
/**
* @brief ADC shunt voltage conversion times
*/
typedef enum ina3221_conv_time_shunt_adc {
INA3221_CONV_TIME_SADC_140US = 0x0000, /**< 140 us */
INA3221_CONV_TIME_SADC_204US = 0x0008, /**< 204 us */
INA3221_CONV_TIME_SADC_332US = 0x0010, /**< 332 us */
INA3221_CONV_TIME_SADC_588US = 0x0018, /**< 588 us */
INA3221_CONV_TIME_SADC_1100US = 0x0020, /**< 1100 us */
INA3221_CONV_TIME_SADC_2116US = 0x0028, /**< 2116 us */
INA3221_CONV_TIME_SADC_4156US = 0x0030, /**< 4156 us */
INA3221_CONV_TIME_SADC_8244US = 0x0038 /**< 8244 us */
} ina3221_conv_time_shunt_adc_t;
/**
* @brief ADC shunt voltage conversion times bit mask
*/
#define INA3221_CONV_TIME_SADC_MASK (0x0038)
/**
* @brief ADC bus voltage conversion times
*/
typedef enum ina3221_conv_time_bus_adc {
INA3221_CONV_TIME_BADC_140US = 0x0000, /**< 140 us */
INA3221_CONV_TIME_BADC_204US = 0x0040, /**< 204 us */
INA3221_CONV_TIME_BADC_332US = 0x0080, /**< 332 us */
INA3221_CONV_TIME_BADC_588US = 0x00C0, /**< 588 us */
INA3221_CONV_TIME_BADC_1100US = 0x0100, /**< 1100 us */
INA3221_CONV_TIME_BADC_2116US = 0x0140, /**< 2116 us */
INA3221_CONV_TIME_BADC_4156US = 0x0180, /**< 4156 us */
INA3221_CONV_TIME_BADC_8244US = 0x01C0 /**< 8244 us */
} ina3221_conv_time_bus_adc_t;
/**
* @brief ADC bus voltage conversion times bit mask
*/
#define INA3221_CONV_TIME_BADC_MASK (0x01C0)
/**
* @brief Number of samples to calculate average shunt/bus voltage value
*/
typedef enum ina3221_num_samples {
INA3221_NUM_SAMPLES_1 = 0x0000, /**< 1 */
INA3221_NUM_SAMPLES_4 = 0x0200, /**< 4 */
INA3221_NUM_SAMPLES_16 = 0x0400, /**< 16 */
INA3221_NUM_SAMPLES_64 = 0x0600, /**< 64 */
INA3221_NUM_SAMPLES_128 = 0x0800, /**< 128 */
INA3221_NUM_SAMPLES_256 = 0x0A00, /**< 256 */
INA3221_NUM_SAMPLES_512 = 0x0C00, /**< 512 */
INA3221_NUM_SAMPLES_1024 = 0x0E00 /**< 1024 */
} ina3221_num_samples_t;
/**
* @brief Number of samples to calculate average shunt/bus voltage value bit mask
*/
#define INA3221_NUM_SAMPLES_MASK (0x0E00)
/**
* @brief Enable channel flags
*/
typedef enum ina3221_enable_ch {
INA3221_ENABLE_CH1 = 0x4000, /**< Enable channel 1 */
INA3221_ENABLE_CH2 = 0x2000, /**< Enable channel 2 */
INA3221_ENABLE_CH3 = 0x1000 /**< Enable channel 3 */
} ina3221_enable_ch_t;
/**
* @brief Enable channel bit mask
*/
#define INA3221_ENABLE_CH_MASK (INA3221_ENABLE_CH1 \
| INA3221_ENABLE_CH2 \
| INA3221_ENABLE_CH3)
/**
* @brief Enable shunt voltage sum calculation channel flags
*/
typedef enum ina3221_enable_sum_ch {
INA3221_ENABLE_SUM_CH1 = 0x4000, /**< Enable sum channel 1 */
INA3221_ENABLE_SUM_CH2 = 0x2000, /**< Enable sum channel 2 */
INA3221_ENABLE_SUM_CH3 = 0x1000 /**< Enable sum channel 3 */
} ina3221_enable_sum_ch_t;
/**
* @brief Enable shunt voltage sum calculation channel bit mask
*/
#define INA3221_ENABLE_SUM_CH_MASK (INA3221_ENABLE_SUM_CH1 \
| INA3221_ENABLE_SUM_CH2 \
| INA3221_ENABLE_SUM_CH3)
/**
* @brief Enable latch flags
*/
typedef enum ina3221_enable_latch {
INA3221_ENABLE_WARN_LATCH = 0x0800, /**< Enable warning alert latch */
INA3221_ENABLE_CRIT_LATCH = 0x0400 /**< Enable critical alert latch */
} ina3221_enable_latch_t;
/**
* @brief Enable latch bit mask
*/
#define INA3221_ENABLE_LATCH_MASK (INA3221_ENABLE_WARN_LATCH \
| INA3221_ENABLE_CRIT_LATCH)
/**
* @brief Flags @ref ina3221_channel_t
*/
typedef unsigned ina3221_ch_t;
/**
* @brief INA3221 device parameters
*/
typedef struct {
i2c_t i2c; /**< I2C bus */
uint8_t addr; /**< I2C address */
union {
struct {
gpio_t pin_warn; /**< Warning alert pin, becomes configured as input */
gpio_t pin_crit; /**< Critical alert pin, becomes configured as input */
gpio_t pin_tc; /**< Timing control alert pin, becomes configured as input */
gpio_t pin_pv; /**< Power valid alert pin, becomes configured as input */
} pins; /**< Struct of alert pins */
struct {
gpio_t alert_pins[INA3221_NUM_ALERTS]; /**< Array to access alert pins via an index */
} apins; /**< Struct of array of alert pins */
} upins; /**< Union of alert pins as array and as single members */
/*
For each pin 1 bit states whether to use internal pull up resistors.
d = don´t care
[d][d][d][d][pu PV][pu TC][pu CRT][pu WRN]
*/
uint8_t gpio_config; /**< Configure pull up resistors for gpio pins */
uint16_t config; /**< Configuration */
uint16_t rshunt_mohm[INA3221_NUM_CH]; /**< Shunt resistances in mOhm for each channel */
} ina3221_params_t;
/* ( INA3221_ENABLE_CH_1
| INA3221_ENABLE_CH_2
| INA3221_ENABLE_CH_2
| INA3221_NUM_SAMPLES_1
| INA3221_CONV_TIME_BADC_1100US
| INA3221_CONV_TIME_SADC_1100US
| INA3221_MODE_CONTINUOUS_SHUNT_BUS ) */
#define INA3221_DEFCONFIG (0x7127) /**< Default configuration register value */
#define INA3221_RESET (0x8000) /**< Reset configuration register value */
#define INA3221_FLAG_CRIT_ALERT_CH1 (0x0200) /**< Critical alert channel 1 flag */
#define INA3221_FLAG_CRIT_ALERT_CH2 (0x0100) /**< Critical alert channel 2 flag */
#define INA3221_FLAG_CRIT_ALERT_CH3 (0x0080) /**< Critical alert channel 3 flag */
#define INA3221_FLAG_SHUNT_SUM_ALERT (0x0040) /**< Alert channel sum flag */
#define INA3221_FLAG_WARN_ALERT_CH1 (0x0020) /**< Warning alert channel 1 flag */
#define INA3221_FLAG_WARN_ALERT_CH2 (0x0010) /**< Warning alert channel 2 flag */
#define INA3221_FLAG_WARN_ALERT_CH3 (0x0008) /**< Warning alert channel 3 flag */
#define INA3221_FLAG_POWER_VALID (0x0004) /**< Power valid flag */
#define INA3221_FLAG_TIMING_CONTROL (0x0002) /**< Timing Control flag */
#define INA3221_FLAG_CONV_READY (0x0001) /**< Conversion ready flag */
/**
* @brief Flags bit mask
*/
#define INA3221_FLAGS_MASK (INA3221_FLAG_CRIT_ALERT_CH1 \
| INA3221_FLAG_CRIT_ALERT_CH2 \
| INA3221_FLAG_CRIT_ALERT_CH3 \
| INA3221_FLAG_SHUNT_SUM_ALERT \
| INA3221_FLAG_WARN_ALERT_CH1 \
| INA3221_FLAG_WARN_ALERT_CH2 \
| INA3221_FLAG_WARN_ALERT_CH3 \
| INA3221_FLAG_POWER_VALID \
| INA3221_FLAG_TIMING_CONTROL \
| INA3221_FLAG_CONV_READY)
/**
* @brief INA3221 device handle struct
*/
typedef struct {
ina3221_params_t params; /**< Device parameters */
#if IS_USED(MODULE_INA3221_ALERTS) || defined(DOXYGEN)
union {
struct {
ina3221_alert_cb_t warn_cb; /**< Warning alert callback: executed when WRN is pulled low */
ina3221_alert_cb_t crit_cb; /**< Critical alert callback: executed when CRT is pulled low */
ina3221_alert_cb_t tc_cb; /**< Timing control alert callback: executed when TC is pulled low */
ina3221_alert_cb_t pv_cb; /**< Power Valid alert callback: executed when PV is pulled low */
};
struct {
ina3221_alert_cb_t alert_callbacks[INA3221_NUM_ALERTS]; /**< Array to access alert callbacks via an index */
};
};
union {
struct {
void *arg_warn_cb; /**< Argument for Warning alert callback */
void *arg_crit_cb; /**< Argument for Critical alert callback */
void *arg_tc_cb; /**< Argument for Timing control alert callback */
void *arg_pv_cb; /**< Argument for Power Valid alert callback */
};
struct {
void *alert_callback_arguments[INA3221_NUM_ALERTS]; /**< Array to access alert callback arguments via an index */
};
};
#endif /* MODULE_INA3221_ALERTS */
} ina3221_t;
/**
* @brief Reset device, i.e. apply default configuration
*
* @param[in,out] dev Device handle
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return -ENODEV, if device could not be reset
* @return @see i2c_read_regs
* @return @see i2c_write_regs
*/
int ina3221_reset(ina3221_t *dev);
/**
* @brief Initialize device
*
* @param[out] dev Device handle
* @param[in] params Device parameters
*
* @return 0, on success
* @return -EFAULT, if NULL pointer was passed
* @return -EIO, if I2C bus acquirement failed
* @return -ENXIO, if manufacturer ID or DIE ID does not match
* @return -ENODEV, if reset failed
* @return -EINVAL, if configuration could not be applied
* @return @see i2c_read_regs
* @return @see i2c_write_regs
*/
int ina3221_init(ina3221_t *dev, const ina3221_params_t *params);
#if IS_USED(MODULE_INA3221_ALERTS) || defined(DOXYGEN)
/**
* @brief Enable alert and set callback and argument for alert @p alert
*
* @param[in,out] dev Device handle
* @param[in] alert Alert index
* @param[in] cb Alert callback
* @param[in] arg Alert callback argument
*
* @return 0, on success
* @return -ERANGE, if @p alert was out of bounds
* @return -ENUTSUP, if alert pin was initialized with GPIO_UNDEF
*/
int ina3221_enable_alert(ina3221_t *dev, ina3221_alert_t alert,
ina3221_alert_cb_t cb, void *arg);
/**
* @brief Wrapper around @ref ina3221_enable_alert,
* to enable warning alert
*
* @param[in,out] dev Device handle
* @param[in] cb Alert callback
* @param[in] arg Alert callback argument
*
* @return @see ina3221_enable_alert
*/
static inline int ina3221_enable_warning_alert(ina3221_t *dev,
ina3221_alert_cb_t cb,
void *arg)
{
return ina3221_enable_alert(dev, INA3221_ALERT_WRN, cb, arg);
}
/**
* @brief Wrapper around @ref ina3221_enable_alert,
* to enable critical alert
*
* @param[in,out] dev Device handle
* @param[in] cb Alert callback
* @param[in] arg Alert callback argument
*
* @return @see ina3221_enable_alert
*/
static inline int ina3221_enable_critical_alert(ina3221_t *dev,
ina3221_alert_cb_t cb,
void *arg)
{
return ina3221_enable_alert(dev, INA3221_ALERT_CRT, cb, arg);
}
/**
* @brief Wrapper around @ref ina3221_enable_alert,
* to enable timing control alert
*
* @param[in,out] dev Device handle
* @param[in] cb Alert callback
* @param[in] arg Alert callback argument
*
* @return @see ina3221_enable_alert
*/
static inline int ina3221_enable_timing_control_alert(ina3221_t *dev,
ina3221_alert_cb_t cb,
void *arg)
{
return ina3221_enable_alert(dev, INA3221_ALERT_TC, cb, arg);
}
/**
* @brief Wrapper around @ref ina3221_enable_alert,
* to enable power valid alert
*
* @param[in,out] dev Device handle
* @param[in] cb Alert callback
* @param[in] arg Alert callback argument
*
* @return @see ina3221_enable_alert
*/
static inline int ina3221_enable_power_valid_alert(ina3221_t *dev,
ina3221_alert_cb_t cb,
void *arg)
{
return ina3221_enable_alert(dev, INA3221_ALERT_PV, cb, arg);
}
/**
* @brief Disable alert @p alert
*
* @param[in,out] dev Device handle
* @param[in] alert Alert index
*
* @return 0, on success
* @return -ERANGE, if @p alert was out of bounds
* @return -ENUTSUP, if alert pin was initialized with GPIO_UNDEF
*/
int ina3221_disable_alert(ina3221_t *dev, ina3221_alert_t alert);
/**
* @brief Wrapper around @ref ina3221_disable_alert,
* to disable warning alert
*
* @param[in,out] dev Device handle
*
* @return @see ina3221_disable_alert
*/
static inline int ina3221_disable_warning_alert(ina3221_t *dev)
{
return ina3221_disable_alert(dev, INA3221_ALERT_WRN);
}
/**
* @brief Wrapper around @ref ina3221_disable_alert,
* to disable critical alert
*
* @param[in,out] dev Device handle
*
* @return @see ina3221_disable_alert
*/
static inline int ina3221_disable_critical_alert(ina3221_t *dev)
{
return ina3221_disable_alert(dev, INA3221_ALERT_CRT);
}
/**
* @brief Wrapper around @ref ina3221_disable_alert,
* to disable timing control alert
*
* @param[in,out] dev Device handle
*
* @return @see ina3221_disable_alert
*/
static inline int ina3221_disable_timing_control_alert(ina3221_t *dev)
{
return ina3221_disable_alert(dev, INA3221_ALERT_TC);
}
/**
* @brief Wrapper around @ref ina3221_disable_alert,
* to disable power valid alert
*
* @param[in,out] dev Device handle
*
* @return @see ina3221_disable_alert
*/
static inline int ina3221_disable_power_valid_alert(ina3221_t *dev)
{
return ina3221_disable_alert(dev, INA3221_ALERT_PV);
}
#endif /* MODULE_INA3221_ALERTS */
/**
* @brief Get enabled measurement channels from configuration value
*
* @param[in] cfg Configuration value
*
* @return Channel flags @ref ina3221_channel_t
*/
static inline ina3221_ch_t ina3221_config_get_enabled_channels(uint16_t cfg)
{
return ((cfg & INA3221_ENABLE_CH1) ? INA3221_CH1 : 0) |
((cfg & INA3221_ENABLE_CH2) ? INA3221_CH2 : 0) |
((cfg & INA3221_ENABLE_CH3) ? INA3221_CH3 : 0);
}
/**
* @brief Enable measurement channels in configuration value
*
* @param[out] cfg Configuration value
* @param[in] ch Channel flags @ref ina3221_channel_t
*/
static inline void ina3221_config_set_enabled_channels(uint16_t *cfg, ina3221_ch_t ch)
{
assert(cfg);
uint16_t u16 = ((ch & INA3221_CH1) ? INA3221_ENABLE_CH1 : 0) |
((ch & INA3221_CH2) ? INA3221_ENABLE_CH2 : 0) |
((ch & INA3221_CH3) ? INA3221_ENABLE_CH3 : 0);
*cfg &= ~INA3221_ENABLE_CH_MASK;
*cfg |= (u16 & INA3221_ENABLE_CH_MASK);
}
/**
* @brief Get number of samples from configuration value
*
* @param[in] cfg Configuration value
*
* @return Number of samples
*/
static inline ina3221_num_samples_t ina3221_config_get_num_samples(uint16_t cfg)
{
return (ina3221_num_samples_t)(cfg & INA3221_NUM_SAMPLES_MASK);
}
/**
* @brief Set number of samples in configuration value
*
* @param[out] cfg Configuration value
* @param[in] ns Number of samples
*/
static inline void ina3221_config_set_num_samples(uint16_t *cfg,
ina3221_num_samples_t ns)
{
assert(cfg);
uint16_t u16 = (uint16_t)ns;
*cfg &= ~INA3221_NUM_SAMPLES_MASK;
*cfg |= (u16 & INA3221_NUM_SAMPLES_MASK);
}
/**
* @brief Get bus voltage conversion time from configuration value
*
* @param[in] cfg Configuration value
*
* @return Bus conversion time
*/
static inline ina3221_conv_time_bus_adc_t ina3221_config_get_conv_time_bus(uint16_t cfg)
{
return (ina3221_conv_time_bus_adc_t)(cfg & INA3221_CONV_TIME_BADC_MASK);
}
/**
* @brief Set bus voltage conversion time in configuration value
*
* @param[out] cfg Configuration value
* @param[in] bus Bus conversion time
*/
static inline void ina3221_config_set_conv_time_bus(uint16_t *cfg,
ina3221_conv_time_bus_adc_t bus)
{
assert(cfg);
uint16_t u16 = (uint16_t)bus;
*cfg &= ~INA3221_CONV_TIME_BADC_MASK;
*cfg |= (u16 & INA3221_CONV_TIME_BADC_MASK);
}
/**
* @brief Get shunt voltage conversion time from configuration value
*
* @param[in] cfg Configuration value
*
* @return Shunt conversion time
*/
static inline ina3221_conv_time_shunt_adc_t ina3221_config_get_conv_time_shunt(uint16_t cfg)
{
return (ina3221_conv_time_shunt_adc_t)(cfg & INA3221_CONV_TIME_SADC_MASK);
}
/**
* @brief Set shunt voltage conversion time in configuration value
*
* @param[out] cfg Configuration value
* @param[in] shunt Shunt conversion time
*/
static inline void ina3221_config_set_conv_time_shunt(uint16_t *cfg,
ina3221_conv_time_shunt_adc_t shunt)
{
assert(cfg);
uint16_t u16 = (uint16_t)shunt;
*cfg &= ~INA3221_CONV_TIME_SADC_MASK;
*cfg |= (u16 & INA3221_CONV_TIME_SADC_MASK);
}
/**
* @brief Get operation mode from configuration value
*
* @param[in] cfg Configuration value
*
* @return Operation mode
*/
static inline ina3221_mode_t ina3221_config_get_mode(uint16_t cfg)
{
return (ina3221_mode_t)(cfg & INA3221_MODE_MASK);
}
/**
* @brief Set operation mode in configuration value
*
* @param[out] cfg Configuration value
* @param[in] mode Operation mode
*/
static inline void ina3221_config_set_mode(uint16_t *cfg,
ina3221_mode_t mode)
{
assert(cfg);
uint16_t u16 = (uint16_t)mode;
*cfg &= ~INA3221_MODE_MASK;
*cfg |= (u16 & INA3221_MODE_MASK);
}
/**
* @brief Write configuration register value to configuration register
*
* @param[in,out] dev Device handle
* @param[in] cfg Configuration register value
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_write_regs
*/
int ina3221_set_config(ina3221_t *dev, uint16_t cfg);
/**
* @brief Read currently saved configuration register value
*
* @param[in] dev Device handle
* @param[out] cfg Pointer to configuration register output value
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
*/
int ina3221_get_config(const ina3221_t *dev, uint16_t *cfg);
/**
* @brief Enable channels
*
* @param[in,out] dev Device handle
* @param[in] ch Channel flags @ref ina3221_channel_t
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
* @return @see i2c_write_regs
*/
int ina3221_set_enable_channel(ina3221_t *dev, ina3221_ch_t ch);
/**
* @brief Read which channels are currently enabled
*
* @param[in] dev Device handle
* @param[out] ch Channel flags @ref ina3221_channel_t
*/
static inline void ina3221_get_enable_channel(const ina3221_t *dev,
ina3221_ch_t *ch)
{
*ch = ina3221_config_get_enabled_channels(dev->params.config);
}
/**
* @brief Update number of samples and write to configuration register
*
* @param[in, out] dev Device handle
* @param[in] ns Number of samples
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
* @return @see i2c_write_regs
*/
int ina3221_set_num_samples(ina3221_t *dev, ina3221_num_samples_t ns);
/**
* @brief Read currently applied number of samples
*
* @param[in] dev Device handle
* @param[out] ns Pointer to number of samples output variable
*/
static inline void ina3221_get_num_samples(const ina3221_t *dev,
ina3221_num_samples_t *ns)
{
*ns = ina3221_config_get_num_samples(dev->params.config);
}
/**
* @brief Update conversion time of bus voltage ADC and write to configuration register
*
* @param[in,out] dev Device handle
* @param[in] ctb Bus voltage conversion time
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
* @return @see i2c_write_regs
*/
int ina3221_set_conv_time_bus_adc(ina3221_t *dev,
ina3221_conv_time_bus_adc_t ctb);
/**
* @brief Read currently applied bus voltage ADC conversion time
*
* @param[in] dev Device handle
* @param[out] ctb Pointer to bus voltage ADC conversion time output variable
*/
static inline void ina3221_get_conv_time_bus_adc(const ina3221_t *dev,
ina3221_conv_time_bus_adc_t *ctb)
{
*ctb = ina3221_config_get_conv_time_bus(dev->params.config);
}
/**
* @brief Update conversion time of shunt voltage ADC and write to configuration register
*
* @param[in,out] dev Device handle
* @param[in] cts Shunt voltage conversion time value
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
* @return @see i2c_write_regs
*/
int ina3221_set_conv_time_shunt_adc(ina3221_t *dev,
ina3221_conv_time_shunt_adc_t cts);
/**
* @brief Read currently applied shunt voltage ADC conversion time
*
* @param[in] dev Device handle
* @param[out] cts Pointer to shunt voltage ADC conversion time output variable
*/
static inline void ina3221_get_conv_time_shunt_adc(const ina3221_t *dev,
ina3221_conv_time_shunt_adc_t *cts)
{
*cts = ina3221_config_get_conv_time_shunt(dev->params.config);
}
/**
* @brief Update device operation mode
*
* @param[in,out] dev Device handle
* @param[in] mode Operation mode value
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
* @return @see i2c_write_regs
*/
int ina3221_set_mode(ina3221_t *dev, ina3221_mode_t mode);
/**
* @brief Read currently applied device operation mode
*
* @param[in] dev Device handle
* @param[out] mode Pointer to device operation mode output variable
*/
static inline void ina3221_get_mode(const ina3221_t *dev, ina3221_mode_t *mode)
{
*mode = ina3221_config_get_mode(dev->params.config);
}
/**
* @brief Enable channels for shunt voltage sum calculation
*
* @param[in] dev Device handle
* @param[in] ch Channel flags @ref ina3221_channel_t
*
* @note The channel must also have been enabled with
* @ref ina3221_set_enable_channel
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
* @return @see i2c_write_regs
*/
int ina3221_set_enable_sum_channel(const ina3221_t *dev, ina3221_ch_t ch);
/**
* @brief Read enabled channels for shunt voltage sum calculation
*
* @param[in] dev Device handle
* @param[out] ch Channel flags @ref ina3221_channel_t
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
*/
int ina3221_get_enable_sum_channel(const ina3221_t *dev, ina3221_ch_t *ch);
/**
* @brief Enable latches for critical/warning alert pins
*
* @param[in] dev Device handle
* @param[in] warn Enable warning alert latch
* @param[in] crit Enable critical alert latch
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
* @return @see i2c_write_regs
*/
int ina3221_set_latch(const ina3221_t *dev, bool warn, bool crit);
/**
* @brief Read enabled latches for critical/warning alert pins
*
* @param[in] dev Device handle
* @param[in] warn Will be true if warning alert is latched
* @param[in] crit Will be true if critical alert is latched
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
*/
int ina3221_get_latch(const ina3221_t *dev, bool *warn, bool *crit);
/**
* @brief Set critical shunt voltage alert limit to @p in_uv for each channel in @p ch
*
* @param[in] dev Device handle
* @param[in] ch Channel flags @ref ina3221_channel_t
* @param[in] in_uv Critical shunt voltage limit in uV
*
* @return Flags of which channel´s limits could be updated
*/
ina3221_ch_t ina3221_set_crit_alert_limit(const ina3221_t *dev,
ina3221_ch_t ch, int32_t in_uv);
/**
* @brief Read critical shunt voltage alert limit for each channel in @p ch
*
* @param[in] dev Device handle
* @param[in] ch Channel flags @ref ina3221_channel_t
* @param[out] out_uv Output array of shunt voltage alert limits in uV
*
* @return Flags of which channel´s values were written to @p out_uv
*/
ina3221_ch_t ina3221_get_crit_alert_limit(const ina3221_t *dev, ina3221_ch_t ch,
int32_t out_uv[INA3221_NUM_CH]);
/**
* @brief Set warning shunt voltage alert limit to @p in_uv for each channel in @p ch
*
* @param[in] dev Device handle
* @param[in] ch Channel flags @ref ina3221_channel_t
* @param[in] in_uv Warning shunt voltage limit in uV
*
* @return Flags of which channel´s limits could be updated
*/
ina3221_ch_t ina3221_set_warn_alert_limit(const ina3221_t *dev,
ina3221_ch_t ch, int32_t in_uv);
/**
* @brief Read warning shunt voltage alert limit for each channel in @p ch
*
* @param[in] dev Device handle
* @param[in] ch Channel flags @ref ina3221_channel_t
* @param[out] out_uv Output array of shunt voltage alert limits in uV
*
* @return Flags of which channel´s values could be read
*/
ina3221_ch_t ina3221_get_warn_alert_limit(const ina3221_t *dev, ina3221_ch_t ch,
int32_t out_uv[INA3221_NUM_CH]);
/**
* @brief Set shunt voltage sum alert limit to @p in_uv
*
* @param[in] dev Device handle
* @param[in] in_uv Shunt voltage sum limit in uV
*
* @return 0, on success
* @return -ERANGE, if @p in_uv was not in [INA3221_MIN_SHUNT_SUM_UV; INA3221_MAX_SHUNT_SUM_UV]
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_write_regs
*/
int ina3221_set_shunt_voltage_sum_alert_limit(const ina3221_t *dev,
int32_t in_uv);
/**
* @brief Read shunt voltage sum alert limit
*
* @param[in] dev Device handle
* @param[out] out_uv Pointer to sum voltage sum alert limit output variable
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
*/
int ina3221_get_shunt_voltage_sum_alert_limit(const ina3221_t *dev,
int32_t *out_uv);
/**
* @brief Set bus voltage power valid upper limit to @p in_mv
*
* @param[in] dev Device handle
* @param[in] in_mv Bus voltage power valid upper limit in mV
*
* @return 0, on success
* @return -ERANGE, if @p in_mv was not in [INA3221_MIN_BUS_MV; INA3221_MAX_BUS_MV]
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_write_regs
*/
int ina3221_set_power_valid_upper_limit(const ina3221_t *dev, int32_t in_mv);
/**
* @brief Read bus voltage power valid upper limit
*
* @param[in] dev Device handle
* @param[out] out_mv Pointer to bus voltage power valid upper limit output variable
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
*/
int ina3221_get_power_valid_upper_limit(const ina3221_t *dev, int32_t *out_mv);
/**
* @brief Set bus voltage power valid lower limit to @p in_mv
*
* @param[in] dev Device handle
* @param[in] in_mv Bus voltage power valid lower limit in mV
*
* @return 0, on success
* @return -ERANGE, if @p in_mv was not in [INA3221_MIN_BUS_MV; INA3221_MAX_BUS_MV]
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_write_regs
*/
int ina3221_set_power_valid_lower_limit(const ina3221_t *dev, int32_t in_mv);
/**
* @brief Read bus voltage power valid lower limit
*
* @param[in] dev Device handle
* @param[out] out_mv Pointer to bus voltage power valid lower limit output variable
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
*/
int ina3221_get_power_valid_lower_limit(const ina3221_t *dev, int32_t *out_mv);
/**
* @brief Read status flags
*
* @param[in] dev Device handle
* @param[out] flags Pointer to flags output variable
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
*/
int ina3221_read_flags(const ina3221_t *dev, uint16_t *flags);
/**
* @brief Read sum of shunt voltages
*
* @param[in] dev Device handle
* @param[out] out_uv Pointer to shunt voltage sum output variable
* @param[out] flags Pointer to flags output variable, may be NULL
*
* @return 0, on success
* @return -EIO, if I2C bus acquirement failed
* @return @see i2c_read_regs
*/
int ina3221_read_shunt_sum_uv(const ina3221_t *dev, int32_t *out_uv,
uint16_t *flags);
/**
* @brief Read shunt voltages for each channel in @p ch
*
* @param[in] dev Device handle
* @param[out] out_uv Array of output shunt voltage values in uV
* @param[out] flags Pointer to flags output variable, may be NULL
*
* @return Flags of which channel´s shunt voltage drops could be read
*/
ina3221_ch_t ina3221_read_shunt_uv(const ina3221_t *dev,
int32_t out_uv[INA3221_NUM_CH],
uint16_t *flags);
/**
* @brief Read bus voltages for each channel in @p ch
*
* @param[in] dev Device handle
* @param[out] out_mv Array of output bus voltage values in mV
* @param[out] flags Pointer to flags output variable, may be NULL
*
* @return Flags of which channel´s bus voltage values could be read
*/
ina3221_ch_t ina3221_read_bus_mv(const ina3221_t *dev,
int16_t out_mv[INA3221_NUM_CH],
uint16_t *flags);
/**
* @brief Calculate current for each channel in @p ch
*
* @param[in] ch Flags of which values in @p in_mohm and @p in_uv are valid
* @param[in] in_mohm Array of input shunt resistors in mOhm
* @param[in] in_uv Array of input shunt voltage values in uV
* @param[out] out_ua Array of output current values in uA
*/
void ina3221_calculate_current_ua(ina3221_ch_t ch,
const uint16_t in_mohm[INA3221_NUM_CH],
const int32_t in_uv[INA3221_NUM_CH],
int32_t out_ua[INA3221_NUM_CH]);
/**
* @brief Calculate power from bus voltage and current values
*
* @param[in] ch Flags of which values in @p in_mv and @p in_ua are valid
* @param[in] in_mv Array of input bus voltage values in mV
* @param[in] in_ua Array of input current values in uA
* @param[out] out_mw Array of output power values in mW
*/
void ina3221_calculate_power_uw(ina3221_ch_t ch,
const int16_t in_mv[INA3221_NUM_CH],
const int32_t in_ua[INA3221_NUM_CH],
int32_t out_mw[INA3221_NUM_CH]);
/**
* @brief Set operation mode to INA3221_MODE_TRIGGER_SHUNT_ONLY
* to trigger shunt voltage measurement
*
* @param dev Device handle
*/
#define INA3221_TRIGGER_SHUNT(dev) \
ina3221_set_mode(dev, INA3221_MODE_TRIGGER_SHUNT_ONLY)
/**
* @brief Set operation mode to INA3221_MODE_TRIGGER_BUS_ONLY
* to trigger bus voltage measurement
*
* @param dev Device handle
*/
#define INA3221_TRIGGER_BUS(dev) \
ina3221_set_mode(dev, INA3221_MODE_TRIGGER_BUS_ONLY)
/**
* @brief Set operation mode to INA3221_MODE_TRIGGER_SHUNT_BUS
* to trigger shunt and bus voltage measurement
*
* @param dev Device handle
*/
#define INA3221_TRIGGER_SHUNT_AND_BUS(dev) \
ina3221_set_mode(dev, INA3221_MODE_TRIGGER_SHUNT_BUS)
#ifdef __cplusplus
}
#endif
#endif /* INA3221_H */
/** @} */