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:
Bård Skaflestad 2022-02-01 18:03:03 +01:00
parent b5ded12847
commit 27564f0610
4 changed files with 2639 additions and 0 deletions

View File

@ -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
View 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
View 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

File diff suppressed because it is too large Load Diff