summaryrefslogtreecommitdiffstats
path: root/src/boost/libs/mpi
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/boost/libs/mpi/build/Jamfile.v2190
-rw-r--r--src/boost/libs/mpi/build/__init__.py10
-rw-r--r--src/boost/libs/mpi/example/cartesian_communicator.cpp42
-rw-r--r--src/boost/libs/mpi/example/generate_collect.cpp129
-rw-r--r--src/boost/libs/mpi/example/generate_collect_optional.cpp113
-rw-r--r--src/boost/libs/mpi/example/global_min.cpp31
-rw-r--r--src/boost/libs/mpi/example/hello_world.cpp33
-rw-r--r--src/boost/libs/mpi/example/hello_world_broadcast.cpp28
-rw-r--r--src/boost/libs/mpi/example/hello_world_groups.cpp46
-rw-r--r--src/boost/libs/mpi/example/hello_world_nonblocking.cpp36
-rw-r--r--src/boost/libs/mpi/example/in_place_global_min.cpp29
-rw-r--r--src/boost/libs/mpi/example/parallel_example.cpp196
-rw-r--r--src/boost/libs/mpi/example/python/hello_world.py16
-rw-r--r--src/boost/libs/mpi/example/random_content.cpp98
-rw-r--r--src/boost/libs/mpi/example/random_gather.cpp31
-rw-r--r--src/boost/libs/mpi/example/random_min.cpp30
-rw-r--r--src/boost/libs/mpi/example/random_scatter.cpp37
-rw-r--r--src/boost/libs/mpi/example/reduce_performance_test.cpp138
-rw-r--r--src/boost/libs/mpi/example/string_cat.cpp46
-rw-r--r--src/boost/libs/mpi/index.html16
-rw-r--r--src/boost/libs/mpi/meta/libraries.json16
-rw-r--r--src/boost/libs/mpi/src/broadcast.cpp142
-rw-r--r--src/boost/libs/mpi/src/cartesian_communicator.cpp179
-rw-r--r--src/boost/libs/mpi/src/communicator.cpp328
-rw-r--r--src/boost/libs/mpi/src/computation_tree.cpp72
-rw-r--r--src/boost/libs/mpi/src/content_oarchive.cpp20
-rw-r--r--src/boost/libs/mpi/src/environment.cpp258
-rw-r--r--src/boost/libs/mpi/src/error_string.cpp31
-rw-r--r--src/boost/libs/mpi/src/exception.cpp24
-rw-r--r--src/boost/libs/mpi/src/graph_communicator.cpp73
-rw-r--r--src/boost/libs/mpi/src/group.cpp106
-rw-r--r--src/boost/libs/mpi/src/intercommunicator.cpp54
-rw-r--r--src/boost/libs/mpi/src/mpi_datatype_cache.cpp66
-rw-r--r--src/boost/libs/mpi/src/mpi_datatype_oarchive.cpp19
-rw-r--r--src/boost/libs/mpi/src/offsets.cpp96
-rw-r--r--src/boost/libs/mpi/src/packed_iarchive.cpp21
-rw-r--r--src/boost/libs/mpi/src/packed_oarchive.cpp19
-rw-r--r--src/boost/libs/mpi/src/packed_skeleton_iarchive.cpp25
-rw-r--r--src/boost/libs/mpi/src/packed_skeleton_oarchive.cpp22
-rw-r--r--src/boost/libs/mpi/src/point_to_point.cpp102
-rw-r--r--src/boost/libs/mpi/src/python/collectives.cpp144
-rw-r--r--src/boost/libs/mpi/src/python/datatypes.cpp27
-rw-r--r--src/boost/libs/mpi/src/python/documentation.cpp671
-rw-r--r--src/boost/libs/mpi/src/python/module.cpp55
-rw-r--r--src/boost/libs/mpi/src/python/py_communicator.cpp133
-rw-r--r--src/boost/libs/mpi/src/python/py_environment.cpp123
-rw-r--r--src/boost/libs/mpi/src/python/py_exception.cpp54
-rw-r--r--src/boost/libs/mpi/src/python/py_nonblocking.cpp255
-rw-r--r--src/boost/libs/mpi/src/python/py_request.cpp111
-rw-r--r--src/boost/libs/mpi/src/python/py_timer.cpp48
-rw-r--r--src/boost/libs/mpi/src/python/request_with_value.hpp71
-rw-r--r--src/boost/libs/mpi/src/python/serialize.cpp79
-rw-r--r--src/boost/libs/mpi/src/python/skeleton_and_content.cpp173
-rw-r--r--src/boost/libs/mpi/src/python/status.cpp41
-rw-r--r--src/boost/libs/mpi/src/python/utility.hpp43
-rw-r--r--src/boost/libs/mpi/src/request.cpp239
-rw-r--r--src/boost/libs/mpi/src/text_skeleton_oarchive.cpp20
-rw-r--r--src/boost/libs/mpi/src/timer.cpp25
-rw-r--r--src/boost/libs/mpi/test/Jamfile.v255
-rw-r--r--src/boost/libs/mpi/test/all_gather_test.cpp148
-rw-r--r--src/boost/libs/mpi/test/all_reduce_test.cpp309
-rw-r--r--src/boost/libs/mpi/test/all_to_all_test.cpp113
-rw-r--r--src/boost/libs/mpi/test/block_nonblock_test.cpp95
-rw-r--r--src/boost/libs/mpi/test/broadcast_stl_test.cpp75
-rw-r--r--src/boost/libs/mpi/test/broadcast_test.cpp159
-rw-r--r--src/boost/libs/mpi/test/cartesian_topology_init_test.cpp88
-rw-r--r--src/boost/libs/mpi/test/cartesian_topology_test.cpp193
-rw-r--r--src/boost/libs/mpi/test/debugger.cpp48
-rw-r--r--src/boost/libs/mpi/test/debugger.hpp26
-rw-r--r--src/boost/libs/mpi/test/gather_test.cpp165
-rw-r--r--src/boost/libs/mpi/test/gps_position.hpp69
-rw-r--r--src/boost/libs/mpi/test/graph_topology_test.cpp141
-rw-r--r--src/boost/libs/mpi/test/groups_test.cpp59
-rw-r--r--src/boost/libs/mpi/test/is_mpi_op_test.cpp33
-rw-r--r--src/boost/libs/mpi/test/mt_init_test.cpp37
-rw-r--r--src/boost/libs/mpi/test/mt_level_test.cpp108
-rw-r--r--src/boost/libs/mpi/test/non_blocking_any_source.cpp63
-rw-r--r--src/boost/libs/mpi/test/nonblocking_test.cpp247
-rw-r--r--src/boost/libs/mpi/test/pointer_test.cpp42
-rw-r--r--src/boost/libs/mpi/test/python/all_gather_test.py25
-rw-r--r--src/boost/libs/mpi/test/python/all_reduce_test.py29
-rw-r--r--src/boost/libs/mpi/test/python/all_to_all_test.py30
-rw-r--r--src/boost/libs/mpi/test/python/broadcast_test.py29
-rw-r--r--src/boost/libs/mpi/test/python/gather_test.py32
-rw-r--r--src/boost/libs/mpi/test/python/generators.py23
-rw-r--r--src/boost/libs/mpi/test/python/nonblocking_test.py131
-rw-r--r--src/boost/libs/mpi/test/python/reduce_test.py31
-rw-r--r--src/boost/libs/mpi/test/python/ring_test.py42
-rw-r--r--src/boost/libs/mpi/test/python/scan_test.py29
-rw-r--r--src/boost/libs/mpi/test/python/scatter_test.py36
-rw-r--r--src/boost/libs/mpi/test/python/skeleton_content_test.cpp37
-rw-r--r--src/boost/libs/mpi/test/python/skeleton_content_test.py75
-rw-r--r--src/boost/libs/mpi/test/reduce_test.cpp237
-rw-r--r--src/boost/libs/mpi/test/ring_test.cpp124
-rw-r--r--src/boost/libs/mpi/test/scan_test.cpp228
-rw-r--r--src/boost/libs/mpi/test/scatter_test.cpp233
-rw-r--r--src/boost/libs/mpi/test/sendrecv_test.cpp62
-rw-r--r--src/boost/libs/mpi/test/sendrecv_vector.cpp95
-rw-r--r--src/boost/libs/mpi/test/skeleton_content_test.cpp200
-rw-r--r--src/boost/libs/mpi/test/version_test.cpp60
-rw-r--r--src/boost/libs/mpi/test/wait_all_vector_test.cpp46
-rw-r--r--src/boost/libs/mpi/test/wait_any_test.cpp64
102 files changed, 9451 insertions, 0 deletions
diff --git a/src/boost/libs/mpi/build/Jamfile.v2 b/src/boost/libs/mpi/build/Jamfile.v2
new file mode 100644
index 00000000..1f423463
--- /dev/null
+++ b/src/boost/libs/mpi/build/Jamfile.v2
@@ -0,0 +1,190 @@
+# Copyright (C) 2005, 2006 The Trustees of Indiana University.
+# Copyright (C) 2005 Douglas Gregor <doug.gregor -at- gmail.com>
+# Copyright (c) 2018 Stefan Seefeld
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Authors: Douglas Gregor
+# Andrew Lumsdaine
+
+import mpi ;
+import indirect ;
+import python ;
+import option ;
+import regex ;
+
+#
+# The `version-suffix` rule really belongs into python.jam, and
+# should be moved there. `split-version` is only duplicated here
+# as a prerequisite. (See https://github.com/boostorg/build/pull/290)
+#
+
+
+# Validate the version string and extract the major/minor part we care about.
+#
+local rule split-version ( version )
+{
+ local major-minor = [ MATCH "^([0-9]+)\.([0-9]+)(.*)$" : $(version) : 1 2 3 ] ;
+ if ! $(major-minor[2]) || $(major-minor[3])
+ {
+ ECHO "Warning: \"using python\" expects a two part (major, minor) version number; got" $(version) instead ;
+
+ # Add a zero to account for the missing digit if necessary.
+ major-minor += 0 ;
+ }
+
+ return $(major-minor[1]) $(major-minor[2]) ;
+}
+
+# Define a version suffix for libraries depending on Python.
+# For example, Boost.Python built for Python 2.7 uses the suffix "27"
+rule version-suffix ( version )
+{
+ local major-minor = [ split-version $(version) ] ;
+ local suffix = $(major-minor:J="") ;
+ return $(suffix) ;
+}
+
+
+# Python build id (for Python libraries only).
+python-id = [ option.get "python-buildid" ] ;
+if $(python-id)
+{
+ PYTHON_ID = [ regex.replace $(python-id) "[*\\/:.\"\']" _ ] ;
+}
+
+rule python-tag ( name : type ? : property-set )
+{
+ local result = $(name) ;
+ if $(type) in STATIC_LIB SHARED_LIB IMPORT_LIB
+ {
+ local version = [ $(property-set).get <python> ] ;
+ local lib-suffix = [ version-suffix $(version) ] ;
+ result = $(result)$(lib-suffix) ;
+ }
+ if $(type) in STATIC_LIB SHARED_LIB IMPORT_LIB && $(PYTHON_ID)
+ {
+ result = $(result)-$(PYTHON_ID) ;
+ }
+
+ # forward to the boost tagging rule
+ return [ tag $(result) : $(type) : $(property-set) ] ;
+}
+
+mpi_python_libs = ;
+
+if [ mpi.configured ]
+{
+
+project boost/mpi
+ : source-location ../src
+ ;
+
+lib boost_mpi
+ :
+ broadcast.cpp
+ cartesian_communicator.cpp
+ communicator.cpp
+ computation_tree.cpp
+ content_oarchive.cpp
+ environment.cpp
+ error_string.cpp
+ exception.cpp
+ graph_communicator.cpp
+ group.cpp
+ intercommunicator.cpp
+ mpi_datatype_cache.cpp
+ mpi_datatype_oarchive.cpp
+ packed_iarchive.cpp
+ packed_oarchive.cpp
+ packed_skeleton_iarchive.cpp
+ packed_skeleton_oarchive.cpp
+ point_to_point.cpp
+ request.cpp
+ text_skeleton_oarchive.cpp
+ timer.cpp
+ offsets.cpp
+ : # Requirements
+ <library>../../serialization/build//boost_serialization
+ <library>/mpi//mpi [ mpi.extra-requirements ]
+ <define>BOOST_MPI_SOURCE=1
+ <link>shared:<define>BOOST_MPI_DYN_LINK=1
+ <local-visibility>global
+ : # Default build
+ <link>shared
+ : # Usage requirements
+ <library>../../serialization/build//boost_serialization
+ <library>/mpi//mpi [ mpi.extra-requirements ]
+ ;
+
+ if [ python.configured ]
+ {
+ lib boost_mpi_python
+ : # Sources
+ python/serialize.cpp
+ : # Requirements
+ <library>boost_mpi
+ <library>/mpi//mpi [ mpi.extra-requirements ]
+ <library>/boost/python//boost_python
+ <link>shared:<define>BOOST_MPI_DYN_LINK=1
+ <link>shared:<define>BOOST_MPI_PYTHON_DYN_LINK=1
+ <link>shared:<define>BOOST_PYTHON_DYN_LINK=1
+ <define>BOOST_MPI_PYTHON_SOURCE=1
+ <python-debugging>on:<define>BOOST_DEBUG_PYTHON
+ -<tag>@$(BOOST_JAMROOT_MODULE)%$(BOOST_JAMROOT_MODULE).tag
+ <tag>@$(__name__).python-tag
+ <conditional>@python.require-py
+ <local-visibility>global
+
+ : # Default build
+ <link>shared
+ : # Usage requirements
+ <library>/mpi//mpi [ mpi.extra-requirements ]
+ ;
+
+ python-extension mpi
+ : # Sources
+ python/collectives.cpp
+ python/py_communicator.cpp
+ python/datatypes.cpp
+ python/documentation.cpp
+ python/py_environment.cpp
+ python/py_nonblocking.cpp
+ python/py_exception.cpp
+ python/module.cpp
+ python/py_request.cpp
+ python/skeleton_and_content.cpp
+ python/status.cpp
+ python/py_timer.cpp
+ : # Requirements
+ <library>/boost/python//boost_python
+ <library>boost_mpi_python
+ <library>boost_mpi
+ <library>/mpi//mpi [ mpi.extra-requirements ]
+ <link>shared:<define>BOOST_MPI_DYN_LINK=1
+ <link>shared:<define>BOOST_MPI_PYTHON_DYN_LINK=1
+ <link>shared:<define>BOOST_PYTHON_DYN_LINK=1
+ <link>shared <runtime-link>shared
+ <python-debugging>on:<define>BOOST_DEBUG_PYTHON
+ ;
+
+ mpi_python_libs = boost_mpi_python mpi ;
+ }
+}
+else if ! ( --without-mpi in [ modules.peek : ARGV ] )
+{
+ message boost_mpi
+ : "warning: skipping optional Message Passing Interface (MPI) library."
+ : "note: to enable MPI support, add \"using mpi ;\" to user-config.jam."
+ : "note: to suppress this message, pass \"--without-mpi\" to bjam."
+ : "note: otherwise, you can safely ignore this message."
+ ;
+}
+else
+{
+ alias boost_mpi ;
+}
+
+boost-install boost_mpi $(mpi_python_libs) ;
diff --git a/src/boost/libs/mpi/build/__init__.py b/src/boost/libs/mpi/build/__init__.py
new file mode 100644
index 00000000..ffd3862e
--- /dev/null
+++ b/src/boost/libs/mpi/build/__init__.py
@@ -0,0 +1,10 @@
+import sys
+if sys.platform == 'linux2':
+ import DLFCN as dl
+ flags = sys.getdlopenflags()
+ sys.setdlopenflags(dl.RTLD_NOW|dl.RTLD_GLOBAL)
+ import mpi
+ sys.setdlopenflags(flags)
+else:
+ import mpi
+
diff --git a/src/boost/libs/mpi/example/cartesian_communicator.cpp b/src/boost/libs/mpi/example/cartesian_communicator.cpp
new file mode 100644
index 00000000..7b011041
--- /dev/null
+++ b/src/boost/libs/mpi/example/cartesian_communicator.cpp
@@ -0,0 +1,42 @@
+// Copyright Alain Miniussi 2014.
+// Distributed under the Boost Software License, Version 1.0.
+// (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Alain Miniussi
+
+#include <vector>
+#include <iostream>
+
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/collectives.hpp>
+#include <boost/mpi/environment.hpp>
+#include <boost/mpi/cartesian_communicator.hpp>
+
+namespace mpi = boost::mpi;
+// Curly brace init make this useless, but
+// - Need to support obsolete like g++ 4.3.x. for some reason
+// - Can't conditionnaly compile with bjam (unless you find
+// the doc, and read it, which would only make sense if you
+// actually wan't to use bjam, which does not (make sense))
+typedef mpi::cartesian_dimension cd;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env;
+ mpi::communicator world;
+
+ if (world.size() != 24) return -1;
+ mpi::cartesian_dimension dims[] = {cd(2, true), cd(3,true), cd(4,true)};
+ mpi::cartesian_communicator cart(world, mpi::cartesian_topology(dims));
+ for (int r = 0; r < cart.size(); ++r) {
+ cart.barrier();
+ if (r == cart.rank()) {
+ std::vector<int> c = cart.coordinates(r);
+ std::cout << "rk :" << r << " coords: "
+ << c[0] << ' ' << c[1] << ' ' << c[2] << '\n';
+ }
+ }
+ return 0;
+}
+
diff --git a/src/boost/libs/mpi/example/generate_collect.cpp b/src/boost/libs/mpi/example/generate_collect.cpp
new file mode 100644
index 00000000..5579d50d
--- /dev/null
+++ b/src/boost/libs/mpi/example/generate_collect.cpp
@@ -0,0 +1,129 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// An example using Boost.MPI's split() operation on communicators to
+// create separate data-generating processes and data-collecting
+// processes.
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <cstdlib>
+#include <boost/serialization/vector.hpp>
+namespace mpi = boost::mpi;
+
+enum message_tags { msg_data_packet, msg_broadcast_data, msg_finished };
+
+void generate_data(mpi::communicator local, mpi::communicator world)
+{
+ using std::srand;
+ using std::rand;
+
+ // The rank of the collector within the world communicator
+ int master_collector = local.size();
+
+ srand(time(0) + world.rank());
+
+ // Send out several blocks of random data to the collectors.
+ int num_data_blocks = rand() % 3 + 1;
+ for (int block = 0; block < num_data_blocks; ++block) {
+ // Generate some random data
+ int num_samples = rand() % 1000;
+ std::vector<int> data;
+ for (int i = 0; i < num_samples; ++i) {
+ data.push_back(rand());
+ }
+
+ // Send our data to the master collector process.
+ std::cout << "Generator #" << local.rank() << " sends some data..."
+ << std::endl;
+ world.send(master_collector, msg_data_packet, data);
+ }
+
+ // Wait for all of the generators to complete
+ (local.barrier)();
+
+ // The first generator will send the message to the master collector
+ // indicating that we're done.
+ if (local.rank() == 0)
+ world.send(master_collector, msg_finished);
+}
+
+void collect_data(mpi::communicator local, mpi::communicator world)
+{
+ // The rank of the collector within the world communicator
+ int master_collector = world.size() - local.size();
+
+ if (world.rank() == master_collector) {
+ while (true) {
+ // Wait for a message
+ mpi::status msg = world.probe();
+ if (msg.tag() == msg_data_packet) {
+ // Receive the packet of data
+ std::vector<int> data;
+ world.recv(msg.source(), msg.tag(), data);
+
+ // Tell each of the collectors that we'll be broadcasting some data
+ for (int dest = 1; dest < local.size(); ++dest)
+ local.send(dest, msg_broadcast_data, msg.source());
+
+ // Broadcast the actual data.
+ broadcast(local, data, 0);
+ } else if (msg.tag() == msg_finished) {
+ // Receive the message
+ world.recv(msg.source(), msg.tag());
+
+ // Tell each of the collectors that we're finished
+ for (int dest = 1; dest < local.size(); ++dest)
+ local.send(dest, msg_finished);
+
+ break;
+ }
+ }
+ } else {
+ while (true) {
+ // Wait for a message from the master collector
+ mpi::status msg = local.probe();
+ if (msg.tag() == msg_broadcast_data) {
+ // Receive the broadcast message
+ int originator;
+ local.recv(msg.source(), msg.tag(), originator);
+
+ // Receive the data broadcasted from the master collector
+ std::vector<int> data;
+ broadcast(local, data, 0);
+
+ std::cout << "Collector #" << local.rank()
+ << " is processing data from generator #" << originator
+ << "." << std::endl;
+ } else if (msg.tag() == msg_finished) {
+ // Receive the message
+ local.recv(msg.source(), msg.tag());
+
+ break;
+ }
+ }
+ }
+}
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ if (world.size() < 3) {
+ if (world.rank() == 0) {
+ std::cerr << "Error: this example requires at least 3 processes."
+ << std::endl;
+ }
+ env.abort(-1);
+ }
+
+ bool is_generator = world.rank() < 2 * world.size() / 3;
+ mpi::communicator local = world.split(is_generator? 0 : 1);
+ if (is_generator) generate_data(local, world);
+ else collect_data(local, world);
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/generate_collect_optional.cpp b/src/boost/libs/mpi/example/generate_collect_optional.cpp
new file mode 100644
index 00000000..3aa3888c
--- /dev/null
+++ b/src/boost/libs/mpi/example/generate_collect_optional.cpp
@@ -0,0 +1,113 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// An example using Boost.MPI's split() operation on communicators to
+// create separate data-generating processes and data-collecting
+// processes using boost::optional for broadcasting.
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <cstdlib>
+#include <boost/serialization/vector.hpp>
+#include <boost/serialization/optional.hpp>
+namespace mpi = boost::mpi;
+
+enum message_tags { msg_data_packet, msg_finished };
+
+void generate_data(mpi::communicator local, mpi::communicator world)
+{
+ using std::srand;
+ using std::rand;
+
+ // The rank of the collector within the world communicator
+ int master_collector = local.size();
+
+ srand(time(0) + world.rank());
+
+ // Send out several blocks of random data to the collectors.
+ int num_data_blocks = rand() % 3 + 1;
+ for (int block = 0; block < num_data_blocks; ++block) {
+ // Generate some random dataa
+ int num_samples = rand() % 1000;
+ std::vector<int> data;
+ for (int i = 0; i < num_samples; ++i) {
+ data.push_back(rand());
+ }
+
+ // Send our data to the master collector process.
+ std::cout << "Generator #" << local.rank() << " sends some data..."
+ << std::endl;
+ world.send(master_collector, msg_data_packet, data);
+ }
+
+ // Wait for all of the generators to complete
+ (local.barrier)();
+
+ // The first generator will send the message to the master collector
+ // indicating that we're done.
+ if (local.rank() == 0)
+ world.send(master_collector, msg_finished);
+}
+
+void collect_data(mpi::communicator local, mpi::communicator world)
+{
+ // The rank of the collector within the world communicator
+ int master_collector = world.size() - local.size();
+
+ if (world.rank() == master_collector) {
+ while (true) {
+ // Wait for a message
+ mpi::status msg = world.probe();
+ if (msg.tag() == msg_data_packet) {
+ // Receive the packet of data into a boost::optional
+ boost::optional<std::vector<int> > data;
+ data = std::vector<int>();
+ world.recv(msg.source(), msg.source(), *data);
+
+ // Broadcast the actual data.
+ broadcast(local, data, 0);
+ } else if (msg.tag() == msg_finished) {
+ // Receive the message
+ world.recv(msg.source(), msg.tag());
+
+ // Broadcast to each collector to tell them we've finished.
+ boost::optional<std::vector<int> > data;
+ broadcast(local, data, 0);
+ break;
+ }
+ }
+ } else {
+ boost::optional<std::vector<int> > data;
+ do {
+ // Wait for a broadcast from the master collector
+ broadcast(local, data, 0);
+ if (data) {
+ std::cout << "Collector #" << local.rank()
+ << " is processing data." << std::endl;
+ }
+ } while (data);
+ }
+}
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ if (world.size() < 4) {
+ if (world.rank() == 0) {
+ std::cerr << "Error: this example requires at least 4 processes."
+ << std::endl;
+ }
+ env.abort(-1);
+ }
+
+ bool is_generator = world.rank() < 2 * world.size() / 3;
+ mpi::communicator local = world.split(is_generator? 0 : 1);
+ if (is_generator) generate_data(local, world);
+ else collect_data(local, world);
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/global_min.cpp b/src/boost/libs/mpi/example/global_min.cpp
new file mode 100644
index 00000000..68de57df
--- /dev/null
+++ b/src/boost/libs/mpi/example/global_min.cpp
@@ -0,0 +1,31 @@
+// Copyright (C) 2013 Alain Miniussi <alain.miniussi@oca.eu>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// An example using Boost.MPI's all_reduce() that compute the minimum
+// of each process's value and broadcast the result to all the processes.
+
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <cstdlib>
+namespace mpi = boost::mpi;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ std::srand(world.rank());
+ int my_number = std::rand();
+ int minimum;
+
+ all_reduce(world, my_number, minimum, mpi::minimum<int>());
+
+ if (world.rank() == 0) {
+ std::cout << "The minimum value is " << minimum << std::endl;
+ }
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/hello_world.cpp b/src/boost/libs/mpi/example/hello_world.cpp
new file mode 100644
index 00000000..7095c172
--- /dev/null
+++ b/src/boost/libs/mpi/example/hello_world.cpp
@@ -0,0 +1,33 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A simple Hello, world! example using Boost.MPI message passing.
+
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <boost/serialization/string.hpp> // Needed to send/receive strings!
+namespace mpi = boost::mpi;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ if (world.rank() == 0) {
+ world.send(1, 0, std::string("Hello"));
+ std::string msg;
+ world.recv(1, 1, msg);
+ std::cout << msg << "!" << std::endl;
+ } else {
+ std::string msg;
+ world.recv(0, 0, msg);
+ std::cout << msg << ", ";
+ std::cout.flush();
+ world.send(0, 1, std::string("world"));
+ }
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/hello_world_broadcast.cpp b/src/boost/libs/mpi/example/hello_world_broadcast.cpp
new file mode 100644
index 00000000..4ffe239e
--- /dev/null
+++ b/src/boost/libs/mpi/example/hello_world_broadcast.cpp
@@ -0,0 +1,28 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A simple Hello, world! example using Boost.MPI broadcast()
+
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <boost/serialization/string.hpp> // Needed to send/receive strings!
+namespace mpi = boost::mpi;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ std::string value;
+ if (world.rank() == 0) {
+ value = "Hello, World!";
+ }
+
+ broadcast(world, value, 0);
+
+ std::cout << "Process #" << world.rank() << " says " << value << std::endl;
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/hello_world_groups.cpp b/src/boost/libs/mpi/example/hello_world_groups.cpp
new file mode 100644
index 00000000..7493aa4a
--- /dev/null
+++ b/src/boost/libs/mpi/example/hello_world_groups.cpp
@@ -0,0 +1,46 @@
+// Copyright (C) 2013 Andreas Hehn <hehn@phys.ethz.ch>, ETH Zurich
+// based on
+// hellp-world_broadcast.cpp (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A simple Hello world! example
+// using boost::mpi::group and boost::mpi::broadcast()
+
+#include <stdexcept>
+#include <boost/mpi/environment.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/group.hpp>
+#include <boost/mpi/collectives.hpp>
+
+#include <boost/serialization/string.hpp>
+
+namespace mpi = boost::mpi;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+ if(world.size() < 2)
+ throw std::runtime_error("Please run with at least 2 MPI processes!");
+
+ int group_a_ranks[2] = {0,1};
+
+ mpi::group world_group = world.group();
+ mpi::group group_a = world_group.include(group_a_ranks,group_a_ranks+2);
+
+ mpi::communicator comm_a(world,group_a);
+
+ std::string value("Hello world!");
+ if(comm_a)
+ {
+ if(comm_a.rank() == 0) {
+ value = "Hello group a!";
+ }
+ mpi::broadcast(comm_a, value, 0);
+ }
+ std::cout << "Process #" << world.rank() << " says " << value << std::endl;
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/hello_world_nonblocking.cpp b/src/boost/libs/mpi/example/hello_world_nonblocking.cpp
new file mode 100644
index 00000000..c65247b9
--- /dev/null
+++ b/src/boost/libs/mpi/example/hello_world_nonblocking.cpp
@@ -0,0 +1,36 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A simple Hello, world! example using Boost.MPI message passing.
+
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <boost/serialization/string.hpp> // Needed to send/receive strings!
+namespace mpi = boost::mpi;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ if (world.rank() == 0) {
+ mpi::request reqs[2];
+ std::string msg, out_msg = "Hello";
+ reqs[0] = world.isend(1, 0, out_msg);
+ reqs[1] = world.irecv(1, 1, msg);
+ mpi::wait_all(reqs, reqs + 2);
+ std::cout << msg << "!" << std::endl;
+ } else {
+ mpi::request reqs[2];
+ std::string msg, out_msg = "world";
+ reqs[0] = world.isend(0, 1, out_msg);
+ reqs[1] = world.irecv(0, 0, msg);
+ mpi::wait_all(reqs, reqs + 2);
+ std::cout << msg << ", ";
+ }
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/in_place_global_min.cpp b/src/boost/libs/mpi/example/in_place_global_min.cpp
new file mode 100644
index 00000000..7da2cf2b
--- /dev/null
+++ b/src/boost/libs/mpi/example/in_place_global_min.cpp
@@ -0,0 +1,29 @@
+// Copyright (C) 2013 Alain Miniussi <alain.miniussi@oca.eu>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// An example using Boost.MPI's all_reduce() that compute the minimum
+// of each process's value and broadcast the result to all the processes.
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <cstdlib>
+namespace mpi = boost::mpi;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ std::srand(world.rank());
+ int my_number = std::rand();
+
+ all_reduce(world, my_number, mpi::minimum<int>());
+
+ if (world.rank() == 0) {
+ std::cout << "The minimum value is " << my_number << std::endl;
+ }
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/parallel_example.cpp b/src/boost/libs/mpi/example/parallel_example.cpp
new file mode 100644
index 00000000..00347d51
--- /dev/null
+++ b/src/boost/libs/mpi/example/parallel_example.cpp
@@ -0,0 +1,196 @@
+// Copyright (C) 2005-2006 Matthias Troyer
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// An example of a parallel Monte Carlo simulation using some nodes to produce
+// data and others to aggregate the data
+#include <iostream>
+
+#include <boost/mpi.hpp>
+#include <boost/random/parallel.hpp>
+#include <boost/random.hpp>
+#include <boost/foreach.hpp>
+#include <iostream>
+#include <cstdlib>
+
+namespace mpi = boost::mpi;
+
+enum {sample_tag, sample_skeleton_tag, sample_broadcast_tag, quit_tag};
+
+
+void calculate_samples(int sample_length)
+{
+ int num_samples = 100;
+ std::vector<double> sample(sample_length);
+
+ // setup communicator by splitting
+
+ mpi::communicator world;
+ mpi::communicator calculate_communicator = world.split(0);
+
+ unsigned int num_calculate_ranks = calculate_communicator.size();
+
+ // the master of the accumulaion ranks is the first of them, hence
+ // with a rank just one after the last calculation rank
+ int master_accumulate_rank = num_calculate_ranks;
+
+ // the master of the calculation ranks sends the skeleton of the sample
+ // to the master of the accumulation ranks
+
+ if (world.rank()==0)
+ world.send(master_accumulate_rank,sample_skeleton_tag,mpi::skeleton(sample));
+
+ // next we extract the content of the sample vector, to be used in sending
+ // the content later on
+
+ mpi::content sample_content = mpi::get_content(sample);
+
+ // now intialize the parallel random number generator
+
+ boost::lcg64 engine(
+ boost::random::stream_number = calculate_communicator.rank(),
+ boost::random::total_streams = calculate_communicator.size()
+ );
+
+ boost::variate_generator<boost::lcg64&,boost::uniform_real<> >
+ rng(engine,boost::uniform_real<>());
+
+ for (unsigned int i=0; i<num_samples/num_calculate_ranks+1;++i) {
+
+ // calculate sample by filling the vector with random numbers
+ // note that std::generate will not work since it takes the generator
+ // by value, and boost::ref cannot be used as a generator.
+ // boost::ref should be fixed so that it can be used as generator
+
+ BOOST_FOREACH(double& x, sample)
+ x = rng();
+
+ // send sample to accumulation ranks
+ // Ideally we want to do this as a broadcast with an inter-communicator
+ // between the calculation and accumulation ranks. MPI2 should support
+ // this, but here we present an MPI1 compatible solution.
+
+ // send content of sample to first (master) accumulation process
+
+ world.send(master_accumulate_rank,sample_tag,sample_content);
+
+ // gather some results from all calculation ranks
+
+ double local_result = sample[0];
+ std::vector<double> gathered_results(calculate_communicator.size());
+ mpi::all_gather(calculate_communicator,local_result,gathered_results);
+ }
+
+ // we are done: the master tells the accumulation ranks to quit
+ if (world.rank()==0)
+ world.send(master_accumulate_rank,quit_tag);
+}
+
+
+
+void accumulate_samples()
+{
+ std::vector<double> sample;
+
+ // setup the communicator for all accumulation ranks by splitting
+
+ mpi::communicator world;
+ mpi::communicator accumulate_communicator = world.split(1);
+
+ bool is_master_accumulate_rank = accumulate_communicator.rank()==0;
+
+ // the master receives the sample skeleton
+
+ if (is_master_accumulate_rank)
+ world.recv(0,sample_skeleton_tag,mpi::skeleton(sample));
+
+ // and broadcasts it to all accumulation ranks
+ mpi::broadcast(accumulate_communicator,mpi::skeleton(sample),0);
+
+ // next we extract the content of the sample vector, to be used in receiving
+ // the content later on
+
+ mpi::content sample_content = mpi::get_content(sample);
+
+ // accumulate until quit is called
+ double sum=0.;
+ while (true) {
+
+
+ // the accumulation master checks whether we should quit
+ if (world.iprobe(0,quit_tag)) {
+ world.recv(0,quit_tag);
+ for (int i=1; i<accumulate_communicator.size();++i)
+ accumulate_communicator.send(i,quit_tag);
+ std::cout << sum << "\n";
+ break; // We're done
+ }
+
+ // the otehr accumulation ranks check whether we should quit
+ if (accumulate_communicator.iprobe(0,quit_tag)) {
+ accumulate_communicator.recv(0,quit_tag);
+ std::cout << sum << "\n";
+ break; // We're done
+ }
+
+ // check whether the master accumulation rank has received a sample
+ if (world.iprobe(mpi::any_source,sample_tag)) {
+ BOOST_ASSERT(is_master_accumulate_rank);
+
+ // receive the content
+ world.recv(mpi::any_source,sample_tag,sample_content);
+
+ // now we need to braodcast
+ // the problam is we do not have a non-blocking broadcast that we could
+ // abort if we receive a quit message from the master. We thus need to
+ // first tell all accumulation ranks to start a broadcast. If the sample
+ // is small, we could just send the sample in this message, but here we
+ // optimize the code for large samples, so that the overhead of these
+ // sends can be ignored, and we count on an optimized broadcast
+ // implementation with O(log N) complexity
+
+ for (int i=1; i<accumulate_communicator.size();++i)
+ accumulate_communicator.send(i,sample_broadcast_tag);
+
+ // now broadcast the contents of the sample to all accumulate ranks
+ mpi::broadcast(accumulate_communicator,sample_content,0);
+
+ // and handle the sample by summing the appropriate value
+ sum += sample[0];
+ }
+
+ // the other accumulation ranks wait for a mesage to start the broadcast
+ if (accumulate_communicator.iprobe(0,sample_broadcast_tag)) {
+ BOOST_ASSERT(!is_master_accumulate_rank);
+
+ accumulate_communicator.recv(0,sample_broadcast_tag);
+
+ // receive broadcast of the sample contents
+ mpi::broadcast(accumulate_communicator,sample_content,0);
+
+ // and handle the sample
+
+ // and handle the sample by summing the appropriate value
+ sum += sample[accumulate_communicator.rank()];
+ }
+ }
+}
+
+int main(int argc, char** argv)
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ // half of the processes generate, the others accumulate
+ // the sample size is just the number of accumulation ranks
+ if (world.rank() < world.size()/2)
+ calculate_samples(world.size()-world.size()/2);
+ else
+ accumulate_samples();
+
+ return 0;
+}
+
+
diff --git a/src/boost/libs/mpi/example/python/hello_world.py b/src/boost/libs/mpi/example/python/hello_world.py
new file mode 100644
index 00000000..73312483
--- /dev/null
+++ b/src/boost/libs/mpi/example/python/hello_world.py
@@ -0,0 +1,16 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+import boost.parallel.mpi as mpi
+
+if mpi.world.rank == 0:
+ mpi.world.send(1, 0, 'Hello')
+ msg = mpi.world.recv(1, 1)
+ print msg,'!'
+else:
+ msg = mpi.world.recv(0, 0)
+ print msg,', ',
+ mpi.world.send(0, 1, 'world')
diff --git a/src/boost/libs/mpi/example/random_content.cpp b/src/boost/libs/mpi/example/random_content.cpp
new file mode 100644
index 00000000..22fd06d8
--- /dev/null
+++ b/src/boost/libs/mpi/example/random_content.cpp
@@ -0,0 +1,98 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// An example using Boost.MPI's skeletons and content to optimize
+// communication.
+#include <boost/mpi.hpp>
+#include <boost/serialization/list.hpp>
+#include <algorithm>
+#include <functional>
+#include <numeric>
+#include <iostream>
+#include <stdlib.h>
+namespace mpi = boost::mpi;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ if (world.size() < 2 || world.size() > 4) {
+ if (world.rank() == 0)
+ std::cerr << "error: please execute this program with 2-4 processes.\n";
+ world.abort(-1);
+ }
+
+ if (world.rank() == 0) {
+ int list_len = 50;
+ int iterations = 10;
+
+ if (argc > 1) list_len = atoi(argv[1]);
+ if (argc > 2) iterations = atoi(argv[2]);
+
+ if (list_len <= 0) {
+ std::cerr << "error: please specific a list length greater than zero.\n";
+ world.abort(-1);
+ }
+
+ // Generate the list and broadcast its structure
+ std::list<int> l(list_len);
+ broadcast(world, mpi::skeleton(l), 0);
+
+ // Generate content several times and broadcast out that content
+ mpi::content c = mpi::get_content(l);
+ for (int i = 0; i < iterations; ++i) {
+ do {
+ std::generate(l.begin(), l.end(), &random);
+ } while (std::find_if(l.begin(), l.end(),
+ std::bind1st(std::not_equal_to<int>(), 0))
+ == l.end());
+
+
+ std::cout << "Iteration #" << i << ": sending content"
+ << " (min = " << *std::min_element(l.begin(), l.end())
+ << ", max = " << *std::max_element(l.begin(), l.end())
+ << ", avg = "
+ << std::accumulate(l.begin(), l.end(), 0)/l.size()
+ << ").\n";
+
+ broadcast(world, c, 0);
+ }
+
+ // Notify the slaves that we're done by sending all zeroes
+ std::fill(l.begin(), l.end(), 0);
+ broadcast(world, c, 0);
+
+ } else {
+ // Receive the content and build up our own list
+ std::list<int> l;
+ broadcast(world, mpi::skeleton(l), 0);
+
+ mpi::content c = mpi::get_content(l);
+ int i = 0;
+ do {
+ broadcast(world, c, 0);
+
+ if (std::find_if(l.begin(), l.end(),
+ std::bind1st(std::not_equal_to<int>(), 0)) == l.end())
+ break;
+
+ if (world.rank() == 1)
+ std::cout << "Iteration #" << i << ": max value = "
+ << *std::max_element(l.begin(), l.end()) << ".\n";
+ else if (world.rank() == 2)
+ std::cout << "Iteration #" << i << ": min value = "
+ << *std::min_element(l.begin(), l.end()) << ".\n";
+ else if (world.rank() == 3)
+ std::cout << "Iteration #" << i << ": avg value = "
+ << std::accumulate(l.begin(), l.end(), 0)/l.size()
+ << ".\n";
+ ++i;
+ } while (true);
+ }
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/random_gather.cpp b/src/boost/libs/mpi/example/random_gather.cpp
new file mode 100644
index 00000000..17825f15
--- /dev/null
+++ b/src/boost/libs/mpi/example/random_gather.cpp
@@ -0,0 +1,31 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// An example using Boost.MPI's gather(): [main]
+
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <cstdlib>
+namespace mpi = boost::mpi;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ std::srand(time(0) + world.rank());
+ int my_number = std::rand();
+ if (world.rank() == 0) {
+ std::vector<int> all_numbers;
+ gather(world, my_number, all_numbers, 0);
+ for (int proc = 0; proc < world.size(); ++proc)
+ std::cout << "Process #" << proc << " thought of " << all_numbers[proc]
+ << std::endl;
+ } else {
+ gather(world, my_number, 0);
+ }
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/random_min.cpp b/src/boost/libs/mpi/example/random_min.cpp
new file mode 100644
index 00000000..d0a67ee0
--- /dev/null
+++ b/src/boost/libs/mpi/example/random_min.cpp
@@ -0,0 +1,30 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// An example using Boost.MPI's reduce() to compute a minimum value.
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <cstdlib>
+namespace mpi = boost::mpi;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ std::srand(time(0) + world.rank());
+ int my_number = std::rand();
+
+ if (world.rank() == 0) {
+ int minimum;
+ reduce(world, my_number, minimum, mpi::minimum<int>(), 0);
+ std::cout << "The minimum value is " << minimum << std::endl;
+ } else {
+ reduce(world, my_number, mpi::minimum<int>(), 0);
+ }
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/random_scatter.cpp b/src/boost/libs/mpi/example/random_scatter.cpp
new file mode 100644
index 00000000..fc8879e4
--- /dev/null
+++ b/src/boost/libs/mpi/example/random_scatter.cpp
@@ -0,0 +1,37 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// An example using Boost.MPI's gather(): [main]
+
+#include <boost/mpi.hpp>
+#include <boost/mpi/collectives.hpp>
+#include <iostream>
+#include <cstdlib>
+#include <vector>
+
+namespace mpi = boost::mpi;
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ std::srand(time(0) + world.rank());
+ std::vector<int> all;
+ int mine = -1;
+ if (world.rank() == 0) {
+ all.resize(world.size());
+ std::generate(all.begin(), all.end(), std::rand);
+ }
+ mpi::scatter(world, all, mine, 0);
+ for (int r = 0; r < world.size(); ++r) {
+ world.barrier();
+ if (r == world.rank()) {
+ std::cout << "Rank " << r << " got " << mine << '\n';
+ }
+ }
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/reduce_performance_test.cpp b/src/boost/libs/mpi/example/reduce_performance_test.cpp
new file mode 100644
index 00000000..40411b14
--- /dev/null
+++ b/src/boost/libs/mpi/example/reduce_performance_test.cpp
@@ -0,0 +1,138 @@
+// Copyright (C) 2006 Trustees of Indiana University
+//
+// Authors: Douglas Gregor
+// Andrew Lumsdaine
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Performance test of the reduce() collective
+#include <boost/mpi.hpp>
+#include <boost/lexical_cast.hpp>
+
+namespace mpi = boost::mpi;
+
+struct add_int {
+ int operator()(int x, int y) const { return x + y; }
+};
+
+struct wrapped_int
+{
+ wrapped_int() : value(0) { }
+ wrapped_int(int value) : value(value) { }
+
+ template<typename Archiver>
+ void serialize(Archiver& ar, const unsigned int /*version*/) {
+ ar & value;
+ }
+
+ int value;
+};
+
+inline wrapped_int operator+(wrapped_int x, wrapped_int y)
+{
+ return wrapped_int(x.value + y.value);
+}
+
+namespace boost { namespace mpi {
+ template<> struct is_mpi_datatype<wrapped_int> : mpl::true_ { };
+} }
+
+struct serialized_int
+{
+ serialized_int() : value(0) { }
+ serialized_int(int value) : value(value) { }
+
+ template<typename Archiver>
+ void serialize(Archiver& ar, const unsigned int /*version*/) {
+ ar & value;
+ }
+
+ int value;
+};
+
+inline serialized_int operator+(serialized_int x, serialized_int y)
+{
+ return serialized_int(x.value + y.value);
+}
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ int repeat_count = 100;
+ int outer_repeat_count = 2;
+
+ if (argc > 1) repeat_count = boost::lexical_cast<int>(argv[1]);
+ if (argc > 2) outer_repeat_count = boost::lexical_cast<int>(argv[2]);
+
+ if (world.rank() == 0)
+ std::cout << "# of processors: " << world.size() << std::endl
+ << "# of iterations: " << repeat_count << std::endl;
+
+ int value = world.rank();
+ int result;
+ wrapped_int wi_value = world.rank();
+ wrapped_int wi_result;
+ serialized_int si_value = world.rank();
+ serialized_int si_result;
+
+ // Spin for a while...
+ for (int i = 0; i < repeat_count/10; ++i) {
+ reduce(world, value, result, std::plus<int>(), 0);
+ reduce(world, value, result, add_int(), 0);
+ reduce(world, wi_value, wi_result, std::plus<wrapped_int>(), 0);
+ reduce(world, si_value, si_result, std::plus<serialized_int>(), 0);
+ }
+
+ for (int outer = 0; outer < outer_repeat_count; ++outer) {
+ // Raw MPI
+ mpi::timer time;
+ for (int i = 0; i < repeat_count; ++i) {
+ MPI_Reduce(&value, &result, 1, MPI_INT, MPI_SUM, 0, MPI_COMM_WORLD);
+ }
+ double reduce_raw_mpi_total_time = time.elapsed();
+
+ // MPI_INT/MPI_SUM case
+ time.restart();
+ for (int i = 0; i < repeat_count; ++i) {
+ reduce(world, value, result, std::plus<int>(), 0);
+ }
+ double reduce_int_sum_total_time = time.elapsed();
+
+ // MPI_INT/MPI_Op case
+ time.restart();
+ for (int i = 0; i < repeat_count; ++i) {
+ reduce(world, value, result, add_int(), 0);
+ }
+ double reduce_int_op_total_time = time.elapsed();
+
+ // MPI_Datatype/MPI_Op case
+ time.restart();
+ for (int i = 0; i < repeat_count; ++i) {
+ reduce(world, wi_value, wi_result, std::plus<wrapped_int>(), 0);
+ }
+ double reduce_type_op_total_time = time.elapsed();
+
+ // Serialized/MPI_Op case
+ time.restart();
+ for (int i = 0; i < repeat_count; ++i) {
+ reduce(world, si_value, si_result, std::plus<serialized_int>(), 0);
+ }
+ double reduce_ser_op_total_time = time.elapsed();
+
+
+ if (world.rank() == 0)
+ std::cout << "\nInvocation\tElapsed Time (seconds)"
+ << "\nRaw MPI\t\t\t" << reduce_raw_mpi_total_time
+ << "\nMPI_INT/MPI_SUM\t\t" << reduce_int_sum_total_time
+ << "\nMPI_INT/MPI_Op\t\t" << reduce_int_op_total_time
+ << "\nMPI_Datatype/MPI_Op\t" << reduce_type_op_total_time
+ << "\nSerialized/MPI_Op\t" << reduce_ser_op_total_time
+ << std::endl;
+ }
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/example/string_cat.cpp b/src/boost/libs/mpi/example/string_cat.cpp
new file mode 100644
index 00000000..5602d596
--- /dev/null
+++ b/src/boost/libs/mpi/example/string_cat.cpp
@@ -0,0 +1,46 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// An example using Boost.MPI's reduce() to concatenate strings.
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <string>
+#include <boost/serialization/string.hpp> // Important for sending strings!
+namespace mpi = boost::mpi;
+
+/* Defining STRING_CONCAT_COMMUTATIVE lies to Boost.MPI by forcing it
+ * to assume that string concatenation is commutative, which it is
+ * not. However, doing so illustrates how the results of a reduction
+ * can change when a non-commutative operator is assumed to be
+ * commutative.
+ */
+#ifdef STRING_CONCAT_COMMUTATIVE
+namespace boost { namespace mpi {
+
+template<>
+struct is_commutative<std::plus<std::string>, std::string> : mpl::true_ { };
+
+} } // end namespace boost::mpi
+#endif
+
+int main(int argc, char* argv[])
+{
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ std::string names[10] = { "zero ", "one ", "two ", "three ", "four ",
+ "five ", "six ", "seven ", "eight ", "nine " };
+
+ std::string result;
+ reduce(world,
+ world.rank() < 10? names[world.rank()] : std::string("many "),
+ result, std::plus<std::string>(), 0);
+
+ if (world.rank() == 0)
+ std::cout << "The result is " << result << std::endl;
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/index.html b/src/boost/libs/mpi/index.html
new file mode 100644
index 00000000..5dad703a
--- /dev/null
+++ b/src/boost/libs/mpi/index.html
@@ -0,0 +1,16 @@
+
+<!--
+Copyright 2005-2007 Daniel James.
+Distributed under the Boost Software License, Version 1.0. (See accompanying
+file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
+-->
+
+<html>
+<head>
+ <meta http-equiv="refresh" content="0; URL=../../doc/html/mpi.html">
+</head>
+<body>
+Automatic redirection failed, please go to
+<a href="../../doc/html/mpi.html">../../doc/html/mpi.html</a>
+</body>
+</html>
diff --git a/src/boost/libs/mpi/meta/libraries.json b/src/boost/libs/mpi/meta/libraries.json
new file mode 100644
index 00000000..ab1e98fb
--- /dev/null
+++ b/src/boost/libs/mpi/meta/libraries.json
@@ -0,0 +1,16 @@
+{
+ "key": "mpi",
+ "name": "MPI",
+ "authors": [
+ "Douglas Gregor",
+ "Matthias Troyer"
+ ],
+ "description": "Message Passing Interface library, for use in distributed-memory parallel application programming.",
+ "category": [
+ "Concurrent"
+ ],
+ "maintainers": [
+ "K. Noel Belcourt <kbelco -at- sandia.gov>",
+ "Alain Miniussi <alain.miniussi -at- oca.eu>"
+ ]
+}
diff --git a/src/boost/libs/mpi/src/broadcast.cpp b/src/boost/libs/mpi/src/broadcast.cpp
new file mode 100644
index 00000000..e171b486
--- /dev/null
+++ b/src/boost/libs/mpi/src/broadcast.cpp
@@ -0,0 +1,142 @@
+// Copyright 2005 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Message Passing Interface 1.1 -- Section 4.4. Broadcast
+
+#include <boost/mpi/config.hpp>
+#include <boost/mpi/collectives/broadcast.hpp>
+#include <boost/mpi/skeleton_and_content.hpp>
+#include <boost/mpi/detail/point_to_point.hpp>
+#include <boost/mpi/environment.hpp>
+#include <cassert>
+
+namespace boost { namespace mpi {
+
+template<>
+void
+broadcast<const packed_oarchive>(const communicator& comm,
+ const packed_oarchive& oa,
+ int root)
+{
+ // Only the root can broadcast the packed_oarchive
+ assert(comm.rank() == root);
+
+ int size = comm.size();
+ if (size < 2) return;
+
+ // Determine maximum tag value
+ int tag = environment::collectives_tag();
+
+ // Broadcast data to all nodes
+ std::vector<request> requests(size-1);
+ std::vector<request>::iterator it = requests.begin();
+ for (int dest = 0; dest < size; ++dest) {
+ if (dest != root) {
+ *it++ = detail::packed_archive_isend(comm, dest, tag, oa);
+ }
+ }
+ wait_all(requests.begin(), requests.end());
+}
+
+template<>
+void
+broadcast<packed_oarchive>(const communicator& comm, packed_oarchive& oa,
+ int root)
+{
+ broadcast(comm, const_cast<const packed_oarchive&>(oa), root);
+}
+
+template<>
+void
+broadcast<packed_iarchive>(const communicator& comm, packed_iarchive& ia,
+ int root)
+{
+ int size = comm.size();
+ if (size < 2) return;
+
+ // Determine maximum tag value
+ int tag = environment::collectives_tag();
+
+ // Receive data from the root.
+ if (comm.rank() != root) {
+ MPI_Status status;
+ detail::packed_archive_recv(comm, root, tag, ia, status);
+ } else {
+ // Broadcast data to all nodes
+ std::vector<request> requests(size-1);
+ std::vector<request>::iterator it = requests.begin();
+ for (int dest = 0; dest < size; ++dest) {
+ if (dest != root) {
+ *it++ = detail::packed_archive_isend(comm, dest, tag, ia);
+ }
+ }
+ wait_all(requests.begin(), requests.end());
+ }
+}
+
+template<>
+void
+broadcast<const packed_skeleton_oarchive>(const communicator& comm,
+ const packed_skeleton_oarchive& oa,
+ int root)
+{
+ broadcast(comm, oa.get_skeleton(), root);
+}
+
+template<>
+void
+broadcast<packed_skeleton_oarchive>(const communicator& comm,
+ packed_skeleton_oarchive& oa, int root)
+{
+ broadcast(comm, oa.get_skeleton(), root);
+}
+
+template<>
+void
+broadcast<packed_skeleton_iarchive>(const communicator& comm,
+ packed_skeleton_iarchive& ia, int root)
+{
+ broadcast(comm, ia.get_skeleton(), root);
+}
+
+template<>
+void broadcast<content>(const communicator& comm, content& c, int root)
+{
+ broadcast(comm, const_cast<const content&>(c), root);
+}
+
+template<>
+void broadcast<const content>(const communicator& comm, const content& c,
+ int root)
+{
+#if defined(BOOST_MPI_BCAST_BOTTOM_WORKS_FINE)
+ BOOST_MPI_CHECK_RESULT(MPI_Bcast,
+ (MPI_BOTTOM, 1, c.get_mpi_datatype(),
+ root, comm));
+#else
+ if (comm.size() < 2)
+ return;
+
+ // Some versions of LAM/MPI behave badly when broadcasting using
+ // MPI_BOTTOM, so we'll instead use manual send/recv operations.
+ if (comm.rank() == root) {
+ for (int p = 0; p < comm.size(); ++p) {
+ if (p != root) {
+ BOOST_MPI_CHECK_RESULT(MPI_Send,
+ (MPI_BOTTOM, 1, c.get_mpi_datatype(),
+ p, environment::collectives_tag(), comm));
+ }
+ }
+ } else {
+ BOOST_MPI_CHECK_RESULT(MPI_Recv,
+ (MPI_BOTTOM, 1, c.get_mpi_datatype(),
+ root, environment::collectives_tag(),
+ comm, MPI_STATUS_IGNORE));
+ }
+#endif
+}
+
+} } // end namespace boost::mpi
diff --git a/src/boost/libs/mpi/src/cartesian_communicator.cpp b/src/boost/libs/mpi/src/cartesian_communicator.cpp
new file mode 100644
index 00000000..a46f0bc2
--- /dev/null
+++ b/src/boost/libs/mpi/src/cartesian_communicator.cpp
@@ -0,0 +1,179 @@
+
+// Copyright Alain Miniussi 2014.
+// Distributed under the Boost Software License, Version 1.0.
+// (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Alain Miniussi
+
+#include <algorithm>
+#include <cassert>
+
+#include <boost/mpi/cartesian_communicator.hpp>
+
+namespace boost { namespace mpi {
+
+namespace {
+ template <typename T, typename A>
+ T* c_data(std::vector<T,A>& v) { return &(v[0]); }
+}
+
+std::ostream&
+operator<<(std::ostream& out, cartesian_dimension const& d) {
+ out << '(' << d.size << ',';
+ if (d.periodic) {
+ out << "periodic";
+ } else {
+ out << "bounded";
+ }
+ out << ')';
+ return out;
+}
+
+std::ostream&
+operator<<(std::ostream& out, cartesian_topology const& topo) {
+ out << '{';
+ int const sz = topo.size();
+ for (int i = 0; i < sz; ++i) {
+ out << topo[i];
+ if ( i < (sz-1) ) {
+ out << ',';
+ }
+ }
+ out << '}';
+ return out;
+}
+
+cartesian_communicator::cartesian_communicator(const communicator& comm,
+ const cartesian_topology& topology,
+ bool reorder )
+ : communicator(MPI_COMM_NULL, comm_attach)
+{
+ std::vector<int> dims(topology.size());
+ std::vector<int> periodic(topology.size());
+ int tsz = topology.size();
+ for(int i = 0; i < tsz; ++i) {
+ dims[i] = topology[i].size;
+ periodic[i] = topology[i].periodic;
+ }
+ // Fill the gaps, if any
+ if (std::count(dims.begin(), dims.end(), 0) > 0) {
+ cartesian_dimensions(comm, dims);
+ }
+ MPI_Comm newcomm;
+ BOOST_MPI_CHECK_RESULT(MPI_Cart_create,
+ ((MPI_Comm)comm, dims.size(),
+ c_data(dims), c_data(periodic),
+ int(reorder), &newcomm));
+ if(newcomm != MPI_COMM_NULL) {
+ comm_ptr.reset(new MPI_Comm(newcomm), comm_free());
+ }
+}
+
+cartesian_communicator::cartesian_communicator(const cartesian_communicator& comm,
+ const std::vector<int>& keep )
+ : communicator(MPI_COMM_NULL, comm_attach)
+{
+ int const max_dims = comm.ndims();
+ int const nbkept = keep.size();
+ assert(nbkept <= max_dims);
+ std::vector<int> bitset(max_dims, int(false));
+ for(int i = 0; i < nbkept; ++i) {
+ assert(keep[i] < max_dims);
+ bitset[keep[i]] = true;
+ }
+
+ MPI_Comm newcomm;
+ BOOST_MPI_CHECK_RESULT(MPI_Cart_sub,
+ ((MPI_Comm)comm, c_data(bitset), &newcomm));
+ if(newcomm != MPI_COMM_NULL) {
+ comm_ptr.reset(new MPI_Comm(newcomm), comm_free());
+ }
+}
+
+int
+cartesian_communicator::ndims() const {
+ int n = -1;
+ BOOST_MPI_CHECK_RESULT(MPI_Cartdim_get,
+ (MPI_Comm(*this), &n));
+ return n;
+}
+
+int
+cartesian_communicator::rank(const std::vector<int>& coords ) const {
+ int r = -1;
+ assert(int(coords.size()) == ndims());
+ BOOST_MPI_CHECK_RESULT(MPI_Cart_rank,
+ (MPI_Comm(*this), c_data(const_cast<std::vector<int>&>(coords)),
+ &r));
+ return r;
+}
+
+std::pair<int, int>
+cartesian_communicator::shifted_ranks(int dim, int disp) const {
+ std::pair<int, int> r(-1,-1);
+ assert(0 <= dim && dim < ndims());
+ BOOST_MPI_CHECK_RESULT(MPI_Cart_shift,
+ (MPI_Comm(*this), dim, disp, &(r.first), &(r.second)));
+ return r;
+}
+
+std::vector<int>
+cartesian_communicator::coordinates(int rk) const {
+ std::vector<int> cbuf(ndims());
+ BOOST_MPI_CHECK_RESULT(MPI_Cart_coords,
+ (MPI_Comm(*this), rk, cbuf.size(), c_data(cbuf) ));
+ return cbuf;
+}
+
+void
+cartesian_communicator::topology( cartesian_topology& topo,
+ std::vector<int>& coords ) const {
+ int ndims = this->ndims();
+ topo.resize(ndims);
+ coords.resize(ndims);
+ std::vector<int> cdims(ndims);
+ std::vector<int> cperiods(ndims);
+ BOOST_MPI_CHECK_RESULT(MPI_Cart_get,
+ (MPI_Comm(*this), ndims, c_data(cdims), c_data(cperiods), c_data(coords)));
+ cartesian_topology res(cdims.begin(), cperiods.begin(), ndims);
+ topo.swap(res);
+}
+
+cartesian_topology
+cartesian_communicator::topology() const {
+ cartesian_topology topo(ndims());
+ std::vector<int> coords;
+ topology(topo, coords);
+ return topo;
+}
+
+void
+cartesian_topology::split(std::vector<int>& dims, std::vector<bool>& periodics) const {
+ int ndims = size();
+ dims.resize(ndims);
+ periodics.resize(ndims);
+ for(int i = 0; i < ndims; ++i) {
+ cartesian_dimension const& d = (*this)[i];
+ dims[i] = d.size;
+ periodics[i] = d.periodic;
+ }
+}
+
+std::vector<int>&
+cartesian_dimensions(int sz, std::vector<int>& dims) {
+ int min = 1;
+ int const dimsz = dims.size();
+ for(int i = 0; i < dimsz; ++i) {
+ if (dims[i] > 0) {
+ min *= dims[i];
+ }
+ }
+ int leftover = sz % min;
+
+ BOOST_MPI_CHECK_RESULT(MPI_Dims_create,
+ (sz-leftover, dims.size(), c_data(dims)));
+ return dims;
+}
+
+} } // end namespace boost::mpi
diff --git a/src/boost/libs/mpi/src/communicator.cpp b/src/boost/libs/mpi/src/communicator.cpp
new file mode 100644
index 00000000..016c6af4
--- /dev/null
+++ b/src/boost/libs/mpi/src/communicator.cpp
@@ -0,0 +1,328 @@
+// Copyright (C) 2005, 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/group.hpp>
+#include <boost/mpi/intercommunicator.hpp>
+#include <boost/mpi/graph_communicator.hpp>
+#include <boost/mpi/cartesian_communicator.hpp>
+#include <boost/mpi/skeleton_and_content.hpp>
+#include <boost/mpi/detail/point_to_point.hpp>
+
+namespace boost { namespace mpi {
+
+/***************************************************************************
+ * status *
+ ***************************************************************************/
+bool status::cancelled() const
+{
+ int flag = 0;
+ BOOST_MPI_CHECK_RESULT(MPI_Test_cancelled, (&m_status, &flag));
+ return flag != 0;
+}
+
+/***************************************************************************
+ * communicator *
+ ***************************************************************************/
+
+communicator::communicator()
+{
+ comm_ptr.reset(new MPI_Comm(MPI_COMM_WORLD));
+}
+
+communicator::communicator(const MPI_Comm& comm, comm_create_kind kind)
+{
+ if (comm == MPI_COMM_NULL)
+ /* MPI_COMM_NULL indicates that the communicator is not usable. */
+ return;
+
+ switch (kind) {
+ case comm_duplicate:
+ {
+ MPI_Comm newcomm;
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_dup, (comm, &newcomm));
+ comm_ptr.reset(new MPI_Comm(newcomm), comm_free());
+ MPI_Comm_set_errhandler(newcomm, MPI_ERRORS_RETURN);
+ break;
+ }
+
+ case comm_take_ownership:
+ comm_ptr.reset(new MPI_Comm(comm), comm_free());
+ break;
+
+ case comm_attach:
+ comm_ptr.reset(new MPI_Comm(comm));
+ break;
+ }
+}
+
+communicator::communicator(const communicator& comm,
+ const boost::mpi::group& subgroup)
+{
+ MPI_Comm newcomm;
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_create,
+ ((MPI_Comm)comm, (MPI_Group)subgroup, &newcomm));
+ if(newcomm != MPI_COMM_NULL)
+ comm_ptr.reset(new MPI_Comm(newcomm), comm_free());
+}
+
+int communicator::size() const
+{
+ int size_;
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_size, (MPI_Comm(*this), &size_));
+ return size_;
+}
+
+int communicator::rank() const
+{
+ int rank_;
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_rank, (MPI_Comm(*this), &rank_));
+ return rank_;
+}
+
+boost::mpi::group communicator::group() const
+{
+ MPI_Group gr;
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_group, ((MPI_Comm)*this, &gr));
+ return boost::mpi::group(gr, /*adopt=*/true);
+}
+
+void communicator::send(int dest, int tag) const
+{
+ BOOST_MPI_CHECK_RESULT(MPI_Send,
+ (MPI_BOTTOM, 0, MPI_PACKED,
+ dest, tag, MPI_Comm(*this)));
+}
+
+status communicator::recv(int source, int tag) const
+{
+ status stat;
+ BOOST_MPI_CHECK_RESULT(MPI_Recv,
+ (MPI_BOTTOM, 0, MPI_PACKED,
+ source, tag, MPI_Comm(*this), &stat.m_status));
+ return stat;
+}
+
+optional<status> communicator::iprobe(int source, int tag) const
+{
+ typedef optional<status> result_type;
+
+ status stat;
+ int flag;
+ BOOST_MPI_CHECK_RESULT(MPI_Iprobe,
+ (source, tag, MPI_Comm(*this), &flag,
+ &stat.m_status));
+ if (flag) return stat;
+ else return result_type();
+}
+
+status communicator::probe(int source, int tag) const
+{
+ status stat;
+ BOOST_MPI_CHECK_RESULT(MPI_Probe,
+ (source, tag, MPI_Comm(*this), &stat.m_status));
+ return stat;
+}
+
+void (communicator::barrier)() const
+{
+ BOOST_MPI_CHECK_RESULT(MPI_Barrier, (MPI_Comm(*this)));
+}
+
+
+communicator::operator MPI_Comm() const
+{
+ if (comm_ptr) return *comm_ptr;
+ else return MPI_COMM_NULL;
+}
+
+communicator communicator::split(int color) const
+{
+ return split(color, rank());
+}
+
+communicator communicator::split(int color, int key) const
+{
+ MPI_Comm newcomm;
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_split,
+ (MPI_Comm(*this), color, key, &newcomm));
+ return communicator(newcomm, comm_take_ownership);
+}
+
+optional<intercommunicator> communicator::as_intercommunicator() const
+{
+ int flag;
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_test_inter, ((MPI_Comm)*this, &flag));
+ if (flag)
+ return intercommunicator(comm_ptr);
+ else
+ return optional<intercommunicator>();
+}
+
+bool communicator::has_graph_topology() const
+{
+ bool is_graph = false;
+ // topology test not allowed on MPI_NULL_COMM
+ if (bool(*this)) {
+ int status;
+ BOOST_MPI_CHECK_RESULT(MPI_Topo_test, ((MPI_Comm)*this, &status));
+ is_graph = status == MPI_GRAPH;
+ }
+ return is_graph;
+}
+
+optional<graph_communicator> communicator::as_graph_communicator() const
+{
+ if (has_graph_topology()) {
+ return graph_communicator(comm_ptr);
+ } else {
+ return optional<graph_communicator>();
+ }
+}
+
+bool communicator::has_cartesian_topology() const
+{
+ bool is_cart = false;
+ // topology test not allowed on MPI_NULL_COM
+ if (bool(*this)) {
+ int status;
+ BOOST_MPI_CHECK_RESULT(MPI_Topo_test, ((MPI_Comm)*this, &status));
+ is_cart = status == MPI_CART;
+ }
+ return is_cart;
+}
+
+optional<cartesian_communicator> communicator::as_cartesian_communicator() const
+{
+ if (has_cartesian_topology()) {
+ return cartesian_communicator(comm_ptr);
+ } else {
+ return optional<cartesian_communicator>();
+ }
+}
+
+void communicator::abort(int errcode) const
+{
+ BOOST_MPI_CHECK_RESULT(MPI_Abort, (MPI_Comm(*this), errcode));
+ std::abort();
+}
+
+/*************************************************************
+ * archived send/recv *
+ *************************************************************/
+template<>
+void
+communicator::send<packed_oarchive>(int dest, int tag,
+ const packed_oarchive& ar) const
+{
+ detail::packed_archive_send(*this, dest, tag, ar);
+}
+
+template<>
+void
+communicator::send<packed_skeleton_oarchive>
+ (int dest, int tag, const packed_skeleton_oarchive& ar) const
+{
+ this->send(dest, tag, ar.get_skeleton());
+}
+
+template<>
+void communicator::send<content>(int dest, int tag, const content& c) const
+{
+ BOOST_MPI_CHECK_RESULT(MPI_Send,
+ (MPI_BOTTOM, 1, c.get_mpi_datatype(),
+ dest, tag, MPI_Comm(*this)));
+}
+
+template<>
+status
+communicator::recv<packed_iarchive>(int source, int tag,
+ packed_iarchive& ar) const
+{
+ status stat;
+ detail::packed_archive_recv(*this, source, tag, ar,
+ stat.m_status);
+ return stat;
+}
+
+template<>
+status
+communicator::recv<packed_skeleton_iarchive>
+ (int source, int tag, packed_skeleton_iarchive& ar) const
+{
+ return this->recv(source, tag, ar.get_skeleton());
+}
+
+template<>
+status
+communicator::recv<const content>(int source, int tag, const content& c) const
+{
+ status stat;
+ BOOST_MPI_CHECK_RESULT(MPI_Recv,
+ (MPI_BOTTOM, 1, c.get_mpi_datatype(),
+ source, tag, MPI_Comm(*this), &stat.m_status));
+ return stat;
+}
+
+/*************************************************************
+ * non-blocking send/recv *
+ *************************************************************/
+template<>
+request
+communicator::isend<packed_oarchive>(int dest, int tag,
+ const packed_oarchive& ar) const
+{
+ return detail::packed_archive_isend(*this, dest, tag, ar);
+}
+
+template<>
+request
+communicator::isend<packed_skeleton_oarchive>
+ (int dest, int tag, const packed_skeleton_oarchive& ar) const
+{
+ return this->isend(dest, tag, ar.get_skeleton());
+}
+
+template<>
+request communicator::isend<content>(int dest, int tag, const content& c) const
+{
+ return request::make_bottom_send(*this, dest, tag, c.get_mpi_datatype());
+}
+
+request communicator::isend(int dest, int tag) const
+{
+ return request::make_empty_send(*this, dest, tag);
+}
+
+template<>
+request
+communicator::irecv<packed_skeleton_iarchive>
+ (int source, int tag, packed_skeleton_iarchive& ar) const
+{
+ return this->irecv(source, tag, ar.get_skeleton());
+}
+
+template<>
+request
+communicator::irecv<const content>(int source, int tag,
+ const content& c) const
+{
+ return request::make_bottom_recv(*this, source, tag, c.get_mpi_datatype());
+}
+
+request communicator::irecv(int source, int tag) const
+{
+ return request::make_empty_recv(*this, source, tag);
+}
+
+bool operator==(const communicator& comm1, const communicator& comm2)
+{
+ int result;
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_compare,
+ (MPI_Comm(comm1), MPI_Comm(comm2), &result));
+ return result == MPI_IDENT;
+}
+
+} } // end namespace boost::mpi
diff --git a/src/boost/libs/mpi/src/computation_tree.cpp b/src/boost/libs/mpi/src/computation_tree.cpp
new file mode 100644
index 00000000..60de534d
--- /dev/null
+++ b/src/boost/libs/mpi/src/computation_tree.cpp
@@ -0,0 +1,72 @@
+// Copyright (C) 2005 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Compute parents, children, levels, etc. to effect a parallel
+// computation tree.
+
+#include <boost/mpi/detail/computation_tree.hpp>
+
+namespace boost { namespace mpi { namespace detail {
+
+int computation_tree::default_branching_factor = 3;
+
+computation_tree
+::computation_tree(int rank, int size, int root, int branching_factor)
+ : rank(rank), size(size), root(root),
+ branching_factor_(branching_factor > 1? branching_factor
+ /* default */: default_branching_factor),
+ level_(0)
+{
+ // The position in the tree, once we've adjusted for non-zero
+ // roots.
+ int n = (rank + size - root) % size;
+ int sum = 0;
+ int term = 1;
+
+ /* The level is the smallest value of k such that
+
+ f^0 + f^1 + ... + f^k > n
+
+ for branching factor f and index n in the tree. */
+ while (sum <= n) {
+ ++level_;
+ term *= branching_factor_;
+ sum += term;
+ }
+}
+
+int computation_tree::level_index(int n) const
+{
+ int sum = 0;
+ int term = 1;
+ while (n--) {
+ sum += term;
+ term *= branching_factor_;
+ }
+ return sum;
+}
+
+int computation_tree::parent() const
+{
+ if (rank == root) return rank;
+ int n = rank + size - 1 - root;
+ return ((n % size / branching_factor_) + root) % size ;
+}
+
+int computation_tree::child_begin() const
+{
+ // Zero-based index of this node
+ int n = (rank + size - root) % size;
+
+ // Compute the index of the child (in a zero-based tree)
+ int child_index = level_index(level_ + 1)
+ + branching_factor_ * (n - level_index(level_));
+
+ if (child_index >= size) return root;
+ else return (child_index + root) % size;
+}
+
+} } } // end namespace boost::mpi::detail
diff --git a/src/boost/libs/mpi/src/content_oarchive.cpp b/src/boost/libs/mpi/src/content_oarchive.cpp
new file mode 100644
index 00000000..0ec1ece3
--- /dev/null
+++ b/src/boost/libs/mpi/src/content_oarchive.cpp
@@ -0,0 +1,20 @@
+// (C) Copyright 2005 Matthias Troyer
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Matthias Troyer
+
+#define BOOST_ARCHIVE_SOURCE
+#include <boost/archive/detail/archive_serializer_map.hpp>
+#include <boost/archive/impl/archive_serializer_map.ipp>
+#include <boost/mpi/skeleton_and_content.hpp>
+
+namespace boost { namespace archive { namespace detail {
+// explicitly instantiate all required template functions
+
+template class archive_serializer_map<mpi::detail::content_oarchive> ;
+template class archive_serializer_map<boost::mpi::detail::ignore_skeleton_oarchive<boost::mpi::detail::content_oarchive> >;
+template class archive_serializer_map<boost::mpi::detail::ignore_skeleton_oarchive<boost::mpi::detail::mpi_datatype_oarchive> >;
+} } }
diff --git a/src/boost/libs/mpi/src/environment.cpp b/src/boost/libs/mpi/src/environment.cpp
new file mode 100644
index 00000000..72b0359f
--- /dev/null
+++ b/src/boost/libs/mpi/src/environment.cpp
@@ -0,0 +1,258 @@
+// Copyright (C) 2005-2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Message Passing Interface 1.1 -- 7.1.1. Environmental Inquiries
+#include <boost/mpi/environment.hpp>
+#include <boost/mpi/exception.hpp>
+#include <boost/mpi/detail/mpi_datatype_cache.hpp>
+#include <boost/core/uncaught_exceptions.hpp>
+#include <cassert>
+#include <string>
+#include <exception>
+#include <stdexcept>
+#include <ostream>
+
+namespace boost { namespace mpi {
+namespace threading {
+std::istream& operator>>(std::istream& in, level& l)
+{
+ std::string tk;
+ in >> tk;
+ if (!in.bad()) {
+ if (tk == "single") {
+ l = single;
+ } else if (tk == "funneled") {
+ l = funneled;
+ } else if (tk == "serialized") {
+ l = serialized;
+ } else if (tk == "multiple") {
+ l = multiple;
+ } else {
+ in.setstate(std::ios::badbit);
+ }
+ }
+ return in;
+}
+
+std::ostream& operator<<(std::ostream& out, level l)
+{
+ switch(l) {
+ case single:
+ out << "single";
+ break;
+ case funneled:
+ out << "funneled";
+ break;
+ case serialized:
+ out << "serialized";
+ break;
+ case multiple:
+ out << "multiple";
+ break;
+ default:
+ out << "<level error>[" << int(l) << ']';
+ out.setstate(std::ios::badbit);
+ break;
+ }
+ return out;
+}
+
+} // namespace threading
+
+#ifdef BOOST_MPI_HAS_NOARG_INITIALIZATION
+environment::environment(bool abrt)
+ : i_initialized(false),
+ abort_on_exception(abrt)
+{
+ if (!initialized()) {
+ BOOST_MPI_CHECK_RESULT(MPI_Init, (0, 0));
+ i_initialized = true;
+ }
+
+#if (2 <= MPI_VERSION)
+ MPI_Comm_set_errhandler(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
+#else
+ MPI_Errhandler_set(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
+#endif
+}
+
+environment::environment(threading::level mt_level, bool abrt)
+ : i_initialized(false),
+ abort_on_exception(abrt)
+{
+ // It is not clear that we can pass null in MPI_Init_thread.
+ int dummy_thread_level = 0;
+ if (!initialized()) {
+ BOOST_MPI_CHECK_RESULT(MPI_Init_thread,
+ (0, 0, int(mt_level), &dummy_thread_level ));
+ i_initialized = true;
+ }
+
+#if (2 <= MPI_VERSION)
+ MPI_Comm_set_errhandler(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
+#else
+ MPI_Errhandler_set(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
+#endif
+}
+#endif
+
+environment::environment(int& argc, char** &argv, bool abrt)
+ : i_initialized(false),
+ abort_on_exception(abrt)
+{
+ if (!initialized()) {
+ BOOST_MPI_CHECK_RESULT(MPI_Init, (&argc, &argv));
+ i_initialized = true;
+ }
+
+#if (2 <= MPI_VERSION)
+ MPI_Comm_set_errhandler(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
+#else
+ MPI_Errhandler_set(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
+#endif
+}
+
+environment::environment(int& argc, char** &argv, threading::level mt_level,
+ bool abrt)
+ : i_initialized(false),
+ abort_on_exception(abrt)
+{
+ // It is not clear that we can pass null in MPI_Init_thread.
+ int dummy_thread_level = 0;
+ if (!initialized()) {
+ BOOST_MPI_CHECK_RESULT(MPI_Init_thread,
+ (&argc, &argv, int(mt_level), &dummy_thread_level));
+ i_initialized = true;
+ }
+
+#if (2 <= MPI_VERSION)
+ MPI_Comm_set_errhandler(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
+#else
+ MPI_Errhandler_set(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
+#endif
+}
+
+environment::~environment()
+{
+ if (i_initialized) {
+ if (boost::core::uncaught_exceptions() > 0 && abort_on_exception) {
+ abort(-1);
+ } else if (!finalized()) {
+ detail::mpi_datatype_cache().clear();
+ BOOST_MPI_CHECK_RESULT(MPI_Finalize, ());
+ }
+ }
+}
+
+void environment::abort(int errcode)
+{
+ BOOST_MPI_CHECK_RESULT(MPI_Abort, (MPI_COMM_WORLD, errcode));
+}
+
+bool environment::initialized()
+{
+ int flag;
+ BOOST_MPI_CHECK_RESULT(MPI_Initialized, (&flag));
+ return flag != 0;
+}
+
+bool environment::finalized()
+{
+ int flag;
+ BOOST_MPI_CHECK_RESULT(MPI_Finalized, (&flag));
+ return flag != 0;
+}
+
+int environment::max_tag()
+{
+ int* max_tag_value;
+ int found = 0;
+
+#if (2 <= MPI_VERSION)
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_get_attr,
+ (MPI_COMM_WORLD, MPI_TAG_UB, &max_tag_value, &found));
+#else
+ BOOST_MPI_CHECK_RESULT(MPI_Attr_get,
+ (MPI_COMM_WORLD, MPI_TAG_UB, &max_tag_value, &found));
+#endif
+ assert(found != 0);
+ return *max_tag_value - num_reserved_tags;
+}
+
+int environment::collectives_tag()
+{
+ return max_tag() + 1;
+}
+
+optional<int> environment::host_rank()
+{
+ int* host;
+ int found = 0;
+
+#if (2 <= MPI_VERSION)
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_get_attr,
+ (MPI_COMM_WORLD, MPI_HOST, &host, &found));
+#else
+ BOOST_MPI_CHECK_RESULT(MPI_Attr_get,
+ (MPI_COMM_WORLD, MPI_HOST, &host, &found));
+#endif
+ if (!found || *host == MPI_PROC_NULL)
+ return optional<int>();
+ else
+ return *host;
+}
+
+optional<int> environment::io_rank()
+{
+ int* io;
+ int found = 0;
+
+#if (2 <= MPI_VERSION)
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_get_attr,
+ (MPI_COMM_WORLD, MPI_IO, &io, &found));
+#else
+ BOOST_MPI_CHECK_RESULT(MPI_Attr_get,
+ (MPI_COMM_WORLD, MPI_IO, &io, &found));
+#endif
+ if (!found || *io == MPI_PROC_NULL)
+ return optional<int>();
+ else
+ return *io;
+}
+
+std::string environment::processor_name()
+{
+ char name[MPI_MAX_PROCESSOR_NAME];
+ int len;
+
+ BOOST_MPI_CHECK_RESULT(MPI_Get_processor_name, (name, &len));
+ return std::string(name, len);
+}
+
+threading::level environment::thread_level()
+{
+ int level;
+
+ BOOST_MPI_CHECK_RESULT(MPI_Query_thread, (&level));
+ return static_cast<threading::level>(level);
+}
+
+bool environment::is_main_thread()
+{
+ int isit;
+
+ BOOST_MPI_CHECK_RESULT(MPI_Is_thread_main, (&isit));
+ return static_cast<bool>(isit);
+}
+
+std::pair<int, int> environment::version()
+{
+ int major, minor;
+ BOOST_MPI_CHECK_RESULT(MPI_Get_version, (&major, &minor));
+ return std::make_pair(major, minor);
+}
+
+} } // end namespace boost::mpi
diff --git a/src/boost/libs/mpi/src/error_string.cpp b/src/boost/libs/mpi/src/error_string.cpp
new file mode 100644
index 00000000..a6db48a6
--- /dev/null
+++ b/src/boost/libs/mpi/src/error_string.cpp
@@ -0,0 +1,31 @@
+// Copyright (C) 2018 Alain Miniussi <alain.miniussi -at- oca.eu>.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+#include <sstream>
+#include <boost/mpi/error_string.hpp>
+
+namespace boost { namespace mpi {
+
+std::string error_string(int err)
+{
+ char buffer[MPI_MAX_ERROR_STRING];
+ int len;
+ int status = MPI_Error_string(err, buffer, &len);
+ if (status == MPI_SUCCESS) {
+ return std::string(buffer);
+ } else {
+ std::ostringstream out;
+ if (status == MPI_ERR_ARG) {
+ out << "<invalid MPI error code " << err << ">";
+ } else {
+ out << "<got error " << status
+ << " while probing MPI error " << err << ">";
+ }
+ return out.str();
+ }
+}
+
+} }
diff --git a/src/boost/libs/mpi/src/exception.cpp b/src/boost/libs/mpi/src/exception.cpp
new file mode 100644
index 00000000..aaae9e37
--- /dev/null
+++ b/src/boost/libs/mpi/src/exception.cpp
@@ -0,0 +1,24 @@
+// Copyright (C) 2007 Trustees of Indiana University
+
+// Authors: Douglas Gregor
+// Andrew Lumsdaine
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+#include <boost/mpi/exception.hpp>
+#include <boost/mpi/error_string.hpp>
+
+namespace boost { namespace mpi {
+
+exception::exception(const char* routine, int result_code)
+ : routine_(routine), result_code_(result_code)
+{
+ message.append(routine_);
+ message.append(": ");
+ message.append(error_string(result_code));
+}
+
+exception::~exception() throw() { }
+
+} } // end namespace boost::mpi
diff --git a/src/boost/libs/mpi/src/graph_communicator.cpp b/src/boost/libs/mpi/src/graph_communicator.cpp
new file mode 100644
index 00000000..586b57bf
--- /dev/null
+++ b/src/boost/libs/mpi/src/graph_communicator.cpp
@@ -0,0 +1,73 @@
+// Copyright (C) 2007 Trustees of Indiana University
+
+// Authors: Douglas Gregor
+// Andrew Lumsdaine
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+#include <boost/mpi/graph_communicator.hpp>
+
+namespace boost { namespace mpi {
+
+// Incidence Graph requirements
+std::pair<detail::comm_out_edge_iterator, detail::comm_out_edge_iterator>
+out_edges(int vertex, const graph_communicator& comm)
+{
+ int nneighbors = out_degree(vertex, comm);
+ shared_array<int> neighbors(new int[nneighbors]);
+ BOOST_MPI_CHECK_RESULT(MPI_Graph_neighbors,
+ ((MPI_Comm)comm, vertex, nneighbors, neighbors.get()));
+ return std::make_pair(detail::comm_out_edge_iterator(vertex, neighbors, 0),
+ detail::comm_out_edge_iterator(vertex, neighbors,
+ nneighbors));
+}
+
+int out_degree(int vertex, const graph_communicator& comm)
+{
+ int nneighbors;
+ BOOST_MPI_CHECK_RESULT(MPI_Graph_neighbors_count,
+ ((MPI_Comm)comm, vertex, &nneighbors));
+ return nneighbors;
+}
+
+// Adjacency Graph requirements
+std::pair<detail::comm_adj_iterator, detail::comm_adj_iterator>
+adjacent_vertices(int vertex, const graph_communicator& comm)
+{
+ int nneighbors = out_degree(vertex, comm);
+ shared_array<int> neighbors(new int[nneighbors]);
+ BOOST_MPI_CHECK_RESULT(MPI_Graph_neighbors,
+ ((MPI_Comm)comm, vertex, nneighbors, neighbors.get()));
+ return std::make_pair(detail::comm_adj_iterator(neighbors, 0),
+ detail::comm_adj_iterator(neighbors, nneighbors));
+}
+
+// Edge List Graph requirements
+std::pair<detail::comm_edge_iterator, detail::comm_edge_iterator>
+edges(const graph_communicator& comm);
+
+std::pair<detail::comm_edge_iterator, detail::comm_edge_iterator>
+edges(const graph_communicator& comm)
+{
+ int nnodes, nedges;
+ BOOST_MPI_CHECK_RESULT(MPI_Graphdims_get, ((MPI_Comm)comm, &nnodes, &nedges));
+
+ shared_array<int> indices(new int[nnodes]);
+ shared_array<int> edges(new int[nedges]);
+ BOOST_MPI_CHECK_RESULT(MPI_Graph_get,
+ ((MPI_Comm)comm, nnodes, nedges,
+ indices.get(), edges.get()));
+ return std::make_pair(detail::comm_edge_iterator(indices, edges),
+ detail::comm_edge_iterator(nedges));
+}
+
+
+int num_edges(const graph_communicator& comm)
+{
+ int nnodes, nedges;
+ BOOST_MPI_CHECK_RESULT(MPI_Graphdims_get, ((MPI_Comm)comm, &nnodes, &nedges));
+ return nedges;
+}
+
+} } // end namespace boost::mpi
diff --git a/src/boost/libs/mpi/src/group.cpp b/src/boost/libs/mpi/src/group.cpp
new file mode 100644
index 00000000..034d08f7
--- /dev/null
+++ b/src/boost/libs/mpi/src/group.cpp
@@ -0,0 +1,106 @@
+// Copyright (C) 2007 Trustees of Indiana University
+
+// Authors: Douglas Gregor
+// Andrew Lumsdaine
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+#include <boost/mpi/group.hpp>
+#include <boost/mpi/communicator.hpp>
+
+namespace boost { namespace mpi {
+
+group::group(const MPI_Group& in_group, bool adopt)
+{
+ if (in_group != MPI_GROUP_EMPTY) {
+ if (adopt) group_ptr.reset(new MPI_Group(in_group), group_free());
+ else group_ptr.reset(new MPI_Group(in_group));
+ }
+}
+
+optional<int> group::rank() const
+{
+ if (!group_ptr)
+ return optional<int>();
+
+ int rank;
+ BOOST_MPI_CHECK_RESULT(MPI_Group_rank, (*group_ptr, &rank));
+ if (rank == MPI_UNDEFINED)
+ return optional<int>();
+ else
+ return rank;
+}
+
+int group::size() const
+{
+ if (!group_ptr)
+ return 0;
+
+ int size;
+ BOOST_MPI_CHECK_RESULT(MPI_Group_size, (*group_ptr, &size));
+ return size;
+}
+
+bool operator==(const group& g1, const group& g2)
+{
+ int result;
+ BOOST_MPI_CHECK_RESULT(MPI_Group_compare,
+ ((MPI_Group)g1, (MPI_Group)g2, &result));
+ return result == MPI_IDENT;
+}
+
+group operator|(const group& g1, const group& g2)
+{
+ MPI_Group result;
+ BOOST_MPI_CHECK_RESULT(MPI_Group_union,
+ ((MPI_Group)g1, (MPI_Group)g2, &result));
+ return group(result, /*adopt=*/true);
+}
+
+group operator&(const group& g1, const group& g2)
+{
+ MPI_Group result;
+ BOOST_MPI_CHECK_RESULT(MPI_Group_intersection,
+ ((MPI_Group)g1, (MPI_Group)g2, &result));
+ return group(result, /*adopt=*/true);
+}
+
+group operator-(const group& g1, const group& g2)
+{
+ MPI_Group result;
+ BOOST_MPI_CHECK_RESULT(MPI_Group_difference,
+ ((MPI_Group)g1, (MPI_Group)g2, &result));
+ return group(result, /*adopt=*/true);
+}
+
+template<>
+int*
+group::translate_ranks(int* first, int* last, const group& to_group, int* out)
+{
+ BOOST_MPI_CHECK_RESULT(MPI_Group_translate_ranks,
+ ((MPI_Group)*this,
+ last-first,
+ first,
+ (MPI_Group)to_group,
+ out));
+ return out + (last - first);
+}
+
+template<> group group::include(int* first, int* last)
+{
+ MPI_Group result;
+ BOOST_MPI_CHECK_RESULT(MPI_Group_incl,
+ ((MPI_Group)*this, last - first, first, &result));
+ return group(result, /*adopt=*/true);
+}
+
+template<> group group::exclude(int* first, int* last)
+{
+ MPI_Group result;
+ BOOST_MPI_CHECK_RESULT(MPI_Group_excl,
+ ((MPI_Group)*this, last - first, first, &result));
+ return group(result, /*adopt=*/true);
+}
+
+} } // end namespace boost::mpi
diff --git a/src/boost/libs/mpi/src/intercommunicator.cpp b/src/boost/libs/mpi/src/intercommunicator.cpp
new file mode 100644
index 00000000..6b072853
--- /dev/null
+++ b/src/boost/libs/mpi/src/intercommunicator.cpp
@@ -0,0 +1,54 @@
+// Copyright (C) 2007 Trustees of Indiana University
+
+// Authors: Douglas Gregor
+// Andrew Lumsdaine
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+#include <boost/mpi/intercommunicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include <boost/mpi/group.hpp>
+
+namespace boost { namespace mpi {
+
+intercommunicator::intercommunicator(const communicator& local,
+ int local_leader,
+ const communicator& peer,
+ int remote_leader)
+{
+ MPI_Comm comm;
+ BOOST_MPI_CHECK_RESULT(MPI_Intercomm_create,
+ ((MPI_Comm)local, local_leader,
+ (MPI_Comm)peer, remote_leader,
+ environment::collectives_tag(), &comm));
+ comm_ptr.reset(new MPI_Comm(comm), comm_free());
+}
+
+boost::mpi::group intercommunicator::local_group() const
+{
+ return this->group();
+}
+
+int intercommunicator::remote_size() const
+{
+ int size;
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_remote_size, ((MPI_Comm)*this, &size));
+ return size;
+}
+
+boost::mpi::group intercommunicator::remote_group() const
+{
+ MPI_Group gr;
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_remote_group, ((MPI_Comm)*this, &gr));
+ return boost::mpi::group(gr, /*adopt=*/true);
+}
+
+communicator intercommunicator::merge(bool high) const
+{
+ MPI_Comm comm;
+ BOOST_MPI_CHECK_RESULT(MPI_Intercomm_merge, ((MPI_Comm)*this, high, &comm));
+ return communicator(comm, comm_take_ownership);
+}
+
+} } // end namespace boost::mpi
diff --git a/src/boost/libs/mpi/src/mpi_datatype_cache.cpp b/src/boost/libs/mpi/src/mpi_datatype_cache.cpp
new file mode 100644
index 00000000..8437e9d0
--- /dev/null
+++ b/src/boost/libs/mpi/src/mpi_datatype_cache.cpp
@@ -0,0 +1,66 @@
+// (C) Copyright 2005 Matthias Troyer
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Matthias Troyer
+
+#include <boost/archive/detail/archive_serializer_map.hpp>
+#include <boost/mpi/detail/mpi_datatype_cache.hpp>
+#include <map>
+
+namespace boost { namespace mpi { namespace detail {
+
+ typedef std::map<std::type_info const*,MPI_Datatype,type_info_compare>
+ stored_map_type;
+
+ struct mpi_datatype_map::implementation
+ {
+ stored_map_type map;
+ };
+
+ mpi_datatype_map::mpi_datatype_map()
+ {
+ impl = new implementation();
+ }
+
+ void mpi_datatype_map::clear()
+ {
+ // do not free after call to MPI_FInalize
+ int finalized=0;
+ BOOST_MPI_CHECK_RESULT(MPI_Finalized,(&finalized));
+ if (!finalized) {
+ // ignore errors in the destructor
+ for (stored_map_type::iterator it=impl->map.begin(); it != impl->map.end(); ++it)
+ MPI_Type_free(&(it->second));
+ }
+ }
+
+
+ mpi_datatype_map::~mpi_datatype_map()
+ {
+ clear();
+ delete impl;
+ }
+
+ MPI_Datatype mpi_datatype_map::get(const std::type_info* t)
+ {
+ stored_map_type::iterator pos = impl->map.find(t);
+ if (pos != impl->map.end())
+ return pos->second;
+ else
+ return MPI_DATATYPE_NULL;
+ }
+
+ void mpi_datatype_map::set(const std::type_info* t, MPI_Datatype datatype)
+ {
+ impl->map[t] = datatype;
+ }
+
+ mpi_datatype_map& mpi_datatype_cache()
+ {
+ static mpi_datatype_map cache;
+ return cache;
+ }
+} } }
diff --git a/src/boost/libs/mpi/src/mpi_datatype_oarchive.cpp b/src/boost/libs/mpi/src/mpi_datatype_oarchive.cpp
new file mode 100644
index 00000000..05d58427
--- /dev/null
+++ b/src/boost/libs/mpi/src/mpi_datatype_oarchive.cpp
@@ -0,0 +1,19 @@
+// (C) Copyright 2005 Matthias Troyer
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Matthias Troyer
+
+#define BOOST_ARCHIVE_SOURCE
+#include <boost/archive/detail/archive_serializer_map.hpp>
+#include <boost/archive/impl/archive_serializer_map.ipp>
+#include <boost/mpi/detail/mpi_datatype_oarchive.hpp>
+
+namespace boost { namespace archive { namespace detail {
+// explicitly instantiate all required template functions
+
+template class archive_serializer_map<mpi::detail::mpi_datatype_oarchive> ;
+
+} } }
diff --git a/src/boost/libs/mpi/src/offsets.cpp b/src/boost/libs/mpi/src/offsets.cpp
new file mode 100644
index 00000000..2382d456
--- /dev/null
+++ b/src/boost/libs/mpi/src/offsets.cpp
@@ -0,0 +1,96 @@
+// Copyright Alain Miniussi 2014.
+// Distributed under the Boost Software License, Version 1.0.
+// (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Alain Miniussi
+
+#include <boost/mpi/detail/offsets.hpp>
+#include <boost/mpi/detail/antiques.hpp>
+
+namespace boost { namespace mpi {
+namespace detail {
+
+// Convert a sequence of sizes [S0..Sn] to a sequence displacement
+// [O0..On] where O[0] = 0 and O[k+1] = O[k]+S[k].
+void
+sizes2offsets(int const* sizes, int* offsets, int n)
+{
+ offsets[0] = 0;
+ for(int i = 1; i < n; ++i) {
+ offsets[i] = offsets[i-1] + sizes[i-1];
+ }
+}
+
+// Convert a sequence of sizes [S0..Sn] to a sequence displacement
+// [O0..On] where O[0] = 0 and O[k+1] = O[k]+S[k].
+void
+sizes2offsets(std::vector<int> const& sizes, std::vector<int>& offsets)
+{
+ int sz = sizes.size();
+ offsets.resize(sz);
+ sizes2offsets(c_data(sizes), c_data(offsets), sz);
+}
+
+// Given a sequence of sizes (typically the number of records dispatched
+// to each process in a scater) and a sequence of displacements (typically the
+// slot index at with those record starts), convert the later to a number
+// of skipped slots.
+void
+offsets2skipped(int const* sizes, int const* offsets, int* skipped, int n)
+{
+ skipped[0] = 0;
+ for(int i = 1; i < n; ++i) {
+ skipped[i] -= offsets[i-1] + sizes[i-1];
+ }
+}
+
+// Reconstruct offsets from sizes assuming no padding.
+// Only takes place if on the root process and if
+// displs are not already provided.
+// If memory was allocated, returns a pointer to it
+// otherwise null.
+int*
+make_offsets(communicator const& comm, int const* sizes, int const* displs, int root)
+{
+ if (root == -1 || root == comm.rank()) {
+ assert(sizes);
+ if (!displs) {
+ int nproc = comm.size();
+ int* offsets = new int[nproc];
+ displs = offsets;
+ sizes2offsets(sizes, offsets, nproc);
+ return offsets;
+ } else {
+ return 0;
+ }
+ } else {
+ return 0;
+ }
+}
+
+// Reconstruct skip slots from sizes and offsets.
+// Only takes place if on the root process and if
+// displs are provided.
+// If memory was allocated, returns a pointer to it
+// otherwise null.
+int*
+make_skipped_slots(communicator const& comm, int const* sizes, int const* displs, int root)
+{
+ if (root == -1 || root == comm.rank()) {
+ assert(sizes);
+ if (displs) {
+ int nproc = comm.size();
+ int* skipped = new int[nproc];
+ std::copy(displs, displs+nproc, skipped);
+ offsets2skipped(sizes, displs, skipped, nproc);
+ return skipped;
+ } else {
+ return 0;
+ }
+ } else {
+ return 0;
+ }
+}
+}
+}}
diff --git a/src/boost/libs/mpi/src/packed_iarchive.cpp b/src/boost/libs/mpi/src/packed_iarchive.cpp
new file mode 100644
index 00000000..a0ea5a6b
--- /dev/null
+++ b/src/boost/libs/mpi/src/packed_iarchive.cpp
@@ -0,0 +1,21 @@
+// (C) Copyright 2005 Matthias Troyer
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Matthias Troyer
+
+#define BOOST_ARCHIVE_SOURCE
+#include <boost/mpi/packed_iarchive.hpp>
+
+#include <boost/archive/detail/archive_serializer_map.hpp>
+#include <boost/archive/impl/archive_serializer_map.ipp>
+
+namespace boost { namespace archive {
+
+// explicitly instantiate all required templates
+
+template class detail::archive_serializer_map<mpi::packed_iarchive> ;
+
+} } // end namespace boost::archive
diff --git a/src/boost/libs/mpi/src/packed_oarchive.cpp b/src/boost/libs/mpi/src/packed_oarchive.cpp
new file mode 100644
index 00000000..d340a40f
--- /dev/null
+++ b/src/boost/libs/mpi/src/packed_oarchive.cpp
@@ -0,0 +1,19 @@
+// (C) Copyright 2005 Matthias Troyer
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Matthias Troyer
+
+#define BOOST_ARCHIVE_SOURCE
+#include <boost/mpi/packed_oarchive.hpp>
+#include <boost/archive/detail/archive_serializer_map.hpp>
+#include <boost/archive/impl/archive_serializer_map.ipp>
+
+namespace boost { namespace archive {
+// explicitly instantiate all required templates
+
+template class detail::archive_serializer_map<mpi::packed_oarchive> ;
+
+} } // end namespace boost::archive
diff --git a/src/boost/libs/mpi/src/packed_skeleton_iarchive.cpp b/src/boost/libs/mpi/src/packed_skeleton_iarchive.cpp
new file mode 100644
index 00000000..97115c20
--- /dev/null
+++ b/src/boost/libs/mpi/src/packed_skeleton_iarchive.cpp
@@ -0,0 +1,25 @@
+// (C) Copyright 2005 Matthias Troyer
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Matthias Troyer
+
+#define BOOST_ARCHIVE_SOURCE
+
+#include <boost/archive/detail/archive_serializer_map.hpp>
+#include <boost/archive/impl/archive_serializer_map.ipp>
+#include <boost/mpi/skeleton_and_content.hpp>
+
+namespace boost { namespace archive {
+
+// explicitly instantiate all required templates
+
+// template class basic_binary_iarchive<mpi::packed_skeleton_iarchive> ;
+template class detail::archive_serializer_map<mpi::packed_skeleton_iarchive> ;
+template class detail::archive_serializer_map<
+ mpi::detail::forward_skeleton_iarchive<
+ boost::mpi::packed_skeleton_iarchive, boost::mpi::packed_iarchive> > ;
+
+} } // end namespace boost::archive
diff --git a/src/boost/libs/mpi/src/packed_skeleton_oarchive.cpp b/src/boost/libs/mpi/src/packed_skeleton_oarchive.cpp
new file mode 100644
index 00000000..5cfecc07
--- /dev/null
+++ b/src/boost/libs/mpi/src/packed_skeleton_oarchive.cpp
@@ -0,0 +1,22 @@
+// (C) Copyright 2005 Matthias Troyer
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Matthias Troyer
+
+#define BOOST_ARCHIVE_SOURCE
+#include <boost/mpi/skeleton_and_content.hpp>
+#include <boost/archive/detail/archive_serializer_map.hpp>
+#include <boost/archive/impl/archive_serializer_map.ipp>
+
+namespace boost { namespace archive {
+// explicitly instantiate all required templates
+
+template class detail::archive_serializer_map<mpi::packed_skeleton_oarchive> ;
+template class detail::archive_serializer_map<
+ mpi::detail::forward_skeleton_oarchive<
+ boost::mpi::packed_skeleton_oarchive, boost::mpi::packed_oarchive> > ;
+
+} } // end namespace boost::archive
diff --git a/src/boost/libs/mpi/src/point_to_point.cpp b/src/boost/libs/mpi/src/point_to_point.cpp
new file mode 100644
index 00000000..6fc0ad17
--- /dev/null
+++ b/src/boost/libs/mpi/src/point_to_point.cpp
@@ -0,0 +1,102 @@
+// Copyright 2005 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Message Passing Interface 1.1 -- Section 3. MPI Point-to-point
+
+/* There is the potential for optimization here. We could keep around
+ a "small message" buffer of size N that we just receive into by
+ default. If the message is N - sizeof(int) bytes or smaller, it can
+ just be sent with that buffer. If it's larger, we send the first N
+ - sizeof(int) bytes in the first packet followed by another
+ packet. The size of the second packet will be stored in an integer
+ at the end of the first packet.
+
+ We will introduce this optimization later, when we have more
+ performance test cases and have met our functionality goals. */
+
+#include <boost/mpi/detail/point_to_point.hpp>
+#include <boost/mpi/datatype.hpp>
+#include <boost/mpi/exception.hpp>
+#include <boost/mpi/request.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/detail/antiques.hpp>
+#include <cassert>
+
+namespace boost { namespace mpi { namespace detail {
+
+void
+packed_archive_send(communicator const& comm, int dest, int tag,
+ const packed_oarchive& ar)
+{
+#if defined(BOOST_MPI_USE_IMPROBE)
+ {
+ void *buf = detail::unconst(ar.address());
+ BOOST_MPI_CHECK_RESULT(MPI_Send,
+ (buf, ar.size(), MPI_PACKED,
+ dest, tag, comm));
+ }
+#else
+ {
+ std::size_t const& size = ar.size();
+ BOOST_MPI_CHECK_RESULT(MPI_Send,
+ (detail::unconst(&size), 1,
+ get_mpi_datatype(size),
+ dest, tag, comm));
+ BOOST_MPI_CHECK_RESULT(MPI_Send,
+ (detail::unconst(ar.address()), size,
+ MPI_PACKED,
+ dest, tag, comm));
+ }
+#endif
+}
+
+request
+packed_archive_isend(communicator const& comm, int dest, int tag,
+ const packed_oarchive& ar)
+{
+ return request::make_packed_send(comm, dest, tag,
+ detail::unconst(ar.address()), ar.size());
+}
+
+request
+packed_archive_isend(communicator const& comm, int dest, int tag,
+ const packed_iarchive& ar)
+{
+ return request::make_packed_send(comm, dest, tag,
+ detail::unconst(ar.address()), ar.size());
+}
+
+void
+packed_archive_recv(communicator const& comm, int source, int tag, packed_iarchive& ar,
+ MPI_Status& status)
+{
+#if defined(BOOST_MPI_USE_IMPROBE)
+ {
+ MPI_Message msg;
+ BOOST_MPI_CHECK_RESULT(MPI_Mprobe, (source, tag, comm, &msg, &status));
+ int count;
+ BOOST_MPI_CHECK_RESULT(MPI_Get_count, (&status, MPI_PACKED, &count));
+ ar.resize(count);
+ BOOST_MPI_CHECK_RESULT(MPI_Mrecv, (ar.address(), count, MPI_PACKED, &msg, &status));
+ }
+#else
+ {
+ std::size_t count;
+ BOOST_MPI_CHECK_RESULT(MPI_Recv,
+ (&count, 1, get_mpi_datatype(count),
+ source, tag, comm, &status));
+
+ // Prepare input buffer and receive the message
+ ar.resize(count);
+ BOOST_MPI_CHECK_RESULT(MPI_Recv,
+ (ar.address(), count, MPI_PACKED,
+ status.MPI_SOURCE, status.MPI_TAG,
+ comm, &status));
+ }
+#endif
+}
+
+} } } // end namespace boost::mpi::detail
diff --git a/src/boost/libs/mpi/src/python/collectives.cpp b/src/boost/libs/mpi/src/python/collectives.cpp
new file mode 100644
index 00000000..fc4bf7b4
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/collectives.cpp
@@ -0,0 +1,144 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file status.cpp
+ *
+ * This file reflects the Boost.MPI @c status class into
+ * Python.
+ */
+#include <boost/python.hpp>
+#include <boost/mpi.hpp>
+#include <boost/mpi/python/serialize.hpp>
+
+using namespace boost::python;
+using namespace boost::mpi;
+
+namespace boost { namespace mpi { namespace python {
+
+extern const char* all_gather_docstring;
+extern const char* all_reduce_docstring;
+extern const char* all_to_all_docstring;
+extern const char* broadcast_docstring;
+extern const char* gather_docstring;
+extern const char* reduce_docstring;
+extern const char* scan_docstring;
+extern const char* scatter_docstring;
+
+object all_gather(const communicator& comm, object value)
+{
+ std::vector<object> values;
+ boost::mpi::all_gather(comm, value, values);
+
+ boost::python::list l;
+ for (int i = 0; i < comm.size(); ++i)
+ l.append(values[i]);
+ return boost::python::tuple(l);
+}
+
+object all_to_all(const communicator& comm, object in_values)
+{
+ // Build input values
+ std::vector<object> in_values_vec(comm.size());
+ object iterator = object(handle<>(PyObject_GetIter(in_values.ptr())));
+ for (int i = 0; i < comm.size(); ++i)
+ in_values_vec[i] = object(handle<>(PyIter_Next(iterator.ptr())));
+
+ std::vector<object> out_values_vec(comm.size());
+ boost::mpi::all_to_all(comm, in_values_vec, out_values_vec);
+
+ boost::python::list l;
+ for (int i = 0; i < comm.size(); ++i)
+ l.append(out_values_vec[i]);
+ return boost::python::tuple(l);
+}
+
+object broadcast(const communicator& comm, object value, int root)
+{
+ boost::mpi::broadcast(comm, value, root);
+ return value;
+}
+
+object gather(const communicator& comm, object value, int root)
+{
+ if (comm.rank() == root) {
+ std::vector<object> values;
+ boost::mpi::gather(comm, value, values, root);
+
+ boost::python::list l;
+ for (int i = 0; i < comm.size(); ++i)
+ l.append(values[i]);
+ return boost::python::tuple(l);
+ } else {
+ boost::mpi::gather(comm, value, root);
+ return object();
+ }
+}
+
+object reduce(const communicator& comm, object value, object op, int root)
+{
+ if (comm.rank() == root) {
+ object out_value;
+ boost::mpi::reduce(comm, value, out_value, op, root);
+ return out_value;
+ } else {
+ boost::mpi::reduce(comm, value, op, root);
+ return object();
+ }
+}
+
+object scatter(const communicator& comm, object values, int root)
+{
+ object result;
+
+ if (comm.rank() == root) {
+ std::vector<object> values_vec(comm.size());
+ object iterator = object(handle<>(PyObject_GetIter(values.ptr())));
+ for (int i = 0; i < comm.size(); ++i)
+ values_vec[i] = object(handle<>(PyIter_Next(iterator.ptr())));
+
+ boost::mpi::scatter(comm, values_vec, result, root);
+ } else {
+ boost::mpi::scatter(comm, result, root);
+ }
+ return result;
+}
+
+void export_collectives()
+{
+ using boost::python::arg;
+
+ def("all_reduce",
+ (object (*)(const communicator&, const object&, object))&all_reduce,
+ (arg("comm") = communicator(), arg("value"), arg("op")),
+ all_reduce_docstring);
+ def("all_gather", &all_gather,
+ (arg("comm") = communicator(), arg("value") = object()),
+ all_gather_docstring);
+ def("all_to_all", &all_to_all,
+ (arg("comm") = communicator(), arg("values") = object()),
+ all_to_all_docstring);
+ def("broadcast", &broadcast,
+ (arg("comm") = communicator(), arg("value") = object(), arg("root")),
+ broadcast_docstring);
+ def("gather", &gather,
+ (arg("comm") = communicator(), arg("value") = object(), arg("root")),
+ gather_docstring);
+ def("reduce", &reduce,
+ (arg("comm") = communicator(), arg("value"), arg("op"),
+ arg("root")),
+ reduce_docstring);
+ def("scan",
+ (object (*)(const communicator&, const object&, object))&scan,
+ (arg("comm") = communicator(), arg("value"), arg("op")),
+ scan_docstring);
+ def("scatter", &scatter,
+ (arg("comm") = communicator(), arg("values") = object(), arg("root")),
+ scatter_docstring);
+}
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/datatypes.cpp b/src/boost/libs/mpi/src/python/datatypes.cpp
new file mode 100644
index 00000000..48d7f6f7
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/datatypes.cpp
@@ -0,0 +1,27 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file datatypes.cpp
+ *
+ * This file provides datatypes support for Boost.MPI in Python.
+ */
+#include <boost/mpi/python/serialize.hpp>
+#include <boost/mpi.hpp>
+
+namespace boost { namespace mpi { namespace python {
+
+void export_datatypes()
+{
+#if PY_MAJOR_VERSION < 3
+ register_serialized(long(0), &PyInt_Type);
+#endif
+ register_serialized(false, &PyBool_Type);
+ register_serialized(double(0.0), &PyFloat_Type);
+}
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/documentation.cpp b/src/boost/libs/mpi/src/python/documentation.cpp
new file mode 100644
index 00000000..ef28fb3b
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/documentation.cpp
@@ -0,0 +1,671 @@
+// (C) Copyright 2005 The Trustees of Indiana University.
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file documentation.cpp
+ *
+ * This file contains all of the documentation strings for the
+ * Boost.MPI Python bindings.
+ */
+namespace boost { namespace mpi { namespace python {
+
+const char* module_docstring =
+ "The boost.mpi module contains Python wrappers for Boost.MPI.\n"
+ "Boost.MPI is a C++ interface to the Message Passing Interface 1.1,\n"
+ "a high-performance message passing library for parallel programming.\n"
+ "\n"
+ "This module supports the most commonly used subset of MPI 1.1. All\n"
+ "communication operations can transmit any Python object that can be\n"
+ "pickled and unpickled, along with C++-serialized data types and\n"
+ "separation of the structure of a data type from its content.\n"
+ "Collectives that have a user-supplied functions,\n"
+ "such as reduce() or scan(), accept arbitrary Python functions, and\n"
+ "all collectives can operate on any serializable or picklable data type.\n"
+ "\n"
+ "IMPORTANT MODULE DATA\n"
+ " any_source This constant may be used for the source parameter of\n"
+ " receive and probe operations to indicate that a\n"
+ " message may be received from any source.\n"
+ "\n"
+ " any_tag This constant may be used for the tag parameter of\n"
+ " receive or probe operations to indicate that a send\n"
+ " with any tag will be matched.\n"
+ "\n"
+ " collectives_tag Returns the reserved tag value used by the Boost.MPI\n"
+ " implementation for collective operations. Although\n"
+ " users are not permitted to use this tag to send or\n"
+ " receive messages with this tag, it may be useful when\n"
+ " monitoring communication patterns.\n"
+ "\n"
+ " host_rank If there is a host process, this is the rank of that\n"
+ " that process. Otherwise, this value will be None. MPI\n"
+ " does not define the meaning of a \"host\" process: \n"
+ " consult the documentation for your MPI implementation.\n"
+ "\n"
+ " io_rank The rank of a process that can perform input/output\n"
+ " via the standard facilities. If every process can\n"
+ " perform I/O using the standard facilities, this value\n"
+ " will be the same as any_source. If no process can\n"
+ " perform I/O, this value will be None.\n"
+ "\n"
+ " max_tag The maximum value that may be used for the tag\n"
+ " parameter of send/receive operations. This value will\n"
+ " be somewhat smaller than the value of MPI_TAG_UB,\n"
+ " because the Boost.MPI implementation reserves some\n"
+ " tags for collective operations.\n"
+ "\n"
+ " processor_name The name of this processor. The actual form of the\n"
+ " of the name is unspecified, but may be documented by\n"
+ " the underlying MPI implementation.\n"
+ "\n"
+ " rank The rank of this process in the \"world\" communicator.\n"
+ "\n"
+ " size The number of processes in the \"world\" communicator.\n"
+ " that process. Otherwise, this value will be None. MPI\n"
+ " does not define the meaning of a \"host\" process: \n"
+ "\n"
+ " world The \"world\" communicator from which all other\n"
+ " communicators will be derived. This is the equivalent\n"
+ " of MPI_COMM_WORLD.\n"
+ "\n"
+ "TRANSMITTING USER-DEFINED DATA\n"
+ " Boost.MPI can transmit user-defined data in several different ways.\n"
+ " Most importantly, it can transmit arbitrary Python objects by pickling\n"
+ " them at the sender and unpickling them at the receiver, allowing\n"
+ " arbitrarily complex Python data structures to interoperate with MPI.\n"
+ "\n"
+ " Boost.MPI also supports efficient serialization and transmission of\n"
+ " C++ objects (that have been exposed to Python) through its C++\n"
+ " interface. Any C++ type that provides (de-)serialization routines that\n"
+ " meet the requirements of the Boost.Serialization library is eligible\n"
+ " for this optimization, but the type must be registered in advance. To\n"
+ " register a C++ type, invoke the C++ function:\n"
+ " boost::mpi::python::register_serialized\n"
+ "\n"
+ " Finally, Boost.MPI supports separation of the structure of an object\n"
+ " from the data it stores, allowing the two pieces to be transmitted\n"
+ " separately. This \"skeleton/content\" mechanism, described in more\n"
+ " detail in a later section, is a communication optimization suitable\n"
+ " for problems with fixed data structures whose internal data changes\n"
+ " frequently.\n"
+ "\n"
+ "COLLECTIVES\n"
+ " Boost.MPI supports all of the MPI collectives (scatter, reduce, scan,\n"
+ " broadcast, etc.) for any type of data that can be transmitted with the\n"
+ " point-to-point communication operations. For the MPI collectives that\n"
+ " require a user-specified operation (e.g., reduce and scan), the\n"
+ " operation can be an arbitrary Python function. For instance, one could\n"
+ " concatenate strings with all_reduce:\n\n"
+ " mpi.all_reduce(my_string, lambda x,y: x + y)\n\n"
+ " The following module-level functions implement MPI collectives:\n"
+ " all_gather Gather the values from all processes.\n"
+ " all_reduce Combine the results from all processes.\n"
+ " all_to_all Every process sends data to every other process.\n"
+ " broadcast Broadcast data from one process to all other processes.\n"
+ " gather Gather the values from all processes to the root.\n"
+ " reduce Combine the results from all processes to the root.\n"
+ " scan Prefix reduction of the values from all processes.\n"
+ " scatter Scatter the values stored at the root to all processes.\n"
+ "\n"
+ "SKELETON/CONTENT MECHANISM\n"
+ " Boost.MPI provides a skeleton/content mechanism that allows the\n"
+ " transfer of large data structures to be split into two separate stages,\n"
+ " with the `skeleton' (or, `shape') of the data structure sent first and\n"
+ " the content (or, `data') of the data structure sent later, potentially\n"
+ " several times, so long as the structure has not changed since the\n"
+ " skeleton was transferred. The skeleton/content mechanism can improve\n"
+ " performance when the data structure is large and its shape is fixed,\n"
+ " because while the skeleton requires serialization (it has an unknown\n"
+ " size), the content transfer is fixed-size and can be done without\n"
+ " extra copies.\n"
+ "\n"
+ " To use the skeleton/content mechanism from Python, you must first\n"
+ " register the type of your data structure with the skeleton/content\n"
+ " mechanism *from C++*. The registration function is\n"
+ " boost::mpi::python::register_skeleton_and_content\n"
+ " and resides in the <boost/mpi/python.hpp> header.\n"
+ "\n"
+ " Once you have registered your C++ data structures, you can extract\n"
+ " the skeleton for an instance of that data structure with skeleton().\n"
+ " The resulting SkeletonProxy can be transmitted via the normal send\n"
+ " routine, e.g.,\n\n"
+ " mpi.world.send(1, 0, skeleton(my_data_structure))\n\n"
+ " SkeletonProxy objects can be received on the other end via recv(),\n"
+ " which stores a newly-created instance of your data structure with the\n"
+ " same `shape' as the sender in its `object' attribute:\n\n"
+ " shape = mpi.world.recv(0, 0)\n"
+ " my_data_structure = shape.object\n\n"
+ " Once the skeleton has been transmitted, the content (accessed via \n"
+ " get_content) can be transmitted in much the same way. Note, however,\n"
+ " that the receiver also specifies get_content(my_data_structure) in its\n"
+ " call to receive:\n\n"
+ " if mpi.rank == 0:\n"
+ " mpi.world.send(1, 0, get_content(my_data_structure))\n"
+ " else:\n"
+ " mpi.world.recv(0, 0, get_content(my_data_structure))\n\n"
+ " Of course, this transmission of content can occur repeatedly, if the\n"
+ " values in the data structure--but not its shape--changes.\n"
+ "\n"
+ " The skeleton/content mechanism is a structured way to exploit the\n"
+ " interaction between custom-built MPI datatypes and MPI_BOTTOM, to\n"
+ " eliminate extra buffer copies.\n"
+ "\n"
+ "C++/PYTHON MPI COMPATIBILITY\n"
+ " Boost.MPI is a C++ library whose facilities have been exposed to Python\n"
+ " via the Boost.Python library. Since the Boost.MPI Python bindings are\n"
+ " build directly on top of the C++ library, and nearly every feature of\n"
+ " C++ library is available in Python, hybrid C++/Python programs using\n"
+ " Boost.MPI can interact, e.g., sending a value from Python but receiving\n"
+ " that value in C++ (or vice versa). However, doing so requires some\n"
+ " care. Because Python objects are dynamically typed, Boost.MPI transfers\n"
+ " type information along with the serialized form of the object, so that\n"
+ " the object can be received even when its type is not known. This\n"
+ " mechanism differs from its C++ counterpart, where the static types of\n"
+ " transmitted values are always known.\n"
+ "\n"
+ " The only way to communicate between the C++ and Python views on \n"
+ " Boost.MPI is to traffic entirely in Python objects. For Python, this is\n"
+ " the normal state of affairs, so nothing will change. For C++, this\n"
+ " means sending and receiving values of type boost::python::object, from\n"
+ " the Boost.Python library. For instance, say we want to transmit an\n"
+ " integer value from Python:\n\n"
+ " comm.send(1, 0, 17)\n\n"
+ " In C++, we would receive that value into a Python object and then\n"
+ " `extract' an integer value:\n\n"
+ " boost::python::object value;\n"
+ " comm.recv(0, 0, value);\n"
+ " int int_value = boost::python::extract<int>(value);\n\n"
+ " In the future, Boost.MPI will be extended to allow improved\n"
+ " interoperability with the C++ Boost.MPI and the C MPI bindings.\n"
+ ;
+
+/***********************************************************
+ * environment documentation *
+ ***********************************************************/
+const char* environment_init_docstring =
+ "Initialize the MPI environment. Users should not need to call\n"
+ "this function directly, because the MPI environment will be\n"
+ "automatically initialized when the Boost.MPI module is loaded.\n";
+
+const char* environment_finalize_docstring =
+ "Finalize (shut down) the MPI environment. Users only need to\n"
+ "invoke this function if MPI should be shut down before program\n"
+ "termination. Boost.MPI will automatically finalize the MPI\n"
+ "environment when the program exits.\n";
+
+const char* environment_abort_docstring =
+ "Aborts all MPI processes and returns to the environment. The\n"
+ "precise behavior will be defined by the underlying MPI\n"
+ "implementation. This is equivalent to a call to MPI_Abort with\n"
+ "MPI_COMM_WORLD.\n"
+ "errcode is the error code to return from aborted processes.\n";
+
+const char* environment_initialized_docstring =
+ "Determine if the MPI environment has already been initialized.\n";
+
+const char* environment_finalized_docstring =
+ "Determine if the MPI environment has already been finalized.\n";
+
+/***********************************************************
+ * nonblocking documentation *
+ ***********************************************************/
+const char* request_list_init_docstring=
+ "Without arguments, constructs an empty RequestList.\n"
+ "With one argument `iterable', copies request objects from this\n"
+ "iterable to the new RequestList.\n";
+
+const char* nonblocking_wait_any_docstring =
+ "Waits until any of the given requests has been completed. It provides\n"
+ "functionality equivalent to MPI_Waitany.\n"
+ "\n"
+ "requests must be a RequestList instance.\n"
+ "\n"
+ "Returns a triple (value, status, index) consisting of received value\n"
+ "(or None), the Status object for the completed request, and its index\n"
+ "in the RequestList.\n";
+
+const char* nonblocking_test_any_docstring =
+ "Tests if any of the given requests have been completed, but does not wait\n"
+ "for completion. It provides functionality equivalent to MPI_Testany.\n"
+ "\n"
+ "requests must be a RequestList instance.\n"
+ "\n"
+ "Returns a triple (value, status, index) like wait_any or None if no request\n"
+ "is complete.\n";
+
+const char* nonblocking_wait_all_docstring =
+ "Waits until all of the given requests have been completed. It provides\n"
+ "functionality equivalent to MPI_Waitall.\n"
+ "\n"
+ "requests must be a RequestList instance.\n"
+ "\n"
+ "If the second parameter `callable' is provided, it is called with each\n"
+ "completed request's received value (or None) and it s Status object as\n"
+ "its arguments. The calls occur in the order given by the `requests' list.\n";
+
+const char* nonblocking_test_all_docstring =
+ "Tests if all of the given requests have been completed. It provides\n"
+ "functionality equivalent to MPI_Testall.\n"
+ "\n"
+ "Returns True if all requests have been completed.\n"
+ "\n"
+ "requests must be a RequestList instance.\n"
+ "\n"
+ "If the second parameter `callable' is provided, it is called with each\n"
+ "completed request's received value (or None) and it s Status object as\n"
+ "its arguments. The calls occur in the order given by the `requests' list.\n";
+
+const char* nonblocking_wait_some_docstring =
+ "Waits until at least one of the given requests has completed. It\n"
+ "then completes all of the requests it can, partitioning the input\n"
+ "sequence into pending requests followed by completed requests.\n"
+ "\n"
+ "This routine provides functionality equivalent to MPI_Waitsome.\n"
+ "\n"
+ "Returns the index of the first completed request."
+ "\n"
+ "requests must be a RequestList instance.\n"
+ "\n"
+ "If the second parameter `callable' is provided, it is called with each\n"
+ "completed request's received value (or None) and it s Status object as\n"
+ "its arguments. The calls occur in the order given by the `requests' list.\n";
+
+const char* nonblocking_test_some_docstring =
+ "Tests to see if any of the given requests has completed. It completes\n"
+ "all of the requests it can, partitioning the input sequence into pending\n"
+ "requests followed by completed requests. This routine is similar to\n"
+ "wait_some, but does not wait until any requests have completed.\n"
+ "\n"
+ "This routine provides functionality equivalent to MPI_Testsome.\n"
+ "\n"
+ "Returns the index of the first completed request."
+ "\n"
+ "requests must be a RequestList instance.\n"
+ "\n"
+ "If the second parameter `callable' is provided, it is called with each\n"
+ "completed request's received value (or None) and it s Status object as\n"
+ "its arguments. The calls occur in the order given by the `requests' list.\n";
+
+/***********************************************************
+ * exception documentation *
+ ***********************************************************/
+const char* exception_docstring =
+ "Instances of this class will be thrown when an MPI error\n"
+ "occurs. MPI failures that trigger these exceptions may or may not\n"
+ "be recoverable, depending on the underlying MPI implementation.\n"
+ "Consult the documentation for your MPI implementation to determine\n"
+ "the effect of MPI errors.\n";
+
+const char* exception_what_docstring =
+ "A description of the error that occured. At present, this refers\n"
+ "only to the name of the MPI routine that failed.\n";
+
+const char* exception_routine_docstring =
+ "The name of the MPI routine that reported the error.\n";
+
+const char* exception_result_code_docstring =
+ "The result code returned from the MPI routine that reported the\n"
+ "error.\n";
+
+/***********************************************************
+ * collectives documentation *
+ ***********************************************************/
+const char* all_gather_docstring =
+ "all_gather is a collective algorithm that collects the values\n"
+ "stored at each process into a tuple of values indexed by the\n"
+ "process number they came from. all_gather is (semantically) a\n"
+ "gather followed by a broadcast. The same tuple of values is\n"
+ "returned to all processes.\n";
+
+const char* all_reduce_docstring =
+ "all_reduce is a collective algorithm that combines the values\n"
+ "stored by each process into a single value. The values can be\n"
+ "combined arbitrarily, specified via any function. The values\n"
+ "a1, a2, .., ap provided by p processors will be combined by the\n"
+ "binary function op into the result\n"
+ " op(a1, op(a2, ... op(ap-1,ap)))\n"
+ "that will be returned to all processes. This function is the\n"
+ "equivalent of calling all_gather() and then applying the built-in\n"
+ "reduce() function to the returned sequence. op is assumed to be\n"
+ "associative.\n";
+
+const char* all_to_all_docstring =
+ "all_to_all is a collective algorithm that transmits values from\n"
+ "every process to every other process. On process i, the jth value\n"
+ "of the values sequence is sent to process j and placed in the ith\n"
+ "position of the tuple that will be returned from all_to_all.\n";
+
+const char* broadcast_docstring =
+ "broadcast is a collective algorithm that transfers a value from an\n"
+ "arbitrary root process to every other process that is part of the\n"
+ "given communicator (comm). The root parameter must be the same for\n"
+ "every process. The value parameter need only be specified at the root\n"
+ "root. broadcast() returns the same broadcasted value to every process.\n";
+
+const char* gather_docstring =
+ "gather is a collective algorithm that collects the values\n"
+ "stored at each process into a tuple of values at the root\n"
+ "process. This tuple is indexed by the process number that the\n"
+ "value came from, and will be returned only by the root process.\n"
+ "All other processes return None.\n";
+
+const char* reduce_docstring =
+ "reduce is a collective algorithm that combines the values\n"
+ "stored by each process into a single value at the root. The\n"
+ "values can be combined arbitrarily, specified via any function.\n"
+ "The values a1, a2, .., ap provided by p processors will be\n"
+ "combined by the binary function op into the result\n"
+ " op(a1, op(a2, ... op(ap-1,ap)))\n"
+ "that will be returned on the root process. This function is the\n"
+ "equivalent of calling gather() to the root and then applying the\n"
+ "built-in reduce() function to the returned sequence. All non-root\n"
+ "processes return None. op is assumed to be associative.\n";
+
+const char* scan_docstring =
+ "@c scan computes a prefix reduction of values from all processes.\n"
+ "It is a collective algorithm that combines the values stored by\n"
+ "each process with the values of all processes with a smaller rank.\n"
+ "The values can be arbitrarily combined, specified via a binary\n"
+ "function op. If each process i provides the value ai, then scan\n"
+ "returns op(a1, op(a2, ... op(ai-1, ai))) to the ith process. op is\n"
+ "assumed to be associative. This routine is the equivalent of an\n"
+ "all_gather(), followed by a built-in reduce() on the first i+1\n"
+ "values in the resulting sequence on processor i. op is assumed\n"
+ "to be associative.\n";
+
+const char* scatter_docstring =
+ "scatter is a collective algorithm that scatters the values stored\n"
+ "in the root process (as a container with comm.size elements) to\n"
+ "all of the processes in the communicator. The values parameter \n"
+ "(only significant at the root) is indexed by the process number to\n"
+ "which the corresponding value will be sent. The value received by \n"
+ "each process is returned from scatter.\n";
+
+/***********************************************************
+ * communicator documentation *
+ ***********************************************************/
+const char* communicator_docstring =
+ "The Communicator class abstracts a set of communicating\n"
+ "processes in MPI. All of the processes that belong to a certain\n"
+ "communicator can determine the size of the communicator, their rank\n"
+ "within the communicator, and communicate with any other processes\n"
+ "in the communicator.\n";
+
+const char* communicator_default_constructor_docstring =
+ "Build a new Boost.MPI Communicator instance for MPI_COMM_WORLD.\n";
+
+const char* communicator_rank_docstring =
+ "Returns the rank of the process in the communicator, which will be a\n"
+ "value in [0, size).\n";
+
+const char* communicator_size_docstring =
+ "Returns the number of processes in the communicator.\n";
+
+const char* communicator_send_docstring =
+ "This routine executes a potentially blocking send with the given\n"
+ "tag to the process with rank dest. It can be received by the\n"
+ "destination process with a matching recv call. The value will be\n"
+ "transmitted in one of several ways:\n"
+ "\n"
+ " - For C++ objects registered via register_serialized(), the value\n"
+ " will be serialized and transmitted.\n"
+ "\n"
+ " - For SkeletonProxy objects, the skeleton of the object will be\n"
+ " serialized and transmitted.\n"
+ "\n"
+ " - For Content objects, the content will be transmitted directly.\n"
+ " This content can be received by a matching recv/irecv call that\n"
+ " provides a suitable `buffer' argument.\n"
+ "\n"
+ " - For all other Python objects, the value will be pickled and\n"
+ " transmitted.\n";
+
+const char* communicator_recv_docstring =
+ "This routine blocks until it receives a message from the process\n"
+ "source with the given tag. If the source parameter is not specified,\n"
+ "the message can be received from any process. Likewise, if the tag\n"
+ "parameter is not specified, a message with any tag can be received.\n"
+ "If return_status is True, returns a tuple containing the received\n"
+ "object followed by a Status object describing the communication.\n"
+ "Otherwise, recv() returns just the received object.\n"
+ "\n"
+ "When receiving the content of a data type that has been sent separately\n"
+ "from its skeleton, user code must provide a value for the `buffer'\n"
+ "argument. This value should be the Content object returned from\n"
+ "get_content().\n";
+
+const char* communicator_isend_docstring =
+ "This routine executes a nonblocking send with the given\n"
+ "tag to the process with rank dest. It can be received by the\n"
+ "destination process with a matching recv call. The value will be\n"
+ "transmitted in the same way as with send().\n"
+ "This routine returns a Request object, which can be used to query\n"
+ "when the transmission has completed, wait for its completion, or\n"
+ "cancel the transmission.\n";
+
+const char* communicator_irecv_docstring =
+ "This routine initiates a non-blocking receive from the process\n"
+ "source with the given tag. If the source parameter is not specified,\n"
+ "the message can be received from any process. Likewise, if the tag\n"
+ "parameter is not specified, a message with any tag can be received.\n"
+ "This routine returns a Request object, which can be used to query\n"
+ "when the transmission has completed, wait for its completion, or\n"
+ "cancel the transmission. The received value be accessible\n"
+ "through the `value' attribute of the Request object once transmission\n"
+ "has completed.\n"
+ "\n"
+ "As with the recv() routine, when receiving the content of a data type\n"
+ "that has been sent separately from its skeleton, user code must provide\n"
+ "a value for the `buffer' argument. This value should be the Content\n"
+ "object returned from get_content().\n";
+
+ const char* communicator_probe_docstring =
+ "This operation waits until a message matching (source, tag)\n"
+ "is available to be received. It then returns information about\n"
+ "that message. If source is omitted, a message from any process\n"
+ "will match. If tag is omitted, a message with any tag will match.\n"
+ "The actual source and tag can be retrieved from the returned Status\n"
+ "object. To check if a message is available without blocking, use\n"
+ "iprobe.\n";
+
+const char* communicator_iprobe_docstring =
+ "This operation determines if a message matching (source, tag) is\n"
+ "available to be received. If so, it returns information about that\n"
+ "message; otherwise, it returns None. If source is omitted, a message\n"
+ "from any process will match. If tag is omitted, a message with any\n"
+ "tag will match. The actual source and tag can be retrieved from the\n"
+ "returned Status object. To wait for a message to become available, use\n"
+ "probe.\n";
+
+const char* communicator_barrier_docstring =
+ "Wait for all processes within a communicator to reach the\n"
+ "barrier.\n";
+
+const char* communicator_split_docstring =
+ "Split the communicator into multiple, disjoint communicators\n"
+ "each of which is based on a particular color. This is a\n"
+ "collective operation that returns a new communicator that is a\n"
+ "subgroup of this. This routine is functionally equivalent to\n"
+ "MPI_Comm_split.\n\n"
+ "color is the color of this process. All processes with the\n"
+ "same color value will be placed into the same group.\n\n"
+ "If provided, key is a key value that will be used to determine\n"
+ "the ordering of processes with the same color in the resulting\n"
+ "communicator. If omitted, the key will default to the rank of\n"
+ "the process in the current communicator.\n\n"
+ "Returns a new Communicator instance containing all of the \n"
+ "processes in this communicator that have the same color.\n";
+
+const char* communicator_abort_docstring =
+ "Makes a \"best attempt\" to abort all of the tasks in the group of\n"
+ "this communicator. Depending on the underlying MPI\n"
+ "implementation, this may either abort the entire program (and\n"
+ "possibly return errcode to the environment) or only abort\n"
+ "some processes, allowing the others to continue. Consult the\n"
+ "documentation for your MPI implementation. This is equivalent to\n"
+ "a call to MPI_Abort\n\n"
+ "errcode is the error code to return from aborted processes.\n";
+
+/***********************************************************
+ * request documentation *
+ ***********************************************************/
+const char* request_docstring =
+ "The Request class contains information about a non-blocking send\n"
+ "or receive and will be returned from isend or irecv, respectively.\n"
+ "When a Request object represents a completed irecv, the `value' \n"
+ "attribute will contain the received value.\n";
+
+const char* request_with_value_docstring =
+ "This class is an implementation detail. Any call that accepts a\n"
+ "Request also accepts a RequestWithValue, and vice versa.\n";
+
+const char* request_wait_docstring =
+ "Wait until the communication associated with this request has\n"
+ "completed. For a request that is associated with an isend(), returns\n"
+ "a Status object describing the communication. For an irecv()\n"
+ "operation, returns the received value by default. However, when\n"
+ "return_status=True, a (value, status) pair is returned by a\n"
+ "completed irecv request.\n";
+
+const char* request_test_docstring =
+ "Determine whether the communication associated with this request\n"
+ "has completed successfully. If so, returns the Status object\n"
+ "describing the communication (for an isend request) or a tuple\n"
+ "containing the received value and a Status object (for an irecv\n"
+ "request). Note that once test() returns a Status object, the\n"
+ "request has completed and wait() should not be called.\n";
+
+const char* request_cancel_docstring =
+ "Cancel a pending communication, assuming it has not already been\n"
+ "completed.\n";
+
+const char* request_value_docstring =
+ "If this request originated in an irecv(), this property makes the"
+ "sent value accessible once the request completes.\n"
+ "\n"
+ "If no value is available, ValueError is raised.\n";
+
+/***********************************************************
+ * skeleton/content documentation *
+ ***********************************************************/
+const char* object_without_skeleton_docstring =
+ "The ObjectWithoutSkeleton class is an exception class used only\n"
+ "when the skeleton() or get_content() function is called with an\n"
+ "object that is not supported by the skeleton/content mechanism.\n"
+ "All C++ types for which skeletons and content can be transmitted\n"
+ "must be registered with the C++ routine:\n"
+ " boost::mpi::python::register_skeleton_and_content\n";
+
+const char* object_without_skeleton_object_docstring =
+ "The object on which skeleton() or get_content() was invoked.\n";
+
+const char* skeleton_proxy_docstring =
+ "The SkeletonProxy class is used to represent the skeleton of an\n"
+ "object. The SkeletonProxy can be used as the value parameter of\n"
+ "send() or isend() operations, but instead of transmitting the\n"
+ "entire object, only its skeleton (\"shape\") will be sent, without\n"
+ "the actual data. Its content can then be transmitted, separately.\n"
+ "\n"
+ "User code cannot generate SkeletonProxy instances directly. To\n"
+ "refer to the skeleton of an object, use skeleton(object). Skeletons\n"
+ "can also be received with the recv() and irecv() methods.\n"
+ "\n"
+ "Note that the skeleton/content mechanism can only be used with C++\n"
+ "types that have been explicitly registered.\n";
+
+const char* skeleton_proxy_object_docstring =
+ "The actual object whose skeleton is represented by this proxy object.\n";
+
+const char* content_docstring =
+ "The content is a proxy class that represents the content of an object,\n"
+ "which can be separately sent or received from its skeleton.\n"
+ "\n"
+ "User code cannot generate content instances directly. Call the\n"
+ "get_content() routine to retrieve the content proxy for a particular\n"
+ "object. The content instance can be used with any of the send() or\n"
+ "recv() variants. Note that get_content() can only be used with C++\n"
+ "data types that have been explicitly registered with the Python\n"
+ "skeleton/content mechanism.\n";
+
+const char* skeleton_docstring =
+ "The skeleton function retrieves the SkeletonProxy for its object\n"
+ "parameter, allowing the transmission of the skeleton (or \"shape\")\n"
+ "of the object separately from its data. The skeleton/content mechanism\n"
+ "is useful when a large data structure remains structurally the same\n"
+ "throughout a computation, but its content (i.e., the values in the\n"
+ "structure) changes several times. Tranmission of the content part does\n"
+ "not require any serialization or unnecessary buffer copies, so it is\n"
+ "very efficient for large data structures.\n"
+ "\n"
+ "Only C++ types that have been explicitly registered with the Boost.MPI\n"
+ "Python library can be used with the skeleton/content mechanism. Use:\b"
+ " boost::mpi::python::register_skeleton_and_content\n";
+
+const char* get_content_docstring =
+ "The get_content function retrieves the content for its object parameter,\n"
+ "allowing the transmission of the data in a data structure separately\n"
+ "from its skeleton (or \"shape\"). The skeleton/content mechanism\n"
+ "is useful when a large data structure remains structurally the same\n"
+ "throughout a computation, but its content (i.e., the values in the\n"
+ "structure) changes several times. Tranmission of the content part does\n"
+ "not require any serialization or unnecessary buffer copies, so it is\n"
+ "very efficient for large data structures.\n"
+ "\n"
+ "Only C++ types that have been explicitly registered with the Boost.MPI\n"
+ "Python library can be used with the skeleton/content mechanism. Use:\b"
+ " boost::mpi::python::register_skeleton_and_content\n";
+
+/***********************************************************
+ * status documentation *
+ ***********************************************************/
+const char* status_docstring =
+ "The Status class stores information about a given message, including\n"
+ "its source, tag, and whether the message transmission was cancelled\n"
+ "or resulted in an error.\n";
+
+const char* status_source_docstring =
+ "The source of the incoming message.\n";
+
+const char* status_tag_docstring =
+ "The tag of the incoming message.\n";
+
+const char* status_error_docstring =
+ "The error code associated with this transmission.\n";
+
+const char* status_cancelled_docstring =
+ "Whether this transmission was cancelled.\n";
+
+/***********************************************************
+ * timer documentation *
+ ***********************************************************/
+const char* timer_docstring =
+ "The Timer class is a simple wrapper around the MPI timing facilities.\n";
+
+const char* timer_default_constructor_docstring =
+ "Initializes the timer. After this call, elapsed == 0.\n";
+
+const char* timer_restart_docstring =
+ "Restart the timer, after which elapsed == 0.\n";
+
+const char* timer_elapsed_docstring =
+ "The time elapsed since initialization or the last restart(),\n"
+ "whichever is more recent.\n";
+
+const char* timer_elapsed_min_docstring =
+ "Returns the minimum non-zero value that elapsed may return\n"
+ "This is the resolution of the timer.\n";
+
+const char* timer_elapsed_max_docstring =
+ "Return an estimate of the maximum possible value of elapsed. Note\n"
+ "that this routine may return too high a value on some systems.\n";
+
+const char* timer_time_is_global_docstring =
+ "Determines whether the elapsed time values are global times or\n"
+ "local processor times.\n";
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/module.cpp b/src/boost/libs/mpi/src/python/module.cpp
new file mode 100644
index 00000000..ec369684
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/module.cpp
@@ -0,0 +1,55 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file module.cpp
+ *
+ * This file provides the top-level module for the Boost.MPI Python
+ * bindings.
+ */
+#include <boost/python.hpp>
+#include <boost/mpi.hpp>
+
+using namespace boost::python;
+using namespace boost::mpi;
+
+namespace boost { namespace mpi { namespace python {
+
+extern void export_environment();
+extern void export_exception();
+extern void export_collectives();
+extern void export_communicator();
+extern void export_datatypes();
+extern void export_request();
+extern void export_status();
+extern void export_timer();
+extern void export_nonblocking();
+
+extern const char* module_docstring;
+
+BOOST_PYTHON_MODULE(mpi)
+{
+ // Setup module documentation
+ scope().attr("__doc__") = module_docstring;
+ scope().attr("__author__") = "Douglas Gregor <doug.gregor@gmail.com>";
+ scope().attr("__date__") = "$LastChangedDate$";
+ scope().attr("__version__") = "$Revision$";
+ scope().attr("__copyright__") = "Copyright (C) 2006 Douglas Gregor";
+ scope().attr("__license__") = "http://www.boost.org/LICENSE_1_0.txt";
+
+ export_environment();
+ export_exception();
+ export_communicator();
+ export_collectives();
+ export_datatypes();
+ export_request();
+ export_status();
+ export_timer();
+ export_nonblocking();
+}
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/py_communicator.cpp b/src/boost/libs/mpi/src/python/py_communicator.cpp
new file mode 100644
index 00000000..6e53f56f
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/py_communicator.cpp
@@ -0,0 +1,133 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file communicator.cpp
+ *
+ * This file reflects the Boost.MPI @c communicator class into
+ * Python.
+ */
+#include <boost/python.hpp>
+#include <boost/mpi.hpp>
+#include <boost/mpi/python/serialize.hpp>
+#include "request_with_value.hpp"
+
+using namespace boost::python;
+using namespace boost::mpi;
+
+namespace boost { namespace mpi { namespace python {
+
+extern const char* communicator_docstring;
+extern const char* communicator_default_constructor_docstring;
+extern const char* communicator_rank_docstring;
+extern const char* communicator_size_docstring;
+extern const char* communicator_send_docstring;
+extern const char* communicator_recv_docstring;
+extern const char* communicator_isend_docstring;
+extern const char* communicator_irecv_docstring;
+extern const char* communicator_probe_docstring;
+extern const char* communicator_iprobe_docstring;
+extern const char* communicator_barrier_docstring;
+extern const char* communicator_split_docstring;
+extern const char* communicator_split_key_docstring;
+extern const char* communicator_abort_docstring;
+
+object
+communicator_recv(const communicator& comm, int source, int tag,
+ bool return_status)
+{
+ using boost::python::make_tuple;
+
+ object result;
+ status stat = comm.recv(source, tag, result);
+ if (return_status)
+ return make_tuple(result, stat);
+ else
+ return result;
+}
+
+request_with_value
+communicator_irecv(const communicator& comm, int source, int tag)
+{
+ boost::shared_ptr<object> result(new object());
+ request_with_value req(comm.irecv(source, tag, *result));
+ req.m_internal_value = result;
+ return req;
+}
+
+object
+communicator_iprobe(const communicator& comm, int source, int tag)
+{
+ if (boost::optional<status> result = comm.iprobe(source, tag))
+ return object(*result);
+ else
+ return object();
+}
+
+extern void export_skeleton_and_content(class_<communicator>&);
+
+void export_communicator()
+{
+ using boost::python::arg;
+ using boost::python::object;
+
+ class_<communicator> comm("Communicator", communicator_docstring);
+ comm
+ .def(init<>())
+ .add_property("rank", &communicator::rank, communicator_rank_docstring)
+ .add_property("size", &communicator::size, communicator_size_docstring)
+ .def("send",
+ (void (communicator::*)(int, int, const object&) const)
+ &communicator::send<object>,
+ (arg("dest"), arg("tag") = 0, arg("value") = object()),
+ communicator_send_docstring)
+ .def("recv", &communicator_recv,
+ (arg("source") = any_source, arg("tag") = any_tag,
+ arg("return_status") = false),
+ communicator_recv_docstring)
+ .def("isend",
+ (request (communicator::*)(int, int, const object&) const)
+ &communicator::isend<object>,
+ (arg("dest"), arg("tag") = 0, arg("value") = object()),
+ communicator_isend_docstring)
+ .def("irecv", &communicator_irecv,
+ (arg("source") = any_source, arg("tag") = any_tag),
+ communicator_irecv_docstring)
+ .def("probe", &communicator::probe,
+ (arg("source") = any_source, arg("tag") = any_tag),
+ communicator_probe_docstring)
+ .def("iprobe", &communicator_iprobe,
+ (arg("source") = any_source, arg("tag") = any_tag),
+ communicator_iprobe_docstring)
+ .def("barrier", &communicator::barrier, communicator_barrier_docstring)
+ .def("__nonzero__", &communicator::operator bool)
+ .def("split",
+ (communicator (communicator::*)(int) const)&communicator::split,
+ (arg("color")), communicator_split_docstring)
+ .def("split",
+ (communicator (communicator::*)(int, int) const)&communicator::split,
+ (arg("color"), arg("key")))
+ .def("abort", &communicator::abort, arg("errcode"),
+ communicator_abort_docstring)
+ ;
+
+ // Module-level attributes
+ scope().attr("any_source") = any_source;
+ scope().attr("any_tag") = any_tag;
+
+ {
+ communicator world;
+ scope().attr("world") = world;
+ scope().attr("rank") = world.rank();
+ scope().attr("size") = world.size();
+ }
+
+ // Export skeleton and content
+ export_skeleton_and_content(comm);
+}
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/py_environment.cpp b/src/boost/libs/mpi/src/python/py_environment.cpp
new file mode 100644
index 00000000..27af7804
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/py_environment.cpp
@@ -0,0 +1,123 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file environment.cpp
+ *
+ * This file reflects the Boost.MPI "environment" class into Python
+ * methods at module level.
+ */
+
+#include <locale>
+#include <string>
+#include <boost/python.hpp>
+#include <boost/mpi.hpp>
+
+using namespace boost::python;
+using namespace boost::mpi;
+
+namespace boost { namespace mpi { namespace python {
+
+extern const char* environment_init_docstring;
+extern const char* environment_finalize_docstring;
+extern const char* environment_abort_docstring;
+extern const char* environment_initialized_docstring;
+extern const char* environment_finalized_docstring;
+
+/**
+ * The environment used by the Boost.MPI Python module. This will be
+ * zero-initialized before it is used.
+ */
+static environment* env;
+
+bool mpi_init(list python_argv, bool abort_on_exception)
+{
+ // If MPI is already initialized, do nothing.
+ if (environment::initialized())
+ return false;
+
+#if PY_MAJOR_VERSION >= 3
+ #ifdef BOOST_MPI_HAS_NOARG_INITIALIZATION
+ env = new environment(abort_on_exception);
+ #else
+ #error No argument initialization, supported from MPI 1.2 and up, is needed when using Boost.MPI with Python 3.x
+ #endif
+#else
+
+ // Convert Python argv into C-style argc/argv.
+ int my_argc = extract<int>(python_argv.attr("__len__")());
+ char** my_argv = new char*[my_argc];
+ for (int arg = 0; arg < my_argc; ++arg)
+ my_argv[arg] = strdup(extract<const char*>(python_argv[arg]));
+
+ // Initialize MPI
+ int mpi_argc = my_argc;
+ char** mpi_argv = my_argv;
+ env = new environment(mpi_argc, mpi_argv, abort_on_exception);
+
+ // If anything changed, convert C-style argc/argv into Python argv
+ if (mpi_argv != my_argv)
+ PySys_SetArgv(mpi_argc, mpi_argv);
+
+ for (int arg = 0; arg < mpi_argc; ++arg)
+ free(mpi_argv[arg]);
+ delete [] mpi_argv;
+#endif
+
+ return true;
+}
+
+void mpi_finalize()
+{
+ if (env) {
+ delete env;
+ env = 0;
+ }
+}
+
+void export_environment()
+{
+ using boost::python::arg;
+
+ def("init", mpi_init, (arg("argv"), arg("abort_on_exception") = true),
+ environment_init_docstring);
+ def("finalize", mpi_finalize, environment_finalize_docstring);
+
+ // Setup initialization and finalization code
+ if (!environment::initialized()) {
+ // MPI_Init from sys.argv
+ object sys = object(handle<>(PyImport_ImportModule("sys")));
+ mpi_init(extract<list>(sys.attr("argv")), true);
+
+ // Setup MPI_Finalize call when the program exits
+ object atexit = object(handle<>(PyImport_ImportModule("atexit")));
+ object finalize = scope().attr("finalize");
+ atexit.attr("register")(finalize);
+ }
+
+ def("abort", &environment::abort, arg("errcode"),
+ environment_abort_docstring);
+ def("initialized", &environment::initialized,
+ environment_initialized_docstring);
+ def("finalized", &environment::finalized,
+ environment_finalized_docstring);
+ scope().attr("max_tag") = environment::max_tag();
+ scope().attr("collectives_tag") = environment::collectives_tag();
+ scope().attr("processor_name") = environment::processor_name();
+
+ if (optional<int> host_rank = environment::host_rank())
+ scope().attr("host_rank") = *host_rank;
+ else
+ scope().attr("host_rank") = object();
+
+ if (optional<int> io_rank = environment::io_rank())
+ scope().attr("io_rank") = *io_rank;
+ else
+ scope().attr("io_rank") = object();
+}
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/py_exception.cpp b/src/boost/libs/mpi/src/python/py_exception.cpp
new file mode 100644
index 00000000..fee48c4d
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/py_exception.cpp
@@ -0,0 +1,54 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+// Copyright (C) 2005 The Trustees of Indiana University.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file exception.cpp
+ *
+ * This file reflects the Boost.MPI @c mpi_error class into
+ * Python.
+ */
+#include <boost/python.hpp>
+#include <boost/mpi/exception.hpp>
+#include <string>
+#include <boost/lexical_cast.hpp>
+#include "utility.hpp"
+
+using namespace boost::python;
+using namespace boost::mpi;
+
+namespace boost { namespace mpi { namespace python {
+
+extern const char* exception_docstring;
+extern const char* exception_what_docstring;
+extern const char* exception_routine_docstring;
+extern const char* exception_result_code_docstring;
+
+str exception_str(const exception& e)
+{
+ return str(std::string(e.what()) +
+ " (code " + lexical_cast<std::string>(e.result_code())+")");
+}
+
+void export_exception()
+{
+ using boost::python::arg;
+ using boost::python::object;
+
+ object type =
+ class_<exception>
+ ("Exception", exception_docstring, no_init)
+ .add_property("what", &exception::what, exception_what_docstring)
+ .add_property("routine", &exception::what, exception_routine_docstring)
+ .add_property("result_code", &exception::result_code,
+ exception_result_code_docstring)
+ .def("__str__", &exception_str)
+ ;
+ translate_exception<exception>::declare(type);
+}
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/py_nonblocking.cpp b/src/boost/libs/mpi/src/python/py_nonblocking.cpp
new file mode 100644
index 00000000..69fae2fd
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/py_nonblocking.cpp
@@ -0,0 +1,255 @@
+// (C) Copyright 2007
+// Douglas Gregor <doug.gregor -at- gmail.com>
+// Andreas Kloeckner <inform -at- tiker.net>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor, Andreas Kloeckner
+
+/** @file py_nonblocking.cpp
+ *
+ * This file reflects the Boost.MPI nonblocking operations into Python
+ * functions.
+ */
+
+#include <vector>
+#include <iterator>
+#include <algorithm>
+#include <boost/operators.hpp>
+#include <boost/python.hpp>
+#include <boost/python/stl_iterator.hpp>
+#include <boost/python/suite/indexing/vector_indexing_suite.hpp>
+#include <boost/mpi.hpp>
+#include "request_with_value.hpp"
+
+using namespace std;
+using namespace boost::python;
+using namespace boost::mpi;
+
+
+
+
+namespace
+{
+ template <class ValueType, class RequestIterator>
+ class py_call_output_iterator :
+ public boost::output_iterator_helper<
+ py_call_output_iterator<ValueType, RequestIterator> >
+ {
+ private:
+ object m_callable;
+ RequestIterator m_request_iterator;
+
+ public:
+ explicit py_call_output_iterator(object callable,
+ const RequestIterator &req_it)
+ : m_callable(callable), m_request_iterator(req_it)
+ { }
+
+ py_call_output_iterator &operator=(ValueType const &v)
+ {
+ m_callable((m_request_iterator++)->get_value_or_none(), v);
+ return *this;
+ }
+ };
+
+
+
+
+ typedef std::vector<python::request_with_value> request_list;
+ typedef py_call_output_iterator<status, request_list::iterator>
+ status_value_iterator;
+
+
+
+
+ std::shared_ptr<request_list> make_request_list_from_py_list(object iterable)
+ {
+ std::shared_ptr<request_list> result(new request_list);
+ std::copy(
+ stl_input_iterator<python::request_with_value>(iterable),
+ stl_input_iterator<python::request_with_value>(),
+ back_inserter(*result));
+ return result;
+ }
+
+
+
+
+ class request_list_indexing_suite :
+ public vector_indexing_suite<request_list, false, request_list_indexing_suite>
+ {
+ public:
+ // FIXME: requests are not comparable, thus __contains__ makes no sense.
+ // Unfortunately, indexing_suites insist on having __contains__ available.
+ // Just make it error out for now.
+
+ static bool
+ contains(request_list& container, request const& key)
+ {
+ PyErr_SetString(PyExc_NotImplementedError, "mpi requests are not comparable");
+ throw error_already_set();
+ }
+ };
+
+
+
+
+ void check_request_list_not_empty(const request_list &requests)
+ {
+ if (requests.size() == 0)
+ {
+ PyErr_SetString(PyExc_ValueError, "cannot wait on an empty request vector");
+ throw error_already_set();
+ }
+
+ }
+
+
+
+
+
+ object wrap_wait_any(request_list &requests)
+ {
+ check_request_list_not_empty(requests);
+
+ pair<status, request_list::iterator> result =
+ wait_any(requests.begin(), requests.end());
+
+ return boost::python::make_tuple(
+ result.second->get_value_or_none(),
+ result.first,
+ distance(requests.begin(), result.second));
+ }
+
+
+
+
+ object wrap_test_any(request_list &requests)
+ {
+ check_request_list_not_empty(requests);
+ ::boost::optional<pair<status, request_list::iterator> > result =
+ test_any(requests.begin(), requests.end());
+
+ if (result)
+ return boost::python::make_tuple(
+ result->second->get_value_or_none(),
+ result->first,
+ distance(requests.begin(), result->second));
+ else
+ return object();
+ }
+
+
+
+
+
+ void wrap_wait_all(request_list &requests, object py_callable)
+ {
+ check_request_list_not_empty(requests);
+ if (py_callable != object())
+ wait_all(requests.begin(), requests.end(),
+ status_value_iterator(py_callable, requests.begin()));
+ else
+ wait_all(requests.begin(), requests.end());
+ }
+
+
+
+
+ bool wrap_test_all(request_list &requests, object py_callable)
+ {
+ check_request_list_not_empty(requests);
+ if (py_callable != object())
+ return bool(test_all(requests.begin(), requests.end(),
+ status_value_iterator(py_callable, requests.begin())));
+ else
+ return bool(test_all(requests.begin(), requests.end()));
+ }
+
+
+
+
+ int wrap_wait_some(request_list &requests, object py_callable)
+ {
+ check_request_list_not_empty(requests);
+ request_list::iterator first_completed;
+ if (py_callable != object())
+ first_completed = wait_some(requests.begin(), requests.end(),
+ status_value_iterator(py_callable, requests.begin())).second;
+ else
+ first_completed = wait_some(requests.begin(), requests.end());
+
+ return distance(requests.begin(), first_completed);
+ }
+
+
+
+
+ int wrap_test_some(request_list &requests, object py_callable)
+ {
+ check_request_list_not_empty(requests);
+ request_list::iterator first_completed;
+ if (py_callable != object())
+ first_completed = test_some(requests.begin(), requests.end(),
+ status_value_iterator(py_callable, requests.begin())).second;
+ else
+ first_completed = test_some(requests.begin(), requests.end());
+
+ return distance(requests.begin(), first_completed);
+ }
+}
+
+
+
+
+namespace boost { namespace mpi { namespace python {
+
+extern const char* request_list_init_docstring;
+extern const char* request_list_append_docstring;
+
+extern const char* nonblocking_wait_any_docstring;
+extern const char* nonblocking_test_any_docstring;
+extern const char* nonblocking_wait_all_docstring;
+extern const char* nonblocking_test_all_docstring;
+extern const char* nonblocking_wait_some_docstring;
+extern const char* nonblocking_test_some_docstring;
+
+void export_nonblocking()
+{
+ using boost::python::arg;
+
+ {
+ typedef request_list cl;
+ class_<cl>("RequestList", "A list of Request objects.")
+ .def("__init__", make_constructor(make_request_list_from_py_list),
+ /*arg("iterable"),*/ request_list_init_docstring)
+ .def(request_list_indexing_suite())
+ ;
+ }
+
+ def("wait_any", wrap_wait_any,
+ (arg("requests")),
+ nonblocking_wait_any_docstring);
+ def("test_any", wrap_test_any,
+ (arg("requests")),
+ nonblocking_test_any_docstring);
+
+ def("wait_all", wrap_wait_all,
+ (arg("requests"), arg("callable") = object()),
+ nonblocking_wait_all_docstring);
+ def("test_all", wrap_test_all,
+ (arg("requests"), arg("callable") = object()),
+ nonblocking_test_all_docstring);
+
+ def("wait_some", wrap_wait_some,
+ (arg("requests"), arg("callable") = object()),
+ nonblocking_wait_some_docstring);
+ def("test_some", wrap_test_some,
+ (arg("requests"), arg("callable") = object()),
+ nonblocking_test_some_docstring);
+}
+
+} } }
diff --git a/src/boost/libs/mpi/src/python/py_request.cpp b/src/boost/libs/mpi/src/python/py_request.cpp
new file mode 100644
index 00000000..55b2b5ae
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/py_request.cpp
@@ -0,0 +1,111 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file request.cpp
+ *
+ * This file reflects the Boost.MPI @c request class into
+ * Python.
+ */
+#include <boost/python.hpp>
+#include <boost/mpi.hpp>
+#include "request_with_value.hpp"
+
+using namespace boost::python;
+using namespace boost::mpi;
+
+const object python::request_with_value::get_value() const
+{
+ if (m_internal_value.get())
+ return *m_internal_value;
+ else if (m_external_value)
+ return *m_external_value;
+ else
+ {
+ PyErr_SetString(PyExc_ValueError, "request value not available");
+ throw boost::python::error_already_set();
+ }
+}
+
+const object python::request_with_value::get_value_or_none() const
+{
+ if (m_internal_value.get())
+ return *m_internal_value;
+ else if (m_external_value)
+ return *m_external_value;
+ else
+ return object();
+}
+
+const object python::request_with_value::wrap_wait()
+{
+ status stat = request::wait();
+ if (m_internal_value.get() || m_external_value)
+ return boost::python::make_tuple(get_value(), stat);
+ else
+ return object(stat);
+}
+
+const object python::request_with_value::wrap_test()
+{
+ ::boost::optional<status> stat = request::test();
+ if (stat)
+ {
+ if (m_internal_value.get() || m_external_value)
+ return boost::python::make_tuple(get_value(), *stat);
+ else
+ return object(*stat);
+ }
+ else
+ return object();
+}
+
+
+namespace boost { namespace mpi { namespace python {
+
+const object request_test(request &req)
+{
+ ::boost::optional<status> stat = req.test();
+ if (stat)
+ return object(*stat);
+ else
+ return object();
+}
+
+extern const char* request_docstring;
+extern const char* request_with_value_docstring;
+extern const char* request_wait_docstring;
+extern const char* request_test_docstring;
+extern const char* request_cancel_docstring;
+extern const char* request_value_docstring;
+
+void export_request()
+{
+ using boost::python::arg;
+ using boost::python::object;
+
+ {
+ typedef request cl;
+ class_<cl>("Request", request_docstring, no_init)
+ .def("wait", &cl::wait, request_wait_docstring)
+ .def("test", &request_test, request_test_docstring)
+ .def("cancel", &cl::cancel, request_cancel_docstring)
+ ;
+ }
+ {
+ typedef request_with_value cl;
+ class_<cl, bases<request> >(
+ "RequestWithValue", request_with_value_docstring, no_init)
+ .def("wait", &cl::wrap_wait, request_wait_docstring)
+ .def("test", &cl::wrap_test, request_test_docstring)
+ ;
+ }
+
+ implicitly_convertible<request, request_with_value>();
+}
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/py_timer.cpp b/src/boost/libs/mpi/src/python/py_timer.cpp
new file mode 100644
index 00000000..88b1b406
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/py_timer.cpp
@@ -0,0 +1,48 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file timer.cpp
+ *
+ * This file reflects the Boost.MPI @c timer class into
+ * Python.
+ */
+#include <boost/python.hpp>
+#include <boost/mpi/timer.hpp>
+
+using namespace boost::python;
+using namespace boost::mpi;
+
+namespace boost { namespace mpi { namespace python {
+
+extern const char* timer_docstring;
+extern const char* timer_default_constructor_docstring;
+extern const char* timer_restart_docstring;
+extern const char* timer_elapsed_docstring;
+extern const char* timer_elapsed_min_docstring;
+extern const char* timer_elapsed_max_docstring;
+extern const char* timer_time_is_global_docstring;
+
+void export_timer()
+{
+ using boost::python::arg;
+ using boost::python::object;
+
+ class_<timer>("Timer", timer_docstring)
+ .def(init<>())
+ .def("restart", &timer::restart, timer_restart_docstring)
+ .add_property("elapsed", &timer::elapsed, timer_elapsed_docstring)
+ .add_property("elapsed_min", &timer::elapsed_min,
+ timer_elapsed_min_docstring)
+ .add_property("elapsed_max", &timer::elapsed_max,
+ timer_elapsed_max_docstring)
+ .add_property("time_is_global", &timer::time_is_global,
+ timer_time_is_global_docstring)
+ ;
+}
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/request_with_value.hpp b/src/boost/libs/mpi/src/python/request_with_value.hpp
new file mode 100644
index 00000000..04d0c53e
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/request_with_value.hpp
@@ -0,0 +1,71 @@
+// (C) Copyright 2006
+// Douglas Gregor <doug.gregor -at- gmail.com>
+// Andreas Kloeckner <inform -at- tiker.net>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor, Andreas Kloeckner
+
+#ifndef BOOST_MPI_PYTHON_REQUEST_WITH_VALUE_HPP
+#define BOOST_MPI_PYTHON_REQUEST_WITH_VALUE_HPP
+
+#include <boost/python.hpp>
+#include <boost/mpi.hpp>
+
+namespace boost { namespace mpi { namespace python {
+
+ /** This wrapper adds a @c boost::python::object value to the @c
+ * boost::mpi::request structure, for the benefit of @c irecv() requests.
+ *
+ * In order to be able to return the value of his requests to the user, we
+ * need a handle that we can update to contain the transmitted value once the
+ * request completes. Since we're passing the address on to irecv to fill at
+ * any time in the future, this address may not change over time.
+ *
+ * There are two possible cases:
+ * - plain irecv()
+ * - skeleton-content irecv()
+ *
+ * In the first case, we need to own the storage from this object, the
+ * m_internal_value is used for this. In the second case, the updated
+ * python::object is part of a boost::mpi::python::content object: the
+ * m_external_value field handles this case. Furthermore, in the latter case,
+ * we now have a lifetime dependency on that content object; this can be
+ * handled with the BPL's with_custodian_and_ward facility.
+ *
+ * Since requests and request_with_value are supposed to be copyconstructible,
+ * we can't put the handle immediately inside this instance. Moreover, since
+ * we need to be able to put request_with_value inside request_vectors, any
+ * values we own must be held in a shared_ptr instance.
+ */
+
+ class request_with_value : public request
+ {
+ private:
+ boost::shared_ptr<boost::python::object> m_internal_value;
+ boost::python::object *m_external_value;
+
+ public:
+ request_with_value()
+ : m_external_value(0)
+ { }
+ request_with_value(const request &req)
+ : request(req), m_external_value(0)
+ { }
+
+ const boost::python::object get_value() const;
+ const boost::python::object get_value_or_none() const;
+
+ const boost::python::object wrap_wait();
+ const boost::python::object wrap_test();
+
+ friend request_with_value communicator_irecv(const communicator &, int, int);
+ friend request_with_value communicator_irecv_content(
+ const communicator&, int, int, content&);
+ };
+
+} } }
+
+#endif
diff --git a/src/boost/libs/mpi/src/python/serialize.cpp b/src/boost/libs/mpi/src/python/serialize.cpp
new file mode 100644
index 00000000..92004a34
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/serialize.cpp
@@ -0,0 +1,79 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file serialize.cpp
+ *
+ * This file provides Boost.Serialization support for Python objects.
+ */
+#include <boost/mpi/python/serialize.hpp>
+#include <boost/mpi/python/skeleton_and_content.hpp>
+#include <boost/mpi.hpp>
+
+namespace boost { namespace python {
+
+struct pickle::data_t {
+ object module;
+ object dumps;
+ object loads;
+};
+
+
+/// Data used for communicating with the Python `pickle' module.
+pickle::data_t* pickle::data;
+
+str pickle::dumps(object obj, int protocol)
+{
+ if (!data) initialize_data();
+ return extract<str>((data->dumps)(obj, protocol));
+}
+
+object pickle::loads(str s)
+{
+ if (!data) initialize_data();
+ return ((data->loads)(s));
+}
+
+void pickle::initialize_data()
+{
+ data = new data_t;
+ data->module = object(handle<>(PyImport_ImportModule("pickle")));
+ data->dumps = data->module.attr("dumps");
+ data->loads = data->module.attr("loads");
+}
+
+} } // end namespace boost::python
+
+BOOST_PYTHON_DIRECT_SERIALIZATION_ARCHIVE_IMPL(
+ ::boost::mpi::packed_iarchive,
+ ::boost::mpi::packed_oarchive)
+
+namespace boost { namespace mpi { namespace python { namespace detail {
+
+ boost::python::object skeleton_proxy_base_type;
+
+ // A map from Python type objects to skeleton/content handlers
+ typedef std::map<PyTypeObject*, skeleton_content_handler>
+ skeleton_content_handlers_type;
+
+ BOOST_MPI_PYTHON_DECL skeleton_content_handlers_type skeleton_content_handlers;
+
+ bool
+ skeleton_and_content_handler_registered(PyTypeObject* type)
+ {
+ return
+ skeleton_content_handlers.find(type) != skeleton_content_handlers.end();
+ }
+
+ void
+ register_skeleton_and_content_handler(PyTypeObject* type,
+ const skeleton_content_handler& handler)
+ {
+ skeleton_content_handlers[type] = handler;
+ }
+
+} } } } // end namespace boost::mpi::python::detail
diff --git a/src/boost/libs/mpi/src/python/skeleton_and_content.cpp b/src/boost/libs/mpi/src/python/skeleton_and_content.cpp
new file mode 100644
index 00000000..d5376c14
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/skeleton_and_content.cpp
@@ -0,0 +1,173 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file skeleton_and_content.cpp
+ *
+ * This file reflects the skeleton/content facilities into Python.
+ */
+#include <boost/mpi/python/skeleton_and_content.hpp>
+#include <boost/mpi/python/serialize.hpp>
+#include <boost/python/list.hpp>
+#include <typeinfo>
+#include <list>
+#include "utility.hpp"
+#include "request_with_value.hpp"
+
+using namespace boost::python;
+using namespace boost::mpi;
+
+namespace boost { namespace mpi { namespace python {
+
+namespace detail {
+ typedef std::map<PyTypeObject*, skeleton_content_handler>
+ skeleton_content_handlers_type;
+
+// We're actually importing skeleton_content_handlers from skeleton_and_content.cpp.
+#if defined(BOOST_HAS_DECLSPEC) && (defined(BOOST_MPI_PYTHON_DYN_LINK) || defined(BOOST_ALL_DYN_LINK))
+# define BOOST_SC_DECL __declspec(dllimport)
+#else
+# define BOOST_SC_DECL
+#endif
+
+ extern BOOST_SC_DECL skeleton_content_handlers_type skeleton_content_handlers;
+}
+
+/**
+ * An exception that will be thrown when the object passed to the
+ * Python version of skeleton() does not have a skeleton.
+ */
+struct object_without_skeleton : public std::exception {
+ explicit object_without_skeleton(object value) : value(value) { }
+ virtual ~object_without_skeleton() throw() { }
+
+ object value;
+};
+
+str object_without_skeleton_str(const object_without_skeleton& e)
+{
+ return str("\nThe skeleton() or get_content() function was invoked for a Python\n"
+ "object that is not supported by the Boost.MPI skeleton/content\n"
+ "mechanism. To transfer objects via skeleton/content, you must\n"
+ "register the C++ type of this object with the C++ function:\n"
+ " boost::mpi::python::register_skeleton_and_content()\n"
+ "Object: " + str(e.value) + "\n");
+}
+
+/**
+ * Extract the "skeleton" from a Python object. In truth, all we're
+ * doing at this point is verifying that the object is a C++ type that
+ * has been registered for the skeleton/content mechanism.
+ */
+object skeleton(object value)
+{
+ PyTypeObject* type = value.ptr()->ob_type;
+ detail::skeleton_content_handlers_type::iterator pos =
+ detail::skeleton_content_handlers.find(type);
+ if (pos == detail::skeleton_content_handlers.end())
+ throw object_without_skeleton(value);
+ else
+ return pos->second.get_skeleton_proxy(value);
+}
+
+/**
+ * Extract the "content" from a Python object, which must be a C++
+ * type that has been registered for the skeleton/content mechanism.
+ */
+content get_content(object value)
+{
+ PyTypeObject* type = value.ptr()->ob_type;
+ detail::skeleton_content_handlers_type::iterator pos =
+ detail::skeleton_content_handlers.find(type);
+ if (pos == detail::skeleton_content_handlers.end())
+ throw object_without_skeleton(value);
+ else
+ return pos->second.get_content(value);
+}
+
+/// Send the content part of a Python object.
+void
+communicator_send_content(const communicator& comm, int dest, int tag,
+ const content& c)
+{
+ comm.send(dest, tag, c.base());
+}
+
+/// Receive the content of a Python object. We return the object
+/// received, not the content wrapper.
+object
+communicator_recv_content(const communicator& comm, int source, int tag,
+ const content& c, bool return_status)
+{
+ using boost::python::make_tuple;
+
+ status stat = comm.recv(source, tag, c.base());
+ if (return_status)
+ return make_tuple(c.object, stat);
+ else
+ return c.object;
+}
+
+/// Receive the content of a Python object. The request object's value
+/// attribute will reference the object whose content is being
+/// received, not the content wrapper.
+request_with_value
+communicator_irecv_content(const communicator& comm, int source, int tag,
+ content& c)
+{
+ request_with_value req(comm.irecv(source, tag, c.base()));
+ req.m_external_value = &c.object;
+ return req;
+}
+
+extern const char* object_without_skeleton_docstring;
+extern const char* object_without_skeleton_object_docstring;
+extern const char* skeleton_proxy_docstring;
+extern const char* skeleton_proxy_object_docstring;
+extern const char* content_docstring;
+extern const char* skeleton_docstring;
+extern const char* get_content_docstring;
+
+void export_skeleton_and_content(class_<communicator>& comm)
+{
+ using boost::python::arg;
+
+ // Expose the object_without_skeleton exception
+ object type =
+ class_<object_without_skeleton>
+ ("ObjectWithoutSkeleton", object_without_skeleton_docstring, no_init)
+ .def_readonly("object", &object_without_skeleton::value,
+ object_without_skeleton_object_docstring)
+ .def("__str__", &object_without_skeleton_str)
+ ;
+ translate_exception<object_without_skeleton>::declare(type);
+
+ // Expose the Python variants of "skeleton_proxy" and "content", and
+ // their generator functions.
+ detail::skeleton_proxy_base_type =
+ class_<skeleton_proxy_base>("SkeletonProxy", skeleton_proxy_docstring,
+ no_init)
+ .def_readonly("object", &skeleton_proxy_base::object,
+ skeleton_proxy_object_docstring);
+ class_<content>("Content", content_docstring, no_init);
+ def("skeleton", &skeleton, arg("object"), skeleton_docstring);
+ def("get_content", &get_content, arg("object"), get_content_docstring);
+
+ // Expose communicator send/recv operations for content.
+ comm
+ .def("send", communicator_send_content,
+ (arg("dest"), arg("tag") = 0, arg("value")))
+ .def("recv", communicator_recv_content,
+ (arg("source") = any_source, arg("tag") = any_tag, arg("buffer"),
+ arg("return_status") = false))
+ .def("irecv", communicator_irecv_content,
+ (arg("source") = any_source, arg("tag") = any_tag, arg("buffer")),
+ with_custodian_and_ward_postcall<0, 4>()
+ );
+}
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/status.cpp b/src/boost/libs/mpi/src/python/status.cpp
new file mode 100644
index 00000000..a74221a7
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/status.cpp
@@ -0,0 +1,41 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+/** @file status.cpp
+ *
+ * This file reflects the Boost.MPI @c status class into
+ * Python.
+ */
+#include <boost/python.hpp>
+#include <boost/mpi.hpp>
+
+using namespace boost::python;
+using namespace boost::mpi;
+
+namespace boost { namespace mpi { namespace python {
+
+extern const char* status_docstring;
+extern const char* status_source_docstring;
+extern const char* status_tag_docstring;
+extern const char* status_error_docstring;
+extern const char* status_cancelled_docstring;
+
+void export_status()
+{
+ using boost::python::arg;
+ using boost::python::object;
+
+ class_<status>("Status", status_docstring, no_init)
+ .add_property("source", &status::source, status_source_docstring)
+ .add_property("tag", &status::tag, status_tag_docstring)
+ .add_property("error", &status::error, status_error_docstring)
+ .add_property("cancelled", &status::cancelled, status_cancelled_docstring)
+ ;
+}
+
+} } } // end namespace boost::mpi::python
diff --git a/src/boost/libs/mpi/src/python/utility.hpp b/src/boost/libs/mpi/src/python/utility.hpp
new file mode 100644
index 00000000..ed001676
--- /dev/null
+++ b/src/boost/libs/mpi/src/python/utility.hpp
@@ -0,0 +1,43 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+#ifndef BOOST_MPI_PYTHON_UTILITY_HPP
+#define BOOST_MPI_PYTHON_UTILITY_HPP
+
+/** @file utility.hpp
+ *
+ * This file is a utility header for the Boost.MPI Python bindings.
+ */
+#include <boost/python.hpp>
+
+namespace boost { namespace mpi { namespace python {
+
+template<typename E>
+class translate_exception
+{
+ explicit translate_exception(boost::python::object type) : type(type) { }
+
+public:
+ static void declare(boost::python::object type)
+ {
+ using boost::python::register_exception_translator;
+ register_exception_translator<E>(translate_exception(type));
+ }
+
+ void operator()(const E& e) const
+ {
+ using boost::python::object;
+ PyErr_SetObject(type.ptr(), object(e).ptr());
+ }
+
+private:
+ boost::python::object type;
+};
+
+} } } // end namespace boost::mpi::python
+
+#endif // BOOST_MPI_PYTHON_UTILITY_HPP
diff --git a/src/boost/libs/mpi/src/request.cpp b/src/boost/libs/mpi/src/request.cpp
new file mode 100644
index 00000000..3ba5695a
--- /dev/null
+++ b/src/boost/libs/mpi/src/request.cpp
@@ -0,0 +1,239 @@
+// Copyright (C) 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+#include <boost/mpi/request.hpp>
+#include <boost/mpi/status.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/detail/request_handlers.hpp>
+
+namespace boost { namespace mpi {
+
+request::request()
+ : m_handler() {}
+
+void
+request::preserve(boost::shared_ptr<void> d) {
+ if (!m_preserved) {
+ m_preserved = d;
+ } else {
+ boost::shared_ptr<void> cdr = m_preserved;
+ typedef std::pair<boost::shared_ptr<void>, boost::shared_ptr<void> > cons;
+ boost::shared_ptr<cons> p(new cons(d, cdr));
+ m_preserved = p;
+ }
+}
+request request::make_dynamic() { return request(new dynamic_handler()); }
+
+request
+request::make_bottom_send(communicator const& comm, int dest, int tag, MPI_Datatype tp) {
+ trivial_handler* handler = new trivial_handler;
+ BOOST_MPI_CHECK_RESULT(MPI_Isend,
+ (MPI_BOTTOM, 1, tp,
+ dest, tag, comm, &handler->m_request));
+ return request(handler);
+}
+
+request
+request::make_empty_send(communicator const& comm, int dest, int tag) {
+ trivial_handler* handler = new trivial_handler;
+ BOOST_MPI_CHECK_RESULT(MPI_Isend,
+ (MPI_BOTTOM, 0, MPI_PACKED,
+ dest, tag, comm, &handler->m_request));
+ return request(handler);
+}
+
+request
+request::make_bottom_recv(communicator const& comm, int dest, int tag, MPI_Datatype tp) {
+ trivial_handler* handler = new trivial_handler;
+ BOOST_MPI_CHECK_RESULT(MPI_Irecv,
+ (MPI_BOTTOM, 1, tp,
+ dest, tag, comm, &handler->m_request));
+ return request(handler);
+}
+
+request
+request::make_empty_recv(communicator const& comm, int dest, int tag) {
+ trivial_handler* handler = new trivial_handler;
+ BOOST_MPI_CHECK_RESULT(MPI_Irecv,
+ (MPI_BOTTOM, 0, MPI_PACKED,
+ dest, tag, comm, &handler->m_request));
+ return request(handler);
+}
+
+request
+request::make_packed_send(communicator const& comm, int dest, int tag, void const* buffer, std::size_t n) {
+#if defined(BOOST_MPI_USE_IMPROBE)
+ {
+ trivial_handler* handler = new trivial_handler;
+ BOOST_MPI_CHECK_RESULT(MPI_Isend,
+ (const_cast<void*>(buffer), n, MPI_PACKED,
+ dest, tag, comm, &handler->m_request));
+ return request(handler);
+ }
+#else
+ {
+ dynamic_handler *handler = new dynamic_handler;
+ request req(handler);
+ shared_ptr<std::size_t> size(new std::size_t(n));
+ req.preserve(size);
+ BOOST_MPI_CHECK_RESULT(MPI_Isend,
+ (size.get(), 1,
+ get_mpi_datatype(*size),
+ dest, tag, comm, handler->m_requests));
+ BOOST_MPI_CHECK_RESULT(MPI_Isend,
+ (const_cast<void*>(buffer), *size,
+ MPI_PACKED,
+ dest, tag, comm, handler->m_requests+1));
+ return req;
+ }
+#endif
+}
+
+/***************************************************************************
+ * handlers *
+ ***************************************************************************/
+
+request::handler::~handler() {}
+
+optional<MPI_Request&>
+request::legacy_handler::trivial() {
+ return boost::none;
+}
+
+bool
+request::legacy_handler::active() const {
+ return m_requests[0] != MPI_REQUEST_NULL || m_requests[1] != MPI_REQUEST_NULL;
+}
+
+// trivial handler
+
+request::trivial_handler::trivial_handler()
+ : m_request(MPI_REQUEST_NULL) {}
+
+status
+request::trivial_handler::wait()
+{
+ status result;
+ BOOST_MPI_CHECK_RESULT(MPI_Wait, (&m_request, &result.m_status));
+ return result;
+}
+
+
+optional<status>
+request::trivial_handler::test()
+{
+ status result;
+ int flag = 0;
+ BOOST_MPI_CHECK_RESULT(MPI_Test,
+ (&m_request, &flag, &result.m_status));
+ return flag != 0? optional<status>(result) : optional<status>();
+}
+
+void
+request::trivial_handler::cancel()
+{
+ BOOST_MPI_CHECK_RESULT(MPI_Cancel, (&m_request));
+}
+
+bool
+request::trivial_handler::active() const
+{
+ return m_request != MPI_REQUEST_NULL;
+}
+
+optional<MPI_Request&>
+request::trivial_handler::trivial()
+{
+ return m_request;
+}
+
+// dynamic handler
+
+request::dynamic_handler::dynamic_handler()
+{
+ m_requests[0] = MPI_REQUEST_NULL;
+ m_requests[1] = MPI_REQUEST_NULL;
+}
+
+status
+request::dynamic_handler::wait()
+{
+ // This request is a send of a serialized type, broken into two
+ // separate messages. Complete both sends at once.
+ MPI_Status stats[2];
+ int error_code = MPI_Waitall(2, m_requests, stats);
+ if (error_code == MPI_ERR_IN_STATUS) {
+ // Dig out which status structure has the error, and use that
+ // one when throwing the exception.
+ if (stats[0].MPI_ERROR == MPI_SUCCESS
+ || stats[0].MPI_ERROR == MPI_ERR_PENDING)
+ boost::throw_exception(exception("MPI_Waitall", stats[1].MPI_ERROR));
+ else
+ boost::throw_exception(exception("MPI_Waitall", stats[0].MPI_ERROR));
+ } else if (error_code != MPI_SUCCESS) {
+ // There was an error somewhere in the MPI_Waitall call; throw
+ // an exception for it.
+ boost::throw_exception(exception("MPI_Waitall", error_code));
+ }
+
+ // No errors. Returns the first status structure.
+ status result;
+ result.m_status = stats[0];
+ return result;
+}
+
+optional<status>
+request::dynamic_handler::test()
+{
+ // This request is a send of a serialized type, broken into two
+ // separate messages. We only get a result if both complete.
+ MPI_Status stats[2];
+ int flag = 0;
+ int error_code = MPI_Testall(2, m_requests, &flag, stats);
+ if (error_code == MPI_ERR_IN_STATUS) {
+ // Dig out which status structure has the error, and use that
+ // one when throwing the exception.
+ if (stats[0].MPI_ERROR == MPI_SUCCESS
+ || stats[0].MPI_ERROR == MPI_ERR_PENDING)
+ boost::throw_exception(exception("MPI_Testall", stats[1].MPI_ERROR));
+ else
+ boost::throw_exception(exception("MPI_Testall", stats[0].MPI_ERROR));
+ } else if (error_code != MPI_SUCCESS) {
+ // There was an error somewhere in the MPI_Testall call; throw
+ // an exception for it.
+ boost::throw_exception(exception("MPI_Testall", error_code));
+ }
+
+ // No errors. Returns the second status structure if the send has
+ // completed.
+ if (flag != 0) {
+ status result;
+ result.m_status = stats[1];
+ return result;
+ } else {
+ return optional<status>();
+ }
+}
+
+void
+request::dynamic_handler::cancel()
+{
+ BOOST_MPI_CHECK_RESULT(MPI_Cancel, (&m_requests[0]));
+ BOOST_MPI_CHECK_RESULT(MPI_Cancel, (&m_requests[1]));
+}
+
+bool
+request::dynamic_handler::active() const
+{
+ return (m_requests[0] != MPI_REQUEST_NULL
+ || m_requests[1] != MPI_REQUEST_NULL);
+}
+
+optional<MPI_Request&>
+request::dynamic_handler::trivial() {
+ return boost::none;
+}
+
+} } // end namespace boost::mpi
diff --git a/src/boost/libs/mpi/src/text_skeleton_oarchive.cpp b/src/boost/libs/mpi/src/text_skeleton_oarchive.cpp
new file mode 100644
index 00000000..7817ea78
--- /dev/null
+++ b/src/boost/libs/mpi/src/text_skeleton_oarchive.cpp
@@ -0,0 +1,20 @@
+// (C) Copyright 2005 Matthias Troyer
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Matthias Troyer
+
+#define BOOST_ARCHIVE_SOURCE
+#include <boost/mpi/detail/text_skeleton_oarchive.hpp>
+
+#include <boost/archive/detail/archive_serializer_map.hpp>
+#include <boost/archive/impl/archive_serializer_map.ipp>
+
+namespace boost { namespace archive {
+// explicitly instantiate all required templates
+
+// template class detail::archive_serializer_map<text_oarchive>;
+
+} } // end namespace boost::archive
diff --git a/src/boost/libs/mpi/src/timer.cpp b/src/boost/libs/mpi/src/timer.cpp
new file mode 100644
index 00000000..e3a7dfc4
--- /dev/null
+++ b/src/boost/libs/mpi/src/timer.cpp
@@ -0,0 +1,25 @@
+// Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+#include <boost/mpi/timer.hpp>
+#include <boost/mpi/exception.hpp>
+
+namespace boost { namespace mpi {
+
+bool timer::time_is_global()
+{
+ int* is_global;
+ int found = 0;
+
+ BOOST_MPI_CHECK_RESULT(MPI_Comm_get_attr,
+ (MPI_COMM_WORLD, MPI_WTIME_IS_GLOBAL, &is_global,
+ &found));
+ if (!found)
+ return false;
+ else
+ return *is_global != 0;
+}
+
+} } // end namespace boost::mpi
diff --git a/src/boost/libs/mpi/test/Jamfile.v2 b/src/boost/libs/mpi/test/Jamfile.v2
new file mode 100644
index 00000000..ac9ac253
--- /dev/null
+++ b/src/boost/libs/mpi/test/Jamfile.v2
@@ -0,0 +1,55 @@
+# Support for the Message Passing Interface (MPI)
+#
+# (C) Copyright 2005, 2006 Trustees of Indiana University
+# (C) Copyright 2005 Douglas Gregor
+#
+# Distributed under the Boost Software License, Version 1.0. (See accompanying
+# file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt.)
+#
+# Authors: Douglas Gregor
+# Andrew Lumsdaine
+
+project : requirements <library>/boost//mpi ;
+import mpi : mpi-test ;
+
+if [ mpi.configured ]
+{
+test-suite mpi
+ :
+ [ mpi-test version_test : : : 1 ]
+ [ mpi-test block_nonblock_test-b2nb : block_nonblock_test.cpp : : 2 ]
+ [ mpi-test block_nonblock_test-nb2b : block_nonblock_test.cpp : : 2 ]
+ [ mpi-test random_gather : ../example/random_gather.cpp : : 2 ]
+ [ mpi-test random_scatter : ../example/random_scatter.cpp : : 2 ]
+ [ mpi-test cartesian_communicator : ../example/cartesian_communicator.cpp : : 24 ]
+ [ mpi-test cartesian_topology_init_test : : : 1 ]
+ [ mpi-test broadcast_stl_test : : : 2 ]
+ [ mpi-test all_gather_test : : : 1 2 11 ]
+ [ mpi-test all_reduce_test : : : 1 2 11 ]
+ [ mpi-test all_to_all_test : : : 1 2 11 ]
+ [ mpi-test broadcast_test : : : 2 17 ]
+ [ mpi-test gather_test : : : 1 2 11 ]
+ [ mpi-test is_mpi_op_test : : : 1 ]
+ [ mpi-test mt_level_test : : : 1 ]
+ [ mpi-test mt_init_test : : : 1 4 ]
+ # Note: Microsoft MPI fails nonblocking_test on 1 processor
+ [ mpi-test nonblocking_test : : : 2 11 24 ]
+ [ mpi-test reduce_test ]
+ [ mpi-test ring_test : : : 2 3 4 7 8 13 17 ]
+ [ mpi-test sendrecv_test : : : 1 4 7 48 ]
+ [ mpi-test wait_any_test : : : 1 4 7 20 ]
+ [ mpi-test wait_all_vector_test : : : 2 ]
+ [ mpi-test scan_test ]
+ [ mpi-test scatter_test ]
+ # Note: Microsoft MPI fails all skeleton-content tests
+ [ mpi-test skeleton_content_test : : : 2 3 4 7 8 13 17 ]
+ [ mpi-test graph_topology_test : : : 2 7 13 ]
+ [ mpi-test cartesian_topology_test : : : 24 ]
+ [ mpi-test pointer_test : : : 2 ]
+ [ mpi-test groups_test ]
+ # tests that require -std=c++11
+ [ mpi-test sendrecv_vector : : : 2 ]
+ # Intel MPI 2018 and older are axtected to fail:
+ [ mpi-test non_blocking_any_source : : : 2 17 ]
+ ;
+}
diff --git a/src/boost/libs/mpi/test/all_gather_test.cpp b/src/boost/libs/mpi/test/all_gather_test.cpp
new file mode 100644
index 00000000..c2385f61
--- /dev/null
+++ b/src/boost/libs/mpi/test/all_gather_test.cpp
@@ -0,0 +1,148 @@
+// Copyright (C) 2005-2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the all_gather() collective.
+
+#include <algorithm>
+
+#include <boost/mpi/collectives/all_gather.hpp>
+#include <boost/mpi/collectives/all_gatherv.hpp>
+#include <boost/mpi/environment.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/serialization/string.hpp>
+#include <boost/serialization/list.hpp>
+#include <boost/iterator/counting_iterator.hpp>
+#include <boost/lexical_cast.hpp>
+
+#define BOOST_TEST_MODULE mpi_all_gather
+#include <boost/test/included/unit_test.hpp>
+
+#include "gps_position.hpp"
+
+namespace mpi = boost::mpi;
+
+template<typename Generator>
+void
+all_gather_test(const mpi::communicator& comm, Generator generator,
+ std::string kind)
+{
+ typedef typename Generator::result_type value_type;
+ value_type value = generator(comm.rank());
+
+ std::vector<value_type> values;
+ if (comm.rank() == 0) {
+ std::cout << "Gathering " << kind << "...";
+ std::cout.flush();
+ }
+
+ mpi::all_gather(comm, value, values);
+
+ std::vector<value_type> expected_values;
+ for (int p = 0; p < comm.size(); ++p)
+ expected_values.push_back(generator(p));
+ BOOST_CHECK(values == expected_values);
+ if (comm.rank() == 0 && values == expected_values)
+ std::cout << "OK." << std::endl;
+
+ (comm.barrier)();
+}
+
+template<typename Generator>
+void
+all_gatherv_test(const mpi::communicator& comm, Generator generator,
+ std::string kind)
+{
+ typedef typename Generator::result_type value_type;
+ using boost::mpi::all_gatherv;
+
+ std::vector<value_type> myvalues, expected, values;
+ std::vector<int> sizes;
+ for(int r = 0; r < comm.size(); ++r) {
+ value_type value = generator(r);
+ sizes.push_back(r+1);
+ for (int k=0; k < r+1; ++k) {
+ expected.push_back(value);
+ if(comm.rank() == r) {
+ myvalues.push_back(value);
+ }
+ }
+ }
+ if (comm.rank() == 0) {
+ std::cout << "Gathering " << kind << "...";
+ std::cout.flush();
+ }
+
+ mpi::all_gatherv(comm, myvalues, values, sizes);
+
+ BOOST_CHECK(values == expected);
+
+ if (comm.rank() == 0 && values == expected)
+ std::cout << "OK." << std::endl;
+
+ (comm.barrier)();
+}
+
+// Generates integers to test with gather()
+struct int_generator
+{
+ typedef int result_type;
+
+ int operator()(int p) const { return 17 + p; }
+};
+
+// Generates GPS positions to test with gather()
+struct gps_generator
+{
+ typedef gps_position result_type;
+
+ gps_position operator()(int p) const
+ {
+ return gps_position(39 + p, 16, 20.2799);
+ }
+};
+
+struct string_generator
+{
+ typedef std::string result_type;
+
+ std::string operator()(int p) const
+ {
+ std::string result = boost::lexical_cast<std::string>(p);
+ result += " rosebud";
+ if (p != 1) result += 's';
+ return result;
+ }
+};
+
+struct string_list_generator
+{
+ typedef std::list<std::string> result_type;
+
+ std::list<std::string> operator()(int p) const
+ {
+ std::list<std::string> result;
+ for (int i = 0; i <= p; ++i) {
+ std::string value = boost::lexical_cast<std::string>(i);
+ result.push_back(value);
+ }
+ return result;
+ }
+};
+
+BOOST_AUTO_TEST_CASE(all_gather)
+{
+ boost::mpi::environment env;
+ mpi::communicator comm;
+ all_gather_test(comm, int_generator(), "integers");
+ all_gather_test(comm, gps_generator(), "GPS positions");
+ all_gather_test(comm, string_generator(), "string");
+ all_gather_test(comm, string_list_generator(), "list of strings");
+
+ all_gatherv_test(comm, int_generator(), "integers");
+ all_gatherv_test(comm, gps_generator(), "GPS positions");
+ all_gatherv_test(comm, string_generator(), "string");
+ all_gatherv_test(comm, string_list_generator(), "list of strings");
+}
diff --git a/src/boost/libs/mpi/test/all_reduce_test.cpp b/src/boost/libs/mpi/test/all_reduce_test.cpp
new file mode 100644
index 00000000..99fee9b8
--- /dev/null
+++ b/src/boost/libs/mpi/test/all_reduce_test.cpp
@@ -0,0 +1,309 @@
+// Copyright (C) 2005, 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the all_reduce() collective.
+#include <boost/mpi/collectives/all_reduce.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include <vector>
+#include <algorithm>
+#include <boost/serialization/string.hpp>
+#include <boost/iterator/counting_iterator.hpp>
+#include <boost/lexical_cast.hpp>
+#include <numeric>
+
+#define BOOST_TEST_MODULE mpi_all_reduce
+#include <boost/test/included/unit_test.hpp>
+
+using boost::mpi::communicator;
+
+// A simple point class that we can build, add, compare, and
+// serialize.
+struct point
+{
+ point() : x(0), y(0), z(0) { }
+ point(int x, int y, int z) : x(x), y(y), z(z) { }
+
+ int x;
+ int y;
+ int z;
+
+ private:
+ template<typename Archiver>
+ void serialize(Archiver& ar, unsigned int /*version*/)
+ {
+ ar & x & y & z;
+ }
+
+ friend class boost::serialization::access;
+};
+
+std::ostream& operator<<(std::ostream& out, const point& p)
+{
+ return out << p.x << ' ' << p.y << ' ' << p.z;
+}
+
+bool operator==(const point& p1, const point& p2)
+{
+ return p1.x == p2.x && p1.y == p2.y && p1.z == p2.z;
+}
+
+bool operator!=(const point& p1, const point& p2)
+{
+ return !(p1 == p2);
+}
+
+point operator+(const point& p1, const point& p2)
+{
+ return point(p1.x + p2.x, p1.y + p2.y, p1.z + p2.z);
+}
+
+// test lexical order
+bool operator<(const point& p1, const point& p2)
+{
+ return (p1.x < p2.x
+ ? true
+ : (p1.x > p2.x
+ ? false
+ : p1.y < p2.y ));
+}
+
+namespace boost { namespace mpi {
+
+ template <>
+ struct is_mpi_datatype<point> : public mpl::true_ { };
+
+} } // end namespace boost::mpi
+
+template<typename Generator, typename Op>
+void
+all_reduce_one_test(const communicator& comm, Generator generator,
+ const char* type_kind, Op op, const char* op_kind,
+ typename Generator::result_type init, bool in_place)
+{
+ typedef typename Generator::result_type value_type;
+ value_type value = generator(comm.rank());
+
+ using boost::mpi::all_reduce;
+ using boost::mpi::inplace;
+
+ if (comm.rank() == 0) {
+ std::cout << "Reducing to " << op_kind << " of " << type_kind << "...";
+ std::cout.flush();
+ }
+
+ value_type result_value;
+ if (in_place) {
+ all_reduce(comm, inplace(value), op);
+ result_value = value;
+ } else {
+ result_value = all_reduce(comm, value, op);
+ }
+
+ // Compute expected result
+ std::vector<value_type> generated_values;
+ for (int p = 0; p < comm.size(); ++p)
+ generated_values.push_back(generator(p));
+ value_type expected_result = std::accumulate(generated_values.begin(),
+ generated_values.end(),
+ init, op);
+ BOOST_CHECK(result_value == expected_result);
+ if (result_value == expected_result && comm.rank() == 0)
+ std::cout << "OK." << std::endl;
+
+ (comm.barrier)();
+}
+
+template<typename Generator, typename Op>
+void
+all_reduce_array_test(const communicator& comm, Generator generator,
+ const char* type_kind, Op op, const char* op_kind,
+ typename Generator::result_type init, bool in_place)
+{
+ typedef typename Generator::result_type value_type;
+ value_type value = generator(comm.rank());
+ std::vector<value_type> send(10, value);
+
+ using boost::mpi::all_reduce;
+ using boost::mpi::inplace;
+
+ if (comm.rank() == 0) {
+ char const* place = in_place ? "in place" : "out of place";
+ std::cout << "Reducing (" << place << ") array to " << op_kind << " of " << type_kind << "...";
+ std::cout.flush();
+ }
+ std::vector<value_type> result;
+ if (in_place) {
+ all_reduce(comm, inplace(&(send[0])), send.size(), op);
+ result.swap(send);
+ } else {
+ std::vector<value_type> recv(10, value_type());
+ all_reduce(comm, &(send[0]), send.size(), &(recv[0]), op);
+ result.swap(recv);
+ }
+
+ // Compute expected result
+ std::vector<value_type> generated_values;
+ for (int p = 0; p < comm.size(); ++p)
+ generated_values.push_back(generator(p));
+ value_type expected_result = std::accumulate(generated_values.begin(),
+ generated_values.end(),
+ init, op);
+
+ bool got_expected_result = (std::equal_range(result.begin(), result.end(),
+ expected_result)
+ == std::make_pair(result.begin(), result.end()));
+ BOOST_CHECK(got_expected_result);
+ if (got_expected_result && comm.rank() == 0)
+ std::cout << "OK." << std::endl;
+
+ (comm.barrier)();
+}
+
+// Test the 4 families of all reduce: (value, array) X (in place, out of place)
+template<typename Generator, typename Op>
+void
+all_reduce_test(const communicator& comm, Generator generator,
+ const char* type_kind, Op op, const char* op_kind,
+ typename Generator::result_type init)
+{
+ const bool in_place = true;
+ const bool out_of_place = false;
+ all_reduce_one_test(comm, generator, type_kind, op, op_kind, init, in_place);
+ all_reduce_one_test(comm, generator, type_kind, op, op_kind, init, out_of_place);
+ all_reduce_array_test(comm, generator, type_kind, op, op_kind,
+ init, in_place);
+ all_reduce_array_test(comm, generator, type_kind, op, op_kind,
+ init, out_of_place);
+}
+
+// Generates integers to test with all_reduce()
+struct int_generator
+{
+ typedef int result_type;
+
+ int_generator(int base = 1) : base(base) { }
+
+ int operator()(int p) const { return base + p; }
+
+ private:
+ int base;
+};
+
+// Generate points to test with all_reduce()
+struct point_generator
+{
+ typedef point result_type;
+
+ point_generator(point origin) : origin(origin) { }
+
+ point operator()(int p) const
+ {
+ return point(origin.x + 1, origin.y + 1, origin.z + 1);
+ }
+
+ private:
+ point origin;
+};
+
+struct string_generator
+{
+ typedef std::string result_type;
+
+ std::string operator()(int p) const
+ {
+ std::string result = boost::lexical_cast<std::string>(p);
+ result += " rosebud";
+ if (p != 1) result += 's';
+ return result;
+ }
+};
+
+struct secret_int_bit_and
+{
+ int operator()(int x, int y) const { return x & y; }
+};
+
+struct wrapped_int
+{
+ wrapped_int() : value(0) { }
+ explicit wrapped_int(int value) : value(value) { }
+
+ template<typename Archive>
+ void serialize(Archive& ar, unsigned int /* version */)
+ {
+ ar & value;
+ }
+
+ int value;
+};
+
+wrapped_int operator+(const wrapped_int& x, const wrapped_int& y)
+{
+ return wrapped_int(x.value + y.value);
+}
+
+bool operator==(const wrapped_int& x, const wrapped_int& y)
+{
+ return x.value == y.value;
+}
+
+bool operator<(const wrapped_int& x, const wrapped_int& y)
+{
+ return x.value < y.value;
+}
+
+// Generates wrapped_its to test with all_reduce()
+struct wrapped_int_generator
+{
+ typedef wrapped_int result_type;
+
+ wrapped_int_generator(int base = 1) : base(base) { }
+
+ wrapped_int operator()(int p) const { return wrapped_int(base + p); }
+
+ private:
+ int base;
+};
+
+namespace boost { namespace mpi {
+
+// Make std::plus<wrapped_int> commutative.
+template<>
+struct is_commutative<std::plus<wrapped_int>, wrapped_int>
+ : mpl::true_ { };
+
+} } // end namespace boost::mpi
+
+BOOST_AUTO_TEST_CASE(test_all_reduce)
+{
+ using namespace boost::mpi;
+ environment env;
+ communicator comm;
+
+ // Built-in MPI datatypes with built-in MPI operations
+ all_reduce_test(comm, int_generator(), "integers", std::plus<int>(), "sum", 0);
+ all_reduce_test(comm, int_generator(), "integers", std::multiplies<int>(), "product", 1);
+ all_reduce_test(comm, int_generator(), "integers", maximum<int>(), "maximum", 0);
+ all_reduce_test(comm, int_generator(), "integers", minimum<int>(), "minimum", 2);
+
+ // User-defined MPI datatypes with operations that have the
+ // same name as built-in operations.
+ all_reduce_test(comm, point_generator(point(0,0,0)), "points", std::plus<point>(),
+ "sum", point());
+
+ // Built-in MPI datatypes with user-defined operations
+ all_reduce_test(comm, int_generator(17), "integers", secret_int_bit_and(),
+ "bitwise and", -1);
+
+ // Arbitrary types with user-defined, commutative operations.
+ all_reduce_test(comm, wrapped_int_generator(17), "wrapped integers",
+ std::plus<wrapped_int>(), "sum", wrapped_int(0));
+
+ // Arbitrary types with (non-commutative) user-defined operations
+ all_reduce_test(comm, string_generator(), "strings",
+ std::plus<std::string>(), "concatenation", std::string());
+}
diff --git a/src/boost/libs/mpi/test/all_to_all_test.cpp b/src/boost/libs/mpi/test/all_to_all_test.cpp
new file mode 100644
index 00000000..0e27dcf0
--- /dev/null
+++ b/src/boost/libs/mpi/test/all_to_all_test.cpp
@@ -0,0 +1,113 @@
+// Copyright (C) 2005, 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the all_to_all() collective.
+#include <boost/mpi/collectives/all_to_all.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include <algorithm>
+#include "gps_position.hpp"
+#include <boost/serialization/string.hpp>
+#include <boost/serialization/list.hpp>
+#include <boost/iterator/counting_iterator.hpp>
+#include <boost/lexical_cast.hpp>
+
+#define BOOST_TEST_MODULE mpi_all_to_all
+#include <boost/test/included/unit_test.hpp>
+
+using boost::mpi::communicator;
+
+using boost::mpi::packed_skeleton_iarchive;
+using boost::mpi::packed_skeleton_oarchive;
+
+template<typename Generator>
+void
+all_to_all_test(const communicator& comm, Generator generator,
+ const char* kind)
+{
+ typedef typename Generator::result_type value_type;
+
+ using boost::mpi::all_to_all;
+
+ std::vector<value_type> in_values;
+ for (int p = 0; p < comm.size(); ++p)
+ in_values.push_back(generator((p + 1) * (comm.rank() + 1)));
+
+ if (comm.rank() == 0) {
+ std::cout << "Performing all-to-all operation on " << kind << "...";
+ std::cout.flush();
+ }
+ std::vector<value_type> out_values;
+ all_to_all(comm, in_values, out_values);
+
+ for (int p = 0; p < comm.size(); ++p) {
+ BOOST_CHECK(out_values[p] == generator((p + 1) * (comm.rank() + 1)));
+ }
+
+ if (comm.rank() == 0) {
+ std::cout << " done." << std::endl;
+ }
+
+ (comm.barrier)();
+}
+
+// Generates integers to test with all_to_all()
+struct int_generator
+{
+ typedef int result_type;
+
+ int operator()(int p) const { return 17 + p; }
+};
+
+// Generates GPS positions to test with all_to_all()
+struct gps_generator
+{
+ typedef gps_position result_type;
+
+ gps_position operator()(int p) const
+ {
+ return gps_position(39 + p, 16, 20.2799);
+ }
+};
+
+struct string_generator
+{
+ typedef std::string result_type;
+
+ std::string operator()(int p) const
+ {
+ std::string result = boost::lexical_cast<std::string>(p);
+ result += " rosebud";
+ if (p != 1) result += 's';
+ return result;
+ }
+};
+
+struct string_list_generator
+{
+ typedef std::list<std::string> result_type;
+
+ std::list<std::string> operator()(int p) const
+ {
+ std::list<std::string> result;
+ for (int i = 0; i <= p; ++i) {
+ std::string value = boost::lexical_cast<std::string>(i);
+ result.push_back(value);
+ }
+ return result;
+ }
+};
+
+BOOST_AUTO_TEST_CASE(all_to_all_check)
+{
+ boost::mpi::environment env;
+ communicator comm;
+
+ all_to_all_test(comm, int_generator(), "integers");
+ all_to_all_test(comm, gps_generator(), "GPS positions");
+ all_to_all_test(comm, string_generator(), "string");
+ all_to_all_test(comm, string_list_generator(), "list of strings");
+}
diff --git a/src/boost/libs/mpi/test/block_nonblock_test.cpp b/src/boost/libs/mpi/test/block_nonblock_test.cpp
new file mode 100644
index 00000000..3088b655
--- /dev/null
+++ b/src/boost/libs/mpi/test/block_nonblock_test.cpp
@@ -0,0 +1,95 @@
+#include <vector>
+#include <iostream>
+#include <iterator>
+#include <typeinfo>
+
+#include <boost/mpi.hpp>
+#include <boost/serialization/vector.hpp>
+#include <boost/core/demangle.hpp>
+
+//#include "debugger.cpp"
+
+#define BOOST_TEST_MODULE mpi_nonblocking
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+template<typename T>
+bool test(mpi::communicator const& comm, std::vector<T> const& ref, bool iswap, bool alloc)
+{
+
+ int rank = comm.rank();
+ if (rank == 0) {
+ std::cout << "Testing with type " << boost::core::demangle(typeid(T).name()) << '\n';
+ if (iswap) {
+ std::cout << "Blockin send, non blocking receive.\n";
+ } else {
+ std::cout << "Non blockin send, blocking receive.\n";
+ }
+ if (alloc) {
+ std::cout << "Explicitly allocate space for the receiver.\n";
+ } else {
+ std::cout << "Do not explicitly allocate space for the receiver.\n";
+ }
+ }
+ if (rank == 0) {
+ std::vector<T> data;
+ if (alloc) {
+ data.resize(ref.size());
+ }
+ if (iswap) {
+ mpi::request req = comm.irecv(1, 0, data);
+ req.wait();
+ } else {
+ comm.recv(1, 0, data);
+ }
+ std::cout << "Process 0 received " << data.size() << " elements :" << std::endl;
+ std::copy(data.begin(), data.end(), std::ostream_iterator<T>(std::cout, " "));
+ std::cout << std::endl;
+ std::cout << "While expecting " << ref.size() << " elements :" << std::endl;
+ std::copy(ref.begin(), ref.end(), std::ostream_iterator<T>(std::cout, " "));
+ std::cout << std::endl;
+ return (data == ref);
+ } else {
+ if (rank == 1) {
+ std::vector<T> vec = ref;
+ if (iswap) {
+ comm.send(0, 0, vec);
+ } else {
+ mpi::request req = comm.isend(0, 0, vec);
+ req.wait();
+ }
+ }
+ return true;
+ }
+}
+
+BOOST_AUTO_TEST_CASE(non_blocking)
+{
+ mpi::environment env;
+ mpi::communicator world;
+
+ BOOST_TEST_REQUIRE(world.size() > 1);
+
+ std::vector<int> integers(13); // don't assume we're lucky
+ for(int i = 0; i < int(integers.size()); ++i) {
+ integers[i] = i;
+ }
+
+ std::vector<std::string> strings(13); // don't assume we're lucky
+ for(int i = 0; i < int(strings.size()); ++i) {
+ std::ostringstream fmt;
+ fmt << "S" << i;
+ strings[i] = fmt.str();
+ }
+
+ BOOST_CHECK(test(world, integers, true, true));
+ BOOST_CHECK(test(world, integers, true, false));
+ BOOST_CHECK(test(world, strings, true, true));
+ BOOST_CHECK(test(world, strings, true, false));
+
+ BOOST_CHECK(test(world, integers, false, true));
+ BOOST_CHECK(test(world, integers, false, false));
+ BOOST_CHECK(test(world, strings, false, true));
+ BOOST_CHECK(test(world, strings, false, false));
+}
diff --git a/src/boost/libs/mpi/test/broadcast_stl_test.cpp b/src/boost/libs/mpi/test/broadcast_stl_test.cpp
new file mode 100644
index 00000000..705d6ccb
--- /dev/null
+++ b/src/boost/libs/mpi/test/broadcast_stl_test.cpp
@@ -0,0 +1,75 @@
+// Copyright (C) 2005, 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the broadcast() collective.
+#include <algorithm>
+#include <vector>
+#include <map>
+
+#include <boost/mpi/collectives/broadcast.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+
+#include <boost/serialization/string.hpp>
+#include <boost/serialization/vector.hpp>
+#include <boost/serialization/map.hpp>
+
+#define BOOST_TEST_MODULE mpi_broadcast_stl
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+typedef std::vector<std::map<int, double> > sparse;
+
+template<typename T>
+void
+broadcast_test(const mpi::communicator& comm, const T& bc_value,
+ std::string const& kind, int root) {
+ using boost::mpi::broadcast;
+
+ T value;
+ if (comm.rank() == root) {
+ value = bc_value;
+ std::cout << "Broadcasting " << kind << " from root " << root << "...";
+ std::cout.flush();
+ }
+
+
+ broadcast(comm, value, root);
+ BOOST_CHECK(value == bc_value);
+ if (comm.rank() == root) {
+ if (value == bc_value) {
+ std::cout << "OK." << std::endl;
+ } else {
+ std::cout << "FAIL." << std::endl;
+ }
+ }
+ comm.barrier();
+}
+
+template<typename T>
+void
+broadcast_test(const mpi::communicator& comm, const T& bc_value,
+ std::string const& kind)
+{
+ for (int root = 0; root < comm.size(); ++root) {
+ broadcast_test(comm, bc_value, kind, root);
+ }
+}
+
+BOOST_AUTO_TEST_CASE(broadcast_stl)
+{
+ boost::mpi::environment env;
+
+ mpi::communicator comm;
+ BOOST_TEST_REQUIRE(comm.size() > 1);
+
+ sparse s;
+ s.resize(2);
+ s[0][12] = 0.12;
+ s[1][13] = 1.13;
+ broadcast_test(comm, s, "sparse");
+}
diff --git a/src/boost/libs/mpi/test/broadcast_test.cpp b/src/boost/libs/mpi/test/broadcast_test.cpp
new file mode 100644
index 00000000..50081aad
--- /dev/null
+++ b/src/boost/libs/mpi/test/broadcast_test.cpp
@@ -0,0 +1,159 @@
+// Copyright (C) 2005, 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the broadcast() collective.
+#include <boost/mpi/collectives/broadcast.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include <algorithm>
+#include "gps_position.hpp"
+#include <boost/serialization/string.hpp>
+#include <boost/serialization/list.hpp>
+#include <boost/mpi/skeleton_and_content.hpp>
+#include <boost/iterator/counting_iterator.hpp>
+//#include "debugger.hpp"
+
+#define BOOST_TEST_MODULE mpi_broadcast
+#include <boost/test/included/unit_test.hpp>
+
+using boost::mpi::communicator;
+
+using boost::mpi::packed_skeleton_iarchive;
+using boost::mpi::packed_skeleton_oarchive;
+
+template<typename T>
+void
+broadcast_test(const communicator& comm, const T& bc_value,
+ const char* kind, int root = -1)
+{
+ if (root == -1) {
+ for (root = 0; root < comm.size(); ++root)
+ broadcast_test(comm, bc_value, kind, root);
+ } else {
+ using boost::mpi::broadcast;
+
+ T value;
+ if (comm.rank() == root) {
+ value = bc_value;
+ std::cout << "Broadcasting " << kind << " from root " << root << "...";
+ std::cout.flush();
+ }
+
+ broadcast(comm, value, root);
+ BOOST_CHECK(value == bc_value);
+ if (comm.rank() == root && value == bc_value)
+ std::cout << "OK." << std::endl;
+ }
+
+ (comm.barrier)();
+}
+
+void
+test_skeleton_and_content(const communicator& comm, int root = 0)
+{
+ using boost::mpi::content;
+ using boost::mpi::get_content;
+ using boost::make_counting_iterator;
+ using boost::mpi::broadcast;
+
+ int list_size = comm.size() + 7;
+ if (comm.rank() == root) {
+ // Fill in the seed data
+ std::list<int> original_list;
+ for (int i = 0; i < list_size; ++i)
+ original_list.push_back(i);
+
+ // Build up the skeleton
+ packed_skeleton_oarchive oa(comm);
+ oa << original_list;
+
+ // Broadcast the skeleton
+ std::cout << "Broadcasting integer list skeleton from root " << root
+ << "..." << std::flush;
+ broadcast(comm, oa, root);
+ std::cout << "OK." << std::endl;
+
+ // Broadcast the content
+ std::cout << "Broadcasting integer list content from root " << root
+ << "..." << std::flush;
+ {
+ content c = get_content(original_list);
+ broadcast(comm, c, root);
+ }
+ std::cout << "OK." << std::endl;
+
+ // Reverse the list, broadcast the content again
+ std::reverse(original_list.begin(), original_list.end());
+ std::cout << "Broadcasting reversed integer list content from root "
+ << root << "..." << std::flush;
+ {
+ content c = get_content(original_list);
+ broadcast(comm, c, root);
+ }
+ std::cout << "OK." << std::endl;
+
+ } else {
+ // Allocate some useless data, to try to get the addresses of the
+ // list<int>'s used later to be different across processes.
+ std::list<int> junk_list(comm.rank() * 3 + 1, 17);
+
+ // Receive the skeleton
+ packed_skeleton_iarchive ia(comm);
+ broadcast(comm, ia, root);
+
+ // Build up a list to match the skeleton, and make sure it has the
+ // right structure (we have no idea what the data will be).
+ std::list<int> transferred_list;
+ ia >> transferred_list;
+ BOOST_CHECK((int)transferred_list.size() == list_size);
+
+ // Receive the content and check it
+ broadcast(comm, get_content(transferred_list), root);
+ bool list_content_ok = std::equal(make_counting_iterator(0),
+ make_counting_iterator(list_size),
+ transferred_list.begin());
+ BOOST_CHECK(list_content_ok);
+
+ // Receive the reversed content and check it
+ broadcast(comm, get_content(transferred_list), root);
+ bool rlist_content_ok = std::equal(make_counting_iterator(0),
+ make_counting_iterator(list_size),
+ transferred_list.rbegin());
+ BOOST_CHECK(rlist_content_ok);
+ if (!(list_content_ok && rlist_content_ok)) {
+ if (comm.rank() == 1) {
+ std::cout
+ << "\n##### You might want to check for BOOST_MPI_BCAST_BOTTOM_WORKS_FINE "
+ << "in boost/mpi/config.hpp.\n\n";
+ }
+ }
+ }
+
+ (comm.barrier)();
+}
+
+BOOST_AUTO_TEST_CASE(broadcast_check)
+{
+ boost::mpi::environment env;
+ communicator comm;
+
+ BOOST_TEST_REQUIRE(comm.size() > 1);
+
+ // Check transfer of individual objects
+ broadcast_test(comm, 17, "integers");
+ broadcast_test(comm, gps_position(39,16,20.2799), "GPS positions");
+ broadcast_test(comm, gps_position(26,25,30.0), "GPS positions");
+ broadcast_test(comm, std::string("Rosie"), "string");
+
+ std::list<std::string> strings;
+ strings.push_back("Hello");
+ strings.push_back("MPI");
+ strings.push_back("World");
+ broadcast_test(comm, strings, "list of strings");
+
+ test_skeleton_and_content(comm, 0);
+ test_skeleton_and_content(comm, 1);
+}
diff --git a/src/boost/libs/mpi/test/cartesian_topology_init_test.cpp b/src/boost/libs/mpi/test/cartesian_topology_init_test.cpp
new file mode 100644
index 00000000..6138a2e6
--- /dev/null
+++ b/src/boost/libs/mpi/test/cartesian_topology_init_test.cpp
@@ -0,0 +1,88 @@
+
+// Copyright Alain Miniussi 2014.
+// Distributed under the Boost Software License, Version 1.0.
+// (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Alain Miniussi
+
+#include <vector>
+#include <list>
+#include <iostream>
+#include <sstream>
+#include <iterator>
+#include <algorithm>
+#include <functional>
+
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/collectives.hpp>
+#include <boost/array.hpp>
+#include <boost/mpi/environment.hpp>
+#include <boost/mpi/cartesian_communicator.hpp>
+
+#define BOOST_TEST_MODULE mpi_cartesian_topology_init
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+BOOST_AUTO_TEST_CASE(cartesian_dimension_init)
+{
+ // Curly brace initialization syntax not supported on (very) old gnu
+ // This typedef keeps things shorter
+ typedef mpi::cartesian_dimension cd;
+
+ {
+ // Check the basic ctor
+ mpi::cartesian_dimension def;
+ mpi::cartesian_topology t1(10);
+ BOOST_CHECK(t1.stl() == std::vector<mpi::cartesian_dimension>(10, def));
+ }
+#if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
+ {
+ // Intializer list ctor vs range based
+ int dims[] = {2,3,4};
+ bool per[] = {true, false, true};
+ mpi::cartesian_topology t1(dims, per);
+ mpi::cartesian_topology t2({{2,true},{3, false},{4, true}});
+ BOOST_CHECK(t1.size() == 3);
+ BOOST_CHECK(t1 == t2);
+ }
+#endif
+ // Container based ctor only available as a replacement for initializer list ctor
+ {
+ // seq ctor vs C array ctor
+ mpi::cartesian_dimension d[] = {cd(2,true),cd(3, false),cd(4, true)};
+ std::list<mpi::cartesian_dimension> seq;
+ std::copy(d, d+3, std::back_inserter(seq));
+ mpi::cartesian_topology t1(seq);
+ mpi::cartesian_topology t2(d);
+ BOOST_CHECK(t1 == t2);
+ }
+ {
+ // Check range based with array based ctor.
+ boost::array<mpi::cartesian_dimension, 3> d = {{cd(2,true),cd(3, false),cd(4, true)}};
+ int dims[] = {2,3,4};
+ bool per[] = {true, false, true};
+ mpi::cartesian_topology t1(dims, per);
+ mpi::cartesian_topology t2(d);
+ BOOST_CHECK(t1.size() == 3);
+ BOOST_CHECK(t1 == t2);
+ }
+ {
+ // Iterator based ctor vs C array based ctor
+ mpi::cartesian_dimension d[] = {cd(2,true),cd(3, false),cd(4, true)};
+ std::vector<mpi::cartesian_dimension> vdims(d, d+3);
+ mpi::cartesian_topology t1(vdims);
+ mpi::cartesian_topology t2(d);
+ BOOST_CHECK(t1.size() == 3);
+ BOOST_CHECK(t1 == t2);
+ BOOST_CHECK(!(t1 != t2));
+ t1[1].periodic = true;
+ BOOST_CHECK(t1 != t2);
+ t1[2].periodic = false;
+ t1[2].size = 0;
+ vdims.push_back(mpi::cartesian_dimension(3, false));
+ mpi::cartesian_topology t3(vdims);
+ BOOST_CHECK(t1 != t3);
+ }
+}
diff --git a/src/boost/libs/mpi/test/cartesian_topology_test.cpp b/src/boost/libs/mpi/test/cartesian_topology_test.cpp
new file mode 100644
index 00000000..d63fc500
--- /dev/null
+++ b/src/boost/libs/mpi/test/cartesian_topology_test.cpp
@@ -0,0 +1,193 @@
+// Copyright Alain Miniussi 2014.
+// Distributed under the Boost Software License, Version 1.0.
+// (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Alain Miniussi
+
+#include <vector>
+#include <iostream>
+#include <sstream>
+#include <iterator>
+#include <algorithm>
+#include <functional>
+
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/collectives.hpp>
+#include <boost/mpi/environment.hpp>
+#include <boost/mpi/cartesian_communicator.hpp>
+
+#define BOOST_TEST_MODULE mpi_cartesian_topolohy
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+struct topo_minimum {
+ mpi::cartesian_dimension
+ operator()(mpi::cartesian_dimension const& d1,
+ mpi::cartesian_dimension const& d2 ) const {
+ return mpi::cartesian_dimension(std::min(d1.size, d2.size),
+ d1.periodic && d2.periodic);
+ }
+};
+
+std::string topology_description( mpi::cartesian_topology const& topo ) {
+ std::ostringstream out;
+ std::copy(topo.begin(), topo.end(), std::ostream_iterator<mpi::cartesian_dimension>(out, " "));
+ out << std::flush;
+ return out.str();
+}
+
+// Check that everyone agrees on the coordinates
+void test_coordinates_consistency( mpi::cartesian_communicator const& cc,
+ std::vector<int> const& coords )
+{
+ cc.barrier(); // flush IOs for nice printing
+ bool master = cc.rank() == 0;
+ if (master) {
+ std::cout << "Test coordinates consistency.\n";
+ }
+ for(int p = 0; p < cc.size(); ++p) {
+ std::vector<int> min(cc.ndims());
+ std::vector<int> local(cc.coordinates(p));
+ mpi::reduce(cc, &local.front(), local.size(),
+ &(min[0]), mpi::minimum<int>(), p);
+ cc.barrier();
+ if (p == cc.rank()) {
+ BOOST_CHECK(std::equal(coords.begin(), coords.end(), min.begin()));
+ std::ostringstream out;
+ out << "proc " << p << " at (";
+ std::copy(min.begin(), min.end(), std::ostream_iterator<int>(out, " "));
+ out << ")\n";
+ std::cout << out.str();
+ }
+ }
+}
+
+void test_shifted_coords( mpi::cartesian_communicator const& cc, int pos, mpi::cartesian_dimension desc, int dim )
+{
+ if (desc.periodic) {
+ for (int i = -(desc.size); i < desc.size; ++i) {
+ std::pair<int,int> rks = cc.shifted_ranks(dim, i);
+ int src = cc.coordinates(rks.first)[dim];
+ int dst = cc.coordinates(rks.second)[dim];
+ if (pos == (dim/2)) {
+ std::ostringstream out;
+ out << "Rank " << cc.rank() << ", dim. " << dim << ", pos " << pos << ", in " << desc << ' ';
+ out << "shifted pos: " << src << ", " << dst << '\n';
+ std::cout << out.str();
+ }
+ }
+ }
+}
+
+void test_shifted_coords( mpi::cartesian_communicator const& cc)
+{
+ cc.barrier(); // flush IOs for nice printing
+ std::vector<int> coords;
+ mpi::cartesian_topology topo(cc.ndims());
+ cc.topology(topo, coords);
+ bool master = cc.rank() == 0;
+ if (master) {
+ std::cout << "Testing shifts with topology " << topo << '\n';
+ }
+ for(int i = 0; i < cc.ndims(); ++i) {
+ if (master) {
+ std::cout << " for dimension " << i << ": " << topo[i] << '\n';
+ }
+ test_shifted_coords( cc, coords[i], topo[i], i );
+ }
+}
+
+void test_topology_consistency( mpi::cartesian_communicator const& cc)
+{
+ cc.barrier(); // flush IOs for nice printing
+ mpi::cartesian_topology itopo(cc.ndims());
+ mpi::cartesian_topology otopo(cc.ndims());
+ std::vector<int> coords(cc.ndims());
+ cc.topology(itopo, coords);
+ bool master = cc.rank() == 0;
+ if (master) {
+ std::cout << "Test topology consistency of" << itopo << "(on master)\n";
+ std::cout << "Check that everyone agrees on the dimensions.\n";
+ }
+ mpi::all_reduce(cc,
+ &(itopo[0]), itopo.size(), &(otopo[0]),
+ topo_minimum());
+ BOOST_CHECK(std::equal(itopo.begin(), itopo.end(), otopo.begin()));
+ if (master) {
+ std::cout << "We agree on " << topology_description(otopo) << '\n';
+ }
+ test_coordinates_consistency( cc, coords );
+}
+
+void test_cartesian_topology( mpi::cartesian_communicator const& cc)
+{
+ BOOST_CHECK(cc.has_cartesian_topology());
+ for( int r = 0; r < cc.size(); ++r) {
+ cc.barrier();
+ if (r == cc.rank()) {
+ std::vector<int> coords = cc.coordinates(r);
+ std::cout << "Process of cartesian rank " << cc.rank()
+ << " has coordinates (";
+ std::copy(coords.begin(), coords.end(), std::ostream_iterator<int>(std::cout," "));
+ std::cout << ")\n";
+ }
+ }
+ test_topology_consistency(cc);
+ test_shifted_coords(cc);
+ std::vector<int> even;
+ for(int i = 0; i < cc.ndims(); i += 2) {
+ even.push_back(i);
+ }
+ cc.barrier();
+ mpi::cartesian_communicator cce(cc, even);
+}
+
+void test_cartesian_topology( mpi::communicator const& world, mpi::cartesian_topology const& topo)
+{
+ mpi::cartesian_communicator cc(world, topo, true);
+ if (cc) {
+ BOOST_CHECK(cc.has_cartesian_topology());
+ BOOST_CHECK(cc.ndims() == int(topo.size()));
+ if (cc.rank() == 0) {
+ std::cout << "Asked topology " << topo << ", got " << cc.topology() << '\n';
+ }
+ test_cartesian_topology(cc);
+ } else {
+ std::ostringstream out;
+ out << world.rank() << " was left outside the cartesian grid\n";
+ std::cout << out.str();
+ }
+}
+
+BOOST_AUTO_TEST_CASE(cartesian_topology)
+{
+ mpi::environment env;
+ mpi::communicator world;
+
+ int const ndim = world.size() >= 24 ? 3 : 2;
+ mpi::cartesian_topology topo(ndim);
+ typedef mpi::cartesian_dimension cd;
+ if (topo.size() == 3) {
+ topo[0] = cd(2,true);
+ topo[1] = cd(3,false);
+ topo[2] = cd(4, true);
+ } else {
+ if (world.size() >= 6) {
+ topo[0] = cd(2,true);
+ topo[1] = cd(3, false);
+ } else {
+ topo[0] = cd(1,true);
+ topo[1] = cd(1, false);
+ }
+ }
+ test_cartesian_topology( world, topo);
+#if !defined(BOOST_NO_CXX11_DEFAULTED_MOVES)
+ world.barrier();
+ if (world.rank()==0) {
+ std::cout << "Testing move constructor.\n";
+ }
+ test_cartesian_topology( world, std::move(topo));
+#endif
+}
diff --git a/src/boost/libs/mpi/test/debugger.cpp b/src/boost/libs/mpi/test/debugger.cpp
new file mode 100644
index 00000000..90093e79
--- /dev/null
+++ b/src/boost/libs/mpi/test/debugger.cpp
@@ -0,0 +1,48 @@
+// Copyright AlainMiniussi 20014 - 20015.
+// Distributed under the Boost Software License, Version 1.0.
+// (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+#include <cstdlib>
+
+#include "debugger.hpp"
+
+std::vector<int> extract_paused_ranks(int argc, char** argv) {
+ std::vector<int> paused;
+ for (int i=1; i < argc; ++i) {
+ paused.push_back(std::atoi(argv[i]));
+ }
+ return paused;
+}
+
+void wait_for_debugger(std::vector<int> const& processes, boost::mpi::communicator const& comm)
+{
+ int i = 1;
+ bool waiting = std::find(processes.begin(), processes.end(), comm.rank()) != processes.end();
+ for (int r = 0; r < comm.size(); ++r) {
+ if (comm.rank() == r) {
+ std::cout << "Rank " << comm.rank() << " has PID " << getpid();
+ if (waiting) {
+ std::cout << " and is waiting.";
+ }
+ std::cout << std::endl;
+ }
+ comm.barrier();
+ }
+ if (std::find(processes.begin(), processes.end(), comm.rank()) != processes.end()) {
+ while (i != 0) {
+ sleep(5);
+ }
+ }
+ std::cout << "Rank " << comm.rank() << " will proceed.\n";
+}
+
+void wait_for_debugger(boost::mpi::communicator const& comm)
+{
+ std::vector<int> all;
+ for (int r = 0; r < comm.size(); ++r) {
+ all.push_back(r);
+ }
+ wait_for_debugger(all, comm);
+}
+
diff --git a/src/boost/libs/mpi/test/debugger.hpp b/src/boost/libs/mpi/test/debugger.hpp
new file mode 100644
index 00000000..71abf51d
--- /dev/null
+++ b/src/boost/libs/mpi/test/debugger.hpp
@@ -0,0 +1,26 @@
+// Copyright AlainMiniussi 20014 - 20015.
+// Distributed under the Boost Software License, Version 1.0.
+// (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+#include <vector>
+#include "boost/mpi/communicator.hpp"
+
+/**
+ * @brief Extract the MPI rank to pause.
+ *
+ * Right now, just atois alla the parameters in argv....
+ */
+std::vector<int> extract_paused_ranks(int argc, char** argv);
+
+/**
+ * @print Print rank pid map and wait if requested.
+ * @param processes Wait if our rank is in there.
+ * @param comm The communicator to consider.
+ *
+ * Once the debugger has attached to the process, it is expected to
+ * set the local variable 'i' to 0 to let the process restarts.
+ */
+void wait_for_debugger(std::vector<int> const& processes, boost::mpi::communicator const& comm);
+/** @override */
+void wait_for_debugger(boost::mpi::communicator const& comm);
diff --git a/src/boost/libs/mpi/test/gather_test.cpp b/src/boost/libs/mpi/test/gather_test.cpp
new file mode 100644
index 00000000..9bef7acd
--- /dev/null
+++ b/src/boost/libs/mpi/test/gather_test.cpp
@@ -0,0 +1,165 @@
+// Copyright (C) 2005, 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the gather() and gatherv() collectives.
+#include <boost/mpi/collectives/gather.hpp>
+#include <boost/mpi/collectives/gatherv.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include "gps_position.hpp"
+#include <boost/serialization/string.hpp>
+#include <boost/serialization/list.hpp>
+#include <boost/iterator/counting_iterator.hpp>
+#include <boost/lexical_cast.hpp>
+
+#define BOOST_TEST_MODULE mpi_gather
+#include <boost/test/included/unit_test.hpp>
+
+using boost::mpi::communicator;
+
+template<typename Generator>
+void
+gather_test(const communicator& comm, Generator generator,
+ const char* kind, int root = -1)
+{
+ typedef typename Generator::result_type value_type;
+ value_type value = generator(comm.rank());
+
+ if (root == -1) {
+ for (root = 0; root < comm.size(); ++root)
+ gather_test(comm, generator, kind, root);
+ } else {
+ using boost::mpi::gather;
+
+ std::vector<value_type> values;
+ if (comm.rank() == root) {
+ std::cout << "Gathering " << kind << " from root "
+ << root << "..." << std::endl;
+ }
+
+ gather(comm, value, values, root);
+
+ if (comm.rank() == root) {
+ std::vector<value_type> expected_values;
+ for (int p = 0; p < comm.size(); ++p)
+ expected_values.push_back(generator(p));
+ BOOST_CHECK(values == expected_values);
+ } else {
+ BOOST_CHECK(values.empty());
+ }
+ }
+
+ (comm.barrier)();
+}
+
+
+template<typename Generator>
+void
+gatherv_test(const communicator& comm, Generator generator,
+ const char* kind, int root = -1)
+{
+ typedef typename Generator::result_type value_type;
+
+ if (root == -1) {
+ for (root = 0; root < comm.size(); ++root)
+ gatherv_test(comm, generator, kind, root);
+ } else {
+ using boost::mpi::gatherv;
+
+ int mysize = comm.rank() + 1;
+ int nprocs = comm.size();
+
+ // process p will send p+1 identical generator(p) elements
+ std::vector<value_type> myvalues(mysize, generator(comm.rank()));
+
+ if (comm.rank() == root) {
+ std::vector<value_type> values((nprocs*(nprocs+1))/2);
+ std::vector<int> sizes(comm.size());
+ for (int p = 0; p < comm.size(); ++p)
+ sizes[p] = p + 1;
+
+ std::cout << "Gatheringv " << kind << " from root "
+ << root << "..." << std::endl;
+
+ gatherv(comm, myvalues, &values[0], sizes, root);
+
+ std::vector<value_type> expected_values;
+ for (int p = 0; p < comm.size(); ++p)
+ for (int i = 0; i < p+1; ++i)
+ expected_values.push_back(generator(p));
+
+ BOOST_CHECK(values == expected_values);
+ } else {
+ gatherv(comm, myvalues, root);
+ }
+ }
+
+ (comm.barrier)();
+}
+
+//
+// Generators to test with gather/gatherv
+//
+struct int_generator
+{
+ typedef int result_type;
+
+ int operator()(int p) const { return 17 + p; }
+};
+
+struct gps_generator
+{
+ typedef gps_position result_type;
+
+ gps_position operator()(int p) const
+ {
+ return gps_position(39 + p, 16, 20.2799);
+ }
+};
+
+struct string_generator
+{
+ typedef std::string result_type;
+
+ std::string operator()(int p) const
+ {
+ std::string result = boost::lexical_cast<std::string>(p);
+ result += " rosebud";
+ if (p != 1) result += 's';
+ return result;
+ }
+};
+
+struct string_list_generator
+{
+ typedef std::list<std::string> result_type;
+
+ std::list<std::string> operator()(int p) const
+ {
+ std::list<std::string> result;
+ for (int i = 0; i <= p; ++i) {
+ std::string value = boost::lexical_cast<std::string>(i);
+ result.push_back(value);
+ }
+ return result;
+ }
+};
+
+BOOST_AUTO_TEST_CASE(gather_check)
+{
+ boost::mpi::environment env;
+ communicator comm;
+
+ gather_test(comm, int_generator(), "integers");
+ gather_test(comm, gps_generator(), "GPS positions");
+ gather_test(comm, string_generator(), "string");
+ gather_test(comm, string_list_generator(), "list of strings");
+
+ gatherv_test(comm, int_generator(), "integers");
+ gatherv_test(comm, gps_generator(), "GPS positions");
+ gatherv_test(comm, string_generator(), "string");
+ gatherv_test(comm, string_list_generator(), "list of strings");
+}
diff --git a/src/boost/libs/mpi/test/gps_position.hpp b/src/boost/libs/mpi/test/gps_position.hpp
new file mode 100644
index 00000000..e6910c44
--- /dev/null
+++ b/src/boost/libs/mpi/test/gps_position.hpp
@@ -0,0 +1,69 @@
+#ifndef GPS_POSITION_HPP
+#define GPS_POSITION_HPP
+
+// Copyright Matthias Troyer
+// 2005. Distributed under the Boost Software License, Version
+// 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+#include <boost/mpi/datatype_fwd.hpp>
+#include <boost/mpl/and.hpp>
+#include <boost/serialization/export.hpp>
+#include <boost/shared_ptr.hpp>
+#include <iostream>
+
+class gps_position
+{
+private:
+ friend class boost::serialization::access;
+ // When the class Archive corresponds to an output archive, the
+ // & operator is defined similar to <<. Likewise, when the class Archive
+ // is a type of input archive the & operator is defined similar to >>.
+ template<class Archive>
+ void serialize(Archive & ar, const unsigned int version)
+ {
+ ar & degrees & minutes & seconds;
+ }
+ int degrees;
+ int minutes;
+ float seconds;
+public:
+ gps_position(){};
+ gps_position(int d, int m, float s) :
+ degrees(d), minutes(m), seconds(s)
+ {}
+
+ friend std::ostream& operator<<(std::ostream& out, const gps_position& g);
+
+ friend bool operator==(const gps_position& x, const gps_position& y)
+ {
+ return (x.degrees == y.degrees
+ && x.minutes == y.minutes
+ && x.seconds == y.seconds);
+ }
+
+ inline friend bool operator!=(const gps_position& x, const gps_position& y)
+ {
+ return !(x == y);
+ }
+};
+
+inline
+std::ostream& operator<<(std::ostream& out, const gps_position& g) {
+ out << "gps{" << g.degrees << 'd' << g.minutes << 'm' << g.seconds << "s}";
+ return out;
+}
+
+namespace boost { namespace mpi {
+
+ template <>
+ struct is_mpi_datatype<gps_position>
+ : public mpl::and_
+ <
+ is_mpi_datatype<int>,
+ is_mpi_datatype<float>
+ >
+ {};
+
+} }
+#endif
diff --git a/src/boost/libs/mpi/test/graph_topology_test.cpp b/src/boost/libs/mpi/test/graph_topology_test.cpp
new file mode 100644
index 00000000..4d781265
--- /dev/null
+++ b/src/boost/libs/mpi/test/graph_topology_test.cpp
@@ -0,0 +1,141 @@
+// Copyright (C) 2007 Trustees of Indiana University
+
+// Authors: Douglas Gregor
+// Andrew Lumsdaine
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the communicator that passes data around a ring and
+// verifies that the same data makes it all the way. Should test all
+// of the various kinds of data that can be sent (primitive types, POD
+// types, serializable objects, etc.)
+#include <boost/mpi/graph_communicator.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include <boost/graph/adjacency_list.hpp>
+#include <boost/lexical_cast.hpp>
+#include <boost/graph/erdos_renyi_generator.hpp>
+#include <boost/random/linear_congruential.hpp>
+#include <boost/graph/iteration_macros.hpp>
+#include <boost/graph/isomorphism.hpp>
+#include <algorithm> // for random_shuffle
+#include <boost/serialization/vector.hpp>
+#include <boost/mpi/collectives/broadcast.hpp>
+#include <boost/config.hpp>
+
+#define BOOST_TEST_MODULE mpi_graph_topology
+#include <boost/test/included/unit_test.hpp>
+
+#if defined(BOOST_NO_CXX98_RANDOM_SHUFFLE)
+
+#include <random>
+
+std::default_random_engine gen;
+
+template<typename RandomIt> void random_shuffle( RandomIt first, RandomIt last )
+{
+ std::shuffle( first, last, gen );
+}
+
+#else
+
+using std::random_shuffle;
+
+#endif // #if defined(BOOST_NO_CXX98_RANDOM_SHUFFLE)
+
+
+using boost::mpi::communicator;
+using boost::mpi::graph_communicator;
+using namespace boost;
+
+BOOST_AUTO_TEST_CASE(graph_topology)
+{
+ boost::function_requires< IncidenceGraphConcept<graph_communicator> >();
+ boost::function_requires< AdjacencyGraphConcept<graph_communicator> >();
+ boost::function_requires< VertexListGraphConcept<graph_communicator> >();
+ boost::function_requires< EdgeListGraphConcept<graph_communicator> >();
+
+ double prob = 0.1;
+
+ boost::mpi::environment env;
+ communicator world;
+
+ // Random number generator
+ minstd_rand gen;
+
+ // Build a random graph with as many vertices as there are processes
+ typedef adjacency_list<listS, vecS, bidirectionalS> Graph;
+ sorted_erdos_renyi_iterator<minstd_rand, Graph>
+ first(gen, world.size(), prob), last;
+ Graph graph(first, last, world.size());
+
+ // Display the original graph
+ if (world.rank() == 0) {
+ std::cout << "Original, random graph:\n";
+ BGL_FORALL_VERTICES(v, graph, Graph) {
+ BGL_FORALL_OUTEDGES(v, e, graph, Graph) {
+ std::cout << source(e, graph) << " -> " << target(e, graph)
+ << std::endl;
+ }
+ }
+ }
+
+ // Create an arbitrary mapping from vertices to integers
+ typedef property_map<Graph, vertex_index_t>::type GraphVertexIndexMap;
+ std::vector<int> graph_alt_index_vec(num_vertices(graph));
+ iterator_property_map<int*, GraphVertexIndexMap>
+ graph_alt_index(&graph_alt_index_vec[0], get(vertex_index, graph));
+
+ // Rank 0 will populate the alternative index vector
+ if (world.rank() == 0) {
+ int index = 0;
+ BGL_FORALL_VERTICES(v, graph, Graph)
+ put(graph_alt_index, v, index++);
+
+ ::random_shuffle(graph_alt_index_vec.begin(), graph_alt_index_vec.end());
+ }
+ broadcast(world, graph_alt_index_vec, 0);
+
+ // Display the original graph with the remapping
+ if (world.rank() == 0) {
+ std::cout << "Original, random graph with remapped vertex numbers:\n";
+ BGL_FORALL_VERTICES(v, graph, Graph) {
+ BGL_FORALL_OUTEDGES(v, e, graph, Graph) {
+ std::cout << get(graph_alt_index, source(e, graph)) << " -> "
+ << get(graph_alt_index, target(e, graph)) << std::endl;
+ }
+ }
+ }
+
+ // Create a communicator with a topology equivalent to the graph
+ graph_communicator graph_comm(world, graph, graph_alt_index, false);
+
+ // The communicator's topology should have the same number of
+ // vertices and edges and the original graph
+ BOOST_CHECK((int)num_vertices(graph) == num_vertices(graph_comm));
+ BOOST_CHECK((int)num_edges(graph) == num_edges(graph_comm));
+
+ // Display the communicator graph
+ if (graph_comm.rank() == 0) {
+ std::cout << "Communicator graph:\n";
+ BGL_FORALL_VERTICES(v, graph_comm, graph_communicator) {
+ BGL_FORALL_OUTEDGES(v, e, graph_comm, graph_communicator) {
+ std::cout << source(e, graph_comm) << " -> " << target(e, graph_comm)
+ << std::endl;
+ }
+ }
+
+ std::cout << "Communicator graph via edges():\n";
+ BGL_FORALL_EDGES(e, graph_comm, graph_communicator)
+ std::cout << source(e, graph_comm) << " -> " << target(e, graph_comm)
+ << std::endl;
+ }
+ (graph_comm.barrier)();
+
+ // Verify the isomorphism
+ if (graph_comm.rank() == 0)
+ std::cout << "Verifying isomorphism..." << std::endl;
+ BOOST_CHECK(verify_isomorphism(graph, graph_comm, graph_alt_index));
+}
diff --git a/src/boost/libs/mpi/test/groups_test.cpp b/src/boost/libs/mpi/test/groups_test.cpp
new file mode 100644
index 00000000..ce81d182
--- /dev/null
+++ b/src/boost/libs/mpi/test/groups_test.cpp
@@ -0,0 +1,59 @@
+// Copyright (C) 2013 Andreas Hehn <hehn@phys.ethz.ch>, ETH Zurich
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of communicators created from groups.
+
+#include <boost/mpi/environment.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/group.hpp>
+#include <vector>
+#include <algorithm>
+
+#define BOOST_TEST_MODULE mpi_group_test
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+template <typename T>
+struct iota
+{
+ iota() : state(0){};
+ T operator()()
+ {
+ return state++;
+ }
+ T state;
+};
+
+void group_test(const mpi::communicator& comm)
+{
+ std::vector<int> grp_a_ranks(comm.size() / 2);
+ std::generate(grp_a_ranks.begin(),grp_a_ranks.end(),iota<int>());
+
+ mpi::group grp_a = comm.group().include(grp_a_ranks.begin(),grp_a_ranks.end());
+ mpi::group grp_b = comm.group().exclude(grp_a_ranks.begin(),grp_a_ranks.end());
+
+ mpi::communicator part_a(comm,grp_a);
+ mpi::communicator part_b(comm,grp_b);
+
+ if(part_a)
+ {
+ std::cout << "comm rank: " << comm.rank() << " -> part_a rank:" << part_a.rank() << std::endl;
+ BOOST_CHECK(part_a.rank() == comm.rank());
+ }
+ if(part_b)
+ {
+ std::cout << "comm rank: " << comm.rank() << " -> part_b rank:" << part_b.rank() << std::endl;
+ BOOST_CHECK(part_b.rank() == comm.rank() - comm.size()/2);
+ }
+}
+
+BOOST_AUTO_TEST_CASE(group)
+{
+ mpi::environment env;
+ mpi::communicator comm;
+ group_test(comm);
+}
diff --git a/src/boost/libs/mpi/test/is_mpi_op_test.cpp b/src/boost/libs/mpi/test/is_mpi_op_test.cpp
new file mode 100644
index 00000000..d023c3f7
--- /dev/null
+++ b/src/boost/libs/mpi/test/is_mpi_op_test.cpp
@@ -0,0 +1,33 @@
+// Copyright (C) 2005-2006 Douglas Gregor <doug.gregor@gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the is_mpi_op functionality.
+#include <boost/mpi/operations.hpp>
+#include <boost/mpi/environment.hpp>
+#include <boost/type_traits/is_base_and_derived.hpp>
+
+#define BOOST_TEST_MODULE mpi_is_mpi_op_test
+#include <boost/test/included/unit_test.hpp>
+
+using namespace boost::mpi;
+using namespace std;
+using boost::is_base_and_derived;
+
+BOOST_AUTO_TEST_CASE(mpi_basic_op)
+{
+ boost::mpi::environment env;
+
+ // Check each predefined MPI_Op type that we support directly.
+ BOOST_TEST((is_mpi_op<minimum<float>, float>::op() == MPI_MIN));
+ BOOST_TEST((is_mpi_op<plus<double>, double>::op() == MPI_SUM));
+ BOOST_TEST((is_mpi_op<multiplies<long>, long>::op() == MPI_PROD));
+ BOOST_TEST((is_mpi_op<logical_and<int>, int>::op() == MPI_LAND));
+ BOOST_TEST((is_mpi_op<bitwise_and<int>, int>::op() == MPI_BAND));
+ BOOST_TEST((is_mpi_op<logical_or<int>, int>::op() == MPI_LOR));
+ BOOST_TEST((is_mpi_op<bitwise_or<int>, int>::op() == MPI_BOR));
+ BOOST_TEST((is_mpi_op<logical_xor<int>, int>::op() == MPI_LXOR));
+ BOOST_TEST((is_mpi_op<bitwise_xor<int>, int>::op() == MPI_BXOR));
+}
diff --git a/src/boost/libs/mpi/test/mt_init_test.cpp b/src/boost/libs/mpi/test/mt_init_test.cpp
new file mode 100644
index 00000000..cfd0f57b
--- /dev/null
+++ b/src/boost/libs/mpi/test/mt_init_test.cpp
@@ -0,0 +1,37 @@
+// Copyright (C) 2013 Alain Miniussi <alain.miniussi@oca.eu>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// test threading::level operations
+
+#include <boost/mpi.hpp>
+#include <iostream>
+#include <sstream>
+
+#define BOOST_TEST_MODULE mpi_mt_init
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+void
+test_mt_init(std::string s)
+{
+ mpi::threading::level required = mpi::threading::level(-1);
+ std::istringstream in(s);
+ in >> required;
+ BOOST_CHECK(!in.bad());
+ BOOST_CHECK(mpi::environment::thread_level() >= mpi::threading::single);
+ BOOST_CHECK(mpi::environment::thread_level() <= mpi::threading::multiple);
+}
+
+BOOST_AUTO_TEST_CASE(mt_init)
+{
+ mpi::environment env;
+ mpi::communicator comm;
+ test_mt_init("single");
+ test_mt_init("funneled");
+ test_mt_init("serialized");
+ test_mt_init("multiple");
+}
diff --git a/src/boost/libs/mpi/test/mt_level_test.cpp b/src/boost/libs/mpi/test/mt_level_test.cpp
new file mode 100644
index 00000000..a72e9a8a
--- /dev/null
+++ b/src/boost/libs/mpi/test/mt_level_test.cpp
@@ -0,0 +1,108 @@
+// Copyright (C) 2013 Alain Miniussi <alain.miniussi@oca.eu>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// test threading::level operations
+
+#include <boost/mpi/environment.hpp>
+#include <iostream>
+#include <sstream>
+
+#define BOOST_TEST_MODULE mpi_level_test
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+void
+test_threading_level_io(mpi::threading::level orig) {
+ std::ostringstream out;
+ namespace mt = boost::mpi::threading;
+ mt::level printed = mt::level(-1);
+
+ out << orig;
+ BOOST_CHECK(out.good());
+ std::string orig_str(out.str());
+ std::cout << "orig string:" << orig_str << '\n';
+ std::istringstream in(orig_str);
+ in >> printed;
+ BOOST_CHECK(!in.bad());
+ std::cout << "orig: " << orig << ", printed: " << printed << std::endl;
+ BOOST_CHECK(orig == printed);
+}
+
+void
+test_threading_levels_io() {
+ namespace mt = boost::mpi::threading;
+ test_threading_level_io(mt::single);
+ test_threading_level_io(mt::funneled);
+ test_threading_level_io(mt::serialized);
+ test_threading_level_io(mt::multiple);
+}
+
+void
+test_threading_level_cmp() {
+ namespace mt = boost::mpi::threading;
+ BOOST_CHECK(mt::single == mt::single);
+ BOOST_CHECK(mt::funneled == mt::funneled);
+ BOOST_CHECK(mt::serialized == mt::serialized);
+ BOOST_CHECK(mt::multiple == mt::multiple);
+
+ BOOST_CHECK(mt::single != mt::funneled);
+ BOOST_CHECK(mt::single != mt::serialized);
+ BOOST_CHECK(mt::single != mt::multiple);
+
+ BOOST_CHECK(mt::funneled != mt::single);
+ BOOST_CHECK(mt::funneled != mt::serialized);
+ BOOST_CHECK(mt::funneled != mt::multiple);
+
+ BOOST_CHECK(mt::serialized != mt::single);
+ BOOST_CHECK(mt::serialized != mt::funneled);
+ BOOST_CHECK(mt::serialized != mt::multiple);
+
+ BOOST_CHECK(mt::multiple != mt::single);
+ BOOST_CHECK(mt::multiple != mt::funneled);
+ BOOST_CHECK(mt::multiple != mt::serialized);
+
+ BOOST_CHECK(mt::single < mt::funneled);
+ BOOST_CHECK(mt::funneled > mt::single);
+ BOOST_CHECK(mt::single < mt::serialized);
+ BOOST_CHECK(mt::serialized > mt::single);
+ BOOST_CHECK(mt::single < mt::multiple);
+ BOOST_CHECK(mt::multiple > mt::single);
+
+ BOOST_CHECK(mt::funneled < mt::serialized);
+ BOOST_CHECK(mt::serialized > mt::funneled);
+ BOOST_CHECK(mt::funneled < mt::multiple);
+ BOOST_CHECK(mt::multiple > mt::funneled);
+
+ BOOST_CHECK(mt::serialized < mt::multiple);
+ BOOST_CHECK(mt::multiple > mt::serialized);
+
+ BOOST_CHECK(mt::single <= mt::single);
+ BOOST_CHECK(mt::single <= mt::funneled);
+ BOOST_CHECK(mt::funneled >= mt::single);
+ BOOST_CHECK(mt::single <= mt::serialized);
+ BOOST_CHECK(mt::serialized >= mt::single);
+ BOOST_CHECK(mt::single <= mt::multiple);
+ BOOST_CHECK(mt::multiple >= mt::single);
+
+ BOOST_CHECK(mt::funneled <= mt::funneled);
+ BOOST_CHECK(mt::funneled <= mt::serialized);
+ BOOST_CHECK(mt::serialized >= mt::funneled);
+ BOOST_CHECK(mt::funneled <= mt::multiple);
+ BOOST_CHECK(mt::multiple >= mt::funneled);
+
+ BOOST_CHECK(mt::serialized <= mt::serialized);
+ BOOST_CHECK(mt::serialized <= mt::multiple);
+ BOOST_CHECK(mt::multiple >= mt::serialized);
+
+ BOOST_CHECK(mt::multiple <= mt::multiple);
+}
+
+BOOST_AUTO_TEST_CASE(mt_level)
+{
+ test_threading_levels_io();
+ test_threading_level_cmp();
+}
diff --git a/src/boost/libs/mpi/test/non_blocking_any_source.cpp b/src/boost/libs/mpi/test/non_blocking_any_source.cpp
new file mode 100644
index 00000000..dde8f92c
--- /dev/null
+++ b/src/boost/libs/mpi/test/non_blocking_any_source.cpp
@@ -0,0 +1,63 @@
+// Copyright (C) 2018 Steffen Hirschmann
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Test any_source on serialized data
+#include <vector>
+#include <iostream>
+#include <iterator>
+#include <boost/mpi.hpp>
+#include <boost/serialization/vector.hpp>
+
+#define BOOST_TEST_MODULE mpi_non_blockin_any_source
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+std::string ok(bool b) {
+ return b ? "ok" : "ko";
+}
+
+BOOST_AUTO_TEST_CASE(non_blocking_any)
+{
+ mpi::environment env;
+ mpi::communicator world;
+ int rank = world.rank();
+#if BOOST_MPI_VERSION < 3
+ if (rank == 0) {
+ std::cout << "\nExpected failure with MPI standard < 3 ("
+ << BOOST_MPI_VERSION << "." << BOOST_MPI_SUBVERSION
+ << " detected)\n\n";
+ }
+ return;
+#endif
+ if (rank == 0) {
+ std::vector<boost::mpi::request> req;
+ std::vector<std::vector<int> > data(world.size() - 1);
+ for (int i = 1; i < world.size(); ++i) {
+ req.push_back(world.irecv(mpi::any_source, 0, data[i - 1]));
+ }
+ boost::mpi::wait_all(req.begin(), req.end());
+ std::vector<bool> check(world.size()-1, false);
+ for (int i = 0; i < world.size() - 1; ++i) {
+ std::cout << "Process 0 received:" << std::endl;
+ std::copy(data[i].begin(), data[i].end(), std::ostream_iterator<int>(std::cout, " "));
+ std::cout << std::endl;
+ int idx = data[i].size();
+ BOOST_CHECK(std::equal_range(data[i].begin(), data[i].end(), idx)
+ == std::make_pair(data[i].begin(), data[i].end()));
+ check[idx-1] = true;
+ }
+ for(int i = 0; i < world.size() - 1; ++i) {
+ std::cout << "Received from " << i+1 << " is " << ok(check[i]) << '\n';
+ }
+ BOOST_CHECK(std::equal_range(check.begin(), check.end(), true)
+ == std::make_pair(check.begin(), check.end()));
+ } else {
+ std::vector<int> vec(rank, rank);
+ mpi::request req = world.isend(0, 0, vec);
+ req.wait();
+ }
+}
diff --git a/src/boost/libs/mpi/test/nonblocking_test.cpp b/src/boost/libs/mpi/test/nonblocking_test.cpp
new file mode 100644
index 00000000..8e3eb099
--- /dev/null
+++ b/src/boost/libs/mpi/test/nonblocking_test.cpp
@@ -0,0 +1,247 @@
+// Copyright (C) 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the nonblocking point-to-point operations.
+#include <boost/mpi/nonblocking.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include "gps_position.hpp"
+#include <boost/lexical_cast.hpp>
+#include <boost/serialization/string.hpp>
+#include <boost/serialization/list.hpp>
+#include <iterator>
+#include <algorithm>
+//#include "debugger.cpp"
+
+#define BOOST_TEST_MODULE mpi_non_blockin_test
+#include <boost/test/included/unit_test.hpp>
+
+using boost::mpi::communicator;
+using boost::mpi::request;
+using boost::mpi::status;
+
+enum method_kind {
+ mk_wait_any, mk_test_any, mk_wait_all, mk_wait_all_keep,
+ mk_test_all, mk_test_all_keep, mk_wait_some, mk_wait_some_keep,
+ mk_test_some, mk_test_some_keep,
+ mk_test_size
+};
+
+static const char* method_kind_names[mk_test_size] = {
+ "wait_any",
+ "test_any",
+ "wait_all",
+ "wait_all (keep results)",
+ "test_all",
+ "test_all (keep results)",
+ "wait_some",
+ "wait_some (keep results)",
+ "test_some",
+ "test_some (keep results)"
+};
+
+
+template<typename T>
+void
+nonblocking_tests( const communicator& comm, const T* values, int num_values,
+ const char* kind, bool composite)
+{
+ nonblocking_test(comm, values, num_values, kind, mk_wait_any);
+ nonblocking_test(comm, values, num_values, kind, mk_test_any);
+ //wait_for_debugger(comm);
+ nonblocking_test(comm, values, num_values, kind, mk_wait_all);
+ nonblocking_test(comm, values, num_values, kind, mk_wait_all_keep);
+ if (!composite) {
+ nonblocking_test(comm, values, num_values, kind, mk_test_all);
+ nonblocking_test(comm, values, num_values, kind, mk_test_all_keep);
+ }
+ nonblocking_test(comm, values, num_values, kind, mk_wait_some);
+ nonblocking_test(comm, values, num_values, kind, mk_wait_some_keep);
+ nonblocking_test(comm, values, num_values, kind, mk_test_some);
+ nonblocking_test(comm, values, num_values, kind, mk_test_some_keep);
+}
+
+template<typename T>
+void
+nonblocking_test(const communicator& comm, const T* values, int num_values,
+ const char* kind, method_kind method)
+{
+ using boost::mpi::wait_any;
+ using boost::mpi::test_any;
+ using boost::mpi::wait_all;
+ using boost::mpi::test_all;
+ using boost::mpi::wait_some;
+ using boost::mpi::test_some;
+
+ int next = (comm.rank() + 1) % comm.size();
+ int prev = (comm.rank() + comm.size() - 1) % comm.size();
+
+ if (comm.rank() == 0) {
+ std::cout << "Testing " << method_kind_names[method]
+ << " with " << kind << "...";
+ std::cout.flush();
+ }
+
+ typedef std::pair<status, std::vector<request>::iterator>
+ status_iterator_pair;
+
+ T incoming_value;
+ std::vector<T> incoming_values(num_values);
+ std::vector<request> reqs;
+ // Send/receive the first value
+ reqs.push_back(comm.isend(next, 0, values[0]));
+ reqs.push_back(comm.irecv(prev, 0, incoming_value));
+
+ if (method != mk_wait_any && method != mk_test_any) {
+#ifndef LAM_MPI
+ // We've run into problems here (with 0-length messages) with
+ // LAM/MPI on Mac OS X and x86-86 Linux. Will investigate
+ // further at a later time, but the problem only seems to occur
+ // when using shared memory, not TCP.
+
+ // Send/receive an empty message
+ reqs.push_back(comm.isend(next, 1));
+ reqs.push_back(comm.irecv(prev, 1));
+#endif
+
+ // Send/receive an array
+ reqs.push_back(comm.isend(next, 2, values, num_values));
+ reqs.push_back(comm.irecv(prev, 2, &incoming_values.front(), num_values));
+ }
+
+ switch (method) {
+ case mk_wait_any:
+ if (wait_any(reqs.begin(), reqs.end()).second == reqs.begin())
+ reqs[1].wait();
+ else
+ reqs[0].wait();
+ break;
+
+ case mk_test_any:
+ {
+ boost::optional<status_iterator_pair> result;
+ do {
+ result = test_any(reqs.begin(), reqs.end());
+ } while (!result);
+ if (result->second == reqs.begin())
+ reqs[1].wait();
+ else
+ reqs[0].wait();
+ break;
+ }
+
+ case mk_wait_all:
+ wait_all(reqs.begin(), reqs.end());
+ break;
+
+ case mk_wait_all_keep:
+ {
+ std::vector<status> stats;
+ wait_all(reqs.begin(), reqs.end(), std::back_inserter(stats));
+ }
+ break;
+
+ case mk_test_all:
+ while (!test_all(reqs.begin(), reqs.end())) { /* Busy wait */ }
+ break;
+
+ case mk_test_all_keep:
+ {
+ std::vector<status> stats;
+ while (!test_all(reqs.begin(), reqs.end(), std::back_inserter(stats)))
+ /* Busy wait */;
+ }
+ break;
+
+ case mk_wait_some:
+ {
+ std::vector<request>::iterator pos = reqs.end();
+ do {
+ pos = wait_some(reqs.begin(), pos);
+ } while (pos != reqs.begin());
+ }
+ break;
+
+ case mk_wait_some_keep:
+ {
+ std::vector<status> stats;
+ std::vector<request>::iterator pos = reqs.end();
+ do {
+ pos = wait_some(reqs.begin(), pos, std::back_inserter(stats)).second;
+ } while (pos != reqs.begin());
+ }
+ break;
+
+ case mk_test_some:
+ {
+ std::vector<request>::iterator pos = reqs.end();
+ do {
+ pos = test_some(reqs.begin(), pos);
+ } while (pos != reqs.begin());
+ }
+ break;
+
+ case mk_test_some_keep:
+ {
+ std::vector<status> stats;
+ std::vector<request>::iterator pos = reqs.end();
+ do {
+ pos = test_some(reqs.begin(), pos, std::back_inserter(stats)).second;
+ } while (pos != reqs.begin());
+ }
+ break;
+
+ default:
+ BOOST_CHECK(false);
+ }
+
+ if (comm.rank() == 0) {
+ bool okay = true;
+
+ if (!((incoming_value == values[0])))
+ okay = false;
+
+ if (method != mk_wait_any && method != mk_test_any
+ && !std::equal(incoming_values.begin(), incoming_values.end(),
+ values))
+ okay = false;
+
+ if (okay)
+ std::cout << "OK." << std::endl;
+ else
+ std::cerr << "ERROR!" << std::endl;
+ }
+
+ BOOST_CHECK(incoming_value == values[0]);
+
+ if (method != mk_wait_any && method != mk_test_any)
+ BOOST_CHECK(std::equal(incoming_values.begin(), incoming_values.end(),
+ values));
+}
+
+BOOST_AUTO_TEST_CASE(nonblocking)
+{
+ boost::mpi::environment env;
+ communicator comm;
+
+ int int_array[3] = {17, 42, 256};
+ nonblocking_tests(comm, int_array, 3, "integers", false);
+
+ gps_position gps_array[2] = {
+ gps_position(17, 42, .06),
+ gps_position(42, 17, .06)
+ };
+ nonblocking_tests(comm, gps_array, 2, "gps positions", false);
+
+ std::string string_array[2] = { "Hello", "World" };
+ nonblocking_tests(comm, string_array, 2, "strings", true);
+
+ std::list<std::string> lst_of_strings;
+ for (int i = 0; i < comm.size(); ++i)
+ lst_of_strings.push_back(boost::lexical_cast<std::string>(i));
+
+ nonblocking_tests(comm, &lst_of_strings, 1, "list of strings", true);
+}
diff --git a/src/boost/libs/mpi/test/pointer_test.cpp b/src/boost/libs/mpi/test/pointer_test.cpp
new file mode 100644
index 00000000..65428e75
--- /dev/null
+++ b/src/boost/libs/mpi/test/pointer_test.cpp
@@ -0,0 +1,42 @@
+// Copyright (C) 2005, 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+
+// a test of pointer serialization
+#include <boost/mpi.hpp>
+#include <boost/serialization/shared_ptr.hpp>
+
+#define BOOST_TEST_MODULE mpi_pointer
+#include <boost/test/included/unit_test.hpp>
+
+class A
+{
+ public:
+ int i;
+ template<class Archive>
+ void serialize(Archive & ar, const unsigned int version)
+ {
+ ar & i;
+ }
+};
+
+BOOST_AUTO_TEST_CASE(pointer)
+{
+ boost::mpi::environment env;
+ boost::mpi::communicator world;
+
+ if (world.rank() == 0) {
+ boost::shared_ptr<A> p(new A);
+ p->i = 42;
+ world.send(1, 0, p);
+ } else if (world.rank() == 1) {
+ boost::shared_ptr<A> p;
+ world.recv(0, 0, p);
+ std::cout << p->i << std::endl;
+ BOOST_CHECK(p->i==42);
+ }
+}
+
diff --git a/src/boost/libs/mpi/test/python/all_gather_test.py b/src/boost/libs/mpi/test/python/all_gather_test.py
new file mode 100644
index 00000000..824dc892
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/all_gather_test.py
@@ -0,0 +1,25 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Test all_gather() collective.
+
+import boost.parallel.mpi as mpi
+from generators import *
+
+def all_gather_test(comm, generator, kind):
+ if comm.rank == 0: print ("Gathering %s..." % (kind,)),
+ my_value = generator(comm.rank)
+ result = mpi.all_gather(comm, my_value)
+ for p in range(0, comm.size):
+ assert result[p] == generator(p)
+ if comm.rank == 0: print "OK."
+
+ return
+
+all_gather_test(mpi.world, int_generator, "integers")
+all_gather_test(mpi.world, gps_generator, "GPS positions")
+all_gather_test(mpi.world, string_generator, "strings")
+all_gather_test(mpi.world, string_list_generator, "list of strings")
diff --git a/src/boost/libs/mpi/test/python/all_reduce_test.py b/src/boost/libs/mpi/test/python/all_reduce_test.py
new file mode 100644
index 00000000..c3285e65
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/all_reduce_test.py
@@ -0,0 +1,29 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Test all_reduce() collective.
+
+import boost.parallel.mpi as mpi
+from generators import *
+
+def all_reduce_test(comm, generator, kind, op, op_kind):
+ if comm.rank == 0:
+ print ("Reducing to %s of %s..." % (op_kind, kind)),
+ my_value = generator(comm.rank)
+ result = mpi.all_reduce(comm, my_value, op)
+ expected_result = generator(0);
+ for p in range(1, comm.size):
+ expected_result = op(expected_result, generator(p))
+
+ assert result == expected_result
+ if comm.rank == 0:
+ print "OK."
+ return
+
+all_reduce_test(mpi.world, int_generator, "integers", lambda x,y:x + y, "sum")
+all_reduce_test(mpi.world, int_generator, "integers", lambda x,y:x * y, "product")
+all_reduce_test(mpi.world, string_generator, "strings", lambda x,y:x + y, "concatenation")
+all_reduce_test(mpi.world, string_list_generator, "list of strings", lambda x,y:x + y, "concatenation")
diff --git a/src/boost/libs/mpi/test/python/all_to_all_test.py b/src/boost/libs/mpi/test/python/all_to_all_test.py
new file mode 100644
index 00000000..b149bf0d
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/all_to_all_test.py
@@ -0,0 +1,30 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Test all_to_all() collective.
+
+import boost.parallel.mpi as mpi
+from generators import *
+
+def all_to_all_test(comm, generator, kind):
+ if comm.rank == 0:
+ print ("All-to-all transmission of %s..." % (kind,)),
+
+ values = list()
+ for p in range(0, comm.size):
+ values.append(generator(p))
+ result = mpi.all_to_all(comm, values)
+
+ for p in range(0, comm.size):
+ assert result[p] == generator(comm.rank)
+
+ if comm.rank == 0: print "OK."
+ return
+
+all_to_all_test(mpi.world, int_generator, "integers")
+all_to_all_test(mpi.world, gps_generator, "GPS positions")
+all_to_all_test(mpi.world, string_generator, "strings")
+all_to_all_test(mpi.world, string_list_generator, "list of strings")
diff --git a/src/boost/libs/mpi/test/python/broadcast_test.py b/src/boost/libs/mpi/test/python/broadcast_test.py
new file mode 100644
index 00000000..dbd53d1b
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/broadcast_test.py
@@ -0,0 +1,29 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Test broadcast() collective.
+
+import boost.parallel.mpi as mpi
+
+def broadcast_test(comm, value, kind, root):
+ if comm.rank == root:
+ print ("Broadcasting %s from root %d..." % (kind, root)),
+
+ got_value = mpi.broadcast(comm, value, root)
+ assert got_value == value
+ if comm.rank == root:
+ print "OK."
+ return
+
+broadcast_test(mpi.world, 17, 'integer', 0)
+broadcast_test(mpi.world, 17, 'integer', 1)
+broadcast_test(mpi.world, 'Hello, World!', 'string', 0)
+broadcast_test(mpi.world, 'Hello, World!', 'string', 1)
+broadcast_test(mpi.world, ['Hello', 'MPI', 'Python', 'World'],
+ 'list of strings', 0)
+broadcast_test(mpi.world, ['Hello', 'MPI', 'Python', 'World'],
+ 'list of strings', 1)
+
diff --git a/src/boost/libs/mpi/test/python/gather_test.py b/src/boost/libs/mpi/test/python/gather_test.py
new file mode 100644
index 00000000..d56b3a41
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/gather_test.py
@@ -0,0 +1,32 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Test gather() collective.
+
+import boost.parallel.mpi as mpi
+from generators import *
+
+def gather_test(comm, generator, kind, root):
+ if comm.rank == root:
+ print ("Gathering %s to root %d..." % (kind, root)),
+ my_value = generator(comm.rank)
+ result = mpi.gather(comm, my_value, root)
+ if comm.rank == root:
+ for p in range(0, comm.size):
+ assert result[p] == generator(p)
+ print "OK."
+ else:
+ assert result == None
+ return
+
+gather_test(mpi.world, int_generator, "integers", 0)
+gather_test(mpi.world, int_generator, "integers", 1)
+gather_test(mpi.world, gps_generator, "GPS positions", 0)
+gather_test(mpi.world, gps_generator, "GPS positions", 1)
+gather_test(mpi.world, string_generator, "strings", 0)
+gather_test(mpi.world, string_generator, "strings", 1)
+gather_test(mpi.world, string_list_generator, "list of strings", 0)
+gather_test(mpi.world, string_list_generator, "list of strings", 1)
diff --git a/src/boost/libs/mpi/test/python/generators.py b/src/boost/libs/mpi/test/python/generators.py
new file mode 100644
index 00000000..8cdd3e05
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/generators.py
@@ -0,0 +1,23 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Value generators used in the Boost.MPI Python regression tests
+def int_generator(p):
+ return 17 + p
+
+def gps_generator(p):
+ return (39 + p, 16, 20.2799)
+
+def string_generator(p):
+ result = "%d rosebud" % p;
+ if p != 1: result = result + 's'
+ return result
+
+def string_list_generator(p):
+ result = list()
+ for i in range(0,p):
+ result.append(str(i))
+ return result
diff --git a/src/boost/libs/mpi/test/python/nonblocking_test.py b/src/boost/libs/mpi/test/python/nonblocking_test.py
new file mode 100644
index 00000000..73b451c5
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/nonblocking_test.py
@@ -0,0 +1,131 @@
+# (C) Copyright 2007
+# Andreas Kloeckner <inform -at- tiker.net>
+#
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+#
+# Authors: Andreas Kloeckner
+
+
+
+
+import boost.mpi as mpi
+import random
+import sys
+
+MAX_GENERATIONS = 20
+TAG_DEBUG = 0
+TAG_DATA = 1
+TAG_TERMINATE = 2
+TAG_PROGRESS_REPORT = 3
+
+
+
+
+class TagGroupListener:
+ """Class to help listen for only a given set of tags.
+
+ This is contrived: Typicallly you could just listen for
+ mpi.any_tag and filter."""
+ def __init__(self, comm, tags):
+ self.tags = tags
+ self.comm = comm
+ self.active_requests = {}
+
+ def wait(self):
+ for tag in self.tags:
+ if tag not in self.active_requests:
+ self.active_requests[tag] = self.comm.irecv(tag=tag)
+ requests = mpi.RequestList(self.active_requests.values())
+ data, status, index = mpi.wait_any(requests)
+ del self.active_requests[status.tag]
+ return status, data
+
+ def cancel(self):
+ for r in self.active_requests.itervalues():
+ r.cancel()
+ #r.wait()
+ self.active_requests = {}
+
+
+
+def rank0():
+ sent_histories = (mpi.size-1)*15
+ print "sending %d packets on their way" % sent_histories
+ send_reqs = mpi.RequestList()
+ for i in range(sent_histories):
+ dest = random.randrange(1, mpi.size)
+ send_reqs.append(mpi.world.isend(dest, TAG_DATA, []))
+
+ mpi.wait_all(send_reqs)
+
+ completed_histories = []
+ progress_reports = {}
+ dead_kids = []
+
+ tgl = TagGroupListener(mpi.world,
+ [TAG_DATA, TAG_DEBUG, TAG_PROGRESS_REPORT, TAG_TERMINATE])
+
+ def is_complete():
+ for i in progress_reports.values():
+ if i != sent_histories:
+ return False
+ return len(dead_kids) == mpi.size-1
+
+ while True:
+ status, data = tgl.wait()
+
+ if status.tag == TAG_DATA:
+ #print "received completed history %s from %d" % (data, status.source)
+ completed_histories.append(data)
+ if len(completed_histories) == sent_histories:
+ print "all histories received, exiting"
+ for rank in range(1, mpi.size):
+ mpi.world.send(rank, TAG_TERMINATE, None)
+ elif status.tag == TAG_PROGRESS_REPORT:
+ progress_reports[len(data)] = progress_reports.get(len(data), 0) + 1
+ elif status.tag == TAG_DEBUG:
+ print "[DBG %d] %s" % (status.source, data)
+ elif status.tag == TAG_TERMINATE:
+ dead_kids.append(status.source)
+ else:
+ print "unexpected tag %d from %d" % (status.tag, status.source)
+
+ if is_complete():
+ break
+
+ print "OK"
+
+def comm_rank():
+ while True:
+ data, status = mpi.world.recv(return_status=True)
+ if status.tag == TAG_DATA:
+ mpi.world.send(0, TAG_PROGRESS_REPORT, data)
+ data.append(mpi.rank)
+ if len(data) >= MAX_GENERATIONS:
+ dest = 0
+ else:
+ dest = random.randrange(1, mpi.size)
+ mpi.world.send(dest, TAG_DATA, data)
+ elif status.tag == TAG_TERMINATE:
+ from time import sleep
+ mpi.world.send(0, TAG_TERMINATE, 0)
+ break
+ else:
+ print "[DIRECTDBG %d] unexpected tag %d from %d" % (mpi.rank, status.tag, status.source)
+
+
+def main():
+ # this program sends around messages consisting of lists of visited nodes
+ # randomly. After MAX_GENERATIONS, they are returned to rank 0.
+
+ if mpi.rank == 0:
+ rank0()
+ else:
+ comm_rank()
+
+
+
+if __name__ == "__main__":
+ main()
diff --git a/src/boost/libs/mpi/test/python/reduce_test.py b/src/boost/libs/mpi/test/python/reduce_test.py
new file mode 100644
index 00000000..65f09f5b
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/reduce_test.py
@@ -0,0 +1,31 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Test reduce() collective.
+
+import boost.parallel.mpi as mpi
+from generators import *
+
+def reduce_test(comm, generator, kind, op, op_kind, root):
+ if comm.rank == root:
+ print ("Reducing to %s of %s at root %d..." % (op_kind, kind, root)),
+ my_value = generator(comm.rank)
+ result = mpi.reduce(comm, my_value, op, root)
+ if comm.rank == root:
+ expected_result = generator(0);
+ for p in range(1, comm.size):
+ expected_result = op(expected_result, generator(p))
+ assert result == expected_result
+ print "OK."
+ else:
+ assert result == None
+ return
+
+reduce_test(mpi.world, int_generator, "integers", lambda x,y:x + y, "sum", 0)
+reduce_test(mpi.world, int_generator, "integers", lambda x,y:x * y, "product", 1)
+reduce_test(mpi.world, int_generator, "integers", min, "minimum", 0)
+reduce_test(mpi.world, string_generator, "strings", lambda x,y:x + y, "concatenation", 0)
+reduce_test(mpi.world, string_list_generator, "list of strings", lambda x,y:x + y, "concatenation", 0)
diff --git a/src/boost/libs/mpi/test/python/ring_test.py b/src/boost/libs/mpi/test/python/ring_test.py
new file mode 100644
index 00000000..3c8b5b92
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/ring_test.py
@@ -0,0 +1,42 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Test basic communication.
+
+import boost.parallel.mpi as mpi
+
+def ring_test(comm, value, kind, root):
+ next_peer = (comm.rank + 1) % comm.size;
+ prior_peer = (comm.rank + comm.size - 1) % comm.size;
+
+ if comm.rank == root:
+ print ("Passing %s around a ring from root %d..." % (kind, root)),
+ comm.send(next_peer, 0, value)
+ (other_value, stat) = comm.recv(return_status = True)
+ assert value == other_value
+ assert stat.source == prior_peer
+ assert stat.tag == 0
+ else:
+ msg = comm.probe()
+ other_value = comm.recv(msg.source, msg.tag)
+ assert value == other_value
+ comm.send(next_peer, 0, other_value)
+
+ comm.barrier()
+ if comm.rank == root:
+ print "OK"
+ pass
+
+if mpi.world.size < 2:
+ print "ERROR: ring_test.py must be executed with more than one process"
+ mpi.world.abort(-1);
+
+ring_test(mpi.world, 17, 'integers', 0)
+ring_test(mpi.world, 17, 'integers', 1)
+ring_test(mpi.world, 'Hello, World!', 'string', 0)
+ring_test(mpi.world, 'Hello, World!', 'string', 1)
+ring_test(mpi.world, ['Hello', 'MPI', 'Python', 'World'], 'list of strings', 0)
+ring_test(mpi.world, ['Hello', 'MPI', 'Python', 'World'], 'list of strings', 1)
diff --git a/src/boost/libs/mpi/test/python/scan_test.py b/src/boost/libs/mpi/test/python/scan_test.py
new file mode 100644
index 00000000..193a6a44
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/scan_test.py
@@ -0,0 +1,29 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Test scan() collective.
+
+import boost.parallel.mpi as mpi
+from generators import *
+
+def scan_test(comm, generator, kind, op, op_kind):
+ if comm.rank == 0:
+ print ("Prefix reduction to %s of %s..." % (op_kind, kind)),
+ my_value = generator(comm.rank)
+ result = mpi.scan(comm, my_value, op)
+ expected_result = generator(0);
+ for p in range(1, comm.rank+1):
+ expected_result = op(expected_result, generator(p))
+
+ assert result == expected_result
+ if comm.rank == 0:
+ print "OK."
+ return
+
+scan_test(mpi.world, int_generator, "integers", lambda x,y:x + y, "sum")
+scan_test(mpi.world, int_generator, "integers", lambda x,y:x * y, "product")
+scan_test(mpi.world, string_generator, "strings", lambda x,y:x + y, "concatenation")
+scan_test(mpi.world, string_list_generator, "list of strings", lambda x,y:x + y, "concatenation")
diff --git a/src/boost/libs/mpi/test/python/scatter_test.py b/src/boost/libs/mpi/test/python/scatter_test.py
new file mode 100644
index 00000000..e0bad967
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/scatter_test.py
@@ -0,0 +1,36 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Test scatter() collective.
+
+import boost.parallel.mpi as mpi
+from generators import *
+
+def scatter_test(comm, generator, kind, root):
+ if comm.rank == root:
+ print ("Scattering %s from root %d..." % (kind, root)),
+
+ if comm.rank == root:
+ values = list()
+ for p in range(0, comm.size):
+ values.append(generator(p))
+ result = mpi.scatter(comm, values, root = root)
+ else:
+ result = mpi.scatter(comm, root = root);
+
+ assert result == generator(comm.rank)
+
+ if comm.rank == root: print "OK."
+ return
+
+scatter_test(mpi.world, int_generator, "integers", 0)
+scatter_test(mpi.world, int_generator, "integers", 1)
+scatter_test(mpi.world, gps_generator, "GPS positions", 0)
+scatter_test(mpi.world, gps_generator, "GPS positions", 1)
+scatter_test(mpi.world, string_generator, "strings", 0)
+scatter_test(mpi.world, string_generator, "strings", 1)
+scatter_test(mpi.world, string_list_generator, "list of strings", 0)
+scatter_test(mpi.world, string_list_generator, "list of strings", 1)
diff --git a/src/boost/libs/mpi/test/python/skeleton_content_test.cpp b/src/boost/libs/mpi/test/python/skeleton_content_test.cpp
new file mode 100644
index 00000000..7ccae00e
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/skeleton_content_test.cpp
@@ -0,0 +1,37 @@
+// (C) Copyright 2006 Douglas Gregor <doug.gregor -at- gmail.com>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// Authors: Douglas Gregor
+
+#include <boost/parallel/mpi/python.hpp>
+#include <boost/python.hpp>
+#include <boost/serialization/list.hpp>
+using namespace boost::python;
+
+template<typename T>
+boost::python::list list_to_python(const std::list<T>& value) {
+ boost::python::list result;
+ for (typename std::list<T>::const_iterator i = value.begin();
+ i != value.end(); ++i)
+ result.append(*i);
+ return result;
+}
+
+BOOST_PYTHON_MODULE(skeleton_content)
+{
+ using boost::python::arg;
+
+ class_<std::list<int> >("list_int")
+ .def("push_back", &std::list<int>::push_back, arg("value"))
+ .def("pop_back", &std::list<int>::pop_back)
+ .def("reverse", &std::list<int>::reverse)
+ .def(boost::python::self == boost::python::self)
+ .def(boost::python::self != boost::python::self)
+ .add_property("size", &std::list<int>::size)
+ .def("to_python", &list_to_python<int>);
+
+ boost::parallel::mpi::python::register_skeleton_and_content<std::list<int> >();
+}
diff --git a/src/boost/libs/mpi/test/python/skeleton_content_test.py b/src/boost/libs/mpi/test/python/skeleton_content_test.py
new file mode 100644
index 00000000..1dfde3cf
--- /dev/null
+++ b/src/boost/libs/mpi/test/python/skeleton_content_test.py
@@ -0,0 +1,75 @@
+# Copyright (C) 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+# Use, modification and distribution is subject to the Boost Software
+# License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+# http://www.boost.org/LICENSE_1_0.txt)
+
+# Test skeleton/content
+
+import boost.parallel.mpi as mpi
+import skeleton_content
+
+def test_skeleton_and_content(comm, root, manual_broadcast = True):
+ assert manual_broadcast
+
+ # Setup data
+ list_size = comm.size + 7
+ original_list = skeleton_content.list_int()
+ for i in range(0,list_size):
+ original_list.push_back(i)
+
+ if comm.rank == root:
+ # Broadcast skeleton
+ print ("Broadcasting integer list skeleton from root %d..." % (root)),
+ if manual_broadcast:
+ for p in range(0,comm.size):
+ if p != comm.rank:
+ comm.send(p, 0, value = mpi.skeleton(original_list))
+ print "OK."
+
+ # Broadcast content
+ print ("Broadcasting integer list content from root %d..." % (root)),
+ if manual_broadcast:
+ for p in range(0,comm.size):
+ if p != comm.rank:
+ comm.send(p, 0, value = mpi.get_content(original_list))
+
+ print "OK."
+
+ # Broadcast reversed content
+ original_list.reverse()
+ print ("Broadcasting reversed integer list content from root %d..." % (root)),
+ if manual_broadcast:
+ for p in range(0,comm.size):
+ if p != comm.rank:
+ comm.send(p, 0, value = mpi.get_content(original_list))
+
+ print "OK."
+ else:
+ # Allocate some useless data, to try to get the addresses of
+ # the underlying lists used later to be different across
+ # processors.
+ junk_list = skeleton_content.list_int()
+ for i in range(0,comm.rank * 3 + 1):
+ junk_list.push_back(i)
+
+ # Receive the skeleton of the list
+ if manual_broadcast:
+ transferred_list_skeleton = comm.recv(root, 0)
+ assert transferred_list_skeleton.object.size == list_size
+
+ # Receive the content and check it
+ transferred_list = transferred_list_skeleton.object
+ if manual_broadcast:
+ comm.recv(root, 0, mpi.get_content(transferred_list))
+ assert transferred_list == original_list
+
+ # Receive the content (again) and check it
+ original_list.reverse()
+ if manual_broadcast:
+ comm.recv(root, 0, mpi.get_content(transferred_list))
+ assert transferred_list == original_list
+
+
+test_skeleton_and_content(mpi.world, 0)
+test_skeleton_and_content(mpi.world, 1)
diff --git a/src/boost/libs/mpi/test/reduce_test.cpp b/src/boost/libs/mpi/test/reduce_test.cpp
new file mode 100644
index 00000000..44532cd5
--- /dev/null
+++ b/src/boost/libs/mpi/test/reduce_test.cpp
@@ -0,0 +1,237 @@
+// Copyright (C) 2005, 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the reduce() collective.
+#include <boost/mpi/collectives/reduce.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include <algorithm>
+#include <boost/serialization/string.hpp>
+#include <boost/iterator/counting_iterator.hpp>
+#include <boost/lexical_cast.hpp>
+#include <numeric>
+
+#define BOOST_TEST_MODULE mpi_reduce_test
+#include <boost/test/included/unit_test.hpp>
+
+using boost::mpi::communicator;
+
+// A simple point class that we can build, add, compare, and
+// serialize.
+struct point
+{
+ point() : x(0), y(0), z(0) { }
+ point(int x, int y, int z) : x(x), y(y), z(z) { }
+
+ int x;
+ int y;
+ int z;
+
+ private:
+ template<typename Archiver>
+ void serialize(Archiver& ar, unsigned int /*version*/)
+ {
+ ar & x & y & z;
+ }
+
+ friend class boost::serialization::access;
+};
+
+std::ostream& operator<<(std::ostream& out, const point& p)
+{
+ return out << p.x << ' ' << p.y << ' ' << p.z;
+}
+
+bool operator==(const point& p1, const point& p2)
+{
+ return p1.x == p2.x && p1.y == p2.y && p1.z == p2.z;
+}
+
+bool operator!=(const point& p1, const point& p2)
+{
+ return !(p1 == p2);
+}
+
+point operator+(const point& p1, const point& p2)
+{
+ return point(p1.x + p2.x, p1.y + p2.y, p1.z + p2.z);
+}
+
+namespace boost { namespace mpi {
+
+ template <>
+ struct is_mpi_datatype<point> : public mpl::true_ { };
+
+} } // end namespace boost::mpi
+
+template<typename Generator, typename Op>
+void
+reduce_test(const communicator& comm, Generator generator,
+ const char* type_kind, Op op, const char* op_kind,
+ typename Generator::result_type init,
+ int root = -1)
+{
+ typedef typename Generator::result_type value_type;
+ value_type value = generator(comm.rank());
+
+ if (root == -1) {
+ for (root = 0; root < comm.size(); ++root)
+ reduce_test(comm, generator, type_kind, op, op_kind, init, root);
+ } else {
+ using boost::mpi::reduce;
+
+ if (comm.rank() == root) {
+ std::cout << "Reducing to " << op_kind << " of " << type_kind
+ << " at root " << root << "...";
+ std::cout.flush();
+
+ value_type result_value;
+ reduce(comm, value, result_value, op, root);
+
+ // Compute expected result
+ std::vector<value_type> generated_values;
+ for (int p = 0; p < comm.size(); ++p)
+ generated_values.push_back(generator(p));
+ value_type expected_result = std::accumulate(generated_values.begin(),
+ generated_values.end(),
+ init, op);
+ BOOST_CHECK(result_value == expected_result);
+ if (result_value == expected_result)
+ std::cout << "OK." << std::endl;
+ } else {
+ reduce(comm, value, op, root);
+ }
+ }
+
+ (comm.barrier)();
+}
+
+// Generates integers to test with reduce()
+struct int_generator
+{
+ typedef int result_type;
+
+ int_generator(int base = 1) : base(base) { }
+
+ int operator()(int p) const { return base + p; }
+
+ private:
+ int base;
+};
+
+// Generate points to test with reduce()
+struct point_generator
+{
+ typedef point result_type;
+
+ point_generator(point origin) : origin(origin) { }
+
+ point operator()(int p) const
+ {
+ return point(origin.x + 1, origin.y + 1, origin.z + 1);
+ }
+
+ private:
+ point origin;
+};
+
+struct string_generator
+{
+ typedef std::string result_type;
+
+ std::string operator()(int p) const
+ {
+ std::string result = boost::lexical_cast<std::string>(p);
+ result += " rosebud";
+ if (p != 1) result += 's';
+ return result;
+ }
+};
+
+struct secret_int_bit_and
+{
+ int operator()(int x, int y) const { return x & y; }
+};
+
+struct wrapped_int
+{
+ wrapped_int() : value(0) { }
+ explicit wrapped_int(int value) : value(value) { }
+
+ template<typename Archive>
+ void serialize(Archive& ar, unsigned int /* version */)
+ {
+ ar & value;
+ }
+
+ int value;
+};
+
+wrapped_int operator+(const wrapped_int& x, const wrapped_int& y)
+{
+ return wrapped_int(x.value + y.value);
+}
+
+bool operator==(const wrapped_int& x, const wrapped_int& y)
+{
+ return x.value == y.value;
+}
+
+// Generates wrapped_its to test with reduce()
+struct wrapped_int_generator
+{
+ typedef wrapped_int result_type;
+
+ wrapped_int_generator(int base = 1) : base(base) { }
+
+ wrapped_int operator()(int p) const { return wrapped_int(base + p); }
+
+ private:
+ int base;
+};
+
+namespace boost { namespace mpi {
+
+// Make std::plus<wrapped_int> commutative.
+template<>
+struct is_commutative<std::plus<wrapped_int>, wrapped_int>
+ : mpl::true_ { };
+
+} } // end namespace boost::mpi
+
+BOOST_AUTO_TEST_CASE(reduce_check)
+{
+ using namespace boost::mpi;
+ environment env;
+
+ communicator comm;
+
+ // Built-in MPI datatypes with built-in MPI operations
+ reduce_test(comm, int_generator(), "integers", std::plus<int>(), "sum", 0);
+ reduce_test(comm, int_generator(), "integers", std::multiplies<int>(),
+ "product", 1);
+ reduce_test(comm, int_generator(), "integers", maximum<int>(),
+ "maximum", 0);
+ reduce_test(comm, int_generator(), "integers", minimum<int>(),
+ "minimum", 2);
+
+ // User-defined MPI datatypes with operations that have the
+ // same name as built-in operations.
+ reduce_test(comm, point_generator(point(0,0,0)), "points",
+ std::plus<point>(), "sum", point());
+
+ // Built-in MPI datatypes with user-defined operations
+ reduce_test(comm, int_generator(17), "integers", secret_int_bit_and(),
+ "bitwise and", -1);
+
+ // Arbitrary types with user-defined, commutative operations.
+ reduce_test(comm, wrapped_int_generator(17), "wrapped integers",
+ std::plus<wrapped_int>(), "sum", wrapped_int(0));
+
+ // Arbitrary types with (non-commutative) user-defined operations
+ reduce_test(comm, string_generator(), "strings",
+ std::plus<std::string>(), "concatenation", std::string());
+}
diff --git a/src/boost/libs/mpi/test/ring_test.cpp b/src/boost/libs/mpi/test/ring_test.cpp
new file mode 100644
index 00000000..573d3d38
--- /dev/null
+++ b/src/boost/libs/mpi/test/ring_test.cpp
@@ -0,0 +1,124 @@
+// Copyright (C) 2005, 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the communicator that passes data around a ring and
+// verifies that the same data makes it all the way. Should test all
+// of the various kinds of data that can be sent (primitive types, POD
+// types, serializable objects, etc.)
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include <algorithm>
+#include "gps_position.hpp"
+#include <boost/serialization/string.hpp>
+#include <boost/serialization/list.hpp>
+//#include "debugger.cpp"
+
+#define BOOST_TEST_MODULE mpi_reduce_ring
+#include <boost/test/included/unit_test.hpp>
+
+using boost::mpi::communicator;
+using boost::mpi::status;
+
+template<typename T>
+void
+ring_test(const communicator& comm, const T& pass_value, const char* kind,
+ int root = 0)
+{
+ T transferred_value;
+
+ int rank = comm.rank();
+ int size = comm.size();
+
+ if (rank == root) {
+ std::cout << "Passing " << kind << " around a ring from root " << root
+ << "...";
+ comm.send((rank + 1) % size, 0, pass_value);
+ comm.recv((rank + size - 1) % size, 0, transferred_value);
+ BOOST_CHECK(transferred_value == pass_value);
+ if (transferred_value == pass_value) std::cout << " OK." << std::endl;
+ } else {
+ comm.recv((rank + size - 1) % size, 0, transferred_value);
+ BOOST_CHECK(transferred_value == pass_value);
+ comm.send((rank + 1) % size, 0, transferred_value);
+ }
+
+ (comm.barrier)();
+}
+
+
+template<typename T>
+void
+ring_array_test(const communicator& comm, const T* pass_values,
+ int n, const char* kind, int root = 0)
+{
+ T* transferred_values = new T[n];
+ int rank = comm.rank();
+ int size = comm.size();
+
+ if (rank == root) {
+
+ std::cout << "Passing " << kind << " array around a ring from root "
+ << root << "...";
+ comm.send((rank + 1) % size, 0, pass_values, n);
+ comm.recv((rank + size - 1) % size, 0, transferred_values, n);
+ bool okay = std::equal(pass_values, pass_values + n,
+ transferred_values);
+ BOOST_CHECK(okay);
+ if (okay) std::cout << " OK." << std::endl;
+ } else {
+ status stat = comm.probe(boost::mpi::any_source, 0);
+ boost::optional<int> num_values = stat.template count<T>();
+ if (boost::mpi::is_mpi_datatype<T>())
+ BOOST_CHECK(num_values && *num_values == n);
+ else
+ BOOST_CHECK(!num_values || *num_values == n);
+ comm.recv(stat.source(), 0, transferred_values, n);
+ BOOST_CHECK(std::equal(pass_values, pass_values + n,
+ transferred_values));
+ comm.send((rank + 1) % size, 0, transferred_values, n);
+ }
+ (comm.barrier)();
+ delete [] transferred_values;
+}
+
+enum color_t {red, green, blue};
+BOOST_IS_MPI_DATATYPE(color_t)
+
+BOOST_AUTO_TEST_CASE(ring)
+{
+ boost::mpi::environment env;
+ communicator comm;
+
+ BOOST_TEST_REQUIRE(comm.size() > 1);
+
+ // Check transfer of individual objects
+ ring_test(comm, 17, "integers", 0);
+ ring_test(comm, 17, "integers", 1);
+ ring_test(comm, red, "enums", 1);
+ ring_test(comm, red, "enums", 1);
+ ring_test(comm, gps_position(39,16,20.2799), "GPS positions", 0);
+ ring_test(comm, gps_position(26,25,30.0), "GPS positions", 1);
+ ring_test(comm, std::string("Rosie"), "string", 0);
+
+ std::list<std::string> strings;
+ strings.push_back("Hello");
+ strings.push_back("MPI");
+ strings.push_back("World");
+ ring_test(comm, strings, "list of strings", 1);
+
+ // Check transfer of arrays
+ int int_array[2] = { 17, 42 };
+ ring_array_test(comm, int_array, 2, "integer", 1);
+ gps_position gps_position_array[2] = {
+ gps_position(39,16,20.2799),
+ gps_position(26,25,30.0)
+ };
+ ring_array_test(comm, gps_position_array, 2, "GPS position", 1);
+
+ std::string string_array[3] = { "Hello", "MPI", "World" };
+ ring_array_test(comm, string_array, 3, "string", 0);
+ ring_array_test(comm, string_array, 3, "string", 1);
+}
diff --git a/src/boost/libs/mpi/test/scan_test.cpp b/src/boost/libs/mpi/test/scan_test.cpp
new file mode 100644
index 00000000..579ce8de
--- /dev/null
+++ b/src/boost/libs/mpi/test/scan_test.cpp
@@ -0,0 +1,228 @@
+// Copyright (C) 2005, 2006 Douglas Gregor <doug.gregor -at- gmail.com>.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the scan() collective.
+#include <boost/mpi/collectives/scan.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include <algorithm>
+#include <boost/serialization/string.hpp>
+#include <boost/iterator/counting_iterator.hpp>
+#include <boost/lexical_cast.hpp>
+#include <numeric>
+
+#define BOOST_TEST_MODULE mpi_scan_test
+#include <boost/test/included/unit_test.hpp>
+
+
+using boost::mpi::communicator;
+
+// A simple point class that we can build, add, compare, and
+// serialize.
+struct point
+{
+ point() : x(0), y(0), z(0) { }
+ point(int x, int y, int z) : x(x), y(y), z(z) { }
+
+ int x;
+ int y;
+ int z;
+
+ private:
+ template<typename Archiver>
+ void serialize(Archiver& ar, unsigned int /*version*/)
+ {
+ ar & x & y & z;
+ }
+
+ friend class boost::serialization::access;
+};
+
+std::ostream& operator<<(std::ostream& out, const point& p)
+{
+ return out << p.x << ' ' << p.y << ' ' << p.z;
+}
+
+bool operator==(const point& p1, const point& p2)
+{
+ return p1.x == p2.x && p1.y == p2.y && p1.z == p2.z;
+}
+
+bool operator!=(const point& p1, const point& p2)
+{
+ return !(p1 == p2);
+}
+
+point operator+(const point& p1, const point& p2)
+{
+ return point(p1.x + p2.x, p1.y + p2.y, p1.z + p2.z);
+}
+
+namespace boost { namespace mpi {
+
+ template <>
+ struct is_mpi_datatype<point> : public mpl::true_ { };
+
+} } // end namespace boost::mpi
+
+template<typename Generator, typename Op>
+void
+scan_test(const communicator& comm, Generator generator,
+ const char* type_kind, Op op, const char* op_kind)
+{
+ typedef typename Generator::result_type value_type;
+ value_type value = generator(comm.rank());
+ using boost::mpi::scan;
+
+ if (comm.rank() == 0) {
+ std::cout << "Prefix reducing to " << op_kind << " of " << type_kind
+ << "...";
+ std::cout.flush();
+ }
+
+ value_type result_value;
+ scan(comm, value, result_value, op);
+ value_type scan_result = scan(comm, value, op);
+ BOOST_CHECK(scan_result == result_value);
+
+ // Compute expected result
+ std::vector<value_type> generated_values;
+ for (int p = 0; p < comm.size(); ++p)
+ generated_values.push_back(generator(p));
+ std::vector<value_type> expected_results(comm.size());
+ std::partial_sum(generated_values.begin(), generated_values.end(),
+ expected_results.begin(), op);
+ BOOST_CHECK(result_value == expected_results[comm.rank()]);
+ if (comm.rank() == 0) std::cout << "Done." << std::endl;
+
+ (comm.barrier)();
+}
+
+// Generates integers to test with scan()
+struct int_generator
+{
+ typedef int result_type;
+
+ int_generator(int base = 1) : base(base) { }
+
+ int operator()(int p) const { return base + p; }
+
+ private:
+ int base;
+};
+
+// Generate points to test with scan()
+struct point_generator
+{
+ typedef point result_type;
+
+ point_generator(point origin) : origin(origin) { }
+
+ point operator()(int p) const
+ {
+ return point(origin.x + 1, origin.y + 1, origin.z + 1);
+ }
+
+ private:
+ point origin;
+};
+
+struct string_generator
+{
+ typedef std::string result_type;
+
+ std::string operator()(int p) const
+ {
+ std::string result = boost::lexical_cast<std::string>(p);
+ result += " rosebud";
+ if (p != 1) result += 's';
+ return result;
+ }
+};
+
+struct secret_int_bit_and
+{
+ int operator()(int x, int y) const { return x & y; }
+};
+
+struct wrapped_int
+{
+ wrapped_int() : value(0) { }
+ explicit wrapped_int(int value) : value(value) { }
+
+ template<typename Archive>
+ void serialize(Archive& ar, unsigned int /* version */)
+ {
+ ar & value;
+ }
+
+ int value;
+};
+
+wrapped_int operator+(const wrapped_int& x, const wrapped_int& y)
+{
+ return wrapped_int(x.value + y.value);
+}
+
+bool operator==(const wrapped_int& x, const wrapped_int& y)
+{
+ return x.value == y.value;
+}
+
+// Generates wrapped_its to test with scan()
+struct wrapped_int_generator
+{
+ typedef wrapped_int result_type;
+
+ wrapped_int_generator(int base = 1) : base(base) { }
+
+ wrapped_int operator()(int p) const { return wrapped_int(base + p); }
+
+ private:
+ int base;
+};
+
+namespace boost { namespace mpi {
+
+// Make std::plus<wrapped_int> commutative.
+template<>
+struct is_commutative<std::plus<wrapped_int>, wrapped_int>
+ : mpl::true_ { };
+
+} } // end namespace boost::mpi
+
+BOOST_AUTO_TEST_CASE(scan_check)
+{
+ using namespace boost::mpi;
+ environment env;
+ communicator comm;
+
+ // Built-in MPI datatypes with built-in MPI operations
+ scan_test(comm, int_generator(), "integers", std::plus<int>(), "sum");
+ scan_test(comm, int_generator(), "integers", std::multiplies<int>(),
+ "product");
+ scan_test(comm, int_generator(), "integers", maximum<int>(),
+ "maximum");
+ scan_test(comm, int_generator(), "integers", minimum<int>(),
+ "minimum");
+
+ // User-defined MPI datatypes with operations that have the
+ // same name as built-in operations.
+ scan_test(comm, point_generator(point(0,0,0)), "points",
+ std::plus<point>(), "sum");
+
+ // Built-in MPI datatypes with user-defined operations
+ scan_test(comm, int_generator(17), "integers", secret_int_bit_and(),
+ "bitwise and");
+
+ // Arbitrary types with user-defined, commutative operations.
+ scan_test(comm, wrapped_int_generator(17), "wrapped integers",
+ std::plus<wrapped_int>(), "sum");
+
+ // Arbitrary types with (non-commutative) user-defined operations
+ scan_test(comm, string_generator(), "strings",
+ std::plus<std::string>(), "concatenation");
+}
diff --git a/src/boost/libs/mpi/test/scatter_test.cpp b/src/boost/libs/mpi/test/scatter_test.cpp
new file mode 100644
index 00000000..f116bb34
--- /dev/null
+++ b/src/boost/libs/mpi/test/scatter_test.cpp
@@ -0,0 +1,233 @@
+// Copyright (C) 2005, 2006 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the scatter() and scatterv() collectives.
+#include <iterator>
+#include <boost/mpi/collectives/scatter.hpp>
+#include <boost/mpi/collectives/scatterv.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include "gps_position.hpp"
+#include <boost/serialization/string.hpp>
+#include <boost/serialization/list.hpp>
+#include <boost/iterator/counting_iterator.hpp>
+#include <boost/lexical_cast.hpp>
+
+#define BOOST_TEST_MODULE mpi_scatter
+#include <boost/test/included/unit_test.hpp>
+
+using namespace boost::mpi;
+
+template<typename Generator>
+void
+scatter_test(const communicator& comm, Generator generator,
+ const char* kind, int root = -1)
+{
+ typedef typename Generator::result_type value_type;
+
+ if (root == -1) {
+ for (root = 0; root < comm.size(); ++root)
+ scatter_test(comm, generator, kind, root);
+ } else {
+ using boost::mpi::scatter;
+
+ value_type value;
+
+ if (comm.rank() == root) {
+ std::vector<value_type> values;
+
+ for (int p = 0; p < comm.size(); ++p)
+ values.push_back(generator(p));
+
+ std::cout << "Scattering " << kind << " from root "
+ << root << "..." << std::endl;
+
+ scatter(comm, values, value, root);
+ } else {
+ scatter(comm, value, root);
+ }
+
+ BOOST_CHECK(value == generator(comm.rank()));
+ }
+
+ comm.barrier();
+}
+
+
+//
+// Generators to test with scatter/scatterv
+//
+struct int_generator
+{
+ typedef int result_type;
+
+ int operator()(int p) const { return 17 + p; }
+};
+
+struct gps_generator
+{
+ typedef gps_position result_type;
+
+ gps_position operator()(int p) const
+ {
+ return gps_position(39 + p, 16, 20.2799);
+ }
+};
+
+struct string_generator
+{
+ typedef std::string result_type;
+
+ std::string operator()(int p) const
+ {
+ std::string result = boost::lexical_cast<std::string>(p);
+ result += " rosebud";
+ if (p != 1) result += 's';
+ return result;
+ }
+};
+
+struct string_list_generator
+{
+ typedef std::list<std::string> result_type;
+
+ std::list<std::string> operator()(int p) const
+ {
+ std::list<std::string> result;
+ for (int i = 0; i <= p; ++i) {
+ std::string value = boost::lexical_cast<std::string>(i);
+ result.push_back(value);
+ }
+ return result;
+ }
+};
+
+std::ostream&
+operator<<(std::ostream& out, std::list<std::string> const& l) {
+ out << '[';
+ std::copy(l.begin(), l.end(), std::ostream_iterator<std::string>(out, " "));
+ out << ']';
+ return out;
+}
+
+template<typename Generator>
+void
+scatterv_test(const communicator& comm, Generator generator,
+ const char* kind, int root = -1)
+{
+ typedef typename Generator::result_type value_type;
+
+ if (root == -1) {
+ for (root = 0; root < comm.size(); ++root)
+ scatterv_test(comm, generator, kind, root);
+ } else {
+ using boost::mpi::scatterv;
+
+ int mysize = comm.rank() + 1;
+ std::vector<value_type> myvalues(mysize);
+
+ if (comm.rank() == root) {
+ std::vector<value_type> values;
+ std::vector<int> sizes(comm.size());
+
+ // process p will receive p+1 identical generator(p) elements
+ for (int p = 0; p < comm.size(); ++p) {
+ for (int i = 0; i < p+1; ++i)
+ values.push_back(generator(p));
+ sizes[p] = p + 1;
+ }
+
+ std::cout << "Scatteringv " << kind << " from root "
+ << root << "..." << std::endl;
+ assert(mysize == sizes[comm.rank()]);
+ scatterv(comm, values, sizes, &(myvalues[0]), root);
+ } else {
+ scatterv(comm, &(myvalues[0]), mysize, root);
+ }
+
+ for (int i = 0; i < mysize; ++i)
+ BOOST_CHECK(myvalues[i] == generator(comm.rank()));
+ }
+
+ comm.barrier();
+}
+
+template<typename Generator>
+void
+scatterd_test(const communicator& comm, Generator generator,
+ const char* kind, int root = -1)
+{
+ typedef typename Generator::result_type value_type;
+
+ if (root == -1) {
+ for (root = 0; root < comm.size(); ++root)
+ scatterv_test(comm, generator, kind, root);
+ } else {
+ using boost::mpi::scatterv;
+
+ int mysize = comm.rank() + 1;
+ std::vector<value_type> myvalues(mysize);
+
+ if (comm.rank() == root) {
+ std::vector<value_type> values;
+ std::vector<int> sizes(comm.size());
+ std::vector<int> displs(comm.size());
+ value_type noise = generator(comm.size()+1);
+ // process p will receive a payload of p+1 identical generator(p) elements
+ // root will insert pseudo random pading between each payload.
+ int shift = 0; // the current position of next payload in source array
+ for (int p = 0; p < comm.size(); ++p) {
+ int size = p+1;
+ int pad = p % 3;
+ // padding
+ for (int i = 0; i < pad; ++i) {
+ values.push_back(noise);
+ }
+ // payload
+ for (int i = 0; i < size; ++i)
+ values.push_back(generator(p));
+ shift += pad;
+ displs[p] = shift;
+ sizes[p] = size;
+ shift += size;
+ }
+
+ std::cout << "Scatteringv " << kind << " from root "
+ << root << "..." << std::endl;
+ assert(mysize == sizes[comm.rank()]);
+ scatterv(comm, values, sizes, displs, &(myvalues[0]), mysize, root);
+ } else {
+ scatterv(comm, &(myvalues[0]), mysize, root);
+ }
+
+ for (int i = 0; i < mysize; ++i)
+ BOOST_CHECK(myvalues[i] == generator(comm.rank()));
+ }
+
+ comm.barrier();
+}
+
+
+BOOST_AUTO_TEST_CASE(simple_scatter)
+{
+ environment env;
+ communicator comm;
+
+ scatter_test(comm, int_generator(), "integers");
+ scatter_test(comm, gps_generator(), "GPS positions");
+ scatter_test(comm, string_generator(), "string");
+ scatter_test(comm, string_list_generator(), "list of strings");
+
+ scatterv_test(comm, int_generator(), "integers");
+ scatterv_test(comm, gps_generator(), "GPS positions");
+ scatterv_test(comm, string_generator(), "string");
+ scatterv_test(comm, string_list_generator(), "list of strings");
+
+ scatterd_test(comm, int_generator(), "integers");
+ scatterd_test(comm, gps_generator(), "GPS positions");
+ scatterd_test(comm, string_generator(), "string");
+ scatterd_test(comm, string_list_generator(), "list of strings");
+}
diff --git a/src/boost/libs/mpi/test/sendrecv_test.cpp b/src/boost/libs/mpi/test/sendrecv_test.cpp
new file mode 100644
index 00000000..801b2629
--- /dev/null
+++ b/src/boost/libs/mpi/test/sendrecv_test.cpp
@@ -0,0 +1,62 @@
+// Copyright Alain Miniussi 20014.
+// Distributed under the Boost Software License, Version 1.0.
+// (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the sendrecv() operation.
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include <vector>
+#include <algorithm>
+#include <boost/serialization/string.hpp>
+#include <boost/iterator/counting_iterator.hpp>
+#include <boost/lexical_cast.hpp>
+#include <numeric>
+
+#define BOOST_TEST_MODULE mpi_sendrecv
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+struct blob {
+ blob(int i) : value(i) {}
+ int value;
+ template<class Archive>
+ void serialize(Archive& s, const unsigned int version) {
+ s & value;
+ }
+};
+
+std::ostream& operator<<(std::ostream& out, blob const& b) {
+ out << "blob(" << b.value << ")";
+ return out;
+}
+
+bool operator==(blob const& b1, blob const& b2) {
+ return b1.value == b2.value;
+}
+
+template<typename T>
+void test_sendrecv(mpi::communicator& com) {
+ int const wrank = com.rank();
+ int const wsize = com.size();
+ int const wnext((wrank + 1) % wsize);
+ int const wprev((wrank + wsize - 1) % wsize);
+ T recv(-1);
+ com.sendrecv(wnext, 1, T(wrank), wprev, 1, recv);
+ for(int r = 0; r < wsize; ++r) {
+ com.barrier();
+ if (r == wrank) {
+ std::cout << "rank " << wrank << " received " << recv << " from " << wprev << '\n';
+ }
+ }
+ BOOST_CHECK(recv == T(wprev));
+}
+
+BOOST_AUTO_TEST_CASE(sendrecv)
+{
+ mpi::environment env;
+ mpi::communicator world;
+ test_sendrecv<int>(world);
+ test_sendrecv<blob>(world);
+}
diff --git a/src/boost/libs/mpi/test/sendrecv_vector.cpp b/src/boost/libs/mpi/test/sendrecv_vector.cpp
new file mode 100644
index 00000000..dc139285
--- /dev/null
+++ b/src/boost/libs/mpi/test/sendrecv_vector.cpp
@@ -0,0 +1,95 @@
+// Author: K. Noel Belcourt <kbelco -at- sandia.gov>
+
+// Distributed under the Boost Software License, Version 1.0.
+// (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+#if defined(__cplusplus) && (201103L <= __cplusplus)
+
+#include <array>
+#include <cassert>
+#include <vector>
+
+#include "boost/mpi/environment.hpp"
+#include "boost/mpi/communicator.hpp"
+
+using std::array;
+using std::vector;
+
+namespace mpi = boost::mpi;
+
+struct blob : array<int, 9>, array<double, 3>, array<char, 8> {
+};
+
+namespace boost {
+ namespace mpi {
+
+ template <>
+ struct is_mpi_datatype<blob> : mpl::true_ {
+ };
+
+ template <>
+ MPI_Datatype get_mpi_datatype<blob>(const blob& b) {
+ array<unsigned long, 3> block_lengths{
+ { 9, 3, 8 }
+ };
+
+ array<MPI_Aint, 3> displacements{
+ { 0, 40, 64 }
+ };
+
+ array<MPI_Datatype, 3> datatypes{
+ { MPI_INT, MPI_DOUBLE, MPI_CHAR }
+ };
+
+ MPI_Datatype blob_type;
+ MPI_Type_create_struct(
+ block_lengths.size()
+ , reinterpret_cast<int*>(block_lengths.data())
+ , displacements.data()
+ , datatypes.data()
+ , &blob_type);
+
+ MPI_Type_commit(&blob_type);
+ return blob_type;
+ }
+
+ } // namespace mpi
+} // namespace boost
+
+#endif // defined(__cplusplus)
+
+
+int main(int argc, char* argv[]) {
+#if defined(__cplusplus) && (201103L <= __cplusplus)
+
+ mpi::environment env(argc, argv);
+ mpi::communicator world;
+
+ vector<blob> data;
+
+ if (world.rank() == 0) {
+ int size = 10000000;
+ data.resize(size);
+ // initialize data at vector ends
+ blob& b1= data[0];
+ array<int, 9>& i = b1;
+ i[0] = -1;
+ blob& b2= data[size-1];
+ array<int, 9>& d = b2;
+ d[2] = -17;
+ world.send(1, 0, data);
+ } else {
+ world.recv(0, 0, data);
+ // check data at vector ends
+ blob& b1 = data[0];
+ array<int, 9>& i = b1;
+ assert(i[0] == -1);
+ // blob& b2 = data[data.size()-1];
+ // array<int, 9>& d = b2;
+ // assert(d[2] == -17);
+ }
+#endif // defined(__cplusplus)
+
+ return 0;
+}
diff --git a/src/boost/libs/mpi/test/skeleton_content_test.cpp b/src/boost/libs/mpi/test/skeleton_content_test.cpp
new file mode 100644
index 00000000..6a42fd5e
--- /dev/null
+++ b/src/boost/libs/mpi/test/skeleton_content_test.cpp
@@ -0,0 +1,200 @@
+// Copyright 2005 Douglas Gregor.
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// A test of the communicator that transmits skeletons and
+// content for data types.
+#include <boost/mpi/communicator.hpp>
+#include <boost/mpi/environment.hpp>
+#include <boost/serialization/list.hpp>
+#include <boost/mpi/skeleton_and_content.hpp>
+#include <boost/mpi/nonblocking.hpp>
+#include <algorithm>
+#include <boost/iterator/counting_iterator.hpp>
+#include <boost/mpi/collectives/broadcast.hpp>
+
+#define BOOST_TEST_MODULE mpi_skeleton_content
+#include <boost/test/included/unit_test.hpp>
+
+using boost::mpi::communicator;
+
+using boost::mpi::packed_skeleton_iarchive;
+using boost::mpi::packed_skeleton_oarchive;
+
+void
+test_skeleton_and_content(const communicator& comm, int root,
+ bool manual_broadcast)
+{
+ using boost::mpi::skeleton;
+ using boost::mpi::content;
+ using boost::mpi::get_content;
+ using boost::make_counting_iterator;
+ using boost::mpi::broadcast;
+
+ int list_size = comm.size() + 7;
+ if (comm.rank() == root) {
+ // Fill in the seed data
+ std::list<int> original_list;
+ for (int i = 0; i < list_size; ++i)
+ original_list.push_back(i);
+
+ std::cout << "Broadcasting integer list skeleton from root " << root
+ << "...";
+ if (manual_broadcast) {
+ // Broadcast the skeleton (manually)
+ for (int p = 0; p < comm.size(); ++p)
+ if (p != root) comm.send(p, 0, skeleton(original_list));
+ } else {
+ broadcast(comm, skeleton(original_list), root);
+ }
+ std::cout << "OK." << std::endl;
+
+ // Broadcast the content (manually)
+ std::cout << "Broadcasting integer list content from root " << root
+ << "...";
+ {
+ content c = get_content(original_list);
+ for (int p = 0; p < comm.size(); ++p)
+ if (p != root) comm.send(p, 1, c);
+ }
+ std::cout << "OK." << std::endl;
+
+ // Reverse the list, broadcast the content again
+ std::reverse(original_list.begin(), original_list.end());
+ std::cout << "Broadcasting reversed integer list content from root "
+ << root << "...";
+ {
+ content c = get_content(original_list);
+ for (int p = 0; p < comm.size(); ++p)
+ if (p != root) comm.send(p, 2, c);
+ }
+ std::cout << "OK." << std::endl;
+ } else {
+ // Allocate some useless data, to try to get the addresses of the
+ // list<int>'s used later to be different across processes.
+ std::list<int> junk_list(comm.rank() * 3 + 1, 17);
+
+ // Receive the skeleton to build up the transferred list
+ std::list<int> transferred_list;
+ if (manual_broadcast) {
+ comm.recv(root, 0, skeleton(transferred_list));
+ } else {
+ broadcast(comm, skeleton(transferred_list), root);
+ }
+ BOOST_CHECK((int)transferred_list.size() == list_size);
+
+ // Receive the content and check it
+ comm.recv(root, 1, get_content(transferred_list));
+ BOOST_CHECK(std::equal(make_counting_iterator(0),
+ make_counting_iterator(list_size),
+ transferred_list.begin()));
+
+ // Receive the reversed content and check it
+ comm.recv(root, 2, get_content(transferred_list));
+ BOOST_CHECK(std::equal(make_counting_iterator(0),
+ make_counting_iterator(list_size),
+ transferred_list.rbegin()));
+ }
+
+ (comm.barrier)();
+}
+
+void
+test_skeleton_and_content_nonblocking(const communicator& comm, int root)
+{
+ using boost::mpi::skeleton;
+ using boost::mpi::content;
+ using boost::mpi::get_content;
+ using boost::make_counting_iterator;
+ using boost::mpi::broadcast;
+ using boost::mpi::request;
+ using boost::mpi::wait_all;
+
+ int list_size = comm.size() + 7;
+ if (comm.rank() == root) {
+ // Fill in the seed data
+ std::list<int> original_list;
+ for (int i = 0; i < list_size; ++i)
+ original_list.push_back(i);
+
+ std::cout << "Non-blocking broadcast of integer list skeleton from root " << root
+ << "...";
+
+ // Broadcast the skeleton (manually)
+ {
+ std::vector<request> reqs;
+ for (int p = 0; p < comm.size(); ++p)
+ if (p != root)
+ reqs.push_back(comm.isend(p, 0, skeleton(original_list)));
+ wait_all(reqs.begin(), reqs.end());
+ }
+ std::cout << "OK." << std::endl;
+
+ // Broadcast the content (manually)
+ std::cout << "Non-blocking broadcast of integer list content from root " << root
+ << "...";
+ {
+ content c = get_content(original_list);
+ std::vector<request> reqs;
+ for (int p = 0; p < comm.size(); ++p)
+ if (p != root) reqs.push_back(comm.isend(p, 1, c));
+ wait_all(reqs.begin(), reqs.end());
+ }
+ std::cout << "OK." << std::endl;
+
+ // Reverse the list, broadcast the content again
+ std::reverse(original_list.begin(), original_list.end());
+ std::cout << "Non-blocking broadcast of reversed integer list content from root "
+ << root << "...";
+ {
+ std::vector<request> reqs;
+ content c = get_content(original_list);
+ for (int p = 0; p < comm.size(); ++p)
+ if (p != root) reqs.push_back(comm.isend(p, 2, c));
+ wait_all(reqs.begin(), reqs.end());
+ }
+ std::cout << "OK." << std::endl;
+ } else {
+ // Allocate some useless data, to try to get the addresses of the
+ // list<int>'s used later to be different across processes.
+ std::list<int> junk_list(comm.rank() * 3 + 1, 17);
+
+ // Receive the skeleton to build up the transferred list
+ std::list<int> transferred_list;
+ request req = comm.irecv(root, 0, skeleton(transferred_list));
+ req.wait();
+ BOOST_CHECK((int)transferred_list.size() == list_size);
+
+ // Receive the content and check it
+ req = comm.irecv(root, 1, get_content(transferred_list));
+ req.wait();
+ BOOST_CHECK(std::equal(make_counting_iterator(0),
+ make_counting_iterator(list_size),
+ transferred_list.begin()));
+
+ // Receive the reversed content and check it
+ req = comm.irecv(root, 2, get_content(transferred_list));
+ req.wait();
+ BOOST_CHECK(std::equal(make_counting_iterator(0),
+ make_counting_iterator(list_size),
+ transferred_list.rbegin()));
+ }
+
+ (comm.barrier)();
+}
+
+BOOST_AUTO_TEST_CASE(sendrecv)
+{
+ boost::mpi::environment env;
+ communicator comm;
+ BOOST_TEST_REQUIRE(comm.size() > 1);
+
+ test_skeleton_and_content(comm, 0, true);
+ test_skeleton_and_content(comm, 0, false);
+ test_skeleton_and_content(comm, 1, true);
+ test_skeleton_and_content(comm, 1, false);
+ test_skeleton_and_content_nonblocking(comm, 0);
+ test_skeleton_and_content_nonblocking(comm, 1);
+}
diff --git a/src/boost/libs/mpi/test/version_test.cpp b/src/boost/libs/mpi/test/version_test.cpp
new file mode 100644
index 00000000..0895b74f
--- /dev/null
+++ b/src/boost/libs/mpi/test/version_test.cpp
@@ -0,0 +1,60 @@
+// Copyright (C) 2013 Alain Miniussi <alain.miniussi@oca.eu>
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+// test mpi version
+
+#include <boost/mpi/environment.hpp>
+#include <boost/mpi/communicator.hpp>
+#include <iostream>
+
+#define BOOST_TEST_MODULE mpi_version
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+void
+test_version(mpi::communicator const& comm) {
+#if defined(MPI_VERSION)
+ int mpi_version = MPI_VERSION;
+ int mpi_subversion = MPI_SUBVERSION;
+#else
+ int mpi_version = 0;
+ int mpi_subversion = 0;
+#endif
+
+ std::pair<int,int> version = mpi::environment::version();
+ if (comm.rank() == 0) {
+ std::cout << "MPI Version: " << version.first << ',' << version.second << '\n';
+ }
+ BOOST_CHECK(version.first == mpi_version);
+ BOOST_CHECK(version.second == mpi_subversion);
+}
+
+std::string
+yesno(bool b) {
+ return b ? std::string("yes") : std::string("no");
+}
+
+void
+report_features(mpi::communicator const& comm) {
+ if (comm.rank() == 0) {
+ std::cout << "Assuming working MPI_Improbe:" <<
+#if defined(BOOST_MPI_USE_IMPROBE)
+ "yes" << '\n';
+#else
+ "no" << '\n';
+#endif
+ }
+}
+
+BOOST_AUTO_TEST_CASE(version)
+{
+ mpi::environment env;
+ mpi::communicator world;
+
+ test_version(world);
+ report_features(world);
+}
diff --git a/src/boost/libs/mpi/test/wait_all_vector_test.cpp b/src/boost/libs/mpi/test/wait_all_vector_test.cpp
new file mode 100644
index 00000000..0e4c800b
--- /dev/null
+++ b/src/boost/libs/mpi/test/wait_all_vector_test.cpp
@@ -0,0 +1,46 @@
+// Copyright (C) 2017 Alain Miniussi & Vincent Chabannes
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+#include <string>
+#include <iostream>
+#include <sstream>
+#include <vector>
+
+#include <boost/mpi.hpp>
+#include <boost/mpi/nonblocking.hpp>
+#include <boost/serialization/string.hpp>
+
+#define BOOST_TEST_MODULE mpi_wait_any
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+BOOST_AUTO_TEST_CASE(wait_any)
+{
+ mpi::environment env;
+ mpi::communicator comm;
+
+ int rank = comm.rank();
+ int const sz = 10;
+ std::vector<int> data;
+ std::vector< mpi::request> reqs;
+ if ( rank == 0 ) {
+ for ( int i=0; i<sz; ++i ) {
+ data.push_back( i );
+ }
+ reqs.push_back( comm.isend(1, 0, data) );
+ } else if ( rank == 1 ) {
+ reqs.push_back( comm.irecv(0, 0, data) );
+ }
+ mpi::wait_all( reqs.begin(), reqs.end() );
+
+ if ( rank == 1 ) {
+ BOOST_CHECK(data.size() == sz);
+ for ( int i=0; i<sz; ++i ) {
+ BOOST_CHECK(data[i] == i);
+ }
+ }
+}
diff --git a/src/boost/libs/mpi/test/wait_any_test.cpp b/src/boost/libs/mpi/test/wait_any_test.cpp
new file mode 100644
index 00000000..e1f4c556
--- /dev/null
+++ b/src/boost/libs/mpi/test/wait_any_test.cpp
@@ -0,0 +1,64 @@
+// Copyright (C) 2017 Alain Miniussi & Steffen Hirschmann
+
+// Use, modification and distribution is subject to the Boost Software
+// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
+// http://www.boost.org/LICENSE_1_0.txt)
+
+#include <string>
+#include <iostream>
+#include <sstream>
+#include <vector>
+#include <set>
+
+#include <boost/mpi.hpp>
+#include <boost/mpi/nonblocking.hpp>
+#include <boost/serialization/string.hpp>
+
+#define BOOST_TEST_MODULE mpi_wait_any
+#include <boost/test/included/unit_test.hpp>
+
+namespace mpi = boost::mpi;
+
+BOOST_AUTO_TEST_CASE(wait_any)
+{
+ mpi::environment env;
+ mpi::communicator world;
+
+ std::vector<std::string> ss(world.size());
+ typedef std::vector<mpi::request> requests;
+ requests rreqs;
+
+ std::set<int> pending_senders;
+ for (int i = 0; i < world.size(); ++i) {
+ rreqs.push_back(world.irecv(i, i, ss[i]));
+ pending_senders.insert(i);
+ }
+
+ std::ostringstream fmt;
+ std::string msg = "Hello, World! this is ";
+ fmt << msg << world.rank();
+
+ requests sreqs;
+ for (int i = 0; i < world.size(); ++i) {
+ sreqs.push_back(world.isend(i, world.rank(), fmt.str()));
+ }
+
+ for (int i = 0; i < world.size(); ++i) {
+ std::pair<mpi::status, requests::iterator> completed = mpi::wait_any(rreqs.begin(), rreqs.end());
+ std::ostringstream out;
+ out << "Proc " << world.rank() << " got message from " << completed.first.source() << '\n';
+ std::cout << out.str();
+ }
+
+ for (int i = 0; i < world.size(); ++i) {
+ std::ostringstream fmt;
+ fmt << msg << i;
+ std::vector<std::string>::iterator found = std::find(ss.begin(), ss.end(), fmt.str());
+ BOOST_CHECK(found != ss.end());
+ fmt.str("");
+ fmt << "Proc " << world.rank() << " Got msg from " << i << '\n';
+ std::cout << fmt.str();
+ }
+
+ mpi::wait_all(sreqs.begin(), sreqs.end());
+}