aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86
diff options
context:
space:
mode:
authorAndreas Herrmann <andreas.herrmann3@amd.com>2010-03-19 07:09:22 -0400
committerGreg Kroah-Hartman <gregkh@suse.de>2010-04-01 19:01:54 -0400
commitb89725179b5fc5ea585930420c7df568de0a97be (patch)
tree153f64f908780ef295b24c06bc697c19103b0ce9 /arch/x86
parent9fcb3fd97bccf736945cccdd74005d64be9602e1 (diff)
x86, amd: Restrict usage of c1e_idle()
commit 035a02c1e1de31888e8b6adac0ff667971ac04db upstream. Currently c1e_idle returns true for all CPUs greater than or equal to family 0xf model 0x40. This covers too many CPUs. Meanwhile a respective erratum for the underlying problem was filed (#400). This patch adds the logic to check whether erratum #400 applies to a given CPU. Especially for CPUs where SMI/HW triggered C1e is not supported, c1e_idle() doesn't need to be used. We can check this by looking at the respective OSVW bit for erratum #400. Signed-off-by: Andreas Herrmann <andreas.herrmann3@amd.com> LKML-Reference: <20100319110922.GA19614@alberich.amd.com> Signed-off-by: H. Peter Anvin <hpa@zytor.com> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
Diffstat (limited to 'arch/x86')
-rw-r--r--arch/x86/include/asm/msr-index.h2
-rw-r--r--arch/x86/kernel/process.c32
2 files changed, 26 insertions, 8 deletions
diff --git a/arch/x86/include/asm/msr-index.h b/arch/x86/include/asm/msr-index.h
index 1cd58cdbc03f..4604e6a54d36 100644
--- a/arch/x86/include/asm/msr-index.h
+++ b/arch/x86/include/asm/msr-index.h
@@ -105,6 +105,8 @@
105#define MSR_AMD64_PATCH_LEVEL 0x0000008b 105#define MSR_AMD64_PATCH_LEVEL 0x0000008b
106#define MSR_AMD64_NB_CFG 0xc001001f 106#define MSR_AMD64_NB_CFG 0xc001001f
107#define MSR_AMD64_PATCH_LOADER 0xc0010020 107#define MSR_AMD64_PATCH_LOADER 0xc0010020
108#define MSR_AMD64_OSVW_ID_LENGTH 0xc0010140
109#define MSR_AMD64_OSVW_STATUS 0xc0010141
108#define MSR_AMD64_IBSFETCHCTL 0xc0011030 110#define MSR_AMD64_IBSFETCHCTL 0xc0011030
109#define MSR_AMD64_IBSFETCHLINAD 0xc0011031 111#define MSR_AMD64_IBSFETCHLINAD 0xc0011031
110#define MSR_AMD64_IBSFETCHPHYSAD 0xc0011032 112#define MSR_AMD64_IBSFETCHPHYSAD 0xc0011032
diff --git a/arch/x86/kernel/process.c b/arch/x86/kernel/process.c
index c9b3522b6b46..999c8a6a4e7a 100644
--- a/arch/x86/kernel/process.c
+++ b/arch/x86/kernel/process.c
@@ -519,21 +519,37 @@ static int __cpuinit mwait_usable(const struct cpuinfo_x86 *c)
519} 519}
520 520
521/* 521/*
522 * Check for AMD CPUs, which have potentially C1E support 522 * Check for AMD CPUs, where APIC timer interrupt does not wake up CPU from C1e.
523 * For more information see
524 * - Erratum #400 for NPT family 0xf and family 0x10 CPUs
525 * - Erratum #365 for family 0x11 (not affected because C1e not in use)
523 */ 526 */
524static int __cpuinit check_c1e_idle(const struct cpuinfo_x86 *c) 527static int __cpuinit check_c1e_idle(const struct cpuinfo_x86 *c)
525{ 528{
529 u64 val;
526 if (c->x86_vendor != X86_VENDOR_AMD) 530 if (c->x86_vendor != X86_VENDOR_AMD)
527 return 0; 531 goto no_c1e_idle;
528
529 if (c->x86 < 0x0F)
530 return 0;
531 532
532 /* Family 0x0f models < rev F do not have C1E */ 533 /* Family 0x0f models < rev F do not have C1E */
533 if (c->x86 == 0x0f && c->x86_model < 0x40) 534 if (c->x86 == 0x0F && c->x86_model >= 0x40)
534 return 0; 535 return 1;
535 536
536 return 1; 537 if (c->x86 == 0x10) {
538 /*
539 * check OSVW bit for CPUs that are not affected
540 * by erratum #400
541 */
542 rdmsrl(MSR_AMD64_OSVW_ID_LENGTH, val);
543 if (val >= 2) {
544 rdmsrl(MSR_AMD64_OSVW_STATUS, val);
545 if (!(val & BIT(1)))
546 goto no_c1e_idle;
547 }
548 return 1;
549 }
550
551no_c1e_idle:
552 return 0;
537} 553}
538 554
539static cpumask_var_t c1e_mask; 555static cpumask_var_t c1e_mask;