| // SPDX-License-Identifier: GPL-2.0-or-later | 
 | /* | 
 |  * The Netronix embedded controller is a microcontroller found in some | 
 |  * e-book readers designed by the original design manufacturer Netronix, Inc. | 
 |  * It contains RTC, battery monitoring, system power management, and PWM | 
 |  * functionality. | 
 |  * | 
 |  * This driver implements access to the RTC time and date. | 
 |  * | 
 |  * Copyright 2020 Jonathan Neuschäfer <j.neuschaefer@gmx.net> | 
 |  */ | 
 |  | 
 | #include <linux/mfd/ntxec.h> | 
 | #include <linux/module.h> | 
 | #include <linux/platform_device.h> | 
 | #include <linux/regmap.h> | 
 | #include <linux/rtc.h> | 
 | #include <linux/types.h> | 
 |  | 
 | struct ntxec_rtc { | 
 | 	struct device *dev; | 
 | 	struct ntxec *ec; | 
 | }; | 
 |  | 
 | #define NTXEC_REG_WRITE_YEAR	0x10 | 
 | #define NTXEC_REG_WRITE_MONTH	0x11 | 
 | #define NTXEC_REG_WRITE_DAY	0x12 | 
 | #define NTXEC_REG_WRITE_HOUR	0x13 | 
 | #define NTXEC_REG_WRITE_MINUTE	0x14 | 
 | #define NTXEC_REG_WRITE_SECOND	0x15 | 
 |  | 
 | #define NTXEC_REG_READ_YEAR_MONTH	0x20 | 
 | #define NTXEC_REG_READ_MDAY_HOUR	0x21 | 
 | #define NTXEC_REG_READ_MINUTE_SECOND	0x23 | 
 |  | 
 | static int ntxec_read_time(struct device *dev, struct rtc_time *tm) | 
 | { | 
 | 	struct ntxec_rtc *rtc = dev_get_drvdata(dev); | 
 | 	unsigned int value; | 
 | 	int res; | 
 |  | 
 | retry: | 
 | 	res = regmap_read(rtc->ec->regmap, NTXEC_REG_READ_MINUTE_SECOND, &value); | 
 | 	if (res < 0) | 
 | 		return res; | 
 |  | 
 | 	tm->tm_min = value >> 8; | 
 | 	tm->tm_sec = value & 0xff; | 
 |  | 
 | 	res = regmap_read(rtc->ec->regmap, NTXEC_REG_READ_MDAY_HOUR, &value); | 
 | 	if (res < 0) | 
 | 		return res; | 
 |  | 
 | 	tm->tm_mday = value >> 8; | 
 | 	tm->tm_hour = value & 0xff; | 
 |  | 
 | 	res = regmap_read(rtc->ec->regmap, NTXEC_REG_READ_YEAR_MONTH, &value); | 
 | 	if (res < 0) | 
 | 		return res; | 
 |  | 
 | 	tm->tm_year = (value >> 8) + 100; | 
 | 	tm->tm_mon = (value & 0xff) - 1; | 
 |  | 
 | 	/* | 
 | 	 * Read the minutes/seconds field again. If it changed since the first | 
 | 	 * read, we can't assume that the values read so far are consistent, | 
 | 	 * and should start from the beginning. | 
 | 	 */ | 
 | 	res = regmap_read(rtc->ec->regmap, NTXEC_REG_READ_MINUTE_SECOND, &value); | 
 | 	if (res < 0) | 
 | 		return res; | 
 |  | 
 | 	if (tm->tm_min != value >> 8 || tm->tm_sec != (value & 0xff)) | 
 | 		goto retry; | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int ntxec_set_time(struct device *dev, struct rtc_time *tm) | 
 | { | 
 | 	struct ntxec_rtc *rtc = dev_get_drvdata(dev); | 
 |  | 
 | 	/* | 
 | 	 * To avoid time overflows while we're writing the full date/time, | 
 | 	 * set the seconds field to zero before doing anything else. For the | 
 | 	 * next 59 seconds (plus however long it takes until the RTC's next | 
 | 	 * update of the second field), the seconds field will not overflow | 
 | 	 * into the other fields. | 
 | 	 */ | 
 | 	struct reg_sequence regs[] = { | 
 | 		{ NTXEC_REG_WRITE_SECOND, ntxec_reg8(0) }, | 
 | 		{ NTXEC_REG_WRITE_YEAR, ntxec_reg8(tm->tm_year - 100) }, | 
 | 		{ NTXEC_REG_WRITE_MONTH, ntxec_reg8(tm->tm_mon + 1) }, | 
 | 		{ NTXEC_REG_WRITE_DAY, ntxec_reg8(tm->tm_mday) }, | 
 | 		{ NTXEC_REG_WRITE_HOUR, ntxec_reg8(tm->tm_hour) }, | 
 | 		{ NTXEC_REG_WRITE_MINUTE, ntxec_reg8(tm->tm_min) }, | 
 | 		{ NTXEC_REG_WRITE_SECOND, ntxec_reg8(tm->tm_sec) }, | 
 | 	}; | 
 |  | 
 | 	return regmap_multi_reg_write(rtc->ec->regmap, regs, ARRAY_SIZE(regs)); | 
 | } | 
 |  | 
 | static const struct rtc_class_ops ntxec_rtc_ops = { | 
 | 	.read_time = ntxec_read_time, | 
 | 	.set_time = ntxec_set_time, | 
 | }; | 
 |  | 
 | static int ntxec_rtc_probe(struct platform_device *pdev) | 
 | { | 
 | 	struct rtc_device *dev; | 
 | 	struct ntxec_rtc *rtc; | 
 |  | 
 | 	pdev->dev.of_node = pdev->dev.parent->of_node; | 
 |  | 
 | 	rtc = devm_kzalloc(&pdev->dev, sizeof(*rtc), GFP_KERNEL); | 
 | 	if (!rtc) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	rtc->dev = &pdev->dev; | 
 | 	rtc->ec = dev_get_drvdata(pdev->dev.parent); | 
 | 	platform_set_drvdata(pdev, rtc); | 
 |  | 
 | 	dev = devm_rtc_allocate_device(&pdev->dev); | 
 | 	if (IS_ERR(dev)) | 
 | 		return PTR_ERR(dev); | 
 |  | 
 | 	dev->ops = &ntxec_rtc_ops; | 
 | 	dev->range_min = RTC_TIMESTAMP_BEGIN_2000; | 
 | 	dev->range_max = 9025257599LL; /* 2255-12-31 23:59:59 */ | 
 |  | 
 | 	return devm_rtc_register_device(dev); | 
 | } | 
 |  | 
 | static struct platform_driver ntxec_rtc_driver = { | 
 | 	.driver = { | 
 | 		.name = "ntxec-rtc", | 
 | 	}, | 
 | 	.probe = ntxec_rtc_probe, | 
 | }; | 
 | module_platform_driver(ntxec_rtc_driver); | 
 |  | 
 | MODULE_AUTHOR("Jonathan Neuschäfer <j.neuschaefer@gmx.net>"); | 
 | MODULE_DESCRIPTION("RTC driver for Netronix EC"); | 
 | MODULE_LICENSE("GPL"); | 
 | MODULE_ALIAS("platform:ntxec-rtc"); |