python_bridge.hpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338
  1. // This file is part of OpenCV project.
  2. // It is subject to the license terms in the LICENSE file found in the top-level directory
  3. // of this distribution and at http://opencv.org/license.html.
  4. //
  5. // Copyright (C) 2021 Intel Corporation
  6. #ifndef OPENCV_GAPI_PYTHON_BRIDGE_HPP
  7. #define OPENCV_GAPI_PYTHON_BRIDGE_HPP
  8. #include <opencv2/gapi.hpp>
  9. #include <opencv2/gapi/garg.hpp>
  10. #include <opencv2/gapi/gopaque.hpp>
  11. #include <opencv2/gapi/render/render_types.hpp> // Prim
  12. #define ID(T, E) T
  13. #define ID_(T, E) ID(T, E),
  14. #define WRAP_ARGS(T, E, G) \
  15. G(T, E)
  16. #define SWITCH(type, LIST_G, HC) \
  17. switch(type) { \
  18. LIST_G(HC, HC) \
  19. default: \
  20. GAPI_Assert(false && "Unsupported type"); \
  21. }
  22. using cv::gapi::wip::draw::Prim;
  23. #define GARRAY_TYPE_LIST_G(G, G2) \
  24. WRAP_ARGS(bool , cv::gapi::ArgType::CV_BOOL, G) \
  25. WRAP_ARGS(int , cv::gapi::ArgType::CV_INT, G) \
  26. WRAP_ARGS(int64_t , cv::gapi::ArgType::CV_INT64, G) \
  27. WRAP_ARGS(double , cv::gapi::ArgType::CV_DOUBLE, G) \
  28. WRAP_ARGS(float , cv::gapi::ArgType::CV_FLOAT, G) \
  29. WRAP_ARGS(std::string , cv::gapi::ArgType::CV_STRING, G) \
  30. WRAP_ARGS(cv::Point , cv::gapi::ArgType::CV_POINT, G) \
  31. WRAP_ARGS(cv::Point2f , cv::gapi::ArgType::CV_POINT2F, G) \
  32. WRAP_ARGS(cv::Size , cv::gapi::ArgType::CV_SIZE, G) \
  33. WRAP_ARGS(cv::Rect , cv::gapi::ArgType::CV_RECT, G) \
  34. WRAP_ARGS(cv::Scalar , cv::gapi::ArgType::CV_SCALAR, G) \
  35. WRAP_ARGS(cv::Mat , cv::gapi::ArgType::CV_MAT, G) \
  36. WRAP_ARGS(Prim , cv::gapi::ArgType::CV_DRAW_PRIM, G) \
  37. WRAP_ARGS(cv::GArg , cv::gapi::ArgType::CV_ANY, G) \
  38. WRAP_ARGS(cv::GMat , cv::gapi::ArgType::CV_GMAT, G2) \
  39. #define GOPAQUE_TYPE_LIST_G(G, G2) \
  40. WRAP_ARGS(bool , cv::gapi::ArgType::CV_BOOL, G) \
  41. WRAP_ARGS(int , cv::gapi::ArgType::CV_INT, G) \
  42. WRAP_ARGS(int64_t , cv::gapi::ArgType::CV_INT64, G) \
  43. WRAP_ARGS(double , cv::gapi::ArgType::CV_DOUBLE, G) \
  44. WRAP_ARGS(float , cv::gapi::ArgType::CV_FLOAT, G) \
  45. WRAP_ARGS(std::string , cv::gapi::ArgType::CV_STRING, G) \
  46. WRAP_ARGS(cv::Point , cv::gapi::ArgType::CV_POINT, G) \
  47. WRAP_ARGS(cv::Point2f , cv::gapi::ArgType::CV_POINT2F, G) \
  48. WRAP_ARGS(cv::Size , cv::gapi::ArgType::CV_SIZE, G) \
  49. WRAP_ARGS(cv::GArg , cv::gapi::ArgType::CV_ANY, G) \
  50. WRAP_ARGS(cv::Rect , cv::gapi::ArgType::CV_RECT, G2) \
  51. namespace cv {
  52. namespace gapi {
  53. // NB: cv.gapi.CV_BOOL in python
  54. enum ArgType {
  55. CV_BOOL,
  56. CV_INT,
  57. CV_INT64,
  58. CV_DOUBLE,
  59. CV_FLOAT,
  60. CV_STRING,
  61. CV_POINT,
  62. CV_POINT2F,
  63. CV_SIZE,
  64. CV_RECT,
  65. CV_SCALAR,
  66. CV_MAT,
  67. CV_GMAT,
  68. CV_DRAW_PRIM,
  69. CV_ANY,
  70. };
  71. GAPI_EXPORTS_W inline cv::GInferOutputs infer(const String& name, const cv::GInferInputs& inputs)
  72. {
  73. return infer<Generic>(name, inputs);
  74. }
  75. GAPI_EXPORTS_W inline GInferOutputs infer(const std::string& name,
  76. const cv::GOpaque<cv::Rect>& roi,
  77. const GInferInputs& inputs)
  78. {
  79. return infer<Generic>(name, roi, inputs);
  80. }
  81. GAPI_EXPORTS_W inline GInferListOutputs infer(const std::string& name,
  82. const cv::GArray<cv::Rect>& rois,
  83. const GInferInputs& inputs)
  84. {
  85. return infer<Generic>(name, rois, inputs);
  86. }
  87. GAPI_EXPORTS_W inline GInferListOutputs infer2(const std::string& name,
  88. const cv::GMat in,
  89. const GInferListInputs& inputs)
  90. {
  91. return infer2<Generic>(name, in, inputs);
  92. }
  93. } // namespace gapi
  94. namespace detail {
  95. template <template <typename> class Wrapper, typename T>
  96. struct WrapType { using type = Wrapper<T>; };
  97. template <template <typename> class T, typename... Types>
  98. using MakeVariantType = cv::util::variant<typename WrapType<T, Types>::type...>;
  99. template<typename T> struct ArgTypeTraits;
  100. #define DEFINE_TYPE_TRAITS(T, E) \
  101. template <> \
  102. struct ArgTypeTraits<T> { \
  103. static constexpr const cv::gapi::ArgType type = E; \
  104. }; \
  105. GARRAY_TYPE_LIST_G(DEFINE_TYPE_TRAITS, DEFINE_TYPE_TRAITS)
  106. } // namespace detail
  107. class GAPI_EXPORTS_W_SIMPLE GOpaqueT
  108. {
  109. public:
  110. GOpaqueT() = default;
  111. using Storage = cv::detail::MakeVariantType<cv::GOpaque, GOPAQUE_TYPE_LIST_G(ID_, ID)>;
  112. template<typename T>
  113. GOpaqueT(cv::GOpaque<T> arg) : m_type(cv::detail::ArgTypeTraits<T>::type), m_arg(arg) { };
  114. GAPI_WRAP GOpaqueT(gapi::ArgType type) : m_type(type)
  115. {
  116. #define HC(T, K) case K: \
  117. m_arg = cv::GOpaque<T>(); \
  118. break;
  119. SWITCH(type, GOPAQUE_TYPE_LIST_G, HC)
  120. #undef HC
  121. }
  122. cv::detail::GOpaqueU strip() {
  123. #define HC(T, K) case Storage:: index_of<cv::GOpaque<T>>(): \
  124. return cv::util::get<cv::GOpaque<T>>(m_arg).strip(); \
  125. SWITCH(m_arg.index(), GOPAQUE_TYPE_LIST_G, HC)
  126. #undef HC
  127. GAPI_Assert(false);
  128. }
  129. GAPI_WRAP gapi::ArgType type() { return m_type; }
  130. const Storage& arg() const { return m_arg; }
  131. private:
  132. gapi::ArgType m_type;
  133. Storage m_arg;
  134. };
  135. class GAPI_EXPORTS_W_SIMPLE GArrayT
  136. {
  137. public:
  138. GArrayT() = default;
  139. using Storage = cv::detail::MakeVariantType<cv::GArray, GARRAY_TYPE_LIST_G(ID_, ID)>;
  140. template<typename T>
  141. GArrayT(cv::GArray<T> arg) : m_type(cv::detail::ArgTypeTraits<T>::type), m_arg(arg) { };
  142. GAPI_WRAP GArrayT(gapi::ArgType type) : m_type(type)
  143. {
  144. #define HC(T, K) case K: \
  145. m_arg = cv::GArray<T>(); \
  146. break;
  147. SWITCH(type, GARRAY_TYPE_LIST_G, HC)
  148. #undef HC
  149. }
  150. cv::detail::GArrayU strip() {
  151. #define HC(T, K) case Storage:: index_of<cv::GArray<T>>(): \
  152. return cv::util::get<cv::GArray<T>>(m_arg).strip(); \
  153. SWITCH(m_arg.index(), GARRAY_TYPE_LIST_G, HC)
  154. #undef HC
  155. GAPI_Assert(false);
  156. }
  157. GAPI_WRAP gapi::ArgType type() { return m_type; }
  158. const Storage& arg() const { return m_arg; }
  159. private:
  160. gapi::ArgType m_type;
  161. Storage m_arg;
  162. };
  163. namespace gapi {
  164. namespace wip {
  165. class GAPI_EXPORTS_W_SIMPLE GOutputs
  166. {
  167. public:
  168. GOutputs() = default;
  169. GOutputs(const std::string& id, cv::GKernel::M outMeta, cv::GArgs &&ins);
  170. GAPI_WRAP cv::GMat getGMat();
  171. GAPI_WRAP cv::GScalar getGScalar();
  172. GAPI_WRAP cv::GArrayT getGArray(cv::gapi::ArgType type);
  173. GAPI_WRAP cv::GOpaqueT getGOpaque(cv::gapi::ArgType type);
  174. private:
  175. class Priv;
  176. std::shared_ptr<Priv> m_priv;
  177. };
  178. GOutputs op(const std::string& id, cv::GKernel::M outMeta, cv::GArgs&& args);
  179. template <typename... T>
  180. GOutputs op(const std::string& id, cv::GKernel::M outMeta, T&&... args)
  181. {
  182. return op(id, outMeta, cv::GArgs{cv::GArg(std::forward<T>(args))... });
  183. }
  184. } // namespace wip
  185. } // namespace gapi
  186. } // namespace cv
  187. cv::gapi::wip::GOutputs cv::gapi::wip::op(const std::string& id,
  188. cv::GKernel::M outMeta,
  189. cv::GArgs&& args)
  190. {
  191. cv::gapi::wip::GOutputs outputs{id, outMeta, std::move(args)};
  192. return outputs;
  193. }
  194. class cv::gapi::wip::GOutputs::Priv
  195. {
  196. public:
  197. Priv(const std::string& id, cv::GKernel::M outMeta, cv::GArgs &&ins);
  198. cv::GMat getGMat();
  199. cv::GScalar getGScalar();
  200. cv::GArrayT getGArray(cv::gapi::ArgType);
  201. cv::GOpaqueT getGOpaque(cv::gapi::ArgType);
  202. private:
  203. int output = 0;
  204. std::unique_ptr<cv::GCall> m_call;
  205. };
  206. cv::gapi::wip::GOutputs::Priv::Priv(const std::string& id, cv::GKernel::M outMeta, cv::GArgs &&args)
  207. {
  208. cv::GKinds kinds;
  209. kinds.reserve(args.size());
  210. std::transform(args.begin(), args.end(), std::back_inserter(kinds),
  211. [](const cv::GArg& arg) { return arg.opaque_kind; });
  212. m_call.reset(new cv::GCall{cv::GKernel{id, {}, outMeta, {}, std::move(kinds), {}}});
  213. m_call->setArgs(std::move(args));
  214. }
  215. cv::GMat cv::gapi::wip::GOutputs::Priv::getGMat()
  216. {
  217. m_call->kernel().outShapes.push_back(cv::GShape::GMAT);
  218. // ...so _empty_ constructor is passed here.
  219. m_call->kernel().outCtors.emplace_back(cv::util::monostate{});
  220. return m_call->yield(output++);
  221. }
  222. cv::GScalar cv::gapi::wip::GOutputs::Priv::getGScalar()
  223. {
  224. m_call->kernel().outShapes.push_back(cv::GShape::GSCALAR);
  225. // ...so _empty_ constructor is passed here.
  226. m_call->kernel().outCtors.emplace_back(cv::util::monostate{});
  227. return m_call->yieldScalar(output++);
  228. }
  229. cv::GArrayT cv::gapi::wip::GOutputs::Priv::getGArray(cv::gapi::ArgType type)
  230. {
  231. m_call->kernel().outShapes.push_back(cv::GShape::GARRAY);
  232. #define HC(T, K) \
  233. case K: \
  234. m_call->kernel().outCtors.emplace_back(cv::detail::GObtainCtor<cv::GArray<T>>::get()); \
  235. return cv::GArrayT(m_call->yieldArray<T>(output++)); \
  236. SWITCH(type, GARRAY_TYPE_LIST_G, HC)
  237. #undef HC
  238. }
  239. cv::GOpaqueT cv::gapi::wip::GOutputs::Priv::getGOpaque(cv::gapi::ArgType type)
  240. {
  241. m_call->kernel().outShapes.push_back(cv::GShape::GOPAQUE);
  242. #define HC(T, K) \
  243. case K: \
  244. m_call->kernel().outCtors.emplace_back(cv::detail::GObtainCtor<cv::GOpaque<T>>::get()); \
  245. return cv::GOpaqueT(m_call->yieldOpaque<T>(output++)); \
  246. SWITCH(type, GOPAQUE_TYPE_LIST_G, HC)
  247. #undef HC
  248. }
  249. cv::gapi::wip::GOutputs::GOutputs(const std::string& id,
  250. cv::GKernel::M outMeta,
  251. cv::GArgs &&ins) :
  252. m_priv(new cv::gapi::wip::GOutputs::Priv(id, outMeta, std::move(ins)))
  253. {
  254. }
  255. cv::GMat cv::gapi::wip::GOutputs::getGMat()
  256. {
  257. return m_priv->getGMat();
  258. }
  259. cv::GScalar cv::gapi::wip::GOutputs::getGScalar()
  260. {
  261. return m_priv->getGScalar();
  262. }
  263. cv::GArrayT cv::gapi::wip::GOutputs::getGArray(cv::gapi::ArgType type)
  264. {
  265. return m_priv->getGArray(type);
  266. }
  267. cv::GOpaqueT cv::gapi::wip::GOutputs::getGOpaque(cv::gapi::ArgType type)
  268. {
  269. return m_priv->getGOpaque(type);
  270. }
  271. #endif // OPENCV_GAPI_PYTHON_BRIDGE_HPP