2019-05-30 02:57:55 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2015-09-29 21:55:33 +03:00
/**
* timer - ti - 32 k . c - OMAP2 32 k Timer Support
*
* Copyright ( C ) 2009 Nokia Corporation
*
* Update to use new clocksource / clockevent layers
* Author : Kevin Hilman , MontaVista Software , Inc . < source @ mvista . com >
* Copyright ( C ) 2007 MontaVista Software , Inc .
*
* Original driver :
* Copyright ( C ) 2005 Nokia Corporation
* Author : Paul Mundt < paul . mundt @ nokia . com >
* Juha Yrjölä < juha . yrjola @ nokia . com >
* OMAP Dual - mode timer framework support by Timo Teras
*
* Some parts based off of TI ' s 24 xx code :
*
* Copyright ( C ) 2004 - 2009 Texas Instruments , Inc .
*
* Roughly modelled after the OMAP1 MPU timer code .
* Added OMAP4 support - Santosh Shilimkar < santosh . shilimkar @ ti . com >
*
* Copyright ( C ) 2015 Texas Instruments Incorporated - http : //www.ti.com
*/
# include <linux/init.h>
# include <linux/time.h>
# include <linux/sched_clock.h>
# include <linux/clocksource.h>
# include <linux/of.h>
# include <linux/of_address.h>
/*
* 32 KHz clocksource . . . always available , on pretty most chips except
* OMAP 730 and 1510. Other timers could be used as clocksources , with
* higher resolution in free - running counter modes ( e . g . 12 MHz xtal ) ,
* but systems won ' t necessarily want to spend resources that way .
*/
# define OMAP2_32KSYNCNT_REV_OFF 0x0
# define OMAP2_32KSYNCNT_REV_SCHEME (0x3 << 30)
# define OMAP2_32KSYNCNT_CR_OFF_LOW 0x10
# define OMAP2_32KSYNCNT_CR_OFF_HIGH 0x30
struct ti_32k {
void __iomem * base ;
void __iomem * counter ;
struct clocksource cs ;
} ;
static inline struct ti_32k * to_ti_32k ( struct clocksource * cs )
{
return container_of ( cs , struct ti_32k , cs ) ;
}
2016-12-21 22:32:01 +03:00
static u64 notrace ti_32k_read_cycles ( struct clocksource * cs )
2015-09-29 21:55:33 +03:00
{
struct ti_32k * ti = to_ti_32k ( cs ) ;
2016-12-21 22:32:01 +03:00
return ( u64 ) readl_relaxed ( ti - > counter ) ;
2015-09-29 21:55:33 +03:00
}
static struct ti_32k ti_32k_timer = {
. cs = {
. name = " 32k_counter " ,
. rating = 250 ,
. read = ti_32k_read_cycles ,
. mask = CLOCKSOURCE_MASK ( 32 ) ,
2018-08-02 09:35:16 +03:00
. flags = CLOCK_SOURCE_IS_CONTINUOUS ,
2015-09-29 21:55:33 +03:00
} ,
} ;
static u64 notrace omap_32k_read_sched_clock ( void )
{
return ti_32k_read_cycles ( & ti_32k_timer . cs ) ;
}
2016-06-07 00:29:03 +03:00
static int __init ti_32k_timer_init ( struct device_node * np )
2015-09-29 21:55:33 +03:00
{
int ret ;
ti_32k_timer . base = of_iomap ( np , 0 ) ;
if ( ! ti_32k_timer . base ) {
pr_err ( " Can't ioremap 32k timer base \n " ) ;
2016-06-07 00:29:03 +03:00
return - ENXIO ;
2015-09-29 21:55:33 +03:00
}
2018-08-08 16:14:59 +03:00
if ( ! of_machine_is_compatible ( " ti,am43 " ) )
ti_32k_timer . cs . flags | = CLOCK_SOURCE_SUSPEND_NONSTOP ;
2015-09-29 21:55:33 +03:00
ti_32k_timer . counter = ti_32k_timer . base ;
/*
* 32 k sync Counter IP register offsets vary between the highlander
* version and the legacy ones .
*
* The ' SCHEME ' bits ( 30 - 31 ) of the revision register is used to identify
* the version .
*/
if ( readl_relaxed ( ti_32k_timer . base + OMAP2_32KSYNCNT_REV_OFF ) &
OMAP2_32KSYNCNT_REV_SCHEME )
ti_32k_timer . counter + = OMAP2_32KSYNCNT_CR_OFF_HIGH ;
else
ti_32k_timer . counter + = OMAP2_32KSYNCNT_CR_OFF_LOW ;
ret = clocksource_register_hz ( & ti_32k_timer . cs , 32768 ) ;
if ( ret ) {
pr_err ( " 32k_counter: can't register clocksource \n " ) ;
2016-06-07 00:29:03 +03:00
return ret ;
2015-09-29 21:55:33 +03:00
}
sched_clock_register ( omap_32k_read_sched_clock , 32 , 32768 ) ;
pr_info ( " OMAP clocksource: 32k_counter at 32768 Hz \n " ) ;
2016-06-07 00:29:03 +03:00
return 0 ;
2015-09-29 21:55:33 +03:00
}
2017-05-26 17:56:11 +03:00
TIMER_OF_DECLARE ( ti_32k_timer , " ti,omap-counter32k " ,
2015-09-29 21:55:33 +03:00
ti_32k_timer_init ) ;