123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249 |
- // 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) 2018-2020 Intel Corporation
- #ifndef OPENCV_GAPI_GOCLKERNEL_HPP
- #define OPENCV_GAPI_GOCLKERNEL_HPP
- #include <vector>
- #include <functional>
- #include <map>
- #include <unordered_map>
- #include <opencv2/core/mat.hpp>
- #include <opencv2/gapi/gcommon.hpp>
- #include <opencv2/gapi/gkernel.hpp>
- #include <opencv2/gapi/garg.hpp>
- // FIXME: namespace scheme for backends?
- namespace cv {
- namespace gimpl
- {
- // Forward-declare an internal class
- class GOCLExecutable;
- } // namespace gimpl
- namespace gapi
- {
- /**
- * @brief This namespace contains G-API OpenCL backend functions, structures, and symbols.
- */
- namespace ocl
- {
- /**
- * \addtogroup gapi_std_backends G-API Standard Backends
- * @{
- */
- /**
- * @brief Get a reference to OCL backend.
- *
- * At the moment, the OCL backend is built atop of OpenCV
- * "Transparent API" (T-API), see cv::UMat for details.
- *
- * @sa gapi_std_backends
- */
- GAPI_EXPORTS cv::gapi::GBackend backend();
- /** @} */
- } // namespace ocl
- } // namespace gapi
- // Represents arguments which are passed to a wrapped OCL function
- // FIXME: put into detail?
- class GAPI_EXPORTS GOCLContext
- {
- public:
- // Generic accessor API
- template<typename T>
- const T& inArg(int input) { return m_args.at(input).get<T>(); }
- // Syntax sugar
- const cv::UMat& inMat(int input);
- cv::UMat& outMatR(int output); // FIXME: Avoid cv::Mat m = ctx.outMatR()
- const cv::Scalar& inVal(int input);
- cv::Scalar& outValR(int output); // FIXME: Avoid cv::Scalar s = ctx.outValR()
- template<typename T> std::vector<T>& outVecR(int output) // FIXME: the same issue
- {
- return outVecRef(output).wref<T>();
- }
- template<typename T> T& outOpaqueR(int output) // FIXME: the same issue
- {
- return outOpaqueRef(output).wref<T>();
- }
- protected:
- detail::VectorRef& outVecRef(int output);
- detail::OpaqueRef& outOpaqueRef(int output);
- std::vector<GArg> m_args;
- std::unordered_map<std::size_t, GRunArgP> m_results;
- friend class gimpl::GOCLExecutable;
- };
- class GAPI_EXPORTS GOCLKernel
- {
- public:
- // This function is kernel's execution entry point (does the processing work)
- using F = std::function<void(GOCLContext &)>;
- GOCLKernel();
- explicit GOCLKernel(const F& f);
- void apply(GOCLContext &ctx);
- protected:
- F m_f;
- };
- // FIXME: This is an ugly ad-hoc implementation. TODO: refactor
- namespace detail
- {
- template<class T> struct ocl_get_in;
- template<> struct ocl_get_in<cv::GMat>
- {
- static cv::UMat get(GOCLContext &ctx, int idx) { return ctx.inMat(idx); }
- };
- template<> struct ocl_get_in<cv::GScalar>
- {
- static cv::Scalar get(GOCLContext &ctx, int idx) { return ctx.inVal(idx); }
- };
- template<typename U> struct ocl_get_in<cv::GArray<U> >
- {
- static const std::vector<U>& get(GOCLContext &ctx, int idx) { return ctx.inArg<VectorRef>(idx).rref<U>(); }
- };
- template<typename U> struct ocl_get_in<cv::GOpaque<U> >
- {
- static const U& get(GOCLContext &ctx, int idx) { return ctx.inArg<OpaqueRef>(idx).rref<U>(); }
- };
- template<class T> struct ocl_get_in
- {
- static T get(GOCLContext &ctx, int idx) { return ctx.inArg<T>(idx); }
- };
- struct tracked_cv_umat{
- //TODO Think if T - API could reallocate UMat to a proper size - how do we handle this ?
- //tracked_cv_umat(cv::UMat& m) : r{(m)}, original_data{m.getMat(ACCESS_RW).data} {}
- tracked_cv_umat(cv::UMat& m) : r(m), original_data{ nullptr } {}
- cv::UMat &r; // FIXME: It was a value (not a reference) before.
- // Actually OCL backend should allocate its internal data!
- uchar* original_data;
- operator cv::UMat& (){ return r;}
- void validate() const{
- //if (r.getMat(ACCESS_RW).data != original_data)
- //{
- // util::throw_error
- // (std::logic_error
- // ("OpenCV kernel output parameter was reallocated. \n"
- // "Incorrect meta data was provided ?"));
- //}
- }
- };
- template<typename... Outputs>
- void postprocess_ocl(Outputs&... outs)
- {
- struct
- {
- void operator()(tracked_cv_umat* bm) { bm->validate(); }
- void operator()(...) { }
- } validate;
- //dummy array to unfold parameter pack
- int dummy[] = { 0, (validate(&outs), 0)... };
- cv::util::suppress_unused_warning(dummy);
- }
- template<class T> struct ocl_get_out;
- template<> struct ocl_get_out<cv::GMat>
- {
- static tracked_cv_umat get(GOCLContext &ctx, int idx)
- {
- auto& r = ctx.outMatR(idx);
- return{ r };
- }
- };
- template<> struct ocl_get_out<cv::GScalar>
- {
- static cv::Scalar& get(GOCLContext &ctx, int idx)
- {
- return ctx.outValR(idx);
- }
- };
- template<typename U> struct ocl_get_out<cv::GArray<U> >
- {
- static std::vector<U>& get(GOCLContext &ctx, int idx) { return ctx.outVecR<U>(idx); }
- };
- template<typename U> struct ocl_get_out<cv::GOpaque<U> >
- {
- static U& get(GOCLContext &ctx, int idx) { return ctx.outOpaqueR<U>(idx); }
- };
- template<typename, typename, typename>
- struct OCLCallHelper;
- // FIXME: probably can be simplified with std::apply or analogue.
- template<typename Impl, typename... Ins, typename... Outs>
- struct OCLCallHelper<Impl, std::tuple<Ins...>, std::tuple<Outs...> >
- {
- template<typename... Inputs>
- struct call_and_postprocess
- {
- template<typename... Outputs>
- static void call(Inputs&&... ins, Outputs&&... outs)
- {
- //not using a std::forward on outs is deliberate in order to
- //cause compilation error, by trying to bind rvalue references to lvalue references
- Impl::run(std::forward<Inputs>(ins)..., outs...);
- postprocess_ocl(outs...);
- }
- };
- template<int... IIs, int... OIs>
- static void call_impl(GOCLContext &ctx, detail::Seq<IIs...>, detail::Seq<OIs...>)
- {
- //TODO: Make sure that OpenCV kernels do not reallocate memory for output parameters
- //by comparing it's state (data ptr) before and after the call.
- //Convert own::Scalar to cv::Scalar before call kernel and run kernel
- //convert cv::Scalar to own::Scalar after call kernel and write back results
- call_and_postprocess<decltype(ocl_get_in<Ins>::get(ctx, IIs))...>::call(ocl_get_in<Ins>::get(ctx, IIs)..., ocl_get_out<Outs>::get(ctx, OIs)...);
- }
- static void call(GOCLContext &ctx)
- {
- call_impl(ctx,
- typename detail::MkSeq<sizeof...(Ins)>::type(),
- typename detail::MkSeq<sizeof...(Outs)>::type());
- }
- };
- } // namespace detail
- template<class Impl, class K>
- class GOCLKernelImpl: public cv::detail::OCLCallHelper<Impl, typename K::InArgs, typename K::OutArgs>,
- public cv::detail::KernelTag
- {
- using P = detail::OCLCallHelper<Impl, typename K::InArgs, typename K::OutArgs>;
- public:
- using API = K;
- static cv::gapi::GBackend backend() { return cv::gapi::ocl::backend(); }
- static cv::GOCLKernel kernel() { return GOCLKernel(&P::call); }
- };
- #define GAPI_OCL_KERNEL(Name, API) struct Name: public cv::GOCLKernelImpl<Name, API>
- } // namespace cv
- #endif // OPENCV_GAPI_GOCLKERNEL_HPP
|