Loading [MathJax]/extensions/tex2jax.js

DUNE PDELab (unstable)

hybridutilities.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: LicenseRef-GPL-2.0-only-with-DUNE-exception
5#ifndef DUNE_COMMON_HYBRIDUTILITIES_HH
6#define DUNE_COMMON_HYBRIDUTILITIES_HH
7
8#include <tuple>
9#include <utility>
10
14#include <dune/common/indices.hh>
15#include <dune/common/integersequence.hh>
17
18
19
20namespace Dune {
21namespace Hybrid {
22
23namespace Impl {
24
25 // Try if std::tuple_size is implemented for class
26 template<class T>
27 constexpr auto size(const T&, const PriorityTag<2>&)
28 -> decltype(std::integral_constant<std::size_t,std::tuple_size<T>::value>())
29 {
30 return {};
31 }
32
33 // Try if there's a static constexpr size() method
34 template<class T>
35 constexpr auto size(const T&, const PriorityTag<1>&)
36 -> decltype(std::integral_constant<std::size_t,T::size()>())
37 {
38 return {};
39 }
40
41 // As a last resort try if there's a non-static size()
42 template<class T>
43 constexpr auto size(const T& t, const PriorityTag<0>&)
44 {
45 return t.size();
46 }
47
48} // namespace Impl
49
50
51
73template<class T>
74constexpr auto size(const T& t)
75{
76 return Impl::size(t, PriorityTag<42>());
77}
78
79
80
81namespace Impl {
82
83 template<class Container, class Index,
84 std::enable_if_t<IsTuple<std::decay_t<Container>>::value, int> = 0>
85 constexpr decltype(auto) elementAt(Container&& c, Index&&, PriorityTag<2>)
86 {
87 return std::get<std::decay_t<Index>::value>(c);
88 }
89
90 template<class T, T... t, class Index>
91 constexpr decltype(auto) elementAt(std::integer_sequence<T, t...> c, Index, PriorityTag<1>)
92 {
93 return Dune::get<Index::value>(c);
94 }
95
96 template<class Container, class Index>
97 constexpr decltype(auto) elementAt(Container&& c, Index&& i, PriorityTag<0>)
98 {
99 return c[i];
100 }
101
102} // namespace Impl
103
104
105
126template<class Container, class Index>
127constexpr decltype(auto) elementAt(Container&& c, Index&& i)
128{
129 return Impl::elementAt(std::forward<Container>(c), std::forward<Index>(i), PriorityTag<42>());
130}
131
132
133
134namespace Impl {
135
136 template<class Begin, class End,
137 std::enable_if_t<IsIntegralConstant<Begin>::value and IsIntegralConstant<End>::value, int> = 0>
138 constexpr auto integralRange(const Begin& /*begin*/, const End& /*end*/, const PriorityTag<1>&)
139 {
140 static_assert(Begin::value <= End::value, "You cannot create an integralRange where end<begin");
142 }
143
144 template<class Begin, class End>
145 constexpr auto integralRange(const Begin& begin, const End& end, const PriorityTag<0>&)
146 {
147 assert(begin<=end && "You cannot create an integralRange where end<begin");
148 return Dune::IntegralRange<End>(begin, end);
149 }
150
151} // namespace Impl
152
153
154
172template<class Begin, class End>
173constexpr auto integralRange(const Begin& begin, const End& end)
174{
175 return Impl::integralRange(begin, end, PriorityTag<42>());
176}
177
191template<class End>
192constexpr auto integralRange(const End& end)
193{
195}
196
197
198
199namespace Impl {
200
201 template<class T>
202 constexpr void evaluateFoldExpression(std::initializer_list<T>&&)
203 {}
204
205 template<class Range, class F, class Index, Index... i>
206 constexpr void forEachIndex(Range&& range, F&& f, std::integer_sequence<Index, i...>)
207 {
208 evaluateFoldExpression<int>({(f(Hybrid::elementAt(range, std::integral_constant<Index,i>())), 0)...});
209 }
210
211 template<class F, class Index, Index... i>
212 constexpr void forEach(std::integer_sequence<Index, i...> /*range*/, F&& f, PriorityTag<2>)
213 {
214 evaluateFoldExpression<int>({(f(std::integral_constant<Index,i>()), 0)...});
215 }
216
217
218 template<class Range, class F,
219 std::enable_if_t<IsIntegralConstant<decltype(Hybrid::size(std::declval<Range>()))>::value, int> = 0>
220 constexpr void forEach(Range&& range, F&& f, PriorityTag<1>)
221 {
222 auto size = Hybrid::size(range);
223 auto indices = std::make_index_sequence<size>();
224 (forEachIndex)(std::forward<Range>(range), std::forward<F>(f), indices);
225 }
226
227 template<class Range, class F>
228 constexpr void forEach(Range&& range, F&& f, PriorityTag<0>)
229 {
230 for(auto&& e : range)
231 f(e);
232 }
233
234} // namespace Impl
235
236
237
256template<class Range, class F>
257constexpr void forEach(Range&& range, F&& f)
258{
259 Impl::forEach(std::forward<Range>(range), std::forward<F>(f), PriorityTag<42>());
260}
261
262
263
279template<class Range, class T, class F>
280constexpr T accumulate(Range&& range, T value, F&& f)
281{
282 forEach(std::forward<Range>(range), [&](auto&& entry) {
283 value = f(value, entry);
284 });
285 return value;
286}
287
288
289
290namespace Impl {
291
292 struct Id {
293 template<class T>
294 constexpr T operator()(T&& x) const {
295 return std::forward<T>(x);
296 }
297 };
298
299 template<class IfFunc, class ElseFunc>
300 constexpr decltype(auto) ifElse(std::true_type, IfFunc&& ifFunc, ElseFunc&& /*elseFunc*/)
301 {
302 return ifFunc(Id{});
303 }
304
305 template<class IfFunc, class ElseFunc>
306 constexpr decltype(auto) ifElse(std::false_type, IfFunc&& /*ifFunc*/, ElseFunc&& elseFunc)
307 {
308 return elseFunc(Id{});
309 }
310
311 template<class IfFunc, class ElseFunc>
312 decltype(auto) ifElse(const bool& condition, IfFunc&& ifFunc, ElseFunc&& elseFunc)
313 {
314 if (condition)
315 return ifFunc(Id{});
316 else
317 return elseFunc(Id{});
318 }
319
320} // namespace Impl
321
322
323
344template<class Condition, class IfFunc, class ElseFunc>
345decltype(auto) ifElse(const Condition& condition, IfFunc&& ifFunc, ElseFunc&& elseFunc)
346{
347 return Impl::ifElse(condition, std::forward<IfFunc>(ifFunc), std::forward<ElseFunc>(elseFunc));
348}
349
357template<class Condition, class IfFunc>
358void ifElse(const Condition& condition, IfFunc&& ifFunc)
359{
360 ifElse(condition, std::forward<IfFunc>(ifFunc), [](auto&&) {});
361}
362
363
364
365namespace Impl {
366
367 struct Max {
368 template<class... Args>
369 constexpr decltype(auto) operator()(Args&&... args) const
370 {
371 using T = std::common_type_t<Args...>;
372 return std::max({T(args)...});
373 }
374 };
375
376 struct Min {
377 template<class... Args>
378 constexpr decltype(auto) operator()(Args&&... args) const
379 {
380 using T = std::common_type_t<Args...>;
381 return std::min({T(args)...});
382 }
383 };
384
385} // namespace Impl
386
387
418template<class Functor>
420
421 static_assert(std::is_default_constructible_v<Functor>,
422 "Operator in integral expressions shall be constexpr default constructible");
423
424 inline static constexpr Functor _functor = Functor{};
425
426public:
427
437 template<class... Args>
438 constexpr decltype(auto) operator()(const Args&... args) const
439 {
440 if constexpr (std::conjunction_v<IsCompileTimeConstant<Args>...>)
441 {
442 constexpr auto result = _functor(Args::value...);
443 // apply functor on integral constant arguments and return an integral constant of the result
444 // this is guaranteed to be evaluated at compile-time
445 return std::integral_constant<std::remove_cv_t<decltype(result)>,result>{};
446 } else {
447 // apply functor directly on arguments and return the result of the functor
448 // (integral constants are likely to be casted to underlying type)
449 // this not is guaranteed to be evaluated at compile-time although is possible if expression is constexpr
450 return _functor(args...);
451 }
452 }
453};
454
459template<class Functor>
460constexpr HybridFunctor<Functor> hybridFunctor(const Functor&)
461{
462 return {};
463}
464
485inline constexpr auto max = hybridFunctor(Impl::Max{});
486
507inline constexpr auto min = hybridFunctor(Impl::Min{});
508
529inline constexpr auto plus = hybridFunctor(std::plus<>{});
530
551inline constexpr auto minus = hybridFunctor(std::minus<>{});
552
573inline constexpr auto equal_to = hybridFunctor(std::equal_to<>{});
574
575
587template<class T1, class T2>
588[[deprecated("Use Hybrid::equal_to instead!")]] constexpr auto equals(T1&& t1, T2&& t2){
589 return equal_to(std::forward<T1>(t1), std::forward<T2>(t2));
590}
591
592
593namespace Impl {
594
595 // This overload is selected if the passed value is already a compile time constant.
596 template<class Result, class T, T t0, T... tt, class ValueType, ValueType value, class Branches, class ElseBranch>
597 constexpr Result switchCases(std::integer_sequence<T, t0, tt...>, const std::integral_constant<ValueType, value>& /*value*/, Branches&& branches, ElseBranch&& elseBranch)
598 {
599 // In case we pass a value known at compile time, we no longer have to do
600 // a dynamic to static dispatch via recursion. The only thing that's left
601 // is to check if the value is contained in the passed range.
602 // If this is true, we have to pass it to the branches callback
603 // as an appropriate integral_constant type. Otherwise we have to
604 // execute the else callback.
605 if constexpr (((t0 == value) || ... || (tt == value)))
606 return branches(std::integral_constant<T, value>{});
607 else
608 return elseBranch();
609 }
610
611 // This overload is selected if the passed value is dynamic.
612 template<class Result, class T, class Value, class Branches, class ElseBranch>
613 constexpr Result switchCases(std::integer_sequence<T>, const Value& /*value*/, Branches&& /*branches*/, ElseBranch&& elseBranch)
614 {
615 return elseBranch();
616 }
617
618 template<class Result, class T, T t0, T... tt, class Value, class Branches, class ElseBranch>
619 constexpr Result switchCases(std::integer_sequence<T, t0, tt...>, const Value& value, Branches&& branches, ElseBranch&& elseBranch)
620 {
621 if (t0 == value)
622 return branches(std::integral_constant<T, t0>());
623 else
624 return Impl::switchCases<Result>(std::integer_sequence<T, tt...>(), value, branches, elseBranch);
625 }
626
627 // This overload is selected if the range of cases is an IntegralRange
628 template <class Result, class T, class Value, class Branches, class ElseBranch>
629 constexpr Result switchCases(IntegralRange<T> range, const Value& value, Branches&& branches, ElseBranch&& elseBranch)
630 {
631 return range.contains(value) ? branches(T(value)) : elseBranch();
632 }
633
634 // This overload is selected if the range of cases is a StaticIntegralRange
635 template <class Result, class T, T to, T from, class Value, class Branches, class ElseBranch>
636 constexpr Result switchCases(StaticIntegralRange<T, to, from> range, const Value& value, Branches&& branches, ElseBranch&& elseBranch)
637 {
638 using seq = typename decltype(range)::integer_sequence;
639 return Impl::switchCases<Result>(seq{}, value, branches, elseBranch);
640 }
641
642} // namespace Impl
643
644
645
673template<class Cases, class Value, class Branches, class ElseBranch>
674constexpr decltype(auto) switchCases(const Cases& cases, const Value& value, Branches&& branches, ElseBranch&& elseBranch)
675{
676 return Impl::switchCases<decltype(elseBranch())>(cases, value, std::forward<Branches>(branches), std::forward<ElseBranch>(elseBranch));
677}
678
698template<class Cases, class Value, class Branches>
699constexpr void switchCases(const Cases& cases, const Value& value, Branches&& branches)
700{
701 Impl::switchCases<void>(cases, value, std::forward<Branches>(branches),
702 []{ assert(false && "value not found in range"); });
703}
704
723template <class T, class Value, class Branches>
724constexpr void switchCases(IntegralRange<T> range, const Value& value, Branches&& branches)
725{
726 assert(range.contains(value) && "value not found in range");
727 branches(T(value));
728}
729
730} // namespace Hybrid
731} // namespace Dune
732
733
734#endif // #ifndef DUNE_COMMON_HYBRIDUTILITIES_HH
Adapter of a hybrid functor that maintains results hybrid.
Definition: hybridutilities.hh:419
dynamic integer range for use in range-based for loops
Definition: rangeutilities.hh:177
static integer range for use in range-based for loops
Definition: rangeutilities.hh:230
Traits for type conversions and type information.
Implements a vector constructed from a given type representing a field and a compile-time given size.
constexpr index_constant< 0 > _0
Compile time index with value 0.
Definition: indices.hh:52
void ifElse(const Condition &condition, IfFunc &&ifFunc)
A conditional expression.
Definition: hybridutilities.hh:358
constexpr auto size(const T &t)
Size query.
Definition: hybridutilities.hh:74
constexpr auto integralRange(const End &end)
Create an integral range starting from 0.
Definition: hybridutilities.hh:192
constexpr auto minus
Function object for performing subtraction.
Definition: hybridutilities.hh:551
constexpr auto equals(T1 &&t1, T2 &&t2)
Equality comparison.
Definition: hybridutilities.hh:588
constexpr void forEach(Range &&range, F &&f)
Range based for loop.
Definition: hybridutilities.hh:257
constexpr auto equal_to
Function object for performing equality comparison.
Definition: hybridutilities.hh:573
constexpr auto max
Function object that returns the greater of the given values.
Definition: hybridutilities.hh:485
constexpr auto plus
Function object for performing addition.
Definition: hybridutilities.hh:529
constexpr auto min
Function object that returns the smaller of the given values.
Definition: hybridutilities.hh:507
constexpr decltype(auto) elementAt(Container &&c, Index &&i)
Get element at given position from container.
Definition: hybridutilities.hh:127
constexpr T accumulate(Range &&range, T value, F &&f)
Accumulate values.
Definition: hybridutilities.hh:280
constexpr void switchCases(IntegralRange< T > range, const Value &value, Branches &&branches)
Switch statement.
Definition: hybridutilities.hh:724
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
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
Utilities for reduction like operations on ranges.
Check if T is an std::integral_constant<I, i>
Definition: typetraits.hh:384
Helper class for tagging priorities.
Definition: typeutilities.hh:73
Utilities for type computations, constraining overloads, ...
Creative Commons License   |  Legal Statements / Impressum  |  Hosted by TU Dresden & Uni Heidelberg  |  generated with Hugo v0.111.3 (Apr 5, 23:02, 2025)