Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Plett OCP Hysteresis Model #3593

Merged
merged 58 commits into from
May 11, 2024
Merged
Show file tree
Hide file tree
Changes from 18 commits
Commits
Show all changes
58 commits
Select commit Hold shift + click to select a range
01dc550
first tests passed
mleot Dec 5, 2023
809af01
updated citations
mleot Dec 6, 2023
a9ce212
clean up ocp script
mleot Dec 6, 2023
13875ce
might be step in wrong direction
mleot Dec 7, 2023
80a03d6
now passes TestDFN unit test well posed
tanner-leo Dec 16, 2023
7d2026c
now passes TestDFN unit test well posed
tanner-leo Dec 16, 2023
551b023
passing MPM integration test
tanner-leo Dec 21, 2023
ab3f87d
nearly passing MPM + DFN tests
tanner-leo Dec 21, 2023
a9f09d0
Merge branch 'pr/mleot/3593' into pr/mleot/3593-1
tanner-leo Dec 21, 2023
fc91c08
Merge branch 'pybamm-team:develop' into plett-ocp-hysteresis
mleot Apr 5, 2024
9388adf
style: pre-commit fixes
pre-commit-ci[bot] Apr 5, 2024
7d4aa19
add citations
mleot Apr 17, 2024
2463e2a
style: pre-commit fixes
pre-commit-ci[bot] Apr 17, 2024
1bcb524
remove unessesary citation
mleot Apr 17, 2024
3e89fb4
passes all unit tests
mleot Apr 17, 2024
27e075f
style: pre-commit fixes
pre-commit-ci[bot] Apr 17, 2024
2e7d81b
move Q from lithium ion function to submodel coupled variable
mleot Apr 17, 2024
5ae648b
Merge branch 'plett-ocp-hysteresis' of https://github.com/mleot/pybam…
mleot Apr 17, 2024
e2fa1ed
clean up changes
mleot Apr 17, 2024
9b0a4aa
style: pre-commit fixes
pre-commit-ci[bot] Apr 17, 2024
e37031c
Merge branch 'develop' into plett-ocp-hysteresis
mleot Apr 17, 2024
2ed9dea
working example notebook and temporary revert to dQ as a function
mleot Apr 18, 2024
c4f96c6
style: pre-commit fixes
pre-commit-ci[bot] Apr 18, 2024
5fd1555
revert to functional form of Q
mleot Apr 18, 2024
957755e
renaming to dchs
mleot Apr 18, 2024
3324f40
Merge branch 'develop' into plett-ocp-hysteresis
mleot Apr 18, 2024
d0be637
clean up
mleot Apr 18, 2024
c2825aa
update docs build script
mleot Apr 19, 2024
a16283a
update notebook
mleot Apr 19, 2024
cc64fb4
compatabilize composite model
mleot Apr 19, 2024
7952dbd
add test
mleot Apr 19, 2024
0ad1869
style: pre-commit fixes
pre-commit-ci[bot] Apr 19, 2024
9e46865
remove unessesary statements
mleot Apr 19, 2024
746f811
style: pre-commit fixes
pre-commit-ci[bot] Apr 19, 2024
e9dd55f
remove unused
mleot Apr 19, 2024
4f8bbbc
Merge branch 'develop' into plett-ocp-hysteresis
mleot Apr 19, 2024
69beb4e
Merge branch 'develop' into plett-ocp-hysteresis
mleot Apr 30, 2024
d379fb7
pre-compatibilize psd+composite electrode model
mleot Apr 30, 2024
00a2e8e
style: pre-commit fixes
pre-commit-ci[bot] Apr 30, 2024
8ad8214
update notebook name
mleot May 2, 2024
2f947d9
rename to Wycisk from DCHS
mleot May 2, 2024
9566f50
update parameters required & names
mleot May 2, 2024
57240b8
Merge remote-tracking branch 'upstream/develop' into pr/mleot/3593
mleot May 2, 2024
ca414cb
Merge branch 'develop' into plett-ocp-hysteresis
mleot May 2, 2024
668c5fe
Merge branch 'plett-ocp-hysteresis' of https://github.com/mleot/pybam…
mleot May 2, 2024
eb2fe95
change if statement for code cov
mleot May 2, 2024
647be7b
add docs
mleot May 3, 2024
ec466f0
style: pre-commit fixes
pre-commit-ci[bot] May 3, 2024
6cab440
Merge branch 'develop' into plett-ocp-hysteresis
mleot May 3, 2024
e72f799
update toctree
mleot May 3, 2024
f02551e
add to changelog
mleot May 5, 2024
48fe678
Merge branch 'develop' into plett-ocp-hysteresis
mleot May 6, 2024
31b7115
move functions inside submodel
mleot May 7, 2024
66f0ef1
Merge branch 'develop' into plett-ocp-hysteresis
mleot May 7, 2024
1a8268c
fix change requests
mleot May 9, 2024
5c169e2
Merge branch 'develop' into plett-ocp-hysteresis
mleot May 9, 2024
f6fc3ad
getting rid of functional form Q
mleot May 11, 2024
d8f451e
Merge branch 'develop' into plett-ocp-hysteresis
mleot May 11, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
13 changes: 13 additions & 0 deletions pybamm/CITATIONS.bib
Original file line number Diff line number Diff line change
Expand Up @@ -691,3 +691,16 @@ @article{landesfeind2019temperature
year={2019},
publisher={The Electrochemical Society}
}

@article{Wycisk2022,
title = {Modified Plett-model for modeling voltage hysteresis in lithium-ion cells},
journal = {Journal of Energy Storage},
volume = {52},
pages = {105016},
year = {2022},
issn = {2352-152X},
doi = {https://doi.org/10.1016/j.est.2022.105016},
url = {https://www.sciencedirect.com/science/article/pii/S2352152X22010192},
author = {Dominik Wycisk and Marc Oldenburger and Marc Gerry Stoye and Toni Mrkonjic and Arnulf Latz},
keywords = {Lithium-ion battery, Voltage hysteresis, Plett-model, Silicon–graphite anode},
}
4 changes: 2 additions & 2 deletions pybamm/models/full_battery_models/base_battery_model.py
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ class BatteryModelOptions(pybamm.FuzzyDict):
reactions.
* "open-circuit potential" : str
Sets the model for the open circuit potential. Can be "single"
(default), "current sigmoid", or "MSMR". If "MSMR" then the "particle"
(default), "current sigmoid", "Plett", or "MSMR". If "MSMR" then the "particle"
option must also be "MSMR". A 2-tuple can be provided for different
behaviour in negative and positive electrodes.
* "operating mode" : str
Expand Down Expand Up @@ -263,7 +263,7 @@ def __init__(self, extra_options):
"stress and reaction-driven",
],
"number of MSMR reactions": ["none"],
"open-circuit potential": ["single", "current sigmoid", "MSMR"],
"open-circuit potential": ["single", "current sigmoid", "MSMR", "Plett"],
mleot marked this conversation as resolved.
Show resolved Hide resolved
"operating mode": [
"current",
"voltage",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -248,6 +248,9 @@ def set_open_circuit_potential_submodel(self):
ocp_model = ocp_submodels.SingleOpenCircuitPotential
elif ocp_option == "current sigmoid":
ocp_model = ocp_submodels.CurrentSigmoidOpenCircuitPotential
elif ocp_option == "Plett":
pybamm.citations.register("Wycisk2022")
ocp_model = ocp_submodels.PlettOpenCircuitPotential
elif ocp_option == "MSMR":
ocp_model = ocp_submodels.MSMROpenCircuitPotential
self.submodels[f"{domain} {phase} open-circuit potential"] = ocp_model(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,3 +2,4 @@
from .single_ocp import SingleOpenCircuitPotential
from .current_sigmoid_ocp import CurrentSigmoidOpenCircuitPotential
from .msmr_ocp import MSMROpenCircuitPotential
from .plett_ocp import PlettOpenCircuitPotential
mleot marked this conversation as resolved.
Show resolved Hide resolved
150 changes: 150 additions & 0 deletions pybamm/models/submodels/interface/open_circuit_potential/plett_ocp.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,150 @@
#
# from Wycisk 2022
#
import pybamm
from . import BaseOpenCircuitPotential


class PlettOpenCircuitPotential(BaseOpenCircuitPotential):
mleot marked this conversation as resolved.
Show resolved Hide resolved
def get_fundamental_variables(self):
domain, Domain = self.domain_Domain
phase_name = self.phase_name
h = pybamm.Variable(
f"{Domain} electrode {phase_name}hysteresis state",
domains={
"primary": f"{domain} electrode",
"secondary": "current collector",
},
)
return {
f"{Domain} electrode {phase_name}hysteresis state": h,
}

def get_coupled_variables(self, variables):
domain, Domain = self.domain_Domain
phase_name = self.phase_name
phase = self.phase

# get dQ, change in capacity with respect to stoichiometry
c_max = self.phase_param.c_max
epsilon_s_av = self.phase_param.epsilon_s_av
V_electrode = self.param.A_cc * self.domain_param.L
Li_max = c_max * V_electrode * epsilon_s_av
Q_max = Li_max * self.param.F / 3600
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is already a variable f"{Domain} electrode capacity [A.h]" defined in the active material model that does this calculation also accounting for any change in active material fraction

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Originally I was calculating dQ(sto) by means of a function in the lithium ion parameters (main parameters for a phase I believe), but @valentinsulzer suggested placing it in the submodel class as a coupled variable.

Since dQ(sto) should be a constant value, I realized this could just be a scalar. Although, I am not sure whether it needs to be an equation as d(sto) may be different depending on the number of steps during solving. I still haven't tested this (looked at actual data or results) with the change from I made, so I am not sure if I will actually get the results I expect. All the more reason to make an example notebook.

I will make the example notebook and assess whether the function form of Q was necessary or not.

dQ = Q_max
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This isn't dQ, right, it's just the electrode capacity? I think later on you are writing dU/dQ = dU/dx * dx/dQ? Can you point to which equation the paper you've implemented as they seem to express it a few ways in the paper?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah my thinking here is that dQ(sto) would be equal to Q*sto, so maybe I need to multiply it by sto.

Suggested change
dQ = Q_max
dQ = Q_max * sto_bulk

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe it wasn't dQ. I believe I have since fixed it, and reverted back to the functional form of Q and taking the differential of that.

Here is the equation:
image


if self.reaction == "lithium-ion main":
T = variables[f"{Domain} electrode temperature [K]"]
h = variables[f"{Domain} electrode {phase_name}hysteresis state"]
# For "particle-size distribution" models, take distribution version
# of c_s_surf that depends on particle size.
domain_options = getattr(self.options, domain)
if domain_options["particle size"] == "distribution":
sto_surf = variables[
f"{Domain} {phase_name}particle surface stoichiometry distribution"
]
# If variable was broadcast, take only the orphan
if isinstance(sto_surf, pybamm.Broadcast) and isinstance(
T, pybamm.Broadcast
):
sto_surf = sto_surf.orphans[0]
T = T.orphans[0]
T = pybamm.PrimaryBroadcast(T, [f"{domain} particle size"])
h = pybamm.PrimaryBroadcast(h, [f"{domain} particle size"])
else:
sto_surf = variables[
f"{Domain} {phase_name}particle surface stoichiometry"
]
# If variable was broadcast, take only the orphan
if isinstance(sto_surf, pybamm.Broadcast) and isinstance(
T, pybamm.Broadcast
):
sto_surf = sto_surf.orphans[0]
T = T.orphans[0]

ocp_surf_eq = self.phase_param.U(sto_surf, T)
dUdT = self.phase_param.dUdT(sto_surf)

# Bulk OCP is from the average SOC and temperature
sto_bulk = variables[f"{Domain} electrode {phase_name}stoichiometry"]
T_bulk = pybamm.xyz_average(pybamm.size_average(T))
ocp_bulk_eq = self.phase_param.U(sto_bulk, T_bulk)

c_scale = self.phase_param.c_max
variables[f"Total lithium in {phase} phase in {domain} electrode [mol]"] = (
sto_bulk * c_scale
) # c_s_vol * L * A

H = self.phase_param.H(sto_surf)
variables[f"{Domain} electrode {phase_name}equilibrium OCP [V]"] = (
ocp_surf_eq
)
variables[f"{Domain} electrode {phase_name}bulk equilibrium OCP [V]"] = (
ocp_bulk_eq
)
variables[f"{Domain} electrode {phase_name}OCP hysteresis [V]"] = H

dU = self.phase_param.U(sto_surf, T_bulk).diff(sto_surf)
# dQ = self.phase_param.Q(sto_surf).diff(sto_surf)
dQdU = dQ / dU
variables[
f"{Domain} electrode {phase_name}differential capacity [A.s.V-1]"
] = dQdU
variables[
f"{Domain} electrode {phase_name}hysteresis state distribution"
] = h
H_x_av = pybamm.x_average(H)
h_x_av = pybamm.x_average(h)
# check if psd
if domain_options["particle size"] == "distribution":
if f"{domain} electrode" in sto_surf.domains["primary"]:
ocp_surf = ocp_surf_eq + H * h
elif f"{domain} particle size" in sto_surf.domains["primary"]:
# check if MPM Model
if "current collector" in sto_surf.domains["secondary"]:
ocp_surf = ocp_surf_eq + H_x_av * h_x_av
# must be DFN with PSD model
else:
ocp_surf = ocp_surf_eq + H * h
else:
ocp_surf = ocp_surf_eq + H_x_av * h_x_av
else:
ocp_surf = ocp_surf_eq + H * h
H_s_av = pybamm.size_average(H_x_av)
h_s_av = pybamm.size_average(h_x_av)

ocp_bulk = ocp_bulk_eq + H_s_av * h_s_av

variables.update(self._get_standard_ocp_variables(ocp_surf, ocp_bulk, dUdT))
return variables

def set_rhs(self, variables):
domain, Domain = self.domain_Domain
phase_name = self.phase_name

current = self.param.current_with_time
current = variables[f"{Domain} electrode interfacial current density [A.m-2]"]
# current = current.orphans[0]
# current = current.SecondaryBroadcast(current,f'{domain} electrode')
Q_cell = variables[f"{Domain} electrode capacity [A.h]"]
dQdU = variables[
f"{Domain} electrode {phase_name}differential capacity [A.s.V-1]"
]
dQdU = dQdU.orphans[0]
mleot marked this conversation as resolved.
Show resolved Hide resolved
K = self.phase_param.K
K_x = self.phase_param.K_x
h = variables[f"{Domain} electrode {phase_name}hysteresis state"]
# h_x_av = variables[f'X-averaged {domain} electrode {phase_name}hysteresis state']

dhdt = (
K * (current / (Q_cell * (dQdU**K_x))) * (1 - pybamm.sign(current) * h)
) #! current is backwards for a halfcell
mleot marked this conversation as resolved.
Show resolved Hide resolved
self.rhs[h] = dhdt

def set_initial_conditions(self, variables):
domain, Domain = self.domain_Domain
phase_name = self.phase_name
h = variables[f"{Domain} electrode {phase_name}hysteresis state"]
# h_av = variables[f'{Domain} electrode {phase_name}hysteresis state distribution']
# self.initial_conditions[h_av] = pybamm.Scalar(0)
self.initial_conditions[h] = pybamm.Scalar(0)
19 changes: 19 additions & 0 deletions pybamm/parameters/lithium_ion_parameters.py
Original file line number Diff line number Diff line change
Expand Up @@ -535,6 +535,12 @@ def _set_parameters(self):
eps_c_init_av = pybamm.xyz_average(
self.epsilon_s * pybamm.r_average(self.c_init)
)
# if self.options['open-circuit potential'] == 'Plett':
self.K = pybamm.Parameter(f"{pref}{Domain} particle hysteresis decay rate")
self.K_x = pybamm.Parameter(
f"{pref}{Domain} particle hysteresis switching factor"
)
mleot marked this conversation as resolved.
Show resolved Hide resolved
self.h_init = pybamm.Scalar(0)
mleot marked this conversation as resolved.
Show resolved Hide resolved

if self.options["open-circuit potential"] != "MSMR":
self.U_init = self.U(self.sto_init_av, main.T_init)
Expand Down Expand Up @@ -630,6 +636,17 @@ def U(self, sto, T, lithiation=None):
out.print_name = r"U_\mathrm{p}(c^\mathrm{surf}_\mathrm{s,p}, T)"
return out

def H(self, sto):
mleot marked this conversation as resolved.
Show resolved Hide resolved
"""Dimensional hysteresis [V]"""
Domain = self.domain.capitalize()
# tol = pybamm.settings.tolerances["U__c_s"]
# sto = pybamm.maximum(pybamm.minimum(sto,1-tol),tol)
inputs = {f"{self.phase_prefactor}{Domain} particle stoichiometry": sto}
h_ref = pybamm.FunctionParameter(
f"{self.phase_prefactor}{Domain} electrode OCP hysteresis [V]", inputs
)
return h_ref

def dUdT(self, sto):
"""
Dimensional entropic change of the open-circuit potential [V.K-1]
Expand All @@ -645,6 +662,8 @@ def dUdT(self, sto):
inputs,
)

# def dQdU(self,sto)

def X_j(self, index):
"Available host sites indexed by reaction j"
domain = self.domain
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,25 @@ def test_current_sigmoid_ocp(self):
modeltest = tests.StandardModelTest(model, parameter_values=parameter_values)
modeltest.test_all(skip_output_tests=True)

def test_plett_ocp(self):
options = {"open-circuit potential": ("Plett", "single")}
model = pybamm.lithium_ion.MPM(options)
parameter_values = pybamm.ParameterValues("Chen2020")
parameter_values = pybamm.get_size_distribution_parameters(parameter_values)
parameter_values.update(
{
"Negative electrode OCP [V]" "": parameter_values[
"Negative electrode OCP [V]"
],
"Negative particle hysteresis decay rate": 1,
"Negative particle hysteresis switching factor": 1,
"Negative electrode OCP hysteresis [V]": lambda sto: 1,
},
check_already_exists=False,
)
modeltest = tests.StandardModelTest(model, parameter_values=parameter_values)
modeltest.test_all(skip_output_tests=True)

def test_voltage_control(self):
options = {"operating mode": "voltage"}
model = pybamm.lithium_ion.MPM(options)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -406,6 +406,10 @@ def test_well_posed_current_sigmoid_ocp(self):
options = {"open-circuit potential": "current sigmoid"}
self.check_well_posedness(options)

def test_well_posed_plett_ocp(self):
options = {"open-circuit potential": "Plett"}
self.check_well_posedness(options)

def test_well_posed_msmr(self):
options = {
"open-circuit potential": "MSMR",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,13 @@ def test_well_posed_current_sigmoid_ocp_with_psd(self):
}
self.check_well_posedness(options)

def test_well_posed_plett_ocp_with_psd(self):
options = {
"open-circuit potential": "Plett",
"particle size": "distribution",
}
self.check_well_posedness(options)

def test_well_posed_external_circuit_explicit_power(self):
options = {"operating mode": "explicit power"}
self.check_well_posedness(options)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -116,6 +116,13 @@ def test_msmr(self):
model = pybamm.lithium_ion.MPM(options)
model.check_well_posedness()

def test_plett_ocp(self):
options = {
"open-circuit potential": "Plett",
}
model = pybamm.lithium_ion.MPM(options)
model.check_well_posedness()


class TestMPMExternalCircuits(TestCase):
def test_well_posed_voltage(self):
Expand Down
Loading