libcu++ 2.1.0 includes an mdspan implementation for nvcc for c++14 and later. I tried to implement the mdspan sample code from cppreference (https://en.cppreference.com/w/cpp/container/mdspan) with nvcc 12.0 and libcu++ 2.1.0
I noticed two problems.
First, I am unable to construct the mdspan
in the same way as in the example.
no instance of constructor "mdspan" matches the argument list argument types are: (int *, int, int)
Second, the access to mdspan
via operator[]
does not compile
error: no operator "[]" matches these operands operand types are: cuda::std::__4::mdspan<int, cuda::std::__4::extents<std::size_t, 2UL, 3UL, 2UL>, cuda::std::__4::layout_right, cuda::std::__4::default_accessor> [ std::size_t ]
1. How does one specify an extent in the constructor which is not known at compile-time?
2. How to access the data of mdspan
?
Below is my code which does not compile with nvcc -Ilibcudacxx-2.1.0/include/ -std=c++17 main.cu -o main
.
#include <cstddef>
#include <vector>
#include <cstdio>
#include <cuda/std/mdspan>
int main()
{
std::vector v{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12};
//error: no instance of constructor "mdspan" matches the argument list. argument types are: (int *, int, int)
//auto ms2 = cuda::std::mdspan(v.data(), 2, 6);
//auto ms3 = cuda::std::mdspan(v.data(), 2, 3, 2);
//no compilation error with compile-time extents
cuda::std::mdspan<int, cuda::std::extents<std::size_t, 2,6>> ms2(v.data());
cuda::std::mdspan<int, cuda::std::extents<std::size_t, 2,3,2>> ms3(v.data());
// write data using 2D view
for (std::size_t i = 0; i != ms2.extent(0); i++)
for (std::size_t j = 0; j != ms2.extent(1); j++)
//no operator "[]" matches these operands. operand types are: cuda::std::__4::mdspan<int, cuda::std::__4::extents<std::size_t, 2UL, 6UL>, cuda::std::__4::layout_right, cuda::std::__4::default_accessor<int>> [ std::size_t ]
ms2[i, j] = i * 1000 + j;
// read back using 3D view
for (std::size_t i = 0; i != ms3.extent(0); i++)
{
printf("slice @ i = %lu\n", i);
for (std::size_t j = 0; j != ms3.extent(1); j++)
{
for (std::size_t k = 0; k != ms3.extent(2); k++)
printf("%d ", ms3[i, j, k]);
printf("\n");
}
}
}
The libcu++ implementation of
mdspan
is based on the reference implementation by Kokkos which has some caveats in the readme:The reason for the reference implementation using
operator()
instead ofoperator[]
(see section "C++20" above) is thatoperator[]
with multiple arguments is a C++23 feature so it can't be ported to earlier C++ versions. Therefore one needs:and
instead.
The constructor that is used in the sample is implemented for C++17 and later (C++14 does not have the necessary CTAD*/deduction guides), or at least they have a test for it. CTAD might be turned off for some (host-) compilers/versions, but there is an alternative way of specifying dynamic extents without CTAD:
*CTAD: Class Template Argument Deduction