Dune TypeTree (unstable)

traversal.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_TRAVERSAL_HH
7#define DUNE_TYPETREE_TRAVERSAL_HH
8
9#include <utility>
10
11#include <dune/common/hybridutilities.hh>
12#include <dune/common/std/type_traits.hh>
13
14#include <dune/typetree/childextraction.hh>
15#include <dune/typetree/treepath.hh>
16#include <dune/typetree/visitor.hh>
17
18namespace Dune {
19 namespace TypeTree {
20
26#ifndef DOXYGEN
28 struct NoOp
29 {
30 template<class... T>
31 constexpr void operator()(T&&...) const { /* do nothing */ }
32 };
33#endif
34
35 namespace Detail {
36
37 // SFINAE template check that Tree has a degree() function and a child() function accepting integer indices
38 template<class Tree>
39 using DynamicTraversalConcept = decltype((
40 std::declval<Tree>().degree(),
41 std::declval<Tree>().child(0u)
42 ));
43
44 // SFINAE template check that Tree has static (constexpr) function Tree::degree()
45 template<class Tree>
46 using StaticTraversalConcept = decltype((
47 std::integral_constant<std::size_t, Tree::degree()>{}
48 ));
49
50
51 template<class Tree, TreePathType::Type pathType, class Prefix,
52 std::enable_if_t<Tree::isLeaf, int> = 0>
53 constexpr auto leafTreePathTuple(Prefix prefix)
54 {
55 return std::make_tuple(prefix);
56 }
57
58 template<class Tree, TreePathType::Type pathType, class Prefix,
59 std::enable_if_t<not Tree::isLeaf, int> = 0>
60 constexpr auto leafTreePathTuple(Prefix prefix);
61
62 template<class Tree, TreePathType::Type pathType, class Prefix, std::size_t... indices,
63 std::enable_if_t<(Tree::isComposite or (Tree::isPower and (pathType!=TreePathType::dynamic))), int> = 0>
64 constexpr auto leafTreePathTuple(Prefix prefix, std::index_sequence<indices...>)
65 {
66 return std::tuple_cat(Detail::leafTreePathTuple<TypeTree::Child<Tree,indices>, pathType>(Dune::TypeTree::push_back(prefix, Dune::index_constant<indices>{}))...);
67 }
68
69 template<class Tree, TreePathType::Type pathType, class Prefix, std::size_t... indices,
70 std::enable_if_t<(Tree::isPower and (pathType==TreePathType::dynamic)), int> = 0>
71 constexpr auto leafTreePathTuple(Prefix prefix, std::index_sequence<indices...>)
72 {
73 return std::tuple_cat(Detail::leafTreePathTuple<TypeTree::Child<Tree,indices>, pathType>(Dune::TypeTree::push_back(prefix, indices))...);
74 }
75
76 template<class Tree, TreePathType::Type pathType, class Prefix,
77 std::enable_if_t<not Tree::isLeaf, int>>
78 constexpr auto leafTreePathTuple(Prefix prefix)
79 {
80 return Detail::leafTreePathTuple<Tree, pathType>(prefix, std::make_index_sequence<Tree::degree()>{});
81 }
82
83 /* The signature is the same as for the public applyToTree
84 * function in Dune::Typetree, despite the additionally passed
85 * treePath argument. The path passed here is associated to
86 * the tree and the relative paths of the children (wrt. to tree)
87 * are appended to this. Hence the behavior of the public function
88 * is resembled by passing an empty treePath.
89 */
90
91 /*
92 * This is the overload for leaf traversal
93 */
94 template<class T, class TreePath, class V,
95 std::enable_if_t<std::decay_t<T>::isLeaf, int> = 0>
96 void applyToTree(T&& tree, TreePath treePath, V&& visitor)
97 {
98 visitor.leaf(tree, treePath);
99 }
100
101 /*
102 * This is the general overload doing child traversal.
103 */
104 template<class T, class TreePath, class V,
105 std::enable_if_t<not std::decay_t<T>::isLeaf, int> = 0>
106 void applyToTree(T&& tree, TreePath treePath, V&& visitor)
107 {
108 using Tree = std::remove_reference_t<T>;
109 using Visitor = std::remove_reference_t<V>;
110 visitor.pre(tree, treePath);
111
112 // check which type of traversal is supported by the tree
113 using allowDynamicTraversal = Dune::Std::is_detected<DynamicTraversalConcept,Tree>;
114 using allowStaticTraversal = Dune::Std::is_detected<StaticTraversalConcept,Tree>;
115
116 // the tree must support either dynamic or static traversal
117 static_assert(allowDynamicTraversal::value || allowStaticTraversal::value);
118
119 // the visitor may specify preferred dynamic traversal
120 using preferDynamicTraversal = std::bool_constant<Visitor::treePathType == TreePathType::dynamic>;
121
122 // create a dynamic or static index range
123 auto indices = [&]{
124 if constexpr(preferDynamicTraversal::value && allowDynamicTraversal::value)
125 return Dune::range(std::size_t(tree.degree()));
126 else
127 return Dune::range(tree.degree());
128 }();
129
130 if constexpr(allowDynamicTraversal::value || allowStaticTraversal::value) {
131 Hybrid::forEach(indices, [&](auto i) {
132 auto&& child = tree.child(i);
133 using Child = std::decay_t<decltype(child)>;
134
135 visitor.beforeChild(tree, child, treePath, i);
136
137 // This requires that visitor.in(...) can always be instantiated,
138 // even if there's a single child only.
139 if (i>0)
140 visitor.in(tree, treePath);
141
142 constexpr bool visitChild = Visitor::template VisitChild<Tree,Child,TreePath>::value;
143 if constexpr(visitChild) {
144 auto childTreePath = Dune::TypeTree::push_back(treePath, i);
145 applyToTree(child, childTreePath, visitor);
146 }
147
148 visitor.afterChild(tree, child, treePath, i);
149 });
150 }
151 visitor.post(tree, treePath);
152 }
153
154 /* Traverse tree and visit each node. The signature is the same
155 * as for the public forEachNode function in Dune::Typtree,
156 * despite the additionally passed treePath argument. The path
157 * passed here is associated to the tree and the relative
158 * paths of the children (wrt. to tree) are appended to this.
159 * Hence the behavior of the public function is resembled
160 * by passing an empty treePath.
161 */
162 template<class T, class TreePath, class PreFunc, class LeafFunc, class PostFunc>
163 void forEachNode(T&& tree, TreePath treePath, PreFunc&& preFunc, LeafFunc&& leafFunc, PostFunc&& postFunc)
164 {
165 using Tree = std::decay_t<T>;
166 if constexpr(Tree::isLeaf) {
167 leafFunc(tree, treePath);
168 } else {
169 preFunc(tree, treePath);
170
171 // check which type of traversal is supported by the tree, prefer dynamic traversal
172 using allowDynamicTraversal = Dune::Std::is_detected<DynamicTraversalConcept,Tree>;
173 using allowStaticTraversal = Dune::Std::is_detected<StaticTraversalConcept,Tree>;
174
175 // the tree must support either dynamic or static traversal
176 static_assert(allowDynamicTraversal::value || allowStaticTraversal::value);
177
178 if constexpr(allowDynamicTraversal::value) {
179 // Specialization for dynamic traversal
180 for (std::size_t i = 0; i < tree.degree(); ++i) {
181 auto childTreePath = Dune::TypeTree::push_back(treePath, i);
182 forEachNode(tree.child(i), childTreePath, preFunc, leafFunc, postFunc);
183 }
184 } else if constexpr(allowStaticTraversal::value) {
185 // Specialization for static traversal
186 auto indices = std::make_index_sequence<Tree::degree()>{};
187 Hybrid::forEach(indices, [&](auto i) {
188 auto childTreePath = Dune::TypeTree::push_back(treePath, i);
189 forEachNode(tree.child(i), childTreePath, preFunc, leafFunc, postFunc);
190 });
191 }
192 postFunc(tree, treePath);
193 }
194 }
195
196 } // namespace Detail
197
198
199 // ********************************************************************************
200 // Public Interface
201 // ********************************************************************************
202
216 template<class Tree, TreePathType::Type pathType=TreePathType::dynamic>
217 constexpr auto leafTreePathTuple()
218 {
219 return Detail::leafTreePathTuple<std::decay_t<Tree>, pathType>(hybridTreePath());
220 }
221
223
237 template<typename Tree, typename Visitor>
238 void applyToTree(Tree&& tree, Visitor&& visitor)
239 {
240 Detail::applyToTree(tree, hybridTreePath(), visitor);
241 }
242
258 template<class Tree, class PreNodeFunc, class LeafNodeFunc, class PostNodeFunc>
259 void forEachNode(Tree&& tree, PreNodeFunc&& preNodeFunc, LeafNodeFunc&& leafNodeFunc, PostNodeFunc&& postNodeFunc)
260 {
261 Detail::forEachNode(tree, hybridTreePath(), preNodeFunc, leafNodeFunc, postNodeFunc);
262 }
263
273 template<class Tree, class NodeFunc>
274 void forEachNode(Tree&& tree, NodeFunc&& nodeFunc)
275 {
276 Detail::forEachNode(tree, hybridTreePath(), nodeFunc, nodeFunc, NoOp{});
277 }
278
288 template<class Tree, class LeafFunc>
289 void forEachLeafNode(Tree&& tree, LeafFunc&& leafFunc)
290 {
291 Detail::forEachNode(tree, hybridTreePath(), NoOp{}, leafFunc, NoOp{});
292 }
293
295
296 } // namespace TypeTree
297} //namespace Dune
298
299#endif // DUNE_TYPETREE_TRAVERSAL_HH
std::size_t degree(const Node &node)
Returns the degree of node as run time information.
Definition: nodeinterface.hh:79
constexpr HybridTreePath< T..., std::size_t > push_back(const HybridTreePath< T... > &tp, std::size_t i)
Appends a run time index to a HybridTreePath.
Definition: treepath.hh:413
constexpr auto hybridTreePath(const T &... t)
Constructs a new HybridTreePath from the given indices.
Definition: treepath.hh:309
constexpr auto treePath(const T &... t)
Constructs a new HybridTreePath from the given indices.
Definition: treepath.hh:323
void forEachNode(Tree &&tree, NodeFunc &&nodeFunc)
Traverse tree and visit each node.
Definition: traversal.hh:274
constexpr auto leafTreePathTuple()
Create tuple of tree paths to leafs.
Definition: traversal.hh:217
void forEachLeafNode(Tree &&tree, LeafFunc &&leafFunc)
Traverse tree and visit each leaf node.
Definition: traversal.hh:289
void applyToTree(Tree &&tree, Visitor &&visitor)
Apply visitor to TypeTree.
Definition: traversal.hh:238
typename impl::_Child< Node, indices... >::type Child
Template alias for the type of a child node given by a list of child indices.
Definition: childextraction.hh:224
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
Creative Commons License   |  Legal Statements / Impressum  |  Hosted by TU Dresden & Uni Heidelberg  |  generated with Hugo v0.111.3 (Apr 3, 22:46, 2025)