@@ -37,14 +37,14 @@ template <class P, class T, bool post_dom>
37
37
class cfg_dominators_templatet
38
38
{
39
39
public:
40
- typedef std::set<T > target_sett;
40
+ typedef std::set<std:: size_t > target_sett;
41
41
42
42
struct nodet
43
43
{
44
44
target_sett dominators;
45
45
};
46
46
47
- typedef procedure_local_cfg_baset< nodet, P, T> cfgt;
47
+ typedef cfg_basic_blockst<procedure_local_cfg_baset, nodet, P, T> cfgt;
48
48
cfgt cfg;
49
49
50
50
void operator ()(P &program);
@@ -63,43 +63,71 @@ class cfg_dominators_templatet
63
63
return cfg.get_node (program_point);
64
64
}
65
65
66
- // / Returns true if the program point corresponding to \p rhs_node is
67
- // / dominated by program point \p lhs. Saves node lookup compared to the
68
- // / dominates overload that takes two program points, so this version is
69
- // / preferable if you intend to check more than one potential dominator.
70
- // / Note by definition all program points dominate themselves.
71
- bool dominates (T lhs, const nodet &rhs_node) const
66
+ // / Get the basic-block graph node index for \p program_point
67
+ std::size_t get_node_index (const T &program_point) const
68
+ {
69
+ return cfg.get_node_index (program_point);
70
+ }
71
+
72
+ // / Returns true if basic block \p lhs [post]dominates \p rhs
73
+ bool basic_block_dominates (std::size_t lhs, std::size_t rhs) const
72
74
{
73
- return rhs_node .dominators .count (lhs);
75
+ return cfg[rhs] .dominators .count (lhs);
74
76
}
75
77
78
+ // / Returns true if program point \p lhs [post]dominates \p rhs
79
+ bool dominates_same_block (T lhs, T rhs, std::size_t block) const ;
80
+
76
81
// / Returns true if program point \p lhs dominates \p rhs.
77
82
// / Note by definition all program points dominate themselves.
78
83
bool dominates (T lhs, T rhs) const
79
84
{
80
- return dominates (lhs, get_node (rhs));
85
+ const auto lhs_block = cfg.entry_map .at (lhs);
86
+ const auto rhs_block = cfg.entry_map .at (rhs);
87
+
88
+ if (lhs == rhs)
89
+ return true ;
90
+
91
+ if (lhs_block != rhs_block)
92
+ return basic_block_dominates (lhs_block, rhs_block);
93
+ else
94
+ return dominates_same_block (lhs, rhs, lhs_block);
81
95
}
82
96
83
- // / Returns true if the program point for \p program_point_node is reachable
97
+ // / Returns true if the basic block \p basic_block_node is reachable
84
98
// / from the entry point. Saves a lookup compared to the overload taking a
85
99
// / program point, so use this overload if you already have the node.
86
- bool program_point_reachable (const nodet &program_point_node ) const
100
+ bool basic_block_reachable (const nodet &basic_block_node ) const
87
101
{
88
102
// Dominator analysis walks from the entry point, so a side-effect is to
89
103
// identify unreachable program points (those which don't dominate even
90
104
// themselves).
91
- return !program_point_node .dominators .empty ();
105
+ return !basic_block_node .dominators .empty ();
92
106
}
93
107
94
- // / Returns true if the program point for \p program_point_node is reachable
108
+ // / Returns true if the basic block \p basic_block_node is reachable
95
109
// / from the entry point. Saves a lookup compared to the overload taking a
96
- // / program point, so use this overload if you already have the node.
110
+ // / program point, so use this overload if you already have the node index.
111
+ bool basic_block_reachable (std::size_t block) const
112
+ {
113
+ return basic_block_reachable (cfg[block]);
114
+ }
115
+
116
+ // / Returns true if the program point for \p program_point_node is reachable
117
+ // / from the entry point.
97
118
bool program_point_reachable (T program_point) const
98
119
{
99
120
// Dominator analysis walks from the entry point, so a side-effect is to
100
121
// identify unreachable program points (those which don't dominate even
101
122
// themselves).
102
- return program_point_reachable (get_node (program_point));
123
+ return basic_block_reachable (get_node_index (program_point));
124
+ }
125
+
126
+ // / Returns the set of dominator blocks for a given basic block, including
127
+ // / itself. The result is a set of indices usable with this class' operator[].
128
+ const target_sett &basic_block_dominators (std::size_t block) const
129
+ {
130
+ return cfg[block].dominators ;
103
131
}
104
132
105
133
T entry_node;
@@ -140,7 +168,7 @@ void cfg_dominators_templatet<P, T, post_dom>::initialise(P &program)
140
168
template <class P , class T , bool post_dom>
141
169
void cfg_dominators_templatet<P, T, post_dom>::fixedpoint(P &program)
142
170
{
143
- std::list<T > worklist;
171
+ std::list<typename cfgt::node_indext > worklist;
144
172
145
173
if (cfgt::nodes_empty (program))
146
174
return ;
@@ -149,23 +177,24 @@ void cfg_dominators_templatet<P, T, post_dom>::fixedpoint(P &program)
149
177
entry_node = cfgt::get_last_node (program);
150
178
else
151
179
entry_node = cfgt::get_first_node (program);
152
- typename cfgt::nodet &n = cfg.get_node (entry_node);
153
- n.dominators .insert (entry_node);
180
+ const auto entry_node_index = cfg.get_node_index (entry_node);
181
+ typename cfgt::nodet &n = cfg[entry_node_index];
182
+ n.dominators .insert (entry_node_index);
154
183
155
184
for (typename cfgt::edgest::const_iterator
156
185
s_it=(post_dom?n.in :n.out ).begin ();
157
186
s_it!=(post_dom?n.in :n.out ).end ();
158
187
++s_it)
159
- worklist.push_back (cfg[ s_it->first ]. PC );
188
+ worklist.push_back (s_it->first );
160
189
161
190
while (!worklist.empty ())
162
191
{
163
192
// get node from worklist
164
- T current= worklist.front ();
193
+ const auto current = worklist.front ();
165
194
worklist.pop_front ();
166
195
167
196
bool changed=false ;
168
- typename cfgt::nodet &node = cfg. get_node ( current) ;
197
+ typename cfgt::nodet &node = cfg[ current] ;
169
198
if (node.dominators .empty ())
170
199
{
171
200
for (const auto &edge : (post_dom ? node.out : node.in ))
@@ -222,12 +251,33 @@ void cfg_dominators_templatet<P, T, post_dom>::fixedpoint(P &program)
222
251
{
223
252
for (const auto &edge : (post_dom ? node.in : node.out ))
224
253
{
225
- worklist.push_back (cfg[ edge.first ]. PC );
254
+ worklist.push_back (edge.first );
226
255
}
227
256
}
228
257
}
229
258
}
230
259
260
+ template <class P , class T , bool post_dom>
261
+ bool cfg_dominators_templatet<P, T, post_dom>::dominates_same_block(
262
+ T lhs,
263
+ T rhs,
264
+ std::size_t block) const
265
+ {
266
+ // Special case when the program points belong to the same block: lhs
267
+ // dominates rhs iff it is <= rhs in program order (or the reverse if we're
268
+ // a postdominator analysis)
269
+
270
+ for (const auto &instruction : cfg[block].block )
271
+ {
272
+ if (instruction == lhs)
273
+ return !post_dom;
274
+ else if (instruction == rhs)
275
+ return post_dom;
276
+ }
277
+
278
+ UNREACHABLE; // Entry map is inconsistent with block members?
279
+ }
280
+
231
281
// / Pretty-print a single node in the dominator tree. Supply a specialisation if
232
282
// / operator<< is not sufficient.
233
283
// / \par parameters: `node` to print and stream `out` to pretty-print it to
@@ -248,22 +298,20 @@ inline void dominators_pretty_print_node(
248
298
template <class P , class T , bool post_dom>
249
299
void cfg_dominators_templatet<P, T, post_dom>::output(std::ostream &out) const
250
300
{
251
- for (const auto &n : cfg.keys () )
301
+ for (typename cfgt::node_indext i = 0 ; i < cfg.size (); ++i )
252
302
{
253
- const auto &node = cfg.get_node (n);
254
-
255
- dominators_pretty_print_node (n, out);
303
+ out << " Block " << dominators_pretty_print_node (cfg[i].block .at (0 ), out);
256
304
if (post_dom)
257
305
out << " post-dominated by " ;
258
306
else
259
307
out << " dominated by " ;
260
308
bool first=true ;
261
- for (const auto &d : node .dominators )
309
+ for (const auto &d : cfg[i] .dominators )
262
310
{
263
311
if (!first)
264
312
out << " , " ;
265
313
first=false ;
266
- dominators_pretty_print_node (d , out);
314
+ dominators_pretty_print_node (cfg[d]. block . at ( 0 ) , out);
267
315
}
268
316
out << " \n " ;
269
317
}
0 commit comments