You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
130 lines
4.0 KiB
130 lines
4.0 KiB
10 years ago
|
/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
||
|
#include <AP_HAL.h>
|
||
|
#include <AP_Common.h>
|
||
|
#include <AP_Math.h>
|
||
|
#include "AP_BattMonitor.h"
|
||
|
#include "AP_BattMonitor_SMBus_I2C.h"
|
||
|
|
||
|
extern const AP_HAL::HAL& hal;
|
||
|
|
||
|
#include <AP_HAL.h>
|
||
|
|
||
|
#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
|