@@ -26,20 +26,44 @@ template <class P, class T, bool post_dom>
26
26
class cfg_dominators_templatet
27
27
{
28
28
public:
29
- typedef std::set<T > target_sett;
29
+ typedef std::set<std:: size_t > target_sett;
30
30
31
31
struct nodet
32
32
{
33
33
target_sett dominators;
34
34
};
35
35
36
- typedef procedure_local_cfg_baset< nodet, P, T> cfgt;
36
+ typedef cfg_basic_blockst<procedure_local_cfg_baset, nodet, P, T> cfgt;
37
37
cfgt cfg;
38
38
39
39
void operator ()(P &program);
40
40
41
41
T entry_node;
42
42
43
+ // / Returns true if basic block \p lhs [post]dominates \p rhs
44
+ bool basic_block_dominates (std::size_t lhs, std::size_t rhs) const
45
+ {
46
+ return cfg[rhs].dominators .count (lhs);
47
+ }
48
+
49
+ // / Returns true if program point \p lhs [post]dominates \p rhs
50
+ bool dominates (T lhs, T rhs) const ;
51
+
52
+ const target_sett &basic_block_dominators (std::size_t block) const
53
+ {
54
+ return cfg[block].dominators ;
55
+ }
56
+
57
+ bool basic_block_reachable (std::size_t block) const
58
+ {
59
+ return !cfg[block].dominators .empty ();
60
+ }
61
+
62
+ bool instruction_reachable (T instruction) const
63
+ {
64
+ return basic_block_reachable (cfg.entry_map .at (instruction));
65
+ }
66
+
43
67
void output (std::ostream &) const ;
44
68
45
69
protected:
@@ -76,32 +100,33 @@ void cfg_dominators_templatet<P, T, post_dom>::initialise(P &program)
76
100
template <class P , class T , bool post_dom>
77
101
void cfg_dominators_templatet<P, T, post_dom>::fixedpoint(P &program)
78
102
{
79
- std::list<T > worklist;
103
+ std::list<typename cfgt::node_indext > worklist;
80
104
81
105
if (cfg.nodes_empty (program))
82
106
return ;
83
107
84
108
if (post_dom)
85
- entry_node=cfg. get_last_node (program);
109
+ entry_node = cfgt:: get_last_node (program);
86
110
else
87
- entry_node=cfg.get_first_node (program);
88
- typename cfgt::nodet &n=cfg[cfg.entry_map [entry_node]];
89
- n.dominators .insert (entry_node);
111
+ entry_node = cfgt::get_first_node (program);
112
+ const auto entry_node_index = cfg.entry_map [entry_node];
113
+ typename cfgt::nodet &n = cfg[entry_node_index];
114
+ n.dominators .insert (entry_node_index);
90
115
91
116
for (typename cfgt::edgest::const_iterator
92
117
s_it=(post_dom?n.in :n.out ).begin ();
93
118
s_it!=(post_dom?n.in :n.out ).end ();
94
119
++s_it)
95
- worklist.push_back (cfg[ s_it->first ]. PC );
120
+ worklist.push_back (s_it->first );
96
121
97
122
while (!worklist.empty ())
98
123
{
99
124
// get node from worklist
100
- T current= worklist.front ();
125
+ const auto current = worklist.front ();
101
126
worklist.pop_front ();
102
127
103
128
bool changed=false ;
104
- typename cfgt::nodet &node= cfg[cfg. entry_map [ current] ];
129
+ typename cfgt::nodet &node = cfg[current];
105
130
if (node.dominators .empty ())
106
131
{
107
132
for (const auto &edge : (post_dom ? node.out : node.in ))
@@ -158,12 +183,39 @@ void cfg_dominators_templatet<P, T, post_dom>::fixedpoint(P &program)
158
183
{
159
184
for (const auto &edge : (post_dom ? node.in : node.out ))
160
185
{
161
- worklist.push_back (cfg[ edge.first ]. PC );
186
+ worklist.push_back (edge.first );
162
187
}
163
188
}
164
189
}
165
190
}
166
191
192
+ template <class P , class T , bool post_dom>
193
+ bool cfg_dominators_templatet<P, T, post_dom>::dominates(T lhs, T rhs) const
194
+ {
195
+ if (lhs == rhs)
196
+ return true ;
197
+
198
+ const auto lhs_block = cfg.entry_map .at (lhs);
199
+ const auto rhs_block = cfg.entry_map .at (rhs);
200
+
201
+ if (lhs_block != rhs_block)
202
+ return basic_block_dominates (lhs_block, rhs_block);
203
+
204
+ // Special case when the program points belong to the same block: lhs
205
+ // dominates rhs iff it is <= rhs in program order (or the reverse if we're
206
+ // a postdominator analysis)
207
+
208
+ for (const auto &instruction : cfg[lhs_block].block )
209
+ {
210
+ if (instruction == lhs)
211
+ return !post_dom;
212
+ else if (instruction == rhs)
213
+ return post_dom;
214
+ }
215
+
216
+ UNREACHABLE; // Entry map is inconsistent with block members?
217
+ }
218
+
167
219
// / Pretty-print a single node in the dominator tree. Supply a specialisation if
168
220
// / operator<< is not sufficient.
169
221
// / \par parameters: `node` to print and stream `out` to pretty-print it to
@@ -184,22 +236,20 @@ inline void dominators_pretty_print_node(
184
236
template <class P , class T , bool post_dom>
185
237
void cfg_dominators_templatet<P, T, post_dom>::output(std::ostream &out) const
186
238
{
187
- for (const auto &node : cfg.entry_map )
239
+ for (typename cfgt::node_indext i = 0 ; i < cfg.size (); ++i )
188
240
{
189
- auto n=node.first ;
190
-
191
- dominators_pretty_print_node (n, out);
241
+ out << " Block " << dominators_pretty_print_node (cfg[i].block .at (0 ), out);
192
242
if (post_dom)
193
243
out << " post-dominated by " ;
194
244
else
195
245
out << " dominated by " ;
196
246
bool first=true ;
197
- for (const auto &d : cfg[node. second ].dominators )
247
+ for (const auto &d : cfg[i ].dominators )
198
248
{
199
249
if (!first)
200
250
out << " , " ;
201
251
first=false ;
202
- dominators_pretty_print_node (d , out);
252
+ dominators_pretty_print_node (cfg[d]. block . at ( 0 ) , out);
203
253
}
204
254
out << " \n " ;
205
255
}
0 commit comments