2011-07-25 20:36:42 +04:00
/*
2012-07-12 00:13:16 +04:00
* Copyright ( C ) 2012 Altera Corporation
2011-07-25 20:36:42 +04:00
* Copyright ( c ) 2011 Picochip Ltd . , Jamie Iles
*
2012-07-12 00:13:16 +04:00
* Modified from mach - picoxcell / time . c
*
2011-07-25 20:36:42 +04:00
* 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 .
*
2012-07-12 00:13:16 +04:00
* 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 , see < http : //www.gnu.org/licenses/>.
2011-07-25 20:36:42 +04:00
*/
# include <linux/dw_apb_timer.h>
# include <linux/of.h>
# include <linux/of_address.h>
# include <linux/of_irq.h>
2013-06-04 13:37:36 +04:00
# include <linux/clk.h>
2013-06-02 10:39:40 +04:00
# include <linux/sched_clock.h>
2011-07-25 20:36:42 +04:00
static void timer_get_base_and_rate ( struct device_node * np ,
void __iomem * * base , u32 * rate )
{
2013-06-04 13:37:36 +04:00
struct clk * timer_clk ;
struct clk * pclk ;
2011-07-25 20:36:42 +04:00
* base = of_iomap ( np , 0 ) ;
if ( ! * base )
panic ( " Unable to map regs for %s " , np - > name ) ;
2013-06-04 13:37:36 +04:00
/*
* Not all implementations use a periphal clock , so don ' t panic
* if it ' s not present
*/
pclk = of_clk_get_by_name ( np , " pclk " ) ;
if ( ! IS_ERR ( pclk ) )
if ( clk_prepare_enable ( pclk ) )
pr_warn ( " pclk for %s is present, but could not be activated \n " ,
np - > name ) ;
timer_clk = of_clk_get_by_name ( np , " timer " ) ;
if ( IS_ERR ( timer_clk ) )
goto try_clock_freq ;
if ( ! clk_prepare_enable ( timer_clk ) ) {
* rate = clk_get_rate ( timer_clk ) ;
return ;
}
try_clock_freq :
2012-07-12 00:13:16 +04:00
if ( of_property_read_u32 ( np , " clock-freq " , rate ) & &
of_property_read_u32 ( np , " clock-frequency " , rate ) )
2013-06-04 13:37:36 +04:00
panic ( " No clock nor clock-frequency property for %s " , np - > name ) ;
2011-07-25 20:36:42 +04:00
}
2012-07-12 00:13:16 +04:00
static void add_clockevent ( struct device_node * event_timer )
2011-07-25 20:36:42 +04:00
{
void __iomem * iobase ;
struct dw_apb_clock_event_device * ced ;
u32 irq , rate ;
irq = irq_of_parse_and_map ( event_timer , 0 ) ;
2013-05-29 12:11:17 +04:00
if ( irq = = 0 )
2011-07-25 20:36:42 +04:00
panic ( " No IRQ for clock event timer " ) ;
timer_get_base_and_rate ( event_timer , & iobase , & rate ) ;
ced = dw_apb_clockevent_init ( 0 , event_timer - > name , 300 , iobase , irq ,
rate ) ;
if ( ! ced )
panic ( " Unable to initialise clockevent device " ) ;
dw_apb_clockevent_register ( ced ) ;
}
2013-06-04 13:37:02 +04:00
static void __iomem * sched_io_base ;
static u32 sched_rate ;
2012-07-12 00:13:16 +04:00
static void add_clocksource ( struct device_node * source_timer )
2011-07-25 20:36:42 +04:00
{
void __iomem * iobase ;
struct dw_apb_clocksource * cs ;
u32 rate ;
timer_get_base_and_rate ( source_timer , & iobase , & rate ) ;
cs = dw_apb_clocksource_init ( 300 , source_timer - > name , iobase , rate ) ;
if ( ! cs )
panic ( " Unable to initialise clocksource device " ) ;
dw_apb_clocksource_start ( cs ) ;
dw_apb_clocksource_register ( cs ) ;
2013-06-04 13:37:02 +04:00
/*
* Fallback to use the clocksource as sched_clock if no separate
* timer is found . sched_io_base then points to the current_value
* register of the clocksource timer .
*/
sched_io_base = iobase + 0x04 ;
sched_rate = rate ;
}
2011-07-25 20:36:42 +04:00
2012-07-12 00:13:16 +04:00
static u32 read_sched_clock ( void )
2011-07-25 20:36:42 +04:00
{
2011-12-15 15:19:23 +04:00
return __raw_readl ( sched_io_base ) ;
2011-07-25 20:36:42 +04:00
}
2012-07-12 00:13:16 +04:00
static const struct of_device_id sptimer_ids [ ] __initconst = {
2011-07-25 20:36:42 +04:00
{ . compatible = " picochip,pc3x2-rtc " } ,
2012-07-12 00:13:16 +04:00
{ . compatible = " snps,dw-apb-timer-sp " } ,
2011-07-25 20:36:42 +04:00
{ /* Sentinel */ } ,
} ;
2012-07-12 00:13:16 +04:00
static void init_sched_clock ( void )
2011-07-25 20:36:42 +04:00
{
struct device_node * sched_timer ;
2012-07-12 00:13:16 +04:00
sched_timer = of_find_matching_node ( NULL , sptimer_ids ) ;
2013-06-04 13:37:02 +04:00
if ( sched_timer ) {
timer_get_base_and_rate ( sched_timer , & sched_io_base ,
& sched_rate ) ;
of_node_put ( sched_timer ) ;
}
2011-07-25 20:36:42 +04:00
2013-06-04 13:37:02 +04:00
setup_sched_clock ( read_sched_clock , 32 , sched_rate ) ;
2011-07-25 20:36:42 +04:00
}
2013-06-04 13:38:42 +04:00
static int num_called ;
static void __init dw_apb_timer_init ( struct device_node * timer )
2011-07-25 20:36:42 +04:00
{
2013-06-04 13:38:42 +04:00
switch ( num_called ) {
case 0 :
pr_debug ( " %s: found clockevent timer \n " , __func__ ) ;
add_clockevent ( timer ) ;
of_node_put ( timer ) ;
break ;
case 1 :
pr_debug ( " %s: found clocksource timer \n " , __func__ ) ;
add_clocksource ( timer ) ;
of_node_put ( timer ) ;
init_sched_clock ( ) ;
break ;
default :
break ;
}
2011-07-25 20:36:42 +04:00
2013-06-04 13:38:42 +04:00
num_called + + ;
2011-07-25 20:36:42 +04:00
}
2013-06-04 13:38:42 +04:00
CLOCKSOURCE_OF_DECLARE ( pc3x2_timer , " picochip,pc3x2-timer " , dw_apb_timer_init ) ;
CLOCKSOURCE_OF_DECLARE ( apb_timer , " snps,dw-apb-timer-osc " , dw_apb_timer_init ) ;