include/cuda/experimental/__launch/launch.cuh
File members: include/cuda/experimental/__launch/launch.cuh
//===----------------------------------------------------------------------===//
//
// Part of CUDA Experimental in CUDA C++ Core Libraries,
// under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
// SPDX-FileCopyrightText: Copyright (c) 2024 NVIDIA CORPORATION & AFFILIATES.
//
//===----------------------------------------------------------------------===//
#ifndef _CUDAX__LAUNCH_LAUNCH
#define _CUDAX__LAUNCH_LAUNCH
#include <cuda_runtime.h>
#include <cuda/std/__exception/cuda_error.h>
#include <cuda/stream_ref>
#include <cuda/experimental/__launch/configuration.cuh>
#include <cuda/experimental/__launch/launch_transform.cuh>
#include <cuda/experimental/__utility/ensure_current_device.cuh>
#if _CCCL_STD_VER >= 2017
namespace cuda::experimental
{
namespace detail
{
template <typename Config, typename Kernel, class... Args>
__global__ void kernel_launcher(const Config conf, Kernel kernel_fn, Args... args)
{
kernel_fn(conf, args...);
}
template <typename Kernel, class... Args>
__global__ void kernel_launcher_no_config(Kernel kernel_fn, Args... args)
{
kernel_fn(args...);
}
template <typename Config, typename Kernel, typename... Args>
_CCCL_NODISCARD cudaError_t
launch_impl(::cuda::stream_ref stream, Config conf, const Kernel& kernel_fn, const Args&... args)
{
static_assert(!::cuda::std::is_same_v<decltype(conf.dims), uninit_t>,
"Can't launch a configuration without hierarchy dimensions");
cudaLaunchConfig_t config{};
cudaError_t status = cudaSuccess;
constexpr bool has_cluster_level = has_level<cluster_level, decltype(conf.dims)>;
constexpr unsigned int num_attrs_needed = detail::kernel_config_count_attr_space(conf) + has_cluster_level;
cudaLaunchAttribute attrs[num_attrs_needed == 0 ? 1 : num_attrs_needed];
config.attrs = &attrs[0];
config.numAttrs = 0;
config.stream = stream.get();
status = detail::apply_kernel_config(conf, config, reinterpret_cast<void*>(kernel_fn));
if (status != cudaSuccess)
{
return status;
}
config.blockDim = conf.dims.extents(thread, block);
config.gridDim = conf.dims.extents(block, grid);
if constexpr (has_cluster_level)
{
auto cluster_dims = conf.dims.extents(block, cluster);
config.attrs[config.numAttrs].id = cudaLaunchAttributeClusterDimension;
config.attrs[config.numAttrs].val.clusterDim = {
static_cast<unsigned int>(cluster_dims.x),
static_cast<unsigned int>(cluster_dims.y),
static_cast<unsigned int>(cluster_dims.z)};
config.numAttrs++;
}
// TODO lower to cudaLaunchKernelExC?
return cudaLaunchKernelEx(&config, kernel_fn, args...);
}
} // namespace detail
template <typename... Args, typename... Config, typename Dimensions, typename Kernel>
void launch(
::cuda::stream_ref stream, const kernel_config<Dimensions, Config...>& conf, const Kernel& kernel, Args&&... args)
{
__ensure_current_device __dev_setter(stream);
cudaError_t status;
auto combined = conf.combine_with_default(kernel);
if constexpr (::cuda::std::is_invocable_v<Kernel, kernel_config<Dimensions, Config...>, as_kernel_arg_t<Args>...>)
{
auto launcher = detail::kernel_launcher<decltype(combined), Kernel, as_kernel_arg_t<Args>...>;
status = detail::launch_impl(
stream,
combined,
launcher,
combined,
kernel,
static_cast<as_kernel_arg_t<Args>>(detail::__launch_transform(stream, std::forward<Args>(args)))...);
}
else
{
static_assert(::cuda::std::is_invocable_v<Kernel, as_kernel_arg_t<Args>...>);
auto launcher = detail::kernel_launcher_no_config<Kernel, as_kernel_arg_t<Args>...>;
status = detail::launch_impl(
stream,
combined,
launcher,
kernel,
static_cast<as_kernel_arg_t<Args>>(detail::__launch_transform(stream, std::forward<Args>(args)))...);
}
if (status != cudaSuccess)
{
::cuda::__throw_cuda_error(status, "Failed to launch a kernel");
}
}
template <typename... ExpArgs, typename... ActArgs, typename... Config, typename Dimensions>
void launch(::cuda::stream_ref stream,
const kernel_config<Dimensions, Config...>& conf,
void (*kernel)(kernel_config<Dimensions, Config...>, ExpArgs...),
ActArgs&&... args)
{
__ensure_current_device __dev_setter(stream);
cudaError_t status = detail::launch_impl(
stream, //
conf,
kernel,
conf,
static_cast<as_kernel_arg_t<ActArgs>>(detail::__launch_transform(stream, std::forward<ActArgs>(args)))...);
if (status != cudaSuccess)
{
::cuda::__throw_cuda_error(status, "Failed to launch a kernel");
}
}
template <typename... ExpArgs, typename... ActArgs, typename... Config, typename Dimensions>
void launch(::cuda::stream_ref stream,
const kernel_config<Dimensions, Config...>& conf,
void (*kernel)(ExpArgs...),
ActArgs&&... args)
{
__ensure_current_device __dev_setter(stream);
cudaError_t status = detail::launch_impl(
stream, //
conf,
kernel,
static_cast<as_kernel_arg_t<ActArgs>>(detail::__launch_transform(stream, std::forward<ActArgs>(args)))...);
if (status != cudaSuccess)
{
::cuda::__throw_cuda_error(status, "Failed to launch a kernel");
}
}
} // namespace cuda::experimental
#endif // _CCCL_STD_VER >= 2017
#endif // _CUDAX__LAUNCH_LAUNCH