|  | // Copyright (c) 2015-2016 The Khronos Group 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_VAL_BASIC_BLOCK_H_ | 
|  | #define SOURCE_VAL_BASIC_BLOCK_H_ | 
|  |  | 
|  | #include <bitset> | 
|  | #include <cstdint> | 
|  | #include <functional> | 
|  | #include <memory> | 
|  | #include <vector> | 
|  |  | 
|  | #include "source/latest_version_spirv_header.h" | 
|  |  | 
|  | namespace spvtools { | 
|  | namespace val { | 
|  |  | 
|  | enum BlockType : uint32_t { | 
|  | kBlockTypeUndefined, | 
|  | kBlockTypeSelection, | 
|  | kBlockTypeLoop, | 
|  | kBlockTypeMerge, | 
|  | kBlockTypeBreak, | 
|  | kBlockTypeContinue, | 
|  | kBlockTypeReturn, | 
|  | kBlockTypeCOUNT  ///< Total number of block types. (must be the last element) | 
|  | }; | 
|  |  | 
|  | class Instruction; | 
|  |  | 
|  | // This class represents a basic block in a SPIR-V module | 
|  | class BasicBlock { | 
|  | public: | 
|  | /// Constructor for a BasicBlock | 
|  | /// | 
|  | /// @param[in] id The ID of the basic block | 
|  | explicit BasicBlock(uint32_t id); | 
|  |  | 
|  | /// Returns the id of the BasicBlock | 
|  | uint32_t id() const { return id_; } | 
|  |  | 
|  | /// Returns the predecessors of the BasicBlock | 
|  | const std::vector<BasicBlock*>* predecessors() const { | 
|  | return &predecessors_; | 
|  | } | 
|  |  | 
|  | /// Returns the predecessors of the BasicBlock | 
|  | std::vector<BasicBlock*>* predecessors() { return &predecessors_; } | 
|  |  | 
|  | /// Returns the successors of the BasicBlock | 
|  | const std::vector<BasicBlock*>* successors() const { return &successors_; } | 
|  |  | 
|  | /// Returns the successors of the BasicBlock | 
|  | std::vector<BasicBlock*>* successors() { return &successors_; } | 
|  |  | 
|  | /// Returns the structural successors of the BasicBlock | 
|  | std::vector<BasicBlock*>* structural_predecessors() { | 
|  | return &structural_predecessors_; | 
|  | } | 
|  |  | 
|  | /// Returns the structural predecessors of the BasicBlock | 
|  | const std::vector<BasicBlock*>* structural_predecessors() const { | 
|  | return &structural_predecessors_; | 
|  | } | 
|  |  | 
|  | /// Returns the structural successors of the BasicBlock | 
|  | std::vector<BasicBlock*>* structural_successors() { | 
|  | return &structural_successors_; | 
|  | } | 
|  |  | 
|  | /// Returns the structural predecessors of the BasicBlock | 
|  | const std::vector<BasicBlock*>* structural_successors() const { | 
|  | return &structural_successors_; | 
|  | } | 
|  |  | 
|  | /// Returns true if the block is reachable in the CFG. | 
|  | bool reachable() const { return reachable_; } | 
|  |  | 
|  | /// Returns true if the block is structurally reachable in the CFG. | 
|  | bool structurally_reachable() const { return structurally_reachable_; } | 
|  |  | 
|  | /// Returns true if BasicBlock is of the given type | 
|  | bool is_type(BlockType type) const { | 
|  | if (type == kBlockTypeUndefined) return type_.none(); | 
|  | return type_.test(type); | 
|  | } | 
|  |  | 
|  | /// Sets the reachability of the basic block in the CFG | 
|  | void set_reachable(bool reachability) { reachable_ = reachability; } | 
|  |  | 
|  | /// Sets the structural reachability of the basic block in the CFG | 
|  | void set_structurally_reachable(bool reachability) { | 
|  | structurally_reachable_ = reachability; | 
|  | } | 
|  |  | 
|  | /// Sets the type of the BasicBlock | 
|  | void set_type(BlockType type) { | 
|  | if (type == kBlockTypeUndefined) | 
|  | type_.reset(); | 
|  | else | 
|  | type_.set(type); | 
|  | } | 
|  |  | 
|  | /// Sets the immediate dominator of this basic block | 
|  | /// | 
|  | /// @param[in] dom_block The dominator block | 
|  | void SetImmediateDominator(BasicBlock* dom_block); | 
|  |  | 
|  | /// Sets the immediate dominator of this basic block | 
|  | /// | 
|  | /// @param[in] dom_block The dominator block | 
|  | void SetImmediateStructuralDominator(BasicBlock* dom_block); | 
|  |  | 
|  | /// Sets the immediate post dominator of this basic block | 
|  | /// | 
|  | /// @param[in] pdom_block The post dominator block | 
|  | void SetImmediateStructuralPostDominator(BasicBlock* pdom_block); | 
|  |  | 
|  | /// Returns the immediate dominator of this basic block | 
|  | BasicBlock* immediate_dominator(); | 
|  |  | 
|  | /// Returns the immediate dominator of this basic block | 
|  | const BasicBlock* immediate_dominator() const; | 
|  |  | 
|  | /// Returns the immediate dominator of this basic block | 
|  | BasicBlock* immediate_structural_dominator(); | 
|  |  | 
|  | /// Returns the immediate dominator of this basic block | 
|  | const BasicBlock* immediate_structural_dominator() const; | 
|  |  | 
|  | /// Returns the immediate post dominator of this basic block | 
|  | BasicBlock* immediate_structural_post_dominator(); | 
|  |  | 
|  | /// Returns the immediate post dominator of this basic block | 
|  | const BasicBlock* immediate_structural_post_dominator() const; | 
|  |  | 
|  | /// Returns the label instruction for the block, or nullptr if not set. | 
|  | const Instruction* label() const { return label_; } | 
|  |  | 
|  | //// Registers the label instruction for the block. | 
|  | void set_label(const Instruction* t) { label_ = t; } | 
|  |  | 
|  | /// Registers the terminator instruction for the block. | 
|  | void set_terminator(const Instruction* t) { terminator_ = t; } | 
|  |  | 
|  | /// Returns the terminator instruction for the block. | 
|  | const Instruction* terminator() const { return terminator_; } | 
|  |  | 
|  | /// Adds @p next BasicBlocks as successors of this BasicBlock | 
|  | void RegisterSuccessors( | 
|  | const std::vector<BasicBlock*>& next = std::vector<BasicBlock*>()); | 
|  |  | 
|  | /// Returns true if the id of the BasicBlock matches | 
|  | bool operator==(const BasicBlock& other) const { return other.id_ == id_; } | 
|  |  | 
|  | /// Returns true if the id of the BasicBlock matches | 
|  | bool operator==(const uint32_t& other_id) const { return other_id == id_; } | 
|  |  | 
|  | /// Returns true if this block dominates the other block. | 
|  | /// Assumes dominators have been computed. | 
|  | bool dominates(const BasicBlock& other) const; | 
|  |  | 
|  | /// Returns true if this block structurally dominates the other block. | 
|  | /// Assumes structural dominators have been computed. | 
|  | bool structurally_dominates(const BasicBlock& other) const; | 
|  |  | 
|  | /// Returns true if this block structurally postdominates the other block. | 
|  | /// Assumes structural dominators have been computed. | 
|  | bool structurally_postdominates(const BasicBlock& other) const; | 
|  |  | 
|  | void RegisterStructuralSuccessor(BasicBlock* block) { | 
|  | block->structural_predecessors_.push_back(this); | 
|  | structural_successors_.push_back(block); | 
|  | } | 
|  |  | 
|  | /// @brief A BasicBlock dominator iterator class | 
|  | /// | 
|  | /// This iterator will iterate over the (post)dominators of the block | 
|  | class DominatorIterator { | 
|  | public: | 
|  | using iterator_category = std::forward_iterator_tag; | 
|  | using value_type = BasicBlock*; | 
|  | using pointer = value_type*; | 
|  | using reference = value_type&; | 
|  | using difference_type = std::ptrdiff_t; | 
|  |  | 
|  | /// @brief Constructs the end of dominator iterator | 
|  | /// | 
|  | /// This will create an iterator which will represent the element | 
|  | /// before the root node of the dominator tree | 
|  | DominatorIterator(); | 
|  |  | 
|  | /// @brief Constructs an iterator for the given block which points to | 
|  | ///        @p block | 
|  | /// | 
|  | /// @param block          The block which is referenced by the iterator | 
|  | /// @param dominator_func This function will be called to get the immediate | 
|  | ///                       (post)dominator of the current block | 
|  | DominatorIterator( | 
|  | const BasicBlock* block, | 
|  | std::function<const BasicBlock*(const BasicBlock*)> dominator_func); | 
|  |  | 
|  | /// @brief Advances the iterator | 
|  | DominatorIterator& operator++(); | 
|  |  | 
|  | /// @brief Returns the current element | 
|  | const BasicBlock*& operator*(); | 
|  |  | 
|  | friend bool operator==(const DominatorIterator& lhs, | 
|  | const DominatorIterator& rhs); | 
|  |  | 
|  | private: | 
|  | const BasicBlock* current_; | 
|  | std::function<const BasicBlock*(const BasicBlock*)> dom_func_; | 
|  | }; | 
|  |  | 
|  | /// Returns a dominator iterator which points to the current block | 
|  | const DominatorIterator dom_begin() const; | 
|  |  | 
|  | /// Returns a dominator iterator which points to the current block | 
|  | DominatorIterator dom_begin(); | 
|  |  | 
|  | /// Returns a dominator iterator which points to one element past the first | 
|  | /// block | 
|  | const DominatorIterator dom_end() const; | 
|  |  | 
|  | /// Returns a dominator iterator which points to one element past the first | 
|  | /// block | 
|  | DominatorIterator dom_end(); | 
|  |  | 
|  | /// Returns a dominator iterator which points to the current block | 
|  | const DominatorIterator structural_dom_begin() const; | 
|  |  | 
|  | /// Returns a dominator iterator which points to the current block | 
|  | DominatorIterator structural_dom_begin(); | 
|  |  | 
|  | /// Returns a dominator iterator which points to one element past the first | 
|  | /// block | 
|  | const DominatorIterator structural_dom_end() const; | 
|  |  | 
|  | /// Returns a dominator iterator which points to one element past the first | 
|  | /// block | 
|  | DominatorIterator structural_dom_end(); | 
|  |  | 
|  | /// Returns a post dominator iterator which points to the current block | 
|  | const DominatorIterator structural_pdom_begin() const; | 
|  | /// Returns a post dominator iterator which points to the current block | 
|  | DominatorIterator structural_pdom_begin(); | 
|  |  | 
|  | /// Returns a post dominator iterator which points to one element past the | 
|  | /// last block | 
|  | const DominatorIterator structural_pdom_end() const; | 
|  |  | 
|  | /// Returns a post dominator iterator which points to one element past the | 
|  | /// last block | 
|  | DominatorIterator structural_pdom_end(); | 
|  |  | 
|  | private: | 
|  | /// Id of the BasicBlock | 
|  | const uint32_t id_; | 
|  |  | 
|  | /// Pointer to the immediate dominator of the BasicBlock | 
|  | BasicBlock* immediate_dominator_; | 
|  |  | 
|  | /// Pointer to the immediate structural dominator of the BasicBlock | 
|  | BasicBlock* immediate_structural_dominator_; | 
|  |  | 
|  | /// Pointer to the immediate structural post dominator of the BasicBlock | 
|  | BasicBlock* immediate_structural_post_dominator_; | 
|  |  | 
|  | /// The set of predecessors of the BasicBlock | 
|  | std::vector<BasicBlock*> predecessors_; | 
|  |  | 
|  | /// The set of successors of the BasicBlock | 
|  | std::vector<BasicBlock*> successors_; | 
|  |  | 
|  | /// The type of the block | 
|  | std::bitset<kBlockTypeCOUNT> type_; | 
|  |  | 
|  | /// True if the block is reachable in the CFG | 
|  | bool reachable_; | 
|  |  | 
|  | /// True if the block is structurally reachable in the CFG | 
|  | bool structurally_reachable_; | 
|  |  | 
|  | /// label of this block, if any. | 
|  | const Instruction* label_; | 
|  |  | 
|  | /// Terminator of this block. | 
|  | const Instruction* terminator_; | 
|  |  | 
|  | std::vector<BasicBlock*> structural_predecessors_; | 
|  | std::vector<BasicBlock*> structural_successors_; | 
|  | }; | 
|  |  | 
|  | /// @brief Returns true if the iterators point to the same element or if both | 
|  | ///        iterators point to the @p dom_end block | 
|  | bool operator==(const BasicBlock::DominatorIterator& lhs, | 
|  | const BasicBlock::DominatorIterator& rhs); | 
|  |  | 
|  | /// @brief Returns true if the iterators point to different elements and they | 
|  | ///        do not both point to the @p dom_end block | 
|  | bool operator!=(const BasicBlock::DominatorIterator& lhs, | 
|  | const BasicBlock::DominatorIterator& rhs); | 
|  |  | 
|  | }  // namespace val | 
|  | }  // namespace spvtools | 
|  |  | 
|  | #endif  // SOURCE_VAL_BASIC_BLOCK_H_ |