#include #include #include #include #include #include #include namespace migraphx { inline namespace MIGRAPHX_INLINE_NS { void dead_code_elimination::apply(program& p) const { p.remove_unused_modules(); } void dead_code_elimination::apply(module& m) const { auto last = std::prev(m.end()); for(auto ins : iterator_for(m)) { // Skip the first instruction, since we always process the previous // instruction if(ins == m.begin()) continue; const auto i = std::prev(ins); // Skip the last instruction if(i == last) break; // Skip instruction with empty shape as output unless its a builtin or undefined or identity if(i->get_shape().elements() == 0 and i->name().front() != '@' and i->name() != "undefined" and i->name() != "identity") continue; assert(std::distance(m.begin(), i) <= std::distance(m.begin(), last)); std::unordered_set visited; fix([&](auto self, auto leaf) { if(not m.has_instruction(leaf)) return; if(leaf->outputs().empty()) { // Dont visit inputs twice if(not visited.insert(leaf).second) return; std::unordered_set args(leaf->inputs().begin(), leaf->inputs().end()); leaf->clear_arguments(); assert(std::distance(m.begin(), leaf) < std::distance(m.begin(), last)); assert(leaf != ins); if(leaf->name() != "@param") m.move_instruction(leaf, m.end()); for(auto arg : args) self(arg); } })(i); } m.remove_instructions(std::next(last), m.end()); } } // namespace MIGRAPHX_INLINE_NS } // namespace migraphx