/* Copyright 2009, 2010 SINTEF ICT, Applied Mathematics. Copyright 2019 SINTEF Digital, Mathematics and Cybernetics. 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 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if HAVE_CUDA #include #include #include #include #endif namespace Opm { template struct AMGSmootherArgsHelper { static auto args(const PropertyTree& prm) { using SmootherArgs = typename Dune::Amg::SmootherTraits::Arguments; SmootherArgs smootherArgs; smootherArgs.iterations = prm.get("iterations", 1); // smootherArgs.overlap=SmootherArgs::vertex; // smootherArgs.overlap=SmootherArgs::none; // smootherArgs.overlap=SmootherArgs::aggregate; smootherArgs.relaxationFactor = prm.get("relaxation", 1.0); return smootherArgs; } }; template struct AMGSmootherArgsHelper> { static auto args(const PropertyTree& prm) { using Smoother = Opm::ParallelOverlappingILU0; using SmootherArgs = typename Dune::Amg::SmootherTraits::Arguments; SmootherArgs smootherArgs; smootherArgs.iterations = prm.get("iterations", 1); const int iluwitdh = prm.get("iluwidth", 0); smootherArgs.setN(iluwitdh); const MILU_VARIANT milu = convertString2Milu(prm.get("milutype", std::string("ilu"))); smootherArgs.setMilu(milu); // smootherArgs.overlap=SmootherArgs::vertex; // smootherArgs.overlap=SmootherArgs::none; // smootherArgs.overlap=SmootherArgs::aggregate; smootherArgs.relaxationFactor = prm.get("relaxation", 1.0); return smootherArgs; } }; template typename AMGHelper::Criterion AMGHelper::criterion(const PropertyTree& prm) { Criterion criterion(15, prm.get("coarsenTarget", 1200)); criterion.setDefaultValuesIsotropic(2); criterion.setAlpha(prm.get("alpha", 0.33)); criterion.setBeta(prm.get("beta", 1e-5)); criterion.setMaxLevel(prm.get("maxlevel", 15)); criterion.setSkipIsolated(prm.get("skip_isolated", false)); criterion.setNoPreSmoothSteps(prm.get("pre_smooth", 1)); criterion.setNoPostSmoothSteps(prm.get("post_smooth", 1)); criterion.setDebugLevel(prm.get("verbosity", 0)); // As the default we request to accumulate data to 1 process always as our matrix // graph might be unsymmetric and hence not supported by the PTScotch/ParMetis // calls in DUNE. Accumulating to 1 skips PTScotch/ParMetis criterion.setAccumulate(static_cast(prm.get("accumulate", 1))); criterion.setProlongationDampingFactor(prm.get("prolongationdamping", 1.6)); criterion.setMaxDistance(prm.get("maxdistance", 2)); criterion.setMaxConnectivity(prm.get("maxconnectivity", 15)); criterion.setMaxAggregateSize(prm.get("maxaggsize", 6)); criterion.setMinAggregateSize(prm.get("minaggsize", 4)); return criterion; } template template typename AMGHelper::PrecPtr AMGHelper:: makeAmgPreconditioner(const Operator& op, const PropertyTree& prm, bool useKamg) { auto crit = criterion(prm); auto sargs = AMGSmootherArgsHelper::args(prm); if (useKamg) { using Type = Dune::DummyUpdatePreconditioner>; return std::make_shared(op, crit, sargs, prm.get("max_krylov", 1), prm.get("min_reduction", 1e-1)); } else { using Type = Dune::Amg::AMGCPR; return std::make_shared(op, crit, sargs); } } template struct StandardPreconditioners { static void add() { using namespace Dune; using O = Operator; using C = Comm; using F = PreconditionerFactory; using M = typename F::Matrix; using V = typename F::Vector; using P = PropertyTree; F::addCreator("ILU0", [](const O& op, const P& prm, const std::function&, std::size_t, const C& comm) { return createParILU(op, prm, comm, 0); }); F::addCreator("ParOverILU0", [](const O& op, const P& prm, const std::function&, std::size_t, const C& comm) { return createParILU(op, prm, comm, prm.get("ilulevel", 0)); }); F::addCreator("ILUn", [](const O& op, const P& prm, const std::function&, std::size_t, const C& comm) { return createParILU(op, prm, comm, prm.get("ilulevel", 0)); }); F::addCreator("Jac", [](const O& op, const P& prm, const std::function&, std::size_t, const C& comm) { const int n = prm.get("repeats", 1); const double w = prm.get("relaxation", 1.0); return wrapBlockPreconditioner>>(comm, op.getmat(), n, w); }); F::addCreator("GS", [](const O& op, const P& prm, const std::function&, std::size_t, const C& comm) { const int n = prm.get("repeats", 1); const double w = prm.get("relaxation", 1.0); return wrapBlockPreconditioner>>(comm, op.getmat(), n, w); }); F::addCreator("SOR", [](const O& op, const P& prm, const std::function&, std::size_t, const C& comm) { const int n = prm.get("repeats", 1); const double w = prm.get("relaxation", 1.0); return wrapBlockPreconditioner>>(comm, op.getmat(), n, w); }); F::addCreator("SSOR", [](const O& op, const P& prm, const std::function&, std::size_t, const C& comm) { const int n = prm.get("repeats", 1); const double w = prm.get("relaxation", 1.0); return wrapBlockPreconditioner>>(comm, op.getmat(), n, w); }); // Only add AMG preconditioners to the factory if the operator // is the overlapping schwarz operator. This could be extended // later, but at this point no other operators are compatible // with the AMG hierarchy construction. if constexpr (std::is_same_v>) { F::addCreator("amg", [](const O& op, const P& prm, const std::function&, std::size_t, const C& comm) { const std::string smoother = prm.get("smoother", "ParOverILU0"); if (smoother == "ILU0" || smoother == "ParOverILU0") { using Smoother = Opm::ParallelOverlappingILU0; auto crit = AMGHelper::criterion(prm); auto sargs = AMGSmootherArgsHelper::args(prm); return std::make_shared>(op, crit, sargs, comm); } else { OPM_THROW(std::invalid_argument, "Properties: No smoother with name " + smoother + "."); } }); } F::addCreator("cpr", [](const O& op, const P& prm, const std::function weightsCalculator, std::size_t pressureIndex, const C& comm) { assert(weightsCalculator); if (pressureIndex == std::numeric_limits::max()) { OPM_THROW(std::logic_error, "Pressure index out of bounds. It needs to specified for CPR"); } using LevelTransferPolicy = Opm::PressureTransferPolicy; return std::make_shared>(op, prm, weightsCalculator, pressureIndex, comm); }); F::addCreator("cprt", [](const O& op, const P& prm, const std::function weightsCalculator, std::size_t pressureIndex, const C& comm) { assert(weightsCalculator); if (pressureIndex == std::numeric_limits::max()) { OPM_THROW(std::logic_error, "Pressure index out of bounds. It needs to specified for CPR"); } using LevelTransferPolicy = Opm::PressureTransferPolicy; return std::make_shared>(op, prm, weightsCalculator, pressureIndex, comm); }); if constexpr (std::is_same_v>) { F::addCreator("cprw", [](const O& op, const P& prm, const std::function weightsCalculator, std::size_t pressureIndex, const C& comm) { assert(weightsCalculator); if (pressureIndex == std::numeric_limits::max()) { OPM_THROW(std::logic_error, "Pressure index out of bounds. It needs to specified for CPR"); } using LevelTransferPolicy = Opm::PressureBhpTransferPolicy; return std::make_shared>(op, prm, weightsCalculator, pressureIndex, comm); }); } #if HAVE_CUDA F::addCreator("CUILU0", [](const O& op, const P& prm, const std::function&, std::size_t, const C& comm) { const double w = prm.get("relaxation", 1.0); using field_type = typename V::field_type; using CuILU0 = typename Opm::cuistl::CuSeqILU0, Opm::cuistl::CuVector>; auto cuILU0 = std::make_shared(op.getmat(), w); auto adapted = std::make_shared>(cuILU0); auto wrapped = std::make_shared>(adapted, comm); return wrapped; }); #endif } static typename PreconditionerFactory::PrecPtr createParILU(const Operator& op, const PropertyTree& prm, const Comm& comm, const int ilulevel) { using F = PreconditionerFactory; using M = typename F::Matrix; using V = typename F::Vector; const double w = prm.get("relaxation", 1.0); const bool redblack = prm.get("redblack", false); const bool reorder_spheres = prm.get("reorder_spheres", false); // Already a parallel preconditioner. Need to pass comm, but no need to wrap it in a BlockPreconditioner. if (ilulevel == 0) { const size_t num_interior = interiorIfGhostLast(comm); return std::make_shared>( op.getmat(), comm, w, Opm::MILU_VARIANT::ILU, num_interior, redblack, reorder_spheres); } else { return std::make_shared>( op.getmat(), comm, ilulevel, w, Opm::MILU_VARIANT::ILU, redblack, reorder_spheres); } } /// Helper method to determine if the local partitioning has the /// K interior cells from [0, K-1] and ghost cells from [K, N-1]. /// Returns K if true, otherwise returns N. This is motivated by /// usage in the ParallelOverlappingILU0 preconditioner. static size_t interiorIfGhostLast(const Comm& comm) { size_t interior_count = 0; size_t highest_interior_index = 0; const auto& is = comm.indexSet(); for (const auto& ind : is) { if (Comm::OwnerSet::contains(ind.local().attribute())) { ++interior_count; highest_interior_index = std::max(highest_interior_index, ind.local().local()); } } if (highest_interior_index + 1 == interior_count) { return interior_count; } else { return is.size(); } } }; template struct StandardPreconditioners { static void add() { using namespace Dune; using O = Operator; using C = Dune::Amg::SequentialInformation; using F = PreconditionerFactory; using M = typename F::Matrix; using V = typename F::Vector; using P = PropertyTree; F::addCreator("ILU0", [](const O& op, const P& prm, const std::function&, std::size_t) { const double w = prm.get("relaxation", 1.0); return std::make_shared>( op.getmat(), 0, w, Opm::MILU_VARIANT::ILU); }); F::addCreator("ParOverILU0", [](const O& op, const P& prm, const std::function&, std::size_t) { const double w = prm.get("relaxation", 1.0); const int n = prm.get("ilulevel", 0); return std::make_shared>( op.getmat(), n, w, Opm::MILU_VARIANT::ILU); }); F::addCreator("ILUn", [](const O& op, const P& prm, const std::function&, std::size_t) { const int n = prm.get("ilulevel", 0); const double w = prm.get("relaxation", 1.0); return std::make_shared>( op.getmat(), n, w, Opm::MILU_VARIANT::ILU); }); F::addCreator("Jac", [](const O& op, const P& prm, const std::function&, std::size_t) { const int n = prm.get("repeats", 1); const double w = prm.get("relaxation", 1.0); return wrapPreconditioner>(op.getmat(), n, w); }); F::addCreator("GS", [](const O& op, const P& prm, const std::function&, std::size_t) { const int n = prm.get("repeats", 1); const double w = prm.get("relaxation", 1.0); return wrapPreconditioner>(op.getmat(), n, w); }); F::addCreator("SOR", [](const O& op, const P& prm, const std::function&, std::size_t) { const int n = prm.get("repeats", 1); const double w = prm.get("relaxation", 1.0); return wrapPreconditioner>(op.getmat(), n, w); }); F::addCreator("SSOR", [](const O& op, const P& prm, const std::function&, std::size_t) { const int n = prm.get("repeats", 1); const double w = prm.get("relaxation", 1.0); return wrapPreconditioner>(op.getmat(), n, w); }); // Only add AMG preconditioners to the factory if the operator // is an actual matrix operator. if constexpr (std::is_same_v>) { F::addCreator("amg", [](const O& op, const P& prm, const std::function&, std::size_t) { const std::string smoother = prm.get("smoother", "ParOverILU0"); if (smoother == "ILU0" || smoother == "ParOverILU0") { using Smoother = SeqILU; return AMGHelper::template makeAmgPreconditioner(op, prm); } else if (smoother == "Jac") { using Smoother = SeqJac; return AMGHelper::template makeAmgPreconditioner(op, prm); } else if (smoother == "SOR") { using Smoother = SeqSOR; return AMGHelper::template makeAmgPreconditioner(op, prm); } else if (smoother == "SSOR") { using Smoother = SeqSSOR; return AMGHelper::template makeAmgPreconditioner(op, prm); } else if (smoother == "ILUn") { using Smoother = SeqILU; return AMGHelper::template makeAmgPreconditioner(op, prm); } else { OPM_THROW(std::invalid_argument, "Properties: No smoother with name " + smoother + "."); } }); F::addCreator("kamg", [](const O& op, const P& prm, const std::function&, std::size_t) { const std::string smoother = prm.get("smoother", "ParOverILU0"); if (smoother == "ILU0" || smoother == "ParOverILU0") { using Smoother = SeqILU; return AMGHelper::template makeAmgPreconditioner(op, prm, true); } else if (smoother == "Jac") { using Smoother = SeqJac; return AMGHelper::template makeAmgPreconditioner(op, prm, true); } else if (smoother == "SOR") { using Smoother = SeqSOR; return AMGHelper::template makeAmgPreconditioner(op, prm, true); // } else if (smoother == "GS") { // using Smoother = SeqGS; // return makeAmgPreconditioner(op, prm, true); } else if (smoother == "SSOR") { using Smoother = SeqSSOR; return AMGHelper::template makeAmgPreconditioner(op, prm, true); } else if (smoother == "ILUn") { using Smoother = SeqILU; return AMGHelper::template makeAmgPreconditioner(op, prm, true); } else { OPM_THROW(std::invalid_argument, "Properties: No smoother with name " + smoother + "."); } }); F::addCreator("famg", [](const O& op, const P& prm, const std::function&, std::size_t) { auto crit = AMGHelper::criterion(prm); Dune::Amg::Parameters parms; parms.setNoPreSmoothSteps(1); parms.setNoPostSmoothSteps(1); return wrapPreconditioner>(op, crit, parms); }); } if constexpr (std::is_same_v>) { F::addCreator("cprw", [](const O& op, const P& prm, const std::function& weightsCalculator, std::size_t pressureIndex) { if (pressureIndex == std::numeric_limits::max()) { OPM_THROW(std::logic_error, "Pressure index out of bounds. It needs to specified for CPR"); } using LevelTransferPolicy = Opm::PressureBhpTransferPolicy; return std::make_shared>(op, prm, weightsCalculator, pressureIndex); }); } F::addCreator("cpr", [](const O& op, const P& prm, const std::function& weightsCalculator, std::size_t pressureIndex) { if (pressureIndex == std::numeric_limits::max()) { OPM_THROW(std::logic_error, "Pressure index out of bounds. It needs to specified for CPR"); } using LevelTransferPolicy = Opm::PressureTransferPolicy; return std::make_shared>(op, prm, weightsCalculator, pressureIndex); }); F::addCreator("cprt", [](const O& op, const P& prm, const std::function& weightsCalculator, std::size_t pressureIndex) { if (pressureIndex == std::numeric_limits::max()) { OPM_THROW(std::logic_error, "Pressure index out of bounds. It needs to specified for CPR"); } using LevelTransferPolicy = Opm::PressureTransferPolicy; return std::make_shared>(op, prm, weightsCalculator, pressureIndex); }); #if HAVE_CUDA F::addCreator("CUILU0", [](const O& op, const P& prm, const std::function&, std::size_t) { const double w = prm.get("relaxation", 1.0); using field_type = typename V::field_type; using CuILU0 = typename Opm::cuistl::CuSeqILU0, Opm::cuistl::CuVector>; return std::make_shared>(std::make_shared(op.getmat(), w)); }); F::addCreator("CUILU0Float", [](const O& op, const P& prm, const std::function&, std::size_t) { const double w = prm.get("relaxation", 1.0); using block_type = typename V::block_type; using VTo = Dune::BlockVector>; using matrix_type_to = typename Dune::BCRSMatrix>; using CuILU0 = typename Opm::cuistl::CuSeqILU0, Opm::cuistl::CuVector>; using Adapter = typename Opm::cuistl::PreconditionerAdapter; using Converter = typename Opm::cuistl::PreconditionerConvertFieldTypeAdapter; auto converted = std::make_shared(op.getmat()); auto adapted = std::make_shared(std::make_shared(converted->getConvertedMatrix(), w)); converted->setUnderlyingPreconditioner(adapted); return converted; }); #endif } }; template PreconditionerFactory::PreconditionerFactory() { } template PreconditionerFactory& PreconditionerFactory::instance() { static PreconditionerFactory singleton; return singleton; } template typename PreconditionerFactory::PrecPtr PreconditionerFactory:: doCreate(const Operator& op, const PropertyTree& prm, const std::function weightsCalculator, std::size_t pressureIndex) { if (!defAdded_) { StandardPreconditioners::add(); defAdded_ = true; } const std::string& type = prm.get("type", "ParOverILU0"); auto it = creators_.find(type); if (it == creators_.end()) { std::ostringstream msg; msg << "Preconditioner type " << type << " is not registered in the factory. Available types are: "; for (const auto& prec : creators_) { msg << prec.first << ' '; } msg << std::endl; OPM_THROW(std::invalid_argument, msg.str()); } return it->second(op, prm, weightsCalculator, pressureIndex); } template typename PreconditionerFactory::PrecPtr PreconditionerFactory:: doCreate(const Operator& op, const PropertyTree& prm, const std::function weightsCalculator, std::size_t pressureIndex, const Comm& comm) { if (!defAdded_) { StandardPreconditioners::add(); defAdded_ = true; } const std::string& type = prm.get("type", "ParOverILU0"); auto it = parallel_creators_.find(type); if (it == parallel_creators_.end()) { std::ostringstream msg; msg << "Parallel preconditioner type " << type << " is not registered in the factory. Available types are: "; for (const auto& prec : parallel_creators_) { msg << prec.first << ' '; } msg << std::endl; OPM_THROW(std::invalid_argument, msg.str()); } return it->second(op, prm, weightsCalculator, pressureIndex, comm); } template void PreconditionerFactory:: doAddCreator(const std::string& type, Creator c) { creators_[type] = c; } template void PreconditionerFactory:: doAddCreator(const std::string& type, ParCreator c) { parallel_creators_[type] = c; } template typename PreconditionerFactory::PrecPtr PreconditionerFactory:: create(const Operator& op, const PropertyTree& prm, const std::function& weightsCalculator, std::size_t pressureIndex) { return instance().doCreate(op, prm, weightsCalculator, pressureIndex); } template typename PreconditionerFactory::PrecPtr PreconditionerFactory:: create(const Operator& op, const PropertyTree& prm, const std::function& weightsCalculator, const Comm& comm, std::size_t pressureIndex) { return instance().doCreate(op, prm, weightsCalculator, pressureIndex, comm); } template typename PreconditionerFactory::PrecPtr PreconditionerFactory:: create(const Operator& op, const PropertyTree& prm, const Comm& comm, std::size_t pressureIndex) { return instance().doCreate(op, prm, std::function(), pressureIndex, comm); } template void PreconditionerFactory:: addCreator(const std::string& type, Creator creator) { instance().doAddCreator(type, creator); } template void PreconditionerFactory:: addCreator(const std::string& type, ParCreator creator) { instance().doAddCreator(type, creator); } using CommSeq = Dune::Amg::SequentialInformation; template using OpFSeq = Dune::MatrixAdapter>, Dune::BlockVector>, Dune::BlockVector>>; template using OpBSeq = Dune::MatrixAdapter>, Dune::BlockVector>, Dune::BlockVector>>; template using OpW = WellModelMatrixAdapter>, Dune::BlockVector>, Dune::BlockVector>, overlap>; template using OpWG = WellModelGhostLastMatrixAdapter>, Dune::BlockVector>, Dune::BlockVector>, overlap>; #if HAVE_MPI using CommPar = Dune::OwnerOverlapCopyCommunication; template using OpFPar = Dune::OverlappingSchwarzOperator>, Dune::BlockVector>, Dune::BlockVector>, CommPar>; template using OpBPar = Dune::OverlappingSchwarzOperator>, Dune::BlockVector>, Dune::BlockVector>, CommPar>; #define INSTANCE_PF_PAR(Dim) \ template class PreconditionerFactory,CommPar>; \ template class PreconditionerFactory,CommPar>; \ template class PreconditionerFactory,CommPar>; \ template class PreconditionerFactory,CommPar>; \ template class PreconditionerFactory,CommPar>; \ template class PreconditionerFactory,CommSeq>; #endif #define INSTANCE_PF_SEQ(Dim) \ template class PreconditionerFactory,CommSeq>; \ template class PreconditionerFactory,CommSeq>; \ template class PreconditionerFactory,CommSeq>; \ template class PreconditionerFactory,CommSeq>; #if HAVE_MPI #define INSTANCE_PF(Dim) \ INSTANCE_PF_PAR(Dim) \ INSTANCE_PF_SEQ(Dim) #else #define INSTANCE_PF(Dim) \ INSTANCE_PF_SEQ(Dim) #endif }