3
0
mirror of https://github.com/triqs/dft_tools synced 2025-01-12 22:18:23 +01:00

[mpi] vector, test and draft of doc

This commit is contained in:
Hartmut Hafermann 2014-07-16 17:36:46 +02:00 committed by Olivier Parcollet
parent 38cfef4e9f
commit ebbb2f0b25
6 changed files with 272 additions and 3 deletions

View File

@ -0,0 +1,5 @@
# Doxygen sources
#set_property(GLOBAL APPEND PROPERTY DOXYGEN_SOURCES ${TRIQS_SOURCE_DIR}/triqs/mpi/mpi.hpp)
all_tests()

View File

@ -0,0 +1,152 @@
MPI
===============
.. warning::
Library of beta quality.
More functionality may be added in the future.
Introduction
--------------
The purpose of the MPi library is to provide a simplified, C++-style API to the MPI routines for standard types
(those for which an MPI type exists) and for composite higher-level objects, in particular the TRIQS arrays and Green's functions.
The communication routines in the C API of the MPI library have require several parameters, such as the ``reduce`` operation:
.. code-block:: c
int MPI_Reduce(void *sendbuf, void *recvbuf, int count,
MPI_Datatype datatype, MPI_Op op, int root, MPI_Comm comm)
In principle, all parameters except for the communicator and id of the root process can be determined from the variable or object to be transmitted.
In most cases, we use ``MPI_COMM_WORLD`` as the communicator, take the id 0 for the root process and use ``MPI_SUM`` as the operation.
This allows us to write
.. code-block:: c
int a = 5;
triqs::mpi::reduce(a);
Such an interface is simpler to use and much less error prone. For higher-level objects, such as vectors or higher-dimensional arrays, the simplifcation is even more significant. Take the scatter and gather operations as examples:
.. code-block:: c
int MPI_Scatter(void *sendbuf, int sendcount, MPI_Datatype sendtype,
void *recvbuf, int recvcount, MPI_Datatype recvtype, int root,
MPI_Comm comm)
.. code-block:: c
int MPI_Gather(void *sendbuf, int sendcount, MPI_Datatype sendtype,
void *recvbuf, int recvcount, MPI_Datatype recvtype, int root,
MPI_Comm comm)
In order to scatter a (contiguos) multidimensional array across all nodes, apply some operations to it and gather it back on the master one requires several lines of relatively complex code.
The leading dimension of the array needs to be sliced, slice length and adress of the first element of each slice have to be computed and finally the MPI C API function has to be called.
This can be packaged in the library once and for all.
Using the library these operations look as follows:
.. code-block:: c
triqs::arrays::array<int, 3> A(8, 8, 8); // a three-dimensional array
triqs::mpi::scatter(A);
//do something with the corresponding part of A on each node
triqs::mpi::gather(A);
All index computations are encapsulated in the triqs::mpi library calls.
In principle, the Boost.MPI library provides a similar interface for basic types and standard library containers. Transmission of the data however requires serialization and doubles the required memory. This poses a severe limitation when large amounts of data are to be transmitted.
In this library, we employ metaprogramming techniques for type deduction as well as a lazy mechanism to avoid unecessary copyies of data.
MPI reference
----------------
In this document, we describe the use of the TRIQS MPI library. For more information on MPI, see, e.g., the `open MPI web pages <http://www.open-mpi.org>`_ or consult the MPI reference manual. For more information on Boost.MPI, refer to the `Boost library documentation <http://www.boost.org>`_.
Supported functions and types
------------------------------
Currently, the TRIQS MPI library supports the following operations::
reduce
allreduce
broadcast
scatter
gather
allgather
These routines have the same meaning as their corresponding MPI analogues.
They work for all 'basic' types, i.e. types for which a native MPI-type exists. These are::
int
long
unsigned long
double
float
std::complex<double>
We also support ``std::vector<T>`` for ``T`` being a basic type, as well as the types provided by the TRIQS ``array`` and TRIQS ``gf`` libraries.
In addition, the library provides a mechanism to enable MPI support for custom containers based on the array or gf libraries.
Basic usage
-------------
The syntax is inspired by Boost.MPI. In order to create an MPI environment, set up the communicator and broadcast a variable, use the following code block:
.. code-block:: c
int main(int argc, char* argv[]) {
mpi::environment env(argc, argv);
mpi::communicator world;
int a = 5;
broadcast(a, world);
}
The declaration of the communicator is optional. If no communicator is passed to the routine, ``MPI_COMM_WORLD`` is used by default.
All collective operations have the same signature. They take up to three arguments:
.. code-block:: c
reduce(T const &x, communicator = {}, int root = 0)
Here T can be any supported type. The communicator is optional. By default, the data will be collected on (or transmitted from) the process with id 0.
Headers
--------------
Support for basic types is provided by the header ``triqs/mpi/base.hpp`` and for vectors and arrays by ``triqs/mpi/vector.hpp`` and ``tiqs/mpi/array.hpp``. For custom container types, the header ``triqs/mpi/generic.hpp`` is required. Support for Boost.MPI is provided by the ``triqs/mpi/boost.hpp`` header file.
For convenience, we provide the header::
triqs/mpi.hpp
which includes the headers for basic, vector, array and generic type support.
Doxygen documentation
-------------------------
The :doxy:`full C++ parameter documentation<triqs::utility::parameters>` and
the :doxy:`parameter_defaults documentation<triqs::utility::parameter_defaults>` are available here.
MPI example
-------------
.. triqs_example:: ./mpi_0.cpp
Simple MPI example.

View File

@ -0,0 +1,30 @@
#include <triqs/arrays.hpp>
#include <triqs/mpi.hpp>
#include <iostream>
using namespace triqs;
using namespace triqs::arrays;
using namespace triqs::mpi;
int main(int argc, char *argv[]) {
mpi::environment env(argc, argv);
mpi::communicator world;
int a = 5;
broadcast(a);
reduce_in_place(a);
array<int, 2> A(2,10); A()=1;
std::cout<<A<<std::endl;
scatter(A);
A += world.rank();
gather(A);
std::cout<<A<<std::endl;
return 0;
}

View File

@ -0,0 +1,78 @@
/*******************************************************************************
*
* TRIQS: a Toolbox for Research in Interacting Quantum Systems
*
* Copyright (C) 2013 by O. Parcollet
*
* TRIQS is free software: you can redistribute it and/or modify it under the
* terms of the GNU General Public License as published by the Free Software
* Foundation, either version 3 of the License, or (at your option) any later
* version.
*
* TRIQS is distributed in the hope that it will be useful, but WITHOUT ANY
* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License along with
* TRIQS. If not, see <http://www.gnu.org/licenses/>.
*
******************************************************************************/
#include <iostream>
#include <type_traits>
#include <triqs/arrays.hpp>
#include <triqs/mpi.hpp>
#include <iostream>
#include <fstream>
#include <sstream>
using namespace triqs;
using namespace triqs::arrays;
using namespace triqs::mpi;
template <typename T>
std::ostream & operator << (std::ostream & out, std::vector<T> const & v) {
for (size_t i =0; i<v.size(); ++i) out<< v[i];
return out;
}
int main(int argc, char* argv[]) {
mpi::environment env(argc, argv);
mpi::communicator world;
// using ARR = array<double,2>;
using VEC = std::vector<std::complex<double>>;
VEC A(7), B(7), AA(7);
clef::placeholder<0> i_;
clef::placeholder<1> j_;
triqs::clef::make_expr(A)[i_] << i_+1;
B = mpi::scatter(A, world);
VEC C = mpi::scatter(A, world);
std::ofstream out("node" + std::to_string(world.rank()));
out << " A = " << A << std::endl;
out << " B = " << B << std::endl;
out << " C = " << C << std::endl;
for(auto &x: B) x *= -1;
for(auto &x: AA) x = 0;
AA = mpi::gather(B, world);
out << " AA = " << AA << std::endl;
mpi::broadcast(AA, world);
out << " cast AA = " << AA << std::endl;
for (auto &x : AA) x = 0;
AA = mpi::allgather(B, world);
out << " AA = " << AA << std::endl;
}

View File

@ -116,7 +116,7 @@ namespace mpi {
static T invoke(tag::allreduce, communicator c, T a, int root) { static T invoke(tag::allreduce, communicator c, T a, int root) {
T b; T b;
MPI_Allreduce(&a, &b, 1, D(), MPI_SUM, root, c.get()); MPI_Allreduce(&a, &b, 1, D(), MPI_SUM, c.get());
return b; return b;
} }

View File

@ -38,7 +38,11 @@ namespace mpi {
} }
// ----------- // -----------
static void broadcast(communicator c, std::vector<T> &a, int root) { MPI_Bcast(a.data(), a.size(), D(), root, c.get()); } static void broadcast(communicator c, std::vector<T> &a, int root) {
size_t s=a.size();
MPI_Bcast(&s, 1, mpi_datatype<size_t>::invoke(), root, c.get());
if(c.rank() != root) a.resize(s);
MPI_Bcast(a.data(), a.size(), D(), root, c.get()); }
// ----------- // -----------
static std::vector<T> invoke(tag::reduce, communicator c, T const &a, int root) { static std::vector<T> invoke(tag::reduce, communicator c, T const &a, int root) {
@ -91,7 +95,7 @@ namespace mpi {
// ----------- // -----------
static std::vector<T> invoke(tag::allgather, communicator c, std::vector<T> const &a, int root) { static std::vector<T> invoke(tag::allgather, communicator c, std::vector<T> const &a, int root) {
long size = reduce(a.size(), c, root); long size = allreduce(a.size(), c, root);
std::vector<T> b(size); std::vector<T> b(size);
auto recvcounts = std::vector<int>(c.size()); auto recvcounts = std::vector<int>(c.size());