Index: src/ia32/assembler-ia32.cc |
diff --git a/src/ia32/assembler-ia32.cc b/src/ia32/assembler-ia32.cc |
index 19e69b319121dc786f6695706641a320e860874e..baa2076344fe1a90a97d3dfac3f6efb80b23e99f 100644 |
--- a/src/ia32/assembler-ia32.cc |
+++ b/src/ia32/assembler-ia32.cc |
@@ -40,15 +40,15 @@ |
#if V8_TARGET_ARCH_IA32 |
+#if V8_LIBC_MSVCRT |
+#include <intrin.h> // _xgetbv() |
+#endif |
#if V8_OS_MACOSX |
#include <sys/sysctl.h> |
#endif |
#include "src/base/bits.h" |
#include "src/base/cpu.h" |
-#if V8_OS_WIN |
-#include "src/base/win32-headers.h" |
-#endif |
#include "src/disassembler.h" |
#include "src/macro-assembler.h" |
#include "src/v8.h" |
@@ -61,13 +61,30 @@ namespace internal { |
namespace { |
-bool EnableAVX() { |
+#if !V8_LIBC_MSVCRT |
+ |
+V8_INLINE uint64_t _xgetbv(unsigned int xcr) { |
+ unsigned eax, edx; |
+ // Check xgetbv; this uses a .byte sequence instead of the instruction |
+ // directly because older assemblers do not include support for xgetbv and |
+ // there is no easy way to conditionally compile based on the assembler |
+ // used. |
+ __asm__ volatile(".byte 0x0f, 0x01, 0xd0" : "=a"(eax), "=d"(edx) : "c"(xcr)); |
+ return static_cast<uint64_t>(eax) | (static_cast<uint64_t>(edx) << 32); |
+} |
+ |
+#define _XCR_XFEATURE_ENABLED_MASK 0 |
+ |
+#endif // !V8_LIBC_MSVCRT |
+ |
+ |
+bool OSHasAVXSupport() { |
#if V8_OS_MACOSX |
// Mac OS X up to 10.9 has a bug where AVX transitions were indeed being |
// caused by ISRs, so we detect that here and disable AVX in that case. |
char buffer[128]; |
size_t buffer_size = arraysize(buffer); |
- int ctl_name[] = { CTL_KERN , KERN_OSRELEASE }; |
+ int ctl_name[] = {CTL_KERN, KERN_OSRELEASE}; |
if (sysctl(ctl_name, 2, buffer, &buffer_size, nullptr, 0) != 0) { |
V8_Fatal(__FILE__, __LINE__, "V8 failed to get kernel version"); |
} |
@@ -78,21 +95,10 @@ bool EnableAVX() { |
*period_pos = '\0'; |
long kernel_version_major = strtol(buffer, nullptr, 10); // NOLINT |
if (kernel_version_major <= 13) return false; |
-#elif V8_OS_WIN |
- // The same problem seems to appear on Windows XP and Vista. |
- OSVERSIONINFOEX osvi; |
- DWORDLONG mask = 0; |
- memset(&osvi, 0, sizeof(osvi)); |
- osvi.dwOSVersionInfoSize = sizeof(osvi); |
- osvi.dwMajorVersion = 6; |
- osvi.dwMinorVersion = 1; |
- VER_SET_CONDITION(mask, VER_MAJORVERSION, VER_GREATER_EQUAL); |
- VER_SET_CONDITION(mask, VER_MINORVERSION, VER_GREATER_EQUAL); |
- if (!VerifyVersionInfo(&osvi, VER_MAJORVERSION | VER_MINORVERSION, mask)) { |
- return false; |
- } |
#endif // V8_OS_MACOSX |
- return FLAG_enable_avx; |
+ // Check whether OS claims to support AVX. |
+ uint64_t feature_mask = _xgetbv(_XCR_XFEATURE_ENABLED_MASK); |
+ return (feature_mask & 0x6) == 0x6; |
} |
} // namespace |
@@ -108,8 +114,14 @@ void CpuFeatures::ProbeImpl(bool cross_compile) { |
if (cpu.has_sse41() && FLAG_enable_sse4_1) supported_ |= 1u << SSE4_1; |
if (cpu.has_sse3() && FLAG_enable_sse3) supported_ |= 1u << SSE3; |
- if (cpu.has_avx() && EnableAVX()) supported_ |= 1u << AVX; |
- if (cpu.has_fma3() && FLAG_enable_fma3) supported_ |= 1u << FMA3; |
+ if (cpu.has_avx() && FLAG_enable_avx && cpu.has_osxsave() && |
+ OSHasAVXSupport()) { |
+ supported_ |= 1u << AVX; |
+ } |
+ if (cpu.has_fma3() && FLAG_enable_fma3 && cpu.has_osxsave() && |
+ OSHasAVXSupport()) { |
+ supported_ |= 1u << FMA3; |
+ } |
if (strcmp(FLAG_mcpu, "auto") == 0) { |
if (cpu.is_atom()) supported_ |= 1u << ATOM; |
} else if (strcmp(FLAG_mcpu, "atom") == 0) { |