Loading [MathJax]/extensions/tex2jax.js

DUNE PDELab (unstable)

nodes.hh
1// -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
2// vi: set et ts=4 sw=2 sts=2:
3
4// SPDX-FileCopyrightText: Copyright © DUNE Project contributors, see file AUTHORS.md
5// SPDX-License-Identifier: LicenseRef-GPL-2.0-only-with-DUNE-exception OR LGPL-3.0-or-later
6
7#ifndef DUNE_FUNCTIONS_FUNCTIONSPACEBASES_NODES_HH
8#define DUNE_FUNCTIONS_FUNCTIONSPACEBASES_NODES_HH
9
10#include <cassert>
11#include <memory>
12
13#include <dune/common/indices.hh>
14
15#include <dune/typetree/leafnode.hh>
16#include <dune/typetree/powernode.hh>
17#include <dune/typetree/dynamicpowernode.hh>
18#include <dune/typetree/compositenode.hh>
19#include <dune/typetree/traversal.hh>
20
21namespace Dune {
22 namespace Functions {
23
24
25 namespace Impl {
26
27 // This class encapsulates the access to the setOffset()
28 // and setTreeIndex() methods of a node. This way we
29 // can hide the methods from the user but still provide
30 // access where this is needed.
31 struct BasisNodeSetupHelper
32 {
33
34 template<class Node, class size_type>
35 static void setOffset(Node& node, const size_type offset)
36 {
37 node.setOffset(offset);
38 }
39
40 template<class Node, class size_type>
41 static void setTreeIndex(Node& node, const size_type index)
42 {
43 node.setTreeIndex(index);
44 }
45
46 };
47
48
49 } // end namespace Impl
50
51
52 class BasisNodeMixin
53 {
54
55 friend struct Impl::BasisNodeSetupHelper;
56
57 public:
58
59 using size_type = std::size_t;
60
61 BasisNodeMixin() :
62 offset_(0),
63 size_(0),
64 treeIndex_(0)
65 {}
66
67 size_type localIndex(size_type i) const
68 {
69 assert(i < size_);
70 return offset_ + i;
71 }
72
73 size_type size() const
74 {
75 return size_;
76 }
77
78 size_type treeIndex() const
79 {
80 return treeIndex_;
81 }
82
83 protected:
84
85 size_type offset() const
86 {
87 return offset_;
88 }
89
90 void setOffset(const size_type offset)
91 {
92 offset_ = offset;
93 }
94
95 void setSize(const size_type size)
96 {
97 size_ = size;
98 }
99
100 void setTreeIndex(size_type treeIndex)
101 {
102 treeIndex_ = treeIndex;
103 }
104
105 private:
106
107 size_type offset_;
108 size_type size_;
109 size_type treeIndex_;
110
111 };
112
113
114 class LeafBasisNode :
115 public BasisNodeMixin,
116 public TypeTree::LeafNode
117 {};
118
119
120
121 template<typename Node, typename Element>
122 class InnerBasisNodeMixin
123 : public BasisNodeMixin
124 {
125 public:
126
127 void bind(const Element& entity)
128 {
129 Node& self = *static_cast<Node*>(this);
130 std::size_t offset = this->offset();
131 Dune::Hybrid::forEach(Dune::range(self.degree()), [&](auto i) {
132 bindTree(self.child(i), entity, offset);
133 offset += self.child(i).size();
134 });
135 this->setSize(offset - this->offset());
136 }
137
138 };
139
140
141
142 template<typename T, std::size_t n>
143 class PowerBasisNode :
144 public InnerBasisNodeMixin<PowerBasisNode<T, n>, typename T::Element>,
145 public TypeTree::PowerNode<T,n>
146 {
147
148 using Node = TypeTree::PowerNode<T,n>;
149
150 public:
151
152 using Element = typename T::Element;
153
154 PowerBasisNode() = default;
155
156 PowerBasisNode(const typename Node::NodeStorage& children) :
157 Node(children)
158 {}
159
160 const Element& element() const
161 {
162 return this->child(Dune::Indices::_0).element();
163 }
164
165 };
166
167
168
169 template<typename T>
170 class DynamicPowerBasisNode :
171 public InnerBasisNodeMixin<DynamicPowerBasisNode<T>, typename T::Element>,
172 public TypeTree::DynamicPowerNode<T>
173 {
174
175 using Node = TypeTree::DynamicPowerNode<T>;
176
177 public:
178
179 using Element = typename T::Element;
180
181 DynamicPowerBasisNode (std::size_t children)
182 : Node(children)
183 {}
184
185 DynamicPowerBasisNode (typename Node::NodeStorage children)
186 : Node(std::move(children))
187 {}
188
189 const Element& element() const
190 {
191 return this->child(0).element();
192 }
193
194 };
195
196
197 template<typename... T>
198 class CompositeBasisNode :
199 public InnerBasisNodeMixin<CompositeBasisNode<T...>, typename TypeTree::CompositeNode<T...>::template Child<0>::Type::Element>,
200 public TypeTree::CompositeNode<T...>
201 {
202
203 using Node = TypeTree::CompositeNode<T...>;
204
205 public:
206
207 using Element = typename Node::template Child<0>::Type::Element;
208
209 CompositeBasisNode() = default;
210
211 CompositeBasisNode(const typename Node::NodeStorage& children) :
212 Node(children)
213 {}
214
215 explicit CompositeBasisNode(const T&... children) :
216 Node(children...)
217 {}
218
219 template<typename... Children>
220 explicit CompositeBasisNode(const std::shared_ptr<Children>&... children) :
221 Node(children...)
222 {}
223
224 const Element& element() const
225 {
226 return this->child(Dune::Indices::_0).element();
227 }
228
229 };
230
231
232 template<typename Tree, typename Entity>
233 void bindTree(Tree& tree, const Entity& entity, std::size_t offset = 0)
234 {
235 Impl::BasisNodeSetupHelper::setOffset(tree, offset);
236 tree.bind(entity);
237 }
238
239 template<typename Tree>
240 void initializeTree(Tree& tree, std::size_t treeIndexOffset = 0)
241 {
242 Dune::TypeTree::forEachNode(tree, [&](auto& node, const auto& treePath) {
243 Impl::BasisNodeSetupHelper::setTreeIndex(node, treeIndexOffset);
244 ++treeIndexOffset;
245 });
246 }
247
248
249 } // namespace Functions
250
251} // namespace Dune
252
253#endif // DUNE_FUNCTIONS_FUNCTIONSPACEBASES_NODES_HH
std::tuple< std::shared_ptr< Children >... > NodeStorage
The type used for storing the children.
Definition: compositenode.hh:36
std::vector< ChildStorageType > NodeStorage
The type used for storing the children.
Definition: dynamicpowernode.hh:67
std::array< std::shared_ptr< T >, k > NodeStorage
The type used for storing the children.
Definition: powernode.hh:77
constexpr index_constant< 0 > _0
Compile time index with value 0.
Definition: indices.hh:52
constexpr void forEach(Range &&range, F &&f)
Range based for loop.
Definition: hybridutilities.hh:257
static constexpr IntegralRange< std::decay_t< T > > range(T &&from, U &&to) noexcept
free standing function for setting up a range based for loop over an integer range for (auto i: range...
Definition: rangeutilities.hh:294
constexpr auto treePath(const T &... t)
Constructs a new HybridTreePath from the given indices.
Definition: treepath.hh:323
void forEachNode(Tree &&tree, PreNodeFunc &&preNodeFunc, LeafNodeFunc &&leafNodeFunc, PostNodeFunc &&postNodeFunc)
Traverse tree and visit each node.
Definition: traversal.hh:259
ImplementationDefined child(Node &&node, Indices... indices)
Extracts the child of a node given by a sequence of compile-time and run-time indices.
Definition: childextraction.hh:127
Dune namespace.
Definition: alignedallocator.hh:13
constexpr std::integral_constant< std::size_t, sizeof...(II)> size(std::integer_sequence< T, II... >)
Return the size of the sequence.
Definition: integersequence.hh:75
STL namespace.
Creative Commons License   |  Legal Statements / Impressum  |  Hosted by TU Dresden & Uni Heidelberg  |  generated with Hugo v0.111.3 (Apr 2, 23:03, 2025)