2019-03-20 13:32:29 +01:00
// SPDX-License-Identifier: GPL-2.0+
2014-06-06 14:35:42 -07:00
/*
* APM X - Gene SoC Real Time Clock Driver
*
* Copyright ( c ) 2014 , Applied Micro Circuits Corporation
* Author : Rameshwar Prasad Sahu < rsahu @ apm . com >
* Loc Ho < lho @ apm . com >
*/
2019-03-20 13:32:30 +01:00
# include <linux/clk.h>
# include <linux/delay.h>
2014-06-06 14:35:42 -07:00
# include <linux/init.h>
2019-03-20 13:32:30 +01:00
# include <linux/io.h>
2014-06-06 14:35:42 -07:00
# include <linux/module.h>
# include <linux/of.h>
# include <linux/platform_device.h>
# include <linux/rtc.h>
2019-03-20 13:32:30 +01:00
# include <linux/slab.h>
2014-06-06 14:35:42 -07:00
/* RTC CSR Registers */
# define RTC_CCVR 0x00
# define RTC_CMR 0x04
# define RTC_CLR 0x08
# define RTC_CCR 0x0C
# define RTC_CCR_IE BIT(0)
# define RTC_CCR_MASK BIT(1)
# define RTC_CCR_EN BIT(2)
# define RTC_CCR_WEN BIT(3)
# define RTC_STAT 0x10
# define RTC_STAT_BIT BIT(0)
# define RTC_RSTAT 0x14
# define RTC_EOI 0x18
# define RTC_VER 0x1C
struct xgene_rtc_dev {
struct rtc_device * rtc ;
void __iomem * csr_base ;
struct clk * clk ;
unsigned int irq_wake ;
2014-04-14 12:09:04 -06:00
unsigned int irq_enabled ;
2014-06-06 14:35:42 -07:00
} ;
static int xgene_rtc_read_time ( struct device * dev , struct rtc_time * tm )
{
struct xgene_rtc_dev * pdata = dev_get_drvdata ( dev ) ;
2019-03-20 13:32:32 +01:00
rtc_time64_to_tm ( readl ( pdata - > csr_base + RTC_CCVR ) , tm ) ;
2018-02-19 16:23:55 +01:00
return 0 ;
2014-06-06 14:35:42 -07:00
}
2019-03-20 13:32:33 +01:00
static int xgene_rtc_set_time ( struct device * dev , struct rtc_time * tm )
2014-06-06 14:35:42 -07:00
{
struct xgene_rtc_dev * pdata = dev_get_drvdata ( dev ) ;
/*
* NOTE : After the following write , the RTC_CCVR is only reflected
* after the update cycle of 1 seconds .
*/
2019-03-20 13:32:33 +01:00
writel ( ( u32 ) rtc_tm_to_time64 ( tm ) , pdata - > csr_base + RTC_CLR ) ;
2014-06-06 14:35:42 -07:00
readl ( pdata - > csr_base + RTC_CLR ) ; /* Force a barrier */
return 0 ;
}
static int xgene_rtc_read_alarm ( struct device * dev , struct rtc_wkalrm * alrm )
{
struct xgene_rtc_dev * pdata = dev_get_drvdata ( dev ) ;
2019-03-20 13:32:31 +01:00
/* If possible, CMR should be read here */
2019-03-20 13:32:32 +01:00
rtc_time64_to_tm ( 0 , & alrm - > time ) ;
2014-06-06 14:35:42 -07:00
alrm - > enabled = readl ( pdata - > csr_base + RTC_CCR ) & RTC_CCR_IE ;
return 0 ;
}
static int xgene_rtc_alarm_irq_enable ( struct device * dev , u32 enabled )
{
struct xgene_rtc_dev * pdata = dev_get_drvdata ( dev ) ;
u32 ccr ;
ccr = readl ( pdata - > csr_base + RTC_CCR ) ;
if ( enabled ) {
ccr & = ~ RTC_CCR_MASK ;
ccr | = RTC_CCR_IE ;
} else {
ccr & = ~ RTC_CCR_IE ;
ccr | = RTC_CCR_MASK ;
}
writel ( ccr , pdata - > csr_base + RTC_CCR ) ;
return 0 ;
}
2014-04-14 12:09:04 -06:00
static int xgene_rtc_alarm_irq_enabled ( struct device * dev )
{
struct xgene_rtc_dev * pdata = dev_get_drvdata ( dev ) ;
return readl ( pdata - > csr_base + RTC_CCR ) & RTC_CCR_IE ? 1 : 0 ;
}
2014-06-06 14:35:42 -07:00
static int xgene_rtc_set_alarm ( struct device * dev , struct rtc_wkalrm * alrm )
{
struct xgene_rtc_dev * pdata = dev_get_drvdata ( dev ) ;
2019-03-20 13:32:32 +01:00
writel ( ( u32 ) rtc_tm_to_time64 ( & alrm - > time ) , pdata - > csr_base + RTC_CMR ) ;
2014-06-06 14:35:42 -07:00
xgene_rtc_alarm_irq_enable ( dev , alrm - > enabled ) ;
return 0 ;
}
static const struct rtc_class_ops xgene_rtc_ops = {
. read_time = xgene_rtc_read_time ,
2019-03-20 13:32:33 +01:00
. set_time = xgene_rtc_set_time ,
2014-06-06 14:35:42 -07:00
. read_alarm = xgene_rtc_read_alarm ,
. set_alarm = xgene_rtc_set_alarm ,
. alarm_irq_enable = xgene_rtc_alarm_irq_enable ,
} ;
static irqreturn_t xgene_rtc_interrupt ( int irq , void * id )
{
2019-03-20 13:32:30 +01:00
struct xgene_rtc_dev * pdata = id ;
2014-06-06 14:35:42 -07:00
/* Check if interrupt asserted */
if ( ! ( readl ( pdata - > csr_base + RTC_STAT ) & RTC_STAT_BIT ) )
return IRQ_NONE ;
/* Clear interrupt */
readl ( pdata - > csr_base + RTC_EOI ) ;
rtc_update_irq ( pdata - > rtc , 1 , RTC_IRQF | RTC_AF ) ;
return IRQ_HANDLED ;
}
static int xgene_rtc_probe ( struct platform_device * pdev )
{
struct xgene_rtc_dev * pdata ;
int ret ;
int irq ;
pdata = devm_kzalloc ( & pdev - > dev , sizeof ( * pdata ) , GFP_KERNEL ) ;
if ( ! pdata )
return - ENOMEM ;
platform_set_drvdata ( pdev , pdata ) ;
2019-10-06 18:29:20 +08:00
pdata - > csr_base = devm_platform_ioremap_resource ( pdev , 0 ) ;
2014-06-06 14:35:42 -07:00
if ( IS_ERR ( pdata - > csr_base ) )
return PTR_ERR ( pdata - > csr_base ) ;
2019-03-20 13:32:27 +01:00
pdata - > rtc = devm_rtc_allocate_device ( & pdev - > dev ) ;
if ( IS_ERR ( pdata - > rtc ) )
return PTR_ERR ( pdata - > rtc ) ;
2014-06-06 14:35:42 -07:00
irq = platform_get_irq ( pdev , 0 ) ;
2019-07-30 11:15:39 -07:00
if ( irq < 0 )
2014-06-06 14:35:42 -07:00
return irq ;
ret = devm_request_irq ( & pdev - > dev , irq , xgene_rtc_interrupt , 0 ,
dev_name ( & pdev - > dev ) , pdata ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " Could not request IRQ \n " ) ;
return ret ;
}
pdata - > clk = devm_clk_get ( & pdev - > dev , NULL ) ;
if ( IS_ERR ( pdata - > clk ) ) {
dev_err ( & pdev - > dev , " Couldn't get the clock for RTC \n " ) ;
return - ENODEV ;
}
2014-04-14 12:09:04 -06:00
ret = clk_prepare_enable ( pdata - > clk ) ;
if ( ret )
return ret ;
2014-06-06 14:35:42 -07:00
/* Turn on the clock and the crystal */
writel ( RTC_CCR_EN , pdata - > csr_base + RTC_CCR ) ;
2014-04-14 12:09:04 -06:00
ret = device_init_wakeup ( & pdev - > dev , 1 ) ;
if ( ret ) {
clk_disable_unprepare ( pdata - > clk ) ;
return ret ;
}
2014-06-06 14:35:42 -07:00
2019-03-20 13:32:27 +01:00
pdata - > rtc - > ops = & xgene_rtc_ops ;
2019-03-20 13:32:28 +01:00
pdata - > rtc - > range_max = U32_MAX ;
2019-03-20 13:32:27 +01:00
2020-11-09 17:34:08 +01:00
ret = devm_rtc_register_device ( pdata - > rtc ) ;
2019-03-20 13:32:27 +01:00
if ( ret ) {
clk_disable_unprepare ( pdata - > clk ) ;
return ret ;
}
2014-06-06 14:35:42 -07:00
return 0 ;
}
static int xgene_rtc_remove ( struct platform_device * pdev )
{
struct xgene_rtc_dev * pdata = platform_get_drvdata ( pdev ) ;
xgene_rtc_alarm_irq_enable ( & pdev - > dev , 0 ) ;
device_init_wakeup ( & pdev - > dev , 0 ) ;
clk_disable_unprepare ( pdata - > clk ) ;
return 0 ;
}
2017-11-08 13:08:10 +01:00
static int __maybe_unused xgene_rtc_suspend ( struct device * dev )
2014-06-06 14:35:42 -07:00
{
struct platform_device * pdev = to_platform_device ( dev ) ;
struct xgene_rtc_dev * pdata = platform_get_drvdata ( pdev ) ;
int irq ;
irq = platform_get_irq ( pdev , 0 ) ;
2014-04-14 12:09:04 -06:00
/*
* If this RTC alarm will be used for waking the system up ,
* don ' t disable it of course . Else we just disable the alarm
* and await suspension .
*/
2014-06-06 14:35:42 -07:00
if ( device_may_wakeup ( & pdev - > dev ) ) {
if ( ! enable_irq_wake ( irq ) )
pdata - > irq_wake = 1 ;
} else {
2014-04-14 12:09:04 -06:00
pdata - > irq_enabled = xgene_rtc_alarm_irq_enabled ( dev ) ;
2014-06-06 14:35:42 -07:00
xgene_rtc_alarm_irq_enable ( dev , 0 ) ;
2014-04-14 12:09:04 -06:00
clk_disable_unprepare ( pdata - > clk ) ;
2014-06-06 14:35:42 -07:00
}
return 0 ;
}
2017-11-08 13:08:10 +01:00
static int __maybe_unused xgene_rtc_resume ( struct device * dev )
2014-06-06 14:35:42 -07:00
{
struct platform_device * pdev = to_platform_device ( dev ) ;
struct xgene_rtc_dev * pdata = platform_get_drvdata ( pdev ) ;
int irq ;
2014-04-14 12:09:04 -06:00
int rc ;
2014-06-06 14:35:42 -07:00
irq = platform_get_irq ( pdev , 0 ) ;
2014-04-14 12:09:04 -06:00
2014-06-06 14:35:42 -07:00
if ( device_may_wakeup ( & pdev - > dev ) ) {
if ( pdata - > irq_wake ) {
disable_irq_wake ( irq ) ;
pdata - > irq_wake = 0 ;
}
} else {
2014-04-14 12:09:04 -06:00
rc = clk_prepare_enable ( pdata - > clk ) ;
if ( rc ) {
dev_err ( dev , " Unable to enable clock error %d \n " , rc ) ;
return rc ;
}
xgene_rtc_alarm_irq_enable ( dev , pdata - > irq_enabled ) ;
2014-06-06 14:35:42 -07:00
}
return 0 ;
}
static SIMPLE_DEV_PM_OPS ( xgene_rtc_pm_ops , xgene_rtc_suspend , xgene_rtc_resume ) ;
# ifdef CONFIG_OF
static const struct of_device_id xgene_rtc_of_match [ ] = {
{ . compatible = " apm,xgene-rtc " } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , xgene_rtc_of_match ) ;
# endif
static struct platform_driver xgene_rtc_driver = {
. probe = xgene_rtc_probe ,
. remove = xgene_rtc_remove ,
. driver = {
. name = " xgene-rtc " ,
. pm = & xgene_rtc_pm_ops ,
. of_match_table = of_match_ptr ( xgene_rtc_of_match ) ,
} ,
} ;
module_platform_driver ( xgene_rtc_driver ) ;
MODULE_DESCRIPTION ( " APM X-Gene SoC RTC driver " ) ;
MODULE_AUTHOR ( " Rameshwar Sahu <rsahu@apm.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;