2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2010-08-10 18:02:14 -07:00
/*
* An I2C driver for the Intersil ISL 12022
*
* Author : Roman Fietze < roman . fietze @ telemotive . de >
*
* Based on the Philips PCF8563 RTC
* by Alessandro Zummo < a . zummo @ towertech . it > .
*/
# include <linux/bcd.h>
2013-07-03 15:07:49 -07:00
# include <linux/err.h>
2022-11-04 12:02:25 +01:00
# include <linux/hwmon.h>
2023-01-10 16:08:06 +02:00
# include <linux/i2c.h>
# include <linux/module.h>
# include <linux/regmap.h>
# include <linux/rtc.h>
# include <linux/slab.h>
2010-08-10 18:02:14 -07:00
2023-01-10 16:08:03 +02:00
# include <asm/byteorder.h>
2010-08-10 18:02:14 -07:00
/* ISL register offsets */
# define ISL12022_REG_SC 0x00
# define ISL12022_REG_MN 0x01
# define ISL12022_REG_HR 0x02
# define ISL12022_REG_DT 0x03
# define ISL12022_REG_MO 0x04
# define ISL12022_REG_YR 0x05
# define ISL12022_REG_DW 0x06
# define ISL12022_REG_SR 0x07
# define ISL12022_REG_INT 0x08
2022-11-04 12:02:25 +01:00
# define ISL12022_REG_BETA 0x0d
# define ISL12022_REG_TEMP_L 0x28
2010-08-10 18:02:14 -07:00
/* ISL register bits */
# define ISL12022_HR_MIL (1 << 7) /* military or 24 hour time */
# define ISL12022_SR_LBAT85 (1 << 2)
# define ISL12022_SR_LBAT75 (1 << 1)
# define ISL12022_INT_WRTC (1 << 6)
2022-11-04 12:02:25 +01:00
# define ISL12022_BETA_TSE (1 << 7)
2010-08-10 18:02:14 -07:00
2022-11-04 12:02:25 +01:00
static umode_t isl12022_hwmon_is_visible ( const void * data ,
enum hwmon_sensor_types type ,
u32 attr , int channel )
{
if ( type = = hwmon_temp & & attr = = hwmon_temp_input )
return 0444 ;
return 0 ;
}
/*
* A user - initiated temperature conversion is not started by this function ,
* so the temperature is updated once every ~ 60 seconds .
*/
static int isl12022_hwmon_read_temp ( struct device * dev , long * mC )
{
2023-01-10 16:08:02 +02:00
struct regmap * regmap = dev_get_drvdata ( dev ) ;
2022-11-04 12:02:25 +01:00
int temp , ret ;
2023-01-10 16:08:03 +02:00
__le16 buf ;
2022-11-04 12:02:25 +01:00
2023-01-10 16:08:03 +02:00
ret = regmap_bulk_read ( regmap , ISL12022_REG_TEMP_L , & buf , sizeof ( buf ) ) ;
2022-11-04 12:02:25 +01:00
if ( ret )
return ret ;
/*
* Temperature is represented as a 10 - bit number , unit half - Kelvins .
*/
2023-01-10 16:08:03 +02:00
temp = le16_to_cpu ( buf ) ;
2022-11-04 12:02:25 +01:00
temp * = 500 ;
temp - = 273000 ;
* mC = temp ;
return 0 ;
}
static int isl12022_hwmon_read ( struct device * dev ,
enum hwmon_sensor_types type ,
u32 attr , int channel , long * val )
{
if ( type = = hwmon_temp & & attr = = hwmon_temp_input )
return isl12022_hwmon_read_temp ( dev , val ) ;
return - EOPNOTSUPP ;
}
static const struct hwmon_channel_info * isl12022_hwmon_info [ ] = {
HWMON_CHANNEL_INFO ( temp , HWMON_T_INPUT ) ,
NULL
} ;
static const struct hwmon_ops isl12022_hwmon_ops = {
. is_visible = isl12022_hwmon_is_visible ,
. read = isl12022_hwmon_read ,
} ;
static const struct hwmon_chip_info isl12022_hwmon_chip_info = {
. ops = & isl12022_hwmon_ops ,
. info = isl12022_hwmon_info ,
} ;
static void isl12022_hwmon_register ( struct device * dev )
{
2023-01-10 16:08:02 +02:00
struct regmap * regmap = dev_get_drvdata ( dev ) ;
2022-11-04 12:02:25 +01:00
struct device * hwmon ;
int ret ;
if ( ! IS_REACHABLE ( CONFIG_HWMON ) )
return ;
2023-01-10 16:08:02 +02:00
ret = regmap_update_bits ( regmap , ISL12022_REG_BETA ,
2022-11-04 12:02:25 +01:00
ISL12022_BETA_TSE , ISL12022_BETA_TSE ) ;
if ( ret ) {
dev_warn ( dev , " unable to enable temperature sensor \n " ) ;
return ;
}
2023-01-10 16:08:02 +02:00
hwmon = devm_hwmon_device_register_with_info ( dev , " isl12022 " , regmap ,
2022-11-04 12:02:25 +01:00
& isl12022_hwmon_chip_info ,
NULL ) ;
if ( IS_ERR ( hwmon ) )
dev_warn ( dev , " unable to register hwmon device: %pe \n " , hwmon ) ;
}
2010-08-10 18:02:14 -07:00
/*
* In the routines that deal directly with the isl12022 hardware , we use
* rtc_time - - month 0 - 11 , hour 0 - 23 , yr = calendar year - epoch .
*/
2018-02-20 23:55:40 +01:00
static int isl12022_rtc_read_time ( struct device * dev , struct rtc_time * tm )
2010-08-10 18:02:14 -07:00
{
2023-01-10 16:08:02 +02:00
struct regmap * regmap = dev_get_drvdata ( dev ) ;
2010-08-10 18:02:14 -07:00
uint8_t buf [ ISL12022_REG_INT + 1 ] ;
int ret ;
2022-09-21 13:46:23 +02:00
ret = regmap_bulk_read ( regmap , ISL12022_REG_SC , buf , sizeof ( buf ) ) ;
2010-08-10 18:02:14 -07:00
if ( ret )
return ret ;
if ( buf [ ISL12022_REG_SR ] & ( ISL12022_SR_LBAT85 | ISL12022_SR_LBAT75 ) ) {
2022-09-21 13:46:19 +02:00
dev_warn ( dev ,
2023-01-10 16:08:05 +02:00
" voltage dropped below %u%%, date and time is not reliable. \n " ,
2010-08-10 18:02:14 -07:00
buf [ ISL12022_REG_SR ] & ISL12022_SR_LBAT85 ? 85 : 75 ) ;
}
2022-09-21 13:46:19 +02:00
dev_dbg ( dev ,
2023-01-10 16:08:05 +02:00
" raw data is sec=%02x, min=%02x, hr=%02x, mday=%02x, mon=%02x, year=%02x, wday=%02x, sr=%02x, int=%02x " ,
2010-08-10 18:02:14 -07:00
buf [ ISL12022_REG_SC ] ,
buf [ ISL12022_REG_MN ] ,
buf [ ISL12022_REG_HR ] ,
buf [ ISL12022_REG_DT ] ,
buf [ ISL12022_REG_MO ] ,
buf [ ISL12022_REG_YR ] ,
buf [ ISL12022_REG_DW ] ,
buf [ ISL12022_REG_SR ] ,
buf [ ISL12022_REG_INT ] ) ;
tm - > tm_sec = bcd2bin ( buf [ ISL12022_REG_SC ] & 0x7F ) ;
tm - > tm_min = bcd2bin ( buf [ ISL12022_REG_MN ] & 0x7F ) ;
tm - > tm_hour = bcd2bin ( buf [ ISL12022_REG_HR ] & 0x3F ) ;
tm - > tm_mday = bcd2bin ( buf [ ISL12022_REG_DT ] & 0x3F ) ;
tm - > tm_wday = buf [ ISL12022_REG_DW ] & 0x07 ;
tm - > tm_mon = bcd2bin ( buf [ ISL12022_REG_MO ] & 0x1F ) - 1 ;
tm - > tm_year = bcd2bin ( buf [ ISL12022_REG_YR ] ) + 100 ;
2022-09-21 13:46:20 +02:00
dev_dbg ( dev , " %s: %ptR \n " , __func__ , tm ) ;
2010-08-10 18:02:14 -07:00
2018-02-19 16:23:56 +01:00
return 0 ;
2010-08-10 18:02:14 -07:00
}
2018-02-20 23:55:40 +01:00
static int isl12022_rtc_set_time ( struct device * dev , struct rtc_time * tm )
2010-08-10 18:02:14 -07:00
{
2023-01-10 16:08:02 +02:00
struct regmap * regmap = dev_get_drvdata ( dev ) ;
2010-08-10 18:02:14 -07:00
int ret ;
uint8_t buf [ ISL12022_REG_DW + 1 ] ;
2022-09-21 13:46:20 +02:00
dev_dbg ( dev , " %s: %ptR \n " , __func__ , tm ) ;
2010-08-10 18:02:14 -07:00
2022-09-21 13:46:23 +02:00
/* Ensure the write enable bit is set. */
ret = regmap_update_bits ( regmap , ISL12022_REG_INT ,
ISL12022_INT_WRTC , ISL12022_INT_WRTC ) ;
if ( ret )
return ret ;
2010-08-10 18:02:14 -07:00
/* hours, minutes and seconds */
buf [ ISL12022_REG_SC ] = bin2bcd ( tm - > tm_sec ) ;
buf [ ISL12022_REG_MN ] = bin2bcd ( tm - > tm_min ) ;
2010-08-10 18:02:21 -07:00
buf [ ISL12022_REG_HR ] = bin2bcd ( tm - > tm_hour ) | ISL12022_HR_MIL ;
2010-08-10 18:02:14 -07:00
buf [ ISL12022_REG_DT ] = bin2bcd ( tm - > tm_mday ) ;
/* month, 1 - 12 */
buf [ ISL12022_REG_MO ] = bin2bcd ( tm - > tm_mon + 1 ) ;
/* year and century */
buf [ ISL12022_REG_YR ] = bin2bcd ( tm - > tm_year % 100 ) ;
buf [ ISL12022_REG_DW ] = tm - > tm_wday & 0x07 ;
2023-01-10 16:08:02 +02:00
return regmap_bulk_write ( regmap , ISL12022_REG_SC , buf , sizeof ( buf ) ) ;
2010-08-10 18:02:14 -07:00
}
static const struct rtc_class_ops isl12022_rtc_ops = {
. read_time = isl12022_rtc_read_time ,
. set_time = isl12022_rtc_set_time ,
} ;
2022-09-21 13:46:23 +02:00
static const struct regmap_config regmap_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. use_single_write = true ,
} ;
2022-06-10 18:23:43 +02:00
static int isl12022_probe ( struct i2c_client * client )
2010-08-10 18:02:14 -07:00
{
2023-01-10 16:08:02 +02:00
struct rtc_device * rtc ;
struct regmap * regmap ;
2010-08-10 18:02:14 -07:00
if ( ! i2c_check_functionality ( client - > adapter , I2C_FUNC_I2C ) )
return - ENODEV ;
2023-01-10 16:08:02 +02:00
regmap = devm_regmap_init_i2c ( client , & regmap_config ) ;
if ( IS_ERR ( regmap ) ) {
2022-09-21 13:46:23 +02:00
dev_err ( & client - > dev , " regmap allocation failed \n " ) ;
2023-01-10 16:08:02 +02:00
return PTR_ERR ( regmap ) ;
2022-09-21 13:46:23 +02:00
}
2023-01-10 16:08:02 +02:00
dev_set_drvdata ( & client - > dev , regmap ) ;
2022-11-04 12:02:25 +01:00
isl12022_hwmon_register ( & client - > dev ) ;
2023-01-10 16:08:02 +02:00
rtc = devm_rtc_allocate_device ( & client - > dev ) ;
if ( IS_ERR ( rtc ) )
return PTR_ERR ( rtc ) ;
2022-09-21 13:46:16 +02:00
2023-01-10 16:08:02 +02:00
rtc - > ops = & isl12022_rtc_ops ;
rtc - > range_min = RTC_TIMESTAMP_BEGIN_2000 ;
rtc - > range_max = RTC_TIMESTAMP_END_2099 ;
2022-09-21 13:46:16 +02:00
2023-01-10 16:08:02 +02:00
return devm_rtc_register_device ( rtc ) ;
2010-08-10 18:02:14 -07:00
}
2014-10-13 15:52:39 -07:00
static const struct of_device_id isl12022_dt_match [ ] = {
2015-02-16 15:58:31 -08:00
{ . compatible = " isl,isl12022 " } , /* for backward compat., don't use */
{ . compatible = " isil,isl12022 " } ,
2014-08-08 14:20:20 -07:00
{ } ,
} ;
2015-07-30 18:18:46 +02:00
MODULE_DEVICE_TABLE ( of , isl12022_dt_match ) ;
2014-08-08 14:20:20 -07:00
2010-08-10 18:02:14 -07:00
static const struct i2c_device_id isl12022_id [ ] = {
{ " isl12022 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , isl12022_id ) ;
static struct i2c_driver isl12022_driver = {
. driver = {
. name = " rtc-isl12022 " ,
2023-01-10 16:08:04 +02:00
. of_match_table = isl12022_dt_match ,
2010-08-10 18:02:14 -07:00
} ,
2022-06-10 18:23:43 +02:00
. probe_new = isl12022_probe ,
2010-08-10 18:02:14 -07:00
. id_table = isl12022_id ,
} ;
2012-03-23 15:02:31 -07:00
module_i2c_driver ( isl12022_driver ) ;
2010-08-10 18:02:14 -07:00
MODULE_AUTHOR ( " roman.fietze@telemotive.de " ) ;
MODULE_DESCRIPTION ( " ISL 12022 RTC driver " ) ;
MODULE_LICENSE ( " GPL " ) ;