/* Copyright 2014 SINTEF ICT, Applied Mathematics. Copyright 2014 IRIS AS. Copyright 2015 Dr. Blatt - HPC-Simulation-Software & Services Copyright 2015 NTNU Copyright 2015 Statoil 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 . */ #ifndef OPM_CPRPRECONDITIONER_HEADER_INCLUDED #define OPM_CPRPRECONDITIONER_HEADER_INCLUDED #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Opm { 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. //// template struct CPRSelector { /// \brief The information about the parallelization and communication typedef Dune::Amg::SequentialInformation ParallelInformation; /// \brief The operator type; typedef Dune::MatrixAdapter Operator; /// \brief The type of the preconditioner used for the elliptic part. typedef Dune::SeqILU0 EllipticPreconditioner; /// \brief The type of the unique pointer to the preconditioner of the elliptic part. typedef std::unique_ptr EllipticPreconditionerPointer; /// \brief type of AMG used to precondition the elliptic system. typedef EllipticPreconditioner Smoother; typedef Dune::Amg::AMG AMG; /// \brief creates an Operator from the matrix /// \param M The matrix to use. /// \param p The parallel information to use. static Operator* makeOperator(const M& m, const P&) { return new Operator(m); } }; #if HAVE_MPI /// \copydoc CPRSelector template struct CPRSelector > { /// \brief The information about the parallelization and communication typedef Dune::OwnerOverlapCopyCommunication ParallelInformation; /// \brief The operator type; typedef Dune::OverlappingSchwarzOperator Operator; /// \brief The type of the preconditioner used for the elliptic part. //typedef Dune::BlockPreconditioner > //EllipticPreconditioner; //typedef ParallelRestrictedOverlappingSchwarz > //EllipticPreconditioner; typedef ParallelOverlappingILU0 EllipticPreconditioner; /// \brief The type of the unique pointer to the preconditioner of the elliptic part. typedef std::unique_ptr > > EllipticPreconditionerPointer; typedef EllipticPreconditioner Smoother; typedef Dune::Amg::AMG AMG; /// \brief creates an Operator from the matrix /// \param M The matrix to use. /// \param p The parallel information to use. static Operator* makeOperator(const M& m, const ParallelInformation& p) { return new Operator(m, p); } }; //! \brief Creates the deleter needed for the parallel ILU preconditioners. //! \tparam ILU The type of the underlying sequential ILU preconditioner. //! \tparam I1 The global index type. //! \tparam I2 The local index type. //! \param ilu A reference to the wrapped preconditioner //! \param p The parallel information for template parameter deduction. template AdditionalObjectDeleter createParallelDeleter(ILU& ilu, const Dune::OwnerOverlapCopyCommunication& p) { (void) p; return AdditionalObjectDeleter(ilu); } #endif //! \brief Creates and initializes a unique pointer to an sequential ILU0 preconditioner. //! \param A The matrix of the linear system to solve. //! \param relax The relaxation factor to use. template std::shared_ptr > createILU0Ptr(const M& A, double relax, const Dune::Amg::SequentialInformation&) { return std::shared_ptr >(new Dune::SeqILU0( A, relax) ); } //! \brief Creates and initializes a shared pointer to an ILUn preconditioner. //! \param A The matrix of the linear system to solve. //! \param ilu_n The n parameter for the extension of the nonzero pattern. //! \param relax The relaxation factor to use. template std::shared_ptr > createILUnPtr(const M& A, int ilu_n, double relax, const Dune::Amg::SequentialInformation&) { return std::shared_ptr >(new Dune::SeqILUn( A, ilu_n, relax) ); } #if HAVE_MPI template struct SelectParallelILUSharedPtr { typedef std::shared_ptr< Dune::BlockPreconditioner< typename ILU::range_type, typename ILU::domain_type, Dune::OwnerOverlapCopyCommunication, ILU > > type; }; //! \brief Creates and initializes a shared pointer to an ILUn preconditioner. //! \param A The matrix of the linear system to solve. //! \param relax The relaxation factor to use. /// \param comm The object describing the parallelization information and communication. template typename SelectParallelILUSharedPtr, I1, I2>::type createILU0Ptr(const M& A, double relax, const Dune::OwnerOverlapCopyCommunication& comm) { typedef Dune::BlockPreconditioner< X, X, Dune::OwnerOverlapCopyCommunication, Dune::SeqILU0 > PointerType; Dune::SeqILU0* ilu = nullptr; int ilu_setup_successful = 1; std::string message; try { ilu = new Dune::SeqILU0(A, relax); } catch ( Dune::MatrixBlockError error ) { message = error.what(); std::cerr<<"Exception occured on process " << comm.communicator().rank() << " during " << "setup of ILU0 preconditioner with message: " << message<, I1, I2> ::type ( new PointerType(*ilu, comm), createParallelDeleter(*ilu, comm)); } //! \brief Creates and initializes a shared pointer to an ILUn preconditioner. //! \param A The matrix of the linear system to solve. //! \param ilu_n The n parameter for the extension of the nonzero pattern. //! \param relax The relaxation factor to use. /// \param comm The object describing the parallelization information and communication. template typename SelectParallelILUSharedPtr, I1, I2>::type createILUnPtr(const M& A, int ilu_n, double relax, const Dune::OwnerOverlapCopyCommunication& comm) { typedef Dune::BlockPreconditioner< X, X, Dune::OwnerOverlapCopyCommunication, Dune::SeqILUn > PointerType; Dune::SeqILUn* ilu = new Dune::SeqILUn( A, ilu_n, relax); return typename SelectParallelILUSharedPtr, I1, I2>::type (new PointerType(*ilu, comm),createParallelDeleter(*ilu, comm)); } #endif /// \brief Creates the elliptic preconditioner (ILU0) /// \param Ae The matrix of the elliptic system. /// \param relax The relaxation parameter for ILU0 template std::unique_ptr > createEllipticPreconditionerPointer(const M& Ae, double relax, const Dune::Amg::SequentialInformation&) { return std::unique_ptr >(new Dune::SeqILU0(Ae, relax)); } #if HAVE_MPI /// \brief Creates the elliptic preconditioner (ILU0) /// \param Ae The matrix of the elliptic system. /// \param relax The relaxation parameter for ILU0. /// \param comm The object describing the parallelization information and communication. template typename CPRSelector > ::EllipticPreconditionerPointer createEllipticPreconditionerPointer(const M& Ae, double relax, const Dune::OwnerOverlapCopyCommunication& comm) { typedef typename CPRSelector > ::EllipticPreconditioner ParallelPreconditioner; //Dune::SeqILU0* ilu=new Dune::SeqILU0(Ae, relax); typedef typename CPRSelector > ::EllipticPreconditionerPointer EllipticPreconditionerPointer; return EllipticPreconditionerPointer(new ParallelPreconditioner(Ae, comm, relax)); } #endif // #if HAVE_MPI /// \brief Creates the elliptic preconditioner (ILU0) /// \param opA The operator representing the matrix of the system. /// \param relax The relaxation parameter for ILU0. /// \param comm The object describing the parallelization information and communication. // \param amgPtr The unique_ptr to be filled (return) template inline void createAMGPreconditionerPointer( Op& opA, const double relax, const P& comm, std::unique_ptr< AMG >& amgPtr ) { // type of matrix typedef typename Op::matrix_type M; // The coupling metric used in the AMG typedef Dune::Amg::FirstDiagonal CouplingMetric; // The coupling criterion used in the AMG typedef Dune::Amg::SymmetricCriterion CritBase; // The coarsening criterion used in the AMG typedef Dune::Amg::CoarsenCriterion Criterion; // TODO: revise choice of parameters int coarsenTarget=1200; Criterion criterion(15,coarsenTarget); criterion.setDebugLevel( 0 ); // no debug information, 1 for printing hierarchy information criterion.setDefaultValuesIsotropic(2); criterion.setNoPostSmoothSteps( 1 ); criterion.setNoPreSmoothSteps( 1 ); // for DUNE 2.2 we also need to pass the smoother args typedef typename AMG::Smoother Smoother; typedef typename Dune::Amg::SmootherTraits::Arguments SmootherArgs; SmootherArgs smootherArgs; smootherArgs.iterations = 1; smootherArgs.relaxationFactor = relax; amgPtr.reset( new AMG(opA, criterion, smootherArgs, comm ) ); } } // end namespace ISTLUtility struct CPRParameter { double cpr_relax_; double cpr_solver_tol_; int cpr_ilu_n_; int cpr_max_ell_iter_; bool cpr_use_amg_; bool cpr_use_bicgstab_; bool cpr_solver_verbose_; CPRParameter() { reset(); } CPRParameter( const parameter::ParameterGroup& param) { // reset values to default reset(); cpr_relax_ = param.getDefault("cpr_relax", cpr_relax_); cpr_solver_tol_ = param.getDefault("cpr_solver_tol", cpr_solver_tol_); cpr_ilu_n_ = param.getDefault("cpr_ilu_n", cpr_ilu_n_); cpr_max_ell_iter_ = param.getDefault("cpr_max_elliptic_iter",cpr_max_ell_iter_); cpr_use_amg_ = param.getDefault("cpr_use_amg", cpr_use_amg_); cpr_use_bicgstab_ = param.getDefault("cpr_use_bicgstab", cpr_use_bicgstab_); cpr_solver_verbose_ = param.getDefault("cpr_solver_verbose", cpr_solver_verbose_); } void reset() { cpr_relax_ = 1.0; cpr_solver_tol_ = 1e-2; cpr_ilu_n_ = 0; cpr_max_ell_iter_ = 25; cpr_use_amg_ = true; cpr_use_bicgstab_ = true; cpr_solver_verbose_ = false; } }; /*! \brief CPR preconditioner. This is a two-stage preconditioner, combining an elliptic-type partial solution with ILU0 for the whole system. \tparam M The matrix type to operate on \tparam X Type of the update \tparam Y Type of the defect \tparam P Type of the parallel information. If not provided this will be Dune::Amg::SequentialInformation. The preconditioner is parallel if this is Dune::OwnerOverlapCopyCommunication */ template class CPRPreconditioner : public Dune::Preconditioner { // prohibit copying for now CPRPreconditioner( const CPRPreconditioner& ); public: //! \brief The type describing the parallel information typedef P ParallelInformation; //! \brief The matrix type the preconditioner is for. typedef typename Dune::remove_const::type matrix_type; //! \brief The domain type of the preconditioner. typedef X domain_type; //! \brief The range type of the preconditioner. typedef Y range_type; //! \brief The field type of the preconditioner. typedef typename X::field_type field_type; // define the category enum { //! \brief The category the preconditioner is part of. category = std::is_same::value? Dune::SolverCategory::sequential:Dune::SolverCategory::overlapping }; typedef ISTLUtility::CPRSelector CPRSelectorType ; //! \brief Elliptic Operator typedef typename CPRSelectorType::Operator Operator; //! \brief preconditioner for the whole system (here either ILU(0) or ILU(n) typedef Dune::Preconditioner WholeSystemPreconditioner; //! \brief type of the unique pointer to the ilu-0 preconditioner //! used the for the elliptic system typedef typename CPRSelectorType::EllipticPreconditionerPointer EllipticPreconditionerPointer; //! \brief amg preconditioner for the elliptic system typedef typename CPRSelectorType::AMG AMG; /*! \brief Constructor. Constructor gets all parameters to operate the prec. \param A The matrix to operate on. \param Ae The top-left elliptic part of A. \param relax The ILU0 relaxation factor. \param useAMG if true, AMG is used as a preconditioner for the elliptic sub-system, otherwise ilu-0 (default) \param useBiCG if true, BiCG solver is used (default), otherwise CG solver \param paralleInformation The information about the parallelization, if this is a parallel run */ CPRPreconditioner (const CPRParameter& param, const M& A, const M& Ae, const ParallelInformation& comm=ParallelInformation(), const ParallelInformation& commAe=ParallelInformation()) : param_( param ), A_(A), Ae_(Ae), de_( Ae_.N() ), ve_( Ae_.M() ), dmodified_( A_.N() ), opAe_(CPRSelectorType::makeOperator(Ae_, commAe)), precond_(), // ilu0 preconditioner for elliptic system amg_(), // amg preconditioner for elliptic system pre_(), // copy A will be made be the preconditioner vilu_( A_.N() ), comm_(comm), commAe_(commAe) { // create appropriate preconditioner for elliptic system createEllipticPreconditioner( param_.cpr_use_amg_, commAe_ ); // create the preconditioner for the whole system. if( param_.cpr_ilu_n_ == 0 ) { pre_ = ISTLUtility::createILU0Ptr( A_, param_.cpr_relax_, comm_ ); } else { pre_ = ISTLUtility::createILUnPtr( A_, param_.cpr_ilu_n_, param_.cpr_relax_, comm_ ); } } /*! \brief Prepare the preconditioner. \copydoc Preconditioner::pre(X&,Y&) */ virtual void pre (X& /*x*/, Y& /*b*/) { } /*! \brief Apply the preconditoner. \copydoc Preconditioner::apply(X&,const Y&) */ virtual void apply (X& v, const Y& d) { // Extract part of d corresponding to elliptic part. // Note: Assumes that the elliptic part comes first. std::copy_n(d.begin(), de_.size(), de_.begin()); // Solve elliptic part, extend solution to full. // reset result ve_ = 0; solveElliptic( ve_, de_ ); //reset return value v = 0.0; // Again assuming that the elliptic part comes first. std::copy(ve_.begin(), ve_.end(), v.begin()); // Subtract elliptic residual from initial residual. // dmodified = d - A * vfull dmodified_ = d; A_.mmv(v, dmodified_); // A is not parallel, do communication manually. comm_.copyOwnerToAll(dmodified_, dmodified_); // Apply Preconditioner for whole system (relax will be applied already) pre_->apply( vilu_, dmodified_); // don't apply relaxation if relax_ == 1 if( std::abs( param_.cpr_relax_ - 1.0 ) < 1e-12 ) { v += vilu_; } else { v *= param_.cpr_relax_; v += vilu_; } } /*! \brief Clean up. \copydoc Preconditioner::post(X&) */ virtual void post (X& /*x*/) { } protected: void solveElliptic(Y& x, Y& de) { // Linear solver parameters const double tolerance = param_.cpr_solver_tol_; const int maxit = param_.cpr_max_ell_iter_; const int verbosity = ( param_.cpr_solver_verbose_ && comm_.communicator().rank()==0 ) ? 1 : 0; // operator result containing iterations etc. Dune::InverseOperatorResult result; // the scalar product chooser typedef Dune::ScalarProductChooser ScalarProductChooser; // the scalar product. std::unique_ptr sp(ScalarProductChooser::construct(commAe_)); if( amg_ ) { // Solve system with AMG if( param_.cpr_use_bicgstab_ ) { Dune::BiCGSTABSolver linsolve(*opAe_, *sp, (*amg_), tolerance, maxit, verbosity); linsolve.apply(x, de, result); } else { Dune::CGSolver linsolve(*opAe_, *sp, (*amg_), tolerance, maxit, verbosity); linsolve.apply(x, de, result); } } else { assert( precond_ ); // Solve system with ILU-0 if( param_.cpr_use_bicgstab_ ) { Dune::BiCGSTABSolver linsolve(*opAe_, *sp, (*precond_), tolerance, maxit, verbosity); linsolve.apply(x, de, result); } else { Dune::CGSolver linsolve(*opAe_, *sp, (*precond_), tolerance, maxit, verbosity); linsolve.apply(x, de, result); } } if (!result.converged) { OPM_THROW(LinearSolverProblem, "CPRPreconditioner failed to solve elliptic subsystem."); } } //! \brief Parameter collection for CPR const CPRParameter& param_; //! \brief The matrix for the full linear problem. const matrix_type& A_; //! \brief The elliptic part of the matrix. const matrix_type& Ae_; //! \brief temporary variables for elliptic solve Y de_, ve_, dmodified_; //! \brief elliptic operator std::unique_ptr opAe_; //! \brief ILU0 preconditioner for the elliptic system EllipticPreconditionerPointer precond_; //! \brief AMG preconditioner with ILU0 smoother std::unique_ptr< AMG > amg_; //! \brief The preconditioner for the whole system //! //! We have to use a shared_ptr instead of a unique_ptr //! as we need to use a custom allocator based on dynamic //! information. But for unique_ptr the type of this deleter //! has to be available at coompile time. std::shared_ptr< WholeSystemPreconditioner > pre_; //! \brief temporary variables for ILU solve Y vilu_; //! \brief The information about the parallelization of the whole system. const P& comm_; //! \brief The information about the parallelization of the elliptic part //! of the system const P& commAe_; protected: void createEllipticPreconditioner( const bool amg, const P& comm ) { if( amg ) { ISTLUtility::createAMGPreconditionerPointer( *opAe_ , param_.cpr_relax_, comm, amg_ ); } else { precond_ = ISTLUtility::createEllipticPreconditionerPointer( Ae_, param_.cpr_relax_, comm); } } }; } // namespace Opm #endif // OPM_CPRPRECONDITIONER_HEADER_INCLUDED