/* Copyright 2017 Dr. Blatt - HPC-Simulation-Software & Services 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 . */ #ifndef OPM_AMG_HEADER_INCLUDED #define OPM_AMG_HEADER_INCLUDED #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Dune { namespace Amg { template class UnSymmetricCriterion; } } namespace Opm { namespace Amg { template class Element { public: enum { /* @brief We preserve the sign.*/ is_sign_preserving = true }; template typename M::field_type operator()(const M& m) const { return m[Row][Column]; } }; } // namespace Amg } // namespace Opm namespace Dune { template class MatrixBlock; } namespace Opm { namespace Detail { /** * \brief Creates a MatrixAdapter as an operator * * The first argument is used to specify the return type using function overloading. * \param matrix The matrix to wrap. */ template Dune::MatrixAdapter createOperator(const Dune::MatrixAdapter&, const M& matrix, const T&) { return Dune::MatrixAdapter(matrix); } /** * \brief Creates a MatrixAdapter as an operator, storing it in a unique_ptr. * * The first argument is used to specify the return type using function overloading. * \param matrix The matrix to wrap. */ template std::unique_ptr< Dune::MatrixAdapter > createOperatorPtr(const Dune::MatrixAdapter&, const M& matrix, const T&) { return std::make_unique< Dune::MatrixAdapter >(matrix); } /** * \brief Creates an OverlappingSchwarzOperator as an operator. * * The first argument is used to specify the return type using function overloading. * \param matrix The matrix to wrap. * \param comm The object encapsulating the parallelization information. */ template Dune::OverlappingSchwarzOperator createOperator(const Dune::OverlappingSchwarzOperator&, const M& matrix, const T& comm) { return Dune::OverlappingSchwarzOperator(matrix, comm); } //! \brief Applies diagonal scaling to the discretization Matrix (Scheichl, 2003) //! //! See section 3.2.3 of Scheichl, Masson: Decoupling and Block Preconditioning for //! Sedimentary Basin Simulations, 2003. //! \param op The operator that stems from the discretization. //! \param comm The communication objecte describing the data distribution. //! \param pressureEqnIndex The index of the pressure in the matrix block //! \retun A pair of the scaled matrix and the associated operator- template std::tuple, Operator> scaleMatrixDRS(const Operator& op, const Communication& comm, std::size_t pressureEqnIndex, const Vector& weights, const Opm::CPRParameter& param) { using Matrix = typename Operator::matrix_type; using Block = typename Matrix::block_type; using BlockVector = typename Vector::block_type; std::unique_ptr matrix(new Matrix(op.getmat())); if (param.cpr_use_drs_) { const auto endi = matrix->end(); for (auto i = matrix->begin(); i != endi; ++i) { const BlockVector& bw = weights[i.index()]; const auto endj = (*i).end(); for (auto j = (*i).begin(); j != endj; ++j) { Block& block = *j; BlockVector& bvec = block[pressureEqnIndex]; // should introduce limits which also change the weights block.mtv(bw, bvec); } } } return std::make_tuple(std::move(matrix), createOperator(op, *matrix, comm)); } //! \brief Applies diagonal scaling to the discretization Matrix (Scheichl, 2003) //! //! See section 3.2.3 of Scheichl, Masson: Decoupling and Block Preconditioning for //! Sedimentary Basin Simulations, 2003. //! \param vector The vector to scale //! \param pressureEqnIndex The index of the pressure in the matrix block template void scaleVectorDRS(Vector& vector, std::size_t pressureEqnIndex, const Opm::CPRParameter& param, const Vector& weights) { using Block = typename Vector::block_type; if (param.cpr_use_drs_) { for (std::size_t j = 0; j < vector.size(); ++j) { Block& block = vector[j]; const Block& bw = weights[j]; block[pressureEqnIndex] = bw.dot(block); } } } //! \brief TMP to create the scalar pendant to a real block matrix, vector, smoother, etc. //! //! \code //! using Scalar = ScalarType::value; //! \endcode //! will get the corresponding scalar type. template struct ScalarType { }; template struct ScalarType > { typedef Dune::FieldVector value; }; template struct ScalarType > { typedef Dune::FieldMatrix value; }; template struct ScalarType > { typedef Dune::MatrixBlock value; }; template struct ScalarType > { typedef Ewoms::MatrixBlock value; }; template struct ScalarType > { using ScalarBlock = typename ScalarType::value; using ScalarAllocator = typename Allocator::template rebind::other; typedef Dune::BCRSMatrix value; }; template struct ScalarType > { using ScalarBlock = typename ScalarType::value; using ScalarAllocator = typename Allocator::template rebind::other; typedef Dune::BlockVector value; }; template struct ScalarType > { typedef Dune::SeqScalarProduct::value> value; }; #define ComposeScalarTypeForSeqPrecond(PREC) \ template \ struct ScalarType > \ { \ typedef PREC::value, \ typename ScalarType::value, \ typename ScalarType::value, \ l> value; \ } ComposeScalarTypeForSeqPrecond(Dune::SeqJac); ComposeScalarTypeForSeqPrecond(Dune::SeqSOR); ComposeScalarTypeForSeqPrecond(Dune::SeqSSOR); ComposeScalarTypeForSeqPrecond(Dune::SeqGS); ComposeScalarTypeForSeqPrecond(Dune::SeqILU0); ComposeScalarTypeForSeqPrecond(Dune::SeqILUn); #undef ComposeScalarTypeForSeqPrecond template struct ScalarType > { typedef Dune::Richardson::value, typename ScalarType::value> value; }; template struct ScalarType > { typedef Dune::OverlappingSchwarzOperator::value, typename ScalarType::value, typename ScalarType::value, C> value; }; template struct ScalarType > { typedef Dune::MatrixAdapter::value, typename ScalarType::value, typename ScalarType::value> value; }; template struct ScalarType > { typedef Dune::OverlappingSchwarzScalarProduct::value, C> value; }; template struct ScalarType > { typedef Dune::NonoverlappingSchwarzScalarProduct::value, C> value; }; template struct ScalarType > { typedef Dune::BlockPreconditioner::value, typename ScalarType::value, C, typename ScalarType::value> value; }; template struct ScalarType > { typedef ParallelOverlappingILU0::value, typename ScalarType::value, typename ScalarType::value, C> value; }; template struct ScalarType,N> > > { using value = Dune::Amg::CoarsenCriterion::value>, Dune::Amg::FirstDiagonal> >; }; template struct ScalarType,N> > > { using value = Dune::Amg::CoarsenCriterion::value>, Dune::Amg::FirstDiagonal> >; }; template struct OneComponentCriterionType {}; template struct OneComponentCriterionType,N> >, COMPONENT_INDEX, VARIABLE_INDEX> { using value = Dune::Amg::CoarsenCriterion, Opm::Amg::Element > >; }; template struct OneComponentCriterionType,N> >, COMPONENT_INDEX, VARIABLE_INDEX> { using value = Dune::Amg::CoarsenCriterion, Opm::Amg::Element > >; }; template class OneComponentAggregationLevelTransferPolicy; /** * @brief A policy class for solving the coarse level system using one step of AMG. * @tparam O The type of the linear operator used. * @tparam S The type of the smoother used in AMG. * @tparam C The type of the crition used for the aggregation within AMG. * @tparam C1 The type of the information about the communication. Either * Dune::OwnerOverlapCopyCommunication or Dune::SequentialInformation. */ template class OneStepAMGCoarseSolverPolicy { public: typedef P LevelTransferPolicy; /** @brief The type of the linear operator used. */ typedef O Operator; /** @brief The type of the range and domain of the operator. */ typedef typename O::range_type X; /** @brief The type of the crition used for the aggregation within AMG.*/ typedef C Criterion; /** @brief The type of the communication used for AMG.*/ typedef typename P::ParallelInformation Communication; /** @brief The type of the smoother used in AMG. */ typedef S Smoother; /** @brief The type of the arguments used for constructing the smoother. */ typedef typename Dune::Amg::SmootherTraits::Arguments SmootherArgs; /** @brief The type of the AMG construct on the coarse level.*/ typedef Dune::Amg::AMG AMGType; /** * @brief Constructs the coarse solver policy. * @param args The arguments used for constructing the smoother. * @param c The crition used for the aggregation within AMG. */ OneStepAMGCoarseSolverPolicy(const CPRParameter* param, const SmootherArgs& args, const Criterion& c) : param_(param), smootherArgs_(args), criterion_(c) {} /** @brief Copy constructor. */ OneStepAMGCoarseSolverPolicy(const OneStepAMGCoarseSolverPolicy& other) : param_(other.param_), coarseOperator_(other.coarseOperator_), smootherArgs_(other.smootherArgs_), criterion_(other.criterion_) {} private: /** * @brief A wrapper that makes an inverse operator out of AMG. * * The operator will use one step of AMG to approximately solve * the coarse level system. */ struct AMGInverseOperator : public Dune::InverseOperator { AMGInverseOperator(const CPRParameter* param, const typename AMGType::Operator& op, const Criterion& crit, const typename AMGType::SmootherArgs& args, const Communication& comm) : param_(param), amg_(), smoother_(), op_(op), comm_(comm) { if ( param_->cpr_use_amg_ ) { amg_.reset(new AMGType(op, crit,args, comm)); } else { typename Dune::Amg::ConstructionTraits::Arguments cargs; cargs.setMatrix(op.getmat()); cargs.setComm(comm); cargs.setArgs(args); smoother_.reset(Dune::Amg::ConstructionTraits::construct(cargs)); } } #if DUNE_VERSION_NEWER(DUNE_ISTL, 2, 6) Dune::SolverCategory::Category category() const override { return std::is_same::value ? Dune::SolverCategory::sequential : Dune::SolverCategory::overlapping; } #endif void apply(X& x, X& b, double reduction, Dune::InverseOperatorResult& res) { DUNE_UNUSED_PARAMETER(reduction); DUNE_UNUSED_PARAMETER(res); #if DUNE_VERSION_NEWER(DUNE_ISTL, 2, 6) auto sp = Dune::createScalarProduct(comm_, op_.category()); #else using Chooser = Dune::ScalarProductChooser; auto sp = Chooser::construct(comm_); #endif Dune::Preconditioner* prec = amg_.get(); if ( ! amg_ ) { prec = smoother_.get(); } // Linear solver parameters const double tolerance = param_->cpr_solver_tol_; const int maxit = param_->cpr_max_ell_iter_; int verbosity = 0; if (comm_.communicator().rank() == 0) { verbosity = param_->cpr_solver_verbose_; } if ( param_->cpr_ell_solvetype_ == 0) { #if DUNE_VERSION_NEWER(DUNE_ISTL, 2, 6) Dune::BiCGSTABSolver solver(const_cast(op_), *sp, *prec, tolerance, maxit, verbosity); solver.apply(x,b,res); #else // Category of preconditioner will be checked at compile time. Therefore we need // to cast to the derived class if ( !amg_ ) { Dune::BiCGSTABSolver solver(const_cast(op_), *sp, reinterpret_cast(*prec), tolerance, maxit, verbosity); solver.apply(x,b,res); } else { Dune::BiCGSTABSolver solver(const_cast(op_), *sp, reinterpret_cast(*prec), tolerance, maxit, verbosity); solver.apply(x,b,res); } #endif } else if (param_->cpr_ell_solvetype_ == 1) { #if DUNE_VERSION_NEWER(DUNE_ISTL, 2, 6) Dune::CGSolver solver(const_cast(op_), *sp, *prec, tolerance, maxit, verbosity); solver.apply(x,b,res); #else // Category of preconditioner will be checked at compile time. Therefore we need // to cast to the derived class if ( !amg_ ) { Dune::CGSolver solver(const_cast(op_), *sp, reinterpret_cast(*prec), tolerance, maxit, verbosity); solver.apply(x,b,res); } else { Dune::CGSolver solver(const_cast(op_), *sp, reinterpret_cast(*prec), tolerance, maxit, verbosity); solver.apply(x,b,res); } #endif } else { #if DUNE_VERSION_NEWER(DUNE_ISTL, 2, 6) Dune::LoopSolver solver(const_cast(op_), *sp, *prec, tolerance, maxit, verbosity); solver.apply(x,b,res); #else if ( !amg_ ) { Dune::LoopSolver solver(const_cast(op_), *sp, reinterpret_cast(*prec), tolerance, maxit, verbosity); solver.apply(x,b,res); } else { Dune::LoopSolver solver(const_cast(op_), *sp, reinterpret_cast(*prec), tolerance, maxit, verbosity); solver.apply(x,b,res); } #endif } // Warn if unknown options. if (param_->cpr_ell_solvetype_ > 2 && comm_.communicator().rank() == 0) { OpmLog::warning("cpr_ell_solver_type_unknown", "Unknown CPR elliptic solver type specification, using LoopSolver."); } #if ! DUNE_VERSION_NEWER(DUNE_ISTL, 2, 6) delete sp; #endif } void apply(X& x, X& b, Dune::InverseOperatorResult& res) { return apply(x,b,1e-8,res); } ~AMGInverseOperator() {} AMGInverseOperator(const AMGInverseOperator& other) : x_(other.x_), amg_(other.amg_) { } private: const CPRParameter* param_; X x_; std::unique_ptr amg_; std::unique_ptr smoother_; const typename AMGType::Operator& op_; const Communication& comm_; }; public: /** @brief The type of solver constructed for the coarse level. */ typedef AMGInverseOperator CoarseLevelSolver; /** * @brief Constructs a coarse level solver. * * @param transferPolicy The policy describing the transfer between levels. * @return A pointer to the constructed coarse level solver. */ template CoarseLevelSolver* createCoarseLevelSolver(LTP& transferPolicy) { coarseOperator_=transferPolicy.getCoarseLevelOperator(); const LevelTransferPolicy& transfer = reinterpret_cast(transferPolicy); AMGInverseOperator* inv = new AMGInverseOperator(param_, *coarseOperator_, criterion_, smootherArgs_, transfer.getCoarseLevelCommunication()); return inv; //std::shared_ptr >(inv); } private: /** @brief The coarse level operator. */ std::shared_ptr coarseOperator_; /** @brief The parameters for the CPR preconditioner. */ const CPRParameter* param_; /** @brief The arguments used to construct the smoother. */ SmootherArgs smootherArgs_; /** @brief The coarsening criterion. */ Criterion criterion_; }; template Smoother* constructSmoother(const Operator& op, const typename Dune::Amg::SmootherTraits::Arguments& smargs, const Communication& comm) { typename Dune::Amg::ConstructionTraits::Arguments args; args.setMatrix(op.getmat()); args.setComm(comm); args.setArgs(smargs); return Dune::Amg::ConstructionTraits::construct(args); } template const Dune::Amg::OverlapVertex* buildOverlapVertices(const G& graph, const C& pinfo, Dune::Amg::AggregatesMap& aggregates, const S& overlap, std::size_t& overlapCount) { // count the overlap vertices. overlapCount = 0; const auto& lookup=pinfo.globalLookup(); for ( const auto& vertex: graph ) { const auto* pair = lookup.pair(vertex); if(pair!=0 && overlap.contains(pair->local().attribute())) ++overlapCount; } // Allocate space using Vertex = typename G::VertexDescriptor; using OverlapVertex = Dune::Amg::OverlapVertex; auto* overlapVertices = new OverlapVertex[overlapCount==0 ? 1 : overlapCount]; if(overlapCount==0) return overlapVertices; // Initialize them overlapCount=0; for ( const auto& vertex: graph ) { const auto* pair = lookup.pair(vertex); if(pair!=0 && overlap.contains(pair->local().attribute())) { overlapVertices[overlapCount].aggregate = &aggregates[pair->local()]; overlapVertices[overlapCount].vertex = pair->local(); ++overlapCount; } } std::sort(overlapVertices, overlapVertices+overlapCount, [](const OverlapVertex& v1, const OverlapVertex& v2) { return *v1.aggregate < *v2.aggregate; }); // due to the sorting the isolated aggregates (to be skipped) are at the end. return overlapVertices; } template void buildCoarseSparseMatrix(M& coarseMatrix, G& fineGraph, const V& visitedMap, const C& pinfo, Dune::Amg::AggregatesMap& aggregates, const S& overlap) { using OverlapVertex = Dune::Amg ::OverlapVertex; std::size_t count; const OverlapVertex* overlapVertices = buildOverlapVertices(fineGraph, pinfo, aggregates, overlap, count); // Reset the visited flags of all vertices. // As the isolated nodes will be skipped we simply mark them as visited #ifndef NDEBUG const auto UNAGGREGATED = Dune::Amg::AggregatesMap::UNAGGREGATED; #endif const auto ISOLATED = Dune::Amg::AggregatesMap::ISOLATED; for ( const auto& vertex: fineGraph ) { assert(aggregates[vertex] != UNAGGREGATED); put(visitedMap, vertex, aggregates[vertex]==ISOLATED); } Dune::Amg::SparsityBuilder sparsityBuilder(coarseMatrix); Dune::Amg::ConnectivityConstructor::examine(fineGraph, visitedMap, pinfo, aggregates, overlap, overlapVertices, overlapVertices+count, sparsityBuilder); delete[] overlapVertices; } template void buildCoarseSparseMatrix(M& coarseMatrix, G& fineGraph, const V& visitedMap, const Dune::Amg::SequentialInformation& pinfo, Dune::Amg::AggregatesMap& aggregates, const S&) { // Reset the visited flags of all vertices. // As the isolated nodes will be skipped we simply mark them as visited #ifndef NDEBUG const auto UNAGGREGATED = Dune::Amg::AggregatesMap::UNAGGREGATED; #endif const auto ISOLATED = Dune::Amg::AggregatesMap::ISOLATED; for(const auto& vertex: fineGraph ) { assert(aggregates[vertex] != UNAGGREGATED); put(visitedMap, vertex, aggregates[vertex]==ISOLATED); } Dune::Amg::SparsityBuilder sparsityBuilder(coarseMatrix); Dune::Amg::ConnectivityConstructor ::examine(fineGraph, visitedMap, pinfo, aggregates, sparsityBuilder); } } // end namespace Detail /** * @brief A LevelTransferPolicy that uses aggregation to construct the coarse level system. * @tparam Operator The type of the fine level operator. * @tparam Criterion The criterion that describes the aggregation procedure. * @tparam Communication The class that describes the communication pattern. */ template class OneComponentAggregationLevelTransferPolicy : public Dune::Amg::LevelTransferPolicy::value> { typedef Dune::Amg::AggregatesMap AggregatesMap; public: using CoarseOperator = typename Detail::ScalarType::value; typedef Dune::Amg::LevelTransferPolicy FatherType; typedef Communication ParallelInformation; public: OneComponentAggregationLevelTransferPolicy(const Criterion& crit, const Communication& comm, bool cpr_pressure_aggregation) : criterion_(crit), communication_(&const_cast(comm)), cpr_pressure_aggregation_(cpr_pressure_aggregation) {} void createCoarseLevelSystem(const Operator& fineOperator) { prolongDamp_ = 1; if ( cpr_pressure_aggregation_ ) { #if DUNE_VERSION_NEWER(DUNE_ISTL, 2, 6) typedef Dune::Amg::PropertiesGraphCreator GraphCreator; #else typedef Dune::Amg::PropertiesGraphCreator GraphCreator; #endif typedef typename GraphCreator::PropertiesGraph PropertiesGraph; typedef typename GraphCreator::GraphTuple GraphTuple; typedef typename PropertiesGraph::VertexDescriptor Vertex; std::vector excluded(fineOperator.getmat().N(), false); using OverlapFlags = Dune::NegateSet; GraphTuple graphs = GraphCreator::create(fineOperator, excluded, *communication_, OverlapFlags()); aggregatesMap_.reset(new AggregatesMap(std::get<1>(graphs)->maxVertex()+1)); int noAggregates, isoAggregates, oneAggregates, skippedAggregates; using std::get; std::tie(noAggregates, isoAggregates, oneAggregates, skippedAggregates) = aggregatesMap_->buildAggregates(fineOperator.getmat(), *(get<1>(graphs)), criterion_, true); using CommunicationArgs = typename Dune::Amg::ConstructionTraits::Arguments; CommunicationArgs commArgs(communication_->communicator(), communication_->getSolverCategory()); coarseLevelCommunication_.reset(Dune::Amg::ConstructionTraits::construct(commArgs)); using Iterator = typename std::vector::iterator; using std::get; auto visitedMap = get(Dune::Amg::VertexVisitedTag(), *(get<1>(graphs))); communication_->buildGlobalLookup(fineOperator.getmat().N()); std::size_t aggregates = Dune::Amg::IndicesCoarsener ::coarsen(*communication_, *get<1>(graphs), visitedMap, *aggregatesMap_, *coarseLevelCommunication_, noAggregates); GraphCreator::free(graphs); coarseLevelCommunication_->buildGlobalLookup(aggregates); Dune::Amg::AggregatesPublisher ::publish(*aggregatesMap_, *communication_, coarseLevelCommunication_->globalLookup()); std::vector& visited=excluded; std::fill(visited.begin(), visited.end(), false); Dune::IteratorPropertyMap visitedMap2(visited.begin(), Dune::IdentityMap()); using CoarseMatrix = typename CoarseOperator::matrix_type; coarseLevelMatrix_.reset(new CoarseMatrix(aggregates, aggregates, CoarseMatrix::row_wise)); Detail::buildCoarseSparseMatrix(*coarseLevelMatrix_, *get<0>(graphs), visitedMap2, *communication_, *aggregatesMap_, OverlapFlags()); delete get<0>(graphs); communication_->freeGlobalLookup(); if( static_cast(this->coarseLevelMatrix_->N()) < criterion_.coarsenTarget()) { coarseLevelCommunication_->freeGlobalLookup(); } calculateCoarseEntries(fineOperator.getmat()); } else { using CoarseMatrix = typename CoarseOperator::matrix_type; const auto& fineLevelMatrix = fineOperator.getmat(); coarseLevelMatrix_.reset(new CoarseMatrix(fineLevelMatrix.N(), fineLevelMatrix.M(), CoarseMatrix::row_wise)); auto createIter = coarseLevelMatrix_->createbegin(); for ( const auto& row: fineLevelMatrix ) { for ( auto col = row.begin(), cend = row.end(); col != cend; ++col) { createIter.insert(col.index()); } ++createIter; } auto coarseRow = coarseLevelMatrix_->begin(); for ( const auto& row: fineLevelMatrix ) { auto coarseCol = coarseRow->begin(); for ( auto col = row.begin(), cend = row.end(); col != cend; ++col, ++coarseCol ) { assert( col.index() == coarseCol.index() ); *coarseCol = (*col)[COMPONENT_INDEX][VARIABLE_INDEX]; } ++coarseRow; } coarseLevelCommunication_.reset(communication_, [](Communication*){}); } this->lhs_.resize(this->coarseLevelMatrix_->M()); this->rhs_.resize(this->coarseLevelMatrix_->N()); using OperatorArgs = typename Dune::Amg::ConstructionTraits::Arguments; OperatorArgs oargs(*coarseLevelMatrix_, *coarseLevelCommunication_); this->operator_.reset(Dune::Amg::ConstructionTraits::construct(oargs)); } template void calculateCoarseEntries(const M& fineMatrix) { *coarseLevelMatrix_ = 0; for(auto row = fineMatrix.begin(), rowEnd = fineMatrix.end(); row != rowEnd; ++row) { const auto& i = (*aggregatesMap_)[row.index()]; if(i != AggregatesMap::ISOLATED) { for(auto entry = row->begin(), entryEnd = row->end(); entry != entryEnd; ++entry) { const auto& j = (*aggregatesMap_)[entry.index()]; if ( j != AggregatesMap::ISOLATED ) { (*coarseLevelMatrix_)[i][j] += (*entry)[COMPONENT_INDEX][VARIABLE_INDEX]; } } } } } void moveToCoarseLevel(const typename FatherType::FineRangeType& fine) { // Set coarse vector to zero this->rhs_=0; if ( cpr_pressure_aggregation_ ) { auto end = fine.end(), begin=fine.begin(); for(auto block=begin; block != end; ++block) { const auto& vertex = (*aggregatesMap_)[block-begin]; if(vertex != AggregatesMap::ISOLATED) { this->rhs_[vertex] += (*block)[COMPONENT_INDEX]; } } } else { auto end = fine.end(), begin=fine.begin(); for(auto block=begin; block != end; ++block) { this->rhs_[block-begin] = (*block)[COMPONENT_INDEX]; } } this->lhs_=0; } void moveToFineLevel(typename FatherType::FineDomainType& fine) { if( cpr_pressure_aggregation_ ) { this->lhs_ *= prolongDamp_; auto end=fine.end(), begin=fine.begin(); for(auto block=begin; block != end; ++block) { const auto& vertex = (*aggregatesMap_)[block-begin]; if(vertex != AggregatesMap::ISOLATED) (*block)[COMPONENT_INDEX] += this->lhs_[vertex]; } communication_->copyOwnerToAll(fine,fine); } else { auto end=fine.end(), begin=fine.begin(); for(auto block=begin; block != end; ++block) { (*block)[COMPONENT_INDEX] = this->lhs_[block-begin]; } } } OneComponentAggregationLevelTransferPolicy* clone() const { return new OneComponentAggregationLevelTransferPolicy(*this); } const Communication& getCoarseLevelCommunication() const { return *coarseLevelCommunication_; } private: typename Operator::matrix_type::field_type prolongDamp_; std::shared_ptr aggregatesMap_; Criterion criterion_; Communication* communication_; std::shared_ptr coarseLevelCommunication_; std::shared_ptr coarseLevelMatrix_; bool cpr_pressure_aggregation_; }; /** * \brief An algebraic twolevel or multigrid approach for solving blackoil (supports CPR with and without AMG) * * This preconditioner first decouples the component used for coarsening using a simple scaling * approach (e.g. Scheichl, Masson 2013,\see scaleMatrixDRS). Then it constructs the * coarse level system. The coupling is defined by the weights corresponding to the element located at * (COMPONENT_INDEX, VARIABLE_INDEX) in the block matrix. Then the coarse level system is constructed * either by extracting these elements, or by applying aggregation to them directly. This coarse level * can be solved either by AMG or by ILU. The preconditioner is configured using CPRParameter. * \tparam O The type of the operator (encapsulating a BCRSMatrix). * \tparam S The type of the smoother. * \tparam C The type of coarsening criterion to use. * \tparam P The type of the class describing the parallelization. * \tparam COMPONENT_INDEX The index of the component to use for coarsening (usually water). * \tparam VARIABLE_INDEX The index of the variable to use for coarsening (usually pressure). */ template class BlackoilAmg : public Dune::Preconditioner { public: /** \brief The type of the operator (encapsulating a BCRSMatrix). */ using Operator = O; /** \brief The type of coarsening criterion to use. */ using Criterion = C; /** \brief The type of the class describing the parallelization. */ using Communication = P; /** \brief The type of the smoother. */ using Smoother = S; /** \brief The type of the smoother arguments for construction. */ using SmootherArgs = typename Dune::Amg::SmootherTraits::Arguments; protected: using Matrix = typename Operator::matrix_type; using CoarseOperator = typename Detail::ScalarType::value; using CoarseSmoother = typename Detail::ScalarType::value; using FineCriterion = typename Detail::OneComponentCriterionType::value; using CoarseCriterion = typename Detail::ScalarType::value; using LevelTransferPolicy = OneComponentAggregationLevelTransferPolicy; using CoarseSolverPolicy = Detail::OneStepAMGCoarseSolverPolicy; using TwoLevelMethod = Dune::Amg::TwoLevelMethod; public: #if DUNE_VERSION_NEWER(DUNE_ISTL, 2, 6) Dune::SolverCategory::Category category() const override { return std::is_same::value ? Dune::SolverCategory::sequential : Dune::SolverCategory::overlapping; } #else // define the category enum { //! \brief The category the precondtioner is part of. category = Operator::category }; #endif /** * \brief Constructor. * \param param The parameters used for configuring the solver. * \param fineOperator The operator of the fine level. * \param criterion The criterion describing the coarsening approach. * \param smargs The arguments for constructing the smoother. * \param comm The information about the parallelization. */ BlackoilAmg(const CPRParameter& param, const typename TwoLevelMethod::FineDomainType& weights, const Operator& fineOperator, const Criterion& criterion, const SmootherArgs& smargs, const Communication& comm) : param_(param), weights_(weights), scaledMatrixOperator_(Detail::scaleMatrixDRS(fineOperator, comm, COMPONENT_INDEX, weights, param)), smoother_(Detail::constructSmoother(std::get<1>(scaledMatrixOperator_), smargs, comm)), levelTransferPolicy_(criterion, comm, param.cpr_pressure_aggregation_), coarseSolverPolicy_(¶m, smargs, criterion), twoLevelMethod_(std::get<1>(scaledMatrixOperator_), smoother_, levelTransferPolicy_, coarseSolverPolicy_, 0, 1) {} void pre(typename TwoLevelMethod::FineDomainType& x, typename TwoLevelMethod::FineRangeType& b) { twoLevelMethod_.pre(x,b); } void post(typename TwoLevelMethod::FineDomainType& x) { twoLevelMethod_.post(x); } void apply(typename TwoLevelMethod::FineDomainType& v, const typename TwoLevelMethod::FineRangeType& d) { auto scaledD = d; Detail::scaleVectorDRS(scaledD, COMPONENT_INDEX, param_, weights_); twoLevelMethod_.apply(v, scaledD); } private: const CPRParameter& param_; const typename TwoLevelMethod::FineDomainType& weights_; std::tuple, Operator> scaledMatrixOperator_; std::shared_ptr smoother_; LevelTransferPolicy levelTransferPolicy_; CoarseSolverPolicy coarseSolverPolicy_; TwoLevelMethod twoLevelMethod_; }; namespace ISTLUtility { /// /// \brief A traits class for selecting the types of the preconditioner. /// /// \tparam M The type of the matrix. /// \tparam X The type of the domain of the linear problem. /// \tparam Y The type of the range of the linear problem. /// \tparam P The type of the parallel information. /// \tparam C The type of the coarsening criterion to use. /// \tparam index The pressure index. //// template struct BlackoilAmgSelector { using Criterion = C; using Selector = CPRSelector; using ParallelInformation = typename Selector::ParallelInformation; using Operator = typename Selector::Operator; using Smoother = typename Selector::EllipticPreconditioner; using AMG = BlackoilAmg; }; } // end namespace ISTLUtility } // end namespace Opm #endif