|  | // 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_IF_CONVERSION_H_ | 
|  | #define SOURCE_OPT_IF_CONVERSION_H_ | 
|  |  | 
|  | #include "source/opt/basic_block.h" | 
|  | #include "source/opt/ir_builder.h" | 
|  | #include "source/opt/pass.h" | 
|  | #include "source/opt/types.h" | 
|  |  | 
|  | namespace spvtools { | 
|  | namespace opt { | 
|  |  | 
|  | // See optimizer.hpp for documentation. | 
|  | class IfConversion : public Pass { | 
|  | public: | 
|  | const char* name() const override { return "if-conversion"; } | 
|  | Status Process() override; | 
|  |  | 
|  | IRContext::Analysis GetPreservedAnalyses() override { | 
|  | return IRContext::kAnalysisDefUse | IRContext::kAnalysisDominatorAnalysis | | 
|  | IRContext::kAnalysisInstrToBlockMapping | IRContext::kAnalysisCFG | | 
|  | IRContext::kAnalysisNameMap | IRContext::kAnalysisConstants | | 
|  | IRContext::kAnalysisTypes; | 
|  | } | 
|  |  | 
|  | private: | 
|  | // Returns true if |id| is a valid type for use with OpSelect. OpSelect only | 
|  | // allows scalars, vectors and pointers as valid inputs. | 
|  | bool CheckType(uint32_t id); | 
|  |  | 
|  | // Returns the basic block containing |id|. | 
|  | BasicBlock* GetBlock(uint32_t id); | 
|  |  | 
|  | // Returns the basic block for the |predecessor|'th index predecessor of | 
|  | // |phi|. | 
|  | BasicBlock* GetIncomingBlock(Instruction* phi, uint32_t predecessor); | 
|  |  | 
|  | // Returns the instruction defining the |predecessor|'th index of |phi|. | 
|  | Instruction* GetIncomingValue(Instruction* phi, uint32_t predecessor); | 
|  |  | 
|  | // Returns the id of a OpCompositeConstruct boolean vector. The composite has | 
|  | // the same number of elements as |vec_data_ty| and each member is |cond|. | 
|  | // |where| indicates the location in |block| to insert the composite | 
|  | // construct. If necessary, this function will also construct the necessary | 
|  | // type instructions for the boolean vector. | 
|  | uint32_t SplatCondition(analysis::Vector* vec_data_ty, uint32_t cond, | 
|  | InstructionBuilder* builder); | 
|  |  | 
|  | // Returns true if none of |phi|'s users are in |block|. | 
|  | bool CheckPhiUsers(Instruction* phi, BasicBlock* block); | 
|  |  | 
|  | // Returns |false| if |block| is not appropriate to transform. Only | 
|  | // transforms blocks with two predecessors. Neither incoming block can be | 
|  | // dominated by |block|. Both predecessors must share a common dominator that | 
|  | // is terminated by a conditional branch. | 
|  | bool CheckBlock(BasicBlock* block, DominatorAnalysis* dominators, | 
|  | BasicBlock** common); | 
|  |  | 
|  | // Moves |inst| to |target_block| if it does not already dominate the block. | 
|  | // Any instructions that |inst| depends on are move if necessary.  It is | 
|  | // assumed that |inst| can be hoisted to |target_block| as defined by | 
|  | // |CanHoistInstruction|.  |dominators| is the dominator analysis for the | 
|  | // function that contains |target_block|. | 
|  | void HoistInstruction(Instruction* inst, BasicBlock* target_block, | 
|  | DominatorAnalysis* dominators); | 
|  |  | 
|  | // Returns true if it is legal to move |inst| and the instructions it depends | 
|  | // on to |target_block| if they do not already dominate |target_block|. | 
|  | bool CanHoistInstruction(Instruction* inst, BasicBlock* target_block, | 
|  | DominatorAnalysis* dominators); | 
|  | }; | 
|  |  | 
|  | }  //  namespace opt | 
|  | }  //  namespace spvtools | 
|  |  | 
|  | #endif  //  SOURCE_OPT_IF_CONVERSION_H_ |