
9e40bd860f
ProcessorCount: Update documentation
Acked-by: Kitware Robot <kwrobot@kitware.com>
Merge-request: !10407
258 lines
9.2 KiB
CMake
258 lines
9.2 KiB
CMake
# Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
# file LICENSE.rst or https://cmake.org/licensing for details.
|
|
|
|
#[=======================================================================[.rst:
|
|
ProcessorCount
|
|
--------------
|
|
|
|
This module provides the following function to determine the number of
|
|
processors/cores:
|
|
|
|
.. command:: ProcessorCount
|
|
|
|
.. code-block:: cmake
|
|
|
|
ProcessorCount(<variable>)
|
|
|
|
Sets a local variable named ``<variable>`` to the number of logical CPU cores
|
|
available on the machine, if the information can be determined. If
|
|
successful, the variable is guaranteed to be set to a positive integer (>=1).
|
|
If the processor count cannot be determined, it is set to 0.
|
|
|
|
Currently, this functionality is implemented for AIX, Cygwin, FreeBSD, Haiku,
|
|
HPUX, Linux, macOS, QNX, Sun and Windows.
|
|
|
|
This function provides an approximation of the number of compute cores
|
|
available on the current machine, making it useful for parallel building and
|
|
testing. It is meant to help utilize as much of the machine as seems
|
|
reasonable, though users should consider other workloads running on the
|
|
machine before using its full capacity for parallel tasks.
|
|
|
|
.. versionchanged:: 3.15
|
|
On Linux, returns the container CPU count instead of the host CPU count.
|
|
|
|
.. note::
|
|
|
|
This module relies on system-dependent commands to determine the number of
|
|
processors, which may not always provide accurate information in certain
|
|
environments. A more generally accurate logical CPU count can be also
|
|
obtained with the :command:`cmake_host_system_information`:
|
|
|
|
.. code-block:: cmake
|
|
|
|
cmake_host_system_information(RESULT n QUERY NUMBER_OF_LOGICAL_CORES)
|
|
|
|
Examples
|
|
^^^^^^^^
|
|
|
|
Using ``ProcessorCount`` module in a :option:`ctest -S` dashboard script:
|
|
|
|
.. code-block:: cmake
|
|
|
|
include(ProcessorCount)
|
|
ProcessorCount(n)
|
|
if(NOT n EQUAL 0)
|
|
set(CTEST_BUILD_FLAGS -j${n})
|
|
set(ctest_test_args ${ctest_test_args} PARALLEL_LEVEL ${n})
|
|
endif()
|
|
#]=======================================================================]
|
|
|
|
# A more reliable way might be to compile a small C program that uses the CPUID
|
|
# instruction, but that again requires compiler support or compiling assembler
|
|
# code.
|
|
|
|
function(ProcessorCount var)
|
|
# Unknown:
|
|
set(count 0)
|
|
|
|
if(WIN32)
|
|
# Windows:
|
|
set(count "$ENV{NUMBER_OF_PROCESSORS}")
|
|
#message("ProcessorCount: WIN32, trying environment variable")
|
|
endif()
|
|
|
|
if(NOT count)
|
|
# Mac, FreeBSD, OpenBSD (systems with sysctl):
|
|
find_program(ProcessorCount_cmd_sysctl sysctl
|
|
PATHS /usr/sbin /sbin)
|
|
mark_as_advanced(ProcessorCount_cmd_sysctl)
|
|
if(ProcessorCount_cmd_sysctl)
|
|
execute_process(COMMAND ${ProcessorCount_cmd_sysctl} -n hw.ncpu
|
|
ERROR_QUIET
|
|
OUTPUT_STRIP_TRAILING_WHITESPACE
|
|
OUTPUT_VARIABLE count)
|
|
#message("ProcessorCount: trying sysctl '${ProcessorCount_cmd_sysctl}'")
|
|
endif()
|
|
endif()
|
|
|
|
if(NOT count)
|
|
# Linux (systems with nproc):
|
|
# Prefer nproc to getconf if available as getconf may return the host CPU count in Linux containers
|
|
find_program(ProcessorCount_cmd_nproc nproc)
|
|
mark_as_advanced(ProcessorCount_cmd_nproc)
|
|
if(ProcessorCount_cmd_nproc)
|
|
execute_process(COMMAND ${ProcessorCount_cmd_nproc}
|
|
ERROR_QUIET
|
|
OUTPUT_STRIP_TRAILING_WHITESPACE
|
|
OUTPUT_VARIABLE count)
|
|
#message("ProcessorCount: trying nproc '${ProcessorCount_cmd_nproc}'")
|
|
endif()
|
|
endif()
|
|
|
|
if(NOT count)
|
|
# Linux (systems with getconf):
|
|
find_program(ProcessorCount_cmd_getconf getconf)
|
|
mark_as_advanced(ProcessorCount_cmd_getconf)
|
|
if(ProcessorCount_cmd_getconf)
|
|
execute_process(COMMAND ${ProcessorCount_cmd_getconf} _NPROCESSORS_ONLN
|
|
ERROR_QUIET
|
|
OUTPUT_STRIP_TRAILING_WHITESPACE
|
|
OUTPUT_VARIABLE count)
|
|
#message("ProcessorCount: trying getconf '${ProcessorCount_cmd_getconf}'")
|
|
endif()
|
|
endif()
|
|
|
|
if(NOT count)
|
|
# HPUX (systems with machinfo):
|
|
find_program(ProcessorCount_cmd_machinfo machinfo
|
|
PATHS /usr/contrib/bin)
|
|
mark_as_advanced(ProcessorCount_cmd_machinfo)
|
|
if(ProcessorCount_cmd_machinfo)
|
|
execute_process(COMMAND ${ProcessorCount_cmd_machinfo}
|
|
ERROR_QUIET
|
|
OUTPUT_STRIP_TRAILING_WHITESPACE
|
|
OUTPUT_VARIABLE machinfo_output)
|
|
string(REGEX MATCHALL "Number of CPUs = ([0-9]+)" procs "${machinfo_output}")
|
|
set(count "${CMAKE_MATCH_1}")
|
|
if(NOT count)
|
|
string(REGEX MATCHALL "([0-9]+) logical processors" procs "${machinfo_output}")
|
|
set(count "${CMAKE_MATCH_1}")
|
|
endif()
|
|
#message("ProcessorCount: trying machinfo '${ProcessorCount_cmd_machinfo}'")
|
|
else()
|
|
find_program(ProcessorCount_cmd_mpsched mpsched)
|
|
mark_as_advanced(ProcessorCount_cmd_mpsched)
|
|
if(ProcessorCount_cmd_mpsched)
|
|
execute_process(COMMAND ${ProcessorCount_cmd_mpsched} -s
|
|
OUTPUT_QUIET
|
|
ERROR_STRIP_TRAILING_WHITESPACE
|
|
ERROR_VARIABLE mpsched_output)
|
|
string(REGEX MATCHALL "Processor Count *: *([0-9]+)" procs "${mpsched_output}")
|
|
set(count "${CMAKE_MATCH_1}")
|
|
#message("ProcessorCount: trying mpsched -s '${ProcessorCount_cmd_mpsched}'")
|
|
endif()
|
|
endif()
|
|
endif()
|
|
|
|
if(NOT count)
|
|
# AIX (systems with lsconf):
|
|
find_program(ProcessorCount_cmd_lsconf lsconf
|
|
PATHS /usr/sbin)
|
|
mark_as_advanced(ProcessorCount_cmd_lsconf)
|
|
if(ProcessorCount_cmd_lsconf)
|
|
execute_process(COMMAND ${ProcessorCount_cmd_lsconf}
|
|
ERROR_QUIET
|
|
OUTPUT_STRIP_TRAILING_WHITESPACE
|
|
OUTPUT_VARIABLE lsconf_output)
|
|
string(REGEX MATCHALL "Number Of Processors: ([0-9]+)" procs "${lsconf_output}")
|
|
set(count "${CMAKE_MATCH_1}")
|
|
#message("ProcessorCount: trying lsconf '${ProcessorCount_cmd_lsconf}'")
|
|
endif()
|
|
endif()
|
|
|
|
if(NOT count)
|
|
# QNX (systems with pidin):
|
|
find_program(ProcessorCount_cmd_pidin pidin)
|
|
mark_as_advanced(ProcessorCount_cmd_pidin)
|
|
if(ProcessorCount_cmd_pidin)
|
|
execute_process(COMMAND ${ProcessorCount_cmd_pidin} info
|
|
ERROR_QUIET
|
|
OUTPUT_STRIP_TRAILING_WHITESPACE
|
|
OUTPUT_VARIABLE pidin_output)
|
|
string(REGEX MATCHALL "Processor[0-9]+: " procs "${pidin_output}")
|
|
list(LENGTH procs count)
|
|
#message("ProcessorCount: trying pidin '${ProcessorCount_cmd_pidin}'")
|
|
endif()
|
|
endif()
|
|
|
|
if(NOT count)
|
|
# Sun (systems where psrinfo tool is available)
|
|
find_program(ProcessorCount_cmd_psrinfo psrinfo PATHS /usr/sbin /sbin)
|
|
mark_as_advanced(ProcessorCount_cmd_psrinfo)
|
|
if (ProcessorCount_cmd_psrinfo)
|
|
execute_process(COMMAND ${ProcessorCount_cmd_psrinfo} -p -v
|
|
ERROR_QUIET
|
|
OUTPUT_STRIP_TRAILING_WHITESPACE
|
|
OUTPUT_VARIABLE psrinfo_output)
|
|
string(REGEX MATCHALL "has [0-9]+ virtual processor" procs "${psrinfo_output}")
|
|
set(count "")
|
|
foreach(proc ${procs})
|
|
string(REGEX MATCH "has ([0-9]+) virtual" res ${proc})
|
|
math(EXPR count "${count} + ${CMAKE_MATCH_1}")
|
|
endforeach()
|
|
#message("ProcessorCount: trying '${ProcessorCount_cmd_psrinfo}' -p -v")
|
|
else()
|
|
# Sun (systems where uname -X emits "NumCPU" in its output):
|
|
find_program(ProcessorCount_cmd_uname uname)
|
|
mark_as_advanced(ProcessorCount_cmd_uname)
|
|
if(ProcessorCount_cmd_uname)
|
|
execute_process(COMMAND ${ProcessorCount_cmd_uname} -X
|
|
ERROR_QUIET
|
|
OUTPUT_STRIP_TRAILING_WHITESPACE
|
|
OUTPUT_VARIABLE uname_X_output)
|
|
string(REGEX MATCHALL "NumCPU = ([0-9]+)" procs "${uname_X_output}")
|
|
set(count "${CMAKE_MATCH_1}")
|
|
#message("ProcessorCount: trying uname -X '${ProcessorCount_cmd_uname}'")
|
|
endif()
|
|
endif()
|
|
endif()
|
|
|
|
# Execute this code when all previously attempted methods return empty
|
|
# output:
|
|
#
|
|
if(NOT count)
|
|
# Systems with /proc/cpuinfo:
|
|
set(cpuinfo_file /proc/cpuinfo)
|
|
if(EXISTS "${cpuinfo_file}")
|
|
cmake_policy(PUSH)
|
|
cmake_policy(SET CMP0159 NEW) # file(STRINGS) with REGEX updates CMAKE_MATCH_<n>
|
|
file(STRINGS "${cpuinfo_file}" procs REGEX "^processor.: [0-9]+$")
|
|
cmake_policy(POP)
|
|
list(LENGTH procs count)
|
|
#message("ProcessorCount: trying cpuinfo '${cpuinfo_file}'")
|
|
endif()
|
|
endif()
|
|
|
|
if(NOT count)
|
|
# Haiku
|
|
find_program(ProcessorCount_cmd_sysinfo sysinfo)
|
|
mark_as_advanced(ProcessorCount_cmd_sysinfo)
|
|
if(ProcessorCount_cmd_sysinfo)
|
|
execute_process(COMMAND ${ProcessorCount_cmd_sysinfo}
|
|
ERROR_QUIET
|
|
OUTPUT_STRIP_TRAILING_WHITESPACE
|
|
OUTPUT_VARIABLE sysinfo_X_output)
|
|
string(REGEX MATCHALL "\nCPU #[0-9]+:" procs "\n${sysinfo_X_output}")
|
|
list(LENGTH procs count)
|
|
#message("ProcessorCount: trying sysinfo '${ProcessorCount_cmd_sysinfo}'")
|
|
endif()
|
|
endif()
|
|
|
|
# Since cygwin builds of CMake do not define WIN32 anymore, but they still
|
|
# run on Windows, and will still have this env var defined:
|
|
#
|
|
if(NOT count)
|
|
set(count "$ENV{NUMBER_OF_PROCESSORS}")
|
|
#message("ProcessorCount: last fallback, trying environment variable")
|
|
endif()
|
|
|
|
# Ensure an integer return (avoid inadvertently returning an empty string
|
|
# or an error string)... If it's not a decimal integer, return 0:
|
|
#
|
|
if(NOT count MATCHES "^[0-9]+$")
|
|
set(count 0)
|
|
endif()
|
|
|
|
set(${var} ${count} PARENT_SCOPE)
|
|
endfunction()
|