Goal: A weighted fusion of coulomb counting SOC and OCV SOC. A high accuracy and low-cost implementation.
Why? Coulomb counting on its own can accumulate errors due to repeated integration. We also cannot predict the current SOC using only coulomb counting, which is why we must incorporate OCV Voltage mapping.
System inputs:
- Sampling time = dt
- Pack current = I[n]
- Previous Pack current = I[n-1]
- Pack voltage (under load) = V[n]
- Previous Pack voltage (under load) = V[n-1]
- SOC = SOC[n]
- Previous SOC = SOC[n-1]
Coulomb Counting:
...
Taken into discrete-time for embedded systems, the new equation is:
SOC[n] = SOC[n-1] + (I[n] * dt) / C
UPDATE: The code uses the trapezoidal rule for better approximation.
Integrated current = 0.5 * (current[n] + current[n-1]) * (time)
Code Block |
---|
float integrated_current = 0.5f * (float)(bms->pack_current + s_storage.last_current) * (d_time);
s_storage.i_soc = s_storage.averaged_soc + (integrated_current / bms->config.pack_capacity); |
Voltage Mapping (Simpler version implemented currently):
We will use a 0.2C discharge curve. We plan on doing this with the Chroma 6200D HV power supply. Once we have the discharge curve of the pack we can map voltage values to specific SOC values. The voltage value will represent the voltage UNDER LOAD. There will be inaccuracy due to varying voltages under load.
A 0.2C curve is roughly ~7.76A, which is what we expect our car to be at during cruising speed.
Note: The current implementation as of September 22, 2024 does not use a 0.2C discharge curve. It instead uses the discharge curve found on this website https://lygte-info.dk/review/batteries2012/LG%2021700%20M50%205000mAh%20%28Grey%29%20UK.html I have taken the data off of the 0.2A per cell. Which is ~1.6A on the pack. That is why we have a predict_voltage() function, that accounts for the voltage lost by the internal resistance.
The general math behind this is:
Use a lookup table to find 2 voltage values such that VOLTAGE_LOW <= Pack_Voltage <= VOLTAGE_HIGH.
VOLTAGE_LOW and VOLTAGE_HIGH will map to a SOC %. We can interpolate between the two values to find a better approximation of the current SOC %.
OCV SOC FUTURE EXPLORATION:
This algorithm raises issues that will be addressed through empirical data collection. OCV mapping allows us to predict the current SOC when there is no load on the pack.
Issue #1 = How do we map OCV - SOC?
Discharge testing.
...