python - 通过 cmake 从 C++ 扩展构建 Python 子模块

标签 python c++ cmake module setuptools

我正在尝试通过 cmake 将 c++ 扩展作为子模块合并到现有的 python 库中。构建 C++ 扩展可以正常工作并将其作为 python 模块导入,但不能作为头库的子模块。

我有以下目录结构:

frontend/
   foo.py
   bar.py
   backend/
      backend.cpp

扩展通过 pybind 绑定(bind)到一个 python 模块:
PYBIND11_MODULE(backend, m)
{
    m.doc() = "backend c++ implementation"; // optional module docstring
    m.def("method", &method, "The method I want to call from python.");
}

在 CMakeLists.txt 中,相关行是:
pybind11_add_module(backend "frontend/backend/backend.cpp")

我已按照 here 的说明进行操作和 here编写 setup.py 脚本。我猜最重要的几行如下所示:
from setuptools import setup, Extension, find_packages
from setuptools.command.build_ext import build_ext
from setuptools.command.test import test as TestCommand

class CMakeExtension(Extension):
    def __init__(self, name, sourcedir=".", sources=[]):
        Extension.__init__(self, name, sources=[])


class CMakeBuild(build_ext):
    def run(self):
        build_directory = os.path.abspath(self.build_temp)
        if not os.path.exists(self.build_temp):
            os.makedirs(self.build_temp)

        cmake_list_dir = os.path.abspath(os.path.dirname(__file__))
        print("-" * 10, "Running CMake prepare", "-" * 40)
        subprocess.check_call(
            ["cmake", cmake_list_dir], cwd=self.build_temp,
        )

        print("-" * 10, "Building extensions", "-" * 40)
        cmake_cmd = ["cmake", "--build", "."] + self.build_args
        subprocess.check_call(cmake_cmd, cwd=self.build_temp)

        # Move from build temp to final position
        for ext in self.extensions:
            self.move_output(ext)

    def move_output(self, ext):
        build_temp = Path(self.build_temp).resolve()
        dest_path = Path(self.get_ext_fullpath(ext.name)).resolve()
        source_path = build_temp / self.get_ext_filename(ext.name)
        dest_directory = dest_path.parents[0]
        dest_directory.mkdir(parents=True, exist_ok=True)
        self.copy_file(source_path, dest_path)


extensions = [CMakeExtension("backend")]

setup(
    name="frontend",
    packages=["frontend"],
    ext_modules=extensions,
    cmdclass=dict(build_ext=CMakeBuild),
)

但这并不能使 backend frontend 的子模块,而是一个独立的模块。所以这有效:
from backend import method

但是为了避免与其他库的命名问题,我想要的是:
from frontend.backend import method

将 pybinding 或扩展调用中的命名更改为 extensions = [CMakeExtension("frontend.backend")]不幸的是没有解决我的问题,安装程序找不到 backend.<platform>.so然后共享库,因为它查找 frontend/backend.<platform>.so ,不存在。我该如何解决这个问题?

最佳答案

我想我已经通过以下几行解决了这个问题:

setup.py 文件中的更改:

ext_modules = [
    Extension(
        "frontend.backend", sources=["frontend/backend/backend.cpp"]
    )
]

CMakeLists.txt 文件中的更改:
pybind11_add_module(backend "frontend/backend/backend.cpp")
set_target_properties( backend
    PROPERTIES
    ARCHIVE_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/frontend"
    LIBRARY_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/frontend"
    RUNTIME_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/frontend"
)

共享库对象 backend.platform.so 必须位于 frontend 目录中。 pybind 模块名称和源文件 .cpp 都不应包含任何“。”在名称中,因为 get_ext_fullpath()来自 build_ext 的方法将被点分割。只有前端目录包含 初始化 .py 文件。

关于python - 通过 cmake 从 C++ 扩展构建 Python 子模块,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61956365/

相关文章:

c++ - Qt pro 文件对应的 CMakeList

python - 连接几个文件删除标题行

python - 设置为 dict Python

python - 我请求的页面不代表与BeautifulSoup4相同的结构

带有 Twisted 或 Node.js 的 Python

C++ 预处理器宏和返回语句

c++ - 如何判断当前窗口是否为事件窗口?

c++ - 在聚合初始化期间从后面的成员表达式中引用早期成员是否定义了行为?

opengl - 尝试构建支持 OpenGL 的 OpenCV 2.4.1

command-line - 如何使用ctest将参数传递给memcheck?