// Modified or adapted from https://github.com/sparkfun/SparkFun_BQ27441_Arduino_Library // // Originally Licensed under the MIT /* The MIT License (MIT) Copyright (c) 2016 SparkFun Electronics, Inc. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /****************************************************************************** SparkFunBQ27441.cpp BQ27441 Arduino Library Main Source File Jim Lindblom @ SparkFun Electronics May 9, 2016 https://github.com/sparkfun/SparkFun_BQ27441_Arduino_Library Implementation of all features of the BQ27441 LiPo Fuel Gauge. Hardware Resources: - Arduino Development Board - SparkFun Battery Babysitter Development environment specifics: Arduino 1.6.7 SparkFun Battery Babysitter v1.0 Arduino Uno (any 'duino should do) ******************************************************************************/ #include "SparkFunBQ27441.h" /***************************************************************************** ************************** Initialization Functions ************************* *****************************************************************************/ // Initializes class variables BQ27441::BQ27441() : _deviceAddress(BQ72441_I2C_ADDRESS), _sealFlag(false), _userConfigControl(false) { } static uint8_t constrain(uint8_t x, uint8_t min, uint8_t max) { if (x < min) return min; if (x > max) return max; return x; } // Initializes I2C and verifies communication with the BQ27441. bool BQ27441::begin(void) { uint16_t deviceID = 0; // Assume the I2C is already initialized from other things deviceID = deviceType(); // Read deviceType from BQ27441 if (deviceID == BQ27441_DEVICE_ID) { return true; // If device ID is valid, return true } return false; // Otherwise return false } // Configures the design capacity of the connected battery. bool BQ27441::setCapacity(uint16_t capacity) { // Write to STATE subclass (82) of BQ27441 extended memory. // Offset 0x0A (10) // Design capacity is a 2-byte piece of data - MSB first // Unit: mAh uint8_t capMSB = capacity >> 8; uint8_t capLSB = capacity & 0x00FF; uint8_t capacityData[2] = {capMSB, capLSB}; return writeExtendedData(BQ27441_ID_STATE, 10, capacityData, 2); } // Configures the design energy of the connected battery. bool BQ27441::setDesignEnergy(uint16_t energy) { // Write to STATE subclass (82) of BQ27441 extended memory. // Offset 0x0C (12) // Design energy is a 2-byte piece of data - MSB first // Unit: mWh uint8_t enMSB = energy >> 8; uint8_t enLSB = energy & 0x00FF; uint8_t energyData[2] = {enMSB, enLSB}; return writeExtendedData(BQ27441_ID_STATE, 12, energyData, 2); } // Configures the terminate voltage. bool BQ27441::setTerminateVoltage(uint16_t voltage) { // Write to STATE subclass (82) of BQ27441 extended memory. // Offset 0x0F (16) // Termiante voltage is a 2-byte piece of data - MSB first // Unit: mV // Min 2500, Max 3700 if(voltage<2500) voltage=2500; if(voltage>3700) voltage=3700; uint8_t tvMSB = voltage >> 8; uint8_t tvLSB = voltage & 0x00FF; uint8_t tvData[2] = {tvMSB, tvLSB}; return writeExtendedData(BQ27441_ID_STATE, 16, tvData, 2); } // Configures taper rate of connected battery. bool BQ27441::setTaperRate(uint16_t rate) { // Write to STATE subclass (82) of BQ27441 extended memory. // Offset 0x1B (27) // Termiante voltage is a 2-byte piece of data - MSB first // Unit: 0.1h // Max 2000 if(rate>2000) rate=2000; uint8_t trMSB = rate >> 8; uint8_t trLSB = rate & 0x00FF; uint8_t trData[2] = {trMSB, trLSB}; return writeExtendedData(BQ27441_ID_STATE, 27, trData, 2); } /***************************************************************************** ********************** Battery Characteristics Functions ******************** *****************************************************************************/ // Reads and returns the battery voltage uint16_t BQ27441::voltage(void) { return readWord(BQ27441_COMMAND_VOLTAGE); } // Reads and returns the specified current measurement int16_t BQ27441::current(current_measure type) { int16_t current = 0; switch (type) { case AVG: current = (int16_t) readWord(BQ27441_COMMAND_AVG_CURRENT); break; case STBY: current = (int16_t) readWord(BQ27441_COMMAND_STDBY_CURRENT); break; case MAX: current = (int16_t) readWord(BQ27441_COMMAND_MAX_CURRENT); break; } return current; } // Reads and returns the specified capacity measurement uint16_t BQ27441::capacity(capacity_measure type) { uint16_t capacity = 0; switch (type) { case REMAIN: return readWord(BQ27441_COMMAND_REM_CAPACITY); break; case FULL: return readWord(BQ27441_COMMAND_FULL_CAPACITY); break; case AVAIL: capacity = readWord(BQ27441_COMMAND_NOM_CAPACITY); break; case AVAIL_FULL: capacity = readWord(BQ27441_COMMAND_AVAIL_CAPACITY); break; case REMAIN_F: capacity = readWord(BQ27441_COMMAND_REM_CAP_FIL); break; case REMAIN_UF: capacity = readWord(BQ27441_COMMAND_REM_CAP_UNFL); break; case FULL_F: capacity = readWord(BQ27441_COMMAND_FULL_CAP_FIL); break; case FULL_UF: capacity = readWord(BQ27441_COMMAND_FULL_CAP_UNFL); break; case DESIGN: capacity = readWord(BQ27441_EXTENDED_CAPACITY); } return capacity; } // Reads and returns measured average power int16_t BQ27441::power(void) { return (int16_t) readWord(BQ27441_COMMAND_AVG_POWER); } // Reads and returns specified state of charge measurement uint16_t BQ27441::soc(soc_measure type) { uint16_t socRet = 0; switch (type) { case FILTERED: socRet = readWord(BQ27441_COMMAND_SOC); break; case UNFILTERED: socRet = readWord(BQ27441_COMMAND_SOC_UNFL); break; } return socRet; } // Reads and returns specified state of health measurement uint8_t BQ27441::soh(soh_measure type) { uint16_t sohRaw = readWord(BQ27441_COMMAND_SOH); uint8_t sohStatus = sohRaw >> 8; uint8_t sohPercent = sohRaw & 0x00FF; if (type == PERCENT) return sohPercent; else return sohStatus; } // Reads and returns specified temperature measurement uint16_t BQ27441::temperature(temp_measure type) { uint16_t temp = 0; switch (type) { case BATTERY: temp = readWord(BQ27441_COMMAND_TEMP); break; case INTERNAL_TEMP: temp = readWord(BQ27441_COMMAND_INT_TEMP); break; } return temp; } /***************************************************************************** ************************** GPOUT Control Functions ************************** *****************************************************************************/ // Get GPOUT polarity setting (active-high or active-low) bool BQ27441::GPOUTPolarity(void) { uint16_t opConfigRegister = opConfig(); return (opConfigRegister & BQ27441_OPCONFIG_GPIOPOL); } // Set GPOUT polarity to active-high or active-low bool BQ27441::setGPOUTPolarity(bool activeHigh) { uint16_t oldOpConfig = opConfig(); // Check to see if we need to update opConfig: if ((activeHigh && (oldOpConfig & BQ27441_OPCONFIG_GPIOPOL)) || (!activeHigh && !(oldOpConfig & BQ27441_OPCONFIG_GPIOPOL))) return true; uint16_t newOpConfig = oldOpConfig; if (activeHigh) newOpConfig |= BQ27441_OPCONFIG_GPIOPOL; else newOpConfig &= ~(BQ27441_OPCONFIG_GPIOPOL); return writeOpConfig(newOpConfig); } // Get GPOUT function (BAT_LOW or SOC_INT) bool BQ27441::GPOUTFunction(void) { uint16_t opConfigRegister = opConfig(); return (opConfigRegister & BQ27441_OPCONFIG_BATLOWEN); } // Set GPOUT function to BAT_LOW or SOC_INT bool BQ27441::setGPOUTFunction(gpout_function function) { uint16_t oldOpConfig = opConfig(); // Check to see if we need to update opConfig: if ((function && (oldOpConfig & BQ27441_OPCONFIG_BATLOWEN)) || (!function && !(oldOpConfig & BQ27441_OPCONFIG_BATLOWEN))) return true; // Modify BATLOWN_EN bit of opConfig: uint16_t newOpConfig = oldOpConfig; if (function) newOpConfig |= BQ27441_OPCONFIG_BATLOWEN; else newOpConfig &= ~(BQ27441_OPCONFIG_BATLOWEN); // Write new opConfig return writeOpConfig(newOpConfig); } // Get SOC1_Set Threshold - threshold to set the alert flag uint8_t BQ27441::SOC1SetThreshold(void) { return readExtendedData(BQ27441_ID_DISCHARGE, 0); } // Get SOC1_Clear Threshold - threshold to clear the alert flag uint8_t BQ27441::SOC1ClearThreshold(void) { return readExtendedData(BQ27441_ID_DISCHARGE, 1); } // Set the SOC1 set and clear thresholds to a percentage bool BQ27441::setSOC1Thresholds(uint8_t set, uint8_t clear) { uint8_t thresholds[2]; thresholds[0] = constrain(set, 0, 100); thresholds[1] = constrain(clear, 0, 100); return writeExtendedData(BQ27441_ID_DISCHARGE, 0, thresholds, 2); } // Get SOCF_Set Threshold - threshold to set the alert flag uint8_t BQ27441::SOCFSetThreshold(void) { return readExtendedData(BQ27441_ID_DISCHARGE, 2); } // Get SOCF_Clear Threshold - threshold to clear the alert flag uint8_t BQ27441::SOCFClearThreshold(void) { return readExtendedData(BQ27441_ID_DISCHARGE, 3); } // Set the SOCF set and clear thresholds to a percentage bool BQ27441::setSOCFThresholds(uint8_t set, uint8_t clear) { uint8_t thresholds[2]; thresholds[0] = constrain(set, 0, 100); thresholds[1] = constrain(clear, 0, 100); return writeExtendedData(BQ27441_ID_DISCHARGE, 2, thresholds, 2); } // Check if the SOC1 flag is set bool BQ27441::socFlag(void) { uint16_t flagState = flags(); return flagState & BQ27441_FLAG_SOC1; } // Check if the SOCF flag is set bool BQ27441::socfFlag(void) { uint16_t flagState = flags(); return flagState & BQ27441_FLAG_SOCF; } // Check if the ITPOR flag is set bool BQ27441::itporFlag(void) { uint16_t flagState = flags(); return flagState & BQ27441_FLAG_ITPOR; } // Check if the FC flag is set bool BQ27441::fcFlag(void) { uint16_t flagState = flags(); return flagState & BQ27441_FLAG_FC; } // Check if the CHG flag is set bool BQ27441::chgFlag(void) { uint16_t flagState = flags(); return flagState & BQ27441_FLAG_CHG; } // Check if the DSG flag is set bool BQ27441::dsgFlag(void) { uint16_t flagState = flags(); return flagState & BQ27441_FLAG_DSG; } // Get the SOC_INT interval delta uint8_t BQ27441::sociDelta(void) { return readExtendedData(BQ27441_ID_STATE, 26); } // Set the SOC_INT interval delta to a value between 1 and 100 bool BQ27441::setSOCIDelta(uint8_t delta) { uint8_t soci = constrain(delta, 0, 100); return writeExtendedData(BQ27441_ID_STATE, 26, &soci, 1); } // Pulse the GPOUT pin - must be in SOC_INT mode bool BQ27441::pulseGPOUT(void) { return executeControlWord(BQ27441_CONTROL_PULSE_SOC_INT); } /***************************************************************************** *************************** Control Sub-Commands **************************** *****************************************************************************/ // Read the device type - should be 0x0421 uint16_t BQ27441::deviceType(void) { return readControlWord(BQ27441_CONTROL_DEVICE_TYPE); } // Enter configuration mode - set userControl if calling from an Arduino sketch // and you want control over when to exitConfig bool BQ27441::enterConfig(bool userControl) { if (userControl) _userConfigControl = true; if (sealed()) { _sealFlag = true; unseal(); // Must be unsealed before making changes } if (executeControlWord(BQ27441_CONTROL_SET_CFGUPDATE)) { int16_t timeout = BQ72441_I2C_TIMEOUT; while ((timeout--) && (!(flags() & BQ27441_FLAG_CFGUPMODE))) vTaskDelay(pdMS_TO_TICKS(1)); if (timeout > 0) return true; } return false; } // Exit configuration mode with the option to perform a resimulation bool BQ27441::exitConfig(bool resim) { // There are two methods for exiting config mode: // 1. Execute the EXIT_CFGUPDATE command // 2. Execute the SOFT_RESET command // EXIT_CFGUPDATE exits config mode _without_ an OCV (open-circuit voltage) // measurement, and without resimulating to update unfiltered-SoC and SoC. // If a new OCV measurement or resimulation is desired, SOFT_RESET or // EXIT_RESIM should be used to exit config mode. if (resim) { if (softReset()) { int16_t timeout = BQ72441_I2C_TIMEOUT; while ((timeout--) && ((flags() & BQ27441_FLAG_CFGUPMODE))) vTaskDelay(pdMS_TO_TICKS(1)); if (timeout > 0) { if (_sealFlag) seal(); // Seal back up if we IC was sealed coming in return true; } } return false; } else { return executeControlWord(BQ27441_CONTROL_EXIT_CFGUPDATE); } } // Read the flags() command uint16_t BQ27441::flags(void) { return readWord(BQ27441_COMMAND_FLAGS); } // Read the CONTROL_STATUS subcommand of control() uint16_t BQ27441::status(void) { return readControlWord(BQ27441_CONTROL_STATUS); } /***************************** Private Functions *****************************/ // Check if the BQ27441-G1A is sealed or not. bool BQ27441::sealed(void) { uint16_t stat = status(); return stat & BQ27441_STATUS_SS; } // Seal the BQ27441-G1A bool BQ27441::seal(void) { return readControlWord(BQ27441_CONTROL_SEALED); } // UNseal the BQ27441-G1A bool BQ27441::unseal(void) { // To unseal the BQ27441, write the key to the control // command. Then immediately write the same key to control again. if (readControlWord(BQ27441_UNSEAL_KEY)) { return readControlWord(BQ27441_UNSEAL_KEY); } return false; } // Read the 16-bit opConfig register from extended data uint16_t BQ27441::opConfig(void) { return readWord(BQ27441_EXTENDED_OPCONFIG); } // Write the 16-bit opConfig register in extended data bool BQ27441::writeOpConfig(uint16_t value) { uint8_t opConfigMSB = value >> 8; uint8_t opConfigLSB = value & 0x00FF; uint8_t opConfigData[2] = {opConfigMSB, opConfigLSB}; // OpConfig register location: BQ27441_ID_REGISTERS id, offset 0 return writeExtendedData(BQ27441_ID_REGISTERS, 0, opConfigData, 2); } // Issue a soft-reset to the BQ27441-G1A bool BQ27441::softReset(void) { return executeControlWord(BQ27441_CONTROL_SOFT_RESET); } // Read a 16-bit command word from the BQ27441-G1A uint16_t BQ27441::readWord(uint16_t subAddress) { uint8_t data[2]; i2cReadBytes(subAddress, data, 2); return ((uint16_t) data[1] << 8) | data[0]; } // Read a 16-bit subcommand() from the BQ27441-G1A's control() uint16_t BQ27441::readControlWord(uint16_t function) { uint8_t subCommandMSB = (function >> 8); uint8_t subCommandLSB = (function & 0x00FF); uint8_t command[3] = {0, subCommandLSB, subCommandMSB}; uint8_t data[2] = {0, 0}; i2cWriteBytes((uint8_t) 0, command, 2); if (i2cReadBytes((uint8_t) 0, data, 2)) { return ((uint16_t)data[1] << 8) | data[0]; } return false; } // Execute a subcommand() from the BQ27441-G1A's control() bool BQ27441::executeControlWord(uint16_t function) { uint8_t subCommandMSB = (function >> 8); uint8_t subCommandLSB = (function & 0x00FF); uint8_t command[2] = {subCommandLSB, subCommandMSB}; if (i2cWriteBytes((uint8_t) 0, command, 2)) return true; return false; } /***************************************************************************** ************************** Extended Data Commands *************************** *****************************************************************************/ // Issue a BlockDataControl() command to enable BlockData access bool BQ27441::blockDataControl(void) { uint8_t enableByte = 0x00; return i2cWriteBytes(BQ27441_EXTENDED_CONTROL, &enableByte, 1); } // Issue a DataClass() command to set the data class to be accessed bool BQ27441::blockDataClass(uint8_t id) { return i2cWriteBytes(BQ27441_EXTENDED_DATACLASS, &id, 1); } // Issue a DataBlock() command to set the data block to be accessed bool BQ27441::blockDataOffset(uint8_t offset) { return i2cWriteBytes(BQ27441_EXTENDED_DATABLOCK, &offset, 1); } // Read the current checksum using BlockDataCheckSum() uint8_t BQ27441::blockDataChecksum(void) { uint8_t csum; i2cReadBytes(BQ27441_EXTENDED_CHECKSUM, &csum, 1); return csum; } // Use BlockData() to read a byte from the loaded extended data uint8_t BQ27441::readBlockData(uint8_t offset) { uint8_t ret; uint8_t address = offset + BQ27441_EXTENDED_BLOCKDATA; i2cReadBytes(address, &ret, 1); return ret; } // Use BlockData() to write a byte to an offset of the loaded data bool BQ27441::writeBlockData(uint8_t offset, uint8_t data) { uint8_t address = offset + BQ27441_EXTENDED_BLOCKDATA; return i2cWriteBytes(address, &data, 1); } // Read all 32 bytes of the loaded extended data and compute a // checksum based on the values. uint8_t BQ27441::computeBlockChecksum(void) { uint8_t data[32]; i2cReadBytes(BQ27441_EXTENDED_BLOCKDATA, data, 32); uint8_t csum = 0; for (int i=0; i<32; i++) { csum += data[i]; } csum = 255 - csum; return csum; } // Use the BlockDataCheckSum() command to write a checksum value bool BQ27441::writeBlockChecksum(uint8_t csum) { return i2cWriteBytes(BQ27441_EXTENDED_CHECKSUM, &csum, 1); } // Read a byte from extended data specifying a class ID and position offset uint8_t BQ27441::readExtendedData(uint8_t classID, uint8_t offset) { uint8_t retData = 0; if (!_userConfigControl) enterConfig(false); if (!blockDataControl()) // // enable block data memory control return false; // Return false if enable fails if (!blockDataClass(classID)) // Write class ID using DataBlockClass() return false; blockDataOffset(offset / 32); // Write 32-bit block offset (usually 0) computeBlockChecksum(); // Compute checksum going in retData = readBlockData(offset % 32); // Read from offset (limit to 0-31) if (!_userConfigControl) exitConfig(); return retData; } // Write a specified number of bytes to extended data specifying a // class ID, position offset. bool BQ27441::writeExtendedData(uint8_t classID, uint8_t offset, uint8_t * data, uint8_t len) { if (len > 32) return false; if (!_userConfigControl) enterConfig(false); if (!blockDataControl()) // // enable block data memory control return false; // Return false if enable fails if (!blockDataClass(classID)) // Write class ID using DataBlockClass() return false; blockDataOffset(offset / 32); // Write 32-bit block offset (usually 0) computeBlockChecksum(); // Compute checksum going in // Write data bytes: for (int i = 0; i < len; i++) { // Write to offset, mod 32 if offset is greater than 32 // The blockDataOffset above sets the 32-bit block writeBlockData((offset % 32) + i, data[i]); } // Write new checksum using BlockDataChecksum (0x60) uint8_t newCsum = computeBlockChecksum(); // Compute the new checksum writeBlockChecksum(newCsum); if (!_userConfigControl) exitConfig(); return true; } /***************************************************************************** ************************ I2C Read and Write Routines ************************ *****************************************************************************/ // Read a specified number of bytes over I2C at a given subAddress int16_t BQ27441::i2cReadBytes(uint8_t subAddress, uint8_t * dest, uint8_t count) { int16_t timeout = BQ72441_I2C_TIMEOUT; i2c_master_write_read_device(BQ72441_I2C_NUM, _deviceAddress, &subAddress, 1, dest, count, timeout); return timeout; } // Write a specified number of bytes over I2C to a given subAddress uint16_t BQ27441::i2cWriteBytes(uint8_t subAddress, uint8_t * src, uint8_t count) { int16_t timeout = BQ72441_I2C_TIMEOUT; // prepend the subAddress to the data. uint8_t w_buff[count+1] = {0}; w_buff[0] = subAddress; for (int i = 0; i < count; i ++) { w_buff[i+1] = src[i]; } esp_err_t ret = i2c_master_write_to_device(BQ72441_I2C_NUM, _deviceAddress, src, count+1, timeout); return ret == ESP_OK; } BQ27441 lipo;