d4774ead43
fixed issues with ocl nv12 cvt kernel finisged ocl nv12-to-rgba kernel, update dx-interop samples. (ocl rgba-to-nv12 kernel will be added later) an attempt to fix build issue fix for non opencl build issue fix typo fix compilation warnings fix compile issue for Mac (OpenCL) add convertion from rgba to nv12 (still need to debug kernel) remove empty line at the EOF fixed compilation warning
226 lines
4.9 KiB
C++
226 lines
4.9 KiB
C++
/*
|
|
// Sample demonstrating interoperability of OpenCV UMat with Direct X surface
|
|
// Base class for Direct X application
|
|
*/
|
|
#include <string>
|
|
#include <iostream>
|
|
#include <queue>
|
|
|
|
#include "opencv2/core.hpp"
|
|
#include "opencv2/core/directx.hpp"
|
|
#include "opencv2/core/ocl.hpp"
|
|
#include "opencv2/imgproc.hpp"
|
|
#include "opencv2/videoio.hpp"
|
|
|
|
#include "winapp.hpp"
|
|
|
|
#define SAFE_RELEASE(p) if (p) { p->Release(); p = NULL; }
|
|
|
|
|
|
class Timer
|
|
{
|
|
public:
|
|
enum UNITS
|
|
{
|
|
USEC = 0,
|
|
MSEC,
|
|
SEC
|
|
};
|
|
|
|
Timer() : m_t0(0), m_diff(0)
|
|
{
|
|
m_tick_frequency = (float)cv::getTickFrequency();
|
|
|
|
m_unit_mul[USEC] = 1000000;
|
|
m_unit_mul[MSEC] = 1000;
|
|
m_unit_mul[SEC] = 1;
|
|
}
|
|
|
|
void start()
|
|
{
|
|
m_t0 = cv::getTickCount();
|
|
}
|
|
|
|
void stop()
|
|
{
|
|
m_diff = cv::getTickCount() - m_t0;
|
|
}
|
|
|
|
float time(UNITS u = UNITS::MSEC)
|
|
{
|
|
float sec = m_diff / m_tick_frequency;
|
|
|
|
return sec * m_unit_mul[u];
|
|
}
|
|
|
|
public:
|
|
float m_tick_frequency;
|
|
int64 m_t0;
|
|
int64 m_diff;
|
|
int m_unit_mul[3];
|
|
};
|
|
|
|
|
|
class D3DSample : public WinApp
|
|
{
|
|
public:
|
|
enum MODE
|
|
{
|
|
MODE_CPU,
|
|
MODE_GPU_RGBA,
|
|
MODE_GPU_NV12
|
|
};
|
|
|
|
D3DSample(int width, int height, std::string& window_name, cv::VideoCapture& cap) :
|
|
WinApp(width, height, window_name)
|
|
{
|
|
m_shutdown = false;
|
|
m_mode = MODE_CPU;
|
|
m_modeStr[0] = cv::String("Processing on CPU");
|
|
m_modeStr[1] = cv::String("Processing on GPU RGBA");
|
|
m_modeStr[2] = cv::String("Processing on GPU NV12");
|
|
m_demo_processing = false;
|
|
m_cap = cap;
|
|
}
|
|
|
|
~D3DSample() {}
|
|
|
|
virtual int create() { return WinApp::create(); }
|
|
virtual int render() = 0;
|
|
virtual int cleanup()
|
|
{
|
|
m_shutdown = true;
|
|
return WinApp::cleanup();
|
|
}
|
|
|
|
protected:
|
|
virtual LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
switch (message)
|
|
{
|
|
case WM_CHAR:
|
|
if (wParam == '1')
|
|
{
|
|
m_mode = MODE_CPU;
|
|
return 0;
|
|
}
|
|
if (wParam == '2')
|
|
{
|
|
m_mode = MODE_GPU_RGBA;
|
|
return 0;
|
|
}
|
|
if (wParam == '3')
|
|
{
|
|
m_mode = MODE_GPU_NV12;
|
|
return 0;
|
|
}
|
|
else if (wParam == VK_SPACE)
|
|
{
|
|
m_demo_processing = !m_demo_processing;
|
|
return 0;
|
|
}
|
|
else if (wParam == VK_ESCAPE)
|
|
{
|
|
return cleanup();
|
|
}
|
|
break;
|
|
|
|
case WM_CLOSE:
|
|
return cleanup();
|
|
|
|
case WM_DESTROY:
|
|
::PostQuitMessage(0);
|
|
return 0;
|
|
}
|
|
|
|
return ::DefWindowProc(hWnd, message, wParam, lParam);
|
|
}
|
|
|
|
// do render at idle
|
|
virtual int idle() { return render(); }
|
|
|
|
protected:
|
|
bool m_shutdown;
|
|
bool m_demo_processing;
|
|
MODE m_mode;
|
|
cv::String m_modeStr[3];
|
|
cv::VideoCapture m_cap;
|
|
cv::Mat m_frame_bgr;
|
|
cv::Mat m_frame_rgba;
|
|
Timer m_timer;
|
|
};
|
|
|
|
|
|
static void help()
|
|
{
|
|
printf(
|
|
"\nSample demonstrating interoperability of DirectX and OpenCL with OpenCV.\n"
|
|
"Hot keys: \n"
|
|
" SPACE - turn processing on/off\n"
|
|
" 1 - process DX surface through OpenCV on CPU\n"
|
|
" 2 - process DX RGBA surface through OpenCV on GPU (via OpenCL)\n"
|
|
" 3 - process DX NV12 surface through OpenCV on GPU (via OpenCL)\n"
|
|
" ESC - exit\n\n");
|
|
}
|
|
|
|
|
|
static const char* keys =
|
|
{
|
|
"{c camera | true | use camera or not}"
|
|
"{f file | | movie file name }"
|
|
"{h help | false | print help info }"
|
|
};
|
|
|
|
|
|
template <typename TApp>
|
|
int d3d_app(int argc, char** argv, std::string& title)
|
|
{
|
|
cv::CommandLineParser parser(argc, argv, keys); \
|
|
bool useCamera = parser.has("camera"); \
|
|
string file = parser.get<string>("file"); \
|
|
bool showHelp = parser.get<bool>("help"); \
|
|
|
|
if (showHelp)
|
|
help();
|
|
|
|
parser.printMessage();
|
|
|
|
cv::VideoCapture cap;
|
|
|
|
if (useCamera)
|
|
cap.open(0);
|
|
else
|
|
cap.open(file.c_str());
|
|
|
|
if (!cap.isOpened())
|
|
{
|
|
printf("can not open camera or video file\n");
|
|
return -1;
|
|
}
|
|
|
|
int width = (int)cap.get(CAP_PROP_FRAME_WIDTH);
|
|
int height = (int)cap.get(CAP_PROP_FRAME_HEIGHT);
|
|
|
|
std::string wndname = title;
|
|
|
|
TApp app(width, height, wndname, cap);
|
|
|
|
try
|
|
{
|
|
app.create();
|
|
return app.run();
|
|
}
|
|
|
|
catch (cv::Exception& e)
|
|
{
|
|
std::cerr << "Exception: " << e.what() << std::endl;
|
|
return 10;
|
|
}
|
|
|
|
catch (...)
|
|
{
|
|
std::cerr << "FATAL ERROR: Unknown exception" << std::endl;
|
|
return 11;
|
|
}
|
|
}
|