mirror of
https://github.com/PixarAnimationStudios/OpenSubdiv
synced 2025-01-13 01:40:15 +00:00
5c54962d3b
build osd and examples with DX/CL interop only of cmake finds cl_d3d11.h or cl_d3d11_ext.h.
351 lines
11 KiB
C++
351 lines
11 KiB
C++
//
|
|
// Copyright 2015 Pixar
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "Apache License")
|
|
// with the following modification; you may not use this file except in
|
|
// compliance with the Apache License and the following modification to it:
|
|
// Section 6. Trademarks. is deleted and replaced with:
|
|
//
|
|
// 6. Trademarks. This License does not grant permission to use the trade
|
|
// names, trademarks, service marks, or product names of the Licensor
|
|
// and its affiliates, except as required to comply with Section 4(c) of
|
|
// the License and to reproduce the content of the NOTICE file.
|
|
//
|
|
// You may obtain a copy of the Apache License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the Apache License with the above modification is
|
|
// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
// KIND, either express or implied. See the Apache License for the specific
|
|
// language governing permissions and limitations under the Apache License.
|
|
//
|
|
|
|
#define CL_USE_DEPRECATED_OPENCL_2_0_APIS
|
|
|
|
#include "clDeviceContext.h"
|
|
|
|
#if defined(_WIN32)
|
|
#include <windows.h>
|
|
#elif defined(__APPLE__)
|
|
#include <OpenGL/OpenGL.h>
|
|
#else
|
|
#include <GL/glx.h>
|
|
#endif
|
|
|
|
#include <cstdio>
|
|
#include <cstring>
|
|
#include <string>
|
|
|
|
#if defined(OPENSUBDIV_HAS_DX11SDK)
|
|
#include <D3D11.h>
|
|
|
|
#if defined(OPENSUBDIV_HAS_CL_D3D11_H)
|
|
#include <CL/cl_d3d11.h>
|
|
#elif defined(OPENSUBDIV_HAS_CL_D3D11_EXT_H)
|
|
#include <CL/cl_d3d11_ext.h>
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define message(...) // fprintf(stderr, __VA_ARGS__)
|
|
#define error(...) fprintf(stderr, __VA_ARGS__)
|
|
|
|
// returns the first found platform.
|
|
//
|
|
static cl_platform_id
|
|
findPlatform() {
|
|
cl_uint numPlatforms;
|
|
cl_int ciErrNum = clGetPlatformIDs(0, NULL, &numPlatforms);
|
|
|
|
if (ciErrNum != CL_SUCCESS) {
|
|
error("Error %d in clGetPlatformIDs call.\n", ciErrNum);
|
|
return NULL;
|
|
}
|
|
if (numPlatforms == 0) {
|
|
error("No OpenCL platform found.\n");
|
|
return NULL;
|
|
}
|
|
|
|
cl_platform_id *clPlatformIDs = new cl_platform_id[numPlatforms];
|
|
ciErrNum = clGetPlatformIDs(numPlatforms, clPlatformIDs, NULL);
|
|
char chBuffer[1024];
|
|
for (cl_uint i = 0; i < numPlatforms; ++i) {
|
|
ciErrNum = clGetPlatformInfo(clPlatformIDs[i], CL_PLATFORM_NAME,
|
|
1024, chBuffer,NULL);
|
|
if (ciErrNum == CL_SUCCESS) {
|
|
cl_platform_id platformId = clPlatformIDs[i];
|
|
delete[] clPlatformIDs;
|
|
return platformId;
|
|
}
|
|
}
|
|
delete[] clPlatformIDs;
|
|
return NULL;
|
|
}
|
|
|
|
|
|
// returns the device in clDevices which supports the extension.
|
|
//
|
|
static int
|
|
findExtensionSupportedDevice(cl_device_id *clDevices,
|
|
int numDevices,
|
|
const char *extensionName) {
|
|
// find a device that supports sharing with GL/D3D11
|
|
// (SLI / X-fire configurations)
|
|
cl_int ciErrNum;
|
|
|
|
for (int i = 0; i < numDevices; ++i) {
|
|
// get extensions string size
|
|
size_t extensionSize;
|
|
ciErrNum = clGetDeviceInfo(clDevices[i],
|
|
CL_DEVICE_EXTENSIONS, 0, NULL,
|
|
&extensionSize );
|
|
|
|
if (ciErrNum != CL_SUCCESS) {
|
|
error("Error %d in clGetDeviceInfo\n", ciErrNum);
|
|
return -1;
|
|
}
|
|
|
|
if (extensionSize>0) {
|
|
// get extensions string
|
|
char *extensions = new char[extensionSize];
|
|
ciErrNum = clGetDeviceInfo(clDevices[i], CL_DEVICE_EXTENSIONS,
|
|
extensionSize, extensions,
|
|
&extensionSize);
|
|
if (ciErrNum != CL_SUCCESS) {
|
|
error("Error %d in clGetDeviceInfo\n", ciErrNum);
|
|
delete[] extensions;
|
|
continue;
|
|
}
|
|
std::string extString(extensions);
|
|
delete[] extensions;
|
|
|
|
// parse string. This is bit deficient since the extentions
|
|
// is space separated.
|
|
//
|
|
// The actual string would be "cl_khr_d3d11_sharing"
|
|
// or "cl_nv_d3d11_sharing"
|
|
if (extString.find(extensionName) != std::string::npos) {
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
// --------------------------------------------------------------------------
|
|
|
|
CLDeviceContext::CLDeviceContext() :
|
|
_clContext(NULL), _clCommandQueue(NULL) {
|
|
}
|
|
|
|
CLDeviceContext::~CLDeviceContext() {
|
|
|
|
if (_clCommandQueue)
|
|
clReleaseCommandQueue(_clCommandQueue);
|
|
if (_clContext)
|
|
clReleaseContext(_clContext);
|
|
}
|
|
|
|
/*static*/
|
|
bool
|
|
CLDeviceContext::HAS_CL_VERSION_1_1 () {
|
|
|
|
#ifdef OPENSUBDIV_HAS_CLEW
|
|
static bool clewInitialized = false;
|
|
static bool clewLoadSuccess;
|
|
if (not clewInitialized) {
|
|
clewInitialized = true;
|
|
clewLoadSuccess = clewInit() == CLEW_SUCCESS;
|
|
if (not clewLoadSuccess) {
|
|
error("Loading OpenCL failed.\n");
|
|
}
|
|
}
|
|
return clewLoadSuccess;
|
|
#endif
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
CLDeviceContext::Initialize() {
|
|
|
|
#ifdef OPENSUBDIV_HAS_CLEW
|
|
if (!clGetPlatformIDs) {
|
|
error("Error clGetPlatformIDs function not bound.\n");
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
cl_int ciErrNum;
|
|
cl_platform_id cpPlatform = findPlatform();
|
|
|
|
#if defined(_WIN32)
|
|
cl_context_properties props[] = {
|
|
CL_GL_CONTEXT_KHR, (cl_context_properties)wglGetCurrentContext(),
|
|
CL_WGL_HDC_KHR, (cl_context_properties)wglGetCurrentDC(),
|
|
CL_CONTEXT_PLATFORM, (cl_context_properties)cpPlatform,
|
|
0
|
|
};
|
|
#elif defined(__APPLE__)
|
|
CGLContextObj kCGLContext = CGLGetCurrentContext();
|
|
CGLShareGroupObj kCGLShareGroup = CGLGetShareGroup(kCGLContext);
|
|
cl_context_properties props[] = {
|
|
CL_CONTEXT_PROPERTY_USE_CGL_SHAREGROUP_APPLE, (cl_context_properties)kCGLShareGroup,
|
|
0
|
|
};
|
|
#else
|
|
cl_context_properties props[] = {
|
|
CL_GL_CONTEXT_KHR, (cl_context_properties)glXGetCurrentContext(),
|
|
CL_GLX_DISPLAY_KHR, (cl_context_properties)glXGetCurrentDisplay(),
|
|
CL_CONTEXT_PLATFORM, (cl_context_properties)cpPlatform,
|
|
0
|
|
};
|
|
#endif
|
|
|
|
#if defined(__APPLE__)
|
|
_clContext = clCreateContext(props, 0, NULL, clLogMessagesToStdoutAPPLE,
|
|
NULL, &ciErrNum);
|
|
if (ciErrNum != CL_SUCCESS) {
|
|
error("Error %d in clCreateContext\n", ciErrNum);
|
|
return false;
|
|
}
|
|
|
|
size_t devicesSize = 0;
|
|
clGetGLContextInfoAPPLE(_clContext, kCGLContext,
|
|
CL_CGL_DEVICES_FOR_SUPPORTED_VIRTUAL_SCREENS_APPLE,
|
|
0, NULL, &devicesSize);
|
|
int numDevices = int(devicesSize / sizeof(cl_device_id));
|
|
if (numDevices == 0) {
|
|
error("No sharable devices.\n");
|
|
return false;
|
|
}
|
|
cl_device_id *clDevices = new cl_device_id[numDevices];
|
|
clGetGLContextInfoAPPLE(_clContext, kCGLContext,
|
|
CL_CGL_DEVICES_FOR_SUPPORTED_VIRTUAL_SCREENS_APPLE,
|
|
numDevices * sizeof(cl_device_id), clDevices, NULL);
|
|
int clDeviceUsed = 0;
|
|
|
|
#else // not __APPLE__
|
|
|
|
// get the number of GPU devices available to the platform
|
|
cl_uint numDevices = 0;
|
|
clGetDeviceIDs(cpPlatform, CL_DEVICE_TYPE_GPU, 0, NULL, &numDevices);
|
|
if (numDevices == 0) {
|
|
error("No CL GPU device found.\n");
|
|
return false;
|
|
}
|
|
|
|
// create the device list
|
|
cl_device_id *clDevices = new cl_device_id[numDevices];
|
|
clGetDeviceIDs(cpPlatform, CL_DEVICE_TYPE_GPU, numDevices, clDevices, NULL);
|
|
|
|
const char *extension = "cl_khr_gl_sharing";
|
|
int clDeviceUsed = findExtensionSupportedDevice(clDevices, numDevices,
|
|
extension);
|
|
|
|
if (clDeviceUsed < 0) {
|
|
error("No device found that supports CL/GL context sharing\n");
|
|
delete[] clDevices;
|
|
return false;
|
|
}
|
|
|
|
_clContext = clCreateContext(props, 1, &clDevices[clDeviceUsed],
|
|
NULL, NULL, &ciErrNum);
|
|
|
|
#endif // not __APPLE__
|
|
|
|
if (ciErrNum != CL_SUCCESS) {
|
|
error("Error %d in clCreateContext\n", ciErrNum);
|
|
delete[] clDevices;
|
|
return false;
|
|
}
|
|
|
|
_clCommandQueue = clCreateCommandQueue(_clContext, clDevices[clDeviceUsed],
|
|
0, &ciErrNum);
|
|
delete[] clDevices;
|
|
if (ciErrNum != CL_SUCCESS) {
|
|
error("Error %d in clCreateCommandQueue\n", ciErrNum);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
bool
|
|
CLD3D11DeviceContext::Initialize(ID3D11DeviceContext *d3dDeviceContext) {
|
|
|
|
#if defined(OPENSUBDIV_HAS_DX11SDK) && \
|
|
(defined(OPENSUBDIV_HAS_CL_D3D11_H) || defined(OPENSUBDIV_HAS_CL_D3D11_EXT_H))
|
|
|
|
_d3dDeviceContext = d3dDeviceContext;
|
|
|
|
cl_int ciErrNum;
|
|
cl_platform_id cpPlatform = findPlatform();
|
|
|
|
ID3D11Device *device;
|
|
d3dDeviceContext->GetDevice(&device);
|
|
|
|
#if defined(OPENSUBDIV_HAS_CL_D3D11_H)
|
|
cl_context_properties props[] = {
|
|
CL_CONTEXT_D3D11_DEVICE_KHR, (cl_context_properties)device,
|
|
CL_CONTEXT_PLATFORM, (cl_context_properties)cpPlatform,
|
|
0
|
|
};
|
|
#elif defined(OPENSUBDIV_HAS_CL_D3D11_EXT_H)
|
|
cl_context_properties props[] = {
|
|
CL_CONTEXT_D3D11_DEVICE_NV, (cl_context_properties)device,
|
|
CL_CONTEXT_PLATFORM, (cl_context_properties)cpPlatform,
|
|
0
|
|
};
|
|
#endif
|
|
|
|
// get the number of GPU devices available to the platform
|
|
cl_uint numDevices = 0;
|
|
clGetDeviceIDs(cpPlatform, CL_DEVICE_TYPE_GPU, 0, NULL, &numDevices);
|
|
if (numDevices == 0) {
|
|
error("No CL GPU device found.\n");
|
|
return false;
|
|
}
|
|
|
|
// create the device list
|
|
cl_device_id *clDevices = new cl_device_id[numDevices];
|
|
clGetDeviceIDs(cpPlatform, CL_DEVICE_TYPE_GPU, numDevices, clDevices, NULL);
|
|
|
|
// we're cheating a little bit.
|
|
// try to find both cl_khr_d3d11_sharing and cl_nv_d3d11_sharing.
|
|
const char *extension = "_d3d11_sharing";
|
|
int clDeviceUsed = findExtensionSupportedDevice(clDevices, numDevices,
|
|
extension);
|
|
|
|
if (clDeviceUsed < 0) {
|
|
error("No device found that supports CL/D3D11 context sharing\n");
|
|
delete[] clDevices;
|
|
return false;
|
|
}
|
|
|
|
_clContext = clCreateContext(props, 1, &clDevices[clDeviceUsed],
|
|
NULL, NULL, &ciErrNum);
|
|
if (ciErrNum != CL_SUCCESS) {
|
|
error("Error %d in clCreateContext\n", ciErrNum);
|
|
delete[] clDevices;
|
|
return false;
|
|
}
|
|
|
|
_clCommandQueue = clCreateCommandQueue(_clContext, clDevices[clDeviceUsed],
|
|
0, &ciErrNum);
|
|
delete[] clDevices;
|
|
if (ciErrNum != CL_SUCCESS) {
|
|
error("Error %d in clCreateCommandQueue\n", ciErrNum);
|
|
return false;
|
|
}
|
|
return true;
|
|
#else
|
|
(void)d3dDeviceContext; // unused
|
|
return false;
|
|
#endif
|
|
}
|