| // Copyright (c) 2018 Google LLC. |
| // |
| // 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_VECTOR_DCE_H_ |
| #define SOURCE_OPT_VECTOR_DCE_H_ |
| |
| #include <unordered_map> |
| #include <vector> |
| |
| #include "source/opt/mem_pass.h" |
| #include "source/util/bit_vector.h" |
| |
| namespace spvtools { |
| namespace opt { |
| |
| class VectorDCE : public MemPass { |
| private: |
| using LiveComponentMap = std::unordered_map<uint32_t, utils::BitVector>; |
| |
| // According to the SPEC the maximum size for a vector is 16. See the data |
| // rules in the universal validation rules (section 2.16.1). |
| enum { kMaxVectorSize = 16 }; |
| |
| struct WorkListItem { |
| WorkListItem() : instruction(nullptr), components(kMaxVectorSize) {} |
| |
| Instruction* instruction; |
| utils::BitVector components; |
| }; |
| |
| public: |
| VectorDCE() : all_components_live_(kMaxVectorSize) { |
| for (uint32_t i = 0; i < kMaxVectorSize; i++) { |
| all_components_live_.Set(i); |
| } |
| } |
| |
| const char* name() const override { return "vector-dce"; } |
| Status Process() override; |
| |
| IRContext::Analysis GetPreservedAnalyses() override { |
| return IRContext::kAnalysisDefUse | IRContext::kAnalysisCFG | |
| IRContext::kAnalysisInstrToBlockMapping | |
| IRContext::kAnalysisLoopAnalysis | IRContext::kAnalysisDecorations | |
| IRContext::kAnalysisDominatorAnalysis | IRContext::kAnalysisNameMap | |
| IRContext::kAnalysisConstants | IRContext::kAnalysisTypes; |
| } |
| |
| private: |
| // Runs the vector dce pass on |function|. Returns true if |function| was |
| // modified. |
| bool VectorDCEFunction(Function* function); |
| |
| // Identifies the live components of the vectors that are results of |
| // instructions in |function|. The results are stored in |live_components|. |
| void FindLiveComponents(Function* function, |
| LiveComponentMap* live_components); |
| |
| // Rewrites instructions in |function| that are dead or partially dead. If an |
| // instruction does not have an entry in |live_components|, then it is not |
| // changed. Returns true if |function| was modified. |
| bool RewriteInstructions(Function* function, |
| const LiveComponentMap& live_components); |
| |
| // Rewrites the OpCompositeInsert instruction |current_inst| to avoid |
| // unnecessary computes given that the only components of the result that are |
| // live are |live_components|. |
| // |
| // If the value being inserted is not live, then the result of |current_inst| |
| // is replaced by the composite input to |current_inst|. |
| // |
| // If the composite input to |current_inst| is not live, then it is replaced |
| // by and OpUndef in |current_inst|. |
| bool RewriteInsertInstruction(Instruction* current_inst, |
| const utils::BitVector& live_components); |
| |
| // Returns true if the result of |inst| is a vector or a scalar. |
| bool HasVectorOrScalarResult(const Instruction* inst) const; |
| |
| // Returns true if the result of |inst| is a scalar. |
| bool HasVectorResult(const Instruction* inst) const; |
| |
| // Returns true if the result of |inst| is a vector. |
| bool HasScalarResult(const Instruction* inst) const; |
| |
| // Adds |work_item| to |work_list| if it is not already live according to |
| // |live_components|. |live_components| is updated to indicate that |
| // |work_item| is now live. |
| void AddItemToWorkListIfNeeded(WorkListItem work_item, |
| LiveComponentMap* live_components, |
| std::vector<WorkListItem>* work_list); |
| |
| // Marks the components |live_elements| of the uses in |current_inst| as live |
| // according to |live_components|. If they were not live before, then they are |
| // added to |work_list|. |
| void MarkUsesAsLive(Instruction* current_inst, |
| const utils::BitVector& live_elements, |
| LiveComponentMap* live_components, |
| std::vector<WorkListItem>* work_list); |
| |
| // Marks the uses in the OpVectorShuffle instruction in |current_item| as live |
| // based on the live components in |current_item|. If anything becomes live |
| // they are added to |work_list| and |live_components| is updated |
| // accordingly. |
| void MarkVectorShuffleUsesAsLive(const WorkListItem& current_item, |
| VectorDCE::LiveComponentMap* live_components, |
| std::vector<WorkListItem>* work_list); |
| |
| // Marks the uses in the OpCompositeInsert instruction in |current_item| as |
| // live based on the live components in |current_item|. If anything becomes |
| // live they are added to |work_list| and |live_components| is updated |
| // accordingly. |
| void MarkInsertUsesAsLive(const WorkListItem& current_item, |
| LiveComponentMap* live_components, |
| std::vector<WorkListItem>* work_list); |
| |
| // Marks the uses in the OpCompositeExtract instruction |current_inst| as |
| // live. If anything becomes live they are added to |work_list| and |
| // |live_components| is updated accordingly. |
| void MarkExtractUseAsLive(const Instruction* current_inst, |
| const utils::BitVector& live_elements, |
| LiveComponentMap* live_components, |
| std::vector<WorkListItem>* work_list); |
| |
| // Marks the uses in the OpCompositeConstruct instruction |current_inst| as |
| // live. If anything becomes live they are added to |work_list| and |
| // |live_components| is updated accordingly. |
| void MarkCompositeContructUsesAsLive(WorkListItem work_item, |
| LiveComponentMap* live_components, |
| std::vector<WorkListItem>* work_list); |
| |
| // A BitVector that can always be used to say that all components of a vector |
| // are live. |
| utils::BitVector all_components_live_; |
| }; |
| |
| } // namespace opt |
| } // namespace spvtools |
| |
| #endif // SOURCE_OPT_VECTOR_DCE_H_ |