Path: blob/master/samples/directx/d3d9ex_interop.cpp
16337 views
/*1// A sample program demonstrating interoperability of OpenCV cv::UMat with Direct X surface2// At first, the data obtained from video file or camera and placed onto Direct X surface,3// following mapping of this Direct X surface to OpenCV cv::UMat and call cv::Blur function.4// The result is mapped back to Direct X surface and rendered through Direct X API.5*/67#define WIN32_LEAN_AND_MEAN8#include <windows.h>9#include <d3d9.h>1011#include "opencv2/core.hpp"12#include "opencv2/core/directx.hpp"13#include "opencv2/core/ocl.hpp"14#include "opencv2/imgproc.hpp"15#include "opencv2/videoio.hpp"1617#include "d3dsample.hpp"1819#pragma comment (lib, "d3d9.lib")202122class D3D9ExWinApp : public D3DSample23{24public:25D3D9ExWinApp(int width, int height, std::string& window_name, cv::VideoCapture& cap) :26D3DSample(width, height, window_name, cap) {}2728~D3D9ExWinApp() {}2930int create(void)31{32// base initialization33D3DSample::create();3435// initialize DirectX36HRESULT r;3738r = ::Direct3DCreate9Ex(D3D_SDK_VERSION, &m_pD3D9Ex);39if (FAILED(r))40{41return EXIT_FAILURE;42}4344DWORD flags = D3DCREATE_HARDWARE_VERTEXPROCESSING |45D3DCREATE_PUREDEVICE |46D3DCREATE_NOWINDOWCHANGES |47D3DCREATE_MULTITHREADED |48D3DCREATE_FPU_PRESERVE;4950D3DPRESENT_PARAMETERS d3dpp;51::ZeroMemory(&d3dpp, sizeof(D3DPRESENT_PARAMETERS));5253d3dpp.Windowed = true;54d3dpp.Flags = 0;55d3dpp.BackBufferCount = 0;56d3dpp.BackBufferFormat = D3DFMT_A8R8G8B8;57d3dpp.BackBufferHeight = m_height;58d3dpp.BackBufferWidth = m_width;59d3dpp.MultiSampleType = D3DMULTISAMPLE_NONE;60d3dpp.SwapEffect = D3DSWAPEFFECT_DISCARD;61d3dpp.hDeviceWindow = m_hWnd;62d3dpp.PresentationInterval = D3DPRESENT_INTERVAL_IMMEDIATE;63d3dpp.FullScreen_RefreshRateInHz = D3DPRESENT_RATE_DEFAULT;6465r = m_pD3D9Ex->CreateDeviceEx(D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL, m_hWnd, flags, &d3dpp, NULL, &m_pD3D9DevEx);66if (FAILED(r))67{68return EXIT_FAILURE;69}7071r = m_pD3D9DevEx->GetBackBuffer(0, 0, D3DBACKBUFFER_TYPE_MONO, &m_pBackBuffer);72if (FAILED(r))73{74return EXIT_FAILURE;75}7677r = m_pD3D9DevEx->CreateOffscreenPlainSurface(m_width, m_height, D3DFMT_A8R8G8B8, D3DPOOL_DEFAULT, &m_pSurface, NULL);78if (FAILED(r))79{80std::cerr << "Can't create surface for result" << std::endl;81return EXIT_FAILURE;82}8384// initialize OpenCL context of OpenCV lib from DirectX85if (cv::ocl::haveOpenCL())86{87m_oclCtx = cv::directx::ocl::initializeContextFromDirect3DDevice9(m_pD3D9DevEx);88}8990m_oclDevName = cv::ocl::useOpenCL() ?91cv::ocl::Context::getDefault().device(0).name() :92"No OpenCL device";9394return EXIT_SUCCESS;95} // create()969798// get media data on DX surface for further processing99int get_surface(LPDIRECT3DSURFACE9* ppSurface)100{101HRESULT r;102103if (!m_cap.read(m_frame_bgr))104return EXIT_FAILURE;105106cv::cvtColor(m_frame_bgr, m_frame_rgba, cv::COLOR_BGR2BGRA);107108D3DLOCKED_RECT memDesc = { 0, NULL };109RECT rc = { 0, 0, m_width, m_height };110111r = m_pSurface->LockRect(&memDesc, &rc, 0);112if (FAILED(r))113{114return r;115}116117cv::Mat m(m_height, m_width, CV_8UC4, memDesc.pBits, memDesc.Pitch);118// copy video frame data to surface119m_frame_rgba.copyTo(m);120121r = m_pSurface->UnlockRect();122if (FAILED(r))123{124return r;125}126127*ppSurface = m_pSurface;128129return EXIT_SUCCESS;130} // get_surface()131132133// process and render media data134int render()135{136try137{138if (m_shutdown)139return EXIT_SUCCESS;140141// capture user input once142MODE mode = m_mode == MODE_GPU_NV12 ? MODE_GPU_RGBA : m_mode;143144HRESULT r;145LPDIRECT3DSURFACE9 pSurface;146147r = get_surface(&pSurface);148if (FAILED(r))149{150return EXIT_FAILURE;151}152153m_timer.reset();154m_timer.start();155156switch (mode)157{158case MODE_CPU:159{160// process video frame on CPU161D3DLOCKED_RECT memDesc = { 0, NULL };162RECT rc = { 0, 0, m_width, m_height };163164r = pSurface->LockRect(&memDesc, &rc, 0);165if (FAILED(r))166{167return EXIT_FAILURE;168}169170cv::Mat m(m_height, m_width, CV_8UC4, memDesc.pBits, memDesc.Pitch);171172if (m_demo_processing)173{174// blur D3D9 surface with OpenCV on CPU175cv::blur(m, m, cv::Size(15, 15));176}177178r = pSurface->UnlockRect();179if (FAILED(r))180{181return EXIT_FAILURE;182}183184break;185}186187case MODE_GPU_RGBA:188{189// process video frame on GPU190cv::UMat u;191192cv::directx::convertFromDirect3DSurface9(pSurface, u);193194if (m_demo_processing)195{196// blur D3D9 surface with OpenCV on GPU with OpenCL197cv::blur(u, u, cv::Size(15, 15));198}199200cv::directx::convertToDirect3DSurface9(u, pSurface);201202break;203}204205} // switch206207m_timer.stop();208209print_info(pSurface, m_mode, m_timer.getTimeMilli(), m_oclDevName);210211// traditional DX render pipeline:212// BitBlt surface to backBuffer and flip backBuffer to frontBuffer213r = m_pD3D9DevEx->StretchRect(pSurface, NULL, m_pBackBuffer, NULL, D3DTEXF_NONE);214if (FAILED(r))215{216return EXIT_FAILURE;217}218219// present the back buffer contents to the display220r = m_pD3D9DevEx->Present(NULL, NULL, NULL, NULL);221if (FAILED(r))222{223return EXIT_FAILURE;224}225226} // try227228catch (cv::Exception& e)229{230std::cerr << "Exception: " << e.what() << std::endl;231return 10;232}233234return EXIT_SUCCESS;235} // render()236237238void print_info(LPDIRECT3DSURFACE9 pSurface, int mode, double time, cv::String oclDevName)239{240HDC hDC;241242HRESULT r = pSurface->GetDC(&hDC);243if (FAILED(r))244{245return;246}247248HFONT hFont = (HFONT)::GetStockObject(SYSTEM_FONT);249250HFONT hOldFont = (HFONT)::SelectObject(hDC, hFont);251252if (hOldFont)253{254TEXTMETRIC tm;255::GetTextMetrics(hDC, &tm);256257char buf[256];258int y = 0;259260buf[0] = 0;261sprintf(buf, "mode: %s", m_modeStr[mode].c_str());262::TextOut(hDC, 0, y, buf, (int)strlen(buf));263264y += tm.tmHeight;265buf[0] = 0;266sprintf(buf, m_demo_processing ? "blur frame" : "copy frame");267::TextOut(hDC, 0, y, buf, (int)strlen(buf));268269y += tm.tmHeight;270buf[0] = 0;271sprintf(buf, "time: %4.1f msec", time);272::TextOut(hDC, 0, y, buf, (int)strlen(buf));273274y += tm.tmHeight;275buf[0] = 0;276sprintf(buf, "OpenCL device: %s", oclDevName.c_str());277::TextOut(hDC, 0, y, buf, (int)strlen(buf));278279::SelectObject(hDC, hOldFont);280}281282r = pSurface->ReleaseDC(hDC);283284return;285} // print_info()286287288int cleanup(void)289{290SAFE_RELEASE(m_pSurface);291SAFE_RELEASE(m_pBackBuffer);292SAFE_RELEASE(m_pD3D9DevEx);293SAFE_RELEASE(m_pD3D9Ex);294D3DSample::cleanup();295return EXIT_SUCCESS;296} // cleanup()297298private:299LPDIRECT3D9EX m_pD3D9Ex;300LPDIRECT3DDEVICE9EX m_pD3D9DevEx;301LPDIRECT3DSURFACE9 m_pBackBuffer;302LPDIRECT3DSURFACE9 m_pSurface;303cv::ocl::Context m_oclCtx;304cv::String m_oclPlatformName;305cv::String m_oclDevName;306};307308309// main func310int main(int argc, char** argv)311{312std::string title = "D3D9Ex interop sample";313return d3d_app<D3D9ExWinApp>(argc, argv, title);314}315316317