// This file is part of OpenCV project. // It is subject to the license terms in the LICENSE file found in the top-level directory // of this distribution and at http://opencv.org/license.html. // // Copyright (C) 2017, Intel Corporation, all rights reserved. // Third party copyrights are property of their respective owners. #include "precomp.hpp" #include "op_halide.hpp" #ifdef HAVE_HALIDE #include #endif // HAVE_HALIDE namespace cv { namespace dnn { #ifdef HAVE_HALIDE Halide::Buffer wrapToHalideBuffer(const Mat& mat) { int n, c, w, h; getCanonicalSize(mat.size, &w, &h, &c, &n); return wrapToHalideBuffer(mat, {w, h, c, n}); } Halide::Buffer wrapToHalideBuffer(const Mat& mat, const std::vector& sizes) { Halide::Buffer buffer((float*)mat.data, sizes); buffer.set_host_dirty(); // Indicate that data is on CPU. return buffer; } Halide::Buffer<> halideBuffer(const Ptr& ptr) { CV_Assert(!ptr.empty()); return ptr.dynamicCast()->buffer; } std::vector > halideBuffers(const std::vector >& ptrs) { std::vector > vec; vec.reserve(ptrs.size()); for (const Ptr& ptr : ptrs) { vec.push_back(halideBuffer(ptr)); } return vec; } void getCanonicalSize(const Halide::Buffer<>& buffer, int* width, int* height, int* channels, int* batch) { CV_Assert(buffer.dimensions() == 4); *width = buffer.extent(0); *height = buffer.extent(1); *channels = buffer.extent(2); *batch = buffer.extent(3); } HalideBackendNode::HalideBackendNode(const Halide::Func& func) : BackendNode(DNN_BACKEND_HALIDE), funcs(1, func) {} HalideBackendNode::HalideBackendNode(const std::vector& funcs) : BackendNode(DNN_BACKEND_HALIDE), funcs(funcs) {} HalideBackendNode::HalideBackendNode(const Ptr& base, const Halide::Func& top) : BackendNode(DNN_BACKEND_HALIDE), funcs(base->funcs) { funcs.back() = top; } HalideBackendWrapper::HalideBackendWrapper(int targetId, const cv::Mat& m) : BackendWrapper(DNN_BACKEND_HALIDE, targetId) { buffer = wrapToHalideBuffer(m); if (targetId == DNN_TARGET_CPU) { return; } else if (targetId == DNN_TARGET_OPENCL) { buffer.copy_to_device(halide_opencl_device_interface()); } else CV_Error(Error::StsNotImplemented, "Unknown target identifier"); } HalideBackendWrapper::HalideBackendWrapper(const Ptr& base, const MatShape& shape) : BackendWrapper(DNN_BACKEND_HALIDE, base->targetId) { int w, h, c, n; getCanonicalSize(shape, &w, &h, &c, &n); Halide::Buffer baseBuffer = halideBuffer(base); buffer = Halide::Buffer((float*)baseBuffer.raw_buffer()->host, {w, h, c, n}); if (baseBuffer.has_device_allocation()) { buffer.raw_buffer()->device = baseBuffer.raw_buffer()->device; buffer.raw_buffer()->device_interface = baseBuffer.raw_buffer()->device_interface; buffer.set_device_dirty(); } else { buffer.set_host_dirty(); // Indicate that data is on CPU. CV_Assert(targetId == DNN_TARGET_CPU); } } void HalideBackendWrapper::copyToHost() { CV_Assert(targetId == DNN_TARGET_CPU || buffer.device_dirty()); if (buffer.device_dirty()) { buffer.device_sync(); buffer.copy_to_host(); } } #endif // HAVE_HALIDE void getCanonicalSize(const MatSize& size, int* width, int* height, int* channels, int* batch) { const int dims = size.p[-1]; CV_Assert(dims == 2 || dims == 4); *batch = size[0]; *channels = size[1]; if (dims == 4) { *width = size[3]; *height = size[2]; } else { *width = 1; *height = 1; } } void getCanonicalSize(const MatShape& shape, int* width, int* height, int* channels, int* batch) { const int dims = shape.size(); CV_Assert(dims == 2 || dims == 4); *batch = shape[0]; *channels = shape[1]; if (dims == 4) { *width = shape[3]; *height = shape[2]; } else { *width = 1; *height = 1; } } void compileHalide(std::vector &outputs, Ptr& node, int targetId) { #ifdef HAVE_HALIDE CV_Assert(!node.empty()); Halide::Func& top = node.dynamicCast()->funcs.back(); int outW, outH, outC, outN; Halide::Var x("x"), y("y"), c("c"), n("n"); getCanonicalSize(outputs[0].size, &outW, &outH, &outC, &outN); top.bound(x, 0, outW).bound(y, 0, outH) .bound(c, 0, outC).bound(n, 0, outN); Halide::Target target = Halide::get_host_target(); target.set_feature(Halide::Target::NoAsserts); if (targetId == DNN_TARGET_OPENCL) { target.set_feature(Halide::Target::OpenCL); } CV_Assert(target.supported()); top.compile_jit(target); #endif // HAVE_HALIDE } void forwardHalide(std::vector > &outputs, const Ptr& node) { #ifdef HAVE_HALIDE CV_Assert(!node.empty()); Halide::Func& top = node.dynamicCast()->funcs.back(); auto outputBuffers = halideBuffers(outputs); top.realize(Halide::Realization(outputBuffers)); #endif // HAVE_HALIDE } bool haveHalide() { #ifdef HAVE_HALIDE return true; #else return false; #endif // HAVE_HALIDE } } // namespace dnn } // namespace cv