diff --git a/dumux/discretization/cellcentered/tpfa/globalfvgeometry.hh b/dumux/discretization/cellcentered/tpfa/globalfvgeometry.hh index 88ac6b0eb2a41f1403e7958e2460469127bee04e..fa0ee62666e0ee2e8fb591dfb12438773847f007 100644 --- a/dumux/discretization/cellcentered/tpfa/globalfvgeometry.hh +++ b/dumux/discretization/cellcentered/tpfa/globalfvgeometry.hh @@ -166,7 +166,7 @@ public: friend inline FVElementGeometry localView(const CCTpfaGlobalFVGeometry& global) { return FVElementGeometry(global); } -private: +//private: //! Get a sub control volume with a global scv index const SubControlVolume& scv(IndexType scvIdx) const @@ -186,6 +186,7 @@ private: return scvfIndicesOfScv_[scvIdx]; } +private: const Problem& problem_() const { return *problemPtr_; } diff --git a/dumux/implicit/model.hh b/dumux/implicit/model.hh index 3fdaf5169993f047da78842a599aebe421f4a1e8..fb14955585f92ff413c873e034b84387eca4f399 100644 --- a/dumux/implicit/model.hh +++ b/dumux/implicit/model.hh @@ -51,6 +51,10 @@ class ImplicitModel // The local jacobian needs to be able to modify objects during the assembly friend typename GET_PROP_TYPE(TypeTag, LocalJacobian); + // the primary variable need to modify the volume variables when global caching is enabled + friend class PrimaryVariableSwitch<TypeTag>; + friend typename GET_PROP_TYPE(TypeTag, PrimaryVariableSwitch); + using Implementation = typename GET_PROP_TYPE(TypeTag, Model); using Problem = typename GET_PROP_TYPE(TypeTag, Problem); using GridView = typename GET_PROP_TYPE(TypeTag, GridView); @@ -87,10 +91,11 @@ class ImplicitModel enum { isBox = GET_PROP_VALUE(TypeTag, ImplicitIsBox) }; enum { dofCodim = isBox ? dim : 0 }; +public: + // copying a model is not a good idea - ImplicitModel(const ImplicitModel &); + ImplicitModel(const ImplicitModel &) = delete; -public: /*! * \brief The constructor. */ @@ -172,7 +177,7 @@ public: if (isBox) { - for (int i = 0; i < element.subEntities(dim); ++i) + for (unsigned int i = 0; i < element.subEntities(dim); ++i) { int globalI = vertexMapper().subIndex(element, i, dim); residual[globalI] += localResidual().residual(i); diff --git a/dumux/implicit/properties.hh b/dumux/implicit/properties.hh index a2e77749196378b6eccf88b2ba521718be1839dd..9d8014568e79c80a171beb47f00c716886d3637c 100644 --- a/dumux/implicit/properties.hh +++ b/dumux/implicit/properties.hh @@ -85,6 +85,7 @@ NEW_PROP_TAG(EnableGlobalFluxVariablesCache); //! specifies if data on flux vars NEW_PROP_TAG(BoundaryVariables); //!< Data required to calculate fluxes over boundary faces in cc models(outflow) NEW_PROP_TAG(ConstantBoundaryConditions); //!< boundary data is stored in case the BC are constant NEW_PROP_TAG(DiscretizationMethod); //!< Property for the used discretization method +NEW_PROP_TAG(PrimaryVariableSwitch); //!< The primary variable switch needed for compositional models // Specify the forms of fluxes that should be considered in the model // also, specify their corresponding flux variables @@ -139,9 +140,6 @@ NEW_PROP_TAG(ImplicitMaxTimeStepDivisions); //! indicate whether discretization is box or not NEW_PROP_TAG(ImplicitIsBox); -//! The type of primary variable switch for compositional models -NEW_PROP_TAG(PrimaryVariableSwitch); - //! the local fem space used for the AMG backend NEW_PROP_TAG(ImplicitLocalFemMap); diff --git a/dumux/implicit/propertydefaults.hh b/dumux/implicit/propertydefaults.hh index e49075d087a630c02bef24bf50218021ce57afaf..08d9fd97f2a34ff4070eec7dd0b69a7fb4450b31 100644 --- a/dumux/implicit/propertydefaults.hh +++ b/dumux/implicit/propertydefaults.hh @@ -38,6 +38,7 @@ #include <dumux/porousmediumflow/implicit/fluxvariables.hh> #include <dumux/porousmediumflow/implicit/fluxvariablescache.hh> #include <dumux/porousmediumflow/nonisothermal/implicit/localresidual.hh> +#include <dumux/porousmediumflow/compositional/primaryvariableswitch.hh> #include <dumux/discretization/volumevariables.hh> #include <dumux/discretization/darcyslaw.hh> @@ -154,6 +155,8 @@ SET_BOOL_PROP(ImplicitBase, EnableGlobalFluxVariablesCache, false); //! boundary conditions are not stationary by default SET_BOOL_PROP(ImplicitBase, ConstantBoundaryConditions, false); +SET_TYPE_PROP(ImplicitBase, PrimaryVariableSwitch, NoPrimaryVariableSwitch<TypeTag> ); + //! Set the type of a global jacobian matrix from the solution types SET_PROP(ImplicitBase, JacobianMatrix) { diff --git a/dumux/porousmediumflow/compositional/primaryvariableswitch.hh b/dumux/porousmediumflow/compositional/primaryvariableswitch.hh index 539ef151f14ec27183111af2dad62a6131417b33..e8b9a87d2f140c15749c07fb63a47cee66cb4723 100644 --- a/dumux/porousmediumflow/compositional/primaryvariableswitch.hh +++ b/dumux/porousmediumflow/compositional/primaryvariableswitch.hh @@ -28,6 +28,14 @@ namespace Dumux { +/*! + * \ingroup ImplicitModel + * \brief Empty class for models without pri var switch + */ +template<class TypeTag> +class NoPrimaryVariableSwitch +{}; + /*! * \ingroup ImplicitModel * \brief The primary variable switch controlling the phase presence state variable @@ -115,7 +123,9 @@ public: * \param problem The problem * \param curSol The current solution to be updated / modified */ - bool update(Problem& problem, SolutionVector& curSol) + template<class T = TypeTag> + typename std::enable_if<!GET_PROP_VALUE(T, EnableGlobalVolumeVariablesCache), bool>::type + update(Problem& problem, SolutionVector& curSol) { bool switched = false; visited_.assign(phasePresence_.size(), false); @@ -156,6 +166,48 @@ public: return switched; } + template<class T = TypeTag> + typename std::enable_if<GET_PROP_VALUE(T, EnableGlobalVolumeVariablesCache), bool>::type + update(Problem& problem, SolutionVector& curSol) + { + bool switched = false; + visited_.assign(phasePresence_.size(), false); + for (const auto& element : elements(problem.gridView())) + { + // make sure FVElementGeometry is bound to the element + auto fvGeometry = localView(problem.model().globalFvGeometry()); + fvGeometry.bindElement(element); + + auto& curGlobalVolVars = problem.model().nonConstCurGlobalVolVars(); + + for (auto&& scv : scvs(fvGeometry)) + { + auto dofIdxGlobal = scv.dofIndex(); + if (!visited_[dofIdxGlobal]) + { + // Note this implies that volume variables don't differ + // in any sub control volume associated with the dof! + visited_[dofIdxGlobal] = true; + // Compute temporary volVars on which grounds we decide + // if we need to switch the primary variables + curGlobalVolVars.volVars(dofIdxGlobal).update(curSol[dofIdxGlobal], problem, element, scv); + const auto& globalPos = scv.dofPosition(); + + if (asImp_().update_(curSol[dofIdxGlobal], curGlobalVolVars.volVars(dofIdxGlobal), dofIdxGlobal, globalPos)) + switched = true; + + } + } + } + + // make sure that if there was a variable switch in an + // other partition we will also set the switch flag for our partition. + if (problem.gridView().comm().size() > 1) + switched = problem.gridView().comm().max(switched); + + return switched; + } + protected: //! return actual implementation (static polymorphism)