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;
 | 
						|
    }
 | 
						|
}
 |