/* Copyright 2019, 2020 SINTEF Digital, Mathematics and Cybernetics. Copyright 2020 Equinor. 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_FLEXIBLE_SOLVER_HEADER_INCLUDED #define OPM_FLEXIBLE_SOLVER_HEADER_INCLUDED #include #include #include #include namespace Dune { /// A solver class that encapsulates all needed objects for a linear solver /// (operator, scalar product, iterative solver and preconditioner) and sets /// them up based on runtime parameters, using the PreconditionerFactory for /// setting up preconditioners. template class FlexibleSolver : public Dune::InverseOperator { public: using MatrixType = MatrixTypeT; using VectorType = VectorTypeT; /// Base class type of the operator passed to the solver. using AbstractOperatorType = Dune::AssembledLinearOperator; /// Base class type of the contained preconditioner. using AbstractPrecondType = Dune::PreconditionerWithUpdate; /// Create a sequential solver. FlexibleSolver(AbstractOperatorType& op, const boost::property_tree::ptree& prm, const std::function& weightsCalculator = std::function()); /// Create a parallel solver (if Comm is e.g. OwnerOverlapCommunication). template FlexibleSolver(AbstractOperatorType& op, const Comm& comm, const boost::property_tree::ptree& prm, const std::function& weightsCalculator = std::function()); virtual void apply(VectorType& x, VectorType& rhs, Dune::InverseOperatorResult& res) override; virtual void apply(VectorType& x, VectorType& rhs, double reduction, Dune::InverseOperatorResult& res) override; /// Access the contained preconditioner. AbstractPrecondType& preconditioner(); virtual Dune::SolverCategory::Category category() const override; private: using AbstractScalarProductType = Dune::ScalarProduct; using AbstractSolverType = Dune::InverseOperator; // Machinery for making sequential or parallel operators/preconditioners/scalar products. template void initOpPrecSp(AbstractOperatorType& op, const boost::property_tree::ptree& prm, const std::function weightsCalculator, const Comm& comm); void initOpPrecSp(AbstractOperatorType& op, const boost::property_tree::ptree& prm, const std::function weightsCalculator, const Dune::Amg::SequentialInformation&); void initSolver(const boost::property_tree::ptree& prm, const bool is_iorank); // Main initialization routine. // Call with Comm == Dune::Amg::SequentialInformation to get a serial solver. template void init(AbstractOperatorType& op, const Comm& comm, const boost::property_tree::ptree& prm, const std::function weightsCalculator); AbstractOperatorType* linearoperator_for_solver_; std::shared_ptr linearoperator_for_precond_; std::shared_ptr preconditioner_; std::shared_ptr scalarproduct_; std::shared_ptr linsolver_; }; } // namespace Dune #endif // OPM_FLEXIBLE_SOLVER_HEADER_INCLUDED