Loading [MathJax]/extensions/tex2jax.js

Dune TypeTree (unstable)

childextraction.hh
1// -*- tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
2// vi: set et ts=8 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_CHILDEXTRACTION_HH
7#define DUNE_TYPETREE_CHILDEXTRACTION_HH
8
9#include <type_traits>
10#include <utility>
11
12#include <dune/common/concept.hh>
13#include <dune/common/documentation.hh>
14#include <dune/common/typetraits.hh>
15#include <dune/common/shared_ptr.hh>
16
17#include <dune/typetree/treepath.hh>
18
19
20namespace Dune {
21 namespace TypeTree {
22
27
28#ifndef DOXYGEN
29
30 namespace Impl {
31
32 // check at run time whether index is a valid child index
33 template <class Node, class Index>
34 std::true_type checkChildIndex (Node const& node, Index i)
35 {
36 assert(std::size_t(i) < node.degree() && "Child index out of range");
37 return {};
38 }
39
40 // check at compile time whether index is a valid index
41 template <class Node, std::size_t i>
42 std::bool_constant<(i < Node::degree())> checkChildIndex (Node const& node, index_constant<i>)
43 {
44 static_assert(i < Node::degree(), "Child index out of range");
45 return {};
46 }
47
48 // finally return the node itself if no further indices are provided. Break condition
49 // for the recursion over the node childs.
50 template<class Node>
51 decltype(auto) childImpl (Node&& node)
52 {
53 return std::forward<Node>(node);
54 }
55
56 template<class NodePtr>
57 auto childStorageImpl (NodePtr&& nodePtr)
58 {
59 return std::forward<NodePtr>(nodePtr);
60 }
61
62 // recursively call `node.child(...)` with the given indices
63 template<class Node, class I0, class... I>
64 decltype(auto) childImpl (Node&& node, I0 i0, [[maybe_unused]] I... i)
65 {
66 auto valid = checkChildIndex(node,i0);
67 if constexpr (valid)
68 return childImpl(node.child(i0),i...);
69 else
70 return;
71 }
72
73 // recursively call `node.childStorage(...)` with the given indices
74 template<class NodePtr, class I0, class... I>
75 decltype(auto) childStorageImpl (NodePtr&& nodePtr, I0 i0, [[maybe_unused]] I... i)
76 {
77 auto valid = checkChildIndex(*nodePtr,i0);
78 if constexpr (valid)
79 return childStorageImpl(nodePtr->childStorage(i0),i...);
80 else
81 return;
82 }
83
84 // forward to the impl methods by extracting the indices from the treepath
85 template<class Node, class... Indices, std::size_t... i>
86 decltype(auto) child (Node&& node, [[maybe_unused]] HybridTreePath<Indices...> tp, std::index_sequence<i...>)
87 {
88 return childImpl(std::forward<Node>(node),treePathEntry<i>(tp)...);
89 }
90
91 // forward to the impl methods by extracting the indices from the treepath
92 template<class NodePtr, class... Indices, std::size_t... i>
93 decltype(auto) childStorage (NodePtr&& nodePtr, [[maybe_unused]] HybridTreePath<Indices...> tp, std::index_sequence<i...>)
94 {
95 return childStorageImpl(std::forward<NodePtr>(nodePtr),treePathEntry<i>(tp)...);
96 }
97
98 } // end namespace Impl
99
100#endif // DOXYGEN
101
103
125 template<typename Node, typename... Indices>
126#ifdef DOXYGEN
127 ImplementationDefined child (Node&& node, Indices... indices)
128#else
129 decltype(auto) child (Node&& node, Indices... indices)
130#endif
131 {
132 return Impl::childImpl(std::forward<Node>(node),indices...);
133 }
134
135 template<typename Node, typename... Indices>
136#ifdef DOXYGEN
137 ImplementationDefined childStorage (Node&& node, Indices... indices)
138#else
139 auto childStorage (Node&& node, Indices... indices)
140#endif
141 {
142 static_assert(sizeof...(Indices) > 0, "childStorage() cannot be called with an empty list of child indices");
143 return Impl::childStorageImpl(&node,indices...);
144 }
145
147
170 template<typename Node, typename... Indices>
171#ifdef DOXYGEN
172 ImplementationDefined child (Node&& node, HybridTreePath<Indices...> treePath)
173#else
174 decltype(auto) child (Node&& node, HybridTreePath<Indices...> tp)
175#endif
176 {
177 return Impl::child(std::forward<Node>(node),tp,std::index_sequence_for<Indices...>{});
178 }
179
180 template<typename Node, typename... Indices>
181#ifdef DOXYGEN
182 ImplementationDefined child (Node&& node, HybridTreePath<Indices...> treePath)
183#else
184 auto childStorage (Node&& node, HybridTreePath<Indices...> tp)
185#endif
186 {
187 static_assert(sizeof...(Indices) > 0, "childStorage() cannot be called with an empty TreePath");
188 return Impl::childStorage(&node,tp,std::index_sequence_for<Indices...>{});
189 }
190
191
192#ifndef DOXYGEN
193
194 namespace impl {
195
196 template<typename T>
197 struct filter_void
198 {
199 using type = T;
200 };
201
202 template<>
203 struct filter_void<void>
204 {};
205
206 template<typename Node, std::size_t... indices>
207 struct _Child
208 : public filter_void<std::decay_t<decltype(child(std::declval<Node>(),index_constant<indices>{}...))>>
209 {};
210
211 }
212
213#endif // DOXYGEN
214
216
223 template<typename Node, std::size_t... indices>
224 using Child = typename impl::_Child<Node,indices...>::type;
225
226
227#ifndef DOXYGEN
228
229 namespace impl {
230
231 template<typename Node, typename TreePath>
232 struct _ChildForTreePath
233 {
234 using type = typename std::decay<decltype(child(std::declval<Node>(),std::declval<TreePath>()))>::type;
235 };
236
237 }
238
239#endif // DOXYGEN
240
242
250 template<typename Node, typename TreePath>
251 using ChildForTreePath = typename impl::_ChildForTreePath<Node,TreePath>::type;
252
253
254#ifndef DOXYGEN
255
256 namespace impl {
257
258 // By default, types are flat indices if they are integral
259 template<typename T>
260 struct _is_flat_index
261 {
262 using type = std::is_integral<T>;
263 };
264
265 // And so is any index_constant
266 template<std::size_t i>
267 struct _is_flat_index<index_constant<i>>
268 {
269 using type = std::true_type;
270 };
271
272 }
273
274#endif // DOXYGEN
275
277 /*
278 * This type trait can be used to check whether T is a flat index (i.e. either `std::size_t`
279 * or `index_constant`). The type trait normalizes T before doing the check, so it will also
280 * work correctly for references and cv-qualified types.
281 */
282 template<typename T>
283 using is_flat_index = typename impl::_is_flat_index<std::decay_t<T>>::type;
284
285#ifndef DOXYGEN
286
287 namespace impl {
288
289 // helper function for check in member child() functions that tolerates being passed something that
290 // isn't a TreePath. It will just return 0 in that case
291
292 template<typename T>
293 constexpr typename std::enable_if<
295 bool
296 >::type
297 _non_empty_tree_path (T)
298 {
299 return false;
300 }
301
302 template<typename T>
303 constexpr typename std::enable_if<
305 bool
306 >::type
307 _non_empty_tree_path (T t)
308 {
309 return treePathSize(t) > 0;
310 }
311
312 }
313
314#endif // DOXYGEN
315
317
318 } // namespace TypeTree
319} //namespace Dune
320
321#endif // DUNE_TYPETREE_CHILDEXTRACTION_HH
A hybrid version of TreePath that supports both compile time and run time indices.
Definition: treepath.hh:155
std::size_t degree(const Node &node)
Returns the degree of node as run time information.
Definition: nodeinterface.hh:79
constexpr std::size_t treePathSize(const HybridTreePath< T... > &)
Returns the size (number of components) of the given HybridTreePath.
Definition: treepath.hh:331
constexpr auto treePath(const T &... t)
Constructs a new HybridTreePath from the given indices.
Definition: treepath.hh:323
typename impl::_is_flat_index< std::decay_t< T > >::type is_flat_index
Type trait that determines whether T is a flat index in the context of child extraction.
Definition: childextraction.hh:283
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
typename impl::_ChildForTreePath< Node, TreePath >::type ChildForTreePath
Template alias for the type of a child node given by a TreePath or a HybridTreePath type.
Definition: childextraction.hh:251
ImplementationDefined child(Node &&node, HybridTreePath< Indices... > treePath)
Extracts the child of a node given by a HybridTreePath object.
Definition: childextraction.hh:172
Creative Commons License   |  Legal Statements / Impressum  |  Hosted by TU Dresden & Uni Heidelberg  |  generated with Hugo v0.111.3 (Apr 3, 22:46, 2025)