From 726a97f977a4bf7381c36c5c5c383fd3327f18d8 Mon Sep 17 00:00:00 2001 From: hbruintjes Date: Fri, 1 Jul 2016 11:45:34 +0200 Subject: [PATCH] Update pybind to latest Former-commit-id: 52f76bea6e10036b7dff577f80a88bfba4b61dc3 --- stormpy/resources/pybind11/.appveyor.yml | 6 +- stormpy/resources/pybind11/.gitignore | 2 + stormpy/resources/pybind11/.travis.yml | 14 +- stormpy/resources/pybind11/CMakeLists.txt | 327 +++++++----------- stormpy/resources/pybind11/README.md | 7 +- stormpy/resources/pybind11/docs/advanced.rst | 115 ++++-- stormpy/resources/pybind11/docs/basics.rst | 33 +- stormpy/resources/pybind11/docs/changelog.rst | 43 ++- stormpy/resources/pybind11/docs/classes.rst | 19 +- stormpy/resources/pybind11/docs/compiling.rst | 162 ++------- stormpy/resources/pybind11/docs/conf.py | 4 +- stormpy/resources/pybind11/docs/faq.rst | 21 +- .../resources/pybind11/docs/limitations.rst | 6 +- stormpy/resources/pybind11/docs/release.rst | 8 +- .../resources/pybind11/example/example10.cpp | 2 +- .../resources/pybind11/example/example11.cpp | 4 + .../resources/pybind11/example/example11.py | 7 +- .../resources/pybind11/example/example11.ref | 13 + .../resources/pybind11/example/example12.cpp | 12 +- .../resources/pybind11/example/example17.cpp | 1 + .../resources/pybind11/example/example17.py | 10 +- .../resources/pybind11/example/example17.ref | 1 + .../resources/pybind11/example/example2.cpp | 2 +- .../resources/pybind11/example/example4.cpp | 2 +- .../resources/pybind11/example/example5.cpp | 6 + .../resources/pybind11/example/example5.py | 3 + .../resources/pybind11/example/example5.ref | 1 + .../resources/pybind11/example/example6.cpp | 10 +- stormpy/resources/pybind11/example/issues.cpp | 34 +- stormpy/resources/pybind11/example/issues.py | 17 + stormpy/resources/pybind11/example/issues.ref | 6 + .../resources/pybind11/example/run_test.py | 11 +- .../pybind11/include/pybind11/attr.h | 34 +- .../pybind11/include/pybind11/cast.h | 24 +- .../pybind11/include/pybind11/common.h | 23 +- .../pybind11/include/pybind11/descr.h | 2 +- .../pybind11/include/pybind11/eigen.h | 21 +- .../pybind11/include/pybind11/numpy.h | 14 +- .../pybind11/include/pybind11/pybind11.h | 118 +++++-- .../pybind11/include/pybind11/pytypes.h | 34 +- .../pybind11/include/pybind11/stl_bind.h | 34 +- .../resources/pybind11/pybind11/_version.py | 2 +- 42 files changed, 676 insertions(+), 539 deletions(-) diff --git a/stormpy/resources/pybind11/.appveyor.yml b/stormpy/resources/pybind11/.appveyor.yml index 5e283af1e..80e02f4f4 100644 --- a/stormpy/resources/pybind11/.appveyor.yml +++ b/stormpy/resources/pybind11/.appveyor.yml @@ -19,8 +19,6 @@ install: build_script: - echo Running cmake... - cd c:\projects\pybind11 - - cmake -G "%CMAKE_PLATFORM%" -DPYTHON_INCLUDE_DIR:PATH=%PYTHON_DIR%/include -DPYTHON_LIBRARY:FILEPATH=%PYTHON_DIR%/libs/python34.lib -DPYTHON_EXECUTABLE:FILEPATH=%PYTHON_DIR%/python.exe + - cmake -G "%CMAKE_PLATFORM%" -DPYTHON_EXECUTABLE:FILEPATH=%PYTHON_DIR%/python.exe - set MSBuildLogger="C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" - - set MSBuildOptions=/v:m /p:Configuration=%Configuration% /logger:%MSBuildLogger% - - msbuild %MSBuildOptions% pybind11.sln - - ctest -C %Configuration% + - cmake --build . --config %Configuration% --target check -- /v:m /logger:%MSBuildLogger% diff --git a/stormpy/resources/pybind11/.gitignore b/stormpy/resources/pybind11/.gitignore index a7a49560d..16701940f 100644 --- a/stormpy/resources/pybind11/.gitignore +++ b/stormpy/resources/pybind11/.gitignore @@ -4,7 +4,9 @@ Makefile cmake_install.cmake .DS_Store /example/example.so +/example/example.cpython*.so /example/example.pyd +/example/example*.dll *.sln *.sdf *.opensdf diff --git a/stormpy/resources/pybind11/.travis.yml b/stormpy/resources/pybind11/.travis.yml index e52319742..3f9cb501c 100644 --- a/stormpy/resources/pybind11/.travis.yml +++ b/stormpy/resources/pybind11/.travis.yml @@ -9,6 +9,7 @@ addons: sources: - ubuntu-toolchain-r-test - deadsnakes + - kubuntu-backports # cmake 2.8.12 packages: - g++-4.8 - g++-4.8-multilib @@ -17,23 +18,24 @@ addons: - python3.5-dev - python3.5-venv - python3.5-dev:i386 + - cmake matrix: include: - os: linux compiler: gcc-4.8 - script: + install: - pyvenv-3.5 venv - - cmake -DPYBIND11_PYTHON_VERSION=3.5 -DPYTHON_INCLUDE_DIR:PATH=/usr/include/python3.5m -DPYTHON_LIBRARY:FILEPATH=/usr/lib/x86_64-linux-gnu/libpython3.5m.so -DPYTHON_EXECUTABLE:FILEPATH=`pwd`/venv/bin/python3.5 -DCMAKE_CXX_COMPILER=g++-4.8 - - make -j 2 - source venv/bin/activate + - pip install -U pip wheel - pip install numpy - - CTEST_OUTPUT_ON_FAILURE=TRUE make test + script: + - CXX=g++-4.8 cmake -DPYBIND11_PYTHON_VERSION=3.5 + - CTEST_OUTPUT_ON_FAILURE=TRUE make check -j 2 - os: osx compiler: clang script: - cmake -DPYBIND11_PYTHON_VERSION=2.7 - - make -j 2 - - CTEST_OUTPUT_ON_FAILURE=TRUE make test + - CTEST_OUTPUT_ON_FAILURE=TRUE make check -j 2 #- os: linux #compiler: gcc-4.8 #script: diff --git a/stormpy/resources/pybind11/CMakeLists.txt b/stormpy/resources/pybind11/CMakeLists.txt index 3f10bd375..7d2c991ab 100644 --- a/stormpy/resources/pybind11/CMakeLists.txt +++ b/stormpy/resources/pybind11/CMakeLists.txt @@ -5,235 +5,166 @@ # 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) +cmake_minimum_required(VERSION 2.8.12) project(pybind11) -option(PYBIND11_INSTALL "Install pybind11 header files?" ON) +# Check if pybind11 is being used directly or via add_subdirectory +set(PYBIND11_MASTER_PROJECT OFF) +if (CMAKE_CURRENT_SOURCE_DIR STREQUAL CMAKE_SOURCE_DIR) + set(PYBIND11_MASTER_PROJECT ON) +endif() + +option(PYBIND11_INSTALL "Install pybind11 header files?" ${PYBIND11_MASTER_PROJECT}) +option(PYBIND11_TEST "Build pybind11 test suite?" ${PYBIND11_MASTER_PROJECT}) # Add a CMake parameter for choosing a desired Python version set(PYBIND11_PYTHON_VERSION "" CACHE STRING "Python version to use for compiling the example application") -include(CheckCXXCompilerFlag) - -# Set a default build configuration if none is specified. 'MinSizeRel' produces the smallest binaries -if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) - message(STATUS "Setting build type to 'MinSizeRel' as none was specified.") - 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") -endif() -string(TOUPPER "${CMAKE_BUILD_TYPE}" U_CMAKE_BUILD_TYPE) - +list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_LIST_DIR}/tools") set(Python_ADDITIONAL_VERSIONS 3.4 3.5 3.6 3.7) -if (NOT ${PYBIND11_PYTHON_VERSION} STREQUAL "") - find_package(PythonLibs ${PYBIND11_PYTHON_VERSION} EXACT) - if (NOT PYTHONLIBS_FOUND) - find_package(PythonLibs ${PYBIND11_PYTHON_VERSION} REQUIRED) - endif() -else() - find_package(PythonLibs REQUIRED) -endif() -# The above sometimes returns version numbers like "3.4.3+"; the "+" must be removed for the next line to work -string(REPLACE "+" "" PYTHONLIBS_VERSION_STRING "+${PYTHONLIBS_VERSION_STRING}") -find_package(PythonInterp ${PYTHONLIBS_VERSION_STRING} EXACT REQUIRED) +find_package(PythonLibsNew ${PYBIND11_PYTHON_VERSION} REQUIRED) + +include(CheckCXXCompilerFlag) -if (CMAKE_CXX_COMPILER_ID MATCHES "Clang" OR CMAKE_CXX_COMPILER_ID MATCHES "GNU" OR CMAKE_CXX_COMPILER_ID MATCHES "Intel") - CHECK_CXX_COMPILER_FLAG("-std=c++14" HAS_CPP14_FLAG) - CHECK_CXX_COMPILER_FLAG("-std=c++11" HAS_CPP11_FLAG) +if(NOT MSVC AND NOT PYBIND11_CPP_STANDARD) + check_cxx_compiler_flag("-std=c++14" HAS_CPP14_FLAG) + check_cxx_compiler_flag("-std=c++11" HAS_CPP11_FLAG) if (HAS_CPP14_FLAG) - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++14") + set(PYBIND11_CPP_STANDARD -std=c++14) elseif (HAS_CPP11_FLAG) - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11") + set(PYBIND11_CPP_STANDARD -std=c++11) else() message(FATAL_ERROR "Unsupported compiler -- pybind11 requires C++11 support!") endif() - # Enable link time optimization and set the default symbol - # visibility to hidden (very important to obtain small binaries) - if (NOT ${U_CMAKE_BUILD_TYPE} MATCHES DEBUG) - # Default symbol visibility - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fvisibility=hidden") - - # Check for Link Time Optimization support - # (GCC/Clang) - CHECK_CXX_COMPILER_FLAG("-flto" HAS_LTO_FLAG) - if (HAS_LTO_FLAG) - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -flto") - endif() - - # Intel equivalent to LTO is called IPO - if (CMAKE_CXX_COMPILER_ID MATCHES "Intel") - CHECK_CXX_COMPILER_FLAG("-ipo" HAS_IPO_FLAG) - if (HAS_IPO_FLAG) - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -ipo") - endif() - endif() - endif() + set(PYBIND11_CPP_STANDARD ${PYBIND11_CPP_STANDARD} CACHE STRING + "C++ standard flag, e.g. -std=c++11 or -std=c++14. Defaults to latest available.") endif() -# Compile with compiler warnings turned on -if(MSVC) - if(CMAKE_CXX_FLAGS MATCHES "/W[0-4]") - string(REGEX REPLACE "/W[0-4]" "/W4" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - else() - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /W4") +# Cache variables so pybind11_add_module can be used in parent projects +set(PYBIND11_INCLUDE_DIR "${CMAKE_CURRENT_LIST_DIR}/include" CACHE INTERNAL "") +set(PYTHON_INCLUDE_DIRS ${PYTHON_INCLUDE_DIRS} CACHE INTERNAL "") +set(PYTHON_LIBRARIES ${PYTHON_LIBRARIES} CACHE INTERNAL "") +set(PYTHON_MODULE_PREFIX ${PYTHON_MODULE_PREFIX} CACHE INTERNAL "") +set(PYTHON_MODULE_EXTENSION ${PYTHON_MODULE_EXTENSION} CACHE INTERNAL "") + +# Build a Python extension module: +# pybind11_add_module( source1 [source2 ...]) +# +function(pybind11_add_module target_name) + add_library(${target_name} MODULE ${ARGN}) + target_include_directories(${target_name} PUBLIC ${PYBIND11_INCLUDE_DIR} ${PYTHON_INCLUDE_DIRS}) + + # The prefix and extension are provided by FindPythonLibsNew.cmake + set_target_properties(${target_name} PROPERTIES PREFIX "${PYTHON_MODULE_PREFIX}") + set_target_properties(${target_name} PROPERTIES SUFFIX "${PYTHON_MODULE_EXTENSION}") + + if(WIN32 OR CYGWIN) + # Link against the Python shared library on Windows + target_link_libraries(${target_name} PRIVATE ${PYTHON_LIBRARIES}) + elseif(APPLE) + # It's quite common to have multiple copies of the same Python version + # installed on one's system. E.g.: one copy from the OS and another copy + # that's statically linked into an application like Blender or Maya. + # If we link our plugin library against the OS Python here and import it + # into Blender or Maya later on, this will cause segfaults when multiple + # conflicting Python instances are active at the same time (even when they + # are of the same version). + + # Windows is not affected by this issue since it handles DLL imports + # differently. The solution for Linux and Mac OS is simple: we just don't + # link against the Python library. The resulting shared library will have + # missing symbols, but that's perfectly fine -- they will be resolved at + # import time. + + target_link_libraries(${target_name} PRIVATE "-undefined dynamic_lookup") endif() -elseif ("${CMAKE_CXX_COMPILER_ID}" MATCHES "Clang" OR "${CMAKE_CXX_COMPILER_ID}" MATCHES "GNU") - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall -Wextra") -endif() + if(NOT MSVC) + # Make sure C++11/14 are enabled + target_compile_options(${target_name} PUBLIC ${PYBIND11_CPP_STANDARD}) -# Check if Eigen is available -set(CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/tools") -find_package(Eigen3 QUIET) - -# Include path for pybind11 header files -include_directories(include) - -# Include path for Python header files -include_directories(${PYTHON_INCLUDE_DIR}) - -set(PYBIND11_HEADERS - include/pybind11/attr.h - include/pybind11/cast.h - include/pybind11/common.h - include/pybind11/complex.h - include/pybind11/descr.h - include/pybind11/eigen.h - include/pybind11/functional.h - include/pybind11/numpy.h - include/pybind11/operators.h - include/pybind11/pybind11.h - include/pybind11/pytypes.h - include/pybind11/stl.h - include/pybind11/stl_bind.h - include/pybind11/typeid.h -) - -set(PYBIND11_EXAMPLES - example/example1.cpp - example/example2.cpp - example/example3.cpp - example/example4.cpp - example/example5.cpp - example/example6.cpp - example/example7.cpp - example/example8.cpp - example/example9.cpp - example/example10.cpp - example/example11.cpp - example/example12.cpp - example/example13.cpp - example/example14.cpp - example/example15.cpp - example/example16.cpp - example/example17.cpp - example/issues.cpp -) - -if (EIGEN3_FOUND) - include_directories(${EIGEN3_INCLUDE_DIR}) - list(APPEND PYBIND11_EXAMPLES example/eigen.cpp) - add_definitions(-DPYBIND11_TEST_EIGEN) - message(STATUS "Building Eigen testcase") -else() - message(STATUS "NOT Building Eigen testcase") -endif() + # Enable link time optimization and set the default symbol + # visibility to hidden (very important to obtain small binaries) + string(TOUPPER "${CMAKE_BUILD_TYPE}" U_CMAKE_BUILD_TYPE) + if (NOT ${U_CMAKE_BUILD_TYPE} MATCHES DEBUG) + # Check for Link Time Optimization support (GCC/Clang) + check_cxx_compiler_flag("-flto" HAS_LTO_FLAG) + if(HAS_LTO_FLAG AND NOT CYGWIN) + target_compile_options(${target_name} PRIVATE -flto) + endif() -# Create the binding library -add_library(example SHARED - ${PYBIND11_HEADERS} - example/example.cpp - ${PYBIND11_EXAMPLES} -) - -# Don't add a 'lib' prefix to the shared library -set_target_properties(example PROPERTIES PREFIX "") - -# Always write the output file directly into the 'example' directory (even on MSVC) -set(CompilerFlags - LIBRARY_OUTPUT_DIRECTORY LIBRARY_OUTPUT_DIRECTORY_RELEASE LIBRARY_OUTPUT_DIRECTORY_DEBUG - LIBRARY_OUTPUT_DIRECTORY_MINSIZEREL LIBRARY_OUTPUT_DIRECTORY_RELWITHDEBINFO - RUNTIME_OUTPUT_DIRECTORY RUNTIME_OUTPUT_DIRECTORY_RELEASE RUNTIME_OUTPUT_DIRECTORY_DEBUG - RUNTIME_OUTPUT_DIRECTORY_MINSIZEREL RUNTIME_OUTPUT_DIRECTORY_RELWITHDEBINFO) - -foreach(CompilerFlag ${CompilerFlags}) - set_target_properties(example PROPERTIES ${CompilerFlag} ${PROJECT_SOURCE_DIR}/example) -endforeach() - -if (WIN32) - if (MSVC) + # Intel equivalent to LTO is called IPO + if(CMAKE_CXX_COMPILER_ID MATCHES "Intel") + check_cxx_compiler_flag("-ipo" HAS_IPO_FLAG) + if(HAS_IPO_FLAG) + target_compile_options(${target_name} PRIVATE -ipo) + endif() + endif() + + # Default symbol visibility + target_compile_options(${target_name} PRIVATE "-fvisibility=hidden") + + # Strip unnecessary sections of the binary on Linux/Mac OS + if(CMAKE_STRIP) + if(APPLE) + add_custom_command(TARGET ${target_name} POST_BUILD + COMMAND ${CMAKE_STRIP} -u -r $) + else() + add_custom_command(TARGET ${target_name} POST_BUILD + COMMAND ${CMAKE_STRIP} $) + endif() + endif() + endif() + elseif(MSVC) # /MP enables multithreaded builds (relevant when there are many files), /bigobj is # needed for bigger binding projects due to the limit to 64k addressable sections - set_property(TARGET example APPEND PROPERTY COMPILE_OPTIONS /MP /bigobj) - # Enforce size-based optimization and link time code generation on MSVC - # (~30% smaller binaries in experiments); do nothing in debug mode. - set_property(TARGET example APPEND PROPERTY COMPILE_OPTIONS - "$<$:/Os>" "$<$:/GL>" - "$<$:/Os>" "$<$:/GL>" - "$<$:/Os>" "$<$:/GL>" - ) - set_property(TARGET example APPEND_STRING PROPERTY LINK_FLAGS_RELEASE "/LTCG ") - set_property(TARGET example APPEND_STRING PROPERTY LINK_FLAGS_MINSIZEREL "/LTCG ") - set_property(TARGET example APPEND_STRING PROPERTY LINK_FLAGS_RELWITHDEBINFO "/LTCG ") - endif() + target_compile_options(${target_name} PRIVATE /MP /bigobj) - # .PYD file extension on Windows - set_target_properties(example PROPERTIES SUFFIX ".pyd") - - # Link against the Python shared library - target_link_libraries(example ${PYTHON_LIBRARY}) -elseif (UNIX) - # It's quite common to have multiple copies of the same Python version - # installed on one's system. E.g.: one copy from the OS and another copy - # that's statically linked into an application like Blender or Maya. - # If we link our plugin library against the OS Python here and import it - # into Blender or Maya later on, this will cause segfaults when multiple - # conflicting Python instances are active at the same time (even when they - # are of the same version). - - # Windows is not affected by this issue since it handles DLL imports - # differently. The solution for Linux and Mac OS is simple: we just don't - # link against the Python library. The resulting shared library will have - # missing symbols, but that's perfectly fine -- they will be resolved at - # import time. - - # .SO file extension on Linux/Mac OS - set_target_properties(example PROPERTIES SUFFIX ".so") - - # Optimize for a small binary size - if (NOT ${U_CMAKE_BUILD_TYPE} MATCHES DEBUG) - set_target_properties(example PROPERTIES COMPILE_FLAGS "-Os") + # Enforce link time code generation on MSVC, except in debug mode + target_compile_options(${target_name} PRIVATE $<$>:/GL>) + + # Fancy generator expressions don't work with linker flags, for reasons unknown + set_property(TARGET ${target_name} APPEND_STRING PROPERTY LINK_FLAGS_RELEASE /LTCG) + set_property(TARGET ${target_name} APPEND_STRING PROPERTY LINK_FLAGS_MINSIZEREL /LTCG) + set_property(TARGET ${target_name} APPEND_STRING PROPERTY LINK_FLAGS_RELWITHDEBINFO /LTCG) endif() +endfunction() - # Strip unnecessary sections of the binary on Linux/Mac OS - if(APPLE) - set_target_properties(example PROPERTIES MACOSX_RPATH ".") - set_target_properties(example PROPERTIES LINK_FLAGS "-undefined dynamic_lookup ") - if (NOT ${U_CMAKE_BUILD_TYPE} MATCHES DEBUG) - add_custom_command(TARGET example POST_BUILD COMMAND strip -u -r ${PROJECT_SOURCE_DIR}/example/example.so) - endif() +# Compile with compiler warnings turned on +function(pybind11_enable_warnings target_name) + if(MSVC) + target_compile_options(${target_name} PRIVATE /W4) else() - if (NOT ${U_CMAKE_BUILD_TYPE} MATCHES DEBUG) - add_custom_command(TARGET example POST_BUILD COMMAND strip ${PROJECT_SOURCE_DIR}/example/example.so) - endif() + target_compile_options(${target_name} PRIVATE -Wall -Wextra -Wconversion) endif() -endif() +endfunction() -enable_testing() - -set(RUN_TEST ${PYTHON_EXECUTABLE} ${CMAKE_CURRENT_SOURCE_DIR}/example/run_test.py) -if (MSVC OR CMAKE_CXX_COMPILER_ID MATCHES "Intel") - set(RUN_TEST ${RUN_TEST} --relaxed) +if (PYBIND11_TEST) + enable_testing() + add_subdirectory(example) + add_custom_target(check COMMAND ${CMAKE_CTEST_COMMAND} -C $ DEPENDS example) endif() -foreach(VALUE ${PYBIND11_EXAMPLES}) - string(REGEX REPLACE "^example/(.+).cpp$" "\\1" EXAMPLE_NAME "${VALUE}") - add_test(NAME ${EXAMPLE_NAME} COMMAND ${RUN_TEST} ${EXAMPLE_NAME}) -endforeach() - if (PYBIND11_INSTALL) - install(FILES ${PYBIND11_HEADERS} DESTINATION include/pybind11) + set(PYBIND11_HEADERS + include/pybind11/attr.h + include/pybind11/cast.h + include/pybind11/common.h + include/pybind11/complex.h + include/pybind11/descr.h + include/pybind11/eigen.h + include/pybind11/functional.h + include/pybind11/numpy.h + include/pybind11/operators.h + include/pybind11/pybind11.h + include/pybind11/pytypes.h + include/pybind11/stl.h + include/pybind11/stl_bind.h + include/pybind11/typeid.h + ) + + install(FILES ${PYBIND11_HEADERS} DESTINATION include/pybind11) endif() diff --git a/stormpy/resources/pybind11/README.md b/stormpy/resources/pybind11/README.md index 376f56295..a313c6f54 100644 --- a/stormpy/resources/pybind11/README.md +++ b/stormpy/resources/pybind11/README.md @@ -90,6 +90,7 @@ In addition to the core functionality, pybind11 provides some extra goodies: 2. GCC (any non-ancient version with C++11 support) 3. Microsoft Visual Studio 2015 or newer 4. Intel C++ compiler v15 or newer +5. Cygwin/GCC (tested on 2.5.1) ## About @@ -101,8 +102,10 @@ Axel Huebl, @hulucc, Sergey Lyskov Johan Mabille, -Tomasz Miąsko, and -Ben Pritchard. +Tomasz Miąsko, +Dean Moldovan, +Ben Pritchard, and +Boris Schäling. ### License diff --git a/stormpy/resources/pybind11/docs/advanced.rst b/stormpy/resources/pybind11/docs/advanced.rst index 032cf8593..c257fa2d4 100644 --- a/stormpy/resources/pybind11/docs/advanced.rst +++ b/stormpy/resources/pybind11/docs/advanced.rst @@ -139,8 +139,19 @@ its return value upon execution. }; } +This example demonstrates using python named parameters in C++ callbacks which +requires using ``py::cpp_function`` as a wrapper. Usage is similar to defining +methods of classes: + +.. code-block:: cpp + + py::cpp_function func_cpp() { + return py::cpp_function([](int i) { return i+1; }, + py::arg("number")); + } + After including the extra header file :file:`pybind11/functional.h`, it is almost -trivial to generate binding code for both of these functions. +trivial to generate binding code for all of these functions. .. code-block:: cpp @@ -151,13 +162,14 @@ trivial to generate binding code for both of these functions. m.def("func_arg", &func_arg); m.def("func_ret", &func_ret); + m.def("func_cpp", &func_cpp); return m.ptr(); } The following interactive session shows how to call them from Python. -.. code-block:: python +.. code-block:: pycon $ python >>> import example @@ -169,7 +181,9 @@ The following interactive session shows how to call them from Python. >>> square_plus_1 = example.func_ret(square) >>> square_plus_1(4) 17L - >>> + >>> plus_1 = func_cpp() + >>> plus_1(number=43) + 44L .. note:: @@ -283,9 +297,8 @@ The binding code also needs a few minor adaptations (highlighted): PYBIND11_PLUGIN(example) { py::module m("example", "pybind11 example plugin"); - py::class_ animal(m, "Animal"); + py::class_, PyAnimal /* <--- trampoline*/> animal(m, "Animal"); animal - .alias() .def(py::init<>()) .def("go", &Animal::go); @@ -297,15 +310,15 @@ The binding code also needs a few minor adaptations (highlighted): return m.ptr(); } -Importantly, the trampoline helper class is used as the template argument to -:class:`class_`, and a call to :func:`class_::alias` informs the binding -generator that this is merely an alias for the underlying type ``Animal``. -Following this, we are able to define a constructor as usual. +Importantly, pybind11 is made aware of the trampoline trampoline helper class +by specifying it as the *third* template argument to :class:`class_`. The +second argument with the unique pointer is simply the default holder type used +by pybind11. Following this, we are able to define a constructor as usual. The Python session below shows how to override ``Animal::go`` and invoke it via a virtual method call. -.. code-block:: python +.. code-block:: pycon >>> from example import * >>> d = Dog() @@ -319,14 +332,7 @@ a virtual method call. >>> call_go(c) u'meow! meow! meow! ' -.. warning:: - - Both :func:`PYBIND11_OVERLOAD` and :func:`PYBIND11_OVERLOAD_PURE` are - macros, which means that they can get confused by commas in a template - argument such as ``PYBIND11_OVERLOAD(MyReturnValue, myFunc)``. In - this case, the preprocessor assumes that the comma indicates the beginnning - of the next parameter. Use a ``typedef`` to bind the template to another - name and use it in the macro to avoid this problem. +Please take a look at the :ref:`macro_notes` before using this feature. .. seealso:: @@ -335,6 +341,21 @@ a virtual method call. detail. +.. _macro_notes: + +General notes regarding convenience macros +========================================== + +pybind11 provides a few convenience macros such as +:func:`PYBIND11_MAKE_OPAQUE` and :func:`PYBIND11_DECLARE_HOLDER_TYPE`, and +``PYBIND11_OVERLOAD_*``. Since these are "just" macros that are evaluated +in the preprocessor (which has no concept of types), they *will* get confused +by commas in a template argument such as ``PYBIND11_OVERLOAD(MyReturnValue, myFunc)``. In this case, the preprocessor assumes that the comma indicates +the beginnning of the next parameter. Use a ``typedef`` to bind the template to +another name and use it in the macro to avoid this problem. + + Global Interpreter Lock (GIL) ============================= @@ -369,9 +390,8 @@ be realized as follows (important changes highlighted): PYBIND11_PLUGIN(example) { py::module m("example", "pybind11 example plugin"); - py::class_ animal(m, "Animal"); + py::class_, PyAnimal> animal(m, "Animal"); animal - .alias() .def(py::init<>()) .def("go", &Animal::go); @@ -441,12 +461,14 @@ functions. The default policy is :enum:`return_value_policy::automatic`. | | See below for a description of what all of these different policies do. | +--------------------------------------------------+----------------------------------------------------------------------------+ | :enum:`return_value_policy::automatic_reference` | As above, but use policy :enum:`return_value_policy::reference` when the | -| | return value is a pointer. You probably won't need to use this. | +| | return value is a pointer. This is the default conversion policy for | +| | function arguments when calling Python functions manually from C++ code | +| | (i.e. via handle::operator()). You probably won't need to use this. | +--------------------------------------------------+----------------------------------------------------------------------------+ | :enum:`return_value_policy::take_ownership` | Reference an existing object (i.e. do not create a new copy) and take | | | ownership. Python will call the destructor and delete operator when the | | | object's reference count reaches zero. Undefined behavior ensues when the | -| | C++ side does the same.. | +| | C++ side does the same. | +--------------------------------------------------+----------------------------------------------------------------------------+ | :enum:`return_value_policy::copy` | Create a new copy of the returned object, which will be owned by Python. | | | This policy is comparably safe because the lifetimes of the two instances | @@ -504,6 +526,14 @@ The following example snippet shows a use case of the non-determinism and segmentation faults, hence it is worth spending the time to understand all the different options in the table above. + It is worth highlighting one common issue where a method (e.g. a getter) + returns a reference (or pointer) to the first attribute of a class. In this + case, the class and attribute will be located at the same address in + memory, which pybind11 will recongnize and return the parent instance + instead of creating a new Python object that represents the attribute. + Here, the :enum:`return_value_policy::reference_internal` policy should be + used rather than relying on the automatic one. + .. note:: The next section on :ref:`call_policies` discusses *call policies* that can be @@ -587,6 +617,31 @@ Python side: py::implicitly_convertible(); +.. note:: + + Implicit conversions from ``A`` to ``B`` only work when ``B`` is a custom + data type that is exposed to Python via pybind11. + +.. _static_properties: + +Static properties +================= + +The section on :ref:`properties` discussed the creation of instance properties +that are implemented in terms of C++ getters and setters. + +Static properties can also be created in a similar way to expose getters and +setters of static class attributes. It is important to note that the implicit +``self`` argument also exists in this case and is used to pass the Python +``type`` subclass instance. This parameter will often not be needed by the C++ +side, and the following example illustrates how to instantiate a lambda getter +function that ignores it: + +.. code-block:: cpp + + py::class_(m, "Foo") + .def_property_readonly_static("foo", [](py::object /* self */) { return Foo(); }); + Unique pointers =============== @@ -723,6 +778,9 @@ There are two ways to resolve this issue: class Child : public std::enable_shared_from_this { }; + +Please take a look at the :ref:`macro_notes` before using this feature. + .. seealso:: The file :file:`example/example8.cpp` contains a complete example that @@ -836,7 +894,7 @@ Suppose we bind the following function and call it from Python, the following happens: -.. code-block:: python +.. code-block:: pycon >>> v = [5, 6] >>> append_1(v) @@ -865,7 +923,7 @@ functions: In this case, properties can be read and written in their entirety. However, an ``append`` operaton involving such a list type has no effect: -.. code-block:: python +.. code-block:: pycon >>> m = MyClass() >>> m.contents = [5, 6] @@ -905,6 +963,7 @@ with a name in Python, and to define a set of available operations: }, py::keep_alive<0, 1>()) /* Keep vector alive while iterator is used */ // .... +Please take a look at the :ref:`macro_notes` before using this feature. .. seealso:: @@ -1146,7 +1205,7 @@ entirely on the C++ side and can be crunched down into a tight, optimized loop by the compiler. The result is returned as a NumPy array of type ``numpy.dtype.float64``. -.. code-block:: python +.. code-block:: pycon >>> x = np.array([[1, 3],[5, 7]]) >>> y = np.array([[2, 4],[6, 8]]) @@ -1318,7 +1377,7 @@ Another aspect worth highlighting is that the "preview" of the default argument in the function signature is generated using the object's ``__repr__`` method. If not available, the signature may not be very helpful, e.g.: -.. code-block:: python +.. code-block:: pycon FUNCTIONS ... @@ -1523,7 +1582,7 @@ Generating documentation using Sphinx Sphinx [#f4]_ has the ability to inspect the signatures and documentation strings in pybind11-based extension modules to automatically generate beautiful -documentation in a variety formats. The pbtest repository [#f5]_ contains a +documentation in a variety formats. The python_example repository [#f5]_ contains a simple example repository which uses this approach. There are two potential gotchas when using this approach: first, make sure that @@ -1551,4 +1610,4 @@ work, it is important that all lines are indented consistently, i.e.: )mydelimiter"); .. [#f4] http://www.sphinx-doc.org -.. [#f5] http://github.com/pybind/pbtest +.. [#f5] http://github.com/pybind/python_example diff --git a/stormpy/resources/pybind11/docs/basics.rst b/stormpy/resources/pybind11/docs/basics.rst index b1765c5c9..1f40f198d 100644 --- a/stormpy/resources/pybind11/docs/basics.rst +++ b/stormpy/resources/pybind11/docs/basics.rst @@ -128,7 +128,7 @@ Assuming that the created file :file:`example.so` (:file:`example.pyd` on Window is located in the current directory, the following interactive Python session shows how to load and execute the example. -.. code-block:: python +.. code-block:: pycon $ python Python 2.7.10 (default, Aug 22 2015, 20:33:39) @@ -157,7 +157,7 @@ metadata into :func:`module::def`. With this modified binding code, we can now call the function using keyword arguments, which is a more readable alternative particularly for functions taking many parameters: -.. code-block:: python +.. code-block:: pycon >>> import example >>> example.add(i=1, j=2) @@ -165,7 +165,7 @@ particularly for functions taking many parameters: The keyword names also appear in the function signatures within the documentation. -.. code-block:: python +.. code-block:: pycon >>> help(example) @@ -177,6 +177,21 @@ The keyword names also appear in the function signatures within the documentatio A function which adds two numbers +A shorter notation for named arguments is also available: + +.. code-block:: cpp + + // regular notation + m.def("add1", &add, py::arg("i"), py::arg("j")); + // shorthand + using namespace pybind11::literals; + m.def("add2", &add, "i"_a, "j"_a); + +The :var:`_a` suffix forms a C++11 literal which is equivalent to :class:`arg`. +Note that the literal operator must first be made visible with the directive +``using namespace pybind11::literals``. This does not bring in anything else +from the ``pybind11`` namespace except for literals. + .. _default_args: Default arguments @@ -201,7 +216,7 @@ using an extension of :class:`arg`: The default values also appear within the documentation. -.. code-block:: python +.. code-block:: pycon >>> help(example) @@ -213,6 +228,15 @@ The default values also appear within the documentation. A function which adds two numbers +The shorthand notation is also available for default arguments: + +.. code-block:: cpp + + // regular notation + m.def("add1", &add, py::arg("i") = 1, py::arg("j") = 2); + // shorthand + m.def("add2", &add, "i"_a=1, "j"_a=2); + .. _supported_types: Supported data types @@ -283,4 +307,3 @@ as arguments and return values, refer to the section on binding :ref:`classes`. .. [#f1] In practice, implementation and binding code will generally be located in separate files. - diff --git a/stormpy/resources/pybind11/docs/changelog.rst b/stormpy/resources/pybind11/docs/changelog.rst index 632f2be84..6560072a6 100644 --- a/stormpy/resources/pybind11/docs/changelog.rst +++ b/stormpy/resources/pybind11/docs/changelog.rst @@ -3,24 +3,61 @@ Changelog ######### -1.8 (Not yet released) +Starting with version 1.8, pybind11 releases use a +[semantic versioning](http://semver.org) policy. + +Breaking changes queued for v2.0.0 (Not yet released) +----------------------------------------------------- +* Redesigned virtual call mechanism and user-facing syntax (see + https://github.com/pybind/pybind11/commit/86d825f3302701d81414ddd3d38bcd09433076bc) + +* Remove ``handle.call()`` method + +1.9.0 (Not yet released) +------------------------ +* Queued changes: ``py::eval*``, map indexing suite, documentation for indexing suites. + +1.8.0 (June 14, 2016) ---------------------- +* Redesigned CMake build system which exports a convenient + ``pybind11_add_module`` function to parent projects. +* ``std::vector<>`` type bindings analogous to Boost.Python's ``indexing_suite`` +* Transparent conversion of sparse and dense Eigen matrices and vectors (``eigen.h``) +* Added an ``ExtraFlags`` template argument to the NumPy ``array_t<>`` wrapper + to disable an enforced cast that may lose precision, e.g. to create overloads + for different precisions and complex vs real-valued matrices. * Prevent implicit conversion of floating point values to integral types in function arguments -* Transparent conversion of sparse and dense Eigen data types * Fixed incorrect default return value policy for functions returning a shared pointer +* Don't allow registering a type via ``class_`` twice * Don't allow casting a ``None`` value into a C++ lvalue reference * Fixed a crash in ``enum_::operator==`` that was triggered by the ``help()`` command * Improved detection of whether or not custom C++ types can be copy/move-constructed * Extended ``str`` type to also work with ``bytes`` instances +* Added a ``"name"_a`` user defined string literal that is equivalent to ``py::arg("name")``. +* When specifying function arguments via ``py::arg``, the test that verifies + the number of arguments now runs at compile time. * Added ``[[noreturn]]`` attribute to ``pybind11_fail()`` to quench some compiler warnings +* List function arguments in exception text when the dispatch code cannot find + a matching overload +* Added ``PYBIND11_OVERLOAD_NAME`` and ``PYBIND11_OVERLOAD_PURE_NAME`` macros which + can be used to override virtual methods whose name differs in C++ and Python + (e.g. ``__call__`` and ``operator()``) * Various minor ``iterator`` and ``make_iterator()`` improvements +* Transparently support ``__bool__`` on Python 2.x and Python 3.x +* Fixed issue with destructor of unpickled object not being called * Minor CMake build system improvements on Windows +* New ``pybind11::args`` and ``pybind11::kwargs`` types to create functions which + take an arbitrary number of arguments and keyword arguments +* New syntax to call a Python function from C++ using ``*args`` and ``*kwargs`` +* The functions ``def_property_*`` now correctly process docstring arguments (these + formerly caused a segmentation fault) * Many ``mkdoc.py`` improvements (enumerations, template arguments, ``DOC()`` macro accepts more arguments) -* Documentation improvements (pickling support, ``keep_alive``) +* Cygwin support +* Documentation improvements (pickling support, ``keep_alive``, macro usage) 1.7 (April 30, 2016) ---------------------- diff --git a/stormpy/resources/pybind11/docs/classes.rst b/stormpy/resources/pybind11/docs/classes.rst index c98f8da2e..5afb21edc 100644 --- a/stormpy/resources/pybind11/docs/classes.rst +++ b/stormpy/resources/pybind11/docs/classes.rst @@ -44,7 +44,7 @@ constructor's parameters as template arguments and wraps the corresponding constructor (see the :ref:`custom_constructors` section for details). An interactive Python session demonstrating this example is shown below: -.. code-block:: python +.. code-block:: pycon % python >>> import example @@ -73,7 +73,7 @@ Binding lambda functions Note how ``print(p)`` produced a rather useless summary of our data structure in the example above: -.. code-block:: python +.. code-block:: pycon >>> print(p) @@ -99,11 +99,13 @@ Lambda function instead: Both stateless [#f1]_ and stateful lambda closures are supported by pybind11. With the above change, the same Python code now produces the following output: -.. code-block:: python +.. code-block:: pycon >>> print(p) +.. _properties: + Instance and static fields ========================== @@ -120,7 +122,7 @@ method also exists for ``const`` fields. This makes it possible to write -.. code-block:: python +.. code-block:: pycon >>> p = example.Pet('Molly') >>> p.name @@ -160,7 +162,8 @@ the setter and getter functions: Similar functions :func:`class_::def_readwrite_static`, :func:`class_::def_readonly_static` :func:`class_::def_property_static`, and :func:`class_::def_property_readonly_static` are provided for binding - static variables and properties. + static variables and properties. Please also see the section on + :ref:`static_properties` in the advanced part of the documentation. .. _inheritance: @@ -212,7 +215,7 @@ Alternatively, we can also assign a name to the previously bound ``Pet`` Functionality-wise, both approaches are completely equivalent. Afterwards, instances will expose fields and methods of both types: -.. code-block:: python +.. code-block:: pycon >>> p = example.Dog('Molly') >>> p.name @@ -253,7 +256,7 @@ sequence. The overload signatures are also visible in the method's docstring: -.. code-block:: python +.. code-block:: pycon >>> help(example.Pet) @@ -319,7 +322,7 @@ constructor. The :func:`enum_::export_values` function exports the enum entries into the parent scope, which should be skipped for newer C++11-style strongly typed enums. -.. code-block:: python +.. code-block:: pycon >>> p = Pet('Lucy', Pet.Cat) >>> p.type diff --git a/stormpy/resources/pybind11/docs/compiling.rst b/stormpy/resources/pybind11/docs/compiling.rst index 5f4d09836..9ec3bc273 100644 --- a/stormpy/resources/pybind11/docs/compiling.rst +++ b/stormpy/resources/pybind11/docs/compiling.rst @@ -7,9 +7,9 @@ Building with setuptools For projects on PyPI, building with setuptools is the way to go. Sylvain Corlay has kindly provided an example project which shows how to set up everything, including automatic generation of documentation using Sphinx. Please refer to -the [pbtest]_ repository. +the [python_example]_ repository. -.. [pbtest] https://github.com/pybind/pbtest +.. [python_example] https://github.com/pybind/python_example Building with cppimport ======================== @@ -26,146 +26,28 @@ Building with cppimport Building with CMake =================== -For C++ codebases that already have an existing CMake-based build system, the -following snippet should be a good starting point to create bindings across -platforms. It assumes that the code is located in a file named -:file:`example.cpp`, and that the pybind11 repository is located in a -subdirectory named :file:`pybind11`. +For C++ codebases that have an existing CMake-based build system, a Python +extension module can be created with just a few lines of code: .. code-block:: cmake - cmake_minimum_required(VERSION 2.8) - + cmake_minimum_required(VERSION 2.8.12) project(example) - # Add a CMake parameter for choosing a desired Python version - set(EXAMPLE_PYTHON_VERSION "" CACHE STRING - "Python version to use for compiling the example library") - - include(CheckCXXCompilerFlag) - - # Set a default build configuration if none is specified. - # 'MinSizeRel' produces the smallest binaries - if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) - message(STATUS "Setting build type to 'MinSizeRel' as none was specified.") - 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") - endif() - string(TOUPPER "${CMAKE_BUILD_TYPE}" U_CMAKE_BUILD_TYPE) - - # Try to autodetect Python (can be overridden manually if needed) - set(Python_ADDITIONAL_VERSIONS 3.4 3.5 3.6 3.7) - if (NOT ${EXAMPLE_PYTHON_VERSION} STREQUAL "") - find_package(PythonLibs ${EXAMPLE_PYTHON_VERSION} EXACT) - if (NOT PYTHONLIBS_FOUND) - find_package(PythonLibs ${EXAMPLE_PYTHON_VERSION} REQUIRED) - endif() - else() - find_package(PythonLibs REQUIRED) - endif() - - # The above sometimes returns version numbers like "3.4.3+"; - # the "+" must be removed for the next lines to work - string(REPLACE "+" "" PYTHONLIBS_VERSION_STRING "+${PYTHONLIBS_VERSION_STRING}") - - # Uncomment the following line if you will also require a matching Python interpreter - # find_package(PythonInterp ${PYTHONLIBS_VERSION_STRING} EXACT REQUIRED) - - if (CMAKE_CXX_COMPILER_ID MATCHES "Clang" OR CMAKE_CXX_COMPILER_ID MATCHES "GNU") - CHECK_CXX_COMPILER_FLAG("-std=c++14" HAS_CPP14_FLAG) - CHECK_CXX_COMPILER_FLAG("-std=c++11" HAS_CPP11_FLAG) - - if (HAS_CPP14_FLAG) - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++14") - elseif (HAS_CPP11_FLAG) - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11") - else() - message(FATAL_ERROR "Unsupported compiler -- at least C++11 support is needed!") - endif() - - # Enable link time optimization and set the default symbol - # visibility to hidden (very important to obtain small binaries) - if (NOT ${U_CMAKE_BUILD_TYPE} MATCHES DEBUG) - # Default symbol visibility - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fvisibility=hidden") - - # Check for Link Time Optimization support - CHECK_CXX_COMPILER_FLAG("-flto" HAS_LTO_FLAG) - if (HAS_LTO_FLAG) - set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -flto") - endif() - endif() - endif() - - # Include path for Python header files - include_directories(${PYTHON_INCLUDE_DIR}) - - # Include path for pybind11 header files -- this may need to be - # changed depending on your setup - include_directories(${PROJECT_SOURCE_DIR}/pybind11/include) - - # Create the binding library - add_library(example SHARED - example.cpp - # ... extra files go here ... - ) - - # Don't add a 'lib' prefix to the shared library - set_target_properties(example PROPERTIES PREFIX "") - - if (WIN32) - if (MSVC) - # /MP enables multithreaded builds (relevant when there are many files), /bigobj is - # needed for bigger binding projects due to the limit to 64k addressable sections - set_property(TARGET example APPEND PROPERTY COMPILE_OPTIONS /MP /bigobj) - # Enforce size-based optimization and link time code generation on MSVC - # (~30% smaller binaries in experiments); do nothing in debug mode. - set_property(TARGET example APPEND PROPERTY COMPILE_OPTIONS - "$<$:/Os>" "$<$:/GL>" - "$<$:/Os>" "$<$:/GL>" - "$<$:/Os>" "$<$:/GL>" - ) - set_property(TARGET example APPEND_STRING PROPERTY LINK_FLAGS_RELEASE "/LTCG ") - set_property(TARGET example APPEND_STRING PROPERTY LINK_FLAGS_MINSIZEREL "/LTCG ") - set_property(TARGET example APPEND_STRING PROPERTY LINK_FLAGS_RELWITHDEBINFO "/LTCG ") - endif() - - # .PYD file extension on Windows - set_target_properties(example PROPERTIES SUFFIX ".pyd") - - # Link against the Python shared library - target_link_libraries(example ${PYTHON_LIBRARY}) - elseif (UNIX) - # It's quite common to have multiple copies of the same Python version - # installed on one's system. E.g.: one copy from the OS and another copy - # that's statically linked into an application like Blender or Maya. - # If we link our plugin library against the OS Python here and import it - # into Blender or Maya later on, this will cause segfaults when multiple - # conflicting Python instances are active at the same time (even when they - # are of the same version). - - # Windows is not affected by this issue since it handles DLL imports - # differently. The solution for Linux and Mac OS is simple: we just don't - # link against the Python library. The resulting shared library will have - # missing symbols, but that's perfectly fine -- they will be resolved at - # import time. - - # .SO file extension on Linux/Mac OS - set_target_properties(example PROPERTIES SUFFIX ".so") - - # Strip unnecessary sections of the binary on Linux/Mac OS - if(APPLE) - set_target_properties(example PROPERTIES MACOSX_RPATH ".") - set_target_properties(example PROPERTIES LINK_FLAGS "-undefined dynamic_lookup ") - if (NOT ${U_CMAKE_BUILD_TYPE} MATCHES DEBUG) - add_custom_command(TARGET example POST_BUILD - COMMAND strip -u -r ${PROJECT_BINARY_DIR}/example.so) - endif() - else() - if (NOT ${U_CMAKE_BUILD_TYPE} MATCHES DEBUG) - add_custom_command(TARGET example POST_BUILD - COMMAND strip ${PROJECT_BINARY_DIR}/example.so) - endif() - endif() - endif() + add_subdirectory(pybind11) + pybind11_add_module(example example.cpp) + +This assumes that the pybind11 repository is located in a subdirectory named +:file:`pybind11` and that the code is located in a file named :file:`example.cpp`. +The CMake command ``add_subdirectory`` will import a function with the signature +``pybind11_add_module( source1 [source2 ...])``. It will take care of all +the details needed to build a Python extension module on any platform. + +The target Python version can be selected by setting the ``PYBIND11_PYTHON_VERSION`` +variable before adding the pybind11 subdirectory. Alternatively, an exact Python +installation can be specified by setting ``PYTHON_EXECUTABLE``. + +A working sample project, including a way to invoke CMake from :file:`setup.py` for +PyPI integration, can be found in the [cmake_example]_ repository. + +.. [cmake_example] https://github.com/pybind/cmake_example diff --git a/stormpy/resources/pybind11/docs/conf.py b/stormpy/resources/pybind11/docs/conf.py index 9d25838d2..4e2c4cef2 100644 --- a/stormpy/resources/pybind11/docs/conf.py +++ b/stormpy/resources/pybind11/docs/conf.py @@ -56,9 +56,9 @@ author = 'Wenzel Jakob' # built documents. # # The short X.Y version. -version = '1.0' +version = '1.9' # The full version, including alpha/beta/rc tags. -release = '1.0' +release = '1.9.dev0' # The language for content autogenerated by Sphinx. Refer to documentation # for a list of supported languages. diff --git a/stormpy/resources/pybind11/docs/faq.rst b/stormpy/resources/pybind11/docs/faq.rst index b6c91a8bd..788b3b893 100644 --- a/stormpy/resources/pybind11/docs/faq.rst +++ b/stormpy/resources/pybind11/docs/faq.rst @@ -24,24 +24,17 @@ The Python interpreter immediately crashes when importing my module See item 2 of the first answer. -CMake doesn't detect the right Python version, or it finds mismatched interpreter and library versions -====================================================================================================== +CMake doesn't detect the right Python version +============================================= -The Python detection logic of CMake is flawed and can sometimes fail to find -the desired Python version, or it chooses mismatched interpreter and library -versions. A longer discussion is available on the pybind11 issue tracker -[#f1]_, though this is ultimately not a pybind11 issue. - -To force the build system to choose a particular version, delete CMakeCache.txt -and then invoke CMake as follows: +The CMake-based build system will try to automatically detect the installed +version of Python and link against that. When this fails, or when there are +multiple versions of Python and it finds the wrong one, delete +``CMakeCache.txt`` and then invoke CMake as follows: .. code-block:: bash - cmake -DPYTHON_EXECUTABLE:FILEPATH=<...> \ - -DPYTHON_LIBRARY:FILEPATH=<...> \ - -DPYTHON_INCLUDE_DIR:PATH=<...> . - -.. [#f1] http://github.com/pybind/pybind11/issues/99 + cmake -DPYTHON_EXECUTABLE:FILEPATH= . Limitations involving reference arguments ========================================= diff --git a/stormpy/resources/pybind11/docs/limitations.rst b/stormpy/resources/pybind11/docs/limitations.rst index 434d09b37..c6100d167 100644 --- a/stormpy/resources/pybind11/docs/limitations.rst +++ b/stormpy/resources/pybind11/docs/limitations.rst @@ -12,7 +12,11 @@ certain limitations: - Multiple inheritance relationships on the C++ side cannot be mapped to Python. -Both of these features could be implemented but would lead to a significant +- The NumPy interface ``pybind11::array`` greatly simplifies accessing + numerical data from C++ (and vice versa), but it's not a full-blown array + class like ``Eigen::Array`` or ``boost.multi_array``. + +All of these features could be implemented but would lead to a significant increase in complexity. I've decided to draw the line here to keep this project simple and compact. Users who absolutely require these features are encouraged to fork pybind11. diff --git a/stormpy/resources/pybind11/docs/release.rst b/stormpy/resources/pybind11/docs/release.rst index d8f9db854..43934ecf3 100644 --- a/stormpy/resources/pybind11/docs/release.rst +++ b/stormpy/resources/pybind11/docs/release.rst @@ -2,18 +2,20 @@ To release a new version of pybind11: - Update the version number and push to pypi - Update ``pybind11/_version.py`` (set release version, remove 'dev') - - Tag release date in ``doc/changelog.rst``. + - Update version in ``docs/conf.py`` + - Tag release date in ``docs/changelog.rst``. - ``git add`` and ``git commit``. - ``git tag -a vX.Y -m 'vX.Y release'``. - ``git push`` - ``git push --tags``. - ``python setup.py sdist upload``. - ``python setup.py bdist_wheel upload``. -- Update conda-forge (https://github.com/conda-forge/pybind11-feedstock) - - change version number in ``meta.yml`` +- Update conda-forge (https://github.com/conda-forge/pybind11-feedstock) via PR + - change version number in ``recipe/meta.yml`` - update checksum to match the one computed by pypi - Get back to work - Update ``_version.py`` (add 'dev' and increment minor). + - Update version in ``docs/conf.py`` - Update version macros in ``include/pybind11/common.h`` - ``git add`` and ``git commit``. ``git push`` diff --git a/stormpy/resources/pybind11/example/example10.cpp b/stormpy/resources/pybind11/example/example10.cpp index cbe737e72..06528c2f0 100644 --- a/stormpy/resources/pybind11/example/example10.cpp +++ b/stormpy/resources/pybind11/example/example10.cpp @@ -13,7 +13,7 @@ double my_func(int x, float y, double z) { std::cout << "my_func(x:int=" << x << ", y:float=" << y << ", z:float=" << z << ")" << std::endl; - return x*y*z; + return (float) x*y*z; } std::complex my_func3(std::complex c) { diff --git a/stormpy/resources/pybind11/example/example11.cpp b/stormpy/resources/pybind11/example/example11.cpp index 799fa6226..4b3c6d08b 100644 --- a/stormpy/resources/pybind11/example/example11.cpp +++ b/stormpy/resources/pybind11/example/example11.cpp @@ -55,4 +55,8 @@ void init_ex11(py::module &m) { m.def("args_function", &args_function); m.def("args_kwargs_function", &args_kwargs_function); + + using namespace py::literals; + m.def("kw_func_udl", &kw_func, "x"_a, "y"_a=300); + m.def("kw_func_udl_z", &kw_func, "x"_a, "y"_a=0); } diff --git a/stormpy/resources/pybind11/example/example11.py b/stormpy/resources/pybind11/example/example11.py index ff35be2e4..09ba13e6c 100755 --- a/stormpy/resources/pybind11/example/example11.py +++ b/stormpy/resources/pybind11/example/example11.py @@ -6,12 +6,14 @@ import pydoc sys.path.append('.') from example import kw_func, kw_func2, kw_func3, kw_func4, call_kw_func -from example import args_function, args_kwargs_function +from example import args_function, args_kwargs_function, kw_func_udl, kw_func_udl_z print(pydoc.render_doc(kw_func, "Help on %s")) print(pydoc.render_doc(kw_func2, "Help on %s")) print(pydoc.render_doc(kw_func3, "Help on %s")) print(pydoc.render_doc(kw_func4, "Help on %s")) +print(pydoc.render_doc(kw_func_udl, "Help on %s")) +print(pydoc.render_doc(kw_func_udl_z, "Help on %s")) kw_func(5, 10) kw_func(5, y=10) @@ -39,3 +41,6 @@ call_kw_func(kw_func2) args_function('arg1_value', 'arg2_value', 3) args_kwargs_function('arg1_value', 'arg2_value', arg3='arg3_value', arg4=4) + +kw_func_udl(x=5, y=10) +kw_func_udl_z(x=5) diff --git a/stormpy/resources/pybind11/example/example11.ref b/stormpy/resources/pybind11/example/example11.ref index f4c23aecd..a693b6ceb 100644 --- a/stormpy/resources/pybind11/example/example11.ref +++ b/stormpy/resources/pybind11/example/example11.ref @@ -18,6 +18,16 @@ Help on built-in function kw_func4 in module example kkww__ffuunncc44(...) kw_func4(myList : list = [13L, 17L]) -> NoneType +Help on built-in function kw_func_udl in module example + +kkww__ffuunncc__uuddll(...) + kw_func_udl(x : int, y : int = 300L) -> NoneType + +Help on built-in function kw_func_udl_z in module example + +kkww__ffuunncc__uuddll__zz(...) + kw_func_udl_z(x : int, y : int = 0L) -> NoneType + kw_func(x=5, y=10) kw_func(x=5, y=10) kw_func(x=5, y=10) @@ -40,3 +50,6 @@ got argument: arg1_value got argument: arg2_value got keyword argument: arg3 -> arg3_value got keyword argument: arg4 -> 4 + +kw_func(x=5, y=10) +kw_func(x=5, y=0) diff --git a/stormpy/resources/pybind11/example/example12.cpp b/stormpy/resources/pybind11/example/example12.cpp index 5cc8dc87c..e5555f53a 100644 --- a/stormpy/resources/pybind11/example/example12.cpp +++ b/stormpy/resources/pybind11/example/example12.cpp @@ -82,15 +82,11 @@ void runExample12Virtual(Example12 *ex) { } void init_ex12(py::module &m) { - /* Important: use the wrapper type as a template - argument to class_<>, but use the original name - to denote the type */ - py::class_(m, "Example12") - /* Declare that 'PyExample12' is really an alias for the original type 'Example12' */ - .alias() + /* Important: indicate the trampoline class PyExample12 using the third + argument to py::class_. The second argument with the unique pointer + is simply the default holder type used by pybind11. */ + py::class_, PyExample12>(m, "Example12") .def(py::init()) - /* Copy constructor (not needed in this case, but should generally be declared in this way) */ - .def(py::init()) /* Reference original class in function definitions */ .def("run", &Example12::run) .def("run_bool", &Example12::run_bool) diff --git a/stormpy/resources/pybind11/example/example17.cpp b/stormpy/resources/pybind11/example/example17.cpp index 8ae4cad08..8fd4ad642 100644 --- a/stormpy/resources/pybind11/example/example17.cpp +++ b/stormpy/resources/pybind11/example/example17.cpp @@ -29,6 +29,7 @@ void init_ex17(py::module &m) { .def(pybind11::init()); pybind11::bind_vector(m, "VectorInt"); + pybind11::bind_vector(m, "VectorBool"); pybind11::bind_vector(m, "VectorEl"); diff --git a/stormpy/resources/pybind11/example/example17.py b/stormpy/resources/pybind11/example/example17.py index 65e586bcb..feae30708 100644 --- a/stormpy/resources/pybind11/example/example17.py +++ b/stormpy/resources/pybind11/example/example17.py @@ -1,7 +1,7 @@ #!/usr/bin/env python from __future__ import print_function -from example import VectorInt, El, VectorEl, VectorVectorEl +from example import VectorInt, El, VectorEl, VectorVectorEl, VectorBool v_int = VectorInt([0, 0]) print(len(v_int)) @@ -38,3 +38,11 @@ vv_a = VectorVectorEl() vv_a.append(v_a) vv_b = vv_a[0] print(vv_b) + +vv_c = VectorBool() +for i in range(10): + vv_c.append(i % 2 == 0) +for i in range(10): + if vv_c[i] != (i % 2 == 0): + print("Error!") +print(vv_c) diff --git a/stormpy/resources/pybind11/example/example17.ref b/stormpy/resources/pybind11/example/example17.ref index 55e47a68d..cc271f3a9 100644 --- a/stormpy/resources/pybind11/example/example17.ref +++ b/stormpy/resources/pybind11/example/example17.ref @@ -8,3 +8,4 @@ VectorInt[3, 0, 99, 2, 3] VectorInt[0, 99, 2, 3] VectorEl[El{1}, El{2}] VectorEl[El{1}, El{2}] +VectorBool[1, 0, 1, 0, 1, 0, 1, 0, 1, 0] diff --git a/stormpy/resources/pybind11/example/example2.cpp b/stormpy/resources/pybind11/example/example2.cpp index 9b91baaad..4b15823f2 100644 --- a/stormpy/resources/pybind11/example/example2.cpp +++ b/stormpy/resources/pybind11/example/example2.cpp @@ -154,7 +154,7 @@ void init_ex2(py::module &m) { .def("get_list", &Example2::get_list, "Return a Python list") .def("get_list_2", &Example2::get_list_2, "Return a C++ list") .def("get_set", &Example2::get_set, "Return a Python set") - .def("get_set2", &Example2::get_set, "Return a C++ set") + .def("get_set2", &Example2::get_set_2, "Return a C++ set") .def("get_array", &Example2::get_array, "Return a C++ array") .def("print_dict", &Example2::print_dict, "Print entries of a Python dictionary") .def("print_dict_2", &Example2::print_dict_2, "Print entries of a C++ dictionary") diff --git a/stormpy/resources/pybind11/example/example4.cpp b/stormpy/resources/pybind11/example/example4.cpp index 281eafed5..7e17864e0 100644 --- a/stormpy/resources/pybind11/example/example4.cpp +++ b/stormpy/resources/pybind11/example/example4.cpp @@ -38,7 +38,7 @@ void test_function2(EMyEnumeration k) { float test_function3(int i) { std::cout << "test_function(" << i << ")" << std::endl; - return i / 2.f; + return (float) i / 2.f; } py::bytes return_bytes() { diff --git a/stormpy/resources/pybind11/example/example5.cpp b/stormpy/resources/pybind11/example/example5.cpp index 11d37d0ef..0e1d2cd13 100644 --- a/stormpy/resources/pybind11/example/example5.cpp +++ b/stormpy/resources/pybind11/example/example5.cpp @@ -60,6 +60,11 @@ std::function test_callback4() { return [](int i) { return i+1; }; } +py::cpp_function test_callback5() { + return py::cpp_function([](int i) { return i+1; }, + py::arg("number")); +} + void init_ex5(py::module &m) { py::class_ pet_class(m, "Pet"); pet_class @@ -82,6 +87,7 @@ void init_ex5(py::module &m) { m.def("test_callback2", &test_callback2); m.def("test_callback3", &test_callback3); m.def("test_callback4", &test_callback4); + m.def("test_callback5", &test_callback5); /* Test cleanup of lambda closure */ diff --git a/stormpy/resources/pybind11/example/example5.py b/stormpy/resources/pybind11/example/example5.py index ef90cfd07..1361c7519 100755 --- a/stormpy/resources/pybind11/example/example5.py +++ b/stormpy/resources/pybind11/example/example5.py @@ -29,6 +29,7 @@ from example import test_callback1 from example import test_callback2 from example import test_callback3 from example import test_callback4 +from example import test_callback5 from example import test_cleanup def func1(): @@ -49,5 +50,7 @@ print(test_callback1(partial(func3, "Partial object with one argument"))) test_callback3(lambda i: i + 1) f = test_callback4() print("func(43) = %i" % f(43)) +f = test_callback5() +print("func(number=43) = %i" % f(number=43)) test_cleanup() diff --git a/stormpy/resources/pybind11/example/example5.ref b/stormpy/resources/pybind11/example/example5.ref index a9c7d465f..f32ba3af1 100644 --- a/stormpy/resources/pybind11/example/example5.ref +++ b/stormpy/resources/pybind11/example/example5.ref @@ -24,3 +24,4 @@ False Callback function 3 called : Partial object with one argument False func(43) = 44 +func(number=43) = 44 diff --git a/stormpy/resources/pybind11/example/example6.cpp b/stormpy/resources/pybind11/example/example6.cpp index 26552b7e3..e0bfb9e70 100644 --- a/stormpy/resources/pybind11/example/example6.cpp +++ b/stormpy/resources/pybind11/example/example6.cpp @@ -133,22 +133,22 @@ void init_ex6(py::module &m) { .def("__reversed__", [](const Sequence &s) -> Sequence { return s.reversed(); }) /// Slicing protocol (optional) .def("__getitem__", [](const Sequence &s, py::slice slice) -> Sequence* { - py::ssize_t start, stop, step, slicelength; + size_t start, stop, step, slicelength; if (!slice.compute(s.size(), &start, &stop, &step, &slicelength)) throw py::error_already_set(); Sequence *seq = new Sequence(slicelength); - for (int i=0; i base(m2, "DispatchIssue"); - base.alias() + py::class_, DispatchIssue>(m2, "DispatchIssue") .def(py::init<>()) .def("dispatch", &Base::dispatch); @@ -108,4 +107,35 @@ void init_issues(py::module &m) { // (no id): don't cast doubles to ints m2.def("expect_float", [](float f) { return f; }); m2.def("expect_int", [](int i) { return i; }); + + // (no id): don't invoke Python dispatch code when instantiating C++ + // classes that were not extended on the Python side + struct A { + virtual ~A() {} + virtual void f() { std::cout << "A.f()" << std::endl; } + }; + + struct PyA : A { + PyA() { std::cout << "PyA.PyA()" << std::endl; } + + void f() override { + std::cout << "PyA.f()" << std::endl; + PYBIND11_OVERLOAD(void, A, f); + } + }; + + auto call_f = [](A *a) { a->f(); }; + + pybind11::class_, PyA>(m2, "A") + .def(py::init<>()) + .def("f", &A::f); + + m2.def("call_f", call_f); + + try { + py::class_(m2, "Placeholder"); + throw std::logic_error("Expected an exception!"); + } catch (std::runtime_error &) { + /* All good */ + } } diff --git a/stormpy/resources/pybind11/example/issues.py b/stormpy/resources/pybind11/example/issues.py index d075e8340..257f08e18 100644 --- a/stormpy/resources/pybind11/example/issues.py +++ b/stormpy/resources/pybind11/example/issues.py @@ -9,6 +9,7 @@ from example.issues import Placeholder, return_vec_of_reference_wrapper from example.issues import iterator_passthrough from example.issues import ElementList, ElementA, print_element from example.issues import expect_float, expect_int +from example.issues import A, call_f import gc print_cchar("const char *") @@ -55,3 +56,19 @@ except Exception as e: print("Failed as expected: " + str(e)) print(expect_float(12)) + +class B(A): + def __init__(self): + super(B, self).__init__() + + def f(self): + print("In python f()") + +print("C++ version") +a = A() +call_f(a) + +print("Python version") +b = B() +call_f(b) + diff --git a/stormpy/resources/pybind11/example/issues.ref b/stormpy/resources/pybind11/example/issues.ref index 4888ea55b..58cc7985f 100644 --- a/stormpy/resources/pybind11/example/issues.ref +++ b/stormpy/resources/pybind11/example/issues.ref @@ -12,3 +12,9 @@ Failed as expected: Incompatible function arguments. The following argument type 1. (int) -> int Invoked with: 5.2 12.0 +C++ version +A.f() +Python version +PyA.PyA() +PyA.f() +In python f() diff --git a/stormpy/resources/pybind11/example/run_test.py b/stormpy/resources/pybind11/example/run_test.py index d0967c6e2..70ce4a6c0 100755 --- a/stormpy/resources/pybind11/example/run_test.py +++ b/stormpy/resources/pybind11/example/run_test.py @@ -2,6 +2,7 @@ import sys import os import re import subprocess +import difflib remove_unicode_marker = re.compile(r'u(\'[^\']*\')') remove_long_marker = re.compile(r'([0-9])L') @@ -36,11 +37,7 @@ def sanitize(lines): line = "" lines[i] = line - lines = '\n'.join(sorted([l for l in lines if l != ""])) - - print('==================') - print(lines) - return lines + return '\n'.join(sorted([l for l in lines if l != ""])) path = os.path.dirname(__file__) if path != '': @@ -69,4 +66,8 @@ elif output == reference: exit(0) else: print('Test "%s" FAILED!' % name) + print('--- output') + print('+++ reference') + print(''.join(difflib.ndiff(output.splitlines(keepends=True), + reference.splitlines(keepends=True)))) exit(-1) diff --git a/stormpy/resources/pybind11/include/pybind11/attr.h b/stormpy/resources/pybind11/include/pybind11/attr.h index d63a44e9f..1e063a5f9 100644 --- a/stormpy/resources/pybind11/include/pybind11/attr.h +++ b/stormpy/resources/pybind11/include/pybind11/attr.h @@ -18,23 +18,29 @@ template struct arg_t; /// Annotation for keyword arguments struct arg { - arg(const char *name) : name(name) { } - template arg_t operator=(const T &value); - template arg_t operator=(T const (&value)[N]); + constexpr explicit arg(const char *name) : name(name) { } + + template + constexpr arg_t operator=(const T &value) const { return {name, value}; } + template + constexpr arg_t operator=(T const (&value)[N]) const { + return operator=((const T *) value); + } + const char *name; }; /// Annotation for keyword arguments with default values template struct arg_t : public arg { - arg_t(const char *name, const T &value, const char *descr = nullptr) + constexpr arg_t(const char *name, const T &value, const char *descr = nullptr) : arg(name), value(value), descr(descr) { } T value; const char *descr; }; -template arg_t arg::operator=(const T &value) { return arg_t(name, value); } -template arg_t arg::operator=(T const (&value)[N]) { - return operator=((const T *) value); +inline namespace literals { +/// String literal version of arg +constexpr arg operator"" _a(const char *name, size_t) { return arg(name); } } /// Annotation for methods @@ -65,6 +71,7 @@ enum op_type : int; struct undefined_t; template struct op_; template struct init; +template struct init_alias; inline void keep_alive_impl(int Nurse, int Patient, handle args, handle ret); /// Internal data structure which holds metadata about a keyword argument @@ -314,5 +321,18 @@ template struct process_attributes { } }; +/// Compile-time integer sum +constexpr size_t constexpr_sum() { return 0; } +template +constexpr size_t constexpr_sum(T n, Ts... ns) { return n + constexpr_sum(ns...); } + +/// Check the number of named arguments at compile time +template ::value...), + size_t self = constexpr_sum(std::is_same::value...)> +constexpr bool expected_num_args(size_t nargs) { + return named == 0 || (self + named) == nargs; +} + NAMESPACE_END(detail) NAMESPACE_END(pybind11) diff --git a/stormpy/resources/pybind11/include/pybind11/cast.h b/stormpy/resources/pybind11/include/pybind11/cast.h index 31f5cb5a6..a4e76aab5 100644 --- a/stormpy/resources/pybind11/include/pybind11/cast.h +++ b/stormpy/resources/pybind11/include/pybind11/cast.h @@ -471,13 +471,13 @@ public: ssize_t length; int err = PYBIND11_BYTES_AS_STRING_AND_SIZE(load_src.ptr(), &buffer, &length); if (err == -1) { PyErr_Clear(); return false; } // TypeError - value = std::string(buffer, length); + value = std::string(buffer, (size_t) length); success = true; return true; } static handle cast(const std::string &src, return_value_policy /* policy */, handle /* parent */) { - return PyUnicode_FromStringAndSize(src.c_str(), src.length()); + return PyUnicode_FromStringAndSize(src.c_str(), (ssize_t) src.length()); } PYBIND11_TYPE_CASTER(std::string, _(PYBIND11_STRING_NAME)); @@ -520,17 +520,17 @@ public: if (temp) { int err = PYBIND11_BYTES_AS_STRING_AND_SIZE(temp.ptr(), (char **) &buffer, &length); if (err == -1) { buffer = nullptr; } // TypeError - length = length / sizeof(wchar_t) - 1; ++buffer; // Skip BOM + length = length / (ssize_t) sizeof(wchar_t) - 1; ++buffer; // Skip BOM } #endif if (!buffer) { PyErr_Clear(); return false; } - value = std::wstring(buffer, length); + value = std::wstring(buffer, (size_t) length); success = true; return true; } static handle cast(const std::wstring &src, return_value_policy /* policy */, handle /* parent */) { - return PyUnicode_FromWideChar(src.c_str(), src.length()); + return PyUnicode_FromWideChar(src.c_str(), (ssize_t) src.length()); } PYBIND11_TYPE_CASTER(std::wstring, _(PYBIND11_STRING_NAME)); @@ -570,7 +570,7 @@ public: static handle cast(const wchar_t *src, return_value_policy /* policy */, handle /* parent */) { if (src == nullptr) return handle(Py_None).inc_ref(); - return PyUnicode_FromWideChar(src, wcslen(src)); + return PyUnicode_FromWideChar(src, (ssize_t) wcslen(src)); } static handle cast(wchar_t src, return_value_policy /* policy */, handle /* parent */) { @@ -660,7 +660,7 @@ public: std::get<1>(value).load(kwargs, convert); return true; } - + static handle cast(const type &src, return_value_policy policy, handle parent) { return cast(src, policy, parent, typename make_index_sequence::type()); } @@ -842,16 +842,18 @@ template object handle::operator()(Args&&... args) const { - tuple args_tuple = pybind11::make_tuple(std::forward(args)...); +template object handle::operator()(Args&&... args) const { + tuple args_tuple = pybind11::make_tuple(std::forward(args)...); object result(PyObject_CallObject(m_ptr, args_tuple.ptr()), false); if (!result) throw error_already_set(); return result; } -template object handle::call(Args &&... args) const { - return operator()(std::forward(args)...); +template object handle::call(Args &&... args) const { + return operator()(std::forward(args)...); } inline object handle::operator()(detail::args_proxy args) const { diff --git a/stormpy/resources/pybind11/include/pybind11/common.h b/stormpy/resources/pybind11/include/pybind11/common.h index e60684fc2..3075fb3c6 100644 --- a/stormpy/resources/pybind11/include/pybind11/common.h +++ b/stormpy/resources/pybind11/include/pybind11/common.h @@ -31,7 +31,7 @@ #endif #define PYBIND11_VERSION_MAJOR 1 -#define PYBIND11_VERSION_MINOR 8 +#define PYBIND11_VERSION_MINOR 9 /// Include Python header, disable linking to pythonX_d.lib on Windows in debug mode #if defined(_MSC_VER) @@ -116,7 +116,7 @@ extern "C" { struct _Py_atomic_address { void *value; }; PyAPI_DATA(_Py_atomic_address) _PyThreadState_Current; -}; +} #endif #define PYBIND11_TRY_NEXT_OVERLOAD ((PyObject *) 1) // special failure return code @@ -151,7 +151,9 @@ enum class return_value_policy : uint8_t { automatic = 0, /** As above, but use policy return_value_policy::reference when the return - value is a pointer. You probably won't need to use this. */ + value is a pointer. This is the default conversion policy for function + arguments when calling Python functions manually from C++ code (i.e. via + handle::operator()). You probably won't need to use this. */ automatic_reference, /** Reference an existing object (i.e. do not create a new copy) and take @@ -197,22 +199,23 @@ struct buffer_info { size_t itemsize; // Size of individual items in bytes size_t size; // Total number of entries std::string format; // For homogeneous buffers, this should be set to format_descriptor::value - int ndim; // Number of dimensions + size_t ndim; // Number of dimensions std::vector shape; // Shape of the tensor (1 entry per dimension) std::vector strides; // Number of entries between adjacent entries (for each per dimension) buffer_info() : ptr(nullptr), view(nullptr) {} - buffer_info(void *ptr, size_t itemsize, const std::string &format, int ndim, + buffer_info(void *ptr, size_t itemsize, const std::string &format, size_t ndim, const std::vector &shape, const std::vector &strides) : ptr(ptr), itemsize(itemsize), size(1), format(format), ndim(ndim), shape(shape), strides(strides) { - for (int i=0; ibuf), itemsize(view->itemsize), size(1), format(view->format), - ndim(view->ndim), shape(view->ndim), strides(view->ndim), view(view) { - for (int i = 0; i < view->ndim; ++i) { + : ptr(view->buf), itemsize((size_t) view->itemsize), size(1), format(view->format), + ndim((size_t) view->ndim), shape((size_t) view->ndim), strides((size_t) view->ndim), view(view) { + for (size_t i = 0; i < (size_t) view->ndim; ++i) { shape[i] = (size_t) view->shape[i]; strides[i] = (size_t) view->strides[i]; size *= shape[i]; @@ -262,7 +265,7 @@ struct internals { std::unordered_map registered_instances; // void * -> PyObject* std::unordered_set, overload_hash> inactive_overload_cache; #if defined(WITH_THREAD) - int tstate = 0; + decltype(PyThread_create_key()) tstate = 0; // Usually an int but a long on Cygwin64 with Python 3.x PyInterpreterState *istate = nullptr; #endif }; diff --git a/stormpy/resources/pybind11/include/pybind11/descr.h b/stormpy/resources/pybind11/include/pybind11/descr.h index 1b65f68c8..6c1d86432 100644 --- a/stormpy/resources/pybind11/include/pybind11/descr.h +++ b/stormpy/resources/pybind11/include/pybind11/descr.h @@ -139,7 +139,7 @@ protected: const T *it = ptr; while (*it++ != (T) 0) ; - return it - ptr; + return static_cast(it - ptr); } const std::type_info **m_types = nullptr; diff --git a/stormpy/resources/pybind11/include/pybind11/eigen.h b/stormpy/resources/pybind11/include/pybind11/eigen.h index 96daf48dd..718107947 100644 --- a/stormpy/resources/pybind11/include/pybind11/eigen.h +++ b/stormpy/resources/pybind11/include/pybind11/eigen.h @@ -10,9 +10,20 @@ #pragma once #include "numpy.h" + +#if defined(__GNUG__) || defined(__clang__) +# pragma GCC diagnostic push +# pragma GCC diagnostic ignored "-Wconversion" +# pragma GCC diagnostic ignored "-Wdeprecated-declarations" +#endif + #include #include +#if defined(__GNUG__) || defined(__clang__) +# pragma GCC diagnostic pop +#endif + #if defined(_MSC_VER) #pragma warning(push) #pragma warning(disable: 4127) // warning C4127: Conditional expression is constant @@ -63,7 +74,7 @@ struct type_caster::value>::t auto strides = Strides(info.strides[0] / sizeof(Scalar), 0); value = Eigen::Map( - (Scalar *) info.ptr, info.shape[0], 1, strides); + (Scalar *) info.ptr, typename Strides::Index(info.shape[0]), 1, strides); } else if (info.ndim == 2) { typedef Eigen::Stride Strides; @@ -76,7 +87,9 @@ struct type_caster::value>::t info.strides[rowMajor ? 1 : 0] / sizeof(Scalar)); value = Eigen::Map( - (Scalar *) info.ptr, info.shape[0], info.shape[1], strides); + (Scalar *) info.ptr, + typename Strides::Index(info.shape[0]), + typename Strides::Index(info.shape[1]), strides); } else { return false; } @@ -117,8 +130,8 @@ struct type_caster::value>::t { (size_t) src.rows(), (size_t) src.cols() }, /* Strides (in bytes) for each index */ - { sizeof(Scalar) * (rowMajor ? src.cols() : 1), - sizeof(Scalar) * (rowMajor ? 1 : src.rows()) } + { sizeof(Scalar) * (rowMajor ? (size_t) src.cols() : 1), + sizeof(Scalar) * (rowMajor ? 1 : (size_t) src.rows()) } )).release(); } } diff --git a/stormpy/resources/pybind11/include/pybind11/numpy.h b/stormpy/resources/pybind11/include/pybind11/numpy.h index f97c790ad..0180d95cd 100644 --- a/stormpy/resources/pybind11/include/pybind11/numpy.h +++ b/stormpy/resources/pybind11/include/pybind11/numpy.h @@ -109,7 +109,7 @@ public: if (descr == nullptr) pybind11_fail("NumPy: unsupported buffer format '" + info.format + "'!"); object tmp(api.PyArray_NewFromDescr_( - api.PyArray_Type_, descr, info.ndim, (Py_intptr_t *) &info.shape[0], + api.PyArray_Type_, descr, (int) info.ndim, (Py_intptr_t *) &info.shape[0], (Py_intptr_t *) &info.strides[0], info.ptr, 0, nullptr), false); if (info.ptr && tmp) tmp = object(api.PyArray_NewCopy_(tmp.ptr(), -1 /* any order */), false); @@ -261,7 +261,7 @@ private: while (buffer_shape_iter != buffer.shape.rend()) { if (*shape_iter == *buffer_shape_iter) - *strides_iter = static_cast(*buffer_strides_iter); + *strides_iter = static_cast(*buffer_strides_iter); else *strides_iter = 0; @@ -286,12 +286,12 @@ private: }; template -bool broadcast(const std::array& buffers, int& ndim, std::vector& shape) { - ndim = std::accumulate(buffers.begin(), buffers.end(), 0, [](int res, const buffer_info& buf) { +bool broadcast(const std::array& buffers, size_t& ndim, std::vector& shape) { + ndim = std::accumulate(buffers.begin(), buffers.end(), size_t(0), [](size_t res, const buffer_info& buf) { return std::max(res, buf.ndim); }); - shape = std::vector(static_cast(ndim), 1); + shape = std::vector(ndim, 1); bool trivial_broadcast = true; for (size_t i = 0; i < N; ++i) { auto res_iter = shape.rbegin(); @@ -329,7 +329,7 @@ struct vectorize_helper { std::array buffers {{ args.request()... }}; /* Determine dimensions parameters of output array */ - int ndim = 0; + size_t ndim = 0; std::vector shape(0); bool trivial_broadcast = broadcast(buffers, ndim, shape); @@ -337,7 +337,7 @@ struct vectorize_helper { std::vector strides(ndim); if (ndim > 0) { strides[ndim-1] = sizeof(Return); - for (int i = ndim - 1; i > 0; --i) { + for (size_t i = ndim - 1; i > 0; --i) { strides[i - 1] = strides[i] * shape[i]; size *= shape[i]; } diff --git a/stormpy/resources/pybind11/include/pybind11/pybind11.h b/stormpy/resources/pybind11/include/pybind11/pybind11.h index 34869ebae..e20096a8e 100644 --- a/stormpy/resources/pybind11/include/pybind11/pybind11.h +++ b/stormpy/resources/pybind11/include/pybind11/pybind11.h @@ -72,6 +72,9 @@ protected: /// Special internal constructor for functors, lambda functions, etc. template void initialize(Func &&f, Return (*)(Args...), const Extra&... extra) { + static_assert(detail::expected_num_args(sizeof...(Args)), + "The number of named arguments does not match the function signature"); + struct capture { typename std::remove_reference::type f; }; /* Store the function including any extra state it might have (e.g. a lambda capture object) */ @@ -206,12 +209,6 @@ protected: } #endif - if (!rec->args.empty() && (int) rec->args.size() != args) - pybind11_fail( - "cpp_function(): function \"" + std::string(rec->name) + "\" takes " + - std::to_string(args) + " arguments, but " + std::to_string(rec->args.size()) + - " pybind11::arg entries were specified!"); - rec->signature = strdup(signature.c_str()); rec->args.shrink_to_fit(); rec->is_constructor = !strcmp(rec->name, "__init__") || !strcmp(rec->name, "__setstate__"); @@ -336,8 +333,8 @@ protected: *it = overloads; /* Need to know how many arguments + keyword arguments there are to pick the right overload */ - size_t nargs = PyTuple_GET_SIZE(args), - nkwargs = kwargs ? PyDict_Size(kwargs) : 0; + size_t nargs = (size_t) PyTuple_GET_SIZE(args), + nkwargs = kwargs ? (size_t) PyDict_Size(kwargs) : 0; handle parent = nargs > 0 ? PyTuple_GET_ITEM(args, 0) : nullptr, result = PYBIND11_TRY_NEXT_OVERLOAD; @@ -503,7 +500,7 @@ public: NAMESPACE_BEGIN(detail) /// Generic support for creating new Python heap types class generic_type : public object { - template friend class class_; + template friend class class_; public: PYBIND11_OBJECT_DEFAULT(generic_type, object, PyType_Check) protected: @@ -520,6 +517,14 @@ protected: } } + auto &internals = get_internals(); + auto tindex = std::type_index(*(rec->type)); + + if (internals.registered_types_cpp.find(tindex) != + internals.registered_types_cpp.end()) + pybind11_fail("generic_type: type \"" + std::string(rec->name) + + "\" is already registered!"); + object type_holder(PyType_Type.tp_alloc(&PyType_Type, 0), false); object name(PYBIND11_FROM_STRING(rec->name), false); auto type = (PyHeapTypeObject*) type_holder.ptr(); @@ -528,12 +533,11 @@ protected: pybind11_fail("generic_type: unable to create type object!"); /* Register supplemental type information in C++ dict */ - auto &internals = get_internals(); detail::type_info *tinfo = new detail::type_info(); tinfo->type = (PyTypeObject *) type; tinfo->type_size = rec->type_size; tinfo->init_holder = rec->init_holder; - internals.registered_types_cpp[std::type_index(*(rec->type))] = tinfo; + internals.registered_types_cpp[tindex] = tinfo; internals.registered_types_py[type] = tinfo; object scope_module; @@ -547,7 +551,7 @@ protected: : std::string(rec->name)); /* Basic type attributes */ type->ht_type.tp_name = strdup(full_name.c_str()); - type->ht_type.tp_basicsize = rec->instance_size; + type->ht_type.tp_basicsize = (ssize_t) rec->instance_size; type->ht_type.tp_base = (PyTypeObject *) rec->base_handle.ptr(); rec->base_handle.inc_ref(); @@ -702,14 +706,14 @@ protected: view->ndim = 1; view->internal = info; view->buf = info->ptr; - view->itemsize = info->itemsize; + view->itemsize = (ssize_t) info->itemsize; view->len = view->itemsize; for (auto s : info->shape) view->len *= s; if ((flags & PyBUF_FORMAT) == PyBUF_FORMAT) view->format = const_cast(info->format.c_str()); if ((flags & PyBUF_STRIDES) == PyBUF_STRIDES) { - view->ndim = info->ndim; + view->ndim = (int) info->ndim; view->strides = (ssize_t *) &info->strides[0]; view->shape = (ssize_t *) &info->shape[0]; } @@ -721,7 +725,7 @@ protected: }; NAMESPACE_END(detail) -template > +template , typename type_alias = type> class class_ : public detail::generic_type { public: typedef detail::instance instance_type; @@ -743,6 +747,11 @@ public: detail::process_attributes::init(extra..., &record); detail::generic_type::initialize(&record); + + if (!std::is_same::value) { + auto &instances = pybind11::detail::get_internals().registered_types_cpp; + instances[std::type_index(typeid(type_alias))] = instances[std::type_index(typeid(type))]; + } } template @@ -780,6 +789,12 @@ public: return *this; } + template + class_ &def(const detail::init_alias &init, const Extra&... extra) { + init.template execute(*this, extra...); + return *this; + } + template class_& def_buffer(Func &&func) { struct capture { Func func; }; capture *ptr = new capture { std::forward(func) }; @@ -842,9 +857,20 @@ public: template class_ &def_property_static(const char *name, const cpp_function &fget, const cpp_function &fset, const Extra& ...extra) { auto rec_fget = get_function_record(fget), rec_fset = get_function_record(fset); + char *doc_prev = rec_fget->doc; /* 'extra' field may include a property-specific documentation string */ detail::process_attributes::init(extra..., rec_fget); - if (rec_fset) + if (rec_fget->doc && rec_fget->doc != doc_prev) { + free(doc_prev); + rec_fget->doc = strdup(rec_fget->doc); + } + if (rec_fset) { + doc_prev = rec_fset->doc; detail::process_attributes::init(extra..., rec_fset); + if (rec_fset->doc && rec_fset->doc != doc_prev) { + free(doc_prev); + rec_fset->doc = strdup(rec_fset->doc); + } + } pybind11::str doc_obj = pybind11::str(rec_fget->doc ? rec_fget->doc : ""); object property( PyObject_CallFunctionObjArgs((PyObject *) &PyProperty_Type, fget.ptr() ? fget.ptr() : Py_None, @@ -856,11 +882,6 @@ public: return *this; } - template class_ alias() { - auto &instances = pybind11::detail::get_internals().registered_types_cpp; - instances[std::type_index(typeid(target))] = instances[std::type_index(typeid(type))]; - return *this; - } private: /// Initialize holder object, variant 1: object derives from enable_shared_from_this template @@ -959,9 +980,31 @@ private: NAMESPACE_BEGIN(detail) template struct init { - template void execute(pybind11::class_ &class_, const Extra&... extra) const { + template ::value, int>::type = 0> + void execute(pybind11::class_ &class_, const Extra&... extra) const { /// Function which calls a specific C++ in-place constructor - class_.def("__init__", [](Base *instance, Args... args) { new (instance) Base(args...); }, extra...); + class_.def("__init__", [](Base *self_, Args... args) { new (self_) Base(args...); }, extra...); + } + + template ::value && + std::is_constructible::value, int>::type = 0> + void execute(pybind11::class_ &class_, const Extra&... extra) const { + handle cl_type = class_; + class_.def("__init__", [cl_type](handle self_, Args... args) { + if (self_.get_type() == cl_type) + new (self_.cast()) Base(args...); + else + new (self_.cast()) Alias(args...); + }, extra...); + } + + template ::value && + !std::is_constructible::value, int>::type = 0> + void execute(pybind11::class_ &class_, const Extra&... extra) const { + class_.def("__init__", [](Alias *self_, Args... args) { new (self_) Alias(args...); }, extra...); } }; @@ -985,26 +1028,36 @@ PYBIND11_NOINLINE inline void keep_alive_impl(int Nurse, int Patient, handle arg (void) wr.release(); } -template struct iterator_state { Iterator it, end; }; +template struct iterator_state { + Iterator it, end; + bool first; +}; NAMESPACE_END(detail) template detail::init init() { return detail::init(); } -template iterator make_iterator(Iterator first, Iterator last, Extra&&... extra) { +template ()), + typename... Extra> +iterator make_iterator(Iterator first, Iterator last, Extra &&... extra) { typedef detail::iterator_state state; if (!detail::get_type_info(typeid(state))) { class_(handle(), "") .def("__iter__", [](state &s) -> state& { return s; }) - .def("__next__", [](state &s) -> decltype(*std::declval()) { + .def("__next__", [](state &s) -> ValueType { + if (!s.first) + ++s.it; + else + s.first = false; if (s.it == s.end) throw stop_iteration(); - return *s.it++; + return *s.it; }, return_value_policy::reference_internal, std::forward(extra)...); } - return (iterator) cast(state { first, last }); + return (iterator) cast(state { first, last, true }); } template iterator make_iterator(Type &value, Extra&&... extra) { @@ -1046,6 +1099,11 @@ template void implicitly_convertible() * can be handy to prevent cases where callbacks issued from an external * thread would otherwise constantly construct and destroy thread state data * structures. + * + * See the Python bindings of NanoGUI (http://github.com/wjakob/nanogui) for an + * example which uses features 2 and 3 to migrate the Python thread of + * execution to another thread (to run the event loop on the original thread, + * in this case). */ class gil_scoped_acquire { @@ -1123,7 +1181,7 @@ public: gil_scoped_release(bool disassoc = false) : disassoc(disassoc) { tstate = PyEval_SaveThread(); if (disassoc) { - int key = detail::get_internals().tstate; + auto key = detail::get_internals().tstate; #if PY_MAJOR_VERSION < 3 PyThread_delete_key_value(key); #else @@ -1136,7 +1194,7 @@ public: return; PyEval_RestoreThread(tstate); if (disassoc) { - int key = detail::get_internals().tstate; + auto key = detail::get_internals().tstate; #if PY_MAJOR_VERSION < 3 PyThread_delete_key_value(key); #endif diff --git a/stormpy/resources/pybind11/include/pybind11/pytypes.h b/stormpy/resources/pybind11/include/pybind11/pytypes.h index 4da205b4f..6eee186db 100644 --- a/stormpy/resources/pybind11/include/pybind11/pytypes.h +++ b/stormpy/resources/pybind11/include/pybind11/pytypes.h @@ -39,10 +39,13 @@ public: inline detail::accessor attr(const char *key) const; inline pybind11::str str() const; template T cast() const; - template + template + #if __cplusplus > 201103L [[deprecated("call(...) was deprecated in favor of operator()(...)")]] + #endif object call(Args&&... args) const; - template object operator()(Args&&... args) const; + template + object operator()(Args&&... args) const; inline object operator()(detail::args_proxy args) const; inline object operator()(detail::args_proxy f_args, detail::kwargs_proxy kwargs) const; operator bool() const { return m_ptr != nullptr; } @@ -323,10 +326,10 @@ public: PYBIND11_OBJECT_DEFAULT(iterable, object, detail::PyIterable_Check) }; -inline detail::accessor handle::operator[](handle key) const { return detail::accessor(ptr(), key.ptr(), false); } -inline detail::accessor handle::operator[](const char *key) const { return detail::accessor(ptr(), key, false); } -inline detail::accessor handle::attr(handle key) const { return detail::accessor(ptr(), key.ptr(), true); } -inline detail::accessor handle::attr(const char *key) const { return detail::accessor(ptr(), key, true); } +inline detail::accessor handle::operator[](handle key) const { return detail::accessor(*this, key, false); } +inline detail::accessor handle::operator[](const char *key) const { return detail::accessor(*this, key, false); } +inline detail::accessor handle::attr(handle key) const { return detail::accessor(*this, key, true); } +inline detail::accessor handle::attr(const char *key) const { return detail::accessor(*this, key, true); } inline iterator handle::begin() const { return iterator(PyObject_GetIter(ptr()), false); } inline iterator handle::end() const { return iterator(nullptr, false); } inline detail::args_proxy handle::operator*() const { return detail::args_proxy(*this); } @@ -336,7 +339,7 @@ public: PYBIND11_OBJECT_DEFAULT(str, object, detail::PyUnicode_Check_Permissive) str(const std::string &s) - : object(PyUnicode_FromStringAndSize(s.c_str(), s.length()), false) { + : object(PyUnicode_FromStringAndSize(s.c_str(), (ssize_t) s.length()), false) { if (!m_ptr) pybind11_fail("Could not allocate string object!"); } @@ -352,7 +355,7 @@ public: int err = PYBIND11_BYTES_AS_STRING_AND_SIZE(temp.ptr(), &buffer, &length); if (err == -1) pybind11_fail("Unable to extract string contents! (invalid type)"); - return std::string(buffer, length); + return std::string(buffer, (size_t) length); } }; @@ -370,7 +373,7 @@ public: PYBIND11_OBJECT_DEFAULT(bytes, object, PYBIND11_BYTES_CHECK) bytes(const std::string &s) - : object(PYBIND11_BYTES_FROM_STRING_AND_SIZE(s.data(), s.size()), false) { + : object(PYBIND11_BYTES_FROM_STRING_AND_SIZE(s.data(), (ssize_t) s.size()), false) { if (!m_ptr) pybind11_fail("Could not allocate bytes object!"); } @@ -380,7 +383,7 @@ public: int err = PYBIND11_BYTES_AS_STRING_AND_SIZE(m_ptr, &buffer, &length); if (err == -1) pybind11_fail("Unable to extract bytes contents!"); - return std::string(buffer, length); + return std::string(buffer, (size_t) length); } }; @@ -463,9 +466,12 @@ public: m_ptr = PySlice_New(start.ptr(), stop.ptr(), step.ptr()); if (!m_ptr) pybind11_fail("Could not allocate slice object!"); } - bool compute(ssize_t length, ssize_t *start, ssize_t *stop, ssize_t *step, ssize_t *slicelength) const { - return PySlice_GetIndicesEx((PYBIND11_SLICE_OBJECT *) m_ptr, length, - start, stop, step, slicelength) == 0; + bool compute(size_t length, size_t *start, size_t *stop, size_t *step, + size_t *slicelength) const { + return PySlice_GetIndicesEx((PYBIND11_SLICE_OBJECT *) m_ptr, + (ssize_t) length, (ssize_t *) start, + (ssize_t *) stop, (ssize_t *) step, + (ssize_t *) slicelength) == 0; } }; @@ -501,7 +507,7 @@ public: if (!m_ptr) pybind11_fail("Could not allocate dict object!"); } size_t size() const { return (size_t) PyDict_Size(m_ptr); } - detail::dict_iterator begin() const { return (++detail::dict_iterator(ptr(), 0)); } + detail::dict_iterator begin() const { return (++detail::dict_iterator(*this, 0)); } detail::dict_iterator end() const { return detail::dict_iterator(); } void clear() const { PyDict_Clear(ptr()); } }; diff --git a/stormpy/resources/pybind11/include/pybind11/stl_bind.h b/stormpy/resources/pybind11/include/pybind11/stl_bind.h index 37839826d..a67f3ca0e 100644 --- a/stormpy/resources/pybind11/include/pybind11/stl_bind.h +++ b/stormpy/resources/pybind11/include/pybind11/stl_bind.h @@ -135,6 +135,8 @@ template , typename holder_ty pybind11::class_, holder_type> bind_vector(pybind11::module &m, std::string const &name, Args&&... args) { using Vector = std::vector; using SizeType = typename Vector::size_type; + using DiffType = typename Vector::difference_type; + using ItType = typename Vector::iterator; using Class_ = pybind11::class_; Class_ cl(m, name.c_str(), std::forward(args)...); @@ -161,7 +163,9 @@ pybind11::class_, holder_type> bind_vector(pybind11::m throw; } }); - cl.def("append", (void (Vector::*) (const T &)) & Vector::push_back, + + cl.def("append", + [](Vector &v, const T &value) { v.push_back(value); }, arg("x"), "Add an item to the end of the list"); @@ -176,7 +180,7 @@ pybind11::class_, holder_type> bind_vector(pybind11::m cl.def("insert", [](Vector &v, SizeType i, const T &x) { - v.insert(v.begin() + i, x); + v.insert(v.begin() + (DiffType) i, x); }, arg("i") , arg("x"), "Insert an item at a given position." @@ -198,7 +202,7 @@ pybind11::class_, holder_type> bind_vector(pybind11::m if (i >= v.size()) throw pybind11::index_error(); T t = v[i]; - v.erase(v.begin() + i); + v.erase(v.begin() + (DiffType) i); return t; }, arg("i"), @@ -213,7 +217,7 @@ pybind11::class_, holder_type> bind_vector(pybind11::m ); cl.def("__getitem__", - [](const Vector &v, SizeType i) { + [](const Vector &v, SizeType i) -> T { if (i >= v.size()) throw pybind11::index_error(); return v[i]; @@ -232,7 +236,7 @@ pybind11::class_, holder_type> bind_vector(pybind11::m [](Vector &v, SizeType i) { if (i >= v.size()) throw pybind11::index_error(); - v.erase(v.begin() + i); + v.erase(v.begin() + typename Vector::difference_type(i)); }, "Delete list elements using a slice object" ); @@ -241,7 +245,7 @@ pybind11::class_, holder_type> bind_vector(pybind11::m cl.def("__iter__", [](Vector &v) { - return pybind11::make_iterator(v.begin(), v.end()); + return pybind11::make_iterator(v.begin(), v.end()); }, pybind11::keep_alive<0, 1>() /* Essential: keep list alive while iterator exists */ ); @@ -249,7 +253,7 @@ pybind11::class_, holder_type> bind_vector(pybind11::m /// Slicing protocol cl.def("__getitem__", [](const Vector &v, slice slice) -> Vector * { - ssize_t start, stop, step, slicelength; + size_t start, stop, step, slicelength; if (!slice.compute(v.size(), &start, &stop, &step, &slicelength)) throw pybind11::error_already_set(); @@ -257,7 +261,7 @@ pybind11::class_, holder_type> bind_vector(pybind11::m Vector *seq = new Vector(); seq->reserve((size_t) slicelength); - for (int i=0; ipush_back(v[start]); start += step; } @@ -269,14 +273,14 @@ pybind11::class_, holder_type> bind_vector(pybind11::m cl.def("__setitem__", [](Vector &v, slice slice, const Vector &value) { - ssize_t start, stop, step, slicelength; + size_t start, stop, step, slicelength; if (!slice.compute(v.size(), &start, &stop, &step, &slicelength)) throw pybind11::error_already_set(); - if ((size_t) slicelength != value.size()) + if (slicelength != value.size()) throw std::runtime_error("Left and right hand size of slice assignment have different sizes!"); - for (int i=0; i, holder_type> bind_vector(pybind11::m cl.def("__delitem__", [](Vector &v, slice slice) { - ssize_t start, stop, step, slicelength; + size_t start, stop, step, slicelength; if (!slice.compute(v.size(), &start, &stop, &step, &slicelength)) throw pybind11::error_already_set(); if (step == 1 && false) { - v.erase(v.begin() + start, v.begin() + start + slicelength); + v.erase(v.begin() + (DiffType) start, v.begin() + DiffType(start + slicelength)); } else { - for (ssize_t i = 0; i < slicelength; ++i) { - v.erase(v.begin() + start); + for (size_t i = 0; i < slicelength; ++i) { + v.erase(v.begin() + DiffType(start)); start += step - 1; } } diff --git a/stormpy/resources/pybind11/pybind11/_version.py b/stormpy/resources/pybind11/pybind11/_version.py index afa4cc9be..be66f0732 100644 --- a/stormpy/resources/pybind11/pybind11/_version.py +++ b/stormpy/resources/pybind11/pybind11/_version.py @@ -1,2 +1,2 @@ -version_info = (1, 8, 'dev0') +version_info = (1, 9, 'dev0') __version__ = '.'.join(map(str, version_info))