This is a hardware and software development of a low voltage battery management syetem, which is mainly designed for our autonomous BEV and self-driving car. However, it could also be applied to other electric devices which need an monitoring system for low voltage battery.
The repository includes:
- The Altium PCB design of LV-BMS with either low-side MOSFET switch control(version 2) or high-side(version 3).
- Source code for the microcontroller based on C++, which is compatible for all version(1~3) design, and is written and exploited in Code Composer Studio.
- Texas Instruments TivaWare Peripheral Driver Library for the control of microcontroller. datasheet
- Concept and Functionality of the LV-BMS
- Development of Hardware Part
- Concept of Software Part
- Installation
The main idea that we need a battery management system is that lithium battery has a higher energy density, higher efficiency and lighter weight in comparison with other art of battery such as LiFePo4, which is crucial in our case to gain better performence, but it could also be life-threatening when something goes wrong. Therefore, we need a monitoring system, BMS, to keep supervising the status of the lithium battery stack during the execution of whole system and take defensive action if it is needed. The status at here means the voltages
, current
and temperatures
of the battery stack. To be more specific, voltages could be the cell voltages
as well as the entire voltage
and current is the entire current
that flows out the battery stack and temperatures could be the cell temperatures
as well as the MOSFETs' temperatures
.
BMS monitor these three status to make sure they are all operating in the acceptable range. if not, then BMS will cut off the power supply of the whole system (LV-Battery) by switching the MOSFETs to off so that no further damage could happen and if the stutas are back to normal again, then BMS will turn on the power suppy again.
The general structure of the hardware PCB design is based on the following functional parts:
-
Analog Front End (AFE)
, which is mainly responsible for collecting the parameters of the battery pack such as voltages, current as well as temperatures by using internal ADC measures. It could also control the current between the cells in order to avoid overloading and underloading, which is called cell-balancing. At here we pick BQ76920 from TI as our AFE since it has not only all the basic features that we need but also some other extra features e.g. hardware protection features, charge/discharge low-side NCH FET drivers, alert interrupt to host microcontroller, simple I2C compatible interface (CRC option) -
Inter-Integrated Circuit (I2C) Interface
, which is really crucial to realize the communication between our host microcontroller (uC),TM4C123GH6PM
, and AFE ,BQ76920
. Through the I2C communication, uC can read value from AFE register, send commmed to AFE or even write value into the register in AFE and AFE can also sent alert to uC so that uC could take an action to deal with the errors -
Microcontroller (uC)
, which is like the brain of the system and where we flash our code into in order to realize all the functions that we need. Here isTM4C123GH6PM
chosen as our uC because it has lots of features to utilize and there is also a compatible Peripheral Driver Library on hand. -
FET Driver
, which is needed if we want to have a high-side switch control feature according to our AFE, BQ76920, and provides the advantages of avoiding ground disconnection in the system and continuous communication between the battery pack and host system. Here BQ76200 is chosen as it is directly competible and circuit design is also provide in both BQ76200 and BQ76920 datasheet.Here is an overview of how BQ and uC structured:
Other features that we need for our autonomous BEV and self-driving car:
CAN Bus system
, which is a communication system between our uC and centeralmain control unit (MCU)
in our cars so that MCU could know what is going on right now in LV-system and whether every thing is fine. For some cases, MCU could even take some preventative steps to protect the car if something in LV-system goes wrong.Analog to digital converter (ADC) in uC
, which is exploited for more temperatures monitoring such as Battery stack temperatures and MOSFETs' temperatures.
You could find all the components and their footprints that are utilized on the PCB in the Bill of Materials (BOM). Here we just list the most important components:
- Microcontroller: TM4C123GH6PM
- Analog Front End: BQ76920 without CRC
- FET Driver: BQ76200
- MOSFETs: AUIRFS8409-7P
- 3.3V CAN Bus Transceivers: SN65HVD230
Other useful datasheets for more understanding of the circuit design:
-
BQ relevant:
-
I2C relevant:
-
Microcontroller relevant:
As described above, we use an AFE, BQ76920, to realize the parameters monitoring and MOSFETs control. Here we would like to explain more about the funtionality behind BQ76920
. It is also helpful to go deep into datasheet as it not only provides you a better overview of hardware but also better understanding of the algorithm in software code.
BQ76920 is a pre-programmed chips that already has the function like voltages, current and temperatures measurement and it would automatically keep monitoring/update the value in a certain amount of time. If something goes wrong, the internal system status will be set to bit 0 and automatically disable the charge and discharge control (CHG & DSG Pins set to 0). Therefore, in order to use BQ76920, it is rather important to grasp the concept of how to let microcontroller communicate with the BQ's register and control the BQ's status through I2C than understand how internal measurement works. here is an important system status of BQ76920 in register:
Take cell voltages measurement for example, it update the cell voltages at a rate of 250 ms
and store the value inside its register for other functions utiliztion such as overvoltage and undervoltage check. This means that if BQ76920 updates the voltage and finds out they are out of the limit range, it will internally set the corresponding system status to 0 (means error) and since system status are not all 1, it will then automatically turn off the power supply by shuting down the CHG or DSG Pin to bit 0, which is bit 1 (means on) when everything is fine, and send a alert signal to our microcontroller. This all happen in BQ internally by itself. Thus, you might wonder why we need a microcontroller then if everything works internally. The reason is that we need a microcontroller to clear the alert/error bit. By that it means that microcontroller will keep demanding the BQ to update its system status, in this case keep measuring the cell voltages, and until the voltages are back into acceptable range microcontroller can then successfully clear the alert/error bit and BQ would then allow the charge or discharge control by setting CHG or DSG Pin to bit 1.
In order to grasp the deep understanding of how I2C works, it is highly recommanded that you go through and deep into this article "Understanding the I2C Bus". In the article you should understand how I2C recognize the start as well as the end bit from SCL (Serial Clock) & SDA (Serial Data) and how the signals are read. Here we would like to show some I2C signal from Oscilloscope:
As mentioned above, BQ76920 itself only support the low-side switch feature. However, in some cases it would benefit more if we would able to use high-side switch. That is why we need a FET Driver to convert the signal of CHG/DSG from BQ76920 to higher voltage. The idea behind BQ76200 is using a charge pump capacitor
, which works like an extra small battery connected in series with low-voltage battery, as you could see in the circuit example between the VDDCP and BAT pins
. During the operation the charge pump capacitor will keep charging and discharging process to store and release the energy. By doing so we could raise our gate-voltage of MOSFETs to surpass the source-voltage of MOSFETs, which is same as low-voltage battery voltage, and achieve a positive gate-source-voltage to turn on the high-side MOSFETs.
Here is how the voltage from CHG and DSG changes after the FET driver:
It is also interesting to see the charging and discharging process in charge pump capacitor:
If everything works fine the gate-source-voltage
of the MOSFETs would looks like this:
Here is the capacity of the charge pump capacitor very critical !!!
If the value is too small, then the capacitor can't store enough energy and it will soon be all released, which leads to just some peaks of higher voltage as the images below show:
If the capacity is too big, then the capacitor does not have enough time to reach the needed energy since the charging rate is too low. Therefore, after several testing we have chosen $2.2 uF$ instead of $470nF$ according to datasheet
.
Since the microcontroller can't understand the analog value such as 3.3V or 1A and its internal measurement only send back a bit by bit signal e.g. in a range of 0 ~ 4096 corresponding to 12 bits signal, we need to convert/calculate this bit by bit signal back to the corresponding analog value range and we will recieve the so called digital value. Here is the general equation of ADC:
For example if we have the following situation:
- ADC signal bit length: 12 bits
- Voltage range: 0 ~ 4.3V
- ADC measured value: 3000
then our converted digital value would be:
which means the measured ADC valus 3000 is corresponding to 3.15V.
Generally, we have 4 functional parts: Microcontroller (TM4C)
, Analog Front End (BQ76920)
, FET driver (BQ76200)
, MOSFETs
for power contro.
We place it in such a way that the analog part, digital part, power part are as separated as possible from each other so that we have cleaner signals. As the image below shown, which is based on LV-BMS version 1, you can see that the polygons are divided into 3:
Polygons and netz distribution based on LV-BMS version 1:
Net von Polygons | Analog Part | Digital Part | Power Part |
---|---|---|---|
Top Layer | GND | GND | (komplizierter) |
GND Layer | GND | GND | GND |
VCC Layer | V_REF (3.0V) | +3.3V | Bat + |
Bottom Layer | GND | GND | GND |
We have created our header files and source files based on following different functions we needed:
ADC
: enable the ADC function in microcontroller and include the convert voltage and temperature functions for monitoringI2C
: create abq769x0 class
for AFE (BQ) and enable I2C communication with microcontrollerTimer
: create timer interrupt to percisely call a function at a certain rate and percise delay function, which is better then using SysCtlDelay(5000000)GPIO
: General Purpose Input/Output, to enable or activate the pins on microcontroller and set the pin as input or output (3.3V)CANDriver
: create a basic CAN class including basic CAN bus initialization and communication functions like sending and receiving messagesTestCAN
: inherit from CANDriver class. This builds the structure of our CAN message in byte level.
The main idea of code is to update BMS status at the rate of GlobalBMSUpdate
function and a timer interrupt to percisely call this function at this rate. For our AFE/BQ-chip, we have created a class for it, which has lots of features such as current status, min/max limit of each status and BQ slave adress and functions like I2C communication, set values, get current status values, update status and enable charging/discharging. By doing so, you could design a LV system with even more BQ-chips by create more BQ objects and don't need to worry about the code being messy as well as confused. However, we just create 1 BQ object as we only need 1 BQ76920 in our case. This BQ class also support other chips in BQ769x0 family, which could monitor more voltages and temperatures.
Additionaly, we have enabled the ADC function in microcontroller since our microcontroller has lots of features and we want more ADCs to monitor more temperatures and utilize for backup voltages in case BQ is somehow not working.
Overall steps and ideas in main function (main.cpp):
- Include all libraries and header files we needed
- Create BMS object as global variable,
BMS
- Define global BMS update function,
GlobalBMSUpdate()
, forTimerBMSUpdatesInterruptBegin()
function - Set up system clock & Initialization of all TM4C Systems (GPIO, ADC, Timer, CAN)
- Set up system clock at 80 MHz
- By GPIO is mainly focusing on port enabling from TM4C (Port A-F) and also Pin type for FET Driver and LED
- By ADC are all the necessary settings for ADC measurement such as ADC 1,2 , sample sequence, step configuration, ADC enable
- Start the
TimerDelayInterruptBegin()
function so that we could have a precise delay for later use - Set up the rx_pin, the tx_pin, and the rate for CAN
- Enable charge pump and pack+ monitor for FET Driver
- Initialize the I2C/BQ-chip (AFE) from
BMS
Enable I2C Communication !!!!!
- Initialize all the temperature and voltage buffer
- Boot up BQ-Chip by outputting 3.3V from a GPIO Pin of TM4C for just a short time (around 2ms) ⇒ BQ always need an Activation at first before it starts working (change from SHIP mode to NORMAL mode)
- Test I2C functionality by trying to write (I2CWriteRegister) something in the register of BQ and checking the equality from reading the register (I2CReadRegister)
- If communication succeed, enable I2C voltage and temperature ADC readings and switch CC_EN on
- Initialize and set up the alert interrupt IRS (GPIO_RISING_EDGE interrupt)
- Get the ADC offset and gain
BMS
system set up- Call the
TimerBMSUpdatesInterruptBegin()
function to run the GlobalBMSUpadate, which would enable CHG and DSG simultaneously, in each 250 second - Activate the ADC function in uC to read the temperatures
- check the Temperature status. if too high, then BMS shutdown the battery
- Send out the message through the CAN bus to let our
Main Control Unit
in car know the status of LV system
For more information and understanding, explore yourself in the code, which is pretty well commented. The main and biggest part of the code is the I2C part ( I2C.hpp & I2C.cpp )
- Code written in C++
- Code Composer Studio (CCS) with setting of
TM4C123GH6PM
microcontroller for flashing - TivaWare Peripheral Driver Library
-
Clone this repository
if you download this repository directly without any changing, your directory should have the following structure:
<dataset root directory (e.g. Low-Voltage-BMS)>/ assets/ Hardware_PCB_Design/ LV-BMS_high-side/ ... LV-BMS_21_V2.PrjPcb LV-BMS_V2_21E.PcbDoc ... LV-BMS_low-side/ ... LV-BMS_21_V3_low-side.PrjPcb LV-BMS_V3_21E_low-side.PcbDoc ... Software_Code/ LV-BMS_Code/ ... tidriverlib-master/ ...
Please keep the Software part structure as it is, otherwise it won't work in CCS unless you change the path to where you put the
tidriverlib-master
!!! -
Download Altium and activate a licence in order to use it
-
imoprt all the schematic libraries, PCB libraries and compiled libraries of the conponents from
Component Footprint Import
directory. Please check out online tutorials to see how to import Footprint from integrated libraries (.IntLib
), schematic libraries (.SchLib
) and PCB libraries (.PcbLib
) or check this links: Working with Integrated Libraries in Altium Designer 2021, Working with Integrated Libraries in Altium Designer 2020 -
Here are some useful links to download and import your own footprint of components you needed
-
In order to open the PCB project, click the file with
.PrjPcb
extension. This will automatically open all the schematics and PCBs file which belong to the Project.For instance: LV-BMS_21_V2.PrjPcb
-
-
Download Code Composer Studio
- Set the microcontroller to
TM4C123GH6PM
- Make sure the path to the TivaWare Peripheral Driver Library (
tidriverlib-master
) is correctly pointed to in the properties of the project. Right click on theKIT21_LVBMS_Code
and then go toProperties
. Do the followings:- Go to
CCS Build -> ARM Compiler -> Include Options
# Make sure this exists on the top of the first block (Add dir to #include search path) ${PROJECT_ROOT}\..\tidriverlib-master # If it's not, add this line by clicking "Add..." on the top right corner to add the library path and also move it up to the top/first one by clicking "Move Up" ${PROJECT_ROOT}\..\tidriverlib-master # If you have change the path of "tidriverlib-master", please add a relative path from project root to where your "tidriverlib-master" is, such as: ${PROJECT_ROOT}\relative\path\to\tidriverlib-master
- Go to
CCS Build -> ARM Linker -> File Search Path
# Make sure this exists on the top of the first block (Include library file or command file as input) ${PROJECT_ROOT}\..\tidriverlib-master\driverlib.lib # If it's not, add this line by clicking "Add..." on the top right corner to add the library path and also move it up to the top/first one by clicking "Move Up" ${PROJECT_ROOT}\..\tidriverlib-master\driverlib.lib # If you have change the path of "tidriverlib-master", please add a relative path from project root to where your "tidriverlib-master" is, such as: ${PROJECT_ROOT}\relative\path\to\tidriverlib-master\driverlib.lib
- Go to
- Right click on the
KIT21_LVBMS_Code
and then go toProperties
. Go toCCS Build -> ARM Linker -> Basic Options
and findSet C system stack size
. Make sure thesystem stack size
is greater than or at least2048
! Otherwise it can't successfully flash the code due to insufficient memory issue
- Set the microcontroller to
-
There you are, done! Feel free to explore yourself and change the Hardware design or software code to your need