我正在尝试使用Swig为大型C ++项目生成接口文件。项目的头文件根据编译器类型进行一些检查,例如
#if defined(__clang__)
...
#if defined(__llvm__)
所以我必须将编译器标志传递给Swig。具体来说,在我看来,我应该像这样打电话:
swig -D__clang__ ...
然而,这似乎并没有减轻我得到的错误消息,因为我仍然得到相同的错误。
我不清楚我是否错误地设置了编译器标志,或者我是否未设置所有编译器标志。所以问题是双重的:
我是通过调用
正确设置编译器标志的swig -D__clang__ ...... xxx.i
或者我应该以某种方式在xxx.i接口文件中定义这些参数?
答案 0 :(得分:0)
理查德是对的。大多数编译器似乎都支持-dM
命令(至少clang
和gcc
似乎)。在我的例子中,我想看看Android构建系统设置了哪些预处理器宏。这有点复杂,因为Android NDK有许多编译器,所有编译器都略有不同。因此,我创建了以下python 3脚本来保存所有宏包含到文件:
import os, sys
# Define the toolchain root
root = "C:\\Android\\android-sdk\\ndk-bundle\\toolchains\\"
# Define the directory where you want all of the macro definitions to go
output = "output\\"
os.makedirs(output, exist_ok=True)
# Get a list of all of the files
files = [ ( os.path.join(dp, f), f ) for dp, dn, fn in os.walk(os.path.expanduser(root)) for f in fn ]
# We only want executables
files = [ file for file in files if file[1].endswith("exe") ]
# Now, let's limit that to executables that contain one of the following keys
keys = [ "gcc", "g++", "clang", "c++", "cpp" ]
files = [ file for file in files if any(key in file[1] for key in keys) ]
# and that exclude the following keys
keys = [ "-ar", "-nm", "-ranlib", "filt", "-format", "-tidy" ]
files = [ file for file in files if all(key not in file[1] for key in keys) ]
# Now, for each of these files, we want to see what preprocessor macros are being set.
call = " -dM -E - < NUL > "
for file in files:
command = file[0] + call + output + file[1].replace(".exe", ".txt" ).replace(root,"")
print(command)
os.system( command )
如果您将此文件称为“introspect.py”,则使用只是
python introspect.py
这个脚本的作用是查找应该包含编译器的*.exe
目录(在我的系统root
上)的所有C:\\Android\\android-sdk\\ndk-bundle\\toolchains\\
文件(对linux系统进行更改)某处。具体来说,此脚本将查找具有其中一个术语
[ "gcc", "g++", "clang", "c++", "cpp" ]
名称中的,不包含带
的文件[ "-ar", "-nm", "-ranlib", "filt", "-format", "-tidy" ]
名字中的。最后,每个找到的可执行文件都以
格式运行compiler.exe -dM -E - < NUL > output\compiler.txt
其中compiler
类似于clang
或gcc
或g++
。请注意,此脚本强制创建output
目录。在我的系统上运行后,output
目录包含文件(警告:通过NDK确实有很多编译器可用):
aarch64-linux-android-c++.txt
aarch64-linux-android-cpp.txt
aarch64-linux-android-g++.txt
aarch64-linux-android-gcc-4.9.txt
aarch64-linux-android-gcc-4.9.x.txt
aarch64-linux-android-gcc.txt
arm-linux-androideabi-c++.txt
arm-linux-androideabi-cpp.txt
arm-linux-androideabi-g++.txt
arm-linux-androideabi-gcc-4.9.txt
arm-linux-androideabi-gcc-4.9.x.txt
arm-linux-androideabi-gcc.txt
clang++.txt
clang.txt
clang_32.txt
i686-linux-android-c++.txt
i686-linux-android-cpp.txt
i686-linux-android-g++.txt
i686-linux-android-gcc-4.9.txt
i686-linux-android-gcc-4.9.x.txt
i686-linux-android-gcc.txt
mips64el-linux-android-c++.txt
mips64el-linux-android-cpp.txt
mips64el-linux-android-g++.txt
mips64el-linux-android-gcc-4.9.txt
mips64el-linux-android-gcc-4.9.x.txt
mips64el-linux-android-gcc.txt
mipsel-linux-android-c++.txt
mipsel-linux-android-cpp.txt
mipsel-linux-android-g++.txt
mipsel-linux-android-gcc-4.9.txt
mipsel-linux-android-gcc-4.9.x.txt
mipsel-linux-android-gcc.txt
x86_64-linux-android-c++.txt
x86_64-linux-android-cpp.txt
x86_64-linux-android-g++.txt
x86_64-linux-android-gcc-4.9.txt
x86_64-linux-android-gcc-4.9.x.txt
x86_64-linux-android-gcc.txt
这些文件的前几行看起来像
#define __DBL_MIN_EXP__ (-1021)
#define __UINT_LEAST16_MAX__ 65535
#define __ATOMIC_ACQUIRE 2
#define __FLT_MIN__ 1.17549435082228750796873653722224568e-38F
#define __GCC_IEC_559_COMPLEX 2
#define __UINT_LEAST8_TYPE__ unsigned char
#define __INTMAX_C(c) c ## L
#define __CHAR_BIT__ 8
#define __UINT8_MAX__ 255
#define __ANDROID__ 1
#define __WINT_MAX__ 4294967295U
#define __ORDER_LITTLE_ENDIAN__ 1234
#define __SIZE_MAX__ 18446744073709551615UL
#define __WCHAR_MAX__ 4294967295U
#define __GCC_HAVE_SYNC_COMPARE_AND_SWAP_1 1
#define __GCC_HAVE_SYNC_COMPARE_AND_SWAP_2 1
#define __GCC_HAVE_SYNC_COMPARE_AND_SWAP_4 1
事实证明,其中许多文件都是相同的,但clang
和gcc
文件之间存在显着差异。
答案 1 :(得分:0)
你应该有一个.i文件来定义你想要生成的接口。
您只需使用#define __clang__
。
另请注意,swig不需要所有编译器标志:它不会编译您的项目,但它只会生成额外的cxx文件。在.i文件中只声明标志(如果需要),允许swig选择正确的类和方法。