2021-08-23 20:16:12 +03:00
// SPDX-License-Identifier: GPL-2.0-only
/*
* Real time clocks driver for MStar / SigmaStar ARMv7 SoCs .
* Based on " Real Time Clock driver for msb252x. " that was contained
* in various MStar kernels .
*
* ( C ) 2019 Daniel Palmer
* ( C ) 2021 Romain Perier
*/
# include <linux/clk.h>
# include <linux/delay.h>
2021-09-27 09:37:23 +03:00
# include <linux/io.h>
2021-08-23 20:16:12 +03:00
# include <linux/module.h>
# include <linux/mod_devicetable.h>
# include <linux/platform_device.h>
# include <linux/rtc.h>
/* Registers */
# define REG_RTC_CTRL 0x00
# define REG_RTC_FREQ_CW_L 0x04
# define REG_RTC_FREQ_CW_H 0x08
# define REG_RTC_LOAD_VAL_L 0x0C
# define REG_RTC_LOAD_VAL_H 0x10
# define REG_RTC_MATCH_VAL_L 0x14
# define REG_RTC_MATCH_VAL_H 0x18
# define REG_RTC_STATUS_INT 0x1C
# define REG_RTC_CNT_VAL_L 0x20
# define REG_RTC_CNT_VAL_H 0x24
/* Control bits for REG_RTC_CTRL */
# define SOFT_RSTZ_BIT BIT(0)
# define CNT_EN_BIT BIT(1)
# define WRAP_EN_BIT BIT(2)
# define LOAD_EN_BIT BIT(3)
# define READ_EN_BIT BIT(4)
# define INT_MASK_BIT BIT(5)
# define INT_FORCE_BIT BIT(6)
# define INT_CLEAR_BIT BIT(7)
/* Control bits for REG_RTC_STATUS_INT */
# define RAW_INT_BIT BIT(0)
# define ALM_INT_BIT BIT(1)
struct msc313_rtc {
struct rtc_device * rtc_dev ;
void __iomem * rtc_base ;
} ;
static int msc313_rtc_read_alarm ( struct device * dev , struct rtc_wkalrm * alarm )
{
struct msc313_rtc * priv = dev_get_drvdata ( dev ) ;
unsigned long seconds ;
seconds = readw ( priv - > rtc_base + REG_RTC_MATCH_VAL_L )
2021-09-28 16:46:54 +03:00
| ( ( unsigned long ) readw ( priv - > rtc_base + REG_RTC_MATCH_VAL_H ) < < 16 ) ;
2021-08-23 20:16:12 +03:00
rtc_time64_to_tm ( seconds , & alarm - > time ) ;
if ( ! ( readw ( priv - > rtc_base + REG_RTC_CTRL ) & INT_MASK_BIT ) )
alarm - > enabled = 1 ;
return 0 ;
}
static int msc313_rtc_alarm_irq_enable ( struct device * dev , unsigned int enabled )
{
struct msc313_rtc * priv = dev_get_drvdata ( dev ) ;
u16 reg ;
reg = readw ( priv - > rtc_base + REG_RTC_CTRL ) ;
if ( enabled )
reg & = ~ INT_MASK_BIT ;
else
reg | = INT_MASK_BIT ;
writew ( reg , priv - > rtc_base + REG_RTC_CTRL ) ;
return 0 ;
}
static int msc313_rtc_set_alarm ( struct device * dev , struct rtc_wkalrm * alarm )
{
struct msc313_rtc * priv = dev_get_drvdata ( dev ) ;
unsigned long seconds ;
seconds = rtc_tm_to_time64 ( & alarm - > time ) ;
writew ( ( seconds & 0xFFFF ) , priv - > rtc_base + REG_RTC_MATCH_VAL_L ) ;
writew ( ( seconds > > 16 ) & 0xFFFF , priv - > rtc_base + REG_RTC_MATCH_VAL_H ) ;
msc313_rtc_alarm_irq_enable ( dev , alarm - > enabled ) ;
return 0 ;
}
static bool msc313_rtc_get_enabled ( struct msc313_rtc * priv )
{
return readw ( priv - > rtc_base + REG_RTC_CTRL ) & CNT_EN_BIT ;
}
static void msc313_rtc_set_enabled ( struct msc313_rtc * priv )
{
u16 reg ;
reg = readw ( priv - > rtc_base + REG_RTC_CTRL ) ;
reg | = CNT_EN_BIT ;
writew ( reg , priv - > rtc_base + REG_RTC_CTRL ) ;
}
static int msc313_rtc_read_time ( struct device * dev , struct rtc_time * tm )
{
struct msc313_rtc * priv = dev_get_drvdata ( dev ) ;
u32 seconds ;
u16 reg ;
if ( ! msc313_rtc_get_enabled ( priv ) )
return - EINVAL ;
reg = readw ( priv - > rtc_base + REG_RTC_CTRL ) ;
writew ( reg | READ_EN_BIT , priv - > rtc_base + REG_RTC_CTRL ) ;
/* Wait for HW latch done */
while ( readw ( priv - > rtc_base + REG_RTC_CTRL ) & READ_EN_BIT )
udelay ( 1 ) ;
seconds = readw ( priv - > rtc_base + REG_RTC_CNT_VAL_L )
2021-09-28 16:46:54 +03:00
| ( ( unsigned long ) readw ( priv - > rtc_base + REG_RTC_CNT_VAL_H ) < < 16 ) ;
2021-08-23 20:16:12 +03:00
rtc_time64_to_tm ( seconds , tm ) ;
return 0 ;
}
static int msc313_rtc_set_time ( struct device * dev , struct rtc_time * tm )
{
struct msc313_rtc * priv = dev_get_drvdata ( dev ) ;
unsigned long seconds ;
u16 reg ;
seconds = rtc_tm_to_time64 ( tm ) ;
writew ( seconds & 0xFFFF , priv - > rtc_base + REG_RTC_LOAD_VAL_L ) ;
writew ( ( seconds > > 16 ) & 0xFFFF , priv - > rtc_base + REG_RTC_LOAD_VAL_H ) ;
/* Enable load for loading value into internal RTC counter */
reg = readw ( priv - > rtc_base + REG_RTC_CTRL ) ;
writew ( reg | LOAD_EN_BIT , priv - > rtc_base + REG_RTC_CTRL ) ;
/* Wait for HW latch done */
while ( readw ( priv - > rtc_base + REG_RTC_CTRL ) & LOAD_EN_BIT )
udelay ( 1 ) ;
msc313_rtc_set_enabled ( priv ) ;
return 0 ;
}
static const struct rtc_class_ops msc313_rtc_ops = {
. read_time = msc313_rtc_read_time ,
. set_time = msc313_rtc_set_time ,
. read_alarm = msc313_rtc_read_alarm ,
. set_alarm = msc313_rtc_set_alarm ,
. alarm_irq_enable = msc313_rtc_alarm_irq_enable ,
} ;
static irqreturn_t msc313_rtc_interrupt ( s32 irq , void * dev_id )
{
struct msc313_rtc * priv = dev_get_drvdata ( dev_id ) ;
u16 reg ;
reg = readw ( priv - > rtc_base + REG_RTC_STATUS_INT ) ;
if ( ! ( reg & ALM_INT_BIT ) )
return IRQ_NONE ;
reg = readw ( priv - > rtc_base + REG_RTC_CTRL ) ;
reg | = INT_CLEAR_BIT ;
reg & = ~ INT_FORCE_BIT ;
writew ( reg , priv - > rtc_base + REG_RTC_CTRL ) ;
rtc_update_irq ( priv - > rtc_dev , 1 , RTC_IRQF | RTC_AF ) ;
return IRQ_HANDLED ;
}
static int msc313_rtc_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct msc313_rtc * priv ;
unsigned long rate ;
struct clk * clk ;
int ret ;
int irq ;
priv = devm_kzalloc ( & pdev - > dev , sizeof ( struct msc313_rtc ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
priv - > rtc_base = devm_platform_ioremap_resource ( pdev , 0 ) ;
if ( IS_ERR ( priv - > rtc_base ) )
return PTR_ERR ( priv - > rtc_base ) ;
irq = platform_get_irq ( pdev , 0 ) ;
if ( irq < 0 )
return - EINVAL ;
priv - > rtc_dev = devm_rtc_allocate_device ( dev ) ;
if ( IS_ERR ( priv - > rtc_dev ) )
return PTR_ERR ( priv - > rtc_dev ) ;
priv - > rtc_dev - > ops = & msc313_rtc_ops ;
priv - > rtc_dev - > range_max = U32_MAX ;
ret = devm_request_irq ( dev , irq , msc313_rtc_interrupt , IRQF_SHARED ,
dev_name ( & pdev - > dev ) , & pdev - > dev ) ;
if ( ret ) {
dev_err ( dev , " Could not request IRQ \n " ) ;
return ret ;
}
2022-12-02 21:45:30 +03:00
clk = devm_clk_get_enabled ( dev , NULL ) ;
2021-08-23 20:16:12 +03:00
if ( IS_ERR ( clk ) ) {
dev_err ( dev , " No input reference clock \n " ) ;
return PTR_ERR ( clk ) ;
}
rate = clk_get_rate ( clk ) ;
writew ( rate & 0xFFFF , priv - > rtc_base + REG_RTC_FREQ_CW_L ) ;
writew ( ( rate > > 16 ) & 0xFFFF , priv - > rtc_base + REG_RTC_FREQ_CW_H ) ;
platform_set_drvdata ( pdev , priv ) ;
return devm_rtc_register_device ( priv - > rtc_dev ) ;
}
static const struct of_device_id msc313_rtc_of_match_table [ ] = {
{ . compatible = " mstar,msc313-rtc " } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , msc313_rtc_of_match_table ) ;
static struct platform_driver msc313_rtc_driver = {
. probe = msc313_rtc_probe ,
. driver = {
. name = " msc313-rtc " ,
. of_match_table = msc313_rtc_of_match_table ,
} ,
} ;
module_platform_driver ( msc313_rtc_driver ) ;
MODULE_AUTHOR ( " Daniel Palmer <daniel@thingy.jp> " ) ;
MODULE_AUTHOR ( " Romain Perier <romain.perier@gmail.com> " ) ;
MODULE_DESCRIPTION ( " MStar RTC Driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;