ReservoirCouplingSlave.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_SLAVE_HPP
21#define OPM_RESERVOIR_COUPLING_SLAVE_HPP
22
25#include <opm/input/eclipse/Schedule/Schedule.hpp>
28#include <opm/common/OpmLog/OpmLog.hpp>
29
30#include <mpi.h>
31
32#include <vector>
33
34namespace Opm {
35
36template <class Scalar>
37class ReservoirCouplingSlaveReportStep;
38
39template <class Scalar>
41public:
50
52 const Parallel::Communication &comm, const Schedule &schedule, const SimulatorTimer &timer
53 );
54 bool activated() const { return activated_; }
56 const Parallel::Communication& getComm() const { return comm_; }
57 MPI_Comm getMasterComm() const { return slave_master_comm_; }
58 const std::string& getSlaveName() const { return slave_name_; }
59 const std::map<std::string, std::string>& getSlaveToMasterGroupNameMap() const {
60 return slave_to_master_group_map_; }
63 bool hasMasterGroupNodePressure(const std::string& gname) const;
64
67 bool hasMasterInjectionTarget(const std::string& gname, const Phase phase) const;
68
71 bool hasMasterProductionLimits(const std::string& gname) const;
72
75 bool hasMasterProductionTarget(const std::string& gname) const;
86 bool isSlaveGroup(const std::string& group_name) const;
87
92
97
98 ReservoirCoupling::Logger& logger() { return this->logger_; }
99 ReservoirCoupling::Logger& logger() const { return this->logger_; }
102 Scalar masterGroupNodePressure(const std::string& gname) const;
103
106 const std::map<std::string, Scalar>& masterGroupNodePressures() const;
107
110 std::pair<Scalar, Group::InjectionCMode> masterInjectionTarget(
111 const std::string& gname, const Phase phase) const;
112
115 const MasterProductionLimits& masterProductionLimits(const std::string& gname) const;
116
119 std::pair<Scalar, Group::ProductionCMode> masterProductionTarget(const std::string& gname) const;
120 void maybeActivate(int report_step);
121 std::size_t numSlaveGroups() const { return this->slave_group_order_.size(); }
129 std::pair<std::size_t, std::size_t> receiveNumGroupConstraintsFromMaster() const;
132 void receiveInjectionGroupTargetsFromMaster(std::size_t num_targets);
133
136 void receiveMasterGroupNodePressuresFromMaster(std::size_t num_pressures);
137
141 std::pair<std::size_t, bool> receiveNumMasterGroupNodePressuresFromMaster();
142
145 void receiveProductionGroupConstraintsFromMaster(std::size_t num_targets);
146
148 void sendInjectionDataToMaster(const std::vector<SlaveGroupInjectionData> &injection_data) const;
150 void sendProductionDataToMaster(const std::vector<SlaveGroupProductionData> &production_data) const;
151 void setDeferredLogger(DeferredLogger *deferred_logger) {
152 this->logger_.setDeferredLogger(deferred_logger);
153 }
159
162 void markSlaveGroupsInSchedule(Schedule& schedule, int report_step_idx);
163
164 const std::string& slaveGroupIdxToGroupName(std::size_t group_idx) const {
165 return this->slave_group_order_.at(group_idx);
166 }
167 bool terminated() const { return this->terminated_; }
168
178
189
190private:
191 void checkGrupSlavGroupNames_();
192 std::pair<double, bool> getGrupSlavActivationDateAndCheckHistoryMatchingMode_() const;
193 bool historyMatchingMode_() const { return this->history_matching_mode_; }
194 std::size_t numMasterGroups_() const { return this->slave_to_master_group_map_.size(); }
195 void receiveMasterGroupNamesFromMasterProcess_();
196 void receiveSlaveNameFromMasterProcess_();
197 void saveMasterGroupNamesAsMapAndEstablishOrder_(const std::vector<char>& group_names);
198 void sendActivationDateToMasterProcess_();
199 void sendActivationHandshakeToMasterProcess_() const;
200 void sendSimulationStartDateToMasterProcess_() const;
201
202 const Parallel::Communication &comm_;
203 const Schedule& schedule_;
204 const SimulatorTimer &timer_;
205 // MPI parent communicator for a slave process
206 MPI_Comm slave_master_comm_{MPI_COMM_NULL};
207 std::map<std::string, std::string> slave_to_master_group_map_;
208 bool activated_{false};
209 // True if the slave was terminated by the master process
210 bool terminated_{false};
211 // True if no GRUPMAST keyword in the master schedule and no GRUPSLAV keyword in the slave schedule
212 bool history_matching_mode_{false};
213 std::string slave_name_; // This is the slave name as defined in the master process
214 mutable ReservoirCoupling::Logger logger_;
215 // Order of the slave groups. A mapping from slave group index to slave group name.
216 // The indices are determined by the order the master process sends us the group names, see
217 // receiveMasterGroupNamesFromMasterProcess_()
218 // Later, the master process will send us group name indices, and not the group names themselves,
219 // so we use this mapping to recover the slave group names from the indices.
220 std::map<std::size_t, std::string> slave_group_order_;
221 // Stores data that changes for a single report step or for timesteps within a report step.
222 std::unique_ptr<ReservoirCouplingSlaveReportStep<Scalar>> report_step_data_{nullptr};
223};
224
225} // namespace Opm
226
227#endif // OPM_RESERVOIR_COUPLING_SLAVE_HPP
Definition: DeferredLogger.hpp:57
Definition: ReservoirCoupling.hpp:42
void setDeferredLogger(DeferredLogger *deferred_logger)
Definition: ReservoirCoupling.hpp:54
void clearDeferredLogger()
Definition: ReservoirCoupling.hpp:47
Definition: ReservoirCouplingSlave.hpp:40
const std::map< std::string, std::string > & getSlaveToMasterGroupNameMap() const
Definition: ReservoirCouplingSlave.hpp:59
ReservoirCouplingSlave(const Parallel::Communication &comm, const Schedule &schedule, const SimulatorTimer &timer)
const std::string & getSlaveName() const
Definition: ReservoirCouplingSlave.hpp:58
void setFirstSubstepOfSyncTimestep(bool value)
bool lastReceivedMasterGroupNodePressuresIsFinal() const
Whether the most recent master-group-node-pressures receive carried the is_final flag.
Scalar masterGroupNodePressure(const std::string &gname) const
Get the master-computed network-leaf node pressure for a master group.
void sendProductionDataToMaster(const std::vector< SlaveGroupProductionData > &production_data) const
std::size_t numSlaveGroups() const
Definition: ReservoirCouplingSlave.hpp:121
bool isFirstSubstepOfSyncTimestep() const
std::pair< Scalar, Group::ProductionCMode > masterProductionTarget(const std::string &gname) const
Get the master-imposed production target and control mode for a group.
const Parallel::Communication & getComm() const
Definition: ReservoirCouplingSlave.hpp:56
bool isSlaveGroup(const std::string &group_name) const
std::pair< std::size_t, bool > receiveNumMasterGroupNodePressuresFromMaster()
Receive the master-group-node-pressures header from master.
void receiveCoupledNetworkActiveStatusFromMaster()
Receive the master's single-bool flag for the current sync timestep and mirror it into the slave's lo...
std::pair< std::size_t, std::size_t > receiveNumGroupConstraintsFromMaster() const
const MasterProductionLimits & masterProductionLimits(const std::string &gname) const
Get the master-imposed per-rate-type production limits for a group.
void receiveInjectionGroupTargetsFromMaster(std::size_t num_targets)
Receive injection group targets from master and store them locally.
void receiveProductionGroupConstraintsFromMaster(std::size_t num_targets)
Receive production group constraints from master and store them locally.
bool connectedToMasterCoupledNetwork() const
Whether this slave is connected to the master's cross-rescoup network this sync step.
bool isLastSubstepOfSyncTimestep() const
Check if this is the last substep within a "sync" timestep.
void setDeferredLogger(DeferredLogger *deferred_logger)
Definition: ReservoirCouplingSlave.hpp:151
MPI_Comm getMasterComm() const
Definition: ReservoirCouplingSlave.hpp:57
void clearDeferredLogger()
Definition: ReservoirCouplingSlave.hpp:55
bool hasMasterInjectionTarget(const std::string &gname, const Phase phase) const
Check if a master-imposed injection target exists for a group and phase.
const std::map< std::string, Scalar > & masterGroupNodePressures() const
Get the full map of master-computed network-leaf node pressures.
void sendNextReportDateToMasterProcess() const
std::pair< Scalar, Group::InjectionCMode > masterInjectionTarget(const std::string &gname, const Phase phase) const
Get the master-imposed injection target and control mode for a group and phase.
ReservoirCoupling::Logger & logger()
Definition: ReservoirCouplingSlave.hpp:98
bool maybeReceiveTerminateSignalFromMaster()
Blocking receive for terminate/continue signal from master.
bool hasMasterGroupNodePressure(const std::string &gname) const
Check if a master-computed network-leaf node pressure exists for a master group.
const std::string & slaveGroupIdxToGroupName(std::size_t group_idx) const
Definition: ReservoirCouplingSlave.hpp:164
bool terminated() const
Definition: ReservoirCouplingSlave.hpp:167
void sendInjectionDataToMaster(const std::vector< SlaveGroupInjectionData > &injection_data) const
bool hasMasterProductionTarget(const std::string &gname) const
Check if a master-imposed production target exists for a group.
bool activated() const
Definition: ReservoirCouplingSlave.hpp:54
ReservoirCoupling::Logger & logger() const
Definition: ReservoirCouplingSlave.hpp:99
void setLastSubstepOfSyncTimestep(bool value)
Set whether this is the last substep within a "sync" timestep.
void maybeActivate(int report_step)
bool hasMasterProductionLimits(const std::string &gname) const
Check if master-imposed per-rate-type production limits exist for a group.
void receiveTerminateAndDisconnect()
Receive terminate signal from master and disconnect the intercommunicator.
void receiveMasterGroupNodePressuresFromMaster(std::size_t num_pressures)
Receive master-computed network-leaf node pressures from master.
void markSlaveGroupsInSchedule(Schedule &schedule, int report_step_idx)
Mark slave groups in the Schedule as production/injection groups.
typename ReservoirCoupling::MasterProductionLimits< Scalar > MasterProductionLimits
Definition: ReservoirCouplingSlave.hpp:48
Definition: SimulatorTimer.hpp:39
Dune::Communication< MPIComm > Communication
Definition: ParallelCommunication.hpp:30
MessageTag
Definition: ReservoirCoupling.hpp:134
Phase
Phase indices for reservoir coupling, we currently only support black-oil phases (oil,...
Definition: ReservoirCoupling.hpp:159
Definition: blackoilbioeffectsmodules.hh:45
Definition: ReservoirCoupling.hpp:238
Master-computed network-leaf node pressure for a single master group.
Definition: ReservoirCoupling.hpp:281
Per-rate-type production limits received from master hierarchy. A value of -1 means no limit defined ...
Definition: ReservoirCoupling.hpp:266
Definition: ReservoirCoupling.hpp:190
Definition: ReservoirCoupling.hpp:248
Definition: ReservoirCoupling.hpp:232
Definition: ReservoirCoupling.hpp:214