| // SPDX-License-Identifier: GPL-2.0-only | 
 | /* | 
 |  * Real time clocks driver for MStar/SigmaStar ARMv7 SoCs. | 
 |  * Based on "Real Time Clock driver for msb252x." that was contained | 
 |  * in various MStar kernels. | 
 |  * | 
 |  * (C) 2019 Daniel Palmer | 
 |  * (C) 2021 Romain Perier | 
 |  */ | 
 |  | 
 | #include <linux/clk.h> | 
 | #include <linux/delay.h> | 
 | #include <linux/io.h> | 
 | #include <linux/module.h> | 
 | #include <linux/mod_devicetable.h> | 
 | #include <linux/platform_device.h> | 
 | #include <linux/rtc.h> | 
 |  | 
 | /* Registers */ | 
 | #define REG_RTC_CTRL		0x00 | 
 | #define REG_RTC_FREQ_CW_L	0x04 | 
 | #define REG_RTC_FREQ_CW_H	0x08 | 
 | #define REG_RTC_LOAD_VAL_L	0x0C | 
 | #define REG_RTC_LOAD_VAL_H	0x10 | 
 | #define REG_RTC_MATCH_VAL_L	0x14 | 
 | #define REG_RTC_MATCH_VAL_H	0x18 | 
 | #define REG_RTC_STATUS_INT	0x1C | 
 | #define REG_RTC_CNT_VAL_L	0x20 | 
 | #define REG_RTC_CNT_VAL_H	0x24 | 
 |  | 
 | /* Control bits for REG_RTC_CTRL */ | 
 | #define SOFT_RSTZ_BIT		BIT(0) | 
 | #define CNT_EN_BIT		BIT(1) | 
 | #define WRAP_EN_BIT		BIT(2) | 
 | #define LOAD_EN_BIT		BIT(3) | 
 | #define READ_EN_BIT		BIT(4) | 
 | #define INT_MASK_BIT		BIT(5) | 
 | #define INT_FORCE_BIT		BIT(6) | 
 | #define INT_CLEAR_BIT		BIT(7) | 
 |  | 
 | /* Control bits for REG_RTC_STATUS_INT */ | 
 | #define RAW_INT_BIT		BIT(0) | 
 | #define ALM_INT_BIT		BIT(1) | 
 |  | 
 | struct msc313_rtc { | 
 | 	struct rtc_device *rtc_dev; | 
 | 	void __iomem *rtc_base; | 
 | }; | 
 |  | 
 | static int msc313_rtc_read_alarm(struct device *dev, struct rtc_wkalrm *alarm) | 
 | { | 
 | 	struct msc313_rtc *priv = dev_get_drvdata(dev); | 
 | 	unsigned long seconds; | 
 |  | 
 | 	seconds = readw(priv->rtc_base + REG_RTC_MATCH_VAL_L) | 
 | 			| ((unsigned long)readw(priv->rtc_base + REG_RTC_MATCH_VAL_H) << 16); | 
 |  | 
 | 	rtc_time64_to_tm(seconds, &alarm->time); | 
 |  | 
 | 	if (!(readw(priv->rtc_base + REG_RTC_CTRL) & INT_MASK_BIT)) | 
 | 		alarm->enabled = 1; | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int msc313_rtc_alarm_irq_enable(struct device *dev, unsigned int enabled) | 
 | { | 
 | 	struct msc313_rtc *priv = dev_get_drvdata(dev); | 
 | 	u16 reg; | 
 |  | 
 | 	reg = readw(priv->rtc_base + REG_RTC_CTRL); | 
 | 	if (enabled) | 
 | 		reg &= ~INT_MASK_BIT; | 
 | 	else | 
 | 		reg |= INT_MASK_BIT; | 
 | 	writew(reg, priv->rtc_base + REG_RTC_CTRL); | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int msc313_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alarm) | 
 | { | 
 | 	struct msc313_rtc *priv = dev_get_drvdata(dev); | 
 | 	unsigned long seconds; | 
 |  | 
 | 	seconds = rtc_tm_to_time64(&alarm->time); | 
 | 	writew((seconds & 0xFFFF), priv->rtc_base + REG_RTC_MATCH_VAL_L); | 
 | 	writew((seconds >> 16) & 0xFFFF, priv->rtc_base + REG_RTC_MATCH_VAL_H); | 
 |  | 
 | 	msc313_rtc_alarm_irq_enable(dev, alarm->enabled); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static bool msc313_rtc_get_enabled(struct msc313_rtc *priv) | 
 | { | 
 | 	return readw(priv->rtc_base + REG_RTC_CTRL) & CNT_EN_BIT; | 
 | } | 
 |  | 
 | static void msc313_rtc_set_enabled(struct msc313_rtc *priv) | 
 | { | 
 | 	u16 reg; | 
 |  | 
 | 	reg = readw(priv->rtc_base + REG_RTC_CTRL); | 
 | 	reg |= CNT_EN_BIT; | 
 | 	writew(reg, priv->rtc_base + REG_RTC_CTRL); | 
 | } | 
 |  | 
 | static int msc313_rtc_read_time(struct device *dev, struct rtc_time *tm) | 
 | { | 
 | 	struct msc313_rtc *priv = dev_get_drvdata(dev); | 
 | 	u32 seconds; | 
 | 	u16 reg; | 
 |  | 
 | 	if (!msc313_rtc_get_enabled(priv)) | 
 | 		return -EINVAL; | 
 |  | 
 | 	reg = readw(priv->rtc_base + REG_RTC_CTRL); | 
 | 	writew(reg | READ_EN_BIT, priv->rtc_base + REG_RTC_CTRL); | 
 |  | 
 | 	/* Wait for HW latch done */ | 
 | 	while (readw(priv->rtc_base + REG_RTC_CTRL) & READ_EN_BIT) | 
 | 		udelay(1); | 
 |  | 
 | 	seconds = readw(priv->rtc_base + REG_RTC_CNT_VAL_L) | 
 | 			| ((unsigned long)readw(priv->rtc_base + REG_RTC_CNT_VAL_H) << 16); | 
 |  | 
 | 	rtc_time64_to_tm(seconds, tm); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int msc313_rtc_set_time(struct device *dev, struct rtc_time *tm) | 
 | { | 
 | 	struct msc313_rtc *priv = dev_get_drvdata(dev); | 
 | 	unsigned long seconds; | 
 | 	u16 reg; | 
 |  | 
 | 	seconds = rtc_tm_to_time64(tm); | 
 | 	writew(seconds & 0xFFFF, priv->rtc_base + REG_RTC_LOAD_VAL_L); | 
 | 	writew((seconds >> 16) & 0xFFFF, priv->rtc_base + REG_RTC_LOAD_VAL_H); | 
 |  | 
 | 	/* Enable load for loading value into internal RTC counter */ | 
 | 	reg = readw(priv->rtc_base + REG_RTC_CTRL); | 
 | 	writew(reg | LOAD_EN_BIT, priv->rtc_base + REG_RTC_CTRL); | 
 |  | 
 | 	/* Wait for HW latch done */ | 
 | 	while (readw(priv->rtc_base + REG_RTC_CTRL) & LOAD_EN_BIT) | 
 | 		udelay(1); | 
 | 	msc313_rtc_set_enabled(priv); | 
 | 	return 0; | 
 | } | 
 |  | 
 | static const struct rtc_class_ops msc313_rtc_ops = { | 
 | 	.read_time = msc313_rtc_read_time, | 
 | 	.set_time = msc313_rtc_set_time, | 
 | 	.read_alarm = msc313_rtc_read_alarm, | 
 | 	.set_alarm = msc313_rtc_set_alarm, | 
 | 	.alarm_irq_enable = msc313_rtc_alarm_irq_enable, | 
 | }; | 
 |  | 
 | static irqreturn_t msc313_rtc_interrupt(s32 irq, void *dev_id) | 
 | { | 
 | 	struct msc313_rtc *priv = dev_get_drvdata(dev_id); | 
 | 	u16 reg; | 
 |  | 
 | 	reg = readw(priv->rtc_base + REG_RTC_STATUS_INT); | 
 | 	if (!(reg & ALM_INT_BIT)) | 
 | 		return IRQ_NONE; | 
 |  | 
 | 	reg = readw(priv->rtc_base + REG_RTC_CTRL); | 
 | 	reg |= INT_CLEAR_BIT; | 
 | 	reg &= ~INT_FORCE_BIT; | 
 | 	writew(reg, priv->rtc_base + REG_RTC_CTRL); | 
 |  | 
 | 	rtc_update_irq(priv->rtc_dev, 1, RTC_IRQF | RTC_AF); | 
 |  | 
 | 	return IRQ_HANDLED; | 
 | } | 
 |  | 
 | static int msc313_rtc_probe(struct platform_device *pdev) | 
 | { | 
 | 	struct device *dev = &pdev->dev; | 
 | 	struct msc313_rtc *priv; | 
 | 	unsigned long rate; | 
 | 	struct clk *clk; | 
 | 	int ret; | 
 | 	int irq; | 
 |  | 
 | 	priv = devm_kzalloc(&pdev->dev, sizeof(struct msc313_rtc), GFP_KERNEL); | 
 | 	if (!priv) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	priv->rtc_base = devm_platform_ioremap_resource(pdev, 0); | 
 | 	if (IS_ERR(priv->rtc_base)) | 
 | 		return PTR_ERR(priv->rtc_base); | 
 |  | 
 | 	irq = platform_get_irq(pdev, 0); | 
 | 	if (irq < 0) | 
 | 		return -EINVAL; | 
 |  | 
 | 	priv->rtc_dev = devm_rtc_allocate_device(dev); | 
 | 	if (IS_ERR(priv->rtc_dev)) | 
 | 		return PTR_ERR(priv->rtc_dev); | 
 |  | 
 | 	priv->rtc_dev->ops = &msc313_rtc_ops; | 
 | 	priv->rtc_dev->range_max = U32_MAX; | 
 |  | 
 | 	ret = devm_request_irq(dev, irq, msc313_rtc_interrupt, IRQF_SHARED, | 
 | 			       dev_name(&pdev->dev), &pdev->dev); | 
 | 	if (ret) { | 
 | 		dev_err(dev, "Could not request IRQ\n"); | 
 | 		return ret; | 
 | 	} | 
 |  | 
 | 	clk = devm_clk_get_enabled(dev, NULL); | 
 | 	if (IS_ERR(clk)) { | 
 | 		dev_err(dev, "No input reference clock\n"); | 
 | 		return PTR_ERR(clk); | 
 | 	} | 
 |  | 
 | 	rate = clk_get_rate(clk); | 
 | 	writew(rate & 0xFFFF, priv->rtc_base + REG_RTC_FREQ_CW_L); | 
 | 	writew((rate >> 16) & 0xFFFF, priv->rtc_base + REG_RTC_FREQ_CW_H); | 
 |  | 
 | 	platform_set_drvdata(pdev, priv); | 
 |  | 
 | 	return devm_rtc_register_device(priv->rtc_dev); | 
 | } | 
 |  | 
 | static const struct of_device_id msc313_rtc_of_match_table[] = { | 
 | 	{ .compatible = "mstar,msc313-rtc" }, | 
 | 	{ } | 
 | }; | 
 | MODULE_DEVICE_TABLE(of, msc313_rtc_of_match_table); | 
 |  | 
 | static struct platform_driver msc313_rtc_driver = { | 
 | 	.probe = msc313_rtc_probe, | 
 | 	.driver = { | 
 | 		.name = "msc313-rtc", | 
 | 		.of_match_table = msc313_rtc_of_match_table, | 
 | 	}, | 
 | }; | 
 |  | 
 | module_platform_driver(msc313_rtc_driver); | 
 |  | 
 | MODULE_AUTHOR("Daniel Palmer <daniel@thingy.jp>"); | 
 | MODULE_AUTHOR("Romain Perier <romain.perier@gmail.com>"); | 
 | MODULE_DESCRIPTION("MStar RTC Driver"); | 
 | MODULE_LICENSE("GPL v2"); |