dominator_tree.hpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492
  1. //=======================================================================
  2. // Copyright (C) 2005-2009 Jongsoo Park <jongsoo.park -at- gmail.com>
  3. //
  4. // Distributed under the Boost Software License, Version 1.0.
  5. // (See accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt)
  7. //=======================================================================
  8. #ifndef BOOST_GRAPH_DOMINATOR_HPP
  9. #define BOOST_GRAPH_DOMINATOR_HPP
  10. #include <boost/config.hpp>
  11. #include <deque>
  12. #include <set>
  13. #include <boost/graph/depth_first_search.hpp>
  14. #include <boost/concept/assert.hpp>
  15. // Dominator tree computation
  16. namespace boost {
  17. namespace detail {
  18. /**
  19. * An extended time_stamper which also records vertices for each dfs number
  20. */
  21. template<class TimeMap, class VertexVector, class TimeT, class Tag>
  22. class time_stamper_with_vertex_vector
  23. : public base_visitor<
  24. time_stamper_with_vertex_vector<TimeMap, VertexVector, TimeT, Tag> >
  25. {
  26. public :
  27. typedef Tag event_filter;
  28. time_stamper_with_vertex_vector(TimeMap timeMap, VertexVector& v,
  29. TimeT& t)
  30. : timeStamper_(timeMap, t), v_(v) { }
  31. template<class Graph>
  32. void
  33. operator()(const typename property_traits<TimeMap>::key_type& v,
  34. const Graph& g)
  35. {
  36. timeStamper_(v, g);
  37. v_[timeStamper_.m_time] = v;
  38. }
  39. private :
  40. time_stamper<TimeMap, TimeT, Tag> timeStamper_;
  41. VertexVector& v_;
  42. };
  43. /**
  44. * A convenient way to create a time_stamper_with_vertex_vector
  45. */
  46. template<class TimeMap, class VertexVector, class TimeT, class Tag>
  47. time_stamper_with_vertex_vector<TimeMap, VertexVector, TimeT, Tag>
  48. stamp_times_with_vertex_vector(TimeMap timeMap, VertexVector& v, TimeT& t,
  49. Tag)
  50. {
  51. return time_stamper_with_vertex_vector<TimeMap, VertexVector, TimeT,
  52. Tag>(timeMap, v, t);
  53. }
  54. template<class Graph, class IndexMap, class TimeMap, class PredMap,
  55. class DomTreePredMap>
  56. class dominator_visitor
  57. {
  58. typedef typename graph_traits<Graph>::vertex_descriptor Vertex;
  59. typedef typename graph_traits<Graph>::vertices_size_type VerticesSizeType;
  60. public :
  61. /**
  62. * @param g [in] the target graph of the dominator tree
  63. * @param entry [in] the entry node of g
  64. * @param domTreePredMap [out] the immediate dominator map
  65. * (parent map in dominator tree)
  66. */
  67. dominator_visitor(const Graph& g, const Vertex& entry,
  68. DomTreePredMap domTreePredMap)
  69. : semi_(num_vertices(g)),
  70. ancestor_(num_vertices(g), graph_traits<Graph>::null_vertex()),
  71. samedom_(ancestor_),
  72. best_(semi_),
  73. semiMap_(make_iterator_property_map(semi_.begin(),
  74. get(vertex_index, g))),
  75. ancestorMap_(make_iterator_property_map(ancestor_.begin(),
  76. get(vertex_index, g))),
  77. bestMap_(make_iterator_property_map(best_.begin(),
  78. get(vertex_index, g))),
  79. buckets_(num_vertices(g)),
  80. bucketMap_(make_iterator_property_map(buckets_.begin(),
  81. get(vertex_index, g))),
  82. entry_(entry),
  83. domTreePredMap_(domTreePredMap),
  84. numOfVertices_(num_vertices(g)),
  85. samedomMap(make_iterator_property_map(samedom_.begin(),
  86. get(vertex_index, g)))
  87. {
  88. }
  89. void
  90. operator()(const Vertex& n, const TimeMap& dfnumMap,
  91. const PredMap& parentMap, const Graph& g)
  92. {
  93. if (n == entry_) return;
  94. const Vertex p(get(parentMap, n));
  95. Vertex s(p);
  96. // 1. Calculate the semidominator of n,
  97. // based on the semidominator thm.
  98. // * Semidominator thm. : To find the semidominator of a node n,
  99. // consider all predecessors v of n in the CFG (Control Flow Graph).
  100. // - If v is a proper ancestor of n in the spanning tree
  101. // (so dfnum(v) < dfnum(n)), then v is a candidate for semi(n)
  102. // - If v is a non-ancestor of n (so dfnum(v) > dfnum(n))
  103. // then for each u that is an ancestor of v (or u = v),
  104. // Let semi(u) be a candidate for semi(n)
  105. // of all these candidates, the one with lowest dfnum is
  106. // the semidominator of n.
  107. // For each predecessor of n
  108. typename graph_traits<Graph>::in_edge_iterator inItr, inEnd;
  109. for (boost::tie(inItr, inEnd) = in_edges(n, g); inItr != inEnd; ++inItr)
  110. {
  111. const Vertex v = source(*inItr, g);
  112. // To deal with unreachable nodes
  113. if (get(dfnumMap, v) < 0 || get(dfnumMap, v) >= numOfVertices_)
  114. continue;
  115. Vertex s2;
  116. if (get(dfnumMap, v) <= get(dfnumMap, n))
  117. s2 = v;
  118. else
  119. s2 = get(semiMap_, ancestor_with_lowest_semi_(v, dfnumMap));
  120. if (get(dfnumMap, s2) < get(dfnumMap, s))
  121. s = s2;
  122. }
  123. put(semiMap_, n, s);
  124. // 2. Calculation of n's dominator is deferred until
  125. // the path from s to n has been linked into the forest
  126. get(bucketMap_, s).push_back(n);
  127. get(ancestorMap_, n) = p;
  128. get(bestMap_, n) = n;
  129. // 3. Now that the path from p to v has been linked into
  130. // the spanning forest, these lines calculate the dominator of v,
  131. // based on the dominator thm., or else defer the calculation
  132. // until y's dominator is known
  133. // * Dominator thm. : On the spanning-tree path below semi(n) and
  134. // above or including n, let y be the node
  135. // with the smallest-numbered semidominator. Then,
  136. //
  137. // idom(n) = semi(n) if semi(y)=semi(n) or
  138. // idom(y) if semi(y) != semi(n)
  139. typename std::deque<Vertex>::iterator buckItr;
  140. for (buckItr = get(bucketMap_, p).begin();
  141. buckItr != get(bucketMap_, p).end();
  142. ++buckItr)
  143. {
  144. const Vertex v(*buckItr);
  145. const Vertex y(ancestor_with_lowest_semi_(v, dfnumMap));
  146. if (get(semiMap_, y) == get(semiMap_, v))
  147. put(domTreePredMap_, v, p);
  148. else
  149. put(samedomMap, v, y);
  150. }
  151. get(bucketMap_, p).clear();
  152. }
  153. protected :
  154. /**
  155. * Evaluate function in Tarjan's path compression
  156. */
  157. const Vertex
  158. ancestor_with_lowest_semi_(const Vertex& v, const TimeMap& dfnumMap)
  159. {
  160. const Vertex a(get(ancestorMap_, v));
  161. if (get(ancestorMap_, a) != graph_traits<Graph>::null_vertex())
  162. {
  163. const Vertex b(ancestor_with_lowest_semi_(a, dfnumMap));
  164. put(ancestorMap_, v, get(ancestorMap_, a));
  165. if (get(dfnumMap, get(semiMap_, b)) <
  166. get(dfnumMap, get(semiMap_, get(bestMap_, v))))
  167. put(bestMap_, v, b);
  168. }
  169. return get(bestMap_, v);
  170. }
  171. std::vector<Vertex> semi_, ancestor_, samedom_, best_;
  172. PredMap semiMap_, ancestorMap_, bestMap_;
  173. std::vector< std::deque<Vertex> > buckets_;
  174. iterator_property_map<typename std::vector<std::deque<Vertex> >::iterator,
  175. IndexMap> bucketMap_;
  176. const Vertex& entry_;
  177. DomTreePredMap domTreePredMap_;
  178. const VerticesSizeType numOfVertices_;
  179. public :
  180. PredMap samedomMap;
  181. };
  182. } // namespace detail
  183. /**
  184. * @brief Build dominator tree using Lengauer-Tarjan algorithm.
  185. * It takes O((V+E)log(V+E)) time.
  186. *
  187. * @pre dfnumMap, parentMap and verticesByDFNum have dfs results corresponding
  188. * indexMap.
  189. * If dfs has already run before,
  190. * this function would be good for saving computations.
  191. * @pre Unreachable nodes must be masked as
  192. * graph_traits<Graph>::null_vertex in parentMap.
  193. * @pre Unreachable nodes must be masked as
  194. * (std::numeric_limits<VerticesSizeType>::max)() in dfnumMap.
  195. *
  196. * @param domTreePredMap [out] : immediate dominator map (parent map
  197. * in dom. tree)
  198. *
  199. * @note reference Appel. p. 452~453. algorithm 19.9, 19.10.
  200. *
  201. * @todo : Optimization in Finding Dominators in Practice, Loukas Georgiadis
  202. */
  203. template<class Graph, class IndexMap, class TimeMap, class PredMap,
  204. class VertexVector, class DomTreePredMap>
  205. void
  206. lengauer_tarjan_dominator_tree_without_dfs
  207. (const Graph& g,
  208. const typename graph_traits<Graph>::vertex_descriptor& entry,
  209. const IndexMap& /*indexMap*/,
  210. TimeMap dfnumMap, PredMap parentMap, VertexVector& verticesByDFNum,
  211. DomTreePredMap domTreePredMap)
  212. {
  213. // Typedefs and concept check
  214. typedef typename graph_traits<Graph>::vertex_descriptor Vertex;
  215. typedef typename graph_traits<Graph>::vertices_size_type VerticesSizeType;
  216. BOOST_CONCEPT_ASSERT(( BidirectionalGraphConcept<Graph> ));
  217. const VerticesSizeType numOfVertices = num_vertices(g);
  218. if (numOfVertices == 0) return;
  219. // 1. Visit each vertex in reverse post order and calculate sdom.
  220. detail::dominator_visitor<Graph, IndexMap, TimeMap, PredMap, DomTreePredMap>
  221. visitor(g, entry, domTreePredMap);
  222. VerticesSizeType i;
  223. for (i = 0; i < numOfVertices; ++i)
  224. {
  225. const Vertex u(verticesByDFNum[numOfVertices - 1 - i]);
  226. if (u != graph_traits<Graph>::null_vertex())
  227. visitor(u, dfnumMap, parentMap, g);
  228. }
  229. // 2. Now all the deferred dominator calculations,
  230. // based on the second clause of the dominator thm., are performed
  231. for (i = 0; i < numOfVertices; ++i)
  232. {
  233. const Vertex n(verticesByDFNum[i]);
  234. if (n == entry || n == graph_traits<Graph>::null_vertex())
  235. continue;
  236. Vertex u = get(visitor.samedomMap, n);
  237. if (u != graph_traits<Graph>::null_vertex())
  238. {
  239. put(domTreePredMap, n, get(domTreePredMap, u));
  240. }
  241. }
  242. }
  243. /**
  244. * Unlike lengauer_tarjan_dominator_tree_without_dfs,
  245. * dfs is run in this function and
  246. * the result is written to dfnumMap, parentMap, vertices.
  247. *
  248. * If the result of dfs required after this algorithm,
  249. * this function can eliminate the need of rerunning dfs.
  250. */
  251. template<class Graph, class IndexMap, class TimeMap, class PredMap,
  252. class VertexVector, class DomTreePredMap>
  253. void
  254. lengauer_tarjan_dominator_tree
  255. (const Graph& g,
  256. const typename graph_traits<Graph>::vertex_descriptor& entry,
  257. const IndexMap& indexMap,
  258. TimeMap dfnumMap, PredMap parentMap, VertexVector& verticesByDFNum,
  259. DomTreePredMap domTreePredMap)
  260. {
  261. // Typedefs and concept check
  262. typedef typename graph_traits<Graph>::vertices_size_type VerticesSizeType;
  263. BOOST_CONCEPT_ASSERT(( BidirectionalGraphConcept<Graph> ));
  264. // 1. Depth first visit
  265. const VerticesSizeType numOfVertices = num_vertices(g);
  266. if (numOfVertices == 0) return;
  267. VerticesSizeType time =
  268. (std::numeric_limits<VerticesSizeType>::max)();
  269. std::vector<default_color_type>
  270. colors(numOfVertices, color_traits<default_color_type>::white());
  271. depth_first_visit
  272. (g, entry,
  273. make_dfs_visitor
  274. (make_pair(record_predecessors(parentMap, on_tree_edge()),
  275. detail::stamp_times_with_vertex_vector
  276. (dfnumMap, verticesByDFNum, time, on_discover_vertex()))),
  277. make_iterator_property_map(colors.begin(), indexMap));
  278. // 2. Run main algorithm.
  279. lengauer_tarjan_dominator_tree_without_dfs(g, entry, indexMap, dfnumMap,
  280. parentMap, verticesByDFNum,
  281. domTreePredMap);
  282. }
  283. /**
  284. * Use vertex_index as IndexMap and make dfnumMap, parentMap, verticesByDFNum
  285. * internally.
  286. * If we don't need the result of dfs (dfnumMap, parentMap, verticesByDFNum),
  287. * this function would be more convenient one.
  288. */
  289. template<class Graph, class DomTreePredMap>
  290. void
  291. lengauer_tarjan_dominator_tree
  292. (const Graph& g,
  293. const typename graph_traits<Graph>::vertex_descriptor& entry,
  294. DomTreePredMap domTreePredMap)
  295. {
  296. // typedefs
  297. typedef typename graph_traits<Graph>::vertex_descriptor Vertex;
  298. typedef typename graph_traits<Graph>::vertices_size_type VerticesSizeType;
  299. typedef typename property_map<Graph, vertex_index_t>::const_type IndexMap;
  300. typedef
  301. iterator_property_map<typename std::vector<VerticesSizeType>::iterator,
  302. IndexMap> TimeMap;
  303. typedef
  304. iterator_property_map<typename std::vector<Vertex>::iterator, IndexMap>
  305. PredMap;
  306. // Make property maps
  307. const VerticesSizeType numOfVertices = num_vertices(g);
  308. if (numOfVertices == 0) return;
  309. const IndexMap indexMap = get(vertex_index, g);
  310. std::vector<VerticesSizeType> dfnum(numOfVertices, 0);
  311. TimeMap dfnumMap(make_iterator_property_map(dfnum.begin(), indexMap));
  312. std::vector<Vertex> parent(numOfVertices,
  313. graph_traits<Graph>::null_vertex());
  314. PredMap parentMap(make_iterator_property_map(parent.begin(), indexMap));
  315. std::vector<Vertex> verticesByDFNum(parent);
  316. // Run main algorithm
  317. lengauer_tarjan_dominator_tree(g, entry,
  318. indexMap, dfnumMap, parentMap,
  319. verticesByDFNum, domTreePredMap);
  320. }
  321. /**
  322. * Muchnick. p. 182, 184
  323. *
  324. * using iterative bit vector analysis
  325. */
  326. template<class Graph, class IndexMap, class DomTreePredMap>
  327. void
  328. iterative_bit_vector_dominator_tree
  329. (const Graph& g,
  330. const typename graph_traits<Graph>::vertex_descriptor& entry,
  331. const IndexMap& indexMap,
  332. DomTreePredMap domTreePredMap)
  333. {
  334. typedef typename graph_traits<Graph>::vertex_descriptor Vertex;
  335. typedef typename graph_traits<Graph>::vertex_iterator vertexItr;
  336. typedef typename graph_traits<Graph>::vertices_size_type VerticesSizeType;
  337. typedef
  338. iterator_property_map<typename std::vector< std::set<Vertex> >::iterator,
  339. IndexMap> vertexSetMap;
  340. BOOST_CONCEPT_ASSERT(( BidirectionalGraphConcept<Graph> ));
  341. // 1. Finding dominator
  342. // 1.1. Initialize
  343. const VerticesSizeType numOfVertices = num_vertices(g);
  344. if (numOfVertices == 0) return;
  345. vertexItr vi, viend;
  346. boost::tie(vi, viend) = vertices(g);
  347. const std::set<Vertex> N(vi, viend);
  348. bool change = true;
  349. std::vector< std::set<Vertex> > dom(numOfVertices, N);
  350. vertexSetMap domMap(make_iterator_property_map(dom.begin(), indexMap));
  351. get(domMap, entry).clear();
  352. get(domMap, entry).insert(entry);
  353. while (change)
  354. {
  355. change = false;
  356. for (boost::tie(vi, viend) = vertices(g); vi != viend; ++vi)
  357. {
  358. if (*vi == entry) continue;
  359. std::set<Vertex> T(N);
  360. typename graph_traits<Graph>::in_edge_iterator inItr, inEnd;
  361. for (boost::tie(inItr, inEnd) = in_edges(*vi, g); inItr != inEnd; ++inItr)
  362. {
  363. const Vertex p = source(*inItr, g);
  364. std::set<Vertex> tempSet;
  365. std::set_intersection(T.begin(), T.end(),
  366. get(domMap, p).begin(),
  367. get(domMap, p).end(),
  368. std::inserter(tempSet, tempSet.begin()));
  369. T.swap(tempSet);
  370. }
  371. T.insert(*vi);
  372. if (T != get(domMap, *vi))
  373. {
  374. change = true;
  375. get(domMap, *vi).swap(T);
  376. }
  377. } // end of for (boost::tie(vi, viend) = vertices(g)
  378. } // end of while(change)
  379. // 2. Build dominator tree
  380. for (boost::tie(vi, viend) = vertices(g); vi != viend; ++vi)
  381. get(domMap, *vi).erase(*vi);
  382. Graph domTree(numOfVertices);
  383. for (boost::tie(vi, viend) = vertices(g); vi != viend; ++vi)
  384. {
  385. if (*vi == entry) continue;
  386. // We have to iterate through copied dominator set
  387. const std::set<Vertex> tempSet(get(domMap, *vi));
  388. typename std::set<Vertex>::const_iterator s;
  389. for (s = tempSet.begin(); s != tempSet.end(); ++s)
  390. {
  391. typename std::set<Vertex>::iterator t;
  392. for (t = get(domMap, *vi).begin(); t != get(domMap, *vi).end(); )
  393. {
  394. typename std::set<Vertex>::iterator old_t = t;
  395. ++t; // Done early because t may become invalid
  396. if (*old_t == *s) continue;
  397. if (get(domMap, *s).find(*old_t) != get(domMap, *s).end())
  398. get(domMap, *vi).erase(old_t);
  399. }
  400. }
  401. }
  402. for (boost::tie(vi, viend) = vertices(g); vi != viend; ++vi)
  403. {
  404. if (*vi != entry && get(domMap, *vi).size() == 1)
  405. {
  406. Vertex temp = *get(domMap, *vi).begin();
  407. put(domTreePredMap, *vi, temp);
  408. }
  409. }
  410. }
  411. template<class Graph, class DomTreePredMap>
  412. void
  413. iterative_bit_vector_dominator_tree
  414. (const Graph& g,
  415. const typename graph_traits<Graph>::vertex_descriptor& entry,
  416. DomTreePredMap domTreePredMap)
  417. {
  418. typename property_map<Graph, vertex_index_t>::const_type
  419. indexMap = get(vertex_index, g);
  420. iterative_bit_vector_dominator_tree(g, entry, indexMap, domTreePredMap);
  421. }
  422. } // namespace boost
  423. #endif // BOOST_GRAPH_DOMINATOR_HPP