hc
2023-05-26 a23f51ed7a39e452c1037343a84d7db1ca2c5bd7
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
// 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_GPLAIDMLKERNEL_HPP
#define OPENCV_GAPI_GPLAIDMLKERNEL_HPP
 
#include <opencv2/gapi/gkernel.hpp>
#include <opencv2/gapi/garg.hpp>
 
namespace plaidml
{
namespace edsl
{
    class Tensor;
} // namespace edsl
} // namespace plaidml
 
namespace cv
{
namespace gapi
{
namespace plaidml
{
 
GAPI_EXPORTS cv::gapi::GBackend backend();
 
} // namespace plaidml
} // namespace gapi
 
struct GPlaidMLContext
{
    // Generic accessor API
    template<typename T>
    const T& inArg(int input) { return m_args.at(input).get<T>(); }
 
    // Syntax sugar
    const plaidml::edsl::Tensor& inTensor(int input)
    {
        return inArg<plaidml::edsl::Tensor>(input);
    }
 
    plaidml::edsl::Tensor& outTensor(int output)
    {
        return *(m_results.at(output).get<plaidml::edsl::Tensor*>());
    }
 
    std::vector<GArg> m_args;
    std::unordered_map<std::size_t, GArg> m_results;
};
 
class GAPI_EXPORTS GPlaidMLKernel
{
public:
    using F = std::function<void(GPlaidMLContext &)>;
 
    GPlaidMLKernel() = default;
    explicit GPlaidMLKernel(const F& f) : m_f(f) {};
 
    void apply(GPlaidMLContext &ctx) const
    {
        GAPI_Assert(m_f);
        m_f(ctx);
    }
 
protected:
    F m_f;
};
 
 
namespace detail
{
 
template<class T> struct plaidml_get_in;
template<> struct plaidml_get_in<cv::GMat>
{
    static const plaidml::edsl::Tensor& get(GPlaidMLContext& ctx, int idx)
    {
        return ctx.inTensor(idx);
    }
};
 
template<class T> struct plaidml_get_in
{
    static T get(GPlaidMLContext &ctx, int idx) { return ctx.inArg<T>(idx); }
};
 
template<class T> struct plaidml_get_out;
template<> struct plaidml_get_out<cv::GMat>
{
    static plaidml::edsl::Tensor& get(GPlaidMLContext& ctx, int idx)
    {
        return ctx.outTensor(idx);
    }
};
 
template<typename, typename, typename>
struct PlaidMLCallHelper;
 
template<typename Impl, typename... Ins, typename... Outs>
struct PlaidMLCallHelper<Impl, std::tuple<Ins...>, std::tuple<Outs...> >
{
    template<int... IIs, int... OIs>
    static void call_impl(GPlaidMLContext &ctx, detail::Seq<IIs...>, detail::Seq<OIs...>)
    {
        Impl::run(plaidml_get_in<Ins>::get(ctx, IIs)..., plaidml_get_out<Outs>::get(ctx, OIs)...);
    }
 
    static void call(GPlaidMLContext& ctx)
    {
        call_impl(ctx,
                  typename detail::MkSeq<sizeof...(Ins)>::type(),
                  typename detail::MkSeq<sizeof...(Outs)>::type());
    }
};
 
} // namespace detail
 
template<class Impl, class K>
class GPlaidMLKernelImpl: public cv::detail::PlaidMLCallHelper<Impl, typename K::InArgs, typename K::OutArgs>,
                          public cv::detail::KernelTag
{
    using P = detail::PlaidMLCallHelper<Impl, typename K::InArgs, typename K::OutArgs>;
 
public:
    using API = K;
 
    static cv::gapi::GBackend backend()  { return cv::gapi::plaidml::backend(); }
    static cv::GPlaidMLKernel kernel()   { return GPlaidMLKernel(&P::call);     }
};
 
#define GAPI_PLAIDML_KERNEL(Name, API) struct Name: public cv::GPlaidMLKernelImpl<Name, API>
 
} // namespace cv
 
#endif // OPENCV_GAPI_GPLAIDMLKERNEL_HPP