2012-07-06 15:47:17 +01:00
/*
* Delay loops based on the OpenRISC implementation .
*
* Copyright ( C ) 2012 ARM Limited
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* 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 . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*
* Author : Will Deacon < will . deacon @ arm . com >
*/
# include <linux/delay.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/timex.h>
/*
* Default to the loop - based delay implementation .
*/
struct arm_delay_ops arm_delay_ops = {
. delay = __loop_delay ,
. const_udelay = __loop_const_udelay ,
. udelay = __loop_udelay ,
} ;
2012-09-21 18:51:44 +01:00
static const struct delay_timer * delay_timer ;
static bool delay_calibrated ;
int read_current_timer ( unsigned long * timer_val )
{
if ( ! delay_timer )
return - ENXIO ;
* timer_val = delay_timer - > read_current_timer ( ) ;
return 0 ;
}
2012-10-01 14:47:31 +00:00
EXPORT_SYMBOL_GPL ( read_current_timer ) ;
2012-09-21 18:51:44 +01:00
2012-07-06 15:47:17 +01:00
static void __timer_delay ( unsigned long cycles )
{
cycles_t start = get_cycles ( ) ;
while ( ( get_cycles ( ) - start ) < cycles )
cpu_relax ( ) ;
}
static void __timer_const_udelay ( unsigned long xloops )
{
unsigned long long loops = xloops ;
2013-03-28 11:17:55 +01:00
loops * = arm_delay_ops . ticks_per_jiffy ;
2012-07-06 15:47:17 +01:00
__timer_delay ( loops > > UDELAY_SHIFT ) ;
}
static void __timer_udelay ( unsigned long usecs )
{
__timer_const_udelay ( usecs * UDELAY_MULT ) ;
}
2012-09-21 18:51:44 +01:00
void __init register_current_timer_delay ( const struct delay_timer * timer )
2012-07-06 15:47:17 +01:00
{
2012-09-21 18:51:44 +01:00
if ( ! delay_calibrated ) {
pr_info ( " Switching to timer-based delay loop \n " ) ;
delay_timer = timer ;
lpj_fine = timer - > freq / HZ ;
2013-03-28 11:17:55 +01:00
/* cpufreq may scale loops_per_jiffy, so keep a private copy */
arm_delay_ops . ticks_per_jiffy = lpj_fine ;
2012-09-21 18:51:44 +01:00
arm_delay_ops . delay = __timer_delay ;
arm_delay_ops . const_udelay = __timer_const_udelay ;
arm_delay_ops . udelay = __timer_udelay ;
2013-03-28 11:17:55 +01:00
2012-09-21 18:51:44 +01:00
delay_calibrated = true ;
} else {
pr_info ( " Ignoring duplicate/late registration of read_current_timer delay \n " ) ;
}
2012-07-06 15:47:17 +01:00
}
2013-06-17 15:43:14 -04:00
unsigned long calibrate_delay_is_known ( void )
2012-07-06 15:47:17 +01:00
{
2012-09-21 18:51:44 +01:00
delay_calibrated = true ;
2012-07-06 15:47:17 +01:00
return lpj_fine ;
}