080e477b36
Adds a driver for the NeoDesign BMS, with variable cell count.
76 lines
2.5 KiB
C++
76 lines
2.5 KiB
C++
#include <AP_HAL/AP_HAL.h>
|
|
#include <AP_Common/AP_Common.h>
|
|
#include "AP_BattMonitor.h"
|
|
#include "AP_BattMonitor_SMBus_NeoDesign.h"
|
|
|
|
#define BATTMONITOR_ND_CELL_COUNT 0x5C // cell-count register
|
|
#define BATTMONITOR_ND_CELL_START 0x30 // first cell register
|
|
|
|
// Constructor
|
|
AP_BattMonitor_SMBus_NeoDesign::AP_BattMonitor_SMBus_NeoDesign(AP_BattMonitor &mon,
|
|
AP_BattMonitor::BattMonitor_State &mon_state,
|
|
AP_BattMonitor_Params ¶ms,
|
|
AP_HAL::OwnPtr<AP_HAL::I2CDevice> dev)
|
|
: AP_BattMonitor_SMBus(mon, mon_state, params, std::move(dev))
|
|
{
|
|
_pec_supported = true;
|
|
}
|
|
|
|
void AP_BattMonitor_SMBus_NeoDesign::timer()
|
|
{
|
|
uint16_t data;
|
|
uint32_t tnow = AP_HAL::micros();
|
|
|
|
// Get the cell count once, it's not likely to change in flight
|
|
if (_cell_count == 0) {
|
|
if (read_word(BATTMONITOR_ND_CELL_COUNT, data)) {
|
|
_cell_count = data;
|
|
} else {
|
|
return; // something wrong, don't try anything else
|
|
}
|
|
}
|
|
|
|
bool read_all_cells = true;
|
|
for(uint8_t i = 0; i < _cell_count; ++i) {
|
|
if(read_word(BATTMONITOR_ND_CELL_START + i, data)) {
|
|
_state.cell_voltages.cells[i] = data;
|
|
_has_cell_voltages = true;
|
|
} else {
|
|
read_all_cells = false;
|
|
}
|
|
}
|
|
|
|
if (read_all_cells && (_cell_count > 0)) {
|
|
uint32_t summed = 0;
|
|
for (int i = 0; i < _cell_count; i++) {
|
|
summed += _state.cell_voltages.cells[i];
|
|
}
|
|
_state.voltage = (float)summed * 1e-3f;
|
|
_state.last_time_micros = tnow;
|
|
_state.healthy = true;
|
|
} else if (read_word(BATTMONITOR_SMBUS_VOLTAGE, data)) {
|
|
// fallback to the voltage register if we didn't manage to poll the cells
|
|
_state.voltage = (float)data * 1e-3f;
|
|
_state.last_time_micros = tnow;
|
|
_state.healthy = true;
|
|
}
|
|
|
|
// timeout after 5 seconds
|
|
if ((tnow - _state.last_time_micros) > AP_BATTMONITOR_SMBUS_TIMEOUT_MICROS) {
|
|
_state.healthy = false;
|
|
// do not attempt to read any more data from battery
|
|
return;
|
|
}
|
|
|
|
// read current (A)
|
|
if (read_word(BATTMONITOR_SMBUS_CURRENT, data)) {
|
|
_state.current_amps = -(float)((int16_t)data) * 1e-3f;
|
|
_state.last_time_micros = tnow;
|
|
}
|
|
|
|
read_full_charge_capacity();
|
|
read_remaining_capacity();
|
|
read_temp();
|
|
}
|
|
|