2005-04-16 15:20:36 -07: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 12:00:19 +02:00
* Copyright ( C ) 2008 Jiri Hladky < hladky _dot_ jiri _at_ gmail _dot_ com >
2005-04-16 15:20:36 -07: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 .
*/
2016-07-13 20:18:57 -04:00
# include <linux/export.h>
2005-04-16 15:20:36 -07:00
# include <linux/sched.h>
2008-02-06 01:36:42 -08:00
# include <linux/timex.h>
2007-11-14 17:00:41 -08:00
# include <linux/preempt.h>
2005-04-16 15:20:36 -07:00
# include <linux/delay.h>
2006-06-26 00:25:11 -07:00
2005-04-16 15:20:36 -07:00
# include <asm/processor.h>
# include <asm/delay.h>
# include <asm/timer.h>
2015-08-10 12:19:54 +02:00
# include <asm/mwait.h>
2005-04-16 15:20:36 -07:00
# ifdef CONFIG_SMP
2006-06-26 00:25:11 -07:00
# include <asm / smp.h>
2005-04-16 15:20:36 -07:00
# endif
2006-06-26 00:25:11 -07:00
/* simple loop based delay: */
static void delay_loop ( unsigned long loops )
{
2008-07-03 12:35:41 -03:00
asm volatile (
2008-06-02 12:00:19 +02:00
" test %0,%0 \n "
" jz 3f \n "
" jmp 1f \n "
" .align 16 \n "
" 1: jmp 2f \n "
" .align 16 \n "
2008-06-24 09:27:19 -03:00
" 2: dec %0 \n "
2008-06-02 12:00:19 +02:00
" jnz 2b \n "
2008-06-24 09:27:19 -03:00
" 3: dec %0 \n "
2008-06-02 12:00:19 +02:00
: /* we don't need output */
: " a " ( loops )
) ;
2006-06-26 00:25:11 -07:00
}
/* TSC based delay: */
2012-03-09 20:55:10 +01:00
static void delay_tsc ( unsigned long __loops )
2006-06-26 00:25:11 -07:00
{
2015-06-25 18:44:00 +02:00
u64 bclock , now , loops = __loops ;
2008-05-25 11:13:32 -04:00
int cpu ;
2006-06-26 00:25:11 -07:00
2008-05-25 11:13:32 -04:00
preempt_disable ( ) ;
cpu = smp_processor_id ( ) ;
2015-06-25 18:44:08 +02:00
bclock = rdtsc_ordered ( ) ;
2008-05-25 11:13:32 -04:00
for ( ; ; ) {
2015-06-25 18:44:08 +02:00
now = rdtsc_ordered ( ) ;
2008-05-25 11: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 18:44:08 +02:00
bclock = rdtsc_ordered ( ) ;
2008-05-25 11:13:32 -04:00
}
}
2007-11-14 17:00:41 -08:00
preempt_enable ( ) ;
2006-06-26 00:25:11 -07:00
}
2015-08-10 12:19:54 +02: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 .
*/
2016-03-09 21:56:22 +01:00
__monitorx ( raw_cpu_ptr ( & cpu_tss ) , 0 , 0 ) ;
2015-08-10 12:19:54 +02:00
/*
* 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 00:25:11 -07: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 12:19:54 +02:00
if ( delay_fn = = delay_loop )
delay_fn = delay_tsc ;
}
void use_mwaitx_delay ( void )
{
delay_fn = delay_mwaitx ;
2006-06-26 00:25:11 -07:00
}
2012-12-21 14:02:53 -08:00
int read_current_timer ( unsigned long * timer_val )
2006-06-26 00:25:11 -07:00
{
if ( delay_fn = = delay_tsc ) {
2015-06-25 18:44:07 +02:00
* timer_val = rdtsc ( ) ;
2006-06-26 00:25:11 -07:00
return 0 ;
}
return - 1 ;
}
2005-04-16 15:20:36 -07:00
void __delay ( unsigned long loops )
{
2006-06-26 00:25:11 -07:00
delay_fn ( loops ) ;
2005-04-16 15:20:36 -07:00
}
2008-07-03 12:35:41 -03:00
EXPORT_SYMBOL ( __delay ) ;
2005-04-16 15:20:36 -07:00
inline void __const_udelay ( unsigned long xloops )
{
2017-01-19 12:47:30 +01:00
unsigned long lpj = this_cpu_read ( cpu_info . loops_per_jiffy ) ? : loops_per_jiffy ;
2005-04-16 15:20:36 -07:00
int d0 ;
2006-06-26 00:25:11 -07:00
2005-04-16 15:20:36 -07:00
xloops * = 4 ;
2008-07-03 12:35:41 -03:00
asm ( " mull %%edx "
2005-04-16 15:20:36 -07:00
: " =d " ( xloops ) , " =&a " ( d0 )
2017-01-19 12:47:30 +01:00
: " 1 " ( xloops ) , " 0 " ( lpj * ( HZ / 4 ) ) ) ;
2006-06-26 00:25:11 -07:00
__delay ( + + xloops ) ;
2005-04-16 15:20:36 -07:00
}
2008-07-03 12:35:41 -03:00
EXPORT_SYMBOL ( __const_udelay ) ;
2005-04-16 15:20:36 -07:00
void __udelay ( unsigned long usecs )
{
2006-06-26 00:25:11 -07:00
__const_udelay ( usecs * 0x000010c7 ) ; /* 2**32 / 1000000 (rounded up) */
2005-04-16 15:20:36 -07:00
}
2008-07-03 12:35:41 -03:00
EXPORT_SYMBOL ( __udelay ) ;
2005-04-16 15:20:36 -07:00
void __ndelay ( unsigned long nsecs )
{
2006-06-26 00:25:11 -07:00
__const_udelay ( nsecs * 0x00005 ) ; /* 2**32 / 1000000000 (rounded up) */
2005-04-16 15:20:36 -07:00
}
2005-06-23 00:08:33 -07:00
EXPORT_SYMBOL ( __ndelay ) ;