1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/tests/driver_ina3221/main.c
2021-07-07 08:40:02 +02:00

356 lines
11 KiB
C

/*
* 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>
*
* @}
*/
#include <stdio.h>
#include <string.h>
#include "periph/gpio.h"
#include "xtimer.h"
#include "fmt.h"
#include "ina3221_internal.h"
#include "ina3221_params.h"
#include "ina3221.h"
/* Issue critical alert if shunt voltage is above this value */
#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]);
if (status != 0) {
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");
}
else if (status != 0) {
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");
}
else if (status != 0) {
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");
}
else if (status != 0) {
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");
}
else if (status != 0) {
printf("[FAILURE] ina3221_enable_alert INA3221_ALERT_PV: %d\n", status);
return 1;
}
puts("[SUCCESS] ina3221_enable_alert");
#endif
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");
return 1;
}
else {
puts("[SUCCESS] ina3221_read_shunt_uv");
}
if (ch != ina3221_read_bus_mv(&dev, bus_mv, &flags)) {
puts("[FAILURE] ina3221_read_bus_mv");
return 1;
}
else {
puts("[SUCCESS] ina3221_read_bus_mv");
}
if ((status = ina3221_set_enable_sum_channel(&dev, ch)) != 0) {
printf("[FAILURE] ina3221_set_enable_sum_channel: %d\n", status);
return 1;
}
else {
puts("[SUCCESS] ina3221_set_enable_sum_channel");
}
if ((status = ina3221_read_shunt_sum_uv(&dev, &sum_shunt_uv, &flags)) != 0) {
printf("[FAILURE] ina3221_read_shunt_sum_uv: %d\n", status);
return 1;
}
else {
puts("[SUCCESS] ina3221_read_shunt_sum_uv");
}
if ((status = INA3221_TRIGGER_SHUNT_AND_BUS(&dev)) != 0) {
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)) {
if ((status = ina3221_read_flags(&dev, &flags)) != 0) {
printf("[FAILURE] ina3221_read_flags: %d\n", status);
return 1;
}
}
if (ch != ina3221_read_shunt_uv(&dev, shunt_uv, &flags)) {
puts("[FAILURE] ina3221_read_shunt_uv (triggered)");
return 1;
}
else {
puts("[SUCCESS] ina3221_read_shunt_uv (triggered)");
}
if (ch != ina3221_read_bus_mv(&dev, bus_mv, &flags)) {
puts("[FAILURE] ina3221_read_bus_mv (triggered)");
return 1;
}
else {
puts("[SUCCESS] ina3221_read_bus_mv (triggered)");
}
int32_t crit_alert_lim = CRIT_ALERT_LIM_UV;
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");
return 1;
}
else {
puts("[SUCCESS] ina3221_set_crit_alert_limit");
}
int32_t warn_alert_lim = WARN_ALERT_LIM_UV;
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");
return 1;
}
else {
puts("[SUCCESS] ina3221_set_warn_alert_limit");
}
int32_t pv_lower_lim = PV_LOWER_LIM_MV;
if ((status = ina3221_set_power_valid_lower_limit(&dev, pv_lower_lim)) != 0) {
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;
if ((status = ina3221_set_power_valid_upper_limit(&dev, pv_upper_lim)) != 0) {
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;
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);
return 1;
}
else {
puts("[SUCCESS] ina3221_set_shunt_voltage_sum_alert_limit");
}
if ((status = ina3221_set_mode(&dev, INA3221_MODE_CONTINUOUS_SHUNT_BUS)) != 0) {
printf("[FAILURE] ina3221_set_mode: %d\n", status);
return 1;
}
char line_buffer[sizeof(THEAD)];
char col_buffer[sizeof(COL)];
while (1) {
if ((status = ina3221_read_flags(&dev, &flags)) != 0) {
printf("[FAILURE] ina3221_read_flags: %d\n", status);
return 1;
}
if (!(flags & INA3221_FLAG_CONV_READY)) {
xtimer_sleep(2);
continue;
}
if (ch != ina3221_read_shunt_uv(&dev, shunt_uv, NULL)) {
puts("[FAILURE] ina3221_read_shunt_uv");
return 1;
}
if (ch != ina3221_read_bus_mv(&dev, bus_mv, NULL)) {
puts("[FAILURE] ina3221_read_bus_mv");
return 1;
}
ina3221_calculate_current_ua(ch, dev.params.rshunt_mohm, shunt_uv, current_ua);
ina3221_calculate_power_uw(ch, bus_mv, current_ua, power_uw);
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;
}