diff options
author | Rusty Russell <rusty@rustcorp.com.au> | 2008-03-04 07:07:50 -0500 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-04-17 11:40:52 -0400 |
commit | 3c2047cd32b1a8c782d7efab72707e7daa251625 (patch) | |
tree | 79a3481d5f7b637626d0cf346dc1f1042d68c07f | |
parent | 9fc34113f6880b215cbea4e7017fc818700384c2 (diff) |
x86: if we cannot calibrate the TSC, we panic.
The current tsc_init() clears the TSC feature bit if the TSC khz
cannot be calculated, causing us to panic in
arch/x86/kernel/cpu/bugs.c check_config(). We should simply mark it
unstable.
Frankly, someone should take an axe to this code. mark_tsc_unstable()
not only marks it unstable, but sets tsc_enabled to 0, which seems
redundant but is actually important here because means it won't be
used by sched_clock() either. Perhaps a tristate enum "UNUSABLE,
UNSTABLE, OK" would be clearer, and separate mark_tsc_unstable() and
mark_tsc_broken() functions?
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
-rw-r--r-- | arch/x86/kernel/tsc_32.c | 13 |
1 files changed, 5 insertions, 8 deletions
diff --git a/arch/x86/kernel/tsc_32.c b/arch/x86/kernel/tsc_32.c index c2241e04ea5f..68657d8526fb 100644 --- a/arch/x86/kernel/tsc_32.c +++ b/arch/x86/kernel/tsc_32.c | |||
@@ -392,13 +392,15 @@ void __init tsc_init(void) | |||
392 | int cpu; | 392 | int cpu; |
393 | 393 | ||
394 | if (!cpu_has_tsc) | 394 | if (!cpu_has_tsc) |
395 | goto out_no_tsc; | 395 | return; |
396 | 396 | ||
397 | cpu_khz = calculate_cpu_khz(); | 397 | cpu_khz = calculate_cpu_khz(); |
398 | tsc_khz = cpu_khz; | 398 | tsc_khz = cpu_khz; |
399 | 399 | ||
400 | if (!cpu_khz) | 400 | if (!cpu_khz) { |
401 | goto out_no_tsc; | 401 | mark_tsc_unstable("could not calculate TSC khz"); |
402 | return; | ||
403 | } | ||
402 | 404 | ||
403 | printk("Detected %lu.%03lu MHz processor.\n", | 405 | printk("Detected %lu.%03lu MHz processor.\n", |
404 | (unsigned long)cpu_khz / 1000, | 406 | (unsigned long)cpu_khz / 1000, |
@@ -431,9 +433,4 @@ void __init tsc_init(void) | |||
431 | tsc_enabled = 1; | 433 | tsc_enabled = 1; |
432 | 434 | ||
433 | clocksource_register(&clocksource_tsc); | 435 | clocksource_register(&clocksource_tsc); |
434 | |||
435 | return; | ||
436 | |||
437 | out_no_tsc: | ||
438 | setup_clear_cpu_cap(X86_FEATURE_TSC); | ||
439 | } | 436 | } |