如何交叉编译C++并构建成python库

如何交叉编译C++并构建成python库

1. 交叉编译python

  • python版本要和目标平台一致, 本文以python3.10为例
python3 --version
# Python 3.10.12

下载python安装包

wget https://www.python.org/ftp/python/3.10.0/Python-3.10.0.tgz
ls -lh Python-3.10.0.tgz # 查看完整性
tar -xzf Python-3.10.0.tgz
  • 编译
mkdir -p /workspace/python3.10-aarch64
cd /workspace/python3.10-aarch64

echo "ac_cv_file__dev_ptmx=yes" > /workspace/config.site
echo "ac_cv_file__dev_ptc=yes" >> /workspace/config.site
export CONFIG_SITE=/workspace/config.site

/workspace/Python-3.10.0/configure --host=aarch64-linux-gnu --build=x86_64-linux-gnu --prefix=/workspace/python3.10-aarch64/install --enable-shared --enable-ipv6 --disable-static --with-ensurepip=yes

make
make install

2. 交叉编译pybind11

编写交叉编译工具链

aarch64-toolchain.cmake

# aarch64-toolchain.cmake
set(CMAKE_SYSTEM_NAME Linux)
set(CMAKE_SYSTEM_PROCESSOR aarch64)

# 设置交叉编译器
set(CMAKE_C_COMPILER aarch64-linux-gnu-gcc)
set(CMAKE_CXX_COMPILER aarch64-linux-gnu-g++)

# 设置路径(根据实际安装路径调整)
# set(CMAKE_FIND_ROOT_PATH /usr/aarch64-linux-gnu)
set(CMAKE_FIND_ROOT_PATH /workspace/python3.10-aarch64/install)

# 只在目标平台上查找库和头文件
set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER)
set(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY)
set(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY)
set(CMAKE_FIND_ROOT_PATH_MODE_PACKAGE ONLY)

git clone https://github.com/pybind/pybind11.git
cd pybind11/
mkdir build
cd build
cmake -DCMAKE_TOOLCHAIN_FILE=aarch64-toolchain.cmake -DCMAKE_INSTALL_PREFIX=/workspace/python3.10-aarch64/install ..

make
make install

3. 交叉编译demo

demo 组成

.
├── CMakeLists.txt
├── aarch64-toolchain.cmake
├── bindings
│   └── python_bindings.cpp
├── build
├── include
│   └── my_math_lib
│       └── math_operations.h
├── src
│   └── math_operations.cpp
└── test.py

math_operations.h

// include/my_math_lib/math_operations.h

#ifndef MY_MATH_LIB_MATH_OPERATIONS_H
#define MY_MATH_LIB_MATH_OPERATIONS_H

namespace my_math_lib {

class MathOperations {
public:
    int add(int a, int b);
    int subtract(int a, int b);
    int multiply(int a, int b);
    double divide(int a, int b);
};

} // namespace my_math_lib

#endif // MY_MATH_LIB_MATH_OPERATIONS_H

math_operations.cpp

// src/math_operations.cpp

#include "my_math_lib/math_operations.h"
#include <stdexcept>

namespace my_math_lib {

int MathOperations::add(int a, int b) {
    return a + b;
}

int MathOperations::subtract(int a, int b) {
    return a - b;
}

int MathOperations::multiply(int a, int b) {
    return a * b;
}

double MathOperations::divide(int a, int b) {
    if (b == 0) {
        throw std::invalid_argument("Division by zero");
    }
    return static_cast<double>(a) / b;
}

} // namespace my_math_lib

python_bindings.cpp

// bindings/python_bindings.cpp

#include <pybind11/pybind11.h>
#include "my_math_lib/math_operations.h"

namespace py = pybind11;

PYBIND11_MODULE(my_math_lib, m) {
    py::class_<my_math_lib::MathOperations>(m, "MathOperations")
        .def(py::init<>())
        .def("add", &my_math_lib::MathOperations::add)
        .def("subtract", &my_math_lib::MathOperations::subtract)
        .def("multiply", &my_math_lib::MathOperations::multiply)
        .def("divide", &my_math_lib::MathOperations::divide);
}

CMakeLists.txt

cmake_minimum_required(VERSION 3.12)
project(MyMathLib)

# 设置C++标准
set(CMAKE_CXX_STANDARD 14)

# 设置Python路径
set(Python3_ROOT_DIR /workspace/python3.10-aarch64/install)
set(Python3_INCLUDE_DIR /workspace/python3.10-aarch64/install/include/python3.10)
set(Python3_LIBRARY /workspace/python3.10-aarch64/install/lib/libpython3.10.so)

# 查找Python3
find_package(Python3 3.10 REQUIRED COMPONENTS Interpreter Development)

# 查找pybind11
find_package(pybind11 REQUIRED)

# 包含头文件
include_directories(${Python3_INCLUDE_DIRS} /workspace/example/libtest/include)

# 添加pybind11模块
pybind11_add_module(my_math_lib MODULE bindings/python_bindings.cpp src/math_operations.cpp)

# 链接Python库
target_link_libraries(my_math_lib PRIVATE ${Python3_LIBRARIES})

编译

cmake -DCMAKE_TOOLCHAIN_FILE=../aarch64-toolchain.cmake -DPYTHON_EXECUTABLE=$(which python3.10) ..

make

4. python 调用

test.py

import my_math_lib
def main():
    math_ops = my_math_lib.MathOperations()
    print(f"Add: 3 + 4 = {math_ops.add(3, 4)}")
    print(f"Subtract: 10 - 6 = {math_ops.subtract(10, 6)}")
    print(f"Multiply: 2 * 5 = {math_ops.multiply(2, 5)}")
    try:
        print(f"Divide: 8 / 2 = {math_ops.divide(8, 2)}")
        print(f"Divide: 8 / 0 = {math_ops.divide(8, 0)}")
    except ValueError as e:
        print(f"Error: {e}")


if __name__ == "__main__":
    main()

cp my_math_lib.cpython-310-aarch64-linux-gnu.so path/to/lib/
export PYTHONPATH=$PYTHONPATH:path/to/lib
python test.py
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值