mirror of
https://github.com/OPM/opm-simulators.git
synced 2024-12-28 02:00:59 -06:00
166 lines
7.8 KiB
C++
166 lines
7.8 KiB
C++
// -*- 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 <http://www.gnu.org/licenses/>.
|
|
|
|
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
|
|
* \brief Provides wrapper classes for the iterative linear solvers available in
|
|
* dune-istl.
|
|
*
|
|
* In conjunction with a suitable solver backend, solver wrappers work by specifying the
|
|
* "SolverWrapper" property:
|
|
* \code
|
|
* template<class TypeTag>
|
|
* struct LinearSolverWrapper<TypeTag, TTag::YourTypeTag>
|
|
* { using type = Opm::Linear::SolverWrapper$SOLVER<TypeTag>; };
|
|
* \endcode
|
|
*
|
|
* The possible choices for '\c $SOLVER' are:
|
|
* - \c Richardson: A fixpoint solver using the Richardson iteration
|
|
* - \c SteepestDescent: The steepest descent solver
|
|
* - \c ConjugatedGradients: A conjugated gradients solver
|
|
* - \c BiCGStab: A stabilized bi-conjugated gradients solver
|
|
* - \c MinRes: A solver based on the minimized residual algorithm
|
|
* - \c RestartedGMRes: A restarted GMRES solver
|
|
*/
|
|
#ifndef EWOMS_ISTL_SOLVER_WRAPPERS_HH
|
|
#define EWOMS_ISTL_SOLVER_WRAPPERS_HH
|
|
|
|
#include <opm/models/utils/propertysystem.hh>
|
|
#include <opm/models/utils/parametersystem.hh>
|
|
#include <opm/simulators/linalg/linalgproperties.hh>
|
|
|
|
#include <dune/istl/solvers.hh>
|
|
|
|
namespace Opm::Linear {
|
|
|
|
/*!
|
|
* \brief Macro to create a wrapper around an ISTL solver
|
|
*/
|
|
#define EWOMS_WRAP_ISTL_SOLVER(SOLVER_NAME, ISTL_SOLVER_NAME) \
|
|
template <class TypeTag> \
|
|
class SolverWrapper##SOLVER_NAME \
|
|
{ \
|
|
using Scalar = GetPropType<TypeTag, Properties::Scalar>; \
|
|
using OverlappingMatrix = GetPropType<TypeTag, Properties::OverlappingMatrix>; \
|
|
using OverlappingVector = GetPropType<TypeTag, Properties::OverlappingVector>; \
|
|
\
|
|
public: \
|
|
using RawSolver = ISTL_SOLVER_NAME<OverlappingVector>; \
|
|
\
|
|
SolverWrapper##SOLVER_NAME() \
|
|
{} \
|
|
\
|
|
static void registerParameters() \
|
|
{} \
|
|
\
|
|
template <class LinearOperator, class ScalarProduct, class Preconditioner> \
|
|
std::shared_ptr<RawSolver> get(LinearOperator& parOperator, \
|
|
ScalarProduct& parScalarProduct, \
|
|
Preconditioner& parPreCond) \
|
|
{ \
|
|
Scalar tolerance = EWOMS_GET_PARAM(TypeTag, Scalar, \
|
|
LinearSolverTolerance); \
|
|
int maxIter = EWOMS_GET_PARAM(TypeTag, int, LinearSolverMaxIterations);\
|
|
\
|
|
int verbosity = 0; \
|
|
if (parOperator.overlap().myRank() == 0) \
|
|
verbosity = EWOMS_GET_PARAM(TypeTag, int, LinearSolverVerbosity); \
|
|
solver_ = std::make_shared<RawSolver>(parOperator, parScalarProduct, \
|
|
parPreCond, tolerance, maxIter, \
|
|
verbosity); \
|
|
\
|
|
return solver_; \
|
|
} \
|
|
\
|
|
void cleanup() \
|
|
{ solver_.reset(); } \
|
|
\
|
|
private: \
|
|
std::shared_ptr<RawSolver> solver_; \
|
|
};
|
|
|
|
EWOMS_WRAP_ISTL_SOLVER(Richardson, Dune::LoopSolver)
|
|
EWOMS_WRAP_ISTL_SOLVER(SteepestDescent, Dune::GradientSolver)
|
|
EWOMS_WRAP_ISTL_SOLVER(ConjugatedGradients, Dune::CGSolver)
|
|
EWOMS_WRAP_ISTL_SOLVER(BiCGStab, Dune::BiCGSTABSolver)
|
|
EWOMS_WRAP_ISTL_SOLVER(MinRes, Dune::MINRESSolver)
|
|
|
|
/*!
|
|
* \brief Solver wrapper for the restarted GMRES solver of dune-istl.
|
|
*
|
|
* dune-istl uses a slightly different API for this solver than for the others...
|
|
*/
|
|
template <class TypeTag>
|
|
class SolverWrapperRestartedGMRes
|
|
{
|
|
using Scalar = GetPropType<TypeTag, Properties::Scalar>;
|
|
using OverlappingMatrix = GetPropType<TypeTag, Properties::OverlappingMatrix>;
|
|
using OverlappingVector = GetPropType<TypeTag, Properties::OverlappingVector>;
|
|
|
|
public:
|
|
using RawSolver = Dune::RestartedGMResSolver<OverlappingVector>;
|
|
|
|
SolverWrapperRestartedGMRes()
|
|
{}
|
|
|
|
static void registerParameters()
|
|
{
|
|
EWOMS_REGISTER_PARAM(TypeTag, int, GMResRestart,
|
|
"Number of iterations after which the GMRES linear solver is restarted");
|
|
}
|
|
|
|
template <class LinearOperator, class ScalarProduct, class Preconditioner>
|
|
std::shared_ptr<RawSolver> get(LinearOperator& parOperator,
|
|
ScalarProduct& parScalarProduct,
|
|
Preconditioner& parPreCond)
|
|
{
|
|
Scalar tolerance = EWOMS_GET_PARAM(TypeTag, Scalar, LinearSolverTolerance);
|
|
int maxIter = EWOMS_GET_PARAM(TypeTag, int, LinearSolverMaxIterations);
|
|
|
|
int verbosity = 0;
|
|
if (parOperator.overlap().myRank() == 0)
|
|
verbosity = EWOMS_GET_PARAM(TypeTag, int, LinearSolverVerbosity);
|
|
int restartAfter = EWOMS_GET_PARAM(TypeTag, int, GMResRestart);
|
|
solver_ = std::make_shared<RawSolver>(parOperator,
|
|
parScalarProduct,
|
|
parPreCond,
|
|
tolerance,
|
|
restartAfter,
|
|
maxIter,
|
|
verbosity);
|
|
|
|
return solver_;
|
|
}
|
|
|
|
void cleanup()
|
|
{ solver_.reset(); }
|
|
|
|
private:
|
|
std::shared_ptr<RawSolver> solver_;
|
|
};
|
|
|
|
#undef EWOMS_WRAP_ISTL_SOLVER
|
|
|
|
} // namespace Opm::Linear
|
|
|
|
#endif
|