long
またはを使用して、組み込みの 64 ビット データ型を選択するための要件を理解しようとしていますlong long
。long
型の等価性とvsのアラインメント要件を理解するのに苦労していますlong long
。
long
LP64/ ILP64 long long
/LLP64 データ モデルで 64 ビット タイプとして選択する場合のベスト プラクティスまたは基準は何ですか?
関連する質問をいくつか紹介します。などの他のトピックを完全にカバーsizeof(long) <= sizeof(long long)
していますが、同等性やアライメントの期待についてはまったく扱っていません。また、 などの非組み込み型にも大きく依存していuint64_t
ます。
- UNIX システム - 64 ビットとデータ サイズの中立性
- c/c++ の unsigned long/long/int の違いは何ですか?
- 64 ビットでは long または long long を使用することをお勧めします
- 64 ビット データ モデルでの 64 ビット符号なし整数リテラルの指定
背景情報は、使用する組み込みの 64 ビット型を選択するための次のようなコードです。
#if _LP64 || __LP64__ || _ILP64 || __ILP64__
typedef my_u64 unsigned long;
#else
typedef my_u64 unsigned long long;
#endif
i686 および ARMv7 以下は、 を使用しても問題ありませんtypedef my_u64 unsigned long long
。
最初の問題ケース
x86_64 (AMD64) マシンでが有効な場合 typedef my_u64 unsigned long
( が原因でLP64
)、コンパイル エラーが発生します。
$ gcc -mrdrnd test.cc -o test.exe
test.cc: In function ‘int main(int, char**)’:
test.cc:18:22: error: invalid conversion from ‘my_u64* {aka long unsigned int*}’ to ‘long long unsigned int*’ [-fpermissive]
_rdrand64_step(&val);
^
In file included from /usr/lib/gcc/x86_64-linux-gnu/4.9/include/x86intrin.h:46:0,
from test.cc:2:
/usr/lib/gcc/x86_64-linux-gnu/4.9/include/immintrin.h:166:1: note: initializing argument 1 of ‘int _rdrand64_step(long long unsigned int*)’
_rdrand64_step (unsigned long long *__P)
^
2番目の問題ケース
Aarch64 (ARM64) マシンでが有効な場合 typedef my_u64 unsigned long long
(x86_64 エラーを修正するため)、コンパイル エラーが発生します。
$ gcc test.cc -o test.exe
test.cc: In function ‘int main(int, char**)’:
test.cc:21:16: error: invalid conversion from ‘my_u64* {aka long long unsigned int*}’ to ‘const uint64_t* {aka const long unsigned int*}’ [-fpermissive]
vld1q_u64(val);
^
In file included from test.cc:4:0:
/usr/lib/gcc/aarch64-linux-gnu/4.9/include/arm_neon.h:17003:1: note: initializing argument 1 of ‘uint64x2_t vld1q_u64(const uint64_t*)’
vld1q_u64 (const uint64_t *a)
^
$ cat test.cc
#if __x86_64__
#include <x86intrin.h>
#elif __aarch64__
#include <arm_neon.h>
#include <arm_acle.h>
#endif
#if _LP64 || __LP64__
typedef unsigned long my_u64;
#else
typedef unsigned long long my_u64;
#endif
int main(int argc, char* argv[])
{
#if __x86_64__
my_u64 val;
_rdrand64_step(&val);
#elif __aarch64__
my_u64 val[2];
vld1q_u64(val);
#endif
return 0;
}