The distributed-ranges (dr) library is a C++20 library for multi-CPU and multi-GPU computing environments. It provides algorithms, data structures and views tailored to use in multi-node HPC systems and servers with many CPUs and/or GPUs. It takes advantage of parallel processing and MPI communication in distributed memory model as well as parallel processing in shared memory model with many GPUs. The library is designed as replacement for chosen data structures, containers, and algorithms of the C++20 Standard Template Library. If you are familiar with the C++ Template Libraries, and in particular std::ranges (C++20) or ranges-v3 (C++11 -- C++17), switching to dr will be straightforward, but this tutorial will help you get started even if you have never used them. However, we assume that you are familiar with C++, at least in the C++11 standard (C++20 is recommended).
The distributed-ranges library can be used on any system with a working SYCL or g++ compiler. Intel's DPC++ is recommended, and it is required by this tutorial. g++ v. 10, 11 or 12 is also supported, but GPU usage is not possible. Distributed-ranges depends on MPI and oneDPL libraries. DPC++, oneDPL and oneMPI are part of the oneAPI - open-standards based industry initiative. OneAPI and the associated Intel® oneAPI Toolkits and products, help to provide a unified approach to mixed-architecture offload computing. Its approach also ensures interoperability with existing distributed computing standards. It is recommended to install oneAPI components before downloading distributed-ranges.
Currently, there are two ways to start work with distributed-ranges.
If you want to use dr in your application, and your development environment is connected to the Internet, we encourage you to clone the distributed-ranges-tutorial repository and modify examples provided. The cmake files provided in the skeleton repo will download the dr library as a source code and build the examples, there is no need for separate install.
In Linux system (bash shell) download distributed-ranges-tutorial from GitHub and build with the following commands
git clone https://github.com/mateuszpn/distributed-ranges-tutorial
cd distributed-ranges-tutorial
CXX=icpx CC=icx cmake -B build
cmake --build build -j
mpirun -n N ./build/src/example_name
If you have a compiler different than DPC++, change CXX and CC values respectively. Modify the call of mpirun, replacing N with number of mpi processes you want to start, and example_name with an actual example name.
Now you can:
- modify provided examples
- add new source files, modifying src/CMakeList.txt accordingly
- start a new project, using the tutorial as a template
In case your environment is not configured properly or you just prefer a hassle-free code exploration you can use Docker.
git clone https://github.com/mateuszpn/distributed-ranges-tutorial
cd distributed-ranges-tutorial
docker run -it -v $(pwd):/custom-directory-name -u root docker.io/intel/oneapi:latest /bin/bash
cd custom-directory-name
CXX=icpx CC=icx cmake -B build -DENABLE_SYCL=ON
cmake --build build -j
mpirun -n N ./build/src/example_name
where 'custom-directory-name' stands for the name of a directory containing local repo data on a docker volume
If you want to contribute to distributed-ranges or go through more advanced examples, please go to original distributed-ranges GitHub repository
git clone https://github.com/oneapi-src/distributed-ranges
cd distributed-ranges
CXX=icpx CC=icx cmake -B build -DENABLE_SYCL=ON
cmake --build build -j
The distributed-ranges library provides data-structures, algorithms and views designed to be used in two memory models - distributed memory and shared (common) memory. For distributed memory model, MPI is used as communication library between processes. Both model are able to use SYCL devices (GPUs and multi-core CPUs) for calculations.
Algorithms and data structures are designed to take the user off the need to worry about the technical details of their parallelism. An example would be the definition of a distributed vector in memory of multiple nodes connected using MPI.
dr::mp::distributed_vector<double> dv(N);
Such a vector, containing N elements, is automatically distributed among all the nodes involved in the calculation, with individual nodes storing an equal (if possible) amount of data.
Then again, functions such as for_each()
or transform()
allow you to perform in parallel operations on each element of a data structure conforming to dr.
In this way, many of the technical details related to the parallel execution of calculations can remain hidden from the user. On the other hand, a programmer aware of the capabilities of the environment in which the application is run has access to the necessary information.
General namespace used in the library is dr::
For program using a single node with shared memory available for multiple CPUs and one or more GPUs, data structures and algorithms from dr::sp::
namespace are provided.
For distributed memory model, use the dr::mp::
namespace.
Content of distributes-ranges' data structures is distributed over available nodes. For example, segments of dr::mp::distributed_vector
are located in memory of different nodes (mpi processes). Still, global view of the distributed_vector
is uniform, with contiguous indices.
When implementing an algorithm using a distributed data structure such as distributed_vector
, its segmented internal structure must be kept in mind. The issue comes up when the algorithm references cells adjacent to the current one, and the local loop reaches the beginning or end of the segment. At this point, the neighboring cells are in the physical memory of another node!
To support this situation, the concept of halo was introduced. A halo is an area into which the contents of the edge elements of a neighboring segment are copied. Also, changes in the halo are copied to cells in the corresponding segment to maintain the consistency of the entire vector.
Following algorithms are included in distributed-ranges, both in mp and sp versions:
copy()
exclusive_scan()
fill()
for_each()
inclusive_scan()
iota()
reduce()
sort()
transform()
Refer to C++20 documentation for detailed description of how the above functions work.
The examples should be compiled with SYCL compiler and run with.
mpirun -n N ./build/src/example_name
where N
- number of MPI processes. Replace example_name with appropriate name of a file tu run.
The example, performing very simple decoding of encoded string, presents copying data between local and distributed data structures, and a for_each()
loop performing a lambda on each element of the distributed_vector<>
. Please note, that the copy operation affects only local vector on the node 0 (the root argument of copy()
function is 0), and only the node prints the decoded message.
The example shows the distributed nature of dr data structures. The distributed_vector has segments located in each of the nodes performing the example. The nodes introduce themselves at the beginning. You can try different numbers on MPI processes when calling mpirun
.
iota()
function is aware what distributed_vector is, and fills the segments accordingly. Then node 0 prints out the general information about the vector, and every node presents size and content of its local part.
The example simulates the elementary 1-d cellular automaton (ECA). Description of what the automaton is and how it works can be found in wikipedia. Visualisation of the automaton work is available in ASU team webpage.
The ECA calculates the new value of a cell using old value of the cell and old values of the cell's neighbors. Therefore a halo of 1-cell width is used, to get access to neighboring cells' values when the loop eaches end of local segment of a vector.
Additionally, a use of a subrange is presented, and transform()
function, which puts transformed values of input structure to the output structure, element by element. The transforming function is given as lambda newvalue
.
Please note: after each loop the vector content is printed with fmt::print()
. The formatter function for distributed_vector
is rather slow, as it gets the vector element by element, both from local node and remote nodes. You can think about customised, more effective way of results presentation.
This example illustrates adding two distributed,multidimensional arrays. Each array has two dimensions and is initialized by an std::array
. The arrays are populated with sequential values using a distributed version of iota called mp::iota
. A for_each loop is the main part of the code, computing the sum on a specified number of nodes. It takes a lambda copy function along with two input arrays (a and b) and an output array (c) as parameters. The result is printed on a node 0.
Example 5 outlines a method for calculating a 2D 5-point stencil with distributed multidimensional arrays, specifically utilizing dr::mp::distributed_mdarray
. Initially, it involves setting up key parameters like the radius for element exchange between nodes through dr::mp::halo
, and defining the start and end points of the array slice. The example's core is the mp::stencil_for_each
function, which applies a lambda function to two subsets of the array, designated as input and output. The mdspan_stencil_op
lambda function conducts a simple calculation that involves adding together the values of an element and its adjacent elements and subsequently calculating their average. The mp::halo().exchange()
enables values to be shared across distinct nodes, making this process feasible. Ultimately, the outcomes of the calculation are neatly displayed on node 0 using mdspan(), resulting in a clear indication of the modifications made to the 2D array. This example is a practical demonstration of executing stencil operations on distributed arrays.
This example's code demonstrates a 2D pattern search in a distributed, multidimensional array (mp::distributed_mdarray<float, 2>
). It initializes a 2D array, populates it with mp::iota
, converts it to binary values using mp::transform
and defines a pattern of 2x2. A lambda function is used to scan the array and mark occurrences of the pattern in a separate array. The process is similar to the one demonstrated in example5.
This example showcases usage of mp::distributed_sparse_matrix
. It retrieves data from resources/example.mtx
file in root node, and distributes it between all nodes. The root node initializes vector and broadcasts it to every other node. After that, the mp::gemv
operation is performed and result is returned to std::vector<double>
in the root. Finally, the root prints the multiplied vector and the result.
The example 8 is exactly the same as example 7, the only thing that is different is the initialization of the matrix data. Here the matrix is generated inside the code, has different shape and uses random values. Additionally, we print matrix data together with vector and result.