2010-12-18 21:39:30 +08:00
/*
* Copyright ( C ) 2000 - 2001 Deep Blue Solutions
* Copyright ( C ) 2002 Shane Nay ( shane @ minirl . com )
* Copyright ( C ) 2006 - 2007 Pavel Pisa ( ppisa @ pikron . com )
* Copyright ( C ) 2008 Juergen Beisert ( kernel @ pengutronix . de )
* Copyright ( C ) 2010 Freescale Semiconductor , Inc . All Rights Reserved .
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version 2
* of the License , or ( at your option ) any later version .
* 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 . , 51 Franklin Street , Fifth Floor , Boston ,
* MA 02110 - 1301 , USA .
*/
2012-04-29 00:02:37 +08:00
# include <linux/err.h>
2010-12-18 21:39:30 +08:00
# include <linux/interrupt.h>
# include <linux/irq.h>
# include <linux/clockchips.h>
# include <linux/clk.h>
2012-08-20 08:51:45 +08:00
# include <linux/of.h>
2013-03-25 14:57:41 +08:00
# include <linux/of_address.h>
2012-08-20 08:51:45 +08:00
# include <linux/of_irq.h>
2013-03-25 20:04:34 +08:00
# include <linux/stmp_device.h>
2013-06-01 23:39:40 -07:00
# include <linux/sched_clock.h>
2010-12-18 21:39:30 +08:00
# include <asm/mach/time.h>
/*
* There are 2 versions of the timrot on Freescale MXS - based SoCs .
* The v1 on MX23 only gets 16 bits counter , while v2 on MX28
* extends the counter to 32 bits .
*
* The implementation uses two timers , one for clock_event and
* another for clocksource . MX28 uses timrot 0 and 1 , while MX23
* uses 0 and 2.
*/
# define MX23_TIMROT_VERSION_OFFSET 0x0a0
# define MX28_TIMROT_VERSION_OFFSET 0x120
# define BP_TIMROT_MAJOR_VERSION 24
# define BV_TIMROT_VERSION_1 0x01
# define BV_TIMROT_VERSION_2 0x02
# define timrot_is_v1() (timrot_major_version == BV_TIMROT_VERSION_1)
/*
* There are 4 registers for each timrotv2 instance , and 2 registers
* for each timrotv1 . So address step 0x40 in macros below strides
* one instance of timrotv2 while two instances of timrotv1 .
*
* As the result , HW_TIMROT_XXXn ( 1 ) defines the address of timrot1
* on MX28 while timrot2 on MX23 .
*/
/* common between v1 and v2 */
# define HW_TIMROT_ROTCTRL 0x00
# define HW_TIMROT_TIMCTRLn(n) (0x20 + (n) * 0x40)
/* v1 only */
# define HW_TIMROT_TIMCOUNTn(n) (0x30 + (n) * 0x40)
/* v2 only */
# define HW_TIMROT_RUNNING_COUNTn(n) (0x30 + (n) * 0x40)
# define HW_TIMROT_FIXED_COUNTn(n) (0x40 + (n) * 0x40)
# define BM_TIMROT_TIMCTRLn_RELOAD (1 << 6)
# define BM_TIMROT_TIMCTRLn_UPDATE (1 << 7)
# define BM_TIMROT_TIMCTRLn_IRQ_EN (1 << 14)
# define BM_TIMROT_TIMCTRLn_IRQ (1 << 15)
# define BP_TIMROT_TIMCTRLn_SELECT 0
2012-12-21 15:06:15 +01:00
# define BV_TIMROTv1_TIMCTRLn_SELECT__32KHZ_XTAL 0x8
# define BV_TIMROTv2_TIMCTRLn_SELECT__32KHZ_XTAL 0xb
# define BV_TIMROTv2_TIMCTRLn_SELECT__TICK_ALWAYS 0xf
2010-12-18 21:39:30 +08:00
static struct clock_event_device mxs_clockevent_device ;
static enum clock_event_mode mxs_clockevent_mode = CLOCK_EVT_MODE_UNUSED ;
2013-03-25 14:57:41 +08:00
static void __iomem * mxs_timrot_base ;
2010-12-18 21:39:30 +08:00
static u32 timrot_major_version ;
static inline void timrot_irq_disable ( void )
{
2013-03-25 20:04:34 +08:00
__raw_writel ( BM_TIMROT_TIMCTRLn_IRQ_EN , mxs_timrot_base +
HW_TIMROT_TIMCTRLn ( 0 ) + STMP_OFFSET_REG_CLR ) ;
2010-12-18 21:39:30 +08:00
}
static inline void timrot_irq_enable ( void )
{
2013-03-25 20:04:34 +08:00
__raw_writel ( BM_TIMROT_TIMCTRLn_IRQ_EN , mxs_timrot_base +
HW_TIMROT_TIMCTRLn ( 0 ) + STMP_OFFSET_REG_SET ) ;
2010-12-18 21:39:30 +08:00
}
static void timrot_irq_acknowledge ( void )
{
2013-03-25 20:04:34 +08:00
__raw_writel ( BM_TIMROT_TIMCTRLn_IRQ , mxs_timrot_base +
HW_TIMROT_TIMCTRLn ( 0 ) + STMP_OFFSET_REG_CLR ) ;
2010-12-18 21:39:30 +08:00
}
static cycle_t timrotv1_get_cycles ( struct clocksource * cs )
{
return ~ ( ( __raw_readl ( mxs_timrot_base + HW_TIMROT_TIMCOUNTn ( 1 ) )
& 0xffff0000 ) > > 16 ) ;
}
static int timrotv1_set_next_event ( unsigned long evt ,
struct clock_event_device * dev )
{
/* timrot decrements the count */
__raw_writel ( evt , mxs_timrot_base + HW_TIMROT_TIMCOUNTn ( 0 ) ) ;
return 0 ;
}
static int timrotv2_set_next_event ( unsigned long evt ,
struct clock_event_device * dev )
{
/* timrot decrements the count */
__raw_writel ( evt , mxs_timrot_base + HW_TIMROT_FIXED_COUNTn ( 0 ) ) ;
return 0 ;
}
static irqreturn_t mxs_timer_interrupt ( int irq , void * dev_id )
{
struct clock_event_device * evt = dev_id ;
timrot_irq_acknowledge ( ) ;
evt - > event_handler ( evt ) ;
return IRQ_HANDLED ;
}
static struct irqaction mxs_timer_irq = {
. name = " MXS Timer Tick " ,
. dev_id = & mxs_clockevent_device ,
. flags = IRQF_TIMER | IRQF_IRQPOLL ,
. handler = mxs_timer_interrupt ,
} ;
# ifdef DEBUG
static const char * clock_event_mode_label [ ] const = {
[ CLOCK_EVT_MODE_PERIODIC ] = " CLOCK_EVT_MODE_PERIODIC " ,
[ CLOCK_EVT_MODE_ONESHOT ] = " CLOCK_EVT_MODE_ONESHOT " ,
[ CLOCK_EVT_MODE_SHUTDOWN ] = " CLOCK_EVT_MODE_SHUTDOWN " ,
[ CLOCK_EVT_MODE_UNUSED ] = " CLOCK_EVT_MODE_UNUSED "
} ;
# endif /* DEBUG */
static void mxs_set_mode ( enum clock_event_mode mode ,
struct clock_event_device * evt )
{
/* Disable interrupt in timer module */
timrot_irq_disable ( ) ;
if ( mode ! = mxs_clockevent_mode ) {
/* Set event time into the furthest future */
if ( timrot_is_v1 ( ) )
__raw_writel ( 0xffff ,
mxs_timrot_base + HW_TIMROT_TIMCOUNTn ( 1 ) ) ;
else
__raw_writel ( 0xffffffff ,
mxs_timrot_base + HW_TIMROT_FIXED_COUNTn ( 1 ) ) ;
/* Clear pending interrupt */
timrot_irq_acknowledge ( ) ;
}
# ifdef DEBUG
pr_info ( " %s: changing mode from %s to %s \n " , __func__ ,
clock_event_mode_label [ mxs_clockevent_mode ] ,
clock_event_mode_label [ mode ] ) ;
# endif /* DEBUG */
/* Remember timer mode */
mxs_clockevent_mode = mode ;
switch ( mode ) {
case CLOCK_EVT_MODE_PERIODIC :
pr_err ( " %s: Periodic mode is not implemented \n " , __func__ ) ;
break ;
case CLOCK_EVT_MODE_ONESHOT :
timrot_irq_enable ( ) ;
break ;
case CLOCK_EVT_MODE_SHUTDOWN :
case CLOCK_EVT_MODE_UNUSED :
case CLOCK_EVT_MODE_RESUME :
/* Left event sources disabled, no more interrupts appear */
break ;
}
}
static struct clock_event_device mxs_clockevent_device = {
. name = " mxs_timrot " ,
. features = CLOCK_EVT_FEAT_ONESHOT ,
. set_mode = mxs_set_mode ,
. set_next_event = timrotv2_set_next_event ,
. rating = 200 ,
} ;
static int __init mxs_clockevent_init ( struct clk * timer_clk )
{
2013-01-12 11:50:05 +00:00
if ( timrot_is_v1 ( ) )
2010-12-18 21:39:30 +08:00
mxs_clockevent_device . set_next_event = timrotv1_set_next_event ;
2013-01-12 11:50:05 +00:00
mxs_clockevent_device . cpumask = cpumask_of ( 0 ) ;
clockevents_config_and_register ( & mxs_clockevent_device ,
2012-12-21 15:06:16 +01:00
clk_get_rate ( timer_clk ) ,
timrot_is_v1 ( ) ? 0xf : 0x2 ,
2013-01-12 11:50:05 +00:00
timrot_is_v1 ( ) ? 0xfffe : 0xfffffffe ) ;
2010-12-18 21:39:30 +08:00
return 0 ;
}
static struct clocksource clocksource_mxs = {
. name = " mxs_timer " ,
. rating = 200 ,
2011-05-08 17:21:49 +01:00
. read = timrotv1_get_cycles ,
. mask = CLOCKSOURCE_MASK ( 16 ) ,
2010-12-18 21:39:30 +08:00
. flags = CLOCK_SOURCE_IS_CONTINUOUS ,
} ;
2012-11-08 23:39:14 +01:00
static u32 notrace mxs_read_sched_clock_v2 ( void )
{
return ~ readl_relaxed ( mxs_timrot_base + HW_TIMROT_RUNNING_COUNTn ( 1 ) ) ;
}
2010-12-18 21:39:30 +08:00
static int __init mxs_clocksource_init ( struct clk * timer_clk )
{
unsigned int c = clk_get_rate ( timer_clk ) ;
2011-05-08 17:21:49 +01:00
if ( timrot_is_v1 ( ) )
clocksource_register_hz ( & clocksource_mxs , c ) ;
2012-11-08 23:39:14 +01:00
else {
2011-05-08 17:21:49 +01:00
clocksource_mmio_init ( mxs_timrot_base + HW_TIMROT_RUNNING_COUNTn ( 1 ) ,
" mxs_timer " , c , 200 , 32 , clocksource_mmio_readl_down ) ;
2012-11-08 23:39:14 +01:00
setup_sched_clock ( mxs_read_sched_clock_v2 , 32 , c ) ;
}
2010-12-18 21:39:30 +08:00
return 0 ;
}
2013-03-25 14:53:08 +08:00
static void __init mxs_timer_init ( struct device_node * np )
2010-12-18 21:39:30 +08:00
{
2012-04-29 00:02:41 +08:00
struct clk * timer_clk ;
2012-08-20 08:51:45 +08:00
int irq ;
2013-03-25 14:57:41 +08:00
mxs_timrot_base = of_iomap ( np , 0 ) ;
WARN_ON ( ! mxs_timrot_base ) ;
2013-03-25 22:57:14 +08:00
timer_clk = of_clk_get ( np , 0 ) ;
2012-04-29 00:02:41 +08:00
if ( IS_ERR ( timer_clk ) ) {
pr_err ( " %s: failed to get clk \n " , __func__ ) ;
return ;
2012-04-29 00:02:37 +08:00
}
2011-12-20 13:50:11 +08:00
clk_prepare_enable ( timer_clk ) ;
2010-12-18 21:39:30 +08:00
/*
* Initialize timers to a known state
*/
2013-03-25 20:04:34 +08:00
stmp_reset_block ( mxs_timrot_base + HW_TIMROT_ROTCTRL ) ;
2010-12-18 21:39:30 +08:00
/* get timrot version */
timrot_major_version = __raw_readl ( mxs_timrot_base +
2013-03-25 19:41:39 +08:00
( of_device_is_compatible ( np , " fsl,imx23-timrot " ) ?
MX23_TIMROT_VERSION_OFFSET :
2010-12-18 21:39:30 +08:00
MX28_TIMROT_VERSION_OFFSET ) ) ;
timrot_major_version > > = BP_TIMROT_MAJOR_VERSION ;
/* one for clock_event */
__raw_writel ( ( timrot_is_v1 ( ) ?
BV_TIMROTv1_TIMCTRLn_SELECT__32KHZ_XTAL :
2012-12-21 15:06:15 +01:00
BV_TIMROTv2_TIMCTRLn_SELECT__TICK_ALWAYS ) |
2010-12-18 21:39:30 +08:00
BM_TIMROT_TIMCTRLn_UPDATE |
BM_TIMROT_TIMCTRLn_IRQ_EN ,
mxs_timrot_base + HW_TIMROT_TIMCTRLn ( 0 ) ) ;
/* another for clocksource */
__raw_writel ( ( timrot_is_v1 ( ) ?
BV_TIMROTv1_TIMCTRLn_SELECT__32KHZ_XTAL :
2012-12-21 15:06:15 +01:00
BV_TIMROTv2_TIMCTRLn_SELECT__TICK_ALWAYS ) |
2010-12-18 21:39:30 +08:00
BM_TIMROT_TIMCTRLn_RELOAD ,
mxs_timrot_base + HW_TIMROT_TIMCTRLn ( 1 ) ) ;
/* set clocksource timer fixed count to the maximum */
if ( timrot_is_v1 ( ) )
__raw_writel ( 0xffff ,
mxs_timrot_base + HW_TIMROT_TIMCOUNTn ( 1 ) ) ;
else
__raw_writel ( 0xffffffff ,
mxs_timrot_base + HW_TIMROT_FIXED_COUNTn ( 1 ) ) ;
/* init and register the timer to the framework */
mxs_clocksource_init ( timer_clk ) ;
mxs_clockevent_init ( timer_clk ) ;
/* Make irqs happen */
2012-08-20 08:51:45 +08:00
irq = irq_of_parse_and_map ( np , 0 ) ;
2010-12-18 21:39:30 +08:00
setup_irq ( irq , & mxs_timer_irq ) ;
}
2013-04-02 13:30:32 -03:00
CLOCKSOURCE_OF_DECLARE ( mxs , " fsl,timrot " , mxs_timer_init ) ;