Skip to content

Commit 1a28c15

Browse files
committed
Dominator analysis: analyse on basic-block granularity
Previously this used instruction granularity, which especially combined with a naive dominator set (not a dominator tree) can waste a great deal of time and memory computing and storing the dominator sets. Now that cfg_dominatorst works on a basic-block granularity, natural_loopst does as well.
1 parent b625e0a commit 1a28c15

15 files changed

+613
-184
lines changed

jbmc/src/java_bytecode/java_local_variable_table.cpp

+17-11
Original file line numberDiff line numberDiff line change
@@ -85,19 +85,19 @@ struct procedure_local_cfg_baset<
8585
}
8686
}
8787

88-
java_bytecode_convert_methodt::method_offsett
89-
get_first_node(const method_with_amapt &args) const
88+
static java_bytecode_convert_methodt::method_offsett
89+
get_first_node(const method_with_amapt &args)
9090
{
9191
return args.second.begin()->first;
9292
}
9393

94-
java_bytecode_convert_methodt::method_offsett
95-
get_last_node(const method_with_amapt &args) const
94+
static java_bytecode_convert_methodt::method_offsett
95+
get_last_node(const method_with_amapt &args)
9696
{
9797
return (--args.second.end())->first;
9898
}
9999

100-
bool nodes_empty(const method_with_amapt &args) const
100+
static bool nodes_empty(const method_with_amapt &args)
101101
{
102102
return args.second.empty();
103103
}
@@ -442,13 +442,19 @@ static java_bytecode_convert_methodt::method_offsett get_common_dominator(
442442
candidate_dominators;
443443
for(auto v : merge_vars)
444444
{
445-
const auto &dominator_nodeidx=
445+
const auto &var_start_basic_block =
446446
dominator_analysis.cfg.entry_map.at(v->var.start_pc);
447-
const auto &this_var_doms=
448-
dominator_analysis.cfg[dominator_nodeidx].dominators;
449-
for(const auto this_var_dom : this_var_doms)
450-
if(this_var_dom<=first_pc)
451-
candidate_dominators.push_back(this_var_dom);
447+
const auto &this_var_dom_blocks =
448+
dominator_analysis.cfg[var_start_basic_block].dominators;
449+
for(const auto this_var_dom_block_index : this_var_dom_blocks)
450+
{
451+
const auto &this_var_dom_block =
452+
dominator_analysis.cfg[this_var_dom_block_index].block;
453+
// Only consider placing variable declarations at the head of
454+
// a basic block (which conveniently is always a safe choice, even
455+
// for a live range starting midway through a block)
456+
candidate_dominators.push_back(this_var_dom_block.front());
457+
}
452458
}
453459
std::sort(candidate_dominators.begin(), candidate_dominators.end());
454460

src/analyses/cfg_dominators.h

+67-17
Original file line numberDiff line numberDiff line change
@@ -26,20 +26,44 @@ template <class P, class T, bool post_dom>
2626
class cfg_dominators_templatet
2727
{
2828
public:
29-
typedef std::set<T> target_sett;
29+
typedef std::set<std::size_t> target_sett;
3030

3131
struct nodet
3232
{
3333
target_sett dominators;
3434
};
3535

36-
typedef procedure_local_cfg_baset<nodet, P, T> cfgt;
36+
typedef cfg_basic_blockst<procedure_local_cfg_baset, nodet, P, T> cfgt;
3737
cfgt cfg;
3838

3939
void operator()(P &program);
4040

4141
T entry_node;
4242

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+
4367
void output(std::ostream &) const;
4468

4569
protected:
@@ -76,32 +100,33 @@ void cfg_dominators_templatet<P, T, post_dom>::initialise(P &program)
76100
template <class P, class T, bool post_dom>
77101
void cfg_dominators_templatet<P, T, post_dom>::fixedpoint(P &program)
78102
{
79-
std::list<T> worklist;
103+
std::list<typename cfgt::node_indext> worklist;
80104

81105
if(cfg.nodes_empty(program))
82106
return;
83107

84108
if(post_dom)
85-
entry_node=cfg.get_last_node(program);
109+
entry_node = cfgt::get_last_node(program);
86110
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);
90115

91116
for(typename cfgt::edgest::const_iterator
92117
s_it=(post_dom?n.in:n.out).begin();
93118
s_it!=(post_dom?n.in:n.out).end();
94119
++s_it)
95-
worklist.push_back(cfg[s_it->first].PC);
120+
worklist.push_back(s_it->first);
96121

97122
while(!worklist.empty())
98123
{
99124
// get node from worklist
100-
T current=worklist.front();
125+
const auto current = worklist.front();
101126
worklist.pop_front();
102127

103128
bool changed=false;
104-
typename cfgt::nodet &node=cfg[cfg.entry_map[current]];
129+
typename cfgt::nodet &node = cfg[current];
105130
if(node.dominators.empty())
106131
{
107132
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)
158183
{
159184
for(const auto &edge : (post_dom ? node.in : node.out))
160185
{
161-
worklist.push_back(cfg[edge.first].PC);
186+
worklist.push_back(edge.first);
162187
}
163188
}
164189
}
165190
}
166191

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+
167219
/// Pretty-print a single node in the dominator tree. Supply a specialisation if
168220
/// operator<< is not sufficient.
169221
/// \par parameters: `node` to print and stream `out` to pretty-print it to
@@ -184,22 +236,20 @@ inline void dominators_pretty_print_node(
184236
template <class P, class T, bool post_dom>
185237
void cfg_dominators_templatet<P, T, post_dom>::output(std::ostream &out) const
186238
{
187-
for(const auto &node : cfg.entry_map)
239+
for(typename cfgt::node_indext i = 0; i < cfg.size(); ++i)
188240
{
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);
192242
if(post_dom)
193243
out << " post-dominated by ";
194244
else
195245
out << " dominated by ";
196246
bool first=true;
197-
for(const auto &d : cfg[node.second].dominators)
247+
for(const auto &d : cfg[i].dominators)
198248
{
199249
if(!first)
200250
out << ", ";
201251
first=false;
202-
dominators_pretty_print_node(d, out);
252+
dominators_pretty_print_node(cfg[d].block.at(0), out);
203253
}
204254
out << "\n";
205255
}

src/analyses/dependence_graph.cpp

+4-15
Original file line numberDiff line numberDiff line change
@@ -97,22 +97,11 @@ void dep_graph_domaint::control_dependencies(
9797

9898
// we could hard-code assume and goto handling here to improve
9999
// performance
100-
cfg_post_dominatorst::cfgt::entry_mapt::const_iterator e =
101-
pd.cfg.entry_map.find(control_dep_candidate);
102-
103-
INVARIANT(
104-
e != pd.cfg.entry_map.end(), "cfg must have an entry for every location");
105-
106-
const cfg_post_dominatorst::cfgt::nodet &m=
107-
pd.cfg[e->second];
108-
109-
// successors of M
110-
for(const auto &edge : m.out)
100+
for(const auto &candidate_successor :
101+
goto_programt::get_well_formed_instruction_successors(
102+
control_dep_candidate))
111103
{
112-
const cfg_post_dominatorst::cfgt::nodet &m_s=
113-
pd.cfg[edge.first];
114-
115-
if(m_s.dominators.find(to)!=m_s.dominators.end())
104+
if(pd.dominates(to, candidate_successor))
116105
post_dom_one=true;
117106
else
118107
post_dom_all=false;

0 commit comments

Comments
 (0)