CMakeLists.txt 10.5 KB
Newer Older
Jason Rhinelander's avatar
Jason Rhinelander committed
1
2
3
4
5
6
7
8
9
10
# 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.

cmake_minimum_required(VERSION 2.8.12)

option(PYBIND11_WERROR  "Report all warnings as errors"  OFF)
11
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
12
13
14
15

if (CMAKE_CURRENT_SOURCE_DIR STREQUAL CMAKE_SOURCE_DIR)
    # 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
16
    project(pybind11_tests CXX)
Jason Rhinelander's avatar
Jason Rhinelander committed
17
18
19
20

    find_package(pybind11 REQUIRED CONFIG)
endif()

Dean Moldovan's avatar
Dean Moldovan committed
21
22
if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES)
  message(STATUS "Setting tests build type to MinSizeRel as none was specified")
23
24
25
  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
26
27
endif()

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

69
# Invoking cmake with something like:
70
#     cmake -DPYBIND11_TEST_OVERRIDE="test_callbacks.cpp;test_picking.cpp" ..
71
72
73
74
75
76
# 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()

77
78
79
80
81
82
83
# 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()

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

86
87
88
89
# 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
90
  test_exceptions.py
91
  test_local_bindings.py
92
  test_stl.py
93
  test_stl_binders.py
94
95
)

96
97
98
99
set(PYBIND11_CROSS_MODULE_GIL_TESTS
  test_gil_scoped.py
)

Henry Schreiner's avatar
Henry Schreiner committed
100
101
option(DOWNLOAD_EIGEN "Download EIGEN (requires CMake 3.11+)" OFF)

102
103
104
105
106
# 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
107
108
109
  # 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
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
  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()
    if (NOT CMAKE_VERSION VERSION_LESS 3.0)
      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)
        endif()
Jason Rhinelander's avatar
Jason Rhinelander committed
139
140
      endif()
    endif()
Henry Schreiner's avatar
Henry Schreiner committed
141
142
143
144
145
    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
146
  endif()
147
148

  if(EIGEN3_FOUND)
Jason Rhinelander's avatar
Jason Rhinelander committed
149
150
151
152
153
154
    # 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()
155
156
157
    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
158
    message(STATUS "Building tests WITHOUT Eigen, use -DDOWNLOAD_EIGEN on CMake 3.11+ to download")
159
  endif()
Dean Moldovan's avatar
Dean Moldovan committed
160
161
endif()

162
163
# Optional dependency for some tests (boost::variant is only supported with version >= 1.56)
find_package(Boost 1.56)
164

Jason Rhinelander's avatar
Jason Rhinelander committed
165
166
167
168
# Compile with compiler warnings turned on
function(pybind11_enable_warnings target_name)
  if(MSVC)
    target_compile_options(${target_name} PRIVATE /W4)
169
  elseif(CMAKE_CXX_COMPILER_ID MATCHES "(GNU|Intel|Clang)")
170
      target_compile_options(${target_name} PRIVATE -Wall -Wextra -Wconversion -Wcast-qual -Wdeprecated)
Jason Rhinelander's avatar
Jason Rhinelander committed
171
172
173
174
175
  endif()

  if(PYBIND11_WERROR)
    if(MSVC)
      target_compile_options(${target_name} PRIVATE /WX)
176
    elseif(CMAKE_CXX_COMPILER_ID MATCHES "(GNU|Intel|Clang)")
Jason Rhinelander's avatar
Jason Rhinelander committed
177
178
179
      target_compile_options(${target_name} PRIVATE -Werror)
    endif()
  endif()
180

181
182
  if(CMAKE_CXX_STANDARD AND CMAKE_CXX_COMPILER_ID MATCHES "Clang" AND PYTHON_VERSION VERSION_LESS 3.0)
    if(CMAKE_CXX_STANDARD LESS 17)
183
      target_compile_options(${target_name} PUBLIC -Wno-deprecated-register)
184
    else()
185
186
187
      target_compile_options(${target_name} PUBLIC -Wno-register)
    endif()
  endif()
Jason Rhinelander's avatar
Jason Rhinelander committed
188
189
endfunction()

190
set(test_targets pybind11_tests)
Jason Rhinelander's avatar
Jason Rhinelander committed
191

192
193
194
195
196
197
198
199
# 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()
200

201
202
203
204
205
206
207
208
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()

209
set(testdir ${CMAKE_CURRENT_SOURCE_DIR})
210
foreach(target ${test_targets})
211
  set(test_files ${PYBIND11_TEST_FILES})
212
  if(NOT target STREQUAL "pybind11_tests")
213
214
    set(test_files "")
  endif()
Dean Moldovan's avatar
Dean Moldovan committed
215

216
  # Create the binding library
217
218
  pybind11_add_module(${target} THIN_LTO ${target}.cpp ${test_files} ${PYBIND11_HEADERS})
  pybind11_enable_warnings(${target})
219
220

  if(MSVC)
221
    target_compile_options(${target} PRIVATE /utf-8)
Jason Rhinelander's avatar
Jason Rhinelander committed
222
  endif()
Dean Moldovan's avatar
Dean Moldovan committed
223

224
225
  if(EIGEN3_FOUND)
    if (PYBIND11_EIGEN_VIA_TARGET)
226
      target_link_libraries(${target} PRIVATE Eigen3::Eigen)
227
    else()
228
      target_include_directories(${target} PRIVATE ${EIGEN3_INCLUDE_DIR})
229
    endif()
230
231
232
233
234
235
    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)
236
  endif()
Dean Moldovan's avatar
Dean Moldovan committed
237

238
239
  # Always write the output file directly into the 'tests' directory (even on MSVC)
  if(NOT CMAKE_LIBRARY_OUTPUT_DIRECTORY)
240
    set_target_properties(${target} PROPERTIES LIBRARY_OUTPUT_DIRECTORY "${testdir}")
241
242
    foreach(config ${CMAKE_CONFIGURATION_TYPES})
      string(TOUPPER ${config} config)
243
      set_target_properties(${target} PROPERTIES LIBRARY_OUTPUT_DIRECTORY_${config} "${testdir}")
244
245
246
    endforeach()
  endif()
endforeach()
Dean Moldovan's avatar
Dean Moldovan committed
247

248
# Make sure pytest is found or produce a fatal error
249
if(NOT PYBIND11_PYTEST_FOUND)
250
251
252
253
254
255
256
257
  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)")
258
  endif()
Wenzel Jakob's avatar
Wenzel Jakob committed
259
  set(PYBIND11_PYTEST_FOUND TRUE CACHE INTERNAL "")
260
261
endif()

262
263
264
265
266
267
if(CMAKE_VERSION VERSION_LESS 3.2)
  set(PYBIND11_USES_TERMINAL "")
else()
  set(PYBIND11_USES_TERMINAL "USES_TERMINAL")
endif()

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

272
273
274
275
276
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
277
278
279
280
281
282
283
284
285
# 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.
if (NOT PROJECT_NAME STREQUAL "pybind11")
  return()
endif()

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

291
292
293
# Test embedding the interpreter. Provides the `cpptest` target.
add_subdirectory(test_embed)

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