/* * Copyright 2016-2019 Arm Limited * * 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 SPIRV_CROSS_CFG_HPP #define SPIRV_CROSS_CFG_HPP #include "spirv_common.hpp" #include namespace spirv_cross { class Compiler; class CFG { public: CFG(Compiler &compiler, const SPIRFunction &function); Compiler &get_compiler() { return compiler; } const Compiler &get_compiler() const { return compiler; } const SPIRFunction &get_function() const { return func; } uint32_t get_immediate_dominator(uint32_t block) const { auto itr = immediate_dominators.find(block); if (itr != std::end(immediate_dominators)) return itr->second; else return 0; } uint32_t get_visit_order(uint32_t block) const { auto itr = visit_order.find(block); assert(itr != std::end(visit_order)); int v = itr->second.get(); assert(v > 0); return uint32_t(v); } uint32_t find_common_dominator(uint32_t a, uint32_t b) const; const std::vector &get_preceding_edges(uint32_t block) const { auto itr = preceding_edges.find(block); if (itr != std::end(preceding_edges)) return itr->second; else return empty_vector; } const std::vector &get_succeeding_edges(uint32_t block) const { auto itr = succeeding_edges.find(block); if (itr != std::end(succeeding_edges)) return itr->second; else return empty_vector; } template void walk_from(std::unordered_set &seen_blocks, uint32_t block, const Op &op) const { if (seen_blocks.count(block)) return; seen_blocks.insert(block); op(block); for (auto b : get_succeeding_edges(block)) walk_from(seen_blocks, b, op); } private: struct VisitOrder { int &get() { return v; } const int &get() const { return v; } int v = -1; }; Compiler &compiler; const SPIRFunction &func; std::unordered_map> preceding_edges; std::unordered_map> succeeding_edges; std::unordered_map immediate_dominators; std::unordered_map visit_order; std::vector post_order; std::vector empty_vector; void add_branch(uint32_t from, uint32_t to); void build_post_order_visit_order(); void build_immediate_dominators(); bool post_order_visit(uint32_t block); uint32_t visit_count = 0; bool is_back_edge(uint32_t to) const; }; class DominatorBuilder { public: DominatorBuilder(const CFG &cfg); void add_block(uint32_t block); uint32_t get_dominator() const { return dominator; } void lift_continue_block_dominator(); private: const CFG &cfg; uint32_t dominator = 0; }; } // namespace spirv_cross #endif