2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2012-12-18 04:02:44 +04:00
/*
* Copyright ( C ) 2012 Avionic Design GmbH
*/
# include <linux/bcd.h>
2021-10-18 18:36:50 +03:00
# include <linux/bitfield.h>
2012-12-18 04:02:44 +04:00
# include <linux/i2c.h>
# include <linux/module.h>
2021-10-18 18:36:46 +03:00
# include <linux/regmap.h>
2012-12-18 04:02:44 +04:00
# include <linux/rtc.h>
# include <linux/of.h>
2021-04-18 03:20:22 +03:00
# include <linux/pm_wakeirq.h>
2012-12-18 04:02:44 +04:00
2021-07-11 00:14:31 +03:00
# define PCF8523_REG_CONTROL1 0x00
# define PCF8523_CONTROL1_CAP_SEL BIT(7)
# define PCF8523_CONTROL1_STOP BIT(5)
# define PCF8523_CONTROL1_AIE BIT(1)
# define PCF8523_REG_CONTROL2 0x01
# define PCF8523_CONTROL2_AF BIT(3)
# define PCF8523_REG_CONTROL3 0x02
2023-03-15 11:20:20 +03:00
# define PCF8523_CONTROL3_PM GENMASK(7, 5)
2021-10-18 18:36:50 +03:00
# define PCF8523_PM_STANDBY 0x7
2021-07-11 00:14:31 +03:00
# define PCF8523_CONTROL3_BLF BIT(2) /* battery low bit, read-only */
2021-10-15 22:24:00 +03:00
# define PCF8523_CONTROL3_BSF BIT(3)
2021-07-11 00:14:31 +03:00
# define PCF8523_REG_SECONDS 0x03
# define PCF8523_SECONDS_OS BIT(7)
# define PCF8523_REG_MINUTES 0x04
# define PCF8523_REG_HOURS 0x05
# define PCF8523_REG_DAYS 0x06
# define PCF8523_REG_WEEKDAYS 0x07
# define PCF8523_REG_MONTHS 0x08
# define PCF8523_REG_YEARS 0x09
# define PCF8523_REG_MINUTE_ALARM 0x0a
# define PCF8523_REG_HOUR_ALARM 0x0b
# define PCF8523_REG_DAY_ALARM 0x0c
# define PCF8523_REG_WEEKDAY_ALARM 0x0d
2021-04-18 03:20:22 +03:00
# define ALARM_DIS BIT(7)
2021-07-11 00:14:31 +03:00
# define PCF8523_REG_OFFSET 0x0e
# define PCF8523_OFFSET_MODE BIT(7)
2017-09-29 13:23:36 +03:00
2021-07-11 00:14:31 +03:00
# define PCF8523_TMR_CLKOUT_CTRL 0x0f
2021-04-18 03:20:22 +03:00
struct pcf8523 {
struct rtc_device * rtc ;
2021-10-18 18:36:46 +03:00
struct regmap * regmap ;
2021-04-18 03:20:22 +03:00
} ;
2021-10-18 18:36:46 +03:00
static int pcf8523_load_capacitance ( struct pcf8523 * pcf8523 , struct device_node * node )
2012-12-18 04:02:44 +04:00
{
2021-10-18 18:36:46 +03:00
u32 load , value = 0 ;
2012-12-18 04:02:44 +04:00
2019-01-19 12:00:30 +03:00
load = 12500 ;
2021-10-18 18:36:46 +03:00
of_property_read_u32 ( node , " quartz-load-femtofarads " , & load ) ;
2019-01-19 12:00:30 +03:00
switch ( load ) {
default :
2021-10-18 18:36:46 +03:00
dev_warn ( & pcf8523 - > rtc - > dev , " Unknown quartz-load-femtofarads value: %d. Assuming 12500 " ,
2019-01-19 12:00:30 +03:00
load ) ;
2020-08-24 01:36:59 +03:00
fallthrough ;
2019-01-19 12:00:30 +03:00
case 12500 :
2023-03-15 11:20:21 +03:00
value = PCF8523_CONTROL1_CAP_SEL ;
2019-01-19 12:00:30 +03:00
break ;
case 7000 :
break ;
}
2012-12-18 04:02:44 +04:00
2021-10-18 18:36:46 +03:00
return regmap_update_bits ( pcf8523 - > regmap , PCF8523_REG_CONTROL1 ,
PCF8523_CONTROL1_CAP_SEL , value ) ;
2012-12-18 04:02:44 +04:00
}
2021-04-18 03:20:22 +03:00
static irqreturn_t pcf8523_irq ( int irq , void * dev_id )
{
2021-10-18 18:36:46 +03:00
struct pcf8523 * pcf8523 = dev_id ;
u32 value ;
2021-04-18 03:20:22 +03:00
int err ;
2021-10-18 18:36:46 +03:00
err = regmap_read ( pcf8523 - > regmap , PCF8523_REG_CONTROL2 , & value ) ;
2021-04-18 03:20:22 +03:00
if ( err < 0 )
return IRQ_HANDLED ;
2021-07-11 00:14:31 +03:00
if ( value & PCF8523_CONTROL2_AF ) {
value & = ~ PCF8523_CONTROL2_AF ;
2021-10-18 18:36:46 +03:00
regmap_write ( pcf8523 - > regmap , PCF8523_REG_CONTROL2 , value ) ;
2021-04-18 03:20:22 +03:00
rtc_update_irq ( pcf8523 - > rtc , 1 , RTC_IRQF | RTC_AF ) ;
return IRQ_HANDLED ;
}
return IRQ_NONE ;
}
2012-12-18 04:02:44 +04:00
static int pcf8523_rtc_read_time ( struct device * dev , struct rtc_time * tm )
{
2021-10-18 18:36:46 +03:00
struct pcf8523 * pcf8523 = dev_get_drvdata ( dev ) ;
2022-08-29 15:46:39 +03:00
u8 regs [ 10 ] ;
2012-12-18 04:02:44 +04:00
int err ;
2022-08-29 15:46:39 +03:00
err = regmap_bulk_read ( pcf8523 - > regmap , PCF8523_REG_CONTROL1 , regs ,
2021-10-18 18:36:46 +03:00
sizeof ( regs ) ) ;
2012-12-18 04:02:44 +04:00
if ( err < 0 )
return err ;
2022-08-29 15:46:39 +03:00
if ( ( regs [ 0 ] & PCF8523_CONTROL1_STOP ) | | ( regs [ 3 ] & PCF8523_SECONDS_OS ) )
2016-03-03 11:55:47 +03:00
return - EINVAL ;
2012-12-18 04:02:44 +04:00
2022-08-29 15:46:39 +03:00
tm - > tm_sec = bcd2bin ( regs [ 3 ] & 0x7f ) ;
tm - > tm_min = bcd2bin ( regs [ 4 ] & 0x7f ) ;
tm - > tm_hour = bcd2bin ( regs [ 5 ] & 0x3f ) ;
tm - > tm_mday = bcd2bin ( regs [ 6 ] & 0x3f ) ;
tm - > tm_wday = regs [ 7 ] & 0x7 ;
tm - > tm_mon = bcd2bin ( regs [ 8 ] & 0x1f ) - 1 ;
tm - > tm_year = bcd2bin ( regs [ 9 ] ) + 100 ;
2012-12-18 04:02:44 +04:00
2018-02-19 18:23:56 +03:00
return 0 ;
2012-12-18 04:02:44 +04:00
}
static int pcf8523_rtc_set_time ( struct device * dev , struct rtc_time * tm )
{
2021-10-18 18:36:46 +03:00
struct pcf8523 * pcf8523 = dev_get_drvdata ( dev ) ;
u8 regs [ 7 ] ;
2012-12-18 04:02:44 +04:00
int err ;
2021-10-18 18:36:46 +03:00
err = regmap_update_bits ( pcf8523 - > regmap , PCF8523_REG_CONTROL1 ,
PCF8523_CONTROL1_STOP , PCF8523_CONTROL1_STOP ) ;
2012-12-18 04:02:44 +04:00
if ( err < 0 )
return err ;
2021-07-11 00:14:31 +03:00
/* This will purposely overwrite PCF8523_SECONDS_OS */
2021-10-18 18:36:46 +03:00
regs [ 0 ] = bin2bcd ( tm - > tm_sec ) ;
regs [ 1 ] = bin2bcd ( tm - > tm_min ) ;
regs [ 2 ] = bin2bcd ( tm - > tm_hour ) ;
regs [ 3 ] = bin2bcd ( tm - > tm_mday ) ;
regs [ 4 ] = tm - > tm_wday ;
regs [ 5 ] = bin2bcd ( tm - > tm_mon + 1 ) ;
regs [ 6 ] = bin2bcd ( tm - > tm_year - 100 ) ;
err = regmap_bulk_write ( pcf8523 - > regmap , PCF8523_REG_SECONDS , regs ,
sizeof ( regs ) ) ;
2012-12-18 04:02:44 +04:00
if ( err < 0 ) {
/*
* If the time cannot be set , restart the RTC anyway . Note
* that errors are ignored if the RTC cannot be started so
* that we have a chance to propagate the original error .
*/
2021-10-18 18:36:46 +03:00
regmap_update_bits ( pcf8523 - > regmap , PCF8523_REG_CONTROL1 ,
PCF8523_CONTROL1_STOP , 0 ) ;
2012-12-18 04:02:44 +04:00
return err ;
}
2021-10-18 18:36:46 +03:00
return regmap_update_bits ( pcf8523 - > regmap , PCF8523_REG_CONTROL1 ,
PCF8523_CONTROL1_STOP , 0 ) ;
2012-12-18 04:02:44 +04:00
}
2021-04-18 03:20:22 +03:00
static int pcf8523_rtc_read_alarm ( struct device * dev , struct rtc_wkalrm * tm )
{
2021-10-18 18:36:46 +03:00
struct pcf8523 * pcf8523 = dev_get_drvdata ( dev ) ;
u8 regs [ 4 ] ;
u32 value ;
2021-04-18 03:20:22 +03:00
int err ;
2021-10-18 18:36:46 +03:00
err = regmap_bulk_read ( pcf8523 - > regmap , PCF8523_REG_MINUTE_ALARM , regs ,
sizeof ( regs ) ) ;
2021-04-18 03:20:22 +03:00
if ( err < 0 )
return err ;
tm - > time . tm_sec = 0 ;
tm - > time . tm_min = bcd2bin ( regs [ 0 ] & 0x7F ) ;
tm - > time . tm_hour = bcd2bin ( regs [ 1 ] & 0x3F ) ;
tm - > time . tm_mday = bcd2bin ( regs [ 2 ] & 0x3F ) ;
tm - > time . tm_wday = bcd2bin ( regs [ 3 ] & 0x7 ) ;
2021-10-18 18:36:46 +03:00
err = regmap_read ( pcf8523 - > regmap , PCF8523_REG_CONTROL1 , & value ) ;
2021-04-18 03:20:22 +03:00
if ( err < 0 )
return err ;
2021-07-11 00:14:31 +03:00
tm - > enabled = ! ! ( value & PCF8523_CONTROL1_AIE ) ;
2021-04-18 03:20:22 +03:00
2021-10-18 18:36:46 +03:00
err = regmap_read ( pcf8523 - > regmap , PCF8523_REG_CONTROL2 , & value ) ;
2021-04-18 03:20:22 +03:00
if ( err < 0 )
return err ;
2021-07-11 00:14:31 +03:00
tm - > pending = ! ! ( value & PCF8523_CONTROL2_AF ) ;
2021-04-18 03:20:22 +03:00
return 0 ;
}
static int pcf8523_irq_enable ( struct device * dev , unsigned int enabled )
{
2021-10-18 18:36:46 +03:00
struct pcf8523 * pcf8523 = dev_get_drvdata ( dev ) ;
2021-04-18 03:20:22 +03:00
2021-10-18 18:36:46 +03:00
return regmap_update_bits ( pcf8523 - > regmap , PCF8523_REG_CONTROL1 ,
PCF8523_CONTROL1_AIE , enabled ?
PCF8523_CONTROL1_AIE : 0 ) ;
2021-04-18 03:20:22 +03:00
}
static int pcf8523_rtc_set_alarm ( struct device * dev , struct rtc_wkalrm * tm )
{
2021-10-18 18:36:46 +03:00
struct pcf8523 * pcf8523 = dev_get_drvdata ( dev ) ;
2021-04-18 03:20:22 +03:00
u8 regs [ 5 ] ;
int err ;
err = pcf8523_irq_enable ( dev , 0 ) ;
if ( err )
return err ;
2021-10-18 18:36:46 +03:00
err = regmap_write ( pcf8523 - > regmap , PCF8523_REG_CONTROL2 , 0 ) ;
2021-04-18 03:20:22 +03:00
if ( err < 0 )
return err ;
2021-10-18 18:36:46 +03:00
regs [ 0 ] = bin2bcd ( tm - > time . tm_min ) ;
regs [ 1 ] = bin2bcd ( tm - > time . tm_hour ) ;
regs [ 2 ] = bin2bcd ( tm - > time . tm_mday ) ;
regs [ 3 ] = ALARM_DIS ;
err = regmap_bulk_write ( pcf8523 - > regmap , PCF8523_REG_MINUTE_ALARM , regs ,
sizeof ( regs ) ) ;
2021-04-18 03:20:22 +03:00
if ( err < 0 )
return err ;
if ( tm - > enabled )
return pcf8523_irq_enable ( dev , tm - > enabled ) ;
return 0 ;
}
2021-10-18 18:36:50 +03:00
static int pcf8523_param_get ( struct device * dev , struct rtc_param * param )
{
struct pcf8523 * pcf8523 = dev_get_drvdata ( dev ) ;
int ret ;
2022-02-10 03:30:27 +03:00
u32 value ;
2021-10-18 18:36:50 +03:00
2023-03-15 11:20:20 +03:00
switch ( param - > param ) {
2021-10-18 18:36:50 +03:00
case RTC_PARAM_BACKUP_SWITCH_MODE :
ret = regmap_read ( pcf8523 - > regmap , PCF8523_REG_CONTROL3 , & value ) ;
if ( ret < 0 )
return ret ;
value = FIELD_GET ( PCF8523_CONTROL3_PM , value ) ;
2023-03-15 11:20:20 +03:00
switch ( value ) {
2021-10-18 18:36:50 +03:00
case 0x0 :
case 0x4 :
param - > uvalue = RTC_BSM_LEVEL ;
break ;
case 0x1 :
case 0x5 :
param - > uvalue = RTC_BSM_DIRECT ;
break ;
case PCF8523_PM_STANDBY :
param - > uvalue = RTC_BSM_STANDBY ;
break ;
default :
param - > uvalue = RTC_BSM_DISABLED ;
}
break ;
default :
return - EINVAL ;
}
return 0 ;
}
static int pcf8523_param_set ( struct device * dev , struct rtc_param * param )
{
struct pcf8523 * pcf8523 = dev_get_drvdata ( dev ) ;
2022-02-10 03:30:27 +03:00
u8 mode ;
2021-10-18 18:36:50 +03:00
2023-03-15 11:20:20 +03:00
switch ( param - > param ) {
2021-10-18 18:36:50 +03:00
case RTC_PARAM_BACKUP_SWITCH_MODE :
switch ( param - > uvalue ) {
case RTC_BSM_DISABLED :
mode = 0x2 ;
break ;
case RTC_BSM_DIRECT :
mode = 0x1 ;
break ;
case RTC_BSM_LEVEL :
mode = 0x0 ;
break ;
case RTC_BSM_STANDBY :
mode = PCF8523_PM_STANDBY ;
break ;
default :
return - EINVAL ;
}
return regmap_update_bits ( pcf8523 - > regmap , PCF8523_REG_CONTROL3 ,
PCF8523_CONTROL3_PM ,
FIELD_PREP ( PCF8523_CONTROL3_PM , mode ) ) ;
break ;
default :
return - EINVAL ;
}
return 0 ;
}
2013-02-22 04:44:27 +04:00
static int pcf8523_rtc_ioctl ( struct device * dev , unsigned int cmd ,
unsigned long arg )
{
2021-10-18 18:36:46 +03:00
struct pcf8523 * pcf8523 = dev_get_drvdata ( dev ) ;
2021-04-18 03:20:23 +03:00
unsigned int flags = 0 ;
2021-10-18 18:36:46 +03:00
u32 value ;
2018-12-05 18:00:09 +03:00
int ret ;
2013-02-22 04:44:27 +04:00
switch ( cmd ) {
case RTC_VL_READ :
2021-10-18 18:36:46 +03:00
ret = regmap_read ( pcf8523 - > regmap , PCF8523_REG_CONTROL3 , & value ) ;
2018-12-05 18:00:09 +03:00
if ( ret < 0 )
return ret ;
2021-10-15 22:24:00 +03:00
if ( value & PCF8523_CONTROL3_BLF )
2021-04-18 03:20:23 +03:00
flags | = RTC_VL_BACKUP_LOW ;
2021-10-18 18:36:46 +03:00
ret = regmap_read ( pcf8523 - > regmap , PCF8523_REG_SECONDS , & value ) ;
2021-04-18 03:20:23 +03:00
if ( ret < 0 )
return ret ;
2021-07-11 00:14:31 +03:00
if ( value & PCF8523_SECONDS_OS )
2021-04-18 03:20:23 +03:00
flags | = RTC_VL_DATA_INVALID ;
2013-02-22 04:44:27 +04:00
2021-04-18 03:20:23 +03:00
return put_user ( flags , ( unsigned int __user * ) arg ) ;
2013-02-22 04:44:27 +04:00
default :
return - ENOIOCTLCMD ;
}
}
2017-09-29 13:23:36 +03:00
static int pcf8523_rtc_read_offset ( struct device * dev , long * offset )
{
2021-10-18 18:36:46 +03:00
struct pcf8523 * pcf8523 = dev_get_drvdata ( dev ) ;
2017-09-29 13:23:36 +03:00
int err ;
2021-10-18 18:36:46 +03:00
u32 value ;
2017-09-29 13:23:36 +03:00
s8 val ;
2021-10-18 18:36:46 +03:00
err = regmap_read ( pcf8523 - > regmap , PCF8523_REG_OFFSET , & value ) ;
2017-09-29 13:23:36 +03:00
if ( err < 0 )
return err ;
/* sign extend the 7-bit offset value */
val = value < < 1 ;
2021-07-11 00:14:31 +03:00
* offset = ( value & PCF8523_OFFSET_MODE ? 4069 : 4340 ) * ( val > > 1 ) ;
2017-09-29 13:23:36 +03:00
return 0 ;
}
static int pcf8523_rtc_set_offset ( struct device * dev , long offset )
{
2021-10-18 18:36:46 +03:00
struct pcf8523 * pcf8523 = dev_get_drvdata ( dev ) ;
2017-09-29 13:23:36 +03:00
long reg_m0 , reg_m1 ;
2021-10-18 18:36:46 +03:00
u32 value ;
2017-09-29 13:23:36 +03:00
reg_m0 = clamp ( DIV_ROUND_CLOSEST ( offset , 4340 ) , - 64L , 63L ) ;
reg_m1 = clamp ( DIV_ROUND_CLOSEST ( offset , 4069 ) , - 64L , 63L ) ;
if ( abs ( reg_m0 * 4340 - offset ) < abs ( reg_m1 * 4069 - offset ) )
value = reg_m0 & 0x7f ;
else
2021-07-11 00:14:31 +03:00
value = ( reg_m1 & 0x7f ) | PCF8523_OFFSET_MODE ;
2017-09-29 13:23:36 +03:00
2021-10-18 18:36:46 +03:00
return regmap_write ( pcf8523 - > regmap , PCF8523_REG_OFFSET , value ) ;
2017-09-29 13:23:36 +03:00
}
2012-12-18 04:02:44 +04:00
static const struct rtc_class_ops pcf8523_rtc_ops = {
. read_time = pcf8523_rtc_read_time ,
. set_time = pcf8523_rtc_set_time ,
2021-04-18 03:20:22 +03:00
. read_alarm = pcf8523_rtc_read_alarm ,
. set_alarm = pcf8523_rtc_set_alarm ,
. alarm_irq_enable = pcf8523_irq_enable ,
2013-02-22 04:44:27 +04:00
. ioctl = pcf8523_rtc_ioctl ,
2017-09-29 13:23:36 +03:00
. read_offset = pcf8523_rtc_read_offset ,
. set_offset = pcf8523_rtc_set_offset ,
2021-10-18 18:36:50 +03:00
. param_get = pcf8523_param_get ,
. param_set = pcf8523_param_set ,
2012-12-18 04:02:44 +04:00
} ;
2021-10-18 18:36:46 +03:00
static const struct regmap_config regmap_config = {
2023-03-15 11:20:20 +03:00
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = 0x13 ,
2021-10-18 18:36:46 +03:00
} ;
2022-06-10 19:23:43 +03:00
static int pcf8523_probe ( struct i2c_client * client )
2012-12-18 04:02:44 +04:00
{
2021-04-18 03:20:22 +03:00
struct pcf8523 * pcf8523 ;
2019-11-23 12:08:38 +03:00
struct rtc_device * rtc ;
2021-04-18 03:20:22 +03:00
bool wakeup_source = false ;
2021-10-18 18:36:50 +03:00
u32 value ;
2012-12-18 04:02:44 +04:00
int err ;
if ( ! i2c_check_functionality ( client - > adapter , I2C_FUNC_I2C ) )
return - ENODEV ;
2021-04-18 03:20:22 +03:00
pcf8523 = devm_kzalloc ( & client - > dev , sizeof ( struct pcf8523 ) , GFP_KERNEL ) ;
if ( ! pcf8523 )
return - ENOMEM ;
2021-10-18 18:36:46 +03:00
pcf8523 - > regmap = devm_regmap_init_i2c ( client , & regmap_config ) ;
if ( IS_ERR ( pcf8523 - > regmap ) )
return PTR_ERR ( pcf8523 - > regmap ) ;
2021-04-18 03:20:22 +03:00
i2c_set_clientdata ( client , pcf8523 ) ;
2021-10-18 18:36:46 +03:00
rtc = devm_rtc_allocate_device ( & client - > dev ) ;
if ( IS_ERR ( rtc ) )
return PTR_ERR ( rtc ) ;
pcf8523 - > rtc = rtc ;
err = pcf8523_load_capacitance ( pcf8523 , client - > dev . of_node ) ;
2012-12-18 04:02:44 +04:00
if ( err < 0 )
2019-01-19 12:00:30 +03:00
dev_warn ( & client - > dev , " failed to set xtal load capacitance: %d " ,
err ) ;
2012-12-18 04:02:44 +04:00
2021-10-18 18:36:50 +03:00
err = regmap_read ( pcf8523 - > regmap , PCF8523_REG_SECONDS , & value ) ;
2012-12-18 04:02:44 +04:00
if ( err < 0 )
return err ;
2021-10-18 18:36:50 +03:00
if ( value & PCF8523_SECONDS_OS ) {
err = regmap_read ( pcf8523 - > regmap , PCF8523_REG_CONTROL3 , & value ) ;
if ( err < 0 )
return err ;
if ( FIELD_GET ( PCF8523_CONTROL3_PM , value ) = = PCF8523_PM_STANDBY ) {
err = regmap_write ( pcf8523 - > regmap , PCF8523_REG_CONTROL3 ,
value & ~ PCF8523_CONTROL3_PM ) ;
if ( err < 0 )
return err ;
}
}
2020-11-18 03:27:45 +03:00
rtc - > ops = & pcf8523_rtc_ops ;
2020-11-18 03:27:46 +03:00
rtc - > range_min = RTC_TIMESTAMP_BEGIN_2000 ;
rtc - > range_max = RTC_TIMESTAMP_END_2099 ;
2022-03-09 19:22:46 +03:00
set_bit ( RTC_FEATURE_ALARM_RES_MINUTE , rtc - > features ) ;
2022-03-09 19:22:45 +03:00
clear_bit ( RTC_FEATURE_UPDATE_INTERRUPT , rtc - > features ) ;
2021-04-18 03:20:22 +03:00
if ( client - > irq > 0 ) {
2023-01-23 23:02:12 +03:00
unsigned long irqflags = IRQF_TRIGGER_LOW ;
if ( dev_fwnode ( & client - > dev ) )
irqflags = 0 ;
2021-10-18 18:36:46 +03:00
err = regmap_write ( pcf8523 - > regmap , PCF8523_TMR_CLKOUT_CTRL , 0x38 ) ;
2021-04-18 03:20:22 +03:00
if ( err < 0 )
return err ;
err = devm_request_threaded_irq ( & client - > dev , client - > irq ,
NULL , pcf8523_irq ,
2023-01-23 23:02:12 +03:00
IRQF_SHARED | IRQF_ONESHOT | irqflags ,
2021-10-18 18:36:46 +03:00
dev_name ( & rtc - > dev ) , pcf8523 ) ;
2021-04-18 03:20:22 +03:00
if ( err )
return err ;
dev_pm_set_wake_irq ( & client - > dev , client - > irq ) ;
}
wakeup_source = of_property_read_bool ( client - > dev . of_node , " wakeup-source " ) ;
if ( client - > irq > 0 | | wakeup_source )
device_init_wakeup ( & client - > dev , true ) ;
2020-11-18 03:27:45 +03:00
return devm_rtc_register_device ( rtc ) ;
2012-12-18 04:02:44 +04:00
}
static const struct i2c_device_id pcf8523_id [ ] = {
{ " pcf8523 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , pcf8523_id ) ;
static const struct of_device_id pcf8523_of_match [ ] = {
{ . compatible = " nxp,pcf8523 " } ,
2018-12-19 00:52:12 +03:00
{ . compatible = " microcrystal,rv8523 " } ,
2012-12-18 04:02:44 +04:00
{ }
} ;
MODULE_DEVICE_TABLE ( of , pcf8523_of_match ) ;
static struct i2c_driver pcf8523_driver = {
. driver = {
2021-04-18 03:20:21 +03:00
. name = " rtc-pcf8523 " ,
2021-10-18 18:36:49 +03:00
. of_match_table = pcf8523_of_match ,
2012-12-18 04:02:44 +04:00
} ,
2023-05-05 15:11:36 +03:00
. probe = pcf8523_probe ,
2012-12-18 04:02:44 +04:00
. id_table = pcf8523_id ,
} ;
module_i2c_driver ( pcf8523_driver ) ;
MODULE_AUTHOR ( " Thierry Reding <thierry.reding@avionic-design.de> " ) ;
MODULE_DESCRIPTION ( " NXP PCF8523 RTC driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;