2015-01-28 02:52:42 +09:00
/*
2015-11-07 01:31:44 +09:00
* H8S TPU Driver
2015-01-28 02:52:42 +09:00
*
* Copyright 2015 Yoshinori Sato < ysato @ users . sourcefoge . jp >
*
*/
# include <linux/errno.h>
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/clocksource.h>
# include <linux/clk.h>
# include <linux/io.h>
# include <linux/of.h>
2015-11-07 01:31:44 +09:00
# include <linux/of_address.h>
# include <linux/of_irq.h>
2015-01-28 02:52:42 +09:00
2015-11-07 14:18:51 +01:00
# define TCR 0x0
# define TSR 0x5
# define TCNT 0x6
2015-01-28 02:52:42 +09:00
struct tpu_priv {
struct clocksource cs ;
unsigned long mapbase1 ;
unsigned long mapbase2 ;
raw_spinlock_t lock ;
unsigned int cs_enabled ;
} ;
static inline unsigned long read_tcnt32 ( struct tpu_priv * p )
{
unsigned long tcnt ;
tcnt = ctrl_inw ( p - > mapbase1 + TCNT ) < < 16 ;
tcnt | = ctrl_inw ( p - > mapbase2 + TCNT ) ;
return tcnt ;
}
static int tpu_get_counter ( struct tpu_priv * p , unsigned long long * val )
{
unsigned long v1 , v2 , v3 ;
int o1 , o2 ;
o1 = ctrl_inb ( p - > mapbase1 + TSR ) & 0x10 ;
/* Make sure the timer value is stable. Stolen from acpi_pm.c */
do {
o2 = o1 ;
v1 = read_tcnt32 ( p ) ;
v2 = read_tcnt32 ( p ) ;
v3 = read_tcnt32 ( p ) ;
o1 = ctrl_inb ( p - > mapbase1 + TSR ) & 0x10 ;
} while ( unlikely ( ( o1 ! = o2 ) | | ( v1 > v2 & & v1 < v3 )
| | ( v2 > v3 & & v2 < v1 ) | | ( v3 > v1 & & v3 < v2 ) ) ) ;
* val = v2 ;
return o1 ;
}
static inline struct tpu_priv * cs_to_priv ( struct clocksource * cs )
{
return container_of ( cs , struct tpu_priv , cs ) ;
}
static cycle_t tpu_clocksource_read ( struct clocksource * cs )
{
struct tpu_priv * p = cs_to_priv ( cs ) ;
unsigned long flags ;
unsigned long long value ;
raw_spin_lock_irqsave ( & p - > lock , flags ) ;
if ( tpu_get_counter ( p , & value ) )
value + = 0x100000000 ;
raw_spin_unlock_irqrestore ( & p - > lock , flags ) ;
return value ;
}
static int tpu_clocksource_enable ( struct clocksource * cs )
{
struct tpu_priv * p = cs_to_priv ( cs ) ;
WARN_ON ( p - > cs_enabled ) ;
ctrl_outw ( 0 , p - > mapbase1 + TCNT ) ;
ctrl_outw ( 0 , p - > mapbase2 + TCNT ) ;
ctrl_outb ( 0x0f , p - > mapbase1 + TCR ) ;
ctrl_outb ( 0x03 , p - > mapbase2 + TCR ) ;
p - > cs_enabled = true ;
return 0 ;
}
static void tpu_clocksource_disable ( struct clocksource * cs )
{
struct tpu_priv * p = cs_to_priv ( cs ) ;
WARN_ON ( ! p - > cs_enabled ) ;
ctrl_outb ( 0 , p - > mapbase1 + TCR ) ;
ctrl_outb ( 0 , p - > mapbase2 + TCR ) ;
p - > cs_enabled = false ;
}
2015-11-07 01:31:44 +09:00
static struct tpu_priv tpu_priv = {
. cs = {
. name = " H8S_TPU " ,
. rating = 200 ,
. read = tpu_clocksource_read ,
. enable = tpu_clocksource_enable ,
. disable = tpu_clocksource_disable ,
. mask = CLOCKSOURCE_MASK ( sizeof ( unsigned long ) * 8 ) ,
. flags = CLOCK_SOURCE_IS_CONTINUOUS ,
} ,
} ;
2015-01-28 02:52:42 +09:00
# define CH_L 0
# define CH_H 1
2015-11-07 01:31:44 +09:00
static void __init h8300_tpu_init ( struct device_node * node )
2015-01-28 02:52:42 +09:00
{
2015-11-07 01:31:44 +09:00
void __iomem * base [ 2 ] ;
struct clk * clk ;
2015-01-28 02:52:42 +09:00
2015-11-07 01:31:44 +09:00
clk = of_clk_get ( node , 0 ) ;
if ( IS_ERR ( clk ) ) {
pr_err ( " failed to get clock for clocksource \n " ) ;
return ;
2015-01-28 02:52:42 +09:00
}
2015-11-07 01:31:44 +09:00
base [ CH_L ] = of_iomap ( node , CH_L ) ;
if ( ! base [ CH_L ] ) {
pr_err ( " failed to map registers for clocksource \n " ) ;
goto free_clk ;
2015-01-28 02:52:42 +09:00
}
2015-11-07 01:31:44 +09:00
base [ CH_H ] = of_iomap ( node , CH_H ) ;
if ( ! base [ CH_H ] ) {
pr_err ( " failed to map registers for clocksource \n " ) ;
goto unmap_L ;
2015-01-28 02:52:42 +09:00
}
2015-11-07 01:31:44 +09:00
tpu_priv . mapbase1 = ( unsigned long ) base [ CH_L ] ;
tpu_priv . mapbase2 = ( unsigned long ) base [ CH_H ] ;
2015-01-28 02:52:42 +09:00
2015-11-07 01:31:44 +09:00
clocksource_register_hz ( & tpu_priv . cs , clk_get_rate ( clk ) / 64 ) ;
2015-01-28 02:52:42 +09:00
2015-11-07 01:31:44 +09:00
return ;
2015-01-28 02:52:42 +09:00
2015-11-07 01:31:44 +09:00
unmap_L :
iounmap ( base [ CH_H ] ) ;
free_clk :
clk_put ( clk ) ;
2015-01-28 02:52:42 +09:00
}
2015-11-07 01:31:44 +09:00
CLOCKSOURCE_OF_DECLARE ( h8300_tpu , " renesas,tpu " , h8300_tpu_init ) ;