2008-02-06 01:38:59 -08:00
/*
* " RTT as Real Time Clock " driver for AT91SAM9 SoC family
*
* ( C ) 2007 Michel Benoit
*
* Based on rtc - at91rm9200 . c by Rick Bronson
*
* 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 .
*/
2015-07-28 21:49:24 +02:00
# include <linux/clk.h>
2008-02-06 01:38:59 -08:00
# include <linux/interrupt.h>
# include <linux/ioctl.h>
2013-04-29 16:20:35 -07:00
# include <linux/io.h>
2015-07-28 21:49:24 +02:00
# include <linux/kernel.h>
2014-09-23 13:14:44 +02:00
# include <linux/mfd/syscon.h>
2015-07-28 21:49:24 +02:00
# include <linux/module.h>
2015-08-10 16:33:39 +02:00
# include <linux/of.h>
2015-07-28 21:49:24 +02:00
# include <linux/platform_device.h>
2014-09-23 13:14:44 +02:00
# include <linux/regmap.h>
2015-07-28 21:49:24 +02:00
# include <linux/rtc.h>
# include <linux/slab.h>
2015-03-02 10:18:14 +01:00
# include <linux/suspend.h>
2015-07-28 21:49:24 +02:00
# include <linux/time.h>
2008-02-06 01:38:59 -08:00
/*
* This driver uses two configurable hardware resources that live in the
* AT91SAM9 backup power domain ( intended to be powered at all times )
* to implement the Real Time Clock interfaces
*
* - A " Real-time Timer " ( RTT ) counts up in seconds from a base time .
* We can ' t assign the counter value ( CRTV ) . . . but we can reset it .
*
* - One of the " General Purpose Backup Registers " ( GPBRs ) holds the
* base time , normally an offset from the beginning of the POSIX
* epoch ( 1970 - Jan - 1 00 : 00 : 00 UTC ) . Some systems also include the
* local timezone ' s offset .
*
* The RTC ' s value is the RTT counter plus that offset . The RTC ' s alarm
* is likewise a base ( ALMV ) plus that offset .
*
* Not all RTTs will be used as RTCs ; some systems have multiple RTTs to
* choose from , or a " real " RTC module . All systems have multiple GPBR
* registers available , likewise usable for more than " RTC " support .
*/
2014-09-23 13:13:29 +02:00
# define AT91_RTT_MR 0x00 /* Real-time Mode Register */
# define AT91_RTT_RTPRES (0xffff << 0) /* Real-time Timer Prescaler Value */
# define AT91_RTT_ALMIEN (1 << 16) /* Alarm Interrupt Enable */
# define AT91_RTT_RTTINCIEN (1 << 17) /* Real Time Timer Increment Interrupt Enable */
# define AT91_RTT_RTTRST (1 << 18) /* Real Time Timer Restart */
# define AT91_RTT_AR 0x04 /* Real-time Alarm Register */
# define AT91_RTT_ALMV (0xffffffff) /* Alarm Value */
# define AT91_RTT_VR 0x08 /* Real-time Value Register */
# define AT91_RTT_CRTV (0xffffffff) /* Current Real-time Value */
# define AT91_RTT_SR 0x0c /* Real-time Status Register */
# define AT91_RTT_ALMS (1 << 0) /* Real-time Alarm Status */
# define AT91_RTT_RTTINC (1 << 1) /* Real-time Timer Increment */
2008-02-06 01:38:59 -08:00
/*
* We store ALARM_DISABLED in ALMV to record that no alarm is set .
* It ' s also the reset value for that field .
*/
# define ALARM_DISABLED ((u32)~0)
struct sam9_rtc {
void __iomem * rtt ;
struct rtc_device * rtcdev ;
u32 imr ;
2014-09-23 13:14:44 +02:00
struct regmap * gpbr ;
unsigned int gpbr_offset ;
2012-08-14 11:19:22 +02:00
int irq ;
2014-09-23 16:41:07 +02:00
struct clk * sclk ;
2015-03-02 10:18:14 +01:00
bool suspended ;
unsigned long events ;
spinlock_t lock ;
2008-02-06 01:38:59 -08:00
} ;
# define rtt_readl(rtc, field) \
2014-09-23 13:13:52 +02:00
readl ( ( rtc ) - > rtt + AT91_RTT_ # # field )
2008-02-06 01:38:59 -08:00
# define rtt_writel(rtc, field, val) \
2014-09-23 13:13:52 +02:00
writel ( ( val ) , ( rtc ) - > rtt + AT91_RTT_ # # field )
2008-02-06 01:38:59 -08:00
2014-09-23 13:14:44 +02:00
static inline unsigned int gpbr_readl ( struct sam9_rtc * rtc )
{
unsigned int val ;
regmap_read ( rtc - > gpbr , rtc - > gpbr_offset , & val ) ;
return val ;
}
static inline void gpbr_writel ( struct sam9_rtc * rtc , unsigned int val )
{
regmap_write ( rtc - > gpbr , rtc - > gpbr_offset , val ) ;
}
2008-02-06 01:38:59 -08:00
/*
* Read current time and date in RTC
*/
static int at91_rtc_readtime ( struct device * dev , struct rtc_time * tm )
{
struct sam9_rtc * rtc = dev_get_drvdata ( dev ) ;
u32 secs , secs2 ;
u32 offset ;
/* read current time offset */
offset = gpbr_readl ( rtc ) ;
if ( offset = = 0 )
return - EILSEQ ;
/* reread the counter to help sync the two clock domains */
secs = rtt_readl ( rtc , VR ) ;
secs2 = rtt_readl ( rtc , VR ) ;
if ( secs ! = secs2 )
secs = rtt_readl ( rtc , VR ) ;
rtc_time_to_tm ( offset + secs , tm ) ;
dev_dbg ( dev , " %s: %4d-%02d-%02d %02d:%02d:%02d \n " , " readtime " ,
1900 + tm - > tm_year , tm - > tm_mon , tm - > tm_mday ,
tm - > tm_hour , tm - > tm_min , tm - > tm_sec ) ;
return 0 ;
}
/*
* Set current time and date in RTC
*/
static int at91_rtc_settime ( struct device * dev , struct rtc_time * tm )
{
struct sam9_rtc * rtc = dev_get_drvdata ( dev ) ;
int err ;
u32 offset , alarm , mr ;
unsigned long secs ;
dev_dbg ( dev , " %s: %4d-%02d-%02d %02d:%02d:%02d \n " , " settime " ,
1900 + tm - > tm_year , tm - > tm_mon , tm - > tm_mday ,
tm - > tm_hour , tm - > tm_min , tm - > tm_sec ) ;
err = rtc_tm_to_time ( tm , & secs ) ;
if ( err ! = 0 )
return err ;
mr = rtt_readl ( rtc , MR ) ;
/* disable interrupts */
rtt_writel ( rtc , MR , mr & ~ ( AT91_RTT_ALMIEN | AT91_RTT_RTTINCIEN ) ) ;
/* read current time offset */
offset = gpbr_readl ( rtc ) ;
/* store the new base time in a battery backup register */
secs + = 1 ;
gpbr_writel ( rtc , secs ) ;
/* adjust the alarm time for the new base */
alarm = rtt_readl ( rtc , AR ) ;
if ( alarm ! = ALARM_DISABLED ) {
if ( offset > secs ) {
/* time jumped backwards, increase time until alarm */
alarm + = ( offset - secs ) ;
} else if ( ( alarm + offset ) > secs ) {
/* time jumped forwards, decrease time until alarm */
alarm - = ( secs - offset ) ;
} else {
/* time jumped past the alarm, disable alarm */
alarm = ALARM_DISABLED ;
mr & = ~ AT91_RTT_ALMIEN ;
}
rtt_writel ( rtc , AR , alarm ) ;
}
/* reset the timer, and re-enable interrupts */
rtt_writel ( rtc , MR , mr | AT91_RTT_RTTRST ) ;
return 0 ;
}
static int at91_rtc_readalarm ( struct device * dev , struct rtc_wkalrm * alrm )
{
struct sam9_rtc * rtc = dev_get_drvdata ( dev ) ;
struct rtc_time * tm = & alrm - > time ;
u32 alarm = rtt_readl ( rtc , AR ) ;
u32 offset ;
offset = gpbr_readl ( rtc ) ;
if ( offset = = 0 )
return - EILSEQ ;
2010-03-05 13:44:23 -08:00
memset ( alrm , 0 , sizeof ( * alrm ) ) ;
2008-02-06 01:38:59 -08:00
if ( alarm ! = ALARM_DISABLED & & offset ! = 0 ) {
rtc_time_to_tm ( offset + alarm , tm ) ;
dev_dbg ( dev , " %s: %4d-%02d-%02d %02d:%02d:%02d \n " , " readalarm " ,
1900 + tm - > tm_year , tm - > tm_mon , tm - > tm_mday ,
tm - > tm_hour , tm - > tm_min , tm - > tm_sec ) ;
if ( rtt_readl ( rtc , MR ) & AT91_RTT_ALMIEN )
alrm - > enabled = 1 ;
}
return 0 ;
}
static int at91_rtc_setalarm ( struct device * dev , struct rtc_wkalrm * alrm )
{
struct sam9_rtc * rtc = dev_get_drvdata ( dev ) ;
struct rtc_time * tm = & alrm - > time ;
unsigned long secs ;
u32 offset ;
u32 mr ;
int err ;
err = rtc_tm_to_time ( tm , & secs ) ;
if ( err ! = 0 )
return err ;
offset = gpbr_readl ( rtc ) ;
if ( offset = = 0 ) {
/* time is not set */
return - EILSEQ ;
}
mr = rtt_readl ( rtc , MR ) ;
rtt_writel ( rtc , MR , mr & ~ AT91_RTT_ALMIEN ) ;
/* alarm in the past? finish and leave disabled */
if ( secs < = offset ) {
rtt_writel ( rtc , AR , ALARM_DISABLED ) ;
return 0 ;
}
/* else set alarm and maybe enable it */
rtt_writel ( rtc , AR , secs - offset ) ;
if ( alrm - > enabled )
rtt_writel ( rtc , MR , mr | AT91_RTT_ALMIEN ) ;
dev_dbg ( dev , " %s: %4d-%02d-%02d %02d:%02d:%02d \n " , " setalarm " ,
tm - > tm_year , tm - > tm_mon , tm - > tm_mday , tm - > tm_hour ,
tm - > tm_min , tm - > tm_sec ) ;
return 0 ;
}
2011-02-02 17:02:41 -08:00
static int at91_rtc_alarm_irq_enable ( struct device * dev , unsigned int enabled )
{
struct sam9_rtc * rtc = dev_get_drvdata ( dev ) ;
u32 mr = rtt_readl ( rtc , MR ) ;
dev_dbg ( dev , " alarm_irq_enable: enabled=%08x, mr %08x \n " , enabled , mr ) ;
if ( enabled )
rtt_writel ( rtc , MR , mr | AT91_RTT_ALMIEN ) ;
else
rtt_writel ( rtc , MR , mr & ~ AT91_RTT_ALMIEN ) ;
return 0 ;
}
2008-02-06 01:38:59 -08:00
/*
* Provide additional RTC information in / proc / driver / rtc
*/
static int at91_rtc_proc ( struct device * dev , struct seq_file * seq )
{
struct sam9_rtc * rtc = dev_get_drvdata ( dev ) ;
u32 mr = mr = rtt_readl ( rtc , MR ) ;
seq_printf ( seq , " update_IRQ \t : %s \n " ,
( mr & AT91_RTT_RTTINCIEN ) ? " yes " : " no " ) ;
return 0 ;
}
2015-03-02 10:18:14 +01:00
static irqreturn_t at91_rtc_cache_events ( struct sam9_rtc * rtc )
2008-02-06 01:38:59 -08:00
{
u32 sr , mr ;
/* Shared interrupt may be for another device. Note: reading
* SR clears it , so we must only read it in this irq handler !
*/
mr = rtt_readl ( rtc , MR ) & ( AT91_RTT_ALMIEN | AT91_RTT_RTTINCIEN ) ;
2008-03-19 17:01:09 -07:00
sr = rtt_readl ( rtc , SR ) & ( mr > > 16 ) ;
2008-02-06 01:38:59 -08:00
if ( ! sr )
return IRQ_NONE ;
/* alarm status */
if ( sr & AT91_RTT_ALMS )
2015-03-02 10:18:14 +01:00
rtc - > events | = ( RTC_AF | RTC_IRQF ) ;
2008-02-06 01:38:59 -08:00
/* timer update/increment */
if ( sr & AT91_RTT_RTTINC )
2015-03-02 10:18:14 +01:00
rtc - > events | = ( RTC_UF | RTC_IRQF ) ;
return IRQ_HANDLED ;
}
static void at91_rtc_flush_events ( struct sam9_rtc * rtc )
{
if ( ! rtc - > events )
return ;
2008-02-06 01:38:59 -08:00
2015-03-02 10:18:14 +01:00
rtc_update_irq ( rtc - > rtcdev , 1 , rtc - > events ) ;
rtc - > events = 0 ;
2008-02-06 01:38:59 -08:00
2008-04-28 02:12:00 -07:00
pr_debug ( " %s: num=%ld, events=0x%02lx \n " , __func__ ,
2015-03-02 10:18:14 +01:00
rtc - > events > > 8 , rtc - > events & 0x000000FF ) ;
}
2008-02-06 01:38:59 -08:00
2015-03-02 10:18:14 +01:00
/*
* IRQ handler for the RTC
*/
static irqreturn_t at91_rtc_interrupt ( int irq , void * _rtc )
{
struct sam9_rtc * rtc = _rtc ;
int ret ;
spin_lock ( & rtc - > lock ) ;
ret = at91_rtc_cache_events ( rtc ) ;
/* We're called in suspended state */
if ( rtc - > suspended ) {
/* Mask irqs coming from this peripheral */
rtt_writel ( rtc , MR ,
rtt_readl ( rtc , MR ) &
~ ( AT91_RTT_ALMIEN | AT91_RTT_RTTINCIEN ) ) ;
/* Trigger a system wakeup */
pm_system_wakeup ( ) ;
} else {
at91_rtc_flush_events ( rtc ) ;
}
spin_unlock ( & rtc - > lock ) ;
return ret ;
2008-02-06 01:38:59 -08:00
}
static const struct rtc_class_ops at91_rtc_ops = {
. read_time = at91_rtc_readtime ,
. set_time = at91_rtc_settime ,
. read_alarm = at91_rtc_readalarm ,
. set_alarm = at91_rtc_setalarm ,
. proc = at91_rtc_proc ,
2011-02-25 11:13:55 -08:00
. alarm_irq_enable = at91_rtc_alarm_irq_enable ,
2008-02-06 01:38:59 -08:00
} ;
2015-02-13 14:40:48 -08:00
static const struct regmap_config gpbr_regmap_config = {
2014-09-23 13:14:44 +02:00
. reg_bits = 32 ,
. val_bits = 32 ,
. reg_stride = 4 ,
} ;
2008-02-06 01:38:59 -08:00
/*
* Initialize and install RTC driver
*/
2012-12-21 13:09:38 -08:00
static int at91_rtc_probe ( struct platform_device * pdev )
2008-02-06 01:38:59 -08:00
{
2014-09-23 13:14:09 +02:00
struct resource * r ;
2008-02-06 01:38:59 -08:00
struct sam9_rtc * rtc ;
2012-08-14 11:19:22 +02:00
int ret , irq ;
2008-02-06 01:38:59 -08:00
u32 mr ;
2014-09-23 16:41:07 +02:00
unsigned int sclk_rate ;
2008-02-06 01:38:59 -08:00
2012-08-14 11:19:22 +02:00
irq = platform_get_irq ( pdev , 0 ) ;
if ( irq < 0 ) {
dev_err ( & pdev - > dev , " failed to get interrupt resource \n " ) ;
return irq ;
}
2013-04-29 16:20:35 -07:00
rtc = devm_kzalloc ( & pdev - > dev , sizeof ( * rtc ) , GFP_KERNEL ) ;
2008-02-06 01:38:59 -08:00
if ( ! rtc )
return - ENOMEM ;
2012-08-14 11:19:22 +02:00
rtc - > irq = irq ;
2008-03-19 17:01:09 -07:00
/* platform setup code should have handled this; sigh */
if ( ! device_can_wakeup ( & pdev - > dev ) )
device_init_wakeup ( & pdev - > dev , 1 ) ;
2008-02-06 01:38:59 -08:00
platform_set_drvdata ( pdev , rtc ) ;
2014-09-23 13:14:09 +02:00
r = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
rtc - > rtt = devm_ioremap_resource ( & pdev - > dev , r ) ;
if ( IS_ERR ( rtc - > rtt ) )
return PTR_ERR ( rtc - > rtt ) ;
2014-09-23 13:14:44 +02:00
if ( ! pdev - > dev . of_node ) {
/*
* TODO : Remove this code chunk when removing non DT board
* support . Remember to remove the gpbr_regmap_config
* variable too .
*/
void __iomem * gpbr ;
r = platform_get_resource ( pdev , IORESOURCE_MEM , 1 ) ;
gpbr = devm_ioremap_resource ( & pdev - > dev , r ) ;
if ( IS_ERR ( gpbr ) )
return PTR_ERR ( gpbr ) ;
rtc - > gpbr = regmap_init_mmio ( NULL , gpbr ,
& gpbr_regmap_config ) ;
} else {
struct of_phandle_args args ;
ret = of_parse_phandle_with_fixed_args ( pdev - > dev . of_node ,
" atmel,rtt-rtc-time-reg " , 1 , 0 ,
& args ) ;
if ( ret )
return ret ;
rtc - > gpbr = syscon_node_to_regmap ( args . np ) ;
rtc - > gpbr_offset = args . args [ 0 ] ;
}
if ( IS_ERR ( rtc - > gpbr ) ) {
dev_err ( & pdev - > dev , " failed to retrieve gpbr regmap, aborting. \n " ) ;
return - ENOMEM ;
}
2012-02-15 21:24:46 +08:00
2014-09-23 16:41:07 +02:00
rtc - > sclk = devm_clk_get ( & pdev - > dev , NULL ) ;
if ( IS_ERR ( rtc - > sclk ) )
return PTR_ERR ( rtc - > sclk ) ;
ret = clk_prepare_enable ( rtc - > sclk ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " Could not enable slow clock \n " ) ;
return ret ;
}
2015-07-28 21:51:10 +02:00
sclk_rate = clk_get_rate ( rtc - > sclk ) ;
if ( ! sclk_rate | | sclk_rate > AT91_RTT_RTPRES ) {
dev_err ( & pdev - > dev , " Invalid slow clock rate \n " ) ;
ret = - EINVAL ;
goto err_clk ;
}
2008-02-06 01:38:59 -08:00
mr = rtt_readl ( rtc , MR ) ;
/* unless RTT is counting at 1 Hz, re-initialize it */
2014-09-23 16:41:07 +02:00
if ( ( mr & AT91_RTT_RTPRES ) ! = sclk_rate ) {
mr = AT91_RTT_RTTRST | ( sclk_rate & AT91_RTT_RTPRES ) ;
2008-02-06 01:38:59 -08:00
gpbr_writel ( rtc , 0 ) ;
}
/* disable all interrupts (same as on shutdown path) */
mr & = ~ ( AT91_RTT_ALMIEN | AT91_RTT_RTTINCIEN ) ;
rtt_writel ( rtc , MR , mr ) ;
2013-04-29 16:20:35 -07:00
rtc - > rtcdev = devm_rtc_device_register ( & pdev - > dev , pdev - > name ,
& at91_rtc_ops , THIS_MODULE ) ;
2015-07-28 21:46:15 +02:00
if ( IS_ERR ( rtc - > rtcdev ) ) {
ret = PTR_ERR ( rtc - > rtcdev ) ;
goto err_clk ;
}
2008-02-06 01:38:59 -08:00
/* register irq handler after we know what name we'll use */
2013-04-29 16:20:35 -07:00
ret = devm_request_irq ( & pdev - > dev , rtc - > irq , at91_rtc_interrupt ,
2015-03-02 10:18:14 +01:00
IRQF_SHARED | IRQF_COND_SUSPEND ,
dev_name ( & rtc - > rtcdev - > dev ) , rtc ) ;
2008-02-06 01:38:59 -08:00
if ( ret ) {
2012-08-14 11:19:22 +02:00
dev_dbg ( & pdev - > dev , " can't share IRQ %d? \n " , rtc - > irq ) ;
2015-07-28 21:46:15 +02:00
goto err_clk ;
2008-02-06 01:38:59 -08:00
}
/* NOTE: sam9260 rev A silicon has a ROM bug which resets the
* RTT on at least some reboots . If you have that chip , you must
* initialize the time from some external source like a GPS , wall
* clock , discrete RTC , etc
*/
if ( gpbr_readl ( rtc ) = = 0 )
dev_warn ( & pdev - > dev , " %s: SET TIME! \n " ,
2009-03-24 16:38:22 -07:00
dev_name ( & rtc - > rtcdev - > dev ) ) ;
2008-02-06 01:38:59 -08:00
return 0 ;
2015-07-28 21:46:15 +02:00
err_clk :
clk_disable_unprepare ( rtc - > sclk ) ;
return ret ;
2008-02-06 01:38:59 -08:00
}
/*
* Disable and remove the RTC driver
*/
2012-12-21 13:09:38 -08:00
static int at91_rtc_remove ( struct platform_device * pdev )
2008-02-06 01:38:59 -08:00
{
struct sam9_rtc * rtc = platform_get_drvdata ( pdev ) ;
u32 mr = rtt_readl ( rtc , MR ) ;
/* disable all interrupts */
rtt_writel ( rtc , MR , mr & ~ ( AT91_RTT_ALMIEN | AT91_RTT_RTTINCIEN ) ) ;
2015-07-28 21:47:57 +02:00
clk_disable_unprepare ( rtc - > sclk ) ;
2014-09-23 16:41:07 +02:00
2008-02-06 01:38:59 -08:00
return 0 ;
}
static void at91_rtc_shutdown ( struct platform_device * pdev )
{
struct sam9_rtc * rtc = platform_get_drvdata ( pdev ) ;
u32 mr = rtt_readl ( rtc , MR ) ;
rtc - > imr = mr & ( AT91_RTT_ALMIEN | AT91_RTT_RTTINCIEN ) ;
rtt_writel ( rtc , MR , mr & ~ rtc - > imr ) ;
}
2013-04-29 16:20:58 -07:00
# ifdef CONFIG_PM_SLEEP
2008-02-06 01:38:59 -08:00
/* AT91SAM9 RTC Power management control */
2013-04-29 16:20:58 -07:00
static int at91_rtc_suspend ( struct device * dev )
2008-02-06 01:38:59 -08:00
{
2013-04-29 16:20:58 -07:00
struct sam9_rtc * rtc = dev_get_drvdata ( dev ) ;
2008-02-06 01:38:59 -08:00
u32 mr = rtt_readl ( rtc , MR ) ;
/*
* This IRQ is shared with DBGU and other hardware which isn ' t
* necessarily a wakeup event source .
*/
rtc - > imr = mr & ( AT91_RTT_ALMIEN | AT91_RTT_RTTINCIEN ) ;
if ( rtc - > imr ) {
2013-04-29 16:20:58 -07:00
if ( device_may_wakeup ( dev ) & & ( mr & AT91_RTT_ALMIEN ) ) {
2015-03-02 10:18:14 +01:00
unsigned long flags ;
2012-08-14 11:19:22 +02:00
enable_irq_wake ( rtc - > irq ) ;
2015-03-02 10:18:14 +01:00
spin_lock_irqsave ( & rtc - > lock , flags ) ;
rtc - > suspended = true ;
spin_unlock_irqrestore ( & rtc - > lock , flags ) ;
2008-02-06 01:38:59 -08:00
/* don't let RTTINC cause wakeups */
if ( mr & AT91_RTT_RTTINCIEN )
rtt_writel ( rtc , MR , mr & ~ AT91_RTT_RTTINCIEN ) ;
} else
rtt_writel ( rtc , MR , mr & ~ rtc - > imr ) ;
}
return 0 ;
}
2013-04-29 16:20:58 -07:00
static int at91_rtc_resume ( struct device * dev )
2008-02-06 01:38:59 -08:00
{
2013-04-29 16:20:58 -07:00
struct sam9_rtc * rtc = dev_get_drvdata ( dev ) ;
2008-02-06 01:38:59 -08:00
u32 mr ;
if ( rtc - > imr ) {
2015-03-02 10:18:14 +01:00
unsigned long flags ;
2013-04-29 16:20:58 -07:00
if ( device_may_wakeup ( dev ) )
2012-08-14 11:19:22 +02:00
disable_irq_wake ( rtc - > irq ) ;
2008-02-06 01:38:59 -08:00
mr = rtt_readl ( rtc , MR ) ;
rtt_writel ( rtc , MR , mr | rtc - > imr ) ;
2015-03-02 10:18:14 +01:00
spin_lock_irqsave ( & rtc - > lock , flags ) ;
rtc - > suspended = false ;
at91_rtc_cache_events ( rtc ) ;
at91_rtc_flush_events ( rtc ) ;
spin_unlock_irqrestore ( & rtc - > lock , flags ) ;
2008-02-06 01:38:59 -08:00
}
return 0 ;
}
# endif
2013-04-29 16:20:58 -07:00
static SIMPLE_DEV_PM_OPS ( at91_rtc_pm_ops , at91_rtc_suspend , at91_rtc_resume ) ;
2014-09-23 13:14:24 +02:00
# ifdef CONFIG_OF
static const struct of_device_id at91_rtc_dt_ids [ ] = {
{ . compatible = " atmel,at91sam9260-rtt " } ,
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( of , at91_rtc_dt_ids ) ;
# endif
2008-02-06 01:38:59 -08:00
static struct platform_driver at91_rtc_driver = {
2012-02-15 20:51:37 +08:00
. probe = at91_rtc_probe ,
2012-12-21 13:09:38 -08:00
. remove = at91_rtc_remove ,
2008-02-06 01:38:59 -08:00
. shutdown = at91_rtc_shutdown ,
2012-02-15 20:51:37 +08:00
. driver = {
. name = " rtc-at91sam9 " ,
2013-04-29 16:20:58 -07:00
. pm = & at91_rtc_pm_ops ,
2014-09-23 13:14:24 +02:00
. of_match_table = of_match_ptr ( at91_rtc_dt_ids ) ,
2012-02-15 20:51:37 +08:00
} ,
2008-02-06 01:38:59 -08:00
} ;
2012-10-04 17:13:54 -07:00
module_platform_driver ( at91_rtc_driver ) ;
2008-02-06 01:38:59 -08:00
MODULE_AUTHOR ( " Michel Benoit " ) ;
MODULE_DESCRIPTION ( " RTC driver for Atmel AT91SAM9x " ) ;
MODULE_LICENSE ( " GPL " ) ;