123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175 |
- #include <AP_HAL/AP_HAL.h>
- #include <AP_Common/AP_Common.h>
- #include <AP_Math/AP_Math.h>
- #include "AP_BattMonitor.h"
- #include "AP_BattMonitor_SMBus_Maxell.h"
- #include <utility>
- #define BATTMONITOR_SMBUS_MAXELL_NUM_CELLS 6
- uint8_t maxell_cell_ids[] = { 0x3f,
- 0x3e,
- 0x3d,
- 0x3c,
- 0x3b,
- 0x3a};
- #define SMBUS_READ_BLOCK_MAXIMUM_TRANSFER 0x20
- AP_BattMonitor_SMBus_Maxell::AP_BattMonitor_SMBus_Maxell(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))
- {}
- void AP_BattMonitor_SMBus_Maxell::timer()
- {
-
- if (!check_pec_support()) {
- return;
- }
- uint16_t data;
- uint32_t tnow = AP_HAL::micros();
-
- if (read_word(BATTMONITOR_SMBUS_VOLTAGE, data)) {
- _state.voltage = (float)data / 1000.0f;
- _state.last_time_micros = tnow;
- _state.healthy = true;
- }
-
- for (uint8_t i = 0; i < BATTMONITOR_SMBUS_MAXELL_NUM_CELLS; i++) {
- if (read_word(maxell_cell_ids[i], data)) {
- _has_cell_voltages = true;
- _state.cell_voltages.cells[i] = data;
- } else {
- _state.cell_voltages.cells[i] = UINT16_MAX;
- }
- }
-
- if ((tnow - _state.last_time_micros) > AP_BATTMONITOR_SMBUS_TIMEOUT_MICROS) {
- _state.healthy = false;
- return;
- }
-
- if (read_word(BATTMONITOR_SMBUS_CURRENT, data)) {
- _state.current_amps = -(float)((int16_t)data) / 1000.0f;
- _state.last_time_micros = tnow;
- }
- read_full_charge_capacity();
-
- read_remaining_capacity();
- read_temp();
- read_serial_number();
- }
- uint8_t AP_BattMonitor_SMBus_Maxell::read_block(uint8_t reg, uint8_t* data, bool append_zero) const
- {
-
- uint8_t bufflen;
-
- if (!_dev->read_registers(reg, &bufflen, 1)) {
- return 0;
- }
-
- if (bufflen == 0 || bufflen > SMBUS_READ_BLOCK_MAXIMUM_TRANSFER) {
- return 0;
- }
-
- const uint8_t read_size = bufflen + 1 + (_pec_supported ? 1 : 0);
- uint8_t buff[read_size];
-
- if (!_dev->read_registers(reg, buff, read_size)) {
- return 0;
- }
-
- if (_pec_supported) {
- uint8_t pec = get_PEC(AP_BATTMONITOR_SMBUS_I2C_ADDR, reg, true, buff, bufflen+1);
- if (pec != buff[bufflen+1]) {
- return 0;
- }
- }
-
- memcpy(data, &buff[1], bufflen);
-
- if (append_zero) {
- data[bufflen] = '\0';
- }
-
- return bufflen;
- }
- bool AP_BattMonitor_SMBus_Maxell::check_pec_support()
- {
-
- if (_pec_confirmed) {
- return true;
- }
-
- uint16_t data;
- if (!read_word(BATTMONITOR_SMBUS_SPECIFICATION_INFO, data)) {
- return false;
- }
-
- uint8_t version = (data & 0xF0) >> 4;
-
- if (version < 3) {
- _pec_supported = false;
- _pec_confirmed = true;
- return true;
- }
-
- uint8_t buff[SMBUS_READ_BLOCK_MAXIMUM_TRANSFER + 1];
- if (read_block(BATTMONITOR_SMBUS_MANUFACTURE_NAME, buff, true)) {
-
- if (strcmp((char*)buff, "Hitachi maxell") == 0) {
- _pec_supported = false;
- _pec_confirmed = true;
- return true;
- }
- }
-
- _pec_supported = true;
- _pec_confirmed = true;
- return true;
- }
|