2012-11-12 15:07:50 +01:00
/*
* Allwinner A1X SoCs timer handling .
*
* Copyright ( C ) 2012 Maxime Ripard
*
* Maxime Ripard < maxime . ripard @ free - electrons . com >
*
* Based on code from
* Allwinner Technology Co . , Ltd . < www . allwinnertech . com >
* Benn Huang < benn @ allwinnertech . com >
*
* This file is licensed under the terms of the GNU General Public
* License version 2. This program is licensed " as is " without any
* warranty of any kind , whether express or implied .
*/
# include <linux/clk.h>
# include <linux/clockchips.h>
# include <linux/interrupt.h>
# include <linux/irq.h>
# include <linux/irqreturn.h>
2013-07-16 16:45:37 +02:00
# include <linux/sched_clock.h>
2012-11-12 15:07:50 +01:00
# include <linux/of.h>
# include <linux/of_address.h>
# include <linux/of_irq.h>
2013-03-10 17:03:46 +01:00
# define TIMER_IRQ_EN_REG 0x00
2013-07-16 16:45:37 +02:00
# define TIMER_IRQ_EN(val) BIT(val)
2012-11-12 15:07:50 +01:00
# define TIMER_IRQ_ST_REG 0x04
2013-03-10 17:03:46 +01:00
# define TIMER_CTL_REG(val) (0x10 * val + 0x10)
2013-07-16 16:45:37 +02:00
# define TIMER_CTL_ENABLE BIT(0)
2013-07-16 16:45:37 +02:00
# define TIMER_CTL_RELOAD BIT(1)
2013-07-16 16:45:38 +02:00
# define TIMER_CTL_CLK_SRC(val) (((val) & 0x3) << 2)
# define TIMER_CTL_CLK_SRC_OSC24M (1)
# define TIMER_CTL_CLK_PRES(val) (((val) & 0x7) << 4)
2013-07-16 16:45:37 +02:00
# define TIMER_CTL_ONESHOT BIT(7)
2013-07-16 16:45:37 +02:00
# define TIMER_INTVAL_REG(val) (0x10 * (val) + 0x14)
# define TIMER_CNTVAL_REG(val) (0x10 * (val) + 0x18)
2012-11-12 15:07:50 +01:00
2013-10-14 21:07:47 +02:00
# define TIMER_SYNC_TICKS 3
2012-11-12 15:07:50 +01:00
static void __iomem * timer_base ;
2013-07-16 16:45:38 +02:00
static u32 ticks_per_jiffy ;
2012-11-12 15:07:50 +01:00
2013-07-16 16:45:38 +02:00
/*
* When we disable a timer , we need to wait at least for 2 cycles of
* the timer source clock . We will use for that the clocksource timer
* that is already setup and runs at the same frequency than the other
* timers , and we never will be disabled .
*/
static void sun4i_clkevt_sync ( void )
{
u32 old = readl ( timer_base + TIMER_CNTVAL_REG ( 1 ) ) ;
2013-10-14 21:07:47 +02:00
while ( ( old - readl ( timer_base + TIMER_CNTVAL_REG ( 1 ) ) ) < TIMER_SYNC_TICKS )
2013-07-16 16:45:38 +02:00
cpu_relax ( ) ;
}
2013-07-16 16:45:38 +02:00
static void sun4i_clkevt_time_stop ( u8 timer )
{
u32 val = readl ( timer_base + TIMER_CTL_REG ( timer ) ) ;
writel ( val & ~ TIMER_CTL_ENABLE , timer_base + TIMER_CTL_REG ( timer ) ) ;
sun4i_clkevt_sync ( ) ;
}
static void sun4i_clkevt_time_setup ( u8 timer , unsigned long delay )
{
writel ( delay , timer_base + TIMER_INTVAL_REG ( timer ) ) ;
}
static void sun4i_clkevt_time_start ( u8 timer , bool periodic )
{
u32 val = readl ( timer_base + TIMER_CTL_REG ( timer ) ) ;
if ( periodic )
val & = ~ TIMER_CTL_ONESHOT ;
else
val | = TIMER_CTL_ONESHOT ;
2013-07-16 16:45:38 +02:00
writel ( val | TIMER_CTL_ENABLE | TIMER_CTL_RELOAD ,
timer_base + TIMER_CTL_REG ( timer ) ) ;
2013-07-16 16:45:38 +02:00
}
2015-06-18 16:24:37 +05:30
static int sun4i_clkevt_shutdown ( struct clock_event_device * evt )
2012-11-12 15:07:50 +01:00
{
2015-06-18 16:24:37 +05:30
sun4i_clkevt_time_stop ( 0 ) ;
return 0 ;
}
static int sun4i_clkevt_set_oneshot ( struct clock_event_device * evt )
{
sun4i_clkevt_time_stop ( 0 ) ;
sun4i_clkevt_time_start ( 0 , false ) ;
return 0 ;
}
static int sun4i_clkevt_set_periodic ( struct clock_event_device * evt )
{
sun4i_clkevt_time_stop ( 0 ) ;
sun4i_clkevt_time_setup ( 0 , ticks_per_jiffy ) ;
sun4i_clkevt_time_start ( 0 , true ) ;
return 0 ;
2012-11-12 15:07:50 +01:00
}
2013-03-24 11:49:25 +01:00
static int sun4i_clkevt_next_event ( unsigned long evt ,
2012-11-12 15:07:50 +01:00
struct clock_event_device * unused )
{
2013-07-16 16:45:38 +02:00
sun4i_clkevt_time_stop ( 0 ) ;
2013-10-14 21:07:47 +02:00
sun4i_clkevt_time_setup ( 0 , evt - TIMER_SYNC_TICKS ) ;
2013-07-16 16:45:38 +02:00
sun4i_clkevt_time_start ( 0 , false ) ;
2012-11-12 15:07:50 +01:00
return 0 ;
}
2013-03-24 11:49:25 +01:00
static struct clock_event_device sun4i_clockevent = {
. name = " sun4i_tick " ,
2013-11-07 12:01:48 +01:00
. rating = 350 ,
2012-11-12 15:07:50 +01:00
. features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT ,
2015-06-18 16:24:37 +05:30
. set_state_shutdown = sun4i_clkevt_shutdown ,
. set_state_periodic = sun4i_clkevt_set_periodic ,
. set_state_oneshot = sun4i_clkevt_set_oneshot ,
. tick_resume = sun4i_clkevt_shutdown ,
2013-03-24 11:49:25 +01:00
. set_next_event = sun4i_clkevt_next_event ,
2012-11-12 15:07:50 +01:00
} ;
2013-03-24 11:49:25 +01:00
static irqreturn_t sun4i_timer_interrupt ( int irq , void * dev_id )
2012-11-12 15:07:50 +01:00
{
struct clock_event_device * evt = ( struct clock_event_device * ) dev_id ;
writel ( 0x1 , timer_base + TIMER_IRQ_ST_REG ) ;
evt - > event_handler ( evt ) ;
return IRQ_HANDLED ;
}
2013-03-24 11:49:25 +01:00
static struct irqaction sun4i_timer_irq = {
. name = " sun4i_timer0 " ,
2013-10-14 21:07:48 +02:00
. flags = IRQF_TIMER | IRQF_IRQPOLL ,
2013-03-24 11:49:25 +01:00
. handler = sun4i_timer_interrupt ,
. dev_id = & sun4i_clockevent ,
2012-11-12 15:07:50 +01:00
} ;
2013-11-20 00:47:32 +01:00
static u64 notrace sun4i_timer_sched_read ( void )
2013-07-16 16:45:37 +02:00
{
return ~ readl ( timer_base + TIMER_CNTVAL_REG ( 1 ) ) ;
}
2016-06-06 17:59:09 +02:00
static int __init sun4i_timer_init ( struct device_node * node )
2012-11-12 15:07:50 +01:00
{
unsigned long rate = 0 ;
struct clk * clk ;
int ret , irq ;
u32 val ;
timer_base = of_iomap ( node , 0 ) ;
2016-06-06 17:59:09 +02:00
if ( ! timer_base ) {
pr_crit ( " Can't map registers " ) ;
return - ENXIO ;
}
2012-11-12 15:07:50 +01:00
irq = irq_of_parse_and_map ( node , 0 ) ;
2016-06-06 17:59:09 +02:00
if ( irq < = 0 ) {
pr_crit ( " Can't parse IRQ " ) ;
return - EINVAL ;
}
2012-11-12 15:07:50 +01:00
clk = of_clk_get ( node , 0 ) ;
2016-06-06 17:59:09 +02:00
if ( IS_ERR ( clk ) ) {
pr_crit ( " Can't get timer clock " ) ;
return PTR_ERR ( clk ) ;
}
ret = clk_prepare_enable ( clk ) ;
if ( ret ) {
pr_err ( " Failed to prepare clock " ) ;
return ret ;
}
2012-11-12 15:07:50 +01:00
rate = clk_get_rate ( clk ) ;
2013-07-16 16:45:37 +02:00
writel ( ~ 0 , timer_base + TIMER_INTVAL_REG ( 1 ) ) ;
writel ( TIMER_CTL_ENABLE | TIMER_CTL_RELOAD |
TIMER_CTL_CLK_SRC ( TIMER_CTL_CLK_SRC_OSC24M ) ,
timer_base + TIMER_CTL_REG ( 1 ) ) ;
2015-03-30 22:17:10 +02:00
/*
* sched_clock_register does not have priorities , and on sun6i and
* later there is a better sched_clock registered by arm_arch_timer . c
*/
if ( of_machine_is_compatible ( " allwinner,sun4i-a10 " ) | |
of_machine_is_compatible ( " allwinner,sun5i-a13 " ) | |
of_machine_is_compatible ( " allwinner,sun5i-a10s " ) )
sched_clock_register ( sun4i_timer_sched_read , 32 , rate ) ;
2016-06-06 17:59:09 +02:00
ret = clocksource_mmio_init ( timer_base + TIMER_CNTVAL_REG ( 1 ) , node - > name ,
rate , 350 , 32 , clocksource_mmio_readl_down ) ;
if ( ret ) {
pr_err ( " Failed to register clocksource " ) ;
return ret ;
}
2013-07-16 16:45:37 +02:00
2013-07-16 16:45:38 +02:00
ticks_per_jiffy = DIV_ROUND_UP ( rate , HZ ) ;
2012-11-12 15:07:50 +01:00
2013-07-16 16:45:38 +02:00
writel ( TIMER_CTL_CLK_SRC ( TIMER_CTL_CLK_SRC_OSC24M ) ,
2013-07-16 16:45:38 +02:00
timer_base + TIMER_CTL_REG ( 0 ) ) ;
2012-11-12 15:07:50 +01:00
2013-12-02 09:29:35 +00:00
/* Make sure timer is stopped before playing with interrupts */
sun4i_clkevt_time_stop ( 0 ) ;
2014-11-18 23:59:33 +01:00
sun4i_clockevent . cpumask = cpu_possible_mask ;
sun4i_clockevent . irq = irq ;
clockevents_config_and_register ( & sun4i_clockevent , rate ,
TIMER_SYNC_TICKS , 0xffffffff ) ;
2013-03-24 11:49:25 +01:00
ret = setup_irq ( irq , & sun4i_timer_irq ) ;
2016-06-06 17:59:09 +02:00
if ( ret ) {
pr_err ( " failed to setup irq %d \n " , irq ) ;
return ret ;
}
2012-11-12 15:07:50 +01:00
/* Enable timer0 interrupt */
2013-03-10 17:03:46 +01:00
val = readl ( timer_base + TIMER_IRQ_EN_REG ) ;
writel ( val | TIMER_IRQ_EN ( 0 ) , timer_base + TIMER_IRQ_EN_REG ) ;
2016-06-06 17:59:09 +02:00
return ret ;
2012-11-12 15:07:50 +01:00
}
2016-06-07 00:27:44 +02:00
CLOCKSOURCE_OF_DECLARE ( sun4i , " allwinner,sun4i-a10-timer " ,
2013-03-24 11:49:25 +01:00
sun4i_timer_init ) ;