GNU parallel in HPC systems¶
Job campaigns that allocate many small jobs quickly, either using job arrays or custom launcher scripts, should use GNU parallel to reduce the scheduler load. The Slurm scheduler performs 2 jobs,
- allocates resources for a job,
- launches the job steps.
Slurm is designed to allocate resources in an allocation loop that runs periodically, usually every 30-180s, depending on its configuration. If many small jobs are in the queue, then operations triggered during the allocation loop, such as backfilling, become expensive. As a result, the scheduling loop can delay past its period, causing the scheduler to appear slow and unresponsive. GNU parallel executes multiple commands in parallel in a single allocation or even in a single job step, removing the need to allocate resources and reducing the scheduler load.
When should GNU parallel be used?
As a rule of thumb,
- if you are planning to execute jobs campaigns that require more than 10 job allocations per minute, then please consider grouping jobs with GNU parallel;
- if you are planning to execute jobs that cumulatively launch more than 1000 job steps per minute, then please consider grouping job and job steps using GNU parallel.
Using GNU parallel in HPC jobs¶
GNU parallel (command parallel)is a shell tool for executing jobs in parallel using one or more computers. A job can be a single command or a small script that has to be run for each of the lines in the input.
- The jobs are forked from the main job when the parallelcommand executes.
- The parallel command blocks until all forked processes exit.
- You can limit the number of jobs that run in parallel; parallelimplements a form of process pull, where a limited number of processes running in parallel executes the jobs.
Grouping allocations with GNU parallel¶
The scheduler is much more efficient in lunching job steps within a job, where resources have been allocated and there is no need to interact with the resource allocation loop. Job steps are lunched within a job with call to the blocking srun command, so the executable needs to be launched with the srun command within GNU parallel.
#!/bin/bash --login
#SBATCH --job-name=single_job_step
#SBATCH --partition=batch
#SBATCH --qos=normal
#SBATCH --nodes=4
#SBATCH --ntasks-per-node=8
#SBATCH --cpus-per-task=16
#SBATCH --time=02:00:00
#SBATCH --output=%x-%j.out
#SBATCH --error=%x-%j.err
declare test_duration=60
parallel \
  --max-procs "${SLURM_NTASKS}" \
  --max-args 0 \
  srun \
    --nodes=1 \
    --ntasks=1 \
    stress-ng \
      --cpu ${SLURM_CPUS_PER_TASK} \
      --timeout "${test_duration}" \
      ::: {0..1023}
This example script launches a single job step per GNU parallel job. The executable stress-ng is a stress test program.
- The number of GNU parallel jobs (--max-procs) is limited to the number of tasks,SLURM_NTASKS, so that there is no overallocation of GNU parallel jobs. Note that if a jobs step is launched without sufficient resources, then the job step fails.
- Each job step is consuming a single task (--ntasks=1) that has access toSLURM_CPUS_PER_TASKCPUs. Thestress-ngprogram requires the explicit specification of the number of CPUs to use for the CPU benchmark with the--cpuflag.
Slurm environment variables (SLURM_*)
Upon starting a job step, srun reads the options defined in a set of environment variables, most of the starting with the prefix SLURM_. The scheduler it self will set many of these variables with an allocation for a job is created. Some useful variables set and their corresponding allocation (sbatch/salloc) flags are the following.
- --nodes:- SLURM_NNODES
- --ntasks:- SLURM_NTASKS
- --ntasks-per-node:- SLURM_NTASKS_PER_NODE
- --cpus-per-task:- SLURM_CPUS_PER_TASK
Note that some environment variables are evaluated implicitly even if the corresponding option is not defined for the allocation. For instance in the example above we define --nodes and --ntasks-per-node, but not --ntasks, yet the SLURM_NTASKS is set to
SLURM_NTASKS = SLURM_NNODES * SLURM_NTASKS_PER_NODE = 4 * 8 = 24
and its value can be used in the submission script without defining --nodes.
Job allocation rate calculation
- In this example jobs steps of 1 minduration (test_duration) are being launched in parallel from4nodes (--nodes) and8tasks per node (--ntasks-per-node), for a total of32tasks.
- If 32tasks where launched every minute in distinct jobs the job allocation rate would be above the empirical limit of10jobs per minute.
- Thus, the use of GNU parallel is justified.
Grouping allocations and job steps with GNU parallel¶
If a jobs contains a massive amounts of very small job steps, it can be limited by the rate with which job steps can be launched. The scheduler stores some information for each job step in a database, and with multiple small steps launching in parallel the throughput limits of the database system can exceeded affecting every job in the cluster. In such extreme cases, GNU parallel can limit the number of job steps by grouping multiple GNU parallel jobs in a single job step.
There are 2 options when lunching multiple GNU parallel jobs in a single jobs step, the GNU parallel jobs of the step can be launched in a script or in a function.
Submission script
#!/bin/bash --login
#SBATCH --job-name=multi_job_step_in_function
#SBATCH --partition=batch
#SBATCH --qos=normal
#SBATCH --nodes=4
#SBATCH --ntasks-per-node=8
#SBATCH --cpus-per-task=16
#SBATCH --time=02:00:00
#SBATCH --output=%x-%j.out
#SBATCH --error=%x-%j.err
declare test_duration=5
declare substeps=$((1024*48+11))
declare substeps_per_step=48
declare cpus_per_substep=4
declare steps=$(( ${substeps} / ${substeps_per_step} ))
declare remainder_steps=$(( ${substeps} % ${substeps_per_step} ))
if [ ! "${remainder_steps}" -eq "0" ]; then
  steps=$(( ${steps} + 1 ))
fi
run_step() {
  local substeps="${1}"
  local substeps_per_step="${2}"
  local test_duration="${3}"
  local cpus_per_substep="${4}"
  local step_idx="${5}"
  local initial_substep=$(( ${step_idx} * ${substeps_per_step} ))
  local final_substep=$(( ${initial_substep} + ${substeps_per_step} ))
  if [ "${final_substep}" -gt "${substeps}" ]; then
    final_substep=${substeps}
  fi
  final_substep=$(( ${final_substep} - 1 ))
  local max_parallel_substeps=$(( ${SLURM_CPUS_PER_TASK} / ${cpus_per_substep} ))
  parallel \
    --max-procs "${max_parallel_substeps}" \
    --max-args 0 \
      stress-ng \
        --cpu "${cpus_per_substep}" \
        --timeout "${test_duration}" \
        ::: $(seq ${initial_substep} "${final_substep}")
}
export -f run_step
declare final_step=$(( ${steps} - 1 ))
parallel \
  --max-procs "${SLURM_NTASKS}" \
  --max-args 1 \
  srun \
    --nodes=1 \
    --ntasks=1 \
    bash \
      -c "\"run_step ${substeps} ${substeps_per_step} ${test_duration} ${cpus_per_substep} {1}\"" \
      ::: $(seq 0 ${final_step})
When running the GNU parallel job steps in a function, make sure that the function is exported to the environment of srun with the export -f bash builtin command.
Submission script
#!/bin/bash --login
#SBATCH --job-name=multi_job_step_in_script
#SBATCH --partition=batch
#SBATCH --qos=normal
#SBATCH --nodes=4
#SBATCH --ntasks-per-node=8
#SBATCH --cpus-per-task=16
#SBATCH --time=02:00:00
#SBATCH --output=%x-%j.out
#SBATCH --error=%x-%j.err
declare test_duration=5
declare substeps=$((1024*48+11))
declare substeps_per_step=48
declare cpus_per_substep=4
declare steps=$(( ${substeps} / ${substeps_per_step} ))
declare remainder_steps=$(( ${substeps} % ${substeps_per_step} ))
if [ ! "${remainder_steps}" -eq "0" ]; then
  steps=$(( ${steps} + 1 ))
fi
declare final_step=$(( ${steps} - 1 ))
parallel \
  --max-procs "${SLURM_NTASKS}" \
  --max-args 1 \
  srun \
    --nodes=1 \
    --ntasks=1 \
    run_job_step \
      "${substeps}" \
      "${substeps_per_step}" \
      "${test_duration}" \
      "${cpus_per_substep}" \
      ::: $(seq 0 ${final_step})
Ensure that the external script run_job_step is accessible from submission script, for instance, place both scripts in the same directory.
External job step execution script run_job_step
#!/bin/bash --login
declare substeps="${1}"
declare substeps_per_step="${2}"
declare test_duration="${3}"
declare cpus_per_substep="${4}"
declare step_idx="${5}"
declare initial_substep=$(( ${step_idx} * ${substeps_per_step} ))
declare final_substep=$(( ${initial_substep} + ${substeps_per_step} ))
if [ "${final_substep}" -gt "${substeps}" ]; then
  final_substep=${substeps}
fi
final_substep=$(( ${final_substep} - 1 ))
declare max_parallel_substeps=$(( ${SLURM_CPUS_PER_TASK} / ${cpus_per_substep} ))
parallel \
  --max-procs "${max_parallel_substeps}" \
  --max-args 0 \
    stress-ng \
      --cpu "${cpus_per_substep}" \
      --timeout "${test_duration}" \
      ::: $(seq ${initial_substep} "${final_substep}")
Finally, make sure that the run_job_step is executable. The command
chmod u+x run_job_step
Job step launch rate calculation
- Each node contains 128CPUs, and each job requires4CPUs (cpus_per_substep); thus each node has128/4 = 32slots to execute jobs in parallel.
- There are 4nodes (--nodes), so with32slots per node, there are in total128slots to launch job steps in parallel.
- Each jobs has a duration of 5 sec, so each slot runs12jobs per minute.
- If every parallel job was a job step, then 12*128 = 1536jobs steps per minute are launched.
- This is above the empirical threshold of 1000job steps per minute, so grouping steps is justified.
- Grouping creates groups of 48GNU parallel jobs (substeps_per_step) reducing the job step launch rate to1536/48 = 32that is safely below the empirical limit of1000per minute.