/* Copyright 2018 SINTEF Digital, Mathematics and Cybernetics. Copyright 2018 Equinor. 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 3 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 . */ #ifndef OPM_CONVERGENCEREPORT_HEADER_INCLUDED #define OPM_CONVERGENCEREPORT_HEADER_INCLUDED #include #include #include #include #include #include namespace Opm { /// Represents the convergence status of the whole simulator, to /// make it possible to query and store the reasons for /// convergence failures. class ConvergenceReport { public: // ----------- Types ----------- enum Status { AllGood = 0, ReservoirFailed = 1 << 0, WellFailed = 1 << 1 }; enum struct Severity { None = 0, Normal = 1, TooLarge = 2, NotANumber = 3 }; class ReservoirFailure { public: enum struct Type { Invalid, MassBalance, Cnv }; ReservoirFailure(Type t, Severity s, int phase) : type_(t), severity_(s), phase_(phase) { } Type type() const { return type_; } Severity severity() const { return severity_; } int phase() const { return phase_; } private: Type type_; Severity severity_; int phase_; }; class ReservoirConvergenceMetric { public: ReservoirConvergenceMetric(ReservoirFailure::Type t, int phase, double value) : type_(t), phase_(phase), value_(value) { } ReservoirFailure::Type type() const { return type_; } int phase() const { return phase_; } double value() const { return value_; } private: ReservoirFailure::Type type_; int phase_; double value_; }; class WellFailure { public: enum struct Type { Invalid, MassBalance, Pressure, ControlBHP, ControlTHP, ControlRate, Unsolvable, WrongFlowDirection }; WellFailure(Type t, Severity s, int phase, const std::string& well_name) : type_(t), severity_(s), phase_(phase), well_name_(well_name) { } Type type() const { return type_; } Severity severity() const { return severity_; } int phase() const { return phase_; } const std::string& wellName() const { return well_name_; } private: Type type_; Severity severity_; int phase_; std::string well_name_; }; // ----------- Mutating member functions ----------- ConvergenceReport() : ConvergenceReport{0.0} { } explicit ConvergenceReport(const double reportTime) : reportTime_{reportTime} , status_{AllGood} , res_failures_{} , well_failures_{} , wellGroupTargetsViolated_(false) { } void clear() { status_ = AllGood; res_failures_.clear(); well_failures_.clear(); wellGroupTargetsViolated_ = false; } void setReservoirFailed(const ReservoirFailure& rf) { status_ = static_cast(status_ | ReservoirFailed); res_failures_.push_back(rf); } void setWellFailed(const WellFailure& wf) { status_ = static_cast(status_ | WellFailed); well_failures_.push_back(wf); } template void setReservoirConvergenceMetric(Args&&... args) { this->res_convergence_.emplace_back(std::forward(args)...); } void setWellGroupTargetsViolated(const bool wellGroupTargetsViolated) { wellGroupTargetsViolated_ = wellGroupTargetsViolated; } ConvergenceReport& operator+=(const ConvergenceReport& other) { reportTime_ = std::max(reportTime_, other.reportTime_); status_ = static_cast(status_ | other.status_); res_failures_.insert(res_failures_.end(), other.res_failures_.begin(), other.res_failures_.end()); well_failures_.insert(well_failures_.end(), other.well_failures_.begin(), other.well_failures_.end()); res_convergence_.insert(res_convergence_.end(), other.res_convergence_.begin(), other.res_convergence_.end()); assert(reservoirFailed() != res_failures_.empty()); assert(wellFailed() != well_failures_.empty()); wellGroupTargetsViolated_ = (wellGroupTargetsViolated_ || other.wellGroupTargetsViolated_); return *this; } // ----------- Const member functions (queries) ----------- double reportTime() const { return reportTime_; } bool converged() const { return (status_ == AllGood) && !wellGroupTargetsViolated_; } bool reservoirFailed() const { return status_ & ReservoirFailed; } bool wellFailed() const { return status_ & WellFailed; } const std::vector& reservoirFailures() const { return res_failures_; } const std::vector& reservoirConvergence() const { return res_convergence_; } const std::vector& wellFailures() const { return well_failures_; } Severity severityOfWorstFailure() const { // A function to get the worst of two severities. auto smax = [](Severity s1, Severity s2) { return s1 < s2 ? s2 : s1; }; auto s = Severity::None; for (const auto& f : res_failures_) { s = smax(s, f.severity()); } for (const auto& f : well_failures_) { s = smax(s, f.severity()); } return s; } private: // ----------- Member variables ----------- double reportTime_; Status status_; std::vector res_failures_; std::vector well_failures_; std::vector res_convergence_; bool wellGroupTargetsViolated_; }; struct StepReport { int report_step; int current_step; std::vector report; }; std::string to_string(const ConvergenceReport::ReservoirFailure::Type t); std::string to_string(const ConvergenceReport::Severity s); std::string to_string(const ConvergenceReport::WellFailure::Type t); std::string to_string(const ConvergenceReport::WellFailure& wf); } // namespace Opm #endif // OPM_CONVERGENCEREPORT_HEADER_INCLUDED