mirror of
https://github.com/ml-explore/mlx.git
synced 2025-11-01 08:38:12 +08:00
* add fp8 e4m3 converters * add cuda * default saturate to min/max * fix for older OS * fix no gpu/cpu * fix saturate * fix compile
359 lines
12 KiB
CMake
359 lines
12 KiB
CMake
cmake_minimum_required(VERSION 3.25)
|
||
|
||
if(NOT MLX_VERSION)
|
||
file(STRINGS "mlx/version.h" _mlx_h_version REGEX "^#define MLX_VERSION_.*$")
|
||
string(REGEX MATCH "#define MLX_VERSION_MAJOR ([0-9]+)" _ "${_mlx_h_version}")
|
||
set(_major ${CMAKE_MATCH_1})
|
||
string(REGEX MATCH "#define MLX_VERSION_MINOR ([0-9]+)" _ "${_mlx_h_version}")
|
||
set(_minor ${CMAKE_MATCH_1})
|
||
string(REGEX MATCH "#define MLX_VERSION_PATCH ([0-9]+)" _ "${_mlx_h_version}")
|
||
set(_patch ${CMAKE_MATCH_1})
|
||
set(MLX_PROJECT_VERSION "${_major}.${_minor}.${_patch}")
|
||
set(MLX_VERSION ${MLX_PROJECT_VERSION})
|
||
else()
|
||
string(REGEX REPLACE "^([0-9]+\.[0-9]+\.[0-9]+).*" "\\1" MLX_PROJECT_VERSION
|
||
${MLX_VERSION})
|
||
endif()
|
||
|
||
project(
|
||
mlx
|
||
LANGUAGES C CXX
|
||
VERSION ${MLX_PROJECT_VERSION})
|
||
|
||
# ----------------------------- Setup -----------------------------
|
||
set(CMAKE_MODULE_PATH "${PROJECT_SOURCE_DIR}/cmake")
|
||
set(CMAKE_CXX_STANDARD 17)
|
||
set(CMAKE_CXX_STANDARD_REQUIRED ON)
|
||
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
||
set(CMAKE_INSTALL_MESSAGE NEVER)
|
||
set(CMAKE_EXPORT_COMPILE_COMMANDS ON)
|
||
|
||
# ----------------------------- Configuration -----------------------------
|
||
option(MLX_BUILD_TESTS "Build tests for mlx" ON)
|
||
option(MLX_BUILD_EXAMPLES "Build examples for mlx" ON)
|
||
option(MLX_BUILD_BENCHMARKS "Build benchmarks for mlx" OFF)
|
||
option(MLX_BUILD_PYTHON_BINDINGS "Build python bindings for mlx" OFF)
|
||
option(MLX_BUILD_METAL "Build metal backend" ON)
|
||
option(MLX_BUILD_CPU "Build cpu backend" ON)
|
||
option(MLX_BUILD_CUDA "Build cuda backend" OFF)
|
||
option(MLX_METAL_DEBUG "Enhance metal debug workflow" OFF)
|
||
option(MLX_ENABLE_X64_MAC "Enable building for x64 macOS" OFF)
|
||
option(MLX_BUILD_GGUF "Include support for GGUF format" ON)
|
||
option(MLX_BUILD_SAFETENSORS "Include support for safetensors format" ON)
|
||
option(MLX_BUILD_BLAS_FROM_SOURCE "Build OpenBLAS from source code" OFF)
|
||
option(MLX_METAL_JIT "Use JIT compilation for Metal kernels" OFF)
|
||
option(MLX_USE_CCACHE "Use CCache for compilation cache when available" ON)
|
||
option(BUILD_SHARED_LIBS "Build mlx as a shared library" OFF)
|
||
option(USE_SYSTEM_FMT "Use system's provided fmt library" OFF)
|
||
|
||
# --------------------- Processor tests -------------------------
|
||
message(
|
||
STATUS
|
||
"Building MLX for ${CMAKE_SYSTEM_PROCESSOR} processor on ${CMAKE_SYSTEM_NAME}"
|
||
)
|
||
|
||
if(${CMAKE_SYSTEM_NAME} MATCHES "Darwin")
|
||
if(${CMAKE_SYSTEM_PROCESSOR} MATCHES "x86_64")
|
||
if(NOT MLX_ENABLE_X64_MAC)
|
||
message(
|
||
FATAL_ERROR
|
||
"Building for x86_64 on macOS is not supported."
|
||
" If you are on an Apple silicon system, check the build"
|
||
" documentation for possible fixes: "
|
||
"https://ml-explore.github.io/mlx/build/html/install.html#build-from-source"
|
||
)
|
||
else()
|
||
set(MLX_BUILD_METAL OFF)
|
||
message(WARNING "Building for x86_64 arch is not officially supported.")
|
||
endif()
|
||
endif()
|
||
else()
|
||
set(MLX_BUILD_METAL OFF)
|
||
endif()
|
||
|
||
if(MLX_USE_CCACHE)
|
||
find_program(CCACHE_PROGRAM ccache)
|
||
if(CCACHE_PROGRAM)
|
||
set(CMAKE_C_COMPILER_LAUNCHER "${CCACHE_PROGRAM}")
|
||
set(CMAKE_CXX_COMPILER_LAUNCHER "${CCACHE_PROGRAM}")
|
||
set(CMAKE_CUDA_COMPILER_LAUNCHER "${CCACHE_PROGRAM}")
|
||
endif()
|
||
endif()
|
||
|
||
# ----------------------------- Lib -----------------------------
|
||
|
||
include(FetchContent)
|
||
# Avoid warning about DOWNLOAD_EXTRACT_TIMESTAMP in CMake 3.24:
|
||
cmake_policy(SET CMP0135 NEW)
|
||
|
||
add_library(mlx)
|
||
|
||
# Supress warnings: note: parameter passing for argument of type
|
||
# ‘std::pair<float, float>’ when C++17 is enabled changed to match C++14 in GCC
|
||
# 10.1
|
||
target_compile_options(mlx PRIVATE -Wno-psabi)
|
||
|
||
if(MLX_BUILD_CUDA)
|
||
enable_language(CUDA)
|
||
endif()
|
||
|
||
if(MLX_BUILD_METAL)
|
||
find_library(METAL_LIB Metal)
|
||
find_library(FOUNDATION_LIB Foundation)
|
||
find_library(QUARTZ_LIB QuartzCore)
|
||
if(METAL_LIB)
|
||
message(STATUS "Metal found ${METAL_LIB}")
|
||
else()
|
||
message(
|
||
FATAL_ERROR
|
||
"Metal not found. Set MLX_BUILD_METAL=OFF to build without GPU")
|
||
endif()
|
||
|
||
if(MLX_METAL_DEBUG)
|
||
add_compile_definitions(MLX_METAL_DEBUG)
|
||
endif()
|
||
|
||
# Throw an error if xcrun not found
|
||
execute_process(
|
||
COMMAND zsh "-c" "/usr/bin/xcrun -sdk macosx --show-sdk-version"
|
||
OUTPUT_VARIABLE MACOS_SDK_VERSION
|
||
OUTPUT_STRIP_TRAILING_WHITESPACE COMMAND_ERROR_IS_FATAL ANY)
|
||
|
||
if(${MACOS_SDK_VERSION} LESS 14.0)
|
||
message(
|
||
FATAL_ERROR
|
||
"MLX requires macOS SDK >= 14.0 to be built with MLX_BUILD_METAL=ON")
|
||
endif()
|
||
message(STATUS "Building with macOS SDK version ${MACOS_SDK_VERSION}")
|
||
|
||
set(METAL_CPP_URL
|
||
https://developer.apple.com/metal/cpp/files/metal-cpp_macOS15_iOS18.zip)
|
||
|
||
if(NOT CMAKE_OSX_DEPLOYMENT_TARGET STREQUAL "")
|
||
set(XCRUN_FLAGS "-mmacosx-version-min=${CMAKE_OSX_DEPLOYMENT_TARGET}")
|
||
endif()
|
||
execute_process(
|
||
COMMAND
|
||
zsh "-c"
|
||
"echo \"__METAL_VERSION__\" | xcrun -sdk macosx metal ${XCRUN_FLAGS} -E -x metal -P - | tail -1 | tr -d '\n'"
|
||
OUTPUT_VARIABLE MLX_METAL_VERSION COMMAND_ERROR_IS_FATAL ANY)
|
||
FetchContent_Declare(metal_cpp URL ${METAL_CPP_URL})
|
||
|
||
FetchContent_MakeAvailable(metal_cpp)
|
||
target_include_directories(
|
||
mlx PUBLIC $<BUILD_INTERFACE:${metal_cpp_SOURCE_DIR}>
|
||
$<INSTALL_INTERFACE:include/metal_cpp>)
|
||
target_link_libraries(mlx PUBLIC ${METAL_LIB} ${FOUNDATION_LIB} ${QUARTZ_LIB})
|
||
endif()
|
||
|
||
if(CMAKE_SYSTEM_NAME STREQUAL "Linux")
|
||
# With newer clang/gcc versions following libs are implicitly linked, but when
|
||
# building on old distributions they need to be explicitly listed.
|
||
target_link_libraries(mlx PRIVATE dl pthread)
|
||
endif()
|
||
|
||
if(WIN32)
|
||
if(MSVC)
|
||
# GGUF does not build with MSVC.
|
||
set(MLX_BUILD_GGUF OFF)
|
||
# There is no prebuilt OpenBLAS distribution for MSVC.
|
||
set(MLX_BUILD_BLAS_FROM_SOURCE ON)
|
||
endif()
|
||
# Windows implementation of dlfcn.h APIs.
|
||
FetchContent_Declare(
|
||
dlfcn-win32
|
||
GIT_REPOSITORY https://github.com/dlfcn-win32/dlfcn-win32.git
|
||
GIT_TAG v1.4.1
|
||
EXCLUDE_FROM_ALL)
|
||
block()
|
||
set(BUILD_SHARED_LIBS OFF)
|
||
FetchContent_MakeAvailable(dlfcn-win32)
|
||
endblock()
|
||
target_include_directories(mlx PRIVATE "${dlfcn-win32_SOURCE_DIR}/src")
|
||
target_link_libraries(mlx PRIVATE dl)
|
||
endif()
|
||
|
||
if(MLX_BUILD_CPU)
|
||
find_library(ACCELERATE_LIBRARY Accelerate)
|
||
if(ACCELERATE_LIBRARY)
|
||
message(STATUS "Accelerate found ${ACCELERATE_LIBRARY}")
|
||
set(MLX_BUILD_ACCELERATE ON)
|
||
else()
|
||
message(STATUS "Accelerate not found, using default backend.")
|
||
set(MLX_BUILD_ACCELERATE OFF)
|
||
endif()
|
||
|
||
if(MLX_BUILD_ACCELERATE)
|
||
target_link_libraries(mlx PUBLIC ${ACCELERATE_LIBRARY})
|
||
add_compile_definitions(MLX_USE_ACCELERATE)
|
||
add_compile_definitions(ACCELERATE_NEW_LAPACK)
|
||
elseif(MLX_BUILD_BLAS_FROM_SOURCE)
|
||
# Download and build OpenBLAS from source code.
|
||
FetchContent_Declare(
|
||
openblas
|
||
GIT_REPOSITORY https://github.com/OpenMathLib/OpenBLAS.git
|
||
GIT_TAG v0.3.28
|
||
EXCLUDE_FROM_ALL)
|
||
set(BUILD_STATIC_LIBS ON) # link statically
|
||
set(NOFORTRAN ON) # msvc has no fortran compiler
|
||
FetchContent_MakeAvailable(openblas)
|
||
target_link_libraries(mlx PRIVATE openblas)
|
||
target_include_directories(
|
||
mlx PRIVATE "${openblas_SOURCE_DIR}/lapack-netlib/LAPACKE/include"
|
||
"${CMAKE_BINARY_DIR}/generated" "${CMAKE_BINARY_DIR}")
|
||
else()
|
||
if(${CMAKE_HOST_APPLE})
|
||
# The blas shipped in macOS SDK is not supported, search homebrew for
|
||
# openblas instead.
|
||
set(BLA_VENDOR OpenBLAS)
|
||
set(LAPACK_ROOT
|
||
"${LAPACK_ROOT};$ENV{LAPACK_ROOT};/usr/local/opt/openblas")
|
||
endif()
|
||
# Search and link with lapack.
|
||
find_package(LAPACK REQUIRED)
|
||
if(NOT LAPACK_FOUND)
|
||
message(FATAL_ERROR "Must have LAPACK installed")
|
||
endif()
|
||
find_path(LAPACK_INCLUDE_DIRS lapacke.h /usr/include /usr/local/include
|
||
/usr/local/opt/openblas/include)
|
||
message(STATUS "Lapack lib " ${LAPACK_LIBRARIES})
|
||
message(STATUS "Lapack include " ${LAPACK_INCLUDE_DIRS})
|
||
target_include_directories(mlx PRIVATE ${LAPACK_INCLUDE_DIRS})
|
||
target_link_libraries(mlx PRIVATE ${LAPACK_LIBRARIES})
|
||
# List blas after lapack otherwise we may accidentally incldue an old
|
||
# version of lapack.h from the include dirs of blas.
|
||
find_package(BLAS REQUIRED)
|
||
if(NOT BLAS_FOUND)
|
||
message(FATAL_ERROR "Must have BLAS installed")
|
||
endif()
|
||
# TODO find a cleaner way to do this
|
||
find_path(BLAS_INCLUDE_DIRS cblas.h /usr/include /usr/local/include
|
||
$ENV{BLAS_HOME}/include)
|
||
message(STATUS "Blas lib " ${BLAS_LIBRARIES})
|
||
message(STATUS "Blas include " ${BLAS_INCLUDE_DIRS})
|
||
target_include_directories(mlx PRIVATE ${BLAS_INCLUDE_DIRS})
|
||
target_link_libraries(mlx PRIVATE ${BLAS_LIBRARIES})
|
||
endif()
|
||
else()
|
||
set(MLX_BUILD_ACCELERATE OFF)
|
||
endif()
|
||
|
||
message(STATUS "Downloading json")
|
||
FetchContent_Declare(
|
||
json
|
||
URL https://github.com/nlohmann/json/releases/download/v3.11.3/json.tar.xz)
|
||
FetchContent_MakeAvailable(json)
|
||
target_include_directories(
|
||
mlx PRIVATE $<BUILD_INTERFACE:${json_SOURCE_DIR}/single_include/nlohmann>)
|
||
|
||
add_subdirectory(${CMAKE_CURRENT_LIST_DIR}/mlx)
|
||
|
||
target_include_directories(
|
||
mlx PUBLIC $<BUILD_INTERFACE:${CMAKE_CURRENT_LIST_DIR}>
|
||
$<INSTALL_INTERFACE:include>)
|
||
|
||
# Do not add mlx_EXPORTS define for shared library.
|
||
set_target_properties(mlx PROPERTIES DEFINE_SYMBOL "")
|
||
|
||
if(USE_SYSTEM_FMT)
|
||
find_package(fmt REQUIRED)
|
||
else()
|
||
FetchContent_Declare(
|
||
fmt
|
||
GIT_REPOSITORY https://github.com/fmtlib/fmt.git
|
||
GIT_TAG 10.2.1
|
||
EXCLUDE_FROM_ALL)
|
||
FetchContent_MakeAvailable(fmt)
|
||
endif()
|
||
target_link_libraries(mlx PRIVATE $<BUILD_INTERFACE:fmt::fmt-header-only>)
|
||
|
||
if(MLX_BUILD_PYTHON_BINDINGS)
|
||
message(STATUS "Building Python bindings.")
|
||
find_package(
|
||
Python 3.8
|
||
COMPONENTS Interpreter Development.Module
|
||
REQUIRED)
|
||
execute_process(
|
||
COMMAND "${Python_EXECUTABLE}" -m nanobind --cmake_dir
|
||
OUTPUT_STRIP_TRAILING_WHITESPACE
|
||
OUTPUT_VARIABLE nanobind_ROOT)
|
||
find_package(nanobind CONFIG REQUIRED)
|
||
add_subdirectory(${CMAKE_CURRENT_LIST_DIR}/python/src)
|
||
endif()
|
||
|
||
if(MLX_BUILD_TESTS)
|
||
include(CTest)
|
||
add_subdirectory(${CMAKE_CURRENT_LIST_DIR}/tests)
|
||
endif()
|
||
|
||
if(MLX_BUILD_EXAMPLES)
|
||
add_subdirectory(${CMAKE_CURRENT_LIST_DIR}/examples/cpp)
|
||
endif()
|
||
|
||
if(MLX_BUILD_BENCHMARKS)
|
||
add_subdirectory(${CMAKE_CURRENT_LIST_DIR}/benchmarks/cpp)
|
||
endif()
|
||
|
||
# ----------------------------- Installation -----------------------------
|
||
include(GNUInstallDirs)
|
||
|
||
# Install library
|
||
install(
|
||
TARGETS mlx
|
||
EXPORT MLXTargets
|
||
LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
|
||
ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}
|
||
RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}
|
||
INCLUDES
|
||
DESTINATION ${CMAKE_INSTALL_INCLUDEDIR})
|
||
|
||
# Install headers
|
||
install(
|
||
DIRECTORY ${CMAKE_CURRENT_LIST_DIR}/mlx
|
||
DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}
|
||
COMPONENT headers
|
||
FILES_MATCHING
|
||
PATTERN "*.h"
|
||
PATTERN "backend/metal/kernels.h" EXCLUDE)
|
||
|
||
# Install metal dependencies
|
||
if(MLX_BUILD_METAL)
|
||
|
||
# Install metal cpp
|
||
install(
|
||
DIRECTORY ${metal_cpp_SOURCE_DIR}/
|
||
DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/metal_cpp
|
||
COMPONENT metal_cpp_source)
|
||
|
||
endif()
|
||
|
||
# Install cmake config
|
||
set(MLX_CMAKE_BUILD_CONFIG ${CMAKE_BINARY_DIR}/MLXConfig.cmake)
|
||
set(MLX_CMAKE_BUILD_VERSION_CONFIG ${CMAKE_BINARY_DIR}/MLXConfigVersion.cmake)
|
||
set(MLX_CMAKE_INSTALL_MODULE_DIR share/cmake/MLX)
|
||
|
||
install(
|
||
EXPORT MLXTargets
|
||
FILE MLXTargets.cmake
|
||
DESTINATION ${MLX_CMAKE_INSTALL_MODULE_DIR})
|
||
|
||
include(CMakePackageConfigHelpers)
|
||
|
||
write_basic_package_version_file(
|
||
${MLX_CMAKE_BUILD_VERSION_CONFIG}
|
||
COMPATIBILITY SameMajorVersion
|
||
VERSION ${MLX_VERSION})
|
||
|
||
configure_package_config_file(
|
||
${CMAKE_CURRENT_LIST_DIR}/mlx.pc.in ${MLX_CMAKE_BUILD_CONFIG}
|
||
INSTALL_DESTINATION ${MLX_CMAKE_INSTALL_MODULE_DIR}
|
||
NO_CHECK_REQUIRED_COMPONENTS_MACRO
|
||
PATH_VARS CMAKE_INSTALL_LIBDIR CMAKE_INSTALL_INCLUDEDIR
|
||
MLX_CMAKE_INSTALL_MODULE_DIR)
|
||
|
||
install(FILES ${MLX_CMAKE_BUILD_CONFIG} ${MLX_CMAKE_BUILD_VERSION_CONFIG}
|
||
DESTINATION ${MLX_CMAKE_INSTALL_MODULE_DIR})
|
||
|
||
install(DIRECTORY ${CMAKE_MODULE_PATH}/
|
||
DESTINATION ${MLX_CMAKE_INSTALL_MODULE_DIR})
|