#ifndef OPENCV_OPENVX_HAL_HPP_INCLUDED #define OPENCV_OPENVX_HAL_HPP_INCLUDED #include "opencv2/core/hal/interface.h" #include "VX/vx.h" #include "VX/vxu.h" #include //================================================================================================== // utility // ... #if 0 #include #define PRINT(...) printf(__VA_ARGS__) #else #define PRINT(...) #endif #if __cplusplus >= 201103L #include struct Tick { typedef std::chrono::time_point point_t; point_t start; point_t point; Tick() { start = std::chrono::steady_clock::now(); point = std::chrono::steady_clock::now(); } inline int one() { point_t old = point; point = std::chrono::steady_clock::now(); return std::chrono::duration_cast(point - old).count(); } inline int total() { return std::chrono::duration_cast(std::chrono::steady_clock::now() - start).count(); } }; #endif //================================================================================================== // One more OpenVX C++ binding :-) // ... template struct VX_Traits { enum { Type = 0 }; }; template <> struct VX_Traits { enum { Type = VX_DF_IMAGE_U8 }; }; template <> struct VX_Traits { enum { Type = VX_DF_IMAGE_U16 }; }; template <> struct VX_Traits { enum { Type = VX_DF_IMAGE_S16 }; }; struct vxContext; struct vxImage; struct vxErr; struct vxErr { vx_status status; std::string msg; vxErr(vx_status status_, const std::string & msg_) : status(status_), msg(msg_) {} void check() { if (status != VX_SUCCESS) throw *this; } void print() { PRINT("OpenVX HAL impl error: %d (%s)\n", status, msg.c_str()); } static void check(vx_context ctx) { vxErr(vxGetStatus((vx_reference)ctx), "context check").check(); } static void check(vx_image img) { vxErr(vxGetStatus((vx_reference)img), "image check").check(); } static void check(vx_status s) { vxErr(s, "status check").check(); } }; struct vxContext { vx_context ctx; static vxContext * getContext(); private: vxContext() { ctx = vxCreateContext(); vxErr::check(ctx); } ~vxContext() { vxReleaseContext(&ctx); } }; struct vxImage { vx_image img; template vxImage(vxContext &ctx, const T *data, size_t step, int w, int h) { if (h == 1) step = w * sizeof(T); vx_imagepatch_addressing_t addr; addr.dim_x = w; addr.dim_y = h; addr.stride_x = sizeof(T); addr.stride_y = step; addr.scale_x = VX_SCALE_UNITY; addr.scale_y = VX_SCALE_UNITY; addr.step_x = 1; addr.step_y = 1; void *ptrs[] = { (void*)data }; img = vxCreateImageFromHandle(ctx.ctx, VX_Traits::Type, &addr, ptrs, VX_MEMORY_TYPE_HOST); vxErr::check(img); } ~vxImage() { vxErr::check(vxSwapImageHandle(img, NULL, NULL, 1)); vxReleaseImage(&img); } }; //================================================================================================== // real code starts here // ... #define OVX_BINARY_OP(hal_func, ovx_call, ...) \ template \ inline int ovx_hal_##hal_func(const T *a, size_t astep, const T *b, size_t bstep, T *c, size_t cstep, int w, int h, __VA_ARGS__) \ { \ try \ { \ vxContext * ctx = vxContext::getContext(); \ vxImage ia(*ctx, a, astep, w, h); \ vxImage ib(*ctx, b, bstep, w, h); \ vxImage ic(*ctx, c, cstep, w, h); \ ovx_call \ } \ catch (vxErr & e) \ { \ e.print(); \ return CV_HAL_ERROR_UNKNOWN; \ } \ return CV_HAL_ERROR_OK; \ } OVX_BINARY_OP(add, {vxErr::check(vxuAdd(ctx->ctx, ia.img, ib.img, VX_CONVERT_POLICY_SATURATE, ic.img));}) OVX_BINARY_OP(sub, {vxErr::check(vxuSubtract(ctx->ctx, ia.img, ib.img, VX_CONVERT_POLICY_SATURATE, ic.img));}) OVX_BINARY_OP(absdiff, {vxErr::check(vxuAbsDiff(ctx->ctx, ia.img, ib.img, ic.img));}) OVX_BINARY_OP(and, {vxErr::check(vxuAnd(ctx->ctx, ia.img, ib.img, ic.img));}) OVX_BINARY_OP(or, {vxErr::check(vxuOr(ctx->ctx, ia.img, ib.img, ic.img));}) OVX_BINARY_OP(xor, {vxErr::check(vxuXor(ctx->ctx, ia.img, ib.img, ic.img));}) OVX_BINARY_OP(mul, {vxErr::check(vxuMultiply(ctx->ctx, ia.img, ib.img, (float)scale, VX_CONVERT_POLICY_SATURATE, VX_ROUND_POLICY_TO_ZERO, ic.img));}, double scale) inline int ovx_hal_not(const uchar *a, size_t astep, uchar *c, size_t cstep, int w, int h) { try { vxContext * ctx = vxContext::getContext(); vxImage ia(*ctx, a, astep, w, h); vxImage ic(*ctx, c, cstep, w, h); vxErr::check(vxuNot(ctx->ctx, ia.img, ic.img)); } catch (vxErr & e) { e.print(); return CV_HAL_ERROR_UNKNOWN; } return CV_HAL_ERROR_OK; } //================================================================================================== // functions redefinition // ... #undef cv_hal_add8u #define cv_hal_add8u ovx_hal_add #undef cv_hal_add16s #define cv_hal_add16s ovx_hal_add #undef cv_hal_sub8u #define cv_hal_sub8u ovx_hal_sub #undef cv_hal_sub16s #define cv_hal_sub16s ovx_hal_sub #undef cv_hal_absdiff8u #define cv_hal_absdiff8u ovx_hal_absdiff #undef cv_hal_absdiff16s #define cv_hal_absdiff16s ovx_hal_absdiff #undef cv_hal_and8u #define cv_hal_and8u ovx_hal_and #undef cv_hal_or8u #define cv_hal_or8u ovx_hal_or #undef cv_hal_xor8u #define cv_hal_xor8u ovx_hal_xor #undef cv_hal_not8u #define cv_hal_not8u ovx_hal_not #undef cv_hal_mul8u #define cv_hal_mul8u ovx_hal_mul #undef cv_hal_mul16s #define cv_hal_mul16s ovx_hal_mul #endif