auto_contiguous.cpp 2.21 KB
Newer Older
Paul's avatar
Paul committed
1
2
3
#include <migraphx/auto_contiguous.hpp>
#include <migraphx/program.hpp>
#include <migraphx/instruction.hpp>
4
5
#include <migraphx/make_op.hpp>

Paul's avatar
Paul committed
6
#include <migraphx/iterator_for.hpp>
Paul's avatar
Paul committed
7

Paul's avatar
Paul committed
8
namespace migraphx {
Paul's avatar
Paul committed
9
inline namespace MIGRAPHX_INLINE_NS {
Paul's avatar
Paul committed
10

11
void auto_contiguous::apply(module& p) const
Paul's avatar
Paul committed
12
{
Shucai Xiao's avatar
Shucai Xiao committed
13
    std::string key = "std_shape";
14
    for(auto ins : reverse_iterator_for(p))
Paul's avatar
Paul committed
15
    {
16
17
        auto&& attr = ins->get_operator().attributes();
        if((attr.contains(key) and attr.at(key).to<bool>()))
Paul's avatar
Paul committed
18
        {
Shucai Xiao's avatar
Shucai Xiao committed
19
            auto args     = ins->inputs();
20
21
            auto new_args = args;
            std::transform(args.begin(), args.end(), new_args.begin(), [&](auto in) {
Shucai Xiao's avatar
Shucai Xiao committed
22
                if(in->name() == "contiguous")
Shucai Xiao's avatar
Shucai Xiao committed
23
24
25
                {
                    return in;
                }
Shucai Xiao's avatar
Shucai Xiao committed
26
                return p.insert_instruction(ins, make_op("contiguous"), in);
27
28
29
30
31
32
            });

            if(new_args != args)
            {
                p.replace_instruction(ins, ins->get_operator(), new_args);
            }
Paul's avatar
Paul committed
33
34
        }
    }
Shucai Xiao's avatar
Shucai Xiao committed
35
36
37
38

    auto last = std::prev(p.end());
    for(auto ins : iterator_for(p))
    {
Shucai Xiao's avatar
Shucai Xiao committed
39
        // for last instruction that is NOT a return
Shucai Xiao's avatar
Shucai Xiao committed
40
41
        if(ins->outputs().empty() and ins != last)
            continue;
Shucai Xiao's avatar
Shucai Xiao committed
42
43
44
45
46
47
48
        shape s = ins->get_shape();
        if(not s.standard() and s.elements() != 0)
        {
            auto c = p.insert_instruction(std::next(ins), make_op("contiguous"), ins);
            p.replace_instruction(ins, c);
        }
    }
Shucai Xiao's avatar
Shucai Xiao committed
49
50
51

    // if ops used as output param are alias 0, add a contiguous for the output
    // so return outputs with standard shape
Shucai Xiao's avatar
Shucai Xiao committed
52
    if(last->name() == "@return")
Shucai Xiao's avatar
Shucai Xiao committed
53
54
    {
        auto inputs = last->inputs();
Shucai Xiao's avatar
Shucai Xiao committed
55
        for(auto ins : inputs)
Shucai Xiao's avatar
Shucai Xiao committed
56
        {
Shucai Xiao's avatar
Shucai Xiao committed
57
58
            if(ins->name() == "contiguous")
                continue;
Shucai Xiao's avatar
Shucai Xiao committed
59
60

            auto ins_alias = ins->get_operator().output_alias({});
Shucai Xiao's avatar
Shucai Xiao committed
61
62
            if(ins_alias == 0 and ins->get_shape().element_space() !=
                                      ins->inputs().front()->get_shape().element_space())
Shucai Xiao's avatar
Shucai Xiao committed
63
64
65
66
67
68
            {
                auto cont_ins = p.insert_instruction(last, make_op("contiguous"), ins);
                p.replace_instruction(ins, cont_ins);
            }
        }
    }
Paul's avatar
Paul committed
69
70
}

Paul's avatar
Paul committed
71
} // namespace MIGRAPHX_INLINE_NS
Paul's avatar
Paul committed
72
} // namespace migraphx