2006-06-26 00:25:05 -07:00
/*
* linux / kernel / time / clocksource . c
*
* This file contains the functions which manage clocksource drivers .
*
* Copyright ( C ) 2004 , 2005 IBM , John Stultz ( johnstul @ us . ibm . com )
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*
* TODO WishList :
* o Allow clocksource drivers to be unregistered
*/
# include <linux/clocksource.h>
# include <linux/sysdev.h>
# include <linux/init.h>
# include <linux/module.h>
[PATCH] kernel/time/clocksource.c needs struct task_struct on m68k
kernel/time/clocksource.c needs struct task_struct on m68k.
Because it uses spin_unlock_irq(), which, on m68k, uses hardirq_count(), which
uses preempt_count(), which needs to dereference struct task_struct, we
have to include sched.h. Because it would cause a loop inclusion, we
cannot include sched.h in any other of asm-m68k/system.h,
linux/thread_info.h, linux/hardirq.h, which leaves this ugly include in
a C file as the only simple solution.
Signed-off-by: Mathieu Desnoyers <mathieu.desnoyers@polymtl.ca>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Roman Zippel <zippel@linux-m68k.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-02-10 01:43:43 -08:00
# include <linux/sched.h> /* for spin_unlock_irq() using preempt_count() m68k */
2007-02-16 01:28:03 -08:00
# include <linux/tick.h>
2009-08-18 17:09:42 +02:00
# include <linux/kthread.h>
2006-06-26 00:25:05 -07:00
2009-02-12 05:03:34 +00:00
void timecounter_init ( struct timecounter * tc ,
const struct cyclecounter * cc ,
u64 start_tstamp )
{
tc - > cc = cc ;
tc - > cycle_last = cc - > read ( cc ) ;
tc - > nsec = start_tstamp ;
}
2009-11-11 19:06:30 -08:00
EXPORT_SYMBOL_GPL ( timecounter_init ) ;
2009-02-12 05:03:34 +00:00
/**
* timecounter_read_delta - get nanoseconds since last call of this function
* @ tc : Pointer to time counter
*
* When the underlying cycle counter runs over , this will be handled
* correctly as long as it does not run over more than once between
* calls .
*
* The first call to this function for a new time counter initializes
* the time tracking and returns an undefined result .
*/
static u64 timecounter_read_delta ( struct timecounter * tc )
{
cycle_t cycle_now , cycle_delta ;
u64 ns_offset ;
/* read cycle counter: */
cycle_now = tc - > cc - > read ( tc - > cc ) ;
/* calculate the delta since the last timecounter_read_delta(): */
cycle_delta = ( cycle_now - tc - > cycle_last ) & tc - > cc - > mask ;
/* convert to nanoseconds: */
ns_offset = cyclecounter_cyc2ns ( tc - > cc , cycle_delta ) ;
/* update time stamp of timecounter_read_delta() call: */
tc - > cycle_last = cycle_now ;
return ns_offset ;
}
u64 timecounter_read ( struct timecounter * tc )
{
u64 nsec ;
/* increment time by nanoseconds since last call */
nsec = timecounter_read_delta ( tc ) ;
nsec + = tc - > nsec ;
tc - > nsec = nsec ;
return nsec ;
}
2009-11-11 19:06:30 -08:00
EXPORT_SYMBOL_GPL ( timecounter_read ) ;
2009-02-12 05:03:34 +00:00
u64 timecounter_cyc2time ( struct timecounter * tc ,
cycle_t cycle_tstamp )
{
u64 cycle_delta = ( cycle_tstamp - tc - > cycle_last ) & tc - > cc - > mask ;
u64 nsec ;
/*
* Instead of always treating cycle_tstamp as more recent
* than tc - > cycle_last , detect when it is too far in the
* future and treat it as old time stamp instead .
*/
if ( cycle_delta > tc - > cc - > mask / 2 ) {
cycle_delta = ( tc - > cycle_last - cycle_tstamp ) & tc - > cc - > mask ;
nsec = tc - > nsec - cyclecounter_cyc2ns ( tc - > cc , cycle_delta ) ;
} else {
nsec = cyclecounter_cyc2ns ( tc - > cc , cycle_delta ) + tc - > nsec ;
}
return nsec ;
}
2009-11-11 19:06:30 -08:00
EXPORT_SYMBOL_GPL ( timecounter_cyc2time ) ;
2009-02-12 05:03:34 +00:00
2009-11-11 14:05:29 +00:00
/**
* clocks_calc_mult_shift - calculate mult / shift factors for scaled math of clocks
* @ mult : pointer to mult variable
* @ shift : pointer to shift variable
* @ from : frequency to convert from
* @ to : frequency to convert to
* @ minsec : guaranteed runtime conversion range in seconds
*
* The function evaluates the shift / mult pair for the scaled math
* operations of clocksources and clockevents .
*
* @ to and @ from are frequency values in HZ . For clock sources @ to is
* NSEC_PER_SEC = = 1 GHz and @ from is the counter frequency . For clock
* event @ to is the counter frequency and @ from is NSEC_PER_SEC .
*
* The @ minsec conversion range argument controls the time frame in
* seconds which must be covered by the runtime conversion with the
* calculated mult and shift factors . This guarantees that no 64 bit
* overflow happens when the input value of the conversion is
* multiplied with the calculated mult factor . Larger ranges may
* reduce the conversion accuracy by chosing smaller mult and shift
* factors .
*/
void
clocks_calc_mult_shift ( u32 * mult , u32 * shift , u32 from , u32 to , u32 minsec )
{
u64 tmp ;
u32 sft , sftacc = 32 ;
/*
* Calculate the shift factor which is limiting the conversion
* range :
*/
tmp = ( ( u64 ) minsec * from ) > > 32 ;
while ( tmp ) {
tmp > > = 1 ;
sftacc - - ;
}
/*
* Find the conversion shift / mult pair which has the best
* accuracy and fits the maxsec conversion range :
*/
for ( sft = 32 ; sft > 0 ; sft - - ) {
tmp = ( u64 ) to < < sft ;
do_div ( tmp , from ) ;
if ( ( tmp > > sftacc ) = = 0 )
break ;
}
* mult = tmp ;
* shift = sft ;
}
2006-06-26 00:25:05 -07:00
/*[Clocksource internal variables]---------
* curr_clocksource :
2009-08-14 15:47:21 +02:00
* currently selected clocksource .
2006-06-26 00:25:05 -07:00
* clocksource_list :
* linked list with the registered clocksources
2009-08-14 15:47:30 +02:00
* clocksource_mutex :
* protects manipulations to curr_clocksource and the clocksource_list
2006-06-26 00:25:05 -07:00
* override_name :
* Name of the user - specified clocksource .
*/
2009-08-14 15:47:21 +02:00
static struct clocksource * curr_clocksource ;
2006-06-26 00:25:05 -07:00
static LIST_HEAD ( clocksource_list ) ;
2009-08-14 15:47:30 +02:00
static DEFINE_MUTEX ( clocksource_mutex ) ;
2006-06-26 00:25:05 -07:00
static char override_name [ 32 ] ;
2009-09-14 19:49:02 +02:00
static int finished_booting ;
2006-06-26 00:25:05 -07:00
2007-02-16 01:27:43 -08:00
# ifdef CONFIG_CLOCKSOURCE_WATCHDOG
2009-09-11 15:33:05 +02:00
static void clocksource_watchdog_work ( struct work_struct * work ) ;
2007-02-16 01:27:43 -08:00
static LIST_HEAD ( watchdog_list ) ;
static struct clocksource * watchdog ;
static struct timer_list watchdog_timer ;
2009-09-11 15:33:05 +02:00
static DECLARE_WORK ( watchdog_work , clocksource_watchdog_work ) ;
2007-02-16 01:27:43 -08:00
static DEFINE_SPINLOCK ( watchdog_lock ) ;
static cycle_t watchdog_last ;
2009-08-14 15:47:24 +02:00
static int watchdog_running ;
2007-05-09 02:35:15 -07:00
2009-08-18 17:09:42 +02:00
static int clocksource_watchdog_kthread ( void * data ) ;
2009-08-19 11:26:09 +02:00
static void __clocksource_change_rating ( struct clocksource * cs , int rating ) ;
2009-08-14 15:47:25 +02:00
2007-02-16 01:27:43 -08:00
/*
2007-05-09 02:33:40 -07:00
* Interval : 0.5 sec Threshold : 0.0625 s
2007-02-16 01:27:43 -08:00
*/
# define WATCHDOG_INTERVAL (HZ >> 1)
2007-05-09 02:33:40 -07:00
# define WATCHDOG_THRESHOLD (NSEC_PER_SEC >> 4)
2007-02-16 01:27:43 -08:00
2009-08-18 17:09:42 +02:00
static void clocksource_watchdog_work ( struct work_struct * work )
{
/*
* If kthread_run fails the next watchdog scan over the
* watchdog_list will find the unstable clock again .
*/
kthread_run ( clocksource_watchdog_kthread , NULL , " kwatchdog " ) ;
}
2009-08-28 20:25:24 +02:00
static void __clocksource_unstable ( struct clocksource * cs )
2007-02-16 01:27:43 -08:00
{
cs - > flags & = ~ ( CLOCK_SOURCE_VALID_FOR_HRES | CLOCK_SOURCE_WATCHDOG ) ;
2009-08-14 15:47:25 +02:00
cs - > flags | = CLOCK_SOURCE_UNSTABLE ;
2009-09-14 19:49:02 +02:00
if ( finished_booting )
schedule_work ( & watchdog_work ) ;
2007-02-16 01:27:43 -08:00
}
2009-08-28 20:25:24 +02:00
static void clocksource_unstable ( struct clocksource * cs , int64_t delta )
{
printk ( KERN_WARNING " Clocksource %s unstable (delta = %Ld ns) \n " ,
cs - > name , delta ) ;
__clocksource_unstable ( cs ) ;
}
/**
* clocksource_mark_unstable - mark clocksource unstable via watchdog
* @ cs : clocksource to be marked unstable
*
* This function is called instead of clocksource_change_rating from
* cpu hotplug code to avoid a deadlock between the clocksource mutex
* and the cpu hotplug mutex . It defers the update of the clocksource
* to the watchdog thread .
*/
void clocksource_mark_unstable ( struct clocksource * cs )
{
unsigned long flags ;
spin_lock_irqsave ( & watchdog_lock , flags ) ;
if ( ! ( cs - > flags & CLOCK_SOURCE_UNSTABLE ) ) {
if ( list_empty ( & cs - > wd_list ) )
list_add ( & cs - > wd_list , & watchdog_list ) ;
__clocksource_unstable ( cs ) ;
}
spin_unlock_irqrestore ( & watchdog_lock , flags ) ;
}
2007-02-16 01:27:43 -08:00
static void clocksource_watchdog ( unsigned long data )
{
2009-08-14 15:47:25 +02:00
struct clocksource * cs ;
2007-02-16 01:27:43 -08:00
cycle_t csnow , wdnow ;
int64_t wd_nsec , cs_nsec ;
2009-08-14 15:47:25 +02:00
int next_cpu ;
2007-02-16 01:27:43 -08:00
spin_lock ( & watchdog_lock ) ;
2009-08-14 15:47:24 +02:00
if ( ! watchdog_running )
goto out ;
2007-02-16 01:27:43 -08:00
2009-04-21 12:24:00 -07:00
wdnow = watchdog - > read ( watchdog ) ;
2009-08-14 15:47:26 +02:00
wd_nsec = clocksource_cyc2ns ( ( wdnow - watchdog_last ) & watchdog - > mask ,
watchdog - > mult , watchdog - > shift ) ;
2007-02-16 01:27:43 -08:00
watchdog_last = wdnow ;
2009-08-14 15:47:25 +02:00
list_for_each_entry ( cs , & watchdog_list , wd_list ) {
/* Clocksource already marked unstable? */
2009-08-18 17:09:42 +02:00
if ( cs - > flags & CLOCK_SOURCE_UNSTABLE ) {
2009-09-14 19:49:02 +02:00
if ( finished_booting )
schedule_work ( & watchdog_work ) ;
2009-08-14 15:47:25 +02:00
continue ;
2009-08-18 17:09:42 +02:00
}
2009-08-14 15:47:25 +02:00
2009-04-21 12:24:00 -07:00
csnow = cs - > read ( cs ) ;
2007-05-09 02:35:15 -07:00
2009-08-14 15:47:22 +02:00
/* Clocksource initialized ? */
if ( ! ( cs - > flags & CLOCK_SOURCE_WATCHDOG ) ) {
cs - > flags | = CLOCK_SOURCE_WATCHDOG ;
2007-05-09 02:35:15 -07:00
cs - > wd_last = csnow ;
continue ;
}
2009-08-14 15:47:22 +02:00
/* Check the deviation from the watchdog clocksource. */
2009-08-14 15:47:26 +02:00
cs_nsec = clocksource_cyc2ns ( ( csnow - cs - > wd_last ) &
cs - > mask , cs - > mult , cs - > shift ) ;
2009-08-14 15:47:22 +02:00
cs - > wd_last = csnow ;
if ( abs ( cs_nsec - wd_nsec ) > WATCHDOG_THRESHOLD ) {
clocksource_unstable ( cs , cs_nsec - wd_nsec ) ;
continue ;
}
if ( ! ( cs - > flags & CLOCK_SOURCE_VALID_FOR_HRES ) & &
( cs - > flags & CLOCK_SOURCE_IS_CONTINUOUS ) & &
( watchdog - > flags & CLOCK_SOURCE_IS_CONTINUOUS ) ) {
cs - > flags | = CLOCK_SOURCE_VALID_FOR_HRES ;
/*
* We just marked the clocksource as highres - capable ,
* notify the rest of the system as well so that we
* transition into high - res mode :
*/
tick_clock_notify ( ) ;
2007-02-16 01:27:43 -08:00
}
}
2009-08-14 15:47:25 +02:00
/*
* Cycle through CPUs to check if the CPUs stay synchronized
* to each other .
*/
next_cpu = cpumask_next ( raw_smp_processor_id ( ) , cpu_online_mask ) ;
if ( next_cpu > = nr_cpu_ids )
next_cpu = cpumask_first ( cpu_online_mask ) ;
watchdog_timer . expires + = WATCHDOG_INTERVAL ;
add_timer_on ( & watchdog_timer , next_cpu ) ;
2009-08-14 15:47:24 +02:00
out :
2007-02-16 01:27:43 -08:00
spin_unlock ( & watchdog_lock ) ;
}
2009-08-14 15:47:23 +02:00
2009-08-14 15:47:24 +02:00
static inline void clocksource_start_watchdog ( void )
{
if ( watchdog_running | | ! watchdog | | list_empty ( & watchdog_list ) )
return ;
init_timer ( & watchdog_timer ) ;
watchdog_timer . function = clocksource_watchdog ;
watchdog_last = watchdog - > read ( watchdog ) ;
watchdog_timer . expires = jiffies + WATCHDOG_INTERVAL ;
add_timer_on ( & watchdog_timer , cpumask_first ( cpu_online_mask ) ) ;
watchdog_running = 1 ;
}
static inline void clocksource_stop_watchdog ( void )
{
if ( ! watchdog_running | | ( watchdog & & ! list_empty ( & watchdog_list ) ) )
return ;
del_timer ( & watchdog_timer ) ;
watchdog_running = 0 ;
}
2009-08-14 15:47:23 +02:00
static inline void clocksource_reset_watchdog ( void )
{
struct clocksource * cs ;
list_for_each_entry ( cs , & watchdog_list , wd_list )
cs - > flags & = ~ CLOCK_SOURCE_WATCHDOG ;
}
2007-05-09 02:35:15 -07:00
static void clocksource_resume_watchdog ( void )
{
2009-08-14 15:47:23 +02:00
unsigned long flags ;
2010-01-26 12:51:10 +01:00
/*
* We use trylock here to avoid a potential dead lock when
* kgdb calls this code after the kernel has been stopped with
* watchdog_lock held . When watchdog_lock is held we just
* return and accept , that the watchdog might trigger and mark
* the monitored clock source ( usually TSC ) unstable .
*
* This does not affect the other caller clocksource_resume ( )
* because at this point the kernel is UP , interrupts are
* disabled and nothing can hold watchdog_lock .
*/
if ( ! spin_trylock_irqsave ( & watchdog_lock , flags ) )
return ;
2009-08-14 15:47:23 +02:00
clocksource_reset_watchdog ( ) ;
spin_unlock_irqrestore ( & watchdog_lock , flags ) ;
2007-05-09 02:35:15 -07:00
}
2009-08-14 15:47:24 +02:00
static void clocksource_enqueue_watchdog ( struct clocksource * cs )
2007-02-16 01:27:43 -08:00
{
unsigned long flags ;
spin_lock_irqsave ( & watchdog_lock , flags ) ;
if ( cs - > flags & CLOCK_SOURCE_MUST_VERIFY ) {
2009-08-14 15:47:24 +02:00
/* cs is a clocksource to be watched. */
2007-02-16 01:27:43 -08:00
list_add ( & cs - > wd_list , & watchdog_list ) ;
2009-08-14 15:47:24 +02:00
cs - > flags & = ~ CLOCK_SOURCE_WATCHDOG ;
2007-03-25 14:42:51 +02:00
} else {
2009-08-14 15:47:24 +02:00
/* cs is a watchdog. */
2007-03-25 14:42:51 +02:00
if ( cs - > flags & CLOCK_SOURCE_IS_CONTINUOUS )
2007-02-16 01:27:43 -08:00
cs - > flags | = CLOCK_SOURCE_VALID_FOR_HRES ;
2009-08-14 15:47:24 +02:00
/* Pick the best watchdog. */
2007-02-16 01:27:43 -08:00
if ( ! watchdog | | cs - > rating > watchdog - > rating ) {
watchdog = cs ;
/* Reset watchdog cycles */
2009-08-14 15:47:23 +02:00
clocksource_reset_watchdog ( ) ;
2007-02-16 01:27:43 -08:00
}
}
2009-08-14 15:47:24 +02:00
/* Check if the watchdog timer needs to be started. */
clocksource_start_watchdog ( ) ;
2007-02-16 01:27:43 -08:00
spin_unlock_irqrestore ( & watchdog_lock , flags ) ;
}
2009-08-14 15:47:24 +02:00
static void clocksource_dequeue_watchdog ( struct clocksource * cs )
{
struct clocksource * tmp ;
unsigned long flags ;
spin_lock_irqsave ( & watchdog_lock , flags ) ;
if ( cs - > flags & CLOCK_SOURCE_MUST_VERIFY ) {
/* cs is a watched clocksource. */
list_del_init ( & cs - > wd_list ) ;
} else if ( cs = = watchdog ) {
/* Reset watchdog cycles */
clocksource_reset_watchdog ( ) ;
/* Current watchdog is removed. Find an alternative. */
watchdog = NULL ;
list_for_each_entry ( tmp , & clocksource_list , list ) {
if ( tmp = = cs | | tmp - > flags & CLOCK_SOURCE_MUST_VERIFY )
continue ;
if ( ! watchdog | | tmp - > rating > watchdog - > rating )
watchdog = tmp ;
}
}
cs - > flags & = ~ CLOCK_SOURCE_WATCHDOG ;
/* Check if the watchdog timer needs to be stopped. */
clocksource_stop_watchdog ( ) ;
spin_unlock_irqrestore ( & watchdog_lock , flags ) ;
}
2009-08-18 17:09:42 +02:00
static int clocksource_watchdog_kthread ( void * data )
2009-08-14 15:47:25 +02:00
{
struct clocksource * cs , * tmp ;
unsigned long flags ;
2009-08-15 13:20:42 +02:00
LIST_HEAD ( unstable ) ;
2009-08-14 15:47:25 +02:00
2009-08-19 11:26:09 +02:00
mutex_lock ( & clocksource_mutex ) ;
2009-08-14 15:47:25 +02:00
spin_lock_irqsave ( & watchdog_lock , flags ) ;
list_for_each_entry_safe ( cs , tmp , & watchdog_list , wd_list )
if ( cs - > flags & CLOCK_SOURCE_UNSTABLE ) {
list_del_init ( & cs - > wd_list ) ;
2009-08-15 13:20:42 +02:00
list_add ( & cs - > wd_list , & unstable ) ;
2009-08-14 15:47:25 +02:00
}
/* Check if the watchdog timer needs to be stopped. */
clocksource_stop_watchdog ( ) ;
2009-08-15 13:20:42 +02:00
spin_unlock_irqrestore ( & watchdog_lock , flags ) ;
/* Needs to be done outside of watchdog lock */
list_for_each_entry_safe ( cs , tmp , & unstable , wd_list ) {
list_del_init ( & cs - > wd_list ) ;
2009-08-19 11:26:09 +02:00
__clocksource_change_rating ( cs , 0 ) ;
2009-08-15 13:20:42 +02:00
}
2009-08-19 11:26:09 +02:00
mutex_unlock ( & clocksource_mutex ) ;
2009-08-18 17:09:42 +02:00
return 0 ;
2009-08-14 15:47:25 +02:00
}
2009-08-14 15:47:24 +02:00
# else /* CONFIG_CLOCKSOURCE_WATCHDOG */
static void clocksource_enqueue_watchdog ( struct clocksource * cs )
2007-02-16 01:27:43 -08:00
{
if ( cs - > flags & CLOCK_SOURCE_IS_CONTINUOUS )
cs - > flags | = CLOCK_SOURCE_VALID_FOR_HRES ;
}
2007-05-09 02:35:15 -07:00
2009-08-14 15:47:24 +02:00
static inline void clocksource_dequeue_watchdog ( struct clocksource * cs ) { }
2007-05-09 02:35:15 -07:00
static inline void clocksource_resume_watchdog ( void ) { }
2009-09-14 19:49:02 +02:00
static inline int clocksource_watchdog_kthread ( void * data ) { return 0 ; }
2009-08-14 15:47:24 +02:00
# endif /* CONFIG_CLOCKSOURCE_WATCHDOG */
2007-02-16 01:27:43 -08:00
2010-02-02 14:41:41 -08:00
/**
* clocksource_suspend - suspend the clocksource ( s )
*/
void clocksource_suspend ( void )
{
struct clocksource * cs ;
list_for_each_entry_reverse ( cs , & clocksource_list , list )
if ( cs - > suspend )
cs - > suspend ( cs ) ;
}
2007-05-09 02:35:15 -07:00
/**
* clocksource_resume - resume the clocksource ( s )
*/
void clocksource_resume ( void )
{
2007-10-18 23:39:58 -07:00
struct clocksource * cs ;
2007-05-09 02:35:15 -07:00
2009-08-14 15:47:30 +02:00
list_for_each_entry ( cs , & clocksource_list , list )
2007-05-09 02:35:15 -07:00
if ( cs - > resume )
2010-02-02 14:41:39 -08:00
cs - > resume ( cs ) ;
2007-05-09 02:35:15 -07:00
clocksource_resume_watchdog ( ) ;
}
2008-02-15 14:55:54 -06:00
/**
* clocksource_touch_watchdog - Update watchdog
*
* Update the watchdog after exception contexts such as kgdb so as not
2010-01-26 12:51:10 +01:00
* to incorrectly trip the watchdog . This might fail when the kernel
* was stopped in code which holds watchdog_lock .
2008-02-15 14:55:54 -06:00
*/
void clocksource_touch_watchdog ( void )
{
clocksource_resume_watchdog ( ) ;
}
2009-08-18 12:45:10 -05:00
/**
* clocksource_max_deferment - Returns max time the clocksource can be deferred
* @ cs : Pointer to clocksource
*
*/
static u64 clocksource_max_deferment ( struct clocksource * cs )
{
u64 max_nsecs , max_cycles ;
/*
* Calculate the maximum number of cycles that we can pass to the
* cyc2ns function without overflowing a 64 - bit signed result . The
* maximum number of cycles is equal to ULLONG_MAX / cs - > mult which
* is equivalent to the below .
* max_cycles < ( 2 ^ 63 ) / cs - > mult
* max_cycles < 2 ^ ( log2 ( ( 2 ^ 63 ) / cs - > mult ) )
* max_cycles < 2 ^ ( log2 ( 2 ^ 63 ) - log2 ( cs - > mult ) )
* max_cycles < 2 ^ ( 63 - log2 ( cs - > mult ) )
* max_cycles < 1 < < ( 63 - log2 ( cs - > mult ) )
* Please note that we add 1 to the result of the log2 to account for
* any rounding errors , ensure the above inequality is satisfied and
* no overflow will occur .
*/
max_cycles = 1ULL < < ( 63 - ( ilog2 ( cs - > mult ) + 1 ) ) ;
/*
* The actual maximum number of cycles we can defer the clocksource is
* determined by the minimum of max_cycles and cs - > mask .
*/
max_cycles = min_t ( u64 , max_cycles , ( u64 ) cs - > mask ) ;
max_nsecs = clocksource_cyc2ns ( max_cycles , cs - > mult , cs - > shift ) ;
/*
* To ensure that the clocksource does not wrap whilst we are idle ,
* limit the time the clocksource can be deferred by 12.5 % . Please
* note a margin of 12.5 % is used because this can be computed with
* a shift , versus say 10 % which would require division .
*/
return max_nsecs - ( max_nsecs > > 5 ) ;
}
2009-08-14 15:47:21 +02:00
# ifdef CONFIG_GENERIC_TIME
2006-06-26 00:25:05 -07:00
/**
2009-08-14 15:47:21 +02:00
* clocksource_select - Select the best clocksource available
2006-06-26 00:25:05 -07:00
*
2009-08-14 15:47:30 +02:00
* Private function . Must hold clocksource_mutex when called .
2006-06-26 00:25:05 -07:00
*
2007-02-16 01:27:33 -08:00
* Select the clocksource with the best rating , or the clocksource ,
* which is selected by userspace override .
2006-06-26 00:25:05 -07:00
*/
2009-08-14 15:47:21 +02:00
static void clocksource_select ( void )
2006-06-26 00:25:05 -07:00
{
2009-08-14 15:47:21 +02:00
struct clocksource * best , * cs ;
2007-02-16 01:27:43 -08:00
2009-08-14 15:47:30 +02:00
if ( ! finished_booting | | list_empty ( & clocksource_list ) )
2009-08-14 15:47:21 +02:00
return ;
/* First clocksource on the list has the best rating. */
best = list_first_entry ( & clocksource_list , struct clocksource , list ) ;
/* Check for the override clocksource. */
list_for_each_entry ( cs , & clocksource_list , list ) {
if ( strcmp ( cs - > name , override_name ) ! = 0 )
continue ;
/*
* Check to make sure we don ' t switch to a non - highres
* capable clocksource if the tick code is in oneshot
* mode ( highres or nohz )
*/
if ( ! ( cs - > flags & CLOCK_SOURCE_VALID_FOR_HRES ) & &
tick_oneshot_mode_active ( ) ) {
/* Override clocksource cannot be used. */
printk ( KERN_WARNING " Override clocksource %s is not "
" HRT compatible. Cannot switch while in "
" HRT/NOHZ mode \n " , cs - > name ) ;
override_name [ 0 ] = 0 ;
} else
/* Override clocksource can be used. */
best = cs ;
break ;
}
2009-08-14 15:47:30 +02:00
if ( curr_clocksource ! = best ) {
printk ( KERN_INFO " Switching to clocksource %s \n " , best - > name ) ;
curr_clocksource = best ;
timekeeping_notify ( curr_clocksource ) ;
}
2009-08-14 15:47:21 +02:00
}
2006-06-26 00:25:05 -07:00
2009-09-14 19:49:02 +02:00
# else /* CONFIG_GENERIC_TIME */
static inline void clocksource_select ( void ) { }
# endif
2009-08-14 15:47:30 +02:00
/*
* clocksource_done_booting - Called near the end of core bootup
*
* Hack to avoid lots of clocksource churn at boot time .
* We use fs_initcall because we want this to start before
* device_initcall but after subsys_initcall .
*/
static int __init clocksource_done_booting ( void )
{
2010-03-01 12:34:43 -08:00
mutex_lock ( & clocksource_mutex ) ;
curr_clocksource = clocksource_default_clock ( ) ;
mutex_unlock ( & clocksource_mutex ) ;
2009-08-14 15:47:30 +02:00
finished_booting = 1 ;
2009-09-14 19:49:02 +02:00
/*
* Run the watchdog first to eliminate unstable clock sources
*/
clocksource_watchdog_kthread ( NULL ) ;
2009-09-14 19:51:11 +02:00
mutex_lock ( & clocksource_mutex ) ;
2009-08-14 15:47:30 +02:00
clocksource_select ( ) ;
2009-09-14 19:51:11 +02:00
mutex_unlock ( & clocksource_mutex ) ;
2009-08-14 15:47:30 +02:00
return 0 ;
}
fs_initcall ( clocksource_done_booting ) ;
2007-02-16 01:27:33 -08:00
/*
* Enqueue the clocksource sorted by rating
2006-06-26 00:25:05 -07:00
*/
2009-08-14 15:47:21 +02:00
static void clocksource_enqueue ( struct clocksource * cs )
2006-06-26 00:25:05 -07:00
{
2009-08-14 15:47:21 +02:00
struct list_head * entry = & clocksource_list ;
struct clocksource * tmp ;
2007-02-16 01:27:33 -08:00
2009-08-14 15:47:21 +02:00
list_for_each_entry ( tmp , & clocksource_list , list )
2007-02-16 01:27:33 -08:00
/* Keep track of the place, where to insert */
2009-08-14 15:47:21 +02:00
if ( tmp - > rating > = cs - > rating )
entry = & tmp - > list ;
list_add ( & cs - > list , entry ) ;
2006-06-26 00:25:05 -07:00
}
/**
2006-06-26 00:25:14 -07:00
* clocksource_register - Used to install new clocksources
2006-06-26 00:25:05 -07:00
* @ t : clocksource to be registered
*
* Returns - EBUSY if registration fails , zero otherwise .
*/
2009-08-14 15:47:21 +02:00
int clocksource_register ( struct clocksource * cs )
2006-06-26 00:25:05 -07:00
{
2009-08-18 12:45:10 -05:00
/* calculate max idle time permitted for this clocksource */
cs - > max_idle_ns = clocksource_max_deferment ( cs ) ;
2009-08-14 15:47:30 +02:00
mutex_lock ( & clocksource_mutex ) ;
2009-08-14 15:47:21 +02:00
clocksource_enqueue ( cs ) ;
clocksource_select ( ) ;
2009-08-14 15:47:24 +02:00
clocksource_enqueue_watchdog ( cs ) ;
2009-08-14 15:47:30 +02:00
mutex_unlock ( & clocksource_mutex ) ;
2009-08-14 15:47:21 +02:00
return 0 ;
2006-06-26 00:25:05 -07:00
}
2006-06-26 00:25:14 -07:00
EXPORT_SYMBOL ( clocksource_register ) ;
2006-06-26 00:25:05 -07:00
2009-08-19 11:26:09 +02:00
static void __clocksource_change_rating ( struct clocksource * cs , int rating )
{
list_del ( & cs - > list ) ;
cs - > rating = rating ;
clocksource_enqueue ( cs ) ;
clocksource_select ( ) ;
}
2006-06-26 00:25:05 -07:00
/**
2007-02-16 01:27:33 -08:00
* clocksource_change_rating - Change the rating of a registered clocksource
2006-06-26 00:25:05 -07:00
*/
2007-02-16 01:27:33 -08:00
void clocksource_change_rating ( struct clocksource * cs , int rating )
2006-06-26 00:25:05 -07:00
{
2009-08-14 15:47:30 +02:00
mutex_lock ( & clocksource_mutex ) ;
2009-08-19 11:26:09 +02:00
__clocksource_change_rating ( cs , rating ) ;
2009-08-14 15:47:30 +02:00
mutex_unlock ( & clocksource_mutex ) ;
2006-06-26 00:25:05 -07:00
}
2009-08-14 15:47:24 +02:00
EXPORT_SYMBOL ( clocksource_change_rating ) ;
2006-06-26 00:25:05 -07:00
2008-01-30 13:30:02 +01:00
/**
* clocksource_unregister - remove a registered clocksource
*/
void clocksource_unregister ( struct clocksource * cs )
{
2009-08-14 15:47:30 +02:00
mutex_lock ( & clocksource_mutex ) ;
2009-08-14 15:47:24 +02:00
clocksource_dequeue_watchdog ( cs ) ;
2008-01-30 13:30:02 +01:00
list_del ( & cs - > list ) ;
2009-08-14 15:47:21 +02:00
clocksource_select ( ) ;
2009-08-14 15:47:30 +02:00
mutex_unlock ( & clocksource_mutex ) ;
2008-01-30 13:30:02 +01:00
}
2009-08-14 15:47:24 +02:00
EXPORT_SYMBOL ( clocksource_unregister ) ;
2008-01-30 13:30:02 +01:00
2006-12-10 02:21:30 -08:00
# ifdef CONFIG_SYSFS
2006-06-26 00:25:05 -07:00
/**
* sysfs_show_current_clocksources - sysfs interface for current clocksource
* @ dev : unused
* @ buf : char buffer to be filled with clocksource list
*
* Provides sysfs interface for listing current clocksource .
*/
static ssize_t
2008-07-01 18:48:41 +02:00
sysfs_show_current_clocksources ( struct sys_device * dev ,
struct sysdev_attribute * attr , char * buf )
2006-06-26 00:25:05 -07:00
{
2008-02-06 01:36:53 -08:00
ssize_t count = 0 ;
2006-06-26 00:25:05 -07:00
2009-08-14 15:47:30 +02:00
mutex_lock ( & clocksource_mutex ) ;
2008-02-06 01:36:53 -08:00
count = snprintf ( buf , PAGE_SIZE , " %s \n " , curr_clocksource - > name ) ;
2009-08-14 15:47:30 +02:00
mutex_unlock ( & clocksource_mutex ) ;
2006-06-26 00:25:05 -07:00
2008-02-06 01:36:53 -08:00
return count ;
2006-06-26 00:25:05 -07:00
}
/**
* sysfs_override_clocksource - interface for manually overriding clocksource
* @ dev : unused
* @ buf : name of override clocksource
* @ count : length of buffer
*
* Takes input from sysfs interface for manually overriding the default
2009-10-06 12:42:51 +02:00
* clocksource selection .
2006-06-26 00:25:05 -07:00
*/
static ssize_t sysfs_override_clocksource ( struct sys_device * dev ,
2008-07-01 18:48:41 +02:00
struct sysdev_attribute * attr ,
2006-06-26 00:25:05 -07:00
const char * buf , size_t count )
{
size_t ret = count ;
2007-02-16 01:27:33 -08:00
2006-06-26 00:25:05 -07:00
/* strings from sysfs write are not 0 terminated! */
if ( count > = sizeof ( override_name ) )
return - EINVAL ;
/* strip of \n: */
if ( buf [ count - 1 ] = = ' \n ' )
count - - ;
2009-08-14 15:47:30 +02:00
mutex_lock ( & clocksource_mutex ) ;
2006-06-26 00:25:05 -07:00
2007-02-16 01:27:33 -08:00
if ( count > 0 )
memcpy ( override_name , buf , count ) ;
2006-06-26 00:25:05 -07:00
override_name [ count ] = 0 ;
2009-08-14 15:47:21 +02:00
clocksource_select ( ) ;
2006-06-26 00:25:05 -07:00
2009-08-14 15:47:30 +02:00
mutex_unlock ( & clocksource_mutex ) ;
2006-06-26 00:25:05 -07:00
return ret ;
}
/**
* sysfs_show_available_clocksources - sysfs interface for listing clocksource
* @ dev : unused
* @ buf : char buffer to be filled with clocksource list
*
* Provides sysfs interface for listing registered clocksources
*/
static ssize_t
2008-07-01 18:48:41 +02:00
sysfs_show_available_clocksources ( struct sys_device * dev ,
struct sysdev_attribute * attr ,
char * buf )
2006-06-26 00:25:05 -07:00
{
2007-10-18 23:39:58 -07:00
struct clocksource * src ;
2008-02-06 01:36:53 -08:00
ssize_t count = 0 ;
2006-06-26 00:25:05 -07:00
2009-08-14 15:47:30 +02:00
mutex_lock ( & clocksource_mutex ) ;
2007-10-18 23:39:58 -07:00
list_for_each_entry ( src , & clocksource_list , list ) {
2009-06-12 11:29:27 +02:00
/*
* Don ' t show non - HRES clocksource if the tick code is
* in one shot mode ( highres = on or nohz = on )
*/
if ( ! tick_oneshot_mode_active ( ) | |
( src - > flags & CLOCK_SOURCE_VALID_FOR_HRES ) )
2009-01-21 22:53:22 -07:00
count + = snprintf ( buf + count ,
2008-02-06 01:36:53 -08:00
max ( ( ssize_t ) PAGE_SIZE - count , ( ssize_t ) 0 ) ,
" %s " , src - > name ) ;
2006-06-26 00:25:05 -07:00
}
2009-08-14 15:47:30 +02:00
mutex_unlock ( & clocksource_mutex ) ;
2006-06-26 00:25:05 -07:00
2008-02-06 01:36:53 -08:00
count + = snprintf ( buf + count ,
max ( ( ssize_t ) PAGE_SIZE - count , ( ssize_t ) 0 ) , " \n " ) ;
2006-06-26 00:25:05 -07:00
2008-02-06 01:36:53 -08:00
return count ;
2006-06-26 00:25:05 -07:00
}
/*
* Sysfs setup bits :
*/
2008-05-03 14:23:14 +02:00
static SYSDEV_ATTR ( current_clocksource , 0644 , sysfs_show_current_clocksources ,
2006-12-10 02:21:33 -08:00
sysfs_override_clocksource ) ;
2006-06-26 00:25:05 -07:00
2008-05-03 14:23:14 +02:00
static SYSDEV_ATTR ( available_clocksource , 0444 ,
2006-12-10 02:21:33 -08:00
sysfs_show_available_clocksources , NULL ) ;
2006-06-26 00:25:05 -07:00
static struct sysdev_class clocksource_sysclass = {
2007-12-20 02:09:39 +01:00
. name = " clocksource " ,
2006-06-26 00:25:05 -07:00
} ;
static struct sys_device device_clocksource = {
. id = 0 ,
. cls = & clocksource_sysclass ,
} ;
2006-06-26 00:25:06 -07:00
static int __init init_clocksource_sysfs ( void )
2006-06-26 00:25:05 -07:00
{
int error = sysdev_class_register ( & clocksource_sysclass ) ;
if ( ! error )
error = sysdev_register ( & device_clocksource ) ;
if ( ! error )
error = sysdev_create_file (
& device_clocksource ,
& attr_current_clocksource ) ;
if ( ! error )
error = sysdev_create_file (
& device_clocksource ,
& attr_available_clocksource ) ;
return error ;
}
device_initcall ( init_clocksource_sysfs ) ;
2006-12-10 02:21:30 -08:00
# endif /* CONFIG_SYSFS */
2006-06-26 00:25:05 -07:00
/**
* boot_override_clocksource - boot clock override
* @ str : override name
*
* Takes a clocksource = boot argument and uses it
* as the clocksource override name .
*/
static int __init boot_override_clocksource ( char * str )
{
2009-08-14 15:47:30 +02:00
mutex_lock ( & clocksource_mutex ) ;
2006-06-26 00:25:05 -07:00
if ( str )
strlcpy ( override_name , str , sizeof ( override_name ) ) ;
2009-08-14 15:47:30 +02:00
mutex_unlock ( & clocksource_mutex ) ;
2006-06-26 00:25:05 -07:00
return 1 ;
}
__setup ( " clocksource= " , boot_override_clocksource ) ;
/**
* boot_override_clock - Compatibility layer for deprecated boot option
* @ str : override name
*
* DEPRECATED ! Takes a clock = boot argument and uses it
* as the clocksource override name
*/
static int __init boot_override_clock ( char * str )
{
2006-06-26 00:25:12 -07:00
if ( ! strcmp ( str , " pmtmr " ) ) {
printk ( " Warning: clock=pmtmr is deprecated. "
" Use clocksource=acpi_pm. \n " ) ;
return boot_override_clocksource ( " acpi_pm " ) ;
}
printk ( " Warning! clock= boot option is deprecated. "
" Use clocksource=xyz \n " ) ;
2006-06-26 00:25:05 -07:00
return boot_override_clocksource ( str ) ;
}
__setup ( " clock= " , boot_override_clock ) ;