-
Notifications
You must be signed in to change notification settings - Fork 751
/
Copy pathplatform_impl.hpp
220 lines (188 loc) · 7.86 KB
/
platform_impl.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
//==-------------- platform_impl.hpp - SYCL platform -----------------------==//
//
// Part of the LLVM Project, 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
//
//===----------------------------------------------------------------------===//
#pragma once
#include <detail/platform_info.hpp>
#include <detail/plugin.hpp>
#include <sycl/backend.hpp>
#include <sycl/backend_types.hpp>
#include <sycl/detail/cl.h>
#include <sycl/detail/common.hpp>
#include <sycl/detail/ur.hpp>
#include <sycl/info/info_desc.hpp>
namespace sycl {
inline namespace _V1 {
// Forward declaration
class device_selector;
class device;
enum class aspect;
namespace detail {
class device_impl;
// TODO: implement extension management for host device
// TODO: implement parameters treatment for host device
class platform_impl {
public:
/// Constructs platform_impl from a plug-in interoperability platform
/// handle.
///
/// \param APlatform is a raw plug-in platform handle.
/// \param APlugin is a plug-in handle.
explicit platform_impl(ur_platform_handle_t APlatform,
const std::shared_ptr<plugin> &APlugin)
: MPlatform(APlatform), MPlugin(APlugin) {
// Find out backend of the platform
ur_platform_backend_t UrBackend = UR_PLATFORM_BACKEND_UNKNOWN;
APlugin->call_nocheck(urPlatformGetInfo, APlatform,
UR_PLATFORM_INFO_BACKEND,
sizeof(ur_platform_backend_t), &UrBackend, nullptr);
MBackend = convertUrBackend(UrBackend);
}
~platform_impl() = default;
/// Checks if this platform supports extension.
///
/// \param ExtensionName is a string containing extension name.
/// \return true if platform supports specified extension.
bool has_extension(const std::string &ExtensionName) const;
/// Checks if this platform supports usm.
/// Non opencl backends are assumed to support it.
///
/// \return true if platform supports usm.
bool supports_usm() const;
/// Returns all SYCL devices associated with this platform.
///
/// If this platform is a host platform and device type requested is either
/// info::device_type::all or info::device_type::host, resulting vector
/// contains only a single SYCL host device. If there are no devices that
/// match given device type, resulting vector is empty.
///
/// \param DeviceType is a SYCL device type.
/// \return a vector of SYCL devices.
std::vector<device>
get_devices(info::device_type DeviceType = info::device_type::all) const;
/// Queries this SYCL platform for info.
///
/// The return type depends on information being queried.
template <typename Param> typename Param::return_type get_info() const;
/// Queries this SYCL platform for SYCL backend-specific information.
///
/// The return type depends on information being queried.
template <typename Param>
typename Param::return_type get_backend_info() const;
/// Returns the backend of this platform.
backend getBackend(void) const { return MBackend; }
/// Get backend option.
void getBackendOption(const char *frontend_option,
const char **backend_option) const {
const auto &Plugin = getPlugin();
ur_result_t Err = Plugin->call_nocheck(
urPlatformGetBackendOption, MPlatform, frontend_option, backend_option);
Plugin->checkUrResult(Err);
}
/// \return an instance of OpenCL cl_platform_id.
cl_platform_id get() const {
ur_native_handle_t nativeHandle = 0;
getPlugin()->call(urPlatformGetNativeHandle, MPlatform, &nativeHandle);
return ur::cast<cl_platform_id>(nativeHandle);
}
/// Returns raw underlying plug-in platform handle.
///
/// Unlike get() method, this method does not retain handler. It is caller
/// responsibility to make sure that platform stays alive while raw handle
/// is in use.
///
/// \return a raw plug-in platform handle.
const ur_platform_handle_t &getHandleRef() const { return MPlatform; }
/// Returns all available SYCL platforms in the system.
///
/// By default the resulting vector always contains a single SYCL host
/// platform instance. There are means to override this behavior for testing
/// purposes. See environment variables guide for up-to-date instructions.
///
/// \return a vector of all available SYCL platforms.
static std::vector<platform> get_platforms();
// \return the Plugin associated with this platform.
const PluginPtr &getPlugin() const { return MPlugin; }
/// Sets the platform implementation to use another plugin.
///
/// \param PluginPtr is a pointer to a plugin instance
/// \param Backend is the backend that we want this platform to use
void setPlugin(PluginPtr &PluginPtr, backend Backend) {
MPlugin = PluginPtr;
MBackend = Backend;
}
/// Gets the native handle of the SYCL platform.
///
/// \return a native handle.
ur_native_handle_t getNative() const;
/// Indicates if all of the SYCL devices on this platform have the
/// given feature.
///
/// \param Aspect is one of the values in Table 4.20 of the SYCL 2020
/// Provisional Spec.
///
/// \return true all of the SYCL devices on this platform have the
/// given feature.
bool has(aspect Aspect) const;
/// Queries the device_impl cache to return a shared_ptr for the
/// device_impl corresponding to the UrDevice.
///
/// \param UrDevice is the UrDevice whose impl is requested
///
/// \return a shared_ptr<device_impl> corresponding to the device
std::shared_ptr<device_impl> getDeviceImpl(ur_device_handle_t UrDevice);
/// Queries the device_impl cache to either return a shared_ptr
/// for the device_impl corresponding to the UrDevice or add
/// a new entry to the cache
///
/// \param UrDevice is the UrDevice whose impl is requested
///
/// \param PlatormImpl is the Platform for that Device
///
/// \return a shared_ptr<device_impl> corresponding to the device
std::shared_ptr<device_impl>
getOrMakeDeviceImpl(ur_device_handle_t UrDevice,
const std::shared_ptr<platform_impl> &PlatformImpl);
/// Queries the cache to see if the specified UR platform has been seen
/// before. If so, return the cached platform_impl, otherwise create a new
/// one and cache it.
///
/// \param UrPlatform is the UR Platform handle representing the platform
/// \param Plugin is the UR plugin providing the backend for the platform
/// \return the platform_impl representing the UR platform
static std::shared_ptr<platform_impl>
getOrMakePlatformImpl(ur_platform_handle_t UrPlatform,
const PluginPtr &Plugin);
/// Queries the cache for the specified platform based on an input device.
/// If found, returns the the cached platform_impl, otherwise creates a new
/// one and caches it.
///
/// \param UrDevice is the UR device handle for the device whose platform is
/// desired
/// \param Plugin is the UR plugin providing the backend for the device and
/// platform
/// \return the platform_impl that contains the input device
static std::shared_ptr<platform_impl>
getPlatformFromUrDevice(ur_device_handle_t UrDevice, const PluginPtr &Plugin);
// when getting sub-devices for ONEAPI_DEVICE_SELECTOR we may temporarily
// ensure every device is a root one.
bool MAlwaysRootDevice = false;
private:
std::shared_ptr<device_impl> getDeviceImplHelper(ur_device_handle_t UrDevice);
// Helper to filter reportable devices in the platform
template <typename ListT, typename FilterT>
std::vector<int>
filterDeviceFilter(std::vector<ur_device_handle_t> &UrDevices,
ListT *FilterList) const;
ur_platform_handle_t MPlatform = 0;
backend MBackend;
PluginPtr MPlugin;
std::vector<std::weak_ptr<device_impl>> MDeviceCache;
std::mutex MDeviceMapMutex;
};
} // namespace detail
} // namespace _V1
} // namespace sycl