nonblocking.hpp 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738
  1. // Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
  2. // Use, modification and distribution is subject to the Boost Software
  3. // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  4. // http://www.boost.org/LICENSE_1_0.txt)
  5. /** @file nonblocking.hpp
  6. *
  7. * This header defines operations for completing non-blocking
  8. * communication requests.
  9. */
  10. #ifndef BOOST_MPI_NONBLOCKING_HPP
  11. #define BOOST_MPI_NONBLOCKING_HPP
  12. #include <boost/mpi/config.hpp>
  13. #include <vector>
  14. #include <iterator> // for std::iterator_traits
  15. #include <boost/optional.hpp>
  16. #include <utility> // for std::pair
  17. #include <algorithm> // for iter_swap, reverse
  18. #include <boost/static_assert.hpp>
  19. #include <boost/mpi/request.hpp>
  20. #include <boost/mpi/status.hpp>
  21. #include <boost/mpi/exception.hpp>
  22. namespace boost { namespace mpi {
  23. /**
  24. * @brief Wait until any non-blocking request has completed.
  25. *
  26. * This routine takes in a set of requests stored in the iterator
  27. * range @c [first,last) and waits until any of these requests has
  28. * been completed. It provides functionality equivalent to
  29. * @c MPI_Waitany.
  30. *
  31. * @param first The iterator that denotes the beginning of the
  32. * sequence of request objects.
  33. *
  34. * @param last The iterator that denotes the end of the sequence of
  35. * request objects. This may not be equal to @c first.
  36. *
  37. * @returns A pair containing the status object that corresponds to
  38. * the completed operation and the iterator referencing the completed
  39. * request.
  40. */
  41. template<typename ForwardIterator>
  42. std::pair<status, ForwardIterator>
  43. wait_any(ForwardIterator first, ForwardIterator last)
  44. {
  45. using std::advance;
  46. BOOST_ASSERT(first != last);
  47. typedef typename std::iterator_traits<ForwardIterator>::difference_type
  48. difference_type;
  49. bool all_trivial_requests = true;
  50. difference_type n = 0;
  51. ForwardIterator current = first;
  52. while (true) {
  53. // Check if we have found a completed request. If so, return it.
  54. if (current->m_requests[0] != MPI_REQUEST_NULL &&
  55. (current->m_requests[1] != MPI_REQUEST_NULL ||
  56. current->m_handler)) {
  57. if (optional<status> result = current->test())
  58. return std::make_pair(*result, current);
  59. }
  60. // Check if this request (and all others before it) are "trivial"
  61. // requests, e.g., they can be represented with a single
  62. // MPI_Request.
  63. all_trivial_requests =
  64. all_trivial_requests
  65. && !current->m_handler
  66. && current->m_requests[1] == MPI_REQUEST_NULL;
  67. // Move to the next request.
  68. ++n;
  69. if (++current == last) {
  70. // We have reached the end of the list. If all requests thus far
  71. // have been trivial, we can call MPI_Waitany directly, because
  72. // it may be more efficient than our busy-wait semantics.
  73. if (all_trivial_requests) {
  74. std::vector<MPI_Request> requests;
  75. requests.reserve(n);
  76. for (current = first; current != last; ++current)
  77. requests.push_back(current->m_requests[0]);
  78. // Let MPI wait until one of these operations completes.
  79. int index;
  80. status stat;
  81. BOOST_MPI_CHECK_RESULT(MPI_Waitany,
  82. (n, &requests[0], &index, &stat.m_status));
  83. // We don't have a notion of empty requests or status objects,
  84. // so this is an error.
  85. if (index == MPI_UNDEFINED)
  86. boost::throw_exception(exception("MPI_Waitany", MPI_ERR_REQUEST));
  87. // Find the iterator corresponding to the completed request.
  88. current = first;
  89. advance(current, index);
  90. current->m_requests[0] = requests[index];
  91. return std::make_pair(stat, current);
  92. }
  93. // There are some nontrivial requests, so we must continue our
  94. // busy waiting loop.
  95. n = 0;
  96. current = first;
  97. all_trivial_requests = true;
  98. }
  99. }
  100. // We cannot ever get here
  101. BOOST_ASSERT(false);
  102. }
  103. /**
  104. * @brief Test whether any non-blocking request has completed.
  105. *
  106. * This routine takes in a set of requests stored in the iterator
  107. * range @c [first,last) and tests whether any of these requests has
  108. * been completed. This routine is similar to @c wait_any, but will
  109. * not block waiting for requests to completed. It provides
  110. * functionality equivalent to @c MPI_Testany.
  111. *
  112. * @param first The iterator that denotes the beginning of the
  113. * sequence of request objects.
  114. *
  115. * @param last The iterator that denotes the end of the sequence of
  116. * request objects.
  117. *
  118. * @returns If any outstanding requests have completed, a pair
  119. * containing the status object that corresponds to the completed
  120. * operation and the iterator referencing the completed
  121. * request. Otherwise, an empty @c optional<>.
  122. */
  123. template<typename ForwardIterator>
  124. optional<std::pair<status, ForwardIterator> >
  125. test_any(ForwardIterator first, ForwardIterator last)
  126. {
  127. while (first != last) {
  128. // Check if we have found a completed request. If so, return it.
  129. if (optional<status> result = first->test()) {
  130. return std::make_pair(*result, first);
  131. }
  132. ++first;
  133. }
  134. // We found nothing
  135. return optional<std::pair<status, ForwardIterator> >();
  136. }
  137. /**
  138. * @brief Wait until all non-blocking requests have completed.
  139. *
  140. * This routine takes in a set of requests stored in the iterator
  141. * range @c [first,last) and waits until all of these requests have
  142. * been completed. It provides functionality equivalent to
  143. * @c MPI_Waitall.
  144. *
  145. * @param first The iterator that denotes the beginning of the
  146. * sequence of request objects.
  147. *
  148. * @param last The iterator that denotes the end of the sequence of
  149. * request objects.
  150. *
  151. * @param out If provided, an output iterator through which the
  152. * status of each request will be emitted. The @c status objects are
  153. * emitted in the same order as the requests are retrieved from
  154. * @c [first,last).
  155. *
  156. * @returns If an @p out parameter was provided, the value @c out
  157. * after all of the @c status objects have been emitted.
  158. */
  159. template<typename ForwardIterator, typename OutputIterator>
  160. OutputIterator
  161. wait_all(ForwardIterator first, ForwardIterator last, OutputIterator out)
  162. {
  163. typedef typename std::iterator_traits<ForwardIterator>::difference_type
  164. difference_type;
  165. using std::distance;
  166. difference_type num_outstanding_requests = distance(first, last);
  167. std::vector<status> results(num_outstanding_requests);
  168. std::vector<bool> completed(num_outstanding_requests);
  169. while (num_outstanding_requests > 0) {
  170. bool all_trivial_requests = true;
  171. difference_type idx = 0;
  172. for (ForwardIterator current = first; current != last; ++current, ++idx) {
  173. if (!completed[idx]) {
  174. if (optional<status> stat = current->test()) {
  175. // This outstanding request has been completed. We're done.
  176. results[idx] = *stat;
  177. completed[idx] = true;
  178. --num_outstanding_requests;
  179. all_trivial_requests = false;
  180. } else {
  181. // Check if this request (and all others before it) are "trivial"
  182. // requests, e.g., they can be represented with a single
  183. // MPI_Request.
  184. all_trivial_requests =
  185. all_trivial_requests
  186. && !current->m_handler
  187. && current->m_requests[1] == MPI_REQUEST_NULL;
  188. }
  189. }
  190. }
  191. // If we have yet to fulfill any requests and all of the requests
  192. // are trivial (i.e., require only a single MPI_Request to be
  193. // fulfilled), call MPI_Waitall directly.
  194. if (all_trivial_requests
  195. && num_outstanding_requests == (difference_type)results.size()) {
  196. std::vector<MPI_Request> requests;
  197. requests.reserve(num_outstanding_requests);
  198. for (ForwardIterator current = first; current != last; ++current)
  199. requests.push_back(current->m_requests[0]);
  200. // Let MPI wait until all of these operations completes.
  201. std::vector<MPI_Status> stats(num_outstanding_requests);
  202. BOOST_MPI_CHECK_RESULT(MPI_Waitall,
  203. (num_outstanding_requests, &requests[0],
  204. &stats[0]));
  205. for (std::vector<MPI_Status>::iterator i = stats.begin();
  206. i != stats.end(); ++i, ++out) {
  207. status stat;
  208. stat.m_status = *i;
  209. *out = stat;
  210. }
  211. return out;
  212. }
  213. all_trivial_requests = false;
  214. }
  215. return std::copy(results.begin(), results.end(), out);
  216. }
  217. /**
  218. * \overload
  219. */
  220. template<typename ForwardIterator>
  221. void
  222. wait_all(ForwardIterator first, ForwardIterator last)
  223. {
  224. typedef typename std::iterator_traits<ForwardIterator>::difference_type
  225. difference_type;
  226. using std::distance;
  227. difference_type num_outstanding_requests = distance(first, last);
  228. std::vector<bool> completed(num_outstanding_requests);
  229. while (num_outstanding_requests > 0) {
  230. bool all_trivial_requests = true;
  231. difference_type idx = 0;
  232. for (ForwardIterator current = first; current != last; ++current, ++idx) {
  233. if (!completed[idx]) {
  234. if (optional<status> stat = current->test()) {
  235. // This outstanding request has been completed.
  236. completed[idx] = true;
  237. --num_outstanding_requests;
  238. all_trivial_requests = false;
  239. } else {
  240. // Check if this request (and all others before it) are "trivial"
  241. // requests, e.g., they can be represented with a single
  242. // MPI_Request.
  243. all_trivial_requests =
  244. all_trivial_requests
  245. && !current->m_handler
  246. && current->m_requests[1] == MPI_REQUEST_NULL;
  247. }
  248. }
  249. }
  250. // If we have yet to fulfill any requests and all of the requests
  251. // are trivial (i.e., require only a single MPI_Request to be
  252. // fulfilled), call MPI_Waitall directly.
  253. if (all_trivial_requests
  254. && num_outstanding_requests == (difference_type)completed.size()) {
  255. std::vector<MPI_Request> requests;
  256. requests.reserve(num_outstanding_requests);
  257. for (ForwardIterator current = first; current != last; ++current)
  258. requests.push_back(current->m_requests[0]);
  259. // Let MPI wait until all of these operations completes.
  260. BOOST_MPI_CHECK_RESULT(MPI_Waitall,
  261. (num_outstanding_requests, &requests[0],
  262. MPI_STATUSES_IGNORE));
  263. // Signal completion
  264. num_outstanding_requests = 0;
  265. }
  266. }
  267. }
  268. /**
  269. * @brief Tests whether all non-blocking requests have completed.
  270. *
  271. * This routine takes in a set of requests stored in the iterator
  272. * range @c [first,last) and determines whether all of these requests
  273. * have been completed. However, due to limitations of the underlying
  274. * MPI implementation, if any of the requests refers to a
  275. * non-blocking send or receive of a serialized data type, @c
  276. * test_all will always return the equivalent of @c false (i.e., the
  277. * requests cannot all be finished at this time). This routine
  278. * performs the same functionality as @c wait_all, except that this
  279. * routine will not block. This routine provides functionality
  280. * equivalent to @c MPI_Testall.
  281. *
  282. * @param first The iterator that denotes the beginning of the
  283. * sequence of request objects.
  284. *
  285. * @param last The iterator that denotes the end of the sequence of
  286. * request objects.
  287. *
  288. * @param out If provided and all requests hav been completed, an
  289. * output iterator through which the status of each request will be
  290. * emitted. The @c status objects are emitted in the same order as
  291. * the requests are retrieved from @c [first,last).
  292. *
  293. * @returns If an @p out parameter was provided, the value @c out
  294. * after all of the @c status objects have been emitted (if all
  295. * requests were completed) or an empty @c optional<>. If no @p out
  296. * parameter was provided, returns @c true if all requests have
  297. * completed or @c false otherwise.
  298. */
  299. template<typename ForwardIterator, typename OutputIterator>
  300. optional<OutputIterator>
  301. test_all(ForwardIterator first, ForwardIterator last, OutputIterator out)
  302. {
  303. std::vector<MPI_Request> requests;
  304. for (; first != last; ++first) {
  305. // If we have a non-trivial request, then no requests can be
  306. // completed.
  307. if (first->m_handler || first->m_requests[1] != MPI_REQUEST_NULL)
  308. return optional<OutputIterator>();
  309. requests.push_back(first->m_requests[0]);
  310. }
  311. int flag = 0;
  312. int n = requests.size();
  313. std::vector<MPI_Status> stats(n);
  314. BOOST_MPI_CHECK_RESULT(MPI_Testall, (n, &requests[0], &flag, &stats[0]));
  315. if (flag) {
  316. for (int i = 0; i < n; ++i, ++out) {
  317. status stat;
  318. stat.m_status = stats[i];
  319. *out = stat;
  320. }
  321. return out;
  322. } else {
  323. return optional<OutputIterator>();
  324. }
  325. }
  326. /**
  327. * \overload
  328. */
  329. template<typename ForwardIterator>
  330. bool
  331. test_all(ForwardIterator first, ForwardIterator last)
  332. {
  333. std::vector<MPI_Request> requests;
  334. for (; first != last; ++first) {
  335. // If we have a non-trivial request, then no requests can be
  336. // completed.
  337. if (first->m_handler || first->m_requests[1] != MPI_REQUEST_NULL)
  338. return false;
  339. requests.push_back(first->m_requests[0]);
  340. }
  341. int flag = 0;
  342. int n = requests.size();
  343. BOOST_MPI_CHECK_RESULT(MPI_Testall,
  344. (n, &requests[0], &flag, MPI_STATUSES_IGNORE));
  345. return flag != 0;
  346. }
  347. /**
  348. * @brief Wait until some non-blocking requests have completed.
  349. *
  350. * This routine takes in a set of requests stored in the iterator
  351. * range @c [first,last) and waits until at least one of the requests
  352. * has completed. It then completes all of the requests it can,
  353. * partitioning the input sequence into pending requests followed by
  354. * completed requests. If an output iterator is provided, @c status
  355. * objects will be emitted for each of the completed requests. This
  356. * routine provides functionality equivalent to @c MPI_Waitsome.
  357. *
  358. * @param first The iterator that denotes the beginning of the
  359. * sequence of request objects.
  360. *
  361. * @param last The iterator that denotes the end of the sequence of
  362. * request objects. This may not be equal to @c first.
  363. *
  364. * @param out If provided, the @c status objects corresponding to
  365. * completed requests will be emitted through this output iterator.
  366. * @returns If the @p out parameter was provided, a pair containing
  367. * the output iterator @p out after all of the @c status objects have
  368. * been written through it and an iterator referencing the first
  369. * completed request. If no @p out parameter was provided, only the
  370. * iterator referencing the first completed request will be emitted.
  371. */
  372. template<typename BidirectionalIterator, typename OutputIterator>
  373. std::pair<OutputIterator, BidirectionalIterator>
  374. wait_some(BidirectionalIterator first, BidirectionalIterator last,
  375. OutputIterator out)
  376. {
  377. using std::advance;
  378. if (first == last)
  379. return std::make_pair(out, first);
  380. typedef typename std::iterator_traits<BidirectionalIterator>::difference_type
  381. difference_type;
  382. bool all_trivial_requests = true;
  383. difference_type n = 0;
  384. BidirectionalIterator current = first;
  385. BidirectionalIterator start_of_completed = last;
  386. while (true) {
  387. // Check if we have found a completed request.
  388. if (optional<status> result = current->test()) {
  389. using std::iter_swap;
  390. // Emit the resulting status object
  391. *out++ = *result;
  392. // We're expanding the set of completed requests
  393. --start_of_completed;
  394. if (current == start_of_completed) {
  395. // If we have hit the end of the list of pending
  396. // requests. Finish up by fixing the order of the completed
  397. // set to match the order in which we emitted status objects,
  398. // then return.
  399. std::reverse(start_of_completed, last);
  400. return std::make_pair(out, start_of_completed);
  401. }
  402. // Swap the request we just completed with the last request that
  403. // has not yet been tested.
  404. iter_swap(current, start_of_completed);
  405. continue;
  406. }
  407. // Check if this request (and all others before it) are "trivial"
  408. // requests, e.g., they can be represented with a single
  409. // MPI_Request.
  410. all_trivial_requests =
  411. all_trivial_requests
  412. && !current->m_handler
  413. && current->m_requests[1] == MPI_REQUEST_NULL;
  414. // Move to the next request.
  415. ++n;
  416. if (++current == start_of_completed) {
  417. if (start_of_completed != last) {
  418. // We have satisfied some requests. Make the order of the
  419. // completed requests match that of the status objects we've
  420. // already emitted and we're done.
  421. std::reverse(start_of_completed, last);
  422. return std::make_pair(out, start_of_completed);
  423. }
  424. // We have reached the end of the list. If all requests thus far
  425. // have been trivial, we can call MPI_Waitsome directly, because
  426. // it may be more efficient than our busy-wait semantics.
  427. if (all_trivial_requests) {
  428. std::vector<MPI_Request> requests;
  429. std::vector<int> indices(n);
  430. std::vector<MPI_Status> stats(n);
  431. requests.reserve(n);
  432. for (current = first; current != last; ++current)
  433. requests.push_back(current->m_requests[0]);
  434. // Let MPI wait until some of these operations complete.
  435. int num_completed;
  436. BOOST_MPI_CHECK_RESULT(MPI_Waitsome,
  437. (n, &requests[0], &num_completed, &indices[0],
  438. &stats[0]));
  439. // Translate the index-based result of MPI_Waitsome into a
  440. // partitioning on the requests.
  441. int current_offset = 0;
  442. current = first;
  443. for (int index = 0; index < num_completed; ++index, ++out) {
  444. using std::iter_swap;
  445. // Move "current" to the request object at this index
  446. advance(current, indices[index] - current_offset);
  447. current_offset = indices[index];
  448. // Emit the status object
  449. status stat;
  450. stat.m_status = stats[index];
  451. *out = stat;
  452. // Finish up the request and swap it into the "completed
  453. // requests" partition.
  454. current->m_requests[0] = requests[indices[index]];
  455. --start_of_completed;
  456. iter_swap(current, start_of_completed);
  457. }
  458. // We have satisfied some requests. Make the order of the
  459. // completed requests match that of the status objects we've
  460. // already emitted and we're done.
  461. std::reverse(start_of_completed, last);
  462. return std::make_pair(out, start_of_completed);
  463. }
  464. // There are some nontrivial requests, so we must continue our
  465. // busy waiting loop.
  466. n = 0;
  467. current = first;
  468. }
  469. }
  470. // We cannot ever get here
  471. BOOST_ASSERT(false);
  472. }
  473. /**
  474. * \overload
  475. */
  476. template<typename BidirectionalIterator>
  477. BidirectionalIterator
  478. wait_some(BidirectionalIterator first, BidirectionalIterator last)
  479. {
  480. using std::advance;
  481. if (first == last)
  482. return first;
  483. typedef typename std::iterator_traits<BidirectionalIterator>::difference_type
  484. difference_type;
  485. bool all_trivial_requests = true;
  486. difference_type n = 0;
  487. BidirectionalIterator current = first;
  488. BidirectionalIterator start_of_completed = last;
  489. while (true) {
  490. // Check if we have found a completed request.
  491. if (optional<status> result = current->test()) {
  492. using std::iter_swap;
  493. // We're expanding the set of completed requests
  494. --start_of_completed;
  495. // If we have hit the end of the list of pending requests, we're
  496. // done.
  497. if (current == start_of_completed)
  498. return start_of_completed;
  499. // Swap the request we just completed with the last request that
  500. // has not yet been tested.
  501. iter_swap(current, start_of_completed);
  502. continue;
  503. }
  504. // Check if this request (and all others before it) are "trivial"
  505. // requests, e.g., they can be represented with a single
  506. // MPI_Request.
  507. all_trivial_requests =
  508. all_trivial_requests
  509. && !current->m_handler
  510. && current->m_requests[1] == MPI_REQUEST_NULL;
  511. // Move to the next request.
  512. ++n;
  513. if (++current == start_of_completed) {
  514. // If we have satisfied some requests, we're done.
  515. if (start_of_completed != last)
  516. return start_of_completed;
  517. // We have reached the end of the list. If all requests thus far
  518. // have been trivial, we can call MPI_Waitsome directly, because
  519. // it may be more efficient than our busy-wait semantics.
  520. if (all_trivial_requests) {
  521. std::vector<MPI_Request> requests;
  522. std::vector<int> indices(n);
  523. requests.reserve(n);
  524. for (current = first; current != last; ++current)
  525. requests.push_back(current->m_requests[0]);
  526. // Let MPI wait until some of these operations complete.
  527. int num_completed;
  528. BOOST_MPI_CHECK_RESULT(MPI_Waitsome,
  529. (n, &requests[0], &num_completed, &indices[0],
  530. MPI_STATUSES_IGNORE));
  531. // Translate the index-based result of MPI_Waitsome into a
  532. // partitioning on the requests.
  533. int current_offset = 0;
  534. current = first;
  535. for (int index = 0; index < num_completed; ++index) {
  536. using std::iter_swap;
  537. // Move "current" to the request object at this index
  538. advance(current, indices[index] - current_offset);
  539. current_offset = indices[index];
  540. // Finish up the request and swap it into the "completed
  541. // requests" partition.
  542. current->m_requests[0] = requests[indices[index]];
  543. --start_of_completed;
  544. iter_swap(current, start_of_completed);
  545. }
  546. // We have satisfied some requests, so we are done.
  547. return start_of_completed;
  548. }
  549. // There are some nontrivial requests, so we must continue our
  550. // busy waiting loop.
  551. n = 0;
  552. current = first;
  553. }
  554. }
  555. // We cannot ever get here
  556. BOOST_ASSERT(false);
  557. }
  558. /**
  559. * @brief Test whether some non-blocking requests have completed.
  560. *
  561. * This routine takes in a set of requests stored in the iterator
  562. * range @c [first,last) and tests to see if any of the requests has
  563. * completed. It completes all of the requests it can, partitioning
  564. * the input sequence into pending requests followed by completed
  565. * requests. If an output iterator is provided, @c status objects
  566. * will be emitted for each of the completed requests. This routine
  567. * is similar to @c wait_some, but does not wait until any requests
  568. * have completed. This routine provides functionality equivalent to
  569. * @c MPI_Testsome.
  570. *
  571. * @param first The iterator that denotes the beginning of the
  572. * sequence of request objects.
  573. *
  574. * @param last The iterator that denotes the end of the sequence of
  575. * request objects. This may not be equal to @c first.
  576. *
  577. * @param out If provided, the @c status objects corresponding to
  578. * completed requests will be emitted through this output iterator.
  579. * @returns If the @p out parameter was provided, a pair containing
  580. * the output iterator @p out after all of the @c status objects have
  581. * been written through it and an iterator referencing the first
  582. * completed request. If no @p out parameter was provided, only the
  583. * iterator referencing the first completed request will be emitted.
  584. */
  585. template<typename BidirectionalIterator, typename OutputIterator>
  586. std::pair<OutputIterator, BidirectionalIterator>
  587. test_some(BidirectionalIterator first, BidirectionalIterator last,
  588. OutputIterator out)
  589. {
  590. BidirectionalIterator current = first;
  591. BidirectionalIterator start_of_completed = last;
  592. while (current != start_of_completed) {
  593. // Check if we have found a completed request.
  594. if (optional<status> result = current->test()) {
  595. using std::iter_swap;
  596. // Emit the resulting status object
  597. *out++ = *result;
  598. // We're expanding the set of completed requests
  599. --start_of_completed;
  600. // Swap the request we just completed with the last request that
  601. // has not yet been tested.
  602. iter_swap(current, start_of_completed);
  603. continue;
  604. }
  605. // Move to the next request.
  606. ++current;
  607. }
  608. // Finish up by fixing the order of the completed set to match the
  609. // order in which we emitted status objects, then return.
  610. std::reverse(start_of_completed, last);
  611. return std::make_pair(out, start_of_completed);
  612. }
  613. /**
  614. * \overload
  615. */
  616. template<typename BidirectionalIterator>
  617. BidirectionalIterator
  618. test_some(BidirectionalIterator first, BidirectionalIterator last)
  619. {
  620. BidirectionalIterator current = first;
  621. BidirectionalIterator start_of_completed = last;
  622. while (current != start_of_completed) {
  623. // Check if we have found a completed request.
  624. if (optional<status> result = current->test()) {
  625. using std::iter_swap;
  626. // We're expanding the set of completed requests
  627. --start_of_completed;
  628. // Swap the request we just completed with the last request that
  629. // has not yet been tested.
  630. iter_swap(current, start_of_completed);
  631. continue;
  632. }
  633. // Move to the next request.
  634. ++current;
  635. }
  636. return start_of_completed;
  637. }
  638. } } // end namespace boost::mpi
  639. #endif // BOOST_MPI_NONBLOCKING_HPP