// Copyright (c) 2018 The Khronos Group Inc. // Copyright (c) 2018 Valve Corporation // Copyright (c) 2018 LunarG Inc. // // 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. #ifndef SOURCE_OPT_DEAD_INSERT_ELIM_PASS_H_ #define SOURCE_OPT_DEAD_INSERT_ELIM_PASS_H_ #include #include #include #include #include #include #include "source/opt/basic_block.h" #include "source/opt/def_use_manager.h" #include "source/opt/ir_context.h" #include "source/opt/mem_pass.h" #include "source/opt/module.h" namespace spvtools { namespace opt { // See optimizer.hpp for documentation. class DeadInsertElimPass : public MemPass { public: DeadInsertElimPass() = default; const char* name() const override { return "eliminate-dead-inserts"; } Status Process() override; IRContext::Analysis GetPreservedAnalyses() override { return IRContext::kAnalysisDefUse | IRContext::kAnalysisInstrToBlockMapping | IRContext::kAnalysisDecorations | IRContext::kAnalysisCombinators | IRContext::kAnalysisCFG | IRContext::kAnalysisDominatorAnalysis | IRContext::kAnalysisNameMap | IRContext::kAnalysisConstants | IRContext::kAnalysisTypes; } private: // Return the number of subcomponents in the composite type |typeId|. // Return 0 if not a composite type or number of components is not a // 32-bit constant. uint32_t NumComponents(Instruction* typeInst); // Mark all inserts in instruction chain ending at |insertChain| with // indices that intersect with extract indices |extIndices| starting with // index at |extOffset|. Chains are composed solely of Inserts and Phis. // Mark all inserts in chain if |extIndices| is nullptr. void MarkInsertChain(Instruction* insertChain, std::vector* extIndices, uint32_t extOffset, std::unordered_set* visited_phis); // Perform EliminateDeadInsertsOnePass(|func|) until no modification is // made. Return true if modified. bool EliminateDeadInserts(Function* func); // DCE all dead struct, matrix and vector inserts in |func|. An insert is // dead if the value it inserts is never used. Replace any reference to the // insert with its original composite. Return true if modified. Dead inserts // in dependence cycles are not currently eliminated. Dead inserts into // arrays are not currently eliminated. bool EliminateDeadInsertsOnePass(Function* func); // Return true if all extensions in this module are allowed by this pass. bool AllExtensionsSupported() const; // Live inserts std::unordered_set liveInserts_; // Visited phis as insert chain is traversed; used to avoid infinite loop std::unordered_map visitedPhis_; }; } // namespace opt } // namespace spvtools #endif // SOURCE_OPT_DEAD_INSERT_ELIM_PASS_H_