2009-08-31 01:49:04 +04:00
/*
* Copyright ( C ) 2007 - 2009 ST - Ericsson AB
* License terms : GNU General Public License ( GPL ) version 2
* RTC clock driver for the AB3100 Analog Baseband Chip
* Author : Linus Walleij < linus . walleij @ stericsson . com >
*/
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/platform_device.h>
# include <linux/rtc.h>
2010-05-01 20:26:07 +04:00
# include <linux/mfd/abx500.h>
2009-08-31 01:49:04 +04:00
/* Clock rate in Hz */
# define AB3100_RTC_CLOCK_RATE 32768
/*
* The AB3100 RTC registers . These are the same for
* AB3000 and AB3100 .
* Control register :
* Bit 0 : RTC Monitor cleared = 0 , active = 1 , if you set it
* to 1 it remains active until RTC power is lost .
* Bit 1 : 32 kHz Oscillator , 0 = on , 1 = bypass
* Bit 2 : Alarm on , 0 = off , 1 = on
* Bit 3 : 32 kHz buffer disabling , 0 = enabled , 1 = disabled
*/
# define AB3100_RTC 0x53
/* default setting, buffer disabled, alarm on */
# define RTC_SETTING 0x30
/* Alarm when AL0-AL3 == TI0-TI3 */
# define AB3100_AL0 0x56
# define AB3100_AL1 0x57
# define AB3100_AL2 0x58
# define AB3100_AL3 0x59
/* This 48-bit register that counts up at 32768 Hz */
# define AB3100_TI0 0x5a
# define AB3100_TI1 0x5b
# define AB3100_TI2 0x5c
# define AB3100_TI3 0x5d
# define AB3100_TI4 0x5e
# define AB3100_TI5 0x5f
/*
* RTC clock functions and device struct declaration
*/
2015-04-02 06:34:29 +03:00
static int ab3100_rtc_set_mmss ( struct device * dev , time64_t secs )
2009-08-31 01:49:04 +04:00
{
u8 regs [ ] = { AB3100_TI0 , AB3100_TI1 , AB3100_TI2 ,
AB3100_TI3 , AB3100_TI4 , AB3100_TI5 } ;
unsigned char buf [ 6 ] ;
2015-04-02 06:34:29 +03:00
u64 hw_counter = secs * AB3100_RTC_CLOCK_RATE * 2 ;
2009-08-31 01:49:04 +04:00
int err = 0 ;
int i ;
2015-04-02 06:34:29 +03:00
buf [ 0 ] = ( hw_counter ) & 0xFF ;
buf [ 1 ] = ( hw_counter > > 8 ) & 0xFF ;
buf [ 2 ] = ( hw_counter > > 16 ) & 0xFF ;
buf [ 3 ] = ( hw_counter > > 24 ) & 0xFF ;
buf [ 4 ] = ( hw_counter > > 32 ) & 0xFF ;
buf [ 5 ] = ( hw_counter > > 40 ) & 0xFF ;
2009-08-31 01:49:04 +04:00
for ( i = 0 ; i < 6 ; i + + ) {
2010-05-01 20:26:20 +04:00
err = abx500_set_register_interruptible ( dev , 0 ,
2009-08-31 01:49:04 +04:00
regs [ i ] , buf [ i ] ) ;
if ( err )
return err ;
}
/* Set the flag to mark that the clock is now set */
2010-05-01 20:26:20 +04:00
return abx500_mask_and_set_register_interruptible ( dev , 0 ,
2009-08-31 01:49:04 +04:00
AB3100_RTC ,
2010-05-01 20:26:20 +04:00
0x01 , 0x01 ) ;
2009-08-31 01:49:04 +04:00
}
static int ab3100_rtc_read_time ( struct device * dev , struct rtc_time * tm )
{
2015-04-02 06:34:29 +03:00
time64_t time ;
2009-08-31 01:49:04 +04:00
u8 rtcval ;
int err ;
2010-05-01 20:26:20 +04:00
err = abx500_get_register_interruptible ( dev , 0 ,
2009-08-31 01:49:04 +04:00
AB3100_RTC , & rtcval ) ;
if ( err )
return err ;
if ( ! ( rtcval & 0x01 ) ) {
dev_info ( dev , " clock not set (lost power) " ) ;
return - EINVAL ;
} else {
2015-04-02 06:34:29 +03:00
u64 hw_counter ;
2009-08-31 01:49:04 +04:00
u8 buf [ 6 ] ;
/* Read out time registers */
2010-05-01 20:26:20 +04:00
err = abx500_get_register_page_interruptible ( dev , 0 ,
2009-08-31 01:49:04 +04:00
AB3100_TI0 ,
buf , 6 ) ;
if ( err ! = 0 )
return err ;
2015-04-02 06:34:29 +03:00
hw_counter = ( ( u64 ) buf [ 5 ] < < 40 ) | ( ( u64 ) buf [ 4 ] < < 32 ) |
2009-08-31 01:49:04 +04:00
( ( u64 ) buf [ 3 ] < < 24 ) | ( ( u64 ) buf [ 2 ] < < 16 ) |
( ( u64 ) buf [ 1 ] < < 8 ) | ( u64 ) buf [ 0 ] ;
2015-04-02 06:34:29 +03:00
time = hw_counter / ( u64 ) ( AB3100_RTC_CLOCK_RATE * 2 ) ;
2009-08-31 01:49:04 +04:00
}
2015-04-02 06:34:29 +03:00
rtc_time64_to_tm ( time , tm ) ;
2009-08-31 01:49:04 +04:00
return rtc_valid_tm ( tm ) ;
}
static int ab3100_rtc_read_alarm ( struct device * dev , struct rtc_wkalrm * alarm )
{
2015-04-02 06:34:29 +03:00
time64_t time ;
u64 hw_counter ;
2009-08-31 01:49:04 +04:00
u8 buf [ 6 ] ;
u8 rtcval ;
int err ;
/* Figure out if alarm is enabled or not */
2010-05-01 20:26:20 +04:00
err = abx500_get_register_interruptible ( dev , 0 ,
2009-08-31 01:49:04 +04:00
AB3100_RTC , & rtcval ) ;
if ( err )
return err ;
if ( rtcval & 0x04 )
alarm - > enabled = 1 ;
else
alarm - > enabled = 0 ;
/* No idea how this could be represented */
alarm - > pending = 0 ;
/* Read out alarm registers, only 4 bytes */
2010-05-01 20:26:20 +04:00
err = abx500_get_register_page_interruptible ( dev , 0 ,
2009-08-31 01:49:04 +04:00
AB3100_AL0 , buf , 4 ) ;
if ( err )
return err ;
2015-04-02 06:34:29 +03:00
hw_counter = ( ( u64 ) buf [ 3 ] < < 40 ) | ( ( u64 ) buf [ 2 ] < < 32 ) |
2009-08-31 01:49:04 +04:00
( ( u64 ) buf [ 1 ] < < 24 ) | ( ( u64 ) buf [ 0 ] < < 16 ) ;
2015-04-02 06:34:29 +03:00
time = hw_counter / ( u64 ) ( AB3100_RTC_CLOCK_RATE * 2 ) ;
2009-08-31 01:49:04 +04:00
2015-04-02 06:34:29 +03:00
rtc_time64_to_tm ( time , & alarm - > time ) ;
2009-08-31 01:49:04 +04:00
return rtc_valid_tm ( & alarm - > time ) ;
}
static int ab3100_rtc_set_alarm ( struct device * dev , struct rtc_wkalrm * alarm )
{
u8 regs [ ] = { AB3100_AL0 , AB3100_AL1 , AB3100_AL2 , AB3100_AL3 } ;
unsigned char buf [ 4 ] ;
2015-04-02 06:34:29 +03:00
time64_t secs ;
u64 hw_counter ;
2009-08-31 01:49:04 +04:00
int err ;
int i ;
2015-04-02 06:34:29 +03:00
secs = rtc_tm_to_time64 ( & alarm - > time ) ;
hw_counter = secs * AB3100_RTC_CLOCK_RATE * 2 ;
buf [ 0 ] = ( hw_counter > > 16 ) & 0xFF ;
buf [ 1 ] = ( hw_counter > > 24 ) & 0xFF ;
buf [ 2 ] = ( hw_counter > > 32 ) & 0xFF ;
buf [ 3 ] = ( hw_counter > > 40 ) & 0xFF ;
2009-08-31 01:49:04 +04:00
/* Set the alarm */
for ( i = 0 ; i < 4 ; i + + ) {
2010-05-01 20:26:20 +04:00
err = abx500_set_register_interruptible ( dev , 0 ,
2009-08-31 01:49:04 +04:00
regs [ i ] , buf [ i ] ) ;
if ( err )
return err ;
}
/* Then enable the alarm */
2010-05-01 20:26:20 +04:00
return abx500_mask_and_set_register_interruptible ( dev , 0 ,
AB3100_RTC , ( 1 < < 2 ) ,
2009-08-31 01:49:04 +04:00
alarm - > enabled < < 2 ) ;
}
static int ab3100_rtc_irq_enable ( struct device * dev , unsigned int enabled )
{
/*
* It ' s not possible to enable / disable the alarm IRQ for this RTC .
* It does not actually trigger any IRQ : instead its only function is
* to power up the system , if it wasn ' t on . This will manifest as
* a " power up cause " in the AB3100 power driver ( battery charging etc )
* and need to be handled there instead .
*/
if ( enabled )
2010-05-01 20:26:20 +04:00
return abx500_mask_and_set_register_interruptible ( dev , 0 ,
AB3100_RTC , ( 1 < < 2 ) ,
2009-08-31 01:49:04 +04:00
1 < < 2 ) ;
else
2010-05-01 20:26:20 +04:00
return abx500_mask_and_set_register_interruptible ( dev , 0 ,
AB3100_RTC , ( 1 < < 2 ) ,
2009-08-31 01:49:04 +04:00
0 ) ;
}
static const struct rtc_class_ops ab3100_rtc_ops = {
. read_time = ab3100_rtc_read_time ,
2015-04-02 06:34:29 +03:00
. set_mmss64 = ab3100_rtc_set_mmss ,
2009-08-31 01:49:04 +04:00
. read_alarm = ab3100_rtc_read_alarm ,
. set_alarm = ab3100_rtc_set_alarm ,
. alarm_irq_enable = ab3100_rtc_irq_enable ,
} ;
static int __init ab3100_rtc_probe ( struct platform_device * pdev )
{
int err ;
u8 regval ;
struct rtc_device * rtc ;
/* The first RTC register needs special treatment */
2010-05-01 20:26:20 +04:00
err = abx500_get_register_interruptible ( & pdev - > dev , 0 ,
2009-08-31 01:49:04 +04:00
AB3100_RTC , & regval ) ;
if ( err ) {
dev_err ( & pdev - > dev , " unable to read RTC register \n " ) ;
return - ENODEV ;
}
if ( ( regval & 0xFE ) ! = RTC_SETTING ) {
dev_warn ( & pdev - > dev , " not default value in RTC reg 0x%x \n " ,
regval ) ;
}
if ( ( regval & 1 ) = = 0 ) {
/*
* Set bit to detect power loss .
* This bit remains until RTC power is lost .
*/
regval = 1 | RTC_SETTING ;
2010-05-01 20:26:20 +04:00
err = abx500_set_register_interruptible ( & pdev - > dev , 0 ,
2009-08-31 01:49:04 +04:00
AB3100_RTC , regval ) ;
/* Ignore any error on this write */
}
2013-04-30 03:19:30 +04:00
rtc = devm_rtc_device_register ( & pdev - > dev , " ab3100-rtc " ,
& ab3100_rtc_ops , THIS_MODULE ) ;
2009-08-31 01:49:04 +04:00
if ( IS_ERR ( rtc ) ) {
err = PTR_ERR ( rtc ) ;
return err ;
}
2010-09-23 00:04:59 +04:00
platform_set_drvdata ( pdev , rtc ) ;
2009-08-31 01:49:04 +04:00
return 0 ;
}
static struct platform_driver ab3100_rtc_driver = {
. driver = {
. name = " ab3100-rtc " ,
} ,
} ;
2013-04-30 03:18:34 +04:00
module_platform_driver_probe ( ab3100_rtc_driver , ab3100_rtc_probe ) ;
2009-08-31 01:49:04 +04:00
MODULE_AUTHOR ( " Linus Walleij <linus.walleij@stericsson.com> " ) ;
MODULE_DESCRIPTION ( " AB3100 RTC Driver " ) ;
MODULE_LICENSE ( " GPL " ) ;