2019-11-05 12:33:06 +01:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup tests
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Test application for the INA3221 sensor driver
|
|
|
|
*
|
|
|
|
* @author Fabian Hüßler <fabian.huessler@ovgu.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2020-01-11 13:32:01 +01:00
|
|
|
#include <stdio.h>
|
2019-11-05 12:33:06 +01:00
|
|
|
#include <string.h>
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
#include "xtimer.h"
|
|
|
|
#include "fmt.h"
|
|
|
|
#include "ina3221_internal.h"
|
|
|
|
#include "ina3221_params.h"
|
|
|
|
#include "ina3221.h"
|
|
|
|
|
2020-10-02 08:17:06 +02:00
|
|
|
/* Issue critical alert if shunt voltage is above this value */
|
2019-11-05 12:33:06 +01:00
|
|
|
#define CRIT_ALERT_LIM_UV (2500)
|
|
|
|
/* Issue warning alert if shunt voltage is above this value */
|
|
|
|
#define WARN_ALERT_LIM_UV (1600)
|
|
|
|
/* Issue power valid alert if bus voltage is below this value */
|
|
|
|
#define PV_LOWER_LIM_MV (4500)
|
|
|
|
/* Issue power valid alert if bus voltage is above this value */
|
|
|
|
#define PV_UPPER_LIM_MV (5500)
|
|
|
|
/* Issue critical alert if the sum of all shunt voltages is above this value */
|
|
|
|
#define SUM_SHUNT_ALERT_LIM_UV (8000)
|
|
|
|
|
|
|
|
#define COL (" ")
|
|
|
|
#define THEAD ( \
|
|
|
|
"channel | shunt [uV] | bus [mV] | current [uA] | power [uW] ")
|
|
|
|
#define HLINE ( \
|
|
|
|
"--------------+--------------+--------------+--------------+--------------")
|
|
|
|
|
|
|
|
#ifdef MODULE_INA3221_ALERTS
|
|
|
|
static void warning_alert(void *arg)
|
|
|
|
{
|
|
|
|
ina3221_t *dev = (ina3221_t *)arg;
|
|
|
|
|
|
|
|
(void)dev;
|
|
|
|
puts("WARNING");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void critical_alert(void *arg)
|
|
|
|
{
|
|
|
|
ina3221_t *dev = (ina3221_t *)arg;
|
|
|
|
|
|
|
|
(void)dev;
|
|
|
|
puts("CRITICAL");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void timing_control_alert(void *arg)
|
|
|
|
{
|
|
|
|
ina3221_t *dev = (ina3221_t *)arg;
|
|
|
|
|
|
|
|
(void)dev;
|
|
|
|
puts("TIMING CONTROL");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void power_valid_alert(void *arg)
|
|
|
|
{
|
|
|
|
ina3221_t *dev = (ina3221_t *)arg;
|
|
|
|
|
|
|
|
(void)dev;
|
|
|
|
puts("POWER VALID");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
ina3221_t dev;
|
|
|
|
|
|
|
|
for (int32_t i = INA3221_MIN_SHUNT_UV; i <= INA3221_MAX_SHUNT_UV;
|
|
|
|
i += INA3221_SHUNT_VOLTAGE_PRECISION_UV) {
|
|
|
|
int16_t reg = shunt_voltage_uv_to_reg_val(i);
|
|
|
|
int32_t sv = reg_val_to_shunt_voltage_uv(reg);
|
|
|
|
int16_t _reg = shunt_voltage_uv_to_reg_val(sv);
|
|
|
|
if (reg != _reg || i != sv) {
|
|
|
|
printf(
|
|
|
|
"[FAILURE] shunt V conversiom: shunt V: %" PRId32 ", REG: %" PRId16 ", shunt V: %" PRId32 ", REG: %" PRId16 "\n", i, reg, sv,
|
|
|
|
_reg);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int32_t i = INA3221_MIN_BUS_MV; i <= INA3221_MAX_BUS_MV;
|
|
|
|
i += INA3221_BUS_VOLTAGE_PRECISION_MV) {
|
|
|
|
int16_t reg = bus_voltage_mv_to_reg_val(i);
|
|
|
|
int32_t bv = reg_val_to_bus_voltage_mv(reg);
|
|
|
|
int16_t _reg = bus_voltage_mv_to_reg_val(bv);
|
|
|
|
if (reg != _reg || i != bv) {
|
|
|
|
printf(
|
|
|
|
"[FAILURE] bus V conversion: bus V: %" PRId32 ", REG: %" PRId16 ", bus V: %" PRId32 ", REG: %" PRId16 "\n", i, reg, bv,
|
|
|
|
_reg);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int32_t i = INA3221_MIN_SHUNT_SUM_UV; i <= INA3221_MAX_SHUNT_SUM_UV;
|
|
|
|
i += INA3221_SHUNT_VOLTAGE_PRECISION_UV) {
|
|
|
|
int16_t reg = sum_shunt_voltage_uv_to_reg_val(i);
|
|
|
|
int32_t sv = sum_reg_val_to_shunt_voltage_uv(reg);
|
|
|
|
int16_t _reg = sum_shunt_voltage_uv_to_reg_val(sv);
|
|
|
|
if (reg != _reg || i != sv) {
|
|
|
|
printf(
|
|
|
|
"[FAILURE] shunt sum V conversion: sum V: %" PRId32 ", REG: %" PRId16 ", sum V: %" PRId32 ", REG: %" PRId16 "\n", i, reg, sv,
|
|
|
|
_reg);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
status = ina3221_init(&dev, &ina3221_params[0]);
|
2021-02-02 11:16:32 +01:00
|
|
|
if (status != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_init: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_init");
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MODULE_INA3221_ALERTS
|
|
|
|
status = ina3221_enable_warning_alert(&dev, warning_alert, &dev);
|
|
|
|
if (status == -ENOTSUP) {
|
|
|
|
puts("[WARNING] INA3221_ALERT_WRN not supported");
|
|
|
|
}
|
2021-02-02 11:16:32 +01:00
|
|
|
else if (status != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_enable_alert INA3221_ALERT_WRN: %d\n",
|
|
|
|
status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = ina3221_enable_critical_alert(&dev, critical_alert, &dev);
|
|
|
|
if (status == -ENOTSUP) {
|
|
|
|
puts("[WARNING] INA3221_ALERT_CRT not supported");
|
|
|
|
}
|
2021-02-02 11:16:32 +01:00
|
|
|
else if (status != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_enable_alert INA3221_ALERT_CRT: %d\n",
|
|
|
|
status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = ina3221_enable_timing_control_alert(&dev, timing_control_alert,
|
|
|
|
&dev);
|
|
|
|
if (status == -ENOTSUP) {
|
|
|
|
puts("[WARNING] INA3221_ALERT_TC not supported");
|
|
|
|
}
|
2021-02-02 11:16:32 +01:00
|
|
|
else if (status != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_enable_alert INA3221_ALERT_TC: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = ina3221_enable_power_valid_alert(&dev, power_valid_alert, &dev);
|
|
|
|
if (status == -ENOTSUP) {
|
|
|
|
puts("[WARNING] INA3221_ALERT_PV not supported");
|
|
|
|
}
|
2021-02-02 11:16:32 +01:00
|
|
|
else if (status != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_enable_alert INA3221_ALERT_PV: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
puts("[SUCCESS] ina3221_enable_alert");
|
|
|
|
#endif
|
|
|
|
|
2021-02-02 11:16:32 +01:00
|
|
|
int32_t shunt_uv[INA3221_NUM_CH];
|
|
|
|
int16_t bus_mv[INA3221_NUM_CH];
|
|
|
|
int32_t sum_shunt_uv;
|
|
|
|
int32_t current_ua[INA3221_NUM_CH];
|
|
|
|
int32_t power_uw[INA3221_NUM_CH];
|
|
|
|
uint16_t flags;
|
|
|
|
ina3221_ch_t ch = 0;
|
|
|
|
ina3221_get_enable_channel(&dev, &ch);
|
|
|
|
|
|
|
|
if (ch != ina3221_read_shunt_uv(&dev, shunt_uv, &flags)) {
|
|
|
|
puts("[FAILURE] ina3221_read_shunt_uv");
|
2019-11-05 12:33:06 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_read_shunt_uv");
|
|
|
|
}
|
|
|
|
|
2021-02-02 11:16:32 +01:00
|
|
|
if (ch != ina3221_read_bus_mv(&dev, bus_mv, &flags)) {
|
|
|
|
puts("[FAILURE] ina3221_read_bus_mv");
|
2019-11-05 12:33:06 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_read_bus_mv");
|
|
|
|
}
|
|
|
|
|
2021-02-02 11:16:32 +01:00
|
|
|
if ((status = ina3221_set_enable_sum_channel(&dev, ch)) != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_set_enable_sum_channel: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
2021-02-02 11:16:32 +01:00
|
|
|
puts("[SUCCESS] ina3221_set_enable_sum_channel");
|
2019-11-05 12:33:06 +01:00
|
|
|
}
|
|
|
|
|
2021-02-02 11:16:32 +01:00
|
|
|
if ((status = ina3221_read_shunt_sum_uv(&dev, &sum_shunt_uv, &flags)) != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_read_shunt_sum_uv: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_read_shunt_sum_uv");
|
|
|
|
}
|
|
|
|
|
2021-02-02 11:16:32 +01:00
|
|
|
if ((status = INA3221_TRIGGER_SHUNT_AND_BUS(&dev)) != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] INA3221_TRIGGER_SHUNT_AND_BUS: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] INA3221_TRIGGER_SHUNT_AND_BUS");
|
|
|
|
}
|
|
|
|
flags = 0;
|
|
|
|
while (!(flags & INA3221_FLAG_CONV_READY)) {
|
2021-02-02 11:16:32 +01:00
|
|
|
if ((status = ina3221_read_flags(&dev, &flags)) != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_read_flags: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
2021-02-02 11:16:32 +01:00
|
|
|
if (ch != ina3221_read_shunt_uv(&dev, shunt_uv, &flags)) {
|
|
|
|
puts("[FAILURE] ina3221_read_shunt_uv (triggered)");
|
2019-11-05 12:33:06 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_read_shunt_uv (triggered)");
|
|
|
|
}
|
2021-02-02 11:16:32 +01:00
|
|
|
if (ch != ina3221_read_bus_mv(&dev, bus_mv, &flags)) {
|
|
|
|
puts("[FAILURE] ina3221_read_bus_mv (triggered)");
|
2019-11-05 12:33:06 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_read_bus_mv (triggered)");
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t crit_alert_lim = CRIT_ALERT_LIM_UV;
|
2021-02-02 11:16:32 +01:00
|
|
|
ina3221_ch_t crit_ch = INA3221_CH1 | INA3221_CH2 | INA3221_CH3;
|
|
|
|
if (crit_ch != ina3221_set_crit_alert_limit(&dev, crit_ch, crit_alert_lim)) {
|
|
|
|
puts("[FAILURE] ina3221_set_crit_alert_limit");
|
2019-11-05 12:33:06 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_set_crit_alert_limit");
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t warn_alert_lim = WARN_ALERT_LIM_UV;
|
2021-02-02 11:16:32 +01:00
|
|
|
ina3221_ch_t warn_ch = INA3221_CH1 | INA3221_CH2 | INA3221_CH3;
|
|
|
|
if (warn_ch != ina3221_set_warn_alert_limit(&dev, warn_ch, warn_alert_lim)) {
|
|
|
|
puts("[FAILURE] ina3221_set_warn_alert_limit");
|
2019-11-05 12:33:06 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_set_warn_alert_limit");
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t pv_lower_lim = PV_LOWER_LIM_MV;
|
2021-02-02 11:16:32 +01:00
|
|
|
if ((status = ina3221_set_power_valid_lower_limit(&dev, pv_lower_lim)) != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_set_power_valid_lower_limit: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_set_power_valid_lower_limit");
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t pv_upper_lim = PV_UPPER_LIM_MV;
|
2021-02-02 11:16:32 +01:00
|
|
|
if ((status = ina3221_set_power_valid_upper_limit(&dev, pv_upper_lim)) != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_set_power_valid_upper_limit: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_set_power_valid_upper_limit");
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t sum_shunt_alert_lim = SUM_SHUNT_ALERT_LIM_UV;
|
2021-02-02 11:16:32 +01:00
|
|
|
if ((status = ina3221_set_shunt_voltage_sum_alert_limit(&dev, sum_shunt_alert_lim))
|
|
|
|
!= 0) {
|
|
|
|
printf("[FAILURE] ina3221_set_shunt_voltage_sum_alert_limit: %d\n", status);
|
2019-11-05 12:33:06 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[SUCCESS] ina3221_set_shunt_voltage_sum_alert_limit");
|
|
|
|
}
|
|
|
|
|
2021-02-02 11:16:32 +01:00
|
|
|
if ((status = ina3221_set_mode(&dev, INA3221_MODE_CONTINUOUS_SHUNT_BUS)) != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_set_mode: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
2021-02-02 11:16:32 +01:00
|
|
|
char line_buffer[sizeof(THEAD)];
|
|
|
|
char col_buffer[sizeof(COL)];
|
2019-11-05 12:33:06 +01:00
|
|
|
while (1) {
|
2021-02-02 11:16:32 +01:00
|
|
|
if ((status = ina3221_read_flags(&dev, &flags)) != 0) {
|
2019-11-05 12:33:06 +01:00
|
|
|
printf("[FAILURE] ina3221_read_flags: %d\n", status);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (!(flags & INA3221_FLAG_CONV_READY)) {
|
|
|
|
xtimer_sleep(2);
|
|
|
|
continue;
|
|
|
|
}
|
2021-02-02 11:16:32 +01:00
|
|
|
if (ch != ina3221_read_shunt_uv(&dev, shunt_uv, NULL)) {
|
|
|
|
puts("[FAILURE] ina3221_read_shunt_uv");
|
2019-11-05 12:33:06 +01:00
|
|
|
return 1;
|
|
|
|
}
|
2021-02-02 11:16:32 +01:00
|
|
|
if (ch != ina3221_read_bus_mv(&dev, bus_mv, NULL)) {
|
|
|
|
puts("[FAILURE] ina3221_read_bus_mv");
|
2019-11-05 12:33:06 +01:00
|
|
|
return 1;
|
|
|
|
}
|
2021-02-02 11:16:32 +01:00
|
|
|
ina3221_calculate_current_ua(ch, dev.params.rshunt_mohm, shunt_uv, current_ua);
|
|
|
|
ina3221_calculate_power_uw(ch, bus_mv, current_ua, power_uw);
|
2019-11-05 12:33:06 +01:00
|
|
|
puts(THEAD);
|
|
|
|
puts(HLINE);
|
|
|
|
for (int8_t i = 0; i < INA3221_NUM_CH; i++) {
|
|
|
|
char *p = line_buffer;
|
|
|
|
strcpy(col_buffer, COL);
|
|
|
|
fmt_char(col_buffer, '0' + i);
|
|
|
|
p += sprintf(p, "%s|", col_buffer);
|
|
|
|
strcpy(col_buffer, COL);
|
|
|
|
fmt_s32_dec(col_buffer, shunt_uv[i]);
|
|
|
|
p += sprintf(p, "%s|", col_buffer);
|
|
|
|
strcpy(col_buffer, COL);
|
|
|
|
fmt_s16_dec(col_buffer, bus_mv[i]);
|
|
|
|
p += sprintf(p, "%s|", col_buffer);
|
|
|
|
strcpy(col_buffer, COL);
|
|
|
|
fmt_s32_dec(col_buffer, current_ua[i]);
|
|
|
|
p += sprintf(p, "%s|", col_buffer);
|
|
|
|
strcpy(col_buffer, COL);
|
|
|
|
fmt_s32_dec(col_buffer, power_uw[i]);
|
|
|
|
p += sprintf(p, "%s", col_buffer);
|
|
|
|
printf("%s\n", line_buffer);
|
|
|
|
}
|
|
|
|
puts("");
|
|
|
|
snprintf(line_buffer, sizeof(line_buffer),
|
|
|
|
"FLAGS: CF1:%d CF2:%d CF3:%d SF:%d WF1:%d WF2:%d WF3:%d PVF:%d TCF:%d CVRF:%d",
|
|
|
|
(flags & INA3221_FLAG_CRIT_ALERT_CH1) ? 1 : 0,
|
|
|
|
(flags & INA3221_FLAG_CRIT_ALERT_CH2) ? 1 : 0,
|
|
|
|
(flags & INA3221_FLAG_CRIT_ALERT_CH3) ? 1 : 0,
|
|
|
|
(flags & INA3221_FLAG_SHUNT_SUM_ALERT) ? 1 : 0,
|
|
|
|
(flags & INA3221_FLAG_WARN_ALERT_CH1) ? 1 : 0,
|
|
|
|
(flags & INA3221_FLAG_WARN_ALERT_CH2) ? 1 : 0,
|
|
|
|
(flags & INA3221_FLAG_WARN_ALERT_CH3) ? 1 : 0,
|
|
|
|
(flags & INA3221_FLAG_POWER_VALID) ? 1 : 0,
|
|
|
|
(flags & INA3221_FLAG_TIMING_CONTROL) ? 1 : 0,
|
|
|
|
(flags & INA3221_FLAG_CONV_READY) ? 1 : 0);
|
|
|
|
printf("%s\n", line_buffer);
|
|
|
|
puts(HLINE);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|