14#include <Kokkos_Core.hpp>
15#include <Kokkos_StdAlgorithms.hpp>
17#include "detail/tagged_vector.hpp"
18#include "detail/type_seq.hpp"
20#include "discrete_element.hpp"
21#include "discrete_vector.hpp"
26struct SparseDiscreteDomainIterator;
28template <
class... DDims>
36template <
class... Tags>
47template <
class... Tags>
50 using type = TypeSeq<Tags...>;
53template <
class T,
class U>
56template <
class... DDims,
class... ODDims>
62template <
class InputIt1,
class InputIt2>
63KOKKOS_FUNCTION
bool equal(InputIt1 first1, InputIt1 last1, InputIt2 first2)
65 for (; first1 != last1; ++first1, ++first2) {
66 if (!(*first1 == *first2)) {
76 class T =
typename std::iterator_traits<ForwardIt>::value_type,
78KOKKOS_FUNCTION ForwardIt lower_bound(ForwardIt first, ForwardIt last, T
const& value, Compare comp)
81 typename std::iterator_traits<ForwardIt>::difference_type count = last - first;
84 typename std::iterator_traits<ForwardIt>::difference_type
const step = count / 2;
87 if (comp(*it, value)) {
100 class T =
typename std::iterator_traits<ForwardIt>::value_type,
102KOKKOS_FUNCTION
bool binary_search(ForwardIt first, ForwardIt last, T
const& value, Compare comp)
104 first = ::
ddc::detail::lower_bound(first, last, value, comp);
105 return (!(first == last) && !(comp(value, *first)));
111 KOKKOS_FUNCTION DiscreteElementType
112 operator()(DiscreteElement<DDim>
const& delem)
const noexcept
120template <
class... DDims>
127 type_seq_is_unique_v<detail::TypeSeq<DDims...>>,
128 "The dimensions of a SparseDiscreteDomain must be unique");
131 using discrete_element_type = DiscreteElement<DDims...>;
136 detail::TaggedVector<Kokkos::View<DiscreteElementType*, Kokkos::SharedSpace>, DDims...> m_views;
141 return sizeof...(DDims);
147 template <
class... DDoms,
class = std::enable_if_t<(is_sparse_discrete_domain_v<DDoms> && ...)>>
149 : m_views(domains.m_views...)
154
155
157 Kokkos::View<DiscreteElement<DDims>*, Kokkos::SharedSpace>
const&... views)
159 ((m_views[type_seq_rank_v<DDims, detail::TypeSeq<DDims...>>]
160 = Kokkos::View<DiscreteElementType*, Kokkos::SharedSpace>(views.label(), views.size())),
162 Kokkos::DefaultExecutionSpace
const execution_space;
163 ((Kokkos::Experimental::transform(
164 "SparseDiscreteDomainCtor",
167 m_views[type_seq_rank_v<DDims, detail::TypeSeq<DDims...>>],
168 detail::GetUidFn<DDims>())),
170 execution_space.fence(
"SparseDiscreteDomainCtor");
184 template <
class... ODims>
187 if constexpr ((std::is_same_v<DDims, ODims> && ...)) {
188 if (
empty() && other.empty()) {
191 for (std::size_t i = 0; i < m_views.size(); ++i) {
192 if (m_views[i].size() != other.m_views[i].size()) {
196 equal(m_views[i].data(),
197 m_views[i].data() + m_views[i].size(),
198 other.m_views[i].data())) {
208#if !defined(__cpp_impl_three_way_comparison) || __cpp_impl_three_way_comparison < 201902L
210 template <
class...
ODims>
213 return !(*
this ==
other);
219 return (1UL * ... * get<DDims>(m_views).size());
224 return discrete_vector_type(get<DDims>(m_views).size()...);
232 template <
class QueryDDim>
240 return discrete_element_type(get<DDims>(m_views)(0)...);
245 return discrete_element_type(get<DDims>(m_views)(get<DDims>(m_views).size() - 1)...);
269 discrete_vector_type n1,
270 discrete_vector_type n2)
const
278 return DiscreteElement<DDims...>(get<DDims>(m_views)(get<DDims>(dvect))...);
281 template <
class... DElems>
285 sizeof...(DDims) == (0 + ... + DElems::size()),
286 "Invalid number of dimensions");
287 static_assert((is_discrete_element_v<DElems> && ...),
"Expected DiscreteElements");
288 DiscreteElement<DDims...>
const delem(delems...);
289 for (std::size_t i = 0; i <
rank(); ++i) {
290 if (!detail::binary_search(
291 Kokkos::Experimental::begin(m_views[i]),
292 Kokkos::Experimental::end(m_views[i]),
293 detail::array(delem)[i],
301 template <
class... DElems>
303 DElems
const&... delems)
const noexcept
306 sizeof...(DDims) == (0 + ... + DElems::size()),
307 "Invalid number of dimensions");
308 static_assert((is_discrete_element_v<DElems> && ...),
"Expected DiscreteElements");
309 KOKKOS_ASSERT(contains(delems...))
311 (detail::lower_bound(
312 Kokkos::Experimental::begin(get<DDims>(m_views)),
313 Kokkos::Experimental::end(get<DDims>(m_views)),
314 uid<DDims>(take<DDims>(delems...)),
316 - Kokkos::Experimental::begin(get<DDims>(m_views)))...);
330 std::size_t N =
sizeof...(DDims),
331 class DDim0 = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
334 return Kokkos::Experimental::begin(get<DDim0>(m_views));
338 std::size_t N =
sizeof...(DDims),
339 class DDim0 = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
342 return Kokkos::Experimental::end(get<DDim0>(m_views));
346 std::size_t N =
sizeof...(DDims),
347 class DDim0 = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
350 return Kokkos::Experimental::cbegin(get<DDim0>(m_views));
354 std::size_t N =
sizeof...(DDims),
355 class DDim0 = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
358 return Kokkos::Experimental::cend(get<DDim0>(m_views));
362 std::size_t N =
sizeof...(DDims),
363 class = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
370 std::size_t N =
sizeof...(DDims),
371 class = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
385 using discrete_element_type = DiscreteElement<>;
397 template <
class... ODDims>
420#if !defined(__cpp_impl_three_way_comparison) || __cpp_impl_three_way_comparison < 201902L
424 return !(*
this ==
other);
516template <
class... QueryDDims,
class... DDims>
526struct ConvertTypeSeqToSparseDiscreteDomain;
528template <
class... DDims>
529struct ConvertTypeSeqToSparseDiscreteDomain<detail::TypeSeq<DDims...>>
535using convert_type_seq_to_sparse_discrete_domain_t =
536 typename ConvertTypeSeqToSparseDiscreteDomain<T>::type;
541template <
class... DDimsA,
class... DDimsB>
546 using TagSeqA = detail::TypeSeq<DDimsA...>;
547 using TagSeqB = detail::TypeSeq<DDimsB...>;
549 using type_seq_r = type_seq_remove_t<TagSeqA, TagSeqB>;
550 return detail::convert_type_seq_to_sparse_discrete_domain_t<type_seq_r>(DDom_a);
556template <
class... DDimsB,
class... DDimsA>
560 using TagSeqA = detail::TypeSeq<DDimsA...>;
561 using TagSeqB = detail::TypeSeq<DDimsB...>;
563 using type_seq_r = type_seq_remove_t<TagSeqA, TagSeqB>;
564 return detail::convert_type_seq_to_sparse_discrete_domain_t<type_seq_r>(DDom_a);
570template <
typename DDim1,
typename DDim2,
typename DDimA,
typename... DDimsB>
571KOKKOS_FUNCTION
constexpr std::conditional_t<
572 std::is_same_v<DDimA, DDim1>,
579 if constexpr (std::is_same_v<DDimA, DDim1>) {
589template <
typename DDim1,
typename DDim2,
typename... DDimsA,
typename... DDimsB>
595 using TagSeqA = detail::TypeSeq<DDimsA...>;
596 using TagSeqB = detail::TypeSeq<DDim1>;
597 using TagSeqC = detail::TypeSeq<DDim2>;
599 using type_seq_r =
ddc::type_seq_replace_t<TagSeqA, TagSeqB, TagSeqC>;
600 return ddc::detail::convert_type_seq_to_sparse_discrete_domain_t<type_seq_r>(
601 detail::replace_dim_of_1d<
608template <
class... QueryDDims,
class... DDims>
615template <
class... QueryDDims,
class... DDims>
622template <
class... QueryDDims,
class... DDims>
KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteElement()=default
KOKKOS_FUNCTION constexpr bool operator!=(DiscreteVector< OTags... > const &rhs) const noexcept
KOKKOS_DEFAULTED_FUNCTION constexpr DiscreteVector()=default
KOKKOS_FUNCTION constexpr bool operator==(SparseDiscreteDomain const &other) const
static KOKKOS_FUNCTION bool contains(DiscreteElement<>) noexcept
static KOKKOS_FUNCTION constexpr std::size_t rank()
KOKKOS_DEFAULTED_FUNCTION constexpr SparseDiscreteDomain()=default
KOKKOS_FUNCTION constexpr SparseDiscreteDomain(SparseDiscreteDomain< ODDims... > const &domain)
KOKKOS_FUNCTION constexpr DiscreteElement operator()(DiscreteVector<> const &) const noexcept
KOKKOS_FUNCTION constexpr SparseDiscreteDomain take_last(discrete_vector_type n) const
static KOKKOS_FUNCTION DiscreteVector distance_from_front() noexcept
static KOKKOS_FUNCTION constexpr std::size_t size()
static KOKKOS_FUNCTION constexpr bool empty() noexcept
KOKKOS_FUNCTION constexpr SparseDiscreteDomain remove(discrete_vector_type n1, discrete_vector_type n2) const
static KOKKOS_FUNCTION DiscreteVector distance_from_front(DiscreteElement<>) noexcept
KOKKOS_FUNCTION constexpr SparseDiscreteDomain remove_first(discrete_vector_type n) const
static KOKKOS_FUNCTION constexpr discrete_element_type back() noexcept
static KOKKOS_FUNCTION bool contains() noexcept
KOKKOS_FUNCTION constexpr operator bool()
KOKKOS_DEFAULTED_FUNCTION SparseDiscreteDomain & operator=(SparseDiscreteDomain const &x)=default
KOKKOS_DEFAULTED_FUNCTION SparseDiscreteDomain & operator=(SparseDiscreteDomain &&x)=default
KOKKOS_DEFAULTED_FUNCTION SparseDiscreteDomain(SparseDiscreteDomain const &x)=default
KOKKOS_DEFAULTED_FUNCTION ~SparseDiscreteDomain()=default
KOKKOS_FUNCTION constexpr SparseDiscreteDomain take_first(discrete_vector_type n) const
static KOKKOS_FUNCTION constexpr discrete_element_type front() noexcept
static KOKKOS_FUNCTION constexpr discrete_vector_type extents() noexcept
KOKKOS_DEFAULTED_FUNCTION SparseDiscreteDomain(SparseDiscreteDomain &&x)=default
KOKKOS_FUNCTION constexpr SparseDiscreteDomain remove_last(discrete_vector_type n) const
KOKKOS_FUNCTION constexpr discrete_element_type back() const noexcept
KOKKOS_FUNCTION constexpr SparseDiscreteDomain remove_first(discrete_vector_type n) const
static KOKKOS_FUNCTION constexpr std::size_t rank()
KOKKOS_FUNCTION constexpr decltype(auto) operator[](std::size_t n) const
KOKKOS_DEFAULTED_FUNCTION SparseDiscreteDomain & operator=(SparseDiscreteDomain const &x)=default
KOKKOS_FUNCTION constexpr SparseDiscreteDomain take_last(discrete_vector_type n) const
KOKKOS_FUNCTION DiscreteVector< DDims... > distance_from_front(DElems const &... delems) const noexcept
KOKKOS_FUNCTION constexpr auto discrete_elements() const noexcept
KOKKOS_FUNCTION constexpr SparseDiscreteDomain remove(discrete_vector_type n1, discrete_vector_type n2) const
KOKKOS_FUNCTION constexpr SparseDiscreteDomain(DDoms const &... domains)
Construct a SparseDiscreteDomain by copies and merge of domains.
SparseDiscreteDomain(Kokkos::View< DiscreteElement< DDims > *, Kokkos::SharedSpace > const &... views)
Construct a SparseDiscreteDomain with Kokkos::View explicitly listing the discrete elements.
KOKKOS_FUNCTION auto cbegin() const
KOKKOS_FUNCTION constexpr decltype(auto) operator[](std::size_t n)
KOKKOS_DEFAULTED_FUNCTION SparseDiscreteDomain & operator=(SparseDiscreteDomain &&x)=default
KOKKOS_FUNCTION constexpr std::size_t size() const
KOKKOS_FUNCTION constexpr bool operator==(SparseDiscreteDomain< ODims... > const &other) const
KOKKOS_FUNCTION bool contains(DElems const &... delems) const noexcept
KOKKOS_FUNCTION constexpr SparseDiscreteDomain take_first(discrete_vector_type n) const
KOKKOS_DEFAULTED_FUNCTION SparseDiscreteDomain()=default
KOKKOS_DEFAULTED_FUNCTION SparseDiscreteDomain(SparseDiscreteDomain &&x)=default
KOKKOS_FUNCTION constexpr discrete_element_type front() const noexcept
KOKKOS_FUNCTION constexpr operator bool()
KOKKOS_FUNCTION constexpr bool empty() const noexcept
KOKKOS_FUNCTION constexpr DiscreteElement< DDims... > operator()(DiscreteVector< DDims... > const &dvect) const noexcept
KOKKOS_DEFAULTED_FUNCTION SparseDiscreteDomain(SparseDiscreteDomain const &x)=default
KOKKOS_FUNCTION constexpr SparseDiscreteDomain remove_last(discrete_vector_type n) const
KOKKOS_FUNCTION auto end() const
KOKKOS_FUNCTION auto cend() const
friend class SparseDiscreteDomain
KOKKOS_FUNCTION auto begin() const
KOKKOS_FUNCTION constexpr discrete_vector_type extents() const noexcept
KOKKOS_FUNCTION constexpr DiscreteVector< QueryDDim > extent() const noexcept
KOKKOS_DEFAULTED_FUNCTION ~SparseDiscreteDomain()=default
The top-level namespace of DDC.
KOKKOS_FUNCTION constexpr DiscreteElement< QueryDDims... > back(SparseDiscreteDomain< DDims... > const &domain) noexcept
KOKKOS_FUNCTION constexpr SparseDiscreteDomain< QueryDDims... > select(SparseDiscreteDomain< DDims... > const &domain)
KOKKOS_FUNCTION constexpr DiscreteVector< QueryDDims... > extents(SparseDiscreteDomain< DDims... > const &domain) noexcept
KOKKOS_FUNCTION constexpr auto replace_dim_of(SparseDiscreteDomain< DDimsA... > const &DDom_a, SparseDiscreteDomain< DDimsB... > const &DDom_b) noexcept
KOKKOS_FUNCTION constexpr DiscreteElement< QueryDDims... > front(SparseDiscreteDomain< DDims... > const &domain) noexcept
constexpr bool is_sparse_discrete_domain_v
KOKKOS_FUNCTION constexpr auto remove_dims_of(SparseDiscreteDomain< DDimsA... > const &DDom_a) noexcept
Remove the dimensions DDimsB from DDom_a.
KOKKOS_FUNCTION constexpr auto remove_dims_of(SparseDiscreteDomain< DDimsA... > const &DDom_a, SparseDiscreteDomain< DDimsB... > const &DDom_b) noexcept