mirror of https://github.com/pybind/pybind11
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
385 lines
13 KiB
385 lines
13 KiB
#[======================================================[.rst |
|
|
|
Adds the following targets:: |
|
|
|
pybind11::pybind11 - link to headers and pybind11 |
|
pybind11::module - Adds module links |
|
pybind11::embed - Adds embed links |
|
pybind11::lto - Link time optimizations (manual selection) |
|
pybind11::thin_lto - Link time optimizations (manual selection) |
|
pybind11::python_link_helper - Adds link to Python libraries |
|
pybind11::windows_extras - MSVC bigobj and mp for building multithreaded |
|
pybind11::opt_size - avoid optimizations that increase code size |
|
|
|
Adds the following functions:: |
|
|
|
pybind11_strip(target) - strip target after building on linux/macOS |
|
pybind11_find_import(module) - See if a module is installed. |
|
|
|
#]======================================================] |
|
|
|
# CMake 3.10 has an include_guard command, but we can't use that yet |
|
# include_guard(global) (pre-CMake 3.10) |
|
if(TARGET pybind11::lto) |
|
return() |
|
endif() |
|
|
|
# If we are in subdirectory mode, all IMPORTED targets must be GLOBAL. If we |
|
# are in CONFIG mode, they should be "normal" targets instead. |
|
# In CMake 3.11+ you can promote a target to global after you create it, |
|
# which might be simpler than this check. |
|
get_property( |
|
is_config |
|
TARGET pybind11::headers |
|
PROPERTY IMPORTED) |
|
if(NOT is_config) |
|
set(optional_global GLOBAL) |
|
endif() |
|
|
|
# If not run in Python mode, we still would like this to at least |
|
# include pybind11's include directory: |
|
set(pybind11_INCLUDE_DIRS |
|
"${pybind11_INCLUDE_DIR}" |
|
CACHE INTERNAL "Include directory for pybind11 (Python not requested)") |
|
|
|
# --------------------- Shared targets ---------------------------- |
|
|
|
# Build an interface library target: |
|
add_library(pybind11::pybind11 IMPORTED INTERFACE ${optional_global}) |
|
set_property( |
|
TARGET pybind11::pybind11 |
|
APPEND |
|
PROPERTY INTERFACE_LINK_LIBRARIES pybind11::headers) |
|
|
|
# Build a module target: |
|
add_library(pybind11::module IMPORTED INTERFACE ${optional_global}) |
|
set_property( |
|
TARGET pybind11::module |
|
APPEND |
|
PROPERTY INTERFACE_LINK_LIBRARIES pybind11::pybind11) |
|
|
|
# Build an embed library target: |
|
add_library(pybind11::embed IMPORTED INTERFACE ${optional_global}) |
|
set_property( |
|
TARGET pybind11::embed |
|
APPEND |
|
PROPERTY INTERFACE_LINK_LIBRARIES pybind11::pybind11) |
|
|
|
# --------------------------- link helper --------------------------- |
|
|
|
add_library(pybind11::python_link_helper IMPORTED INTERFACE ${optional_global}) |
|
|
|
if(CMAKE_VERSION VERSION_LESS 3.13) |
|
# In CMake 3.11+, you can set INTERFACE properties via the normal methods, and |
|
# this would be simpler. |
|
set_property( |
|
TARGET pybind11::python_link_helper |
|
APPEND |
|
PROPERTY INTERFACE_LINK_LIBRARIES "$<$<PLATFORM_ID:Darwin>:-undefined dynamic_lookup>") |
|
else() |
|
# link_options was added in 3.13+ |
|
# This is safer, because you are ensured the deduplication pass in CMake will not consider |
|
# these separate and remove one but not the other. |
|
set_property( |
|
TARGET pybind11::python_link_helper |
|
APPEND |
|
PROPERTY INTERFACE_LINK_OPTIONS "$<$<PLATFORM_ID:Darwin>:LINKER:-undefined,dynamic_lookup>") |
|
endif() |
|
|
|
# ------------------------ Windows extras ------------------------- |
|
|
|
add_library(pybind11::windows_extras IMPORTED INTERFACE ${optional_global}) |
|
|
|
if(MSVC) # That's also clang-cl |
|
# /bigobj is needed for bigger binding projects due to the limit to 64k |
|
# addressable sections |
|
set_property( |
|
TARGET pybind11::windows_extras |
|
APPEND |
|
PROPERTY INTERFACE_COMPILE_OPTIONS $<$<COMPILE_LANGUAGE:CXX>:/bigobj>) |
|
|
|
# /MP enables multithreaded builds (relevant when there are many files) for MSVC |
|
if("${CMAKE_CXX_COMPILER_ID}" STREQUAL "MSVC") # no Clang no Intel |
|
if(CMAKE_VERSION VERSION_LESS 3.11) |
|
set_property( |
|
TARGET pybind11::windows_extras |
|
APPEND |
|
PROPERTY INTERFACE_COMPILE_OPTIONS $<$<NOT:$<CONFIG:Debug>>:/MP>) |
|
else() |
|
# Only set these options for C++ files. This is important so that, for |
|
# instance, projects that include other types of source files like CUDA |
|
# .cu files don't get these options propagated to nvcc since that would |
|
# cause the build to fail. |
|
set_property( |
|
TARGET pybind11::windows_extras |
|
APPEND |
|
PROPERTY INTERFACE_COMPILE_OPTIONS |
|
$<$<NOT:$<CONFIG:Debug>>:$<$<COMPILE_LANGUAGE:CXX>:/MP>>) |
|
endif() |
|
endif() |
|
endif() |
|
|
|
# ----------------------- Optimize binary size -------------------------- |
|
|
|
add_library(pybind11::opt_size IMPORTED INTERFACE ${optional_global}) |
|
|
|
if(MSVC) |
|
set(PYBIND11_OPT_SIZE /Os) |
|
else() |
|
set(PYBIND11_OPT_SIZE -Os) |
|
endif() |
|
|
|
set_property( |
|
TARGET pybind11::opt_size |
|
APPEND |
|
PROPERTY INTERFACE_COMPILE_OPTIONS $<$<CONFIG:Release>:${PYBIND11_OPT_SIZE}> |
|
$<$<CONFIG:MinSizeRel>:${PYBIND11_OPT_SIZE}> |
|
$<$<CONFIG:RelWithDebInfo>:${PYBIND11_OPT_SIZE}>) |
|
|
|
# ----------------------- Legacy option -------------------------- |
|
|
|
# Warn or error if old variable name used |
|
if(PYBIND11_CPP_STANDARD) |
|
string(REGEX MATCH [[..$]] VAL "${PYBIND11_CPP_STANDARD}") |
|
if(CMAKE_CXX_STANDARD) |
|
if(NOT CMAKE_CXX_STANDARD STREQUAL VAL) |
|
message(WARNING "CMAKE_CXX_STANDARD=${CMAKE_CXX_STANDARD} does not match " |
|
"PYBIND11_CPP_STANDARD=${PYBIND11_CPP_STANDARD}, " |
|
"please remove PYBIND11_CPP_STANDARD from your cache") |
|
endif() |
|
else() |
|
set(supported_standards 11 14 17 20) |
|
if("${VAL}" IN_LIST supported_standards) |
|
message(WARNING "USE -DCMAKE_CXX_STANDARD=${VAL} instead of PYBIND11_CPP_STANDARD") |
|
set(CMAKE_CXX_STANDARD |
|
${VAL} |
|
CACHE STRING "From PYBIND11_CPP_STANDARD") |
|
else() |
|
message(FATAL_ERROR "PYBIND11_CPP_STANDARD should be replaced with CMAKE_CXX_STANDARD " |
|
"(last two chars: ${VAL} not understood as a valid CXX std)") |
|
endif() |
|
endif() |
|
endif() |
|
|
|
# --------------------- Python specifics ------------------------- |
|
|
|
# Check to see which Python mode we are in, new, old, or no python |
|
if(PYBIND11_NOPYTHON) |
|
set(_pybind11_nopython ON) |
|
elseif( |
|
PYBIND11_FINDPYTHON |
|
OR Python_FOUND |
|
OR Python2_FOUND |
|
OR Python3_FOUND) |
|
# New mode |
|
include("${CMAKE_CURRENT_LIST_DIR}/pybind11NewTools.cmake") |
|
|
|
else() |
|
|
|
# Classic mode |
|
include("${CMAKE_CURRENT_LIST_DIR}/pybind11Tools.cmake") |
|
|
|
endif() |
|
|
|
# --------------------- pybind11_find_import ------------------------------- |
|
|
|
if(NOT _pybind11_nopython) |
|
# Check to see if modules are importable. Use REQUIRED to force an error if |
|
# one of the modules is not found. <package_name>_FOUND will be set if the |
|
# package was found (underscores replace dashes if present). QUIET will hide |
|
# the found message, and VERSION will require a minimum version. A successful |
|
# find will cache the result. |
|
function(pybind11_find_import PYPI_NAME) |
|
# CMake variables need underscores (PyPI doesn't care) |
|
string(REPLACE "-" "_" NORM_PYPI_NAME "${PYPI_NAME}") |
|
|
|
# Return if found previously |
|
if(${NORM_PYPI_NAME}_FOUND) |
|
return() |
|
endif() |
|
|
|
set(options "REQUIRED;QUIET") |
|
set(oneValueArgs "VERSION") |
|
cmake_parse_arguments(ARG "${options}" "${oneValueArgs}" "" ${ARGN}) |
|
|
|
if(ARG_REQUIRED) |
|
set(status_level FATAL_ERROR) |
|
else() |
|
set(status_level WARNING) |
|
endif() |
|
|
|
execute_process( |
|
COMMAND |
|
${${_Python}_EXECUTABLE} -c |
|
"from pkg_resources import get_distribution; print(get_distribution('${PYPI_NAME}').version)" |
|
RESULT_VARIABLE RESULT_PRESENT |
|
OUTPUT_VARIABLE PKG_VERSION |
|
ERROR_QUIET) |
|
|
|
string(STRIP "${PKG_VERSION}" PKG_VERSION) |
|
|
|
# If a result is present, this failed |
|
if(RESULT_PRESENT) |
|
set(${NORM_PYPI_NAME}_FOUND |
|
${NORM_PYPI_NAME}-NOTFOUND |
|
CACHE INTERNAL "") |
|
# Always warn or error |
|
message( |
|
${status_level} |
|
"Missing: ${PYPI_NAME} ${ARG_VERSION}\nTry: ${${_Python}_EXECUTABLE} -m pip install ${PYPI_NAME}" |
|
) |
|
else() |
|
if(ARG_VERSION AND PKG_VERSION VERSION_LESS ARG_VERSION) |
|
message( |
|
${status_level} |
|
"Version incorrect: ${PYPI_NAME} ${PKG_VERSION} found, ${ARG_VERSION} required - try upgrading" |
|
) |
|
else() |
|
set(${NORM_PYPI_NAME}_FOUND |
|
YES |
|
CACHE INTERNAL "") |
|
set(${NORM_PYPI_NAME}_VERSION |
|
${PKG_VERSION} |
|
CACHE INTERNAL "") |
|
endif() |
|
if(NOT ARG_QUIET) |
|
message(STATUS "Found ${PYPI_NAME} ${PKG_VERSION}") |
|
endif() |
|
endif() |
|
if(NOT ARG_VERSION OR (NOT PKG_VERSION VERSION_LESS ARG_VERSION)) |
|
# We have successfully found a good version, cache to avoid calling again. |
|
endif() |
|
endfunction() |
|
endif() |
|
|
|
# --------------------- LTO ------------------------------- |
|
|
|
include(CheckCXXCompilerFlag) |
|
|
|
# Checks whether the given CXX/linker flags can compile and link a cxx file. |
|
# cxxflags and linkerflags are lists of flags to use. The result variable is a |
|
# unique variable name for each set of flags: the compilation result will be |
|
# cached base on the result variable. If the flags work, sets them in |
|
# cxxflags_out/linkerflags_out internal cache variables (in addition to |
|
# ${result}). |
|
function(_pybind11_return_if_cxx_and_linker_flags_work result cxxflags linkerflags cxxflags_out |
|
linkerflags_out) |
|
set(CMAKE_REQUIRED_LIBRARIES ${linkerflags}) |
|
check_cxx_compiler_flag("${cxxflags}" ${result}) |
|
if(${result}) |
|
set(${cxxflags_out} |
|
"${cxxflags}" |
|
PARENT_SCOPE) |
|
set(${linkerflags_out} |
|
"${linkerflags}" |
|
PARENT_SCOPE) |
|
endif() |
|
endfunction() |
|
|
|
function(_pybind11_generate_lto target prefer_thin_lto) |
|
if(MINGW) |
|
message(STATUS "${target} disabled (problems with undefined symbols for MinGW for now)") |
|
return() |
|
endif() |
|
|
|
if(CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang") |
|
set(cxx_append "") |
|
set(linker_append "") |
|
if(CMAKE_CXX_COMPILER_ID MATCHES "Clang" AND NOT APPLE) |
|
# Clang Gold plugin does not support -Os; append -O3 to MinSizeRel builds to override it |
|
set(linker_append ";$<$<CONFIG:MinSizeRel>:-O3>") |
|
elseif(CMAKE_CXX_COMPILER_ID MATCHES "GNU" AND NOT MINGW) |
|
set(cxx_append ";-fno-fat-lto-objects") |
|
endif() |
|
|
|
if(CMAKE_SYSTEM_PROCESSOR MATCHES "ppc64le" OR CMAKE_SYSTEM_PROCESSOR MATCHES "mips64") |
|
set(NO_FLTO_ARCH TRUE) |
|
else() |
|
set(NO_FLTO_ARCH FALSE) |
|
endif() |
|
|
|
if(CMAKE_CXX_COMPILER_ID MATCHES "Clang" |
|
AND prefer_thin_lto |
|
AND NOT NO_FLTO_ARCH) |
|
_pybind11_return_if_cxx_and_linker_flags_work( |
|
HAS_FLTO_THIN "-flto=thin${cxx_append}" "-flto=thin${linker_append}" |
|
PYBIND11_LTO_CXX_FLAGS PYBIND11_LTO_LINKER_FLAGS) |
|
endif() |
|
|
|
if(NOT HAS_FLTO_THIN AND NOT NO_FLTO_ARCH) |
|
_pybind11_return_if_cxx_and_linker_flags_work( |
|
HAS_FLTO "-flto${cxx_append}" "-flto${linker_append}" PYBIND11_LTO_CXX_FLAGS |
|
PYBIND11_LTO_LINKER_FLAGS) |
|
endif() |
|
elseif(CMAKE_CXX_COMPILER_ID MATCHES "Intel") |
|
# Intel equivalent to LTO is called IPO |
|
_pybind11_return_if_cxx_and_linker_flags_work(HAS_INTEL_IPO "-ipo" "-ipo" |
|
PYBIND11_LTO_CXX_FLAGS PYBIND11_LTO_LINKER_FLAGS) |
|
elseif(MSVC) |
|
# cmake only interprets libraries as linker flags when they start with a - (otherwise it |
|
# converts /LTCG to \LTCG as if it was a Windows path). Luckily MSVC supports passing flags |
|
# with - instead of /, even if it is a bit non-standard: |
|
_pybind11_return_if_cxx_and_linker_flags_work(HAS_MSVC_GL_LTCG "/GL" "-LTCG" |
|
PYBIND11_LTO_CXX_FLAGS PYBIND11_LTO_LINKER_FLAGS) |
|
endif() |
|
|
|
# Enable LTO flags if found, except for Debug builds |
|
if(PYBIND11_LTO_CXX_FLAGS) |
|
# CONFIG takes multiple values in CMake 3.19+, until then we have to use OR |
|
set(is_debug "$<OR:$<CONFIG:Debug>,$<CONFIG:RelWithDebInfo>>") |
|
set(not_debug "$<NOT:${is_debug}>") |
|
set(cxx_lang "$<COMPILE_LANGUAGE:CXX>") |
|
if(MSVC AND CMAKE_VERSION VERSION_LESS 3.11) |
|
set(genex "${not_debug}") |
|
else() |
|
set(genex "$<AND:${not_debug},${cxx_lang}>") |
|
endif() |
|
set_property( |
|
TARGET ${target} |
|
APPEND |
|
PROPERTY INTERFACE_COMPILE_OPTIONS "$<${genex}:${PYBIND11_LTO_CXX_FLAGS}>") |
|
if(CMAKE_PROJECT_NAME STREQUAL "pybind11") |
|
message(STATUS "${target} enabled") |
|
endif() |
|
else() |
|
if(CMAKE_PROJECT_NAME STREQUAL "pybind11") |
|
message(STATUS "${target} disabled (not supported by the compiler and/or linker)") |
|
endif() |
|
endif() |
|
|
|
if(PYBIND11_LTO_LINKER_FLAGS) |
|
if(CMAKE_VERSION VERSION_LESS 3.11) |
|
set_property( |
|
TARGET ${target} |
|
APPEND |
|
PROPERTY INTERFACE_LINK_LIBRARIES "$<${not_debug}:${PYBIND11_LTO_LINKER_FLAGS}>") |
|
else() |
|
set_property( |
|
TARGET ${target} |
|
APPEND |
|
PROPERTY INTERFACE_LINK_OPTIONS "$<${not_debug}:${PYBIND11_LTO_LINKER_FLAGS}>") |
|
endif() |
|
endif() |
|
endfunction() |
|
|
|
add_library(pybind11::lto IMPORTED INTERFACE ${optional_global}) |
|
_pybind11_generate_lto(pybind11::lto FALSE) |
|
|
|
add_library(pybind11::thin_lto IMPORTED INTERFACE ${optional_global}) |
|
_pybind11_generate_lto(pybind11::thin_lto TRUE) |
|
|
|
# ---------------------- pybind11_strip ----------------------------- |
|
|
|
function(pybind11_strip target_name) |
|
# Strip unnecessary sections of the binary on Linux/macOS |
|
if(CMAKE_STRIP) |
|
if(APPLE) |
|
set(x_opt -x) |
|
endif() |
|
|
|
add_custom_command( |
|
TARGET ${target_name} |
|
POST_BUILD |
|
COMMAND ${CMAKE_STRIP} ${x_opt} $<TARGET_FILE:${target_name}>) |
|
endif() |
|
endfunction()
|
|
|