У нас есть код, который работает на нескольких платформах. Код использует BMI / BMI2 встроенные функции, когда они доступны, как Core i7 5-го поколения. GCC, поставляемый Sun на Solaris 11.3, является определяющим __BMI__
а также __BMI2__
, но он испытывает затруднения при определении местонахождения BMI / BMI2:
$ cat test.cxx
#include <x86intrin.h>
int main(int argc, char* argv[])
{
unsigned long long t = argc;
#if defined(__BMI__) || defined(__BMI2__)
t = _blsr_u64(t);
#endif
return int(t);
}
$ /bin/g++ -march=native test.cxx -o test.exe
test.cxx: In function ‘int main(int, char**)’:
test.cxx:6:18: error: ‘_blsr_u64’ was not declared in this scope
t = _blsr_u64(t);
^
В том числе immintrin.h не имеет значения.
Какой заголовок мы включаем для _blsr_u64
при использовании GCC на Solaris 11.3?
Вот соответствующие определения от GCC:
$ /bin/g++ -march=native -dM -E - < /dev/null | sort | \
/usr/gnu/bin/egrep -i '(sse|aes|rdrnd|rdseed|avx|bmi)'
#define __AES__ 1
#define __AVX__ 1
#define __AVX2__ 1
#define __BMI__ 1
#define __BMI2__ 1
#define __core_avx2 1
#define __core_avx2__ 1
#define __RDRND__ 1
#define __RDSEED__ 1
#define __SSE__ 1
#define __SSE2__ 1
#define __SSE3__ 1
#define __SSE4_1__ 1
#define __SSE4_2__ 1
#define __SSSE3__ 1
#define __tune_core_avx2__ 1
И особенности процессора:
$ isainfo -v
64-bit amd64 applications
avx xsave pclmulqdq aes movbe sse4.2 sse4.1 ssse3 amd_lzcnt popcnt tscp
ahf cx16 sse3 sse2 sse fxsr mmx cmov amd_sysc cx8 tsc fpu prfchw adx
rdseed efs rtm hle bmi2 avx2 bmi1 f16c fma rdrand
И версия GCC:
$ /bin/g++ --version
g++ (GCC) 4.8.2
Copyright (C) 2013 Free Software Foundation, Inc.
Какой заголовок мы включаем для _blsr_u64 при использовании GCC в Solaris 11.3?
Это выглядит как #include <x86intrin.h>
верно.
Проблема была в том, что для вызова компилятора требовалось -march=native -m64
хотя 64-битная версия является родной для машины, а ядро 64-битной:
$ /bin/g++ -march=native -m64 test.cxx -o test.exe
Других решений пока нет …