2015-01-25 22:06:02 +01:00
/*
* Rockchip timer support
*
* Copyright ( C ) Daniel Lezcano < daniel . lezcano @ linaro . org >
*
* 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 .
*/
# include <linux/clk.h>
# include <linux/clockchips.h>
# include <linux/init.h>
# include <linux/interrupt.h>
2017-01-31 15:43:14 +03:00
# include <linux/sched_clock.h>
# include <linux/slab.h>
2015-01-25 22:06:02 +01:00
# include <linux/of.h>
# include <linux/of_address.h>
# include <linux/of_irq.h>
# define TIMER_NAME "rk_timer"
2015-09-25 10:14:56 +08:00
# define TIMER_LOAD_COUNT0 0x00
# define TIMER_LOAD_COUNT1 0x04
2017-01-31 15:43:14 +03:00
# define TIMER_CURRENT_VALUE0 0x08
# define TIMER_CURRENT_VALUE1 0x0C
2016-06-16 16:00:08 +02:00
# define TIMER_CONTROL_REG3288 0x10
# define TIMER_CONTROL_REG3399 0x1c
2015-09-25 10:14:56 +08:00
# define TIMER_INT_STATUS 0x18
2015-01-25 22:06:02 +01:00
2015-09-25 10:14:56 +08:00
# define TIMER_DISABLE 0x0
# define TIMER_ENABLE 0x1
# define TIMER_MODE_FREE_RUNNING (0 << 1)
# define TIMER_MODE_USER_DEFINED_COUNT (1 << 1)
# define TIMER_INT_UNMASK (1 << 2)
2015-01-25 22:06:02 +01:00
2017-01-31 15:43:14 +03:00
struct rk_timer {
2015-01-25 22:06:02 +01:00
void __iomem * base ;
2016-06-16 16:00:08 +02:00
void __iomem * ctrl ;
2017-01-31 15:43:14 +03:00
struct clk * clk ;
struct clk * pclk ;
2015-01-25 22:06:02 +01:00
u32 freq ;
2017-01-31 15:43:14 +03:00
int irq ;
2015-01-25 22:06:02 +01:00
} ;
2017-01-31 15:43:14 +03:00
struct rk_clkevt {
struct clock_event_device ce ;
struct rk_timer timer ;
} ;
2015-01-25 22:06:02 +01:00
2017-01-31 15:43:14 +03:00
static struct rk_clkevt * rk_clkevt ;
static struct rk_timer * rk_clksrc ;
2015-01-25 22:06:02 +01:00
2017-01-31 15:43:14 +03:00
static inline struct rk_timer * rk_timer ( struct clock_event_device * ce )
2016-06-16 16:00:08 +02:00
{
2017-01-31 15:43:14 +03:00
return & container_of ( ce , struct rk_clkevt , ce ) - > timer ;
2016-06-16 16:00:08 +02:00
}
2017-01-31 15:43:14 +03:00
static inline void rk_timer_disable ( struct rk_timer * timer )
2015-01-25 22:06:02 +01:00
{
2017-01-31 15:43:14 +03:00
writel_relaxed ( TIMER_DISABLE , timer - > ctrl ) ;
2015-01-25 22:06:02 +01:00
}
2017-01-31 15:43:14 +03:00
static inline void rk_timer_enable ( struct rk_timer * timer , u32 flags )
2015-01-25 22:06:02 +01:00
{
2017-01-31 15:43:14 +03:00
writel_relaxed ( TIMER_ENABLE | flags , timer - > ctrl ) ;
2015-01-25 22:06:02 +01:00
}
static void rk_timer_update_counter ( unsigned long cycles ,
2017-01-31 15:43:14 +03:00
struct rk_timer * timer )
2015-01-25 22:06:02 +01:00
{
2017-01-31 15:43:14 +03:00
writel_relaxed ( cycles , timer - > base + TIMER_LOAD_COUNT0 ) ;
writel_relaxed ( 0 , timer - > base + TIMER_LOAD_COUNT1 ) ;
2015-01-25 22:06:02 +01:00
}
2017-01-31 15:43:14 +03:00
static void rk_timer_interrupt_clear ( struct rk_timer * timer )
2015-01-25 22:06:02 +01:00
{
2017-01-31 15:43:14 +03:00
writel_relaxed ( 1 , timer - > base + TIMER_INT_STATUS ) ;
2015-01-25 22:06:02 +01:00
}
static inline int rk_timer_set_next_event ( unsigned long cycles ,
struct clock_event_device * ce )
{
2017-01-31 15:43:14 +03:00
struct rk_timer * timer = rk_timer ( ce ) ;
rk_timer_disable ( timer ) ;
rk_timer_update_counter ( cycles , timer ) ;
rk_timer_enable ( timer , TIMER_MODE_USER_DEFINED_COUNT |
TIMER_INT_UNMASK ) ;
2015-01-25 22:06:02 +01:00
return 0 ;
}
2015-06-18 16:24:32 +05:30
static int rk_timer_shutdown ( struct clock_event_device * ce )
2015-01-25 22:06:02 +01:00
{
2017-01-31 15:43:14 +03:00
struct rk_timer * timer = rk_timer ( ce ) ;
rk_timer_disable ( timer ) ;
2015-06-18 16:24:32 +05:30
return 0 ;
}
static int rk_timer_set_periodic ( struct clock_event_device * ce )
{
2017-01-31 15:43:14 +03:00
struct rk_timer * timer = rk_timer ( ce ) ;
rk_timer_disable ( timer ) ;
rk_timer_update_counter ( timer - > freq / HZ - 1 , timer ) ;
rk_timer_enable ( timer , TIMER_MODE_FREE_RUNNING | TIMER_INT_UNMASK ) ;
2015-06-18 16:24:32 +05:30
return 0 ;
2015-01-25 22:06:02 +01:00
}
static irqreturn_t rk_timer_interrupt ( int irq , void * dev_id )
{
struct clock_event_device * ce = dev_id ;
2017-01-31 15:43:14 +03:00
struct rk_timer * timer = rk_timer ( ce ) ;
2015-01-25 22:06:02 +01:00
2017-01-31 15:43:14 +03:00
rk_timer_interrupt_clear ( timer ) ;
2015-01-25 22:06:02 +01:00
2015-06-18 16:24:32 +05:30
if ( clockevent_state_oneshot ( ce ) )
2017-01-31 15:43:14 +03:00
rk_timer_disable ( timer ) ;
2015-01-25 22:06:02 +01:00
ce - > event_handler ( ce ) ;
return IRQ_HANDLED ;
}
2017-01-31 15:43:14 +03:00
static u64 notrace rk_timer_sched_read ( void )
{
return ~ readl_relaxed ( rk_clksrc - > base + TIMER_CURRENT_VALUE0 ) ;
}
static int __init
rk_timer_probe ( struct rk_timer * timer , struct device_node * np )
2015-01-25 22:06:02 +01:00
{
struct clk * timer_clk ;
struct clk * pclk ;
2016-05-31 17:28:55 +02:00
int ret = - EINVAL , irq ;
2017-01-31 15:43:14 +03:00
u32 ctrl_reg = TIMER_CONTROL_REG3288 ;
2015-01-25 22:06:02 +01:00
2017-01-31 15:43:14 +03:00
timer - > base = of_iomap ( np , 0 ) ;
if ( ! timer - > base ) {
2015-01-25 22:06:02 +01:00
pr_err ( " Failed to get base address for '%s' \n " , TIMER_NAME ) ;
2016-05-31 17:28:55 +02:00
return - ENXIO ;
2015-01-25 22:06:02 +01:00
}
2017-01-31 15:43:14 +03:00
if ( of_device_is_compatible ( np , " rockchip,rk3399-timer " ) )
ctrl_reg = TIMER_CONTROL_REG3399 ;
timer - > ctrl = timer - > base + ctrl_reg ;
2015-01-25 22:06:02 +01:00
pclk = of_clk_get_by_name ( np , " pclk " ) ;
if ( IS_ERR ( pclk ) ) {
2016-05-31 17:28:55 +02:00
ret = PTR_ERR ( pclk ) ;
2015-01-25 22:06:02 +01:00
pr_err ( " Failed to get pclk for '%s' \n " , TIMER_NAME ) ;
2016-02-15 09:02:09 +08:00
goto out_unmap ;
2015-01-25 22:06:02 +01:00
}
2016-05-31 17:28:55 +02:00
ret = clk_prepare_enable ( pclk ) ;
if ( ret ) {
2015-01-25 22:06:02 +01:00
pr_err ( " Failed to enable pclk for '%s' \n " , TIMER_NAME ) ;
2016-02-15 09:02:09 +08:00
goto out_unmap ;
2015-01-25 22:06:02 +01:00
}
2017-01-31 15:43:14 +03:00
timer - > pclk = pclk ;
2015-01-25 22:06:02 +01:00
timer_clk = of_clk_get_by_name ( np , " timer " ) ;
if ( IS_ERR ( timer_clk ) ) {
2016-05-31 17:28:55 +02:00
ret = PTR_ERR ( timer_clk ) ;
2015-01-25 22:06:02 +01:00
pr_err ( " Failed to get timer clock for '%s' \n " , TIMER_NAME ) ;
2016-02-15 09:02:09 +08:00
goto out_timer_clk ;
2015-01-25 22:06:02 +01:00
}
2016-05-31 17:28:55 +02:00
ret = clk_prepare_enable ( timer_clk ) ;
if ( ret ) {
2015-01-25 22:06:02 +01:00
pr_err ( " Failed to enable timer clock \n " ) ;
2016-02-15 09:02:09 +08:00
goto out_timer_clk ;
2015-01-25 22:06:02 +01:00
}
2017-01-31 15:43:14 +03:00
timer - > clk = timer_clk ;
2015-01-25 22:06:02 +01:00
2017-01-31 15:43:14 +03:00
timer - > freq = clk_get_rate ( timer_clk ) ;
2015-01-25 22:06:02 +01:00
irq = irq_of_parse_and_map ( np , 0 ) ;
2015-09-20 07:00:10 -07:00
if ( ! irq ) {
2016-05-31 17:28:55 +02:00
ret = - EINVAL ;
2015-01-25 22:06:02 +01:00
pr_err ( " Failed to map interrupts for '%s' \n " , TIMER_NAME ) ;
2016-02-15 09:02:09 +08:00
goto out_irq ;
2015-01-25 22:06:02 +01:00
}
2017-01-31 15:43:14 +03:00
timer - > irq = irq ;
rk_timer_interrupt_clear ( timer ) ;
rk_timer_disable ( timer ) ;
return 0 ;
out_irq :
clk_disable_unprepare ( timer_clk ) ;
out_timer_clk :
clk_disable_unprepare ( pclk ) ;
out_unmap :
iounmap ( timer - > base ) ;
return ret ;
}
static void __init rk_timer_cleanup ( struct rk_timer * timer )
{
clk_disable_unprepare ( timer - > clk ) ;
clk_disable_unprepare ( timer - > pclk ) ;
iounmap ( timer - > base ) ;
}
static int __init rk_clkevt_init ( struct device_node * np )
{
struct clock_event_device * ce ;
int ret = - EINVAL ;
rk_clkevt = kzalloc ( sizeof ( struct rk_clkevt ) , GFP_KERNEL ) ;
if ( ! rk_clkevt ) {
ret = - ENOMEM ;
goto out ;
}
2015-01-25 22:06:02 +01:00
2017-01-31 15:43:14 +03:00
ret = rk_timer_probe ( & rk_clkevt - > timer , np ) ;
if ( ret )
goto out_probe ;
ce = & rk_clkevt - > ce ;
2015-01-25 22:06:02 +01:00
ce - > name = TIMER_NAME ;
2016-06-16 15:57:53 +02:00
ce - > features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT |
CLOCK_EVT_FEAT_DYNIRQ ;
2015-01-25 22:06:02 +01:00
ce - > set_next_event = rk_timer_set_next_event ;
2015-06-18 16:24:32 +05:30
ce - > set_state_shutdown = rk_timer_shutdown ;
ce - > set_state_periodic = rk_timer_set_periodic ;
2017-01-31 15:43:14 +03:00
ce - > irq = rk_clkevt - > timer . irq ;
2016-06-16 15:57:53 +02:00
ce - > cpumask = cpu_possible_mask ;
2015-01-25 22:06:02 +01:00
ce - > rating = 250 ;
2017-01-31 15:43:14 +03:00
ret = request_irq ( rk_clkevt - > timer . irq , rk_timer_interrupt , IRQF_TIMER ,
TIMER_NAME , ce ) ;
2015-01-25 22:06:02 +01:00
if ( ret ) {
2017-01-31 15:43:14 +03:00
pr_err ( " Failed to initialize '%s': %d \n " ,
TIMER_NAME , ret ) ;
2016-02-15 09:02:09 +08:00
goto out_irq ;
2015-01-25 22:06:02 +01:00
}
2017-01-31 15:43:14 +03:00
clockevents_config_and_register ( & rk_clkevt - > ce ,
rk_clkevt - > timer . freq , 1 , UINT_MAX ) ;
2016-05-31 17:28:55 +02:00
return 0 ;
2016-02-15 09:02:09 +08:00
out_irq :
2017-01-31 15:43:14 +03:00
rk_timer_cleanup ( & rk_clkevt - > timer ) ;
out_probe :
kfree ( rk_clkevt ) ;
out :
/* Leave rk_clkevt not NULL to prevent future init */
rk_clkevt = ERR_PTR ( ret ) ;
2016-05-31 17:28:55 +02:00
return ret ;
2015-01-25 22:06:02 +01:00
}
2015-09-25 10:14:56 +08:00
2017-01-31 15:43:14 +03:00
static int __init rk_clksrc_init ( struct device_node * np )
2016-06-16 16:00:08 +02:00
{
2017-01-31 15:43:14 +03:00
int ret = - EINVAL ;
rk_clksrc = kzalloc ( sizeof ( struct rk_timer ) , GFP_KERNEL ) ;
if ( ! rk_clksrc ) {
ret = - ENOMEM ;
goto out ;
}
ret = rk_timer_probe ( rk_clksrc , np ) ;
if ( ret )
goto out_probe ;
rk_timer_update_counter ( UINT_MAX , rk_clksrc ) ;
rk_timer_enable ( rk_clksrc , 0 ) ;
ret = clocksource_mmio_init ( rk_clksrc - > base + TIMER_CURRENT_VALUE0 ,
TIMER_NAME , rk_clksrc - > freq , 250 , 32 ,
clocksource_mmio_readl_down ) ;
if ( ret ) {
2017-09-25 13:46:41 +05:30
pr_err ( " Failed to register clocksource \n " ) ;
2017-01-31 15:43:14 +03:00
goto out_clocksource ;
}
sched_clock_register ( rk_timer_sched_read , 32 , rk_clksrc - > freq ) ;
return 0 ;
out_clocksource :
rk_timer_cleanup ( rk_clksrc ) ;
out_probe :
kfree ( rk_clksrc ) ;
out :
/* Leave rk_clksrc not NULL to prevent future init */
rk_clksrc = ERR_PTR ( ret ) ;
return ret ;
2016-06-16 16:00:08 +02:00
}
2017-01-31 15:43:14 +03:00
static int __init rk_timer_init ( struct device_node * np )
2016-06-16 16:00:08 +02:00
{
2017-01-31 15:43:14 +03:00
if ( ! rk_clkevt )
return rk_clkevt_init ( np ) ;
if ( ! rk_clksrc )
return rk_clksrc_init ( np ) ;
pr_err ( " Too many timer definitions for '%s' \n " , TIMER_NAME ) ;
return - EINVAL ;
2016-06-16 16:00:08 +02:00
}
2017-05-26 16:56:11 +02:00
TIMER_OF_DECLARE ( rk3288_timer , " rockchip,rk3288-timer " , rk_timer_init ) ;
TIMER_OF_DECLARE ( rk3399_timer , " rockchip,rk3399-timer " , rk_timer_init ) ;