diff --git a/libraries/AP_BattMonitor/AP_BattMonitor_SMBus_I2C.cpp b/libraries/AP_BattMonitor/AP_BattMonitor_SMBus_I2C.cpp new file mode 100644 index 0000000000..36dd8437e0 --- /dev/null +++ b/libraries/AP_BattMonitor/AP_BattMonitor_SMBus_I2C.cpp @@ -0,0 +1,129 @@ +/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- +#include +#include +#include +#include "AP_BattMonitor.h" +#include "AP_BattMonitor_SMBus_I2C.h" + +extern const AP_HAL::HAL& hal; + +#include + +#if CONFIG_HAL_BOARD != HAL_BOARD_PX4 + +#define BATTMONITOR_SMBUS_I2C_ADDR 0x0B // default I2C bus address + +#define BATTMONITOR_SMBUS_TEMP 0x08 // temperature register +#define BATTMONITOR_SMBUS_VOLTAGE 0x09 // voltage register +#define BATTMONITOR_SMBUS_FULL_CHARGE_CAPACITY 0x10 // full capacity register +#define BATTMONITOR_SMBUS_BATTERY_STATUS 0x16 // battery status register including alarms +#define BATTMONITOR_SMBUS_DESIGN_CAPACITY 0x18 // design capacity register +#define BATTMONITOR_SMBUS_DESIGN_VOLTAGE 0x19 // design voltage register +#define BATTMONITOR_SMBUS_SERIALNUM 0x1c // serial number register +#define BATTMONITOR_SMBUS_MANUFACTURE_NAME 0x20 // manufacturer name +#define BATTMONITOR_SMBUS_DEVICE_NAME 0x21 // device name +#define BATTMONITOR_SMBUS_DEVICE_CHEMISTRY 0x22 // device chemistry +#define BATTMONITOR_SMBUS_MANUFACTURE_INFO 0x25 // manufacturer info including cell voltage +#define BATTMONITOR_SMBUS_CELL_VOLTAGE 0x28 // cell voltage register +#define BATTMONITOR_SMBUS_CURRENT 0x2a // current register + +// Constructor +AP_BattMonitor_SMBus_I2C::AP_BattMonitor_SMBus_I2C(AP_BattMonitor &mon, uint8_t instance, AP_BattMonitor::BattMonitor_State &mon_state) : + AP_BattMonitor_SMBus(mon, instance, mon_state) +{} + +/// Read the battery voltage and current. Should be called at 10hz +void AP_BattMonitor_SMBus_I2C::read() +{ + uint16_t data; + uint8_t buff[4]; + uint32_t tnow = hal.scheduler->micros(); + + // read voltage + if (read_word(BATTMONITOR_SMBUS_VOLTAGE, data)) { + _state.voltage = (float)data / 1000.0f; + _state.last_time_micros = tnow; + } + + // read current + if (read_block(BATTMONITOR_SMBUS_CURRENT, buff, 4, false) == 4) { + _state.current_amps = (float)((int32_t)((uint32_t)buff[3]<<24 | (uint32_t)buff[2]<<16 | (uint32_t)buff[1]<<8 | (uint32_t)buff[0])) / 1000.0f; + _state.last_time_micros = tnow; + } +} + +// read word from register +// returns true if read was succesful, false if failed +bool AP_BattMonitor_SMBus_I2C::read_word(uint8_t reg, uint16_t& data) const +{ + uint8_t buff[2]; // buffer to hold results + + // get pointer to i2c bus semaphore + AP_HAL::Semaphore* i2c_sem = hal.i2c->get_semaphore(); + + // take i2c bus semaphore + if (!i2c_sem->take_nonblocking()) { + i2c_sem->give(); + return false; + } + + // read two bytes + if (hal.i2c->readRegisters(BATTMONITOR_SMBUS_I2C_ADDR, reg, 2, buff) != 0) { + i2c_sem->give(); + return false; + } + + // convert buffer to word + data = (uint16_t)buff[1]<<8 | (uint16_t)buff[0]; + + // give back i2c semaphore + i2c_sem->give(); + + // return success + return true; +} + +// read_block - returns number of characters read if successful, zero if unsuccessful +uint8_t AP_BattMonitor_SMBus_I2C::read_block(uint8_t reg, uint8_t* data, uint8_t max_len, bool append_zero) const +{ + uint8_t buff[max_len+1]; // buffer to hold results + + // get pointer to i2c bus semaphore + AP_HAL::Semaphore* i2c_sem = hal.i2c->get_semaphore(); + + // take i2c bus semaphore + if (!i2c_sem->take_nonblocking()) { + i2c_sem->give(); + return 0; + } + + // read bytes + if (hal.i2c->readRegisters(BATTMONITOR_SMBUS_I2C_ADDR, reg, max_len+1, buff) != 0) { + i2c_sem->give(); + return 0; + } + + // give back i2c semaphore + i2c_sem->give(); + + // get length + uint8_t bufflen = buff[0]; + + // sanity check length returned by smbus + if (bufflen == 0 || bufflen > max_len) { + return 0; + } + + // copy data + memcpy(data, &buff[1], bufflen); + + // optionally add zero to end + if (append_zero) { + data[bufflen] = '\0'; + } + + // return success + return bufflen; +} + +#endif // CONFIG_HAL_BOARD != HAL_BOARD_PX4 diff --git a/libraries/AP_BattMonitor/AP_BattMonitor_SMBus_I2C.h b/libraries/AP_BattMonitor/AP_BattMonitor_SMBus_I2C.h new file mode 100644 index 0000000000..6079f2b221 --- /dev/null +++ b/libraries/AP_BattMonitor/AP_BattMonitor_SMBus_I2C.h @@ -0,0 +1,30 @@ +/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- +#ifndef AP_BATTMONITOR_SMBUS_I2C_H +#define AP_BATTMONITOR_SMBUS_I2C_H + +#include +#include +#include +#include "AP_BattMonitor_SMBus.h" + +class AP_BattMonitor_SMBus_I2C : public AP_BattMonitor_SMBus +{ +public: + + // Constructor + AP_BattMonitor_SMBus_I2C(AP_BattMonitor &mon, uint8_t instance, AP_BattMonitor::BattMonitor_State &mon_state); + + // Read the battery voltage and current. Should be called at 10hz + void read(); + +private: + + // read word from register + // returns true if read was successful, false if failed + bool read_word(uint8_t reg, uint16_t& data) const; + + // read_block - returns number of characters read if successful, zero if unsuccessful + uint8_t read_block(uint8_t reg, uint8_t* data, uint8_t max_len, bool append_zero) const; +}; + +#endif // AP_BATTMONITOR_SMBUS_I2C_H