2005-04-17 02:20:36 +04:00
/*
* Precise Delay Loops for i386
*
* Copyright ( C ) 1993 Linus Torvalds
* Copyright ( C ) 1997 Martin Mares < mj @ atrey . karlin . mff . cuni . cz >
2008-06-02 14:00:19 +04:00
* Copyright ( C ) 2008 Jiri Hladky < hladky _dot_ jiri _at_ gmail _dot_ com >
2005-04-17 02:20:36 +04:00
*
* The __delay function must _NOT_ be inlined as its execution time
* depends wildly on alignment on many x86 processors . The additional
* jump magic is needed to get the timing stable on all the CPU ' s
* we have to worry about .
*/
2006-06-26 11:25:11 +04:00
# include <linux/module.h>
2005-04-17 02:20:36 +04:00
# include <linux/sched.h>
2008-02-06 12:36:42 +03:00
# include <linux/timex.h>
2007-11-15 04:00:41 +03:00
# include <linux/preempt.h>
2005-04-17 02:20:36 +04:00
# include <linux/delay.h>
2008-02-06 12:36:42 +03:00
# include <linux/init.h>
2006-06-26 11:25:11 +04:00
2005-04-17 02:20:36 +04:00
# include <asm/processor.h>
# include <asm/delay.h>
# include <asm/timer.h>
# ifdef CONFIG_SMP
2006-06-26 11:25:11 +04:00
# include <asm / smp.h>
2005-04-17 02:20:36 +04:00
# endif
2006-06-26 11:25:11 +04:00
/* simple loop based delay: */
static void delay_loop ( unsigned long loops )
{
__asm__ __volatile__ (
2008-06-02 14:00:19 +04:00
" test %0,%0 \n "
" jz 3f \n "
" jmp 1f \n "
" .align 16 \n "
" 1: jmp 2f \n "
" .align 16 \n "
2008-06-24 16:27:19 +04:00
" 2: dec %0 \n "
2008-06-02 14:00:19 +04:00
" jnz 2b \n "
2008-06-24 16:27:19 +04:00
" 3: dec %0 \n "
2008-06-02 14:00:19 +04:00
: /* we don't need output */
: " a " ( loops )
) ;
2006-06-26 11:25:11 +04:00
}
/* TSC based delay: */
static void delay_tsc ( unsigned long loops )
{
unsigned long bclock , now ;
2008-05-25 19:13:32 +04:00
int cpu ;
2006-06-26 11:25:11 +04:00
2008-05-25 19:13:32 +04:00
preempt_disable ( ) ;
cpu = smp_processor_id ( ) ;
2006-06-26 11:25:11 +04:00
rdtscl ( bclock ) ;
2008-05-25 19:13:32 +04:00
for ( ; ; ) {
2006-06-26 11:25:11 +04:00
rdtscl ( now ) ;
2008-05-25 19:13:32 +04:00
if ( ( now - bclock ) > = loops )
break ;
/* Allow RT tasks to run */
preempt_enable ( ) ;
rep_nop ( ) ;
preempt_disable ( ) ;
/*
* It is possible that we moved to another CPU , and
* since TSC ' s are per - cpu we need to calculate
* that . The delay must guarantee that we wait " at
* least " the amount of time. Being moved to another
* CPU could make the wait longer but we just need to
* make sure we waited long enough . Rebalance the
* counter for this CPU .
*/
if ( unlikely ( cpu ! = smp_processor_id ( ) ) ) {
loops - = ( now - bclock ) ;
cpu = smp_processor_id ( ) ;
rdtscl ( bclock ) ;
}
}
2007-11-15 04:00:41 +03:00
preempt_enable ( ) ;
2006-06-26 11:25:11 +04:00
}
/*
* Since we calibrate only once at boot , this
* function should be set once at boot and not changed
*/
static void ( * delay_fn ) ( unsigned long ) = delay_loop ;
void use_tsc_delay ( void )
{
delay_fn = delay_tsc ;
}
2008-02-06 12:36:42 +03:00
int __devinit read_current_timer ( unsigned long * timer_val )
2006-06-26 11:25:11 +04:00
{
if ( delay_fn = = delay_tsc ) {
rdtscl ( * timer_val ) ;
return 0 ;
}
return - 1 ;
}
2005-04-17 02:20:36 +04:00
void __delay ( unsigned long loops )
{
2006-06-26 11:25:11 +04:00
delay_fn ( loops ) ;
2005-04-17 02:20:36 +04:00
}
inline void __const_udelay ( unsigned long xloops )
{
int d0 ;
2006-06-26 11:25:11 +04:00
2005-04-17 02:20:36 +04:00
xloops * = 4 ;
__asm__ ( " mull %0 "
: " =d " ( xloops ) , " =&a " ( d0 )
2006-06-26 11:25:11 +04:00
: " 1 " ( xloops ) , " 0 "
2007-10-19 22:35:04 +04:00
( cpu_data ( raw_smp_processor_id ( ) ) . loops_per_jiffy * ( HZ / 4 ) ) ) ;
2006-06-26 11:25:11 +04:00
__delay ( + + xloops ) ;
2005-04-17 02:20:36 +04:00
}
void __udelay ( unsigned long usecs )
{
2006-06-26 11:25:11 +04:00
__const_udelay ( usecs * 0x000010c7 ) ; /* 2**32 / 1000000 (rounded up) */
2005-04-17 02:20:36 +04:00
}
void __ndelay ( unsigned long nsecs )
{
2006-06-26 11:25:11 +04:00
__const_udelay ( nsecs * 0x00005 ) ; /* 2**32 / 1000000000 (rounded up) */
2005-04-17 02:20:36 +04:00
}
2005-06-23 11:08:33 +04:00
EXPORT_SYMBOL ( __delay ) ;
EXPORT_SYMBOL ( __const_udelay ) ;
EXPORT_SYMBOL ( __udelay ) ;
EXPORT_SYMBOL ( __ndelay ) ;