// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- // vi: set et ts=4 sw=4 sts=4: /* 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 2 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 . Consult the COPYING file in the top-level source directory of this module for the precise wording of the license and the list of copyright holders. */ #ifndef EWOMS_COLLECT_TO_IO_RANK_HH #define EWOMS_COLLECT_TO_IO_RANK_HH #include #include #include #include #include #include #include #include #include #include #include #include namespace Opm { template class CollectDataToIORank { public: typedef typename Vanguard::Grid Grid; typedef typename Grid::CollectiveCommunication CollectiveCommunication; // global id class GlobalCellIndex { int globalId_; int localIndex_; bool isInterior_; public: GlobalCellIndex() : globalId_(-1) , localIndex_(-1) , isInterior_(true) {} void setGhost() { isInterior_ = false; } void setId(int globalId) { globalId_ = globalId; } void setIndex(int localIndex) { localIndex_ = localIndex; } int localIndex () const { return localIndex_; } int id () const { return globalId_; } bool isInterior() const { return isInterior_; } }; typedef typename Dune::PersistentContainer GlobalIndexContainer; static const int dimension = Grid::dimension; typedef typename Grid::LeafGridView GridView; typedef GridView AllGridView; typedef Dune::Point2PointCommunicator P2PCommunicatorType; typedef typename P2PCommunicatorType::MessageBufferType MessageBufferType; typedef std::vector LocalIndexMapType; typedef std::vector IndexMapType; typedef std::vector IndexMapStorageType; class DistributeIndexMapping : public P2PCommunicatorType::DataHandleInterface { protected: const std::vector& distributedGlobalIndex_; IndexMapType& localIndexMap_; IndexMapStorageType& indexMaps_; std::map globalPosition_; std::set& recv_; std::vector& ranks_; public: DistributeIndexMapping(const std::vector& globalIndex, const std::vector& distributedGlobalIndex, IndexMapType& localIndexMap, IndexMapStorageType& indexMaps, std::vector& ranks, std::set& recv, bool isIORank) : distributedGlobalIndex_(distributedGlobalIndex) , localIndexMap_(localIndexMap) , indexMaps_(indexMaps) , globalPosition_() , recv_(recv) , ranks_(ranks) { size_t size = globalIndex.size(); // create mapping globalIndex --> localIndex if ( isIORank ) // ioRank for (size_t index = 0; index < size; ++index) globalPosition_.insert(std::make_pair(globalIndex[index], index)); // we need to create a mapping from local to global if (!indexMaps_.empty()) { if (isIORank) ranks_.resize(size, -1); // for the ioRank create a localIndex to index in global state map IndexMapType& indexMap = indexMaps_.back(); size_t localSize = localIndexMap_.size(); indexMap.resize(localSize); for (size_t i=0; isecond; // Using max should be backwards compatible ranks_[entry] = std::max(ranks_[entry], rank); } if (rankIt != recv_.end()) ++rankIt; } #ifndef NDEBUG for (const auto& rank: ranks_) assert(rank>=0); #endif } } void pack(int link, MessageBufferType& buffer) { // we should only get one link if (link != 0) throw std::logic_error("link in method pack is not 0 as execpted"); // pack all interior global cell id's int size = localIndexMap_.size(); buffer.write(size); for (int index = 0; index < size; ++index) { int globalIdx = distributedGlobalIndex_[localIndexMap_[index]]; buffer.write(globalIdx); } } void unpack(int link, MessageBufferType& buffer) { // get index map for current link IndexMapType& indexMap = indexMaps_[link]; assert(!globalPosition_.empty()); // unpack all interior global cell id's int numCells = 0; buffer.read(numCells); indexMap.resize(numCells); for (int index = 0; index < numCells; ++index) { buffer.read(indexMap[index]); } } }; /// \brief Communication handle to scatter the global index template class ElementIndexScatterHandle { public: ElementIndexScatterHandle(const EquilMapper& sendMapper, const Mapper& recvMapper, std::vector& elementIndices) : sendMapper_(sendMapper), recvMapper_(recvMapper), elementIndices_(elementIndices) {} using DataType = int; bool fixedsize(int /*dim*/, int /*codim*/) { return true; } template std::size_t size(const T&) { return 1; } template void gather(B& buffer, const T& t) { buffer.write(sendMapper_.index(t)); } template void scatter(B& buffer, const T& t, std::size_t) { buffer.read(elementIndices_[recvMapper_.index(t)]); } bool contains(int dim, int codim) { return dim==3 && codim==0; } private: const EquilMapper& sendMapper_; const Mapper& recvMapper_; std::vector& elementIndices_; }; /// \brief Communication handle to scatter the global index template class ElementIndexHandle { public: ElementIndexHandle(const Mapper& mapper, std::vector& elementIndices) : mapper_(mapper), elementIndices_(elementIndices) {} using DataType = int; bool fixedsize(int /*dim*/, int /*codim*/) { return true; } template std::size_t size(const T&) { return 1; } template void gather(B& buffer, const T& t) { buffer.write(elementIndices_[mapper_.index(t)]); } template void scatter(B& buffer, const T& t, std::size_t) { buffer.read(elementIndices_[mapper_.index(t)]); } bool contains(int dim, int codim) { return dim==3 && codim==0; } private: const Mapper& mapper_; std::vector& elementIndices_; }; enum { ioRank = 0 }; static const bool needsReordering = !std::is_same::value; CollectDataToIORank(const Vanguard& vanguard) : toIORankComm_() { // index maps only have to be build when reordering is needed if (!needsReordering && !isParallel()) return; const CollectiveCommunication& comm = vanguard.grid().comm(); { std::set send, recv; typedef typename Vanguard::EquilGrid::LeafGridView EquilGridView; typedef typename Vanguard::GridView LocalGridView; const LocalGridView localGridView = vanguard.gridView(); typedef Dune::MultipleCodimMultipleGeomTypeMapper ElementMapper; ElementMapper elemMapper(localGridView, Dune::mcmgElementLayout()); const auto& cartMapper = vanguard.cartesianIndexMapper(); sortedCartesianIdx_.reserve(vanguard.gridView().size(0)); for(const auto& elem: elements(localGridView)) { auto idx = elemMapper.index(elem); sortedCartesianIdx_.push_back(cartMapper.cartesianIndex(idx)); } std::sort(sortedCartesianIdx_.begin(), sortedCartesianIdx_.end()); localIdxToGlobalIdx_.resize(localGridView.size(0), -1); // the I/O rank receives from all other ranks if (isIORank()) { // We need a mapping from local to global grid, here we // use equilGrid which represents a view on the global grid // reserve memory const size_t globalSize = vanguard.equilGrid().leafGridView().size(0); globalCartesianIndex_.resize(globalSize, -1); const EquilGridView equilGridView = vanguard.equilGrid().leafGridView(); typedef Dune::MultipleCodimMultipleGeomTypeMapper EquilElementMapper; EquilElementMapper equilElemMapper(equilGridView, Dune::mcmgElementLayout()); // Scatter the global index to local index for lookup during restart ElementIndexScatterHandle handle(equilElemMapper, elemMapper, localIdxToGlobalIdx_); vanguard.grid().scatterData(handle); // loop over all elements (global grid) and store Cartesian index auto elemIt = vanguard.equilGrid().leafGridView().template begin<0>(); const auto& elemEndIt = vanguard.equilGrid().leafGridView().template end<0>(); for (; elemIt != elemEndIt; ++elemIt) { int elemIdx = equilElemMapper.index(*elemIt); int cartElemIdx = vanguard.equilCartesianIndexMapper().cartesianIndex(elemIdx); globalCartesianIndex_[elemIdx] = cartElemIdx; } for (int i = 0; i < comm.size(); ++i) { if (i != ioRank) recv.insert(i); } } else { // all other simply send to the I/O rank send.insert(ioRank); // Scatter the global index to local index for lookup during restart // This is a bit hacky since the type differs from the iorank. // But should work since we only receive, i.e. use the second parameter. ElementIndexScatterHandle handle(elemMapper, elemMapper, localIdxToGlobalIdx_); vanguard.grid().scatterData(handle); } // Sync the global element indices ElementIndexHandle handle(elemMapper, localIdxToGlobalIdx_); vanguard.grid().communicate(handle, Dune::InteriorBorder_All_Interface, Dune::ForwardCommunication); localIndexMap_.clear(); const size_t gridSize = vanguard.grid().size(0); localIndexMap_.reserve(gridSize); // store the local Cartesian index IndexMapType distributedCartesianIndex; distributedCartesianIndex.resize(gridSize, -1); // A mapping for the whole grid (including the ghosts) is needed for restarts auto eIt = localGridView.template begin<0>(); const auto& eEndIt = localGridView.template end<0>(); for (; eIt != eEndIt; ++eIt) { const auto element = *eIt; int elemIdx = elemMapper.index(element); distributedCartesianIndex[elemIdx] = vanguard.cartesianIndex(elemIdx); // only store interior element for collection //assert(element.partitionType() == Dune::InteriorEntity); localIndexMap_.push_back(elemIdx); } // insert send and recv linkage to communicator toIORankComm_.insertRequest(send, recv); // need an index map for each rank indexMaps_.clear(); indexMaps_.resize(comm.size()); // distribute global id's to io rank for later association of dof's DistributeIndexMapping distIndexMapping(globalCartesianIndex_, distributedCartesianIndex, localIndexMap_, indexMaps_, globalRanks_, recv, isIORank()); toIORankComm_.exchange(distIndexMapping); } } class PackUnPackCellData : public P2PCommunicatorType::DataHandleInterface { const Opm::data::Solution& localCellData_; Opm::data::Solution& globalCellData_; const IndexMapType& localIndexMap_; const IndexMapStorageType& indexMaps_; public: PackUnPackCellData(const Opm::data::Solution& localCellData, Opm::data::Solution& globalCellData, const IndexMapType& localIndexMap, const IndexMapStorageType& indexMaps, size_t globalSize, bool isIORank) : localCellData_(localCellData) , globalCellData_(globalCellData) , localIndexMap_(localIndexMap) , indexMaps_(indexMaps) { if (isIORank) { // add missing data to global cell data for (const auto& pair : localCellData_) { const std::string& key = pair.first; std::size_t containerSize = globalSize; auto OPM_OPTIM_UNUSED ret = globalCellData_.insert(key, pair.second.dim, std::vector(containerSize), pair.second.target); assert(ret.second); } MessageBufferType buffer; pack(0, buffer); // the last index map is the local one doUnpack(indexMaps.back(), buffer); } } // pack all data associated with link void pack(int link, MessageBufferType& buffer) { // we should only get one link if (link != 0) throw std::logic_error("link in method pack is not 0 as expected"); // write all cell data registered in local state for (const auto& pair : localCellData_) { const auto& data = pair.second.data; // write all data from local data to buffer write(buffer, localIndexMap_, data); } } void doUnpack(const IndexMapType& indexMap, MessageBufferType& buffer) { // we loop over the data as // its order governs the order the data got received. for (auto& pair : localCellData_) { const std::string& key = pair.first; auto& data = globalCellData_.data(key); //write all data from local cell data to buffer read(buffer, indexMap, data); } } // unpack all data associated with link void unpack(int link, MessageBufferType& buffer) { doUnpack(indexMaps_[link], buffer); } protected: template void write(MessageBufferType& buffer, const IndexMapType& localIndexMap, const Vector& vector, unsigned int offset = 0, unsigned int stride = 1) const { unsigned int size = localIndexMap.size(); buffer.write(size); assert(vector.size() >= stride * size); for (unsigned int i=0; i void read(MessageBufferType& buffer, const IndexMapType& indexMap, Vector& vector, unsigned int offset = 0, unsigned int stride = 1) const { unsigned int size = 0; buffer.read(size); assert(size == indexMap.size()); for (unsigned int i=0; ipack(0, buffer); // pass a dummy_link to satisfy virtual class const int dummyLink = -1; this->unpack(dummyLink, buffer); } // pack all data associated with link void pack(int link, MessageBufferType& buffer) { // we should only get one link if (link != 0) { throw std::logic_error { "link in method pack is not 0 as expected" }; } // write all group and network (node/branch) data this->localGroupAndNetworkData_.write(buffer); } // unpack all data associated with link void unpack(int /*link*/, MessageBufferType& buffer) { this->globalGroupAndNetworkData_.read(buffer); } }; class PackUnPackBlockData : public P2PCommunicatorType::DataHandleInterface { const std::map, double>& localBlockData_; std::map, double>& globalBlockValues_; public: PackUnPackBlockData(const std::map, double>& localBlockData, std::map, double>& globalBlockValues, bool isIORank) : localBlockData_(localBlockData) , globalBlockValues_(globalBlockValues) { if (isIORank) { MessageBufferType buffer; pack(0, buffer); // pass a dummyLink to satisfy virtual class int dummyLink = -1; unpack(dummyLink, buffer); } } // pack all data associated with link void pack(int link, MessageBufferType& buffer) { // we should only get one link if (link != 0) throw std::logic_error("link in method pack is not 0 as expected"); // write all block data unsigned int size = localBlockData_.size(); buffer.write(size); for (const auto& map : localBlockData_) { buffer.write(map.first.first); buffer.write(map.first.second); buffer.write(map.second); } } // unpack all data associated with link void unpack(int /*link*/, MessageBufferType& buffer) { // read all block data unsigned int size = 0; buffer.read(size); for (size_t i = 0; i < size; ++i) { std::string name; int idx; double data; buffer.read(name); buffer.read(idx); buffer.read(data); globalBlockValues_[std::make_pair(name, idx)] = data; } } }; class PackUnPackWBPData : public P2PCommunicatorType::DataHandleInterface { const std::map& localWBPData_; std::map& globalWBPValues_; public: PackUnPackWBPData(const std::map& localWBPData, std::map& globalWBPValues, bool isIORank) : localWBPData_(localWBPData) , globalWBPValues_(globalWBPValues) { if (isIORank) { MessageBufferType buffer; pack(0, buffer); // pass a dummyLink to satisfy virtual class int dummyLink = -1; unpack(dummyLink, buffer); } } // pack all data associated with link void pack(int link, MessageBufferType& buffer) { // we should only get one link if (link != 0) throw std::logic_error("link in method pack is not 0 as expected"); // write all block data unsigned int size = localWBPData_.size(); buffer.write(size); for (const auto& [global_index, wbp_value] : localWBPData_) { buffer.write(global_index); buffer.write(wbp_value); } } // unpack all data associated with link void unpack(int /*link*/, MessageBufferType& buffer) { // read all block data unsigned int size = 0; buffer.read(size); for (size_t i = 0; i < size; ++i) { std::size_t idx; double data; buffer.read(idx); buffer.read(data); globalWBPValues_[idx] = data; } } }; // gather solution to rank 0 for EclipseWriter void collect(const Opm::data::Solution& localCellData, const std::map, double>& localBlockData, const std::map& localWBPData, const Opm::data::Wells& localWellData, const Opm::data::GroupAndNetworkValues& localGroupAndNetworkData) { globalCellData_ = {}; globalBlockData_.clear(); globalWBPData_.clear(); globalWellData_.clear(); globalGroupAndNetworkData_.clear(); // index maps only have to be build when reordering is needed if(!needsReordering && !isParallel()) return; // this also linearises the local buffers on ioRank PackUnPackCellData packUnpackCellData { localCellData, this->globalCellData_, this->localIndexMap_, this->indexMaps_, this->numCells(), this->isIORank() }; if (! isParallel()) { // no need to collect anything. return; } PackUnPackWellData packUnpackWellData { localWellData, this->globalWellData_, this->isIORank() }; PackUnPackGroupAndNetworkValues packUnpackGroupAndNetworkData { localGroupAndNetworkData, this->globalGroupAndNetworkData_, this->isIORank() }; PackUnPackBlockData packUnpackBlockData { localBlockData, this->globalBlockData_, this->isIORank() }; PackUnPackWBPData packUnpackWBPData { localWBPData, this->globalWBPData_, this->isIORank() }; toIORankComm_.exchange(packUnpackCellData); toIORankComm_.exchange(packUnpackWellData); toIORankComm_.exchange(packUnpackGroupAndNetworkData); toIORankComm_.exchange(packUnpackBlockData); toIORankComm_.exchange(packUnpackWBPData); #ifndef NDEBUG // mkae sure every process is on the same page toIORankComm_.barrier(); #endif } const std::map& globalWBPData() const { return this->globalWBPData_; } const std::map, double>& globalBlockData() const { return globalBlockData_; } const Opm::data::Solution& globalCellData() const { return globalCellData_; } const Opm::data::Wells& globalWellData() const { return globalWellData_; } const Opm::data::GroupAndNetworkValues& globalGroupAndNetworkData() const { return globalGroupAndNetworkData_; } bool isIORank() const { return toIORankComm_.rank() == ioRank; } bool isParallel() const { return toIORankComm_.size() > 1; } int localIdxToGlobalIdx(unsigned localIdx) const { if (!isParallel()) return localIdx; if (localIdxToGlobalIdx_.empty()) throw std::logic_error("index map is not created on this rank"); if (localIdx > localIdxToGlobalIdx_.size()) throw std::logic_error("local index is outside map range"); return localIdxToGlobalIdx_[localIdx]; } size_t numCells () const { return globalCartesianIndex_.size(); } const std::vector& globalRanks() const { return globalRanks_; } bool isCartIdxOnThisRank(int cartIdx) const { if (!isParallel()) return true; assert(!needsReordering); auto candidate = std::lower_bound(sortedCartesianIdx_.begin(), sortedCartesianIdx_.end(), cartIdx); return (candidate != sortedCartesianIdx_.end() && *candidate == cartIdx); } protected: P2PCommunicatorType toIORankComm_; IndexMapType globalCartesianIndex_; IndexMapType localIndexMap_; IndexMapStorageType indexMaps_; std::vector globalRanks_; Opm::data::Solution globalCellData_; std::map, double> globalBlockData_; std::map globalWBPData_; Opm::data::Wells globalWellData_; Opm::data::GroupAndNetworkValues globalGroupAndNetworkData_; std::vector localIdxToGlobalIdx_; /// \brief sorted list of cartesian indices present- /// /// non-empty only when running in parallel std::vector sortedCartesianIdx_; }; } // end namespace Opm #endif