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>
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>
2015-08-10 13:19:54 +03:00
# include <asm/mwait.h>
2005-04-17 02:20:36 +04:00
# 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 )
{
2008-07-03 19:35:41 +04:00
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: */
2012-03-09 23:55:10 +04:00
static void delay_tsc ( unsigned long __loops )
2006-06-26 11:25:11 +04:00
{
2015-06-25 19:44:00 +03:00
u64 bclock , now , loops = __loops ;
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 ( ) ;
2015-06-25 19:44:08 +03:00
bclock = rdtsc_ordered ( ) ;
2008-05-25 19:13:32 +04:00
for ( ; ; ) {
2015-06-25 19:44:08 +03:00
now = rdtsc_ordered ( ) ;
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 ( ) ;
2015-06-25 19:44:08 +03:00
bclock = rdtsc_ordered ( ) ;
2008-05-25 19:13:32 +04:00
}
}
2007-11-15 04:00:41 +03:00
preempt_enable ( ) ;
2006-06-26 11:25:11 +04:00
}
2015-08-10 13:19:54 +03:00
/*
* On some AMD platforms , MWAITX has a configurable 32 - bit timer , that
* counts with TSC frequency . The input value is the loop of the
* counter , it will exit when the timer expires .
*/
static void delay_mwaitx ( unsigned long __loops )
{
u64 start , end , delay , loops = __loops ;
start = rdtsc_ordered ( ) ;
for ( ; ; ) {
delay = min_t ( u64 , MWAITX_MAX_LOOPS , loops ) ;
/*
* Use cpu_tss as a cacheline - aligned , seldomly
* accessed per - cpu variable as the monitor target .
*/
__monitorx ( this_cpu_ptr ( & cpu_tss ) , 0 , 0 ) ;
/*
* AMD , like Intel , supports the EAX hint and EAX = 0xf
* means , do not enter any deep C - state and we use it
* here in delay ( ) to minimize wakeup latency .
*/
__mwaitx ( MWAITX_DISABLE_CSTATES , delay , MWAITX_ECX_TIMER_ENABLE ) ;
end = rdtsc_ordered ( ) ;
if ( loops < = end - start )
break ;
loops - = end - start ;
start = end ;
}
}
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 )
{
2015-08-10 13:19:54 +03:00
if ( delay_fn = = delay_loop )
delay_fn = delay_tsc ;
}
void use_mwaitx_delay ( void )
{
delay_fn = delay_mwaitx ;
2006-06-26 11:25:11 +04:00
}
2012-12-22 02:02:53 +04:00
int read_current_timer ( unsigned long * timer_val )
2006-06-26 11:25:11 +04:00
{
if ( delay_fn = = delay_tsc ) {
2015-06-25 19:44:07 +03:00
* timer_val = rdtsc ( ) ;
2006-06-26 11:25:11 +04:00
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
}
2008-07-03 19:35:41 +04:00
EXPORT_SYMBOL ( __delay ) ;
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 ;
2008-07-03 19:35:41 +04:00
asm ( " mull %%edx "
2005-04-17 02:20:36 +04:00
: " =d " ( xloops ) , " =&a " ( d0 )
2006-06-26 11:25:11 +04:00
: " 1 " ( xloops ) , " 0 "
2010-12-16 21:14:43 +03:00
( this_cpu_read ( cpu_info . loops_per_jiffy ) * ( HZ / 4 ) ) ) ;
2006-06-26 11:25:11 +04:00
__delay ( + + xloops ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-03 19:35:41 +04:00
EXPORT_SYMBOL ( __const_udelay ) ;
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
}
2008-07-03 19:35:41 +04:00
EXPORT_SYMBOL ( __udelay ) ;
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 ( __ndelay ) ;