diff options
author | Oren Twaig <oren@scalemp.com> | 2014-06-29 06:01:08 -0400 |
---|---|---|
committer | H. Peter Anvin <hpa@zytor.com> | 2014-07-13 20:48:03 -0400 |
commit | 411cf9ee2946492c0ac7eca48422fcf94a723ce5 (patch) | |
tree | a151a728aaf0f631405f5a47ea534b0bffd65cf4 | |
parent | b81975eade8c6495f3c4d6746d22bdc95f617777 (diff) |
x86, vsmp: Remove is_vsmp_box() from apic_is_clustered_box()
When a vSMP Foundation box is detected, the function apic_cluster_num() counts
the number of APIC clusters found. If more than one found, a multi board
configuration is assumed, and TSC marked as unstable. This behavior is
incorrect as vSMP Foundation may use processors from single node only, attached
to memory of other nodes - and such node may have more than one APIC cluster
(typically any recent intel box has more than single APIC_CLUSTERID(x)).
To fix this, we simply remove the code which detects a vSMP Foundation box and
affects apic_is_clusted_box() return value. This can be done because later the
kernel checks by itself if the TSC is stable using the
check_tsc_sync_[source|target]() functions and marks TSC as unstable if needed.
Acked-by: Shai Fultheim <shai@scalemp.com>
Signed-off-by: Oren Twaig <oren@scalemp.com>
Link: http://lkml.kernel.org/r/1404036068-11674-1-git-send-email-oren@scalemp.com
Signed-off-by: H. Peter Anvin <hpa@zytor.com>
-rw-r--r-- | arch/x86/include/asm/apic.h | 8 | ||||
-rw-r--r-- | arch/x86/kernel/apic/apic.c | 60 |
2 files changed, 1 insertions, 67 deletions
diff --git a/arch/x86/include/asm/apic.h b/arch/x86/include/asm/apic.h index 69ed79aa9085..b40ea7e4bae9 100644 --- a/arch/x86/include/asm/apic.h +++ b/arch/x86/include/asm/apic.h | |||
@@ -85,14 +85,6 @@ static inline bool apic_from_smp_config(void) | |||
85 | #include <asm/paravirt.h> | 85 | #include <asm/paravirt.h> |
86 | #endif | 86 | #endif |
87 | 87 | ||
88 | #ifdef CONFIG_X86_64 | ||
89 | extern int is_vsmp_box(void); | ||
90 | #else | ||
91 | static inline int is_vsmp_box(void) | ||
92 | { | ||
93 | return 0; | ||
94 | } | ||
95 | #endif | ||
96 | extern int setup_profiling_timer(unsigned int); | 88 | extern int setup_profiling_timer(unsigned int); |
97 | 89 | ||
98 | static inline void native_apic_mem_write(u32 reg, u32 v) | 90 | static inline void native_apic_mem_write(u32 reg, u32 v) |
diff --git a/arch/x86/kernel/apic/apic.c b/arch/x86/kernel/apic/apic.c index ca1bd75e3de2..6b35d308688c 100644 --- a/arch/x86/kernel/apic/apic.c +++ b/arch/x86/kernel/apic/apic.c | |||
@@ -2451,51 +2451,6 @@ static void apic_pm_activate(void) { } | |||
2451 | 2451 | ||
2452 | #ifdef CONFIG_X86_64 | 2452 | #ifdef CONFIG_X86_64 |
2453 | 2453 | ||
2454 | static int apic_cluster_num(void) | ||
2455 | { | ||
2456 | int i, clusters, zeros; | ||
2457 | unsigned id; | ||
2458 | u16 *bios_cpu_apicid; | ||
2459 | DECLARE_BITMAP(clustermap, NUM_APIC_CLUSTERS); | ||
2460 | |||
2461 | bios_cpu_apicid = early_per_cpu_ptr(x86_bios_cpu_apicid); | ||
2462 | bitmap_zero(clustermap, NUM_APIC_CLUSTERS); | ||
2463 | |||
2464 | for (i = 0; i < nr_cpu_ids; i++) { | ||
2465 | /* are we being called early in kernel startup? */ | ||
2466 | if (bios_cpu_apicid) { | ||
2467 | id = bios_cpu_apicid[i]; | ||
2468 | } else if (i < nr_cpu_ids) { | ||
2469 | if (cpu_present(i)) | ||
2470 | id = per_cpu(x86_bios_cpu_apicid, i); | ||
2471 | else | ||
2472 | continue; | ||
2473 | } else | ||
2474 | break; | ||
2475 | |||
2476 | if (id != BAD_APICID) | ||
2477 | __set_bit(APIC_CLUSTERID(id), clustermap); | ||
2478 | } | ||
2479 | |||
2480 | /* Problem: Partially populated chassis may not have CPUs in some of | ||
2481 | * the APIC clusters they have been allocated. Only present CPUs have | ||
2482 | * x86_bios_cpu_apicid entries, thus causing zeroes in the bitmap. | ||
2483 | * Since clusters are allocated sequentially, count zeros only if | ||
2484 | * they are bounded by ones. | ||
2485 | */ | ||
2486 | clusters = 0; | ||
2487 | zeros = 0; | ||
2488 | for (i = 0; i < NUM_APIC_CLUSTERS; i++) { | ||
2489 | if (test_bit(i, clustermap)) { | ||
2490 | clusters += 1 + zeros; | ||
2491 | zeros = 0; | ||
2492 | } else | ||
2493 | ++zeros; | ||
2494 | } | ||
2495 | |||
2496 | return clusters; | ||
2497 | } | ||
2498 | |||
2499 | static int multi_checked; | 2454 | static int multi_checked; |
2500 | static int multi; | 2455 | static int multi; |
2501 | 2456 | ||
@@ -2540,20 +2495,7 @@ static void dmi_check_multi(void) | |||
2540 | int apic_is_clustered_box(void) | 2495 | int apic_is_clustered_box(void) |
2541 | { | 2496 | { |
2542 | dmi_check_multi(); | 2497 | dmi_check_multi(); |
2543 | if (multi) | 2498 | return multi; |
2544 | return 1; | ||
2545 | |||
2546 | if (!is_vsmp_box()) | ||
2547 | return 0; | ||
2548 | |||
2549 | /* | ||
2550 | * ScaleMP vSMPowered boxes have one cluster per board and TSCs are | ||
2551 | * not guaranteed to be synced between boards | ||
2552 | */ | ||
2553 | if (apic_cluster_num() > 1) | ||
2554 | return 1; | ||
2555 | |||
2556 | return 0; | ||
2557 | } | 2499 | } |
2558 | #endif | 2500 | #endif |
2559 | 2501 | ||