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

标签 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,
)

-compile命令:
python3 setup.py build_ext --inplace --verbose

在日志中,我收到以下消息:
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]

汇编进行得很糟糕,但是警告看起来很讨厌。我该如何摆脱呢?
天真的解决方案
os.environ["CC"] = "clang"
os.environ["CXX"] = "clang++"
error: invalid argument '-std=c++17' not allowed with 'C'失败

最佳答案

gcc(或clang)只是一个前端,它会在给定文件扩展名的情况下调用适当的编译器(c或c++编译器)(。c表示应使用c-compiler编译,.cpp表示应可以使用c++-compiler进行编译),例如参见类似的SO-issue

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

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

剩下的问题是,我们想对不同的源文件使用不同的编译器选项(仅对cpp文件使用-std=c++17)。这可以使用鲜为人知的命令 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,
)

现在用
python setup.py build --verbose

要么
python setup.py build_clib --verbose build_ext -i --verbose

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

上面的代码在构建静态库时使用默认的编译标志,在您的情况下应该足够了。
distutils doesn't offer可以指定附加标志,因此,如果有必要,我们必须切换到提供此功能的 setuptools 或修补build_clib命令。

对于第二种选择,我们必须在上面的setup.py -file中添加以下内容:

#setup.py
...
# adding cflags to build_clib
from distutils import log
from distutils.command.build_clib import build_clib

# use original implementation but with tweaked build_libraries!
class build_clib_with_cflags(build_clib):
    def build_libraries(self, libraries):
            for (lib_name, build_info) in libraries:
                sources = build_info.get('sources')
                if sources is None or not isinstance(sources, (list, tuple)):
                    raise DistutilsSetupError(
                           "in 'libraries' option (library '%s'), "
                           "'sources' must be present and must be "
                           "a list of source filenames" % lib_name)
                sources = list(sources)

                log.info("building '%s' library", lib_name)


                macros = build_info.get('macros')
                include_dirs = build_info.get('include_dirs')
                cflags = build_info.get('cflags')                    # HERE we add cflags
                objects = self.compiler.compile(sources, 
                                                output_dir=self.build_temp,
                                                macros=macros,
                                                include_dirs=include_dirs,
                                                extra_postargs=cflags,        # HERE we use cflags
                                                debug=self.debug)


                self.compiler.create_static_lib(objects, lib_name,
                                                output_dir=self.build_clib,
                                                debug=self.debug)

...
setup(
    ...
    cmdclass={'build_clib': build_clib_with_cflags}, # use our class instead of built-in!
)

现在我们可以在库定义中添加其他编译标志(如果使用setuptools,则可以跳过上一步):
...
myclib = ('myclib', {'sources': ["source2.c"], 'cflags' : ["-O3"]})
...

关于c++ - 在Cython中编译C和C++源代码,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59345649/

相关文章:

c++ - SDL2 面向对象包装器; 1000+ Virtual Function 调用一个框架;有什么办法可以优化这个吗?

C++:误解内存地址和指针的复制值

C++使用类从多维 vector 输出一个char数组

c++ - 类型未声明 C++

java - 为什么这个通用接口(interface)实现无法编译?

c++ - 专门化非模板类的模板化构造函数

C++如何初始化堆中的对象数组

java - “找不到符号”或“无法解析符号”错误是什么意思?

python - CPython可以用Clang编译吗?

gcc - 从 GCC 切换到 Clang 的常见问题是什么?