2006-01-09 17:05:41 +00:00
/*
2007-02-05 11:42:07 +01:00
* linux / arch / arm / mach - at91 / at91rm9200_time . c
2006-01-09 17:05:41 +00:00
*
* Copyright ( C ) 2003 SAN People
* Copyright ( C ) 2003 ATMEL
*
* 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 . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
2007-07-31 01:41:26 +01:00
# include <linux/kernel.h>
2006-01-09 17:05:41 +00:00
# include <linux/interrupt.h>
2006-07-01 23:01:50 +01:00
# include <linux/irq.h>
2007-07-31 01:41:26 +01:00
# include <linux/clockchips.h>
2012-04-04 19:15:15 +02:00
# include <linux/export.h>
2006-01-09 17:05:41 +00:00
# include <asm/mach/time.h>
2008-08-05 16:14:15 +01:00
# include <mach/at91_st.h>
2006-11-30 17:16:43 +01:00
2006-06-19 15:23:41 +01:00
static unsigned long last_crtr ;
2007-07-31 01:41:26 +01:00
static u32 irqmask ;
static struct clock_event_device clkevt ;
2006-06-19 15:23:41 +01:00
2011-10-16 18:17:09 +08:00
# define RM9200_TIMER_LATCH ((AT91_SLOW_CLOCK + HZ / 2) / HZ)
2006-01-09 17:05:41 +00:00
/*
2007-07-31 01:41:26 +01:00
* The ST_CRTR is updated asynchronously to the master clock . . . but
* the updates as seen by the CPU don ' t seem to be strictly monotonic .
* Waiting until we read the same value twice avoids glitching .
2006-01-09 17:05:41 +00:00
*/
2007-07-31 01:41:26 +01:00
static inline unsigned long read_CRTR ( void )
{
2006-01-09 17:05:41 +00:00
unsigned long x1 , x2 ;
2012-02-20 11:07:39 +01:00
x1 = at91_st_read ( AT91_ST_CRTR ) ;
2006-01-09 17:05:41 +00:00
do {
2012-02-20 11:07:39 +01:00
x2 = at91_st_read ( AT91_ST_CRTR ) ;
2007-07-31 01:41:26 +01:00
if ( x1 = = x2 )
break ;
x1 = x2 ;
} while ( 1 ) ;
2006-01-09 17:05:41 +00:00
return x1 ;
}
/*
* IRQ handler for the timer .
*/
2006-10-06 10:53:39 -07:00
static irqreturn_t at91rm9200_timer_interrupt ( int irq , void * dev_id )
2006-01-09 17:05:41 +00:00
{
2012-02-20 11:07:39 +01:00
u32 sr = at91_st_read ( AT91_ST_SR ) & irqmask ;
2006-01-09 17:05:41 +00:00
2009-09-21 09:30:09 +02:00
/*
* irqs should be disabled here , but as the irq is shared they are only
* guaranteed to be off if the timer irq is registered first .
*/
WARN_ON_ONCE ( ! irqs_disabled ( ) ) ;
2007-07-31 01:41:26 +01:00
/* simulate "oneshot" timer with alarm */
if ( sr & AT91_ST_ALMS ) {
clkevt . event_handler ( & clkevt ) ;
return IRQ_HANDLED ;
}
2006-01-09 17:05:41 +00:00
2007-07-31 01:41:26 +01:00
/* periodic mode should handle delayed ticks */
if ( sr & AT91_ST_PITS ) {
u32 crtr = read_CRTR ( ) ;
2006-01-09 17:05:41 +00:00
2011-10-16 18:17:09 +08:00
while ( ( ( crtr - last_crtr ) & AT91_ST_CRTV ) > = RM9200_TIMER_LATCH ) {
last_crtr + = RM9200_TIMER_LATCH ;
2007-07-31 01:41:26 +01:00
clkevt . event_handler ( & clkevt ) ;
}
2006-01-09 17:05:41 +00:00
return IRQ_HANDLED ;
}
2007-07-31 01:41:26 +01:00
/* this irq is shared ... */
return IRQ_NONE ;
2006-01-09 17:05:41 +00:00
}
static struct irqaction at91rm9200_timer_irq = {
. name = " at91_tick " ,
2007-05-08 00:35:39 -07:00
. flags = IRQF_SHARED | IRQF_DISABLED | IRQF_TIMER | IRQF_IRQPOLL ,
2006-01-09 17:05:41 +00:00
. handler = at91rm9200_timer_interrupt
} ;
2009-04-21 12:24:00 -07:00
static cycle_t read_clk32k ( struct clocksource * cs )
2006-06-19 15:26:50 +01:00
{
2007-07-31 01:41:26 +01:00
return read_CRTR ( ) ;
}
2006-06-19 15:26:50 +01:00
2007-07-31 01:41:26 +01:00
static struct clocksource clk32k = {
. name = " 32k_counter " ,
. rating = 150 ,
. read = read_clk32k ,
. mask = CLOCKSOURCE_MASK ( 20 ) ,
. flags = CLOCK_SOURCE_IS_CONTINUOUS ,
} ;
static void
clkevt32k_mode ( enum clock_event_mode mode , struct clock_event_device * dev )
{
/* Disable and flush pending timer interrupts */
2012-02-20 11:07:39 +01:00
at91_st_write ( AT91_ST_IDR , AT91_ST_PITS | AT91_ST_ALMS ) ;
2012-02-20 11:13:13 +01:00
at91_st_read ( AT91_ST_SR ) ;
2006-06-19 15:26:50 +01:00
2007-07-31 01:41:26 +01:00
last_crtr = read_CRTR ( ) ;
switch ( mode ) {
case CLOCK_EVT_MODE_PERIODIC :
/* PIT for periodic irqs; fixed rate of 1/HZ */
irqmask = AT91_ST_PITS ;
2012-02-20 11:07:39 +01:00
at91_st_write ( AT91_ST_PIMR , RM9200_TIMER_LATCH ) ;
2007-07-31 01:41:26 +01:00
break ;
case CLOCK_EVT_MODE_ONESHOT :
/* ALM for oneshot irqs, set by next_event()
* before 32 seconds have passed
*/
irqmask = AT91_ST_ALMS ;
2012-02-20 11:07:39 +01:00
at91_st_write ( AT91_ST_RTAR , last_crtr ) ;
2007-07-31 01:41:26 +01:00
break ;
case CLOCK_EVT_MODE_SHUTDOWN :
case CLOCK_EVT_MODE_UNUSED :
case CLOCK_EVT_MODE_RESUME :
irqmask = 0 ;
break ;
}
2012-02-20 11:07:39 +01:00
at91_st_write ( AT91_ST_IER , irqmask ) ;
2007-07-31 01:41:26 +01:00
}
2006-06-19 15:26:50 +01:00
2007-07-31 01:41:26 +01:00
static int
clkevt32k_next_event ( unsigned long delta , struct clock_event_device * dev )
{
u32 alm ;
int status = 0 ;
BUG_ON ( delta < 2 ) ;
/* The alarm IRQ uses absolute time (now+delta), not the relative
* time ( delta ) in our calling convention . Like all clockevents
* using such " match " hardware , we have a race to defend against .
*
* Our defense here is to have set up the clockevent device so the
* delta is at least two . That way we never end up writing RTAR
* with the value then held in CRTR . . . which would mean the match
* wouldn ' t trigger until 32 seconds later , after CRTR wraps .
*/
alm = read_CRTR ( ) ;
/* Cancel any pending alarm; flush any pending IRQ */
2012-02-20 11:07:39 +01:00
at91_st_write ( AT91_ST_RTAR , alm ) ;
2012-02-20 11:13:13 +01:00
at91_st_read ( AT91_ST_SR ) ;
2006-12-01 10:15:04 +01:00
2007-07-31 01:41:26 +01:00
/* Schedule alarm by writing RTAR. */
alm + = delta ;
2012-02-20 11:07:39 +01:00
at91_st_write ( AT91_ST_RTAR , alm ) ;
2007-07-31 01:41:26 +01:00
return status ;
2006-06-19 15:26:50 +01:00
}
2007-07-31 01:41:26 +01:00
static struct clock_event_device clkevt = {
. name = " at91_tick " ,
. features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT ,
. shift = 32 ,
. rating = 150 ,
. set_next_event = clkevt32k_next_event ,
. set_mode = clkevt32k_mode ,
} ;
2012-02-20 11:07:39 +01:00
void __iomem * at91_st_base ;
2012-04-04 19:15:15 +02:00
EXPORT_SYMBOL_GPL ( at91_st_base ) ;
2012-02-20 11:07:39 +01:00
void __init at91rm9200_ioremap_st ( u32 addr )
{
at91_st_base = ioremap ( addr , 256 ) ;
if ( ! at91_st_base )
panic ( " Impossible to ioremap ST \n " ) ;
}
2006-01-09 17:05:41 +00:00
/*
2007-07-31 01:41:26 +01:00
* ST ( system timer ) module supports both clockevents and clocksource .
2006-01-09 17:05:41 +00:00
*/
void __init at91rm9200_timer_init ( void )
{
2007-07-31 01:41:26 +01:00
/* Disable all timer interrupts, and clear any pending ones */
2012-02-20 11:07:39 +01:00
at91_st_write ( AT91_ST_IDR ,
2007-07-31 01:41:26 +01:00
AT91_ST_PITS | AT91_ST_WDOVF | AT91_ST_RTTINC | AT91_ST_ALMS ) ;
2012-02-20 11:13:13 +01:00
at91_st_read ( AT91_ST_SR ) ;
2006-01-09 17:05:41 +00:00
2006-06-19 15:26:50 +01:00
/* Make IRQs happen for the system timer */
2006-01-09 17:05:41 +00:00
setup_irq ( AT91_ID_SYS , & at91rm9200_timer_irq ) ;
2007-07-31 01:41:26 +01:00
/* The 32KiHz "Slow Clock" (tick every 30517.58 nanoseconds) is used
* directly for the clocksource and all clockevents , after adjusting
* its prescaler from the 1 Hz default .
*/
2012-02-20 11:07:39 +01:00
at91_st_write ( AT91_ST_RTMR , 1 ) ;
2006-01-09 17:05:41 +00:00
2007-07-31 01:41:26 +01:00
/* Setup timer clockevent, with minimum of two ticks (important!!) */
clkevt . mult = div_sc ( AT91_SLOW_CLOCK , NSEC_PER_SEC , clkevt . shift ) ;
clkevt . max_delta_ns = clockevent_delta2ns ( AT91_ST_ALMV , & clkevt ) ;
clkevt . min_delta_ns = clockevent_delta2ns ( 2 , & clkevt ) + 1 ;
2008-12-13 21:20:26 +10:30
clkevt . cpumask = cpumask_of ( 0 ) ;
2007-07-31 01:41:26 +01:00
clockevents_register_device ( & clkevt ) ;
2006-06-19 15:26:50 +01:00
2007-07-31 01:41:26 +01:00
/* register clocksource */
2010-12-13 13:14:55 +00:00
clocksource_register_hz ( & clk32k , AT91_SLOW_CLOCK ) ;
2006-01-09 17:05:41 +00:00
}
struct sys_timer at91rm9200_timer = {
. init = at91rm9200_timer_init ,
} ;
2006-06-19 15:26:50 +01:00