Loading [MathJax]/extensions/tex2jax.js

Dune TypeTree (2.10)

transformation.hh
1// -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
2// vi: set et ts=4 sw=2 sts=2:
3// SPDX-FileCopyrightInfo: Copyright © DUNE Project contributors, see file LICENSE.md in module root
4// SPDX-License-Identifier: LGPL-3.0-or-later OR LicenseRef-GPL-2.0-only-with-PDELab-exception
5
6#ifndef DUNE_TYPETREE_TRANSFORMATION_HH
7#define DUNE_TYPETREE_TRANSFORMATION_HH
8
9#include <array>
10#include <tuple>
11#include <memory>
12#include <utility>
13
14#include <dune/common/hybridutilities.hh>
15#include <dune/common/exceptions.hh>
16#include <dune/common/typetraits.hh>
17#include <dune/typetree/typetraits.hh>
18#include <dune/typetree/nodeinterface.hh>
19#include <dune/typetree/nodetags.hh>
20#include <dune/typetree/utility.hh>
21
22
23namespace Dune {
24 namespace TypeTree {
25
31#ifdef DOXYGEN
32
34
53 template<typename SourceNode, typename Transformation, typename Tag>
54 void registerNodeTransformation(SourceNode*, Transformation*, Tag*);
55
56#else // DOXYGEN
57
68 template<typename S, typename T, typename Tag>
69 struct LookupNodeTransformation
70 {
71
72 typedef decltype(registerNodeTransformation(declptr<S>(),declptr<T>(),declptr<Tag>())) lookup_type;
73
74 typedef typename evaluate_if_meta_function<
75 lookup_type
76 >::type type;
77
78 static_assert((!std::is_same<type,void>::value), "Unable to find valid transformation descriptor");
79 };
80
81#endif // DOXYGEN
82
83
85
94 template<typename SourceTree, typename Transformation, typename Tag = StartTag, bool recursive = true>
96 {
97
98#ifndef DOXYGEN
99
100 typedef typename LookupNodeTransformation<SourceTree,Transformation,typename SourceTree::ImplementationTag>::type NodeTransformation;
101
102 // the type of the new tree that will result from this transformation
103 typedef typename TransformTree<SourceTree,Transformation,NodeTag<SourceTree>,NodeTransformation::recursive>::transformed_type transformed_type;
104
105 // the storage type of the new tree that will result from this transformation
106 typedef typename TransformTree<SourceTree,Transformation,NodeTag<SourceTree>,NodeTransformation::recursive>::transformed_storage_type transformed_storage_type;
107
108#endif // DOXYGEN
109
111 typedef transformed_type type;
112
113 typedef type Type;
114
116 static transformed_type transform(const SourceTree& s, const Transformation& t = Transformation())
117 {
118 return TransformTree<SourceTree,Transformation,NodeTag<SourceTree>,NodeTransformation::recursive>::transform(s,t);
119 }
120
122 static transformed_type transform(const SourceTree& s, Transformation& t)
123 {
124 return TransformTree<SourceTree,Transformation,NodeTag<SourceTree>,NodeTransformation::recursive>::transform(s,t);
125 }
126
128 static transformed_type transform(std::shared_ptr<const SourceTree> sp, const Transformation& t = Transformation())
129 {
130 return TransformTree<SourceTree,Transformation,NodeTag<SourceTree>,NodeTransformation::recursive>::transform(sp,t);
131 }
132
134 static transformed_type transform(std::shared_ptr<const SourceTree> sp, Transformation& t)
135 {
136 return TransformTree<SourceTree,Transformation,NodeTag<SourceTree>,NodeTransformation::recursive>::transform(sp,t);
137 }
138
141 static transformed_storage_type transform_storage(std::shared_ptr<const SourceTree> sp, const Transformation& t = Transformation())
142 {
144 }
145
148 static transformed_storage_type transform_storage(std::shared_ptr<const SourceTree> sp, Transformation& t)
149 {
151 }
152
153
154 };
155
156#ifndef DOXYGEN // internal per-node implementations of the transformation algorithm
157
158 // handle a leaf node - this is easy
159 template<typename S, typename T, bool recursive>
160 struct TransformTree<S,T,LeafNodeTag,recursive>
161 {
162 // get transformed type from specification
163 typedef typename LookupNodeTransformation<S,T,ImplementationTag<S>>::type NodeTransformation;
164
165 typedef typename NodeTransformation::transformed_type transformed_type;
166 typedef typename NodeTransformation::transformed_storage_type transformed_storage_type;
167
168 // delegate instance transformation to per-node specification
169 static transformed_type transform(const S& s, T& t)
170 {
171 return NodeTransformation::transform(s,t);
172 }
173
174 // delegate instance transformation to per-node specification
175 static transformed_type transform(const S& s, const T& t)
176 {
177 return NodeTransformation::transform(s,t);
178 }
179
180 // delegate instance transformation to per-node specification
181 static transformed_type transform(std::shared_ptr<const S> sp, T& t)
182 {
183 return NodeTransformation::transform(sp,t);
184 }
185
186 // delegate instance transformation to per-node specification
187 static transformed_type transform(std::shared_ptr<const S> sp, const T& t)
188 {
189 return NodeTransformation::transform(sp,t);
190 }
191
192 static transformed_storage_type transform_storage(std::shared_ptr<const S> sp, T& t)
193 {
194 return NodeTransformation::transform_storage(sp,t);
195 }
196
197 static transformed_storage_type transform_storage(std::shared_ptr<const S> sp, const T& t)
198 {
199 return NodeTransformation::transform_storage(sp,t);
200 }
201
202 };
203
204
205 // common implementation for non-recursive transformation of non-leaf nodes
206 template<typename S, typename T>
207 struct TransformTreeNonRecursive
208 {
209 // get transformed type from specification
210 typedef typename LookupNodeTransformation<S,T,ImplementationTag<S>>::type NodeTransformation;
211
212 typedef typename NodeTransformation::transformed_type transformed_type;
213 typedef typename NodeTransformation::transformed_storage_type transformed_storage_type;
214
215 // delegate instance transformation to per-node specification
216 static transformed_type transform(const S& s, T& t)
217 {
218 return NodeTransformation::transform(s,t);
219 }
220
221 // delegate instance transformation to per-node specification
222 static transformed_type transform(const S& s, const T& t)
223 {
224 return NodeTransformation::transform(s,t);
225 }
226
227 // delegate instance transformation to per-node specification
228 static transformed_type transform(std::shared_ptr<const S> sp, T& t)
229 {
230 return NodeTransformation::transform(sp,t);
231 }
232
233 // delegate instance transformation to per-node specification
234 static transformed_type transform(std::shared_ptr<const S> sp, const T& t)
235 {
236 return NodeTransformation::transform(sp,t);
237 }
238
239 static transformed_storage_type transform_storage(std::shared_ptr<const S> sp, T& t)
240 {
241 return NodeTransformation::transform_storage(sp,t);
242 }
243
244 static transformed_storage_type transform_storage(std::shared_ptr<const S> sp, const T& t)
245 {
246 return NodeTransformation::transform_storage(sp,t);
247 }
248
249 };
250
251
252 namespace Impl {
253
254 // Helper class to handle recursive power nodes
255 template<class Source, class Transformation, class Tag>
256 class RecursivePowerTransformTree
257 {
258 // We only know two types of tags!
259 static_assert(std::is_same_v<Tag,PowerNodeTag> or std::is_same_v<Tag,DynamicPowerNodeTag>);
260
261 using ChildType = typename Source::ChildType;
262
263 // in case degree is dynamic, provid a vector correctly initialized
264 template<class NodeStorage>
265 static auto node_storage_provider(const std::size_t& degree)
266 {
267 return std::vector<NodeStorage>(degree);
268 }
269
270 // in case degree is static, provid an array
271 template<class NodeStorage, class StaticIndex>
272 static auto node_storage_provider(StaticIndex)
273 {
274 return std::array<NodeStorage,std::size_t(StaticIndex{})>();
275 }
276
277 public:
278 // get transformed type from specification
279 // Handling this transformation in a way that makes the per-node specification easy to write
280 // is a little involved:
281 // The problem is that the transformed power node must be parameterized on the transformed child
282 // type. So we need to transform the child type and pass the transformed child type to an inner
283 // template of the node transformation struct called result (see example of such a specification
284 // further down).
285 using NodeTransformation = typename LookupNodeTransformation<Source,Transformation,ImplementationTag<Source>>::type;
286 using ChildNodeTransformation = typename LookupNodeTransformation<ChildType,Transformation,ImplementationTag<ChildType>>::type;
287
288 private:
289 // Since every child is same type, is enough to get transformation once
290 using ChildTreeTransformation = TransformTree<ChildType,
291 Transformation,
292 NodeTag<ChildType>,
293 ChildNodeTransformation::recursive>;
294
295 // Get transformed type of children
296 using transformed_child_type = typename ChildTreeTransformation::transformed_type;
297 using transformed_child_storage_type = typename ChildTreeTransformation::transformed_storage_type;
298 public:
299 // Apply transformation from children to current node
300 using transformed_type = typename NodeTransformation::template result<transformed_child_type>::type;
301 using transformed_storage_type = typename NodeTransformation::template result<transformed_child_type>::storage_type;
302
303 // Transform an instance of source tree.
304 static transformed_type transform(const Source& source, Transformation& transformation)
305 {
306 auto children_storage = node_storage_provider<std::shared_ptr<transformed_child_type>>(source.degree());
307 for (std::size_t k = 0; k < source.degree(); ++k) {
308 children_storage[k] = ChildTreeTransformation::transform_storage(source.childStorage(k),transformation);
309 }
310 return NodeTransformation::transform(source,transformation,children_storage);
311 }
312
313 // Transform an instance of source tree.
314 static transformed_type transform(const Source& source, const Transformation& transformation)
315 {
316 auto children_storage = node_storage_provider<std::shared_ptr<transformed_child_type>>(source.degree());
317 for (std::size_t k = 0; k < source.degree(); ++k) {
318 children_storage[k] = ChildTreeTransformation::transform_storage(source.childStorage(k),transformation);
319 }
320 return NodeTransformation::transform(source,transformation,children_storage);
321 }
322
323 // Transform an instance of source tree.
324 static transformed_type transform(std::shared_ptr<const Source> source_ptr, Transformation& transformation)
325 {
326 auto children_storage = node_storage_provider<std::shared_ptr<transformed_child_type>>(source_ptr->degree());
327 for (std::size_t k = 0; k < source_ptr->degree(); ++k) {
328 children_storage[k] = ChildTreeTransformation::transform_storage(source_ptr->childStorage(k),transformation);
329 }
330 return NodeTransformation::transform(source_ptr,transformation,children_storage);
331 }
332
333 // Transform an instance of source tree.
334 static transformed_type transform(std::shared_ptr<const Source> source_ptr, const Transformation& transformation)
335 {
336 auto children_storage = node_storage_provider<std::shared_ptr<transformed_child_type>>(source_ptr->degree());
337 for (std::size_t k = 0; k < source_ptr->degree(); ++k) {
338 children_storage[k] = ChildTreeTransformation::transform_storage(source_ptr->childStorage(k),transformation);
339 }
340 return NodeTransformation::transform(source_ptr,transformation,children_storage);
341 }
342
343 // Transform an instance of source tree ptr.
344 static transformed_storage_type transform_storage(std::shared_ptr<const Source> source_ptr, Transformation& transformation)
345 {
346 auto children_storage = node_storage_provider<transformed_child_storage_type>(source_ptr->degree());
347 for (std::size_t k = 0; k < source_ptr->degree(); ++k) {
348 children_storage[k] = ChildTreeTransformation::transform_storage(source_ptr->childStorage(k),transformation);
349 }
350 return NodeTransformation::transform_storage(source_ptr,transformation,children_storage);
351 }
352
353 // Transform an instance of source tree ptr.
354 static transformed_storage_type transform_storage(std::shared_ptr<const Source> source_ptr, const Transformation& transformation)
355 {
356 auto children_storage = node_storage_provider<transformed_child_storage_type>(source_ptr->degree());
357 for (std::size_t k = 0; k < source_ptr->degree(); ++k) {
358 children_storage[k] = ChildTreeTransformation::transform_storage(source_ptr->childStorage(k),transformation);
359 }
360 return NodeTransformation::transform_storage(source_ptr,transformation,children_storage);
361 }
362
363 };
364 } // namespace Impl
365
366 // Recursive version of the PowerNode transformation for static nodes.
367 template<typename Source, typename Transformation>
368 struct TransformTree<Source,Transformation,PowerNodeTag,true>
369 : public Impl::RecursivePowerTransformTree<Source,Transformation,PowerNodeTag>
370 {};
371
372 // Recursive version of the DynamicPowerNode transformation for static nodes.
373 template<typename Source, typename Transformation>
374 struct TransformTree<Source,Transformation,DynamicPowerNodeTag,true>
375 : public Impl::RecursivePowerTransformTree<Source,Transformation,DynamicPowerNodeTag>
376 {};
377
378 // non-recursive version of the PowerNode transformation.
379 template<typename S, typename T>
380 struct TransformTree<S,T,PowerNodeTag,false>
381 : public TransformTreeNonRecursive<S,T>
382 {};
383
384 // non-recursive version of the DynamicPowerNodeTag transformation.
385 template<typename S, typename T>
386 struct TransformTree<S,T,DynamicPowerNodeTag,false>
387 : public TransformTreeNonRecursive<S,T>
388 {};
389
390 // helper struct that does the actual transformation for a composite node. We need this additional struct
391 // to extract the template argument list with the types of all children from the node, which we cannot do
392 // directly in the transformation<> template, as the type passed to transformation<> will usually be a
393 // derived type and will normally have more template arguments than just the children. This declaration
394 // just introduces the type of the helper struct, we always instantiate the specialization defined below;
395 template<typename S, typename Children, typename T>
396 struct transform_composite_node;
397
398 // specialized version of the helper struct which extracts the template argument list with the children from
399 // its second template parameter, which has to be CompositeNode::ChildTypes. Apart from that, the struct is
400 // similar to the one for a PowerNode, but it obviously delegates transformation of the children to the TMP.
401 template<typename S, typename T, typename... C>
402 struct transform_composite_node<S,std::tuple<C...>,T>
403 {
404
405 // transformed type, using the same nested struct trick as the PowerNode
406 typedef ImplementationTag<S> Tag;
407 typedef typename LookupNodeTransformation<S,T,Tag>::type NodeTransformation;
408 typedef typename NodeTransformation::template result<typename TransformTree<C,
409 T,
410 NodeTag<C>,
411 LookupNodeTransformation<C,T,ImplementationTag<C>>::type::recursive
412 >::transformed_type...
413 >::type transformed_type;
414
415 typedef typename NodeTransformation::template result<typename TransformTree<C,
416 T,
417 NodeTag<C>,
418 LookupNodeTransformation<C,T,ImplementationTag<C>>::type::recursive
419 >::transformed_type...
420 >::storage_type transformed_storage_type;
421
422 // Retrieve the transformation descriptor for the child with index i.
423 // This little helper improves really improves the readability of the
424 // transformation functions.
425 template<std::size_t i>
426 struct ChildTransformation
427 : public TransformTree<typename S::template Child<i>::Type,
428 T,
429 NodeTag<typename S::template Child<i>::Type>,
430 LookupNodeTransformation<
431 typename S::template Child<i>::Type,
432 T,
433 ImplementationTag<typename S::template Child<i>::Type>
434 >::type::recursive
435 >
436 {};
437
438 template<std::size_t i, typename Tuple, typename Value>
439 static void setElement(Tuple& tuple, Value&& value)
440 {
441 std::get<i>(tuple) = std::forward<Value>(value);
442 }
443
444 template<typename Trafo, std::size_t... i>
445 static transformed_type transform(const S& s, Trafo&& t, std::index_sequence<i...> indices)
446 {
447 std::tuple<typename ChildTransformation<i>::transformed_storage_type...> storage;
448 Dune::Hybrid::Impl::evaluateFoldExpression<int>({(setElement<i>(storage, ChildTransformation<i>::transform_storage(s.template childStorage<i>(), std::forward<Trafo>(t))),0)...});
449 return NodeTransformation::transform(s, std::forward<Trafo>(t), std::get<i>(storage)...);
450 }
451
452 template<typename Trafo, std::size_t... i>
453 static transformed_storage_type transform_storage(std::shared_ptr<const S> sp, Trafo&& t, std::index_sequence<i...> indices)
454 {
455 std::tuple<typename ChildTransformation<i>::transformed_storage_type...> storage;
456 Dune::Hybrid::Impl::evaluateFoldExpression<int>({(setElement<i>(storage, ChildTransformation<i>::transform_storage(sp->template childStorage<i>(), std::forward<Trafo>(t))),0)...});
457 return NodeTransformation::transform_storage(sp, std::forward<Trafo>(t), std::get<i>(storage)...);
458 }
459 };
460
461
462 // the specialization of transformation<> for the CompositeNode. This just extracts the
463 // CompositeNode::ChildTypes member and forwards to the helper struct
464 template<typename S, typename T>
465 struct TransformTree<S,T,CompositeNodeTag,true>
466 {
467
468 private:
469
470 typedef typename S::ChildTypes ChildTypes;
471
472 static auto child_indices()
473 {
474 return std::make_index_sequence<S::degree()>();
475 }
476
477 public:
478
479 typedef typename transform_composite_node<S,ChildTypes,T>::transformed_type transformed_type;
480 typedef typename transform_composite_node<S,ChildTypes,T>::transformed_storage_type transformed_storage_type;
481
482 static transformed_type transform(const S& s, T& t)
483 {
484 return transform_composite_node<S,ChildTypes,T>::transform(s,t,child_indices());
485 }
486
487 static transformed_type transform(const S& s, const T& t)
488 {
489 return transform_composite_node<S,ChildTypes,T>::transform(s,t,child_indices());
490 }
491
492 static transformed_storage_type transform_storage(std::shared_ptr<const S> sp, T& t)
493 {
494 return transform_composite_node<S,ChildTypes,T>::transform_storage(sp,t,child_indices());
495 }
496
497 static transformed_storage_type transform_storage(std::shared_ptr<const S> sp, const T& t)
498 {
499 return transform_composite_node<S,ChildTypes,T>::transform_storage(sp,t,child_indices());
500 }
501
502 };
503
504 // non-recursive version of the CompositeNode transformation.
505 template<typename S, typename T>
506 struct TransformTree<S,T,CompositeNodeTag,false>
507 : public TransformTreeNonRecursive<S,T>
508 {};
509
510#endif // DOXYGEN
511
513
514 } // namespace TypeTree
515} //namespace Dune
516
517#endif // DUNE_TYPETREE_TRANSFORMATION_HH
std::size_t degree(const Node &node)
Returns the degree of node as run time information.
Definition: nodeinterface.hh:79
void registerNodeTransformation(SourceNode *, Transformation *, Tag *)
Register transformation descriptor to transform SourceNode with Transformation.
Transform a TypeTree.
Definition: transformation.hh:96
static transformed_type transform(std::shared_ptr< const SourceTree > sp, Transformation &t)
Apply transformation to an existing tree s.
Definition: transformation.hh:134
static transformed_type transform(std::shared_ptr< const SourceTree > sp, const Transformation &t=Transformation())
Apply transformation to an existing tree s.
Definition: transformation.hh:128
static transformed_type transform(const SourceTree &s, Transformation &t)
Apply transformation to an existing tree s.
Definition: transformation.hh:122
transformed_type type
The type of the transformed tree.
Definition: transformation.hh:111
static transformed_type transform(const SourceTree &s, const Transformation &t=Transformation())
Apply transformation to an existing tree s.
Definition: transformation.hh:116
static transformed_storage_type transform_storage(std::shared_ptr< const SourceTree > sp, const Transformation &t=Transformation())
Definition: transformation.hh:141
static transformed_storage_type transform_storage(std::shared_ptr< const SourceTree > sp, Transformation &t)
Definition: transformation.hh:148
Meta function that evaluates its argument iff it inherits from meta_function.
Definition: typetraits.hh:142
Creative Commons License   |  Legal Statements / Impressum  |  Hosted by TU Dresden & Uni Heidelberg  |  generated with Hugo v0.111.3 (Apr 3, 22:46, 2025)