8#include <boost/math/special_functions/gamma.hpp>
11#include "cantera/numerics/eigen_dense.h"
15#include <boost/polymorphic_pointer_cast.hpp>
32 m_eedfSolver = make_unique<EEDFTwoTermApproximation>(
this);
36 size_t nGridCells = 301;
43PlasmaPhase::~PlasmaPhase()
45 if (shared_ptr<Solution> soln =
m_soln.lock()) {
46 soln->removeChangedCallback(
this);
47 soln->kinetics()->removeReactionAddedCallback(
this);
62 "No electron species found.");
78 if (cached.
state1 != tempNow || cached.
state2 != electronTempNow) {
84 cached.
state2 = electronTempNow;
100 if ((spec->name ==
"e" || spec->name ==
"Electron") ||
101 (spec->composition.find(
"E") != spec->composition.end() &&
102 spec->composition.size() == 1 &&
103 spec->composition[
"E"] == 1)) {
108 "Cannot add species, {}. "
109 "Only one electron species is allowed.", spec->name);
119 if (shared_ptr<Solution> soln =
m_soln.lock()) {
120 soln->registerChangedCallback(
this, [&]() {
133 eedf[
"energy-levels"] = levels;
135 eedf[
"shape-factor"] = m_isotropicShapeFactor;
140 eedf[
"distribution"] = dist;
143 phaseNode[
"electron-energy-distribution"] = std::move(eedf);
149 if (phaseNode.
hasKey(
"electron-energy-distribution")) {
150 const AnyMap eedf = phaseNode[
"electron-energy-distribution"].as<
AnyMap>();
153 if (eedf.
hasKey(
"shape-factor")) {
157 "isotropic type requires shape-factor key.");
159 if (eedf.
hasKey(
"mean-electron-energy")) {
160 double energy = eedf.
convert(
"mean-electron-energy",
"eV");
164 "isotropic type requires electron-temperature key.");
166 if (eedf.
hasKey(
"energy-levels")) {
167 auto levels = eedf[
"energy-levels"].asVector<
double>();
172 if (!eedf.
hasKey(
"energy-levels")) {
174 "Cannot find key energy-levels.");
176 if (!eedf.
hasKey(
"distribution")) {
178 "Cannot find key distribution.");
180 if (eedf.
hasKey(
"normalize")) {
183 auto levels = eedf[
"energy-levels"].asVector<
double>();
184 auto distribution = eedf[
"distribution"].asVector<
double>(levels.size());
189 if (rootNode.
hasKey(
"electron-collisions")) {
190 for (
const auto& item : rootNode[
"electron-collisions"].asVector<
AnyMap>()) {
191 auto rate = make_shared<ElectronCollisionPlasmaRate>(item);
193 reactants[item[
"target"].asString()] = 1;
195 if (item.hasKey(
"product")) {
196 products[item[
"product"].asString()] = 1;
198 products[item[
"target"].asString()] = 1;
201 if (rate->kind() ==
"ionization") {
203 }
else if (rate->kind() ==
"attachment") {
206 auto R = make_shared<Reaction>(reactants, products, rate);
219 throw CanteraError(
"PlasmaPhase::updateElectronEnergyDistribution",
220 "Invalid for discretized electron energy distribution.");
224 auto ierr =
m_eedfSolver->calculateDistributionFunction();
229 throw CanteraError(
"PlasmaPhase::updateElectronEnergyDistribution",
230 "Call to calculateDistributionFunction failed.");
242 writelog(
"Skipping Te update: EEDF is empty, non-finite, or unnormalized.\n");
245 throw CanteraError(
"PlasmaPhase::updateElectronEnergyDistribution",
257 throw CanteraError(
"PlasmaPhase::normalizeElectronEnergyDistribution",
258 "The norm is negative. This might be caused by bad "
259 "electron energy distribution");
266 if (
type ==
"discretized" ||
267 type ==
"isotropic" ||
268 type ==
"Boltzmann-two-term") {
271 throw CanteraError(
"PlasmaPhase::setElectronEnergyDistributionType",
272 "Unknown type for electron energy distribution.");
279 double x = m_isotropicShapeFactor;
280 double gamma1 = boost::math::tgamma(3.0 / 2.0 / x);
281 double gamma2 = boost::math::tgamma(5.0 / 2.0 / x);
282 double c1 = x * std::pow(gamma2, 1.5) / std::pow(gamma1, 2.5);
283 double c2 = std::pow(gamma2 / gamma1, x);
293 throw CanteraError(
"PlasmaPhase::setElectronTemperature",
294 "Electron temperature cannot be negative.");
347 const auto& rate = boost::polymorphic_pointer_downcast
359 throw CanteraError(
"PlasmaPhase::checkElectronEnergyLevels",
360 "Values of electron energy levels need to be positive and "
361 "monotonically increasing.");
370 throw CanteraError(
"PlasmaPhase::checkElectronEnergyDistribution",
371 "Values of electron energy distribution cannot be negative.");
374 warn_user(
"PlasmaPhase::checkElectronEnergyDistribution",
375 "The value of the last element of electron energy distribution exceed 0.01. "
376 "This indicates that the value of electron energy level is not high enough "
377 "to contain the isotropic distribution at mean electron energy of "
383 span<const double> dist)
412 if (epsilon_m < 0.0) {
413 throw CanteraError(
"PlasmaPhase::updateElectronTemperatureFromEnergyDist",
414 "The electron energy distribution produces negative electron temperature.");
421 m_isotropicShapeFactor = x;
427 if (shared_ptr<Solution> soln =
m_soln.lock()) {
428 shared_ptr<Kinetics> kin = soln->kinetics();
435 set<Reaction*> existing;
437 existing.insert(R.get());
439 for (
size_t i = 0; i < kin->nReactions(); i++) {
440 shared_ptr<Reaction> R = kin->reaction(i);
441 if (R->rate()->type() !=
"electron-collision-plasma"
442 || existing.count(R.get())) {
450 kin->registerReactionAddedCallback(
this, [
this, kin]() {
451 size_t i = kin->nReactions() - 1;
452 if (kin->reaction(i)->type() ==
"electron-collision-plasma") {
453 addCollision(kin->reaction(i));
468 std::dynamic_pointer_cast<ElectronCollisionPlasmaRate>(
collision->rate());
481 if (target.empty()) {
482 throw CanteraError(
"PlasmaPhase::addCollision",
"Error identifying target for"
483 " collision with equation '{}'",
collision->equation());
488 std::dynamic_pointer_cast<ElectronCollisionPlasmaRate>(
collision->rate()));
499 if ((kind ==
"effective" || kind ==
"elastic")) {
505 throw CanteraError(
"PlasmaPhase::addCollision",
"Phase already contains"
506 " an effective/elastic cross section for '{}'.", target);
514 auto levels = rate.energyLevels();
516 auto sections = rate.crossSections();
542 for (
size_t i = 1; i <
m_nPoints - 1; i++) {
548 (h1 * h0) / (h1 + h0);
577 if (interpChanged[i]) {
596 auto cs_array = Eigen::Map<const Eigen::ArrayXd>(
618 "EEDF not initialized");
632 if (!std::isfinite(q_elastic)) {
634 "Non-finite elastic power loss");
647 "Electron mobility is only available for 'Boltzmann-two-term' "
648 "electron energy distributions.");
661 for (
size_t k = 0; k <
m_kk; ++k) {
672 for (
size_t k = 0; k <
m_kk; ++k) {
683 for (
size_t k = 0; k <
m_kk; ++k) {
694 for (
size_t k = 0; k <
m_kk; ++k) {
709 return T_g + X_e * (T_e - T_g);
729 for (
size_t k = 0; k <
nSpecies(); k++) {
738 for (
size_t k = 0; k <
m_kk; k++) {
753 mu[k] += (
RTe() -
RT()) * log(xx);
782 for (
size_t k = 0; k <
m_kk; k++) {
783 ubar[k] =
RT() * (_h[k] - 1.0);
787 ubar[k] =
RTe() * (_h[k] - 1.0);
793 for (
size_t k = 0; k <
m_kk; k++) {
812 for (
size_t k = 0; k <
m_kk; k++) {
824 for (
size_t k = 0; k <
m_kk; k++) {
858 AnyMap state = input_state;
861 if (state.
hasKey(
"electron-temperature")) {
862 state[
"Te"] = state[
"electron-temperature"];
870 if (state.
hasKey(
"gas-temperature")) {
871 state[
"T"] = state[
"gas-temperature"];
874 state[
"T"] = state[
"Tg"];
897 return sigma * E * E;
910 return qJ + qElastic;
EEDF Two-Term approximation solver.
Header for plasma reaction rates parameterized by electron collision cross section and electron energ...
Base class for kinetics managers and also contains the kineticsmgr module documentation (see Kinetics...
Header file for class PlasmaPhase.
Declaration for class Cantera::Species.
A map of string keys to values whose type can vary at runtime.
bool hasKey(const string &key) const
Returns true if the map contains an item named key.
double convert(const string &key, const string &units) const
Convert the item stored by the given key to the units specified in units.
Base class for exceptions thrown by Cantera classes.
Electron collision plasma reaction rate type.
void updateInterpolatedCrossSection(span< const double >)
Update the value of m_crossSectionsInterpolated [m2].
void getGibbs_ref(span< double > g) const override
Returns the vector of the Gibbs function of the reference state at the current temperature of the sol...
void getPartialMolarEnthalpies(span< double > hbar) const override
Returns an array of partial molar enthalpies for the species in the mixture.
vector< double > m_g0_RT
Temporary storage for dimensionless reference state Gibbs energies.
vector< double > m_h0_RT
Temporary storage for dimensionless reference state enthalpies.
span< const double > enthalpy_RT_ref() const
Returns a reference to the dimensionless reference state enthalpy vector.
virtual void updateThermo() const
Update the species reference state thermodynamic functions.
vector< double > m_s0_R
Temporary storage for dimensionless reference state entropies.
void getPartialMolarEntropies(span< double > sbar) const override
Returns an array of partial molar entropies of the species in the solution.
vector< double > m_cp0_R
Temporary storage for dimensionless reference state heat capacities.
bool addSpecies(shared_ptr< Species > spec) override
Add a Species to this Phase.
void getChemPotentials(span< double > mu) const override
Get the species chemical potentials. Units: J/kmol.
void getStandardVolumes_ref(span< double > vol) const override
Get the molar volumes of the species reference states at the current T and P_ref of the solution.
virtual void update_single(size_t k, double T, double &cp_R, double &h_RT, double &s_R) const
Get reference-state properties for a single species.
An error indicating that an unimplemented function has been called.
ValueCache m_cache
Cached for saved calculations within each ThermoPhase.
size_t nSpecies() const
Returns the number of species in the phase.
size_t m_kk
Number of species in the phase.
size_t speciesIndex(const string &name, bool raise=true) const
Returns the index of a species named 'name' within the Phase object.
double temperature() const
Temperature (K).
double meanMolecularWeight() const
The mean molecular weight. Units: (kg/kmol)
virtual double concentration(const size_t k) const
Concentration of species k.
double moleFraction(size_t k) const
Return the mole fraction of a single species.
virtual double density() const
Density (kg/m^3).
double molecularWeight(size_t k) const
Molecular weight of species k.
string name() const
Return the name of the phase.
void checkElectronEnergyDistribution() const
Check the electron energy distribution.
void getStandardChemPotentials(span< double > muStar) const override
Return the standard chemical potentials of the species. Units: J/kmol.
vector< vector< double > > m_energyLevels
Electron energy levels corresponding to the cross section data.
void setCollisions()
Set collisions.
double meanElectronEnergy() const
Mean electron energy [eV].
void getGibbs_ref(span< double > g) const override
Return the reference chemical potentials of the species. Units: J/kmol.
double m_electronTempEquil
Saved electron temperature during an equilibrium solve.
double enthalpy_mole() const override
Return the Molar enthalpy. Units: J/kmol.
size_t m_nPoints
Number of points of electron energy levels.
void setState(const AnyMap &state) override
Set the state using an AnyMap containing any combination of properties supported by the thermodynamic...
void getActivities(span< double > a) const override
Get the array of non-dimensional activities at the current solution temperature, pressure,...
void addCollision(shared_ptr< Reaction > collision)
Add a collision and record the target species.
virtual void setSolution(std::weak_ptr< Solution > soln) override
Set the link to the Solution object that owns this ThermoPhase.
void normalizeElectronEnergyDistribution()
Electron energy distribution norm.
void updateThermo() const override
Update the species reference state thermodynamic functions.
void getPartialMolarEnthalpies(span< double > hbar) const override
Return the partial molar enthalpies of the species in the solution. Units: J/kmol.
vector< size_t > m_targetSpeciesIndices
The collision-target species indices of m_collisions.
void setElectronTemperature(double Te) override
Set the internally stored electron temperature of the phase [K].
void electronEnergyLevelChanged()
When electron energy level changed, plasma properties such as electron-collision reaction rates need ...
double pressure() const override
Return the pressure of the plasma phase. Units: Pa.
double elasticPowerLoss()
The elastic power loss [J/s/m³].
int m_levelNum
Electron energy level change variable.
bool updateInterpolatedCrossSection(size_t k)
Update interpolated cross section of a collision.
bool m_inEquilibrate
Lock flag (default off)
void electronEnergyDistributionChanged()
When electron energy distribution changed, plasma properties such as electron-collision reaction rate...
size_t nElectronEnergyLevels() const
Number of electron levels.
size_t nCollisions() const
Number of electron collision cross sections.
void endEquilibrate() override
Hook called at the end of an equilibrium calculation on this phase.
Eigen::ArrayXd m_electronEnergyDist
Normalized electron energy distribution vector [-] Length: m_nPoints.
double electricField() const
Get the applied electric field strength [V/m].
Eigen::ArrayXd m_electronEnergyLevels
electron energy levels [ev]. Length: m_nPoints
void getActivityCoefficients(span< double > ac) const override
Get the array of non-dimensional activity coefficients at the current solution temperature,...
double meanTemperature() const
Return the mean temperature of the plasma phase. Units: K.
double intrinsicHeating() override
Intrinsic volumetric heating rate [W/m³].
double electronMobility() const
The electron mobility (m²/V/s)
void getParameters(AnyMap &phaseNode) const override
Store the parameters of a ThermoPhase object such that an identical one could be reconstructed using ...
string type() const override
String indicating the thermodynamic model implemented.
void checkElectronEnergyLevels() const
Check the electron energy levels.
void initThermo() override
Initialize the ThermoPhase object after all species have been set up.
void updateElasticElectronEnergyLossCoefficients()
Update elastic electron energy loss coefficients.
void updateElectronTemperatureFromEnergyDist()
Update electron temperature (K) From energy distribution.
string m_distributionType
Electron energy distribution type. Can be "isotropic", "discretized" or "Boltzmann-two-term".
void updateElectronEnergyDistribution()
Update the electron energy distribution.
vector< double > m_elasticElectronEnergyLossCoefficients
Elastic electron energy loss coefficients (eV m3/s)
string m_quadratureMethod
Numerical quadrature method for electron energy distribution.
PlasmaPhase(const string &inputFile="", const string &id="")
Construct and initialize a PlasmaPhase object directly from an input file.
void beginEquilibrate() override
Hook called at the beginning of an equilibrium calculation on this phase.
void setDiscretizedElectronEnergyDist(span< const double > levels, span< const double > distrb)
Set discretized electron energy distribution.
double m_electronTemp
Electron temperature [K].
double RTe() const
Return the Gas Constant multiplied by the current electron temperature [J/kmol].
double intEnergy_mole() const override
Return the molar internal energy. Units: J/kmol.
double entropy_mole() const override
Return the molar entropy. Units: J/kmol/K.
bool m_do_normalizeElectronEnergyDist
Flag of normalizing electron energy distribution.
void updateElectronEnergyDistDifference()
Update electron energy distribution difference.
void updateElasticElectronEnergyLossCoefficient(size_t i)
Updates the elastic electron energy loss coefficient for collision index i.
vector< size_t > m_kElastic
Indices of elastic collisions in m_crossSections.
unique_ptr< EEDFTwoTermApproximation > m_eedfSolver
Solver used to calculate the EEDF based on electron collision rates.
string electronSpeciesName() const
Electron species name.
void setIsotropicElectronEnergyDistribution()
Set isotropic electron energy distribution.
void getPartialMolarVolumes(span< double > vbar) const override
Return the partial molar volumes of the species in the solution. Units: m³/kmol.
Eigen::ArrayXd m_electronEnergyDistDiff
ionization degree for the electron-electron collisions (tmp is the previous one)
void getStandardVolumes(span< double > vol) const override
Return the standard molar volumes of the species. Units: m³/kmol.
void getPartialMolarEntropies(span< double > sbar) const override
Return the partial molar entropies of the species in the solution. Units: J/kmol/K.
double gibbs_mole() const override
Return the molar Gibbs free energy. Units: J/kmol.
double standardConcentration(size_t k=0) const override
Returns the standard concentration , which is used to normalize the generalized concentration.
std::vector< double > m_work
Work array.
bool addSpecies(shared_ptr< Species > spec) override
Add a Species to this Phase.
const shared_ptr< Reaction > collision(size_t i) const
Get the Reaction object associated with electron collision i.
vector< bool > m_interp_cs_ready
The list of whether the interpolated cross sections is ready.
vector< shared_ptr< ElectronCollisionPlasmaRate > > m_collisionRates
The list of shared pointers of collision rates.
void getChemPotentials(span< double > mu) const override
Return the chemical potentials of the species in the solution. Units: J/kmol.
void setElectronEnergyLevels(span< const double > levels)
Set electron energy levels.
vector< shared_ptr< Reaction > > m_collisions
The list of shared pointers of plasma collision reactions.
void setParameters(const AnyMap &phaseNode, const AnyMap &rootNode=AnyMap()) override
Set equation of state parameters from an AnyMap phase description.
void setMeanElectronEnergy(double energy)
Set mean electron energy [eV].
void getStandardVolumes_ref(span< double > vol) const override
Return the molar volumes of the species reference states. Units: m³/kmol.
size_t m_electronSpeciesIndex
Index of electron species.
vector< vector< double > > m_crossSections
Cross section data.
void setElectronEnergyDistributionType(const string &type)
Set electron energy distribution type.
double jouleHeatingPower() const
The joule heating power (W/m³)
vector< size_t > m_kInelastic
Indices of inelastic collisions in m_crossSections.
double electronTemperature() const override
Electron Temperature [K].
void setIsotropicShapeFactor(double x)
Set the shape factor of isotropic electron energy distribution.
void enableNormalizeElectronEnergyDist(bool enable)
Set flag of automatically normalize electron energy distribution.
void getPartialMolarIntEnergies(span< double > ubar) const override
Return the partial molar internal energies of the species in the solution. Units: J/kmol.
int m_distNum
Electron energy distribution change variable.
virtual void endEquilibrate()
Hook called at the end of an equilibrium calculation on this phase.
virtual void setParameters(const AnyMap &phaseNode, const AnyMap &rootNode=AnyMap())
Set equation of state parameters from an AnyMap phase description.
virtual void getParameters(AnyMap &phaseNode) const
Store the parameters of a ThermoPhase object such that an identical one could be reconstructed using ...
virtual void setState(const AnyMap &state)
Set the state using an AnyMap containing any combination of properties supported by the thermodynamic...
double RT() const
Return the Gas Constant multiplied by the current temperature.
virtual void setSolution(std::weak_ptr< Solution > soln)
Set the link to the Solution object that owns this ThermoPhase.
virtual void initThermo()
Initialize the ThermoPhase object after all species have been set up.
void initThermoFile(const string &inputFile, const string &id)
Initialize a ThermoPhase object using an input file.
std::weak_ptr< Solution > m_soln
reference to Solution
virtual void beginEquilibrate()
Hook called at the beginning of an equilibrium calculation on this phase.
MultiSpeciesThermo m_spthermo
Pointer to the calculation manager for species reference-state thermodynamic properties.
virtual double refPressure() const
Returns the reference pressure in Pa.
CachedScalar getScalar(int id)
Get a reference to a CachedValue object representing a scalar (double) with the given id.
int getId()
Get a unique id for a cached value.
Header for a file containing miscellaneous numerical functions.
This file contains definitions for utility functions and text for modules, inputfiles and logging,...
void writelog(const string &fmt, const Args &... args)
Write a formatted message to the screen.
double numericalQuadrature(const string &method, const Eigen::ArrayXd &f, const Eigen::ArrayXd &x)
Numerical integration of a function.
const double Boltzmann
Boltzmann constant [J/K].
const double Avogadro
Avogadro's Number [number/kmol].
const double GasConstant
Universal Gas Constant [J/kmol/K].
const double ElectronCharge
Elementary charge [C].
const double ElectronMass
Electron Mass [kg].
void warn_user(const string &method, const string &msg, const Args &... args)
Print a user warning raised from method as CanteraWarning.
Namespace for the Cantera kernel.
const size_t npos
index returned by functions to indicate "no position"
void checkFinite(const double tmp)
Check to see that a number is finite (not NaN, +Inf or -Inf)
MappedVector asVectorXd(vector< double > &v)
Convenience wrapper for accessing std::vector as an Eigen VectorXd.
span< double > asSpan(Eigen::DenseBase< Derived > &v)
Convenience wrapper for accessing Eigen vector/array/map data as a span.
const double SmallNumber
smallest number to compare to zero.
map< string, double > Composition
Map from string names to doubles.
void checkArraySize(const char *procedure, size_t available, size_t required)
Wrapper for throwing ArraySizeError.
A cached property value and the state at which it was evaluated.
double state2
Value of the second state variable for the state at which value was evaluated, for example density or...
bool validate(double state1New)
Check whether the currently cached value is valid based on a single state variable.
double state1
Value of the first state variable for the state at which value was evaluated, for example temperature...