Skip to content

Commit 49e1e74

Browse files
[SYCL] Hide SYCL service kernels (#4519)
The SYCL runtime may in places benefit from being able to define "service kernels" to use behind-the-scenes work on devices. These kernels may use the same interface as other SYCL kernels, but as such will be handled no different than user-defined kernels. These changes makes a distinction between service kernels and user-defined kernels by placing service kernels in a common namespace (`cl::sycl::detail::__sycl_service_kernel__`). The program manager will not grant unique kernel IDs to service kernels. This prevents service kernels from being visible to the user through kernel_bundle related interfaces. Consequently the runtime may create device images that do not contain any kernel IDs. As such, `sycl::has_kernel_bundle` will now discount any device images that have no kernel IDs. Signed-off-by: Steffen Larsen <steffen.larsen@intel.com>
1 parent 239fb5e commit 49e1e74

File tree

6 files changed

+107
-11
lines changed

6 files changed

+107
-11
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
//==-------- service_kernels.hpp - SYCL service kernel name types ----------==//
2+
//
3+
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4+
// See https://llvm.org/LICENSE.txt for license information.
5+
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6+
//
7+
//===----------------------------------------------------------------------===//
8+
9+
#pragma once
10+
11+
__SYCL_INLINE_NAMESPACE(cl) {
12+
namespace sycl {
13+
namespace detail {
14+
namespace __sycl_service_kernel__ {
15+
16+
class AssertInfoCopier;
17+
18+
} // namespace __sycl_service_kernel__
19+
} // namespace detail
20+
} // namespace sycl
21+
} // __SYCL_INLINE_NAMESPACE(cl)

sycl/include/CL/sycl/queue.hpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212
#include <CL/sycl/detail/assert_happened.hpp>
1313
#include <CL/sycl/detail/common.hpp>
1414
#include <CL/sycl/detail/export.hpp>
15+
#include <CL/sycl/detail/service_kernel_names.hpp>
1516
#include <CL/sycl/device.hpp>
1617
#include <CL/sycl/device_selector.hpp>
1718
#include <CL/sycl/event.hpp>
@@ -79,7 +80,6 @@ class queue;
7980
namespace detail {
8081
class queue_impl;
8182
#if __SYCL_USE_FALLBACK_ASSERT
82-
class AssertInfoCopier;
8383
static event submitAssertCapture(queue &, event &, queue *,
8484
const detail::code_location &);
8585
#endif
@@ -1172,7 +1172,7 @@ event submitAssertCapture(queue &Self, event &Event, queue *SecondaryQueue,
11721172

11731173
auto Acc = Buffer.get_access<access::mode::write>(CGH);
11741174

1175-
CGH.single_task<AssertInfoCopier>([Acc] {
1175+
CGH.single_task<__sycl_service_kernel__::AssertInfoCopier>([Acc] {
11761176
#ifdef __SYCL_DEVICE_ONLY__
11771177
__devicelib_assert_read(&Acc[0]);
11781178
#else

sycl/source/detail/program_manager/program_manager.cpp

+24-3
Original file line numberDiff line numberDiff line change
@@ -1064,6 +1064,16 @@ void ProgramManager::addImages(pi_device_binaries DeviceBinary) {
10641064
auto Result = KSIdMap.insert(std::make_pair(EntriesIt->name, KSId));
10651065
(void)Result;
10661066
assert(Result.second && "Kernel sets are not disjoint");
1067+
1068+
// Skip creating unique kernel ID if it is a service kernel.
1069+
// SYCL service kernels are identified by having
1070+
// __sycl_service_kernel__ in the mangled name, primarily as part of
1071+
// the namespace of the name type.
1072+
if (std::strstr(EntriesIt->name, "__sycl_service_kernel__")) {
1073+
m_ServiceKernels.insert(EntriesIt->name);
1074+
continue;
1075+
}
1076+
10671077
// ... and create a unique kernel ID for the entry
10681078
std::shared_ptr<detail::kernel_id_impl> KernelIDImpl =
10691079
std::make_shared<detail::kernel_id_impl>(EntriesIt->name);
@@ -1352,7 +1362,6 @@ ProgramManager::getSYCLDeviceImagesWithCompatibleState(
13521362
if (!compatibleWithDevice(BinImage, Dev))
13531363
continue;
13541364

1355-
// TODO: Cache kernel_ids
13561365
std::vector<sycl::kernel_id> KernelIDs;
13571366
// Collect kernel names for the image
13581367
pi_device_binary DevBin =
@@ -1362,11 +1371,23 @@ ProgramManager::getSYCLDeviceImagesWithCompatibleState(
13621371
for (_pi_offload_entry EntriesIt = DevBin->EntriesBegin;
13631372
EntriesIt != DevBin->EntriesEnd; ++EntriesIt) {
13641373
auto KernelID = m_KernelIDs.find(EntriesIt->name);
1365-
assert(KernelID != m_KernelIDs.end() &&
1366-
"Kernel ID in device binary missing from cache");
1374+
1375+
if (KernelID == m_KernelIDs.end()) {
1376+
// Service kernels do not have kernel IDs
1377+
assert(m_ServiceKernels.find(EntriesIt->name) !=
1378+
m_ServiceKernels.end() &&
1379+
"Kernel ID in device binary missing from cache");
1380+
continue;
1381+
}
1382+
13671383
KernelIDs.push_back(KernelID->second);
13681384
}
13691385
}
1386+
1387+
// If the image does not contain any non-service kernels we can skip it.
1388+
if (KernelIDs.empty())
1389+
continue;
1390+
13701391
// device_image_impl expects kernel ids to be sorted for fast search
13711392
std::sort(KernelIDs.begin(), KernelIDs.end(), LessByNameComp{});
13721393

sycl/source/detail/program_manager/program_manager.hpp

+10-1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@
2323
#include <memory>
2424
#include <set>
2525
#include <unordered_map>
26+
#include <unordered_set>
2627
#include <vector>
2728

2829
// +++ Entry points referenced by the offload wrapper object {
@@ -281,7 +282,7 @@ class ProgramManager {
281282
/// Maps names of kernels to their unique kernel IDs.
282283
/// TODO: Use std::unordered_set with transparent hash and equality functions
283284
/// when C++20 is enabled for the runtime library.
284-
/// Access must be guarded by the m_KernelIDsMutex mutex
285+
/// Access must be guarded by the m_KernelIDsMutex mutex.
285286
std::unordered_map<std::string, kernel_id> m_KernelIDs;
286287

287288
/// Protects kernel ID cache.
@@ -290,6 +291,14 @@ class ProgramManager {
290291
/// \ref Sync::getGlobalLock() while holding this mutex.
291292
std::mutex m_KernelIDsMutex;
292293

294+
/// Caches all found service kernels to expedite future checks. A SYCL service
295+
/// kernel is a kernel that has not been defined by the user but is instead
296+
/// generated by the SYCL runtime. Service kernel name types must be declared
297+
/// in the sycl::detail::__sycl_service_kernel__ namespace which is
298+
/// exclusively used for this purpose.
299+
/// Access must be guarded by the m_KernelIDsMutex mutex.
300+
std::unordered_set<std::string> m_ServiceKernels;
301+
293302
// Keeps track of pi_program to image correspondence. Needed for:
294303
// - knowing which specialization constants are used in the program and
295304
// injecting their current values before compiling the SPIR-V; the binary

sycl/unittests/SYCL2020/KernelID.cpp

+31-1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
class TestKernel1;
1818
class TestKernel2;
1919
class TestKernel3;
20+
class ServiceKernel1;
2021

2122
__SYCL_INLINE_NAMESPACE(cl) {
2223
namespace sycl {
@@ -57,6 +58,19 @@ template <> struct KernelInfo<TestKernel3> {
5758
static constexpr bool callsAnyThisFreeFunction() { return false; }
5859
};
5960

61+
template <> struct KernelInfo<ServiceKernel1> {
62+
static constexpr unsigned getNumParams() { return 0; }
63+
static const kernel_param_desc_t &getParamDesc(int) {
64+
static kernel_param_desc_t Dummy;
65+
return Dummy;
66+
}
67+
static constexpr const char *getName() {
68+
return "_ZTSN2cl4sycl6detail23__sycl_service_kernel__14ServiceKernel1";
69+
}
70+
static constexpr bool isESIMD() { return false; }
71+
static constexpr bool callsThisItem() { return false; }
72+
static constexpr bool callsAnyThisFreeFunction() { return false; }
73+
};
6074
} // namespace detail
6175
} // namespace sycl
6276
} // __SYCL_INLINE_NAMESPACE(cl)
@@ -84,7 +98,9 @@ generateDefaultImage(std::initializer_list<std::string> Kernels) {
8498

8599
static sycl::unittest::PiImage Imgs[2] = {
86100
generateDefaultImage({"KernelID_TestKernel1", "KernelID_TestKernel3"}),
87-
generateDefaultImage({"KernelID_TestKernel2"})};
101+
generateDefaultImage(
102+
{"KernelID_TestKernel2",
103+
"_ZTSN2cl4sycl6detail23__sycl_service_kernel__14ServiceKernel1"})};
88104
static sycl::unittest::PiImageArray<2> ImgArray{Imgs};
89105

90106
TEST(KernelID, AllProgramKernelIds) {
@@ -106,6 +122,20 @@ TEST(KernelID, AllProgramKernelIds) {
106122
}
107123
}
108124

125+
TEST(KernelID, NoServiceKernelIds) {
126+
const char *ServiceKernel1Name =
127+
sycl::detail::KernelInfo<ServiceKernel1>::getName();
128+
129+
std::vector<sycl::kernel_id> AllKernelIDs = sycl::get_kernel_ids();
130+
131+
auto NoFoundServiceKernelID = std::none_of(
132+
AllKernelIDs.begin(), AllKernelIDs.end(), [=](sycl::kernel_id KernelID) {
133+
return strcmp(KernelID.get_name(), ServiceKernel1Name) == 0;
134+
});
135+
136+
EXPECT_TRUE(NoFoundServiceKernelID);
137+
}
138+
109139
TEST(KernelID, FreeKernelIDEqualsKernelBundleId) {
110140
sycl::platform Plt{sycl::default_selector()};
111141
if (Plt.is_host()) {

sycl/unittests/assert/assert.cpp

+19-4
Original file line numberDiff line numberDiff line change
@@ -51,9 +51,10 @@ template <> struct KernelInfo<TestKernel> {
5151
static constexpr const kernel_param_desc_t Signatures[] = {
5252
{kernel_param_kind_t::kind_accessor, 4062, 0}};
5353

54-
template <> struct KernelInfo<::sycl::detail::AssertInfoCopier> {
54+
template <>
55+
struct KernelInfo<::sycl::detail::__sycl_service_kernel__::AssertInfoCopier> {
5556
static constexpr const char *getName() {
56-
return "_ZTSN2cl4sycl6detail16AssertInfoCopierE";
57+
return "_ZTSN2cl4sycl6detail23__sycl_service_kernel__16AssertInfoCopierE";
5758
}
5859
static constexpr unsigned getNumParams() { return 1; }
5960
static constexpr const kernel_param_desc_t &getParamDesc(unsigned Idx) {
@@ -73,7 +74,7 @@ static sycl::unittest::PiImage generateDefaultImage() {
7374

7475
static const std::string KernelName = "TestKernel";
7576
static const std::string CopierKernelName =
76-
"_ZTSN2cl4sycl6detail16AssertInfoCopierE";
77+
"_ZTSN2cl4sycl6detail23__sycl_service_kernel__16AssertInfoCopierE";
7778

7879
PiPropertySet PropSet;
7980

@@ -98,7 +99,7 @@ static sycl::unittest::PiImage generateCopierKernelImage() {
9899
using namespace sycl::unittest;
99100

100101
static const std::string CopierKernelName =
101-
"_ZTSN2cl4sycl6detail16AssertInfoCopierE";
102+
"_ZTSN2cl4sycl6detail23__sycl_service_kernel__16AssertInfoCopierE";
102103

103104
PiPropertySet PropSet;
104105

@@ -396,3 +397,17 @@ TEST(Assert, TestPositive) {
396397
}
397398
#endif // _WIN32
398399
}
400+
401+
TEST(Assert, TestAssertServiceKernelHidden) {
402+
const char *AssertServiceKernelName = sycl::detail::KernelInfo<
403+
sycl::detail::__sycl_service_kernel__::AssertInfoCopier>::getName();
404+
405+
std::vector<sycl::kernel_id> AllKernelIDs = sycl::get_kernel_ids();
406+
407+
auto NoFoundServiceKernelID = std::none_of(
408+
AllKernelIDs.begin(), AllKernelIDs.end(), [=](sycl::kernel_id KernelID) {
409+
return strcmp(KernelID.get_name(), AssertServiceKernelName) == 0;
410+
});
411+
412+
EXPECT_TRUE(NoFoundServiceKernelID);
413+
}

0 commit comments

Comments
 (0)