13#include <Kokkos_Assert.hpp>
14#include <Kokkos_Macros.hpp>
16#include "detail/type_seq.hpp"
18#include "discrete_element.hpp"
19#include "discrete_vector.hpp"
26template <
class... DDims>
34template <
class... Tags>
45template <
class... Tags>
48 using type = TypeSeq<Tags...>;
51template <
class T,
class U>
54template <
class... DDims,
class... ODDims>
55struct RebindDomain<
DiscreteDomain<DDims...>, detail::TypeSeq<ODDims...>>
62template <
class... DDims>
69 type_seq_is_unique_v<detail::TypeSeq<DDims...>>,
70 "The dimensions of a DiscreteDomain must be unique");
73 using discrete_element_type = DiscreteElement<DDims...>;
78 DiscreteElement<DDims...> m_element_begin;
80 DiscreteElement<DDims...> m_element_end;
85 return sizeof...(DDims);
91 template <
class... DDoms,
class = std::enable_if_t<(is_discrete_domain_v<DDoms> && ...)>>
93 : m_element_begin(domains.front()...)
94 , m_element_end((domains.front() + domains.extents())...)
99
100
101
103 discrete_element_type
const& element_begin,
104 discrete_vector_type
const& size)
105 : m_element_begin(element_begin)
106 , m_element_end(element_begin + size)
120 template <
class... ODims>
123 if (
empty() && other.empty()) {
126 return m_element_begin == other.m_element_begin && m_element_end == other.m_element_end;
129#if !defined(__cpp_impl_three_way_comparison) || __cpp_impl_three_way_comparison < 201902L
131 template <
class...
ODims>
134 return !(*
this ==
other);
140 return (1UL * ... * extent<DDims>().value());
145 return m_element_end - m_element_begin;
148 template <
class QueryDDim>
151 return DiscreteElement<QueryDDim>(m_element_end)
152 - DiscreteElement<QueryDDim>(m_element_begin);
157 return m_element_begin;
162 return discrete_element_type((DiscreteElement<DDims>(m_element_end) - 1)...);
186 discrete_vector_type n1,
187 discrete_vector_type n2)
const
195 return m_element_begin + dvect;
198 template <
class... ODDims>
202 ((DiscreteElement<ODDims>(m_element_begin)
203 <= DiscreteElement<ODDims>(odomain.m_element_begin))
206 ((DiscreteElement<ODDims>(m_element_end)
207 >= DiscreteElement<ODDims>(odomain.m_element_end))
212 DiscreteElement<DDims...>((select_or<DDims>(
213 odomain.m_element_begin,
214 DiscreteElement<DDims>(m_element_begin)))...),
216 (select_or<DDims>(oextents,
DiscreteVector<DDims>(myextents)))...));
219 template <
class... DElems>
223 sizeof...(DDims) == (0 + ... + DElems::size()),
224 "Invalid number of dimensions");
225 static_assert((is_discrete_element_v<DElems> && ...),
"Expected DiscreteElements");
226 DiscreteElement<DDims...>
const delem(delems...);
227 for (std::size_t i = 0; i <
rank(); ++i) {
228 if ((detail::array(delem)[i] < detail::array(m_element_begin)[i])
229 || (detail::array(delem)[i] >= detail::array(m_element_end)[i])) {
236 template <
class... DElems>
238 DElems
const&... delems)
const noexcept
241 sizeof...(DDims) == (0 + ... + DElems::size()),
242 "Invalid number of dimensions");
243 static_assert((is_discrete_element_v<DElems> && ...),
"Expected DiscreteElements");
245 (DiscreteElement<DDims>(take<DDims>(delems...))
246 - DiscreteElement<DDims>(m_element_begin))...);
260 std::size_t N =
sizeof...(DDims),
261 class DDim0 = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
268 std::size_t N =
sizeof...(DDims),
269 class DDim0 = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
276 std::size_t N =
sizeof...(DDims),
277 class DDim0 = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
284 std::size_t N =
sizeof...(DDims),
285 class DDim0 = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
292 std::size_t N =
sizeof...(DDims),
293 class = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
300 std::size_t N =
sizeof...(DDims),
301 class = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
315 using discrete_element_type = DiscreteElement<>;
327 template <
class... DDoms,
class = std::enable_if_t<(is_discrete_domain_v<DDoms> && ...)>>
333
334
335
337 [[
maybe_unused]] discrete_element_type
const& element_begin,
357#if !defined(__cpp_impl_three_way_comparison) || __cpp_impl_three_way_comparison < 201902L
361 return !(*
this ==
other);
422 template <
class... ODims>
460template <
class... QueryDDims,
class... DDims>
470struct ConvertTypeSeqToDiscreteDomain;
472template <
class... DDims>
473struct ConvertTypeSeqToDiscreteDomain<detail::TypeSeq<DDims...>>
479using convert_type_seq_to_discrete_domain_t =
typename ConvertTypeSeqToDiscreteDomain<T>::type;
485template <
typename... DDoms>
486struct cartesian_prod;
488template <
typename... DDims1,
typename... DDims2,
typename... DDomsTail>
491 using type =
typename cartesian_prod<
DiscreteDomain<DDims1..., DDims2...>, DDomsTail...>::type;
494template <
typename... DDims>
501struct cartesian_prod<>
506template <
typename... DDoms>
507using cartesian_prod_t =
typename cartesian_prod<DDoms...>::type;
510template <
class... DDimsA,
class... DDimsB>
515 using TagSeqA = detail::TypeSeq<DDimsA...>;
516 using TagSeqB = detail::TypeSeq<DDimsB...>;
518 using type_seq_r = type_seq_remove_t<TagSeqA, TagSeqB>;
519 return detail::convert_type_seq_to_discrete_domain_t<type_seq_r>(DDom_a);
525template <
class... DDimsB,
class... DDimsA>
528 using TagSeqA = detail::TypeSeq<DDimsA...>;
529 using TagSeqB = detail::TypeSeq<DDimsB...>;
531 using type_seq_r = type_seq_remove_t<TagSeqA, TagSeqB>;
532 return detail::convert_type_seq_to_discrete_domain_t<type_seq_r>(DDom_a);
536template <
typename DDom,
typename... DDims>
537using remove_dims_of_t =
decltype(remove_dims_of<DDims...>(std::declval<DDom>()));
542template <
typename DDim1,
typename DDim2,
typename DDimA,
typename... DDimsB>
543KOKKOS_FUNCTION
constexpr std::conditional_t<
544 std::is_same_v<DDimA, DDim1>,
549 [[maybe_unused]]
DiscreteDomain<DDimsB...>
const& DDom_b)
noexcept
551 if constexpr (std::is_same_v<DDimA, DDim1>) {
561template <
typename DDim1,
typename DDim2,
typename... DDimsA,
typename... DDimsB>
567 using TagSeqA = detail::TypeSeq<DDimsA...>;
568 using TagSeqB = detail::TypeSeq<DDim1>;
569 using TagSeqC = detail::TypeSeq<DDim2>;
571 using type_seq_r =
ddc::type_seq_replace_t<TagSeqA, TagSeqB, TagSeqC>;
572 return ddc::detail::convert_type_seq_to_discrete_domain_t<type_seq_r>(
573 detail::replace_dim_of_1d<
581template <
typename DDom,
typename DDim1,
typename DDim2>
582using replace_dim_of_t =
decltype(replace_dim_of<DDim1, DDim2>(
583 std::declval<DDom>(),
584 std::declval<
typename detail::RebindDomain<DDom, detail::TypeSeq<DDim2>>::type>()));
587template <
class... QueryDDims,
class... DDims>
594template <
class... QueryDDims,
class... DDims>
598 return DiscreteElement<QueryDDims...>(
DiscreteDomain<QueryDDims>(domain).front()...);
601template <
class... QueryDDims,
class... DDims>
605 return DiscreteElement<QueryDDims...>(
DiscreteDomain<QueryDDims>(domain).back()...);
612 DiscreteElement<DDim> m_value = DiscreteElement<DDim>();
615 using iterator_category = std::random_access_iterator_tag;
617 using value_type = DiscreteElement<DDim>;
619 using difference_type = std::ptrdiff_t;
661 if (n >=
static_cast<difference_type>(0)) {
662 m_value +=
static_cast<DiscreteElementType>(n);
664 m_value -=
static_cast<DiscreteElementType>(-n);
671 if (n >=
static_cast<difference_type>(0)) {
672 m_value -=
static_cast<DiscreteElementType>(n);
674 m_value +=
static_cast<DiscreteElementType>(-n);
688 return xx.m_value == yy.m_value;
691#if !defined(__cpp_impl_three_way_comparison) || __cpp_impl_three_way_comparison < 201902L
705 return xx.m_value < yy.m_value;
754 return (yy.m_value > xx.m_value) ? (-
static_cast<difference_type>(yy.m_value - xx.m_value))
755 : (xx.m_value - yy.m_value);
static KOKKOS_FUNCTION constexpr discrete_vector_type extents() noexcept
KOKKOS_DEFAULTED_FUNCTION ~DiscreteDomain()=default
KOKKOS_FUNCTION constexpr DiscreteDomain(discrete_element_type const &element_begin, discrete_vector_type const &size)
Construct a DiscreteDomain starting from element_begin with size points.
KOKKOS_FUNCTION constexpr DiscreteDomain remove(discrete_vector_type n1, discrete_vector_type n2) const
KOKKOS_FUNCTION constexpr DiscreteDomain take_last(discrete_vector_type n) const
static KOKKOS_FUNCTION constexpr std::size_t rank()
KOKKOS_DEFAULTED_FUNCTION DiscreteDomain(DiscreteDomain const &x)=default
static KOKKOS_FUNCTION bool contains() noexcept
KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteDomain()=default
KOKKOS_DEFAULTED_FUNCTION DiscreteDomain & operator=(DiscreteDomain &&x)=default
KOKKOS_FUNCTION constexpr operator bool()
static KOKKOS_FUNCTION constexpr std::size_t size()
static KOKKOS_FUNCTION constexpr bool empty() noexcept
KOKKOS_FUNCTION constexpr DiscreteDomain remove_last(discrete_vector_type n) const
static KOKKOS_FUNCTION bool contains(DiscreteElement<>) noexcept
static KOKKOS_FUNCTION constexpr discrete_element_type back() noexcept
static KOKKOS_FUNCTION DiscreteVector distance_from_front(DiscreteElement<>) noexcept
static KOKKOS_FUNCTION constexpr discrete_element_type front() noexcept
KOKKOS_FUNCTION constexpr DiscreteDomain take_first(discrete_vector_type n) const
KOKKOS_DEFAULTED_FUNCTION DiscreteDomain & operator=(DiscreteDomain const &x)=default
KOKKOS_FUNCTION constexpr DiscreteDomain(DDoms const &... domains)
Construct a DiscreteDomain by copies and merge of domains.
KOKKOS_FUNCTION constexpr bool operator==(DiscreteDomain const &other) const
KOKKOS_FUNCTION constexpr DiscreteDomain remove_first(discrete_vector_type n) const
KOKKOS_FUNCTION constexpr DiscreteDomain restrict_with(DiscreteDomain< ODims... > const &) const
KOKKOS_FUNCTION constexpr DiscreteElement operator()(DiscreteVector<> const &) const noexcept
KOKKOS_DEFAULTED_FUNCTION DiscreteDomain(DiscreteDomain &&x)=default
static KOKKOS_FUNCTION DiscreteVector distance_from_front() noexcept
KOKKOS_DEFAULTED_FUNCTION DiscreteDomain(DiscreteDomain const &x)=default
KOKKOS_DEFAULTED_FUNCTION DiscreteDomain(DiscreteDomain &&x)=default
KOKKOS_FUNCTION constexpr DiscreteDomain remove_last(discrete_vector_type n) const
KOKKOS_FUNCTION bool contains(DElems const &... delems) const noexcept
KOKKOS_FUNCTION constexpr DiscreteDomain take_last(discrete_vector_type n) const
KOKKOS_FUNCTION auto begin() const
KOKKOS_FUNCTION constexpr DiscreteDomain remove(discrete_vector_type n1, discrete_vector_type n2) const
KOKKOS_FUNCTION constexpr auto restrict_with(DiscreteDomain< ODDims... > const &odomain) const
static KOKKOS_FUNCTION constexpr std::size_t rank()
KOKKOS_FUNCTION constexpr bool empty() const noexcept
KOKKOS_FUNCTION constexpr std::size_t size() const
friend class DiscreteDomain
KOKKOS_FUNCTION constexpr discrete_vector_type extents() const noexcept
KOKKOS_FUNCTION constexpr DiscreteVector< QueryDDim > extent() const noexcept
KOKKOS_FUNCTION constexpr discrete_element_type front() const noexcept
KOKKOS_FUNCTION constexpr DiscreteDomain(DDoms const &... domains)
Construct a DiscreteDomain by copies and merge of domains.
KOKKOS_DEFAULTED_FUNCTION DiscreteDomain & operator=(DiscreteDomain &&x)=default
KOKKOS_FUNCTION constexpr DiscreteDomain remove_first(discrete_vector_type n) const
KOKKOS_FUNCTION constexpr bool operator==(DiscreteDomain< ODims... > const &other) const
KOKKOS_FUNCTION auto cbegin() const
KOKKOS_FUNCTION auto end() const
KOKKOS_FUNCTION constexpr DiscreteDomain take_first(discrete_vector_type n) const
KOKKOS_DEFAULTED_FUNCTION ~DiscreteDomain()=default
KOKKOS_FUNCTION auto cend() const
KOKKOS_FUNCTION constexpr decltype(auto) operator[](std::size_t n) const
KOKKOS_FUNCTION DiscreteVector< DDims... > distance_from_front(DElems const &... delems) const noexcept
KOKKOS_FUNCTION constexpr discrete_element_type back() const noexcept
KOKKOS_DEFAULTED_FUNCTION DiscreteDomain & operator=(DiscreteDomain const &x)=default
KOKKOS_FUNCTION constexpr DiscreteElement< DDims... > operator()(DiscreteVector< DDims... > const &dvect) const noexcept
KOKKOS_DEFAULTED_FUNCTION DiscreteDomain()=default
KOKKOS_FUNCTION constexpr operator bool()
KOKKOS_FUNCTION constexpr DiscreteDomain(discrete_element_type const &element_begin, discrete_vector_type const &size)
Construct a DiscreteDomain starting from element_begin with size points.
KOKKOS_FUNCTION constexpr decltype(auto) operator[](std::size_t n)
KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteElement()=default
KOKKOS_FUNCTION constexpr bool operator!=(DiscreteVector< OTags... > const &rhs) const noexcept
KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteVector()=default
The top-level namespace of DDC.
KOKKOS_FUNCTION constexpr DiscreteElement< QueryDDims... > back(DiscreteDomain< DDims... > const &domain) noexcept
KOKKOS_FUNCTION constexpr auto remove_dims_of(DiscreteDomain< DDimsA... > const &DDom_a) noexcept
Remove the dimensions DDimsB from DDom_a.
KOKKOS_FUNCTION constexpr DiscreteElement< QueryDDims... > front(DiscreteDomain< DDims... > const &domain) noexcept
KOKKOS_FUNCTION constexpr DiscreteVector< QueryDDims... > extents(DiscreteDomain< DDims... > const &domain) noexcept
KOKKOS_FUNCTION constexpr auto remove_dims_of(DiscreteDomain< DDimsA... > const &DDom_a, DiscreteDomain< DDimsB... > const &DDom_b) noexcept
KOKKOS_FUNCTION constexpr auto replace_dim_of(DiscreteDomain< DDimsA... > const &DDom_a, DiscreteDomain< DDimsB... > const &DDom_b) noexcept
constexpr bool is_discrete_domain_v
KOKKOS_FUNCTION constexpr DiscreteDomain< QueryDDims... > select(DiscreteDomain< DDims... > const &domain)
KOKKOS_FUNCTION constexpr DiscreteDomainIterator operator--(int)
friend KOKKOS_FUNCTION constexpr bool operator<=(DiscreteDomainIterator const &xx, DiscreteDomainIterator const &yy)
KOKKOS_FUNCTION constexpr DiscreteElement< DDim > operator*() const noexcept
friend KOKKOS_FUNCTION constexpr bool operator>=(DiscreteDomainIterator const &xx, DiscreteDomainIterator const &yy)
KOKKOS_DEFAULTED_FUNCTION DiscreteDomainIterator()=default
KOKKOS_FUNCTION constexpr DiscreteElement< DDim > operator[](difference_type n) const
KOKKOS_FUNCTION constexpr DiscreteDomainIterator & operator--()
KOKKOS_FUNCTION constexpr DiscreteDomainIterator operator++(int)
friend KOKKOS_FUNCTION constexpr DiscreteDomainIterator operator-(DiscreteDomainIterator i, difference_type n)
friend KOKKOS_FUNCTION constexpr DiscreteDomainIterator operator+(DiscreteDomainIterator i, difference_type n)
friend KOKKOS_FUNCTION constexpr bool operator>(DiscreteDomainIterator const &xx, DiscreteDomainIterator const &yy)
friend KOKKOS_FUNCTION constexpr bool operator!=(DiscreteDomainIterator const &xx, DiscreteDomainIterator const &yy)
friend KOKKOS_FUNCTION constexpr bool operator<(DiscreteDomainIterator const &xx, DiscreteDomainIterator const &yy)
KOKKOS_FUNCTION constexpr DiscreteDomainIterator & operator+=(difference_type n)
KOKKOS_FUNCTION constexpr DiscreteDomainIterator & operator++()
friend KOKKOS_FUNCTION constexpr DiscreteDomainIterator operator+(difference_type n, DiscreteDomainIterator i)
friend KOKKOS_FUNCTION constexpr bool operator==(DiscreteDomainIterator const &xx, DiscreteDomainIterator const &yy)
KOKKOS_FUNCTION constexpr DiscreteDomainIterator(DiscreteElement< DDim > value)
KOKKOS_FUNCTION constexpr DiscreteDomainIterator & operator-=(difference_type n)
friend KOKKOS_FUNCTION constexpr difference_type operator-(DiscreteDomainIterator const &xx, DiscreteDomainIterator const &yy)