gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-12-02 10:07:43 33 4
gpt4 key购买 nike

我试图在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/

33 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com