2010-06-18 06:36:31 +02:00
|
|
|
# HG changeset patch
|
|
|
|
# User Keir Fraser <keir.fraser@citrix.com>
|
|
|
|
# Date 1275643111 -3600
|
|
|
|
# Node ID 48e2b07cf01c044bf483bd7fa5408a6f9801416b
|
|
|
|
# Parent f2b1924f20281bc42fa3532c7d82b3ee0700aff4
|
|
|
|
Intel: Add CPUID feature mask support for NHM processors.
|
|
|
|
|
|
|
|
Signed-off-by: Jun Nakajima <jun.nakajima@intel.com>
|
|
|
|
Signed-off-by: Liping Ke <liping.ke@intel.com>
|
|
|
|
|
2010-08-19 16:34:50 +02:00
|
|
|
# HG changeset patch
|
|
|
|
# User Keir Fraser <keir.fraser@citrix.com>
|
|
|
|
# Date 1276604335 -3600
|
|
|
|
# Node ID 2501732e291b001711a0dc1c474bb89ce77f3110
|
|
|
|
# Parent a2cc1db1af9c8f9b148c80f8b2c3f64bde7542f9
|
|
|
|
x86: fix pv cpuid masking
|
|
|
|
|
|
|
|
Invert initial values of the variables parsed into from the command
|
|
|
|
line, so that completely clearing out one or more of the four bit
|
|
|
|
fields is possible.
|
|
|
|
|
|
|
|
Further, consolidate the command line parameter specifications into
|
|
|
|
a single place.
|
|
|
|
|
|
|
|
Finally, as per "Intel Virtualization Technology FlexMigration
|
|
|
|
Application Note" (http://www.intel.com/Assets/PDF/manual/323850.pdf),
|
|
|
|
also handle family 6 model 0x1f.
|
|
|
|
|
|
|
|
What remains open is the question whether pv_cpuid() shouldn't also
|
|
|
|
consume these masks.
|
|
|
|
|
|
|
|
Signed-off-by: Jan Beulich <jbeulich@novell.com>
|
|
|
|
|
2010-09-22 16:40:08 +02:00
|
|
|
Index: xen-4.0.1-testing/xen/arch/x86/cpu/amd.c
|
|
|
|
===================================================================
|
|
|
|
--- xen-4.0.1-testing.orig/xen/arch/x86/cpu/amd.c
|
|
|
|
+++ xen-4.0.1-testing/xen/arch/x86/cpu/amd.c
|
2010-08-19 16:34:50 +02:00
|
|
|
@@ -33,14 +33,6 @@ void start_svm(struct cpuinfo_x86 *c);
|
|
|
|
static char opt_famrev[14];
|
|
|
|
string_param("cpuid_mask_cpu", opt_famrev);
|
|
|
|
|
|
|
|
-/* Finer-grained CPUID feature control. */
|
|
|
|
-static unsigned int opt_cpuid_mask_ecx, opt_cpuid_mask_edx;
|
|
|
|
-integer_param("cpuid_mask_ecx", opt_cpuid_mask_ecx);
|
|
|
|
-integer_param("cpuid_mask_edx", opt_cpuid_mask_edx);
|
|
|
|
-static unsigned int opt_cpuid_mask_ext_ecx, opt_cpuid_mask_ext_edx;
|
|
|
|
-integer_param("cpuid_mask_ext_ecx", opt_cpuid_mask_ext_ecx);
|
|
|
|
-integer_param("cpuid_mask_ext_edx", opt_cpuid_mask_ext_edx);
|
|
|
|
-
|
|
|
|
static inline void wrmsr_amd(unsigned int index, unsigned int lo,
|
|
|
|
unsigned int hi)
|
|
|
|
{
|
|
|
|
@@ -61,7 +53,7 @@ static inline void wrmsr_amd(unsigned in
|
|
|
|
*
|
|
|
|
* The processor revision string parameter has precedene.
|
|
|
|
*/
|
|
|
|
-static void __devinit set_cpuidmask(struct cpuinfo_x86 *c)
|
|
|
|
+static void __devinit set_cpuidmask(const struct cpuinfo_x86 *c)
|
|
|
|
{
|
|
|
|
static unsigned int feat_ecx, feat_edx;
|
|
|
|
static unsigned int extfeat_ecx, extfeat_edx;
|
|
|
|
@@ -76,12 +68,12 @@ static void __devinit set_cpuidmask(stru
|
|
|
|
ASSERT((status == not_parsed) && (smp_processor_id() == 0));
|
|
|
|
status = no_mask;
|
|
|
|
|
|
|
|
- if (opt_cpuid_mask_ecx | opt_cpuid_mask_edx |
|
|
|
|
- opt_cpuid_mask_ext_ecx | opt_cpuid_mask_ext_edx) {
|
|
|
|
- feat_ecx = opt_cpuid_mask_ecx ? : ~0U;
|
|
|
|
- feat_edx = opt_cpuid_mask_edx ? : ~0U;
|
|
|
|
- extfeat_ecx = opt_cpuid_mask_ext_ecx ? : ~0U;
|
|
|
|
- extfeat_edx = opt_cpuid_mask_ext_edx ? : ~0U;
|
|
|
|
+ if (~(opt_cpuid_mask_ecx & opt_cpuid_mask_edx &
|
|
|
|
+ opt_cpuid_mask_ext_ecx & opt_cpuid_mask_ext_edx)) {
|
|
|
|
+ feat_ecx = opt_cpuid_mask_ecx;
|
|
|
|
+ feat_edx = opt_cpuid_mask_edx;
|
|
|
|
+ extfeat_ecx = opt_cpuid_mask_ext_ecx;
|
|
|
|
+ extfeat_edx = opt_cpuid_mask_ext_edx;
|
|
|
|
} else if (*opt_famrev == '\0') {
|
|
|
|
return;
|
|
|
|
} else if (!strcmp(opt_famrev, "fam_0f_rev_c")) {
|
2010-09-22 16:40:08 +02:00
|
|
|
Index: xen-4.0.1-testing/xen/arch/x86/cpu/common.c
|
|
|
|
===================================================================
|
|
|
|
--- xen-4.0.1-testing.orig/xen/arch/x86/cpu/common.c
|
|
|
|
+++ xen-4.0.1-testing/xen/arch/x86/cpu/common.c
|
2010-08-19 16:34:50 +02:00
|
|
|
@@ -22,6 +22,15 @@ static int cachesize_override __cpuinitd
|
|
|
|
static int disable_x86_fxsr __cpuinitdata;
|
|
|
|
static int disable_x86_serial_nr __cpuinitdata;
|
|
|
|
|
|
|
|
+unsigned int __devinitdata opt_cpuid_mask_ecx = ~0u;
|
|
|
|
+integer_param("cpuid_mask_ecx", opt_cpuid_mask_ecx);
|
|
|
|
+unsigned int __devinitdata opt_cpuid_mask_edx = ~0u;
|
|
|
|
+integer_param("cpuid_mask_edx", opt_cpuid_mask_edx);
|
|
|
|
+unsigned int __devinitdata opt_cpuid_mask_ext_ecx = ~0u;
|
2010-06-18 06:36:31 +02:00
|
|
|
+integer_param("cpuid_mask_ext_ecx", opt_cpuid_mask_ext_ecx);
|
2010-08-19 16:34:50 +02:00
|
|
|
+unsigned int __devinitdata opt_cpuid_mask_ext_edx = ~0u;
|
2010-06-18 06:36:31 +02:00
|
|
|
+integer_param("cpuid_mask_ext_edx", opt_cpuid_mask_ext_edx);
|
2010-08-19 16:34:50 +02:00
|
|
|
+
|
|
|
|
struct cpu_dev * cpu_devs[X86_VENDOR_NUM] = {};
|
|
|
|
|
|
|
|
/*
|
2010-09-22 16:40:08 +02:00
|
|
|
Index: xen-4.0.1-testing/xen/arch/x86/cpu/cpu.h
|
|
|
|
===================================================================
|
|
|
|
--- xen-4.0.1-testing.orig/xen/arch/x86/cpu/cpu.h
|
|
|
|
+++ xen-4.0.1-testing/xen/arch/x86/cpu/cpu.h
|
2010-08-19 16:34:50 +02:00
|
|
|
@@ -21,6 +21,9 @@ struct cpu_dev {
|
|
|
|
|
|
|
|
extern struct cpu_dev * cpu_devs [X86_VENDOR_NUM];
|
|
|
|
|
|
|
|
+extern unsigned int opt_cpuid_mask_ecx, opt_cpuid_mask_edx;
|
|
|
|
+extern unsigned int opt_cpuid_mask_ext_ecx, opt_cpuid_mask_ext_edx;
|
|
|
|
+
|
|
|
|
extern int get_model_name(struct cpuinfo_x86 *c);
|
|
|
|
extern void display_cacheinfo(struct cpuinfo_x86 *c);
|
|
|
|
|
2010-09-22 16:40:08 +02:00
|
|
|
Index: xen-4.0.1-testing/xen/arch/x86/cpu/intel.c
|
|
|
|
===================================================================
|
|
|
|
--- xen-4.0.1-testing.orig/xen/arch/x86/cpu/intel.c
|
|
|
|
+++ xen-4.0.1-testing/xen/arch/x86/cpu/intel.c
|
2010-08-19 16:34:50 +02:00
|
|
|
@@ -20,16 +20,6 @@
|
|
|
|
|
|
|
|
extern int trap_init_f00f_bug(void);
|
2010-06-18 06:36:31 +02:00
|
|
|
|
2010-08-19 16:34:50 +02:00
|
|
|
-/*
|
|
|
|
- * opt_cpuid_mask_ecx/edx: cpuid.1[ecx, edx] feature mask.
|
|
|
|
- * For example, E8400[Intel Core 2 Duo Processor series] ecx = 0x0008E3FD,
|
|
|
|
- * edx = 0xBFEBFBFF when executing CPUID.EAX = 1 normally. If you want to
|
|
|
|
- * 'rev down' to E8400, you can set these values in these Xen boot parameters.
|
|
|
|
- */
|
|
|
|
-static unsigned int opt_cpuid_mask_ecx, opt_cpuid_mask_edx;
|
|
|
|
-integer_param("cpuid_mask_ecx", opt_cpuid_mask_ecx);
|
|
|
|
-integer_param("cpuid_mask_edx", opt_cpuid_mask_edx);
|
|
|
|
-
|
2010-06-18 06:36:31 +02:00
|
|
|
static int use_xsave = 1;
|
|
|
|
boolean_param("xsave", use_xsave);
|
2010-08-19 16:34:50 +02:00
|
|
|
|
|
|
|
@@ -40,24 +30,57 @@ boolean_param("xsave", use_xsave);
|
2010-06-18 06:36:31 +02:00
|
|
|
struct movsl_mask movsl_mask __read_mostly;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
-static void __devinit set_cpuidmask(void)
|
2010-08-19 16:34:50 +02:00
|
|
|
+/*
|
|
|
|
+ * opt_cpuid_mask_ecx/edx: cpuid.1[ecx, edx] feature mask.
|
|
|
|
+ * For example, E8400[Intel Core 2 Duo Processor series] ecx = 0x0008E3FD,
|
|
|
|
+ * edx = 0xBFEBFBFF when executing CPUID.EAX = 1 normally. If you want to
|
|
|
|
+ * 'rev down' to E8400, you can set these values in these Xen boot parameters.
|
|
|
|
+ */
|
|
|
|
+static void __devinit set_cpuidmask(const struct cpuinfo_x86 *c)
|
2010-06-18 06:36:31 +02:00
|
|
|
{
|
|
|
|
- unsigned int eax, ebx, ecx, edx, model;
|
2010-08-19 16:34:50 +02:00
|
|
|
+ const char *extra = "";
|
2010-06-18 06:36:31 +02:00
|
|
|
|
|
|
|
- if (!(opt_cpuid_mask_ecx | opt_cpuid_mask_edx))
|
2010-08-19 16:34:50 +02:00
|
|
|
+ if (!~(opt_cpuid_mask_ecx & opt_cpuid_mask_edx &
|
|
|
|
+ opt_cpuid_mask_ext_ecx & opt_cpuid_mask_ext_edx))
|
2010-06-18 06:36:31 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
- cpuid(0x00000001, &eax, &ebx, &ecx, &edx);
|
|
|
|
- model = ((eax & 0xf0000) >> 12) | ((eax & 0xf0) >> 4);
|
|
|
|
- if (!((model == 0x1d) || ((model == 0x17) && ((eax & 0xf) >= 4)))) {
|
2010-08-19 16:34:50 +02:00
|
|
|
- printk(XENLOG_ERR "Cannot set CPU feature mask on CPU#%d\n",
|
|
|
|
- smp_processor_id());
|
|
|
|
+ /* Only family 6 supports this feature */
|
|
|
|
+ switch ((c->x86 == 6) * c->x86_model) {
|
|
|
|
+ case 0x17:
|
|
|
|
+ if ((c->x86_mask & 0x0f) < 4)
|
|
|
|
+ break;
|
|
|
|
+ /* fall through */
|
|
|
|
+ case 0x1d:
|
|
|
|
+ wrmsr(MSR_INTEL_CPUID_FEATURE_MASK,
|
|
|
|
+ opt_cpuid_mask_ecx,
|
|
|
|
+ opt_cpuid_mask_edx);
|
|
|
|
+ if (!~(opt_cpuid_mask_ext_ecx & opt_cpuid_mask_ext_edx))
|
|
|
|
+ return;
|
|
|
|
+ extra = "extended ";
|
|
|
|
+ break;
|
|
|
|
+/*
|
2010-06-18 06:36:31 +02:00
|
|
|
+ * CPU supports this feature if the processor signature meets the following:
|
|
|
|
+ * (CPUID.(EAX=01h):EAX) > 000106A2h, or
|
|
|
|
+ * (CPUID.(EAX=01h):EAX) == 000106Exh, 0002065xh, 000206Cxh, 000206Exh, or 000206Fxh
|
|
|
|
+ *
|
|
|
|
+ */
|
2010-08-19 16:34:50 +02:00
|
|
|
+ case 0x1a:
|
|
|
|
+ if ((c->x86_mask & 0x0f) <= 2)
|
|
|
|
+ break;
|
|
|
|
+ /* fall through */
|
|
|
|
+ case 0x1e: case 0x1f:
|
|
|
|
+ case 0x25: case 0x2c: case 0x2e: case 0x2f:
|
|
|
|
+ wrmsr(MSR_INTEL_CPUID1_FEATURE_MASK,
|
|
|
|
+ opt_cpuid_mask_ecx,
|
|
|
|
+ opt_cpuid_mask_edx);
|
|
|
|
+ wrmsr(MSR_INTEL_CPUID80000001_FEATURE_MASK,
|
|
|
|
+ opt_cpuid_mask_ext_ecx,
|
|
|
|
+ opt_cpuid_mask_ext_edx);
|
2010-06-18 06:36:31 +02:00
|
|
|
return;
|
|
|
|
}
|
2010-08-19 16:34:50 +02:00
|
|
|
|
2010-06-18 06:36:31 +02:00
|
|
|
- wrmsr(MSR_IA32_CPUID_FEATURE_MASK1,
|
|
|
|
- opt_cpuid_mask_ecx ? : ~0u,
|
|
|
|
- opt_cpuid_mask_edx ? : ~0u);
|
2010-08-19 16:34:50 +02:00
|
|
|
+ printk(XENLOG_ERR "Cannot set CPU feature mask on CPU#%d\n",
|
|
|
|
+ smp_processor_id());
|
2010-06-18 06:36:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void __devinit early_intel_workaround(struct cpuinfo_x86 *c)
|
2010-09-22 16:40:08 +02:00
|
|
|
@@ -179,7 +202,7 @@ static void __devinit init_intel(struct
|
2010-06-18 06:36:31 +02:00
|
|
|
|
|
|
|
detect_ht(c);
|
|
|
|
|
|
|
|
- set_cpuidmask();
|
|
|
|
+ set_cpuidmask(c);
|
|
|
|
|
|
|
|
/* Work around errata */
|
|
|
|
Intel_errata_workarounds(c);
|
2010-09-22 16:40:08 +02:00
|
|
|
Index: xen-4.0.1-testing/xen/include/asm-x86/msr-index.h
|
|
|
|
===================================================================
|
|
|
|
--- xen-4.0.1-testing.orig/xen/include/asm-x86/msr-index.h
|
|
|
|
+++ xen-4.0.1-testing/xen/include/asm-x86/msr-index.h
|
2010-08-19 16:34:50 +02:00
|
|
|
@@ -156,8 +156,10 @@
|
|
|
|
#define MSR_P6_EVNTSEL0 0x00000186
|
|
|
|
#define MSR_P6_EVNTSEL1 0x00000187
|
2010-06-18 06:36:31 +02:00
|
|
|
|
2010-08-19 16:34:50 +02:00
|
|
|
-/* MSR for cpuid feature mask */
|
|
|
|
-#define MSR_IA32_CPUID_FEATURE_MASK1 0x00000478
|
|
|
|
+/* MSRs for Intel cpuid feature mask */
|
|
|
|
+#define MSR_INTEL_CPUID_FEATURE_MASK 0x00000478
|
|
|
|
+#define MSR_INTEL_CPUID1_FEATURE_MASK 0x00000130
|
|
|
|
+#define MSR_INTEL_CPUID80000001_FEATURE_MASK 0x00000131
|
2010-06-18 06:36:31 +02:00
|
|
|
|
|
|
|
/* MSRs & bits used for VMX enabling */
|
|
|
|
#define MSR_IA32_VMX_BASIC 0x480
|