opm-simulators/opm/models/pvs/pvsmodel.hh
2024-08-01 13:03:36 +02:00

647 lines
22 KiB
C++

// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
// vi: set et ts=4 sw=4 sts=4:
/*
This file is part of the Open Porous Media project (OPM).
OPM is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 2 of the License, or
(at your option) any later version.
OPM is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with OPM. If not, see <http://www.gnu.org/licenses/>.
Consult the COPYING file in the top-level source directory of this
module for the precise wording of the license and the list of
copyright holders.
*/
/*!
* \file
*
* \copydoc Opm::PvsModel
*/
#ifndef EWOMS_PVS_MODEL_HH
#define EWOMS_PVS_MODEL_HH
#include <opm/common/Exceptions.hpp>
#include <opm/material/densead/Math.hpp>
#include <opm/material/fluidmatrixinteractions/MaterialTraits.hpp>
#include <opm/material/fluidmatrixinteractions/NullMaterial.hpp>
#include <opm/models/common/diffusionmodule.hh>
#include <opm/models/common/energymodule.hh>
#include <opm/models/common/multiphasebasemodel.hh>
#include <opm/models/io/vtkcompositionmodule.hh>
#include <opm/models/io/vtkenergymodule.hh>
#include <opm/models/io/vtkdiffusionmodule.hh>
#include <opm/models/pvs/pvsboundaryratevector.hh>
#include <opm/models/pvs/pvsextensivequantities.hh>
#include <opm/models/pvs/pvsindices.hh>
#include <opm/models/pvs/pvsintensivequantities.hh>
#include <opm/models/pvs/pvslocalresidual.hh>
#include <opm/models/pvs/pvsnewtonmethod.hh>
#include <opm/models/pvs/pvsprimaryvariables.hh>
#include <opm/models/pvs/pvsproperties.hh>
#include <opm/models/pvs/pvsratevector.hh>
#include <iostream>
#include <sstream>
#include <string>
#include <vector>
namespace Opm {
template <class TypeTag>
class PvsModel;
}
namespace Opm::Properties {
namespace TTag {
//! The type tag for the isothermal single phase problems
struct PvsModel
{
using InheritsFrom = std::tuple<VtkDiffusion,
VtkEnergy,
VtkComposition,
VtkPhasePresence,
MultiPhaseBaseModel>; };
} // namespace TTag
//! Use the PVS local jacobian operator for the PVS model
template<class TypeTag>
struct LocalResidual<TypeTag, TTag::PvsModel>
{ using type = Opm::PvsLocalResidual<TypeTag>; };
//! Use the PVS specific newton method for the PVS model
template<class TypeTag>
struct NewtonMethod<TypeTag, TTag::PvsModel>
{ using type = Opm::PvsNewtonMethod<TypeTag>; };
//! the Model property
template<class TypeTag>
struct Model<TypeTag, TTag::PvsModel>
{ using type = Opm::PvsModel<TypeTag>; };
//! the PrimaryVariables property
template<class TypeTag>
struct PrimaryVariables<TypeTag, TTag::PvsModel>
{ using type = Opm::PvsPrimaryVariables<TypeTag>; };
//! the RateVector property
template<class TypeTag>
struct RateVector<TypeTag, TTag::PvsModel>
{ using type = Opm::PvsRateVector<TypeTag>; };
//! the BoundaryRateVector property
template<class TypeTag>
struct BoundaryRateVector<TypeTag, TTag::PvsModel>
{ using type = Opm::PvsBoundaryRateVector<TypeTag>; };
//! the IntensiveQuantities property
template<class TypeTag>
struct IntensiveQuantities<TypeTag, TTag::PvsModel>
{ using type = Opm::PvsIntensiveQuantities<TypeTag>; };
//! the ExtensiveQuantities property
template<class TypeTag>
struct ExtensiveQuantities<TypeTag, TTag::PvsModel>
{ using type = Opm::PvsExtensiveQuantities<TypeTag>; };
//! The indices required by the isothermal PVS model
template<class TypeTag>
struct Indices<TypeTag, TTag::PvsModel>
{ using type = Opm::PvsIndices<TypeTag, /*PVIdx=*/0>; };
//! Disable the energy equation by default
template<class TypeTag>
struct EnableEnergy<TypeTag, TTag::PvsModel>
{ static constexpr bool value = false; };
// disable molecular diffusion by default
template<class TypeTag>
struct EnableDiffusion<TypeTag, TTag::PvsModel>
{ static constexpr bool value = false; };
//! The basis value for the weight of the pressure primary variable
template<class TypeTag>
struct PvsPressureBaseWeight<TypeTag, TTag::PvsModel>
{
using type = GetPropType<TypeTag, Scalar>;
static constexpr type value = 1.0;
};
//! The basis value for the weight of the saturation primary variables
template<class TypeTag>
struct PvsSaturationsBaseWeight<TypeTag, TTag::PvsModel>
{
using type = GetPropType<TypeTag, Scalar>;
static constexpr type value = 1.0;
};
//! The basis value for the weight of the mole fraction primary variables
template<class TypeTag>
struct PvsMoleFractionsBaseWeight<TypeTag, TTag::PvsModel>
{
using type = GetPropType<TypeTag, Scalar>;
static constexpr type value = 1.0;
};
} // namespace Opm::Properties
namespace Opm::Parameters {
//! The verbosity of the model (0 -> do not print anything, 2 -> spam stdout a lot)
template<class TypeTag, class MyTypeTag>
struct PvsVerbosity { using type = Properties::UndefinedProperty; };
// set the model to a medium verbosity
template<class TypeTag>
struct PvsVerbosity<TypeTag, Properties::TTag::PvsModel>
{ static constexpr int value = 1; };
} // namespace Opm::Parameters
namespace Opm {
/*!
* \ingroup PvsModel
*
* \brief A generic compositional multi-phase model using primary-variable
* switching.
*
* This model assumes a flow of \f$M \geq 1\f$ fluid phases
* \f$\alpha\f$, each of which is assumed to be a mixture \f$N \geq
* M\f$ chemical species \f$\kappa\f$.
*
* By default, the standard multi-phase Darcy approach is used to determine
* the velocity, i.e.
* \f[
* \mathbf{v}_\alpha = - \frac{k_{r\alpha}}{\mu_\alpha} \mathbf{K}
* \left(\mathbf{grad}\, p_\alpha - \varrho_{\alpha} \mathbf{g} \right) \;,
* \f]
* although the actual approach which is used can be specified via the
* \c FluxModule property. For example, the velocity model can by
* changed to the Forchheimer approach by
* \code
* template<class TypeTag>
struct FluxModule<TypeTag, TTag::MyProblemTypeTag> { using type = Opm::ForchheimerFluxModule<TypeTag>; };
* \endcode
*
* The core of the model is the conservation mass of each component by
* means of the equation
* \f[
* \sum_\alpha \frac{\partial\;\phi c_\alpha^\kappa S_\alpha }{\partial t}
* - \sum_\alpha \mathrm{div} \left\{ c_\alpha^\kappa \mathbf{v}_\alpha \right\}
* - q^\kappa = 0 \;.
* \f]
*
* To close the system mathematically, \f$M\f$ model equations are
* also required. This model uses the primary variable switching
* assumptions, which are given by:
* \f[
* 0 \stackrel{!}{=}
* f_\alpha = \left\{
* \begin{array}{cl}
* S_\alpha& \quad \text{if phase }\alpha\text{ is not present} \ \
* 1 - \sum_\kappa x_\alpha^\kappa& \quad \text{else}
* \end{array}
* \right.
* \f]
*
* To make this approach applicable, a pseudo primary variable
* <em>phase presence</em> has to be introduced. Its purpose is to
* specify for each phase whether it is present or not. It is a
* <em>pseudo</em> primary variable because it is not directly considered when
* linearizing the system in the Newton method, but after each Newton
* iteration, it gets updated like the "real" primary variables. The
* following rules are used for this update procedure:
*
* <ul>
* <li>If phase \f$\alpha\f$ is present according to the pseudo
* primary variable, but \f$S_\alpha < 0\f$ after the Newton
* update, consider the phase \f$\alpha\f$ disappeared for the
* next iteration and use the set of primary variables which
* correspond to the new phase presence.</li>
* <li>If phase \f$\alpha\f$ is not present according to the pseudo
* primary variable, but the sum of the component mole fractions
* in the phase is larger than 1, i.e. \f$\sum_\kappa
* x_\alpha^\kappa > 1\f$, consider the phase \f$\alpha\f$ present
* in the the next iteration and update the set of primary
* variables to make it consistent with the new phase
* presence.</li>
* <li>In all other cases don't modify the phase presence for phase
* \f$\alpha\f$.</li>
*
* </ul>
*
* The model always requires \f$N\f$ primary variables, but their
* interpretation is dependent on the phase presence:
*
* <ul>
*
* <li>The first primary variable is always interpreted as the
* pressure of the phase with the lowest index \f$PV_0 =
* p_0\f$.</li>
*
* <li>Then, \f$M - 1\f$ "switching primary variables" follow, which
* are interpreted depending in the presence of the first
* \f$M-1\f$ phases: If phase \f$\alpha\f$ is present, its
* saturation \f$S_\alpha = PV_i\f$ is used as primary variable;
* if it is not present, the mole fraction \f$PV_i =
* x_{\alpha^\star}^\alpha\f$ of the component with index
* \f$\alpha\f$ in the phase with the lowest index that is present
* \f$\alpha^\star\f$ is used instead.</li>
*
* <li>Finally, the mole fractions of the \f$N-M\f$ components with
* the largest index in the phase with the lowest index that is
* present \f$x_{\alpha^\star}^\kappa\f$ are used as primary
* variables.</li>
*
* </ul>
*/
template <class TypeTag>
class PvsModel
: public MultiPhaseBaseModel<TypeTag>
{
using ParentType = MultiPhaseBaseModel<TypeTag>;
using Scalar = GetPropType<TypeTag, Properties::Scalar>;
using Simulator = GetPropType<TypeTag, Properties::Simulator>;
using FluidSystem = GetPropType<TypeTag, Properties::FluidSystem>;
using GridView = GetPropType<TypeTag, Properties::GridView>;
using PrimaryVariables = GetPropType<TypeTag, Properties::PrimaryVariables>;
using IntensiveQuantities = GetPropType<TypeTag, Properties::IntensiveQuantities>;
using ElementContext = GetPropType<TypeTag, Properties::ElementContext>;
using Indices = GetPropType<TypeTag, Properties::Indices>;
enum { numPhases = getPropValue<TypeTag, Properties::NumPhases>() };
enum { numComponents = getPropValue<TypeTag, Properties::NumComponents>() };
enum { enableDiffusion = getPropValue<TypeTag, Properties::EnableDiffusion>() };
enum { enableEnergy = getPropValue<TypeTag, Properties::EnableEnergy>() };
using Element = typename GridView::template Codim<0>::Entity;
using ElementIterator = typename GridView::template Codim<0>::Iterator;
using EnergyModule = Opm::EnergyModule<TypeTag, enableEnergy>;
public:
PvsModel(Simulator& simulator)
: ParentType(simulator)
{
verbosity_ = Parameters::get<TypeTag, Parameters::PvsVerbosity>();
numSwitched_ = 0;
}
/*!
* \brief Register all run-time parameters for the PVS compositional model.
*/
static void registerParameters()
{
ParentType::registerParameters();
// register runtime parameters of the VTK output modules
Opm::VtkPhasePresenceModule<TypeTag>::registerParameters();
Opm::VtkCompositionModule<TypeTag>::registerParameters();
if (enableDiffusion)
Opm::VtkDiffusionModule<TypeTag>::registerParameters();
if (enableEnergy)
Opm::VtkEnergyModule<TypeTag>::registerParameters();
Parameters::registerParam<TypeTag, Parameters::PvsVerbosity>
("The verbosity level of the primary variable "
"switching model");
}
/*!
* \copydoc FvBaseDiscretization::name
*/
static std::string name()
{ return "pvs"; }
/*!
* \copydoc FvBaseDiscretization::primaryVarName
*/
std::string primaryVarName(unsigned pvIdx) const
{
std::string s;
if (!(s = EnergyModule::primaryVarName(pvIdx)).empty())
return s;
std::ostringstream oss;
if (pvIdx == Indices::pressure0Idx)
oss << "pressure_" << FluidSystem::phaseName(/*phaseIdx=*/0);
else if (Indices::switch0Idx <= pvIdx
&& pvIdx < Indices::switch0Idx + numPhases - 1)
oss << "switch_" << pvIdx - Indices::switch0Idx;
else if (Indices::switch0Idx + numPhases - 1 <= pvIdx
&& pvIdx < Indices::switch0Idx + numComponents - 1)
oss << "auxMoleFrac^" << FluidSystem::componentName(pvIdx);
else
assert(false);
return oss.str();
}
/*!
* \copydoc FvBaseDiscretization::eqName
*/
std::string eqName(unsigned eqIdx) const
{
std::string s;
if (!(s = EnergyModule::eqName(eqIdx)).empty())
return s;
std::ostringstream oss;
if (Indices::conti0EqIdx <= eqIdx && eqIdx < Indices::conti0EqIdx
+ numComponents) {
unsigned compIdx = eqIdx - Indices::conti0EqIdx;
oss << "continuity^" << FluidSystem::componentName(compIdx);
}
else
assert(false);
return oss.str();
}
/*!
* \copydoc FvBaseDiscretization::updateFailed
*/
void updateFailed()
{
ParentType::updateFailed();
numSwitched_ = 0;
}
/*!
* \copydoc FvBaseDiscretization::updateBegin
*/
void updateBegin()
{
ParentType::updateBegin();
// find the a reference pressure. The first degree of freedom
// might correspond to non-interior entities which would lead
// to an undefined value, so we have to iterate...
size_t nDof = this->numTotalDof();
for (unsigned dofIdx = 0; dofIdx < nDof; ++ dofIdx) {
if (this->dofTotalVolume(dofIdx) > 0.0) {
referencePressure_ =
this->solution(/*timeIdx=*/0)[dofIdx][/*pvIdx=*/Indices::pressure0Idx];
if (referencePressure_ > 0.0)
break;
}
}
}
/*!
* \copydoc FvBaseDiscretization::primaryVarWeight
*/
Scalar primaryVarWeight(unsigned globalDofIdx, unsigned pvIdx) const
{
Scalar tmp = EnergyModule::primaryVarWeight(*this, globalDofIdx, pvIdx);
if (tmp > 0)
// energy related quantity
return tmp;
if (Indices::pressure0Idx == pvIdx) {
return 10 / referencePressure_;
}
if (Indices::switch0Idx <= pvIdx && pvIdx < Indices::switch0Idx
+ numPhases - 1) {
unsigned phaseIdx = pvIdx - Indices::switch0Idx;
if (!this->solution(/*timeIdx=*/0)[globalDofIdx].phaseIsPresent(phaseIdx))
// for saturations, the weight is always 1
return 1;
// for saturations, the PvsMoleSaturationsBaseWeight
// property determines the weight
return getPropValue<TypeTag, Properties::PvsSaturationsBaseWeight>();
}
// for mole fractions, the PvsMoleFractionsBaseWeight
// property determines the weight
return getPropValue<TypeTag, Properties::PvsMoleFractionsBaseWeight>();
}
/*!
* \copydoc FvBaseDiscretization::eqWeight
*/
Scalar eqWeight(unsigned globalDofIdx, unsigned eqIdx) const
{
Scalar tmp = EnergyModule::eqWeight(*this, globalDofIdx, eqIdx);
if (tmp > 0)
// energy related equation
return tmp;
unsigned compIdx = eqIdx - Indices::conti0EqIdx;
assert(compIdx <= numComponents);
// make all kg equal
return FluidSystem::molarMass(compIdx);
}
/*!
* \copydoc FvBaseDiscretization::advanceTimeLevel
*/
void advanceTimeLevel()
{
ParentType::advanceTimeLevel();
numSwitched_ = 0;
}
/*!
* \brief Return true if the primary variables were switched for
* at least one vertex after the last timestep.
*/
bool switched() const
{ return numSwitched_ > 0; }
/*!
* \copydoc FvBaseDiscretization::serializeEntity
*/
template <class DofEntity>
void serializeEntity(std::ostream& outstream, const DofEntity& dofEntity)
{
// write primary variables
ParentType::serializeEntity(outstream, dofEntity);
unsigned dofIdx = static_cast<unsigned>(this->dofMapper().index(dofEntity));
if (!outstream.good())
throw std::runtime_error("Could not serialize DOF "+std::to_string(dofIdx));
outstream << this->solution(/*timeIdx=*/0)[dofIdx].phasePresence() << " ";
}
/*!
* \copydoc FvBaseDiscretization::deserializeEntity
*/
template <class DofEntity>
void deserializeEntity(std::istream& instream, const DofEntity& dofEntity)
{
// read primary variables
ParentType::deserializeEntity(instream, dofEntity);
// read phase presence
unsigned dofIdx = static_cast<unsigned>(this->dofMapper().index(dofEntity));
if (!instream.good())
throw std::runtime_error("Could not deserialize DOF "+std::to_string(dofIdx));
short tmp;
instream >> tmp;
this->solution(/*timeIdx=*/0)[dofIdx].setPhasePresence(tmp);
this->solution(/*timeIdx=*/1)[dofIdx].setPhasePresence(tmp);
}
/*!
* \internal
* \brief Do the primary variable switching after a Newton iteration.
*
* This is an internal method that needs to be public because it
* gets called by the Newton method after an update.
*/
void switchPrimaryVars_()
{
numSwitched_ = 0;
int succeeded;
try {
std::vector<bool> visited(this->numGridDof(), false);
ElementContext elemCtx(this->simulator_);
for (const auto& elem : elements(this->gridView_, Dune::Partitions::interior)) {
elemCtx.updateStencil(elem);
size_t numLocalDof = elemCtx.stencil(/*timeIdx=*/0).numPrimaryDof();
for (unsigned dofIdx = 0; dofIdx < numLocalDof; ++dofIdx) {
unsigned globalIdx = elemCtx.globalSpaceIndex(dofIdx, /*timeIdx=*/0);
if (visited[globalIdx])
continue;
visited[globalIdx] = true;
// compute the intensive quantities of the current degree of freedom
auto& priVars = this->solution(/*timeIdx=*/0)[globalIdx];
elemCtx.updateIntensiveQuantities(priVars, dofIdx, /*timeIdx=*/0);
const IntensiveQuantities& intQuants = elemCtx.intensiveQuantities(dofIdx, /*timeIdx=*/0);
// evaluate primary variable switch
short oldPhasePresence = priVars.phasePresence();
// set the primary variables and the new phase state
// from the current fluid state
priVars.assignNaive(intQuants.fluidState());
if (oldPhasePresence != priVars.phasePresence()) {
if (verbosity_ > 1)
printSwitchedPhases_(elemCtx,
dofIdx,
intQuants.fluidState(),
oldPhasePresence,
priVars);
++numSwitched_;
}
}
}
succeeded = 1;
}
catch (...)
{
std::cout << "rank " << this->simulator_.gridView().comm().rank()
<< " caught an exception during primary variable switching"
<< "\n" << std::flush;
succeeded = 0;
}
succeeded = this->simulator_.gridView().comm().min(succeeded);
if (!succeeded)
throw NumericalProblem("A process did not succeed in adapting the primary variables");
// make sure that if there was a variable switch in an
// other partition we will also set the switch flag
// for our partition.
numSwitched_ = this->gridView_.comm().sum(numSwitched_);
if (verbosity_ > 0)
this->simulator_.model().newtonMethod().endIterMsg()
<< ", num switched=" << numSwitched_;
}
template <class FluidState>
void printSwitchedPhases_(const ElementContext& elemCtx,
unsigned dofIdx,
const FluidState& fs,
short oldPhasePresence,
const PrimaryVariables& newPv) const
{
using FsToolbox = Opm::MathToolbox<typename FluidState::Scalar>;
for (unsigned phaseIdx = 0; phaseIdx < numPhases; ++phaseIdx) {
bool oldPhasePresent = (oldPhasePresence& (1 << phaseIdx)) > 0;
bool newPhasePresent = newPv.phaseIsPresent(phaseIdx);
if (oldPhasePresent == newPhasePresent)
continue;
const auto& pos = elemCtx.pos(dofIdx, /*timeIdx=*/0);
if (oldPhasePresent && !newPhasePresent) {
std::cout << "'" << FluidSystem::phaseName(phaseIdx)
<< "' phase disappears at position " << pos
<< ". saturation=" << fs.saturation(phaseIdx)
<< std::flush;
}
else {
Scalar sumx = 0;
for (unsigned compIdx = 0; compIdx < numComponents; ++compIdx)
sumx += FsToolbox::value(fs.moleFraction(phaseIdx, compIdx));
std::cout << "'" << FluidSystem::phaseName(phaseIdx)
<< "' phase appears at position " << pos
<< " sum x = " << sumx << std::flush;
}
}
std::cout << ", new primary variables: ";
newPv.print();
std::cout << "\n" << std::flush;
}
void registerOutputModules_()
{
ParentType::registerOutputModules_();
// add the VTK output modules which are meaningful for the model
this->addOutputModule(new Opm::VtkPhasePresenceModule<TypeTag>(this->simulator_));
this->addOutputModule(new Opm::VtkCompositionModule<TypeTag>(this->simulator_));
if (enableDiffusion)
this->addOutputModule(new Opm::VtkDiffusionModule<TypeTag>(this->simulator_));
if (enableEnergy)
this->addOutputModule(new Opm::VtkEnergyModule<TypeTag>(this->simulator_));
}
mutable Scalar referencePressure_;
// number of switches of the phase state in the last Newton
// iteration
unsigned numSwitched_;
// verbosity of the model
int verbosity_;
};
}
#endif