2007-10-25 01:34:09 +09:00
/*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
*
* Based on linux / arch / mips / kernel / cevt - r4k . c ,
2013-01-22 12:59:30 +01:00
* linux / arch / mips / jmr3927 / rbhma3100 / setup . c
2007-10-25 01:34:09 +09:00
*
* Copyright 2001 MontaVista Software Inc .
* Copyright ( C ) 2000 - 2001 Toshiba Corporation
* Copyright ( C ) 2007 MIPS Technologies , Inc .
* Copyright ( C ) 2007 Ralf Baechle < ralf @ linux - mips . org >
*/
# include <linux/init.h>
# include <linux/interrupt.h>
2010-10-07 14:08:54 +01:00
# include <linux/irq.h>
2015-03-07 10:30:30 -08:00
# include <linux/sched_clock.h>
2007-10-25 01:34:09 +09:00
# include <asm/time.h>
# include <asm/txx9tmr.h>
# define TCR_BASE (TXx9_TMTCR_CCDE | TXx9_TMTCR_CRE | TXx9_TMTCR_TMODE_ITVL)
# define TIMER_CCD 0 /* 1/2 */
# define TIMER_CLK(imclk) ((imclk) / (2 << TIMER_CCD))
2009-04-24 00:10:36 +09:00
struct txx9_clocksource {
struct clocksource cs ;
struct txx9_tmr_reg __iomem * tmrptr ;
} ;
2007-10-25 01:34:09 +09:00
2016-12-21 20:32:01 +01:00
static u64 txx9_cs_read ( struct clocksource * cs )
2007-10-25 01:34:09 +09:00
{
2009-04-24 00:10:36 +09:00
struct txx9_clocksource * txx9_cs =
container_of ( cs , struct txx9_clocksource , cs ) ;
return __raw_readl ( & txx9_cs - > tmrptr - > trr ) ;
2007-10-25 01:34:09 +09:00
}
/* Use 1 bit smaller width to use full bits in that width */
# define TXX9_CLOCKSOURCE_BITS (TXX9_TIMER_BITS - 1)
2009-04-24 00:10:36 +09:00
static struct txx9_clocksource txx9_clocksource = {
. cs = {
. name = " TXx9 " ,
. rating = 200 ,
. read = txx9_cs_read ,
. mask = CLOCKSOURCE_MASK ( TXX9_CLOCKSOURCE_BITS ) ,
. flags = CLOCK_SOURCE_IS_CONTINUOUS ,
} ,
2007-10-25 01:34:09 +09:00
} ;
2015-03-07 10:30:30 -08:00
static u64 notrace txx9_read_sched_clock ( void )
{
return __raw_readl ( & txx9_clocksource . tmrptr - > trr ) ;
}
2007-10-25 01:34:09 +09:00
void __init txx9_clocksource_init ( unsigned long baseaddr ,
unsigned int imbusclk )
{
struct txx9_tmr_reg __iomem * tmrptr ;
2010-04-26 20:23:11 -07:00
clocksource_register_hz ( & txx9_clocksource . cs , TIMER_CLK ( imbusclk ) ) ;
2007-10-25 01:34:09 +09:00
tmrptr = ioremap ( baseaddr , sizeof ( struct txx9_tmr_reg ) ) ;
__raw_writel ( TCR_BASE , & tmrptr - > tcr ) ;
__raw_writel ( 0 , & tmrptr - > tisr ) ;
__raw_writel ( TIMER_CCD , & tmrptr - > ccdr ) ;
__raw_writel ( TXx9_TMITMR_TZCE , & tmrptr - > itmr ) ;
__raw_writel ( 1 < < TXX9_CLOCKSOURCE_BITS , & tmrptr - > cpra ) ;
__raw_writel ( TCR_BASE | TXx9_TMTCR_TCE , & tmrptr - > tcr ) ;
2009-04-24 00:10:36 +09:00
txx9_clocksource . tmrptr = tmrptr ;
2015-03-07 10:30:30 -08:00
sched_clock_register ( txx9_read_sched_clock , TXX9_CLOCKSOURCE_BITS ,
TIMER_CLK ( imbusclk ) ) ;
2007-10-25 01:34:09 +09:00
}
2009-04-24 00:10:36 +09:00
struct txx9_clock_event_device {
struct clock_event_device cd ;
struct txx9_tmr_reg __iomem * tmrptr ;
} ;
2007-10-25 01:34:09 +09:00
static void txx9tmr_stop_and_clear ( struct txx9_tmr_reg __iomem * tmrptr )
{
/* stop and reset counter */
__raw_writel ( TCR_BASE , & tmrptr - > tcr ) ;
/* clear pending interrupt */
__raw_writel ( 0 , & tmrptr - > tisr ) ;
}
2015-07-06 16:42:00 +05:30
static int txx9tmr_set_state_periodic ( struct clock_event_device * evt )
2007-10-25 01:34:09 +09:00
{
2009-04-24 00:10:36 +09:00
struct txx9_clock_event_device * txx9_cd =
container_of ( evt , struct txx9_clock_event_device , cd ) ;
struct txx9_tmr_reg __iomem * tmrptr = txx9_cd - > tmrptr ;
2007-10-25 01:34:09 +09:00
txx9tmr_stop_and_clear ( tmrptr ) ;
2015-07-06 16:42:00 +05:30
__raw_writel ( TXx9_TMITMR_TIIE | TXx9_TMITMR_TZCE , & tmrptr - > itmr ) ;
/* start timer */
__raw_writel ( ( ( u64 ) ( NSEC_PER_SEC / HZ ) * evt - > mult ) > > evt - > shift ,
& tmrptr - > cpra ) ;
__raw_writel ( TCR_BASE | TXx9_TMTCR_TCE , & tmrptr - > tcr ) ;
return 0 ;
}
static int txx9tmr_set_state_oneshot ( struct clock_event_device * evt )
{
struct txx9_clock_event_device * txx9_cd =
container_of ( evt , struct txx9_clock_event_device , cd ) ;
struct txx9_tmr_reg __iomem * tmrptr = txx9_cd - > tmrptr ;
txx9tmr_stop_and_clear ( tmrptr ) ;
__raw_writel ( TXx9_TMITMR_TIIE , & tmrptr - > itmr ) ;
return 0 ;
}
static int txx9tmr_set_state_shutdown ( struct clock_event_device * evt )
{
struct txx9_clock_event_device * txx9_cd =
container_of ( evt , struct txx9_clock_event_device , cd ) ;
struct txx9_tmr_reg __iomem * tmrptr = txx9_cd - > tmrptr ;
txx9tmr_stop_and_clear ( tmrptr ) ;
__raw_writel ( 0 , & tmrptr - > itmr ) ;
return 0 ;
}
static int txx9tmr_tick_resume ( struct clock_event_device * evt )
{
struct txx9_clock_event_device * txx9_cd =
container_of ( evt , struct txx9_clock_event_device , cd ) ;
struct txx9_tmr_reg __iomem * tmrptr = txx9_cd - > tmrptr ;
txx9tmr_stop_and_clear ( tmrptr ) ;
__raw_writel ( TIMER_CCD , & tmrptr - > ccdr ) ;
__raw_writel ( 0 , & tmrptr - > itmr ) ;
return 0 ;
2007-10-25 01:34:09 +09:00
}
static int txx9tmr_set_next_event ( unsigned long delta ,
struct clock_event_device * evt )
{
2009-04-24 00:10:36 +09:00
struct txx9_clock_event_device * txx9_cd =
container_of ( evt , struct txx9_clock_event_device , cd ) ;
struct txx9_tmr_reg __iomem * tmrptr = txx9_cd - > tmrptr ;
2007-10-25 01:34:09 +09:00
txx9tmr_stop_and_clear ( tmrptr ) ;
/* start timer */
__raw_writel ( delta , & tmrptr - > cpra ) ;
__raw_writel ( TCR_BASE | TXx9_TMTCR_TCE , & tmrptr - > tcr ) ;
return 0 ;
}
2009-04-24 00:10:36 +09:00
static struct txx9_clock_event_device txx9_clock_event_device = {
. cd = {
2015-07-06 16:42:00 +05:30
. name = " TXx9 " ,
. features = CLOCK_EVT_FEAT_PERIODIC |
CLOCK_EVT_FEAT_ONESHOT ,
. rating = 200 ,
. set_state_shutdown = txx9tmr_set_state_shutdown ,
. set_state_periodic = txx9tmr_set_state_periodic ,
. set_state_oneshot = txx9tmr_set_state_oneshot ,
. tick_resume = txx9tmr_tick_resume ,
. set_next_event = txx9tmr_set_next_event ,
2009-04-24 00:10:36 +09:00
} ,
2007-10-25 01:34:09 +09:00
} ;
static irqreturn_t txx9tmr_interrupt ( int irq , void * dev_id )
{
2009-04-24 00:10:36 +09:00
struct txx9_clock_event_device * txx9_cd = dev_id ;
struct clock_event_device * cd = & txx9_cd - > cd ;
struct txx9_tmr_reg __iomem * tmrptr = txx9_cd - > tmrptr ;
2007-10-25 01:34:09 +09:00
2013-01-22 12:59:30 +01:00
__raw_writel ( 0 , & tmrptr - > tisr ) ; /* ack interrupt */
2007-10-25 01:34:09 +09:00
cd - > event_handler ( cd ) ;
return IRQ_HANDLED ;
}
void __init txx9_clockevent_init ( unsigned long baseaddr , int irq ,
unsigned int imbusclk )
{
2009-04-24 00:10:36 +09:00
struct clock_event_device * cd = & txx9_clock_event_device . cd ;
2007-10-25 01:34:09 +09:00
struct txx9_tmr_reg __iomem * tmrptr ;
tmrptr = ioremap ( baseaddr , sizeof ( struct txx9_tmr_reg ) ) ;
txx9tmr_stop_and_clear ( tmrptr ) ;
__raw_writel ( TIMER_CCD , & tmrptr - > ccdr ) ;
__raw_writel ( 0 , & tmrptr - > itmr ) ;
2009-04-24 00:10:36 +09:00
txx9_clock_event_device . tmrptr = tmrptr ;
2007-10-25 01:34:09 +09:00
clockevent_set_clock ( cd , TIMER_CLK ( imbusclk ) ) ;
cd - > max_delta_ns =
clockevent_delta2ns ( 0xffffffff > > ( 32 - TXX9_TIMER_BITS ) , cd ) ;
2017-03-30 21:47:32 +02:00
cd - > max_delta_ticks = 0xffffffff > > ( 32 - TXX9_TIMER_BITS ) ;
2007-10-25 01:34:09 +09:00
cd - > min_delta_ns = clockevent_delta2ns ( 0xf , cd ) ;
2017-03-30 21:47:32 +02:00
cd - > min_delta_ticks = 0xf ;
2007-10-25 01:34:09 +09:00
cd - > irq = irq ;
2020-12-16 21:08:51 +08:00
cd - > cpumask = cpumask_of ( 0 ) ;
2007-10-25 01:34:09 +09:00
clockevents_register_device ( cd ) ;
2020-03-05 17:27:53 +05:30
if ( request_irq ( irq , txx9tmr_interrupt , IRQF_PERCPU | IRQF_TIMER ,
" txx9tmr " , & txx9_clock_event_device ) )
pr_err ( " Failed to request irq %d (txx9tmr) \n " , irq ) ;
2007-10-25 01:34:09 +09:00
printk ( KERN_INFO " TXx9: clockevent device at 0x%lx, irq %d \n " ,
baseaddr , irq ) ;
}
void __init txx9_tmr_init ( unsigned long baseaddr )
{
struct txx9_tmr_reg __iomem * tmrptr ;
tmrptr = ioremap ( baseaddr , sizeof ( struct txx9_tmr_reg ) ) ;
2008-06-24 23:26:38 +09:00
/* Start once to make CounterResetEnable effective */
__raw_writel ( TXx9_TMTCR_CRE | TXx9_TMTCR_TCE , & tmrptr - > tcr ) ;
/* Stop and reset the counter */
2007-10-25 01:34:09 +09:00
__raw_writel ( TXx9_TMTCR_CRE , & tmrptr - > tcr ) ;
__raw_writel ( 0 , & tmrptr - > tisr ) ;
__raw_writel ( 0xffffffff , & tmrptr - > cpra ) ;
__raw_writel ( 0 , & tmrptr - > itmr ) ;
__raw_writel ( 0 , & tmrptr - > ccdr ) ;
__raw_writel ( 0 , & tmrptr - > pgmr ) ;
iounmap ( tmrptr ) ;
}