本文介绍如何在setup.py 中使用cmake.
- 适用场景: 采用c/c++ 为python添加扩展模版
项目组织结构
- top-level CMakeLists.txt 与 setup.py 放在同级目录下
CMakeLists.txt
setup.py
...
setup.py 文件内容
import os
import re
import sys
import platform
import subprocess
import pathlib
from setuptools import setup, Extension
from setuptools.command.build_ext import build_ext
from distutils.version import LooseVersion
class CMakeExtension(Extension):
"""
自定义了 Extension 类,忽略原来的 sources、libraries 等参数,交给 CMake 来处理这些事情
"""
def __init__(self, name, sourcedir=''):
Extension.__init__(self, name, sources=[])
self.sourcedir = os.path.abspath(sourcedir)
class CMakeBuild(build_ext):
"""
自定义了 build_ext 类,对 CMakeExtension 的实例,调用 CMake 和 Make 命令来编译它们
"""
def run(self):
for ext in self.extensions:
if isinstance(ext, CMakeExtension):
self.build_cmake(ext)
super().run()
def build_cmake(self, ext):
cwd = pathlib.Path().absolute()
build_temp = f"{pathlib.Path(self.build_temp)}/{ext.name}"
os.makedirs(build_temp, exist_ok=True)
extdir = pathlib.Path(self.get_ext_fullpath(ext.name))
extdir.mkdir(parents=True, exist_ok=True)
config = "Debug" if self.debug else "Release"
cmake_args = [
"-DCMAKE_LIBRARY_OUTPUT_DIRECTORY=" + str(extdir.parent.absolute()),
"-DCMAKE_BUILD_TYPE=" + config
]
build_args = [
"--config", config,
"--", "-j12"
]
os.chdir(build_temp)
self.spawn(["cmake", f"{str(cwd)}/{ext.name}"] + cmake_args)
if not self.dry_run:
self.spawn(["cmake", "--build", "."] + build_args)
os.chdir(str(cwd))
setup(
name='xxx',
version='0.0.1',
author='xxx',
author_email='xxx',
description='xxx',
long_description='',
ext_modules=[CMakeExtension('.')],
cmdclass=dict(build_ext=CMakeBuild),
zip_safe=False,
)