Skip to content

Commit

Permalink
Fixes from review
Browse files Browse the repository at this point in the history
  • Loading branch information
Beanavil committed Dec 15, 2023
1 parent 43b8e34 commit aae074f
Show file tree
Hide file tree
Showing 6 changed files with 160 additions and 27 deletions.
1 change: 1 addition & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ project(OpenCL-SDK
include(CMakeDependentOption)
option(OPENCL_SDK_BUILD_SAMPLES "Build sample code" ON)
cmake_dependent_option(OPENCL_SDK_BUILD_OPENGL_SAMPLES "Build OpenCL-OpenGL interop sample code" ON OPENCL_SDK_BUILD_SAMPLES OFF)
cmake_dependent_option(OPENCL_SDK_BUILD_VULKAN_SAMPLES "Build OpenCL-Vulkan interop sample code" ON OPENCL_SDK_BUILD_SAMPLES OFF)
cmake_dependent_option(OPENCL_SDK_TEST_SAMPLES "Add CTest to samples (where applicable)" ON OPENCL_SDK_BUILD_SAMPLES OFF)

option(OPENCL_SDK_BUILD_CLINFO "Build clinfo utility" ON)
Expand Down
7 changes: 6 additions & 1 deletion cmake/Dependencies.cmake
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ set(BUILD_SHARED_LIBS OFF CACHE BOOL "Global flag to cause add_library() to crea

# Fetch dependencies
if(OPENCL_SDK_BUILD_SAMPLES)
foreach(DEP IN ITEMS cargs TCLAP Stb Vulkan)
foreach(DEP IN ITEMS cargs TCLAP Stb)
list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_LIST_DIR}/Dependencies/${DEP}")
include(${DEP})
endforeach()
Expand All @@ -45,6 +45,11 @@ if(OPENCL_SDK_BUILD_SAMPLES)
include(${DEP})
endforeach()
endif(OPENCL_SDK_BUILD_OPENGL_SAMPLES)

if(OPENCL_SDK_BUILD_VULKAN_SAMPLES)
list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_LIST_DIR}/Dependencies/Vulkan")
include(Vulkan)
endif(OPENCL_SDK_BUILD_VULKAN_SAMPLES)
endif(OPENCL_SDK_BUILD_SAMPLES)

if(OPENCL_SDK_BUILD_CLINFO)
Expand Down
4 changes: 3 additions & 1 deletion samples/extensions/khr/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -12,9 +12,11 @@
# See the License for the specific language governing permissions and
# limitations under the License.

add_subdirectory(externalmemory)
add_subdirectory(histogram)
if(OPENCL_SDK_BUILD_OPENGL_SAMPLES)
add_subdirectory(conway)
add_subdirectory(nbody)
endif()
if(OPENCL_SDK_BUILD_VULKAN_SAMPLES)
add_subdirectory(externalmemory)
endif()
72 changes: 66 additions & 6 deletions samples/extensions/khr/externalmemory/main.c
Original file line number Diff line number Diff line change
Expand Up @@ -430,7 +430,6 @@ int main(int argc, char* argv[])
buffer_info.size = sizeof(cl_float) * length;
buffer_info.usage =
VK_BUFFER_USAGE_TRANSFER_SRC_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT;
;
buffer_info.sharingMode = VK_SHARING_MODE_EXCLUSIVE;

VkBuffer vk_buf_x, vk_buf_y;
Expand Down Expand Up @@ -485,6 +484,35 @@ int main(int argc, char* argv[])
memcpy(vk_arr_x, arr_x, sizeof(cl_float) * length);
memcpy(vk_arr_y, arr_y, sizeof(cl_float) * length);

#ifdef _WIN32
// Get Vulkan external memory file descriptors for accessing external memory
// with OpenCL.
VkMemoryGetWin32HandleInfoKHR handle_info_x = { 0 };
handle_info_x.sType = VK_STRUCTURE_TYPE_MEMORY_GET_WIN32_HANDLE_INFO_KHR;
handle_info_x.pNext = NULL;
handle_info_x.memory = vk_buf_x_memory;
handle_info_x.handleType = vk_external_memory_handle_type;
HANDLE handle_x;

VkMemoryGetWin32HandleInfoKHR handle_info_y = { 0 };
handle_info_y.sType = VK_STRUCTURE_TYPE_MEMORY_GET_WIN32_HANDLE_INFO_KHR;
handle_info_y.pNext = NULL;
handle_info_y.memory = vk_buf_y_memory;
handle_info_y.handleType = vk_external_memory_handle_type;
HANDLE handle_y;

// We need to get the pointer to the
// vkGetMemoryFdKHR/vkGetMemoryWin32HandleKHR function because it's from
// extension VK_KHR_external_memory_fd. This Vulkan function exports a POSIX
// file descriptor/Windows handle referencing the payload of a Vulkan device
// memory object.
PFN_vkGetMemoryWin32HandleKHR vkGetMemoryWin32Handle;
*(PFN_vkGetMemoryWin32HandleKHR*)&vkGetMemoryWin32Handle =
(PFN_vkGetMemoryWin32HandleKHR)vkGetDeviceProcAddr(
vk_device, "vkGetMemoryWin32HandleKHR");
VK_CHECK(vkGetMemoryWin32Handle(vk_device, &handle_info_x, &handle_x));
VK_CHECK(vkGetMemoryWin32Handle(vk_device, &handle_info_y, &handle_y));
#else
// Get Vulkan external memory file descriptors for accessing external memory
// with OpenCL.
VkMemoryGetFdInfoKHR fd_info_x = { 0 };
Expand All @@ -501,27 +529,40 @@ int main(int argc, char* argv[])
fd_info_y.handleType = vk_external_memory_handle_type;
int fd_y;

// We need to get the pointer to the vkGetMemoryFdKHR function because it's
// from extension VK_KHR_external_memory_fd.
PFN_vkGetMemoryFdKHR vkGetMemoryFdKHR =
// We need to get the pointer to the
// vkGetMemoryFdKHR/vkGetMemoryWin32HandleKHR function because it's from
// extension VK_KHR_external_memory_fd. This Vulkan function exports a POSIX
// file descriptor/Windows handle referencing the payload of a Vulkan device
// memory object.
PFN_vkGetMemoryFdKHR vkGetMemoryFd;
*(PFN_vkGetMemoryFdKHR*)&vkGetMemoryFd =
(PFN_vkGetMemoryFdKHR)vkGetDeviceProcAddr(vk_device,
"vkGetMemoryFdKHR");
VK_CHECK(vkGetMemoryFd(vk_device, &fd_info_x, &fd_x));
VK_CHECK(vkGetMemoryFd(vk_device, &fd_info_y, &fd_y));
#endif

VK_CHECK(vkGetMemoryFdKHR(vk_device, &fd_info_x, &fd_x));
VK_CHECK(vkGetMemoryFdKHR(vk_device, &fd_info_y, &fd_y));

// Create OpenCL buffers from Vulkan external memory file descriptors.
cl_mem_properties ext_mem_props_x[] = {
(cl_mem_properties)CL_EXTERNAL_MEMORY_HANDLE_OPAQUE_FD_KHR,
#ifdef _WIN32
(cl_mem_properties)handle_x,
#else
(cl_mem_properties)fd_x,
#endif
(cl_mem_properties)CL_DEVICE_HANDLE_LIST_KHR,
(cl_mem_properties)(uintptr_t)cl_device,
CL_DEVICE_HANDLE_LIST_END_KHR,
0
};
cl_mem_properties ext_mem_props_y[] = {
(cl_mem_properties)CL_EXTERNAL_MEMORY_HANDLE_OPAQUE_FD_KHR,
#ifdef _WIN32
(cl_mem_properties)handle_y,
#else
(cl_mem_properties)fd_y,
#endif
(cl_mem_properties)CL_DEVICE_HANDLE_LIST_KHR,
(cl_mem_properties)(uintptr_t)cl_device,
CL_DEVICE_HANDLE_LIST_END_KHR,
Expand Down Expand Up @@ -552,6 +593,16 @@ int main(int argc, char* argv[])
OCLERROR_RET(clSetKernelArg(saxpy, 2, sizeof(cl_mem), &cl_buf_y), error,
clbufy);

// Acquire OpenCL memory objects created from Vulkan external memory
// handles.
cl_mem cl_mem_objects[] = { cl_buf_x, cl_buf_y };
clEnqueueAcquireExternalMemObjectsKHR_fn
clEnqueueAcquireExternalMemObjects =
(clEnqueueAcquireExternalMemObjectsKHR_fn)
clGetExtensionFunctionAddressForPlatform(
cl_platform, "clEnqueueAcquireExternalMemObjectsKHR");
clEnqueueAcquireExternalMemObjects(queue, 2, cl_mem_objects, 0, NULL, NULL);

// Launch kernel.
if (diag_opts.verbose)
{
Expand All @@ -570,6 +621,15 @@ int main(int argc, char* argv[])
cl_ulong dev_time;
TIMER_DIFFERENCE(dev_time, dev_start, dev_end)

// Release OpenCL memory objects created from Vulkan external memory
// handles.
clEnqueueReleaseExternalMemObjectsKHR_fn
clEnqueueReleaseExternalMemObjects =
(clEnqueueReleaseExternalMemObjectsKHR_fn)
clGetExtensionFunctionAddressForPlatform(
cl_platform, "clEnqueueReleaseExternalMemObjectsKHR");
clEnqueueReleaseExternalMemObjects(queue, 2, cl_mem_objects, 0, NULL, NULL);

// Concurrently calculate reference saxpy.
if (diag_opts.verbose)
{
Expand Down
79 changes: 72 additions & 7 deletions samples/extensions/khr/externalmemory/main.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -349,7 +349,6 @@ int main(int argc, char* argv[])
buffer_info.size = sizeof(cl_float) * length;
buffer_info.usage =
VK_BUFFER_USAGE_TRANSFER_SRC_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT;
;
buffer_info.sharingMode = VK_SHARING_MODE_EXCLUSIVE;

VkBuffer vk_buf_x, vk_buf_y;
Expand Down Expand Up @@ -404,6 +403,37 @@ int main(int argc, char* argv[])
memcpy(vk_arr_x, arr_x.data(), sizeof(cl_float) * length);
memcpy(vk_arr_y, arr_y.data(), sizeof(cl_float) * length);

#ifdef _WIN32
// Get Vulkan external memory file descriptors for accessing external
// memory with OpenCL.
VkMemoryGetWin32HandleInfoKHR handle_info_x{};
handle_info_x.sType =
VK_STRUCTURE_TYPE_MEMORY_GET_WIN32_HANDLE_INFO_KHR;
handle_info_x.pNext = nullptr;
handle_info_x.memory = vk_buf_x_memory;
handle_info_x.handleType = vk_external_memory_handle_type;
HANDLE handle_x;

VkMemoryGetWin32HandleInfoKHR handle_info_y{};
handle_info_y.sType =
VK_STRUCTURE_TYPE_MEMORY_GET_WIN32_HANDLE_INFO_KHR;
handle_info_y.pNext = nullptr;
handle_info_y.memory = vk_buf_y_memory;
handle_info_y.handleType = vk_external_memory_handle_type;
HANDLE handle_y;

// We need to get the pointer to the
// vkGetMemoryFdKHR/vkGetMemoryWin32HandleKHR function because it's from
// the extension VK_KHR_external_memory_fd. This Vulkan function exports
// a POSIX file descriptor/Windows handle referencing the payload of a
// Vulkan device memory object.
PFN_vkGetMemoryWin32HandleKHR vkGetMemoryWin32Handle;
*(PFN_vkGetMemoryWin32HandleKHR*)&vkGetMemoryWin32Handle =
(PFN_vkGetMemoryWin32HandleKHR)vkGetDeviceProcAddr(
vk_device, "vkGetMemoryWin32HandleKHR");
VK_CHECK(vkGetMemoryWin32Handle(vk_device, &handle_info_x, &handle_x));
VK_CHECK(vkGetMemoryWin32Handle(vk_device, &handle_info_y, &handle_y));
#else
// Get Vulkan external memory file descriptors for accessing external
// memory with OpenCL.
VkMemoryGetFdInfoKHR fd_info_x{};
Expand All @@ -420,27 +450,39 @@ int main(int argc, char* argv[])
fd_info_y.handleType = vk_external_memory_handle_type;
int fd_y;

// We need to get the pointer to the vkGetMemoryFdKHR function because
// it's from extension VK_KHR_external_memory_fd.
PFN_vkGetMemoryFdKHR vkGetMemoryFdKHR =
// We need to get the pointer to the
// vkGetMemoryFdKHR/vkGetMemoryWin32HandleKHR function because it's from
// extension VK_KHR_external_memory_fd. This Vulkan function exports a
// POSIX file descriptor/Windows handle referencing the payload of a
// Vulkan device memory object.
PFN_vkGetMemoryFdKHR vkGetMemoryFd;
*(PFN_vkGetMemoryFdKHR*)&vkGetMemoryFd =
(PFN_vkGetMemoryFdKHR)vkGetDeviceProcAddr(vk_device,
"vkGetMemoryFdKHR");

VK_CHECK(vkGetMemoryFdKHR(vk_device, &fd_info_x, &fd_x));
VK_CHECK(vkGetMemoryFdKHR(vk_device, &fd_info_y, &fd_y));
VK_CHECK(vkGetMemoryFd(vk_device, &fd_info_x, &fd_x));
VK_CHECK(vkGetMemoryFd(vk_device, &fd_info_y, &fd_y));
#endif

// Create OpenCL buffers from Vulkan external memory file descriptors.
std::vector<cl_mem_properties> ext_mem_props_x = {
(cl_mem_properties)CL_EXTERNAL_MEMORY_HANDLE_OPAQUE_FD_KHR,
#ifdef _WIN32
(cl_mem_properties)handle_x,
#else
(cl_mem_properties)fd_x,
#endif
(cl_mem_properties)CL_DEVICE_HANDLE_LIST_KHR,
(cl_mem_properties)cl_device(),
CL_DEVICE_HANDLE_LIST_END_KHR,
0
};
std::vector<cl_mem_properties> ext_mem_props_y = {
(cl_mem_properties)CL_EXTERNAL_MEMORY_HANDLE_OPAQUE_FD_KHR,
#ifdef _WIN32
(cl_mem_properties)handle_y,
#else
(cl_mem_properties)fd_y,
#endif
(cl_mem_properties)CL_DEVICE_HANDLE_LIST_KHR,
(cl_mem_properties)cl_device(),
CL_DEVICE_HANDLE_LIST_END_KHR,
Expand All @@ -457,6 +499,18 @@ int main(int argc, char* argv[])
0 };
cl::CommandQueue queue{ cl_context, cl_device, *queue_props };

// Acquire OpenCL memory objects created from Vulkan external memory
// handles.
std::vector<cl_mem> cl_mem_objects = { cl_buf_x(), cl_buf_y() };
clEnqueueAcquireExternalMemObjectsKHR_fn
clEnqueueAcquireExternalMemObjects =
(clEnqueueAcquireExternalMemObjectsKHR_fn)
clGetExtensionFunctionAddressForPlatform(
cl_platform(), "clEnqueueAcquireExternalMemObjectsKHR");
clEnqueueAcquireExternalMemObjects(
queue(), static_cast<cl_uint>(cl_mem_objects.size()),
cl_mem_objects.data(), 0, nullptr, nullptr);

// Launch kernel.
if (diag_opts.verbose)
{
Expand All @@ -472,6 +526,17 @@ int main(int argc, char* argv[])
cl::WaitForEvents(kernel_run);
auto dev_end = std::chrono::high_resolution_clock::now();

// Release OpenCL memory objects created from Vulkan external memory
// handles.
clEnqueueReleaseExternalMemObjectsKHR_fn
clEnqueueReleaseExternalMemObjects =
(clEnqueueReleaseExternalMemObjectsKHR_fn)
clGetExtensionFunctionAddressForPlatform(
cl_platform(), "clEnqueueReleaseExternalMemObjectsKHR");
clEnqueueReleaseExternalMemObjects(
queue(), static_cast<cl_uint>(cl_mem_objects.size()),
cl_mem_objects.data(), 0, nullptr, nullptr);

// Concurrently calculate reference saxpy.
if (diag_opts.verbose)
{
Expand Down
24 changes: 12 additions & 12 deletions samples/extensions/khr/externalmemory/vulkan_utils.h
Original file line number Diff line number Diff line change
Expand Up @@ -224,24 +224,24 @@ find_suitable_device(VkInstance instance,
// Query OpenCL devices available.
cl_int error = CL_SUCCESS;
bool candidate_found = false;
cl_uint cl_platform_count = 0;
cl_uint platform_count = 0;
struct device_candidate found_candidate = {0};
OCLERROR_RET(clGetPlatformIDs(0, NULL, &cl_platform_count), error, ret);
OCLERROR_RET(clGetPlatformIDs(0, NULL, &platform_count), error, ret);

cl_platform_id* platforms =
(cl_platform_id*)malloc(cl_platform_count * sizeof(cl_platform_id));
OCLERROR_RET(clGetPlatformIDs(cl_platform_count, platforms, NULL), error,
(cl_platform_id*)malloc(platform_count * sizeof(cl_platform_id));
OCLERROR_RET(clGetPlatformIDs(platform_count, platforms, NULL), error,
platforms);

size_t cl_device_count = 0;
const char* uuid_khronos_extension[] = {
CL_KHR_DEVICE_UUID_EXTENSION_NAME
};
for (cl_uint cl_platform_id = 0; cl_platform_id < cl_platform_count;
++cl_platform_id)
for (cl_uint platform_id = 0; platform_id < platform_count;
++platform_id)
{
cl_uint cl_platform_devices_count = 0;
OCLERROR_RET(clGetDeviceIDs(platforms[cl_platform_id],
OCLERROR_RET(clGetDeviceIDs(platforms[platform_id],
CL_DEVICE_TYPE_ALL, 0, NULL,
&cl_platform_devices_count),
error, platforms);
Expand All @@ -250,7 +250,7 @@ find_suitable_device(VkInstance instance,
{
cl_device_id device;
OCLERROR_PAR(device = cl_util_get_device(
cl_platform_id, device_id, CL_DEVICE_TYPE_ALL, &error), error, platforms);
platform_id, device_id, CL_DEVICE_TYPE_ALL, &error), error, platforms);
cl_device_count +=
check_khronos_extensions(device, uuid_khronos_extension, 1);
}
Expand All @@ -264,11 +264,11 @@ find_suitable_device(VkInstance instance,
(struct cl_device_candidate*)malloc(
cl_device_count * sizeof(struct cl_device_candidate));
cl_device_count = 0;
for (cl_uint cl_platform_id = 0; cl_platform_id < cl_platform_count;
++cl_platform_id)
for (cl_uint platform_id = 0; platform_id < platform_count;
++platform_id)
{
cl_uint cl_platform_devices_count = 0;
OCLERROR_RET(clGetDeviceIDs(platforms[cl_platform_id],
OCLERROR_RET(clGetDeviceIDs(platforms[platform_id],
CL_DEVICE_TYPE_ALL, 0, NULL,
&cl_platform_devices_count),
error, candidates);
Expand All @@ -278,7 +278,7 @@ find_suitable_device(VkInstance instance,
++cl_candidate_id, ++cl_device_count)
{
cl_device_id device = cl_util_get_device(
cl_platform_id, cl_candidate_id, CL_DEVICE_TYPE_ALL, &error);
platform_id, cl_candidate_id, CL_DEVICE_TYPE_ALL, &error);
if (check_khronos_extensions(device, uuid_khronos_extension, 1))
{
cl_uchar vk_candidate_uuid[CL_UUID_SIZE_KHR];
Expand Down

0 comments on commit aae074f

Please sign in to comment.