|
| 1 | +// RUN: %{build} -o %t.out |
| 2 | +// RUN: %{run} %t.out |
| 3 | + |
| 4 | +// This test verifies whether sycl::accessor can be used with free function |
| 5 | +// kernels extension. |
| 6 | + |
| 7 | +#include <sycl/ext/oneapi/free_function_queries.hpp> |
| 8 | + |
| 9 | +#include "helpers.hpp" |
| 10 | + |
| 11 | +template <int Dims> |
| 12 | +SYCL_EXT_ONEAPI_FUNCTION_PROPERTY((syclexp::single_task_kernel)) |
| 13 | +void globalScopeSingleFreeFunc( |
| 14 | + sycl::accessor<int, Dims, sycl::access::mode::read_write, |
| 15 | + sycl::access::target::device, |
| 16 | + sycl::access::placeholder::false_t> |
| 17 | + Accessor, |
| 18 | + int Value) { |
| 19 | + for (auto &Elem : Accessor) |
| 20 | + Elem = Value; |
| 21 | +} |
| 22 | +namespace ns { |
| 23 | +template <int Dims> |
| 24 | +SYCL_EXT_ONEAPI_FUNCTION_PROPERTY((syclexp::nd_range_kernel<Dims>)) |
| 25 | +void nsNdRangeFreeFunc(sycl::accessor<int, Dims, sycl::access::mode::read_write, |
| 26 | + sycl::access::target::device, |
| 27 | + sycl::access::placeholder::false_t> |
| 28 | + Accessor, |
| 29 | + int Value) { |
| 30 | + auto Item = syclext::this_work_item::get_nd_item<Dims>().get_global_id(); |
| 31 | + Accessor[Item] = Value; |
| 32 | +} |
| 33 | +} // namespace ns |
| 34 | + |
| 35 | +template <int Dims> |
| 36 | +SYCL_EXT_ONEAPI_FUNCTION_PROPERTY((syclexp::nd_range_kernel<Dims>)) |
| 37 | +void ndRangeFreeFuncMultipleParameters( |
| 38 | + sycl::accessor<int, Dims, sycl::access::mode::read, |
| 39 | + sycl::access::target::device, |
| 40 | + sycl::access::placeholder::false_t> |
| 41 | + InputAAcc, |
| 42 | + sycl::accessor<int, Dims, sycl::access::mode::read, |
| 43 | + sycl::access::target::device, |
| 44 | + sycl::access::placeholder::false_t> |
| 45 | + InputBAcc, |
| 46 | + sycl::accessor<int, Dims, sycl::access::mode::write, |
| 47 | + sycl::access::target::device, |
| 48 | + sycl::access::placeholder::false_t> |
| 49 | + ResultAcc) { |
| 50 | + auto Item = syclext::this_work_item::get_nd_item<Dims>().get_global_id(); |
| 51 | + ResultAcc[Item] = InputAAcc[Item] + InputBAcc[Item]; |
| 52 | +} |
| 53 | + |
| 54 | +// TODO: Need to add checks for a static member functions of a class as free |
| 55 | +// function kernel. |
| 56 | + |
| 57 | +template <auto Func, size_t Dims> |
| 58 | +int runSingleTaskTest(sycl::queue &Queue, sycl::context &Context, |
| 59 | + sycl::range<Dims> NumOfElementsPerDim, |
| 60 | + std::string_view ErrorMessage, |
| 61 | + const int ExpectedResultValue) { |
| 62 | + sycl::kernel UsedKernel = getKernel<Func>(Context); |
| 63 | + std::vector<int> ResultData(NumOfElementsPerDim.size(), 0); |
| 64 | + { |
| 65 | + sycl::buffer<int, Dims> Buffer(ResultData.data(), NumOfElementsPerDim); |
| 66 | + Queue.submit([&](sycl::handler &Handler) { |
| 67 | + sycl::accessor<int, Dims> Accessor{Buffer, Handler}; |
| 68 | + Handler.set_args(Accessor, ExpectedResultValue); |
| 69 | + Handler.single_task(UsedKernel); |
| 70 | + }); |
| 71 | + } |
| 72 | + return performResultCheck(NumOfElementsPerDim.size(), ResultData.data(), |
| 73 | + ErrorMessage, ExpectedResultValue); |
| 74 | +} |
| 75 | + |
| 76 | +template <auto Func, size_t Dims> |
| 77 | +int runNdRangeTest(sycl::queue &Queue, sycl::context &Context, |
| 78 | + sycl::nd_range<Dims> NdRange, std::string_view ErrorMessage, |
| 79 | + const int ExpectedResultValue) { |
| 80 | + sycl::kernel UsedKernel = getKernel<Func>(Context); |
| 81 | + std::vector<int> ResultData(NdRange.get_global_range().size(), 0); |
| 82 | + { |
| 83 | + sycl::buffer<int, Dims> Buffer(ResultData.data(), |
| 84 | + NdRange.get_global_range()); |
| 85 | + Queue.submit([&](sycl::handler &Handler) { |
| 86 | + sycl::accessor<int, Dims> Accessor{Buffer, Handler}; |
| 87 | + Handler.set_args(Accessor, ExpectedResultValue); |
| 88 | + Handler.parallel_for(NdRange, UsedKernel); |
| 89 | + }); |
| 90 | + } |
| 91 | + return performResultCheck(NdRange.get_global_range().size(), |
| 92 | + ResultData.data(), ErrorMessage, |
| 93 | + ExpectedResultValue); |
| 94 | +} |
| 95 | + |
| 96 | +template <auto Func, size_t Dims> |
| 97 | +int runNdRangeTestMultipleParameters(sycl::queue &Queue, sycl::context &Context, |
| 98 | + sycl::nd_range<Dims> NdRange, |
| 99 | + std::string_view ErrorMessage, |
| 100 | + sycl::range<3> Values) { |
| 101 | + sycl::kernel UsedKernel = getKernel<Func>(Context); |
| 102 | + std::vector<int> InputAData(NdRange.get_global_range().size(), Values[0]); |
| 103 | + std::vector<int> InputBData(NdRange.get_global_range().size(), Values[1]); |
| 104 | + std::vector<int> ResultData(NdRange.get_global_range().size(), 0); |
| 105 | + |
| 106 | + { |
| 107 | + sycl::buffer<int, Dims> InputABuffer(InputAData.data(), |
| 108 | + NdRange.get_global_range()); |
| 109 | + sycl::buffer<int, Dims> InputBBuffer(InputBData.data(), |
| 110 | + NdRange.get_global_range()); |
| 111 | + sycl::buffer<int, Dims> ResultBuffer(ResultData.data(), |
| 112 | + NdRange.get_global_range()); |
| 113 | + Queue.submit([&](sycl::handler &Handler) { |
| 114 | + sycl::accessor<int, Dims, sycl::access::mode::read, |
| 115 | + sycl::access::target::device> |
| 116 | + InputAAcc{InputABuffer, Handler}; |
| 117 | + sycl::accessor<int, Dims, sycl::access::mode::read, |
| 118 | + sycl::access::target::device> |
| 119 | + InputBAcc{InputBBuffer, Handler}; |
| 120 | + sycl::accessor<int, Dims, sycl::access::mode::write> ResultAcc{ |
| 121 | + ResultBuffer, Handler}; |
| 122 | + Handler.set_args(InputAAcc, InputBAcc, ResultAcc); |
| 123 | + Handler.parallel_for(NdRange, UsedKernel); |
| 124 | + }); |
| 125 | + } |
| 126 | + return performResultCheck(NdRange.get_global_range().size(), |
| 127 | + ResultData.data(), ErrorMessage, Values[2]); |
| 128 | +} |
| 129 | + |
| 130 | +int main() { |
| 131 | + |
| 132 | + int Failed = 0; |
| 133 | + sycl::queue Queue; |
| 134 | + sycl::context Context = Queue.get_context(); |
| 135 | + { |
| 136 | + // Check that sycl::accessor is supported inside single_task free function |
| 137 | + // kernel |
| 138 | + Failed += runSingleTaskTest<globalScopeSingleFreeFunc<1>, 1>( |
| 139 | + Queue, Context, sycl::range<1>{10}, |
| 140 | + "globalScopeSingleFreeFunc with sycl::accessor<1>", 1); |
| 141 | + Failed += runSingleTaskTest<globalScopeSingleFreeFunc<2>, 2>( |
| 142 | + Queue, Context, sycl::range<2>{10, 10}, |
| 143 | + "globalScopeSingleFreeFunc with sycl::accessor<2>", 2); |
| 144 | + Failed += runSingleTaskTest<globalScopeSingleFreeFunc<3>, 3>( |
| 145 | + Queue, Context, sycl::range<3>{5, 5, 5}, |
| 146 | + "globalScopeSingleFreeFunc with sycl::accessor<3>", 3); |
| 147 | + } |
| 148 | + |
| 149 | + { |
| 150 | + // Check that sycl::accessor is supported inside nd_range free function |
| 151 | + // kernel |
| 152 | + Failed += runNdRangeTest<ns::nsNdRangeFreeFunc<1>, 1>( |
| 153 | + Queue, Context, sycl::nd_range{sycl::range{10}, sycl::range{2}}, |
| 154 | + "ns::nsNdRangeFreeFunc with sycl::accessor<1>", 4); |
| 155 | + Failed += runNdRangeTest<ns::nsNdRangeFreeFunc<2>, 2>( |
| 156 | + Queue, Context, sycl::nd_range{sycl::range{16, 16}, sycl::range{4, 4}}, |
| 157 | + "ns::nsNdRangeFreeFunc with sycl::accessor<2>", 5); |
| 158 | + Failed += runNdRangeTest<ns::nsNdRangeFreeFunc<3>, 3>( |
| 159 | + Queue, Context, |
| 160 | + sycl::nd_range{sycl::range{10, 10, 10}, sycl::range{2, 2, 2}}, |
| 161 | + "ns::nsNdRangeFreeFunc with sycl::accessor<3>", 6); |
| 162 | + } |
| 163 | + |
| 164 | + { |
| 165 | + // Check that multiple sycl::accessor are supported inside nd_range free |
| 166 | + // function kernel |
| 167 | + Failed += |
| 168 | + runNdRangeTestMultipleParameters<ndRangeFreeFuncMultipleParameters<1>, |
| 169 | + 1>( |
| 170 | + Queue, Context, sycl::nd_range{sycl::range{10}, sycl::range{2}}, |
| 171 | + "ndRangeFreeFuncMultipleParameters with multiple sycl::accessor<1>", |
| 172 | + sycl::range{111, 111, 222}); |
| 173 | + Failed += |
| 174 | + runNdRangeTestMultipleParameters<ndRangeFreeFuncMultipleParameters<2>, |
| 175 | + 2>( |
| 176 | + Queue, Context, |
| 177 | + sycl::nd_range{sycl::range{16, 16}, sycl::range{4, 4}}, |
| 178 | + "ndRangeFreeFuncMultipleParameters with multiple sycl::accessor<2>", |
| 179 | + sycl::range{222, 222, 444}); |
| 180 | + Failed += |
| 181 | + runNdRangeTestMultipleParameters<ndRangeFreeFuncMultipleParameters<3>, |
| 182 | + 3>( |
| 183 | + Queue, Context, |
| 184 | + sycl::nd_range{sycl::range{10, 10, 10}, sycl::range{2, 2, 2}}, |
| 185 | + "ndRangeFreeFuncMultipleParameters with multiple sycl::accessor<3>", |
| 186 | + sycl::range{444, 444, 888}); |
| 187 | + } |
| 188 | + return Failed; |
| 189 | +} |
0 commit comments