在 Cython 中编译 C 和 C++ 源代码

Rex*_*rus 4 c++ compiler-errors clang cython clang++

我正在尝试在 Cython 中同时编译 C 和 C++ 源代码。这是我当前的设置:

-setup.py

from distutils.core import setup
from Cython.Build import cythonize
from distutils.extension import Extension
import os

language = "c++"
extra_compile_flags = ["-std=c++17"]
os.environ["CC"] = "clang++"

ext_modules = [
    Extension(
        name="Dummy",
        sources=["mydummy.pyx", "source1.cpp","source2.c"],
        language=language,
        extra_compile_args=extra_compile_flags,
   )
]

ext_modules = cythonize(ext_modules)

setup(
    name="myapp",
    ext_modules=ext_modules,
)
Run Code Online (Sandbox Code Playgroud)

- 编译命令:

python3 setup.py build_ext --inplace --verbose
Run Code Online (Sandbox Code Playgroud)

在日志中我收到以下消息:

clang++ -DNDEBUG -g -fwrapv -O2 -Wall -g -fstack-protector-strong -Wformat -Werror=format-security -Wdate-time -D_FORTIFY_SOURCE=2 -fPIC -I. -I/usr/include/python3.6m -I/usr/include/python3.6m -c /path/source2.c -o build/temp.linux-x86_64-3.6/./path/source2.o -std=c++17 -O3
clang: warning: treating 'c' input as 'c++' when in C++ mode, this behavior is deprecated [-Wdeprecated]
Run Code Online (Sandbox Code Playgroud)

编译失败了,但是警告看起来很糟糕。我怎样才能摆脱它?天真的解决方案

os.environ["CC"] = "clang"
os.environ["CXX"] = "clang++"
Run Code Online (Sandbox Code Playgroud)

失败了error: invalid argument '-std=c++17' not allowed with 'C'

ead*_*ead 5

gcc(或 clang)只是一个前端,它根据已编译文件的文件扩展名调用适当的编译器(c 或 c++ 编译器)(.c 表示它应该使用 c 编译器进行编译,.cpp 表示它应该使用使用 c++-compiler 进行编译),请参见例如这个类似的SO-issue

因此无需将编译器设置为“clang++”,因为它会自动发生。

但是,cython 需要知道它必须生成 cpp 文件(以及接受 c++ 语法)而不是 c 文件。此外,链接器必须知道它必须链接到 cpp-libaries(如果使用 g++/clang++ 进行链接,则会自动完成)。因此,我们需要添加language = "c++"Extension定义,正如 DavidW 的答案中所建议的那样 - 这将解决最后两个问题。

剩下的问题是,我们希望-std=c++17对不同的源文件使用不同的编译器选项(仅适用于 cpp 文件)。这可以使用一个不太为人所知的命令来实现build_clib

#setup.py:

from distutils.core import setup
from Cython.Build import cythonize
from distutils.extension import Extension

ext_modules = [
    Extension(
        name="mydummy",
        sources=["mydummy.pyx", "source1.cpp"]
        language="c++",
        extra_compile_args=["-std=c++17"],
   )
]

ext_modules = cythonize(ext_modules)

myclib = ('myclib', {'sources': ["source2.c"]})

setup(
    name="mydummy",
    libraries=[myclib],
    ext_modules=ext_modules,
)
Run Code Online (Sandbox Code Playgroud)

现在用

python setup.py build --verbose
Run Code Online (Sandbox Code Playgroud)

或者

python setup.py build_clib --verbose build_ext -i --verbose
Run Code Online (Sandbox Code Playgroud)

首先构建一个由 C 代码组成的简单静态库,并将其链接到由 C++ 代码组成的结果扩展。

上面的代码在构建静态库时使用默认编译标志,这对于您的情况来说应该足够了。

distutils 不提供指定附加标志的可能性,因此如果有必要,我们必须切换到setuptools提供此功能的选项,或者修补该build_clib命令。

对于第二种选择,我们必须将以下内容添加到上述setup.py文件中:

python setup.py build --verbose
Run Code Online (Sandbox Code Playgroud)

现在我们可以向库定义添加额外的编译标志(如果setuptools使用则可以跳过上一步):

...
myclib = ('myclib', {'sources': ["source2.c"], 'cflags' : ["-O3"]})
...
Run Code Online (Sandbox Code Playgroud)