aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTony Breeds <tony@bakeyournoodle.com>2007-06-22 02:54:30 -0400
committerPaul Mackerras <paulus@samba.org>2007-06-28 05:19:23 -0400
commit71712b455374a73af042fcfb5002fef5fd25ba44 (patch)
tree872c63855d0ca5c0dd2a8446cda4d6600415cc74
parenta5c631b174e23cab773cf422c1f39b28e7224602 (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.c30
-rw-r--r--arch/powerpc/platforms/iseries/setup.c6
-rw-r--r--include/asm-powerpc/time.h2
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 */
78time_t last_rtc_update; 78time_t last_rtc_update;
79#ifdef CONFIG_PPC_ISERIES 79#ifdef CONFIG_PPC_ISERIES
80unsigned long iSeries_recal_titan = 0; 80static unsigned long __initdata iSeries_recal_titan;
81unsigned long iSeries_recal_tb = 0; 81static signed long __initdata iSeries_recal_tb;
82static 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
559static void iSeries_tb_recal(void) 558static 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 610late_initcall(iSeries_tb_recal);
611
612/* Called from platform early init */
613void __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);
79static void iSeries_pci_final_fixup(void) { } 79static void iSeries_pci_final_fixup(void) { }
80#endif 80#endif
81 81
82extern unsigned long iSeries_recal_tb;
83extern unsigned long iSeries_recal_titan;
84 82
85struct MemoryBlock { 83struct 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
243extern void iSeries_time_init_early(void);
244
243#endif /* __KERNEL__ */ 245#endif /* __KERNEL__ */
244#endif /* __POWERPC_TIME_H */ 246#endif /* __POWERPC_TIME_H */