104 lines
3.4 KiB
C++
104 lines
3.4 KiB
C++
|
// 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) 2019 Intel Corporation
|
||
|
|
||
|
#ifndef OPENCV_GAPI_GTRANSFORM_HPP
|
||
|
#define OPENCV_GAPI_GTRANSFORM_HPP
|
||
|
|
||
|
#include <functional>
|
||
|
#include <type_traits>
|
||
|
#include <utility>
|
||
|
|
||
|
#include <opencv2/gapi/gcommon.hpp>
|
||
|
#include <opencv2/gapi/util/util.hpp>
|
||
|
#include <opencv2/gapi/garg.hpp>
|
||
|
#include <opencv2/gapi/gtype_traits.hpp>
|
||
|
#include <opencv2/gapi/util/compiler_hints.hpp>
|
||
|
#include <opencv2/gapi/gcomputation.hpp>
|
||
|
|
||
|
namespace cv
|
||
|
{
|
||
|
|
||
|
struct GAPI_EXPORTS GTransform
|
||
|
{
|
||
|
// FIXME: consider another simplified
|
||
|
// class instead of GComputation
|
||
|
using F = std::function<GComputation()>;
|
||
|
|
||
|
std::string description;
|
||
|
F pattern;
|
||
|
F substitute;
|
||
|
|
||
|
GTransform(const std::string& d, const F &p, const F &s) : description(d), pattern(p), substitute(s) {}
|
||
|
};
|
||
|
|
||
|
namespace detail
|
||
|
{
|
||
|
|
||
|
template <typename, typename, typename>
|
||
|
struct TransHelper;
|
||
|
|
||
|
template <typename K, typename... Ins, typename Out>
|
||
|
struct TransHelper<K, std::tuple<Ins...>, Out>
|
||
|
{
|
||
|
template <typename Callable, int... IIs, int... OIs>
|
||
|
static GComputation invoke(Callable f, Seq<IIs...>, Seq<OIs...>)
|
||
|
{
|
||
|
const std::tuple<Ins...> ins;
|
||
|
const auto r = tuple_wrap_helper<Out>::get(f(std::get<IIs>(ins)...));
|
||
|
return GComputation(cv::GIn(std::get<IIs>(ins)...),
|
||
|
cv::GOut(std::get<OIs>(r)...));
|
||
|
}
|
||
|
|
||
|
static GComputation get_pattern()
|
||
|
{
|
||
|
return invoke(K::pattern, typename MkSeq<sizeof...(Ins)>::type(),
|
||
|
typename MkSeq<std::tuple_size<typename tuple_wrap_helper<Out>::type>::value>::type());
|
||
|
}
|
||
|
static GComputation get_substitute()
|
||
|
{
|
||
|
return invoke(K::substitute, typename MkSeq<sizeof...(Ins)>::type(),
|
||
|
typename MkSeq<std::tuple_size<typename tuple_wrap_helper<Out>::type>::value>::type());
|
||
|
}
|
||
|
};
|
||
|
} // namespace detail
|
||
|
|
||
|
template <typename, typename>
|
||
|
class GTransformImpl;
|
||
|
|
||
|
template <typename K, typename R, typename... Args>
|
||
|
class GTransformImpl<K, std::function<R(Args...)>> : public cv::detail::TransHelper<K, std::tuple<Args...>, R>,
|
||
|
public cv::detail::TransformTag
|
||
|
{
|
||
|
public:
|
||
|
// FIXME: currently there is no check that transformations' signatures are unique
|
||
|
// and won't be any intersection in graph compilation stage
|
||
|
using API = K;
|
||
|
|
||
|
static GTransform transformation()
|
||
|
{
|
||
|
return GTransform(K::descr(), &K::get_pattern, &K::get_substitute);
|
||
|
}
|
||
|
};
|
||
|
} // namespace cv
|
||
|
|
||
|
#define G_DESCR_HELPER_CLASS(Class) Class##DescrHelper
|
||
|
|
||
|
#define G_DESCR_HELPER_BODY(Class, Descr) \
|
||
|
namespace detail \
|
||
|
{ \
|
||
|
struct G_DESCR_HELPER_CLASS(Class) \
|
||
|
{ \
|
||
|
static constexpr const char *descr() { return Descr; }; \
|
||
|
}; \
|
||
|
}
|
||
|
|
||
|
#define GAPI_TRANSFORM(Class, API, Descr) \
|
||
|
G_DESCR_HELPER_BODY(Class, Descr) \
|
||
|
struct Class final : public cv::GTransformImpl<Class, std::function API>, \
|
||
|
public detail::G_DESCR_HELPER_CLASS(Class)
|
||
|
|
||
|
#endif // OPENCV_GAPI_GTRANSFORM_HPP
|