2009-03-14 00:53:18 +01:00
|
|
|
/* Initialize CPU feature data.
|
|
|
|
This file is part of the GNU C Library.
|
2016-01-04 17:05:18 +01:00
|
|
|
Copyright (C) 2008-2016 Free Software Foundation, Inc.
|
2009-03-14 00:53:18 +01:00
|
|
|
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU Lesser General Public
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
The GNU C Library is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
Lesser General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
2012-02-10 00:18:22 +01:00
|
|
|
License along with the GNU C Library; if not, see
|
|
|
|
<http://www.gnu.org/licenses/>. */
|
2009-03-14 00:53:18 +01:00
|
|
|
|
2009-07-31 20:53:35 +02:00
|
|
|
#include <cpuid.h>
|
2015-08-13 12:37:47 +02:00
|
|
|
#include <cpu-features.h>
|
2009-03-14 00:53:18 +01:00
|
|
|
|
2015-08-13 12:37:47 +02:00
|
|
|
static inline void
|
|
|
|
get_common_indeces (struct cpu_features *cpu_features,
|
2015-11-30 17:53:37 +01:00
|
|
|
unsigned int *family, unsigned int *model,
|
|
|
|
unsigned int *extended_model)
|
2009-06-30 13:39:09 +02:00
|
|
|
{
|
2015-08-13 12:37:47 +02:00
|
|
|
unsigned int eax;
|
|
|
|
__cpuid (1, eax, cpu_features->cpuid[COMMON_CPUID_INDEX_1].ebx,
|
|
|
|
cpu_features->cpuid[COMMON_CPUID_INDEX_1].ecx,
|
|
|
|
cpu_features->cpuid[COMMON_CPUID_INDEX_1].edx);
|
|
|
|
GLRO(dl_x86_cpu_features).cpuid[COMMON_CPUID_INDEX_1].eax = eax;
|
2010-04-04 09:25:46 +02:00
|
|
|
*family = (eax >> 8) & 0x0f;
|
|
|
|
*model = (eax >> 4) & 0x0f;
|
2015-11-30 17:53:37 +01:00
|
|
|
*extended_model = (eax >> 12) & 0xf0;
|
|
|
|
if (*family == 0x0f)
|
|
|
|
{
|
|
|
|
*family += (eax >> 20) & 0xff;
|
|
|
|
*model += *extended_model;
|
|
|
|
}
|
2009-06-30 13:39:09 +02:00
|
|
|
}
|
|
|
|
|
2015-08-13 12:37:47 +02:00
|
|
|
static inline void
|
|
|
|
init_cpu_features (struct cpu_features *cpu_features)
|
2009-03-14 00:53:18 +01:00
|
|
|
{
|
2015-08-13 12:37:47 +02:00
|
|
|
unsigned int ebx, ecx, edx;
|
2010-04-04 09:25:46 +02:00
|
|
|
unsigned int family = 0;
|
|
|
|
unsigned int model = 0;
|
|
|
|
enum cpu_features_kind kind;
|
2009-03-14 00:53:18 +01:00
|
|
|
|
2015-08-18 16:59:49 +02:00
|
|
|
#if !HAS_CPUID
|
2015-08-13 13:52:50 +02:00
|
|
|
if (__get_cpuid_max (0, 0) == 0)
|
|
|
|
{
|
|
|
|
kind = arch_kind_other;
|
|
|
|
goto no_cpuid;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-08-13 12:37:47 +02:00
|
|
|
__cpuid (0, cpu_features->max_cpuid, ebx, ecx, edx);
|
2009-03-14 00:53:18 +01:00
|
|
|
|
|
|
|
/* This spells out "GenuineIntel". */
|
|
|
|
if (ebx == 0x756e6547 && ecx == 0x6c65746e && edx == 0x49656e69)
|
|
|
|
{
|
2015-11-30 17:53:37 +01:00
|
|
|
unsigned int extended_model;
|
|
|
|
|
2010-04-04 09:25:46 +02:00
|
|
|
kind = arch_kind_intel;
|
2009-03-14 00:53:18 +01:00
|
|
|
|
2015-11-30 17:53:37 +01:00
|
|
|
get_common_indeces (cpu_features, &family, &model, &extended_model);
|
2009-06-30 13:39:09 +02:00
|
|
|
|
2015-11-30 17:53:37 +01:00
|
|
|
if (family == 0x06)
|
2010-01-12 20:22:03 +01:00
|
|
|
{
|
2015-08-13 12:37:47 +02:00
|
|
|
ecx = cpu_features->cpuid[COMMON_CPUID_INDEX_1].ecx;
|
2010-04-04 09:25:46 +02:00
|
|
|
model += extended_model;
|
2010-05-27 20:14:18 +02:00
|
|
|
switch (model)
|
2010-01-12 20:22:03 +01:00
|
|
|
{
|
2010-08-25 19:07:37 +02:00
|
|
|
case 0x1c:
|
|
|
|
case 0x26:
|
|
|
|
/* BSF is slow on Atom. */
|
2015-08-13 12:37:47 +02:00
|
|
|
cpu_features->feature[index_Slow_BSF] |= bit_Slow_BSF;
|
2010-08-25 19:07:37 +02:00
|
|
|
break;
|
|
|
|
|
2015-12-15 20:46:54 +01:00
|
|
|
case 0x57:
|
|
|
|
/* Knights Landing. Enable Silvermont optimizations. */
|
2015-12-19 00:47:28 +01:00
|
|
|
cpu_features->feature[index_Prefer_No_VZEROUPPER]
|
|
|
|
|= bit_Prefer_No_VZEROUPPER;
|
2015-12-15 20:46:54 +01:00
|
|
|
|
2013-06-14 20:46:15 +02:00
|
|
|
case 0x37:
|
2015-01-24 02:27:09 +01:00
|
|
|
case 0x4a:
|
|
|
|
case 0x4d:
|
2015-01-24 03:52:45 +01:00
|
|
|
case 0x5a:
|
|
|
|
case 0x5d:
|
2013-06-14 20:46:15 +02:00
|
|
|
/* Unaligned load versions are faster than SSSE3
|
|
|
|
on Silvermont. */
|
2013-06-29 00:28:50 +02:00
|
|
|
#if index_Fast_Unaligned_Load != index_Prefer_PMINUB_for_stringop
|
|
|
|
# error index_Fast_Unaligned_Load != index_Prefer_PMINUB_for_stringop
|
|
|
|
#endif
|
|
|
|
#if index_Fast_Unaligned_Load != index_Slow_SSE4_2
|
|
|
|
# error index_Fast_Unaligned_Load != index_Slow_SSE4_2
|
|
|
|
#endif
|
2015-08-13 12:37:47 +02:00
|
|
|
cpu_features->feature[index_Fast_Unaligned_Load]
|
2013-06-29 00:28:50 +02:00
|
|
|
|= (bit_Fast_Unaligned_Load
|
|
|
|
| bit_Prefer_PMINUB_for_stringop
|
|
|
|
| bit_Slow_SSE4_2);
|
2013-06-14 20:46:15 +02:00
|
|
|
break;
|
|
|
|
|
2011-06-03 13:01:25 +02:00
|
|
|
default:
|
|
|
|
/* Unknown family 0x06 processors. Assuming this is one
|
2011-10-21 04:43:15 +02:00
|
|
|
of Core i3/i5/i7 processors if AVX is available. */
|
2011-06-03 13:01:25 +02:00
|
|
|
if ((ecx & bit_AVX) == 0)
|
|
|
|
break;
|
|
|
|
|
2010-01-12 20:22:03 +01:00
|
|
|
case 0x1a:
|
|
|
|
case 0x1e:
|
|
|
|
case 0x1f:
|
|
|
|
case 0x25:
|
2010-11-12 09:48:52 +01:00
|
|
|
case 0x2c:
|
2010-01-12 20:22:03 +01:00
|
|
|
case 0x2e:
|
|
|
|
case 0x2f:
|
2011-07-19 23:11:54 +02:00
|
|
|
/* Rep string instructions, copy backward, unaligned loads
|
|
|
|
and pminub are fast on Intel Core i3, i5 and i7. */
|
2010-06-30 17:26:11 +02:00
|
|
|
#if index_Fast_Rep_String != index_Fast_Copy_Backward
|
|
|
|
# error index_Fast_Rep_String != index_Fast_Copy_Backward
|
2011-06-24 20:15:32 +02:00
|
|
|
#endif
|
|
|
|
#if index_Fast_Rep_String != index_Fast_Unaligned_Load
|
|
|
|
# error index_Fast_Rep_String != index_Fast_Unaligned_Load
|
2011-07-19 23:11:54 +02:00
|
|
|
#endif
|
|
|
|
#if index_Fast_Rep_String != index_Prefer_PMINUB_for_stringop
|
|
|
|
# error index_Fast_Rep_String != index_Prefer_PMINUB_for_stringop
|
2010-06-30 17:26:11 +02:00
|
|
|
#endif
|
2015-08-13 12:37:47 +02:00
|
|
|
cpu_features->feature[index_Fast_Rep_String]
|
2011-06-24 20:15:32 +02:00
|
|
|
|= (bit_Fast_Rep_String
|
|
|
|
| bit_Fast_Copy_Backward
|
2011-07-19 23:11:54 +02:00
|
|
|
| bit_Fast_Unaligned_Load
|
|
|
|
| bit_Prefer_PMINUB_for_stringop);
|
2010-01-12 20:22:03 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-03-14 00:53:18 +01:00
|
|
|
}
|
|
|
|
/* This spells out "AuthenticAMD". */
|
|
|
|
else if (ebx == 0x68747541 && ecx == 0x444d4163 && edx == 0x69746e65)
|
|
|
|
{
|
2015-11-30 17:53:37 +01:00
|
|
|
unsigned int extended_model;
|
|
|
|
|
2010-04-04 09:25:46 +02:00
|
|
|
kind = arch_kind_amd;
|
2009-03-14 00:53:18 +01:00
|
|
|
|
2015-11-30 17:53:37 +01:00
|
|
|
get_common_indeces (cpu_features, &family, &model, &extended_model);
|
2011-03-05 05:30:08 +01:00
|
|
|
|
2015-08-13 12:37:47 +02:00
|
|
|
ecx = cpu_features->cpuid[COMMON_CPUID_INDEX_1].ecx;
|
2011-03-05 05:30:08 +01:00
|
|
|
|
2011-10-22 02:47:20 +02:00
|
|
|
unsigned int eax;
|
2011-10-21 04:43:15 +02:00
|
|
|
__cpuid (0x80000000, eax, ebx, ecx, edx);
|
|
|
|
if (eax >= 0x80000001)
|
|
|
|
__cpuid (0x80000001,
|
2015-08-13 12:37:47 +02:00
|
|
|
cpu_features->cpuid[COMMON_CPUID_INDEX_80000001].eax,
|
|
|
|
cpu_features->cpuid[COMMON_CPUID_INDEX_80000001].ebx,
|
|
|
|
cpu_features->cpuid[COMMON_CPUID_INDEX_80000001].ecx,
|
|
|
|
cpu_features->cpuid[COMMON_CPUID_INDEX_80000001].edx);
|
2016-01-14 15:36:02 +01:00
|
|
|
|
|
|
|
if (family == 0x15)
|
|
|
|
{
|
|
|
|
/* "Excavator" */
|
|
|
|
if (model >= 0x60 && model <= 0x7f)
|
|
|
|
cpu_features->feature[index_Fast_Unaligned_Load]
|
|
|
|
|= bit_Fast_Unaligned_Load;
|
|
|
|
}
|
2009-03-14 00:53:18 +01:00
|
|
|
}
|
|
|
|
else
|
2010-04-04 09:25:46 +02:00
|
|
|
kind = arch_kind_other;
|
|
|
|
|
2015-08-27 18:06:26 +02:00
|
|
|
/* Support i586 if CX8 is available. */
|
|
|
|
if (HAS_CPU_FEATURE (CX8))
|
|
|
|
cpu_features->feature[index_I586] |= bit_I586;
|
|
|
|
|
|
|
|
/* Support i686 if CMOV is available. */
|
|
|
|
if (HAS_CPU_FEATURE (CMOV))
|
|
|
|
cpu_features->feature[index_I686] |= bit_I686;
|
|
|
|
|
2015-08-13 12:37:47 +02:00
|
|
|
if (cpu_features->max_cpuid >= 7)
|
2013-01-03 18:38:20 +01:00
|
|
|
__cpuid_count (7, 0,
|
2015-08-13 12:37:47 +02:00
|
|
|
cpu_features->cpuid[COMMON_CPUID_INDEX_7].eax,
|
|
|
|
cpu_features->cpuid[COMMON_CPUID_INDEX_7].ebx,
|
|
|
|
cpu_features->cpuid[COMMON_CPUID_INDEX_7].ecx,
|
|
|
|
cpu_features->cpuid[COMMON_CPUID_INDEX_7].edx);
|
2013-01-03 18:38:20 +01:00
|
|
|
|
2012-05-17 05:14:24 +02:00
|
|
|
/* Can we call xgetbv? */
|
2015-08-13 12:37:47 +02:00
|
|
|
if (HAS_CPU_FEATURE (OSXSAVE))
|
2012-01-26 13:45:14 +01:00
|
|
|
{
|
2012-05-17 05:14:24 +02:00
|
|
|
unsigned int xcrlow;
|
|
|
|
unsigned int xcrhigh;
|
|
|
|
asm ("xgetbv" : "=a" (xcrlow), "=d" (xcrhigh) : "c" (0));
|
|
|
|
/* Is YMM and XMM state usable? */
|
|
|
|
if ((xcrlow & (bit_YMM_state | bit_XMM_state)) ==
|
|
|
|
(bit_YMM_state | bit_XMM_state))
|
|
|
|
{
|
|
|
|
/* Determine if AVX is usable. */
|
2015-08-13 12:37:47 +02:00
|
|
|
if (HAS_CPU_FEATURE (AVX))
|
|
|
|
cpu_features->feature[index_AVX_Usable] |= bit_AVX_Usable;
|
2015-01-30 15:50:20 +01:00
|
|
|
#if index_AVX2_Usable != index_AVX_Fast_Unaligned_Load
|
|
|
|
# error index_AVX2_Usable != index_AVX_Fast_Unaligned_Load
|
|
|
|
#endif
|
|
|
|
/* Determine if AVX2 is usable. Unaligned load with 256-bit
|
|
|
|
AVX registers are faster on processors with AVX2. */
|
2015-08-13 12:37:47 +02:00
|
|
|
if (HAS_CPU_FEATURE (AVX2))
|
|
|
|
cpu_features->feature[index_AVX2_Usable]
|
2015-01-30 15:50:20 +01:00
|
|
|
|= bit_AVX2_Usable | bit_AVX_Fast_Unaligned_Load;
|
2015-06-08 13:07:59 +02:00
|
|
|
/* Check if OPMASK state, upper 256-bit of ZMM0-ZMM15 and
|
|
|
|
ZMM16-ZMM31 state are enabled. */
|
|
|
|
if ((xcrlow & (bit_Opmask_state | bit_ZMM0_15_state
|
|
|
|
| bit_ZMM16_31_state)) ==
|
|
|
|
(bit_Opmask_state | bit_ZMM0_15_state | bit_ZMM16_31_state))
|
|
|
|
{
|
|
|
|
/* Determine if AVX512F is usable. */
|
2015-08-13 12:37:47 +02:00
|
|
|
if (HAS_CPU_FEATURE (AVX512F))
|
2015-06-08 13:07:59 +02:00
|
|
|
{
|
2015-08-13 12:37:47 +02:00
|
|
|
cpu_features->feature[index_AVX512F_Usable]
|
2015-06-08 13:07:59 +02:00
|
|
|
|= bit_AVX512F_Usable;
|
|
|
|
/* Determine if AVX512DQ is usable. */
|
2015-08-13 12:37:47 +02:00
|
|
|
if (HAS_CPU_FEATURE (AVX512DQ))
|
|
|
|
cpu_features->feature[index_AVX512DQ_Usable]
|
2015-06-08 13:07:59 +02:00
|
|
|
|= bit_AVX512DQ_Usable;
|
|
|
|
}
|
|
|
|
}
|
2012-10-02 14:05:17 +02:00
|
|
|
/* Determine if FMA is usable. */
|
2015-08-13 12:37:47 +02:00
|
|
|
if (HAS_CPU_FEATURE (FMA))
|
|
|
|
cpu_features->feature[index_FMA_Usable] |= bit_FMA_Usable;
|
2012-05-17 05:14:24 +02:00
|
|
|
/* Determine if FMA4 is usable. */
|
2015-08-13 12:37:47 +02:00
|
|
|
if (HAS_CPU_FEATURE (FMA4))
|
|
|
|
cpu_features->feature[index_FMA4_Usable] |= bit_FMA4_Usable;
|
2012-05-17 05:14:24 +02:00
|
|
|
}
|
2012-01-26 13:45:14 +01:00
|
|
|
}
|
|
|
|
|
2015-08-18 16:59:49 +02:00
|
|
|
#if !HAS_CPUID
|
2015-08-13 13:52:50 +02:00
|
|
|
no_cpuid:
|
|
|
|
#endif
|
|
|
|
|
2015-08-13 12:37:47 +02:00
|
|
|
cpu_features->family = family;
|
|
|
|
cpu_features->model = model;
|
|
|
|
cpu_features->kind = kind;
|
2009-07-30 00:22:28 +02:00
|
|
|
}
|