PreconditionerWithUpdate.hpp
Go to the documentation of this file.
1/*
2 Copyright 2019 SINTEF Digital, Mathematics and Cybernetics.
3
4 This file is part of the Open Porous Media project (OPM).
5
6 OPM is free software: you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation, either version 3 of the License, or
9 (at your option) any later version.
10
11 OPM is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
15
16 You should have received a copy of the GNU General Public License
17 along with OPM. If not, see <http://www.gnu.org/licenses/>.
18*/
19
20#ifndef OPM_PRECONDITIONERWITHUPDATE_HEADER_INCLUDED
21#define OPM_PRECONDITIONERWITHUPDATE_HEADER_INCLUDED
22
23#include <dune/istl/preconditioner.hh>
24#include <memory>
25
26namespace Dune
27{
28
30template <class X, class Y>
31class PreconditionerWithUpdate : public Preconditioner<X, Y>
32{
33public:
34 virtual void update() = 0;
35};
36
37template <class OriginalPreconditioner>
38class DummyUpdatePreconditioner : public PreconditionerWithUpdate<typename OriginalPreconditioner::domain_type,
39 typename OriginalPreconditioner::range_type>
40{
41public:
42 template <class... Args>
44 : orig_precond_(std::forward<Args>(args)...)
45 {
46 }
47
48 using X = typename OriginalPreconditioner::domain_type;
49 using Y = typename OriginalPreconditioner::range_type;
50
51 virtual void pre(X& x, Y& b) override
52 {
53 orig_precond_.pre(x, b);
54 }
55
56 virtual void apply(X& v, const Y& d) override
57 {
58 orig_precond_.apply(v, d);
59 }
60
61 virtual void post(X& x) override
62 {
63 orig_precond_.post(x);
64 }
65
66 virtual SolverCategory::Category category() const override
67 {
68 return orig_precond_.category();
69 }
70
71 // The update() function does nothing for a wrapped preconditioner.
72 virtual void update() override
73 {
74 }
75
76private:
77 OriginalPreconditioner orig_precond_;
78};
79
80template <class OriginalPreconditioner, class... Args>
81std::shared_ptr<DummyUpdatePreconditioner<OriginalPreconditioner>>
82getDummyUpdateWrapper(Args&&... args)
83{
84 return std::make_shared<DummyUpdatePreconditioner<OriginalPreconditioner>>(std::forward<Args>(args)...);
85}
86
89template <class OriginalPreconditioner>
91 virtual std::unique_ptr<
92 Preconditioner<typename OriginalPreconditioner::domain_type, typename OriginalPreconditioner::range_type>>
93 make() = 0;
94};
95
100template <class OriginalPreconditioner, class... Args>
101struct PreconditionerMaker : public GeneralPreconditionerMaker<OriginalPreconditioner> {
102 using GenericPreconditioner = Preconditioner<typename OriginalPreconditioner::domain_type, typename OriginalPreconditioner::range_type>;
103
104 PreconditionerMaker(Args&&... args)
105 : args_(args...)
106 {
107 }
108 std::unique_ptr<GenericPreconditioner>
109 make() override
110 {
111 // return std::unique_ptr<GenericPreconditioner> {new auto(std::make_from_tuple<OriginalPreconditioner>(args_))};
112 return std::apply(
113 [](auto&&... args) {
114 return std::make_unique<OriginalPreconditioner>(std::forward<Args>(args)...);
115 }, args_);
116 }
117 std::tuple<Args...> args_;
118};
119
122template <class OriginalPreconditioner>
123class RebuildOnUpdatePreconditioner : public PreconditionerWithUpdate<typename OriginalPreconditioner::domain_type,
124 typename OriginalPreconditioner::range_type>
125{
126public:
127 template<class... Args>
129 : preconditioner_maker_(std::make_unique<PreconditionerMaker<OriginalPreconditioner, Args...>>(std::forward<Args>(args)...))
130 {
131 update();
132 }
133
134 using X = typename OriginalPreconditioner::domain_type;
135 using Y = typename OriginalPreconditioner::range_type;
136
137 virtual void pre(X& x, Y& b) override
138 {
139 orig_precond_->pre(x, b);
140 }
141
142 virtual void apply(X& v, const Y& d) override
143 {
144 orig_precond_->apply(v, d);
145 }
146
147 virtual void post(X& x) override
148 {
149 orig_precond_->post(x);
150 }
151
152 virtual SolverCategory::Category category() const override
153 {
154 return orig_precond_->category();
155 }
156
157 // Rebuild the preconditioner on update
158 void update() override
159 {
160 orig_precond_ = preconditioner_maker_->make();
161 }
162
163private:
165 using GenericPreconditioner = Preconditioner<typename OriginalPreconditioner::domain_type, typename OriginalPreconditioner::range_type>;
166
167 std::unique_ptr<AbstractMakerType> preconditioner_maker_;
168 std::unique_ptr<GenericPreconditioner> orig_precond_;
169};
170
176template <class OriginalPreconditioner, class... Args>
177auto
179{
180 return std::make_shared<RebuildOnUpdatePreconditioner<OriginalPreconditioner>>(
181 std::forward<Args>(args)...);
182}
183
184} // namespace Dune
185
186#endif // OPM_PRECONDITIONERWITHUPDATE_HEADER_INCLUDED
Definition: PreconditionerWithUpdate.hpp:40
virtual void post(X &x) override
Definition: PreconditionerWithUpdate.hpp:61
DummyUpdatePreconditioner(Args &&... args)
Definition: PreconditionerWithUpdate.hpp:43
typename OriginalPreconditioner::range_type Y
Definition: PreconditionerWithUpdate.hpp:49
virtual void pre(X &x, Y &b) override
Definition: PreconditionerWithUpdate.hpp:51
virtual void apply(X &v, const Y &d) override
Definition: PreconditionerWithUpdate.hpp:56
virtual void update() override
Definition: PreconditionerWithUpdate.hpp:72
virtual SolverCategory::Category category() const override
Definition: PreconditionerWithUpdate.hpp:66
typename OriginalPreconditioner::domain_type X
Definition: PreconditionerWithUpdate.hpp:48
Interface class adding the update() method to the preconditioner interface.
Definition: PreconditionerWithUpdate.hpp:32
Wrapper class of preconditioners that should be reconstructed on update.
Definition: PreconditionerWithUpdate.hpp:125
typename OriginalPreconditioner::range_type Y
Definition: PreconditionerWithUpdate.hpp:135
void update() override
Definition: PreconditionerWithUpdate.hpp:158
virtual SolverCategory::Category category() const override
Definition: PreconditionerWithUpdate.hpp:152
RebuildOnUpdatePreconditioner(Args... args)
Definition: PreconditionerWithUpdate.hpp:128
virtual void apply(X &v, const Y &d) override
Definition: PreconditionerWithUpdate.hpp:142
typename OriginalPreconditioner::domain_type X
Definition: PreconditionerWithUpdate.hpp:134
virtual void pre(X &x, Y &b) override
Definition: PreconditionerWithUpdate.hpp:137
virtual void post(X &x) override
Definition: PreconditionerWithUpdate.hpp:147
Definition: SupportsFaceTag.hpp:27
auto getRebuildOnUpdateWrapper(Args... args)
Wrapper function creating and return a shared pointer to a preconditioner which is reconstructed on u...
Definition: PreconditionerWithUpdate.hpp:178
std::shared_ptr< DummyUpdatePreconditioner< OriginalPreconditioner > > getDummyUpdateWrapper(Args &&... args)
Definition: PreconditionerWithUpdate.hpp:82
Interface class ensuring make function is overriden.
Definition: PreconditionerWithUpdate.hpp:90
virtual std::unique_ptr< Preconditioner< typename OriginalPreconditioner::domain_type, typename OriginalPreconditioner::range_type > > make()=0
Struct implementing a make function which creates a preconditioner Can create arbitrary preconditione...
Definition: PreconditionerWithUpdate.hpp:101
PreconditionerMaker(Args &&... args)
Definition: PreconditionerWithUpdate.hpp:104
Preconditioner< typename OriginalPreconditioner::domain_type, typename OriginalPreconditioner::range_type > GenericPreconditioner
Definition: PreconditionerWithUpdate.hpp:102
std::tuple< Args... > args_
Definition: PreconditionerWithUpdate.hpp:117
std::unique_ptr< GenericPreconditioner > make() override
Definition: PreconditionerWithUpdate.hpp:109