Source code for burnman.eos.vinet

# This file is part of BurnMan - a thermoelastic and thermodynamic toolkit for the Earth and Planetary Sciences
# Copyright (C) 2012 - 2025 by the BurnMan team, released under the GNU
# GPL v2 or later.


import scipy.optimize as opt
from . import equation_of_state as eos
import warnings
from math import exp


[docs] def bulk_modulus_vinet(volume, params): """ compute the bulk modulus as per the Vinet equation of state. Reference bulk modulus should be in :math:`[Pa]`. """ V_rel = volume / params["V_0"] eta = (3.0 / 2.0) * (params["Kprime_0"] - 1.0) K = ( (params["K_0"] * pow(V_rel, -2.0 / 3.0)) * (1 + ((eta * pow(V_rel, 1.0 / 3.0) + 1.0) * (1.0 - pow(V_rel, 1.0 / 3.0)))) * exp(eta * (1.0 - pow(V_rel, 1.0 / 3.0))) ) return K
[docs] def pressure_vinet(V_rel, params): """ Pressure equation for the Vinet equation of state, returns pressure in the same units that are supplied for the reference bulk modulus (params['K_0']), which should be in math:`[Pa]`. """ eta = (3.0 / 2.0) * (params["Kprime_0"] - 1.0) return ( 3.0 * params["K_0"] * (pow(V_rel, -2.0 / 3.0)) * (1.0 - (pow(V_rel, 1.0 / 3.0))) * exp(eta * (1.0 - pow(V_rel, 1.0 / 3.0))) + params["P_0"] )
[docs] def volume_vinet(pressure, params): """ Get the volume at a reference temperature for a given pressure :math:`[Pa]`. Returns molar volume in :math:`[m^3]` """ def func(V): return pressure_vinet(V / params["V_0"], params) - pressure V = opt.brentq(func, 0.1 * params["V_0"], 1.5 * params["V_0"]) return V
[docs] class Vinet(eos.IsothermalEquationOfState): """ Base class for the isothermal Vinet equation of state. References for this equation of state are :cite:`vinet1986` and :cite:`vinet1987`. This equation of state actually predates Vinet by 55 years :cite:`Rydberg1932`, and was investigated further by :cite:`Stacey1981`. .. list-table:: :widths: 25 75 20 :header-rows: 1 * - Parameter - Description - Units * - ``F_0`` - Reference Helmholtz free energy. - :math:`\\text{J/mol}` * - ``P_0`` - Reference pressure. - :math:`\\text{Pa}` * - ``V_0`` - Reference volume. - :math:`\\text{m}^3` * - ``K_0`` - Reference bulk modulus. - :math:`\\text{Pa}` * - ``Kprime_0`` - Pressure derivative of bulk modulus. - Dimensionless """
[docs] def volume(self, pressure, temperature, params): """ Returns volume :math:`[m^3]` as a function of pressure :math:`[Pa]`. """ return volume_vinet(pressure, params)
[docs] def pressure(self, temperature, volume, params): return pressure_vinet(volume / params["V_0"], params)
[docs] def isothermal_bulk_modulus_reuss(self, pressure, temperature, volume, params): """ Returns isothermal bulk modulus :math:`K_T` :math:`[Pa]` as a function of pressure :math:`[Pa]`, temperature :math:`[K]` and volume :math:`[m^3]`. """ return bulk_modulus_vinet(volume, params)
[docs] def shear_modulus(self, pressure, temperature, volume, params): """ Returns shear modulus :math:`G` of the mineral. :math:`[Pa]` Not included in the Vinet EOS, so returns 0. """ return 0.0
[docs] def gibbs_energy(self, pressure, temperature, volume, params): """ Returns the Gibbs free energy :math:`\\mathcal{G}` of the mineral. :math:`[J/mol]` """ # G = int VdP = [PV] - int PdV = E + PV x = pow(volume / params["V_0"], 1.0 / 3.0) eta = (3.0 / 2.0) * (params["Kprime_0"] - 1.0) intPdV = ( 9.0 * params["V_0"] * params["K_0"] / (eta * eta) * ((1.0 - eta * (1.0 - x)) * exp(eta * (1.0 - x)) - 1.0) ) return params["F_0"] - intPdV + volume * pressure
[docs] def validate_parameters(self, params): """ Check for existence and validity of the parameters """ if "F_0" not in params: params["F_0"] = 0.0 if "P_0" not in params: params["P_0"] = 0.0 if "E_0" in params: raise KeyError( "Isothermal equations of state should be " "defined in terms of Helmholtz free energy " "F_0, not internal energy E_0." ) # G is not included in the Vinet EOS so we shall set them to NaN's if "G_0" not in params: params["G_0"] = float("nan") if "Gprime_0" not in params: params["Gprime_0"] = float("nan") # check that all the required keys are in the dictionary expected_keys = ["V_0", "K_0", "Kprime_0"] for k in expected_keys: if k not in params: raise KeyError("params object missing parameter : " + k) # now check that the values are reasonable. I mostly just # made up these values from experience, and we are only # raising a warning. Better way to do this? [IR] if params["V_0"] < 1.0e-7 or params["V_0"] > 1.0e-3: warnings.warn("Unusual value for V_0", stacklevel=2) if params["K_0"] < 1.0e9 or params["K_0"] > 1.0e13: warnings.warn("Unusual value for K_0", stacklevel=2) if params["Kprime_0"] < -5.0 or params["Kprime_0"] > 10.0: warnings.warn("Unusual value for Kprime_0", stacklevel=2)