20#include <Kokkos_Core.hpp>
22#include "detail/dual_discretization.hpp"
23#include "detail/macros.hpp"
25#if defined(KOKKOS_ENABLE_CUDA)
27#elif defined(KOKKOS_ENABLE_HIP)
28# include <hip/hip_runtime.h>
31#if defined(KOKKOS_ENABLE_CUDA) || defined(KOKKOS_ENABLE_HIP)
32# define DDC_DETAIL_DEVICE_THROW_ON_ERROR(val)
33 ddc::detail::device_throw_on_error((val), #val, __FILE__, __LINE__)
40#if defined(KOKKOS_ENABLE_CUDA)
41void device_throw_on_error(
42 cudaError_t
const err,
43 const char*
const func,
44 const char*
const file,
46#elif defined(KOKKOS_ENABLE_HIP)
47void device_throw_on_error(
49 const char*
const func,
50 const char*
const file,
54template <
class DDim,
class MemorySpace>
55using ddim_impl_t =
typename DDim::
template Impl<DDim, MemorySpace>;
71 alignas(T) Kokkos::Array<std::byte,
sizeof(T)> m_data;
77 return reinterpret_cast<T*>(m_data.data());
83 return *
reinterpret_cast<T*>(m_data.data());
89 return reinterpret_cast<T*>(m_data.data());
94extern std::optional<std::map<std::string, std::function<
void()>>> g_discretization_store;
98inline std::optional<DualDiscretization<DDim>> g_discrete_space_dual;
100#if defined(KOKKOS_ENABLE_CUDA)
103__constant__ GpuProxy<ddim_impl_t<DDim, GlobalVariableDeviceSpace>> g_discrete_space_device;
104#elif defined(KOKKOS_ENABLE_HIP)
108__constant__ GpuProxy<ddim_impl_t<DDim, GlobalVariableDeviceSpace>> g_discrete_space_device;
109#elif defined(KOKKOS_ENABLE_SYCL)
112SYCL_EXTERNAL
inline sycl::ext::oneapi::experimental::device_global<
113 GpuProxy<ddim_impl_t<DDim, GlobalVariableDeviceSpace>>>
114 g_discrete_space_device;
117void display_discretization_store(std::ostream& os);
119template <
class Tuple, std::size_t... Ids>
120auto extract_after(Tuple&& t, std::index_sequence<Ids...>)
122 return std::make_tuple(std::move(std::get<Ids + 1>(t))...);
128
129
130
131template <
class DDim,
class... Args>
135 !std::is_same_v<DDim,
typename DDim::discrete_dimension_type>,
136 "Discrete dimensions should inherit from the discretization, not use an alias");
137 if (detail::g_discrete_space_dual<DDim>) {
138 throw std::runtime_error(
"Discrete space function already initialized.");
140 detail::g_discrete_space_dual<DDim>.emplace(std::forward<Args>(args)...);
141 detail::g_discretization_store->emplace(
typeid(DDim).name(), []() {
142 detail::g_discrete_space_dual<DDim>.reset();
144#if defined(KOKKOS_ENABLE_CUDA)
145 DDC_DETAIL_DEVICE_THROW_ON_ERROR(cudaMemcpyToSymbol(
146 detail::g_discrete_space_device<DDim>,
147 &detail::g_discrete_space_dual<DDim>->get_device(),
148 sizeof(detail::g_discrete_space_dual<DDim>->get_device())));
149#elif defined(KOKKOS_ENABLE_HIP)
150 DDC_DETAIL_DEVICE_THROW_ON_ERROR(hipMemcpyToSymbol(
151 detail::g_discrete_space_device<DDim>,
152 &detail::g_discrete_space_dual<DDim>->get_device(),
153 sizeof(detail::g_discrete_space_dual<DDim>->get_device())));
154#elif defined(KOKKOS_ENABLE_SYCL)
155 Kokkos::DefaultExecutionSpace exec;
156 sycl::queue q = exec.sycl_queue();
157 q.memcpy(detail::g_discrete_space_device<DDim>,
158 &detail::g_discrete_space_dual<DDim>->get_device())
164
165
166
167
168template <
class DDim,
class DDimImpl,
class Arg0>
171 init_discrete_space<DDim>(std::move(std::get<0>(a)));
172 return std::get<1>(a);
176
177
178
179
180template <
class DDim,
class DDimImpl,
class Arg0,
class Arg1,
class... Args>
181std::tuple<Arg0, Arg1, Args...>
init_discrete_space(std::tuple<DDimImpl, Arg0, Arg1, Args...>&& a)
183 init_discrete_space<DDim>(std::move(std::get<0>(a)));
184 return detail::extract_after(std::move(a), std::index_sequence_for<Arg0, Arg1, Args...>());
188
189
190
191
195 return detail::g_discrete_space_dual<DDim>.has_value();
199
200
201
202
203
204template <
class DDim,
class MemorySpace = DDC_CURRENT_KOKKOS_SPACE>
205KOKKOS_FUNCTION detail::ddim_impl_t<DDim, MemorySpace>
const&
discrete_space()
208 if constexpr (std::is_same_v<MemorySpace, Kokkos::HostSpace>) {
209 KOKKOS_ASSERT(is_discrete_space_initialized<DDim>())
210 return detail::g_discrete_space_dual<DDim>->get_host();
212#if defined(KOKKOS_ENABLE_CUDA) || defined(KOKKOS_ENABLE_HIP)
213 else if constexpr (std::is_same_v<MemorySpace, detail::GlobalVariableDeviceSpace>) {
214 return *detail::g_discrete_space_device<DDim>;
216#elif defined(KOKKOS_ENABLE_SYCL)
217 else if constexpr (std::is_same_v<MemorySpace, detail::GlobalVariableDeviceSpace>) {
218 return *detail::g_discrete_space_device<DDim>.get();
222 static_assert(std::is_same_v<MemorySpace, MemorySpace>,
"Memory space not handled");
230 assert(is_discrete_space_initialized<DDim>());
231 return detail::g_discrete_space_dual<DDim>->get_host();
236#if defined(KOKKOS_ENABLE_CUDA) || defined(KOKKOS_ENABLE_HIP)
237# undef DDC_DETAIL_DEVICE_THROW_ON_ERROR
ScopeGuard & operator=(ScopeGuard const &x)=delete
ScopeGuard(ScopeGuard &&x) noexcept=delete
ScopeGuard & operator=(ScopeGuard &&x) noexcept=delete
ScopeGuard(int, char **&)
ScopeGuard(ScopeGuard const &x)=delete
The top-level namespace of DDC.
bool is_discrete_space_initialized() noexcept
void init_discrete_space(Args &&... args)
Initialize (emplace) a global singleton discrete space.
detail::ddim_impl_t< DDim, Kokkos::HostSpace > const & host_discrete_space()
KOKKOS_FUNCTION detail::ddim_impl_t< DDim, MemorySpace > const & discrete_space()
Arg0 init_discrete_space(std::tuple< DDimImpl, Arg0 > &&a)
Move construct a global singleton discrete space and pass through the other argument.
std::tuple< Arg0, Arg1, Args... > init_discrete_space(std::tuple< DDimImpl, Arg0, Arg1, Args... > &&a)
Move construct a global singleton discrete space and pass through remaining arguments.