// 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_LICM_PASS_H_ #define SOURCE_OPT_LICM_PASS_H_ #include <queue> #include <vector> #include "source/opt/basic_block.h" #include "source/opt/instruction.h" #include "source/opt/loop_descriptor.h" #include "source/opt/pass.h" namespace spvtools { namespace opt { class LICMPass : public Pass { public: LICMPass() {} const char* name() const override { return "loop-invariant-code-motion"; } Status Process() override; private: // Searches the IRContext for functions and processes each, moving invariants // outside loops within the function where possible. // Returns the status depending on whether or not there was a failure or // change. Pass::Status ProcessIRContext(); // Checks the function for loops, calling ProcessLoop on each one found. // Returns the status depending on whether or not there was a failure or // change. Pass::Status ProcessFunction(Function* f); // Checks for invariants in the loop and attempts to move them to the loops // preheader. Works from inner loop to outer when nested loops are found. // Returns the status depending on whether or not there was a failure or // change. Pass::Status ProcessLoop(Loop* loop, Function* f); // Analyses each instruction in |bb|, hoisting invariants to |pre_header_bb|. // Each child of |bb| wrt to |dom_tree| is pushed to |loop_bbs| // Returns the status depending on whether or not there was a failure or // change. Pass::Status AnalyseAndHoistFromBB(Loop* loop, Function* f, BasicBlock* bb, std::vector<BasicBlock*>* loop_bbs); // Returns true if |bb| is immediately contained in |loop| bool IsImmediatelyContainedInLoop(Loop* loop, Function* f, BasicBlock* bb); // Move the instruction to the preheader of |loop|. // This method will update the instruction to block mapping for the context bool HoistInstruction(Loop* loop, Instruction* inst); }; } // namespace opt } // namespace spvtools #endif // SOURCE_OPT_LICM_PASS_H_