本文介绍了如何检测X32 ABI或环境在preprocessor?的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

X32 是一个ABI为 AMD64 / 使用32位指针的x86_64 的CPU。我们的想法是较大的寄存器集的x86_64与来自32位指针产生的较小的存储器和高速缓存足迹结合。它提供了高达约40%的加速。见X86,X32,和x64架构之间对堆栈溢出,以及Debian的详情维基页面并设置它作为一个chroot环境。

我们已经从环境下的Debian维护者的bug报告。该报告是 adcq 是一个非法指令。内联汇编基于preprocessor宏激活,因此我们不会检测X32正确(或更正确,一点都没有到现在为止)。

最明显的选择(对我)的preprocessor宏类似 __ __ X32 ,但是这不提供。基于和 Debian的建议,它看起来像 __ __ ILP32 可以使用。但我想,因为 _ILP32 __ code_model_small __ 看起来有趣,也更规范的答案。 (我知道那里的编译器的支持,但操作系统不与SSE2,问题)。

什么是可以用于可靠地检测X32 ABI和环境中的preprocessor宏使用锵和GCC时

要清楚,我不是试图修复code。在这一点上。我只是想知道,可以在一个完整的补救使用的宏。


 #CPP -dM<的/ dev / null的|分类
#定义__amd64 1
#定义__amd64__ 1
#定义__ATOMIC_ACQ_REL 4
#定义__ATOMIC_ACQUIRE 2
#定义__ATOMIC_CONSUME 1
#定义__ATOMIC_HLE_ACQUIRE 65536
#定义__ATOMIC_HLE_RELEASE 131072
#定义__ATOMIC_RELAXED 0
#定义__ATOMIC_RELEASE 3
#定义__ATOMIC_SEQ_CST 5
#定义__BIGGEST_ALIGNMENT__ 16
#定义__BYTE_ORDER__ __ORDER_LITTLE_ENDIAN__
#定义__CHAR16_TYPE__短的无符号整型
#定义__CHAR32_TYPE__无符号整型
#定义__CHAR_BIT__ 8
#定义__ $ C $ 1 c_model_small__
#定义__DBL_DECIMAL_DIG__ 17
#定义__DBL_DENORM_MIN__((双)4.94065645841246544177e-324L)
#定义__DBL_DIG__ 15
#定义__DBL_EPSILON__((双)2.22044604925031308085e-16L)
#定义__DBL_HAS_DENORM__ 1
#定义__DBL_HAS_INFINITY__ 1
#定义__DBL_HAS_QUIET_NAN__ 1
#定义__DBL_MANT_DIG__ 53
#定义__DBL_MAX_10_EXP__ 308
#定义__DBL_MAX__((双)1.79769313486231570815e + 308L)
#定义__DBL_MAX_EXP__ 1024
#定义__DBL_MIN_10_EXP__(-307)
#定义__DBL_MIN__((双)2.22507385850720138309e-308L)
#定义__DBL_MIN_EXP__(-1021)
#定义__DEC128_EPSILON__ 1E-33DL
#定义__DEC128_MANT_DIG__ 34
#定义__DEC128_MAX__ 9.999999999999999999999999999999999E6144DL
#定义__DEC128_MAX_EXP__ 6145
#定义__DEC128_MIN__ 1E-6143DL
#定义__DEC128_MIN_EXP__(-6142)
#定义__DEC128_SUBNORMAL_MIN__ 0.000000000000000000000000000000001E-6143DL
#定义__DEC32_EPSILON__ 1E-6DF
#定义__DEC32_MANT_DIG__ 7
#定义__DEC32_MAX__ 9.999999E96DF
#定义__DEC32_MAX_EXP__ 97
#定义__DEC32_MIN__ 1E-95DF
#定义__DEC32_MIN_EXP__(-94)
#定义__DEC32_SUBNORMAL_MIN__ 0.000001E-95DF
#定义__DEC64_EPSILON__ 1E-15DD
#定义__DEC64_MANT_DIG__ 16
#定义__DEC64_MAX__ 9.999999999999999E384DD
#定义__DEC64_MAX_EXP__ 385
#定义__DEC64_MIN__ 1E-383DD
#定义__DEC64_MIN_EXP__(-382)
#定义__DEC64_SUBNORMAL_MIN__ 0.000000000000001E-383DD
#定义__DEC_EVAL_METHOD__ 2
#定义__DECIMAL_BID_FORMAT__ 1
#定义__DECIMAL_DIG__ 21
#定义__ELF__ 1
#定义__FINITE_MATH_ONLY__ 0
#定义__FLOAT_WORD_ORDER__ __ORDER_LITTLE_ENDIAN__
#定义__FLT_DECIMAL_DIG__ 9
#定义__FLT_DENORM_MIN__ 1.40129846432481707092e-45F
#定义__FLT_DIG__ 6
#定义__FLT_EPSILON__ 1.19209289550781250000e-7F
#定义__FLT_EVAL_METHOD__ 0
#定义__FLT_HAS​​_DENORM__ 1
#定义__FLT_HAS​​_INFINITY__ 1
#定义__FLT_HAS​​_QUIET_NAN__ 1
#定义__FLT_MANT_DIG__ 24
#定义__FLT_MAX_10_EXP__ 38
#定义__FLT_MAX__ 3.40282346638528859812e + 38F
#定义__FLT_MAX_EXP__ 128
#定义__FLT_MIN_10_EXP__(-37)
#定义__FLT_MIN__ 1.17549435082228750797e-38F
#定义__FLT_MIN_EXP__(-125)
#定义__FLT_RADIX__ 2
#定义__FXSR__ 1
#定义__GCC_ATOMIC_BOOL_LOCK_FREE 2
#定义__GCC_ATOMIC_CHAR16_T_LOCK_FREE 2
#定义__GCC_ATOMIC_CHAR32_T_LOCK_FREE 2
#定义__GCC_ATOMIC_CHAR_LOCK_FREE 2
#定义__GCC_ATOMIC_INT_LOCK_FREE 2
#定义__GCC_ATOMIC_LLONG_LOCK_FREE 2
#定义__GCC_ATOMIC_LONG_LOCK_FREE 2
#定义__GCC_ATOMIC_POINTER_LOCK_FREE 2
#定义__GCC_ATOMIC_SHORT_LOCK_FREE 2
#定义__GCC_ATOMIC_TEST_AND_SET_TRUEVAL 1
#定义__GCC_ATOMIC_WCHAR_T_LOCK_FREE 2
#定义__GCC_HAVE_DWARF2_CFI_ASM 1
#定义__GCC_HAVE_SYNC_COMPARE_AND_SWAP_1 1
#定义__GCC_HAVE_SYNC_COMPARE_AND_SWAP_2 1
#定义__GCC_HAVE_SYNC_COMPARE_AND_SWAP_4 1
#定义__GCC_HAVE_SYNC_COMPARE_AND_SWAP_8 1
#定义__GCC_IEC_559 2
#定义__GCC_IEC_559_COMPLEX 2
#定义__GNUC__ 5
#定义__GNUC_MINOR__ 2
#定义__GNUC_PATCHLEVEL__ 1
#定义__GNUC_STDC_INLINE__ 1
#定义__gnu_linux__ 1
#定义__GXX_ABI_VERSION 1009
#定义__has_include_next(STR)__has_include_next __(STR)
#定义__has_include(STR)__has_include __(STR)
#定义__ILP32__ 1
#定义_ILP32 1
#定义__INT16_C(三)C
#定义__INT16_MAX__ 0x7FFF的
#定义__INT16_TYPE__短整型
#定义__INT32_C(三)C
#定义__INT32_MAX__为0x7FFFFFFF
#定义__INT32_TYPE__ INT
#定义__INT64_C(三)C ## LL
#定义__INT64_MAX__ 0x7fffffffffffffffLL
#定义__INT64_TYPE__得到long long int
#定义__INT8_C(三)C
#定义__INT8_MAX__到0x7f
#定义__INT8_TYPE__符号字符
#定义__INT_FAST16_MAX__为0x7FFFFFFF
#定义__INT_FAST16_TYPE__ INT
#定义__INT_FAST32_MAX__为0x7FFFFFFF
#定义__INT_FAST32_TYPE__ INT
#定义__INT_FAST64_MAX__ 0x7fffffffffffffffLL
#定义__INT_FAST64_TYPE__得到long long int
#定义__INT_FAST8_MAX__到0x7f
#定义__INT_FAST8_TYPE__符号字符
#定义__INT_LEAST16_MAX__ 0x7FFF的
#定义__INT_LEAST16_TYPE__短整型
#定义__INT_LEAST32_MAX__为0x7FFFFFFF
#定义__INT_LEAST32_TYPE__ INT
#定义__INT_LEAST64_MAX__ 0x7fffffffffffffffLL
#定义__INT_LEAST64_TYPE__得到long long int
#定义__INT_LEAST8_MAX__到0x7f
#定义__INT_LEAST8_TYPE__符号字符
#定义__INT_MAX__为0x7FFFFFFF
#定义__INTMAX_C(三)C ## LL
#定义__INTMAX_MAX__ 0x7fffffffffffffffLL
#定义__INTMAX_TYPE__得到long long int
#定义__INTPTR_MAX__为0x7FFFFFFF
#定义__INTPTR_TYPE__ INT
#定义__k8 1
#定义__k8__ 1
#定义__LDBL_DENORM_MIN__ 3.64519953188247460253e-4951L
#定义__LDBL_DIG__ 18
#定义__LDBL_EPSILON__ 1.08420217248550443401e-19L
#定义__LDBL_HAS_DENORM__ 1
#定义__LDBL_HAS_INFINITY__ 1
#定义__LDBL_HAS_QUIET_NAN__ 1
#定义__LDBL_MANT_DIG__ 64
#定义__LDBL_MAX_10_EXP__ 4932
#定义__LDBL_MAX__ 1.18973149535723176502e + 4932L
#定义__LDBL_MAX_EXP__ 16384
#定义__LDBL_MIN_10_EXP__(-4931)
#定义__LDBL_MIN__ 3.36210314311209350626e-4932L
#定义__LDBL_MIN_EXP__(-16381)
#定义__linux 1
#定义__linux__ 1
linux下的#define 1
#定义__LONG_LONG_MAX__ 0x7fffffffffffffffLL
#定义__LONG_MAX__ 0x7fffffffL
#定义__MMX__ 1
#定义__NO_INLINE__ 1
#定义__ORDER_BIG_ENDIAN__ 4321
#定义__ORDER_LITTLE_ENDIAN__ 1234
#定义__ORDER_PDP_ENDIAN__ 3412
#定义__PRAGMA_REDEFINE_EXTNAME 1
#定义__PTRDIFF_MAX__为0x7FFFFFFF
#定义__PTRDIFF_TYPE__ INT
#定义__REGISTER_ preFIX__
#定义__SCHAR_MAX__到0x7f
#定义__SHRT_MAX__ 0x7FFF的
#定义__SIG_ATOMIC_MAX__为0x7FFFFFFF
#定义__SIG_ATOMIC_MIN__(-__ SIG_ATOMIC_MAX__ - 1)
#定义__SIG_ATOMIC_TYPE__ INT
#定义__SIZE_MAX__ 0xffffffffU
#定义__SIZEOF_DOUBLE__ 8
#定义__SIZEOF_FLOAT128__ 16
#定义__SIZEOF_FLOAT__ 4
#定义__SIZEOF_FLOAT80__ 16
#定义__SIZEOF_INT128__ 16
#定义__SIZEOF_INT__ 4
#定义__SIZEOF_LONG__ 4
#定义__SIZEOF_LONG_DO​​UBLE__ 16
#定义__SIZEOF_LONG_LONG__ 8
#定义__SIZEOF_POINTER__ 4
#定义__SIZEOF_PTRDIFF_T__ 4
#定义__SIZEOF_SHORT__ 2
#定义__SIZEOF_SIZE_T__ 4
#定义__SIZEOF_WCHAR_T__ 4
#定义__SIZEOF_WINT_T__ 4
#定义__SIZE_TYPE__无符号整型
#定义__SSE__ 1
#定义__SSE2__ 1
#定义__SSE2_MATH__ 1
#定义__SSE_MATH__ 1
__STDC__的#define 1
#定义__STDC_HOSTED__ 1
#定义__STDC_IEC_559__ 1
#定义__STDC_IEC_559_COMPLEX__ 1
#定义__STDC_ISO_10646__ 201103L
#定义__STDC_NO_THREADS__ 1
#定义_STDC_ preDEF_H 1
#定义__STDC_UTF_16__ 1
#定义__STDC_UTF_32__ 1
#定义__STDC_VERSION__ 201112L
#定义__UINT16_C(三)C
#定义__UINT16_MAX__ 0xFFFF的
#定义__UINT16_TYPE__短的无符号整型
#定义__UINT32_C(三)C ##ü
#定义__UINT32_MAX__ 0xffffffffU
#定义__UINT32_TYPE__无符号整型
#定义__UINT64_C(三)C ## ULL
#定义__UINT64_MAX__ 0xffffffffffffffffULL
#定义__UINT64_TYPE__长长的无符号整型
#定义__UINT8_C(三)C
#定义__UINT8_MAX__为0xFF
#定义__UINT8_TYPE__ unsigned char型
#定义__UINT_FAST16_MAX__ 0xffffffffU
#定义__UINT_FAST16_TYPE__无符号整型
#定义__UINT_FAST32_MAX__ 0xffffffffU
#定义__UINT_FAST32_TYPE__无符号整型
#定义__UINT_FAST64_MAX__ 0xffffffffffffffffULL
#定义__UINT_FAST64_TYPE__长长的无符号整型
#定义__UINT_FAST8_MAX__为0xFF
#定义__UINT_FAST8_TYPE__ unsigned char型
#定义__UINT_LEAST16_MAX__ 0xFFFF的
#定义__UINT_LEAST16_TYPE__短的无符号整型
#定义__UINT_LEAST32_MAX__ 0xffffffffU
#定义__UINT_LEAST32_TYPE__无符号整型
#定义__UINT_LEAST64_MAX__ 0xffffffffffffffffULL
#定义__UINT_LEAST64_TYPE__长长的无符号整型
#定义__UINT_LEAST8_MAX__为0xFF
#定义__UINT_LEAST8_TYPE__ unsigned char型
#定义__UINTMAX_C(三)C ## ULL
#定义__UINTMAX_MAX__ 0xffffffffffffffffULL
#定义__UINTMAX_TYPE__长长的无符号整型
#定义__UINTPTR_MAX__ 0xffffffffU
#定义__UINTPTR_TYPE__无符号整型
#定义__unix 1
#定义__unix__ 1
#定义UNIX 1
#定义__USER_LABEL_ preFIX__
#定义__VERSION__5.2.1 20150911
#定义__WCHAR_MAX__ 0x7fffffffL
#定义__WCHAR_MIN__(-__ WCHAR_MAX__ - 1)
#定义__WCHAR_TYPE__长整型
#定义__WINT_MAX__ 0xffffffffU
#定义__WINT_MIN__ 0U
#定义__WINT_TYPE__无符号整型
#定义__x86_64 1
#定义__x86_64__ 1


解决方案

有没有出现是predefined宏明确指定的X32环境。 CPP -dM CPP -dM -mx32 ,符号 _ILP32 和 __ ILP32 __ 只为X32定义, _LP64 __ LP64__ 只为没有X32 x86_64的定义。其他一些predefined宏有不同的值,这两个环境。

我想在编译时检测到X32的最直接的方法是检查 __ x86_64的__ SIZE_MAX 宏。前者是predefined(或没有)海湾合作委员会,后者定义在< stdint.h方式>

下面的程序演示这一点。它的工作原理正确地的gcc -m64 的gcc -m32 GCC -mx32 的x86_64的系统上,,并与 GCC 在非x86_64的系统(SPARC)。

 的#include<&stdio.h中GT;
#包括LT&;&stdint.h GT;
诠释主要(无效){
#IFDEF __x86_64__
    #如果SIZE_MAX 0xFFFFFFFF的==
        看跌期权(这是X32);
    #其他
        看跌期权(这是x86_64的而不是X32);
    #万一
#其他
    看跌期权(这不是x64_64);
#万一
}

X32 is an ABI for amd64/x86_64 CPUs using 32-bit pointers. The idea is to combine the larger register set of x86_64 with the smaller memory and cache footprint resulting from 32-bit pointers. It provides up to about a 40% speedup. See Difference between x86, x32, and x64 architectures on Stack Overflow, and the Debian X32 Ports wiki page for details and setting it up as a chroot environment.

We have a bug report from a Debian maintainer under the environment. The report is adcq is an illegal instruction. The inline assembly is activated based on preprocessor macros, so we are not detecting X32 properly (or more correctly, not at all until now).

The most obvious choice (to me) for a preprocessor macro is something like __X32__, but that's not offered. Based on Clang's patch and Debian's suggestion, it looks like __ILP32__ can be used. But I'd like a more canonical answer since _ILP32 and __code_model_small__ look interesting, too. (And I'm aware of issues with SSE2, where the compiler supported it but the OS did not).

What are the preprocessor macros that can be used to reliably detect an X32 ABI and environment when using Clang and GCC?

To be clear, I'm not trying to fix the code at this point. I just want to know the macros that can be used in a complete remediation.


# cpp -dM < /dev/null | sort
#define __amd64 1
#define __amd64__ 1
#define __ATOMIC_ACQ_REL 4
#define __ATOMIC_ACQUIRE 2
#define __ATOMIC_CONSUME 1
#define __ATOMIC_HLE_ACQUIRE 65536
#define __ATOMIC_HLE_RELEASE 131072
#define __ATOMIC_RELAXED 0
#define __ATOMIC_RELEASE 3
#define __ATOMIC_SEQ_CST 5
#define __BIGGEST_ALIGNMENT__ 16
#define __BYTE_ORDER__ __ORDER_LITTLE_ENDIAN__
#define __CHAR16_TYPE__ short unsigned int
#define __CHAR32_TYPE__ unsigned int
#define __CHAR_BIT__ 8
#define __code_model_small__ 1
#define __DBL_DECIMAL_DIG__ 17
#define __DBL_DENORM_MIN__ ((double)4.94065645841246544177e-324L)
#define __DBL_DIG__ 15
#define __DBL_EPSILON__ ((double)2.22044604925031308085e-16L)
#define __DBL_HAS_DENORM__ 1
#define __DBL_HAS_INFINITY__ 1
#define __DBL_HAS_QUIET_NAN__ 1
#define __DBL_MANT_DIG__ 53
#define __DBL_MAX_10_EXP__ 308
#define __DBL_MAX__ ((double)1.79769313486231570815e+308L)
#define __DBL_MAX_EXP__ 1024
#define __DBL_MIN_10_EXP__ (-307)
#define __DBL_MIN__ ((double)2.22507385850720138309e-308L)
#define __DBL_MIN_EXP__ (-1021)
#define __DEC128_EPSILON__ 1E-33DL
#define __DEC128_MANT_DIG__ 34
#define __DEC128_MAX__ 9.999999999999999999999999999999999E6144DL
#define __DEC128_MAX_EXP__ 6145
#define __DEC128_MIN__ 1E-6143DL
#define __DEC128_MIN_EXP__ (-6142)
#define __DEC128_SUBNORMAL_MIN__ 0.000000000000000000000000000000001E-6143DL
#define __DEC32_EPSILON__ 1E-6DF
#define __DEC32_MANT_DIG__ 7
#define __DEC32_MAX__ 9.999999E96DF
#define __DEC32_MAX_EXP__ 97
#define __DEC32_MIN__ 1E-95DF
#define __DEC32_MIN_EXP__ (-94)
#define __DEC32_SUBNORMAL_MIN__ 0.000001E-95DF
#define __DEC64_EPSILON__ 1E-15DD
#define __DEC64_MANT_DIG__ 16
#define __DEC64_MAX__ 9.999999999999999E384DD
#define __DEC64_MAX_EXP__ 385
#define __DEC64_MIN__ 1E-383DD
#define __DEC64_MIN_EXP__ (-382)
#define __DEC64_SUBNORMAL_MIN__ 0.000000000000001E-383DD
#define __DEC_EVAL_METHOD__ 2
#define __DECIMAL_BID_FORMAT__ 1
#define __DECIMAL_DIG__ 21
#define __ELF__ 1
#define __FINITE_MATH_ONLY__ 0
#define __FLOAT_WORD_ORDER__ __ORDER_LITTLE_ENDIAN__
#define __FLT_DECIMAL_DIG__ 9
#define __FLT_DENORM_MIN__ 1.40129846432481707092e-45F
#define __FLT_DIG__ 6
#define __FLT_EPSILON__ 1.19209289550781250000e-7F
#define __FLT_EVAL_METHOD__ 0
#define __FLT_HAS_DENORM__ 1
#define __FLT_HAS_INFINITY__ 1
#define __FLT_HAS_QUIET_NAN__ 1
#define __FLT_MANT_DIG__ 24
#define __FLT_MAX_10_EXP__ 38
#define __FLT_MAX__ 3.40282346638528859812e+38F
#define __FLT_MAX_EXP__ 128
#define __FLT_MIN_10_EXP__ (-37)
#define __FLT_MIN__ 1.17549435082228750797e-38F
#define __FLT_MIN_EXP__ (-125)
#define __FLT_RADIX__ 2
#define __FXSR__ 1
#define __GCC_ATOMIC_BOOL_LOCK_FREE 2
#define __GCC_ATOMIC_CHAR16_T_LOCK_FREE 2
#define __GCC_ATOMIC_CHAR32_T_LOCK_FREE 2
#define __GCC_ATOMIC_CHAR_LOCK_FREE 2
#define __GCC_ATOMIC_INT_LOCK_FREE 2
#define __GCC_ATOMIC_LLONG_LOCK_FREE 2
#define __GCC_ATOMIC_LONG_LOCK_FREE 2
#define __GCC_ATOMIC_POINTER_LOCK_FREE 2
#define __GCC_ATOMIC_SHORT_LOCK_FREE 2
#define __GCC_ATOMIC_TEST_AND_SET_TRUEVAL 1
#define __GCC_ATOMIC_WCHAR_T_LOCK_FREE 2
#define __GCC_HAVE_DWARF2_CFI_ASM 1
#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
#define __GCC_HAVE_SYNC_COMPARE_AND_SWAP_8 1
#define __GCC_IEC_559 2
#define __GCC_IEC_559_COMPLEX 2
#define __GNUC__ 5
#define __GNUC_MINOR__ 2
#define __GNUC_PATCHLEVEL__ 1
#define __GNUC_STDC_INLINE__ 1
#define __gnu_linux__ 1
#define __GXX_ABI_VERSION 1009
#define __has_include_next(STR) __has_include_next__(STR)
#define __has_include(STR) __has_include__(STR)
#define __ILP32__ 1
#define _ILP32 1
#define __INT16_C(c) c
#define __INT16_MAX__ 0x7fff
#define __INT16_TYPE__ short int
#define __INT32_C(c) c
#define __INT32_MAX__ 0x7fffffff
#define __INT32_TYPE__ int
#define __INT64_C(c) c ## LL
#define __INT64_MAX__ 0x7fffffffffffffffLL
#define __INT64_TYPE__ long long int
#define __INT8_C(c) c
#define __INT8_MAX__ 0x7f
#define __INT8_TYPE__ signed char
#define __INT_FAST16_MAX__ 0x7fffffff
#define __INT_FAST16_TYPE__ int
#define __INT_FAST32_MAX__ 0x7fffffff
#define __INT_FAST32_TYPE__ int
#define __INT_FAST64_MAX__ 0x7fffffffffffffffLL
#define __INT_FAST64_TYPE__ long long int
#define __INT_FAST8_MAX__ 0x7f
#define __INT_FAST8_TYPE__ signed char
#define __INT_LEAST16_MAX__ 0x7fff
#define __INT_LEAST16_TYPE__ short int
#define __INT_LEAST32_MAX__ 0x7fffffff
#define __INT_LEAST32_TYPE__ int
#define __INT_LEAST64_MAX__ 0x7fffffffffffffffLL
#define __INT_LEAST64_TYPE__ long long int
#define __INT_LEAST8_MAX__ 0x7f
#define __INT_LEAST8_TYPE__ signed char
#define __INT_MAX__ 0x7fffffff
#define __INTMAX_C(c) c ## LL
#define __INTMAX_MAX__ 0x7fffffffffffffffLL
#define __INTMAX_TYPE__ long long int
#define __INTPTR_MAX__ 0x7fffffff
#define __INTPTR_TYPE__ int
#define __k8 1
#define __k8__ 1
#define __LDBL_DENORM_MIN__ 3.64519953188247460253e-4951L
#define __LDBL_DIG__ 18
#define __LDBL_EPSILON__ 1.08420217248550443401e-19L
#define __LDBL_HAS_DENORM__ 1
#define __LDBL_HAS_INFINITY__ 1
#define __LDBL_HAS_QUIET_NAN__ 1
#define __LDBL_MANT_DIG__ 64
#define __LDBL_MAX_10_EXP__ 4932
#define __LDBL_MAX__ 1.18973149535723176502e+4932L
#define __LDBL_MAX_EXP__ 16384
#define __LDBL_MIN_10_EXP__ (-4931)
#define __LDBL_MIN__ 3.36210314311209350626e-4932L
#define __LDBL_MIN_EXP__ (-16381)
#define __linux 1
#define __linux__ 1
#define linux 1
#define __LONG_LONG_MAX__ 0x7fffffffffffffffLL
#define __LONG_MAX__ 0x7fffffffL
#define __MMX__ 1
#define __NO_INLINE__ 1
#define __ORDER_BIG_ENDIAN__ 4321
#define __ORDER_LITTLE_ENDIAN__ 1234
#define __ORDER_PDP_ENDIAN__ 3412
#define __PRAGMA_REDEFINE_EXTNAME 1
#define __PTRDIFF_MAX__ 0x7fffffff
#define __PTRDIFF_TYPE__ int
#define __REGISTER_PREFIX__
#define __SCHAR_MAX__ 0x7f
#define __SHRT_MAX__ 0x7fff
#define __SIG_ATOMIC_MAX__ 0x7fffffff
#define __SIG_ATOMIC_MIN__ (-__SIG_ATOMIC_MAX__ - 1)
#define __SIG_ATOMIC_TYPE__ int
#define __SIZE_MAX__ 0xffffffffU
#define __SIZEOF_DOUBLE__ 8
#define __SIZEOF_FLOAT128__ 16
#define __SIZEOF_FLOAT__ 4
#define __SIZEOF_FLOAT80__ 16
#define __SIZEOF_INT128__ 16
#define __SIZEOF_INT__ 4
#define __SIZEOF_LONG__ 4
#define __SIZEOF_LONG_DOUBLE__ 16
#define __SIZEOF_LONG_LONG__ 8
#define __SIZEOF_POINTER__ 4
#define __SIZEOF_PTRDIFF_T__ 4
#define __SIZEOF_SHORT__ 2
#define __SIZEOF_SIZE_T__ 4
#define __SIZEOF_WCHAR_T__ 4
#define __SIZEOF_WINT_T__ 4
#define __SIZE_TYPE__ unsigned int
#define __SSE__ 1
#define __SSE2__ 1
#define __SSE2_MATH__ 1
#define __SSE_MATH__ 1
#define __STDC__ 1
#define __STDC_HOSTED__ 1
#define __STDC_IEC_559__ 1
#define __STDC_IEC_559_COMPLEX__ 1
#define __STDC_ISO_10646__ 201103L
#define __STDC_NO_THREADS__ 1
#define _STDC_PREDEF_H 1
#define __STDC_UTF_16__ 1
#define __STDC_UTF_32__ 1
#define __STDC_VERSION__ 201112L
#define __UINT16_C(c) c
#define __UINT16_MAX__ 0xffff
#define __UINT16_TYPE__ short unsigned int
#define __UINT32_C(c) c ## U
#define __UINT32_MAX__ 0xffffffffU
#define __UINT32_TYPE__ unsigned int
#define __UINT64_C(c) c ## ULL
#define __UINT64_MAX__ 0xffffffffffffffffULL
#define __UINT64_TYPE__ long long unsigned int
#define __UINT8_C(c) c
#define __UINT8_MAX__ 0xff
#define __UINT8_TYPE__ unsigned char
#define __UINT_FAST16_MAX__ 0xffffffffU
#define __UINT_FAST16_TYPE__ unsigned int
#define __UINT_FAST32_MAX__ 0xffffffffU
#define __UINT_FAST32_TYPE__ unsigned int
#define __UINT_FAST64_MAX__ 0xffffffffffffffffULL
#define __UINT_FAST64_TYPE__ long long unsigned int
#define __UINT_FAST8_MAX__ 0xff
#define __UINT_FAST8_TYPE__ unsigned char
#define __UINT_LEAST16_MAX__ 0xffff
#define __UINT_LEAST16_TYPE__ short unsigned int
#define __UINT_LEAST32_MAX__ 0xffffffffU
#define __UINT_LEAST32_TYPE__ unsigned int
#define __UINT_LEAST64_MAX__ 0xffffffffffffffffULL
#define __UINT_LEAST64_TYPE__ long long unsigned int
#define __UINT_LEAST8_MAX__ 0xff
#define __UINT_LEAST8_TYPE__ unsigned char
#define __UINTMAX_C(c) c ## ULL
#define __UINTMAX_MAX__ 0xffffffffffffffffULL
#define __UINTMAX_TYPE__ long long unsigned int
#define __UINTPTR_MAX__ 0xffffffffU
#define __UINTPTR_TYPE__ unsigned int
#define __unix 1
#define __unix__ 1
#define unix 1
#define __USER_LABEL_PREFIX__
#define __VERSION__ "5.2.1 20150911"
#define __WCHAR_MAX__ 0x7fffffffL
#define __WCHAR_MIN__ (-__WCHAR_MAX__ - 1)
#define __WCHAR_TYPE__ long int
#define __WINT_MAX__ 0xffffffffU
#define __WINT_MIN__ 0U
#define __WINT_TYPE__ unsigned int
#define __x86_64 1
#define __x86_64__ 1
解决方案

There doesn't appear to be a predefined macro that explicitly specifies an x32 environment. Comparing the output of cpp -dM and cpp -dM -mx32, the symbols _ILP32 and __ILP32__ are defined only for x32, and _LP64 and __LP64__ are defined only for x86_64 without x32. A number of other predefined macros have different values for the two environments.

I think the most straightforward way to detect x32 at compile time is to examine the __x86_64__ and SIZE_MAX macros. The former is predefined (or not) by gcc, and the latter is defined in <stdint.h>.

The following program demonstrates this. It works correctly with gcc -m64, gcc -m32, and gcc -mx32 on an x86_64 system,, and with gcc on a non-x86_64 system (SPARC).

#include <stdio.h>
#include <stdint.h>
int main(void) {
#ifdef __x86_64__
    #if SIZE_MAX == 0xFFFFFFFF
        puts("This is x32");
    #else
        puts("This is x86_64 but not x32");
    #endif
#else
    puts("This is not x64_64");
#endif
}

这篇关于如何检测X32 ABI或环境在preprocessor?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持!

09-10 21:30