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

Refactor get_initial_stoichiometries with custom and default tolerance #3714

Merged
merged 10 commits into from
Jan 24, 2024
31 changes: 23 additions & 8 deletions pybamm/models/full_battery_models/lithium_ion/electrode_soh.py
Original file line number Diff line number Diff line change
Expand Up @@ -664,7 +664,7 @@ def _check_esoh_feasible(self, inputs):
)
)

def get_initial_stoichiometries(self, initial_value):
def get_initial_stoichiometries(self, initial_value, tol=1e-6):
"""
Calculate initial stoichiometries to start off the simulation at a particular
state of charge, given voltage limits, open-circuit potentials, etc defined by
Expand All @@ -677,6 +677,10 @@ def get_initial_stoichiometries(self, initial_value):
If integer, interpreted as SOC, must be between 0 and 1.
If string e.g. "4 V", interpreted as voltage,
must be between V_min and V_max.
tol : float, optional
The tolerance for the solver used to compute the initial stoichiometries.
A lower value results in higher precision but may increase computation time.
Default is 1e-6.

Returns
-------
Expand Down Expand Up @@ -723,7 +727,9 @@ def get_initial_stoichiometries(self, initial_value):
soc_model.initial_conditions[soc] = (V_init - V_min) / (V_max - V_min)
soc_model.variables["soc"] = soc
parameter_values.process_model(soc_model)
initial_soc = pybamm.AlgebraicSolver().solve(soc_model, [0])["soc"].data[0]
initial_soc = (
pybamm.AlgebraicSolver(tol=tol).solve(soc_model, [0])["soc"].data[0]
)
elif isinstance(initial_value, (int, float)):
initial_soc = initial_value
if not 0 <= initial_soc <= 1:
Expand Down Expand Up @@ -766,7 +772,7 @@ def get_min_max_stoichiometries(self):
sol = self.solve(inputs)
return [sol["x_0"], sol["x_100"], sol["y_100"], sol["y_0"]]

def get_initial_ocps(self, initial_value):
def get_initial_ocps(self, initial_value, tol=1e-6):
"""
Calculate initial open-circuit potentials to start off the simulation at a
particular state of charge, given voltage limits, open-circuit potentials, etc
Expand All @@ -776,6 +782,8 @@ def get_initial_ocps(self, initial_value):
----------
initial_value : float
Target SOC, must be between 0 and 1.
tol: float, optional
Tolerance for the solver used in calculating initial stoichiometries.

Returns
-------
Expand All @@ -784,7 +792,7 @@ def get_initial_ocps(self, initial_value):
"""
parameter_values = self.parameter_values
param = self.param
x, y = self.get_initial_stoichiometries(initial_value)
x, y = self.get_initial_stoichiometries(initial_value, tol)
if self.options["open-circuit potential"] == "MSMR":
msmr_pot_model = _get_msmr_potential_model(
self.parameter_values, self.param
Expand Down Expand Up @@ -854,6 +862,7 @@ def get_initial_stoichiometries(
param=None,
known_value="cyclable lithium capacity",
options=None,
tol=1e-6,
):
"""
Calculate initial stoichiometries to start off the simulation at a particular
Expand All @@ -878,14 +887,18 @@ def get_initial_stoichiometries(
options : dict-like, optional
A dictionary of options to be passed to the model, see
:class:`pybamm.BatteryModelOptions`.
tol : float, optional
The tolerance for the solver used to compute the initial stoichiometries.
A lower value results in higher precision but may increase computation time.
Default is 1e-6.

Returns
-------
x, y
The initial stoichiometries that give the desired initial state of charge
"""
esoh_solver = ElectrodeSOHSolver(parameter_values, param, known_value, options)
return esoh_solver.get_initial_stoichiometries(initial_value)
return esoh_solver.get_initial_stoichiometries(initial_value, tol)


def get_min_max_stoichiometries(
Expand Down Expand Up @@ -1014,7 +1027,7 @@ def theoretical_energy_integral(parameter_values, param, inputs, points=100):


def calculate_theoretical_energy(
parameter_values, initial_soc=1.0, final_soc=0.0, points=100
parameter_values, initial_soc=1.0, final_soc=0.0, points=100, tol=1e-6
):
"""
Calculate maximum energy possible from a cell given OCV, initial soc, and final soc
Expand All @@ -1030,14 +1043,16 @@ def calculate_theoretical_energy(
The soc at end of discharge, default 0.0
points : int
The number of points at which to calculate voltage.
tol: float
Tolerance for the solver used in calculating initial and final stoichiometries.
Returns
-------
E
The total energy of the cell in Wh
"""
# Get initial and final stoichiometric values.
x_100, y_100 = get_initial_stoichiometries(initial_soc, parameter_values)
x_0, y_0 = get_initial_stoichiometries(final_soc, parameter_values)
x_100, y_100 = get_initial_stoichiometries(initial_soc, parameter_values, tol=tol)
x_0, y_0 = get_initial_stoichiometries(final_soc, parameter_values, tol=tol)
Q_p = parameter_values.evaluate(pybamm.LithiumIonParameters().p.prim.Q_init)
E = theoretical_energy_integral(
parameter_values,
Expand Down
8 changes: 7 additions & 1 deletion pybamm/parameters/parameter_values.py
Original file line number Diff line number Diff line change
Expand Up @@ -307,14 +307,20 @@ def set_initial_stoichiometries(
known_value="cyclable lithium capacity",
inplace=True,
options=None,
tol=1e-6,
):
"""
Set the initial stoichiometry of each electrode, based on the initial
SOC or voltage
"""
param = param or pybamm.LithiumIonParameters(options)
x, y = pybamm.lithium_ion.get_initial_stoichiometries(
initial_value, self, param=param, known_value=known_value, options=options
initial_value,
self,
param=param,
known_value=known_value,
options=options,
tol=tol,
)
if inplace:
parameter_values = self
Expand Down
Loading