2019-04-08 00:05:40 +03:00
// SPDX-License-Identifier: GPL-2.0
2006-03-27 13:16:43 +04:00
/*
* An rtc / i2c driver for the Dallas DS1672
2006-04-11 09:54:41 +04:00
* Copyright 2005 - 06 Tower Technologies
*
* Author : Alessandro Zummo < a . zummo @ towertech . it >
2006-03-27 13:16:43 +04:00
*/
# include <linux/i2c.h>
# include <linux/rtc.h>
2011-05-27 17:57:25 +04:00
# include <linux/module.h>
2006-03-27 13:16:43 +04:00
/* Registers */
# define DS1672_REG_CNT_BASE 0
# define DS1672_REG_CONTROL 4
# define DS1672_REG_TRICKLE 5
2006-04-11 09:54:41 +04:00
# define DS1672_REG_CONTROL_EOSC 0x80
2006-03-27 13:16:43 +04:00
/*
* In the routines that deal directly with the ds1672 hardware , we use
* rtc_time - - month 0 - 11 , hour 0 - 23 , yr = calendar year - epoch
2019-04-08 00:05:34 +03:00
* Time is set to UTC .
2006-03-27 13:16:43 +04:00
*/
2019-04-08 00:05:37 +03:00
static int ds1672_read_time ( struct device * dev , struct rtc_time * tm )
2006-03-27 13:16:43 +04:00
{
2019-04-08 00:05:37 +03:00
struct i2c_client * client = to_i2c_client ( dev ) ;
2006-03-27 13:16:43 +04:00
unsigned long time ;
2019-04-08 00:05:35 +03:00
unsigned char addr = DS1672_REG_CONTROL ;
2006-03-27 13:16:43 +04:00
unsigned char buf [ 4 ] ;
struct i2c_msg msgs [ ] = {
2012-10-05 04:14:17 +04:00
{ /* setup read ptr */
. addr = client - > addr ,
. len = 1 ,
. buf = & addr
} ,
{ /* read date */
. addr = client - > addr ,
. flags = I2C_M_RD ,
2019-04-08 00:05:35 +03:00
. len = 1 ,
2012-10-05 04:14:17 +04:00
. buf = buf
} ,
2006-03-27 13:16:43 +04:00
} ;
2019-04-08 00:05:35 +03:00
/* read control register */
if ( ( i2c_transfer ( client - > adapter , & msgs [ 0 ] , 2 ) ) ! = 2 ) {
dev_warn ( & client - > dev , " Unable to read the control register \n " ) ;
return - EIO ;
}
if ( buf [ 0 ] & DS1672_REG_CONTROL_EOSC ) {
dev_warn ( & client - > dev , " Oscillator not enabled. Set time to enable. \n " ) ;
return - EINVAL ;
}
addr = DS1672_REG_CNT_BASE ;
msgs [ 1 ] . len = 4 ;
2006-03-27 13:16:43 +04:00
/* read date registers */
if ( ( i2c_transfer ( client - > adapter , & msgs [ 0 ] , 2 ) ) ! = 2 ) {
2008-04-28 13:12:00 +04:00
dev_err ( & client - > dev , " %s: read error \n " , __func__ ) ;
2006-03-27 13:16:43 +04:00
return - EIO ;
}
dev_dbg ( & client - > dev ,
2006-10-04 12:41:53 +04:00
" %s: raw read data - counters=%02x,%02x,%02x,%02x \n " ,
2008-04-28 13:12:00 +04:00
__func__ , buf [ 0 ] , buf [ 1 ] , buf [ 2 ] , buf [ 3 ] ) ;
2006-03-27 13:16:43 +04:00
2019-02-05 21:04:49 +03:00
time = ( ( unsigned long ) buf [ 3 ] < < 24 ) | ( buf [ 2 ] < < 16 ) |
( buf [ 1 ] < < 8 ) | buf [ 0 ] ;
2006-03-27 13:16:43 +04:00
2019-04-08 00:05:38 +03:00
rtc_time64_to_tm ( time , tm ) ;
2006-03-27 13:16:43 +04:00
2019-04-08 00:05:41 +03:00
dev_dbg ( & client - > dev , " %s: tm is %ptR \n " , __func__ , tm ) ;
2006-03-27 13:16:43 +04:00
return 0 ;
}
2019-04-08 00:05:39 +03:00
static int ds1672_set_time ( struct device * dev , struct rtc_time * tm )
2006-03-27 13:16:43 +04:00
{
2019-04-08 00:05:37 +03:00
struct i2c_client * client = to_i2c_client ( dev ) ;
2006-03-27 13:16:43 +04:00
int xfer ;
2006-04-11 09:54:39 +04:00
unsigned char buf [ 6 ] ;
2019-04-08 00:05:39 +03:00
unsigned long secs = rtc_tm_to_time64 ( tm ) ;
2006-03-27 13:16:43 +04:00
buf [ 0 ] = DS1672_REG_CNT_BASE ;
buf [ 1 ] = secs & 0x000000FF ;
buf [ 2 ] = ( secs & 0x0000FF00 ) > > 8 ;
buf [ 3 ] = ( secs & 0x00FF0000 ) > > 16 ;
buf [ 4 ] = ( secs & 0xFF000000 ) > > 24 ;
2008-10-16 09:03:10 +04:00
buf [ 5 ] = 0 ; /* set control reg to enable counting */
2006-03-27 13:16:43 +04:00
2006-04-11 09:54:39 +04:00
xfer = i2c_master_send ( client , buf , 6 ) ;
if ( xfer ! = 6 ) {
2008-04-28 13:12:00 +04:00
dev_err ( & client - > dev , " %s: send: %d \n " , __func__ , xfer ) ;
2006-03-27 13:16:43 +04:00
return - EIO ;
}
return 0 ;
}
2006-10-01 10:28:17 +04:00
static const struct rtc_class_ops ds1672_rtc_ops = {
2019-04-08 00:05:37 +03:00
. read_time = ds1672_read_time ,
2019-04-08 00:05:39 +03:00
. set_time = ds1672_set_time ,
2006-03-27 13:16:43 +04:00
} ;
2022-06-10 19:23:43 +03:00
static int ds1672_probe ( struct i2c_client * client )
2006-03-27 13:16:43 +04:00
{
int err = 0 ;
struct rtc_device * rtc ;
2008-10-16 09:03:10 +04:00
dev_dbg ( & client - > dev , " %s \n " , __func__ ) ;
2006-03-27 13:16:43 +04:00
2008-10-16 09:03:10 +04:00
if ( ! i2c_check_functionality ( client - > adapter , I2C_FUNC_I2C ) )
return - ENODEV ;
2006-03-27 13:16:43 +04:00
2019-04-08 00:05:34 +03:00
rtc = devm_rtc_allocate_device ( & client - > dev ) ;
if ( IS_ERR ( rtc ) )
return PTR_ERR ( rtc ) ;
rtc - > ops = & ds1672_rtc_ops ;
rtc - > range_max = U32_MAX ;
2020-11-09 19:34:08 +03:00
err = devm_rtc_register_device ( rtc ) ;
2019-04-08 00:05:34 +03:00
if ( err )
return err ;
2006-03-27 13:16:43 +04:00
i2c_set_clientdata ( client , rtc ) ;
return 0 ;
}
2017-08-19 22:07:55 +03:00
static const struct i2c_device_id ds1672_id [ ] = {
2008-12-10 00:14:11 +03:00
{ " ds1672 " , 0 } ,
{ }
} ;
2011-04-18 16:16:57 +04:00
MODULE_DEVICE_TABLE ( i2c , ds1672_id ) ;
2008-12-10 00:14:11 +03:00
2021-02-02 14:22:03 +03:00
static const __maybe_unused struct of_device_id ds1672_of_match [ ] = {
2017-03-03 17:29:18 +03:00
{ . compatible = " dallas,ds1672 " } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , ds1672_of_match ) ;
2008-10-16 09:03:10 +04:00
static struct i2c_driver ds1672_driver = {
. driver = {
. name = " rtc-ds1672 " ,
2017-03-03 17:29:18 +03:00
. of_match_table = of_match_ptr ( ds1672_of_match ) ,
} ,
2022-06-10 19:23:43 +03:00
. probe_new = ds1672_probe ,
2008-12-10 00:14:11 +03:00
. id_table = ds1672_id ,
2008-10-16 09:03:10 +04:00
} ;
2012-03-24 02:02:31 +04:00
module_i2c_driver ( ds1672_driver ) ;
2006-03-27 13:16:43 +04:00
MODULE_AUTHOR ( " Alessandro Zummo <a.zummo@towertech.it> " ) ;
MODULE_DESCRIPTION ( " Dallas/Maxim DS1672 timekeeper driver " ) ;
MODULE_LICENSE ( " GPL " ) ;