Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Provide CMake-driver for building libcustom_calls #1345

Open
wants to merge 38 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
38 commits
Select commit Hold shift + click to select a range
3fa41d0
Begin scaffolding for libcustom_call in cmake
mlxd Nov 29, 2024
969240d
Ensure nested cmakes use project scoping
mlxd Nov 29, 2024
2a1b6cc
Remove python driven build for libcustom_calls.so
mlxd Nov 29, 2024
58c27f7
Fix frontend build structure
mlxd Nov 29, 2024
ed676eb
Update Lapack submodule build setup
mlxd Nov 29, 2024
0e1fb47
Build libcustom_calls as a standalone shared lib
mlxd Nov 29, 2024
1e55e5e
Tidy lapack submodule
mlxd Nov 29, 2024
1af2b86
Remove libcustom_calls from setup.py and simplify
mlxd Nov 29, 2024
e3adebf
Update docstring for cmake builder
mlxd Dec 6, 2024
c0fca1f
Merge branch 'main' into cmakeify_frontend
mlxd Dec 6, 2024
5df5dd9
Fix black format
mlxd Dec 6, 2024
45d8aca
Merge branch 'cmakeify_frontend' of github.com:PennyLaneAI/catalyst i…
mlxd Dec 6, 2024
5f9c3ac
Trigger wheel builds
mlxd Dec 6, 2024
5aa20b0
Ensure libcustom follows OS native library naming
mlxd Dec 9, 2024
a31e53f
Merge branch 'main' into cmakeify_frontend
mlxd Dec 9, 2024
d5b6e4b
Fix missing path in setup
mlxd Dec 10, 2024
0ae1613
Merge branch 'main' into cmakeify_frontend
mlxd Dec 10, 2024
fe27420
Merge branch 'main' into cmakeify_frontend
mlxd Dec 10, 2024
67874e7
Force macos wheel builds to ignore universal
mlxd Dec 11, 2024
f5c28d4
Merge branch 'cmakeify_frontend' of github.com:PennyLaneAI/catalyst i…
mlxd Dec 11, 2024
8520a69
Merge branch 'main' into cmakeify_frontend
mlxd Dec 11, 2024
c68f237
Update setup.py
mlxd Dec 11, 2024
bd3d195
Update setup.py
mlxd Dec 11, 2024
4557ed9
Update setup.py
mlxd Dec 11, 2024
33ea71e
Update frontend/catalyst/utils/CMakeLists.txt
mlxd Dec 11, 2024
7b4a034
Update frontend/catalyst/utils/jax_cpu_lapack_kernels/CMakeLists.txt
mlxd Dec 11, 2024
9aee105
remove unneeded branching
mlxd Dec 11, 2024
b73ffde
Remove shared-lib specifics from builder
mlxd Dec 11, 2024
32ceb39
Remove CMake BLAS override option for custom calls lib
mlxd Dec 11, 2024
3f2b6b9
Rename to .so for now
mlxd Dec 11, 2024
bc8c51d
Merge branch 'main' into cmakeify_frontend
mlxd Dec 11, 2024
cc3e77a
Fix cmake target name
mlxd Dec 11, 2024
b6441c3
Merge branch 'cmakeify_frontend' of github.com:PennyLaneAI/catalyst i…
mlxd Dec 11, 2024
9386d07
Force builder to use MacOS 13 as min wheel version
mlxd Dec 13, 2024
8580971
Fix missing list
mlxd Dec 13, 2024
599d987
Force set min mac version for wheel naming
mlxd Dec 13, 2024
7350689
Merge branch 'main' into cmakeify_frontend
mlxd Dec 13, 2024
10e33d3
Add missing platform tag
mlxd Dec 13, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 3 additions & 1 deletion .github/workflows/build-wheel-macos-arm64.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,9 @@ on:
workflow_call:

env:
MACOSX_DEPLOYMENT_TARGET: 14.0
MACOSX_DEPLOYMENT_TARGET: 13.0
_PYTHON_HOST_PLATFORM: "macosx-13.0-arm64"
ARCHFLAGS: "-arch arm64"

concurrency:
group: Build Catalyst Wheel on macOS (arm64)-${{ github.ref }}
Expand Down
2 changes: 2 additions & 0 deletions .github/workflows/build-wheel-macos-x86_64.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@ on:

env:
MACOSX_DEPLOYMENT_TARGET: 13
_PYTHON_HOST_PLATFORM: "macosx-13.0-x86_64"
ARCHFLAGS: "-arch x86_64"

concurrency:
group: Build Catalyst Wheel on macOS (x86_64)-${{ github.ref }}
Expand Down
13 changes: 13 additions & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
cmake_minimum_required(VERSION 3.20)
project(catalyst)

set(LOGO [=[
░█▀▀░█▀█░▀█▀░█▀█░█░░░█░█░█▀▀░▀█▀
░█░░░█▀█░░█░░█▀█░█░░░░█░░▀▀█░░█░
░▀▀▀░▀░▀░░▀░░▀░▀░▀▀▀░░▀░░▀▀▀░░▀░
mlxd marked this conversation as resolved.
Show resolved Hide resolved
]=])
message(${LOGO})

add_subdirectory(frontend)
add_subdirectory(mlir)
add_subdirectory(runtime)
2 changes: 2 additions & 0 deletions frontend/catalyst/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -1 +1,3 @@
project(catalyst_frontend)

add_subdirectory(utils)
39 changes: 30 additions & 9 deletions frontend/catalyst/utils/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -1,9 +1,31 @@
project(catalyst_frontend_utils)
dime10 marked this conversation as resolved.
Show resolved Hide resolved

set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)

# Find the Python interpreter, development headers and NumPy headers
find_package(Python 3
REQUIRED COMPONENTS Interpreter Development.Module
REQUIRED COMPONENTS Interpreter Development.Module NumPy
OPTIONAL_COMPONENTS Development.SABIModule)
include_directories(${PYTHON_INCLUDE_DIRS})

# Build the custom calls library and make `catalyst_jax_cpu_lapack_kernels`
# target visible in the current project directory.
add_subdirectory(jax_cpu_lapack_kernels)

add_library(custom_calls SHARED
${PROJECT_SOURCE_DIR}/libcustom_calls.cpp
)
target_link_libraries(custom_calls PUBLIC "$<LINK_LIBRARY:WHOLE_ARCHIVE,catalyst_jax_cpu_lapack_kernels>")
set_property(TARGET custom_calls PROPERTY ENABLE_EXPORTS OFF)

# If aiming to build additional shared submodules, setting the RPATH $ORIGIN
# value may be useful to avoid explicit library paths.
set_target_properties(custom_calls PROPERTIES BUILD_RPATH "$ORIGIN/")

# Editable installations do not respect `package_data` in setup.py so
# build to mimic extension module expectations from setuptools+distutils
set_target_properties(custom_calls PROPERTIES SUFFIX ".so")

# nanobind suggests including these lines to configure CMake to perform an optimized release build
# by default unless another build type is specified. Without this addition, binding code may run
Expand All @@ -21,13 +43,6 @@ execute_process(

find_package(nanobind CONFIG REQUIRED)

# Get the NumPy include directory
execute_process(
COMMAND "${Python_EXECUTABLE}" -c "import numpy; print(numpy.get_include())"
OUTPUT_VARIABLE NUMPY_INCLUDE_DIR
OUTPUT_STRIP_TRAILING_WHITESPACE
)

# Source file list for `wrapper` module
set(WRAPPER_SRC_FILES
wrapper.cpp
Expand All @@ -39,8 +54,14 @@ set(WRAPPER_SRC_FILES
nanobind_add_module(wrapper STABLE_ABI ${WRAPPER_SRC_FILES})

# Add the NumPy include directory to the library's include paths
target_include_directories(wrapper PRIVATE ${NUMPY_INCLUDE_DIR})
target_include_directories(wrapper PRIVATE ${Python_NumPy_INCLUDE_DIRS})

# Use suffix ".so" rather than ".abi3.so" for library file using Stable ABI
# This is necessary for compatibility with setuptools build extensions
set_target_properties(wrapper PROPERTIES SUFFIX ".so")

# Allow custom calls to be built with explicit linkage against wrapper
add_dependencies(wrapper custom_calls)

# Explicitly set RPATH to allow access to libraries within same directory
set_target_properties(wrapper PROPERTIES BUILD_RPATH "$ORIGIN")
23 changes: 23 additions & 0 deletions frontend/catalyst/utils/jax_cpu_lapack_kernels/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
project(catalyst_jax_cpu_lapack_kernels)

# Create a static library for LAPACK kernels to be included into the
# custom_calls library in the parent scope.

add_library(catalyst_jax_cpu_lapack_kernels STATIC
${PROJECT_SOURCE_DIR}/lapack_kernels.cpp
${PROJECT_SOURCE_DIR}/lapack_kernels_using_lapack.cpp
)

# Ensure fPIC is set for static builds
set_property(TARGET catalyst_jax_cpu_lapack_kernels PROPERTY POSITION_INDEPENDENT_CODE ON)

# Headers are tracked into the target from the current project scope
target_include_directories(catalyst_jax_cpu_lapack_kernels PUBLIC
$<BUILD_INTERFACE:${PROJECT_SOURCE_DIR}>
$<INSTALL_INTERFACE:include>
)

# If building on MacOS avoid the linker complaining about undefined symbols
if(APPLE)
target_link_options(catalyst_jax_cpu_lapack_kernels PUBLIC -undefined dynamic_lookup)
endif()
Original file line number Diff line number Diff line change
Expand Up @@ -113,7 +113,6 @@ jax::Sytrd<std::complex<double>>::FnType GET_SYMBOL(LAPACKE_zhetrd);
} // extern "C"

namespace jax {

static auto init = []() -> int {
RealTrsm<float>::fn = GET_SYMBOL(cblas_strsm);
RealTrsm<double>::fn = GET_SYMBOL(cblas_dtrsm);
Expand Down
119 changes: 28 additions & 91 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,6 @@

# pylint: disable=wrong-import-order

import glob
import os
import platform
import subprocess
Expand Down Expand Up @@ -69,6 +68,14 @@
lightning_dep = f"pennylane-lightning>={lq_min_release}"
kokkos_dep = ""

# Ensure MacOS minimum version is set for wheel & CMake builder
if platform.system() == "Darwin":
if val := os.environ.get("MACOSX_DEPLOYMENT_TARGET"):
MacOS_SDK_VERSION = val
else:
MacOS_SDK_VERSION = "13.0"
os.environ["_PYTHON_HOST_PLATFORM"] = f"macosx-{MacOS_SDK_VERSION}-{platform.machine()}"

requirements = [
pennylane_dep,
lightning_dep,
Expand Down Expand Up @@ -130,30 +137,29 @@


class CMakeExtension(Extension):
"""A setuptools Extension class for modules with a CMake configuration."""
"""setuptools Extension wrapper for CMake.

Provides access to the source directories directly for modules to be compiled.

For example, to build the `libcustom_calls` library, the direct module path can be provided as
```python
CMakeExtension("catalyst.utils.libcustom_calls", sourcedir=frontend_dir)
```
"""

def __init__(self, name, sourcedir=""):
super().__init__(name, sources=[])
Extension.__init__(self, name, sources=[])
self.sourcedir = os.path.abspath(sourcedir)


class UnifiedBuildExt(build_ext):
class CMakeBuild(build_ext):
"""Custom build extension class for the Catalyst Frontend.

This class overrides a number of methods from its parent class
setuptools.command.build_ext.build_ext, the most important of which are:

1. `get_ext_filename`, in order to remove the architecture/python
version suffix of the library name.
2. `build_extension`, in order to handle the compilation of extensions
with CMake configurations, namely the catalyst.utils.wrapper module,
and of generic C/C++ extensions without a CMake configuration, namely
the catalyst.utils.libcustom_calls module, which is currently built
as a plain setuptools Extension.

TODO: Eventually it would be better to build the utils.libcustom_calls
module using a CMake configuration as well, rather than as a setuptools
Extension.
"""

def initialize_options(self):
Expand All @@ -179,12 +185,6 @@ def get_ext_filename(self, fullname):
return filename.replace(suffix, "") + extension

def build_extension(self, ext):
if isinstance(ext, CMakeExtension):
self.build_cmake_extension(ext)
else:
super().build_extension(ext)

def build_cmake_extension(self, ext: CMakeExtension):
"""Configure and build CMake extension."""
cmake_path = "cmake"
ninja_path = "ninja"
Expand Down Expand Up @@ -219,6 +219,10 @@ def build_cmake_extension(self, ext: CMakeExtension):

configure_args += self.cmake_defines

if platform.system() == "Darwin":
# Ensure use of -mmacosx-version-min=X compiler argument
configure_args += [f"-DCMAKE_OSX_DEPLOYMENT_TARGET={MacOS_SDK_VERSION}"]

if "CMAKE_ARGS" in os.environ:
configure_args += os.environ["CMAKE_ARGS"].split(" ")

Expand All @@ -231,85 +235,17 @@ def build_cmake_extension(self, ext: CMakeExtension):
subprocess.check_call(
[cmake_path, "-G", "Ninja", ext.sourcedir] + configure_args, cwd=build_temp
)
subprocess.check_call([cmake_path, "--build", "."] + build_args, cwd=build_temp)


class CustomBuildExtLinux(UnifiedBuildExt):
"""Custom build extension class for Linux platforms

Currently no extra work needs to be performed with respect to the base class
UnifiedBuildExt.
"""


class CustomBuildExtMacos(UnifiedBuildExt):
"""Custom build extension class for macOS platforms

In addition to the work performed by the base class UnifiedBuildExt, this
class also changes the LC_ID_DYLIB that is otherwise constant and equal to
where the shared library was created.
"""

def run(self):
# Run the original build_ext command
super().run()

# Construct library name based on ext suffix (contains python version, architecture and .so)
library_name = "libcustom_calls.so"

package_root = os.path.dirname(__file__)
frontend_path = glob.glob(
os.path.join(package_root, "frontend", "**", library_name), recursive=True
)
build_path = glob.glob(os.path.join("build", "**", library_name), recursive=True)
lib_with_r_path = "@rpath/libcustom_calls.so"

original_path = frontend_path[0] if frontend_path else build_path[0]

# Run install_name_tool to modify LC_ID_DYLIB(other the rpath stays in vars/folder)
subprocess.run(
["/usr/bin/install_name_tool", "-id", lib_with_r_path, original_path],
check=False,
subprocess.check_call(
[cmake_path, "--build", ".", "--verbose"] + build_args, cwd=build_temp
)


# Compile the library of custom calls in the frontend
if system_platform == "Linux":
custom_calls_extension = Extension(
"catalyst.utils.libcustom_calls",
sources=[
"frontend/catalyst/utils/libcustom_calls.cpp",
"frontend/catalyst/utils/jax_cpu_lapack_kernels/lapack_kernels.cpp",
"frontend/catalyst/utils/jax_cpu_lapack_kernels/lapack_kernels_using_lapack.cpp",
],
extra_compile_args=["-std=c++17"],
)
cmdclass = {"build_ext": CustomBuildExtLinux}

elif system_platform == "Darwin":
variables = sysconfig.get_config_vars()
# Here we need to switch the deault to MacOs dynamic lib
variables["LDSHARED"] = variables["LDSHARED"].replace("-bundle", "-dynamiclib")
if sysconfig.get_config_var("LDCXXSHARED"):
variables["LDCXXSHARED"] = variables["LDCXXSHARED"].replace("-bundle", "-dynamiclib")
custom_calls_extension = Extension(
"catalyst.utils.libcustom_calls",
sources=[
"frontend/catalyst/utils/libcustom_calls.cpp",
"frontend/catalyst/utils/jax_cpu_lapack_kernels/lapack_kernels.cpp",
"frontend/catalyst/utils/jax_cpu_lapack_kernels/lapack_kernels_using_lapack.cpp",
],
extra_compile_args=["-std=c++17"],
)
cmdclass = {"build_ext": CustomBuildExtMacos}


project_root_dir = os.path.abspath(os.path.dirname(__file__))
frontend_dir = os.path.join(project_root_dir, "frontend")

ext_modules = [
custom_calls_extension,
CMakeExtension("catalyst.utils.wrapper", sourcedir=frontend_dir),
CMakeExtension("catalyst.utils.libcustom_calls", sourcedir=frontend_dir),
]

options = {"bdist_wheel": {"py_limited_api": "cp312"}} if sys.hexversion >= 0x030C0000 else {}
Expand All @@ -320,6 +256,7 @@ def run(self):
# - `ops`: Path to the compiler operations module.
# - `qjit`: Path to the JIT compiler decorator provided by the compiler.


setup(
classifiers=classifiers,
name="PennyLane-Catalyst",
Expand All @@ -336,7 +273,7 @@ def run(self):
package_dir={"": "frontend"},
include_package_data=True,
ext_modules=ext_modules,
cmdclass=cmdclass,
cmdclass={"build_ext": CMakeBuild},
**description,
options=options,
)
Loading