2019-05-27 09:55:06 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2014-08-26 07:54:55 +04:00
/*
* An RTC driver for Allwinner A31 / A23
*
* Copyright ( c ) 2014 , Chen - Yu Tsai < wens @ csie . org >
*
* based on rtc - sunxi . c
*
* An RTC driver for Allwinner A10 / A20
*
* Copyright ( c ) 2013 , Carlo Caione < carlo . caione @ gmail . com >
*/
2017-01-23 13:41:49 +03:00
# include <linux/clk.h>
# include <linux/clk-provider.h>
2014-08-26 07:54:55 +04:00
# include <linux/delay.h>
# include <linux/err.h>
# include <linux/fs.h>
# include <linux/init.h>
# include <linux/interrupt.h>
# include <linux/io.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/of.h>
# include <linux/of_address.h>
# include <linux/of_device.h>
# include <linux/platform_device.h>
# include <linux/rtc.h>
2017-01-23 13:41:49 +03:00
# include <linux/slab.h>
2014-08-26 07:54:55 +04:00
# include <linux/types.h>
/* Control register */
# define SUN6I_LOSC_CTRL 0x0000
2017-01-23 13:41:48 +03:00
# define SUN6I_LOSC_CTRL_KEY (0x16aa << 16)
2019-08-20 18:19:33 +03:00
# define SUN6I_LOSC_CTRL_AUTO_SWT_BYPASS BIT(15)
2014-08-26 07:54:55 +04:00
# define SUN6I_LOSC_CTRL_ALM_DHMS_ACC BIT(9)
# define SUN6I_LOSC_CTRL_RTC_HMS_ACC BIT(8)
# define SUN6I_LOSC_CTRL_RTC_YMD_ACC BIT(7)
2019-08-20 18:19:33 +03:00
# define SUN6I_LOSC_CTRL_EXT_LOSC_EN BIT(4)
2017-01-23 13:41:48 +03:00
# define SUN6I_LOSC_CTRL_EXT_OSC BIT(0)
2014-08-26 07:54:55 +04:00
# define SUN6I_LOSC_CTRL_ACC_MASK GENMASK(9, 7)
2017-01-23 13:41:49 +03:00
# define SUN6I_LOSC_CLK_PRESCAL 0x0008
2014-08-26 07:54:55 +04:00
/* RTC */
# define SUN6I_RTC_YMD 0x0010
# define SUN6I_RTC_HMS 0x0014
/* Alarm 0 (counter) */
# define SUN6I_ALRM_COUNTER 0x0020
# define SUN6I_ALRM_CUR_VAL 0x0024
# define SUN6I_ALRM_EN 0x0028
# define SUN6I_ALRM_EN_CNT_EN BIT(0)
# define SUN6I_ALRM_IRQ_EN 0x002c
# define SUN6I_ALRM_IRQ_EN_CNT_IRQ_EN BIT(0)
# define SUN6I_ALRM_IRQ_STA 0x0030
# define SUN6I_ALRM_IRQ_STA_CNT_IRQ_PEND BIT(0)
/* Alarm 1 (wall clock) */
# define SUN6I_ALRM1_EN 0x0044
# define SUN6I_ALRM1_IRQ_EN 0x0048
# define SUN6I_ALRM1_IRQ_STA 0x004c
# define SUN6I_ALRM1_IRQ_STA_WEEK_IRQ_PEND BIT(0)
/* Alarm config */
# define SUN6I_ALARM_CONFIG 0x0050
# define SUN6I_ALARM_CONFIG_WAKEUP BIT(0)
2017-08-25 10:42:02 +03:00
# define SUN6I_LOSC_OUT_GATING 0x0060
2018-05-30 21:27:44 +03:00
# define SUN6I_LOSC_OUT_GATING_EN_OFFSET 0
2017-08-25 10:42:02 +03:00
2014-08-26 07:54:55 +04:00
/*
* Get date values
*/
# define SUN6I_DATE_GET_DAY_VALUE(x) ((x) & 0x0000001f)
# define SUN6I_DATE_GET_MON_VALUE(x) (((x) & 0x00000f00) >> 8)
# define SUN6I_DATE_GET_YEAR_VALUE(x) (((x) & 0x003f0000) >> 16)
# define SUN6I_LEAP_GET_VALUE(x) (((x) & 0x00400000) >> 22)
/*
* Get time values
*/
# define SUN6I_TIME_GET_SEC_VALUE(x) ((x) & 0x0000003f)
# define SUN6I_TIME_GET_MIN_VALUE(x) (((x) & 0x00003f00) >> 8)
# define SUN6I_TIME_GET_HOUR_VALUE(x) (((x) & 0x001f0000) >> 16)
/*
* Set date values
*/
# define SUN6I_DATE_SET_DAY_VALUE(x) ((x) & 0x0000001f)
# define SUN6I_DATE_SET_MON_VALUE(x) ((x) << 8 & 0x00000f00)
# define SUN6I_DATE_SET_YEAR_VALUE(x) ((x) << 16 & 0x003f0000)
# define SUN6I_LEAP_SET_VALUE(x) ((x) << 22 & 0x00400000)
/*
* Set time values
*/
# define SUN6I_TIME_SET_SEC_VALUE(x) ((x) & 0x0000003f)
# define SUN6I_TIME_SET_MIN_VALUE(x) ((x) << 8 & 0x00003f00)
# define SUN6I_TIME_SET_HOUR_VALUE(x) ((x) << 16 & 0x001f0000)
/*
* The year parameter passed to the driver is usually an offset relative to
* the year 1900. This macro is used to convert this offset to another one
* relative to the minimum year allowed by the hardware .
*
* The year range is 1970 - 2033. This range is selected to match Allwinner ' s
* driver , even though it is somewhat limited .
*/
# define SUN6I_YEAR_MIN 1970
# define SUN6I_YEAR_OFF (SUN6I_YEAR_MIN - 1900)
2018-12-03 17:58:17 +03:00
/*
* There are other differences between models , including :
*
* - number of GPIO pins that can be configured to hold a certain level
* - crypto - key related registers ( H5 , H6 )
* - boot process related ( super standby , secondary processor entry address )
* registers ( R40 , H6 )
* - SYS power domain controls ( R40 )
* - DCXO controls ( H6 )
* - RC oscillator calibration ( H6 )
*
* These functions are not covered by this driver .
*/
struct sun6i_rtc_clk_data {
unsigned long rc_osc_rate ;
unsigned int fixed_prescaler : 16 ;
unsigned int has_prescaler : 1 ;
unsigned int has_out_clk : 1 ;
2018-12-03 17:58:19 +03:00
unsigned int export_iosc : 1 ;
2019-08-20 18:19:33 +03:00
unsigned int has_losc_en : 1 ;
unsigned int has_auto_swt : 1 ;
2018-12-03 17:58:17 +03:00
} ;
2014-08-26 07:54:55 +04:00
struct sun6i_rtc_dev {
struct rtc_device * rtc ;
2018-12-03 17:58:17 +03:00
const struct sun6i_rtc_clk_data * data ;
2014-08-26 07:54:55 +04:00
void __iomem * base ;
int irq ;
unsigned long alarm ;
2017-01-23 13:41:47 +03:00
2017-01-23 13:41:49 +03:00
struct clk_hw hw ;
struct clk_hw * int_osc ;
struct clk * losc ;
2017-08-25 10:42:02 +03:00
struct clk * ext_losc ;
2017-01-23 13:41:49 +03:00
2017-01-23 13:41:47 +03:00
spinlock_t lock ;
2014-08-26 07:54:55 +04:00
} ;
2017-01-23 13:41:49 +03:00
static struct sun6i_rtc_dev * sun6i_rtc ;
static unsigned long sun6i_rtc_osc_recalc_rate ( struct clk_hw * hw ,
unsigned long parent_rate )
{
struct sun6i_rtc_dev * rtc = container_of ( hw , struct sun6i_rtc_dev , hw ) ;
2018-12-03 17:58:17 +03:00
u32 val = 0 ;
2017-01-23 13:41:49 +03:00
val = readl ( rtc - > base + SUN6I_LOSC_CTRL ) ;
if ( val & SUN6I_LOSC_CTRL_EXT_OSC )
return parent_rate ;
2018-12-03 17:58:17 +03:00
if ( rtc - > data - > fixed_prescaler )
parent_rate / = rtc - > data - > fixed_prescaler ;
if ( rtc - > data - > has_prescaler ) {
val = readl ( rtc - > base + SUN6I_LOSC_CLK_PRESCAL ) ;
val & = GENMASK ( 4 , 0 ) ;
}
2017-01-23 13:41:49 +03:00
return parent_rate / ( val + 1 ) ;
}
static u8 sun6i_rtc_osc_get_parent ( struct clk_hw * hw )
{
struct sun6i_rtc_dev * rtc = container_of ( hw , struct sun6i_rtc_dev , hw ) ;
return readl ( rtc - > base + SUN6I_LOSC_CTRL ) & SUN6I_LOSC_CTRL_EXT_OSC ;
}
static int sun6i_rtc_osc_set_parent ( struct clk_hw * hw , u8 index )
{
struct sun6i_rtc_dev * rtc = container_of ( hw , struct sun6i_rtc_dev , hw ) ;
unsigned long flags ;
u32 val ;
if ( index > 1 )
return - EINVAL ;
spin_lock_irqsave ( & rtc - > lock , flags ) ;
val = readl ( rtc - > base + SUN6I_LOSC_CTRL ) ;
val & = ~ SUN6I_LOSC_CTRL_EXT_OSC ;
val | = SUN6I_LOSC_CTRL_KEY ;
val | = index ? SUN6I_LOSC_CTRL_EXT_OSC : 0 ;
2019-08-20 18:19:33 +03:00
if ( rtc - > data - > has_losc_en ) {
val & = ~ SUN6I_LOSC_CTRL_EXT_LOSC_EN ;
val | = index ? SUN6I_LOSC_CTRL_EXT_LOSC_EN : 0 ;
}
2017-01-23 13:41:49 +03:00
writel ( val , rtc - > base + SUN6I_LOSC_CTRL ) ;
spin_unlock_irqrestore ( & rtc - > lock , flags ) ;
return 0 ;
}
static const struct clk_ops sun6i_rtc_osc_ops = {
. recalc_rate = sun6i_rtc_osc_recalc_rate ,
. get_parent = sun6i_rtc_osc_get_parent ,
. set_parent = sun6i_rtc_osc_set_parent ,
} ;
2018-12-03 17:58:17 +03:00
static void __init sun6i_rtc_clk_init ( struct device_node * node ,
const struct sun6i_rtc_clk_data * data )
2017-01-23 13:41:49 +03:00
{
struct clk_hw_onecell_data * clk_data ;
struct sun6i_rtc_dev * rtc ;
struct clk_init_data init = {
. ops = & sun6i_rtc_osc_ops ,
2018-12-03 17:58:16 +03:00
. name = " losc " ,
2017-01-23 13:41:49 +03:00
} ;
2018-12-03 17:58:19 +03:00
const char * iosc_name = " rtc-int-osc " ;
2017-08-25 10:42:02 +03:00
const char * clkout_name = " osc32k-out " ;
2017-01-23 13:41:49 +03:00
const char * parents [ 2 ] ;
2019-08-20 18:19:33 +03:00
u32 reg ;
2017-01-23 13:41:49 +03:00
rtc = kzalloc ( sizeof ( * rtc ) , GFP_KERNEL ) ;
if ( ! rtc )
return ;
2018-12-03 17:58:17 +03:00
rtc - > data = data ;
2018-12-03 17:58:19 +03:00
clk_data = kzalloc ( struct_size ( clk_data , hws , 3 ) , GFP_KERNEL ) ;
2017-11-22 20:16:18 +03:00
if ( ! clk_data ) {
kfree ( rtc ) ;
2017-01-23 13:41:49 +03:00
return ;
2017-11-22 20:16:18 +03:00
}
2017-07-12 13:59:48 +03:00
2017-01-23 13:41:49 +03:00
spin_lock_init ( & rtc - > lock ) ;
rtc - > base = of_io_request_and_map ( node , 0 , of_node_full_name ( node ) ) ;
2017-02-09 03:16:13 +03:00
if ( IS_ERR ( rtc - > base ) ) {
2017-01-23 13:41:49 +03:00
pr_crit ( " Can't map RTC registers " ) ;
2017-07-19 19:57:02 +03:00
goto err ;
2017-01-23 13:41:49 +03:00
}
2019-08-20 18:19:33 +03:00
reg = SUN6I_LOSC_CTRL_KEY ;
if ( rtc - > data - > has_auto_swt ) {
/* Bypass auto-switch to int osc, on ext losc failure */
reg | = SUN6I_LOSC_CTRL_AUTO_SWT_BYPASS ;
writel ( reg , rtc - > base + SUN6I_LOSC_CTRL ) ;
}
2020-03-08 16:58:48 +03:00
/* Switch to the external, more precise, oscillator, if present */
if ( of_get_property ( node , " clocks " , NULL ) ) {
reg | = SUN6I_LOSC_CTRL_EXT_OSC ;
if ( rtc - > data - > has_losc_en )
reg | = SUN6I_LOSC_CTRL_EXT_LOSC_EN ;
}
2019-08-20 18:19:33 +03:00
writel ( reg , rtc - > base + SUN6I_LOSC_CTRL ) ;
2017-01-23 13:41:49 +03:00
2017-01-29 13:13:43 +03:00
/* Yes, I know, this is ugly. */
sun6i_rtc = rtc ;
2018-12-03 17:58:19 +03:00
/* Only read IOSC name from device tree if it is exported */
if ( rtc - > data - > export_iosc )
of_property_read_string_index ( node , " clock-output-names " , 2 ,
& iosc_name ) ;
2017-01-23 13:41:49 +03:00
rtc - > int_osc = clk_hw_register_fixed_rate_with_accuracy ( NULL ,
2018-12-03 17:58:19 +03:00
iosc_name ,
2017-01-23 13:41:49 +03:00
NULL , 0 ,
2018-12-03 17:58:17 +03:00
rtc - > data - > rc_osc_rate ,
2017-01-23 13:41:49 +03:00
300000000 ) ;
if ( IS_ERR ( rtc - > int_osc ) ) {
pr_crit ( " Couldn't register the internal oscillator \n " ) ;
return ;
}
parents [ 0 ] = clk_hw_get_name ( rtc - > int_osc ) ;
2020-03-08 16:58:48 +03:00
/* If there is no external oscillator, this will be NULL and ... */
2017-01-23 13:41:49 +03:00
parents [ 1 ] = of_clk_get_parent_name ( node , 0 ) ;
rtc - > hw . init = & init ;
init . parent_names = parents ;
2020-03-08 16:58:48 +03:00
/* ... number of clock parents will be 1. */
2017-01-23 13:41:49 +03:00
init . num_parents = of_clk_get_parent_count ( node ) + 1 ;
2017-08-25 10:42:02 +03:00
of_property_read_string_index ( node , " clock-output-names " , 0 ,
& init . name ) ;
2017-01-23 13:41:49 +03:00
rtc - > losc = clk_register ( NULL , & rtc - > hw ) ;
if ( IS_ERR ( rtc - > losc ) ) {
pr_crit ( " Couldn't register the LOSC clock \n " ) ;
return ;
}
2017-08-25 10:42:02 +03:00
of_property_read_string_index ( node , " clock-output-names " , 1 ,
& clkout_name ) ;
2019-08-15 19:00:19 +03:00
rtc - > ext_losc = clk_register_gate ( NULL , clkout_name , init . name ,
2017-08-25 10:42:02 +03:00
0 , rtc - > base + SUN6I_LOSC_OUT_GATING ,
2018-05-30 21:27:44 +03:00
SUN6I_LOSC_OUT_GATING_EN_OFFSET , 0 ,
2017-08-25 10:42:02 +03:00
& rtc - > lock ) ;
if ( IS_ERR ( rtc - > ext_losc ) ) {
pr_crit ( " Couldn't register the LOSC external gate \n " ) ;
return ;
}
clk_data - > num = 2 ;
2017-01-23 13:41:49 +03:00
clk_data - > hws [ 0 ] = & rtc - > hw ;
2017-08-25 10:42:02 +03:00
clk_data - > hws [ 1 ] = __clk_get_hw ( rtc - > ext_losc ) ;
2018-12-03 17:58:19 +03:00
if ( rtc - > data - > export_iosc ) {
clk_data - > hws [ 2 ] = rtc - > int_osc ;
clk_data - > num = 3 ;
}
2017-01-23 13:41:49 +03:00
of_clk_add_hw_provider ( node , of_clk_hw_onecell_get , clk_data ) ;
2017-07-19 19:57:02 +03:00
return ;
err :
kfree ( clk_data ) ;
2017-01-23 13:41:49 +03:00
}
2018-12-03 17:58:17 +03:00
static const struct sun6i_rtc_clk_data sun6i_a31_rtc_data = {
. rc_osc_rate = 667000 , /* datasheet says 600 ~ 700 KHz */
. has_prescaler = 1 ,
} ;
static void __init sun6i_a31_rtc_clk_init ( struct device_node * node )
{
sun6i_rtc_clk_init ( node , & sun6i_a31_rtc_data ) ;
}
CLK_OF_DECLARE_DRIVER ( sun6i_a31_rtc_clk , " allwinner,sun6i-a31-rtc " ,
sun6i_a31_rtc_clk_init ) ;
2017-01-23 13:41:49 +03:00
2018-12-03 17:58:18 +03:00
static const struct sun6i_rtc_clk_data sun8i_a23_rtc_data = {
. rc_osc_rate = 667000 , /* datasheet says 600 ~ 700 KHz */
. has_prescaler = 1 ,
. has_out_clk = 1 ,
} ;
static void __init sun8i_a23_rtc_clk_init ( struct device_node * node )
{
sun6i_rtc_clk_init ( node , & sun8i_a23_rtc_data ) ;
}
CLK_OF_DECLARE_DRIVER ( sun8i_a23_rtc_clk , " allwinner,sun8i-a23-rtc " ,
sun8i_a23_rtc_clk_init ) ;
static const struct sun6i_rtc_clk_data sun8i_h3_rtc_data = {
. rc_osc_rate = 16000000 ,
. fixed_prescaler = 32 ,
. has_prescaler = 1 ,
. has_out_clk = 1 ,
2018-12-03 17:58:19 +03:00
. export_iosc = 1 ,
2018-12-03 17:58:18 +03:00
} ;
static void __init sun8i_h3_rtc_clk_init ( struct device_node * node )
{
sun6i_rtc_clk_init ( node , & sun8i_h3_rtc_data ) ;
}
CLK_OF_DECLARE_DRIVER ( sun8i_h3_rtc_clk , " allwinner,sun8i-h3-rtc " ,
sun8i_h3_rtc_clk_init ) ;
/* As far as we are concerned, clocks for H5 are the same as H3 */
CLK_OF_DECLARE_DRIVER ( sun50i_h5_rtc_clk , " allwinner,sun50i-h5-rtc " ,
sun8i_h3_rtc_clk_init ) ;
2019-08-20 18:19:33 +03:00
static const struct sun6i_rtc_clk_data sun50i_h6_rtc_data = {
. rc_osc_rate = 16000000 ,
. fixed_prescaler = 32 ,
. has_prescaler = 1 ,
. has_out_clk = 1 ,
. export_iosc = 1 ,
. has_losc_en = 1 ,
. has_auto_swt = 1 ,
} ;
static void __init sun50i_h6_rtc_clk_init ( struct device_node * node )
{
sun6i_rtc_clk_init ( node , & sun50i_h6_rtc_data ) ;
}
CLK_OF_DECLARE_DRIVER ( sun50i_h6_rtc_clk , " allwinner,sun50i-h6-rtc " ,
sun50i_h6_rtc_clk_init ) ;
2019-12-05 11:50:54 +03:00
/*
* The R40 user manual is self - conflicting on whether the prescaler is
* fixed or configurable . The clock diagram shows it as fixed , but there
* is also a configurable divider in the RTC block .
*/
static const struct sun6i_rtc_clk_data sun8i_r40_rtc_data = {
. rc_osc_rate = 16000000 ,
. fixed_prescaler = 512 ,
} ;
static void __init sun8i_r40_rtc_clk_init ( struct device_node * node )
{
sun6i_rtc_clk_init ( node , & sun8i_r40_rtc_data ) ;
}
CLK_OF_DECLARE_DRIVER ( sun8i_r40_rtc_clk , " allwinner,sun8i-r40-rtc " ,
sun8i_r40_rtc_clk_init ) ;
2018-12-03 17:58:18 +03:00
static const struct sun6i_rtc_clk_data sun8i_v3_rtc_data = {
. rc_osc_rate = 32000 ,
. has_out_clk = 1 ,
} ;
static void __init sun8i_v3_rtc_clk_init ( struct device_node * node )
{
sun6i_rtc_clk_init ( node , & sun8i_v3_rtc_data ) ;
}
CLK_OF_DECLARE_DRIVER ( sun8i_v3_rtc_clk , " allwinner,sun8i-v3-rtc " ,
sun8i_v3_rtc_clk_init ) ;
2014-08-26 07:54:55 +04:00
static irqreturn_t sun6i_rtc_alarmirq ( int irq , void * id )
{
struct sun6i_rtc_dev * chip = ( struct sun6i_rtc_dev * ) id ;
2017-01-23 13:41:47 +03:00
irqreturn_t ret = IRQ_NONE ;
2014-08-26 07:54:55 +04:00
u32 val ;
2017-01-23 13:41:47 +03:00
spin_lock ( & chip - > lock ) ;
2014-08-26 07:54:55 +04:00
val = readl ( chip - > base + SUN6I_ALRM_IRQ_STA ) ;
if ( val & SUN6I_ALRM_IRQ_STA_CNT_IRQ_PEND ) {
val | = SUN6I_ALRM_IRQ_STA_CNT_IRQ_PEND ;
writel ( val , chip - > base + SUN6I_ALRM_IRQ_STA ) ;
rtc_update_irq ( chip - > rtc , 1 , RTC_AF | RTC_IRQF ) ;
2017-01-23 13:41:47 +03:00
ret = IRQ_HANDLED ;
2014-08-26 07:54:55 +04:00
}
2017-01-23 13:41:47 +03:00
spin_unlock ( & chip - > lock ) ;
2014-08-26 07:54:55 +04:00
2017-01-23 13:41:47 +03:00
return ret ;
2014-08-26 07:54:55 +04:00
}
static void sun6i_rtc_setaie ( int to , struct sun6i_rtc_dev * chip )
{
u32 alrm_val = 0 ;
u32 alrm_irq_val = 0 ;
u32 alrm_wake_val = 0 ;
2017-01-23 13:41:47 +03:00
unsigned long flags ;
2014-08-26 07:54:55 +04:00
if ( to ) {
alrm_val = SUN6I_ALRM_EN_CNT_EN ;
alrm_irq_val = SUN6I_ALRM_IRQ_EN_CNT_IRQ_EN ;
alrm_wake_val = SUN6I_ALARM_CONFIG_WAKEUP ;
} else {
writel ( SUN6I_ALRM_IRQ_STA_CNT_IRQ_PEND ,
chip - > base + SUN6I_ALRM_IRQ_STA ) ;
}
2017-01-23 13:41:47 +03:00
spin_lock_irqsave ( & chip - > lock , flags ) ;
2014-08-26 07:54:55 +04:00
writel ( alrm_val , chip - > base + SUN6I_ALRM_EN ) ;
writel ( alrm_irq_val , chip - > base + SUN6I_ALRM_IRQ_EN ) ;
writel ( alrm_wake_val , chip - > base + SUN6I_ALARM_CONFIG ) ;
2017-01-23 13:41:47 +03:00
spin_unlock_irqrestore ( & chip - > lock , flags ) ;
2014-08-26 07:54:55 +04:00
}
static int sun6i_rtc_gettime ( struct device * dev , struct rtc_time * rtc_tm )
{
struct sun6i_rtc_dev * chip = dev_get_drvdata ( dev ) ;
u32 date , time ;
/*
* read again in case it changes
*/
do {
date = readl ( chip - > base + SUN6I_RTC_YMD ) ;
time = readl ( chip - > base + SUN6I_RTC_HMS ) ;
} while ( ( date ! = readl ( chip - > base + SUN6I_RTC_YMD ) ) | |
( time ! = readl ( chip - > base + SUN6I_RTC_HMS ) ) ) ;
rtc_tm - > tm_sec = SUN6I_TIME_GET_SEC_VALUE ( time ) ;
rtc_tm - > tm_min = SUN6I_TIME_GET_MIN_VALUE ( time ) ;
rtc_tm - > tm_hour = SUN6I_TIME_GET_HOUR_VALUE ( time ) ;
rtc_tm - > tm_mday = SUN6I_DATE_GET_DAY_VALUE ( date ) ;
rtc_tm - > tm_mon = SUN6I_DATE_GET_MON_VALUE ( date ) ;
rtc_tm - > tm_year = SUN6I_DATE_GET_YEAR_VALUE ( date ) ;
rtc_tm - > tm_mon - = 1 ;
/*
* switch from ( data_year - > min ) - relative offset to
* a ( 1900 ) - relative one
*/
rtc_tm - > tm_year + = SUN6I_YEAR_OFF ;
2018-02-19 18:23:56 +03:00
return 0 ;
2014-08-26 07:54:55 +04:00
}
static int sun6i_rtc_getalarm ( struct device * dev , struct rtc_wkalrm * wkalrm )
{
struct sun6i_rtc_dev * chip = dev_get_drvdata ( dev ) ;
2017-01-23 13:41:47 +03:00
unsigned long flags ;
2014-08-26 07:54:55 +04:00
u32 alrm_st ;
u32 alrm_en ;
2017-01-23 13:41:47 +03:00
spin_lock_irqsave ( & chip - > lock , flags ) ;
2014-08-26 07:54:55 +04:00
alrm_en = readl ( chip - > base + SUN6I_ALRM_IRQ_EN ) ;
alrm_st = readl ( chip - > base + SUN6I_ALRM_IRQ_STA ) ;
2017-01-23 13:41:47 +03:00
spin_unlock_irqrestore ( & chip - > lock , flags ) ;
2014-08-26 07:54:55 +04:00
wkalrm - > enabled = ! ! ( alrm_en & SUN6I_ALRM_EN_CNT_EN ) ;
wkalrm - > pending = ! ! ( alrm_st & SUN6I_ALRM_EN_CNT_EN ) ;
2020-03-30 23:12:26 +03:00
rtc_time64_to_tm ( chip - > alarm , & wkalrm - > time ) ;
2014-08-26 07:54:55 +04:00
return 0 ;
}
static int sun6i_rtc_setalarm ( struct device * dev , struct rtc_wkalrm * wkalrm )
{
struct sun6i_rtc_dev * chip = dev_get_drvdata ( dev ) ;
struct rtc_time * alrm_tm = & wkalrm - > time ;
struct rtc_time tm_now ;
unsigned long time_now = 0 ;
unsigned long time_set = 0 ;
unsigned long time_gap = 0 ;
int ret = 0 ;
ret = sun6i_rtc_gettime ( dev , & tm_now ) ;
if ( ret < 0 ) {
dev_err ( dev , " Error in getting time \n " ) ;
return - EINVAL ;
}
2020-03-30 23:12:26 +03:00
time_set = rtc_tm_to_time64 ( alrm_tm ) ;
time_now = rtc_tm_to_time64 ( & tm_now ) ;
2014-08-26 07:54:55 +04:00
if ( time_set < = time_now ) {
dev_err ( dev , " Date to set in the past \n " ) ;
return - EINVAL ;
}
time_gap = time_set - time_now ;
if ( time_gap > U32_MAX ) {
dev_err ( dev , " Date too far in the future \n " ) ;
return - EINVAL ;
}
sun6i_rtc_setaie ( 0 , chip ) ;
writel ( 0 , chip - > base + SUN6I_ALRM_COUNTER ) ;
usleep_range ( 100 , 300 ) ;
writel ( time_gap , chip - > base + SUN6I_ALRM_COUNTER ) ;
chip - > alarm = time_set ;
sun6i_rtc_setaie ( wkalrm - > enabled , chip ) ;
return 0 ;
}
static int sun6i_rtc_wait ( struct sun6i_rtc_dev * chip , int offset ,
unsigned int mask , unsigned int ms_timeout )
{
const unsigned long timeout = jiffies + msecs_to_jiffies ( ms_timeout ) ;
u32 reg ;
do {
reg = readl ( chip - > base + offset ) ;
reg & = mask ;
if ( ! reg )
return 0 ;
} while ( time_before ( jiffies , timeout ) ) ;
return - ETIMEDOUT ;
}
static int sun6i_rtc_settime ( struct device * dev , struct rtc_time * rtc_tm )
{
struct sun6i_rtc_dev * chip = dev_get_drvdata ( dev ) ;
u32 date = 0 ;
u32 time = 0 ;
rtc_tm - > tm_year - = SUN6I_YEAR_OFF ;
rtc_tm - > tm_mon + = 1 ;
date = SUN6I_DATE_SET_DAY_VALUE ( rtc_tm - > tm_mday ) |
SUN6I_DATE_SET_MON_VALUE ( rtc_tm - > tm_mon ) |
SUN6I_DATE_SET_YEAR_VALUE ( rtc_tm - > tm_year ) ;
2020-03-30 23:12:25 +03:00
if ( is_leap_year ( rtc_tm - > tm_year + SUN6I_YEAR_MIN ) )
2014-08-26 07:54:55 +04:00
date | = SUN6I_LEAP_SET_VALUE ( 1 ) ;
time = SUN6I_TIME_SET_SEC_VALUE ( rtc_tm - > tm_sec ) |
SUN6I_TIME_SET_MIN_VALUE ( rtc_tm - > tm_min ) |
SUN6I_TIME_SET_HOUR_VALUE ( rtc_tm - > tm_hour ) ;
/* Check whether registers are writable */
if ( sun6i_rtc_wait ( chip , SUN6I_LOSC_CTRL ,
SUN6I_LOSC_CTRL_ACC_MASK , 50 ) ) {
dev_err ( dev , " rtc is still busy. \n " ) ;
return - EBUSY ;
}
writel ( time , chip - > base + SUN6I_RTC_HMS ) ;
/*
* After writing the RTC HH - MM - SS register , the
* SUN6I_LOSC_CTRL_RTC_HMS_ACC bit is set and it will not
* be cleared until the real writing operation is finished
*/
if ( sun6i_rtc_wait ( chip , SUN6I_LOSC_CTRL ,
SUN6I_LOSC_CTRL_RTC_HMS_ACC , 50 ) ) {
dev_err ( dev , " Failed to set rtc time. \n " ) ;
return - ETIMEDOUT ;
}
writel ( date , chip - > base + SUN6I_RTC_YMD ) ;
/*
* After writing the RTC YY - MM - DD register , the
* SUN6I_LOSC_CTRL_RTC_YMD_ACC bit is set and it will not
* be cleared until the real writing operation is finished
*/
if ( sun6i_rtc_wait ( chip , SUN6I_LOSC_CTRL ,
SUN6I_LOSC_CTRL_RTC_YMD_ACC , 50 ) ) {
dev_err ( dev , " Failed to set rtc time. \n " ) ;
return - ETIMEDOUT ;
}
return 0 ;
}
static int sun6i_rtc_alarm_irq_enable ( struct device * dev , unsigned int enabled )
{
struct sun6i_rtc_dev * chip = dev_get_drvdata ( dev ) ;
if ( ! enabled )
sun6i_rtc_setaie ( enabled , chip ) ;
return 0 ;
}
static const struct rtc_class_ops sun6i_rtc_ops = {
. read_time = sun6i_rtc_gettime ,
. set_time = sun6i_rtc_settime ,
. read_alarm = sun6i_rtc_getalarm ,
. set_alarm = sun6i_rtc_setalarm ,
. alarm_irq_enable = sun6i_rtc_alarm_irq_enable
} ;
2019-08-22 00:00:56 +03:00
# ifdef CONFIG_PM_SLEEP
/* Enable IRQ wake on suspend, to wake up from RTC. */
static int sun6i_rtc_suspend ( struct device * dev )
{
struct sun6i_rtc_dev * chip = dev_get_drvdata ( dev ) ;
if ( device_may_wakeup ( dev ) )
enable_irq_wake ( chip - > irq ) ;
return 0 ;
}
/* Disable IRQ wake on resume. */
static int sun6i_rtc_resume ( struct device * dev )
{
struct sun6i_rtc_dev * chip = dev_get_drvdata ( dev ) ;
if ( device_may_wakeup ( dev ) )
disable_irq_wake ( chip - > irq ) ;
return 0 ;
}
# endif
static SIMPLE_DEV_PM_OPS ( sun6i_rtc_pm_ops ,
sun6i_rtc_suspend , sun6i_rtc_resume ) ;
2014-08-26 07:54:55 +04:00
static int sun6i_rtc_probe ( struct platform_device * pdev )
{
2017-01-23 13:41:49 +03:00
struct sun6i_rtc_dev * chip = sun6i_rtc ;
2014-08-26 07:54:55 +04:00
int ret ;
if ( ! chip )
2017-01-23 13:41:49 +03:00
return - ENODEV ;
2014-08-26 07:54:55 +04:00
platform_set_drvdata ( pdev , chip ) ;
chip - > irq = platform_get_irq ( pdev , 0 ) ;
2019-07-30 21:15:39 +03:00
if ( chip - > irq < 0 )
2014-08-26 07:54:55 +04:00
return chip - > irq ;
ret = devm_request_irq ( & pdev - > dev , chip - > irq , sun6i_rtc_alarmirq ,
0 , dev_name ( & pdev - > dev ) , chip ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " Could not request IRQ \n " ) ;
return ret ;
}
/* clear the alarm counter value */
writel ( 0 , chip - > base + SUN6I_ALRM_COUNTER ) ;
/* disable counter alarm */
writel ( 0 , chip - > base + SUN6I_ALRM_EN ) ;
/* disable counter alarm interrupt */
writel ( 0 , chip - > base + SUN6I_ALRM_IRQ_EN ) ;
/* disable week alarm */
writel ( 0 , chip - > base + SUN6I_ALRM1_EN ) ;
/* disable week alarm interrupt */
writel ( 0 , chip - > base + SUN6I_ALRM1_IRQ_EN ) ;
/* clear counter alarm pending interrupts */
writel ( SUN6I_ALRM_IRQ_STA_CNT_IRQ_PEND ,
chip - > base + SUN6I_ALRM_IRQ_STA ) ;
/* clear week alarm pending interrupts */
writel ( SUN6I_ALRM1_IRQ_STA_WEEK_IRQ_PEND ,
chip - > base + SUN6I_ALRM1_IRQ_STA ) ;
/* disable alarm wakeup */
writel ( 0 , chip - > base + SUN6I_ALARM_CONFIG ) ;
2017-01-23 13:41:49 +03:00
clk_prepare_enable ( chip - > losc ) ;
2017-01-23 13:41:48 +03:00
2019-08-22 00:00:56 +03:00
device_init_wakeup ( & pdev - > dev , 1 ) ;
2020-03-30 23:12:25 +03:00
chip - > rtc = devm_rtc_allocate_device ( & pdev - > dev ) ;
if ( IS_ERR ( chip - > rtc ) )
2014-08-26 07:54:55 +04:00
return PTR_ERR ( chip - > rtc ) ;
2020-03-30 23:12:25 +03:00
chip - > rtc - > ops = & sun6i_rtc_ops ;
chip - > rtc - > range_max = 2019686399LL ; /* 2033-12-31 23:59:59 */
ret = rtc_register_device ( chip - > rtc ) ;
if ( ret )
return ret ;
2014-08-26 07:54:55 +04:00
dev_info ( & pdev - > dev , " RTC enabled \n " ) ;
return 0 ;
}
2018-12-03 17:58:17 +03:00
/*
* As far as RTC functionality goes , all models are the same . The
* datasheets claim that different models have different number of
* registers available for non - volatile storage , but experiments show
* that all SoCs have 16 registers available for this purpose .
*/
2014-08-26 07:54:55 +04:00
static const struct of_device_id sun6i_rtc_dt_ids [ ] = {
{ . compatible = " allwinner,sun6i-a31-rtc " } ,
2018-12-03 17:58:18 +03:00
{ . compatible = " allwinner,sun8i-a23-rtc " } ,
{ . compatible = " allwinner,sun8i-h3-rtc " } ,
2019-05-28 23:30:36 +03:00
{ . compatible = " allwinner,sun8i-r40-rtc " } ,
2018-12-03 17:58:18 +03:00
{ . compatible = " allwinner,sun8i-v3-rtc " } ,
{ . compatible = " allwinner,sun50i-h5-rtc " } ,
2019-08-20 18:19:33 +03:00
{ . compatible = " allwinner,sun50i-h6-rtc " } ,
2014-08-26 07:54:55 +04:00
{ /* sentinel */ } ,
} ;
MODULE_DEVICE_TABLE ( of , sun6i_rtc_dt_ids ) ;
static struct platform_driver sun6i_rtc_driver = {
. probe = sun6i_rtc_probe ,
. driver = {
. name = " sun6i-rtc " ,
. of_match_table = sun6i_rtc_dt_ids ,
2019-08-22 00:00:56 +03:00
. pm = & sun6i_rtc_pm_ops ,
2014-08-26 07:54:55 +04:00
} ,
} ;
2017-01-23 13:41:46 +03:00
builtin_platform_driver ( sun6i_rtc_driver ) ;