12#include <Kokkos_Core.hpp>
14#include "detail/kokkos.hpp"
15#include "detail/type_traits.hpp"
17#include "chunk_common.hpp"
18#include "chunk_span.hpp"
19#include "discrete_domain.hpp"
20#include "discrete_element.hpp"
21#include "discrete_vector.hpp"
22#include "kokkos_allocator.hpp"
26template <
class ElementType,
class,
class Allocator = HostAllocator<ElementType>>
32template <
class ElementType,
class SupportType,
class Allocator>
33class Chunk :
public ChunkCommon<ElementType, SupportType, Kokkos::layout_right>
36 using base_type =
ChunkCommon<ElementType, SupportType, Kokkos::layout_right>;
44 typename Allocator::memory_space>;
51 typename Allocator::memory_space>;
54 using allocation_mdspan_type =
typename base_type::allocation_mdspan_type;
56 using const_allocation_mdspan_type =
typename base_type::const_allocation_mdspan_type;
58 using discrete_domain_type =
typename base_type::discrete_domain_type;
60 using memory_space =
typename Allocator::memory_space;
62 using discrete_element_type =
typename base_type::discrete_element_type;
64 using discrete_vector_type =
typename base_type::discrete_vector_type;
66 using extents_type =
typename base_type::extents_type;
68 using layout_type =
typename base_type::layout_type;
70 using mapping_type =
typename base_type::mapping_type;
72 using element_type =
typename base_type::element_type;
74 using value_type =
typename base_type::value_type;
76 using size_type =
typename base_type::size_type;
78 using data_handle_type =
typename base_type::data_handle_type;
80 using reference =
typename base_type::reference;
82 template <
class,
class,
class>
86 Allocator m_allocator;
96 std::string
const& label,
97 SupportType
const& domain,
98 Allocator allocator = Allocator())
99 : base_type(allocator.allocate(label, domain.size()), domain)
100 , m_allocator(std::move(allocator))
106 explicit Chunk(SupportType
const& domain, Allocator allocator = Allocator())
107 :
Chunk(
"no-label", domain, std::move(allocator))
115
116
118 : base_type(std::move(
static_cast<base_type&>(other)))
119 , m_allocator(std::move(other.m_allocator))
120 , m_label(std::move(other.m_label))
122 other.m_allocation_mdspan
123 = allocation_mdspan_type(
nullptr, other.m_allocation_mdspan.mapping());
128 if (
this->m_allocation_mdspan.data_handle()) {
129 m_allocator.deallocate(
this->data_handle(),
this->size());
137
138
139
142 if (
this->data_handle()) {
143 m_allocator.deallocate(
this->data_handle(),
this->size());
145 static_cast<base_type&>(*
this) = std::move(
static_cast<base_type&>(other));
146 m_allocator = std::move(other.m_allocator);
147 m_label = std::move(other.m_label);
148 other.m_allocation_mdspan
149 = allocation_mdspan_type(
nullptr, other.m_allocation_mdspan.mapping());
155 template <
class... QueryDDims>
156 auto operator[](DiscreteElement<QueryDDims...>
const& slice_spec)
const
158 return view_type(*
this)[slice_spec];
162 template <
class... QueryDDims>
163 auto operator[](DiscreteElement<QueryDDims...>
const& slice_spec)
165 return span_view()[slice_spec];
171 class SFINAESupportType = SupportType,
172 std::enable_if_t<is_discrete_domain_v<SFINAESupportType>,
int> = 0>
175 return span_view()[odomain];
181 class SFINAESupportType = SupportType,
182 std::enable_if_t<is_discrete_domain_v<SFINAESupportType>,
int> = 0>
185 return span_view()[odomain];
189
190
191
194 std::enable_if_t<detail::all_of_v<is_discrete_element_v<DElems>...>,
int> = 0>
195 element_type
const&
operator()(DElems
const&... delems)
const noexcept
198 SupportType::rank() == (0 + ... + DElems::size()),
199 "Invalid number of dimensions");
200 assert(
this->m_domain.contains(delems...));
201 return DDC_MDSPAN_ACCESS_OP(
202 this->m_allocation_mdspan,
203 detail::array(
this->m_domain.distance_from_front(delems...)));
207
208
209
212 std::enable_if_t<detail::all_of_v<is_discrete_vector_v<DVects>...>,
int> = 0,
213 std::enable_if_t<
sizeof...(DVects) != 0,
int> = 0>
214 element_type
const&
operator()(DVects
const&... dvects)
const noexcept
217 SupportType::rank() == (0 + ... + DVects::size()),
218 "Invalid number of dimensions");
219 return DDC_MDSPAN_ACCESS_OP(
220 this->m_allocation_mdspan,
221 detail::array(discrete_vector_type(dvects...)));
225
226
227
230 std::enable_if_t<detail::all_of_v<is_discrete_element_v<DElems>...>,
int> = 0>
231 element_type&
operator()(DElems
const&... delems)
noexcept
234 SupportType::rank() == (0 + ... + DElems::size()),
235 "Invalid number of dimensions");
236 assert(
this->m_domain.contains(delems...));
237 return DDC_MDSPAN_ACCESS_OP(
238 this->m_allocation_mdspan,
239 detail::array(
this->m_domain.distance_from_front(delems...)));
243
244
245
248 std::enable_if_t<detail::all_of_v<is_discrete_vector_v<DVects>...>,
int> = 0,
249 std::enable_if_t<
sizeof...(DVects) != 0,
int> = 0>
250 element_type&
operator()(DVects
const&... dvects)
noexcept
253 SupportType::rank() == (0 + ... + DVects::size()),
254 "Invalid number of dimensions");
255 return DDC_MDSPAN_ACCESS_OP(
256 this->m_allocation_mdspan,
257 detail::array(discrete_vector_type(dvects...)));
261
262
263 char const*
label()
const
265 return m_label.c_str();
269
270
273 return base_type::data_handle();
277
278
281 return base_type::data_handle();
285
286
289 return base_type::allocation_mdspan();
293
294
297 return base_type::allocation_mdspan();
301
302
305 auto s =
this->allocation_mdspan();
306 auto kokkos_layout = detail::build_kokkos_layout(
309 std::make_index_sequence<SupportType::rank()> {});
311 detail::mdspan_to_kokkos_element_t<ElementType, SupportType::rank()>,
312 decltype(kokkos_layout),
313 typename Allocator::memory_space>(s.data_handle(), kokkos_layout);
317
318
321 auto s =
this->allocation_mdspan();
322 auto kokkos_layout = detail::build_kokkos_layout(
325 std::make_index_sequence<SupportType::rank()> {});
327 detail::mdspan_to_kokkos_element_t<ElementType
const, SupportType::rank()>,
328 decltype(kokkos_layout),
329 typename Allocator::memory_space>(s.data_handle(), kokkos_layout);
334 return view_type(*
this);
339 return view_type(*
this);
344 return span_type(*
this);
348template <
class SupportType,
class Allocator>
349Chunk(std::string
const&, SupportType
const&, Allocator)
352template <
class SupportType,
class Allocator>
353Chunk(SupportType
const&, Allocator)
auto operator[](DiscreteElement< QueryDDims... > const &slice_spec) const
Slice out some dimensions.
char const * label() const
Returns the label of the Chunk.
auto operator[](DiscreteDomain< QueryDDims... > const &odomain)
Slice out some dimensions.
const_allocation_mdspan_type allocation_mdspan() const
Provide a mdspan on the memory allocation.
element_type & operator()(DVects const &... dvects) noexcept
Element access using a list of DiscreteVector.
ElementType const * data_handle() const
Access to the underlying allocation pointer.
auto allocation_kokkos_view()
Provide an unmanaged Kokkos::View on the memory allocation.
auto allocation_kokkos_view() const
Provide an unmanaged Kokkos::View on the memory allocation.
Chunk(SupportType const &domain, Allocator allocator=Allocator())
Construct a Chunk on a domain with uninitialized values.
Chunk & operator=(Chunk const &other)=delete
Deleted: use deepcopy instead.
allocation_mdspan_type allocation_mdspan()
Provide a mdspan on the memory allocation.
ElementType * data_handle()
Access to the underlying allocation pointer.
element_type const & operator()(DVects const &... dvects) const noexcept
Element access using a list of DiscreteVector.
view_type span_cview() const
Chunk(Chunk const &other)=delete
Deleted: use deepcopy instead.
Chunk(std::string const &label, SupportType const &domain, Allocator allocator=Allocator())
Construct a labeled Chunk on a domain with uninitialized values.
element_type & operator()(DElems const &... delems) noexcept
Element access using a list of DiscreteElement.
Chunk()=default
Empty Chunk.
Chunk & operator=(Chunk &&other) noexcept
Move-assigns a new value to this field.
Chunk(Chunk &&other) noexcept
Constructs a new Chunk by move.
view_type span_view() const
auto operator[](DiscreteDomain< QueryDDims... > const &odomain) const
Slice out some dimensions.
element_type const & operator()(DElems const &... delems) const noexcept
Element access using a list of DiscreteElement.
auto operator[](DiscreteElement< QueryDDims... > const &slice_spec)
Slice out some dimensions.
friend class DiscreteDomain
The top-level namespace of DDC.
constexpr bool enable_chunk< Chunk< ElementType, SupportType, Allocator > >
Chunk(SupportType const &, Allocator) -> Chunk< typename Allocator::value_type, SupportType, Allocator >
Chunk(std::string const &, SupportType const &, Allocator) -> Chunk< typename Allocator::value_type, SupportType, Allocator >