/* */ #ifndef OPM_ADAPTIVE_TIME_STEPPING_EBOS_HPP #define OPM_ADAPTIVE_TIME_STEPPING_EBOS_HPP #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Opm::Properties { namespace TTag { struct FlowTimeSteppingParameters {}; } template struct SolverRestartFactor { using type = UndefinedProperty; }; template struct SolverGrowthFactor { using type = UndefinedProperty; }; template struct SolverMaxGrowth { using type = UndefinedProperty; }; template struct SolverMaxTimeStepInDays { using type = UndefinedProperty; }; template struct SolverMinTimeStep { using type = UndefinedProperty; }; template struct SolverMaxRestarts { using type = UndefinedProperty; }; template struct SolverVerbosity { using type = UndefinedProperty; }; template struct TimeStepVerbosity { using type = UndefinedProperty; }; template struct InitialTimeStepInDays { using type = UndefinedProperty; }; template struct FullTimeStepInitially { using type = UndefinedProperty; }; template struct TimeStepAfterEventInDays { using type = UndefinedProperty; }; template struct TimeStepControl { using type = UndefinedProperty; }; template struct TimeStepControlTolerance { using type = UndefinedProperty; }; template struct TimeStepControlTargetIterations { using type = UndefinedProperty; }; template struct TimeStepControlTargetNewtonIterations { using type = UndefinedProperty; }; template struct TimeStepControlDecayRate { using type = UndefinedProperty; }; template struct TimeStepControlGrowthRate { using type = UndefinedProperty; }; template struct TimeStepControlDecayDampingFactor { using type = UndefinedProperty; }; template struct TimeStepControlGrowthDampingFactor { using type = UndefinedProperty; }; template struct TimeStepControlFileName { using type = UndefinedProperty; }; template struct MinTimeStepBeforeShuttingProblematicWellsInDays { using type = UndefinedProperty; }; template struct SolverRestartFactor { using type = GetPropType; static constexpr type value = 0.33; }; template struct SolverGrowthFactor { using type = GetPropType; static constexpr type value = 2.0; }; template struct SolverMaxGrowth { using type = GetPropType; static constexpr type value = 3.0; }; template struct SolverMaxTimeStepInDays { using type = GetPropType; static constexpr type value = 365.0; }; template struct SolverMinTimeStep { using type = GetPropType; static constexpr type value = 0.0; }; template struct SolverMaxRestarts { static constexpr int value = 10; }; template struct SolverVerbosity { static constexpr int value = 1; }; template struct TimeStepVerbosity { static constexpr int value = 1; }; template struct InitialTimeStepInDays { using type = GetPropType; static constexpr type value = 1.0; }; template struct FullTimeStepInitially { static constexpr bool value = false; }; template struct TimeStepAfterEventInDays { using type = GetPropType; static constexpr type value = -1.0; }; template struct TimeStepControl { static constexpr auto value = "pid+newtoniteration"; }; template struct TimeStepControlTolerance { using type = GetPropType; static constexpr type value = 1e-1; }; template struct TimeStepControlTargetIterations { static constexpr int value = 30; }; template struct TimeStepControlTargetNewtonIterations { static constexpr int value = 8; }; template struct TimeStepControlDecayRate { using type = GetPropType; static constexpr type value = 0.75; }; template struct TimeStepControlGrowthRate { using type = GetPropType; static constexpr type value = 1.25; }; template struct TimeStepControlDecayDampingFactor { using type = GetPropType; static constexpr type value = 1.0; }; template struct TimeStepControlGrowthDampingFactor { using type = GetPropType; static constexpr type value = 3.2; }; template struct TimeStepControlFileName { static constexpr auto value = "timesteps"; }; template struct MinTimeStepBeforeShuttingProblematicWellsInDays { using type = GetPropType; static constexpr type value = 0.001; }; } // namespace Opm::Properties namespace Opm { // AdaptiveTimeStepping //--------------------- template class AdaptiveTimeSteppingEbos { template class SolutionTimeErrorSolverWrapperEbos : public RelativeChangeInterface { const Solver& solver_; public: SolutionTimeErrorSolverWrapperEbos(const Solver& solver) : solver_(solver) {} /// return || u^n+1 - u^n || / || u^n+1 || double relativeChange() const { return solver_.model().relativeChange(); } }; template void logException_(const E& exception, bool verbose) { if (verbose) { std::string message; message = "Caught Exception: "; message += exception.what(); OpmLog::debug(message); } } public: //! \brief contructor taking parameter object AdaptiveTimeSteppingEbos(const bool terminalOutput = true) : timeStepControl_() , restartFactor_(EWOMS_GET_PARAM(TypeTag, double, SolverRestartFactor)) // 0.33 , growthFactor_(EWOMS_GET_PARAM(TypeTag, double, SolverGrowthFactor)) // 2.0 , maxGrowth_(EWOMS_GET_PARAM(TypeTag, double, SolverMaxGrowth)) // 3.0 , maxTimeStep_(EWOMS_GET_PARAM(TypeTag, double, SolverMaxTimeStepInDays)*24*60*60) // 365.25 , minTimeStep_(EWOMS_GET_PARAM(TypeTag, double, SolverMinTimeStep)) // 0.0 , solverRestartMax_(EWOMS_GET_PARAM(TypeTag, int, SolverMaxRestarts)) // 10 , solverVerbose_(EWOMS_GET_PARAM(TypeTag, int, SolverVerbosity) > 0 && terminalOutput) // 2 , timestepVerbose_(EWOMS_GET_PARAM(TypeTag, int, TimeStepVerbosity) > 0 && terminalOutput) // 2 , suggestedNextTimestep_(EWOMS_GET_PARAM(TypeTag, double, InitialTimeStepInDays)*24*60*60) // 1.0 , fullTimestepInitially_(EWOMS_GET_PARAM(TypeTag, bool, FullTimeStepInitially)) // false , timestepAfterEvent_(EWOMS_GET_PARAM(TypeTag, double, TimeStepAfterEventInDays)*24*60*60) // 1e30 , useNewtonIteration_(false) , minTimeStepBeforeShuttingProblematicWells_(EWOMS_GET_PARAM(TypeTag, double, MinTimeStepBeforeShuttingProblematicWellsInDays)*unit::day) { init_(); } //! \brief contructor taking parameter object //! \param tuning Pointer to ecl TUNING keyword //! \param timeStep current report step AdaptiveTimeSteppingEbos(const Tuning& tuning, const bool terminalOutput = true) : timeStepControl_() , restartFactor_(tuning.TSFCNV) , growthFactor_(tuning.TFDIFF) , maxGrowth_(tuning.TSFMAX) , maxTimeStep_(tuning.TSMAXZ) // 365.25 , minTimeStep_(EWOMS_GET_PARAM(TypeTag, double, SolverMinTimeStep)) // 0.0 , solverRestartMax_(EWOMS_GET_PARAM(TypeTag, int, SolverMaxRestarts)) // 10 , solverVerbose_(EWOMS_GET_PARAM(TypeTag, int, SolverVerbosity) > 0 && terminalOutput) // 2 , timestepVerbose_(EWOMS_GET_PARAM(TypeTag, int, TimeStepVerbosity) > 0 && terminalOutput) // 2 , suggestedNextTimestep_(tuning.TSINIT) // 1.0 , fullTimestepInitially_(EWOMS_GET_PARAM(TypeTag, bool, FullTimeStepInitially)) // false , timestepAfterEvent_(tuning.TMAXWC) // 1e30 , useNewtonIteration_(false) , minTimeStepBeforeShuttingProblematicWells_(EWOMS_GET_PARAM(TypeTag, double, MinTimeStepBeforeShuttingProblematicWellsInDays)*unit::day) { init_(); } static void registerParameters() { // TODO: make sure the help messages are correct (and useful) EWOMS_REGISTER_PARAM(TypeTag, double, SolverRestartFactor, "The factor time steps are elongated after restarts"); EWOMS_REGISTER_PARAM(TypeTag, double, SolverGrowthFactor, "The factor time steps are elongated after a successful substep"); EWOMS_REGISTER_PARAM(TypeTag, double, SolverMaxGrowth, "The maximum factor time steps are elongated after a report step"); EWOMS_REGISTER_PARAM(TypeTag, double, SolverMaxTimeStepInDays, "The maximum size of a time step in days"); EWOMS_REGISTER_PARAM(TypeTag, double, SolverMinTimeStep, "The minimum size of a time step in seconds. If a step cannot converge without getting cut below this step size the simulator will stop"); EWOMS_REGISTER_PARAM(TypeTag, int, SolverMaxRestarts, "The maximum number of breakdowns before a substep is given up and the simulator is terminated"); EWOMS_REGISTER_PARAM(TypeTag, int, SolverVerbosity, "Specify the \"chattiness\" of the non-linear solver itself"); EWOMS_REGISTER_PARAM(TypeTag, int, TimeStepVerbosity, "Specify the \"chattiness\" during the time integration"); EWOMS_REGISTER_PARAM(TypeTag, double, InitialTimeStepInDays, "The size of the initial time step in days"); EWOMS_REGISTER_PARAM(TypeTag, bool, FullTimeStepInitially, "Always attempt to finish a report step using a single substep"); EWOMS_REGISTER_PARAM(TypeTag, double, TimeStepAfterEventInDays, "Time step size of the first time step after an event occurs during the simulation in days"); EWOMS_REGISTER_PARAM(TypeTag, std::string, TimeStepControl, "The algorithm used to determine time-step sizes. valid options are: 'pid' (default), 'pid+iteration', 'pid+newtoniteration', 'iterationcount', 'newtoniterationcount' and 'hardcoded'"); EWOMS_REGISTER_PARAM(TypeTag, double, TimeStepControlTolerance, "The tolerance used by the time step size control algorithm"); EWOMS_REGISTER_PARAM(TypeTag, int, TimeStepControlTargetIterations, "The number of linear iterations which the time step control scheme should aim for (if applicable)"); EWOMS_REGISTER_PARAM(TypeTag, int, TimeStepControlTargetNewtonIterations, "The number of Newton iterations which the time step control scheme should aim for (if applicable)"); EWOMS_REGISTER_PARAM(TypeTag, double, TimeStepControlDecayRate, "The decay rate of the time step size of the number of target iterations is exceeded"); EWOMS_REGISTER_PARAM(TypeTag, double, TimeStepControlGrowthRate, "The growth rate of the time step size of the number of target iterations is undercut"); EWOMS_REGISTER_PARAM(TypeTag, double, TimeStepControlDecayDampingFactor, "The decay rate of the time step decrease when the target iterations is exceeded"); EWOMS_REGISTER_PARAM(TypeTag, double, TimeStepControlGrowthDampingFactor, "The growth rate of the time step increase when the target iterations is undercut"); EWOMS_REGISTER_PARAM(TypeTag, std::string, TimeStepControlFileName, "The name of the file which contains the hardcoded time steps sizes"); EWOMS_REGISTER_PARAM(TypeTag, double, MinTimeStepBeforeShuttingProblematicWellsInDays, "The minimum time step size in days for which problematic wells are not shut"); } /** \brief step method that acts like the solver::step method in a sub cycle of time steps */ template SimulatorReport step(const SimulatorTimer& simulatorTimer, Solver& solver, const bool isEvent, const std::vector* fipnum = nullptr) { SimulatorReport report; const double timestep = simulatorTimer.currentStepLength(); // init last time step as a fraction of the given time step if (suggestedNextTimestep_ < 0) { suggestedNextTimestep_ = restartFactor_ * timestep; } if (fullTimestepInitially_) { suggestedNextTimestep_ = timestep; } // use seperate time step after event if (isEvent && timestepAfterEvent_ > 0) { suggestedNextTimestep_ = timestepAfterEvent_; } auto& ebosSimulator = solver.model().ebosSimulator(); auto& ebosProblem = ebosSimulator.problem(); // create adaptive step timer with previously used sub step size AdaptiveSimulatorTimer substepTimer(simulatorTimer, suggestedNextTimestep_, maxTimeStep_); // counter for solver restarts int restarts = 0; // sub step time loop while (!substepTimer.done()) { // get current delta t const double dt = substepTimer.currentStepLength() ; if (timestepVerbose_) { std::ostringstream ss; boost::posix_time::time_facet* facet = new boost::posix_time::time_facet("%d-%b-%Y"); ss.imbue(std::locale(std::locale::classic(), facet)); ss <<"\nTime step " << substepTimer.currentStepNum() << ", stepsize " << unit::convert::to(substepTimer.currentStepLength(), unit::day) << " days," << " at day " << (double)unit::convert::to(substepTimer.simulationTimeElapsed(), unit::day) << "/" << (double)unit::convert::to(substepTimer.totalTime(), unit::day) << ", date = " << substepTimer.currentDateTime(); OpmLog::info(ss.str()); } SimulatorReportSingle substepReport; std::string causeOfFailure = ""; try { substepReport = solver.step(substepTimer); if (solverVerbose_) { // report number of linear iterations OpmLog::debug("Overall linear iterations used: " + std::to_string(substepReport.total_linear_iterations)); } } catch (const Opm::TooManyIterations& e) { substepReport = solver.failureReport(); causeOfFailure = "Solver convergence failure - Iteration limit reached"; logException_(e, solverVerbose_); // since linearIterations is < 0 this will restart the solver } catch (const Opm::LinearSolverProblem& e) { substepReport = solver.failureReport(); causeOfFailure = "Linear solver convergence failure"; logException_(e, solverVerbose_); // since linearIterations is < 0 this will restart the solver } catch (const Opm::NumericalIssue& e) { substepReport = solver.failureReport(); causeOfFailure = "Solver convergence failure - Numerical problem encountered"; logException_(e, solverVerbose_); // since linearIterations is < 0 this will restart the solver } catch (const std::runtime_error& e) { substepReport = solver.failureReport(); logException_(e, solverVerbose_); // also catch linear solver not converged } catch (const Dune::ISTLError& e) { substepReport = solver.failureReport(); logException_(e, solverVerbose_); // also catch errors in ISTL AMG that occur when time step is too large } catch (const Dune::MatrixBlockError& e) { substepReport = solver.failureReport(); logException_(e, solverVerbose_); // this can be thrown by ISTL's ILU0 in block mode, yet is not an ISTLError } report += substepReport; if (substepReport.converged) { // advance by current dt ++substepTimer; // create object to compute the time error, simply forwards the call to the model SolutionTimeErrorSolverWrapperEbos relativeChange(solver); // compute new time step estimate const int iterations = useNewtonIteration_ ? substepReport.total_newton_iterations : substepReport.total_linear_iterations; double dtEstimate = timeStepControl_->computeTimeStepSize(dt, iterations, relativeChange, substepTimer.simulationTimeElapsed()); assert(dtEstimate > 0); // limit the growth of the timestep size by the growth factor dtEstimate = std::min(dtEstimate, double(maxGrowth_ * dt)); assert(dtEstimate > 0); // further restrict time step size growth after convergence problems if (restarts > 0) { dtEstimate = std::min(growthFactor_ * dt, dtEstimate); // solver converged, reset restarts counter restarts = 0; } // Further restrict time step size if we are in // prediction mode with THP constraints. if (solver.model().wellModel().hasTHPConstraints()) { const double maxPredictionTHPTimestep = 16.0 * unit::day; dtEstimate = std::min(dtEstimate, maxPredictionTHPTimestep); } assert(dtEstimate > 0); if (timestepVerbose_) { std::ostringstream ss; substepReport.reportStep(ss); OpmLog::info(ss.str()); } // write data if outputWriter was provided // if the time step is done we do not need // to write it as this will be done by the simulator // anyway. if (!substepTimer.done()) { if (fipnum) { solver.computeFluidInPlace(*fipnum); } Opm::time::StopWatch perfTimer; perfTimer.start(); ebosProblem.writeOutput(); report.success.output_write_time += perfTimer.secsSinceStart(); } // set new time step length substepTimer.provideTimeStepEstimate(dtEstimate); report.success.converged = substepTimer.done(); substepTimer.setLastStepFailed(false); } else { // in case of no convergence substepTimer.setLastStepFailed(true); // If we have restarted (i.e. cut the timestep) too // many times, we have failed and throw an exception. if (restarts >= solverRestartMax_) { const auto msg = std::string("Solver failed to converge after cutting timestep ") + std::to_string(restarts) + " times."; if (solverVerbose_) { OpmLog::error(msg); } OPM_THROW_NOLOG(Opm::NumericalIssue, msg); } // The new, chopped timestep. const double newTimeStep = restartFactor_ * dt; // If we have restarted (i.e. cut the timestep) too // much, we have failed and throw an exception. if (newTimeStep < minTimeStep_) { const auto msg = std::string("Solver failed to converge after cutting timestep to ") + std::to_string(minTimeStep_) + "\n which is the minimum threshold given" + "by option --solver-min-time-step= \n"; if (solverVerbose_) { OpmLog::error(msg); } OPM_THROW_NOLOG(Opm::NumericalIssue, msg); } // Define utility function for chopping timestep. auto chopTimestep = [&]() { substepTimer.provideTimeStepEstimate(newTimeStep); if (solverVerbose_) { std::string msg; msg = causeOfFailure + "\nTimestep chopped to " + std::to_string(unit::convert::to(substepTimer.currentStepLength(), unit::day)) + " days\n"; OpmLog::problem(msg); } ++restarts; }; const double minimumChoppedTimestep = minTimeStepBeforeShuttingProblematicWells_; if (newTimeStep > minimumChoppedTimestep) { chopTimestep(); } else { // We are below the threshold, and will check if there are any // wells we should close rather than chopping again. std::set failing_wells = consistentlyFailingWells(solver.model().stepReports()); if (failing_wells.empty()) { // Found no wells to close, chop the timestep as above. chopTimestep(); } else { // Close all consistently failing wells. int num_shut_wells = 0; for (const auto& well : failing_wells) { bool was_shut = solver.model().wellModel().forceShutWellByNameIfPredictionMode(well, substepTimer.simulationTimeElapsed()); if (was_shut) { ++num_shut_wells; } } if (num_shut_wells == 0) { // None of the problematic wells were prediction wells, // so none were shut. We must fall back to chopping again. chopTimestep(); } else { substepTimer.provideTimeStepEstimate(dt); if (solverVerbose_) { std::string msg; msg = "\nProblematic well(s) were shut: "; for (const auto& well : failing_wells) { msg += well; msg += " "; } msg += "(retrying timestep)\n"; OpmLog::problem(msg); } } } } } ebosProblem.setNextTimeStepSize(substepTimer.currentStepLength()); } // store estimated time step for next reportStep suggestedNextTimestep_ = substepTimer.currentStepLength(); if (timestepVerbose_) { std::ostringstream ss; substepTimer.report(ss); ss << "Suggested next step size = " << unit::convert::to(suggestedNextTimestep_, unit::day) << " (days)" << std::endl; OpmLog::debug(ss.str()); } if (! std::isfinite(suggestedNextTimestep_)) { // check for NaN suggestedNextTimestep_ = timestep; } return report; } /** \brief Returns the simulator report for the failed substeps of the last * report step. */ double suggestedNextStep() const { return suggestedNextTimestep_; } void setSuggestedNextStep(const double x) { suggestedNextTimestep_ = x; } void updateTUNING(const Tuning& tuning) { restartFactor_ = tuning.TSFCNV; growthFactor_ = tuning.TFDIFF; maxGrowth_ = tuning.TSFMAX; maxTimeStep_ = tuning.TSMAXZ; suggestedNextTimestep_ = tuning.TSINIT; timestepAfterEvent_ = tuning.TMAXWC; } protected: void init_() { // valid are "pid" and "pid+iteration" std::string control = EWOMS_GET_PARAM(TypeTag, std::string, TimeStepControl); // "pid" const double tol = EWOMS_GET_PARAM(TypeTag, double, TimeStepControlTolerance); // 1e-1 if (control == "pid") { timeStepControl_ = TimeStepControlType(new PIDTimeStepControl(tol)); } else if (control == "pid+iteration") { const int iterations = EWOMS_GET_PARAM(TypeTag, int, TimeStepControlTargetIterations); // 30 const double decayDampingFactor = EWOMS_GET_PARAM(TypeTag, double, TimeStepControlDecayDampingFactor); // 1.0 const double growthDampingFactor = EWOMS_GET_PARAM(TypeTag, double, TimeStepControlGrowthDampingFactor); // 3.2 timeStepControl_ = TimeStepControlType(new PIDAndIterationCountTimeStepControl(iterations, decayDampingFactor, growthDampingFactor, tol)); } else if (control == "pid+newtoniteration") { const int iterations = EWOMS_GET_PARAM(TypeTag, int, TimeStepControlTargetNewtonIterations); // 8 const double decayDampingFactor = EWOMS_GET_PARAM(TypeTag, double, TimeStepControlDecayDampingFactor); // 1.0 const double growthDampingFactor = EWOMS_GET_PARAM(TypeTag, double, TimeStepControlGrowthDampingFactor); // 3.2 timeStepControl_ = TimeStepControlType(new PIDAndIterationCountTimeStepControl(iterations, decayDampingFactor, growthDampingFactor, tol)); useNewtonIteration_ = true; } else if (control == "iterationcount") { const int iterations = EWOMS_GET_PARAM(TypeTag, int, TimeStepControlTargetIterations); // 30 const double decayrate = EWOMS_GET_PARAM(TypeTag, double, TimeStepControlDecayRate); // 0.75 const double growthrate = EWOMS_GET_PARAM(TypeTag, double, TimeStepControlGrowthRate); // 1.25 timeStepControl_ = TimeStepControlType(new SimpleIterationCountTimeStepControl(iterations, decayrate, growthrate)); } else if (control == "newtoniterationcount") { const int iterations = EWOMS_GET_PARAM(TypeTag, int, TimeStepControlTargetNewtonIterations); // 8 const double decayrate = EWOMS_GET_PARAM(TypeTag, double, TimeStepControlDecayRate); // 0.75 const double growthrate = EWOMS_GET_PARAM(TypeTag, double, TimeStepControlGrowthRate); // 1.25 timeStepControl_ = TimeStepControlType(new SimpleIterationCountTimeStepControl(iterations, decayrate, growthrate)); useNewtonIteration_ = true; } else if (control == "hardcoded") { const std::string filename = EWOMS_GET_PARAM(TypeTag, std::string, TimeStepControlFileName); // "timesteps" timeStepControl_ = TimeStepControlType(new HardcodedTimeStepControl(filename)); } else OPM_THROW(std::runtime_error,"Unsupported time step control selected "<< control); // make sure growth factor is something reasonable assert(growthFactor_ >= 1.0); } template std::set consistentlyFailingWells(const StepReportVector& sr) { // If there are wells that cause repeated failures, we // close them, and restart the un-chopped timestep. std::ostringstream msg; msg << " Excessive chopping detected in report step " << sr.back().report_step << ", substep " << sr.back().current_step << "\n"; std::set failing_wells; // return empty set if no report exists // well failures in assembly is not yet registred if(sr.back().report.empty()) return failing_wells; const auto& wfs = sr.back().report.back().wellFailures(); for (const auto& wf : wfs) { msg << " Well that failed: " << wf.wellName() << "\n"; } msg.flush(); OpmLog::debug(msg.str()); // Check the last few step reports. const int num_steps = 3; const int rep_step = sr.back().report_step; const int sub_step = sr.back().current_step; const int sr_size = sr.size(); if (sr_size >= num_steps) { for (const auto& wf : wfs) { failing_wells.insert(wf.wellName()); } for (int s = 1; s < num_steps; ++s) { const auto& srep = sr[sr_size - 1 - s]; // Report must be from same report step and substep, otherwise we have // not chopped/retried enough times on this step. if (srep.report_step != rep_step || srep.current_step != sub_step) { break; } // Get the failing wells for this step, that also failed all other steps. std::set failing_wells_step; for (const auto& wf : srep.report.back().wellFailures()) { if (failing_wells.count(wf.wellName()) > 0) { failing_wells_step.insert(wf.wellName()); } } failing_wells.swap(failing_wells_step); } } return failing_wells; } typedef std::unique_ptr TimeStepControlType; TimeStepControlType timeStepControl_; //!< time step control object double restartFactor_; //!< factor to multiply time step with when solver fails to converge double growthFactor_; //!< factor to multiply time step when solver recovered from failed convergence double maxGrowth_; //!< factor that limits the maximum growth of a time step double maxTimeStep_; //!< maximal allowed time step size in days double minTimeStep_; //!< minimal allowed time step size before throwing int solverRestartMax_; //!< how many restart of solver are allowed bool solverVerbose_; //!< solver verbosity bool timestepVerbose_; //!< timestep verbosity double suggestedNextTimestep_; //!< suggested size of next timestep bool fullTimestepInitially_; //!< beginning with the size of the time step from data file double timestepAfterEvent_; //!< suggested size of timestep after an event bool useNewtonIteration_; //!< use newton iteration count for adaptive time step control double minTimeStepBeforeShuttingProblematicWells_; //! < shut problematic wells when time step size in days are less than this }; } #endif // OPM_ADAPTIVE_TIME_STEPPING_EBOS_HPP