/*! ADBMS1818: Multicell Battery Monitors * *@verbatim *The ADBMS1818 is multi-cell battery stack monitor that measures up to 18 series *connected battery cells with a total measurement error of less than 2.2mV. *The cell measurement range of 0V to 5V makes the ADBMS1818 suitable for most *battery chemistries. All 18 cell voltages can be captured in 290uS, and lower *data acquisition rates can be selected for high noise reduction. *Using the ADBMS1818, multiple devices are connected in a daisy-chain with one *host processor connection for all devices, permitting simultaneous cell monitoring *of long, high voltage battery strings. *@endverbatim * * https://www.analog.com/en/products/ADBMS1818.html /TBD * https://www.analog.com/en/design-center/evaluation-hardware-and-software/evaluation-boards-kits/EVAL-ADBMS1818.html /TBD * ********************************************************************************* * Copyright 2019(c) Analog Devices, Inc. * * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * - Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * - Neither the name of Analog Devices, Inc. nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * - The use of this software may or may not infringe the patent rights * of one or more patent holders. This license does not release you * from the requirement that you obtain separate licenses from these * patent holders to use this software. * - Use of the software either in source or binary form, must be run * on or directly connected to an Analog Devices Inc. component. * * THIS SOFTWARE IS PROVIDED BY ANALOG DEVICES "AS IS" AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, NON-INFRINGEMENT, * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL ANALOG DEVICES BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, INTELLECTUAL PROPERTY RIGHTS, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. *******************************************************************************/ //! @ingroup BMS //! @{ //! @defgroup ADBMS1818 ADBMS1818: Multicell Battery Monitor //! @} /*! @file @ingroup ADBMS1818 Library for ADBMS1818 Multicell Battery Monitor */ #include "stdint.h" #include "ADBMS181x.h" #include "ADBMS1818.h" /* Helper function to initialize register limits. */ void ADBMS1818_init_reg_limits(uint8_t total_ic, //Number of ICs in the system cell_asic *ic // A two dimensional array that will store the data ) { for(uint8_t cic=0; cicfailure_pos_limit) || (measure_deltafailure_pos_limit) || (measure_deltaisospi_reverse == true){c_ic = current_ic;} else{c_ic = total_ic - current_ic - 1;} write_buffer[0] = ic[c_ic].pwmb.tx_data[0]; write_buffer[1] = ic[c_ic].pwmb.tx_data[1]; write_buffer[2]= ic[c_ic].pwmb.tx_data[2]; write_buffer[3] = ic[c_ic].sctrlb.tx_data[3]; write_buffer[4] = ic[c_ic].sctrlb.tx_data[4]; write_buffer[5]= ic[c_ic].sctrlb.tx_data[5]; } write_68(total_ic, cmd, write_buffer); } /* Reading the 6813 PWM/Sctrl Register B */ uint8_t ADBMS1818_rdpsb(uint8_t total_ic, //< number of ICs in the daisy chain cell_asic *ic //< a two dimensional array that the function stores the read data ) { uint8_t cmd[4]; uint8_t read_buffer[256]; int8_t pec_error = 0; uint16_t data_pec; uint16_t calc_pec; uint8_t c_ic = 0; cmd[0] = 0x00; cmd[1] = 0x1E; pec_error = read_68(total_ic, cmd, read_buffer); for(uint8_t current_ic =0; current_icisospi_reverse == false){c_ic = current_ic;} else{c_ic = total_ic - current_ic - 1;} for(int byte=0; byte<3;byte++) { ic[c_ic].pwmb.rx_data[byte] = read_buffer[byte+(8*current_ic)]; } for(int byte=3; byte<6;byte++) { ic[c_ic].sctrlb.rx_data[byte] = read_buffer[byte+(8*current_ic)]; } for(int byte=6; byte<8;byte++) { ic[c_ic].pwmb.rx_data[byte] = read_buffer[byte+(8*current_ic)]; ic[c_ic].sctrlb.rx_data[byte] = read_buffer[byte+(8*current_ic)]; } calc_pec = pec15_calc(6,&read_buffer[8*current_ic]); data_pec = read_buffer[7+(8*current_ic)] | (read_buffer[6+(8*current_ic)]<<8); if(calc_pec != data_pec ) { ic[c_ic].pwmb.rx_pec_match = 1; ic[c_ic].sctrlb.rx_pec_match = 1; } else { ic[c_ic].pwmb.rx_pec_match = 0; ic[c_ic].sctrlb.rx_pec_match = 0; } } return(pec_error); } /* Writes the COMM registers of a ADBMS1818 daisy chain */ void ADBMS1818_wrcomm(uint8_t total_ic, //The number of ICs being written to cell_asic *ic //A two dimensional array of the comm data that will be written ) { ADBMS181x_wrcomm(total_ic,ic); } /* Reads COMM registers of a ADBMS1818 daisy chain */ int8_t ADBMS1818_rdcomm(uint8_t total_ic, //Number of ICs in the system cell_asic *ic //A two dimensional array that the function stores the read data. ) { int8_t pec_error = 0; ADBMS181x_rdcomm(total_ic, ic); return(pec_error); } /* Shifts data in COMM register out over ADBMS1818 SPI/I2C port */ void ADBMS1818_stcomm(uint8_t len) //Length of data to be transmitted { ADBMS181x_stcomm(len); } /* Mutes the ADBMS1818 discharge transistors */ void ADBMS1818_mute() { uint8_t cmd[2]; cmd[0] = 0x00; cmd[1] = 0x28; cmd_68(cmd); } /* Clears the ADBMS1818 Mute Discharge */ void ADBMS1818_unmute() { uint8_t cmd[2]; cmd[0] = 0x00; cmd[1] = 0x29; cmd_68(cmd); } /* Helper function that increments PEC counters */ void ADBMS1818_check_pec(uint8_t total_ic,//Number of ICs in the system uint8_t reg, // Type of register cell_asic *ic // A two dimensional array that will store the data ) { ADBMS181x_check_pec(total_ic,reg,ic); } /* Helper Function to reset PEC counters */ void ADBMS1818_reset_crc_count(uint8_t total_ic, //Number of ICs in the system cell_asic *ic // A two dimensional array that will store the data ) { ADBMS181x_reset_crc_count(total_ic,ic); } /* Helper function to initialize CFG variables */ void ADBMS1818_init_cfg(uint8_t total_ic, cell_asic *ic) { ADBMS181x_init_cfg(total_ic,ic); } /* Helper function to set CFGR variable */ void ADBMS1818_set_cfgr(uint8_t nIC, cell_asic *ic, bool refon, bool adcopt, bool gpio[5],bool dcc[12],bool dcto[4], uint16_t uv, uint16_t ov) { ADBMS181x_set_cfgr_refon(nIC,ic,refon); ADBMS181x_set_cfgr_adcopt(nIC,ic,adcopt); ADBMS181x_set_cfgr_gpio(nIC,ic,gpio); ADBMS181x_set_cfgr_dis(nIC,ic,dcc); ADBMS181x_set_cfgr_dcto(nIC,ic,dcto); ADBMS181x_set_cfgr_uv(nIC, ic, uv); ADBMS181x_set_cfgr_ov(nIC, ic, ov); } /* Helper function to set the REFON bit */ void ADBMS1818_set_cfgr_refon(uint8_t nIC, cell_asic *ic, bool refon) { ADBMS181x_set_cfgr_refon(nIC,ic,refon); } /* Helper function to set the adcopt bit */ void ADBMS1818_set_cfgr_adcopt(uint8_t nIC, cell_asic *ic, bool adcopt) { ADBMS181x_set_cfgr_adcopt(nIC,ic,adcopt); } /* Helper function to set GPIO bits */ void ADBMS1818_set_cfgr_gpio(uint8_t nIC, cell_asic *ic,bool gpio[5]) { ADBMS181x_set_cfgr_gpio(nIC,ic,gpio); } /* Helper function to control discharge */ void ADBMS1818_set_cfgr_dis(uint8_t nIC, cell_asic *ic,bool dcc[12]) { ADBMS181x_set_cfgr_dis(nIC,ic,dcc); } /* Helper Function to set uv value in CFG register */ void ADBMS1818_set_cfgr_uv(uint8_t nIC, cell_asic *ic,uint16_t uv) { ADBMS181x_set_cfgr_uv(nIC, ic, uv); } /* Helper Function to set dcto value in CFG register */ void ADBMS1818_set_cfgr_dcto(uint8_t nIC, cell_asic *ic,bool dcto[4]) { ADBMS181x_set_cfgr_dcto(nIC, ic, dcto); } /* Helper function to set OV value in CFG register */ void ADBMS1818_set_cfgr_ov(uint8_t nIC, cell_asic *ic,uint16_t ov) { ADBMS181x_set_cfgr_ov( nIC, ic, ov); } /* Helper Function to initialize the CFGRB data structures */ void ADBMS1818_init_cfgb(uint8_t total_ic,cell_asic *ic) { for (uint8_t current_ic = 0; current_ic0 and i<5) { if(dccbits[i])ic[nIC].configb.tx_data[0] = ic[nIC].configb.tx_data[0]|(0x01<<(i+3)); else ic[nIC].configb.tx_data[0] = ic[nIC].configb.tx_data[0]&(~(0x01<<(i+3))); } if(i>4 and i<7) { if(dccbits[i])ic[nIC].configb.tx_data[1] = ic[nIC].configb.tx_data[1]|(0x01<<(i-5)); else ic[nIC].configb.tx_data[1] = ic[nIC].configb.tx_data[1]&(~(0x01<<(i-5))); } } }