2019-05-31 11:09:37 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2008-02-06 12:38:44 +03:00
/*
* Dallas DS1302 RTC Support
*
2009-08-20 07:31:49 +04:00
* Copyright ( C ) 2002 David McCullough
* Copyright ( C ) 2003 - 2007 Paul Mundt
2008-02-06 12:38:44 +03:00
*/
2009-08-20 07:31:49 +04:00
2016-02-23 13:54:57 +03:00
# include <linux/bcd.h>
2008-02-06 12:38:44 +03:00
# include <linux/init.h>
2016-02-23 13:54:57 +03:00
# include <linux/io.h>
2008-02-06 12:38:44 +03:00
# include <linux/kernel.h>
2016-02-23 13:54:57 +03:00
# include <linux/module.h>
# include <linux/of.h>
2008-02-06 12:38:44 +03:00
# include <linux/rtc.h>
2016-02-23 13:54:57 +03:00
# include <linux/spi/spi.h>
2008-02-06 12:38:44 +03:00
# define RTC_CMD_READ 0x81 /* Read command */
# define RTC_CMD_WRITE 0x80 /* Write command */
2013-07-04 02:07:46 +04:00
# define RTC_CMD_WRITE_ENABLE 0x00 /* Write enable */
# define RTC_CMD_WRITE_DISABLE 0x80 /* Write disable */
2008-02-06 12:38:44 +03:00
# define RTC_ADDR_RAM0 0x20 /* Address of RAM0 */
# define RTC_ADDR_TCR 0x08 /* Address of trickle charge register */
2016-02-23 13:54:57 +03:00
# define RTC_CLCK_BURST 0x1F /* Address of clock burst */
# define RTC_CLCK_LEN 0x08 /* Size of clock burst */
2013-07-04 02:07:46 +04:00
# define RTC_ADDR_CTRL 0x07 /* Address of control register */
2008-02-06 12:38:44 +03:00
# define RTC_ADDR_YEAR 0x06 /* Address of year register */
# define RTC_ADDR_DAY 0x05 /* Address of day of week register */
# define RTC_ADDR_MON 0x04 /* Address of month register */
# define RTC_ADDR_DATE 0x03 /* Address of day of month register */
# define RTC_ADDR_HOUR 0x02 /* Address of hour register */
# define RTC_ADDR_MIN 0x01 /* Address of minute register */
# define RTC_ADDR_SEC 0x00 /* Address of second register */
2016-02-23 13:54:57 +03:00
static int ds1302_rtc_set_time ( struct device * dev , struct rtc_time * time )
2010-05-25 01:33:47 +04:00
{
2016-02-23 13:54:57 +03:00
struct spi_device * spi = dev_get_drvdata ( dev ) ;
u8 buf [ 1 + RTC_CLCK_LEN ] ;
2018-01-23 13:17:27 +03:00
u8 * bp ;
2016-02-23 13:54:57 +03:00
int status ;
/* Enable writing */
bp = buf ;
* bp + + = RTC_ADDR_CTRL < < 1 | RTC_CMD_WRITE ;
* bp + + = RTC_CMD_WRITE_ENABLE ;
status = spi_write_then_read ( spi , buf , 2 ,
NULL , 0 ) ;
2016-04-10 17:59:23 +03:00
if ( status )
2016-02-23 13:54:57 +03:00
return status ;
/* Write registers starting at the first time/date address. */
bp = buf ;
* bp + + = RTC_CLCK_BURST < < 1 | RTC_CMD_WRITE ;
* bp + + = bin2bcd ( time - > tm_sec ) ;
* bp + + = bin2bcd ( time - > tm_min ) ;
* bp + + = bin2bcd ( time - > tm_hour ) ;
* bp + + = bin2bcd ( time - > tm_mday ) ;
* bp + + = bin2bcd ( time - > tm_mon + 1 ) ;
2016-04-10 17:59:24 +03:00
* bp + + = time - > tm_wday + 1 ;
2016-02-23 13:54:57 +03:00
* bp + + = bin2bcd ( time - > tm_year % 100 ) ;
* bp + + = RTC_CMD_WRITE_DISABLE ;
/* use write-then-read since dma from stack is nonportable */
return spi_write_then_read ( spi , buf , sizeof ( buf ) ,
NULL , 0 ) ;
2010-05-25 01:33:47 +04:00
}
2016-02-23 13:54:57 +03:00
static int ds1302_rtc_get_time ( struct device * dev , struct rtc_time * time )
2010-05-25 01:33:47 +04:00
{
2016-02-23 13:54:57 +03:00
struct spi_device * spi = dev_get_drvdata ( dev ) ;
u8 addr = RTC_CLCK_BURST < < 1 | RTC_CMD_READ ;
u8 buf [ RTC_CLCK_LEN - 1 ] ;
int status ;
/* Use write-then-read to get all the date/time registers
* since dma from stack is nonportable
*/
status = spi_write_then_read ( spi , & addr , sizeof ( addr ) ,
buf , sizeof ( buf ) ) ;
if ( status < 0 )
return status ;
/* Decode the registers */
time - > tm_sec = bcd2bin ( buf [ RTC_ADDR_SEC ] ) ;
time - > tm_min = bcd2bin ( buf [ RTC_ADDR_MIN ] ) ;
time - > tm_hour = bcd2bin ( buf [ RTC_ADDR_HOUR ] ) ;
time - > tm_wday = buf [ RTC_ADDR_DAY ] - 1 ;
time - > tm_mday = bcd2bin ( buf [ RTC_ADDR_DATE ] ) ;
time - > tm_mon = bcd2bin ( buf [ RTC_ADDR_MON ] ) - 1 ;
time - > tm_year = bcd2bin ( buf [ RTC_ADDR_YEAR ] ) + 100 ;
2018-02-19 18:23:56 +03:00
return 0 ;
2010-05-25 01:33:47 +04:00
}
rtc: constify rtc_class_ops structures
Check for rtc_class_ops structures that are only passed to
devm_rtc_device_register, rtc_device_register,
platform_device_register_data, all of which declare the corresponding
parameter as const. Declare rtc_class_ops structures that have these
properties as const.
The semantic patch that makes this change is as follows:
(http://coccinelle.lip6.fr/)
// <smpl>
@r disable optional_qualifier@
identifier i;
position p;
@@
static struct rtc_class_ops i@p = { ... };
@ok@
identifier r.i;
expression e1,e2,e3,e4;
position p;
@@
(
devm_rtc_device_register(e1,e2,&i@p,e3)
|
rtc_device_register(e1,e2,&i@p,e3)
|
platform_device_register_data(e1,e2,e3,&i@p,e4)
)
@bad@
position p != {r.p,ok.p};
identifier r.i;
@@
i@p
@depends on !bad disable optional_qualifier@
identifier r.i;
@@
static
+const
struct rtc_class_ops i = { ... };
// </smpl>
Signed-off-by: Julia Lawall <Julia.Lawall@lip6.fr>
Acked-by: Baruch Siach <baruch@tkos.co.il>
Acked-by: Hans Ulli Kroll <ulli.kroll@googlemail.com>
Acked-by: Linus Walleij <linus.walleij@linaro.org>
Acked-by: Thierry Reding <treding@nvidia.com>
Signed-off-by: Alexandre Belloni <alexandre.belloni@free-electrons.com>
2016-08-31 11:05:25 +03:00
static const struct rtc_class_ops ds1302_rtc_ops = {
2016-02-23 13:54:57 +03:00
. read_time = ds1302_rtc_get_time ,
. set_time = ds1302_rtc_set_time ,
} ;
2008-02-06 12:38:44 +03:00
2016-02-23 13:54:57 +03:00
static int ds1302_probe ( struct spi_device * spi )
2008-02-06 12:38:44 +03:00
{
2016-02-23 13:54:57 +03:00
struct rtc_device * rtc ;
u8 addr ;
u8 buf [ 4 ] ;
2018-01-23 13:17:27 +03:00
u8 * bp ;
2016-02-23 13:54:57 +03:00
int status ;
/* Sanity check board setup data. This may be hooked up
* in 3 wire mode , but we don ' t care . Note that unless
* there ' s an inverter in place , this needs SPI_CS_HIGH !
*/
if ( spi - > bits_per_word & & ( spi - > bits_per_word ! = 8 ) ) {
dev_err ( & spi - > dev , " bad word length \n " ) ;
return - EINVAL ;
} else if ( spi - > max_speed_hz > 2000000 ) {
dev_err ( & spi - > dev , " speed is too high \n " ) ;
return - EINVAL ;
} else if ( spi - > mode & SPI_CPHA ) {
dev_err ( & spi - > dev , " bad mode \n " ) ;
return - EINVAL ;
2008-02-06 12:38:44 +03:00
}
2010-05-25 01:33:47 +04:00
2016-02-23 13:54:57 +03:00
addr = RTC_ADDR_CTRL < < 1 | RTC_CMD_READ ;
status = spi_write_then_read ( spi , & addr , sizeof ( addr ) , buf , 1 ) ;
if ( status < 0 ) {
dev_err ( & spi - > dev , " control register read error %d \n " ,
status ) ;
return status ;
2008-02-06 12:38:44 +03:00
}
2016-02-23 13:54:57 +03:00
if ( ( buf [ 0 ] & ~ RTC_CMD_WRITE_DISABLE ) ! = 0 ) {
status = spi_write_then_read ( spi , & addr , sizeof ( addr ) , buf , 1 ) ;
if ( status < 0 ) {
dev_err ( & spi - > dev , " control register read error %d \n " ,
status ) ;
return status ;
}
if ( ( buf [ 0 ] & ~ RTC_CMD_WRITE_DISABLE ) ! = 0 ) {
dev_err ( & spi - > dev , " junk in control register \n " ) ;
return - ENODEV ;
}
}
if ( buf [ 0 ] = = 0 ) {
bp = buf ;
* bp + + = RTC_ADDR_CTRL < < 1 | RTC_CMD_WRITE ;
* bp + + = RTC_CMD_WRITE_DISABLE ;
status = spi_write_then_read ( spi , buf , 2 , NULL , 0 ) ;
if ( status < 0 ) {
dev_err ( & spi - > dev , " control register write error %d \n " ,
status ) ;
return status ;
}
addr = RTC_ADDR_CTRL < < 1 | RTC_CMD_READ ;
status = spi_write_then_read ( spi , & addr , sizeof ( addr ) , buf , 1 ) ;
if ( status < 0 ) {
dev_err ( & spi - > dev ,
" error %d reading control register \n " ,
status ) ;
return status ;
}
if ( buf [ 0 ] ! = RTC_CMD_WRITE_DISABLE ) {
dev_err ( & spi - > dev , " failed to detect chip \n " ) ;
return - ENODEV ;
}
}
2008-02-06 12:38:44 +03:00
2016-02-23 13:54:57 +03:00
spi_set_drvdata ( spi , spi ) ;
2008-02-06 12:38:44 +03:00
2016-02-23 13:54:57 +03:00
rtc = devm_rtc_device_register ( & spi - > dev , " ds1302 " ,
& ds1302_rtc_ops , THIS_MODULE ) ;
if ( IS_ERR ( rtc ) ) {
status = PTR_ERR ( rtc ) ;
dev_err ( & spi - > dev , " error %d registering rtc \n " , status ) ;
return status ;
}
2013-07-04 02:07:46 +04:00
2008-02-06 12:38:44 +03:00
return 0 ;
}
2016-02-23 13:54:57 +03:00
# ifdef CONFIG_OF
static const struct of_device_id ds1302_dt_ids [ ] = {
{ . compatible = " maxim,ds1302 " , } ,
{ /* sentinel */ }
2008-02-06 12:38:44 +03:00
} ;
2016-02-23 13:54:57 +03:00
MODULE_DEVICE_TABLE ( of , ds1302_dt_ids ) ;
# endif
2008-02-06 12:38:44 +03:00
2021-09-23 22:49:20 +03:00
static const struct spi_device_id ds1302_spi_ids [ ] = {
{ . name = " ds1302 " , } ,
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( spi , ds1302_spi_ids ) ;
2016-02-23 13:54:57 +03:00
static struct spi_driver ds1302_driver = {
. driver . name = " rtc-ds1302 " ,
. driver . of_match_table = of_match_ptr ( ds1302_dt_ids ) ,
. probe = ds1302_probe ,
2021-09-23 22:49:20 +03:00
. id_table = ds1302_spi_ids ,
2008-02-06 12:38:44 +03:00
} ;
2016-02-23 13:54:57 +03:00
module_spi_driver ( ds1302_driver ) ;
2008-02-06 12:38:44 +03:00
MODULE_DESCRIPTION ( " Dallas DS1302 RTC driver " ) ;
MODULE_AUTHOR ( " Paul Mundt, David McCullough " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;