auto_contiguous.cpp 2.07 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
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66

    // if ops used as output param are alias 0, add a contiguous for the output
    // so return outputs with standard shape
    if (last->name() == "@return")
    {
        auto inputs = last->inputs();
        for (auto ins : inputs)
        {
            if (ins->name() == "contiguous") continue;

            auto ins_alias = ins->get_operator().output_alias({});
            if (ins_alias == 0)
            {
                auto cont_ins = p.insert_instruction(last, make_op("contiguous"), ins);
                p.replace_instruction(ins, cont_ins);
            }
        }
    }
Paul's avatar
Paul committed
67
68
}

Paul's avatar
Paul committed
69
} // namespace MIGRAPHX_INLINE_NS
Paul's avatar
Paul committed
70
} // namespace migraphx