Example Job Scripts
This page contains a small collection of example job scripts users may find useful for submitting their jobs on Polaris. Additional information on PBS and how to submit these job scripts is available here.
A simple example using a similar script on Polaris is available in the Getting Started Repo.
Comments in PBS scripts
Since #
is required prior to each PBS directive, comments should be added after the directives have been listed in your submission script. If you try to add comments within the directive list, you could experience submission issues due to PBS attempting to read your comment as an additional directive. This includes adding comments on the same line as a directive (i.e., #PBS -q <queue_name> #comment
).
CPU MPI-OpenMP Examples
The following submit.sh
example submits a 1-node job to Polaris with 16 MPI ranks per node and 2 OpenMP threads per rank. See Queues for details on practical limits to node counts and job times for different sizes of jobs.
The hello_affinity
program is a compiled C++ code, which is built via make -f Makefile.nvhpc
in the linked directory after cloning the Getting Started repository.
cd ${PBS_O_WORKDIR}
: change into the working directory from whereqsub
was executed.NNODES= `wc -l < $PBS_NODEFILE`
: one method for determining the total number of nodes allocated to a job.NRANKS_PER_NODE=16
: This is a helper variable to set the number of MPI ranks for each node to 16.NDEPTH=2
: This is a helper variable to space MPI ranks 2 "slots" from each other. In this example, individual threads correspond to a slot. This will be used together with the--cpu-bind
option frommpiexec
and additional binding options are available (e.g.numa
,socket
,core
, etc.).NTHREADS=2
: This is a helper variable to set the number of OpenMP threads per MPI rank.NTOTRANKS=$(( NNODES * NRANKS_PER_NODE))
: This is a helper variable calculating the total number of MPI ranks spanning all nodes in the job.
The following function in the hello_affinity
source code is essential for uniquely identifying the CUDA device even when Multi-Instance GPU (MIG) is enabled, as each physical device will be partitioned into multiple virtual devices, each with unique UUIDs differentiated by the last few characters:
//https://stackoverflow.com/questions/68823023/set-cuda-device-by-uuid
void uuid_print(cudaUUID_t a){
std::cout << "GPU";
std::vector<std::tuple<int, int> > r = {{0,4}, {4,6}, {6,8}, {8,10}, {10,16}};
for (auto t : r){
std::cout << "-";
for (int i = std::get<0>(t); i < std::get<1>(t); i++)
std::cout << std::hex << std::setfill('0') << std::setw(2) << (unsigned)(unsigned char)a.bytes[i];
}
std::cout << std::endl;
}
Zsh users
If you are a zsh
user, you will need to ensure all PBS job submission and shell scripts include the -l
flag following #!/bin/bash
as seen in the example above to ensure your environment is being instantiated properly. zsh
is not officially supported by HPE and support from ALCF will be best effort only.*
Each Polaris compute node has 1 Milan CPU with a total of 32 physical cores, with each core supporting 2 hardware threads (for a total of 64 logical cores).
The process affinity in this example is set up to map each MPI rank to 2 physical cores. Each MPI rank spawns 2 OpenMP threads, so 1 thread per physical core. The OpenMP settings bind each OpenMP thread to a single hardware thread within a core, such that all 32 physical cores are utilized. CPU core IDs 32
to 63
are not mapped to any MPI rank, since they correspond to simultaneous multithreaded (SMT) sibling hardware threads that share the execution resources of the core ids 0
to 31
, respectively.
cd ${PBS_O_WORKDIR}
: change into the working directory from whereqsub
was executed.NNODES= `wc -l < $PBS_NODEFILE`
: one method for determining the total number of nodes allocated to a job.NRANKS_PER_NODE=16
: This is a helper variable to set the number of MPI ranks for each node to 16.NDEPTH=2
: This is a helper variable to space MPI ranks 2 "slots" from each other. In this example, individual threads correspond to a slot. This will be used together with the--cpu-bind
option frommpiexec
and additional binding options are available (e.g.numa
,socket
,core
, etc.).NTHREADS=2
: This is a helper variable to set the number of OpenMP threads per MPI rank.NTOTRANKS=$(( NNODES * NRANKS_PER_NODE))
: This is a helper variable calculating the total number of MPI ranks spanning all nodes in the job.
Information on the use of mpiexec
is available via man mpiexec
. Some notes on the specific options used in the above example follow.
-n ${NTOTRANKS}
: This is specifying the total number of MPI ranks to start.--ppn ${NRANKS_PER_NODE}
: This is specifying the number of MPI ranks to start on each node.--depth=${NDEPTH}
: This is specifying how many cores/threads to space MPI ranks apart on each node.--cpu-bind depth
: This is indicating the number of cores/threads will be bound to MPI ranks based on thedepth
argument.--env OMP_NUM_THREADS=${NTHREADS}
: This is setting the environment variableOMP_NUM_THREADS
to determine the number of OpenMP threads per MPI rank.--env OMP_PLACES=threads
: This is indicating how OpenMP should distribute threads across the resource, in this case across hardware threads.
Hardware threads
This example is similar to the previous, but it exhausts all 64 logical cores available on each compute node CPU. We double the number of MPI ranks to 32, one per each physical core. Using --cpu-bind=core
, the --depth
flag value becomes interpreted by Cray MPICH as spacing in number of physical cores, so NDEPTH=1
ensures that rank 0 is bound to CPU core IDs (0,32)
, the 2 SMT sibling hardware threads that share the first physical core.
Many HPC applications do not benefit from utilizing the CPU's SMT2 capabilities, and such software may achieve better performance by using the previous script such that each of the 32 physical cores only runs a single OpenMP thread.
GPU MPI Examples
Using the CPU job submission examples above as a baseline, there are not many additional changes needed to enable an application to make use of the 4 NVIDIA A100 GPUs on each Polaris node. In the following 2-node example (because #PBS -l select=2
indicates the number of nodes requested), 4 MPI ranks will be started on each node assigning 1 MPI rank to each GPU in a round-robin fashion. A simple example using a similar job submission script on Polaris is available in the Getting Started Repo.
The affinity options NDEPTH=8;
and --cpu-bind depth
or core
are set to ensure that each MPI rank is bound to a separate NUMA node. If OpenMP threading is desired, set NTHREADS=8
for each MPI rank to spawn 1 thread per physical core (all in the same NUMA domain that the rank is bound to). The OpenMP-related options are not needed if your application does not use OpenMP. Nothing additional is required on the mpiexec
command for applications that internally manage GPU devices and handle the binding of MPI/OpenMP processes to GPUs. A small helper script is available for those with applications that rely on MPI to handle the binding of MPI ranks to GPUs. Some notes on this helper script and other key differences with the early CPU example follow.
export MPICH_GPU_SUPPORT_ENABLED=1
For applications that support GPU-enabled MPI (i.e. use MPI to communicate data directly between GPUs), this environment variable is required to enable GPU support in Cray's MPICH. Omitting this will result in a segfault. Support for this also requires that the application was linked against the GPU Transport Layer library (e.g. -lmpi_gtl_cuda), which is automatically included for users by the craype-accel-nvidia80
module in the default environment on Polaris. If this gtl library is not properly linked, then users will see an error message indicating that upon executing the first MPI command that uses a device pointer.
./set_affinity_gpu_polaris.sh
This script is useful for those applications that rely on MPI to bind MPI ranks to GPUs on each node. Such a script is not necessary when the application handles process-gpu binding. This script simply sets the environment variable CUDA_VISIBLE_DEVICES
to a restricted set of GPUs (e.g. each MPI rank sees only one GPU). Otherwise, users would find that all MPI ranks on a node will target the first GPU likely having a negative impact on performance. An example for this script is available in the Getting Started repo and copied below.
Hardware threads
As in the previous hardware threads example, the MPI ranks are spaced apart assuming the user wants to utilize all 64 logical cores (achieved by setting NTHREADS=$NDEPTH=16
and --cpu-bind numa
here).
In this script, we have added -j oe
to the list of PBS options; -j oe
combines stdout and stderr to the same file and uses the stdout filename provided (if provided). -j eo
would do the same but use the stderr filename provided. Without these options, separate files containing stdout and stderr of the job are produced.
Here we compare two bare-bones PBS submission scripts for a CUDA example with and without MPI:
#!/bin/bash
#PBS -l select=2
#PBS -l walltime=00:10:00
#PBS -q debug
#PBS -l filesystems=home
#PBS -A <project-name>
#PBS -o logs/
#PBS -e logs/
# Count number of nodes assigned
NNODES=`wc -l < $PBS_NODEFILE`
# set 1 MPI rank per GPU
NRANKS_PER_NODE=4
# calculate total ranks
NTOTRANKS=$(( NNODES * NRANKS_PER_NODE ))
echo NUM_OF_NODES= ${NNODES} TOTAL_NUM_RANKS= ${NTOTRANKS} RANKS_PER_NODE= ${NRANKS_PER_NODE}
mpiexec -n ${NTOTRANKS} --ppn ${NRANKS_PER_NODE} $HOME/ALCFBeginnersGuide/polaris/examples/01_example_mpi
Setting GPU affinity for each MPI rank
The CUDA_VISIBLE_DEVICES
environment variable is provided for users to set which GPUs on a node are accessible to an application or MPI ranks started on a node.
A copy of the small helper script provided in the Getting Started repo is provided below for reference:
#!/bin/bash -l
num_gpus=4
# need to assign GPUs in reverse order due to topology
# See Polaris Device Affinity Information https://www.alcf.anl.gov/support/user-guides/polaris/hardware-overview/machine-overview/index.html
gpu=$((${num_gpus} - 1 - ${PMI_LOCAL_RANK} % ${num_gpus}))
export CUDA_VISIBLE_DEVICES=$gpu
echo “RANK= ${PMI_RANK} LOCAL_RANK= ${PMI_LOCAL_RANK} gpu= ${gpu}”
exec "$@"
Note
The echo
command prints a helpful message for the user to confirm the desired mapping is achieved. Users are encouraged to edit this file as necessary for their particular needs.
Warning
If planning large-scale runs with many thousands of MPI ranks, it is advised to comment out the echo
command above so as not to have thousands of lines of output written to stdout
.
Using MPS on the GPUs
Documentation for the NVIDIA Multi-Process Service (MPS) can be found here
In the script below, note that if you are going to run this as a multi-node job you will need to do this on every compute node, and you will need to ensure that the paths you specify for CUDA_MPS_PIPE_DIRECTORY
and CUDA_MPS_LOG_DIRECTORY
do not "collide" and end up with all the nodes writing to the same place.
An example is available in the Getting Started Repo and discussed below. The local SSDs or /dev/shm
or incorporation of the node name into the path would all be possible ways of dealing with that issue.
#!/bin/bash -l
export CUDA_MPS_PIPE_DIRECTORY=</path/writeable/by/you>
export CUDA_MPS_LOG_DIRECTORY=</path/writeable/by/you>
CUDA_VISIBLE_DEVICES=0,1,2,3 nvidia-cuda-mps-control -d
echo "start_server -uid $( id -u )" | nvidia-cuda-mps-control
To verify the control service is running:
And the output should look similar to this:
+-----------------------------------------------------------------------------+
| Processes: |
| GPU GI CI PID Type Process name GPU Memory |
| ID ID Usage |
|=============================================================================|
| 0 N/A N/A 58874 C nvidia-cuda-mps-server 27MiB |
| 1 N/A N/A 58874 C nvidia-cuda-mps-server 27MiB |
| 2 N/A N/A 58874 C nvidia-cuda-mps-server 27MiB |
| 3 N/A N/A 58874 C nvidia-cuda-mps-server 27MiB |
+-----------------------------------------------------------------------------+
To shut down the service:
echo "quit" | nvidia-cuda-mps-control
To verify the service shut down properly:
nvidia-smi | grep -B1 -A15 Processes
And the output should look like this:
+-----------------------------------------------------------------------------+
| Processes: |
| GPU GI CI PID Type Process name GPU Memory |
| ID ID Usage |
|=============================================================================|
| No running processes found |
+-----------------------------------------------------------------------------+
Using MPS in Multi-node Jobs
As stated earlier, it is important to start the MPS control service on each node in a job that requires it. An example is available in the Getting Started Repo. The helper script enable_mps_polaris.sh
can be used to start the MPS on a node.
The helper script disable_mps_polaris.sh
can be used to disable MPS at appropriate points during a job script, if needed.
In the example job script submit.sh
below, MPS is first enabled on all nodes in the job using mpiexec -n ${NNODES} --ppn 1
to launch the enablement script using a single MPI rank on each compute node. The application is then run as normally. If desired, a similar one-rank-per-node mpiexec
command can be used to disable MPS on all the nodes in a job.
Single-node Ensemble Calculations Example
In the script below, a set of four applications are launched simultaneously on a single node. Each application runs on 8 MPI ranks and targets a specific GPU using the CUDA_VISIBLE_DEVICES
environment variable. In the first instance, MPI ranks 0-7 will spawn on CPUs 24-31, and GPU 0 is used. This pairing of CPUs and GPU is based on output of the nvidia-smi topo-m
command showing which CPUs share a NUMA domain with each GPU. It is important to background processes using &
and to wait
for all runs to complete before exiting the script or continuing on with additional work. Note, multiple applications can run on the same set of CPU resources, but it may not be optimal depending on the workload. An example is available in the Getting Started Repo.
Multi-node Ensemble Calculations Example
To run multiple concurrent applications on distinct sets of nodes, one simply needs to provide appropriate hostfiles to the mpiexec
command. The split
unix command is one convenient way to create several unique hostfiles, each containing a subset of nodes available to the job. In the 8-node example below, a total of four applications will be launched on separate sets of nodes. The $PBS_NODEFILE
file will be split into several hostfiles, each containing two lines (nodes). These smaller hostfiles are then used as the argument to the --hostfile
argument of mpiexec
to the launch applications. It is important to background processes using &
and to wait
for applications to finish running before leaving the script or continuing on with additional work. Note, multiple applications can run on the same set of CPU resources, but it may not be optimal depending on the workload. An example is available in the Getting Started Repo.
Job array example
In situations where you wish to repeat a job multiple times with a small change each time, such as in a parameter space study, a job array may be an option. Unlike the multi-node ensemble case above, each subjob in a job array is its own job and will have its own initialization and tear-down by PBS. Also, a job array will not block all nodes for the length of the longest running task, as is the case for an ensemble job. Jobs on Polaris cannot share nodes with other jobs, so job arrays on Polaris cannot be used to distribute work to different CPU cores or GPUs on a node. In that case, an ensemble job or using mpiexec
as a parallel launcher can accomplish that goal.
Both ensemble jobs and job arrays become unwieldy and inefficient for very large numbers of tasks. They either have limits to the number of tasks that can be created at once (job arrays) or are unable to refill idle nodes when tasks complete (ensemble jobs). In such cases, a workflow management tool that can manage the running of tasks is recommended.
Job array submission scripts
An example job array submission script:
There are two required options for job arrays in PBS: -r
and -J
.
The -r
option must be set like this:
The -J
option sets the number of subjobs in the array and the value of their array indices. The example script above will run 4 subjobs and space their array indices in increments of 2, so the array indices will be 0, 2, 4, and 6.
The form the -J
option takes is
<start_index>
is the index of the first job in the array
* <end_index>
is the index of the last job in the array
* <skip_index>
is the number of index integers to skip between subjobs
* <num_concurrent>
is the maximum number of subjobs that will run concurrently at one time
Within a subjob, the environment variable PBS_ARRAY_INDEX
will contain the index of the subjob in the array. It can be used in the job script to set the value or paths of inputs or outputs.
Interacting with job arrays
The status of job arrays can be queried with the command:
When interacting with a job array with commands like qdel
or qalter
, include the brackets with the jobid, e.g.:
Limits on job arrays
The number of subjobs in a job array is limited by the number of jobs that can be submitted to the queue.
On Polaris, for the debug queue, that is 1, for preemptable, that is 20, and for prod that is 10.
The limit for prod on Polaris is 10 because 10 is the maximum number of jobs that can be routed by prod to one of the execution queues (small, medium, or large). One note, PBS will allow job array submissions of up to 100 subjobs in prod, however, these job arrays will not run because they will not route to an execution queue. This is a known issue on Polaris.