DDC 0.10.0
Loading...
Searching...
No Matches
chunk_span.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 <cstddef>
8#include <type_traits>
9#include <utility>
10
11#include <Kokkos_Core.hpp>
12
13#include "detail/kokkos.hpp"
14#include "detail/macros.hpp"
15#include "detail/type_seq.hpp"
16#include "detail/type_traits.hpp"
17
18#include "chunk_common.hpp"
19#include "discrete_domain.hpp"
20#include "discrete_element.hpp"
21#include "discrete_vector.hpp"
22
23namespace ddc {
24
25template <class, class, class>
26class Chunk;
27
28template <
29 class ElementType,
30 class SupportType,
31 class LayoutStridedPolicy = Kokkos::layout_right,
32 class MemorySpace = Kokkos::DefaultHostExecutionSpace::memory_space>
33class ChunkSpan;
34
35template <class ElementType, class SupportType, class LayoutStridedPolicy, class MemorySpace>
36inline constexpr bool
38 = true;
39
40template <class ElementType, class SupportType, class LayoutStridedPolicy, class MemorySpace>
41inline constexpr bool
43 = true;
44
45template <class ElementType, class SupportType, class LayoutStridedPolicy, class MemorySpace>
46class ChunkSpan : public ChunkCommon<ElementType, SupportType, LayoutStridedPolicy>
47{
48 static_assert(
49 std::is_same_v<LayoutStridedPolicy, Kokkos::layout_left>
50 || std::is_same_v<LayoutStridedPolicy, Kokkos::layout_right>
51 || std::is_same_v<LayoutStridedPolicy, Kokkos::layout_stride>,
52 "ChunkSpan only supports layout_left, layout_right or layout_stride");
53
54protected:
55 using base_type = ChunkCommon<ElementType, SupportType, LayoutStridedPolicy>;
56
57public:
58 /// type of a span of this full chunk
59 using span_type = ChunkSpan<ElementType, SupportType, LayoutStridedPolicy, MemorySpace>;
60
61 /// type of a view of this full chunk
62 using view_type = ChunkSpan<ElementType const, SupportType, LayoutStridedPolicy, MemorySpace>;
63
64 using discrete_domain_type = typename base_type::discrete_domain_type;
65
66 using memory_space = MemorySpace;
67
68 /// The dereferenceable part of the co-domain but with a different domain, starting at 0
69 using allocation_mdspan_type = typename base_type::allocation_mdspan_type;
70
71 using const_allocation_mdspan_type = typename base_type::const_allocation_mdspan_type;
72
73 using discrete_element_type = typename discrete_domain_type::discrete_element_type;
74
75 using discrete_vector_type = typename discrete_domain_type::discrete_vector_type;
76
77 using extents_type = typename base_type::extents_type;
78
79 using layout_type = typename base_type::layout_type;
80
81 using accessor_type = typename base_type::accessor_type;
82
83 using mapping_type = typename base_type::mapping_type;
84
85 using element_type = typename base_type::element_type;
86
87 using value_type = typename base_type::value_type;
88
89 using size_type = typename base_type::size_type;
90
91 using data_handle_type = typename base_type::data_handle_type;
92
93 using reference = typename base_type::reference;
94
95 template <class, class, class, class>
96 friend class ChunkSpan;
97
98protected:
99 template <class QueryDDim, class... ODDims>
100 KOKKOS_FUNCTION static constexpr auto get_slicer_for(DiscreteVector<ODDims...> const& c)
101 {
102 DDC_IF_NVCC_THEN_PUSH_AND_SUPPRESS(implicit_return_from_non_void_function)
103 if constexpr (in_tags_v<QueryDDim, detail::TypeSeq<ODDims...>>) {
104 return c.template get<QueryDDim>();
105 } else {
106 return Kokkos::full_extent;
107 }
108 DDC_IF_NVCC_THEN_POP
109 }
110
111 template <class QueryDDim, class... ODDims, class... OODDims>
112 KOKKOS_FUNCTION static constexpr auto get_slicer_for(
113 DiscreteDomain<ODDims...> const& c,
114 DiscreteDomain<OODDims...> const& origin)
115 {
116 DDC_IF_NVCC_THEN_PUSH_AND_SUPPRESS(implicit_return_from_non_void_function)
117 if constexpr (in_tags_v<QueryDDim, detail::TypeSeq<ODDims...>>) {
118 DiscreteDomain<QueryDDim> const c_slice(c);
119 DiscreteDomain<QueryDDim> const origin_slice(origin);
120 return std::pair<std::size_t, std::size_t>(
121 c_slice.front() - origin_slice.front(),
122 c_slice.back() + 1 - origin_slice.front());
123 } else {
124 return Kokkos::full_extent;
125 }
126 DDC_IF_NVCC_THEN_POP
127 }
128
129 template <class TypeSeq>
130 struct slicer;
131
132 template <class... DDims>
133 struct slicer<detail::TypeSeq<DDims...>>
134 {
135 template <class... ODDims>
136 KOKKOS_FUNCTION constexpr auto operator()(
137 allocation_mdspan_type const& span,
138 DiscreteVector<ODDims...> const& c) const
139 {
140 return Kokkos::submdspan(span, get_slicer_for<DDims>(c)...);
141 }
142
143 template <class... ODDims, class... OODDims>
144 KOKKOS_FUNCTION constexpr auto operator()(
145 allocation_mdspan_type const& span,
146 DiscreteDomain<ODDims...> const& c,
147 DiscreteDomain<OODDims...> const& origin) const
148 {
149 return Kokkos::submdspan(span, get_slicer_for<DDims>(c, origin)...);
150 }
151 };
152
153public:
154 /// Empty ChunkSpan
155 KOKKOS_DEFAULTED_FUNCTION constexpr ChunkSpan() = default;
156
157 /** Constructs a new ChunkSpan by copy, yields a new view to the same data
158 * @param other the ChunkSpan to copy
159 */
160 KOKKOS_DEFAULTED_FUNCTION constexpr ChunkSpan(ChunkSpan const& other) = default;
161
162 /** Constructs a new ChunkSpan by move
163 * @param other the ChunkSpan to move
164 */
165 KOKKOS_DEFAULTED_FUNCTION constexpr ChunkSpan(ChunkSpan&& other) noexcept = default;
166
167 /** Forbids to construct a ChunkSpan from a rvalue of type Chunk.
168 */
169 template <
170 class OElementType,
171 class Allocator,
172 class = std::enable_if_t<std::is_same_v<typename Allocator::memory_space, MemorySpace>>>
173 ChunkSpan(Chunk<OElementType, SupportType, Allocator>&& other) noexcept = delete;
174
175 /** Constructs a new ChunkSpan from a Chunk, yields a new view to the same data
176 * @param other the Chunk to view
177 */
178 template <
179 class OElementType,
180 class Allocator,
181 class = std::enable_if_t<std::is_same_v<typename Allocator::memory_space, MemorySpace>>>
182 KOKKOS_FUNCTION constexpr explicit ChunkSpan(
183 Chunk<OElementType, SupportType, Allocator>& other) noexcept
184 : base_type(other.m_allocation_mdspan, other.m_domain)
185 {
186 }
187
188 /** Constructs a new ChunkSpan from a Chunk, yields a new view to the same data
189 * @param other the Chunk to view
190 */
191 // Disabled by SFINAE in the case of `ElementType` is not `const` to avoid write access
192 template <
193 class OElementType,
194 class SFINAEElementType = ElementType,
195 class = std::enable_if_t<std::is_const_v<SFINAEElementType>>,
196 class Allocator,
197 class = std::enable_if_t<std::is_same_v<typename Allocator::memory_space, MemorySpace>>>
198 KOKKOS_FUNCTION constexpr explicit ChunkSpan(
199 Chunk<OElementType, SupportType, Allocator> const& other) noexcept
200 : base_type(other.m_allocation_mdspan, other.m_domain)
201 {
202 }
203
204 /** Constructs a new ChunkSpan by copy of a chunk, yields a new view to the same data
205 * @param other the ChunkSpan to move
206 */
207 template <class OElementType>
208 KOKKOS_FUNCTION constexpr explicit ChunkSpan(
209 ChunkSpan<OElementType, SupportType, layout_type, MemorySpace> const& other) noexcept
210 : base_type(other.m_allocation_mdspan, other.m_domain)
211 {
212 }
213
214 /** Constructs a new ChunkSpan from scratch
215 * @param ptr the allocation pointer to the data
216 * @param domain the domain that sustains the view
217 */
218 template <
219 class Mapping = mapping_type,
220 std::enable_if_t<std::is_constructible_v<Mapping, extents_type>, int> = 0>
221 KOKKOS_FUNCTION constexpr ChunkSpan(ElementType* const ptr, SupportType const& domain)
222 : base_type(ptr, domain)
223 {
224 }
225
226 /** Constructs a new ChunkSpan from scratch
227 * @param allocation_mdspan the allocation mdspan to the data
228 * @param domain the domain that sustains the view
229 */
230 KOKKOS_FUNCTION constexpr ChunkSpan(
231 allocation_mdspan_type allocation_mdspan,
232 SupportType const& domain)
233 : base_type(allocation_mdspan, domain)
234 {
235 for (std::size_t i = 0; i < SupportType::rank(); ++i) {
236 KOKKOS_ASSERT(
237 allocation_mdspan.extent(i)
238 == static_cast<std::size_t>(detail::array(domain.extents())[i]))
239 }
240 }
241
242 /** Constructs a new ChunkSpan from scratch
243 * @param view the Kokkos view
244 * @param domain the domain that sustains the view
245 */
246 template <class KokkosView, class = std::enable_if_t<Kokkos::is_view_v<KokkosView>>>
247 KOKKOS_FUNCTION constexpr ChunkSpan(KokkosView const& view, SupportType const& domain) noexcept
248 : ChunkSpan(
249 detail::build_mdspan(view, std::make_index_sequence<SupportType::rank()> {}),
250 domain)
251 {
252 }
253
254 KOKKOS_DEFAULTED_FUNCTION ~ChunkSpan() noexcept = default;
255
256 /** Copy-assigns a new value to this ChunkSpan, yields a new view to the same data
257 * @param other the ChunkSpan to copy
258 * @return *this
259 */
260 KOKKOS_DEFAULTED_FUNCTION constexpr ChunkSpan& operator=(ChunkSpan const& other) = default;
261
262 /** Move-assigns a new value to this ChunkSpan
263 * @param other the ChunkSpan to move
264 * @return *this
265 */
266 KOKKOS_DEFAULTED_FUNCTION constexpr ChunkSpan& operator=(ChunkSpan&& other) noexcept = default;
267
268 /** Slice out some dimensions
269 */
270 template <class... QueryDDims>
271 KOKKOS_FUNCTION constexpr auto operator[](
272 DiscreteElement<QueryDDims...> const& slice_spec) const
273 {
274 using detail::TypeSeq;
275 using QueryDDom = typename detail::RebindDomain<SupportType, TypeSeq<QueryDDims...>>::type;
276 KOKKOS_ASSERT(QueryDDom(this->m_domain).contains(slice_spec))
277 slicer<to_type_seq_t<SupportType>> const slicer;
278 auto subview = slicer(
279 this->allocation_mdspan(),
280 QueryDDom(this->m_domain).distance_from_front(slice_spec));
281 using layout_type = typename decltype(subview)::layout_type;
282 using extents_type = typename decltype(subview)::extents_type;
283 using OutTypeSeqDDims
284 = type_seq_remove_t<to_type_seq_t<SupportType>, TypeSeq<QueryDDims...>>;
285 using OutDDom = typename detail::RebindDomain<SupportType, OutTypeSeqDDims>::type;
286 if constexpr (
287 std::is_same_v<layout_type, Kokkos::Experimental::layout_left_padded<>>
288 || std::is_same_v<layout_type, Kokkos::Experimental::layout_right_padded<>>) {
289 Kokkos::layout_stride::mapping<extents_type> const mapping_stride(subview.mapping());
290 Kokkos::mdspan<ElementType, extents_type, Kokkos::layout_stride> const
291 a(subview.data_handle(), mapping_stride);
292 return ChunkSpan<
293 ElementType,
294 OutDDom,
295 Kokkos::layout_stride,
296 memory_space>(a, OutDDom(this->m_domain));
297 } else {
298 return ChunkSpan<
299 ElementType,
300 OutDDom,
301 layout_type,
302 memory_space>(subview, OutDDom(this->m_domain));
303 }
304 }
305
306 /** Restrict to a subdomain, only valid when SupportType is a DiscreteDomain
307 */
308 template <
309 class... QueryDDims,
310 class SFINAESupportType = SupportType,
311 std::enable_if_t<is_discrete_domain_v<SFINAESupportType>, int> = 0>
312 KOKKOS_FUNCTION constexpr auto operator[](DiscreteDomain<QueryDDims...> const& odomain) const
313 {
314 KOKKOS_ASSERT(
315 odomain.empty()
316 || (DiscreteDomain<QueryDDims...>(this->m_domain).contains(odomain.front())
317 && DiscreteDomain<QueryDDims...>(this->m_domain).contains(odomain.back())))
318 slicer<to_type_seq_t<SupportType>> const slicer;
319 auto subview = slicer(this->allocation_mdspan(), odomain, this->m_domain);
320 using layout_type = typename decltype(subview)::layout_type;
321 using extents_type = typename decltype(subview)::extents_type;
322 if constexpr (
323 std::is_same_v<layout_type, Kokkos::Experimental::layout_left_padded<>>
324 || std::is_same_v<layout_type, Kokkos::Experimental::layout_right_padded<>>) {
325 Kokkos::layout_stride::mapping<extents_type> const mapping_stride(subview.mapping());
326 Kokkos::mdspan<ElementType, extents_type, Kokkos::layout_stride> const
327 a(subview.data_handle(), mapping_stride);
328 return ChunkSpan<
329 ElementType,
330 decltype(this->m_domain.restrict_with(odomain)),
331 Kokkos::layout_stride,
332 memory_space>(a, this->m_domain.restrict_with(odomain));
333 } else {
334 return ChunkSpan<
335 ElementType,
336 decltype(this->m_domain.restrict_with(odomain)),
337 layout_type,
338 memory_space>(subview, this->m_domain.restrict_with(odomain));
339 }
340 }
341
342 /** Element access using a list of DiscreteElement
343 * @param delems discrete elements
344 * @return reference to this element
345 */
346 template <
347 class... DElems,
348 std::enable_if_t<detail::all_of_v<is_discrete_element_v<DElems>...>, int> = 0>
349 KOKKOS_FUNCTION constexpr reference operator()(DElems const&... delems) const noexcept
350 {
351 static_assert(
352 SupportType::rank() == (0 + ... + DElems::size()),
353 "Invalid number of dimensions");
354 KOKKOS_ASSERT(this->m_domain.contains(delems...))
355 return DDC_MDSPAN_ACCESS_OP(
356 this->m_allocation_mdspan,
357 detail::array(this->m_domain.distance_from_front(delems...)));
358 }
359
360 /** Element access using a list of DiscreteVector
361 * @param dvects discrete vectors
362 * @return reference to this element
363 */
364 template <
365 class... DVects,
366 std::enable_if_t<detail::all_of_v<is_discrete_vector_v<DVects>...>, int> = 0,
367 std::enable_if_t<sizeof...(DVects) != 0, int> = 0>
368 KOKKOS_FUNCTION constexpr reference operator()(DVects const&... dvects) const noexcept
369 {
370 static_assert(
371 SupportType::rank() == (0 + ... + DVects::size()),
372 "Invalid number of dimensions");
373 return DDC_MDSPAN_ACCESS_OP(
374 this->m_allocation_mdspan,
375 detail::array(discrete_vector_type(dvects...)));
376 }
377
378 /** Access to the underlying allocation pointer
379 * @return allocation pointer
380 */
381 KOKKOS_FUNCTION constexpr ElementType* data_handle() const
382 {
383 return base_type::data_handle();
384 }
385
386 /** Provide a mdspan on the memory allocation
387 * @return allocation mdspan
388 */
389 KOKKOS_FUNCTION constexpr allocation_mdspan_type allocation_mdspan() const
390 {
391 return base_type::allocation_mdspan();
392 }
393
394 /** Provide a mdspan on the memory allocation
395 * @return allocation mdspan
396 */
397 KOKKOS_FUNCTION constexpr auto allocation_kokkos_view() const
398 {
399 auto s = this->allocation_mdspan();
400 auto kokkos_layout = detail::build_kokkos_layout(
401 s.extents(),
402 s.mapping(),
403 std::make_index_sequence<SupportType::rank()> {});
404 return Kokkos::View<
405 detail::mdspan_to_kokkos_element_t<ElementType, SupportType::rank()>,
406 decltype(kokkos_layout),
407 MemorySpace>(s.data_handle(), kokkos_layout);
408 }
409
410 KOKKOS_FUNCTION constexpr view_type span_cview() const
411 {
412 return view_type(*this);
413 }
414
415 KOKKOS_FUNCTION constexpr span_type span_view() const
416 {
417 return *this;
418 }
419};
420
421template <class DataType, class... Properties, class SupportType>
423 Kokkos::View<DataType, Properties...> const& view,
424 SupportType domain)
425 -> ChunkSpan<
426 detail::kokkos_to_mdspan_element_t<
427 typename Kokkos::View<DataType, Properties...>::data_type>,
428 SupportType,
429 detail::kokkos_to_mdspan_layout_t<
430 typename Kokkos::View<DataType, Properties...>::array_layout>,
431 typename Kokkos::View<DataType, Properties...>::memory_space>;
432
433template <class ElementType, class SupportType, class Allocator>
434ChunkSpan(Chunk<ElementType, SupportType, Allocator>& other) -> ChunkSpan<
435 ElementType,
436 SupportType,
437 Kokkos::layout_right,
438 typename Allocator::memory_space>;
439
440template <class ElementType, class SupportType, class Allocator>
441ChunkSpan(Chunk<ElementType, SupportType, Allocator> const& other) -> ChunkSpan<
442 ElementType const,
443 SupportType,
444 Kokkos::layout_right,
445 typename Allocator::memory_space>;
446
447template <
448 class ElementType,
449 class SupportType,
450 class LayoutStridedPolicy = Kokkos::layout_right,
451 class MemorySpace = Kokkos::HostSpace>
452using ChunkView = ChunkSpan<ElementType const, SupportType, LayoutStridedPolicy, MemorySpace>;
453
454} // namespace ddc
friend class ChunkSpan
KOKKOS_DEFAULTED_FUNCTION constexpr ChunkSpan(ChunkSpan &&other) noexcept=default
Constructs a new ChunkSpan by move.
KOKKOS_FUNCTION constexpr view_type span_cview() const
KOKKOS_FUNCTION constexpr ChunkSpan(Chunk< OElementType, SupportType, Allocator > const &other) noexcept
Constructs a new ChunkSpan from a Chunk, yields a new view to the same data.
KOKKOS_FUNCTION constexpr ChunkSpan(allocation_mdspan_type allocation_mdspan, SupportType const &domain)
Constructs a new ChunkSpan from scratch.
KOKKOS_FUNCTION constexpr auto operator[](DiscreteElement< QueryDDims... > const &slice_spec) const
Slice out some dimensions.
KOKKOS_FUNCTION constexpr ChunkSpan(KokkosView const &view, SupportType const &domain) noexcept
Constructs a new ChunkSpan from scratch.
ChunkSpan(Chunk< OElementType, SupportType, Allocator > &&other) noexcept=delete
Forbids to construct a ChunkSpan from a rvalue of type Chunk.
KOKKOS_FUNCTION constexpr auto operator[](DiscreteDomain< QueryDDims... > const &odomain) const
Restrict to a subdomain, only valid when SupportType is a DiscreteDomain.
KOKKOS_DEFAULTED_FUNCTION constexpr ChunkSpan(ChunkSpan const &other)=default
Constructs a new ChunkSpan by copy, yields a new view to the same data.
static KOKKOS_FUNCTION constexpr auto get_slicer_for(DiscreteDomain< ODDims... > const &c, DiscreteDomain< OODDims... > const &origin)
KOKKOS_FUNCTION constexpr ChunkSpan(ElementType *const ptr, SupportType const &domain)
Constructs a new ChunkSpan from scratch.
static KOKKOS_FUNCTION constexpr auto get_slicer_for(DiscreteVector< ODDims... > const &c)
KOKKOS_DEFAULTED_FUNCTION constexpr ChunkSpan()=default
Empty ChunkSpan.
KOKKOS_FUNCTION constexpr allocation_mdspan_type allocation_mdspan() const
Provide a mdspan on the memory allocation.
KOKKOS_FUNCTION constexpr ElementType * data_handle() const
Access to the underlying allocation pointer.
KOKKOS_FUNCTION constexpr auto allocation_kokkos_view() const
Provide a mdspan on the memory allocation.
KOKKOS_DEFAULTED_FUNCTION constexpr ChunkSpan & operator=(ChunkSpan const &other)=default
Copy-assigns a new value to this ChunkSpan, yields a new view to the same data.
KOKKOS_FUNCTION constexpr ChunkSpan(Chunk< OElementType, SupportType, Allocator > &other) noexcept
Constructs a new ChunkSpan from a Chunk, yields a new view to the same data.
KOKKOS_FUNCTION constexpr ChunkSpan(ChunkSpan< OElementType, SupportType, layout_type, MemorySpace > const &other) noexcept
Constructs a new ChunkSpan by copy of a chunk, yields a new view to the same data.
KOKKOS_DEFAULTED_FUNCTION ~ChunkSpan() noexcept=default
KOKKOS_FUNCTION constexpr span_type span_view() const
KOKKOS_FUNCTION constexpr reference operator()(DElems const &... delems) const noexcept
Element access using a list of DiscreteElement.
KOKKOS_DEFAULTED_FUNCTION constexpr ChunkSpan & operator=(ChunkSpan &&other) noexcept=default
Move-assigns a new value to this ChunkSpan.
KOKKOS_FUNCTION constexpr reference operator()(DVects const &... dvects) const noexcept
Element access using a list of DiscreteVector.
friend class Chunk
Definition chunk.hpp:83
friend class DiscreteDomain
KOKKOS_FUNCTION constexpr bool operator!=(DiscreteVector< OTags... > const &rhs) const noexcept
The top-level namespace of DDC.
constexpr bool enable_borrowed_chunk< ChunkSpan< ElementType, SupportType, LayoutStridedPolicy, MemorySpace > >
ChunkSpan(Chunk< ElementType, SupportType, Allocator > const &other) -> ChunkSpan< ElementType const, SupportType, Kokkos::layout_right, typename Allocator::memory_space >
ChunkSpan(Chunk< ElementType, SupportType, Allocator > &other) -> ChunkSpan< ElementType, SupportType, Kokkos::layout_right, typename Allocator::memory_space >
constexpr bool enable_chunk< ChunkSpan< ElementType, SupportType, LayoutStridedPolicy, MemorySpace > >
KOKKOS_DEDUCTION_GUIDE ChunkSpan(Kokkos::View< DataType, Properties... > const &view, SupportType domain) -> ChunkSpan< detail::kokkos_to_mdspan_element_t< typename Kokkos::View< DataType, Properties... >::data_type >, SupportType, detail::kokkos_to_mdspan_layout_t< typename Kokkos::View< DataType, Properties... >::array_layout >, typename Kokkos::View< DataType, Properties... >::memory_space >