问题
Microsoft 是否真的支持 X32(相对于 X86 和 X64)?
TL;DR 答案
答案是“不,Microsoft 不支持它”。预处理器宏不会导致 X32 的任何标识,命令行选项和 IDE 选项不存在,并且标识此类编译器的字符串不存在。
长答案——第一部分
“X32 没有标头字符串”
无视以下事实:
- 不存在此类功能的官方文档,
- Visual Studio 中没有选项或
cl.exe /?
启用/禁用它存在,并且
strings -el clui.dll
没有显示出这种选择的迹象,
strings -el "%VCINSTALLDIR%\bin\1033\clui.dll" | find "Microsoft (R)"
也没有显示匹配标头字符串的迹象:
4Microsoft (R) C/C++ Optimizing Compiler Version %s
-for Microsoft (R) .NET Framework version %s
(Microsoft (R) C/C++ Optimizing Compiler
FMicrosoft (R) C/C++ Optimizing Compiler Version %s for MIPS R-Series
)Microsoft (R) MIPS Assembler Version %s
CMicrosoft (R) C/C++ Optimizing Compiler Version %s for Renesas SH
<Microsoft (R) C/C++ Optimizing Compiler Version %s for ARM
:Microsoft (R) C/C++ Standard Compiler Version %s for x86
<Microsoft (R) C/C++ Optimizing Compiler Version %s for x86
GMicrosoft (R) 32-bit C/C++ Optimizing Compiler Version %s for PowerPC
@Microsoft (R) C/C++ Optimizing Compiler Version %s for Itanium
<Microsoft (R) C/C++ Optimizing Compiler Version %s for x64
>Microsoft (R) C/C++ Optimizing Compiler Version %s for ARM64
Microsoft (R) MIPS Assembler
在bin\x86_amd64\1033\clui.dll
andbin\x86_arm\1033\clui.dll
文件中可以看到相同的输出,因此并不是一个文件根本没有包含它。
长答案——第二部分
“Windows 不做数据模型”
让我们假设它做到了。你将如何检测它?在 GLIBC 的情况下,__ILP32__
为 x32 和 x86 定义,而__LP64__
为 amd64 定义,表示使用的数据模型。此外,__x86_64__
将针对 AMD64 架构进行定义。如果__x86_64__
已定义且__ILP32__
已定义,则您使用的是 X32 ABI,否则您使用的是 AMD64 ABI。对于 C 来说,这才是最重要的。如果您使用的是汇编代码,这就是 X32 ABI 和 x86 ABI 之间的区别很重要的地方,因此检查__x86_64__
以确定目标体系结构是 64 位并检查__ILP32__
以确定是 32 位还是 64 位 ABI利用。例如:
#ifdef __x86_64__
# ifdef __ILP32__
// Use X32 version of myfunc().
extern long myfunc_x32 (const char *);
long (*myfunc)(const char *) = myfunc_x32;
# else /* !__ILP32__ */
// Use AMD64 version of myfunc().
extern long myfunc_amd64 (const char *);
long (*myfunc)(const char *) = myfunc_amd64;
# endif /* __ILP32__ */
/* !__x86_64__ */
#elif defined __i386__
// Use x86 version of myfunc().
extern long myfunc_x86 (const char *);
long (*myfunc)(const char *) = myfunc_x86;
/* !__i386__ */
#else
// Use generic version of myfunc() since no optimized versions are available.
long myfunc(const char *);
#endif /* __x86_64__ */
但是,在 Windows 上没有指示数据模型的宏。您针对以下架构之一:
- 32 位 x86 (
_M_IX86
)
- 64 位 AMD64 (
_M_AMD64
/ _M_X64
)
- (32 位?)ARM (
_M_ARM
)
理论上可以用_M_AMD64
and_M_X64
独立判断X32是否存在,但如果_M_AMD64
定义了,_M_X64
也定义了。
长答案——第三部分
“坏消息”
最后,在搜索到任何东西(甚至可能是早已被遗忘的材料)之后,没有证据表明 Windows 已经支持或将支持像 Linux 这样的 X32 ABI 的编码。预处理器宏无助于识别 X32,命令行选项和 IDE 选项不存在,识别此类编译器的字符串也不存在。
长答案——新的希望破灭了
“这些不是您要查找的宏”
可以假设使用当前存在的宏进行检查,但在这种情况下它没有帮助,因为 Windows 的 X32 不存在。它与 GLIBC 检查没有什么不同,尽管如果__ILP32__
未定义则启用 X32,而不是启用 X32 _M_X64
。
#ifdef _M_AMD64
# ifndef _M_X64
# define ABI_STR "X32"
# else
# define ABI_STR "AMD64"
# endif
#elif defined _M_IX86
# define ABI_STR "X86"
#else
# error unsupported CPU/architecture
#endif
当然,如果_M_AMD64
定义了,那么_M_X64
也定义了,进一步强化了 Windows 没有 X32 的证据。