graph_communicator.hpp 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575
  1. // Copyright (C) 2007 Trustees of Indiana University
  2. // Authors: Douglas Gregor
  3. // Andrew Lumsdaine
  4. // Use, modification and distribution is subject to the Boost Software
  5. // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt)
  7. /** @file graph_communicator.hpp
  8. *
  9. * This header defines facilities to support MPI communicators with
  10. * graph topologies, using the graph interface defined by the Boost
  11. * Graph Library. One can construct a communicator whose topology is
  12. * described by any graph meeting the requirements of the Boost Graph
  13. * Library's graph concepts. Likewise, any communicator that has a
  14. * graph topology can be viewed as a graph by the Boost Graph
  15. * Library, permitting one to use the BGL's graph algorithms on the
  16. * process topology.
  17. */
  18. #ifndef BOOST_MPI_GRAPH_COMMUNICATOR_HPP
  19. #define BOOST_MPI_GRAPH_COMMUNICATOR_HPP
  20. #include <boost/mpi/communicator.hpp>
  21. #include <vector>
  22. #include <utility>
  23. // Headers required to implement graph topologies
  24. #include <boost/graph/graph_traits.hpp>
  25. #include <boost/graph/properties.hpp>
  26. #include <boost/property_map/property_map.hpp>
  27. #include <boost/iterator/counting_iterator.hpp>
  28. #include <boost/graph/iteration_macros.hpp>
  29. #include <boost/shared_array.hpp>
  30. #include <boost/assert.hpp>
  31. namespace boost { namespace mpi {
  32. /**
  33. * @brief An MPI communicator with a graph topology.
  34. *
  35. * A @c graph_communicator is a communicator whose topology is
  36. * expressed as a graph. Graph communicators have the same
  37. * functionality as (intra)communicators, but also allow one to query
  38. * the relationships among processes. Those relationships are
  39. * expressed via a graph, using the interface defined by the Boost
  40. * Graph Library. The @c graph_communicator class meets the
  41. * requirements of the BGL Graph, Incidence Graph, Adjacency Graph,
  42. * Vertex List Graph, and Edge List Graph concepts.
  43. */
  44. class BOOST_MPI_DECL graph_communicator : public communicator
  45. {
  46. friend class communicator;
  47. /**
  48. * INTERNAL ONLY
  49. *
  50. * Construct a graph communicator given a shared pointer to the
  51. * underlying MPI_Comm. This operation is used for "casting" from a
  52. * communicator to a graph communicator.
  53. */
  54. explicit graph_communicator(const shared_ptr<MPI_Comm>& comm_ptr)
  55. {
  56. #ifndef BOOST_DISABLE_ASSERTS
  57. int status;
  58. BOOST_MPI_CHECK_RESULT(MPI_Topo_test, ((MPI_Comm)*this, &status));
  59. BOOST_ASSERT(status == MPI_GRAPH);
  60. #endif
  61. this->comm_ptr = comm_ptr;
  62. }
  63. public:
  64. /**
  65. * Build a new Boost.MPI graph communicator based on the MPI
  66. * communicator @p comm with graph topology.
  67. *
  68. * @p comm may be any valid MPI communicator. If @p comm is
  69. * MPI_COMM_NULL, an empty communicator (that cannot be used for
  70. * communication) is created and the @p kind parameter is
  71. * ignored. Otherwise, the @p kind parameter determines how the
  72. * Boost.MPI communicator will be related to @p comm:
  73. *
  74. * - If @p kind is @c comm_duplicate, duplicate @c comm to create
  75. * a new communicator. This new communicator will be freed when
  76. * the Boost.MPI communicator (and all copies of it) is
  77. * destroyed. This option is only permitted if the underlying MPI
  78. * implementation supports MPI 2.0; duplication of
  79. * intercommunicators is not available in MPI 1.x.
  80. *
  81. * - If @p kind is @c comm_take_ownership, take ownership of @c
  82. * comm. It will be freed automatically when all of the Boost.MPI
  83. * communicators go out of scope.
  84. *
  85. * - If @p kind is @c comm_attach, this Boost.MPI communicator
  86. * will reference the existing MPI communicator @p comm but will
  87. * not free @p comm when the Boost.MPI communicator goes out of
  88. * scope. This option should only be used when the communicator is
  89. * managed by the user.
  90. */
  91. graph_communicator(const MPI_Comm& comm, comm_create_kind kind)
  92. : communicator(comm, kind)
  93. {
  94. #ifndef BOOST_DISABLE_ASSERTS
  95. int status;
  96. BOOST_MPI_CHECK_RESULT(MPI_Topo_test, ((MPI_Comm)*this, &status));
  97. BOOST_ASSERT(status == MPI_GRAPH);
  98. #endif
  99. }
  100. /**
  101. * Create a new communicator whose topology is described by the
  102. * given graph. The indices of the vertices in the graph will be
  103. * assumed to be the ranks of the processes within the
  104. * communicator. There may be fewer vertices in the graph than
  105. * there are processes in the communicator; in this case, the
  106. * resulting communicator will be a NULL communicator.
  107. *
  108. * @param comm The communicator that the new, graph communicator
  109. * will be based on.
  110. *
  111. * @param graph Any type that meets the requirements of the
  112. * Incidence Graph and Vertex List Graph concepts from the Boost Graph
  113. * Library. This structure of this graph will become the topology
  114. * of the communicator that is returned.
  115. *
  116. * @param reorder Whether MPI is permitted to re-order the process
  117. * ranks within the returned communicator, to better optimize
  118. * communication. If false, the ranks of each process in the
  119. * returned process will match precisely the rank of that process
  120. * within the original communicator.
  121. */
  122. template<typename Graph>
  123. explicit
  124. graph_communicator(const communicator& comm, const Graph& graph,
  125. bool reorder = false);
  126. /**
  127. * Create a new communicator whose topology is described by the
  128. * given graph. The rank map (@p rank) gives the mapping from
  129. * vertices in the graph to ranks within the communicator. There
  130. * may be fewer vertices in the graph than there are processes in
  131. * the communicator; in this case, the resulting communicator will
  132. * be a NULL communicator.
  133. *
  134. * @param comm The communicator that the new, graph communicator
  135. * will be based on. The ranks in @c rank refer to the processes in
  136. * this communicator.
  137. *
  138. * @param graph Any type that meets the requirements of the
  139. * Incidence Graph and Vertex List Graph concepts from the Boost Graph
  140. * Library. This structure of this graph will become the topology
  141. * of the communicator that is returned.
  142. *
  143. * @param rank This map translates vertices in the @c graph into
  144. * ranks within the current communicator. It must be a Readable
  145. * Property Map (see the Boost Property Map library) whose key type
  146. * is the vertex type of the @p graph and whose value type is @c
  147. * int.
  148. *
  149. * @param reorder Whether MPI is permitted to re-order the process
  150. * ranks within the returned communicator, to better optimize
  151. * communication. If false, the ranks of each process in the
  152. * returned process will match precisely the rank of that process
  153. * within the original communicator.
  154. */
  155. template<typename Graph, typename RankMap>
  156. explicit
  157. graph_communicator(const communicator& comm, const Graph& graph,
  158. RankMap rank, bool reorder = false);
  159. protected:
  160. /**
  161. * INTERNAL ONLY
  162. *
  163. * Used by the constructors to create the new communicator with a
  164. * graph topology.
  165. */
  166. template<typename Graph, typename RankMap>
  167. void
  168. setup_graph(const communicator& comm, const Graph& graph, RankMap rank,
  169. bool reorder);
  170. };
  171. /****************************************************************************
  172. * Implementation Details *
  173. ****************************************************************************/
  174. template<typename Graph>
  175. graph_communicator::graph_communicator(const communicator& comm,
  176. const Graph& graph,
  177. bool reorder)
  178. {
  179. this->setup_graph(comm, graph, get(vertex_index, graph), reorder);
  180. }
  181. template<typename Graph, typename RankMap>
  182. graph_communicator::graph_communicator(const communicator& comm,
  183. const Graph& graph,
  184. RankMap rank, bool reorder)
  185. {
  186. this->setup_graph(comm, graph, rank, reorder);
  187. }
  188. template<typename Graph, typename RankMap>
  189. void
  190. graph_communicator::setup_graph(const communicator& comm, const Graph& graph,
  191. RankMap rank, bool reorder)
  192. {
  193. typedef typename graph_traits<Graph>::vertex_descriptor vertex_descriptor;
  194. // Build a mapping from ranks to vertices
  195. std::vector<vertex_descriptor> vertex_with_rank(num_vertices(graph));
  196. if (vertex_with_rank.empty())
  197. return;
  198. BGL_FORALL_VERTICES_T(v, graph, Graph)
  199. vertex_with_rank[get(rank, v)] = v;
  200. // Build the representation of the graph required by
  201. // MPI_Graph_create.
  202. std::vector<int> indices(num_vertices(graph));
  203. std::vector<int> edges;
  204. int nvertices = indices.size();
  205. for (int vertex_index = 0; vertex_index < nvertices; ++vertex_index) {
  206. vertex_descriptor v = vertex_with_rank[vertex_index];
  207. BGL_FORALL_OUTEDGES_T(v, e, graph, Graph)
  208. edges.push_back(get(rank, target(e, graph)));
  209. indices[vertex_index] = edges.size();
  210. }
  211. // Create the new communicator
  212. MPI_Comm newcomm;
  213. BOOST_MPI_CHECK_RESULT(MPI_Graph_create,
  214. ((MPI_Comm)comm,
  215. nvertices,
  216. &indices[0],
  217. edges.empty()? (int*)0 : &edges[0],
  218. reorder,
  219. &newcomm));
  220. this->comm_ptr.reset(new MPI_Comm(newcomm), comm_free());
  221. }
  222. /****************************************************************************
  223. * Communicator with Graph Topology as BGL Graph *
  224. ****************************************************************************/
  225. namespace detail {
  226. /**
  227. * INTERNAL ONLY
  228. *
  229. * The iterator used to access the outgoing edges within a
  230. * communicator's graph topology.
  231. */
  232. class comm_out_edge_iterator
  233. : public iterator_facade<comm_out_edge_iterator,
  234. std::pair<int, int>,
  235. random_access_traversal_tag,
  236. const std::pair<int, int>&,
  237. int>
  238. {
  239. public:
  240. comm_out_edge_iterator() { }
  241. comm_out_edge_iterator(int source, shared_array<int> neighbors, int index)
  242. : edge(source, -1), neighbors(neighbors), index(index) { }
  243. protected:
  244. friend class boost::iterator_core_access;
  245. const std::pair<int, int>& dereference() const
  246. {
  247. edge.second = neighbors[index];
  248. return edge;
  249. }
  250. bool equal(const comm_out_edge_iterator& other) const
  251. {
  252. return (edge.first == other.edge.first
  253. && index == other.index);
  254. }
  255. void increment() { ++index; }
  256. void decrement() { --index; }
  257. void advance(int n) { index += n; }
  258. int distance_to(const comm_out_edge_iterator& other) const
  259. {
  260. return other.index - index;
  261. }
  262. mutable std::pair<int, int> edge;
  263. shared_array<int> neighbors;
  264. int index;
  265. };
  266. /**
  267. * INTERNAL ONLY
  268. *
  269. * The iterator used to access the adjacent vertices within a
  270. * communicator's graph topology.
  271. */
  272. class comm_adj_iterator
  273. : public iterator_facade<comm_adj_iterator,
  274. int,
  275. random_access_traversal_tag,
  276. int,
  277. int>
  278. {
  279. public:
  280. comm_adj_iterator() { }
  281. comm_adj_iterator(shared_array<int> neighbors, int index)
  282. : neighbors(neighbors), index(index) { }
  283. protected:
  284. friend class boost::iterator_core_access;
  285. int dereference() const { return neighbors[index]; }
  286. bool equal(const comm_adj_iterator& other) const
  287. {
  288. return (neighbors == other.neighbors
  289. && index == other.index);
  290. }
  291. void increment() { ++index; }
  292. void decrement() { --index; }
  293. void advance(int n) { index += n; }
  294. int distance_to(const comm_adj_iterator& other) const
  295. {
  296. return other.index - index;
  297. }
  298. shared_array<int> neighbors;
  299. int index;
  300. };
  301. /**
  302. * INTERNAL ONLY
  303. *
  304. * The iterator used to access the edges in a communicator's graph
  305. * topology.
  306. */
  307. class comm_edge_iterator
  308. : public iterator_facade<comm_edge_iterator,
  309. std::pair<int, int>,
  310. forward_traversal_tag,
  311. const std::pair<int, int>&,
  312. int>
  313. {
  314. public:
  315. comm_edge_iterator() { }
  316. /// Constructor for a past-the-end iterator
  317. comm_edge_iterator(int nedges) : edge_index(nedges) { }
  318. comm_edge_iterator(shared_array<int> indices, shared_array<int> edges)
  319. : indices(indices), edges(edges), edge_index(0), edge(0, 0)
  320. { }
  321. protected:
  322. friend class boost::iterator_core_access;
  323. const std::pair<int, int>& dereference() const
  324. {
  325. while (edge_index == indices[edge.first])
  326. ++edge.first;
  327. edge.second = edges[edge_index];
  328. return edge;
  329. }
  330. bool equal(const comm_edge_iterator& other) const
  331. {
  332. return edge_index == other.edge_index;
  333. }
  334. void increment()
  335. {
  336. ++edge_index;
  337. }
  338. shared_array<int> indices;
  339. shared_array<int> edges;
  340. int edge_index;
  341. mutable std::pair<int, int> edge;
  342. };
  343. } // end namespace detail
  344. // Incidence Graph requirements
  345. /**
  346. * @brief Returns the source vertex from an edge in the graph topology
  347. * of a communicator.
  348. */
  349. inline int source(const std::pair<int, int>& edge, const graph_communicator&)
  350. {
  351. return edge.first;
  352. }
  353. /**
  354. * @brief Returns the target vertex from an edge in the graph topology
  355. * of a communicator.
  356. */
  357. inline int target(const std::pair<int, int>& edge, const graph_communicator&)
  358. {
  359. return edge.second;
  360. }
  361. /**
  362. * @brief Returns an iterator range containing all of the edges
  363. * outgoing from the given vertex in a graph topology of a
  364. * communicator.
  365. */
  366. std::pair<detail::comm_out_edge_iterator, detail::comm_out_edge_iterator>
  367. out_edges(int vertex, const graph_communicator& comm);
  368. /**
  369. * @brief Returns the out-degree of a vertex in the graph topology of
  370. * a communicator.
  371. */
  372. int out_degree(int vertex, const graph_communicator& comm);
  373. // Adjacency Graph requirements
  374. /**
  375. * @brief Returns an iterator range containing all of the neighbors of
  376. * the given vertex in the communicator's graph topology.
  377. */
  378. std::pair<detail::comm_adj_iterator, detail::comm_adj_iterator>
  379. adjacent_vertices(int vertex, const graph_communicator& comm);
  380. // Vertex List Graph requirements
  381. /**
  382. * @brief Returns an iterator range that contains all of the vertices
  383. * with the communicator's graph topology, i.e., all of the process
  384. * ranks in the communicator.
  385. */
  386. inline std::pair<counting_iterator<int>, counting_iterator<int> >
  387. vertices(const graph_communicator& comm)
  388. {
  389. return std::make_pair(counting_iterator<int>(0),
  390. counting_iterator<int>(comm.size()));
  391. }
  392. /**
  393. * @brief Returns the number of vertices within the graph topology of
  394. * the communicator, i.e., the number of processes in the
  395. * communicator.
  396. */
  397. inline int num_vertices(const graph_communicator& comm) { return comm.size(); }
  398. // Edge List Graph requirements
  399. /**
  400. * @brief Returns an iterator range that contains all of the edges
  401. * with the communicator's graph topology.
  402. */
  403. std::pair<detail::comm_edge_iterator, detail::comm_edge_iterator>
  404. edges(const graph_communicator& comm);
  405. /**
  406. * @brief Returns the number of edges in the communicator's graph
  407. * topology.
  408. */
  409. int num_edges(const graph_communicator& comm);
  410. // Property Graph requirements
  411. /**
  412. * @brief Returns a property map that maps from vertices in a
  413. * communicator's graph topology to their index values.
  414. *
  415. * Since the vertices are ranks in the communicator, the returned
  416. * property map is the identity property map.
  417. */
  418. inline identity_property_map get(vertex_index_t, const graph_communicator&)
  419. {
  420. return identity_property_map();
  421. }
  422. /**
  423. * @brief Returns the index of a vertex in the communicator's graph
  424. * topology.
  425. *
  426. * Since the vertices are ranks in the communicator, this is the
  427. * identity function.
  428. */
  429. inline int get(vertex_index_t, const graph_communicator&, int vertex)
  430. {
  431. return vertex;
  432. }
  433. } } // end namespace boost::mpi
  434. namespace boost {
  435. /**
  436. * @brief Traits structure that allows a communicator with graph
  437. * topology to be view as a graph by the Boost Graph Library.
  438. *
  439. * The specialization of @c graph_traits for an MPI communicator
  440. * allows a communicator with graph topology to be viewed as a
  441. * graph. An MPI communicator with graph topology meets the
  442. * requirements of the Graph, Incidence Graph, Adjacency Graph, Vertex
  443. * List Graph, and Edge List Graph concepts from the Boost Graph
  444. * Library.
  445. */
  446. template<>
  447. struct graph_traits<mpi::graph_communicator> {
  448. // Graph concept requirements
  449. typedef int vertex_descriptor;
  450. typedef std::pair<int, int> edge_descriptor;
  451. typedef directed_tag directed_category;
  452. typedef disallow_parallel_edge_tag edge_parallel_category;
  453. /**
  454. * INTERNAL ONLY
  455. */
  456. struct traversal_category
  457. : incidence_graph_tag,
  458. adjacency_graph_tag,
  459. vertex_list_graph_tag,
  460. edge_list_graph_tag
  461. {
  462. };
  463. /**
  464. * @brief Returns a vertex descriptor that can never refer to any
  465. * valid vertex.
  466. */
  467. static vertex_descriptor null_vertex() { return -1; }
  468. // Incidence Graph requirements
  469. typedef mpi::detail::comm_out_edge_iterator out_edge_iterator;
  470. typedef int degree_size_type;
  471. // Adjacency Graph requirements
  472. typedef mpi::detail::comm_adj_iterator adjacency_iterator;
  473. // Vertex List Graph requirements
  474. typedef counting_iterator<int> vertex_iterator;
  475. typedef int vertices_size_type;
  476. // Edge List Graph requirements
  477. typedef mpi::detail::comm_edge_iterator edge_iterator;
  478. typedef int edges_size_type;
  479. };
  480. // Property Graph requirements
  481. /**
  482. * INTERNAL ONLY
  483. */
  484. template<>
  485. struct property_map<mpi::graph_communicator, vertex_index_t>
  486. {
  487. typedef identity_property_map type;
  488. typedef identity_property_map const_type;
  489. };
  490. } // end namespace boost
  491. #endif // BOOST_MPI_GRAPH_COMMUNICATOR_HPP