20 const Node*
inest()
const {
return inest_; }
24 auto idom()
const {
return calc_dominance()->idom_; }
25 bool is_root()
const {
return inest_ ==
nullptr; }
31 uint32_t
level()
const {
return level_; }
32 uint32_t
loop_depth()
const {
return sccs().loop_depth_; }
41 auto mut2node()
const {
return mut2node_ | std::views::transform([](
auto p) {
return std::pair{p.first,
const_cast<const Node*
>(p.second)}; }); }
42 auto muts()
const {
return mut2node_ | std::views::keys; }
43 auto nodes()
const {
return mut2node_ | std::views::transform([](
auto p) {
return const_cast<const Node*
>(p.second); }); }
45 size_t num()
const {
return mut2node_.size(); }
56 auto begin()
const {
return mut2node_.cbegin(); }
57 auto end()
const {
return mut2node_.cend(); }
61 const auto&
mut2node() {
return mut2node_; }
62 auto nodes() {
return mut2node_ | std::views::values; }
63 auto muts() {
return mut2node_ | std::views::keys; }
64 auto begin() {
return mut2node_.begin(); }
65 auto end() {
return mut2node_.end(); }
77 template<
bool Forward>
82 return nodes_ | std::views::transform([](Node* n) {
return const_cast<const Node*
>(n); });
89 bool contains(
const Node* n)
const {
return nodes_.contains(n); }
94 auto begin()
const {
return nodes_.cbegin(); }
95 auto end()
const {
return nodes_.cend(); }
99 const auto&
nodes() {
return nodes_; }
100 auto begin() {
return nodes_.begin(); }
101 auto end() {
return nodes_.end(); }
103 absl::flat_hash_set<Node*> nodes_;
112 template<
bool Forward = true>
114 nest().calc_sibl_deps();
115 if constexpr (Forward)
118 return sibl_rev_deps_;
121 template<
bool Forward = true>
128 using SCC = absl::flat_hash_set<const Node*>;
134 const auto&
SCCs() {
return sccs().SCCs_; }
135 const auto&
topo()
const {
return sccs().topo_; }
150 const Node& sccs()
const {
return nest().calc_SCCs(), *
this; }
152 void link(Node* other) { this->sibl_deps_.nodes_.emplace(other), other->sibl_rev_deps_.nodes_.emplace(
this); }
153 void dot(Tab, std::ostream&)
const;
156 using Stack = std::stack<Node*>;
158 uint32_t tarjan(uint32_t, Node*, Stack&);
161 const Node* calc_dominance()
const;
167 uint32_t loop_depth_ : 31 = 0;
168 bool recursive_ : 1 =
false;
172 std::deque<std::unique_ptr<SCC>> topo_;
173 absl::flat_hash_map<const Node*, const SCC*> SCCs_;
174 mutable const Node* idom_ =
nullptr;
177 mutable std::optional<size_t> postorder_number_ = std::nullopt;
180 static constexpr uint32_t Unvisited = uint32_t(-1);
181 uint32_t idx_ = Unvisited;
182 uint32_t low_ : 31 = 0;
183 bool on_stack_ : 1 =
false;
184 Node* curr_child =
nullptr;
205 bool is_recursive()
const {
return calc_SCCs().root()->is_recursive(); }
212 auto muts()
const {
return mut2node_ | std::views::keys; }
213 auto nodes()
const {
return mut2node_ | std::views::transform([](
const auto& p) {
return (
const Node*)p.second.get(); }); }
220 auto begin()
const {
return mut2node_.cbegin(); }
221 auto end()
const {
return mut2node_.cend(); }
224 template<
bool bootstrapping = false>
230 void dot(std::ostream& os)
const;
231 void dot(
const char* file =
nullptr)
const;
232 void dot(std::string s)
const {
dot(s.c_str()); }
236 auto begin() {
return mut2node_.begin(); }
237 auto end() {
return mut2node_.end(); }
240 Node* make_node(Def*,
Node* inest =
nullptr);
241 void calc_sibl_deps(
Node*)
const;
242 void calc_SCCs(
Node*)
const;
243 void assign_postorder_numbers()
const;
246 if (
auto i = mut2node_.find(mut); i != mut2node_.end())
return i->second.get();
250 void calc_sibl_deps()
const {
253 calc_sibl_deps(root_);
257 const Nest& calc_SCCs()
const {
266 absl::flat_hash_map<Def*, std::unique_ptr<Node>> mut2node_;
269 mutable bool siblings_ =
false;
270 mutable bool sccs_ =
false;
std::string unique_name() const
name + "_" + Def::gid
Vars free_vars() const
Compute a global solution by transitively following mutables as well.
const auto & sibl_deps() const
Def * mut() const
The mutable capsulated in this Node or nullptr, if it's a virtual root comprising several Nodes.
bool is_recursive() const
const Children & children() const
auto idom() const
Immediate Dominator for children in connected components.
absl::flat_hash_set< const Node * > SCC
Strongly Connected Component.
const auto & topo() const
Topological sorting of all SCCs.
const Nest & nest() const
bool is_directly_recursive() const
bool is_mutually_recursive() const
uint32_t loop_depth() const
const Node * inest() const
Immediate nester/parent of this Node.
void dot(std::ostream &os) const
bool is_recursive() const
const Node * operator[](Def *mut) const
static const Node * lca(const Node *n, const Node *m)
Least common ancestor of n and m.
Nest & operator=(Nest)=delete
void dot(std::string s) const
const Node * root() const
Vars vars() const
All Vars occurring in this Nest.
Nest(const Nest &)=delete
bool contains(const Def *def) const
bool has_intersection(Set other) const noexcept
Is ?.
The World represents the whole program and manages creation of MimIR nodes (Defs).
auto lookup(const C &container, const K &key)
Yields pointer to element (or the element itself if it is already a pointer), if found and nullptr ot...
Sets< const Var >::Set Vars
GIDMap< Def *, To > MutMap
size_t num() const
Number of children.
bool contains(Def *mut) const
is mut a child?
const Node * operator[](Def *mut) const
bool contains(const Node *n) const