2019-04-01 19:33:49 +03:00
// SPDX-License-Identifier: GPL-2.0
2014-08-09 01:20:22 +04:00
/*
* An I2C driver for the PCF85063 RTC
* Copyright 2014 Rose Technology
*
* Author : Søren Andersen < san @ rosetechnology . dk >
* Maintainers : http : //www.nslu2-linux.org/
2019-04-01 19:08:13 +03:00
*
* Copyright ( C ) 2019 Micro Crystal AG
* Author : Alexandre Belloni < alexandre . belloni @ bootlin . com >
2014-08-09 01:20:22 +04:00
*/
2020-01-24 04:52:38 +03:00
# include <linux/clk-provider.h>
2014-08-09 01:20:22 +04:00
# include <linux/i2c.h>
# include <linux/bcd.h>
# include <linux/rtc.h>
# include <linux/module.h>
2019-04-01 19:08:10 +03:00
# include <linux/of_device.h>
2019-04-01 19:08:12 +03:00
# include <linux/pm_wakeirq.h>
2019-04-01 19:08:10 +03:00
# include <linux/regmap.h>
2014-08-09 01:20:22 +04:00
2016-07-13 00:15:46 +03:00
/*
* Information for this driver was pulled from the following datasheets .
*
2021-06-03 17:34:46 +03:00
* https : //www.nxp.com/docs/en/data-sheet/PCF85063A.pdf
* https : //www.nxp.com/docs/en/data-sheet/PCF85063TP.pdf
2016-07-13 00:15:46 +03:00
*
2021-06-24 15:09:53 +03:00
* PCF85063A - - Rev . 7 — 30 March 2018
2016-07-13 00:15:46 +03:00
* PCF85063TP - - Rev . 4 — 6 May 2015
2019-04-01 19:08:13 +03:00
*
* https : //www.microcrystal.com/fileadmin/Media/Products/RTC/App.Manual/RV-8263-C7_App-Manual.pdf
* RV8263 - - Rev . 1.0 — January 2019
*/
2016-07-13 00:15:46 +03:00
2014-08-09 01:20:22 +04:00
# define PCF85063_REG_CTRL1 0x00 /* status */
2019-01-19 12:00:31 +03:00
# define PCF85063_REG_CTRL1_CAP_SEL BIT(0)
2016-02-09 13:57:27 +03:00
# define PCF85063_REG_CTRL1_STOP BIT(5)
2021-10-15 14:12:08 +03:00
# define PCF85063_REG_CTRL1_EXT_TEST BIT(7)
2014-08-09 01:20:22 +04:00
2019-04-01 19:08:12 +03:00
# define PCF85063_REG_CTRL2 0x01
# define PCF85063_CTRL2_AF BIT(6)
# define PCF85063_CTRL2_AIE BIT(7)
2019-04-01 19:08:15 +03:00
# define PCF85063_REG_OFFSET 0x02
# define PCF85063_OFFSET_SIGN_BIT 6 /* 2's complement sign bit */
# define PCF85063_OFFSET_MODE BIT(7)
# define PCF85063_OFFSET_STEP0 4340
# define PCF85063_OFFSET_STEP1 4069
2020-01-24 04:52:38 +03:00
# define PCF85063_REG_CLKO_F_MASK 0x07 /* frequency mask */
# define PCF85063_REG_CLKO_F_32768HZ 0x00
# define PCF85063_REG_CLKO_F_OFF 0x07
2019-04-01 19:08:14 +03:00
# define PCF85063_REG_RAM 0x03
2014-08-09 01:20:22 +04:00
# define PCF85063_REG_SC 0x04 /* datetime */
2016-02-09 13:57:26 +03:00
# define PCF85063_REG_SC_OS 0x80
2014-08-09 01:20:22 +04:00
2019-04-01 19:08:12 +03:00
# define PCF85063_REG_ALM_S 0x0b
# define PCF85063_AEN BIT(7)
2019-04-01 19:08:11 +03:00
struct pcf85063_config {
struct regmap_config regmap ;
2019-04-01 19:08:12 +03:00
unsigned has_alarms : 1 ;
2019-04-01 19:08:13 +03:00
unsigned force_cap_7000 : 1 ;
2019-04-01 19:08:11 +03:00
} ;
2019-04-01 19:08:10 +03:00
struct pcf85063 {
struct rtc_device * rtc ;
struct regmap * regmap ;
2020-01-24 04:52:38 +03:00
# ifdef CONFIG_COMMON_CLK
struct clk_hw clkout_hw ;
# endif
2019-04-01 19:08:10 +03:00
} ;
2016-07-13 00:15:46 +03:00
2018-02-21 18:09:27 +03:00
static int pcf85063_rtc_read_time ( struct device * dev , struct rtc_time * tm )
2014-08-09 01:20:22 +04:00
{
2019-04-01 19:08:10 +03:00
struct pcf85063 * pcf85063 = dev_get_drvdata ( dev ) ;
2016-02-09 13:57:25 +03:00
int rc ;
u8 regs [ 7 ] ;
/*
* while reading , the time / date registers are blocked and not updated
* anymore until the access is finished . To not lose a second
* event , the access must be finished within one second . So , read all
* time / date registers in one turn .
*/
2019-04-01 19:08:10 +03:00
rc = regmap_bulk_read ( pcf85063 - > regmap , PCF85063_REG_SC , regs ,
sizeof ( regs ) ) ;
if ( rc )
return rc ;
2014-08-09 01:20:22 +04:00
2016-02-09 13:57:26 +03:00
/* if the clock has lost its power it makes no sense to use its time */
if ( regs [ 0 ] & PCF85063_REG_SC_OS ) {
2019-04-01 19:08:10 +03:00
dev_warn ( & pcf85063 - > rtc - > dev , " Power loss detected, invalid time \n " ) ;
2016-02-09 13:57:26 +03:00
return - EINVAL ;
}
2016-02-09 13:57:25 +03:00
tm - > tm_sec = bcd2bin ( regs [ 0 ] & 0x7F ) ;
tm - > tm_min = bcd2bin ( regs [ 1 ] & 0x7F ) ;
tm - > tm_hour = bcd2bin ( regs [ 2 ] & 0x3F ) ; /* rtc hr 0-23 */
tm - > tm_mday = bcd2bin ( regs [ 3 ] & 0x3F ) ;
tm - > tm_wday = regs [ 4 ] & 0x07 ;
tm - > tm_mon = bcd2bin ( regs [ 5 ] & 0x1F ) - 1 ; /* rtc mn 1-12 */
tm - > tm_year = bcd2bin ( regs [ 6 ] ) ;
2016-07-18 12:08:59 +03:00
tm - > tm_year + = 100 ;
2014-08-09 01:20:22 +04:00
2018-02-21 18:07:34 +03:00
return 0 ;
2014-08-09 01:20:22 +04:00
}
2018-02-21 18:09:27 +03:00
static int pcf85063_rtc_set_time ( struct device * dev , struct rtc_time * tm )
2014-08-09 01:20:22 +04:00
{
2019-04-01 19:08:10 +03:00
struct pcf85063 * pcf85063 = dev_get_drvdata ( dev ) ;
2016-02-09 13:57:27 +03:00
int rc ;
2016-07-13 00:15:46 +03:00
u8 regs [ 7 ] ;
2014-08-09 01:20:22 +04:00
2016-02-09 13:57:27 +03:00
/*
* to accurately set the time , reset the divider chain and keep it in
* reset state until all time / date registers are written
*/
2019-04-01 19:08:10 +03:00
rc = regmap_update_bits ( pcf85063 - > regmap , PCF85063_REG_CTRL1 ,
2021-10-15 14:12:08 +03:00
PCF85063_REG_CTRL1_EXT_TEST |
2019-04-01 19:08:10 +03:00
PCF85063_REG_CTRL1_STOP ,
PCF85063_REG_CTRL1_STOP ) ;
if ( rc )
2016-02-09 13:57:27 +03:00
return rc ;
2014-08-09 01:20:22 +04:00
/* hours, minutes and seconds */
2016-02-09 13:57:27 +03:00
regs [ 0 ] = bin2bcd ( tm - > tm_sec ) & 0x7F ; /* clear OS flag */
2014-08-09 01:20:22 +04:00
2016-02-09 13:57:27 +03:00
regs [ 1 ] = bin2bcd ( tm - > tm_min ) ;
regs [ 2 ] = bin2bcd ( tm - > tm_hour ) ;
2014-08-09 01:20:22 +04:00
/* Day of month, 1 - 31 */
2016-02-09 13:57:27 +03:00
regs [ 3 ] = bin2bcd ( tm - > tm_mday ) ;
2014-08-09 01:20:22 +04:00
/* Day, 0 - 6 */
2016-02-09 13:57:27 +03:00
regs [ 4 ] = tm - > tm_wday & 0x07 ;
2014-08-09 01:20:22 +04:00
/* month, 1 - 12 */
2016-02-09 13:57:27 +03:00
regs [ 5 ] = bin2bcd ( tm - > tm_mon + 1 ) ;
2014-08-09 01:20:22 +04:00
/* year and century */
2016-07-18 12:08:59 +03:00
regs [ 6 ] = bin2bcd ( tm - > tm_year - 100 ) ;
2016-02-09 13:57:27 +03:00
/* write all registers at once */
2019-04-01 19:08:10 +03:00
rc = regmap_bulk_write ( pcf85063 - > regmap , PCF85063_REG_SC ,
regs , sizeof ( regs ) ) ;
if ( rc )
2016-02-09 13:57:27 +03:00
return rc ;
2014-08-09 01:20:22 +04:00
2016-07-13 00:15:46 +03:00
/*
* Write the control register as a separate action since the size of
* the register space is different between the PCF85063TP and
* PCF85063A devices . The rollover point can not be used .
*/
2019-04-01 19:08:10 +03:00
return regmap_update_bits ( pcf85063 - > regmap , PCF85063_REG_CTRL1 ,
PCF85063_REG_CTRL1_STOP , 0 ) ;
2014-08-09 01:20:22 +04:00
}
2019-04-01 19:08:12 +03:00
static int pcf85063_rtc_read_alarm ( struct device * dev , struct rtc_wkalrm * alrm )
{
struct pcf85063 * pcf85063 = dev_get_drvdata ( dev ) ;
u8 buf [ 4 ] ;
unsigned int val ;
int ret ;
ret = regmap_bulk_read ( pcf85063 - > regmap , PCF85063_REG_ALM_S ,
buf , sizeof ( buf ) ) ;
if ( ret )
return ret ;
2022-09-21 10:41:41 +03:00
alrm - > time . tm_sec = bcd2bin ( buf [ 0 ] & 0x7f ) ;
alrm - > time . tm_min = bcd2bin ( buf [ 1 ] & 0x7f ) ;
alrm - > time . tm_hour = bcd2bin ( buf [ 2 ] & 0x3f ) ;
alrm - > time . tm_mday = bcd2bin ( buf [ 3 ] & 0x3f ) ;
2019-04-01 19:08:12 +03:00
ret = regmap_read ( pcf85063 - > regmap , PCF85063_REG_CTRL2 , & val ) ;
if ( ret )
return ret ;
alrm - > enabled = ! ! ( val & PCF85063_CTRL2_AIE ) ;
return 0 ;
}
static int pcf85063_rtc_set_alarm ( struct device * dev , struct rtc_wkalrm * alrm )
{
struct pcf85063 * pcf85063 = dev_get_drvdata ( dev ) ;
u8 buf [ 5 ] ;
int ret ;
buf [ 0 ] = bin2bcd ( alrm - > time . tm_sec ) ;
buf [ 1 ] = bin2bcd ( alrm - > time . tm_min ) ;
buf [ 2 ] = bin2bcd ( alrm - > time . tm_hour ) ;
buf [ 3 ] = bin2bcd ( alrm - > time . tm_mday ) ;
buf [ 4 ] = PCF85063_AEN ; /* Do not match on week day */
ret = regmap_update_bits ( pcf85063 - > regmap , PCF85063_REG_CTRL2 ,
PCF85063_CTRL2_AIE | PCF85063_CTRL2_AF , 0 ) ;
if ( ret )
return ret ;
ret = regmap_bulk_write ( pcf85063 - > regmap , PCF85063_REG_ALM_S ,
buf , sizeof ( buf ) ) ;
if ( ret )
return ret ;
return regmap_update_bits ( pcf85063 - > regmap , PCF85063_REG_CTRL2 ,
PCF85063_CTRL2_AIE | PCF85063_CTRL2_AF ,
alrm - > enabled ? PCF85063_CTRL2_AIE | PCF85063_CTRL2_AF : PCF85063_CTRL2_AF ) ;
}
static int pcf85063_rtc_alarm_irq_enable ( struct device * dev ,
unsigned int enabled )
{
struct pcf85063 * pcf85063 = dev_get_drvdata ( dev ) ;
return regmap_update_bits ( pcf85063 - > regmap , PCF85063_REG_CTRL2 ,
PCF85063_CTRL2_AIE ,
enabled ? PCF85063_CTRL2_AIE : 0 ) ;
}
static irqreturn_t pcf85063_rtc_handle_irq ( int irq , void * dev_id )
{
struct pcf85063 * pcf85063 = dev_id ;
unsigned int val ;
int err ;
err = regmap_read ( pcf85063 - > regmap , PCF85063_REG_CTRL2 , & val ) ;
if ( err )
return IRQ_NONE ;
if ( val & PCF85063_CTRL2_AF ) {
rtc_update_irq ( pcf85063 - > rtc , 1 , RTC_IRQF | RTC_AF ) ;
regmap_update_bits ( pcf85063 - > regmap , PCF85063_REG_CTRL2 ,
PCF85063_CTRL2_AIE | PCF85063_CTRL2_AF ,
0 ) ;
return IRQ_HANDLED ;
}
return IRQ_NONE ;
}
2019-04-01 19:08:15 +03:00
static int pcf85063_read_offset ( struct device * dev , long * offset )
{
struct pcf85063 * pcf85063 = dev_get_drvdata ( dev ) ;
long val ;
u32 reg ;
int ret ;
ret = regmap_read ( pcf85063 - > regmap , PCF85063_REG_OFFSET , & reg ) ;
if ( ret < 0 )
return ret ;
val = sign_extend32 ( reg & ~ PCF85063_OFFSET_MODE ,
PCF85063_OFFSET_SIGN_BIT ) ;
if ( reg & PCF85063_OFFSET_MODE )
* offset = val * PCF85063_OFFSET_STEP1 ;
else
* offset = val * PCF85063_OFFSET_STEP0 ;
return 0 ;
}
static int pcf85063_set_offset ( struct device * dev , long offset )
{
struct pcf85063 * pcf85063 = dev_get_drvdata ( dev ) ;
s8 mode0 , mode1 , reg ;
unsigned int error0 , error1 ;
if ( offset > PCF85063_OFFSET_STEP0 * 63 )
return - ERANGE ;
if ( offset < PCF85063_OFFSET_STEP0 * - 64 )
return - ERANGE ;
mode0 = DIV_ROUND_CLOSEST ( offset , PCF85063_OFFSET_STEP0 ) ;
mode1 = DIV_ROUND_CLOSEST ( offset , PCF85063_OFFSET_STEP1 ) ;
error0 = abs ( offset - ( mode0 * PCF85063_OFFSET_STEP0 ) ) ;
error1 = abs ( offset - ( mode1 * PCF85063_OFFSET_STEP1 ) ) ;
if ( mode1 > 63 | | mode1 < - 64 | | error0 < error1 )
reg = mode0 & ~ PCF85063_OFFSET_MODE ;
else
reg = mode1 | PCF85063_OFFSET_MODE ;
return regmap_write ( pcf85063 - > regmap , PCF85063_REG_OFFSET , reg ) ;
}
2019-04-01 19:08:16 +03:00
static int pcf85063_ioctl ( struct device * dev , unsigned int cmd ,
unsigned long arg )
{
struct pcf85063 * pcf85063 = dev_get_drvdata ( dev ) ;
int status , ret = 0 ;
switch ( cmd ) {
case RTC_VL_READ :
ret = regmap_read ( pcf85063 - > regmap , PCF85063_REG_SC , & status ) ;
if ( ret < 0 )
return ret ;
2021-11-08 01:53:48 +03:00
status = ( status & PCF85063_REG_SC_OS ) ? RTC_VL_DATA_INVALID : 0 ;
2019-04-01 19:08:16 +03:00
2019-12-15 01:02:53 +03:00
return put_user ( status , ( unsigned int __user * ) arg ) ;
2019-04-01 19:08:16 +03:00
default :
return - ENOIOCTLCMD ;
}
}
2014-08-09 01:20:22 +04:00
static const struct rtc_class_ops pcf85063_rtc_ops = {
2019-04-01 19:08:12 +03:00
. read_time = pcf85063_rtc_read_time ,
. set_time = pcf85063_rtc_set_time ,
2019-04-01 19:08:15 +03:00
. read_offset = pcf85063_read_offset ,
. set_offset = pcf85063_set_offset ,
2019-04-01 19:08:12 +03:00
. read_alarm = pcf85063_rtc_read_alarm ,
. set_alarm = pcf85063_rtc_set_alarm ,
. alarm_irq_enable = pcf85063_rtc_alarm_irq_enable ,
2019-04-01 19:08:16 +03:00
. ioctl = pcf85063_ioctl ,
2019-04-01 19:08:12 +03:00
} ;
2019-04-01 19:08:14 +03:00
static int pcf85063_nvmem_read ( void * priv , unsigned int offset ,
void * val , size_t bytes )
{
return regmap_read ( priv , PCF85063_REG_RAM , val ) ;
}
static int pcf85063_nvmem_write ( void * priv , unsigned int offset ,
void * val , size_t bytes )
{
return regmap_write ( priv , PCF85063_REG_RAM , * ( u8 * ) val ) ;
}
2019-04-01 19:08:10 +03:00
static int pcf85063_load_capacitance ( struct pcf85063 * pcf85063 ,
2019-04-01 19:08:13 +03:00
const struct device_node * np ,
unsigned int force_cap )
2019-01-19 12:00:31 +03:00
{
2019-04-01 19:08:10 +03:00
u32 load = 7000 ;
u8 reg = 0 ;
2019-01-19 12:00:31 +03:00
2019-04-01 19:08:13 +03:00
if ( force_cap )
load = force_cap ;
else
of_property_read_u32 ( np , " quartz-load-femtofarads " , & load ) ;
2019-01-19 12:00:31 +03:00
switch ( load ) {
default :
2019-04-01 19:08:10 +03:00
dev_warn ( & pcf85063 - > rtc - > dev , " Unknown quartz-load-femtofarads value: %d. Assuming 7000 " ,
2019-01-19 12:00:31 +03:00
load ) ;
2020-08-24 01:36:59 +03:00
fallthrough ;
2019-01-19 12:00:31 +03:00
case 7000 :
break ;
case 12500 :
2019-04-01 19:08:10 +03:00
reg = PCF85063_REG_CTRL1_CAP_SEL ;
2019-01-19 12:00:31 +03:00
break ;
}
2019-04-01 19:08:10 +03:00
return regmap_update_bits ( pcf85063 - > regmap , PCF85063_REG_CTRL1 ,
PCF85063_REG_CTRL1_CAP_SEL , reg ) ;
2019-01-19 12:00:31 +03:00
}
2020-01-24 04:52:38 +03:00
# ifdef CONFIG_COMMON_CLK
/*
* Handling of the clkout
*/
# define clkout_hw_to_pcf85063(_hw) container_of(_hw, struct pcf85063, clkout_hw)
static int clkout_rates [ ] = {
32768 ,
16384 ,
8192 ,
4096 ,
2048 ,
1024 ,
1 ,
0
} ;
static unsigned long pcf85063_clkout_recalc_rate ( struct clk_hw * hw ,
unsigned long parent_rate )
{
struct pcf85063 * pcf85063 = clkout_hw_to_pcf85063 ( hw ) ;
unsigned int buf ;
int ret = regmap_read ( pcf85063 - > regmap , PCF85063_REG_CTRL2 , & buf ) ;
if ( ret < 0 )
return 0 ;
buf & = PCF85063_REG_CLKO_F_MASK ;
return clkout_rates [ buf ] ;
}
static long pcf85063_clkout_round_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long * prate )
{
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( clkout_rates ) ; i + + )
if ( clkout_rates [ i ] < = rate )
return clkout_rates [ i ] ;
return 0 ;
}
static int pcf85063_clkout_set_rate ( struct clk_hw * hw , unsigned long rate ,
unsigned long parent_rate )
{
struct pcf85063 * pcf85063 = clkout_hw_to_pcf85063 ( hw ) ;
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( clkout_rates ) ; i + + )
if ( clkout_rates [ i ] = = rate )
return regmap_update_bits ( pcf85063 - > regmap ,
PCF85063_REG_CTRL2 ,
PCF85063_REG_CLKO_F_MASK , i ) ;
return - EINVAL ;
}
static int pcf85063_clkout_control ( struct clk_hw * hw , bool enable )
{
struct pcf85063 * pcf85063 = clkout_hw_to_pcf85063 ( hw ) ;
unsigned int buf ;
int ret ;
2022-12-12 01:35:53 +03:00
ret = regmap_read ( pcf85063 - > regmap , PCF85063_REG_CTRL2 , & buf ) ;
2020-01-24 04:52:38 +03:00
if ( ret < 0 )
return ret ;
buf & = PCF85063_REG_CLKO_F_MASK ;
if ( enable ) {
if ( buf = = PCF85063_REG_CLKO_F_OFF )
buf = PCF85063_REG_CLKO_F_32768HZ ;
else
return 0 ;
} else {
if ( buf ! = PCF85063_REG_CLKO_F_OFF )
buf = PCF85063_REG_CLKO_F_OFF ;
else
return 0 ;
}
return regmap_update_bits ( pcf85063 - > regmap , PCF85063_REG_CTRL2 ,
PCF85063_REG_CLKO_F_MASK , buf ) ;
}
static int pcf85063_clkout_prepare ( struct clk_hw * hw )
{
return pcf85063_clkout_control ( hw , 1 ) ;
}
static void pcf85063_clkout_unprepare ( struct clk_hw * hw )
{
pcf85063_clkout_control ( hw , 0 ) ;
}
static int pcf85063_clkout_is_prepared ( struct clk_hw * hw )
{
struct pcf85063 * pcf85063 = clkout_hw_to_pcf85063 ( hw ) ;
unsigned int buf ;
int ret = regmap_read ( pcf85063 - > regmap , PCF85063_REG_CTRL2 , & buf ) ;
if ( ret < 0 )
return 0 ;
return ( buf & PCF85063_REG_CLKO_F_MASK ) ! = PCF85063_REG_CLKO_F_OFF ;
}
static const struct clk_ops pcf85063_clkout_ops = {
. prepare = pcf85063_clkout_prepare ,
. unprepare = pcf85063_clkout_unprepare ,
. is_prepared = pcf85063_clkout_is_prepared ,
. recalc_rate = pcf85063_clkout_recalc_rate ,
. round_rate = pcf85063_clkout_round_rate ,
. set_rate = pcf85063_clkout_set_rate ,
} ;
static struct clk * pcf85063_clkout_register_clk ( struct pcf85063 * pcf85063 )
{
struct clk * clk ;
struct clk_init_data init ;
2021-03-11 00:10:26 +03:00
struct device_node * node = pcf85063 - > rtc - > dev . parent - > of_node ;
2021-10-13 10:49:54 +03:00
struct device_node * fixed_clock ;
fixed_clock = of_get_child_by_name ( node , " clock " ) ;
if ( fixed_clock ) {
/*
* skip registering square wave clock when a fixed
* clock has been registered . The fixed clock is
* registered automatically when being referenced .
*/
of_node_put ( fixed_clock ) ;
return NULL ;
}
2020-01-24 04:52:38 +03:00
init . name = " pcf85063-clkout " ;
init . ops = & pcf85063_clkout_ops ;
init . flags = 0 ;
init . parent_names = NULL ;
init . num_parents = 0 ;
pcf85063 - > clkout_hw . init = & init ;
/* optional override of the clockname */
2021-03-11 00:10:26 +03:00
of_property_read_string ( node , " clock-output-names " , & init . name ) ;
2020-01-24 04:52:38 +03:00
/* register the clock */
clk = devm_clk_register ( & pcf85063 - > rtc - > dev , & pcf85063 - > clkout_hw ) ;
if ( ! IS_ERR ( clk ) )
2021-03-11 00:10:26 +03:00
of_clk_add_provider ( node , of_clk_src_simple_get , clk ) ;
2020-01-24 04:52:38 +03:00
return clk ;
}
# endif
2021-11-16 19:47:33 +03:00
enum pcf85063_type {
PCF85063 ,
PCF85063TP ,
PCF85063A ,
RV8263 ,
PCF85063_LAST_ID
} ;
static struct pcf85063_config pcf85063_cfg [ ] = {
[ PCF85063 ] = {
. regmap = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = 0x0a ,
} ,
} ,
[ PCF85063TP ] = {
. regmap = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = 0x0a ,
} ,
} ,
[ PCF85063A ] = {
. regmap = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = 0x11 ,
} ,
. has_alarms = 1 ,
} ,
[ RV8263 ] = {
. regmap = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = 0x11 ,
} ,
. has_alarms = 1 ,
. force_cap_7000 = 1 ,
2019-04-01 19:08:11 +03:00
} ,
2019-04-01 19:08:10 +03:00
} ;
2021-11-16 19:47:33 +03:00
static const struct i2c_device_id pcf85063_ids [ ] ;
2019-04-01 19:08:05 +03:00
static int pcf85063_probe ( struct i2c_client * client )
2014-08-09 01:20:22 +04:00
{
2019-04-01 19:08:10 +03:00
struct pcf85063 * pcf85063 ;
unsigned int tmp ;
2016-10-17 16:53:31 +03:00
int err ;
2021-11-16 19:47:33 +03:00
const struct pcf85063_config * config ;
2019-04-01 19:08:14 +03:00
struct nvmem_config nvmem_cfg = {
. name = " pcf85063_nvram " ,
. reg_read = pcf85063_nvmem_read ,
. reg_write = pcf85063_nvmem_write ,
. type = NVMEM_TYPE_BATTERY_BACKED ,
. size = 1 ,
} ;
2014-08-09 01:20:22 +04:00
dev_dbg ( & client - > dev , " %s \n " , __func__ ) ;
2019-04-01 19:08:10 +03:00
pcf85063 = devm_kzalloc ( & client - > dev , sizeof ( struct pcf85063 ) ,
GFP_KERNEL ) ;
if ( ! pcf85063 )
return - ENOMEM ;
2021-11-16 19:47:33 +03:00
if ( client - > dev . of_node ) {
config = of_device_get_match_data ( & client - > dev ) ;
if ( ! config )
return - ENODEV ;
} else {
enum pcf85063_type type =
i2c_match_id ( pcf85063_ids , client ) - > driver_data ;
if ( type > = PCF85063_LAST_ID )
return - ENODEV ;
config = & pcf85063_cfg [ type ] ;
}
2019-04-01 19:08:11 +03:00
pcf85063 - > regmap = devm_regmap_init_i2c ( client , & config - > regmap ) ;
2019-04-01 19:08:10 +03:00
if ( IS_ERR ( pcf85063 - > regmap ) )
return PTR_ERR ( pcf85063 - > regmap ) ;
i2c_set_clientdata ( client , pcf85063 ) ;
err = regmap_read ( pcf85063 - > regmap , PCF85063_REG_CTRL1 , & tmp ) ;
if ( err ) {
2016-10-17 16:53:31 +03:00
dev_err ( & client - > dev , " RTC chip is not present \n " ) ;
return err ;
}
2019-04-01 19:08:10 +03:00
pcf85063 - > rtc = devm_rtc_allocate_device ( & client - > dev ) ;
if ( IS_ERR ( pcf85063 - > rtc ) )
return PTR_ERR ( pcf85063 - > rtc ) ;
2019-04-01 19:08:13 +03:00
err = pcf85063_load_capacitance ( pcf85063 , client - > dev . of_node ,
config - > force_cap_7000 ? 7000 : 0 ) ;
2019-01-19 12:00:31 +03:00
if ( err < 0 )
dev_warn ( & client - > dev , " failed to set xtal load capacitance: %d " ,
err ) ;
2019-04-01 19:08:10 +03:00
pcf85063 - > rtc - > ops = & pcf85063_rtc_ops ;
pcf85063 - > rtc - > range_min = RTC_TIMESTAMP_BEGIN_2000 ;
pcf85063 - > rtc - > range_max = RTC_TIMESTAMP_END_2099 ;
2022-03-09 19:22:44 +03:00
set_bit ( RTC_FEATURE_ALARM_RES_2S , pcf85063 - > rtc - > features ) ;
2022-03-09 19:22:43 +03:00
clear_bit ( RTC_FEATURE_UPDATE_INTERRUPT , pcf85063 - > rtc - > features ) ;
2021-01-11 02:17:43 +03:00
clear_bit ( RTC_FEATURE_ALARM , pcf85063 - > rtc - > features ) ;
2019-04-01 19:08:12 +03:00
if ( config - > has_alarms & & client - > irq > 0 ) {
2023-01-23 23:02:11 +03:00
unsigned long irqflags = IRQF_TRIGGER_LOW ;
if ( dev_fwnode ( & client - > dev ) )
irqflags = 0 ;
2019-04-01 19:08:12 +03:00
err = devm_request_threaded_irq ( & client - > dev , client - > irq ,
NULL , pcf85063_rtc_handle_irq ,
2023-01-23 23:02:11 +03:00
irqflags | IRQF_ONESHOT ,
2019-04-01 19:08:12 +03:00
" pcf85063 " , pcf85063 ) ;
if ( err ) {
dev_warn ( & pcf85063 - > rtc - > dev ,
" unable to request IRQ, alarms disabled \n " ) ;
} else {
2021-01-11 02:17:43 +03:00
set_bit ( RTC_FEATURE_ALARM , pcf85063 - > rtc - > features ) ;
2019-04-01 19:08:12 +03:00
device_init_wakeup ( & client - > dev , true ) ;
err = dev_pm_set_wake_irq ( & client - > dev , client - > irq ) ;
if ( err )
dev_err ( & pcf85063 - > rtc - > dev ,
" failed to enable irq wake \n " ) ;
}
}
2014-08-09 01:20:22 +04:00
2019-04-01 19:08:14 +03:00
nvmem_cfg . priv = pcf85063 - > regmap ;
2020-11-09 19:34:06 +03:00
devm_rtc_nvmem_register ( pcf85063 - > rtc , & nvmem_cfg ) ;
2019-04-01 19:08:14 +03:00
2020-01-24 04:52:38 +03:00
# ifdef CONFIG_COMMON_CLK
/* register clk in common clk framework */
pcf85063_clkout_register_clk ( pcf85063 ) ;
# endif
2020-11-09 19:34:08 +03:00
return devm_rtc_register_device ( pcf85063 - > rtc ) ;
2014-08-09 01:20:22 +04:00
}
2021-11-16 19:47:33 +03:00
static const struct i2c_device_id pcf85063_ids [ ] = {
2022-04-19 04:44:45 +03:00
{ " pca85073a " , PCF85063A } ,
2021-11-16 19:47:33 +03:00
{ " pcf85063 " , PCF85063 } ,
{ " pcf85063tp " , PCF85063TP } ,
{ " pcf85063a " , PCF85063A } ,
{ " rv8263 " , RV8263 } ,
{ }
2021-02-02 14:22:08 +03:00
} ;
2021-11-16 19:47:33 +03:00
MODULE_DEVICE_TABLE ( i2c , pcf85063_ids ) ;
2021-02-02 14:22:08 +03:00
2021-11-16 19:47:33 +03:00
# ifdef CONFIG_OF
2014-08-09 01:20:22 +04:00
static const struct of_device_id pcf85063_of_match [ ] = {
2022-04-19 04:44:45 +03:00
{ . compatible = " nxp,pca85073a " , . data = & pcf85063_cfg [ PCF85063A ] } ,
2021-11-16 19:47:33 +03:00
{ . compatible = " nxp,pcf85063 " , . data = & pcf85063_cfg [ PCF85063 ] } ,
{ . compatible = " nxp,pcf85063tp " , . data = & pcf85063_cfg [ PCF85063TP ] } ,
{ . compatible = " nxp,pcf85063a " , . data = & pcf85063_cfg [ PCF85063A ] } ,
{ . compatible = " microcrystal,rv8263 " , . data = & pcf85063_cfg [ RV8263 ] } ,
2014-08-09 01:20:22 +04:00
{ }
} ;
MODULE_DEVICE_TABLE ( of , pcf85063_of_match ) ;
# endif
static struct i2c_driver pcf85063_driver = {
. driver = {
. name = " rtc-pcf85063 " ,
. of_match_table = of_match_ptr ( pcf85063_of_match ) ,
} ,
2019-04-01 19:08:05 +03:00
. probe_new = pcf85063_probe ,
2021-11-16 19:47:33 +03:00
. id_table = pcf85063_ids ,
2014-08-09 01:20:22 +04:00
} ;
module_i2c_driver ( pcf85063_driver ) ;
MODULE_AUTHOR ( " Søren Andersen <san@rosetechnology.dk> " ) ;
MODULE_DESCRIPTION ( " PCF85063 RTC driver " ) ;
MODULE_LICENSE ( " GPL " ) ;