| // 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_ |