TRIQS/nda 1.3.0
Multi-dimensional array library for C++
Loading...
Searching...
No Matches
traits.hpp
Go to the documentation of this file.
1// Copyright (c) 2019-2024 Simons Foundation
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0.txt
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14//
15// Authors: Thomas Hahn, Olivier Parcollet, Nils Wentzell
16
22#pragma once
23
24#include <complex>
25#include <cstdint>
26#include <ranges>
27#include <type_traits>
28#include <utility>
29
30namespace nda {
31
50 template <template <typename...> class TMPLT, typename T>
51 struct is_instantiation_of : std::false_type {};
52
53 // Specialization of nda::is_instantiation_of for when it evaluates to true.
54 template <template <typename...> class TMPLT, typename... U>
55 struct is_instantiation_of<TMPLT, TMPLT<U...>> : std::true_type {};
56
58 template <template <typename...> class TMPLT, typename T>
60
62 template <typename T, typename... Ts>
63 static constexpr bool is_any_of = (std::is_same_v<Ts, T> or ... or false);
64
66 template <typename... Ts>
67 static constexpr bool always_true = true;
68
70 template <typename... Ts>
71 static constexpr bool always_false = false;
72
74 template <typename T>
76
78 template <typename S>
79 inline constexpr bool is_scalar_v = std::is_arithmetic_v<std::remove_cvref_t<S>> or is_complex_v<S>;
80
85 template <typename S>
86 inline constexpr bool is_scalar_or_convertible_v = is_scalar_v<S> or std::is_constructible_v<std::complex<double>, S>;
87
92 template <typename S, typename A>
93 inline constexpr bool is_scalar_for_v =
94 (is_scalar_v<typename A::value_type> ? is_scalar_or_convertible_v<S> : std::is_same_v<S, typename A::value_type>);
95
97 template <typename T>
98 inline constexpr bool is_double_or_complex_v = is_complex_v<T> or std::is_same_v<double, std::remove_cvref_t<T>>;
99
101 template <typename T>
103
125 template <typename A>
126 inline constexpr char get_algebra = 'N';
127
128 // Specialization of nda::get_algebra for cvref types.
129 template <typename A>
130 requires(!std::is_same_v<A, std::remove_cvref_t<A>>)
131 inline constexpr char get_algebra<A> = get_algebra<std::remove_cvref_t<A>>;
132
134 template <typename A>
135 requires(std::ranges::contiguous_range<A> or requires { std::declval<A const>().shape(); })
136 constexpr int get_rank = []() {
137 if constexpr (std::ranges::contiguous_range<A>)
138 return 1;
139 else
140 return std::tuple_size_v<std::remove_cvref_t<decltype(std::declval<A const>().shape())>>;
141 }();
142
144 template <typename A>
145 inline constexpr bool is_regular_v = false;
146
147 // Specialization of nda::is_regular_v for cvref types.
148 template <typename A>
149 requires(!std::is_same_v<A, std::remove_cvref_t<A>>)
150 inline constexpr bool is_regular_v<A> = is_regular_v<std::remove_cvref_t<A>>;
151
153 template <typename A>
154 inline constexpr bool is_view_v = false;
155
156 // Specialization of nda::is_view_v for cvref types.
157 template <typename A>
158 requires(!std::is_same_v<A, std::remove_cvref_t<A>>)
159 inline constexpr bool is_view_v<A> = is_view_v<std::remove_cvref_t<A>>;
160
162 template <typename A>
163 inline constexpr bool is_regular_or_view_v = is_regular_v<A> or is_view_v<A>;
164
166 template <typename A>
167 inline constexpr bool is_matrix_or_view_v = is_regular_or_view_v<A> and (get_algebra<A> == 'M') and (get_rank<A> == 2);
168
176 template <typename A>
177 decltype(auto) get_first_element(A const &a) {
178 if constexpr (is_scalar_v<A>) {
179 return a;
180 } else {
181 return [&a]<auto... Is>(std::index_sequence<Is...>) -> decltype(auto) {
182 return a((0 * Is)...); // repeat 0 sizeof...(Is) times
183 }(std::make_index_sequence<get_rank<A>>{});
184 }
185 }
186
191 template <typename A>
192 using get_value_t = std::decay_t<decltype(get_first_element(std::declval<A const>()))>;
193
195 template <typename A0, typename... As>
196 inline constexpr bool have_same_value_type_v = (std::is_same_v<get_value_t<A0>, get_value_t<As>> and ... and true);
197
199 template <typename A0, typename... As>
200 inline constexpr bool have_same_rank_v = ((get_rank<A0> == get_rank<As>)and... and true);
201
222 enum class layout_prop_e : uint64_t {
223 none = 0x0,
224 strided_1d = 0x1,
225 smallest_stride_is_one = 0x2,
226 contiguous = strided_1d | smallest_stride_is_one
227 };
228
238 if (from == layout_prop_e::contiguous) return true;
239 return ((to == layout_prop_e::none) or (to == from));
240 }
241
249 constexpr layout_prop_e operator|(layout_prop_e lhs, layout_prop_e rhs) { return layout_prop_e(uint64_t(lhs) | uint64_t(rhs)); }
250
258 constexpr layout_prop_e operator&(layout_prop_e lhs, layout_prop_e rhs) { return layout_prop_e{uint64_t(lhs) & uint64_t(rhs)}; }
259
266 inline constexpr bool has_strided_1d(layout_prop_e lp) { return uint64_t(lp) & uint64_t(layout_prop_e::strided_1d); }
267
274 inline constexpr bool has_smallest_stride_is_one(layout_prop_e lp) { return uint64_t(lp) & uint64_t(layout_prop_e::smallest_stride_is_one); }
275
282 inline constexpr bool has_contiguous(layout_prop_e lp) { return has_strided_1d(lp) and has_smallest_stride_is_one(lp); }
283
284 // FIXME : I need a NONE for stride_order. For the scalars ...
297 uint64_t stride_order = 0;
298
300 layout_prop_e prop = layout_prop_e::none;
301 };
302
313 if (lhs.stride_order == rhs.stride_order)
314 return layout_info_t{lhs.stride_order, layout_prop_e(uint64_t(lhs.prop) & uint64_t(rhs.prop))};
315 else
316 return layout_info_t{uint64_t(-1), layout_prop_e::none};
317 }
318
320 template <typename A>
322
323 // Specialization of nda::get_layout_info for cvref types.
324 template <typename A>
325 requires(!std::is_same_v<A, std::remove_cvref_t<A>>)
326 inline constexpr layout_info_t get_layout_info<A> = get_layout_info<std::remove_cvref_t<A>>;
327
329 template <typename A>
330 constexpr bool has_contiguous_layout = (has_contiguous(get_layout_info<A>.prop));
331
333 template <typename A>
334 constexpr bool has_layout_strided_1d = (has_strided_1d(get_layout_info<A>.prop));
335
337 template <typename A>
338 constexpr bool has_layout_smallest_stride_is_one = (has_smallest_stride_is_one(get_layout_info<A>.prop));
339
345 long value;
346 };
347
350} // namespace nda
constexpr bool is_regular_v
Constexpr variable that is true if type A is a regular array, i.e. an nda::basic_array.
Definition traits.hpp:145
constexpr char get_algebra
Constexpr variable that specifies the algebra of a type.
Definition traits.hpp:126
constexpr bool is_matrix_or_view_v
Constexpr variable that is true if type A is a regular matrix or a view of a matrix.
Definition traits.hpp:167
constexpr bool have_same_value_type_v
Constexpr variable that is true if all types in As have the same value type as A0.
Definition traits.hpp:196
constexpr int get_rank
Constexpr variable that specifies the rank of an nda::Array or of a contiguous 1-dimensional range.
Definition traits.hpp:136
std::decay_t< decltype(get_first_element(std::declval< A const >()))> get_value_t
Get the value type of an array/view or a scalar type.
Definition traits.hpp:192
constexpr bool have_same_rank_v
Constexpr variable that is true if all types in As have the same rank as A0.
Definition traits.hpp:200
constexpr bool is_view_v
Constexpr variable that is true if type A is a view, i.e. an nda::basic_array_view.
Definition traits.hpp:154
constexpr bool is_regular_or_view_v
Constexpr variable that is true if type A is either a regular array or a view.
Definition traits.hpp:163
decltype(auto) get_first_element(A const &a)
Get the first element of an array/view or simply return the scalar if a scalar is given.
Definition traits.hpp:177
constexpr bool layout_property_compatible(layout_prop_e from, layout_prop_e to)
Checks if two layout properties are compatible with each other.
Definition traits.hpp:237
constexpr bool has_contiguous(layout_prop_e lp)
Checks if a layout property has the contiguous property.
Definition traits.hpp:282
constexpr bool has_layout_smallest_stride_is_one
Constexpr variable that is true if type A has the smallest_stride_is_one nda::layout_prop_e guarantee...
Definition traits.hpp:338
constexpr bool has_strided_1d(layout_prop_e lp)
Checks if a layout property has the strided_1d property.
Definition traits.hpp:266
constexpr bool has_layout_strided_1d
Constexpr variable that is true if type A has the strided_1d nda::layout_prop_e guarantee.
Definition traits.hpp:334
constexpr layout_prop_e operator&(layout_prop_e lhs, layout_prop_e rhs)
Bitwise AND operator for two layout properties.
Definition traits.hpp:258
constexpr layout_prop_e operator|(layout_prop_e lhs, layout_prop_e rhs)
Bitwise OR operator for two layout properties.
Definition traits.hpp:249
constexpr layout_info_t get_layout_info
Constexpr variable that specifies the nda::layout_info_t of type A.
Definition traits.hpp:321
constexpr bool has_smallest_stride_is_one(layout_prop_e lp)
Checks if a layout property has the smallest_stride_is_one property.
Definition traits.hpp:274
constexpr bool has_contiguous_layout
Constexpr variable that is true if type A has the contiguous nda::layout_prop_e guarantee.
Definition traits.hpp:330
layout_prop_e
Compile-time guarantees of the memory layout of an array/view.
Definition traits.hpp:222
constexpr bool is_instantiation_of_v
Constexpr variable that is true if type T is an instantiation of TMPLT (see nda::is_instantiation_of)...
Definition traits.hpp:59
constexpr bool is_complex_v
Constexpr variable that is true if type T is a std::complex type.
Definition traits.hpp:75
constexpr bool is_blas_lapack_v
Alias for nda::is_double_or_complex_v.
Definition traits.hpp:102
static constexpr bool always_false
Constexpr variable that is always false regardless of the types in Ts (used to trigger static_assert)...
Definition traits.hpp:71
constexpr bool is_scalar_for_v
Constexpr variable used to check requirements when initializing an nda::basic_array or nda::basic_arr...
Definition traits.hpp:93
static constexpr bool is_any_of
Constexpr variable that is true if type T is contained in the parameter pack Ts.
Definition traits.hpp:63
constexpr bool is_double_or_complex_v
Constexpr variable that is true if type T is a std::complex type or a double type.
Definition traits.hpp:98
static constexpr bool always_true
Constexpr variable that is always true regardless of the types in Ts.
Definition traits.hpp:67
constexpr bool is_scalar_v
Constexpr variable that is true if type S is a scalar type, i.e. arithmetic or complex.
Definition traits.hpp:79
constexpr bool is_scalar_or_convertible_v
Constexpr variable that is true if type S is a scalar type (see nda::is_scalar_v) or if a std::comple...
Definition traits.hpp:86
A small wrapper around a single long integer to be used as a linear index.
Definition traits.hpp:343
long value
Linear index.
Definition traits.hpp:345
Check if type T is of type TMPLT<...>.
Definition traits.hpp:51
Stores information about the memory layout and the stride order of an array/view.
Definition traits.hpp:295
uint64_t stride_order
Stride order of the array/view.
Definition traits.hpp:297
layout_prop_e prop
Memory layout properties of the array/view.
Definition traits.hpp:300