// -*- 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. */ /*! * \file * \copydoc Opm::Linear::ParallelBiCGStabSolverBackend */ #ifndef EWOMS_PARALLEL_BICGSTAB_BACKEND_HH #define EWOMS_PARALLEL_BICGSTAB_BACKEND_HH #include "linalgproperties.hh" #include "parallelbasebackend.hh" #include "bicgstabsolver.hh" #include "combinedcriterion.hh" #include "istlsparsematrixadapter.hh" #include namespace Opm::Linear { template class ParallelBiCGStabSolverBackend; } // namespace Opm::Linear namespace Opm::Properties { // Create new type tags namespace TTag { struct ParallelBiCGStabLinearSolver { using InheritsFrom = std::tuple; }; } // end namespace TTag template struct LinearSolverBackend { using type = Opm::Linear::ParallelBiCGStabSolverBackend; }; template struct LinearSolverMaxError { using type = GetPropType; static constexpr type value = 1e7; }; } // namespace Opm::Properties namespace Opm { namespace Linear { /*! * \ingroup Linear * * \brief Implements a generic linear solver abstraction. * * Chosing the preconditioner works by setting the "PreconditionerWrapper" property: * * \code * template * struct PreconditionerWrapper * { using type = Opm::Linear::PreconditionerWrapper$PRECONDITIONER; }; * \endcode * * Where the choices possible for '\c $PRECONDITIONER' are: * - \c Jacobi: A Jacobi preconditioner * - \c GaussSeidel: A Gauss-Seidel preconditioner * - \c SSOR: A symmetric successive overrelaxation (SSOR) preconditioner * - \c SOR: A successive overrelaxation (SOR) preconditioner * - \c ILUn: An ILU(n) preconditioner * - \c ILU0: An ILU(0) preconditioner. The results of this * preconditioner are the same as setting the * PreconditionerOrder property to 0 and using the ILU(n) * preconditioner. The reason for the existence of ILU0 is * that it is computationally cheaper because it does not * need to consider things which are only required for * higher orders */ template class ParallelBiCGStabSolverBackend : public ParallelBaseBackend { using ParentType = ParallelBaseBackend; using Scalar = GetPropType; using Simulator = GetPropType; using SparseMatrixAdapter = GetPropType; using ParallelOperator = typename ParentType::ParallelOperator; using OverlappingVector = typename ParentType::OverlappingVector; using ParallelPreconditioner = typename ParentType::ParallelPreconditioner; using ParallelScalarProduct = typename ParentType::ParallelScalarProduct; using MatrixBlock = typename SparseMatrixAdapter::MatrixBlock; using RawLinearSolver = BiCGStabSolver; static_assert(std::is_same >::value, "The ParallelIstlSolverBackend linear solver backend requires the IstlSparseMatrixAdapter"); public: ParallelBiCGStabSolverBackend(const Simulator& simulator) : ParentType(simulator) { } static void registerParameters() { ParentType::registerParameters(); EWOMS_REGISTER_PARAM(TypeTag, Scalar, LinearSolverMaxError, "The maximum residual error which the linear solver tolerates" " without giving up"); } protected: friend ParentType; std::shared_ptr prepareSolver_(ParallelOperator& parOperator, ParallelScalarProduct& parScalarProduct, ParallelPreconditioner& parPreCond) { const auto& gridView = this->simulator_.gridView(); using CCC = CombinedCriterion; Scalar linearSolverTolerance = EWOMS_GET_PARAM(TypeTag, Scalar, LinearSolverTolerance); Scalar linearSolverAbsTolerance = EWOMS_GET_PARAM(TypeTag, Scalar, LinearSolverAbsTolerance); if(linearSolverAbsTolerance < 0.0) linearSolverAbsTolerance = this->simulator_.model().newtonMethod().tolerance() / 100.0; convCrit_.reset(new CCC(gridView.comm(), /*residualReductionTolerance=*/linearSolverTolerance, /*absoluteResidualTolerance=*/linearSolverAbsTolerance, EWOMS_GET_PARAM(TypeTag, Scalar, LinearSolverMaxError))); auto bicgstabSolver = std::make_shared(parPreCond, *convCrit_, parScalarProduct); int verbosity = 0; if (parOperator.overlap().myRank() == 0) verbosity = EWOMS_GET_PARAM(TypeTag, int, LinearSolverVerbosity); bicgstabSolver->setVerbosity(verbosity); bicgstabSolver->setMaxIterations(EWOMS_GET_PARAM(TypeTag, int, LinearSolverMaxIterations)); bicgstabSolver->setLinearOperator(&parOperator); bicgstabSolver->setRhs(this->overlappingb_); return bicgstabSolver; } std::pair runSolver_(std::shared_ptr solver) { bool converged = solver->apply(*this->overlappingx_); return std::make_pair(converged, int(solver->report().iterations())); } void cleanupSolver_() { /* nothing to do */ } std::unique_ptr > convCrit_; }; }} // namespace Linear, Opm #endif