2006-03-27 01:16:43 -08:00
/*
* An rtc / i2c driver for the Dallas DS1672
2006-04-10 22:54:41 -07:00
* Copyright 2005 - 06 Tower Technologies
*
* Author : Alessandro Zummo < a . zummo @ towertech . it >
2006-03-27 01:16:43 -08:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/i2c.h>
# include <linux/rtc.h>
2008-10-15 22:03:10 -07:00
# define DRV_VERSION "0.4"
2006-03-27 01:16:43 -08:00
/* Registers */
# define DS1672_REG_CNT_BASE 0
# define DS1672_REG_CONTROL 4
# define DS1672_REG_TRICKLE 5
2006-04-10 22:54:41 -07:00
# define DS1672_REG_CONTROL_EOSC 0x80
2006-03-27 01:16:43 -08:00
2008-10-15 22:03:10 -07:00
static struct i2c_driver ds1672_driver ;
2006-03-27 01:16:43 -08: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
* Epoch is initialized as 2000. Time is set to UTC .
*/
static int ds1672_get_datetime ( struct i2c_client * client , struct rtc_time * tm )
{
unsigned long time ;
unsigned char addr = DS1672_REG_CNT_BASE ;
unsigned char buf [ 4 ] ;
struct i2c_msg msgs [ ] = {
2008-10-15 22:03:10 -07:00
{ client - > addr , 0 , 1 , & addr } , /* setup read ptr */
{ client - > addr , I2C_M_RD , 4 , buf } , /* read date */
2006-03-27 01:16:43 -08:00
} ;
/* read date registers */
if ( ( i2c_transfer ( client - > adapter , & msgs [ 0 ] , 2 ) ) ! = 2 ) {
2008-04-28 02:12:00 -07:00
dev_err ( & client - > dev , " %s: read error \n " , __func__ ) ;
2006-03-27 01:16:43 -08:00
return - EIO ;
}
dev_dbg ( & client - > dev ,
2006-10-04 04:41:53 -04:00
" %s: raw read data - counters=%02x,%02x,%02x,%02x \n " ,
2008-04-28 02:12:00 -07:00
__func__ , buf [ 0 ] , buf [ 1 ] , buf [ 2 ] , buf [ 3 ] ) ;
2006-03-27 01:16:43 -08:00
time = ( buf [ 3 ] < < 24 ) | ( buf [ 2 ] < < 16 ) | ( buf [ 1 ] < < 8 ) | buf [ 0 ] ;
rtc_time_to_tm ( time , tm ) ;
dev_dbg ( & client - > dev , " %s: tm is secs=%d, mins=%d, hours=%d, "
" mday=%d, mon=%d, year=%d, wday=%d \n " ,
2008-04-28 02:12:00 -07:00
__func__ , tm - > tm_sec , tm - > tm_min , tm - > tm_hour ,
2006-03-27 01:16:43 -08:00
tm - > tm_mday , tm - > tm_mon , tm - > tm_year , tm - > tm_wday ) ;
return 0 ;
}
static int ds1672_set_mmss ( struct i2c_client * client , unsigned long secs )
{
int xfer ;
2006-04-10 22:54:39 -07:00
unsigned char buf [ 6 ] ;
2006-03-27 01:16:43 -08: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-15 22:03:10 -07:00
buf [ 5 ] = 0 ; /* set control reg to enable counting */
2006-03-27 01:16:43 -08:00
2006-04-10 22:54:39 -07:00
xfer = i2c_master_send ( client , buf , 6 ) ;
if ( xfer ! = 6 ) {
2008-04-28 02:12:00 -07:00
dev_err ( & client - > dev , " %s: send: %d \n " , __func__ , xfer ) ;
2006-03-27 01:16:43 -08:00
return - EIO ;
}
return 0 ;
}
static int ds1672_set_datetime ( struct i2c_client * client , struct rtc_time * tm )
{
unsigned long secs ;
dev_dbg ( & client - > dev ,
2006-10-04 04:41:53 -04:00
" %s: secs=%d, mins=%d, hours=%d, "
2006-03-27 01:16:43 -08:00
" mday=%d, mon=%d, year=%d, wday=%d \n " ,
2008-04-28 02:12:00 -07:00
__func__ ,
2006-03-27 01:16:43 -08:00
tm - > tm_sec , tm - > tm_min , tm - > tm_hour ,
tm - > tm_mday , tm - > tm_mon , tm - > tm_year , tm - > tm_wday ) ;
rtc_tm_to_time ( tm , & secs ) ;
return ds1672_set_mmss ( client , secs ) ;
}
static int ds1672_rtc_read_time ( struct device * dev , struct rtc_time * tm )
{
return ds1672_get_datetime ( to_i2c_client ( dev ) , tm ) ;
}
static int ds1672_rtc_set_time ( struct device * dev , struct rtc_time * tm )
{
return ds1672_set_datetime ( to_i2c_client ( dev ) , tm ) ;
}
static int ds1672_rtc_set_mmss ( struct device * dev , unsigned long secs )
{
return ds1672_set_mmss ( to_i2c_client ( dev ) , secs ) ;
}
2006-04-10 22:54:39 -07:00
static int ds1672_get_control ( struct i2c_client * client , u8 * status )
{
unsigned char addr = DS1672_REG_CONTROL ;
struct i2c_msg msgs [ ] = {
2008-10-15 22:03:10 -07:00
{ client - > addr , 0 , 1 , & addr } , /* setup read ptr */
{ client - > addr , I2C_M_RD , 1 , status } , /* read control */
2006-04-10 22:54:39 -07:00
} ;
/* read control register */
if ( ( i2c_transfer ( client - > adapter , & msgs [ 0 ] , 2 ) ) ! = 2 ) {
2008-04-28 02:12:00 -07:00
dev_err ( & client - > dev , " %s: read error \n " , __func__ ) ;
2006-04-10 22:54:39 -07:00
return - EIO ;
}
return 0 ;
}
/* following are the sysfs callback functions */
2008-10-15 22:03:10 -07:00
static ssize_t show_control ( struct device * dev , struct device_attribute * attr ,
char * buf )
2006-04-10 22:54:39 -07:00
{
struct i2c_client * client = to_i2c_client ( dev ) ;
u8 control ;
int err ;
err = ds1672_get_control ( client , & control ) ;
if ( err )
return err ;
2006-04-10 22:54:41 -07:00
return sprintf ( buf , " %s \n " , ( control & DS1672_REG_CONTROL_EOSC )
2008-10-15 22:03:10 -07:00
? " disabled " : " enabled " ) ;
2006-04-10 22:54:39 -07:00
}
2008-10-15 22:03:10 -07:00
2006-04-10 22:54:39 -07:00
static DEVICE_ATTR ( control , S_IRUGO , show_control , NULL ) ;
2006-09-30 23:28:17 -07:00
static const struct rtc_class_ops ds1672_rtc_ops = {
2008-10-15 22:03:10 -07:00
. read_time = ds1672_rtc_read_time ,
. set_time = ds1672_rtc_set_time ,
. set_mmss = ds1672_rtc_set_mmss ,
2006-03-27 01:16:43 -08:00
} ;
2008-10-15 22:03:10 -07:00
static int ds1672_remove ( struct i2c_client * client )
2006-03-27 01:16:43 -08:00
{
struct rtc_device * rtc = i2c_get_clientdata ( client ) ;
2008-10-15 22:03:10 -07:00
if ( rtc )
2006-03-27 01:16:43 -08:00
rtc_device_unregister ( rtc ) ;
return 0 ;
}
2008-10-15 22:03:10 -07:00
static int ds1672_probe ( struct i2c_client * client ,
const struct i2c_device_id * id )
2006-03-27 01:16:43 -08:00
{
int err = 0 ;
2006-04-10 22:54:39 -07:00
u8 control ;
2006-03-27 01:16:43 -08:00
struct rtc_device * rtc ;
2008-10-15 22:03:10 -07:00
dev_dbg ( & client - > dev , " %s \n " , __func__ ) ;
2006-03-27 01:16:43 -08:00
2008-10-15 22:03:10 -07:00
if ( ! i2c_check_functionality ( client - > adapter , I2C_FUNC_I2C ) )
return - ENODEV ;
2006-03-27 01:16:43 -08:00
dev_info ( & client - > dev , " chip found, driver version " DRV_VERSION " \n " ) ;
rtc = rtc_device_register ( ds1672_driver . driver . name , & client - > dev ,
2008-10-15 22:03:10 -07:00
& ds1672_rtc_ops , THIS_MODULE ) ;
2006-03-27 01:16:43 -08:00
2008-10-15 22:03:10 -07:00
if ( IS_ERR ( rtc ) )
return PTR_ERR ( rtc ) ;
2006-03-27 01:16:43 -08:00
i2c_set_clientdata ( client , rtc ) ;
2006-04-10 22:54:39 -07:00
/* read control register */
err = ds1672_get_control ( client , & control ) ;
2006-04-10 22:54:41 -07:00
if ( err )
2006-12-06 20:35:34 -08:00
goto exit_devreg ;
2006-04-10 22:54:39 -07:00
if ( control & DS1672_REG_CONTROL_EOSC )
dev_warn ( & client - > dev , " Oscillator not enabled. "
2008-10-15 22:03:10 -07:00
" Set time to enable. \n " ) ;
2006-04-10 22:54:39 -07:00
/* Register sysfs hooks */
2006-12-06 20:35:34 -08:00
err = device_create_file ( & client - > dev , & dev_attr_control ) ;
if ( err )
goto exit_devreg ;
2006-04-10 22:54:39 -07:00
2006-03-27 01:16:43 -08:00
return 0 ;
2008-10-15 22:03:10 -07:00
exit_devreg :
2006-12-06 20:35:34 -08:00
rtc_device_unregister ( rtc ) ;
2006-03-27 01:16:43 -08:00
return err ;
}
2008-10-15 22:03:10 -07:00
static struct i2c_driver ds1672_driver = {
. driver = {
. name = " rtc-ds1672 " ,
} ,
. probe = & ds1672_probe ,
. remove = & ds1672_remove ,
} ;
2006-03-27 01:16:43 -08:00
static int __init ds1672_init ( void )
{
return i2c_add_driver ( & ds1672_driver ) ;
}
static void __exit ds1672_exit ( void )
{
i2c_del_driver ( & ds1672_driver ) ;
}
MODULE_AUTHOR ( " Alessandro Zummo <a.zummo@towertech.it> " ) ;
MODULE_DESCRIPTION ( " Dallas/Maxim DS1672 timekeeper driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_VERSION ( DRV_VERSION ) ;
module_init ( ds1672_init ) ;
module_exit ( ds1672_exit ) ;