28#ifndef OPM_BLACK_OIL_NEWTON_METHOD_HPP
29#define OPM_BLACK_OIL_NEWTON_METHOD_HPP
31#include <opm/common/Exceptions.hpp>
41#include <opm/material/common/Valgrind.hpp>
50template <
class TypeTag,
class MyTypeTag>
51struct DiscNewtonMethod;
62template <
class TypeTag>
75 static constexpr bool enableBioeffects = getPropValue<TypeTag, Properties::EnableBioeffects>();
76 using BioeffectsModule = BlackOilBioeffectsModule<TypeTag, enableBioeffects>;
78 static const unsigned numEq = getPropValue<TypeTag, Properties::NumEq>();
79 static constexpr bool enableSaltPrecipitation = getPropValue<TypeTag, Properties::EnableSaltPrecipitation>();
92 ParentType::finishInit();
94 wasSwitched_.resize(this->model().numTotalDof(),
false);
102 ParentType::registerParameters();
111 {
return numPriVarsSwitched_; }
122 numPriVarsSwitched_ = 0;
123 ParentType::beginIteration_();
133 const SolutionVector& uLastIter)
138 const int localSwitched = numPriVarsSwitched_;
139 MPI_Allreduce(&localSwitched,
140 &numPriVarsSwitched_,
147 this->simulator_.model().newtonMethod().endIterMsg()
148 <<
", num switched=" << numPriVarsSwitched_;
150 ParentType::endIteration_(uCurrentIter, uLastIter);
155 const SolutionVector& currentSolution,
156 const GlobalEqVector& solutionUpdate,
157 const GlobalEqVector& currentResidual)
159 const auto& comm = this->simulator_.gridView().comm();
163 ParentType::update_(nextSolution,
172 succeeded = comm.min(succeeded);
175 throw NumericalProblem(
"A process did not succeed in adapting the primary variables");
178 numPriVarsSwitched_ = comm.sum(numPriVarsSwitched_);
181 template <
class DofIndices>
183 const SolutionVector& currentSolution,
184 const GlobalEqVector& solutionUpdate,
185 const GlobalEqVector& currentResidual,
186 const DofIndices& dofIndices)
188 const auto zero = 0.0 * solutionUpdate[0];
189 for (
auto dofIdx : dofIndices) {
190 if (solutionUpdate[dofIdx] == zero) {
194 nextSolution[dofIdx],
195 currentSolution[dofIdx],
196 solutionUpdate[dofIdx],
197 currentResidual[dofIdx]);
206 PrimaryVariables& nextValue,
207 const PrimaryVariables& currentValue,
208 const EqVector& update,
209 const EqVector& currentResidual)
211 static constexpr bool enableSolvent =
212 Indices::solventSaturationIdx != std::numeric_limits<unsigned>::max();
213 static constexpr bool enableExtbo =
214 Indices::zFractionIdx != std::numeric_limits<unsigned>::max();
215 static constexpr bool enablePolymer =
216 Indices::polymerConcentrationIdx != std::numeric_limits<unsigned>::max();
217 static constexpr bool enablePolymerWeight =
218 Indices::polymerMoleWeightIdx != std::numeric_limits<unsigned>::max();
219 static constexpr bool enableFullyImplicitThermal =
220 Indices::temperatureIdx != std::numeric_limits<unsigned>::max();
221 static constexpr bool enableFoam =
222 Indices::foamConcentrationIdx != std::numeric_limits<unsigned>::max();
223 static constexpr bool enableBrine =
224 Indices::saltConcentrationIdx != std::numeric_limits<unsigned>::max();
225 static constexpr bool enableMICP = Indices::enableMICP;
227 currentValue.checkDefined();
228 Valgrind::CheckDefined(update);
229 Valgrind::CheckDefined(currentResidual);
232 Scalar deltaSw = 0.0;
233 Scalar deltaSo = 0.0;
234 Scalar deltaSg = 0.0;
235 Scalar deltaSs = 0.0;
237 if (currentValue.primaryVarsMeaningWater() == PrimaryVariables::WaterMeaning::Sw)
239 if constexpr (Indices::waterSwitchIdx != std::numeric_limits<unsigned>::max()) {
240 deltaSw = update[Indices::waterSwitchIdx];
244 if (currentValue.primaryVarsMeaningGas() == PrimaryVariables::GasMeaning::Sg)
246 if constexpr (Indices::compositionSwitchIdx != std::numeric_limits<unsigned>::max()) {
247 deltaSg = update[Indices::compositionSwitchIdx];
251 if (currentValue.primaryVarsMeaningSolvent() == PrimaryVariables::SolventMeaning::Ss) {
252 if constexpr (Indices::solventSaturationIdx != std::numeric_limits<unsigned>::max()) {
253 deltaSs = update[Indices::solventSaturationIdx];
259 Scalar maxSatDelta = std::max(std::abs(deltaSg), std::abs(deltaSo));
260 maxSatDelta = std::max(maxSatDelta, std::abs(deltaSw));
261 maxSatDelta = std::max(maxSatDelta, std::abs(deltaSs));
265 Scalar satAlpha = 1.0;
266 if (maxSatDelta > bparams_.dsMax_) {
267 satAlpha = bparams_.dsMax_ / maxSatDelta;
270 for (
unsigned pvIdx = 0; pvIdx < numEq; ++pvIdx) {
277 Scalar delta = update[pvIdx];
280 if (pvIdx == Indices::pressureSwitchIdx) {
281 if (std::abs(delta) > bparams_.dpMaxRel_ * currentValue[pvIdx]) {
282 delta =
signum(delta) * bparams_.dpMaxRel_ * currentValue[pvIdx];
286 else if (pvIdx == Indices::waterSwitchIdx)
287 if (currentValue.primaryVarsMeaningWater() == PrimaryVariables::WaterMeaning::Sw) {
292 if (delta > currentValue[ Indices::waterSwitchIdx]) {
293 delta = currentValue[ Indices::waterSwitchIdx];
296 else if (pvIdx == Indices::compositionSwitchIdx) {
302 if (currentValue.primaryVarsMeaningGas() == PrimaryVariables::GasMeaning::Sg) {
307 if (delta > currentValue[Indices::compositionSwitchIdx]) {
308 delta = currentValue[Indices::compositionSwitchIdx];
312 else if (enableSolvent && pvIdx == Indices::solventSaturationIdx) {
314 if (currentValue.primaryVarsMeaningSolvent() == PrimaryVariables::SolventMeaning::Ss) {
319 if (delta > currentValue[Indices::solventSaturationIdx]) {
320 delta = currentValue[Indices::solventSaturationIdx];
324 else if (enableExtbo && pvIdx == Indices::zFractionIdx) {
326 const auto& curr = currentValue[Indices::zFractionIdx];
327 delta = std::clamp(delta, curr - Scalar{1.0}, curr);
329 else if (enablePolymerWeight && pvIdx == Indices::polymerMoleWeightIdx) {
330 const double sign = delta >= 0. ? 1. : -1.;
333 const Scalar maxMolarWeightChange = 100.0;
334 delta = sign * std::min(std::abs(delta), maxMolarWeightChange);
337 else if (enableFullyImplicitThermal && pvIdx == Indices::temperatureIdx) {
338 const double sign = delta >= 0. ? 1. : -1.;
339 delta = sign * std::min(std::abs(delta), bparams_.maxTempChange_);
341 else if (enableBrine && pvIdx == Indices::saltConcentrationIdx &&
342 enableSaltPrecipitation &&
343 currentValue.primaryVarsMeaningBrine() == PrimaryVariables::BrineMeaning::Sp)
345 const Scalar maxSaltSaturationChange = 0.1;
346 const Scalar sign = delta >= 0. ? 1. : -1.;
347 delta = sign * std::min(std::abs(delta), maxSaltSaturationChange);
351 nextValue[pvIdx] = currentValue[pvIdx] - delta;
354 if (enableSolvent && pvIdx == Indices::solventSaturationIdx) {
355 if (currentValue.primaryVarsMeaningSolvent() == PrimaryVariables::SolventMeaning::Ss) {
356 nextValue[pvIdx] = std::min(std::max(nextValue[pvIdx], Scalar{0.0}), Scalar{1.0});
361 if (enableExtbo && pvIdx == Indices::zFractionIdx) {
362 nextValue[pvIdx] = std::min(std::max(nextValue[pvIdx], Scalar{0.0}), Scalar{1.0});
366 if (enablePolymer && pvIdx == Indices::polymerConcentrationIdx) {
367 nextValue[pvIdx] = std::max(nextValue[pvIdx], Scalar{0.0});
370 if (enablePolymerWeight && pvIdx == Indices::polymerMoleWeightIdx) {
371 nextValue[pvIdx] = std::max(nextValue[pvIdx], Scalar{0.0});
372 const double polymerConcentration = nextValue[Indices::polymerConcentrationIdx];
373 if (polymerConcentration < 1.e-10) {
374 nextValue[pvIdx] = 0.0;
379 if (enableFoam && pvIdx == Indices::foamConcentrationIdx) {
380 nextValue[pvIdx] = std::max(nextValue[pvIdx], Scalar{0.0});
383 if (enableBrine && pvIdx == Indices::saltConcentrationIdx) {
385 if (!enableSaltPrecipitation ||
386 currentValue.primaryVarsMeaningBrine() == PrimaryVariables::BrineMeaning::Cs)
388 nextValue[pvIdx] = std::max(nextValue[pvIdx], Scalar{0.0});
391 if (enableSaltPrecipitation &&
392 currentValue.primaryVarsMeaningBrine() == PrimaryVariables::BrineMeaning::Sp)
394 nextValue[pvIdx] = std::min(nextValue[pvIdx], Scalar{1.0-1.e-8});
399 if (enableFullyImplicitThermal && pvIdx == Indices::temperatureIdx) {
400 nextValue[pvIdx] = std::clamp(nextValue[pvIdx], bparams_.tempMin_, bparams_.tempMax_);
403 if (pvIdx == Indices::pressureSwitchIdx) {
404 nextValue[pvIdx] = std::clamp(nextValue[pvIdx], bparams_.pressMin_, bparams_.pressMax_);
412 if constexpr (enableBioeffects) {
413 if (pvIdx == Indices::microbialConcentrationIdx) {
414 nextValue[pvIdx] = std::max(nextValue[pvIdx], Scalar{0.0});
416 if (pvIdx == Indices::biofilmVolumeFractionIdx) {
417 nextValue[pvIdx] = std::clamp(nextValue[pvIdx],
419 this->problem().referencePorosity(globalDofIdx, 0) - 1e-8);
421 if constexpr (enableMICP) {
422 if (pvIdx == Indices::oxygenConcentrationIdx) {
423 nextValue[pvIdx] = std::max(nextValue[pvIdx], Scalar{0.0});
425 if (pvIdx == Indices::ureaConcentrationIdx) {
426 nextValue[pvIdx] = std::max(nextValue[pvIdx], Scalar{0.0});
428 if (pvIdx == Indices::calciteVolumeFractionIdx) {
429 nextValue[pvIdx] = std::clamp(nextValue[pvIdx], Scalar{0.0},
430 this->problem().referencePorosity(globalDofIdx, 0) - 1e-8);
439 if (wasSwitched_[globalDofIdx]) {
440 wasSwitched_[globalDofIdx] = nextValue.adaptPrimaryVariables(this->problem(),
442 bparams_.waterSaturationMax_,
443 bparams_.waterOnlyThreshold_,
444 bparams_.priVarOscilationThreshold_);
447 wasSwitched_[globalDofIdx] = nextValue.adaptPrimaryVariables(this->problem(),
449 bparams_.waterSaturationMax_,
450 bparams_.waterOnlyThreshold_);
453 if (wasSwitched_[globalDofIdx]) {
454 ++numPriVarsSwitched_;
456 if (bparams_.projectSaturations_) {
457 nextValue.chopAndNormalizeSaturations();
460 nextValue.checkDefined();
464 int numPriVarsSwitched_{};
466 BlackoilNewtonParams<Scalar> bparams_{};
470 std::vector<bool> wasSwitched_{};
Contains classes extending the black-oil model. \detail This file holds dummy definitions,...
Declares the properties required by the black oil model.
A newton solver which is specific to the black oil model.
Definition: blackoilnewtonmethod.hpp:64
void beginIteration_()
Indicates the beginning of a Newton iteration.
Definition: blackoilnewtonmethod.hpp:120
static void registerParameters()
Register all run-time parameters for the blackoil newton method.
Definition: blackoilnewtonmethod.hpp:100
void update_(SolutionVector &nextSolution, const SolutionVector ¤tSolution, const GlobalEqVector &solutionUpdate, const GlobalEqVector ¤tResidual)
Definition: blackoilnewtonmethod.hpp:154
void finishInit()
Finialize the construction of the object.
Definition: blackoilnewtonmethod.hpp:90
void endIteration_(SolutionVector &uCurrentIter, const SolutionVector &uLastIter)
Indicates that one Newton iteration was finished.
Definition: blackoilnewtonmethod.hpp:132
unsigned numPriVarsSwitched() const
Returns the number of degrees of freedom for which the interpretation has changed for the most recent...
Definition: blackoilnewtonmethod.hpp:110
BlackOilNewtonMethod(Simulator &simulator)
Definition: blackoilnewtonmethod.hpp:82
void updatePrimaryVariables_(unsigned globalDofIdx, PrimaryVariables &nextValue, const PrimaryVariables ¤tValue, const EqVector &update, const EqVector ¤tResidual)
Update a single primary variables object.
Definition: blackoilnewtonmethod.hpp:205
friend ParentType
Definition: blackoilnewtonmethod.hpp:115
void update_(SolutionVector &nextSolution, const SolutionVector ¤tSolution, const GlobalEqVector &solutionUpdate, const GlobalEqVector ¤tResidual, const DofIndices &dofIndices)
Definition: blackoilnewtonmethod.hpp:182
The multi-dimensional Newton method.
Definition: newtonmethod.hh:100
Definition: blackoilmodel.hh:75
Definition: blackoilbioeffectsmodules.hh:45
constexpr int signum(Scalar val)
Template function which returns the sign of a floating point value.
Definition: signum.hh:41
typename Properties::Detail::GetPropImpl< TypeTag, Property >::type::type GetPropType
get the type alias defined in the property (equivalent to old macro GET_PROP_TYPE(....
Definition: propertysystem.hh:233
static void registerParameters()
Registers the parameters in parameter system.