8239090: Improve CPU feature support in VM_Version

Reviewed-by: iveresov, simonis
This commit is contained in:
Paul Hohensee 2020-09-30 17:02:08 +00:00
parent 8f7c9a7b18
commit 9e453d9792
6 changed files with 209 additions and 212 deletions

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, 2019, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2020, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -37,7 +37,9 @@
#define VM_INT_CONSTANTS_CPU(declare_constant, declare_preprocessor_constant, declare_c1_constant, declare_c2_constant, declare_c2_preprocessor_constant) \ #define VM_INT_CONSTANTS_CPU(declare_constant, declare_preprocessor_constant, declare_c1_constant, declare_c2_constant, declare_c2_preprocessor_constant) \
LP64_ONLY(declare_constant(frame::arg_reg_save_area_bytes)) \ LP64_ONLY(declare_constant(frame::arg_reg_save_area_bytes)) \
declare_constant(frame::interpreter_frame_sender_sp_offset) \ declare_constant(frame::interpreter_frame_sender_sp_offset) \
declare_constant(frame::interpreter_frame_last_sp_offset) \ declare_constant(frame::interpreter_frame_last_sp_offset)
#define VM_LONG_CONSTANTS_CPU(declare_constant, declare_preprocessor_constant, declare_c1_constant, declare_c2_constant, declare_c2_preprocessor_constant) \
declare_constant(VM_Version::CPU_CX8) \ declare_constant(VM_Version::CPU_CX8) \
declare_constant(VM_Version::CPU_CMOV) \ declare_constant(VM_Version::CPU_CMOV) \
declare_constant(VM_Version::CPU_FXSR) \ declare_constant(VM_Version::CPU_FXSR) \
@ -68,22 +70,21 @@
declare_constant(VM_Version::CPU_AVX512DQ) \ declare_constant(VM_Version::CPU_AVX512DQ) \
declare_constant(VM_Version::CPU_AVX512PF) \ declare_constant(VM_Version::CPU_AVX512PF) \
declare_constant(VM_Version::CPU_AVX512ER) \ declare_constant(VM_Version::CPU_AVX512ER) \
declare_constant(VM_Version::CPU_AVX512CD) declare_constant(VM_Version::CPU_AVX512CD) \
declare_constant(VM_Version::CPU_AVX512BW) \
#define VM_LONG_CONSTANTS_CPU(declare_constant, declare_preprocessor_constant, declare_c1_constant, declare_c2_constant, declare_c2_preprocessor_constant) \ declare_constant(VM_Version::CPU_AVX512VL) \
declare_preprocessor_constant("VM_Version::CPU_AVX512BW", CPU_AVX512BW) \ declare_constant(VM_Version::CPU_SHA) \
declare_preprocessor_constant("VM_Version::CPU_AVX512VL", CPU_AVX512VL) \ declare_constant(VM_Version::CPU_FMA) \
declare_preprocessor_constant("VM_Version::CPU_SHA", CPU_SHA) \ declare_constant(VM_Version::CPU_VZEROUPPER) \
declare_preprocessor_constant("VM_Version::CPU_FMA", CPU_FMA) \ declare_constant(VM_Version::CPU_AVX512_VPOPCNTDQ) \
declare_preprocessor_constant("VM_Version::CPU_VZEROUPPER", CPU_VZEROUPPER) \ declare_constant(VM_Version::CPU_AVX512_VPCLMULQDQ) \
declare_preprocessor_constant("VM_Version::CPU_AVX512_VPOPCNTDQ", CPU_AVX512_VPOPCNTDQ) \ declare_constant(VM_Version::CPU_AVX512_VAES) \
declare_preprocessor_constant("VM_Version::CPU_AVX512_VPCLMULQDQ", CPU_AVX512_VPCLMULQDQ) \ declare_constant(VM_Version::CPU_AVX512_VNNI) \
declare_preprocessor_constant("VM_Version::CPU_AVX512_VAES", CPU_AVX512_VAES) \ declare_constant(VM_Version::CPU_FLUSH) \
declare_preprocessor_constant("VM_Version::CPU_AVX512_VNNI", CPU_AVX512_VNNI) \ declare_constant(VM_Version::CPU_FLUSHOPT) \
declare_preprocessor_constant("VM_Version::CPU_FLUSH", CPU_FLUSH) \ declare_constant(VM_Version::CPU_CLWB) \
declare_preprocessor_constant("VM_Version::CPU_FLUSHOPT", CPU_FLUSHOPT) \ declare_constant(VM_Version::CPU_AVX512_VBMI2) \
declare_preprocessor_constant("VM_Version::CPU_CLWB", CPU_CLWB) \ declare_constant(VM_Version::CPU_AVX512_VBMI) \
declare_preprocessor_constant("VM_Version::CPU_AVX512_VBMI2", CPU_AVX512_VBMI2) \ declare_constant(VM_Version::CPU_HV)
declare_preprocessor_constant("VM_Version::CPU_AVX512_VBMI", CPU_AVX512_VBMI)
#endif // CPU_X86_VMSTRUCTS_X86_HPP #endif // CPU_X86_VMSTRUCTS_X86_HPP

View File

@ -44,6 +44,7 @@ int VM_Version::_model;
int VM_Version::_stepping; int VM_Version::_stepping;
bool VM_Version::_has_intel_jcc_erratum; bool VM_Version::_has_intel_jcc_erratum;
VM_Version::CpuidInfo VM_Version::_cpuid_info = { 0, }; VM_Version::CpuidInfo VM_Version::_cpuid_info = { 0, };
const char* VM_Version::_features_names[] = { FEATURES_NAMES };
// Address of instruction which causes SEGV // Address of instruction which causes SEGV
address VM_Version::_cpuinfo_segv_addr = 0; address VM_Version::_cpuinfo_segv_addr = 0;
@ -772,65 +773,14 @@ void VM_Version::get_processor_features() {
} }
char buf[512]; char buf[512];
int res = jio_snprintf(buf, sizeof(buf), int res = jio_snprintf(
"(%u cores per cpu, %u threads per core) family %d model %d stepping %d microcode 0x%x" buf, sizeof(buf),
"%s%s%s%s%s%s%s%s%s%s" "%s%s%s%s%s%s%s%s%s%s" "%s%s%s%s%s%s%s%s%s%s" "%s%s%s%s%s%s%s%s%s%s" "%s%s%s%s%s%s", "(%u cores per cpu, %u threads per core) family %d model %d stepping %d microcode 0x%x",
cores_per_cpu(), threads_per_core(),
cores_per_cpu(), threads_per_core(), cpu_family(), _model, _stepping, os::cpu_microcode_revision());
cpu_family(), _model, _stepping, os::cpu_microcode_revision(), assert(res > 0, "not enough temporary space allocated");
assert(exact_log2(CPU_MAX_FEATURE) + 1 == sizeof(_features_names) / sizeof(char*), "wrong size features_names");
(supports_cmov() ? ", cmov" : ""), insert_features_names(buf + res, sizeof(buf) - res, _features_names);
(supports_cmpxchg8() ? ", cx8" : ""),
(supports_fxsr() ? ", fxsr" : ""),
(supports_mmx() ? ", mmx" : ""),
(supports_sse() ? ", sse" : ""),
(supports_sse2() ? ", sse2" : ""),
(supports_sse3() ? ", sse3" : ""),
(supports_ssse3()? ", ssse3": ""),
(supports_sse4_1() ? ", sse4.1" : ""),
(supports_sse4_2() ? ", sse4.2" : ""),
(supports_popcnt() ? ", popcnt" : ""),
(supports_vzeroupper() ? ", vzeroupper" : ""),
(supports_avx() ? ", avx" : ""),
(supports_avx2() ? ", avx2" : ""),
(supports_aes() ? ", aes" : ""),
(supports_clmul() ? ", clmul" : ""),
(supports_erms() ? ", erms" : ""),
(supports_rtm() ? ", rtm" : ""),
(supports_3dnow_prefetch() ? ", 3dnowpref" : ""),
(supports_lzcnt() ? ", lzcnt": ""),
(supports_sse4a() ? ", sse4a": ""),
(supports_ht() ? ", ht": ""),
(supports_tsc() ? ", tsc": ""),
(supports_tscinv_bit() ? ", tscinvbit": ""),
(supports_tscinv() ? ", tscinv": ""),
(supports_bmi1() ? ", bmi1" : ""),
(supports_bmi2() ? ", bmi2" : ""),
(supports_adx() ? ", adx" : ""),
(supports_evex() ? ", avx512f" : ""),
(supports_avx512dq() ? ", avx512dq" : ""),
(supports_avx512pf() ? ", avx512pf" : ""),
(supports_avx512er() ? ", avx512er" : ""),
(supports_avx512cd() ? ", avx512cd" : ""),
(supports_avx512bw() ? ", avx512bw" : ""),
(supports_avx512vl() ? ", avx512vl" : ""),
(supports_avx512_vpopcntdq() ? ", avx512_vpopcntdq" : ""),
(supports_avx512_vpclmulqdq() ? ", avx512_vpclmulqdq" : ""),
(supports_avx512_vbmi() ? ", avx512_vbmi" : ""),
(supports_avx512_vbmi2() ? ", avx512_vbmi2" : ""),
(supports_avx512_vaes() ? ", avx512_vaes" : ""),
(supports_avx512_vnni() ? ", avx512_vnni" : ""),
(supports_sha() ? ", sha" : ""),
(supports_fma() ? ", fma" : ""),
(supports_clflush() ? ", clflush" : ""),
(supports_clflushopt() ? ", clflushopt" : ""),
(supports_clwb() ? ", clwb" : ""));
assert(res > 0, "not enough temporary space allocated"); // increase 'buf' size
_features_string = os::strdup(buf); _features_string = os::strdup(buf);

View File

@ -298,59 +298,87 @@ protected:
static address _cpuinfo_segv_addr; // address of instruction which causes SEGV static address _cpuinfo_segv_addr; // address of instruction which causes SEGV
static address _cpuinfo_cont_addr; // address of instruction after the one which causes SEGV static address _cpuinfo_cont_addr; // address of instruction after the one which causes SEGV
enum Feature_Flag { enum Feature_Flag : uint64_t {
CPU_CX8 = (1 << 0), // next bits are from cpuid 1 (EDX) CPU_CX8 = (1ULL << 0), // next bits are from cpuid 1 (EDX)
CPU_CMOV = (1 << 1), CPU_CMOV = (1ULL << 1),
CPU_FXSR = (1 << 2), CPU_FXSR = (1ULL << 2),
CPU_HT = (1 << 3), CPU_HT = (1ULL << 3),
CPU_MMX = (1 << 4),
CPU_3DNOW_PREFETCH = (1 << 5), // Processor supports 3dnow prefetch and prefetchw instructions CPU_MMX = (1ULL << 4),
// may not necessarily support other 3dnow instructions CPU_3DNOW_PREFETCH = (1ULL << 5), // Processor supports 3dnow prefetch and prefetchw instructions
CPU_SSE = (1 << 6), // may not necessarily support other 3dnow instructions
CPU_SSE2 = (1 << 7), CPU_SSE = (1ULL << 6),
CPU_SSE3 = (1 << 8), // SSE3 comes from cpuid 1 (ECX) CPU_SSE2 = (1ULL << 7),
CPU_SSSE3 = (1 << 9),
CPU_SSE4A = (1 << 10), CPU_SSE3 = (1ULL << 8), // SSE3 comes from cpuid 1 (ECX)
CPU_SSE4_1 = (1 << 11), CPU_SSSE3 = (1ULL << 9),
CPU_SSE4_2 = (1 << 12), CPU_SSE4A = (1ULL << 10),
CPU_POPCNT = (1 << 13), CPU_SSE4_1 = (1ULL << 11),
CPU_LZCNT = (1 << 14),
CPU_TSC = (1 << 15), CPU_SSE4_2 = (1ULL << 12),
CPU_TSCINV = (1 << 16), CPU_POPCNT = (1ULL << 13),
CPU_AVX = (1 << 17), CPU_LZCNT = (1ULL << 14),
CPU_AVX2 = (1 << 18), CPU_TSC = (1ULL << 15),
CPU_AES = (1 << 19),
CPU_ERMS = (1 << 20), // enhanced 'rep movsb/stosb' instructions CPU_TSCINV_BIT = (1ULL << 16),
CPU_CLMUL = (1 << 21), // carryless multiply for CRC CPU_TSCINV = (1ULL << 17),
CPU_BMI1 = (1 << 22), CPU_AVX = (1ULL << 18),
CPU_BMI2 = (1 << 23), CPU_AVX2 = (1ULL << 19),
CPU_RTM = (1 << 24), // Restricted Transactional Memory instructions
CPU_ADX = (1 << 25), CPU_AES = (1ULL << 20),
CPU_AVX512F = (1 << 26), // AVX 512bit foundation instructions CPU_ERMS = (1ULL << 21), // enhanced 'rep movsb/stosb' instructions
CPU_AVX512DQ = (1 << 27), CPU_CLMUL = (1ULL << 22), // carryless multiply for CRC
CPU_AVX512PF = (1 << 28), CPU_BMI1 = (1ULL << 23),
CPU_AVX512ER = (1 << 29),
CPU_AVX512CD = (1 << 30) CPU_BMI2 = (1ULL << 24),
// Keeping sign bit 31 unassigned. CPU_RTM = (1ULL << 25), // Restricted Transactional Memory instructions
CPU_ADX = (1ULL << 26),
CPU_AVX512F = (1ULL << 27), // AVX 512bit foundation instructions
CPU_AVX512DQ = (1ULL << 28),
CPU_AVX512PF = (1ULL << 29),
CPU_AVX512ER = (1ULL << 30),
CPU_AVX512CD = (1ULL << 31),
CPU_AVX512BW = (1ULL << 32), // Byte and word vector instructions
CPU_AVX512VL = (1ULL << 33), // EVEX instructions with smaller vector length
CPU_SHA = (1ULL << 34), // SHA instructions
CPU_FMA = (1ULL << 35), // FMA instructions
CPU_VZEROUPPER = (1ULL << 36), // Vzeroupper instruction
CPU_AVX512_VPOPCNTDQ = (1ULL << 37), // Vector popcount
CPU_AVX512_VPCLMULQDQ = (1ULL << 38), // Vector carryless multiplication
CPU_AVX512_VAES = (1ULL << 39), // Vector AES instruction
CPU_AVX512_VNNI = (1ULL << 40), // Vector Neural Network Instructions
CPU_FLUSH = (1ULL << 41), // flush instruction
CPU_FLUSHOPT = (1ULL << 42), // flusopth instruction
CPU_CLWB = (1ULL << 43), // clwb instruction
CPU_AVX512_VBMI2 = (1ULL << 44), // VBMI2 shift left double instructions
CPU_AVX512_VBMI = (1ULL << 45), // Vector BMI instructions
CPU_HV = (1ULL << 46), // Hypervisor instructions
CPU_MAX_FEATURE = CPU_HV
}; };
#define CPU_AVX512BW ((uint64_t)UCONST64( 0x100000000)) // enums are limited to 31 bit #define FEATURES_NAMES \
#define CPU_AVX512VL ((uint64_t)UCONST64( 0x200000000)) // EVEX instructions with smaller vector length "cx8", "cmov", "fxsr", "ht", \
#define CPU_SHA ((uint64_t)UCONST64( 0x400000000)) // SHA instructions "mmx", "3dnowpref", "sse", "sse2", \
#define CPU_FMA ((uint64_t)UCONST64( 0x800000000)) // FMA instructions "sse3", "ssse3", "sse4a", "sse4.1", \
#define CPU_VZEROUPPER ((uint64_t)UCONST64( 0x1000000000)) // Vzeroupper instruction "sse4.2", "popcnt", "lzcnt", "tsc", \
#define CPU_AVX512_VPOPCNTDQ ((uint64_t)UCONST64( 0x2000000000)) // Vector popcount "tscinvbit", "tscinv", "avx", "avx2", \
#define CPU_AVX512_VPCLMULQDQ ((uint64_t)UCONST64( 0x4000000000)) // Vector carryless multiplication "aes", "erms", "clmul", "bmi1", \
#define CPU_AVX512_VAES ((uint64_t)UCONST64( 0x8000000000)) // Vector AES instructions "bmi2", "rtm", "adx", "avx512f", \
#define CPU_AVX512_VNNI ((uint64_t)UCONST64( 0x10000000000)) // Vector Neural Network Instructions "avx512dq", "avx512pf", "avx512er", "avx512cd", \
#define CPU_FLUSH ((uint64_t)UCONST64( 0x20000000000)) // flush instruction "avx512bw", "avx512vl", "sha", "fma", \
#define CPU_FLUSHOPT ((uint64_t)UCONST64( 0x40000000000)) // flushopt instruction "vzeroupper", "avx512_vpopcntdq", "avx512_vpclmulqdq", "avx512_vaes", \
#define CPU_CLWB ((uint64_t)UCONST64( 0x80000000000)) // clwb instruction "avx512_vnni", "clflush", "clflushopt", "clwb", \
#define CPU_AVX512_VBMI2 ((uint64_t)UCONST64(0x100000000000)) // VBMI2 shift left double instructions "avx512_vmbi2", "avx512_vmbi", "hv"
#define CPU_AVX512_VBMI ((uint64_t)UCONST64(0x200000000000)) // Vector BMI instructions
#define CPU_HV_PRESENT ((uint64_t)UCONST64(0x400000000000)) // for hypervisor detection
// NB! When adding new CPU feature detection consider updating vmStructs_x86.hpp, vmStructs_jvmci.hpp, and VM_Version::get_processor_features(). static const char* _features_names[];
// NB! When adding new CPU feature detection consider updating vmStructs_x86.hpp, vmStructs_jvmci.hpp, and VM_Version::get_processor_features().
enum Extended_Family { enum Extended_Family {
// AMD // AMD
@ -582,13 +610,13 @@ enum Extended_Family {
} }
} }
if (_cpuid_info.std_cpuid1_ecx.bits.hv != 0) if (_cpuid_info.std_cpuid1_ecx.bits.hv != 0)
result |= CPU_HV_PRESENT; result |= CPU_HV;
if (_cpuid_info.sef_cpuid7_ebx.bits.bmi1 != 0) if (_cpuid_info.sef_cpuid7_ebx.bits.bmi1 != 0)
result |= CPU_BMI1; result |= CPU_BMI1;
if (_cpuid_info.std_cpuid1_edx.bits.tsc != 0) if (_cpuid_info.std_cpuid1_edx.bits.tsc != 0)
result |= CPU_TSC; result |= CPU_TSC;
if (_cpuid_info.ext_cpuid7_edx.bits.tsc_invariance != 0) if (_cpuid_info.ext_cpuid7_edx.bits.tsc_invariance != 0)
result |= CPU_TSCINV; result |= CPU_TSCINV_BIT;
if (_cpuid_info.std_cpuid1_ecx.bits.aes != 0) if (_cpuid_info.std_cpuid1_ecx.bits.aes != 0)
result |= CPU_AES; result |= CPU_AES;
if (_cpuid_info.sef_cpuid7_ebx.bits.erms != 0) if (_cpuid_info.sef_cpuid7_ebx.bits.erms != 0)
@ -618,6 +646,7 @@ enum Extended_Family {
if (_cpuid_info.ext_cpuid1_ecx.bits.sse4a != 0) if (_cpuid_info.ext_cpuid1_ecx.bits.sse4a != 0)
result |= CPU_SSE4A; result |= CPU_SSE4A;
} }
// Intel features. // Intel features.
if (is_intel()) { if (is_intel()) {
if (_cpuid_info.ext_cpuid1_ecx.bits.lzcnt_intel != 0) if (_cpuid_info.ext_cpuid1_ecx.bits.lzcnt_intel != 0)
@ -641,6 +670,13 @@ enum Extended_Family {
} }
} }
// Composite features.
if (supports_tscinv_bit() &&
((is_amd_family() && !is_amd_Barcelona()) ||
is_intel_tsc_synched_at_init())) {
result |= CPU_TSCINV;
}
return result; return result;
} }
@ -827,54 +863,54 @@ public:
// //
// Feature identification // Feature identification
// //
static bool supports_cpuid() { return _features != 0; } static bool supports_cpuid() { return _features != 0; }
static bool supports_cmpxchg8() { return (_features & CPU_CX8) != 0; } static bool supports_cmpxchg8() { return (_features & CPU_CX8) != 0; }
static bool supports_cmov() { return (_features & CPU_CMOV) != 0; } static bool supports_cmov() { return (_features & CPU_CMOV) != 0; }
static bool supports_fxsr() { return (_features & CPU_FXSR) != 0; } static bool supports_fxsr() { return (_features & CPU_FXSR) != 0; }
static bool supports_ht() { return (_features & CPU_HT) != 0; } static bool supports_ht() { return (_features & CPU_HT) != 0; }
static bool supports_mmx() { return (_features & CPU_MMX) != 0; } static bool supports_mmx() { return (_features & CPU_MMX) != 0; }
static bool supports_sse() { return (_features & CPU_SSE) != 0; } static bool supports_sse() { return (_features & CPU_SSE) != 0; }
static bool supports_sse2() { return (_features & CPU_SSE2) != 0; } static bool supports_sse2() { return (_features & CPU_SSE2) != 0; }
static bool supports_sse3() { return (_features & CPU_SSE3) != 0; } static bool supports_sse3() { return (_features & CPU_SSE3) != 0; }
static bool supports_ssse3() { return (_features & CPU_SSSE3)!= 0; } static bool supports_ssse3() { return (_features & CPU_SSSE3)!= 0; }
static bool supports_sse4_1() { return (_features & CPU_SSE4_1) != 0; } static bool supports_sse4_1() { return (_features & CPU_SSE4_1) != 0; }
static bool supports_sse4_2() { return (_features & CPU_SSE4_2) != 0; } static bool supports_sse4_2() { return (_features & CPU_SSE4_2) != 0; }
static bool supports_popcnt() { return (_features & CPU_POPCNT) != 0; } static bool supports_popcnt() { return (_features & CPU_POPCNT) != 0; }
static bool supports_avx() { return (_features & CPU_AVX) != 0; } static bool supports_avx() { return (_features & CPU_AVX) != 0; }
static bool supports_avx2() { return (_features & CPU_AVX2) != 0; } static bool supports_avx2() { return (_features & CPU_AVX2) != 0; }
static bool supports_tsc() { return (_features & CPU_TSC) != 0; } static bool supports_tsc() { return (_features & CPU_TSC) != 0; }
static bool supports_aes() { return (_features & CPU_AES) != 0; } static bool supports_aes() { return (_features & CPU_AES) != 0; }
static bool supports_erms() { return (_features & CPU_ERMS) != 0; } static bool supports_erms() { return (_features & CPU_ERMS) != 0; }
static bool supports_clmul() { return (_features & CPU_CLMUL) != 0; } static bool supports_clmul() { return (_features & CPU_CLMUL) != 0; }
static bool supports_rtm() { return (_features & CPU_RTM) != 0; } static bool supports_rtm() { return (_features & CPU_RTM) != 0; }
static bool supports_bmi1() { return (_features & CPU_BMI1) != 0; } static bool supports_bmi1() { return (_features & CPU_BMI1) != 0; }
static bool supports_bmi2() { return (_features & CPU_BMI2) != 0; } static bool supports_bmi2() { return (_features & CPU_BMI2) != 0; }
static bool supports_adx() { return (_features & CPU_ADX) != 0; } static bool supports_adx() { return (_features & CPU_ADX) != 0; }
static bool supports_evex() { return (_features & CPU_AVX512F) != 0; } static bool supports_evex() { return (_features & CPU_AVX512F) != 0; }
static bool supports_avx512dq() { return (_features & CPU_AVX512DQ) != 0; } static bool supports_avx512dq() { return (_features & CPU_AVX512DQ) != 0; }
static bool supports_avx512pf() { return (_features & CPU_AVX512PF) != 0; } static bool supports_avx512pf() { return (_features & CPU_AVX512PF) != 0; }
static bool supports_avx512er() { return (_features & CPU_AVX512ER) != 0; } static bool supports_avx512er() { return (_features & CPU_AVX512ER) != 0; }
static bool supports_avx512cd() { return (_features & CPU_AVX512CD) != 0; } static bool supports_avx512cd() { return (_features & CPU_AVX512CD) != 0; }
static bool supports_avx512bw() { return (_features & CPU_AVX512BW) != 0; } static bool supports_avx512bw() { return (_features & CPU_AVX512BW) != 0; }
static bool supports_avx512vl() { return (_features & CPU_AVX512VL) != 0; } static bool supports_avx512vl() { return (_features & CPU_AVX512VL) != 0; }
static bool supports_avx512vlbw() { return (supports_evex() && supports_avx512bw() && supports_avx512vl()); } static bool supports_avx512vlbw() { return (supports_evex() && supports_avx512bw() && supports_avx512vl()); }
static bool supports_avx512vldq() { return (supports_evex() && supports_avx512dq() && supports_avx512vl()); } static bool supports_avx512vldq() { return (supports_evex() && supports_avx512dq() && supports_avx512vl()); }
static bool supports_avx512vlbwdq() { return (supports_evex() && supports_avx512vl() && static bool supports_avx512vlbwdq() { return (supports_evex() && supports_avx512vl() &&
supports_avx512bw() && supports_avx512dq()); } supports_avx512bw() && supports_avx512dq()); }
static bool supports_avx512novl() { return (supports_evex() && !supports_avx512vl()); } static bool supports_avx512novl() { return (supports_evex() && !supports_avx512vl()); }
static bool supports_avx512nobw() { return (supports_evex() && !supports_avx512bw()); } static bool supports_avx512nobw() { return (supports_evex() && !supports_avx512bw()); }
static bool supports_avx256only() { return (supports_avx2() && !supports_evex()); } static bool supports_avx256only() { return (supports_avx2() && !supports_evex()); }
static bool supports_avxonly() { return ((supports_avx2() || supports_avx()) && !supports_evex()); } static bool supports_avxonly() { return ((supports_avx2() || supports_avx()) && !supports_evex()); }
static bool supports_sha() { return (_features & CPU_SHA) != 0; } static bool supports_sha() { return (_features & CPU_SHA) != 0; }
static bool supports_fma() { return (_features & CPU_FMA) != 0 && supports_avx(); } static bool supports_fma() { return (_features & CPU_FMA) != 0 && supports_avx(); }
static bool supports_vzeroupper() { return (_features & CPU_VZEROUPPER) != 0; } static bool supports_vzeroupper() { return (_features & CPU_VZEROUPPER) != 0; }
static bool supports_avx512_vpopcntdq() { return (_features & CPU_AVX512_VPOPCNTDQ) != 0; } static bool supports_avx512_vpopcntdq() { return (_features & CPU_AVX512_VPOPCNTDQ) != 0; }
static bool supports_avx512_vpclmulqdq() { return (_features & CPU_AVX512_VPCLMULQDQ) != 0; } static bool supports_avx512_vpclmulqdq() { return (_features & CPU_AVX512_VPCLMULQDQ) != 0; }
static bool supports_avx512_vaes() { return (_features & CPU_AVX512_VAES) != 0; } static bool supports_avx512_vaes() { return (_features & CPU_AVX512_VAES) != 0; }
static bool supports_avx512_vnni() { return (_features & CPU_AVX512_VNNI) != 0; } static bool supports_avx512_vnni() { return (_features & CPU_AVX512_VNNI) != 0; }
static bool supports_avx512_vbmi() { return (_features & CPU_AVX512_VBMI) != 0; } static bool supports_avx512_vbmi() { return (_features & CPU_AVX512_VBMI) != 0; }
static bool supports_avx512_vbmi2() { return (_features & CPU_AVX512_VBMI2) != 0; } static bool supports_avx512_vbmi2() { return (_features & CPU_AVX512_VBMI2) != 0; }
static bool supports_hv() { return (_features & CPU_HV_PRESENT) != 0; } static bool supports_hv() { return (_features & CPU_HV) != 0; }
// Intel features // Intel features
static bool is_intel_family_core() { return is_intel() && static bool is_intel_family_core() { return is_intel() &&
@ -917,12 +953,10 @@ public:
// Intel and AMD newer cores support fast timestamps well // Intel and AMD newer cores support fast timestamps well
static bool supports_tscinv_bit() { static bool supports_tscinv_bit() {
return (_features & CPU_TSCINV) != 0; return (_features & CPU_TSCINV_BIT) != 0;
} }
static bool supports_tscinv() { static bool supports_tscinv() {
return supports_tscinv_bit() && return (_features & CPU_TSCINV) != 0;
((is_amd_family() && !is_amd_Barcelona()) ||
is_intel_tsc_synched_at_init());
} }
// Intel Core and newer cpus have fast IDIV instruction (excluding Atom). // Intel Core and newer cpus have fast IDIV instruction (excluding Atom).
@ -998,10 +1032,10 @@ public:
// not synchronize with other memory ops. it needs a preceding // not synchronize with other memory ops. it needs a preceding
// and trailing StoreStore fence // and trailing StoreStore fence
// //
// clwb is an optional, intel-specific instruction optional which // clwb is an optional intel-specific instruction which
// writes back without evicting the line. it also does not // writes back without evicting the line. it also does not
// synchronize with other memory ops. so, it also needs a preceding // synchronize with other memory ops. so, it needs preceding
// and trailing StoreStore fence. // and trailing StoreStore fences.
#ifdef _LP64 #ifdef _LP64
static bool supports_clflush() { static bool supports_clflush() {
@ -1017,13 +1051,12 @@ public:
assert ((!Universe::is_fully_initialized() || (_features & CPU_FLUSH) != 0), "clflush should be available"); assert ((!Universe::is_fully_initialized() || (_features & CPU_FLUSH) != 0), "clflush should be available");
return true; return true;
} }
static bool supports_clflushopt() { return ((_features & CPU_FLUSHOPT) != 0); }
static bool supports_clwb() { return ((_features & CPU_CLWB) != 0); }
#else #else
static bool supports_clflush() { return ((_features & CPU_FLUSH) != 0); } static bool supports_clflush() { return ((_features & CPU_FLUSH) != 0); }
static bool supports_clflushopt() { return false; }
static bool supports_clwb() { return false; }
#endif // _LP64 #endif // _LP64
// Note: CPU_FLUSHOPT and CPU_CLWB bits should always be zero for 32-bit
static bool supports_clflushopt() { return ((_features & CPU_FLUSHOPT) != 0); }
static bool supports_clwb() { return ((_features & CPU_CLWB) != 0); }
// support functions for virtualization detection // support functions for virtualization detection
private: private:

View File

@ -49,7 +49,8 @@
static_field(CompilerToVM::Data, SharedRuntime_ic_miss_stub, address) \ static_field(CompilerToVM::Data, SharedRuntime_ic_miss_stub, address) \
static_field(CompilerToVM::Data, SharedRuntime_handle_wrong_method_stub, address) \ static_field(CompilerToVM::Data, SharedRuntime_handle_wrong_method_stub, address) \
static_field(CompilerToVM::Data, SharedRuntime_deopt_blob_unpack, address) \ static_field(CompilerToVM::Data, SharedRuntime_deopt_blob_unpack, address) \
static_field(CompilerToVM::Data, SharedRuntime_deopt_blob_unpack_with_exception_in_tls, address) \ static_field(CompilerToVM::Data, SharedRuntime_deopt_blob_unpack_with_exception_in_tls, \
address) \
static_field(CompilerToVM::Data, SharedRuntime_deopt_blob_uncommon_trap, address) \ static_field(CompilerToVM::Data, SharedRuntime_deopt_blob_uncommon_trap, address) \
\ \
static_field(CompilerToVM::Data, ThreadLocalAllocBuffer_alignment_reserve, size_t) \ static_field(CompilerToVM::Data, ThreadLocalAllocBuffer_alignment_reserve, size_t) \
@ -752,7 +753,9 @@
#define VM_INT_CONSTANTS_CPU(declare_constant, declare_preprocessor_constant, declare_c1_constant, declare_c2_constant, declare_c2_preprocessor_constant) \ #define VM_INT_CONSTANTS_CPU(declare_constant, declare_preprocessor_constant, declare_c1_constant, declare_c2_constant, declare_c2_preprocessor_constant) \
LP64_ONLY(declare_constant(frame::arg_reg_save_area_bytes)) \ LP64_ONLY(declare_constant(frame::arg_reg_save_area_bytes)) \
declare_constant(frame::interpreter_frame_sender_sp_offset) \ declare_constant(frame::interpreter_frame_sender_sp_offset) \
declare_constant(frame::interpreter_frame_last_sp_offset) \ declare_constant(frame::interpreter_frame_last_sp_offset)
#define VM_LONG_CONSTANTS_CPU(declare_constant, declare_preprocessor_constant, declare_c1_constant, declare_c2_constant, declare_c2_preprocessor_constant) \
declare_constant(VM_Version::CPU_CX8) \ declare_constant(VM_Version::CPU_CX8) \
declare_constant(VM_Version::CPU_CMOV) \ declare_constant(VM_Version::CPU_CMOV) \
declare_constant(VM_Version::CPU_FXSR) \ declare_constant(VM_Version::CPU_FXSR) \
@ -783,23 +786,22 @@
declare_constant(VM_Version::CPU_AVX512DQ) \ declare_constant(VM_Version::CPU_AVX512DQ) \
declare_constant(VM_Version::CPU_AVX512PF) \ declare_constant(VM_Version::CPU_AVX512PF) \
declare_constant(VM_Version::CPU_AVX512ER) \ declare_constant(VM_Version::CPU_AVX512ER) \
declare_constant(VM_Version::CPU_AVX512CD) declare_constant(VM_Version::CPU_AVX512CD) \
declare_constant(VM_Version::CPU_AVX512BW) \
#define VM_LONG_CONSTANTS_CPU(declare_constant, declare_preprocessor_constant, declare_c1_constant, declare_c2_constant, declare_c2_preprocessor_constant) \ declare_constant(VM_Version::CPU_AVX512VL) \
declare_preprocessor_constant("VM_Version::CPU_AVX512BW", CPU_AVX512BW) \ declare_constant(VM_Version::CPU_SHA) \
declare_preprocessor_constant("VM_Version::CPU_AVX512VL", CPU_AVX512VL) \ declare_constant(VM_Version::CPU_FMA) \
declare_preprocessor_constant("VM_Version::CPU_SHA", CPU_SHA) \ declare_constant(VM_Version::CPU_VZEROUPPER) \
declare_preprocessor_constant("VM_Version::CPU_FMA", CPU_FMA) \ declare_constant(VM_Version::CPU_AVX512_VPOPCNTDQ) \
declare_preprocessor_constant("VM_Version::CPU_VZEROUPPER", CPU_VZEROUPPER) \ declare_constant(VM_Version::CPU_AVX512_VPCLMULQDQ) \
declare_preprocessor_constant("VM_Version::CPU_AVX512_VPOPCNTDQ", CPU_AVX512_VPOPCNTDQ) \ declare_constant(VM_Version::CPU_AVX512_VAES) \
declare_preprocessor_constant("VM_Version::CPU_AVX512_VPCLMULQDQ", CPU_AVX512_VPCLMULQDQ) \ declare_constant(VM_Version::CPU_AVX512_VNNI) \
declare_preprocessor_constant("VM_Version::CPU_AVX512_VAES", CPU_AVX512_VAES) \ declare_constant(VM_Version::CPU_FLUSH) \
declare_preprocessor_constant("VM_Version::CPU_AVX512_VNNI", CPU_AVX512_VNNI) \ declare_constant(VM_Version::CPU_FLUSHOPT) \
declare_preprocessor_constant("VM_Version::CPU_FLUSH", CPU_FLUSH) \ declare_constant(VM_Version::CPU_CLWB) \
declare_preprocessor_constant("VM_Version::CPU_FLUSHOPT", CPU_FLUSHOPT) \ declare_constant(VM_Version::CPU_AVX512_VBMI2) \
declare_preprocessor_constant("VM_Version::CPU_CLWB", CPU_CLWB) \ declare_constant(VM_Version::CPU_AVX512_VBMI) \
declare_preprocessor_constant("VM_Version::CPU_AVX512_VBMI2", CPU_AVX512_VBMI2) \ declare_constant(VM_Version::CPU_HV)
declare_preprocessor_constant("VM_Version::CPU_AVX512_VBMI", CPU_AVX512_VBMI)
#endif #endif
@ -822,7 +824,6 @@
#define VM_ADDRESSES_OS(declare_address, declare_preprocessor_address, declare_function) #define VM_ADDRESSES_OS(declare_address, declare_preprocessor_address, declare_function)
#endif #endif
// //
// Instantiation of VMStructEntries, VMTypeEntries and VMIntConstantEntries // Instantiation of VMStructEntries, VMTypeEntries and VMIntConstantEntries
// //

View File

@ -291,6 +291,22 @@ unsigned int Abstract_VM_Version::jvm_version() {
(Abstract_VM_Version::vm_build_number() & 0xFF); (Abstract_VM_Version::vm_build_number() & 0xFF);
} }
void Abstract_VM_Version::insert_features_names(char* buf, size_t buflen, const char* features_names[]) {
uint64_t features = _features;
uint features_names_index = 0;
while (features != 0) {
if (features & 1) {
int res = jio_snprintf(buf, buflen, ", %s", features_names[features_names_index]);
assert(res > 0, "not enough temporary space allocated");
buf += res;
buflen -= res;
}
features >>= 1;
++features_names_index;
}
}
bool Abstract_VM_Version::print_matching_lines_from_file(const char* filename, outputStream* st, const char* keywords_to_match[]) { bool Abstract_VM_Version::print_matching_lines_from_file(const char* filename, outputStream* st, const char* keywords_to_match[]) {
char line[500]; char line[500];
FILE* fp = fopen(filename, "r"); FILE* fp = fopen(filename, "r");

View File

@ -119,13 +119,9 @@ class Abstract_VM_Version: AllStatic {
static const char* jdk_debug_level(); static const char* jdk_debug_level();
static const char* printable_jdk_debug_level(); static const char* printable_jdk_debug_level();
static uint64_t features() { static uint64_t features() { return _features; }
return _features; static const char* features_string() { return _features_string; }
} static void insert_features_names(char* buf, size_t buflen, const char* features_names[]);
static const char* features_string() {
return _features_string;
}
static VirtualizationType get_detected_virtualization() { static VirtualizationType get_detected_virtualization() {
return _detected_virtualization; return _detected_virtualization;