DDC 0.9.0
Loading...
Searching...
No Matches
discrete_element.hpp
1// Copyright (C) The DDC development team, see COPYRIGHT.md file
2//
3// SPDX-License-Identifier: MIT
4
5#pragma once
6
7#include <array>
8#include <cstddef>
9#include <ostream>
10#include <type_traits>
11#include <utility>
12
13#include <Kokkos_Macros.hpp>
14
15#include "detail/macros.hpp"
16#include "detail/type_seq.hpp"
17#include "detail/utils.hpp"
18
19#include "discrete_vector.hpp"
20
21namespace ddc {
22
23template <class...>
24class DiscreteElement;
25
26template <class T>
27struct is_discrete_element : std::false_type
28{
29};
30
31template <class... Tags>
32struct is_discrete_element<DiscreteElement<Tags...>> : std::true_type
33{
34};
35
36template <class T>
37inline constexpr bool is_discrete_element_v = is_discrete_element<T>::value;
38
39
40namespace detail {
41
42template <class... Tags>
43struct ToTypeSeq<DiscreteElement<Tags...>>
44{
45 using type = TypeSeq<Tags...>;
46};
47
48} // namespace detail
49
50/** A DiscreteCoordElement is a scalar that identifies an element of the discrete dimension
51 */
52using DiscreteElementType = std::size_t;
53
54template <class Tag>
55KOKKOS_FUNCTION constexpr DiscreteElementType const& uid(DiscreteElement<Tag> const& tuple) noexcept
56{
57 return tuple.uid();
58}
59
60template <class Tag>
61KOKKOS_FUNCTION constexpr DiscreteElementType& uid(DiscreteElement<Tag>& tuple) noexcept
62{
63 return tuple.uid();
64}
65
66template <class QueryTag, class... Tags>
67KOKKOS_FUNCTION constexpr DiscreteElementType const& uid(
68 DiscreteElement<Tags...> const& tuple) noexcept
69{
70 return tuple.template uid<QueryTag>();
71}
72
73template <class QueryTag, class... Tags>
74KOKKOS_FUNCTION constexpr DiscreteElementType& uid(DiscreteElement<Tags...>& tuple) noexcept
75{
76 return tuple.template uid<QueryTag>();
77}
78
79template <class QueryTag, class... Tags>
80KOKKOS_FUNCTION constexpr DiscreteElement<QueryTag> select_or(
81 DiscreteElement<Tags...> const& arr,
82 DiscreteElement<QueryTag> const& default_value) noexcept
83{
84 if constexpr (in_tags_v<QueryTag, detail::TypeSeq<Tags...>>) {
85 return DiscreteElement<QueryTag>(arr);
86 } else {
87 return default_value;
88 }
89}
90
91template <class... QueryTags, class... Tags>
92KOKKOS_FUNCTION constexpr DiscreteElement<QueryTags...> select(
93 DiscreteElement<Tags...> const& arr) noexcept
94{
95 return DiscreteElement<QueryTags...>(arr);
96}
97
98template <class... QueryTags, class... Tags>
99KOKKOS_FUNCTION constexpr DiscreteElement<QueryTags...> select(
100 DiscreteElement<Tags...>&& arr) noexcept
101{
102 return DiscreteElement<QueryTags...>(std::move(arr));
103}
104
105/// Returns a reference towards the DiscreteElement that contains the QueryTag
106template <
107 class QueryTag,
108 class HeadDElem,
109 class... TailDElems,
110 std::enable_if_t<
111 is_discrete_element_v<HeadDElem> && (is_discrete_element_v<TailDElems> && ...),
112 int>
113 = 1>
114KOKKOS_FUNCTION constexpr auto const& take(HeadDElem const& head, TailDElems const&... tail)
115{
116 DDC_IF_NVCC_THEN_PUSH_AND_SUPPRESS(implicit_return_from_non_void_function)
117 if constexpr (type_seq_contains_v<detail::TypeSeq<QueryTag>, to_type_seq_t<HeadDElem>>) {
118 static_assert(
119 (!type_seq_contains_v<detail::TypeSeq<QueryTag>, to_type_seq_t<TailDElems>> && ...),
120 "ERROR: tag redundant");
121 return head;
122 } else {
123 static_assert(sizeof...(TailDElems) > 0, "ERROR: tag not found");
124 return take<QueryTag>(tail...);
125 }
126 DDC_IF_NVCC_THEN_POP
127}
128
129namespace detail {
130
131/// Returns a reference to the underlying `std::array`
132template <class... Tags>
133KOKKOS_FUNCTION constexpr std::array<DiscreteElementType, sizeof...(Tags)>& array(
134 DiscreteElement<Tags...>& v) noexcept
135{
136 return v.m_values;
137}
138
139/// Returns a reference to the underlying `std::array`
140template <class... Tags>
141KOKKOS_FUNCTION constexpr std::array<DiscreteElementType, sizeof...(Tags)> const& array(
142 DiscreteElement<Tags...> const& v) noexcept
143{
144 return v.m_values;
145}
146
147} // namespace detail
148
149template <class DDim>
150constexpr DiscreteElement<DDim> create_reference_discrete_element() noexcept
151{
152 return DiscreteElement<DDim>(0);
153}
154
155/** A DiscreteElement identifies an element of the discrete dimension
156 *
157 * Each one is tagged by its associated dimensions.
158 */
159template <class... Tags>
160class DiscreteElement
161{
162 using tags_seq = detail::TypeSeq<Tags...>;
163
164 static_assert(
165 type_seq_is_unique_v<tags_seq>,
166 "The dimensions of a DiscreteElement must be unique");
167
168 friend KOKKOS_FUNCTION constexpr std::array<DiscreteElementType, sizeof...(Tags)>& detail::
169 array<Tags...>(DiscreteElement<Tags...>& v) noexcept;
170
171 friend KOKKOS_FUNCTION constexpr std::array<DiscreteElementType, sizeof...(Tags)> const&
172 detail::array<Tags...>(DiscreteElement<Tags...> const& v) noexcept;
173
174private:
175 std::array<DiscreteElementType, sizeof...(Tags)> m_values;
176
177public:
178 using value_type = DiscreteElementType;
179
180 static KOKKOS_FUNCTION constexpr std::size_t size() noexcept
181 {
182 return sizeof...(Tags);
183 }
184
185public:
186 KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteElement() = default;
187
188 KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteElement(DiscreteElement const&) = default;
189
190 KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteElement(DiscreteElement&&) = default;
191
192 template <class... DElems, class = std::enable_if_t<(is_discrete_element_v<DElems> && ...)>>
193 KOKKOS_FUNCTION constexpr explicit DiscreteElement(DElems const&... delems) noexcept
194 : m_values {take<Tags>(delems...).template uid<Tags>()...}
195 {
196 }
197
198 template <
199 class IntegerType,
200 class = std::enable_if_t<std::is_convertible_v<IntegerType, DiscreteElementType>>>
201 KOKKOS_FUNCTION constexpr explicit DiscreteElement(
202 std::array<IntegerType, sizeof...(Tags)> const& values) noexcept
203 : m_values(
204 detail::convert_array_to<
205 DiscreteElementType>(values, std::make_index_sequence<sizeof...(Tags)>()))
206 {
207 }
208
209 template <
210 class... Params,
211 class = std::enable_if_t<(!is_discrete_element_v<Params> && ...)>,
212 class = std::enable_if_t<(std::is_convertible_v<Params, DiscreteElementType> && ...)>,
213 class = std::enable_if_t<sizeof...(Params) == sizeof...(Tags)>>
214 KOKKOS_FUNCTION constexpr explicit DiscreteElement(Params const&... params) noexcept
215 : m_values {static_cast<DiscreteElementType>(params)...}
216 {
217 }
218
219 KOKKOS_DEFAULTED_FUNCTION ~DiscreteElement() = default;
220
221 KOKKOS_DEFAULTED_FUNCTION DiscreteElement& operator=(DiscreteElement const& other) = default;
222
223 KOKKOS_DEFAULTED_FUNCTION DiscreteElement& operator=(DiscreteElement&& other) = default;
224
225 template <class QueryTag>
226 KOKKOS_FUNCTION constexpr value_type& uid() noexcept
227 {
228 static_assert(in_tags_v<QueryTag, tags_seq>, "requested Tag absent from DiscreteElement");
229 return m_values[type_seq_rank_v<QueryTag, tags_seq>];
230 }
231
232 template <class QueryTag>
233 KOKKOS_FUNCTION constexpr value_type const& uid() const noexcept
234 {
235 static_assert(in_tags_v<QueryTag, tags_seq>, "requested Tag absent from DiscreteElement");
236 return m_values[type_seq_rank_v<QueryTag, tags_seq>];
237 }
238
239 template <std::size_t N = sizeof...(Tags)>
240 KOKKOS_FUNCTION constexpr std::enable_if_t<N == 1, value_type&> uid() noexcept
241 {
242 return m_values[0];
243 }
244
245 template <std::size_t N = sizeof...(Tags)>
246 KOKKOS_FUNCTION constexpr std::enable_if_t<N == 1, value_type const&> uid() const noexcept
247 {
248 return m_values[0];
249 }
250
251 template <std::size_t N = sizeof...(Tags), class = std::enable_if_t<N == 1>>
252 KOKKOS_FUNCTION constexpr DiscreteElement& operator++()
253 {
254 ++m_values[0];
255 return *this;
256 }
257
258 template <std::size_t N = sizeof...(Tags), class = std::enable_if_t<N == 1>>
259 KOKKOS_FUNCTION constexpr DiscreteElement operator++(int)
260 {
261 DiscreteElement const tmp = *this;
262 ++m_values[0];
263 return tmp;
264 }
265
266 template <std::size_t N = sizeof...(Tags), class = std::enable_if_t<N == 1>>
267 KOKKOS_FUNCTION constexpr DiscreteElement& operator--()
268 {
269 --m_values[0];
270 return *this;
271 }
272
273 template <std::size_t N = sizeof...(Tags), class = std::enable_if_t<N == 1>>
274 KOKKOS_FUNCTION constexpr DiscreteElement operator--(int)
275 {
276 DiscreteElement const tmp = *this;
277 --m_values[0];
278 return tmp;
279 }
280
281 template <class... OTags>
282 KOKKOS_FUNCTION constexpr DiscreteElement& operator+=(DiscreteVector<OTags...> const& rhs)
283 {
284 static_assert(((type_seq_contains_v<detail::TypeSeq<OTags>, tags_seq>) && ...));
285 ((m_values[type_seq_rank_v<OTags, tags_seq>] += rhs.template get<OTags>()), ...);
286 return *this;
287 }
288
289 template <
290 class IntegralType,
291 std::size_t N = sizeof...(Tags),
292 class = std::enable_if_t<N == 1>,
293 class = std::enable_if_t<std::is_integral_v<IntegralType>>>
294 KOKKOS_FUNCTION constexpr DiscreteElement& operator+=(IntegralType const& rhs)
295 {
296 m_values[0] += rhs;
297 return *this;
298 }
299
300 template <class... OTags>
301 KOKKOS_FUNCTION constexpr DiscreteElement& operator-=(DiscreteVector<OTags...> const& rhs)
302 {
303 static_assert(((type_seq_contains_v<detail::TypeSeq<OTags>, tags_seq>) && ...));
304 ((m_values[type_seq_rank_v<OTags, tags_seq>] -= rhs.template get<OTags>()), ...);
305 return *this;
306 }
307
308 template <
309 class IntegralType,
310 std::size_t N = sizeof...(Tags),
311 class = std::enable_if_t<N == 1>,
312 class = std::enable_if_t<std::is_integral_v<IntegralType>>>
313 KOKKOS_FUNCTION constexpr DiscreteElement& operator-=(IntegralType const& rhs)
314 {
315 m_values[0] -= rhs;
316 return *this;
317 }
318};
319
320inline std::ostream& operator<<(std::ostream& out, DiscreteElement<> const&)
321{
322 out << "()";
323 return out;
324}
325
326template <class Head, class... Tags>
327std::ostream& operator<<(std::ostream& out, DiscreteElement<Head, Tags...> const& arr)
328{
329 out << "(";
330 out << uid<Head>(arr);
331 ((out << ", " << uid<Tags>(arr)), ...);
332 out << ")";
333 return out;
334}
335
336
337template <class... Tags, class... OTags>
338KOKKOS_FUNCTION constexpr bool operator==(
339 DiscreteElement<Tags...> const& lhs,
340 DiscreteElement<OTags...> const& rhs) noexcept
341{
342 return ((lhs.template uid<Tags>() == rhs.template uid<Tags>()) && ...);
343}
344
345#if !defined(__cpp_impl_three_way_comparison) || __cpp_impl_three_way_comparison < 201902L
346// In C++20, `a!=b` shall be automatically translated by the compiler to `!(a==b)`
347template <class... Tags, class... OTags>
348KOKKOS_FUNCTION constexpr bool operator!=(
349 DiscreteElement<Tags...> const& lhs,
350 DiscreteElement<OTags...> const& rhs) noexcept
351{
352 return !(lhs == rhs);
353}
354#endif
355
356template <class Tag>
357KOKKOS_FUNCTION constexpr bool operator<(
358 DiscreteElement<Tag> const& lhs,
359 DiscreteElement<Tag> const& rhs)
360{
361 return lhs.uid() < rhs.uid();
362}
363
364template <class Tag>
365KOKKOS_FUNCTION constexpr bool operator<=(
366 DiscreteElement<Tag> const& lhs,
367 DiscreteElement<Tag> const& rhs)
368{
369 return lhs.uid() <= rhs.uid();
370}
371
372template <class Tag>
373KOKKOS_FUNCTION constexpr bool operator>(
374 DiscreteElement<Tag> const& lhs,
375 DiscreteElement<Tag> const& rhs)
376{
377 return lhs.uid() > rhs.uid();
378}
379
380template <class Tag>
381KOKKOS_FUNCTION constexpr bool operator>=(
382 DiscreteElement<Tag> const& lhs,
383 DiscreteElement<Tag> const& rhs)
384{
385 return lhs.uid() >= rhs.uid();
386}
387
388/// right external binary operators: +, -
389
390template <class... Tags, class... OTags>
391KOKKOS_FUNCTION constexpr DiscreteElement<Tags...> operator+(
392 DiscreteElement<Tags...> const& lhs,
393 DiscreteVector<OTags...> const& rhs)
394{
395 using detail::TypeSeq;
396 static_assert(((type_seq_contains_v<TypeSeq<OTags>, TypeSeq<Tags...>>) && ...));
397 DiscreteElement<Tags...> result(lhs);
398 result += rhs;
399 return result;
400}
401
402template <
403 class Tag,
404 class IntegralType,
405 class = std::enable_if_t<std::is_integral_v<IntegralType>>,
406 class = std::enable_if_t<!is_discrete_vector_v<IntegralType>>>
407KOKKOS_FUNCTION constexpr DiscreteElement<Tag> operator+(
408 DiscreteElement<Tag> const& lhs,
409 IntegralType const& rhs)
410{
411 DiscreteElement<Tag> result(lhs);
412 result += rhs;
413 return result;
414}
415
416template <class... Tags, class... OTags>
417KOKKOS_FUNCTION constexpr DiscreteElement<Tags...> operator-(
418 DiscreteElement<Tags...> const& lhs,
419 DiscreteVector<OTags...> const& rhs)
420{
421 using detail::TypeSeq;
422 static_assert(((type_seq_contains_v<TypeSeq<OTags>, TypeSeq<Tags...>>) && ...));
423 DiscreteElement<Tags...> result(lhs);
424 result -= rhs;
425 return result;
426}
427
428template <
429 class Tag,
430 class IntegralType,
431 class = std::enable_if_t<std::is_integral_v<IntegralType>>,
432 class = std::enable_if_t<!is_discrete_vector_v<IntegralType>>>
433KOKKOS_FUNCTION constexpr DiscreteElement<Tag> operator-(
434 DiscreteElement<Tag> const& lhs,
435 IntegralType const& rhs)
436{
437 DiscreteElement<Tag> result(lhs);
438 result -= rhs;
439 return result;
440}
441
442/// binary operator: -
443
444template <class... Tags, class... OTags>
445KOKKOS_FUNCTION constexpr DiscreteVector<Tags...> operator-(
446 DiscreteElement<Tags...> const& lhs,
447 DiscreteElement<OTags...> const& rhs)
448{
449 static_assert(type_seq_same_v<detail::TypeSeq<Tags...>, detail::TypeSeq<OTags...>>);
450 return DiscreteVector<Tags...>((uid<Tags>(lhs) - uid<Tags>(rhs))...);
451}
452
453} // namespace ddc
KOKKOS_FUNCTION constexpr DiscreteElement & operator+=(DiscreteVector< OTags... > const &rhs)
KOKKOS_FUNCTION constexpr value_type const & uid() const noexcept
KOKKOS_FUNCTION constexpr value_type & uid() noexcept
KOKKOS_DEFAULTED_FUNCTION ~DiscreteElement()=default
KOKKOS_FUNCTION constexpr DiscreteElement(std::array< IntegerType, sizeof...(Tags)> const &values) noexcept
KOKKOS_FUNCTION constexpr DiscreteElement & operator++()
KOKKOS_FUNCTION constexpr DiscreteElement & operator-=(IntegralType const &rhs)
KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteElement(DiscreteElement const &)=default
KOKKOS_FUNCTION constexpr DiscreteElement(DElems const &... delems) noexcept
KOKKOS_FUNCTION constexpr DiscreteElement & operator-=(DiscreteVector< OTags... > const &rhs)
KOKKOS_FUNCTION constexpr DiscreteElement operator++(int)
KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteElement()=default
KOKKOS_FUNCTION constexpr DiscreteElement & operator--()
KOKKOS_DEFAULTED_FUNCTION DiscreteElement & operator=(DiscreteElement const &other)=default
KOKKOS_FUNCTION constexpr DiscreteElement operator--(int)
KOKKOS_FUNCTION constexpr std::enable_if_t< N==1, value_type & > uid() noexcept
static KOKKOS_FUNCTION constexpr std::size_t size() noexcept
KOKKOS_FUNCTION constexpr std::enable_if_t< N==1, value_type const & > uid() const noexcept
KOKKOS_FUNCTION constexpr DiscreteElement & operator+=(IntegralType const &rhs)
KOKKOS_DEFAULTED_FUNCTION DiscreteElement & operator=(DiscreteElement &&other)=default
KOKKOS_FUNCTION constexpr DiscreteElement(Params const &... params) noexcept
KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteElement(DiscreteElement &&)=default
KOKKOS_FUNCTION constexpr bool operator!=(DiscreteVector< OTags... > const &rhs) const noexcept
The top-level namespace of DDC.
KOKKOS_FUNCTION constexpr bool operator<(DiscreteElement< Tag > const &lhs, DiscreteElement< Tag > const &rhs)
KOKKOS_FUNCTION constexpr bool operator==(DiscreteElement< Tags... > const &lhs, DiscreteElement< OTags... > const &rhs) noexcept
KOKKOS_FUNCTION constexpr DiscreteElement< QueryTags... > select(DiscreteElement< Tags... > &&arr) noexcept
constexpr DiscreteElement< DDim > create_reference_discrete_element() noexcept
KOKKOS_FUNCTION constexpr DiscreteElement< QueryTags... > select(DiscreteElement< Tags... > const &arr) noexcept
KOKKOS_FUNCTION constexpr auto const & take(HeadDElem const &head, TailDElems const &... tail)
Returns a reference towards the DiscreteElement that contains the QueryTag.
KOKKOS_FUNCTION constexpr DiscreteElement< Tag > operator-(DiscreteElement< Tag > const &lhs, IntegralType const &rhs)
KOKKOS_FUNCTION constexpr DiscreteVector< Tags... > operator-(DiscreteElement< Tags... > const &lhs, DiscreteElement< OTags... > const &rhs)
binary operator: -
KOKKOS_FUNCTION constexpr DiscreteElement< QueryTag > select_or(DiscreteElement< Tags... > const &arr, DiscreteElement< QueryTag > const &default_value) noexcept
KOKKOS_FUNCTION constexpr DiscreteElementType const & uid(DiscreteElement< Tag > const &tuple) noexcept
KOKKOS_FUNCTION constexpr DiscreteElement< Tags... > operator+(DiscreteElement< Tags... > const &lhs, DiscreteVector< OTags... > const &rhs)
right external binary operators: +, -
constexpr bool is_discrete_element_v
KOKKOS_FUNCTION constexpr bool operator>=(DiscreteElement< Tag > const &lhs, DiscreteElement< Tag > const &rhs)
KOKKOS_FUNCTION constexpr bool operator<=(DiscreteElement< Tag > const &lhs, DiscreteElement< Tag > const &rhs)
KOKKOS_FUNCTION constexpr DiscreteElementType & uid(DiscreteElement< Tags... > &tuple) noexcept
KOKKOS_FUNCTION constexpr bool operator>(DiscreteElement< Tag > const &lhs, DiscreteElement< Tag > const &rhs)
KOKKOS_FUNCTION constexpr DiscreteElement< Tag > operator+(DiscreteElement< Tag > const &lhs, IntegralType const &rhs)
KOKKOS_FUNCTION constexpr DiscreteElementType & uid(DiscreteElement< Tag > &tuple) noexcept
KOKKOS_FUNCTION constexpr DiscreteElement< Tags... > operator-(DiscreteElement< Tags... > const &lhs, DiscreteVector< OTags... > const &rhs)
KOKKOS_FUNCTION constexpr DiscreteElementType const & uid(DiscreteElement< Tags... > const &tuple) noexcept