mirror of
https://github.com/OPM/opm-simulators.git
synced 2025-01-01 12:06:54 -06:00
Add MPI-Aware Accumulator for Inter-Region Flow Rates
This commit introduces a new helper class Opm::EclInterRegFlowMapSingleFIP that wraps an object of type Opm::data::InterRegFlowMap along with the MPI rank's notion of a FIP region array definition (e.g., the local FIPNUM array). The new single-FIP flow map is responsible for accumulating local contributions to the inter-region flows defined by that FIP array. In the case of connections between MPI ranks, the rank that owns the lowest region ID accumulates the associate flow rates. Add unit tests to exercise the new class, including simulating multiple MPI ranks that are communicated to a single I/O rank for summary output purposes.
This commit is contained in:
parent
b5ded12847
commit
27564f0610
@ -31,6 +31,7 @@ list (APPEND MAIN_SOURCE_FILES
|
||||
ebos/eclgenerictracermodel.cc
|
||||
ebos/eclgenericvanguard.cc
|
||||
ebos/eclgenericwriter.cc
|
||||
ebos/eclinterregflows.cc
|
||||
ebos/ecltransmissibility.cc
|
||||
opm/core/props/BlackoilPhases.cpp
|
||||
opm/core/props/phaseUsageFromDeck.cpp
|
||||
@ -140,6 +141,7 @@ list (APPEND TEST_SOURCE_FILES
|
||||
tests/test_convergencereport.cpp
|
||||
tests/test_deferredlogger.cpp
|
||||
tests/test_ecl_output.cc
|
||||
tests/test_eclinterregflows.cpp
|
||||
tests/test_equil.cc
|
||||
tests/test_flexiblesolver.cpp
|
||||
tests/test_glift1.cpp
|
||||
@ -229,6 +231,7 @@ list (APPEND TEST_DATA_FILES
|
||||
# originally generated with the command:
|
||||
# find opm -name '*.h*' -a ! -name '*-pch.hpp' -printf '\t%p\n' | sort
|
||||
list (APPEND PUBLIC_HEADER_FILES
|
||||
ebos/eclinterregflows.hh
|
||||
opm/simulators/flow/countGlobalCells.hpp
|
||||
opm/simulators/flow/BlackoilModelEbos.hpp
|
||||
opm/simulators/flow/BlackoilModelParametersEbos.hpp
|
||||
|
109
ebos/eclinterregflows.cc
Normal file
109
ebos/eclinterregflows.cc
Normal file
@ -0,0 +1,109 @@
|
||||
// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
||||
// vi: set et ts=4 sw=4 sts=4:
|
||||
/*
|
||||
Copyright 2022 Equinor AS
|
||||
|
||||
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 <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <ebos/eclinterregflows.hh>
|
||||
|
||||
#include <algorithm>
|
||||
#include <stdexcept>
|
||||
|
||||
Opm::EclInterRegFlowMapSingleFIP::
|
||||
EclInterRegFlowMapSingleFIP(const std::vector<int>& region)
|
||||
: region_(region.size(), 0)
|
||||
{
|
||||
if (! region.empty()) {
|
||||
this->maxLocalRegionID_= this->maxGlobalRegionID_ =
|
||||
*std::max_element(region.begin(), region.end());
|
||||
}
|
||||
|
||||
std::transform(region.begin(), region.end(),
|
||||
this->region_.begin(),
|
||||
[](const int regID) { return regID - 1; });
|
||||
}
|
||||
|
||||
void
|
||||
Opm::EclInterRegFlowMapSingleFIP::
|
||||
addConnection(const Cell& source,
|
||||
const Cell& destination,
|
||||
const data::InterRegFlowMap::FlowRates& rates)
|
||||
{
|
||||
if (this->isReadFromStream_) {
|
||||
throw std::logic_error {
|
||||
"Cannot add new connection to deserialised object"
|
||||
};
|
||||
}
|
||||
|
||||
if (! (source.isInterior || destination.isInterior)) {
|
||||
// Connection between two cells not on this process. Unlikely, but
|
||||
// nothing to do here.
|
||||
return;
|
||||
}
|
||||
|
||||
const auto r1 = this->region_[ source.activeIndex ];
|
||||
const auto r2 = this->region_[ destination.activeIndex ];
|
||||
|
||||
if (r1 == r2) {
|
||||
// Connection is internal to a region. Nothing to do.
|
||||
return;
|
||||
}
|
||||
|
||||
if ((source.isInterior && destination.isInterior) ||
|
||||
(source.isInterior && (r1 < r2)) ||
|
||||
(destination.isInterior && (r2 < r1)))
|
||||
{
|
||||
// Inter-region connection internal to an MPI rank or this rank owns
|
||||
// the flow rate across this connection.
|
||||
this->iregFlow_.addConnection(r1, r2, rates);
|
||||
}
|
||||
}
|
||||
|
||||
void Opm::EclInterRegFlowMapSingleFIP::compress()
|
||||
{
|
||||
this->iregFlow_.compress(this->maxGlobalRegionID_);
|
||||
}
|
||||
|
||||
void Opm::EclInterRegFlowMapSingleFIP::clear()
|
||||
{
|
||||
this->iregFlow_.clear();
|
||||
this->isReadFromStream_ = false;
|
||||
}
|
||||
|
||||
const Opm::data::InterRegFlowMap&
|
||||
Opm::EclInterRegFlowMapSingleFIP::getInterRegFlows() const
|
||||
{
|
||||
return this->iregFlow_;
|
||||
}
|
||||
|
||||
std::size_t Opm::EclInterRegFlowMapSingleFIP::getLocalMaxRegionID() const
|
||||
{
|
||||
return this->maxLocalRegionID_;
|
||||
}
|
||||
|
||||
bool
|
||||
Opm::EclInterRegFlowMapSingleFIP::
|
||||
assignGlobalMaxRegionID(const std::size_t regID)
|
||||
{
|
||||
if (regID < this->maxLocalRegionID_) {
|
||||
return false;
|
||||
}
|
||||
|
||||
this->maxGlobalRegionID_ = regID;
|
||||
return true;
|
||||
}
|
168
ebos/eclinterregflows.hh
Normal file
168
ebos/eclinterregflows.hh
Normal file
@ -0,0 +1,168 @@
|
||||
// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
||||
// vi: set et ts=4 sw=4 sts=4:
|
||||
/*
|
||||
Copyright 2022 Equinor AS
|
||||
|
||||
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 <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#ifndef ECL_INTERREG_FLOWS_MODULE_HH
|
||||
#define ECL_INTERREG_FLOWS_MODULE_HH
|
||||
|
||||
#include <opm/output/data/InterRegFlowMap.hpp>
|
||||
|
||||
#include <algorithm>
|
||||
#include <cstddef>
|
||||
#include <string>
|
||||
#include <utility>
|
||||
#include <vector>
|
||||
|
||||
/// \file
|
||||
///
|
||||
/// MPI-aware facility for converting collection of tuples of region ID
|
||||
/// pairs and associate flow rates into a sparse (CSR) adjacency matrix
|
||||
/// representation of a graph. Supports O(nnz) compression.
|
||||
|
||||
namespace Opm {
|
||||
|
||||
/// Form CSR adjacency matrix representation of inter-region flow rate
|
||||
/// graph provided as a list of connections between regions on local MPI
|
||||
/// rank. Pertains to a single FIP definition array (e.g., FIPNUM).
|
||||
class EclInterRegFlowMapSingleFIP
|
||||
{
|
||||
public:
|
||||
/// Minimal characteristics of a cell from a simulation grid.
|
||||
struct Cell {
|
||||
/// Cell's active index on local rank.
|
||||
int activeIndex{-1};
|
||||
|
||||
/// Whether or not cell is interior to local rank.
|
||||
bool isInterior{true};
|
||||
};
|
||||
|
||||
/// Constructor
|
||||
///
|
||||
/// \param[in] region Local rank's FIP region definition array.
|
||||
explicit EclInterRegFlowMapSingleFIP(const std::vector<int>& region);
|
||||
|
||||
/// Add flow rate connection between regions.
|
||||
///
|
||||
/// \param[in] source Cell from which the flow nominally originates.
|
||||
///
|
||||
/// \param[in] destination Cell into which flow nominally goes.
|
||||
///
|
||||
/// \param[in] rates Flow rates associated to single connection.
|
||||
///
|
||||
/// If both cells are in the same region, or if neither cell is
|
||||
/// interior to this MPI rank, then this function does nothing. If
|
||||
/// one cell is interior to this MPI rank and the other isn't, then
|
||||
/// this function will include the flow rate contribution if and
|
||||
/// only if the cell with the smallest associate region ID is
|
||||
/// interior to this MPI rank.
|
||||
void addConnection(const Cell& source,
|
||||
const Cell& destination,
|
||||
const data::InterRegFlowMap::FlowRates& rates);
|
||||
|
||||
/// Form CSR adjacency matrix representation of input graph from
|
||||
/// connections established in previous calls to addConnection().
|
||||
///
|
||||
/// Number of rows in the CSR representation is the maximum FIP
|
||||
/// region ID.
|
||||
void compress();
|
||||
|
||||
/// Clear all internal buffers, but preserve allocated capacity.
|
||||
void clear();
|
||||
|
||||
/// Get read-only access to the underlying CSR representation.
|
||||
///
|
||||
/// Mostly intended for summary output purposes.
|
||||
const data::InterRegFlowMap& getInterRegFlows() const;
|
||||
|
||||
/// Retrieve maximum FIP region ID on local MPI rank.
|
||||
std::size_t getLocalMaxRegionID() const;
|
||||
|
||||
/// Assign maximum FIP region ID across all MPI ranks.
|
||||
///
|
||||
/// Fails if global maximum is smaller than local maximum region ID.
|
||||
///
|
||||
/// \param[in] regID Global maximum FIP region ID for this FIP
|
||||
/// definition array across all MPI ranks.
|
||||
///
|
||||
/// \return Whether or not assignment succeeded.
|
||||
bool assignGlobalMaxRegionID(const std::size_t regID);
|
||||
|
||||
/// Serialise internal representation to MPI message buffer
|
||||
///
|
||||
/// \tparam MessageBufferType Linear MPI message buffer. API should
|
||||
/// be similar to Dune::MessageBufferIF
|
||||
///
|
||||
/// \param[in,out] buffer Linear MPI message buffer instance.
|
||||
/// Function appends a partially linearised representation of
|
||||
/// \code *this \endcode to the buffer contents.
|
||||
template <class MessageBufferType>
|
||||
void write(MessageBufferType& buffer) const
|
||||
{
|
||||
// Note: this->region_ is *intentionally* omitted here.
|
||||
buffer.write(this->maxGlobalRegionID_);
|
||||
this->iregFlow_.write(buffer);
|
||||
}
|
||||
|
||||
/// Reconstitute internal object representation from MPI message
|
||||
/// buffer
|
||||
///
|
||||
/// This object (\code *this \endcode) is not usable in subsequent
|
||||
/// calls to \code addConnection() \endcode following a call to
|
||||
/// member function \code read() \endcode.
|
||||
///
|
||||
/// \tparam MessageBufferType Linear MPI message buffer. API should
|
||||
/// be similar to Dune::MessageBufferIF
|
||||
///
|
||||
/// \param[in,out] buffer Linear MPI message buffer instance.
|
||||
/// Function reads a partially linearised representation of \code
|
||||
/// *this \endcode from the buffer contents and advances the
|
||||
/// buffer's read position.
|
||||
template <class MessageBufferType>
|
||||
void read(MessageBufferType& buffer)
|
||||
{
|
||||
auto otherMaxRegionID = 0 * this->maxGlobalRegionID_;
|
||||
buffer.read(otherMaxRegionID);
|
||||
this->maxGlobalRegionID_ = std::max(this->maxGlobalRegionID_, otherMaxRegionID);
|
||||
|
||||
this->iregFlow_.read(buffer);
|
||||
this->isReadFromStream_ = true;
|
||||
}
|
||||
|
||||
private:
|
||||
/// Zero-based FIP region IDs on local MPI rank.
|
||||
std::vector<int> region_{};
|
||||
|
||||
/// Maximum one-based FIP region ID on local MPI rank.
|
||||
std::size_t maxLocalRegionID_{0};
|
||||
|
||||
/// Maximum one-based FIP region ID for this FIP region definition
|
||||
/// array across all MPI ranks.
|
||||
std::size_t maxGlobalRegionID_{0};
|
||||
|
||||
/// Rank-local inter-regional flow map.
|
||||
data::InterRegFlowMap iregFlow_{};
|
||||
|
||||
/// Whether or not this object contains contributions deserialised
|
||||
/// from a stream. For error detection.
|
||||
bool isReadFromStream_{false};
|
||||
};
|
||||
} // namespace Opm
|
||||
|
||||
#endif // ECL_INTERREG_FLOWS_MODULE_HH
|
2359
tests/test_eclinterregflows.cpp
Normal file
2359
tests/test_eclinterregflows.cpp
Normal file
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue
Block a user