NVIDIA DOCA DPA All-to-all Application Guide

1.0

This guide explains all-to-all collective operation example when accelerated using the DPA in NVIDIA® BlueField®-3 DPU.

This reference application shows how the message passing interface (MPI) all-to-all collective can be accelerated on the Data Path Accelerator (DPA). In an MPI collective, all processes in the same job call the collective routine.

Given a communicator of n ranks, the application performs a collective operation in which all processes send and receive the same amount of data from all processes (hence all-to-all).

This document describes how to run the all-to-all example using the DOCA DPA API .

All-to-all is an MPI method. MPI is a standardized and portable message passing standard designed to function on parallel computing architectures. An MPI program is one where several processes run in parallel.

system-design-diagram-version-1-modificationdate-1707755743853-api-v2.png

Each process in the diagram divides its local sendbuf into n blocks (4 in this example), each containing sendcount elements (4 in this example). Process i sends the k-th block of its local sendbuf to process k which places the data in the i-th block of its local recvbuf.

Implementing all-to-all method using DOCA DPA offloads the copying of the elements from the srcbuf to the recvbufs to the DPA, and leaves the CPU free to perform other computations.

The following diagram describes the differences between host-based all-to-all and DPA all-to-all.

all-to-all-non-blocking-version-1-modificationdate-1707755742537-api-v2.png

  • In DPA all-to-all, DPA threads perform all-to-all and the CPU is free to do other computations

  • In host-based all-to-all, CPU must still perform all-to-all at some point and is not completely free for other computations

This application leverages the following DOCA library:

Refer to its programming guide for more information.

  • NVIDIA BlueField-3 platform is required

  • The application can be run on target BlueField or on host.

  • Open MPI version 4.1.5rc2 or greater (included in DOCA’s installation).

Info

Please refer to the NVIDIA DOCA Installation Guide for Linux for details on how to install BlueField-related software.

The installation of DOCA’s reference applications contains the sources of the applications, alongside the matching compilation instructions. This allows for compiling the applications “as-is” and provides the ability to modify the sources, then compile a new version of the application.

Tip

For more information about the applications as well as development and compilation tips, refer to the DOCA Applications page.

The sources of the application can be found under the application’s directory: /opt/mellanox/doca/applications/dpa_all_to_all/.

Compiling All Applications

All DOCA applications are defined under a single meson project. So, by default, the compilation includes all of them.

To build all applications together, run:

Copy
Copied!
            

cd /opt/mellanox/doca/applications/ meson /tmp/build ninja -C /tmp/build

Info

doca_dpa_all_to_all is created under /tmp/build/dpa_all_to_all/.


Compiling DPA All-to-all Application Only

To directly build only all-to-all application:

Copy
Copied!
            

cd /opt/mellanox/doca/applications/ meson /tmp/build -Denable_all_applications=false -Denable_dpa_all_to_all=true ninja -C /tmp/build

Info

doca_dpa_all_to_all is created under /tmp/build/dpa_all_to_all/.

Alternatively, one can set the desired flags in meson_options.txt file instead of providing them in the compilation command line:

  1. Edit the following flags in /opt/mellanox/doca/applications/meson_options.txt:

    • Set enable_all_applications to false

    • Set enable_dpa_all_to_all to true

  2. Run the following compilation commands :

    Copy
    Copied!
                

    cd /opt/mellanox/doca/applications/ meson /tmp/build ninja -C /tmp/build

    Info

    doca_dpa_all_to_all is created under /tmp/build/dpa_all_to_all/.

Troubleshooting

Please refer to the NVIDIA DOCA Troubleshooting Guide for any issue encountered with the compilation of the application .

Prerequisites

MPI is used for compilation and running of this application. Make sure that MPI is installed on your setup (openmpi is provided as part of the installation of doca-tools).

Note

The installation also requires updating the LD_LIBRARY_PATH and PATH environment variable to include MPI. For example, if openmpi is installed under /usr/mpi/gcc/openmpi-4.1.7a1 then updating the environment variables should be like this:

Copy
Copied!
            

export PATH=/usr/mpi/gcc/openmpi-4.1.7a1/bin:${PATH} export LD_LIBRARY_PATH=/usr/mpi/gcc/openmpi-4.1.7a1/lib:${LD_LIBRARY_PATH}


Application Execution

DPA all-to-all application is provided in source form. Therefore, a compilation is required before application can be executed.

  1. Application usage instructions:

    Copy
    Copied!
                

    Usage: doca_dpa_all_to_all [DOCA Flags] [Program Flags]   DOCA Flags: -h, --help Print a help synopsis -v, --version Print program version information -l, --log-level Set the (numeric) log level for the program <10=DISABLE, 20=CRITICAL, 30=ERROR, 40=WARNING, 50=INFO, 60=DEBUG, 70=TRACE> --sdk-log-level Set the SDK (numeric) log level for the program <10=DISABLE, 20=CRITICAL, 30=ERROR, 40=WARNING, 50=INFO, 60=DEBUG, 70=TRACE> -j, --json <path> Parse all command flags from an input json file   Program Flags: -m, --msgsize <Message size> The message size - the size of the sendbuf and recvbuf (in bytes). Must be in multiplies of integer size. Default is size of one integer times the number of processes. -d, --devices <IB device names> IB devices names that supports DPA, separated by comma without spaces (max of two devices). If not provided then a random IB device will be chosen.

    Info

    This usage printout can be printed to the command line using the -h (or --help) option:

    Copy
    Copied!
                

    ./doca_dpa_all_to_all -h

    Info

    For additional information, please refer to section “Command Line Flags”.

  2. CLI example for running the application on host:

    Note

    This is an MPI program, so use mpirun to run the application (with the -np flag to specify the number of processes to run).

    • The following runs the DPA all-to-all application with 8 processes using the default message size (the number of processes, which is 8, times the size of 1 integer) with a random InfiniBand device:

      Copy
      Copied!
                  

      mpirun -np 8 ./doca_dpa_all_to_all

    • The following runs DPA all-to-all application with 8 processes, with 128 bytes as message size, and with mlx5_0 and mlx5_1 as the InfiniBand devices:

      Copy
      Copied!
                  

      mpirun-np 8 ./doca_dpa_all_to_all -m 128 -d "mlx5_0,mlx5_1"

      Note

      The application supports running with a maximum of 16 processes. If you try to run with more processes, an error is printed and the application exits.

  3. The application also supports a JSON-based deployment mode, in which all command-line arguments are provided through a JSON file:

    Copy
    Copied!
                

    ./doca_dpa_all_to_all --json [json_file]

    For example:

    Copy
    Copied!
                

    ./doca_dpa_all_to_all --json ./dpa_all_to_all_params.json

    Note

    Before execution, ensure that the used JSON file contains the correct configuration parameters, especially the InfiniBand device identifiers.

Command Line Flags

Flag Type

Short Flag

Long Flag/JSON Key

Description

JSON Content

General flags

h

help

Prints a help synopsis

N/A

v

version

Prints program version information

N/A

l

log-level

Set the log level for the application:

  • DISABLE=10

  • CRITICAL=20

  • ERROR=30

  • WARNING=40

  • INFO=50

  • DEBUG=60

  • TRACE=70 (requires compilation with TRACE log level support)

Copy
Copied!
            

"log-level": 60

N/A

sdk-log-level

Sets the log level for the program:

  • DISABLE=10

  • CRITICAL=20

  • ERROR=30

  • WARNING=40

  • INFO=50

  • DEBUG=60

  • TRACE=70

Copy
Copied!
            

"sdk-log-level": 40

j

json

Parse all command flags from an input json file

N/A

Program flags

m

msgsize

The message size. The size of the sendbuf and recvbuf (in bytes). Must be in multiples of an integer. The default is size of 1 integer times the number of processes.

Copy
Copied!
            

"msgsize": -1

Note

The value -1 is a placeholder to use the default size, which is only known at run time (because it depends on the number of processes).

d

devices

InfiniBand devices names that support DPA, separated by comma without spaces (max of two devices). If NOT_SET then a random InfiniBand device is chosen.

Copy
Copied!
            

"devices": "NOT_SET"

Info

Refer to DOCA Arg Parser for more information regarding the supported flags and execution modes.


Troubleshooting

Refer to the NVIDIA DOCA Troubleshooting Guide for any issue encountered with the installation or execution of the DOCA applications .

  1. Initialize MPI.

    Copy
    Copied!
                

    MPI_Init(&argc, &argv);

  2. Parse application arguments.

    1. Initialize arg parser resources and register DOCA general parameters.

      Copy
      Copied!
                  

      doca_argp_init();

    2. Register the application’s parameters.

      Copy
      Copied!
                  

      register_all_to_all_params();

    3. Parse the arguments.

      Copy
      Copied!
                  

      doca_argp_start();

      1. The msgsize parameter is the size of the sendbuf and recvbuf (in bytes). It must be in multiples of an integer and at least the number of processes times an integer size.

      2. The devices_param parameter is the names of the InfiniBand devices to use (must support DPA). It can include up to two devices names.

    4. Only let the first process (of rank 0) parse the parameters to then broadcast them to the rest of the processes.

  3. Check and prepare the needed resources for the all_to_all call:

    1. Check the number of processes (maximum is 16).

    2. Check the msgsize. It must be in multiples of integer size and at least the number of processes times integer size.

    3. Allocate the sendbuf and recvbuf according to msgsize.

  4. Prepare the resources required to perform all-to-all method using DOCA DPA:

    1. Initialize DOCA DPA context:

      1. Open DOCA DPA device (DOCA device that supports DPA).

        Copy
        Copied!
                    

        open_dpa_device(&doca_device);

      2. Initialize DOCA DPA context using the opened device.

        Copy
        Copied!
                    

        extern struct doca_dpa_app *dpa_all2all_app;   doca_dpa_create(doca_device, &doca_dpa);   doca_dpa_set_app(doca_dpa, dpa_all2all_app);   doca_dpa_start(doca_dpa);

    2. Initialize the required DOCA Sync Events for the all-to-all:

      1. One completion event for the kernel launch where the subscriber is CPU and the publisher is DPA.

      2. Kernel events, published by remote peer and subscribed to by DPA, as the number of processes.

        Copy
        Copied!
                    

        create_dpa_a2a_events() { // initialize completion event doca_sync_event_create(&comp_event);   doca_sync_event_add_publisher_location_dpa(comp_event);   doca_sync_event_add_subscriber_location_cpu(comp_event);   doca_sync_event_start(comp_event); // initialize kernels events for (i = 0; i < resources->num_ranks; i++) { doca_sync_event_create(&(kernel_events[i]));   doca_sync_event_add_publisher_location_remote_net(kernel_events[i]);   doca_sync_event_add_subscriber_location_dpa(kernel_events[i]);   doca_sync_event_start(kernel_events[i]); } }

    3. Prepare DOCA RDMAs and set them to work on DPA:

      1. Create DOCA RDMAs as the number of processes/ranks.

        Copy
        Copied!
                    

        for (i = 0; i < resources->num_ranks; i++) { doca_rdma_create(&rdma);   rdma_as_doca_ctx = doca_rdma_as_ctx(rdma);   doca_rdma_set_permissions(rdma);   doca_rdma_set_grh_enabled(rdma);   doca_ctx_set_datapath_on_dpa(rdma_as_doca_ctx, doca_dpa);   doca_ctx_start(rdma_as_doca_ctx); }

      2. Connect local DOCA RDMAs to the remote DOCA RDMAs.

        Copy
        Copied!
                    

        connect_dpa_a2a_rdmas();

      3. Get DPA handles for local DOCA RDMAs (so they can be used by DPA kernel) and copy them to DPA heap memory.

        Copy
        Copied!
                    

        for (int i = 0; i < resources->num_ranks; i++) { doca_rdma_get_dpa_handle(rdmas[i], &(rdma_handles[i])); }   doca_dpa_mem_alloc(&dev_ptr_rdma_handles);   doca_dpa_h2d_memcpy(dev_ptr_rdma_handles, rdma_handles);

    4. Prepare the memory required to perform all-to-all method using DOCA Mmap. This includes creating DPA memory handles for sendbuf and recvbuf, getting other processes recvbufs handles, and copying these memory handles and their remote keys and events handlers to DPA heap memory.

      Copy
      Copied!
                  

      prepare_dpa_a2a_memory();

  5. Launch alltoall_kernel using DOCA DPA kernel launch with all required parameters:

    1. Every MPI rank launches a kernel of up to MAX_NUM_THREADS. This example defines MAX_NUM_THREADS as 16.

    2. Launch alltoall_kernel using kernel_launch.

      Copy
      Copied!
                  

      doca_dpa_kernel_launch_update_set();

    3. Each process should perform num_ranks RDMA write operations, with local and remote buffers calculated based on the rank of the process that is performing the RDMA write operation and the rank of the remote process that is being written to. The application iterates over the rank of the remote process.i

      Each process runs num_threads threads on this kernel, therefore the number of RDMA write operations (which is the number of processes) is divided by the number of threads.
      Each thread should wait on its local events to make sure that the remote processes have finished RDMA write operations.
      Each thread should also synchronize its RDMA DPA handles to make sure that the local RDMA operation calls has finished.

      Copy
      Copied!
                  

      for (i = thread_rank; i < num_ranks; i += num_threads) { doca_dpa_dev_rdma_post_write(); doca_dpa_dev_rdma_signal_set(); }   for (i = thread_rank; i < num_ranks; i += num_threads) { doca_dpa_dev_sync_event_wait_gt(); doca_dpa_dev_rdma_synchronize(); }

    4. Wait until alltoall_kernel has finished.

      Copy
      Copied!
                  

      doca_sync_event_wait_gt();

      Note

      Add an MPI barrier after waiting for the event to make sure that all of the processes have finished executing alltoall_kernel.

      Copy
      Copied!
                  

      MPI_Barrier();

      After alltoall_kernel is finished, the recvbuf of all processes contains the expected output of all-to-all method.

  6. Destroy a2a_resources:

    1. Free all DOCA DPA memories.

      Copy
      Copied!
                  

      doca_dpa_mem_free();

    2. Destroy all DOCA Mmaps

      Copy
      Copied!
                  

      doca_mmap_destroy();

    3. Destroy all DOCA RDMAs.

      Copy
      Copied!
                  

      doca_ctx_stop(); doca_rdma_destroy();

    4. Destroy all DOCA Sync Events.

      Copy
      Copied!
                  

      doca_sync_event_destroy();

    5. Destroy DOCA DPA context.

      Copy
      Copied!
                  

      doca_dpa_destroy();

    6. Close DOCA device.

      Copy
      Copied!
                  

      doca_dev_close();

  • /opt/mellanox/doca/applications/dpa_all_to_all/

  • /opt/mellanox/doca/applications/dpa_all_to_all/dpa_all_to_all_params.json

© Copyright 2024, NVIDIA. Last updated on Feb 9, 2024.