scikit-build-core icon indicating copy to clipboard operation
scikit-build-core copied to clipboard

Precompiled Header dont work.

Open petrasvestartas opened this issue 8 months ago • 8 comments
trafficstars

I have the CMakeLists below that has the precompiled header setup.

But with scikit-build, every time I write pip install -e ., the whole c++ project is recompiled. Is there a way to turn on precompiled headers for scikit?


cmake_minimum_required(VERSION 3.15...3.26)

project(cgal_wrapper LANGUAGES CXX)
set(CMAKE_CXX_STANDARD 20)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
set(CMAKE_CXX_EXTENSIONS OFF)

# =====================================================================
# Dependencies
# =====================================================================
include(ExternalProject)

# Define source directories for external dependencies
set(EXTERNAL_DIR "${CMAKE_CURRENT_SOURCE_DIR}/external")
set(CGAL_SOURCE_DIR "${EXTERNAL_DIR}/cgal")
set(BOOST_SOURCE_DIR "${EXTERNAL_DIR}/boost")
set(EIGEN_SOURCE_DIR "${EXTERNAL_DIR}/eigen")

# Create a custom target for all external dependencies
add_custom_target(external_downloads ALL)

# ========================================================================
# CGAL
# ========================================================================
if(NOT EXISTS "${CGAL_SOURCE_DIR}")
    message(STATUS "Downloading CGAL...")
    ExternalProject_Add(
        cgal_download
        URL https://github.com/CGAL/cgal/releases/download/v6.0.1/CGAL-6.0.1.zip
        SOURCE_DIR       "${CGAL_SOURCE_DIR}"
        CONFIGURE_COMMAND ""
        BUILD_COMMAND     ""
        INSTALL_COMMAND   ""
        LOG_DOWNLOAD ON
        UPDATE_COMMAND    ""
        PATCH_COMMAND     ""
        TLS_VERIFY       ON
    )
    add_dependencies(external_downloads cgal_download)
endif()

# ========================================================================
# BOOST
# ========================================================================
if(NOT EXISTS "${BOOST_SOURCE_DIR}")
  message(STATUS "Configuring Boost...")
  ExternalProject_Add(boost_download
      URL https://archives.boost.io/release/1.82.0/source/boost_1_82_0.tar.gz
      SOURCE_DIR "${BOOST_SOURCE_DIR}"
      CONFIGURE_COMMAND ""
      BUILD_COMMAND ""
      INSTALL_COMMAND ${CMAKE_COMMAND} -E copy_directory <SOURCE_DIR>/boost ${BOOST_SOURCE_DIR}/boost
      LOG_DOWNLOAD 
      ON
  )
  
  add_dependencies(external_downloads boost_download)
endif()

# ========================================================================
# EIGEN
# ========================================================================
if(NOT EXISTS "${EIGEN_SOURCE_DIR}")
    message(STATUS "Downloading Eigen...")
    ExternalProject_Add(
        eigen_download
        URL https://gitlab.com/libeigen/eigen/-/archive/3.4.0/eigen-3.4.0.tar.gz
        SOURCE_DIR       "${EIGEN_SOURCE_DIR}"
        CONFIGURE_COMMAND ""
        BUILD_COMMAND     ""
        INSTALL_COMMAND   ""
        LOG_DOWNLOAD ON
        UPDATE_COMMAND    ""
        PATCH_COMMAND     ""
    )
    add_dependencies(external_downloads eigen_download)
endif()

# Add include directories
set(CGAL_INCLUDE_DIR "${CGAL_SOURCE_DIR}/include")
set(BOOST_INCLUDE_DIR "${BOOST_SOURCE_DIR}")
set(EIGEN_INCLUDE_DIR "${EIGEN_SOURCE_DIR}")

# Print include directories for verification
message(STATUS "====================================================")
message(STATUS "CGAL INCLUDE DIRECTORY: ${CGAL_INCLUDE_DIR}")
message(STATUS "BOOST INCLUDE DIRECTORY: ${BOOST_INCLUDE_DIR}")
message(STATUS "EIGEN INCLUDE DIRECTORY: ${EIGEN_INCLUDE_DIR}")
message(STATUS "====================================================")

include_directories(
    ${CGAL_INCLUDE_DIR}
    ${BOOST_INCLUDE_DIR}
    ${EIGEN_INCLUDE_DIR}
)

# Incase linking is not working check the includes
#  add_compile_options(/showIncludes)

# Define libraries as header only
add_definitions(
    -DCGAL_HEADER_ONLY
    -DBOOST_ALL_NO_LIB 
    -DBOOST_ALL_DYN_LINK=0
    -DCGAL_DISABLE_GMP
    -DCGAL_USE_BOOST_MP
)

# =====================================================================
# BOOST and CGAL flags to configure multiprecision
# =====================================================================
set(CGAL_DO_NOT_USE_MPZF ON CACHE BOOL "Do not use MPZF")
set(CGAL_USE_GMPXX OFF CACHE BOOL "Disable GMPXX")
set(CGAL_DISABLE_GMP ON CACHE BOOL "Disable GMP in CGAL")
set(CMAKE_DISABLE_FIND_PACKAGE_GMP ON CACHE BOOL "Disable CMake find package for GMP")
set(CMAKE_DISABLE_FIND_PACKAGE_MPFR ON CACHE BOOL "Disable CMake find package for MPFR")
set(CGAL_NT_USE_BOOST_MP ON CACHE BOOL "Use Boost Multiprecision")

# Eigen is header-only by default, no special defines needed

# =====================================================================
# Binding
# =====================================================================

if (NOT SKBUILD)
  message(WARNING "\
  This CMake file is meant to be executed using 'scikit-build'. Running
  it directly will almost certainly not produce the desired result. If
  you are a user trying to install this package, please use the command
  below, which will install all necessary build dependencies, compile
  the package in an isolated environment, and then install it.
  =====================================================================
   $ pip install .
  =====================================================================
  If you are a software developer, and this is your own package, then
  it is usually much more efficient to install the build dependencies
  in your environment once and use the following command that avoids
  a costly creation of a new virtual environment at every compilation:
  =====================================================================
   $ pip install nanobind scikit-build-core[pyproject]
   $ conda install cmake
   $ pip install --no-build-isolation -ve .
   $ pip install --no-build-isolation -ve . -Ceditable.rebuild=true
  =====================================================================
  You may optionally add -Ceditable.rebuild=true to auto-rebuild when
  the package is imported. Otherwise, you need to re-run the above
  after editing C++ files.")
endif()

# Try to import all Python components potentially needed by nanobind
find_package(Python 3.8
  REQUIRED COMPONENTS Interpreter Development.Module
  OPTIONAL_COMPONENTS Development.SABIModule)

# Import nanobind through CMake's find_package mechanism
find_package(nanobind CONFIG REQUIRED)

# We are now ready to compile the actual extension module
nanobind_add_module(
  compas_cgal_ext
  STABLE_ABI
  NB_STATIC
  src/compas_cgal.cpp
  src/meshing.cpp
)

# Set include directories for the extension
target_include_directories(compas_cgal_ext PRIVATE 
    ${CGAL_INCLUDE_DIR}
    ${BOOST_INCLUDE_DIR}
    ${EIGEN_INCLUDE_DIR}
)

# Simple precompiled header setup
target_precompile_headers(compas_cgal_ext PRIVATE src/stdafx.h)

# Make sure external dependencies are downloaded before building the extension
add_dependencies(compas_cgal_ext external_downloads)

# Install directive for scikit-build-core
install(TARGETS compas_cgal_ext LIBRARY DESTINATION compas_cgal)

###############################################################################
# PCH support
###############################################################################
set(CMAKE_PCH_INSTANTIATE_TEMPLATES ON)

petrasvestartas avatar Mar 04 '25 18:03 petrasvestartas

Have you tried setting tool.scikit-build.build-dir?

henryiii avatar Mar 04 '25 18:03 henryiii

Is there an example of any existing project how to set up this directory?

Image

Full github repo: https://github.com/compas-dev/compas_cgal/tree/da292f3b004e38c11c804bf18c2c4c0265750e5a

petrasvestartas avatar Mar 04 '25 19:03 petrasvestartas

You should make sure that build is in your gitignore, but otherwise, don't think there's anything to setup? Did setting it help?

henryiii avatar Mar 04 '25 20:03 henryiii

It's mostly up to how your developers tend to work. E.g. sometimes it's useful to have the build directory be python-version dependent, sometimes you want a predictable path so that you can tell your CI to run ctest. You can override it as you wish with pip --config-settings.

How we plan to deal with cmake presets might be relevant here.

LecrisUT avatar Mar 04 '25 21:03 LecrisUT

@henryiii

No, I do not know how to setup it correctly, still every time I do pip install -e . precompiled header is recompiled.

Here is the video of verbose compilation: https://www.youtube.com/watch?v=RhqDxqVsSfE

These precompiled headers in CMakelists have no effect:

# Enhanced PCH configuration
set(CMAKE_PCH_INSTANTIATE_TEMPLATES ON)
set(CMAKE_PCH_WARN_INVALID ON)

# Configure PCH for the extension
target_precompile_headers(compas_cgal_ext 
    PRIVATE 
    src/stdafx.h
)

# Install directive for scikit-build-core
install(TARGETS compas_cgal_ext LIBRARY DESTINATION compas_cgal)

Is there any way that you could clone the repository and try it to run?

petrasvestartas avatar Mar 04 '25 21:03 petrasvestartas

Are you using isolated builds? In an isolated build, nanobind gets reinstalled every time, so the paths change. It looks like nanobind's bath is part of the compilation line.

henryiii avatar Mar 04 '25 22:03 henryiii

Does it work when run locally outside of SKBUILD? A dockerfile or github wotkflow would be helpful to navigate the build log

LecrisUT avatar Mar 04 '25 22:03 LecrisUT

Thank you,

This worked:

   $ pip install nanobind scikit-build-core[pyproject]
   $ conda install cmake
   $ pip install --no-build-isolation -ve .
   $ pip install --no-build-isolation -ve . -Ceditable.rebuild=true

Built time decreased from 18 to 8 seconds using precompiled headers.

I am checking it seems stdafx.h is skipped.

petrasvestartas avatar Mar 04 '25 22:03 petrasvestartas