2006-09-27 12:13:19 +04:00
/*
* SuperH On - Chip RTC Support
*
2009-04-16 09:12:22 +04:00
* Copyright ( C ) 2006 - 2009 Paul Mundt
2006-12-08 09:26:15 +03:00
* Copyright ( C ) 2006 Jamie Lenehan
2008-03-06 06:50:53 +03:00
* Copyright ( C ) 2008 Angelo Castello
2006-09-27 12:13:19 +04:00
*
* Based on the old arch / sh / kernel / cpu / rtc . c by :
*
* Copyright ( C ) 2000 Philipp Rumpf < prumpf @ tux . org >
* Copyright ( C ) 1999 Tetsuya Okada & Niibe Yutaka
*
* 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 .
*/
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/bcd.h>
# include <linux/rtc.h>
# include <linux/init.h>
# include <linux/platform_device.h>
# include <linux/seq_file.h>
# include <linux/interrupt.h>
# include <linux/spinlock.h>
2006-12-07 11:23:50 +03:00
# include <linux/io.h>
2009-01-07 01:42:12 +03:00
# include <linux/log2.h>
2009-04-16 09:12:22 +04:00
# include <linux/clk.h>
2007-08-03 09:19:58 +04:00
# include <asm/rtc.h>
2006-09-27 12:13:19 +04:00
2006-12-08 09:26:15 +03:00
# define DRV_NAME "sh-rtc"
2009-08-20 08:25:11 +04:00
# define DRV_VERSION "0.2.3"
2006-09-27 12:13:19 +04:00
# define RTC_REG(r) ((r) * rtc_reg_size)
2006-12-07 11:23:50 +03:00
# define R64CNT RTC_REG(0)
2006-12-08 09:26:15 +03:00
# define RSECCNT RTC_REG(1) /* RTC sec */
# define RMINCNT RTC_REG(2) /* RTC min */
# define RHRCNT RTC_REG(3) /* RTC hour */
# define RWKCNT RTC_REG(4) /* RTC week */
# define RDAYCNT RTC_REG(5) /* RTC day */
# define RMONCNT RTC_REG(6) /* RTC month */
# define RYRCNT RTC_REG(7) /* RTC year */
# define RSECAR RTC_REG(8) /* ALARM sec */
# define RMINAR RTC_REG(9) /* ALARM min */
# define RHRAR RTC_REG(10) /* ALARM hour */
# define RWKAR RTC_REG(11) /* ALARM week */
# define RDAYAR RTC_REG(12) /* ALARM day */
# define RMONAR RTC_REG(13) /* ALARM month */
# define RCR1 RTC_REG(14) /* Control */
# define RCR2 RTC_REG(15) /* Control */
2007-11-26 11:56:31 +03:00
/*
* Note on RYRAR and RCR3 : Up until this point most of the register
* definitions are consistent across all of the available parts . However ,
* the placement of the optional RYRAR and RCR3 ( the RYRAR control
* register used to control RYRCNT / RYRAR compare ) varies considerably
* across various parts , occasionally being mapped in to a completely
* unrelated address space . For proper RYRAR support a separate resource
* would have to be handed off , but as this is purely optional in
* practice , we simply opt not to support it , thereby keeping the code
* quite a bit more simplified .
*/
2006-12-08 09:26:15 +03:00
/* ALARM Bits - or with BCD encoded value */
# define AR_ENB 0x80 /* Enable for alarm cmp */
2006-09-27 12:13:19 +04:00
2008-03-06 06:50:53 +03:00
/* Period Bits */
# define PF_HP 0x100 /* Enable Half Period to support 8,32,128Hz */
# define PF_COUNT 0x200 /* Half periodic counter */
# define PF_OXS 0x400 /* Periodic One x Second */
# define PF_KOU 0x800 /* Kernel or User periodic request 1=kernel */
# define PF_MASK 0xf00
2006-09-27 12:13:19 +04:00
/* RCR1 Bits */
# define RCR1_CF 0x80 /* Carry Flag */
# define RCR1_CIE 0x10 /* Carry Interrupt Enable */
# define RCR1_AIE 0x08 /* Alarm Interrupt Enable */
# define RCR1_AF 0x01 /* Alarm Flag */
/* RCR2 Bits */
# define RCR2_PEF 0x80 /* PEriodic interrupt Flag */
# define RCR2_PESMASK 0x70 /* Periodic interrupt Set */
# define RCR2_RTCEN 0x08 /* ENable RTC */
# define RCR2_ADJ 0x04 /* ADJustment (30-second) */
# define RCR2_RESET 0x02 /* Reset bit */
# define RCR2_START 0x01 /* Start bit */
struct sh_rtc {
2009-04-16 09:12:22 +04:00
void __iomem * regbase ;
unsigned long regsize ;
struct resource * res ;
int alarm_irq ;
int periodic_irq ;
int carry_irq ;
struct clk * clk ;
struct rtc_device * rtc_dev ;
spinlock_t lock ;
unsigned long capabilities ; /* See asm/rtc.h for cap bits */
unsigned short periodic_freq ;
2006-09-27 12:13:19 +04:00
} ;
2009-02-24 16:11:03 +03:00
static int __sh_rtc_interrupt ( struct sh_rtc * rtc )
2006-09-27 12:13:19 +04:00
{
2009-02-24 16:11:03 +03:00
unsigned int tmp , pending ;
2006-09-27 12:13:19 +04:00
tmp = readb ( rtc - > regbase + RCR1 ) ;
2009-02-24 16:11:03 +03:00
pending = tmp & RCR1_CF ;
2006-12-08 09:26:15 +03:00
tmp & = ~ RCR1_CF ;
2006-09-27 12:13:19 +04:00
writeb ( tmp , rtc - > regbase + RCR1 ) ;
2008-03-06 06:50:53 +03:00
/* Users have requested One x Second IRQ */
2009-02-24 16:11:03 +03:00
if ( pending & & rtc - > periodic_freq & PF_OXS )
2008-03-06 06:50:53 +03:00
rtc_update_irq ( rtc - > rtc_dev , 1 , RTC_UF | RTC_IRQF ) ;
2006-09-27 12:13:19 +04:00
2009-02-24 16:11:03 +03:00
return pending ;
2006-09-27 12:13:19 +04:00
}
2009-02-24 16:11:03 +03:00
static int __sh_rtc_alarm ( struct sh_rtc * rtc )
2006-12-08 09:26:15 +03:00
{
2009-02-24 16:11:03 +03:00
unsigned int tmp , pending ;
2006-12-08 09:26:15 +03:00
tmp = readb ( rtc - > regbase + RCR1 ) ;
2009-02-24 16:11:03 +03:00
pending = tmp & RCR1_AF ;
2008-03-06 06:50:53 +03:00
tmp & = ~ ( RCR1_AF | RCR1_AIE ) ;
2009-02-24 16:11:03 +03:00
writeb ( tmp , rtc - > regbase + RCR1 ) ;
2006-12-08 09:26:15 +03:00
2009-02-24 16:11:03 +03:00
if ( pending )
rtc_update_irq ( rtc - > rtc_dev , 1 , RTC_AF | RTC_IRQF ) ;
2008-03-06 06:50:53 +03:00
2009-02-24 16:11:03 +03:00
return pending ;
2006-12-08 09:26:15 +03:00
}
2009-02-24 16:11:03 +03:00
static int __sh_rtc_periodic ( struct sh_rtc * rtc )
2006-09-27 12:13:19 +04:00
{
2008-03-06 06:50:53 +03:00
struct rtc_device * rtc_dev = rtc - > rtc_dev ;
2009-02-24 16:11:03 +03:00
struct rtc_task * irq_task ;
unsigned int tmp , pending ;
2006-09-27 12:13:19 +04:00
2008-03-06 06:50:53 +03:00
tmp = readb ( rtc - > regbase + RCR2 ) ;
2009-02-24 16:11:03 +03:00
pending = tmp & RCR2_PEF ;
2008-03-06 06:50:53 +03:00
tmp & = ~ RCR2_PEF ;
writeb ( tmp , rtc - > regbase + RCR2 ) ;
2009-02-24 16:11:03 +03:00
if ( ! pending )
return 0 ;
2008-03-06 06:50:53 +03:00
/* Half period enabled than one skipped and the next notified */
if ( ( rtc - > periodic_freq & PF_HP ) & & ( rtc - > periodic_freq & PF_COUNT ) )
rtc - > periodic_freq & = ~ PF_COUNT ;
else {
if ( rtc - > periodic_freq & PF_HP )
rtc - > periodic_freq | = PF_COUNT ;
if ( rtc - > periodic_freq & PF_KOU ) {
spin_lock ( & rtc_dev - > irq_task_lock ) ;
2009-02-24 16:11:03 +03:00
irq_task = rtc_dev - > irq_task ;
if ( irq_task )
irq_task - > func ( irq_task - > private_data ) ;
2008-03-06 06:50:53 +03:00
spin_unlock ( & rtc_dev - > irq_task_lock ) ;
} else
rtc_update_irq ( rtc - > rtc_dev , 1 , RTC_PF | RTC_IRQF ) ;
}
2006-09-27 12:13:19 +04:00
2009-02-24 16:11:03 +03:00
return pending ;
}
static irqreturn_t sh_rtc_interrupt ( int irq , void * dev_id )
{
struct sh_rtc * rtc = dev_id ;
int ret ;
spin_lock ( & rtc - > lock ) ;
ret = __sh_rtc_interrupt ( rtc ) ;
spin_unlock ( & rtc - > lock ) ;
return IRQ_RETVAL ( ret ) ;
}
static irqreturn_t sh_rtc_alarm ( int irq , void * dev_id )
{
struct sh_rtc * rtc = dev_id ;
int ret ;
spin_lock ( & rtc - > lock ) ;
ret = __sh_rtc_alarm ( rtc ) ;
spin_unlock ( & rtc - > lock ) ;
return IRQ_RETVAL ( ret ) ;
}
static irqreturn_t sh_rtc_periodic ( int irq , void * dev_id )
{
struct sh_rtc * rtc = dev_id ;
int ret ;
spin_lock ( & rtc - > lock ) ;
ret = __sh_rtc_periodic ( rtc ) ;
2006-09-27 12:13:19 +04:00
spin_unlock ( & rtc - > lock ) ;
2009-02-24 16:11:03 +03:00
return IRQ_RETVAL ( ret ) ;
}
static irqreturn_t sh_rtc_shared ( int irq , void * dev_id )
{
struct sh_rtc * rtc = dev_id ;
int ret ;
spin_lock ( & rtc - > lock ) ;
ret = __sh_rtc_interrupt ( rtc ) ;
ret | = __sh_rtc_alarm ( rtc ) ;
ret | = __sh_rtc_periodic ( rtc ) ;
spin_unlock ( & rtc - > lock ) ;
return IRQ_RETVAL ( ret ) ;
2006-09-27 12:13:19 +04:00
}
2009-08-20 08:25:11 +04:00
static int sh_rtc_irq_set_state ( struct device * dev , int enable )
2006-09-27 12:13:19 +04:00
{
struct sh_rtc * rtc = dev_get_drvdata ( dev ) ;
unsigned int tmp ;
spin_lock_irq ( & rtc - > lock ) ;
tmp = readb ( rtc - > regbase + RCR2 ) ;
if ( enable ) {
2009-08-20 08:25:11 +04:00
rtc - > periodic_freq | = PF_KOU ;
2008-03-06 06:50:53 +03:00
tmp & = ~ RCR2_PEF ; /* Clear PES bit */
tmp | = ( rtc - > periodic_freq & ~ PF_HP ) ; /* Set PES2-0 */
2009-08-20 08:25:11 +04:00
} else {
rtc - > periodic_freq & = ~ PF_KOU ;
2006-09-27 12:13:19 +04:00
tmp & = ~ ( RCR2_PESMASK | RCR2_PEF ) ;
2009-08-20 08:25:11 +04:00
}
2006-09-27 12:13:19 +04:00
writeb ( tmp , rtc - > regbase + RCR2 ) ;
spin_unlock_irq ( & rtc - > lock ) ;
2009-08-20 08:25:11 +04:00
return 0 ;
2006-09-27 12:13:19 +04:00
}
2009-08-20 08:25:11 +04:00
static int sh_rtc_irq_set_freq ( struct device * dev , int freq )
2006-09-27 12:13:19 +04:00
{
struct sh_rtc * rtc = dev_get_drvdata ( dev ) ;
2008-03-06 06:50:53 +03:00
int tmp , ret = 0 ;
2006-09-27 12:13:19 +04:00
spin_lock_irq ( & rtc - > lock ) ;
2008-03-06 06:50:53 +03:00
tmp = rtc - > periodic_freq & PF_MASK ;
2006-09-27 12:13:19 +04:00
2008-03-06 06:50:53 +03:00
switch ( freq ) {
case 0 :
rtc - > periodic_freq = 0x00 ;
break ;
case 1 :
rtc - > periodic_freq = 0x60 ;
break ;
case 2 :
rtc - > periodic_freq = 0x50 ;
break ;
case 4 :
rtc - > periodic_freq = 0x40 ;
break ;
case 8 :
rtc - > periodic_freq = 0x30 | PF_HP ;
break ;
case 16 :
rtc - > periodic_freq = 0x30 ;
break ;
case 32 :
rtc - > periodic_freq = 0x20 | PF_HP ;
break ;
case 64 :
rtc - > periodic_freq = 0x20 ;
break ;
case 128 :
rtc - > periodic_freq = 0x10 | PF_HP ;
break ;
case 256 :
rtc - > periodic_freq = 0x10 ;
break ;
default :
ret = - ENOTSUPP ;
}
2006-09-27 12:13:19 +04:00
2009-09-09 07:13:01 +04:00
if ( ret = = 0 )
2008-03-06 06:50:53 +03:00
rtc - > periodic_freq | = tmp ;
2006-09-27 12:13:19 +04:00
spin_unlock_irq ( & rtc - > lock ) ;
2008-03-06 06:50:53 +03:00
return ret ;
2006-09-27 12:13:19 +04:00
}
2008-03-06 06:50:53 +03:00
static inline void sh_rtc_setaie ( struct device * dev , unsigned int enable )
2006-09-27 12:13:19 +04:00
{
struct sh_rtc * rtc = dev_get_drvdata ( dev ) ;
unsigned int tmp ;
2008-03-06 06:50:53 +03:00
spin_lock_irq ( & rtc - > lock ) ;
2006-09-27 12:13:19 +04:00
2008-03-06 06:50:53 +03:00
tmp = readb ( rtc - > regbase + RCR1 ) ;
2006-09-27 12:13:19 +04:00
2009-04-16 09:12:22 +04:00
if ( enable )
2008-03-06 06:50:53 +03:00
tmp | = RCR1_AIE ;
2009-04-16 09:12:22 +04:00
else
tmp & = ~ RCR1_AIE ;
2006-09-27 12:13:19 +04:00
2008-03-06 06:50:53 +03:00
writeb ( tmp , rtc - > regbase + RCR1 ) ;
2006-09-27 12:13:19 +04:00
2008-03-06 06:50:53 +03:00
spin_unlock_irq ( & rtc - > lock ) ;
2006-09-27 12:13:19 +04:00
}
static int sh_rtc_proc ( struct device * dev , struct seq_file * seq )
{
struct sh_rtc * rtc = dev_get_drvdata ( dev ) ;
unsigned int tmp ;
tmp = readb ( rtc - > regbase + RCR1 ) ;
2008-03-06 06:50:53 +03:00
seq_printf ( seq , " carry_IRQ \t : %s \n " , ( tmp & RCR1_CIE ) ? " yes " : " no " ) ;
2006-09-27 12:13:19 +04:00
tmp = readb ( rtc - > regbase + RCR2 ) ;
seq_printf ( seq , " periodic_IRQ \t : %s \n " ,
2008-03-06 06:50:53 +03:00
( tmp & RCR2_PESMASK ) ? " yes " : " no " ) ;
2006-09-27 12:13:19 +04:00
return 0 ;
}
2009-03-19 13:05:58 +03:00
static inline void sh_rtc_setcie ( struct device * dev , unsigned int enable )
{
struct sh_rtc * rtc = dev_get_drvdata ( dev ) ;
unsigned int tmp ;
spin_lock_irq ( & rtc - > lock ) ;
tmp = readb ( rtc - > regbase + RCR1 ) ;
if ( ! enable )
tmp & = ~ RCR1_CIE ;
else
tmp | = RCR1_CIE ;
writeb ( tmp , rtc - > regbase + RCR1 ) ;
spin_unlock_irq ( & rtc - > lock ) ;
}
2006-09-27 12:13:19 +04:00
static int sh_rtc_ioctl ( struct device * dev , unsigned int cmd , unsigned long arg )
{
2008-03-06 06:50:53 +03:00
struct sh_rtc * rtc = dev_get_drvdata ( dev ) ;
unsigned int ret = 0 ;
2006-09-27 12:13:19 +04:00
switch ( cmd ) {
case RTC_AIE_OFF :
case RTC_AIE_ON :
sh_rtc_setaie ( dev , cmd = = RTC_AIE_ON ) ;
break ;
2008-03-06 06:50:53 +03:00
case RTC_UIE_OFF :
rtc - > periodic_freq & = ~ PF_OXS ;
2009-03-19 13:05:58 +03:00
sh_rtc_setcie ( dev , 0 ) ;
2008-03-06 06:50:53 +03:00
break ;
case RTC_UIE_ON :
rtc - > periodic_freq | = PF_OXS ;
2009-03-19 13:05:58 +03:00
sh_rtc_setcie ( dev , 1 ) ;
2008-03-06 06:50:53 +03:00
break ;
default :
ret = - ENOIOCTLCMD ;
2006-09-27 12:13:19 +04:00
}
return ret ;
}
static int sh_rtc_read_time ( struct device * dev , struct rtc_time * tm )
{
struct platform_device * pdev = to_platform_device ( dev ) ;
struct sh_rtc * rtc = platform_get_drvdata ( pdev ) ;
unsigned int sec128 , sec2 , yr , yr100 , cf_bit ;
do {
unsigned int tmp ;
spin_lock_irq ( & rtc - > lock ) ;
tmp = readb ( rtc - > regbase + RCR1 ) ;
tmp & = ~ RCR1_CF ; /* Clear CF-bit */
tmp | = RCR1_CIE ;
writeb ( tmp , rtc - > regbase + RCR1 ) ;
sec128 = readb ( rtc - > regbase + R64CNT ) ;
2008-10-19 07:28:41 +04:00
tm - > tm_sec = bcd2bin ( readb ( rtc - > regbase + RSECCNT ) ) ;
tm - > tm_min = bcd2bin ( readb ( rtc - > regbase + RMINCNT ) ) ;
tm - > tm_hour = bcd2bin ( readb ( rtc - > regbase + RHRCNT ) ) ;
tm - > tm_wday = bcd2bin ( readb ( rtc - > regbase + RWKCNT ) ) ;
tm - > tm_mday = bcd2bin ( readb ( rtc - > regbase + RDAYCNT ) ) ;
tm - > tm_mon = bcd2bin ( readb ( rtc - > regbase + RMONCNT ) ) - 1 ;
2006-09-27 12:13:19 +04:00
2007-08-03 09:19:58 +04:00
if ( rtc - > capabilities & RTC_CAP_4_DIGIT_YEAR ) {
yr = readw ( rtc - > regbase + RYRCNT ) ;
2008-10-19 07:28:41 +04:00
yr100 = bcd2bin ( yr > > 8 ) ;
2007-08-03 09:19:58 +04:00
yr & = 0xff ;
} else {
yr = readb ( rtc - > regbase + RYRCNT ) ;
2008-10-19 07:28:41 +04:00
yr100 = bcd2bin ( ( yr = = 0x99 ) ? 0x19 : 0x20 ) ;
2007-08-03 09:19:58 +04:00
}
2006-09-27 12:13:19 +04:00
2008-10-19 07:28:41 +04:00
tm - > tm_year = ( yr100 * 100 + bcd2bin ( yr ) ) - 1900 ;
2006-09-27 12:13:19 +04:00
sec2 = readb ( rtc - > regbase + R64CNT ) ;
cf_bit = readb ( rtc - > regbase + RCR1 ) & RCR1_CF ;
spin_unlock_irq ( & rtc - > lock ) ;
} while ( cf_bit ! = 0 | | ( ( sec128 ^ sec2 ) & RTC_BIT_INVERTED ) ! = 0 ) ;
# if RTC_BIT_INVERTED != 0
if ( ( sec128 & RTC_BIT_INVERTED ) )
tm - > tm_sec - - ;
# endif
2009-03-19 13:05:58 +03:00
/* only keep the carry interrupt enabled if UIE is on */
if ( ! ( rtc - > periodic_freq & PF_OXS ) )
sh_rtc_setcie ( dev , 0 ) ;
2007-05-08 06:56:27 +04:00
dev_dbg ( dev , " %s: tm is secs=%d, mins=%d, hours=%d, "
2006-09-27 12:13:19 +04:00
" mday=%d, mon=%d, year=%d, wday=%d \n " ,
2008-04-28 13:12:00 +04:00
__func__ ,
2006-09-27 12:13:19 +04:00
tm - > tm_sec , tm - > tm_min , tm - > tm_hour ,
2006-12-08 08:49:30 +03:00
tm - > tm_mday , tm - > tm_mon + 1 , tm - > tm_year , tm - > tm_wday ) ;
2006-09-27 12:13:19 +04:00
2009-03-19 13:10:44 +03:00
return rtc_valid_tm ( tm ) ;
2006-09-27 12:13:19 +04:00
}
static int sh_rtc_set_time ( struct device * dev , struct rtc_time * tm )
{
struct platform_device * pdev = to_platform_device ( dev ) ;
struct sh_rtc * rtc = platform_get_drvdata ( pdev ) ;
unsigned int tmp ;
int year ;
spin_lock_irq ( & rtc - > lock ) ;
/* Reset pre-scaler & stop RTC */
tmp = readb ( rtc - > regbase + RCR2 ) ;
tmp | = RCR2_RESET ;
2007-07-26 12:31:28 +04:00
tmp & = ~ RCR2_START ;
2006-09-27 12:13:19 +04:00
writeb ( tmp , rtc - > regbase + RCR2 ) ;
2008-10-19 07:28:41 +04:00
writeb ( bin2bcd ( tm - > tm_sec ) , rtc - > regbase + RSECCNT ) ;
writeb ( bin2bcd ( tm - > tm_min ) , rtc - > regbase + RMINCNT ) ;
writeb ( bin2bcd ( tm - > tm_hour ) , rtc - > regbase + RHRCNT ) ;
writeb ( bin2bcd ( tm - > tm_wday ) , rtc - > regbase + RWKCNT ) ;
writeb ( bin2bcd ( tm - > tm_mday ) , rtc - > regbase + RDAYCNT ) ;
writeb ( bin2bcd ( tm - > tm_mon + 1 ) , rtc - > regbase + RMONCNT ) ;
2006-09-27 12:13:19 +04:00
2007-08-03 09:19:58 +04:00
if ( rtc - > capabilities & RTC_CAP_4_DIGIT_YEAR ) {
2008-10-19 07:28:41 +04:00
year = ( bin2bcd ( ( tm - > tm_year + 1900 ) / 100 ) < < 8 ) |
bin2bcd ( tm - > tm_year % 100 ) ;
2007-08-03 09:19:58 +04:00
writew ( year , rtc - > regbase + RYRCNT ) ;
} else {
year = tm - > tm_year % 100 ;
2008-10-19 07:28:41 +04:00
writeb ( bin2bcd ( year ) , rtc - > regbase + RYRCNT ) ;
2007-08-03 09:19:58 +04:00
}
2006-09-27 12:13:19 +04:00
/* Start RTC */
tmp = readb ( rtc - > regbase + RCR2 ) ;
tmp & = ~ RCR2_RESET ;
tmp | = RCR2_RTCEN | RCR2_START ;
writeb ( tmp , rtc - > regbase + RCR2 ) ;
spin_unlock_irq ( & rtc - > lock ) ;
return 0 ;
}
2006-12-08 09:26:15 +03:00
static inline int sh_rtc_read_alarm_value ( struct sh_rtc * rtc , int reg_off )
{
unsigned int byte ;
int value = 0xff ; /* return 0xff for ignored values */
byte = readb ( rtc - > regbase + reg_off ) ;
if ( byte & AR_ENB ) {
byte & = ~ AR_ENB ; /* strip the enable bit */
2008-10-19 07:28:41 +04:00
value = bcd2bin ( byte ) ;
2006-12-08 09:26:15 +03:00
}
return value ;
}
static int sh_rtc_read_alarm ( struct device * dev , struct rtc_wkalrm * wkalrm )
{
struct platform_device * pdev = to_platform_device ( dev ) ;
struct sh_rtc * rtc = platform_get_drvdata ( pdev ) ;
2008-03-06 06:50:53 +03:00
struct rtc_time * tm = & wkalrm - > time ;
2006-12-08 09:26:15 +03:00
spin_lock_irq ( & rtc - > lock ) ;
tm - > tm_sec = sh_rtc_read_alarm_value ( rtc , RSECAR ) ;
tm - > tm_min = sh_rtc_read_alarm_value ( rtc , RMINAR ) ;
tm - > tm_hour = sh_rtc_read_alarm_value ( rtc , RHRAR ) ;
tm - > tm_wday = sh_rtc_read_alarm_value ( rtc , RWKAR ) ;
tm - > tm_mday = sh_rtc_read_alarm_value ( rtc , RDAYAR ) ;
tm - > tm_mon = sh_rtc_read_alarm_value ( rtc , RMONAR ) ;
if ( tm - > tm_mon > 0 )
tm - > tm_mon - = 1 ; /* RTC is 1-12, tm_mon is 0-11 */
tm - > tm_year = 0xffff ;
2007-01-11 10:15:32 +03:00
wkalrm - > enabled = ( readb ( rtc - > regbase + RCR1 ) & RCR1_AIE ) ? 1 : 0 ;
2006-12-08 09:26:15 +03:00
spin_unlock_irq ( & rtc - > lock ) ;
return 0 ;
}
static inline void sh_rtc_write_alarm_value ( struct sh_rtc * rtc ,
int value , int reg_off )
{
/* < 0 for a value that is ignored */
if ( value < 0 )
writeb ( 0 , rtc - > regbase + reg_off ) ;
else
2008-10-19 07:28:41 +04:00
writeb ( bin2bcd ( value ) | AR_ENB , rtc - > regbase + reg_off ) ;
2006-12-08 09:26:15 +03:00
}
2008-03-06 06:50:53 +03:00
static int sh_rtc_check_alarm ( struct rtc_time * tm )
2006-12-08 09:26:15 +03:00
{
/*
* The original rtc says anything > 0xc0 is " don't care " or " match
* all " - most users use 0xff but rtc-dev uses -1 for the same thing.
* The original rtc doesn ' t support years - some things use - 1 and
* some 0xffff . We use - 1 to make out tests easier .
*/
if ( tm - > tm_year = = 0xffff )
tm - > tm_year = - 1 ;
if ( tm - > tm_mon > = 0xff )
tm - > tm_mon = - 1 ;
if ( tm - > tm_mday > = 0xff )
tm - > tm_mday = - 1 ;
if ( tm - > tm_wday > = 0xff )
tm - > tm_wday = - 1 ;
if ( tm - > tm_hour > = 0xff )
tm - > tm_hour = - 1 ;
if ( tm - > tm_min > = 0xff )
tm - > tm_min = - 1 ;
if ( tm - > tm_sec > = 0xff )
tm - > tm_sec = - 1 ;
if ( tm - > tm_year > 9999 | |
tm - > tm_mon > = 12 | |
tm - > tm_mday = = 0 | | tm - > tm_mday > = 32 | |
tm - > tm_wday > = 7 | |
tm - > tm_hour > = 24 | |
tm - > tm_min > = 60 | |
tm - > tm_sec > = 60 )
return - EINVAL ;
return 0 ;
}
static int sh_rtc_set_alarm ( struct device * dev , struct rtc_wkalrm * wkalrm )
{
struct platform_device * pdev = to_platform_device ( dev ) ;
struct sh_rtc * rtc = platform_get_drvdata ( pdev ) ;
unsigned int rcr1 ;
struct rtc_time * tm = & wkalrm - > time ;
int mon , err ;
err = sh_rtc_check_alarm ( tm ) ;
if ( unlikely ( err < 0 ) )
return err ;
spin_lock_irq ( & rtc - > lock ) ;
2007-01-23 07:40:41 +03:00
/* disable alarm interrupt and clear the alarm flag */
2006-12-08 09:26:15 +03:00
rcr1 = readb ( rtc - > regbase + RCR1 ) ;
2008-03-06 06:50:53 +03:00
rcr1 & = ~ ( RCR1_AF | RCR1_AIE ) ;
2007-01-23 07:40:41 +03:00
writeb ( rcr1 , rtc - > regbase + RCR1 ) ;
2006-12-08 09:26:15 +03:00
/* set alarm time */
sh_rtc_write_alarm_value ( rtc , tm - > tm_sec , RSECAR ) ;
sh_rtc_write_alarm_value ( rtc , tm - > tm_min , RMINAR ) ;
sh_rtc_write_alarm_value ( rtc , tm - > tm_hour , RHRAR ) ;
sh_rtc_write_alarm_value ( rtc , tm - > tm_wday , RWKAR ) ;
sh_rtc_write_alarm_value ( rtc , tm - > tm_mday , RDAYAR ) ;
mon = tm - > tm_mon ;
if ( mon > = 0 )
mon + = 1 ;
sh_rtc_write_alarm_value ( rtc , mon , RMONAR ) ;
2007-01-23 07:40:41 +03:00
if ( wkalrm - > enabled ) {
rcr1 | = RCR1_AIE ;
writeb ( rcr1 , rtc - > regbase + RCR1 ) ;
}
2006-12-08 09:26:15 +03:00
spin_unlock_irq ( & rtc - > lock ) ;
return 0 ;
}
2006-09-27 12:13:19 +04:00
static struct rtc_class_ops sh_rtc_ops = {
. ioctl = sh_rtc_ioctl ,
. read_time = sh_rtc_read_time ,
. set_time = sh_rtc_set_time ,
2006-12-08 09:26:15 +03:00
. read_alarm = sh_rtc_read_alarm ,
. set_alarm = sh_rtc_set_alarm ,
2008-03-06 06:50:53 +03:00
. irq_set_state = sh_rtc_irq_set_state ,
. irq_set_freq = sh_rtc_irq_set_freq ,
2006-09-27 12:13:19 +04:00
. proc = sh_rtc_proc ,
} ;
2009-08-20 08:25:11 +04:00
static int __init sh_rtc_probe ( struct platform_device * pdev )
2006-09-27 12:13:19 +04:00
{
struct sh_rtc * rtc ;
struct resource * res ;
2009-03-19 13:10:44 +03:00
struct rtc_time r ;
2009-04-16 09:12:22 +04:00
char clk_name [ 6 ] ;
int clk_id , ret ;
2006-09-27 12:13:19 +04:00
rtc = kzalloc ( sizeof ( struct sh_rtc ) , GFP_KERNEL ) ;
if ( unlikely ( ! rtc ) )
return - ENOMEM ;
spin_lock_init ( & rtc - > lock ) ;
2008-03-06 06:50:53 +03:00
/* get periodic/carry/alarm irqs */
2008-09-10 21:34:44 +04:00
ret = platform_get_irq ( pdev , 0 ) ;
2009-01-07 01:42:11 +03:00
if ( unlikely ( ret < = 0 ) ) {
2008-09-10 21:34:44 +04:00
ret = - ENOENT ;
2009-02-24 16:11:03 +03:00
dev_err ( & pdev - > dev , " No IRQ resource \n " ) ;
2006-09-27 12:13:19 +04:00
goto err_badres ;
}
2009-04-16 09:12:22 +04:00
2008-09-10 21:34:44 +04:00
rtc - > periodic_irq = ret ;
2009-02-24 16:11:03 +03:00
rtc - > carry_irq = platform_get_irq ( pdev , 1 ) ;
rtc - > alarm_irq = platform_get_irq ( pdev , 2 ) ;
2006-09-27 12:13:19 +04:00
res = platform_get_resource ( pdev , IORESOURCE_IO , 0 ) ;
if ( unlikely ( res = = NULL ) ) {
2008-09-10 21:34:44 +04:00
ret = - ENOENT ;
2006-09-27 12:13:19 +04:00
dev_err ( & pdev - > dev , " No IO resource \n " ) ;
goto err_badres ;
}
2009-04-16 09:12:22 +04:00
rtc - > regsize = resource_size ( res ) ;
2006-09-27 12:13:19 +04:00
rtc - > res = request_mem_region ( res - > start , rtc - > regsize , pdev - > name ) ;
if ( unlikely ( ! rtc - > res ) ) {
ret = - EBUSY ;
goto err_badres ;
}
2008-04-25 12:58:42 +04:00
rtc - > regbase = ioremap_nocache ( rtc - > res - > start , rtc - > regsize ) ;
2006-09-27 12:13:19 +04:00
if ( unlikely ( ! rtc - > regbase ) ) {
ret = - EINVAL ;
goto err_badmap ;
}
2009-04-16 09:12:22 +04:00
clk_id = pdev - > id ;
/* With a single device, the clock id is still "rtc0" */
if ( clk_id < 0 )
clk_id = 0 ;
snprintf ( clk_name , sizeof ( clk_name ) , " rtc%d " , clk_id ) ;
rtc - > clk = clk_get ( & pdev - > dev , clk_name ) ;
if ( IS_ERR ( rtc - > clk ) ) {
/*
* No error handling for rtc - > clk intentionally , not all
* platforms will have a unique clock for the RTC , and
* the clk API can handle the struct clk pointer being
* NULL .
*/
rtc - > clk = NULL ;
}
clk_enable ( rtc - > clk ) ;
2007-08-03 09:19:58 +04:00
rtc - > capabilities = RTC_DEF_CAPABILITIES ;
if ( pdev - > dev . platform_data ) {
struct sh_rtc_platform_info * pinfo = pdev - > dev . platform_data ;
/*
* Some CPUs have special capabilities in addition to the
* default set . Add those in here .
*/
rtc - > capabilities | = pinfo - > capabilities ;
}
2009-02-24 16:11:03 +03:00
if ( rtc - > carry_irq < = 0 ) {
/* register shared periodic/carry/alarm irq */
ret = request_irq ( rtc - > periodic_irq , sh_rtc_shared ,
IRQF_DISABLED , " sh-rtc " , rtc ) ;
if ( unlikely ( ret ) ) {
dev_err ( & pdev - > dev ,
" request IRQ failed with %d, IRQ %d \n " , ret ,
rtc - > periodic_irq ) ;
goto err_unmap ;
}
} else {
/* register periodic/carry/alarm irqs */
ret = request_irq ( rtc - > periodic_irq , sh_rtc_periodic ,
IRQF_DISABLED , " sh-rtc period " , rtc ) ;
if ( unlikely ( ret ) ) {
dev_err ( & pdev - > dev ,
" request period IRQ failed with %d, IRQ %d \n " ,
ret , rtc - > periodic_irq ) ;
goto err_unmap ;
}
2008-03-06 06:50:53 +03:00
2009-02-24 16:11:03 +03:00
ret = request_irq ( rtc - > carry_irq , sh_rtc_interrupt ,
IRQF_DISABLED , " sh-rtc carry " , rtc ) ;
if ( unlikely ( ret ) ) {
dev_err ( & pdev - > dev ,
" request carry IRQ failed with %d, IRQ %d \n " ,
ret , rtc - > carry_irq ) ;
free_irq ( rtc - > periodic_irq , rtc ) ;
goto err_unmap ;
}
2008-03-06 06:50:53 +03:00
2009-02-24 16:11:03 +03:00
ret = request_irq ( rtc - > alarm_irq , sh_rtc_alarm ,
IRQF_DISABLED , " sh-rtc alarm " , rtc ) ;
if ( unlikely ( ret ) ) {
dev_err ( & pdev - > dev ,
" request alarm IRQ failed with %d, IRQ %d \n " ,
ret , rtc - > alarm_irq ) ;
free_irq ( rtc - > carry_irq , rtc ) ;
free_irq ( rtc - > periodic_irq , rtc ) ;
goto err_unmap ;
}
2008-03-06 06:50:53 +03:00
}
2009-08-20 08:25:11 +04:00
platform_set_drvdata ( pdev , rtc ) ;
2009-03-19 13:05:58 +03:00
/* everything disabled by default */
2009-08-20 08:25:11 +04:00
sh_rtc_irq_set_freq ( & pdev - > dev , 0 ) ;
sh_rtc_irq_set_state ( & pdev - > dev , 0 ) ;
2009-03-19 13:05:58 +03:00
sh_rtc_setaie ( & pdev - > dev , 0 ) ;
sh_rtc_setcie ( & pdev - > dev , 0 ) ;
2009-03-19 13:10:44 +03:00
2009-08-20 08:25:11 +04:00
rtc - > rtc_dev = rtc_device_register ( " sh " , & pdev - > dev ,
& sh_rtc_ops , THIS_MODULE ) ;
if ( IS_ERR ( rtc - > rtc_dev ) ) {
ret = PTR_ERR ( rtc - > rtc_dev ) ;
free_irq ( rtc - > periodic_irq , rtc ) ;
free_irq ( rtc - > carry_irq , rtc ) ;
free_irq ( rtc - > alarm_irq , rtc ) ;
goto err_unmap ;
}
rtc - > rtc_dev - > max_user_freq = 256 ;
2009-03-19 13:10:44 +03:00
/* reset rtc to epoch 0 if time is invalid */
if ( rtc_read_time ( rtc - > rtc_dev , & r ) < 0 ) {
rtc_time_to_tm ( 0 , & r ) ;
rtc_set_time ( rtc - > rtc_dev , & r ) ;
}
2009-03-19 13:14:41 +03:00
device_init_wakeup ( & pdev - > dev , 1 ) ;
2006-09-27 12:13:19 +04:00
return 0 ;
2008-04-25 12:58:42 +04:00
err_unmap :
2009-04-16 09:12:22 +04:00
clk_disable ( rtc - > clk ) ;
clk_put ( rtc - > clk ) ;
2008-04-25 12:58:42 +04:00
iounmap ( rtc - > regbase ) ;
2006-09-27 12:13:19 +04:00
err_badmap :
release_resource ( rtc - > res ) ;
err_badres :
kfree ( rtc ) ;
return ret ;
}
2009-08-20 08:25:11 +04:00
static int __exit sh_rtc_remove ( struct platform_device * pdev )
2006-09-27 12:13:19 +04:00
{
struct sh_rtc * rtc = platform_get_drvdata ( pdev ) ;
2009-08-20 08:25:11 +04:00
rtc_device_unregister ( rtc - > rtc_dev ) ;
sh_rtc_irq_set_state ( & pdev - > dev , 0 ) ;
2006-09-27 12:13:19 +04:00
sh_rtc_setaie ( & pdev - > dev , 0 ) ;
2009-03-19 13:05:58 +03:00
sh_rtc_setcie ( & pdev - > dev , 0 ) ;
2006-09-27 12:13:19 +04:00
2008-03-06 06:50:53 +03:00
free_irq ( rtc - > periodic_irq , rtc ) ;
2009-04-16 09:12:22 +04:00
2009-02-24 16:11:03 +03:00
if ( rtc - > carry_irq > 0 ) {
free_irq ( rtc - > carry_irq , rtc ) ;
free_irq ( rtc - > alarm_irq , rtc ) ;
}
2008-03-06 06:50:53 +03:00
2008-04-25 12:58:42 +04:00
iounmap ( rtc - > regbase ) ;
2009-08-20 08:25:11 +04:00
release_resource ( rtc - > res ) ;
2008-04-25 12:58:42 +04:00
2009-04-16 09:12:22 +04:00
clk_disable ( rtc - > clk ) ;
clk_put ( rtc - > clk ) ;
2006-09-27 12:13:19 +04:00
platform_set_drvdata ( pdev , NULL ) ;
kfree ( rtc ) ;
return 0 ;
}
2009-04-01 18:45:17 +04:00
static void sh_rtc_set_irq_wake ( struct device * dev , int enabled )
{
struct platform_device * pdev = to_platform_device ( dev ) ;
struct sh_rtc * rtc = platform_get_drvdata ( pdev ) ;
set_irq_wake ( rtc - > periodic_irq , enabled ) ;
2009-04-16 09:12:22 +04:00
2009-04-01 18:45:17 +04:00
if ( rtc - > carry_irq > 0 ) {
set_irq_wake ( rtc - > carry_irq , enabled ) ;
set_irq_wake ( rtc - > alarm_irq , enabled ) ;
}
}
static int sh_rtc_suspend ( struct device * dev )
{
if ( device_may_wakeup ( dev ) )
sh_rtc_set_irq_wake ( dev , 1 ) ;
return 0 ;
}
static int sh_rtc_resume ( struct device * dev )
{
if ( device_may_wakeup ( dev ) )
sh_rtc_set_irq_wake ( dev , 0 ) ;
return 0 ;
}
2009-12-15 05:00:08 +03:00
static const struct dev_pm_ops sh_rtc_dev_pm_ops = {
2009-04-01 18:45:17 +04:00
. suspend = sh_rtc_suspend ,
. resume = sh_rtc_resume ,
} ;
2006-09-27 12:13:19 +04:00
static struct platform_driver sh_rtc_platform_driver = {
. driver = {
2006-12-08 09:26:15 +03:00
. name = DRV_NAME ,
2006-09-27 12:13:19 +04:00
. owner = THIS_MODULE ,
2009-04-01 18:45:17 +04:00
. pm = & sh_rtc_dev_pm_ops ,
2006-09-27 12:13:19 +04:00
} ,
2009-08-20 08:25:11 +04:00
. remove = __exit_p ( sh_rtc_remove ) ,
2006-09-27 12:13:19 +04:00
} ;
static int __init sh_rtc_init ( void )
{
2009-08-20 08:25:11 +04:00
return platform_driver_probe ( & sh_rtc_platform_driver , sh_rtc_probe ) ;
2006-09-27 12:13:19 +04:00
}
static void __exit sh_rtc_exit ( void )
{
platform_driver_unregister ( & sh_rtc_platform_driver ) ;
}
module_init ( sh_rtc_init ) ;
module_exit ( sh_rtc_exit ) ;
MODULE_DESCRIPTION ( " SuperH on-chip RTC driver " ) ;
2006-12-08 09:26:15 +03:00
MODULE_VERSION ( DRV_VERSION ) ;
2008-03-06 06:50:53 +03:00
MODULE_AUTHOR ( " Paul Mundt <lethal@linux-sh.org>, "
" Jamie Lenehan <lenehan@twibble.org>, "
" Angelo Castello <angelo.castello@st.com> " ) ;
2006-09-27 12:13:19 +04:00
MODULE_LICENSE ( " GPL " ) ;
2008-04-11 08:29:25 +04:00
MODULE_ALIAS ( " platform: " DRV_NAME ) ;