| // Copyright (c) 2017 Pierre Moreau |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #include "source/opt/remove_duplicates_pass.h" |
| |
| #include <algorithm> |
| #include <cstring> |
| #include <limits> |
| #include <string> |
| #include <unordered_map> |
| #include <unordered_set> |
| #include <vector> |
| |
| #include "source/opcode.h" |
| #include "source/opt/decoration_manager.h" |
| #include "source/opt/ir_context.h" |
| #include "source/opt/reflect.h" |
| |
| namespace spvtools { |
| namespace opt { |
| |
| Pass::Status RemoveDuplicatesPass::Process() { |
| bool modified = RemoveDuplicateCapabilities(); |
| modified |= RemoveDuplicatesExtInstImports(); |
| modified |= RemoveDuplicateTypes(); |
| modified |= RemoveDuplicateDecorations(); |
| |
| return modified ? Status::SuccessWithChange : Status::SuccessWithoutChange; |
| } |
| |
| bool RemoveDuplicatesPass::RemoveDuplicateCapabilities() const { |
| bool modified = false; |
| |
| if (context()->capabilities().empty()) { |
| return modified; |
| } |
| |
| std::unordered_set<uint32_t> capabilities; |
| for (auto* i = &*context()->capability_begin(); i;) { |
| auto res = capabilities.insert(i->GetSingleWordOperand(0u)); |
| |
| if (res.second) { |
| // Never seen before, keep it. |
| i = i->NextNode(); |
| } else { |
| // It's a duplicate, remove it. |
| i = context()->KillInst(i); |
| modified = true; |
| } |
| } |
| |
| return modified; |
| } |
| |
| bool RemoveDuplicatesPass::RemoveDuplicatesExtInstImports() const { |
| bool modified = false; |
| |
| if (context()->ext_inst_imports().empty()) { |
| return modified; |
| } |
| |
| std::unordered_map<std::string, SpvId> ext_inst_imports; |
| for (auto* i = &*context()->ext_inst_import_begin(); i;) { |
| auto res = ext_inst_imports.emplace( |
| reinterpret_cast<const char*>(i->GetInOperand(0u).words.data()), |
| i->result_id()); |
| if (res.second) { |
| // Never seen before, keep it. |
| i = i->NextNode(); |
| } else { |
| // It's a duplicate, remove it. |
| context()->ReplaceAllUsesWith(i->result_id(), res.first->second); |
| i = context()->KillInst(i); |
| modified = true; |
| } |
| } |
| |
| return modified; |
| } |
| |
| bool RemoveDuplicatesPass::RemoveDuplicateTypes() const { |
| bool modified = false; |
| |
| if (context()->types_values().empty()) { |
| return modified; |
| } |
| |
| std::vector<Instruction*> visited_types; |
| std::vector<Instruction*> to_delete; |
| for (auto* i = &*context()->types_values_begin(); i; i = i->NextNode()) { |
| // We only care about types. |
| if (!spvOpcodeGeneratesType((i->opcode())) && |
| i->opcode() != SpvOpTypeForwardPointer) { |
| continue; |
| } |
| |
| // Is the current type equal to one of the types we have aready visited? |
| SpvId id_to_keep = 0u; |
| // TODO(dneto0): Use a trie to avoid quadratic behaviour? Extract the |
| // ResultIdTrie from unify_const_pass.cpp for this. |
| for (auto j : visited_types) { |
| if (AreTypesEqual(*i, *j, context())) { |
| id_to_keep = j->result_id(); |
| break; |
| } |
| } |
| |
| if (id_to_keep == 0u) { |
| // This is a never seen before type, keep it around. |
| visited_types.emplace_back(i); |
| } else { |
| // The same type has already been seen before, remove this one. |
| context()->KillNamesAndDecorates(i->result_id()); |
| context()->ReplaceAllUsesWith(i->result_id(), id_to_keep); |
| modified = true; |
| to_delete.emplace_back(i); |
| } |
| } |
| |
| for (auto i : to_delete) { |
| context()->KillInst(i); |
| } |
| |
| return modified; |
| } |
| |
| // TODO(pierremoreau): Duplicate decoration groups should be removed. For |
| // example, in |
| // OpDecorate %1 Constant |
| // %1 = OpDecorationGroup |
| // OpDecorate %2 Constant |
| // %2 = OpDecorationGroup |
| // OpGroupDecorate %1 %3 |
| // OpGroupDecorate %2 %4 |
| // group %2 could be removed. |
| bool RemoveDuplicatesPass::RemoveDuplicateDecorations() const { |
| bool modified = false; |
| |
| std::vector<const Instruction*> visited_decorations; |
| |
| analysis::DecorationManager decoration_manager(context()->module()); |
| for (auto* i = &*context()->annotation_begin(); i;) { |
| // Is the current decoration equal to one of the decorations we have aready |
| // visited? |
| bool already_visited = false; |
| // TODO(dneto0): Use a trie to avoid quadratic behaviour? Extract the |
| // ResultIdTrie from unify_const_pass.cpp for this. |
| for (const Instruction* j : visited_decorations) { |
| if (decoration_manager.AreDecorationsTheSame(&*i, j, false)) { |
| already_visited = true; |
| break; |
| } |
| } |
| |
| if (!already_visited) { |
| // This is a never seen before decoration, keep it around. |
| visited_decorations.emplace_back(&*i); |
| i = i->NextNode(); |
| } else { |
| // The same decoration has already been seen before, remove this one. |
| modified = true; |
| i = context()->KillInst(i); |
| } |
| } |
| |
| return modified; |
| } |
| |
| bool RemoveDuplicatesPass::AreTypesEqual(const Instruction& inst1, |
| const Instruction& inst2, |
| IRContext* context) { |
| if (inst1.opcode() != inst2.opcode()) return false; |
| if (!IsTypeInst(inst1.opcode())) return false; |
| |
| const analysis::Type* type1 = |
| context->get_type_mgr()->GetType(inst1.result_id()); |
| const analysis::Type* type2 = |
| context->get_type_mgr()->GetType(inst2.result_id()); |
| if (type1 && type2 && *type1 == *type2) return true; |
| |
| return false; |
| } |
| |
| } // namespace opt |
| } // namespace spvtools |