subgraph.hpp 38 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070
  1. //=======================================================================
  2. // Copyright 2001 University of Notre Dame.
  3. // Authors: Jeremy G. Siek and Lie-Quan Lee
  4. //
  5. // Distributed under the Boost Software License, Version 1.0. (See
  6. // accompanying file LICENSE_1_0.txt or copy at
  7. // http://www.boost.org/LICENSE_1_0.txt)
  8. //=======================================================================
  9. #ifndef BOOST_SUBGRAPH_HPP
  10. #define BOOST_SUBGRAPH_HPP
  11. // UNDER CONSTRUCTION
  12. #include <boost/config.hpp>
  13. #include <list>
  14. #include <vector>
  15. #include <map>
  16. #include <boost/assert.hpp>
  17. #include <boost/graph/graph_traits.hpp>
  18. #include <boost/graph/graph_mutability_traits.hpp>
  19. #include <boost/graph/properties.hpp>
  20. #include <boost/iterator/indirect_iterator.hpp>
  21. #include <boost/static_assert.hpp>
  22. #include <boost/assert.hpp>
  23. #include <boost/type_traits.hpp>
  24. #include <boost/mpl/if.hpp>
  25. #include <boost/mpl/or.hpp>
  26. namespace boost {
  27. struct subgraph_tag { };
  28. /** @name Property Lookup
  29. * The local_property and global_property functions are used to create
  30. * structures that determine the lookup strategy for properties in subgraphs.
  31. * Note that the nested kind member is used to help interoperate with actual
  32. * Property types.
  33. */
  34. //@{
  35. template <typename T>
  36. struct local_property
  37. {
  38. typedef T kind;
  39. local_property(T x) : value(x) { }
  40. T value;
  41. };
  42. template <typename T>
  43. inline local_property<T> local(T x)
  44. { return local_property<T>(x); }
  45. template <typename T>
  46. struct global_property
  47. {
  48. typedef T kind;
  49. global_property(T x) : value(x) { }
  50. T value;
  51. };
  52. template <typename T>
  53. inline global_property<T> global(T x)
  54. { return global_property<T>(x); }
  55. //@}
  56. // Invariants of an induced subgraph:
  57. // - If vertex u is in subgraph g, then u must be in g.parent().
  58. // - If edge e is in subgraph g, then e must be in g.parent().
  59. // - If edge e=(u,v) is in the root graph, then edge e
  60. // is also in any subgraph that contains both vertex u and v.
  61. // The Graph template parameter must have a vertex_index and edge_index
  62. // internal property. It is assumed that the vertex indices are assigned
  63. // automatically by the graph during a call to add_vertex(). It is not
  64. // assumed that the edge vertices are assigned automatically, they are
  65. // explicitly assigned here.
  66. template <typename Graph>
  67. class subgraph {
  68. typedef graph_traits<Graph> Traits;
  69. typedef std::list<subgraph<Graph>*> ChildrenList;
  70. public:
  71. // Graph requirements
  72. typedef typename Traits::vertex_descriptor vertex_descriptor;
  73. typedef typename Traits::edge_descriptor edge_descriptor;
  74. typedef typename Traits::directed_category directed_category;
  75. typedef typename Traits::edge_parallel_category edge_parallel_category;
  76. typedef typename Traits::traversal_category traversal_category;
  77. // IncidenceGraph requirements
  78. typedef typename Traits::out_edge_iterator out_edge_iterator;
  79. typedef typename Traits::degree_size_type degree_size_type;
  80. // AdjacencyGraph requirements
  81. typedef typename Traits::adjacency_iterator adjacency_iterator;
  82. // VertexListGraph requirements
  83. typedef typename Traits::vertex_iterator vertex_iterator;
  84. typedef typename Traits::vertices_size_type vertices_size_type;
  85. // EdgeListGraph requirements
  86. typedef typename Traits::edge_iterator edge_iterator;
  87. typedef typename Traits::edges_size_type edges_size_type;
  88. typedef typename Traits::in_edge_iterator in_edge_iterator;
  89. typedef typename edge_property_type<Graph>::type edge_property_type;
  90. typedef typename vertex_property_type<Graph>::type vertex_property_type;
  91. typedef subgraph_tag graph_tag;
  92. typedef Graph graph_type;
  93. typedef typename graph_property_type<Graph>::type graph_property_type;
  94. // Create the main graph, the root of the subgraph tree
  95. subgraph()
  96. : m_parent(0), m_edge_counter(0)
  97. { }
  98. subgraph(const graph_property_type& p)
  99. : m_graph(p), m_parent(0), m_edge_counter(0)
  100. { }
  101. subgraph(vertices_size_type n, const graph_property_type& p = graph_property_type())
  102. : m_graph(n, p), m_parent(0), m_edge_counter(0), m_global_vertex(n)
  103. {
  104. typename Graph::vertex_iterator v, v_end;
  105. vertices_size_type i = 0;
  106. for(boost::tie(v, v_end) = vertices(m_graph); v != v_end; ++v)
  107. m_global_vertex[i++] = *v;
  108. }
  109. // copy constructor
  110. subgraph(const subgraph& x)
  111. : m_parent(x.m_parent), m_edge_counter(x.m_edge_counter)
  112. , m_global_vertex(x.m_global_vertex), m_global_edge(x.m_global_edge)
  113. {
  114. if(x.is_root())
  115. {
  116. m_graph = x.m_graph;
  117. }
  118. // Do a deep copy (recursive).
  119. // Only the root graph is copied, the subgraphs contain
  120. // only references to the global vertices they own.
  121. typename subgraph<Graph>::children_iterator i,i_end;
  122. boost::tie(i,i_end) = x.children();
  123. for(; i != i_end; ++i)
  124. {
  125. subgraph<Graph> child = this->create_subgraph();
  126. child = *i;
  127. vertex_iterator vi,vi_end;
  128. boost::tie(vi,vi_end) = vertices(*i);
  129. for (;vi!=vi_end;++vi)
  130. {
  131. add_vertex(*vi,child);
  132. }
  133. }
  134. }
  135. ~subgraph() {
  136. for(typename ChildrenList::iterator i = m_children.begin();
  137. i != m_children.end(); ++i)
  138. {
  139. delete *i;
  140. }
  141. }
  142. // Return a null vertex descriptor for the graph.
  143. static vertex_descriptor null_vertex()
  144. { return Traits::null_vertex(); }
  145. // Create a subgraph
  146. subgraph<Graph>& create_subgraph() {
  147. m_children.push_back(new subgraph<Graph>());
  148. m_children.back()->m_parent = this;
  149. return *m_children.back();
  150. }
  151. // Create a subgraph with the specified vertex set.
  152. template <typename VertexIterator>
  153. subgraph<Graph>& create_subgraph(VertexIterator first, VertexIterator last) {
  154. m_children.push_back(new subgraph<Graph>());
  155. m_children.back()->m_parent = this;
  156. for(; first != last; ++first) {
  157. add_vertex(*first, *m_children.back());
  158. }
  159. return *m_children.back();
  160. }
  161. // local <-> global descriptor conversion functions
  162. vertex_descriptor local_to_global(vertex_descriptor u_local) const
  163. { return is_root() ? u_local : m_global_vertex[u_local]; }
  164. vertex_descriptor global_to_local(vertex_descriptor u_global) const {
  165. vertex_descriptor u_local; bool in_subgraph;
  166. if (is_root()) return u_global;
  167. boost::tie(u_local, in_subgraph) = this->find_vertex(u_global);
  168. BOOST_ASSERT(in_subgraph == true);
  169. return u_local;
  170. }
  171. edge_descriptor local_to_global(edge_descriptor e_local) const
  172. { return is_root() ? e_local : m_global_edge[get(get(edge_index, m_graph), e_local)]; }
  173. edge_descriptor global_to_local(edge_descriptor e_global) const
  174. { return is_root() ? e_global : (*m_local_edge.find(get(get(edge_index, root().m_graph), e_global))).second; }
  175. // Is vertex u (of the root graph) contained in this subgraph?
  176. // If so, return the matching local vertex.
  177. std::pair<vertex_descriptor, bool>
  178. find_vertex(vertex_descriptor u_global) const {
  179. if (is_root()) return std::make_pair(u_global, true);
  180. typename LocalVertexMap::const_iterator i = m_local_vertex.find(u_global);
  181. bool valid = i != m_local_vertex.end();
  182. return std::make_pair((valid ? (*i).second : null_vertex()), valid);
  183. }
  184. // Is edge e (of the root graph) contained in this subgraph?
  185. // If so, return the matching local edge.
  186. std::pair<edge_descriptor, bool>
  187. find_edge(edge_descriptor e_global) const {
  188. if (is_root()) return std::make_pair(e_global, true);
  189. typename LocalEdgeMap::const_iterator i =
  190. m_local_edge.find(get(get(edge_index, root().m_graph), e_global));
  191. bool valid = i != m_local_edge.end();
  192. return std::make_pair((valid ? (*i).second : edge_descriptor()), valid);
  193. }
  194. // Return the parent graph.
  195. subgraph& parent() { return *m_parent; }
  196. const subgraph& parent() const { return *m_parent; }
  197. // Return true if this is the root subgraph
  198. bool is_root() const { return m_parent == 0; }
  199. // Return the root graph of the subgraph tree.
  200. subgraph& root()
  201. { return is_root() ? *this : m_parent->root(); }
  202. const subgraph& root() const
  203. { return is_root() ? *this : m_parent->root(); }
  204. // Return the children subgraphs of this graph/subgraph.
  205. // Use a list of pointers because the VC++ std::list doesn't like
  206. // storing incomplete type.
  207. typedef indirect_iterator<
  208. typename ChildrenList::const_iterator
  209. , subgraph<Graph>
  210. , std::bidirectional_iterator_tag
  211. >
  212. children_iterator;
  213. typedef indirect_iterator<
  214. typename ChildrenList::const_iterator
  215. , subgraph<Graph> const
  216. , std::bidirectional_iterator_tag
  217. >
  218. const_children_iterator;
  219. std::pair<const_children_iterator, const_children_iterator> children() const {
  220. return std::make_pair(const_children_iterator(m_children.begin()),
  221. const_children_iterator(m_children.end()));
  222. }
  223. std::pair<children_iterator, children_iterator> children() {
  224. return std::make_pair(children_iterator(m_children.begin()),
  225. children_iterator(m_children.end()));
  226. }
  227. std::size_t num_children() const { return m_children.size(); }
  228. #ifndef BOOST_GRAPH_NO_BUNDLED_PROPERTIES
  229. // Defualt property access delegates the lookup to global properties.
  230. template <typename Descriptor>
  231. typename graph::detail::bundled_result<Graph, Descriptor>::type&
  232. operator[](Descriptor x)
  233. { return is_root() ? m_graph[x] : root().m_graph[local_to_global(x)]; }
  234. template <typename Descriptor>
  235. typename graph::detail::bundled_result<Graph, Descriptor>::type const&
  236. operator[](Descriptor x) const
  237. { return is_root() ? m_graph[x] : root().m_graph[local_to_global(x)]; }
  238. // Local property access returns the local property of the given descripor.
  239. template <typename Descriptor>
  240. typename graph::detail::bundled_result<Graph, Descriptor>::type&
  241. operator[](local_property<Descriptor> x)
  242. { return m_graph[x.value]; }
  243. template <typename Descriptor>
  244. typename graph::detail::bundled_result<Graph, Descriptor>::type const&
  245. operator[](local_property<Descriptor> x) const
  246. { return m_graph[x.value]; }
  247. // Global property access returns the global property associated with the
  248. // given descriptor. This is an alias for the default bundled property
  249. // access operations.
  250. template <typename Descriptor>
  251. typename graph::detail::bundled_result<Graph, Descriptor>::type&
  252. operator[](global_property<Descriptor> x)
  253. { return (*this)[x.value]; }
  254. template <typename Descriptor>
  255. typename graph::detail::bundled_result<Graph, Descriptor>::type const&
  256. operator[](global_property<Descriptor> x) const
  257. { return (*this)[x.value]; }
  258. #endif // BOOST_GRAPH_NO_BUNDLED_PROPERTIES
  259. // private:
  260. typedef typename property_map<Graph, edge_index_t>::type EdgeIndexMap;
  261. typedef typename property_traits<EdgeIndexMap>::value_type edge_index_type;
  262. BOOST_STATIC_ASSERT((!is_same<edge_index_type,
  263. boost::detail::error_property_not_found>::value));
  264. private:
  265. typedef std::vector<vertex_descriptor> GlobalVertexList;
  266. typedef std::vector<edge_descriptor> GlobalEdgeList;
  267. typedef std::map<vertex_descriptor, vertex_descriptor> LocalVertexMap;
  268. typedef std::map<edge_index_type, edge_descriptor> LocalEdgeMap;
  269. // TODO: Should the LocalVertexMap be: map<index_type, descriptor>?
  270. // TODO: Can we relax the indexing requirement if both descriptors are
  271. // LessThanComparable?
  272. // TODO: Should we really be using unorderd_map for improved lookup times?
  273. public: // Probably shouldn't be public....
  274. Graph m_graph;
  275. subgraph<Graph>* m_parent;
  276. edge_index_type m_edge_counter; // for generating unique edge indices
  277. ChildrenList m_children;
  278. GlobalVertexList m_global_vertex; // local -> global
  279. LocalVertexMap m_local_vertex; // global -> local
  280. GlobalEdgeList m_global_edge; // local -> global
  281. LocalEdgeMap m_local_edge; // global -> local
  282. edge_descriptor local_add_edge(vertex_descriptor u_local,
  283. vertex_descriptor v_local,
  284. edge_descriptor e_global)
  285. {
  286. edge_descriptor e_local;
  287. bool inserted;
  288. boost::tie(e_local, inserted) = add_edge(u_local, v_local, m_graph);
  289. put(edge_index, m_graph, e_local, m_edge_counter++);
  290. m_global_edge.push_back(e_global);
  291. m_local_edge[get(get(edge_index, this->root()), e_global)] = e_local;
  292. return e_local;
  293. }
  294. };
  295. template <typename Graph>
  296. struct vertex_bundle_type<subgraph<Graph> >
  297. : vertex_bundle_type<Graph>
  298. { };
  299. template<typename Graph>
  300. struct edge_bundle_type<subgraph<Graph> >
  301. : edge_bundle_type<Graph>
  302. { };
  303. template<typename Graph>
  304. struct graph_bundle_type<subgraph<Graph> >
  305. : graph_bundle_type<Graph>
  306. { };
  307. //===========================================================================
  308. // Functions special to the Subgraph Class
  309. template <typename G>
  310. typename subgraph<G>::vertex_descriptor
  311. add_vertex(typename subgraph<G>::vertex_descriptor u_global,
  312. subgraph<G>& g)
  313. {
  314. BOOST_ASSERT(!g.is_root());
  315. typename subgraph<G>::vertex_descriptor u_local, v_global;
  316. typename subgraph<G>::edge_descriptor e_global;
  317. u_local = add_vertex(g.m_graph);
  318. g.m_global_vertex.push_back(u_global);
  319. g.m_local_vertex[u_global] = u_local;
  320. subgraph<G>& r = g.root();
  321. // remember edge global and local maps
  322. {
  323. typename subgraph<G>::out_edge_iterator ei, ei_end;
  324. for (boost::tie(ei, ei_end) = out_edges(u_global, r);
  325. ei != ei_end; ++ei) {
  326. e_global = *ei;
  327. v_global = target(e_global, r);
  328. if (g.find_vertex(v_global).second == true)
  329. g.local_add_edge(u_local, g.global_to_local(v_global), e_global);
  330. }
  331. }
  332. if (is_directed(g)) { // not necessary for undirected graph
  333. typename subgraph<G>::vertex_iterator vi, vi_end;
  334. typename subgraph<G>::out_edge_iterator ei, ei_end;
  335. for(boost::tie(vi, vi_end) = vertices(r); vi != vi_end; ++vi) {
  336. v_global = *vi;
  337. if (v_global == u_global)
  338. continue; // don't insert self loops twice!
  339. if (!g.find_vertex(v_global).second)
  340. continue; // not a subgraph vertex => try next one
  341. for(boost::tie(ei, ei_end) = out_edges(*vi, r); ei != ei_end; ++ei) {
  342. e_global = *ei;
  343. if(target(e_global, r) == u_global) {
  344. g.local_add_edge(g.global_to_local(v_global), u_local, e_global);
  345. }
  346. }
  347. }
  348. }
  349. return u_local;
  350. }
  351. // NOTE: Descriptors are local unless otherwise noted.
  352. //===========================================================================
  353. // Functions required by the IncidenceGraph concept
  354. template <typename G>
  355. std::pair<typename graph_traits<G>::out_edge_iterator,
  356. typename graph_traits<G>::out_edge_iterator>
  357. out_edges(typename graph_traits<G>::vertex_descriptor v, const subgraph<G>& g)
  358. { return out_edges(v, g.m_graph); }
  359. template <typename G>
  360. typename graph_traits<G>::degree_size_type
  361. out_degree(typename graph_traits<G>::vertex_descriptor v, const subgraph<G>& g)
  362. { return out_degree(v, g.m_graph); }
  363. template <typename G>
  364. typename graph_traits<G>::vertex_descriptor
  365. source(typename graph_traits<G>::edge_descriptor e, const subgraph<G>& g)
  366. { return source(e, g.m_graph); }
  367. template <typename G>
  368. typename graph_traits<G>::vertex_descriptor
  369. target(typename graph_traits<G>::edge_descriptor e, const subgraph<G>& g)
  370. { return target(e, g.m_graph); }
  371. //===========================================================================
  372. // Functions required by the BidirectionalGraph concept
  373. template <typename G>
  374. std::pair<typename graph_traits<G>::in_edge_iterator,
  375. typename graph_traits<G>::in_edge_iterator>
  376. in_edges(typename graph_traits<G>::vertex_descriptor v, const subgraph<G>& g)
  377. { return in_edges(v, g.m_graph); }
  378. template <typename G>
  379. typename graph_traits<G>::degree_size_type
  380. in_degree(typename graph_traits<G>::vertex_descriptor v, const subgraph<G>& g)
  381. { return in_degree(v, g.m_graph); }
  382. template <typename G>
  383. typename graph_traits<G>::degree_size_type
  384. degree(typename graph_traits<G>::vertex_descriptor v, const subgraph<G>& g)
  385. { return degree(v, g.m_graph); }
  386. //===========================================================================
  387. // Functions required by the AdjacencyGraph concept
  388. template <typename G>
  389. std::pair<typename subgraph<G>::adjacency_iterator,
  390. typename subgraph<G>::adjacency_iterator>
  391. adjacent_vertices(typename subgraph<G>::vertex_descriptor v, const subgraph<G>& g)
  392. { return adjacent_vertices(v, g.m_graph); }
  393. //===========================================================================
  394. // Functions required by the VertexListGraph concept
  395. template <typename G>
  396. std::pair<typename subgraph<G>::vertex_iterator,
  397. typename subgraph<G>::vertex_iterator>
  398. vertices(const subgraph<G>& g)
  399. { return vertices(g.m_graph); }
  400. template <typename G>
  401. typename subgraph<G>::vertices_size_type
  402. num_vertices(const subgraph<G>& g)
  403. { return num_vertices(g.m_graph); }
  404. //===========================================================================
  405. // Functions required by the EdgeListGraph concept
  406. template <typename G>
  407. std::pair<typename subgraph<G>::edge_iterator,
  408. typename subgraph<G>::edge_iterator>
  409. edges(const subgraph<G>& g)
  410. { return edges(g.m_graph); }
  411. template <typename G>
  412. typename subgraph<G>::edges_size_type
  413. num_edges(const subgraph<G>& g)
  414. { return num_edges(g.m_graph); }
  415. //===========================================================================
  416. // Functions required by the AdjacencyMatrix concept
  417. template <typename G>
  418. std::pair<typename subgraph<G>::edge_descriptor, bool>
  419. edge(typename subgraph<G>::vertex_descriptor u,
  420. typename subgraph<G>::vertex_descriptor v,
  421. const subgraph<G>& g)
  422. { return edge(u, v, g.m_graph); }
  423. //===========================================================================
  424. // Functions required by the MutableGraph concept
  425. namespace detail {
  426. template <typename Vertex, typename Edge, typename Graph>
  427. void add_edge_recur_down(Vertex u_global, Vertex v_global, Edge e_global,
  428. subgraph<Graph>& g);
  429. template <typename Vertex, typename Edge, typename Children, typename G>
  430. void children_add_edge(Vertex u_global, Vertex v_global, Edge e_global,
  431. Children& c, subgraph<G>* orig)
  432. {
  433. for(typename Children::iterator i = c.begin(); i != c.end(); ++i) {
  434. if ((*i)->find_vertex(u_global).second &&
  435. (*i)->find_vertex(v_global).second)
  436. {
  437. add_edge_recur_down(u_global, v_global, e_global, **i, orig);
  438. }
  439. }
  440. }
  441. template <typename Vertex, typename Edge, typename Graph>
  442. void add_edge_recur_down(Vertex u_global, Vertex v_global, Edge e_global,
  443. subgraph<Graph>& g, subgraph<Graph>* orig)
  444. {
  445. if(&g != orig ) {
  446. // add local edge only if u_global and v_global are in subgraph g
  447. Vertex u_local, v_local;
  448. bool u_in_subgraph, v_in_subgraph;
  449. boost::tie(u_local, u_in_subgraph) = g.find_vertex(u_global);
  450. boost::tie(v_local, v_in_subgraph) = g.find_vertex(v_global);
  451. if(u_in_subgraph && v_in_subgraph) {
  452. g.local_add_edge(u_local, v_local, e_global);
  453. }
  454. }
  455. children_add_edge(u_global, v_global, e_global, g.m_children, orig);
  456. }
  457. template <typename Vertex, typename Graph>
  458. std::pair<typename subgraph<Graph>::edge_descriptor, bool>
  459. add_edge_recur_up(Vertex u_global, Vertex v_global,
  460. const typename Graph::edge_property_type& ep,
  461. subgraph<Graph>& g, subgraph<Graph>* orig)
  462. {
  463. if(g.is_root()) {
  464. typename subgraph<Graph>::edge_descriptor e_global;
  465. bool inserted;
  466. boost::tie(e_global, inserted) = add_edge(u_global, v_global, ep, g.m_graph);
  467. put(edge_index, g.m_graph, e_global, g.m_edge_counter++);
  468. g.m_global_edge.push_back(e_global);
  469. children_add_edge(u_global, v_global, e_global, g.m_children, orig);
  470. return std::make_pair(e_global, inserted);
  471. } else {
  472. return add_edge_recur_up(u_global, v_global, ep, *g.m_parent, orig);
  473. }
  474. }
  475. } // namespace detail
  476. // Add an edge to the subgraph g, specified by the local vertex descriptors u
  477. // and v. In addition, the edge will be added to any (all) other subgraphs that
  478. // contain vertex descriptors u and v.
  479. template <typename G>
  480. std::pair<typename subgraph<G>::edge_descriptor, bool>
  481. add_edge(typename subgraph<G>::vertex_descriptor u,
  482. typename subgraph<G>::vertex_descriptor v,
  483. const typename G::edge_property_type& ep,
  484. subgraph<G>& g)
  485. {
  486. if (g.is_root()) {
  487. // u and v are really global
  488. return detail::add_edge_recur_up(u, v, ep, g, &g);
  489. } else {
  490. typename subgraph<G>::edge_descriptor e_local, e_global;
  491. bool inserted;
  492. boost::tie(e_global, inserted) =
  493. detail::add_edge_recur_up(g.local_to_global(u),
  494. g.local_to_global(v),
  495. ep, g, &g);
  496. e_local = g.local_add_edge(u, v, e_global);
  497. return std::make_pair(e_local, inserted);
  498. }
  499. }
  500. template <typename G>
  501. std::pair<typename subgraph<G>::edge_descriptor, bool>
  502. add_edge(typename subgraph<G>::vertex_descriptor u,
  503. typename subgraph<G>::vertex_descriptor v,
  504. subgraph<G>& g)
  505. { return add_edge(u, v, typename G::edge_property_type(), g); }
  506. namespace detail {
  507. //-------------------------------------------------------------------------
  508. // implementation of remove_edge(u,v,g)
  509. template <typename Vertex, typename Graph>
  510. void remove_edge_recur_down(Vertex u_global, Vertex v_global,
  511. subgraph<Graph>& g);
  512. template <typename Vertex, typename Children>
  513. void children_remove_edge(Vertex u_global, Vertex v_global,
  514. Children& c)
  515. {
  516. for(typename Children::iterator i = c.begin(); i != c.end(); ++i) {
  517. if((*i)->find_vertex(u_global).second &&
  518. (*i)->find_vertex(v_global).second)
  519. {
  520. remove_edge_recur_down(u_global, v_global, **i);
  521. }
  522. }
  523. }
  524. template <typename Vertex, typename Graph>
  525. void remove_edge_recur_down(Vertex u_global, Vertex v_global,
  526. subgraph<Graph>& g)
  527. {
  528. Vertex u_local, v_local;
  529. u_local = g.m_local_vertex[u_global];
  530. v_local = g.m_local_vertex[v_global];
  531. remove_edge(u_local, v_local, g.m_graph);
  532. children_remove_edge(u_global, v_global, g.m_children);
  533. }
  534. template <typename Vertex, typename Graph>
  535. void remove_edge_recur_up(Vertex u_global, Vertex v_global,
  536. subgraph<Graph>& g)
  537. {
  538. if(g.is_root()) {
  539. remove_edge(u_global, v_global, g.m_graph);
  540. children_remove_edge(u_global, v_global, g.m_children);
  541. } else {
  542. remove_edge_recur_up(u_global, v_global, *g.m_parent);
  543. }
  544. }
  545. //-------------------------------------------------------------------------
  546. // implementation of remove_edge(e,g)
  547. template <typename G, typename Edge, typename Children>
  548. void children_remove_edge(Edge e_global, Children& c)
  549. {
  550. for(typename Children::iterator i = c.begin(); i != c.end(); ++i) {
  551. std::pair<typename subgraph<G>::edge_descriptor, bool> found =
  552. (*i)->find_edge(e_global);
  553. if (!found.second) {
  554. continue;
  555. }
  556. children_remove_edge<G>(e_global, (*i)->m_children);
  557. remove_edge(found.first, (*i)->m_graph);
  558. }
  559. }
  560. } // namespace detail
  561. template <typename G>
  562. void
  563. remove_edge(typename subgraph<G>::vertex_descriptor u,
  564. typename subgraph<G>::vertex_descriptor v,
  565. subgraph<G>& g)
  566. {
  567. if(g.is_root()) {
  568. detail::remove_edge_recur_up(u, v, g);
  569. } else {
  570. detail::remove_edge_recur_up(g.local_to_global(u),
  571. g.local_to_global(v), g);
  572. }
  573. }
  574. template <typename G>
  575. void
  576. remove_edge(typename subgraph<G>::edge_descriptor e, subgraph<G>& g)
  577. {
  578. typename subgraph<G>::edge_descriptor e_global = g.local_to_global(e);
  579. #ifndef NDEBUG
  580. std::pair<typename subgraph<G>::edge_descriptor, bool> fe = g.find_edge(e_global);
  581. BOOST_ASSERT(fe.second && fe.first == e);
  582. #endif //NDEBUG
  583. subgraph<G> &root = g.root(); // chase to root
  584. detail::children_remove_edge<G>(e_global, root.m_children);
  585. remove_edge(e_global, root.m_graph); // kick edge from root
  586. }
  587. // This is slow, but there may not be a good way to do it safely otherwise
  588. template <typename Predicate, typename G>
  589. void
  590. remove_edge_if(Predicate p, subgraph<G>& g) {
  591. while (true) {
  592. bool any_removed = false;
  593. typedef typename subgraph<G>::edge_iterator ei_type;
  594. for (std::pair<ei_type, ei_type> ep = edges(g);
  595. ep.first != ep.second; ++ep.first) {
  596. if (p(*ep.first)) {
  597. any_removed = true;
  598. remove_edge(*ep.first, g);
  599. break; /* Since iterators may be invalidated */
  600. }
  601. }
  602. if (!any_removed) break;
  603. }
  604. }
  605. template <typename G>
  606. void
  607. clear_vertex(typename subgraph<G>::vertex_descriptor v, subgraph<G>& g) {
  608. while (true) {
  609. typedef typename subgraph<G>::out_edge_iterator oei_type;
  610. std::pair<oei_type, oei_type> p = out_edges(v, g);
  611. if (p.first == p.second) break;
  612. remove_edge(*p.first, g);
  613. }
  614. }
  615. namespace detail {
  616. template <typename G>
  617. typename subgraph<G>::vertex_descriptor
  618. add_vertex_recur_up(subgraph<G>& g)
  619. {
  620. typename subgraph<G>::vertex_descriptor u_local, u_global;
  621. if (g.is_root()) {
  622. u_global = add_vertex(g.m_graph);
  623. g.m_global_vertex.push_back(u_global);
  624. } else {
  625. u_global = add_vertex_recur_up(*g.m_parent);
  626. u_local = add_vertex(g.m_graph);
  627. g.m_global_vertex.push_back(u_global);
  628. g.m_local_vertex[u_global] = u_local;
  629. }
  630. return u_global;
  631. }
  632. } // namespace detail
  633. template <typename G>
  634. typename subgraph<G>::vertex_descriptor
  635. add_vertex(subgraph<G>& g)
  636. {
  637. typename subgraph<G>::vertex_descriptor u_local, u_global;
  638. if(g.is_root()) {
  639. u_global = add_vertex(g.m_graph);
  640. g.m_global_vertex.push_back(u_global);
  641. u_local = u_global;
  642. } else {
  643. u_global = detail::add_vertex_recur_up(g.parent());
  644. u_local = add_vertex(g.m_graph);
  645. g.m_global_vertex.push_back(u_global);
  646. g.m_local_vertex[u_global] = u_local;
  647. }
  648. return u_local;
  649. }
  650. #if 0
  651. // TODO: Under Construction
  652. template <typename G>
  653. void remove_vertex(typename subgraph<G>::vertex_descriptor u, subgraph<G>& g)
  654. { BOOST_ASSERT(false); }
  655. #endif
  656. //===========================================================================
  657. // Functions required by the PropertyGraph concept
  658. /**
  659. * The global property map returns the global properties associated with local
  660. * descriptors.
  661. */
  662. template <typename GraphPtr, typename PropertyMap, typename Tag>
  663. class subgraph_global_property_map
  664. : public put_get_helper<
  665. typename property_traits<PropertyMap>::reference,
  666. subgraph_global_property_map<GraphPtr, PropertyMap, Tag>
  667. >
  668. {
  669. typedef property_traits<PropertyMap> Traits;
  670. public:
  671. typedef typename mpl::if_<is_const<typename remove_pointer<GraphPtr>::type>,
  672. readable_property_map_tag,
  673. typename Traits::category>::type
  674. category;
  675. typedef typename Traits::value_type value_type;
  676. typedef typename Traits::key_type key_type;
  677. typedef typename Traits::reference reference;
  678. subgraph_global_property_map()
  679. { }
  680. subgraph_global_property_map(GraphPtr g, Tag tag)
  681. : m_g(g), m_tag(tag)
  682. { }
  683. reference operator[](key_type e) const {
  684. PropertyMap pmap = get(m_tag, m_g->root().m_graph);
  685. return m_g->is_root()
  686. ? pmap[e]
  687. : pmap[m_g->local_to_global(e)];
  688. }
  689. GraphPtr m_g;
  690. Tag m_tag;
  691. };
  692. /**
  693. * The local property map returns the local property associated with the local
  694. * descriptors.
  695. */
  696. template <typename GraphPtr, typename PropertyMap, typename Tag>
  697. class subgraph_local_property_map
  698. : public put_get_helper<
  699. typename property_traits<PropertyMap>::reference,
  700. subgraph_local_property_map<GraphPtr, PropertyMap, Tag>
  701. >
  702. {
  703. typedef property_traits<PropertyMap> Traits;
  704. public:
  705. typedef typename mpl::if_<is_const<typename remove_pointer<GraphPtr>::type>,
  706. readable_property_map_tag,
  707. typename Traits::category>::type
  708. category;
  709. typedef typename Traits::value_type value_type;
  710. typedef typename Traits::key_type key_type;
  711. typedef typename Traits::reference reference;
  712. typedef Tag tag;
  713. typedef PropertyMap pmap;
  714. subgraph_local_property_map()
  715. { }
  716. subgraph_local_property_map(GraphPtr g, Tag tag)
  717. : m_g(g), m_tag(tag)
  718. { }
  719. reference operator[](key_type e) const {
  720. // Get property map on the underlying graph.
  721. PropertyMap pmap = get(m_tag, m_g->m_graph);
  722. return pmap[e];
  723. }
  724. GraphPtr m_g;
  725. Tag m_tag;
  726. };
  727. namespace detail {
  728. // Extract the actual tags from local or global property maps so we don't
  729. // try to find non-properties.
  730. template <typename P> struct extract_lg_tag { typedef P type; };
  731. template <typename P> struct extract_lg_tag< local_property<P> > {
  732. typedef P type;
  733. };
  734. template <typename P> struct extract_lg_tag< global_property<P> > {
  735. typedef P type;
  736. };
  737. // NOTE: Mysterious Property template parameter unused in both metafunction
  738. // classes.
  739. struct subgraph_global_pmap {
  740. template <class Tag, class SubGraph, class Property>
  741. struct bind_ {
  742. typedef typename SubGraph::graph_type Graph;
  743. typedef SubGraph* SubGraphPtr;
  744. typedef const SubGraph* const_SubGraphPtr;
  745. typedef typename extract_lg_tag<Tag>::type TagType;
  746. typedef typename property_map<Graph, TagType>::type PMap;
  747. typedef typename property_map<Graph, TagType>::const_type const_PMap;
  748. public:
  749. typedef subgraph_global_property_map<SubGraphPtr, PMap, TagType> type;
  750. typedef subgraph_global_property_map<const_SubGraphPtr, const_PMap, TagType>
  751. const_type;
  752. };
  753. };
  754. struct subgraph_local_pmap {
  755. template <class Tag, class SubGraph, class Property>
  756. struct bind_ {
  757. typedef typename SubGraph::graph_type Graph;
  758. typedef SubGraph* SubGraphPtr;
  759. typedef const SubGraph* const_SubGraphPtr;
  760. typedef typename extract_lg_tag<Tag>::type TagType;
  761. typedef typename property_map<Graph, TagType>::type PMap;
  762. typedef typename property_map<Graph, TagType>::const_type const_PMap;
  763. public:
  764. typedef subgraph_local_property_map<SubGraphPtr, PMap, TagType> type;
  765. typedef subgraph_local_property_map<const_SubGraphPtr, const_PMap, TagType>
  766. const_type;
  767. };
  768. };
  769. // These metafunctions select the corresponding metafunctions above, and
  770. // are used by the choose_pmap metafunction below to specialize the choice
  771. // of local/global property map. By default, we defer to the global
  772. // property.
  773. template <class Tag>
  774. struct subgraph_choose_pmap_helper {
  775. typedef subgraph_global_pmap type;
  776. };
  777. template <class Tag>
  778. struct subgraph_choose_pmap_helper< local_property<Tag> > {
  779. typedef subgraph_local_pmap type;
  780. };
  781. template <class Tag>
  782. struct subgraph_choose_pmap_helper< global_property<Tag> > {
  783. typedef subgraph_global_pmap type;
  784. };
  785. // As above, unless we're requesting vertex_index_t. Then it's always a
  786. // local property map. This enables the correct translation of descriptors
  787. // between local and global layers.
  788. template <>
  789. struct subgraph_choose_pmap_helper<vertex_index_t> {
  790. typedef subgraph_local_pmap type;
  791. };
  792. template <>
  793. struct subgraph_choose_pmap_helper< local_property<vertex_index_t> > {
  794. typedef subgraph_local_pmap type;
  795. };
  796. template <>
  797. struct subgraph_choose_pmap_helper< global_property<vertex_index_t> > {
  798. typedef subgraph_local_pmap type;
  799. };
  800. // Determine the kind of property. If SameType<Tag, vertex_index_t>, then
  801. // the property lookup is always local. Otherwise, the lookup is global.
  802. // NOTE: Property parameter is basically unused.
  803. template <class Tag, class Graph, class Property>
  804. struct subgraph_choose_pmap {
  805. typedef typename subgraph_choose_pmap_helper<Tag>::type Helper;
  806. typedef typename Helper::template bind_<Tag, Graph, Property> Bind;
  807. typedef typename Bind::type type;
  808. typedef typename Bind::const_type const_type;
  809. };
  810. // Used by the vertex/edge property selectors to determine the kind(s) of
  811. // property maps used by the property_map type generator.
  812. struct subgraph_property_generator {
  813. template <class SubGraph, class Property, class Tag>
  814. struct bind_ {
  815. typedef subgraph_choose_pmap<Tag, SubGraph, Property> Choice;
  816. typedef typename Choice::type type;
  817. typedef typename Choice::const_type const_type;
  818. };
  819. };
  820. } // namespace detail
  821. template <>
  822. struct vertex_property_selector<subgraph_tag> {
  823. typedef detail::subgraph_property_generator type;
  824. };
  825. template <>
  826. struct edge_property_selector<subgraph_tag> {
  827. typedef detail::subgraph_property_generator type;
  828. };
  829. // ==================================================
  830. // get(p, g), get(p, g, k), and put(p, g, k, v)
  831. // ==================================================
  832. template <typename G, typename Property>
  833. typename property_map<subgraph<G>, Property>::type
  834. get(Property p, subgraph<G>& g) {
  835. typedef typename property_map< subgraph<G>, Property>::type PMap;
  836. return PMap(&g, p);
  837. }
  838. template <typename G, typename Property>
  839. typename property_map<subgraph<G>, Property>::const_type
  840. get(Property p, const subgraph<G>& g) {
  841. typedef typename property_map< subgraph<G>, Property>::const_type PMap;
  842. return PMap(&g, p);
  843. }
  844. template <typename G, typename Property, typename Key>
  845. typename property_traits<
  846. typename property_map<subgraph<G>, Property>::const_type
  847. >::value_type
  848. get(Property p, const subgraph<G>& g, const Key& k) {
  849. typedef typename property_map< subgraph<G>, Property>::const_type PMap;
  850. PMap pmap(&g, p);
  851. return pmap[k];
  852. }
  853. template <typename G, typename Property, typename Key, typename Value>
  854. void put(Property p, subgraph<G>& g, const Key& k, const Value& val) {
  855. typedef typename property_map< subgraph<G>, Property>::type PMap;
  856. PMap pmap(&g, p);
  857. pmap[k] = val;
  858. }
  859. // ==================================================
  860. // get(global(p), g)
  861. // NOTE: get(global(p), g, k) and put(global(p), g, k, v) not supported
  862. // ==================================================
  863. template <typename G, typename Property>
  864. typename property_map<subgraph<G>, global_property<Property> >::type
  865. get(global_property<Property> p, subgraph<G>& g) {
  866. typedef typename property_map<
  867. subgraph<G>, global_property<Property>
  868. >::type Map;
  869. return Map(&g, p.value);
  870. }
  871. template <typename G, typename Property>
  872. typename property_map<subgraph<G>, global_property<Property> >::const_type
  873. get(global_property<Property> p, const subgraph<G>& g) {
  874. typedef typename property_map<
  875. subgraph<G>, global_property<Property>
  876. >::const_type Map;
  877. return Map(&g, p.value);
  878. }
  879. // ==================================================
  880. // get(local(p), g)
  881. // NOTE: get(local(p), g, k) and put(local(p), g, k, v) not supported
  882. // ==================================================
  883. template <typename G, typename Property>
  884. typename property_map<subgraph<G>, local_property<Property> >::type
  885. get(local_property<Property> p, subgraph<G>& g) {
  886. typedef typename property_map<
  887. subgraph<G>, local_property<Property>
  888. >::type Map;
  889. return Map(&g, p.value);
  890. }
  891. template <typename G, typename Property>
  892. typename property_map<subgraph<G>, local_property<Property> >::const_type
  893. get(local_property<Property> p, const subgraph<G>& g) {
  894. typedef typename property_map<
  895. subgraph<G>, local_property<Property>
  896. >::const_type Map;
  897. return Map(&g, p.value);
  898. }
  899. template <typename G, typename Tag>
  900. inline typename graph_property<G, Tag>::type&
  901. get_property(subgraph<G>& g, Tag tag) {
  902. return get_property(g.m_graph, tag);
  903. }
  904. template <typename G, typename Tag>
  905. inline const typename graph_property<G, Tag>::type&
  906. get_property(const subgraph<G>& g, Tag tag) {
  907. return get_property(g.m_graph, tag);
  908. }
  909. //===========================================================================
  910. // Miscellaneous Functions
  911. template <typename G>
  912. typename subgraph<G>::vertex_descriptor
  913. vertex(typename subgraph<G>::vertices_size_type n, const subgraph<G>& g)
  914. { return vertex(n, g.m_graph); }
  915. //===========================================================================
  916. // Mutability Traits
  917. // Just pull the mutability traits form the underlying graph. Note that this
  918. // will probably fail (badly) for labeled graphs.
  919. template <typename G>
  920. struct graph_mutability_traits< subgraph<G> > {
  921. typedef typename graph_mutability_traits<G>::category category;
  922. };
  923. } // namespace boost
  924. #endif // BOOST_SUBGRAPH_HPP