CMakeLists.txt 10.7 KB
Newer Older
Jason Rhinelander's avatar
Jason Rhinelander committed
1
2
3
4
5
6
7
# CMakeLists.txt -- Build system for the pybind11 test suite
#
# Copyright (c) 2015 Wenzel Jakob <wenzel@inf.ethz.ch>
#
# All rights reserved. Use of this source code is governed by a
# BSD-style license that can be found in the LICENSE file.

8
9
10
11
12
13
14
15
16
17
18
19
cmake_minimum_required(VERSION 3.7)

# VERSION 3.7...3.18, but some versions of VS have a patched CMake 3.11
# that do not work properly with this syntax, so using the following workaround:
if(${CMAKE_VERSION} VERSION_LESS 3.18)
    cmake_policy(VERSION ${CMAKE_MAJOR_VERSION}.${CMAKE_MINOR_VERSION})
else()
    cmake_policy(VERSION 3.18)
endif()

# There's no harm in including a project in a project
project(pybind11_tests CXX)
Jason Rhinelander's avatar
Jason Rhinelander committed
20
21

option(PYBIND11_WERROR  "Report all warnings as errors"  OFF)
22
option(DOWNLOAD_EIGEN "Download EIGEN (requires CMake 3.11+)" OFF)
23
set(PYBIND11_TEST_OVERRIDE "" CACHE STRING "Tests from ;-separated list of *.cpp files will be built instead of all tests")
Jason Rhinelander's avatar
Jason Rhinelander committed
24

25
if(CMAKE_CURRENT_SOURCE_DIR STREQUAL CMAKE_SOURCE_DIR)
Jason Rhinelander's avatar
Jason Rhinelander committed
26
27
28
29
30
    # We're being loaded directly, i.e. not via add_subdirectory, so make this
    # work as its own project and load the pybind11Config to get the tools we need
    find_package(pybind11 REQUIRED CONFIG)
endif()

Dean Moldovan's avatar
Dean Moldovan committed
31
32
if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES)
  message(STATUS "Setting tests build type to MinSizeRel as none was specified")
33
34
35
  set(CMAKE_BUILD_TYPE MinSizeRel CACHE STRING "Choose the type of build." FORCE)
  set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS "Debug" "Release"
    "MinSizeRel" "RelWithDebInfo")
Dean Moldovan's avatar
Dean Moldovan committed
36
37
endif()

38
# Full set of test files (you can override these; see below)
Dean Moldovan's avatar
Dean Moldovan committed
39
set(PYBIND11_TEST_FILES
40
  test_async.cpp
Dean Moldovan's avatar
Dean Moldovan committed
41
  test_buffers.cpp
42
  test_builtin_casters.cpp
Dean Moldovan's avatar
Dean Moldovan committed
43
  test_call_policies.cpp
Dean Moldovan's avatar
Dean Moldovan committed
44
  test_callbacks.cpp
45
  test_chrono.cpp
46
  test_class.cpp
Dean Moldovan's avatar
Dean Moldovan committed
47
  test_constants_and_functions.cpp
48
  test_copy_move.cpp
49
  test_custom_type_casters.cpp
50
  test_docstring_options.cpp
51
  test_eigen.cpp
52
  test_enum.cpp
Dean Moldovan's avatar
Dean Moldovan committed
53
54
  test_eval.cpp
  test_exceptions.cpp
55
  test_factory_constructors.cpp
56
  test_gil_scoped.cpp
57
  test_iostream.cpp
Dean Moldovan's avatar
Dean Moldovan committed
58
  test_kwargs_and_defaults.cpp
59
  test_local_bindings.cpp
Dean Moldovan's avatar
Dean Moldovan committed
60
61
  test_methods_and_attributes.cpp
  test_modules.cpp
Dean Moldovan's avatar
Dean Moldovan committed
62
  test_multiple_inheritance.cpp
63
  test_numpy_array.cpp
Dean Moldovan's avatar
Dean Moldovan committed
64
65
66
67
68
  test_numpy_dtypes.cpp
  test_numpy_vectorize.cpp
  test_opaque_types.cpp
  test_operator_overloading.cpp
  test_pickling.cpp
69
  test_pytypes.cpp
Dean Moldovan's avatar
Dean Moldovan committed
70
  test_sequences_and_iterators.cpp
Dean Moldovan's avatar
Dean Moldovan committed
71
  test_smart_ptr.cpp
72
  test_stl.cpp
Dean Moldovan's avatar
Dean Moldovan committed
73
  test_stl_binders.cpp
74
  test_tagbased_polymorphic.cpp
75
  test_union.cpp
Dean Moldovan's avatar
Dean Moldovan committed
76
77
78
  test_virtual_functions.cpp
)

79
# Invoking cmake with something like:
80
#     cmake -DPYBIND11_TEST_OVERRIDE="test_callbacks.cpp;test_picking.cpp" ..
81
82
83
84
85
86
# lets you override the tests that get compiled and run.  You can restore to all tests with:
#     cmake -DPYBIND11_TEST_OVERRIDE= ..
if (PYBIND11_TEST_OVERRIDE)
  set(PYBIND11_TEST_FILES ${PYBIND11_TEST_OVERRIDE})
endif()

87
88
89
90
91
92
93
# Skip test_async for Python < 3.5
list(FIND PYBIND11_TEST_FILES test_async.cpp PYBIND11_TEST_FILES_ASYNC_I)
if((PYBIND11_TEST_FILES_ASYNC_I GREATER -1) AND ("${PYTHON_VERSION_MAJOR}.${PYTHON_VERSION_MINOR}" VERSION_LESS 3.5))
  message(STATUS "Skipping test_async because Python version ${PYTHON_VERSION_MAJOR}.${PYTHON_VERSION_MINOR} < 3.5")
  list(REMOVE_AT PYBIND11_TEST_FILES ${PYBIND11_TEST_FILES_ASYNC_I})
endif()

94
string(REPLACE ".cpp" ".py" PYBIND11_PYTEST_FILES "${PYBIND11_TEST_FILES}")
Dean Moldovan's avatar
Dean Moldovan committed
95

96
97
98
99
# Contains the set of test files that require pybind11_cross_module_tests to be
# built; if none of these are built (i.e. because TEST_OVERRIDE is used and
# doesn't include them) the second module doesn't get built.
set(PYBIND11_CROSS_MODULE_TESTS
100
  test_exceptions.py
101
  test_local_bindings.py
102
  test_stl.py
103
  test_stl_binders.py
104
105
)

106
107
108
109
set(PYBIND11_CROSS_MODULE_GIL_TESTS
  test_gil_scoped.py
)

110
111
112
113
114
# Check if Eigen is available; if not, remove from PYBIND11_TEST_FILES (but
# keep it in PYBIND11_PYTEST_FILES, so that we get the "eigen is not installed"
# skip message).
list(FIND PYBIND11_TEST_FILES test_eigen.cpp PYBIND11_TEST_FILES_EIGEN_I)
if(PYBIND11_TEST_FILES_EIGEN_I GREATER -1)
Jason Rhinelander's avatar
Jason Rhinelander committed
115
116
117
  # Try loading via newer Eigen's Eigen3Config first (bypassing tools/FindEigen3.cmake).
  # Eigen 3.3.1+ exports a cmake 3.0+ target for handling dependency requirements, but also
  # produces a fatal error if loaded from a pre-3.0 cmake.
Henry Schreiner's avatar
Henry Schreiner committed
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
  if(DOWNLOAD_EIGEN)
    if(CMAKE_VERSION VERSION_LESS 3.11)
      message(FATAL_ERROR "CMake 3.11+ required when using DOWNLOAD_EIGEN")
    endif()

    set(EIGEN3_VERSION_STRING "3.3.7")

    include(FetchContent)
    FetchContent_Declare(
      eigen
      GIT_REPOSITORY https://gitlab.com/libeigen/eigen.git
      GIT_TAG ${EIGEN3_VERSION_STRING}
    )

    FetchContent_GetProperties(eigen)
    if(NOT eigen_POPULATED)
      message(STATUS "Downloading Eigen")
      FetchContent_Populate(eigen)
    endif()

    set(EIGEN3_INCLUDE_DIR ${eigen_SOURCE_DIR})
    set(EIGEN3_FOUND TRUE)
  else()
141
142
143
144
    find_package(Eigen3 3.2.7 QUIET CONFIG)
    if (EIGEN3_FOUND)
      if (EIGEN3_VERSION_STRING AND NOT EIGEN3_VERSION_STRING VERSION_LESS 3.3.1)
        set(PYBIND11_EIGEN_VIA_TARGET TRUE)
Jason Rhinelander's avatar
Jason Rhinelander committed
145
146
      endif()
    endif()
147

Henry Schreiner's avatar
Henry Schreiner committed
148
149
150
151
152
    if (NOT EIGEN3_FOUND)
      # Couldn't load via target, so fall back to allowing module mode finding, which will pick up
      # tools/FindEigen3.cmake
      find_package(Eigen3 3.2.7 QUIET)
    endif()
Jason Rhinelander's avatar
Jason Rhinelander committed
153
  endif()
154
155

  if(EIGEN3_FOUND)
Jason Rhinelander's avatar
Jason Rhinelander committed
156
157
158
159
160
161
    # Eigen 3.3.1+ cmake sets EIGEN3_VERSION_STRING (and hard codes the version when installed
    # rather than looking it up in the cmake script); older versions, and the
    # tools/FindEigen3.cmake, set EIGEN3_VERSION instead.
    if(NOT EIGEN3_VERSION AND EIGEN3_VERSION_STRING)
      set(EIGEN3_VERSION ${EIGEN3_VERSION_STRING})
    endif()
162
163
164
    message(STATUS "Building tests with Eigen v${EIGEN3_VERSION}")
  else()
    list(REMOVE_AT PYBIND11_TEST_FILES ${PYBIND11_TEST_FILES_EIGEN_I})
Henry Schreiner's avatar
Henry Schreiner committed
165
    message(STATUS "Building tests WITHOUT Eigen, use -DDOWNLOAD_EIGEN on CMake 3.11+ to download")
166
  endif()
Dean Moldovan's avatar
Dean Moldovan committed
167
168
endif()

169
170
# Optional dependency for some tests (boost::variant is only supported with version >= 1.56)
find_package(Boost 1.56)
171

Jason Rhinelander's avatar
Jason Rhinelander committed
172
173
174
175
# Compile with compiler warnings turned on
function(pybind11_enable_warnings target_name)
  if(MSVC)
    target_compile_options(${target_name} PRIVATE /W4)
176
  elseif(CMAKE_CXX_COMPILER_ID MATCHES "(GNU|Intel|Clang)")
177
      target_compile_options(${target_name} PRIVATE -Wall -Wextra -Wconversion -Wcast-qual -Wdeprecated)
Jason Rhinelander's avatar
Jason Rhinelander committed
178
179
180
181
182
  endif()

  if(PYBIND11_WERROR)
    if(MSVC)
      target_compile_options(${target_name} PRIVATE /WX)
183
    elseif(CMAKE_CXX_COMPILER_ID MATCHES "(GNU|Intel|Clang)")
Jason Rhinelander's avatar
Jason Rhinelander committed
184
185
186
      target_compile_options(${target_name} PRIVATE -Werror)
    endif()
  endif()
187

188
  # Needs to be readded since the ordering requires these to be after the ones above
189
190
  if(CMAKE_CXX_STANDARD AND CMAKE_CXX_COMPILER_ID MATCHES "Clang" AND PYTHON_VERSION VERSION_LESS 3.0)
    if(CMAKE_CXX_STANDARD LESS 17)
191
      target_compile_options(${target_name} PUBLIC -Wno-deprecated-register)
192
    else()
193
194
195
      target_compile_options(${target_name} PUBLIC -Wno-register)
    endif()
  endif()
Jason Rhinelander's avatar
Jason Rhinelander committed
196
197
endfunction()

198
set(test_targets pybind11_tests)
Jason Rhinelander's avatar
Jason Rhinelander committed
199

200
201
202
203
204
205
206
207
# Build pybind11_cross_module_tests if any test_whatever.py are being built that require it
foreach(t ${PYBIND11_CROSS_MODULE_TESTS})
  list(FIND PYBIND11_PYTEST_FILES ${t} i)
  if (i GREATER -1)
    list(APPEND test_targets pybind11_cross_module_tests)
    break()
  endif()
endforeach()
208

209
210
211
212
213
214
215
216
foreach(t ${PYBIND11_CROSS_MODULE_GIL_TESTS})
  list(FIND PYBIND11_PYTEST_FILES ${t} i)
  if (i GREATER -1)
    list(APPEND test_targets cross_module_gil_utils)
    break()
  endif()
endforeach()

217
set(testdir ${CMAKE_CURRENT_SOURCE_DIR})
218
foreach(target ${test_targets})
219
  set(test_files ${PYBIND11_TEST_FILES})
220
  if(NOT target STREQUAL "pybind11_tests")
221
222
    set(test_files "")
  endif()
Dean Moldovan's avatar
Dean Moldovan committed
223

224
  # Create the binding library
225
226
  pybind11_add_module(${target} THIN_LTO ${target}.cpp ${test_files} ${PYBIND11_HEADERS})
  pybind11_enable_warnings(${target})
227
228

  if(MSVC)
229
    target_compile_options(${target} PRIVATE /utf-8)
Jason Rhinelander's avatar
Jason Rhinelander committed
230
  endif()
Dean Moldovan's avatar
Dean Moldovan committed
231

232
233
  if(EIGEN3_FOUND)
    if (PYBIND11_EIGEN_VIA_TARGET)
234
      target_link_libraries(${target} PRIVATE Eigen3::Eigen)
235
    else()
236
      target_include_directories(${target} PRIVATE ${EIGEN3_INCLUDE_DIR})
237
    endif()
238
239
240
241
242
243
    target_compile_definitions(${target} PRIVATE -DPYBIND11_TEST_EIGEN)
  endif()

  if(Boost_FOUND)
    target_include_directories(${target} PRIVATE ${Boost_INCLUDE_DIRS})
    target_compile_definitions(${target} PRIVATE -DPYBIND11_TEST_BOOST)
244
  endif()
Dean Moldovan's avatar
Dean Moldovan committed
245

246
247
  # Always write the output file directly into the 'tests' directory (even on MSVC)
  if(NOT CMAKE_LIBRARY_OUTPUT_DIRECTORY)
248
    set_target_properties(${target} PROPERTIES LIBRARY_OUTPUT_DIRECTORY "${testdir}")
249
250
    foreach(config ${CMAKE_CONFIGURATION_TYPES})
      string(TOUPPER ${config} config)
251
      set_target_properties(${target} PROPERTIES LIBRARY_OUTPUT_DIRECTORY_${config} "${testdir}")
252
253
254
    endforeach()
  endif()
endforeach()
Dean Moldovan's avatar
Dean Moldovan committed
255

256
# Make sure pytest is found or produce a fatal error
257
if(NOT PYBIND11_PYTEST_FOUND)
258
259
260
261
262
263
264
265
  execute_process(COMMAND ${PYTHON_EXECUTABLE} -c "import pytest; print(pytest.__version__)"
                  RESULT_VARIABLE pytest_not_found OUTPUT_VARIABLE pytest_version ERROR_QUIET)
  if(pytest_not_found)
    message(FATAL_ERROR "Running the tests requires pytest. Please install it manually"
                        " (try: ${PYTHON_EXECUTABLE} -m pip install pytest)")
  elseif(pytest_version VERSION_LESS 3.0)
    message(FATAL_ERROR "Running the tests requires pytest >= 3.0. Found: ${pytest_version}"
                        "Please update it (try: ${PYTHON_EXECUTABLE} -m pip install -U pytest)")
266
  endif()
Wenzel Jakob's avatar
Wenzel Jakob committed
267
  set(PYBIND11_PYTEST_FOUND TRUE CACHE INTERNAL "")
268
269
endif()

Dean Moldovan's avatar
Dean Moldovan committed
270
# A single command to compile and run the tests
271
add_custom_target(pytest COMMAND ${PYTHON_EXECUTABLE} -m pytest ${PYBIND11_PYTEST_FILES}
272
                  DEPENDS ${test_targets} WORKING_DIRECTORY ${testdir} USES_TERMINAL)
273

274
275
276
277
278
if(PYBIND11_TEST_OVERRIDE)
  add_custom_command(TARGET pytest POST_BUILD
    COMMAND ${CMAKE_COMMAND} -E echo "Note: not all tests run: -DPYBIND11_TEST_OVERRIDE is in effect")
endif()

Jason Rhinelander's avatar
Jason Rhinelander committed
279
280
281
282
283
# Add a check target to run all the tests, starting with pytest (we add dependencies to this below)
add_custom_target(check DEPENDS pytest)

# The remaining tests only apply when being built as part of the pybind11 project, but not if the
# tests are being built independently.
284
if(CMAKE_CURRENT_SOURCE_DIR STREQUAL CMAKE_SOURCE_DIR)
Jason Rhinelander's avatar
Jason Rhinelander committed
285
286
287
  return()
endif()

288
# Add a post-build comment to show the primary test suite .so size and, if a previous size, compare it:
289
290
291
292
293
294
295
296
add_custom_command(
  TARGET
    pybind11_tests
  POST_BUILD COMMAND
    ${PYTHON_EXECUTABLE}
    ${CMAKE_CURRENT_SOURCE_DIR}/../tools/libsize.py
    $<TARGET_FILE:pybind11_tests>
    ${CMAKE_CURRENT_BINARY_DIR}/sosize-$<TARGET_FILE_NAME:pybind11_tests>.txt)
297

298
299
300
# Test embedding the interpreter. Provides the `cpptest` target.
add_subdirectory(test_embed)

301
# Test CMake build using functions and targets from subdirectory or installed location
302
add_subdirectory(test_cmake_build)