ReservoirCoupling.hpp
Go to the documentation of this file.
1/*
2 Copyright 2024 Equinor ASA
3
4 This file is part of the Open Porous Media project (OPM).
5
6 OPM is free software: you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation, either version 3 of the License, or
9 (at your option) any later version.
10
11 OPM is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
15
16 You should have received a copy of the GNU General Public License
17 along with OPM. If not, see <http://www.gnu.org/licenses/>.
18*/
19
20#ifndef OPM_RESERVOIR_COUPLING_HPP
21#define OPM_RESERVOIR_COUPLING_HPP
24#include <opm/input/eclipse/Schedule/Group/Group.hpp>
25#include <opm/input/eclipse/Schedule/Group/GuideRate.hpp>
26#include <opm/input/eclipse/Units/Units.hpp>
27
28#include <dune/common/parallel/mpitraits.hh>
29
30#include <fmt/format.h>
31
32#include <mpi.h>
33#include <cmath>
34#include <iostream>
35#include <memory>
36#include <stdexcept>
37#include <vector>
38
39namespace Opm {
40namespace ReservoirCoupling {
41
42class Logger {
43public:
44 Logger() = delete; // No default constructor - must have comm
45 explicit Logger(const Parallel::Communication& comm) : comm_(comm) {}
46
47 void clearDeferredLogger() { deferred_logger_ = nullptr; }
48 void debug(const std::string &msg) const;
49 DeferredLogger& deferredLogger() { return *deferred_logger_; }
50 DeferredLogger& deferredLogger() const { return *deferred_logger_; }
51 bool haveDeferredLogger() const { return deferred_logger_ != nullptr; }
52 void info(const std::string &msg) const;
53 void warning(const std::string &msg) const;
54 void setDeferredLogger(DeferredLogger *deferred_logger) { deferred_logger_ = deferred_logger; }
55
56private:
57 const Parallel::Communication& comm_;
58 DeferredLogger *deferred_logger_ = nullptr;
59};
60
89public:
90 explicit ScopedLoggerGuard(Logger& logger, DeferredLogger* deferred_logger)
91 : logger_(&logger)
92 {
93 logger_->setDeferredLogger(deferred_logger);
94 }
95
97 // Only clear if we still own the logger (not moved-from)
98 if (logger_) {
99 logger_->clearDeferredLogger();
100 }
101 }
102
103 // Prevent copying to ensure single ownership
106
107 // Enable moving - required for returning from functions and std::optional
109 : logger_(other.logger_)
110 {
111 // Transfer ownership: moved-from object becomes inactive and won't clear the logger
112 other.logger_ = nullptr;
113 }
114
116 if (this != &other) {
117 // Clean up current logger before taking ownership of new one
118 if (logger_) {
119 logger_->clearDeferredLogger();
120 }
121 // Transfer ownership from other
122 logger_ = other.logger_;
123 other.logger_ = nullptr;
124 }
125 return *this;
126 }
127
128private:
129 // Use pointer instead of reference to enable move semantics
130 // (references cannot be reassigned, which is required for move operations)
131 Logger* logger_{nullptr};
132};
133
134enum class MessageTag : int {
145 SlaveName,
152};
153
156enum class Phase : std::size_t {
157 Oil = 0, // Matches Opm::Phase::OIL
158 Gas, // Matches Opm::Phase::GAS
159 Water, // Matches Opm::Phase::WATER
160 Count
161};
162
168enum class RateKind {
174};
175
176template <class Scalar>
178 InjectionRates() = default;
179
180 std::array<Scalar, static_cast<std::size_t>(Phase::Count)> rate{};
181 [[nodiscard]] Scalar& operator[](Phase p) noexcept { return rate[static_cast<std::size_t>(p)]; }
182 [[nodiscard]] Scalar operator[](Phase p) const noexcept { return rate[static_cast<std::size_t>(p)]; }
183};
184
185
186// Used to communicate potentials for oil, gas, and water rates between slave and master processes
187template <class Scalar>
189 std::array<Scalar, static_cast<std::size_t>(Phase::Count)> rate{};
190
191 [[nodiscard]] Scalar& operator[](Phase p) noexcept { return rate[static_cast<std::size_t>(p)]; }
192 [[nodiscard]] Scalar operator[](Phase p) const noexcept { return rate[static_cast<std::size_t>(p)]; }
193};
194
195template <class Scalar>
197 ProductionRates() = default;
198
199 explicit ProductionRates(const GuideRate::RateVector& rate_vector)
200 : rate{static_cast<Scalar>(rate_vector.oil_rat),
201 static_cast<Scalar>(rate_vector.gas_rat),
202 static_cast<Scalar>(rate_vector.wat_rat)}
203 {}
204
205 std::array<Scalar, static_cast<std::size_t>(Phase::Count)> rate{};
206 [[nodiscard]] Scalar& operator[](Phase p) noexcept { return rate[static_cast<std::size_t>(p)]; }
207 [[nodiscard]] Scalar operator[](Phase p) const noexcept { return rate[static_cast<std::size_t>(p)]; }
208};
209
210// Slave group production data sent to the corresponding master group for target calculation.
211template <class Scalar>
213 // Group production potentials are used by the master group for guiderate calculations
215 // Production rates are used by the master group in guiderate calculations
216 // when converting the guide rate target to the phase of the master group.
217 ProductionRates<Scalar> surface_rates; // Surface production rates by phase (network=false)
218 // Network surface rates - computed with network=true, meaning efficiency factors
219 // are 1.0 for groups/wells with GEFAC/WEFAC item 3 = "NO"
220 ProductionRates<Scalar> network_surface_rates; // Surface rates for network calculations
221 // Individual phase reservoir production rates - needed when master's parent group
222 // has RESV control mode, so the conversion uses slave's PVT properties
223 ProductionRates<Scalar> reservoir_rates; // Reservoir production rates by phase
224 Scalar voidage_rate{0.0}; // Reservoir voidage replacement rate
225 Scalar gas_reinjection_rate{0.0}; // Reinjection (surface) rate for the gas phase
226};
227
228// Slave group injection data sent to the corresponding master group for target calculation.
229template <class Scalar>
231 InjectionRates<Scalar> surface_rates; // Surface injection rates by phase
232 InjectionRates<Scalar> reservoir_rates; // Reservoir injection rates by phase
233};
234
235template <class Scalar>
237 // To save memory and avoid varying size of the struct when serializing
238 // and deserializing the group name, we use an index instead of the full name.
239 std::size_t group_name_idx; // Index of group name in the master group names vector
240 Scalar target; // Target rate for the group
241 Group::InjectionCMode cmode; // Control mode for the group
242 Phase phase; // Phase the target applies to
243};
244
245template <class Scalar>
247 // To save memory and avoid varying size of the struct when serializing
248 // and deserializing the group name, we use an index instead of the full name.
249 std::size_t group_name_idx; // Index of group name in the master group names vector
250 Scalar target; // Target rate for the group
251 Group::ProductionCMode cmode; // Control mode for the group
252};
253
254// Helper functions
256void customErrorHandler_(MPI_Comm* comm, int* err, const std::string &msg);
257void customErrorHandlerSlave_(MPI_Comm* comm, int* err, ...);
258void customErrorHandlerMaster_(MPI_Comm* comm, int* err, ...);
259void setErrhandler(MPI_Comm comm, bool is_master);
260std::pair<std::vector<char>, std::size_t> serializeStrings(const std::vector<std::string>& data);
261
265inline std::string formatDays(double seconds) {
266 double days = seconds / unit::day;
267 return fmt::format(fmt::runtime("{:.0f}s ({:.2f} days)"), seconds, days);
268}
269
287struct Seconds {
289 static constexpr double abstol = 1e-15;
290
292 static constexpr double reltol = 1e-15;
293
312 static bool compare_eq(double a, double b);
313
319 static bool compare_gt(double a, double b);
320
326 static bool compare_gt_or_eq(double a, double b);
327
333 static bool compare_lt(double a, double b);
334
340 static bool compare_lt_or_eq(double a, double b);
341};
342
343} // namespace ReservoirCoupling
344} // namespace Opm
345
346#endif // OPM_RESERVOIR_COUPLING_HPP
Definition: DeferredLogger.hpp:57
Definition: ReservoirCoupling.hpp:42
void info(const std::string &msg) const
void warning(const std::string &msg) const
void setDeferredLogger(DeferredLogger *deferred_logger)
Definition: ReservoirCoupling.hpp:54
void clearDeferredLogger()
Definition: ReservoirCoupling.hpp:47
Logger(const Parallel::Communication &comm)
Definition: ReservoirCoupling.hpp:45
bool haveDeferredLogger() const
Definition: ReservoirCoupling.hpp:51
DeferredLogger & deferredLogger()
Definition: ReservoirCoupling.hpp:49
void debug(const std::string &msg) const
DeferredLogger & deferredLogger() const
Definition: ReservoirCoupling.hpp:50
Guard for managing DeferredLogger lifecycle in ReservoirCoupling.
Definition: ReservoirCoupling.hpp:88
~ScopedLoggerGuard()
Definition: ReservoirCoupling.hpp:96
ScopedLoggerGuard & operator=(const ScopedLoggerGuard &)=delete
ScopedLoggerGuard(ScopedLoggerGuard &&other) noexcept
Definition: ReservoirCoupling.hpp:108
ScopedLoggerGuard(const ScopedLoggerGuard &)=delete
ScopedLoggerGuard & operator=(ScopedLoggerGuard &&other) noexcept
Definition: ReservoirCoupling.hpp:115
ScopedLoggerGuard(Logger &logger, DeferredLogger *deferred_logger)
Definition: ReservoirCoupling.hpp:90
Dune::Communication< MPIComm > Communication
Definition: ParallelCommunication.hpp:30
std::string formatDays(double seconds)
Format seconds as a human-readable string showing both seconds and days.
Definition: ReservoirCoupling.hpp:265
RateKind
Selects which kind of rate to retrieve from slave group data.
Definition: ReservoirCoupling.hpp:168
void customErrorHandler_(MPI_Comm *comm, int *err, const std::string &msg)
std::pair< std::vector< char >, std::size_t > serializeStrings(const std::vector< std::string > &data)
void customErrorHandlerSlave_(MPI_Comm *comm, int *err,...)
MessageTag
Definition: ReservoirCoupling.hpp:134
void setErrhandler(MPI_Comm comm, bool is_master)
void customErrorHandlerMaster_(MPI_Comm *comm, int *err,...)
Phase convertPhaseToReservoirCouplingPhase(::Opm::Phase phase)
Phase
Phase indices for reservoir coupling, we currently only support black-oil phases (oil,...
Definition: ReservoirCoupling.hpp:156
Definition: blackoilbioeffectsmodules.hh:43
Definition: ReservoirCoupling.hpp:236
Scalar target
Definition: ReservoirCoupling.hpp:240
std::size_t group_name_idx
Definition: ReservoirCoupling.hpp:239
Group::InjectionCMode cmode
Definition: ReservoirCoupling.hpp:241
Phase phase
Definition: ReservoirCoupling.hpp:242
Definition: ReservoirCoupling.hpp:177
Scalar operator[](Phase p) const noexcept
Definition: ReservoirCoupling.hpp:182
std::array< Scalar, static_cast< std::size_t >(Phase::Count)> rate
Definition: ReservoirCoupling.hpp:180
Scalar & operator[](Phase p) noexcept
Definition: ReservoirCoupling.hpp:181
Definition: ReservoirCoupling.hpp:188
std::array< Scalar, static_cast< std::size_t >(Phase::Count)> rate
Definition: ReservoirCoupling.hpp:189
Scalar & operator[](Phase p) noexcept
Definition: ReservoirCoupling.hpp:191
Scalar operator[](Phase p) const noexcept
Definition: ReservoirCoupling.hpp:192
Definition: ReservoirCoupling.hpp:246
Scalar target
Definition: ReservoirCoupling.hpp:250
std::size_t group_name_idx
Definition: ReservoirCoupling.hpp:249
Group::ProductionCMode cmode
Definition: ReservoirCoupling.hpp:251
Definition: ReservoirCoupling.hpp:196
Scalar operator[](Phase p) const noexcept
Definition: ReservoirCoupling.hpp:207
ProductionRates(const GuideRate::RateVector &rate_vector)
Definition: ReservoirCoupling.hpp:199
Scalar & operator[](Phase p) noexcept
Definition: ReservoirCoupling.hpp:206
std::array< Scalar, static_cast< std::size_t >(Phase::Count)> rate
Definition: ReservoirCoupling.hpp:205
Utility class for comparing double values representing epoch dates or elapsed time.
Definition: ReservoirCoupling.hpp:287
static bool compare_gt_or_eq(double a, double b)
Determines if a is greater than b within the specified tolerance.
static bool compare_gt(double a, double b)
Determines if a is greater than b within the specified tolerance.
static bool compare_lt_or_eq(double a, double b)
Determines if a is less than or equal to b within the specified tolerance.
static bool compare_eq(double a, double b)
Determines if two double values are equal within a specified tolerance.
static constexpr double reltol
Relative tolerance used for comparisons.
Definition: ReservoirCoupling.hpp:292
static bool compare_lt(double a, double b)
Determines if a is less than b within the specified tolerance.
static constexpr double abstol
Absolute tolerance used for comparisons.
Definition: ReservoirCoupling.hpp:289
Definition: ReservoirCoupling.hpp:230
InjectionRates< Scalar > surface_rates
Definition: ReservoirCoupling.hpp:231
InjectionRates< Scalar > reservoir_rates
Definition: ReservoirCoupling.hpp:232
Definition: ReservoirCoupling.hpp:212
Scalar gas_reinjection_rate
Definition: ReservoirCoupling.hpp:225
Potentials< Scalar > potentials
Definition: ReservoirCoupling.hpp:214
ProductionRates< Scalar > network_surface_rates
Definition: ReservoirCoupling.hpp:220
ProductionRates< Scalar > reservoir_rates
Definition: ReservoirCoupling.hpp:223
Scalar voidage_rate
Definition: ReservoirCoupling.hpp:224
ProductionRates< Scalar > surface_rates
Definition: ReservoirCoupling.hpp:217