3

longまたはを使用して、組み込みの 64 ビット データ型を選択するための要件を理解しようとしていますlong longlong型の等価性とvsのアラインメント要件を理解するのに苦労していますlong long

longLP64/ ILP64 long long/LLP64 データ モデルで 64 ビット タイプとして選択する場合のベスト プラクティスまたは基準は何ですか?


関連する質問をいくつか紹介します。などの他のトピックを完全にカバーsizeof(long) <= sizeof(long long)していますが、同等性やアライメントの期待についてはまったく扱っていません。また、 などの非組み込み型にも大きく依存していuint64_tます。


背景情報は、使用する組み込みの 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;
}
4

0 に答える 0