Я пытаюсь использовать эффект DirectX ChromaKey, но моя функция застревает на каком-то шаге.
Что я делаю:
- Создание ID2D1Factory1
- Создание ID3D11Device и ID3D11DeviceContext
- Получить DXGIResource из полученной текстуры
- Получить общий дескриптор из DXGIResource
- Открыть DXGIResource как новый ID3D11Texture2D с использованием ID3D11Device
- Получить D3D11_TEXTURE2D_DES
- Копирование ресурса из полученного ID3D11Texture2D к созданному ID3D11Texture2D
- Определяем D2D1_SIZE_U, D2D1_PIXEL_FORMAT, D2D1_BITMAP_PROPERTIES
- Получение рабочего стола DPI из ID2D1Factory1
- Создание D2D1_RENDER_TARGET_PROPERTIES с использованием полученного DPI
- Создание ID2D1RenderTarget с использованием ID2D1Factory1, IDXGISurface и D2D1_30 * * * * 10END_RER_P красный ID2D1Bitmap с использованием ID2D1RenderTarget, IDXGISurface и D2D1_BITMAP_PROPERTIES
- Получение IDXGIDevice с использованием ID3D11Device
- Создание ID2D1Device с использованием ID2D1Factory1 и IDD2DD2D2D2D2D2D2 * * * 10D2DD2D2D2D2DD2D2D2D2D2D2D2D2D2DD2D2D2DD2D2D2D2D2D2D2D2D2D2D2D2D2D2D2D2DD2D2D0eD2D0D2D0D2D0D0eE0D0eE ID2D1DeviceContext
- Установить параметры и D2D1_VECTOR_3F из ID2D1Effect
- Возврат новый ID3D11Texture2D привязанных к ID2D1Bitmap
Вот код.
Определяет:
#define SIZEOF_ARRAY(arr) (sizeof(arr)/sizeof((arr)[0]))
#define RET_HR(_f) hr = _f; if (FAILED(hr)) return hr;
#define RET_HR_NULL( _f, _obj ) hr = _f; if (FAILED(hr) || !_obj) return FAILED(hr) ? hr : S_FALSE;
Создание устройства:
inline CComPtr<ID3D11Device> DX11_CreateDevice(IDXGIAdapter* _pAdapter, ID3D11DeviceContext** _ppD2D1DeviceContext = NULL)
{
DXGI_ADAPTER_DESC dxgiDesc = {};
CComQIPtr<IDXGIAdapter> qpDXGIAdapter(_pAdapter);
if (qpDXGIAdapter)
qpDXGIAdapter->GetDesc(&dxgiDesc);
CComPtr<ID3D11Device> cpD3D11Device;
D3D_FEATURE_LEVEL d3dFeature = (D3D_FEATURE_LEVEL)0;
D3D_FEATURE_LEVEL arrD3DFeatures[] = {
D3D_FEATURE_LEVEL_11_1,
D3D_FEATURE_LEVEL_11_0,
D3D_FEATURE_LEVEL_10_1,
D3D_FEATURE_LEVEL_10_0,
D3D_FEATURE_LEVEL_9_3,
D3D_FEATURE_LEVEL_9_2,
D3D_FEATURE_LEVEL_9_1 };
HRESULT hr = S_OK;
RET_HR(D3D11CreateDevice(
qpDXGIAdapter,
qpDXGIAdapter ? D3D_DRIVER_TYPE_UNKNOWN : D3D_DRIVER_TYPE_HARDWARE,
nullptr,
D3D11_CREATE_DEVICE_BGRA_SUPPORT | D3D11_CREATE_DEVICE_DEBUG,
arrD3DFeatures,
SIZEOF_ARRAY(arrD3DFeatures),
D3D11_SDK_VERSION,
&cpD3D11Device,
&d3dFeature,
_ppD2D1DeviceContext
));
CComQIPtr<ID3D10Multithread> qpMTProtect(cpD3D11Device);
ATLASSERT(qpMTProtect);
if (qpMTProtect)
qpMTProtect->SetMultithreadProtected(TRUE);
return cpD3D11Device;
}
Код ChromaKey:
HRESULT DXV_ChromaKey(ID3D11Texture2D* _pTexIn, ID3D11Texture2D** _pTexOut)
{
// -----------------
HRESULT hr = S_OK;
CComPtr<ID2D1Factory1> cpD2DFactory;
RET_HR_NULL(D2D1CreateFactory(D2D1_FACTORY_TYPE_SINGLE_THREADED, &cpD2DFactory), cpD2DFactory);
// Create the DX11 API device object, and get a corresponding context.
CComPtr<ID3D11DeviceContext> cpD3D11DeviceContext;
CComPtr<IDXGIAdapter> cpNVidiaAdpt = DXGI_AdapterByType(eGPU_NVidia);
CComPtr<ID3D11Device> cpD3D11Device = DX11_CreateDevice(cpNVidiaAdpt, &cpD3D11DeviceContext);
CComPtr<IDXGIResource> cpDXGIResource;
RET_HR_NULL(_pTexIn->QueryInterface(__uuidof(IDXGIResource), (void**)&cpDXGIResource), cpDXGIResource);
HANDLE sharedHandle;
cpDXGIResource->GetSharedHandle(&sharedHandle);
CComPtr<ID3D11Texture2D> cpTexIn;
cpD3D11Device->OpenSharedResource(sharedHandle, __uuidof(ID3D11Resource), (void**)(&cpTexIn));
D3D11_TEXTURE2D_DESC td;
cpTexIn->GetDesc(&td);
td.BindFlags = D3D11_BIND_SHADER_RESOURCE;
td.CPUAccessFlags = D3D11_CPU_ACCESS_READ;
CComPtr<ID3D11Texture2D> cpNewTexture;
RET_HR_NULL(cpD3D11Device->CreateTexture2D(&td, NULL, &cpNewTexture), cpNewTexture);
cpD3D11DeviceContext->CopyResource(cpNewTexture, cpTexIn);
CComPtr<IDXGISurface> cpDXGISurface;
RET_HR_NULL(cpNewTexture->QueryInterface(&cpDXGISurface), cpDXGISurface);
// -----------------
D2D1_SIZE_U size = D2D1::SizeU(200, 200); // Adapt to your own size.
D2D1_PIXEL_FORMAT desc2d;
desc2d.format = DXGI_FORMAT_B8G8R8A8_UNORM;
desc2d.alphaMode = D2D1_ALPHA_MODE_IGNORE; // Adapt to your needs.
D2D1_BITMAP_PROPERTIES bmpprops;
bmpprops.dpiX = 96.0f;
bmpprops.dpiY = 96.0f;
bmpprops.pixelFormat = desc2d;
FLOAT dpiX;
FLOAT dpiY;
cpD2DFactory->GetDesktopDpi(&dpiX, &dpiY);
D2D1_RENDER_TARGET_PROPERTIES d2d1RenderTargetProperties =
D2D1::RenderTargetProperties(
D2D1_RENDER_TARGET_TYPE_DEFAULT,
D2D1::PixelFormat(DXGI_FORMAT_UNKNOWN, D2D1_ALPHA_MODE_PREMULTIPLIED),
dpiX,
dpiY
);
CComPtr<ID2D1RenderTarget> cpRenderTarget;
RET_HR_NULL(cpD2DFactory->CreateDxgiSurfaceRenderTarget(
cpDXGISurface,
&d2d1RenderTargetProperties,
&cpRenderTarget),
cpRenderTarget);
// -----------------
CComPtr<ID2D1Bitmap> cpBitmap;
cpRenderTarget->CreateSharedBitmap(__uuidof(IDXGISurface), cpDXGISurface, &bmpprops, &cpBitmap);
// -----------------
CComPtr<IDXGIDevice> cpDXGIDevice;
// Obtain the underlying DXGI device of the Direct3D11 device.
RET_HR_NULL(cpD3D11Device->QueryInterface(&cpDXGIDevice), cpDXGIDevice);
CComPtr<ID2D1Device> cpD2D1Device;
// Obtain the Direct2D device for 2-D rendering.
RET_HR_NULL(cpD2DFactory->CreateDevice(cpDXGIDevice, &cpD2D1Device), cpD2D1Device);
CComPtr<ID2D1DeviceContext> cpD2DContext;
// Get Direct2D device's corresponding device context object.
RET_HR_NULL(cpD2D1Device->CreateDeviceContext(
D2D1_DEVICE_CONTEXT_OPTIONS_NONE,
&cpD2DContext
), cpD2DContext);
CComPtr<ID2D1Effect> chromakeyEffect;
RET_HR_NULL(cpD2DContext->CreateEffect(CLSID_D2D1ChromaKey, &chromakeyEffect), chromakeyEffect);
D2D1_VECTOR_3F Vector3F;
Vector3F.x = .32f;
Vector3F.y = .99f;
Vector3F.z = .48f;
int nSize = sizeof(Vector3F);
BYTE* pData = (BYTE*)&Vector3F;
chromakeyEffect->SetInput(0, cpBitmap);
RET_HR(chromakeyEffect->SetValue(D2D1_CHROMAKEY_PROP_COLOR, pData, nSize));
RET_HR(chromakeyEffect->SetValue(D2D1_CHROMAKEY_PROP_TOLERANCE, 0.2f));
RET_HR(chromakeyEffect->SetValue(D2D1_CHROMAKEY_PROP_INVERT_ALPHA, false));
RET_HR(chromakeyEffect->SetValue(D2D1_CHROMAKEY_PROP_FEATHER, false));
cpD2DContext->BeginDraw();
cpD2DContext->DrawImage(chromakeyEffect);
RET_HR(cpD2DContext->EndDraw());
return cpNewTexture->QueryInterface(_pTexOut);
// -----------------
}
Точка входа:
TEST_METHOD(UT_DXV_ChromaKey)
{
MTimeout timeOut(__FUNCTION__);
HRESULT hr = S_OK;
DX11Texture::TPtr apTexturePicOrg;
DX_TextureLoadFromFile(cbsChromaCat, &apTexturePicOrg);
Assert::IsTrue(apTexturePicOrg);
CComPtr<ID3D11Texture2D> cpD3D11Texture;
hr = m_arrVidProcessor[0]->DXV_ChromaKey(apTexturePicOrg, &cpD3D11Texture);
CComPtr<IMFFrame> cpMFFrameCnv;
MCreator::MFFrameCreateFromTexture(NULL, cpD3D11Texture, 0, NULL, &cpMFFrameCnv, NULL);
cpMFFrameCnv->MFVideoSaveToFile(L"C:\\temp\\Kosh.bmp");
}
Входное изображение:
Сейчас Я застрял на 13-м шаге:
Создание ID2D1RenderTarget с использованием ID2D1Factory1, IDXGISurface и D2D1_RENDER_TARGET_PROPERTIES
RET_HR_NULL(cpD2DFactory->CreateDxgiSurfaceRenderTarget(
cpDXGISurface,
&d2d1RenderTargetProperties,
&cpRenderTarget),
cpRenderTarget);
Несмотря на D3D11_CREATE_DEVICE_DEBUG
Я получаю только следующую ошибку как HRESULT
из CreateDxgiSurfaceRenderTarget
:
1091 * E_INVALIDARG Один или несколько аргументов недействительны.
Вот содержимое td
и описание cpDXGISurface
:
td {Width=0x00000354 Height=0x000001e0 MipLevels=0x00000001 ...} D3D11_TEXTURE2D_DESC
Width 0x00000354 unsigned int
Height 0x000001e0 unsigned int
MipLevels 0x00000001 unsigned int
ArraySize 0x00000001 unsigned int
Format DXGI_FORMAT_B8G8R8A8_UNORM (0x00000057) DXGI_FORMAT
SampleDesc {Count=0x00000001 Quality=0x00000000 } DXGI_SAMPLE_DESC
Usage D3D11_USAGE_DEFAULT (0x00000000) D3D11_USAGE
BindFlags 0x00000008 unsigned int
CPUAccessFlags 0x00020000 unsigned int
MiscFlags 0x00000002 unsigned int
sd {Width=0x00000354 Height=0x000001e0 Format=DXGI_FORMAT_B8G8R8A8_UNORM (0x00000057) ...} DXGI_SURFACE_DESC
Width 0x00000354 unsigned int
Height 0x000001e0 unsigned int
Format DXGI_FORMAT_B8G8R8A8_UNORM (0x00000057) DXGI_FORMAT
SampleDesc {Count=0x00000001 Quality=0x00000000 } DXGI_SAMPLE_DESC