diff options
author | Tony Breeds <tony@bakeyournoodle.com> | 2007-06-22 02:54:30 -0400 |
---|---|---|
committer | Paul Mackerras <paulus@samba.org> | 2007-06-28 05:19:23 -0400 |
commit | 71712b455374a73af042fcfb5002fef5fd25ba44 (patch) | |
tree | 872c63855d0ca5c0dd2a8446cda4d6600415cc74 | |
parent | a5c631b174e23cab773cf422c1f39b28e7224602 (diff) |
[POWERPC] Move iSeries_tb_recal into its own late_initcall.
Currently iSeries will recalibrate the cputime_factors in the first
settimeofday() call.
It seems the reason for doing this is to ensure a resaonable time delta after
time_init(). On current kernels (with udev), this call is made 40-60 seconds
into the boot process, by moving it to a late initcall it is called
approximately 5 seconds after time_init() is called. This is sufficient to
recalibrate the timebase.
Signed-off-by: Tony Breeds <tony@bakeyournoodle.com>
CC: Stephen Rothwell <sfr@canb.auug.org.au>
Signed-off-by: Paul Mackerras <paulus@samba.org>
-rw-r--r-- | arch/powerpc/kernel/time.c | 30 | ||||
-rw-r--r-- | arch/powerpc/platforms/iseries/setup.c | 6 | ||||
-rw-r--r-- | include/asm-powerpc/time.h | 2 |
3 files changed, 23 insertions, 15 deletions
diff --git a/arch/powerpc/kernel/time.c b/arch/powerpc/kernel/time.c index 43c687a1d76e..66d2db7495aa 100644 --- a/arch/powerpc/kernel/time.c +++ b/arch/powerpc/kernel/time.c | |||
@@ -77,9 +77,8 @@ | |||
77 | /* keep track of when we need to update the rtc */ | 77 | /* keep track of when we need to update the rtc */ |
78 | time_t last_rtc_update; | 78 | time_t last_rtc_update; |
79 | #ifdef CONFIG_PPC_ISERIES | 79 | #ifdef CONFIG_PPC_ISERIES |
80 | unsigned long iSeries_recal_titan = 0; | 80 | static unsigned long __initdata iSeries_recal_titan; |
81 | unsigned long iSeries_recal_tb = 0; | 81 | static signed long __initdata iSeries_recal_tb; |
82 | static unsigned long first_settimeofday = 1; | ||
83 | #endif | 82 | #endif |
84 | 83 | ||
85 | /* The decrementer counts down by 128 every 128ns on a 601. */ | 84 | /* The decrementer counts down by 128 every 128ns on a 601. */ |
@@ -556,10 +555,15 @@ EXPORT_SYMBOL(profile_pc); | |||
556 | * returned by the service processor for the timebase frequency. | 555 | * returned by the service processor for the timebase frequency. |
557 | */ | 556 | */ |
558 | 557 | ||
559 | static void iSeries_tb_recal(void) | 558 | static int __init iSeries_tb_recal(void) |
560 | { | 559 | { |
561 | struct div_result divres; | 560 | struct div_result divres; |
562 | unsigned long titan, tb; | 561 | unsigned long titan, tb; |
562 | |||
563 | /* Make sure we only run on iSeries */ | ||
564 | if (!firmware_has_feature(FW_FEATURE_ISERIES)) | ||
565 | return -ENODEV; | ||
566 | |||
563 | tb = get_tb(); | 567 | tb = get_tb(); |
564 | titan = HvCallXm_loadTod(); | 568 | titan = HvCallXm_loadTod(); |
565 | if ( iSeries_recal_titan ) { | 569 | if ( iSeries_recal_titan ) { |
@@ -600,8 +604,18 @@ static void iSeries_tb_recal(void) | |||
600 | } | 604 | } |
601 | iSeries_recal_titan = titan; | 605 | iSeries_recal_titan = titan; |
602 | iSeries_recal_tb = tb; | 606 | iSeries_recal_tb = tb; |
607 | |||
608 | return 0; | ||
603 | } | 609 | } |
604 | #endif | 610 | late_initcall(iSeries_tb_recal); |
611 | |||
612 | /* Called from platform early init */ | ||
613 | void __init iSeries_time_init_early(void) | ||
614 | { | ||
615 | iSeries_recal_tb = get_tb(); | ||
616 | iSeries_recal_titan = HvCallXm_loadTod(); | ||
617 | } | ||
618 | #endif /* CONFIG_PPC_ISERIES */ | ||
605 | 619 | ||
606 | /* | 620 | /* |
607 | * For iSeries shared processors, we have to let the hypervisor | 621 | * For iSeries shared processors, we have to let the hypervisor |
@@ -765,12 +779,6 @@ int do_settimeofday(struct timespec *tv) | |||
765 | * to the RTC again, or write to the RTC but then they don't call | 779 | * to the RTC again, or write to the RTC but then they don't call |
766 | * settimeofday to perform this operation. | 780 | * settimeofday to perform this operation. |
767 | */ | 781 | */ |
768 | #ifdef CONFIG_PPC_ISERIES | ||
769 | if (firmware_has_feature(FW_FEATURE_ISERIES) && first_settimeofday) { | ||
770 | iSeries_tb_recal(); | ||
771 | first_settimeofday = 0; | ||
772 | } | ||
773 | #endif | ||
774 | 782 | ||
775 | /* Make userspace gettimeofday spin until we're done. */ | 783 | /* Make userspace gettimeofday spin until we're done. */ |
776 | ++vdso_data->tb_update_count; | 784 | ++vdso_data->tb_update_count; |
diff --git a/arch/powerpc/platforms/iseries/setup.c b/arch/powerpc/platforms/iseries/setup.c index 7f5dcee814d4..13a8b1908ded 100644 --- a/arch/powerpc/platforms/iseries/setup.c +++ b/arch/powerpc/platforms/iseries/setup.c | |||
@@ -79,8 +79,6 @@ extern void iSeries_pci_final_fixup(void); | |||
79 | static void iSeries_pci_final_fixup(void) { } | 79 | static void iSeries_pci_final_fixup(void) { } |
80 | #endif | 80 | #endif |
81 | 81 | ||
82 | extern unsigned long iSeries_recal_tb; | ||
83 | extern unsigned long iSeries_recal_titan; | ||
84 | 82 | ||
85 | struct MemoryBlock { | 83 | struct MemoryBlock { |
86 | unsigned long absStart; | 84 | unsigned long absStart; |
@@ -292,8 +290,8 @@ static void __init iSeries_init_early(void) | |||
292 | { | 290 | { |
293 | DBG(" -> iSeries_init_early()\n"); | 291 | DBG(" -> iSeries_init_early()\n"); |
294 | 292 | ||
295 | iSeries_recal_tb = get_tb(); | 293 | /* Snapshot the timebase, for use in later recalibration */ |
296 | iSeries_recal_titan = HvCallXm_loadTod(); | 294 | iSeries_time_init_early(); |
297 | 295 | ||
298 | /* | 296 | /* |
299 | * Initialize the DMA/TCE management | 297 | * Initialize the DMA/TCE management |
diff --git a/include/asm-powerpc/time.h b/include/asm-powerpc/time.h index 2d00e13c981a..d7f5ddfbaac7 100644 --- a/include/asm-powerpc/time.h +++ b/include/asm-powerpc/time.h | |||
@@ -240,5 +240,7 @@ extern void snapshot_timebases(void); | |||
240 | #define snapshot_timebases() do { } while (0) | 240 | #define snapshot_timebases() do { } while (0) |
241 | #endif | 241 | #endif |
242 | 242 | ||
243 | extern void iSeries_time_init_early(void); | ||
244 | |||
243 | #endif /* __KERNEL__ */ | 245 | #endif /* __KERNEL__ */ |
244 | #endif /* __POWERPC_TIME_H */ | 246 | #endif /* __POWERPC_TIME_H */ |