19 const Node*
inest()
const {
return inest_; }
20 bool is_root()
const {
return inest_ ==
nullptr; }
26 uint32_t
level()
const {
return level_; }
27 uint32_t
loop_depth()
const {
return sccs().loop_depth_; }
33 auto muts()
const {
return mut2node_ | std::views::keys; }
37 | std::views::transform([](
const auto& pair) {
return const_cast<const Node*
>(pair.second); });
39 auto nodes() {
return mut2node_ | std::views::values; }
42 return mut2node_ | std::views::transform([](
const auto& pair) {
43 return std::pair{pair.first,
const_cast<const Node*
>(pair.second)};
48 size_t num()
const {
return mut2node_.size(); }
51 if (
auto i = mut2node_.find(
mut); i != mut2node_.end())
return i->second;
59 auto begin() {
return mut2node_.begin(); }
60 auto end() {
return mut2node_.end(); }
61 auto begin()
const {
return mut2node_.cbegin(); }
62 auto end()
const {
return mut2node_.cend(); }
74 template<
bool Forward>
77 absl::flat_hash_set<Node*>&
deps() {
return nodes_; }
79 return nodes_ | std::views::transform([](Node* n) {
return const_cast<const Node*
>(n); });
81 bool contains(
const Node* n)
const {
return nodes_.contains(n); }
84 absl::flat_hash_set<Node*> nodes_;
93 template<
bool Forward = true>
95 nest().calcuate_siblings();
96 if constexpr (Forward)
102 template<
bool Forward = true>
109 using SCC = absl::flat_hash_set<const Node*>;
115 const auto&
SCCs() {
return sccs().SCCs_; }
116 const auto&
topo()
const {
return sccs().topo_; }
131 const Node& sccs()
const {
return nest().calculate_SCCs(), *
this; }
133 void link(Node* other) { this->siblings_.nodes_.emplace(other), other->rev_siblings_.nodes_.emplace(
this); }
134 void dot(Tab, std::ostream&)
const;
137 using Stack = std::stack<Node*>;
139 uint32_t tarjan(uint32_t, Node*, Stack&);
145 uint32_t loop_depth_ : 31 = 0;
146 bool recursive_ : 1 =
false;
150 std::deque<std::unique_ptr<SCC>> topo_;
151 absl::node_hash_map<const Node*, const SCC*> SCCs_;
154 static constexpr uint32_t Unvisited = uint32_t(-1);
155 uint32_t idx_ = Unvisited;
156 uint32_t low_ : 31 = 0;
157 bool on_stack_ : 1 =
false;
158 Node* curr_child =
nullptr;
176 bool is_recursive()
const {
return calculate_SCCs().root()->is_recursive(); }
181 auto muts()
const {
return mut2node_ | std::views::keys; }
183 return mut2node_ | std::views::transform([](
const auto& pair) {
return (
const Node*)pair.second.get(); });
196 void dot(std::ostream& os)
const;
197 void dot(
const char* file =
nullptr)
const;
198 void dot(std::string s)
const {
dot(s.c_str()); }
202 Node* mut2node_nonconst(
Def* mut)
const {
203 if (
auto i = mut2node_.find(mut); i != mut2node_.end())
return i->second.get();
208 Node* make_node(Def*, Node* inest =
nullptr);
209 void sibl(Node*)
const;
210 void find_SCCs(Node*)
const;
212 void calcuate_siblings()
const {
219 const Nest& calculate_SCCs()
const {
228 absl::flat_hash_map<Def*, std::unique_ptr<Node>> mut2node_;
231 mutable bool siblings_ =
false;
232 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 & siblings() 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
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
Same as above.
static const Node * lca(const Node *n, const Node *m)
Least common ancestor of n and m.
void dot(std::string s) const
const Node * mut2node(Def *mut) const
const Node * root() const
Vars vars() const
All Vars occurring in this Nest.
const auto & mut2node() const
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).
Sets< const Var >::Set Vars
GIDMap< Def *, To > MutMap
bool contains(Def *mut) const
const Node * operator[](Def *mut) const
absl::flat_hash_set< Node * > & deps()
bool contains(const Node *n) const