DDC 0.10.0
Loading...
Searching...
No Matches
print.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 <algorithm>
8#include <cstdlib>
9#include <memory>
10#include <ostream>
11#include <sstream>
12#include <type_traits>
13#include <typeinfo>
14#include <utility>
15
16#include <Kokkos_Core.hpp>
17
18#include "chunk_span.hpp"
19#include "discrete_vector.hpp"
20
21#if defined(KOKKOS_COMPILER_GNU) || defined(KOKKOS_COMPILER_CLANG)
22# include <cxxabi.h>
23#endif
24
25namespace ddc {
26namespace detail {
27class ChunkPrinter
28{
29 static constexpr int const s_threshold = 10;
30 // If this ever becomes modifiable by the user, we need to ensure that
31 // s_edgeitems < (s_threshold / 2) stays true.
32 static constexpr int const s_edgeitems = 3;
33
34 std::stringstream m_ss;
35
36 static std::ostream& alignment(std::ostream& os, int level)
37 {
38 for (int i = 0; i <= level; ++i) {
39 os << ' ';
40 }
41 return os;
42 }
43
44 template <class T>
45 std::size_t get_element_width(T const& elem)
46 {
47 m_ss.seekp(0);
48 m_ss << elem;
49 return m_ss.tellp();
50 }
51
52 template <class T>
53 void display_aligned_element(std::ostream& os, T const& elem, std::size_t largest_element)
54 {
55 std::size_t const elem_width = get_element_width(elem);
56
57 for (std::size_t i = 0; i < largest_element - elem_width; ++i) {
58 os << " ";
59 }
60 os << elem;
61 }
62
63 template <class ElementType, class Extents, class Layout, class Accessor>
64 std::ostream& base_case_display(
65 std::ostream& os,
66 Kokkos::mdspan<ElementType, Extents, Layout, Accessor> const& s,
67 std::size_t largest_element,
68 std::size_t beginning,
69 std::size_t end,
70 std::size_t extent)
71 {
72 for (std::size_t i0 = beginning; i0 < end; ++i0) {
73 display_aligned_element(os, s[i0], largest_element);
74 if (i0 < extent - 1) {
75 os << " ";
76 }
77 }
78 return os;
79 }
80
81 template <class ElementType, class Extents, class Layout, class Accessor, std::size_t... Is>
82 std::ostream& recursive_display(
83 std::ostream& os,
84 Kokkos::mdspan<ElementType, Extents, Layout, Accessor> const& s,
85 int level,
86 std::size_t largest_element,
87 std::size_t beginning,
88 std::size_t end,
89 std::index_sequence<Is...>)
90 {
91 for (std::size_t i0 = beginning; i0 < end; ++i0) {
92 print_impl(
93 os,
94 Kokkos::submdspan(s, i0, ((void)Is, Kokkos::full_extent)...),
95 level + 1,
96 largest_element,
97 std::make_index_sequence<sizeof...(Is)>());
98 if (i0 < end - 1) {
99 for (int ndims = 0; ndims < sizeof...(Is); ++ndims) {
100 os << '\n';
101 }
102 alignment(os, level);
103 }
104 }
105
106 return os;
107 }
108
109public:
110 // 0D chunk span
111 template <class ElementType, class Extents, class Layout, class Accessor>
112 std::ostream& print_impl(
113 std::ostream& os,
114 Kokkos::mdspan<ElementType, Extents, Layout, Accessor> const& s,
115 int /*level*/,
116 std::size_t /*largest_element*/,
117 std::index_sequence<>)
118 {
119 return os << *s.data_handle();
120 }
121
122 // Recursively parse the chunk to print it
123 template <
124 class ElementType,
125 class Extents,
126 class Layout,
127 class Accessor,
128 std::size_t I0,
129 std::size_t... Is>
130 std::ostream& print_impl(
131 std::ostream& os,
132 Kokkos::mdspan<ElementType, Extents, Layout, Accessor> const& s,
133 int level,
134 std::size_t largest_element,
135 std::index_sequence<I0, Is...>)
136 {
137 auto extent = s.extent(I0);
138 if constexpr (sizeof...(Is) > 0) {
139 os << '[';
140 if (extent < s_threshold) {
141 recursive_display(
142 os,
143 s,
144 level,
145 largest_element,
146 0,
147 extent,
148 std::make_index_sequence<sizeof...(Is)>());
149 } else {
150 recursive_display(
151 os,
152 s,
153 level,
154 largest_element,
155 0,
156 s_edgeitems,
157 std::make_index_sequence<sizeof...(Is)>());
158 for (int ndims = 0; ndims < sizeof...(Is); ++ndims) {
159 os << '\n';
160 }
161 alignment(os, level);
162 os << "...";
163 for (int ndims = 0; ndims < sizeof...(Is); ++ndims) {
164 os << '\n';
165 }
166 alignment(os, level);
167 recursive_display(
168 os,
169 s,
170 level,
171 largest_element,
172 extent - s_edgeitems,
173 extent,
174 std::make_index_sequence<sizeof...(Is)>());
175 }
176 os << "]";
177 } else {
178 os << "[";
179 if (extent < s_threshold) {
180 base_case_display(os, s, largest_element, 0, extent, extent);
181 } else {
182 base_case_display(os, s, largest_element, 0, s_edgeitems, extent);
183 os << "... ";
184 base_case_display(os, s, largest_element, extent - s_edgeitems, extent, extent);
185 }
186 os << "]";
187 }
188
189 return os;
190 }
191
192 // 0D, we don't need the element size in this case so the actual returned value can be anything.
193 template <class ElementType, class Extents, class Layout, class Accessor>
194 std::size_t find_largest_displayed_element(
195 Kokkos::mdspan<ElementType, Extents, Layout, Accessor> const&,
196 std::index_sequence<>)
197 {
198 return 0;
199 }
200
201 // Find the largest element we have to print to allow alignment (it ignore
202 // element that will be elided).
203 template <
204 class ElementType,
205 class Extents,
206 class Layout,
207 class Accessor,
208 std::size_t I0,
209 std::size_t... Is>
210 std::size_t find_largest_displayed_element(
211 Kokkos::mdspan<ElementType, Extents, Layout, Accessor> const& s,
212 std::index_sequence<I0, Is...>)
213 {
214 std::size_t ret = 0;
215 auto extent = s.extent(I0);
216 if constexpr (sizeof...(Is) > 0) {
217 if (extent < s_threshold) {
218 for (std::size_t i0 = 0; i0 < extent; ++i0) {
219 ret = std::max(
220 ret,
221 find_largest_displayed_element(
222 Kokkos::submdspan(s, i0, ((void)Is, Kokkos::full_extent)...),
223 std::make_index_sequence<sizeof...(Is)>()));
224 }
225 } else {
226 for (std::size_t i0 = 0; i0 < s_edgeitems; ++i0) {
227 ret = std::max(
228 ret,
229 find_largest_displayed_element(
230 Kokkos::submdspan(s, i0, ((void)Is, Kokkos::full_extent)...),
231 std::make_index_sequence<sizeof...(Is)>()));
232 }
233 for (std::size_t i0 = extent - s_edgeitems; i0 < extent; ++i0) {
234 ret = std::max(
235 ret,
236 find_largest_displayed_element(
237 Kokkos::submdspan(s, i0, ((void)Is, Kokkos::full_extent)...),
238 std::make_index_sequence<sizeof...(Is)>()));
239 }
240 }
241 } else {
242 if (extent < s_threshold) {
243 for (std::size_t i0 = 0; i0 < extent; ++i0) {
244 ret = std::max(ret, get_element_width(s[i0]));
245 }
246 } else {
247 for (std::size_t i0 = 0; i0 < s_edgeitems; ++i0) {
248 ret = std::max(ret, get_element_width(s[i0]));
249 }
250 for (std::size_t i0 = extent - s_edgeitems; i0 < extent; ++i0) {
251 ret = std::max(ret, get_element_width(s[i0]));
252 }
253 }
254 }
255
256 return ret;
257 }
258
259 explicit ChunkPrinter(std::ostream const& os)
260 {
261 m_ss.copyfmt(os);
262 }
263};
264
265inline void print_demangled_type_name(std::ostream& os, char const* const mangled_name)
266{
267#if defined(KOKKOS_COMPILER_GNU) || defined(KOKKOS_COMPILER_CLANG)
268 int status;
269
270 std::unique_ptr<char, decltype(std::free)*> const
271 demangled_name(abi::__cxa_demangle(mangled_name, nullptr, nullptr, &status), std::free);
272 if (status != 0) {
273 os << "Error demangling dimension name: " << status;
274 return;
275 }
276
277 os << demangled_name.get();
278#else
279 os << mangled_name;
280#endif
281}
282
283inline void print_single_dim_name(
284 std::ostream& os,
285 std::type_info const& dim,
286 DiscreteVectorElement const size)
287{
288 print_demangled_type_name(os, dim.name());
289 os << '(' << size << ')';
290}
291
292inline void print_dim_name(std::ostream& os, DiscreteVector<> const&)
293{
294 os << "Scalar";
295}
296
297template <class Dim0, class... Dims>
298void print_dim_name(std::ostream& os, DiscreteVector<Dim0, Dims...> const& dd)
299{
300 print_single_dim_name(os, typeid(Dim0), get<Dim0>(dd));
301 ((os << "×", print_single_dim_name(os, typeid(Dims), get<Dims>(dd))), ...);
302}
303
304} // namespace detail
305
306template <class ElementType, class SupportType, class LayoutStridedPolicy, class MemorySpace>
307std::ostream& print_content(
308 std::ostream& os,
309 ChunkSpan<ElementType, SupportType, LayoutStridedPolicy, MemorySpace> const& chunk_span)
310{
311 auto h_chunk_span = create_mirror_view_and_copy(Kokkos::HostSpace(), chunk_span);
312
313 using chunkspan_type = std::remove_cv_t<std::remove_reference_t<decltype(h_chunk_span)>>;
314 using mdspan_type = typename chunkspan_type::allocation_mdspan_type;
315 using extents = typename mdspan_type::extents_type;
316
317 mdspan_type const allocated_mdspan = h_chunk_span.allocation_mdspan();
318
319 ddc::detail::ChunkPrinter printer(os);
320 std::size_t const largest_element = printer.find_largest_displayed_element(
321 allocated_mdspan,
322 std::make_index_sequence<extents::rank()>());
323
324 printer.print_impl(
325 os,
326 allocated_mdspan,
327 0,
328 largest_element,
329 std::make_index_sequence<extents::rank()>());
330
331 return os;
332}
333
334template <class ElementType, class SupportType, class LayoutStridedPolicy, class MemorySpace>
335std::ostream& print_type_info(
336 std::ostream& os,
337 ChunkSpan<ElementType, SupportType, LayoutStridedPolicy, MemorySpace> const& chunk_span)
338{
339 ddc::detail::print_dim_name(os, chunk_span.extents());
340 os << '\n';
341 ddc::detail::print_demangled_type_name(os, typeid(chunk_span).name());
342 os << '\n';
343
344 return os;
345}
346
347template <class ElementType, class SupportType, class LayoutStridedPolicy, class MemorySpace>
348std::ostream& print(
349 std::ostream& os,
350 ChunkSpan<ElementType, SupportType, LayoutStridedPolicy, MemorySpace> const& chunk_span)
351{
352 print_type_info(os, chunk_span);
353 print_content(os, chunk_span);
354
355 return os;
356}
357
358
359template <class ElementType, class SupportType, class LayoutStridedPolicy, class MemorySpace>
360std::ostream& operator<<(
361 std::ostream& os,
362 ChunkSpan<ElementType, SupportType, LayoutStridedPolicy, MemorySpace> const& chunk_span)
363{
364 return print(os, chunk_span);
365}
366
367} // namespace ddc
friend class ChunkSpan
KOKKOS_FUNCTION constexpr bool operator!=(DiscreteVector< OTags... > const &rhs) const noexcept
The top-level namespace of DDC.
std::ostream & print_content(std::ostream &os, ChunkSpan< ElementType, SupportType, LayoutStridedPolicy, MemorySpace > const &chunk_span)
Definition print.hpp:307
std::ostream & print(std::ostream &os, ChunkSpan< ElementType, SupportType, LayoutStridedPolicy, MemorySpace > const &chunk_span)
Definition print.hpp:348
std::ostream & print_type_info(std::ostream &os, ChunkSpan< ElementType, SupportType, LayoutStridedPolicy, MemorySpace > const &chunk_span)
Definition print.hpp:335