181 lines
7.1 KiB
C++
181 lines
7.1 KiB
C++
/*
|
|
* Copyright (C) 2017-2019 Intel Corporation
|
|
*
|
|
* SPDX-License-Identifier: MIT
|
|
*
|
|
*/
|
|
|
|
#pragma once
|
|
#include "runtime/api/dispatch.h"
|
|
#include "runtime/helpers/debug_helpers.h"
|
|
#include "runtime/sharings/sharing.h"
|
|
#include "DXGI1_2.h"
|
|
#include <map>
|
|
#include <vector>
|
|
#include "runtime/helpers/array_count.h"
|
|
|
|
namespace OCLRT {
|
|
namespace D3DTypesHelper {
|
|
struct D3D9 {
|
|
typedef IDirect3DDevice9 D3DDevice;
|
|
typedef IDirect3DQuery9 D3DQuery;
|
|
typedef D3DQUERYTYPE D3DQueryDesc;
|
|
typedef IDirect3DResource9 D3DResource;
|
|
typedef struct {
|
|
} D3DBufferDesc;
|
|
typedef void *D3DBufferObj;
|
|
typedef D3DSURFACE_DESC D3DTexture2dDesc;
|
|
typedef struct {
|
|
} D3DTexture3dDesc;
|
|
typedef IDirect3DSurface9 D3DTexture2d;
|
|
typedef struct {
|
|
} D3DTexture3d;
|
|
};
|
|
struct D3D10 {
|
|
typedef ID3D10Device D3DDevice;
|
|
typedef ID3D10Query D3DQuery;
|
|
typedef D3D10_QUERY_DESC D3DQueryDesc;
|
|
typedef ID3D10Resource D3DResource;
|
|
typedef D3D10_BUFFER_DESC D3DBufferDesc;
|
|
typedef ID3D10Buffer D3DBufferObj;
|
|
typedef D3D10_TEXTURE2D_DESC D3DTexture2dDesc;
|
|
typedef D3D10_TEXTURE3D_DESC D3DTexture3dDesc;
|
|
typedef ID3D10Texture2D D3DTexture2d;
|
|
typedef ID3D10Texture3D D3DTexture3d;
|
|
};
|
|
struct D3D11 {
|
|
typedef ID3D11Device D3DDevice;
|
|
typedef ID3D11Query D3DQuery;
|
|
typedef D3D11_QUERY_DESC D3DQueryDesc;
|
|
typedef ID3D11Resource D3DResource;
|
|
typedef D3D11_BUFFER_DESC D3DBufferDesc;
|
|
typedef ID3D11Buffer D3DBufferObj;
|
|
typedef D3D11_TEXTURE2D_DESC D3DTexture2dDesc;
|
|
typedef D3D11_TEXTURE3D_DESC D3DTexture3dDesc;
|
|
typedef ID3D11Texture2D D3DTexture2d;
|
|
typedef ID3D11Texture3D D3DTexture3d;
|
|
};
|
|
} // namespace D3DTypesHelper
|
|
|
|
enum D3DResourceFlags {
|
|
USAGE_RENDERTARGET = 1,
|
|
MISC_SHARED = 2,
|
|
MISC_SHARED_KEYEDMUTEX = 256,
|
|
MISC_SHARED_NTHANDLE = 2048
|
|
};
|
|
|
|
template <typename D3D>
|
|
class D3DSharingFunctions : public SharingFunctions {
|
|
typedef typename D3D::D3DDevice D3DDevice;
|
|
typedef typename D3D::D3DQuery D3DQuery;
|
|
typedef typename D3D::D3DQueryDesc D3DQueryDesc;
|
|
typedef typename D3D::D3DResource D3DResource;
|
|
typedef typename D3D::D3DBufferDesc D3DBufferDesc;
|
|
typedef typename D3D::D3DBufferObj D3DBufferObj;
|
|
typedef typename D3D::D3DTexture2dDesc D3DTexture2dDesc;
|
|
typedef typename D3D::D3DTexture3dDesc D3DTexture3dDesc;
|
|
typedef typename D3D::D3DTexture2d D3DTexture2d;
|
|
typedef typename D3D::D3DTexture3d D3DTexture3d;
|
|
|
|
public:
|
|
typedef void (*GetDxgiDescFcn)(DXGI_ADAPTER_DESC *dxgiDesc, IDXGIAdapter *adapter, D3DDevice *device);
|
|
|
|
D3DSharingFunctions(D3DDevice *d3dDevice) : d3dDevice(d3dDevice) {
|
|
trackedResources.reserve(128);
|
|
getDxgiDescFcn = &this->getDxgiDesc;
|
|
};
|
|
|
|
uint32_t getId() const override {
|
|
return D3DSharingFunctions<D3D>::sharingId;
|
|
}
|
|
|
|
D3DSharingFunctions() = delete;
|
|
|
|
static const uint32_t sharingId;
|
|
|
|
MOCKABLE_VIRTUAL void createQuery(D3DQuery **query);
|
|
MOCKABLE_VIRTUAL void createBuffer(D3DBufferObj **buffer, unsigned int width);
|
|
MOCKABLE_VIRTUAL void createTexture2d(D3DTexture2d **texture, D3DTexture2dDesc *desc, cl_uint subresource);
|
|
MOCKABLE_VIRTUAL void createTexture3d(D3DTexture3d **texture, D3DTexture3dDesc *desc, cl_uint subresource);
|
|
MOCKABLE_VIRTUAL void getBufferDesc(D3DBufferDesc *bufferDesc, D3DBufferObj *buffer);
|
|
MOCKABLE_VIRTUAL void getTexture2dDesc(D3DTexture2dDesc *textureDesc, D3DTexture2d *texture);
|
|
MOCKABLE_VIRTUAL void getTexture3dDesc(D3DTexture3dDesc *textureDesc, D3DTexture3d *texture);
|
|
MOCKABLE_VIRTUAL void getSharedHandle(D3DResource *resource, void **handle);
|
|
MOCKABLE_VIRTUAL void getSharedNTHandle(D3DResource *resource, void **handle);
|
|
MOCKABLE_VIRTUAL void addRef(D3DResource *resource);
|
|
MOCKABLE_VIRTUAL void release(IUnknown *resource);
|
|
MOCKABLE_VIRTUAL void copySubresourceRegion(D3DResource *dst, cl_uint dstSubresource,
|
|
D3DResource *src, cl_uint srcSubresource);
|
|
MOCKABLE_VIRTUAL void flushAndWait(D3DQuery *query);
|
|
MOCKABLE_VIRTUAL void getDeviceContext(D3DQuery *query);
|
|
MOCKABLE_VIRTUAL void releaseDeviceContext(D3DQuery *query);
|
|
MOCKABLE_VIRTUAL void lockRect(D3DTexture2d *d3dResource, D3DLOCKED_RECT *lockedRect, uint32_t flags);
|
|
MOCKABLE_VIRTUAL void unlockRect(D3DTexture2d *d3dResource);
|
|
MOCKABLE_VIRTUAL void getRenderTargetData(D3DTexture2d *renderTarget, D3DTexture2d *dstSurface);
|
|
MOCKABLE_VIRTUAL void updateSurface(D3DTexture2d *src, D3DTexture2d *dst);
|
|
MOCKABLE_VIRTUAL void updateDevice(D3DResource *resource);
|
|
MOCKABLE_VIRTUAL void checkFormatSupport(DXGI_FORMAT format, UINT *pFormat);
|
|
MOCKABLE_VIRTUAL bool memObjectFormatSupport(cl_mem_object_type object, UINT format);
|
|
|
|
GetDxgiDescFcn getDxgiDescFcn = nullptr;
|
|
|
|
bool isTracked(D3DResource *resource, cl_uint subresource) {
|
|
return std::find(trackedResources.begin(), trackedResources.end(), std::make_pair(resource, subresource)) != trackedResources.end();
|
|
}
|
|
|
|
void track(D3DResource *resource, cl_uint subresource) {
|
|
trackedResources.push_back(std::make_pair(resource, subresource));
|
|
}
|
|
|
|
void untrack(D3DResource *resource, cl_uint subresource) {
|
|
auto element = std::find(trackedResources.begin(), trackedResources.end(), std::make_pair(resource, subresource));
|
|
DEBUG_BREAK_IF(element == trackedResources.end());
|
|
trackedResources.erase(element);
|
|
}
|
|
|
|
void setDevice(D3DDevice *d3dDevice) { this->d3dDevice = d3dDevice; }
|
|
D3DDevice *getDevice() { return d3dDevice; }
|
|
|
|
void fillCreateBufferDesc(D3DBufferDesc &desc, unsigned int width);
|
|
void fillCreateTexture2dDesc(D3DTexture2dDesc &desc, D3DTexture2dDesc *srcDesc, cl_uint subresource);
|
|
void fillCreateTexture3dDesc(D3DTexture3dDesc &desc, D3DTexture3dDesc *srcDesc, cl_uint subresource);
|
|
|
|
std::vector<DXGI_FORMAT> &retrieveTextureFormats(cl_mem_object_type imageType);
|
|
|
|
protected:
|
|
D3DDevice *d3dDevice = nullptr;
|
|
ID3D11DeviceContext *d3d11DeviceContext = nullptr;
|
|
|
|
std::vector<DXGI_FORMAT> DXGINoFormats;
|
|
std::vector<std::pair<D3DResource *, cl_uint>> trackedResources;
|
|
std::map<cl_mem_object_type, std::vector<DXGI_FORMAT>> textureFormatCache;
|
|
static void getDxgiDesc(DXGI_ADAPTER_DESC *dxgiDesc, IDXGIAdapter *adapter, D3DDevice *device);
|
|
};
|
|
|
|
template <typename D3DSharing>
|
|
static inline cl_int getSupportedDXTextureFormats(cl_context context, cl_mem_object_type imageType,
|
|
cl_uint numEntries, DXGI_FORMAT *formats, cl_uint *numImageFormats) {
|
|
Context *pContext = castToObject<Context>(context);
|
|
if (!pContext) {
|
|
return CL_INVALID_CONTEXT;
|
|
}
|
|
|
|
auto pSharing = pContext->getSharing<D3DSharingFunctions<D3DSharing>>();
|
|
if (!pSharing) {
|
|
return CL_INVALID_CONTEXT;
|
|
}
|
|
|
|
auto supported_formats = pSharing->retrieveTextureFormats(imageType);
|
|
|
|
if (formats != nullptr) {
|
|
memcpy_s(formats, sizeof(DXGI_FORMAT) * numEntries, supported_formats.data(), std::min(static_cast<size_t>(numEntries), supported_formats.size()));
|
|
}
|
|
|
|
if (numImageFormats) {
|
|
*numImageFormats = static_cast<cl_uint>(supported_formats.size());
|
|
}
|
|
return CL_SUCCESS;
|
|
}
|
|
|
|
} // namespace OCLRT
|