1*49a51df4SHector Martin // SPDX-License-Identifier: GPL-2.0-only OR MIT
2*49a51df4SHector Martin /*
3*49a51df4SHector Martin * Apple SMC RTC driver
4*49a51df4SHector Martin * Copyright The Asahi Linux Contributors
5*49a51df4SHector Martin */
6*49a51df4SHector Martin
7*49a51df4SHector Martin #include <linux/bitops.h>
8*49a51df4SHector Martin #include <linux/mfd/macsmc.h>
9*49a51df4SHector Martin #include <linux/module.h>
10*49a51df4SHector Martin #include <linux/nvmem-consumer.h>
11*49a51df4SHector Martin #include <linux/of.h>
12*49a51df4SHector Martin #include <linux/platform_device.h>
13*49a51df4SHector Martin #include <linux/rtc.h>
14*49a51df4SHector Martin #include <linux/slab.h>
15*49a51df4SHector Martin
16*49a51df4SHector Martin /* 48-bit RTC */
17*49a51df4SHector Martin #define RTC_BYTES 6
18*49a51df4SHector Martin #define RTC_BITS (8 * RTC_BYTES)
19*49a51df4SHector Martin
20*49a51df4SHector Martin /* 32768 Hz clock */
21*49a51df4SHector Martin #define RTC_SEC_SHIFT 15
22*49a51df4SHector Martin
23*49a51df4SHector Martin struct macsmc_rtc {
24*49a51df4SHector Martin struct device *dev;
25*49a51df4SHector Martin struct apple_smc *smc;
26*49a51df4SHector Martin struct rtc_device *rtc_dev;
27*49a51df4SHector Martin struct nvmem_cell *rtc_offset;
28*49a51df4SHector Martin };
29*49a51df4SHector Martin
macsmc_rtc_get_time(struct device * dev,struct rtc_time * tm)30*49a51df4SHector Martin static int macsmc_rtc_get_time(struct device *dev, struct rtc_time *tm)
31*49a51df4SHector Martin {
32*49a51df4SHector Martin struct macsmc_rtc *rtc = dev_get_drvdata(dev);
33*49a51df4SHector Martin u64 ctr = 0, off = 0;
34*49a51df4SHector Martin time64_t now;
35*49a51df4SHector Martin void *p_off;
36*49a51df4SHector Martin size_t len;
37*49a51df4SHector Martin int ret;
38*49a51df4SHector Martin
39*49a51df4SHector Martin ret = apple_smc_read(rtc->smc, SMC_KEY(CLKM), &ctr, RTC_BYTES);
40*49a51df4SHector Martin if (ret < 0)
41*49a51df4SHector Martin return ret;
42*49a51df4SHector Martin if (ret != RTC_BYTES)
43*49a51df4SHector Martin return -EIO;
44*49a51df4SHector Martin
45*49a51df4SHector Martin p_off = nvmem_cell_read(rtc->rtc_offset, &len);
46*49a51df4SHector Martin if (IS_ERR(p_off))
47*49a51df4SHector Martin return PTR_ERR(p_off);
48*49a51df4SHector Martin if (len < RTC_BYTES) {
49*49a51df4SHector Martin kfree(p_off);
50*49a51df4SHector Martin return -EIO;
51*49a51df4SHector Martin }
52*49a51df4SHector Martin
53*49a51df4SHector Martin memcpy(&off, p_off, RTC_BYTES);
54*49a51df4SHector Martin kfree(p_off);
55*49a51df4SHector Martin
56*49a51df4SHector Martin /* Sign extend from 48 to 64 bits, then arithmetic shift right 15 bits to get seconds */
57*49a51df4SHector Martin now = sign_extend64(ctr + off, RTC_BITS - 1) >> RTC_SEC_SHIFT;
58*49a51df4SHector Martin rtc_time64_to_tm(now, tm);
59*49a51df4SHector Martin
60*49a51df4SHector Martin return ret;
61*49a51df4SHector Martin }
62*49a51df4SHector Martin
macsmc_rtc_set_time(struct device * dev,struct rtc_time * tm)63*49a51df4SHector Martin static int macsmc_rtc_set_time(struct device *dev, struct rtc_time *tm)
64*49a51df4SHector Martin {
65*49a51df4SHector Martin struct macsmc_rtc *rtc = dev_get_drvdata(dev);
66*49a51df4SHector Martin u64 ctr = 0, off = 0;
67*49a51df4SHector Martin int ret;
68*49a51df4SHector Martin
69*49a51df4SHector Martin ret = apple_smc_read(rtc->smc, SMC_KEY(CLKM), &ctr, RTC_BYTES);
70*49a51df4SHector Martin if (ret < 0)
71*49a51df4SHector Martin return ret;
72*49a51df4SHector Martin if (ret != RTC_BYTES)
73*49a51df4SHector Martin return -EIO;
74*49a51df4SHector Martin
75*49a51df4SHector Martin /* This sets the offset such that the set second begins now */
76*49a51df4SHector Martin off = (rtc_tm_to_time64(tm) << RTC_SEC_SHIFT) - ctr;
77*49a51df4SHector Martin return nvmem_cell_write(rtc->rtc_offset, &off, RTC_BYTES);
78*49a51df4SHector Martin }
79*49a51df4SHector Martin
80*49a51df4SHector Martin static const struct rtc_class_ops macsmc_rtc_ops = {
81*49a51df4SHector Martin .read_time = macsmc_rtc_get_time,
82*49a51df4SHector Martin .set_time = macsmc_rtc_set_time,
83*49a51df4SHector Martin };
84*49a51df4SHector Martin
macsmc_rtc_probe(struct platform_device * pdev)85*49a51df4SHector Martin static int macsmc_rtc_probe(struct platform_device *pdev)
86*49a51df4SHector Martin {
87*49a51df4SHector Martin struct apple_smc *smc = dev_get_drvdata(pdev->dev.parent);
88*49a51df4SHector Martin struct macsmc_rtc *rtc;
89*49a51df4SHector Martin
90*49a51df4SHector Martin /*
91*49a51df4SHector Martin * MFD will probe this device even without a node in the device tree,
92*49a51df4SHector Martin * thus bail out early if the SMC on the current machines does not
93*49a51df4SHector Martin * support RTC and has no node in the device tree.
94*49a51df4SHector Martin */
95*49a51df4SHector Martin if (!pdev->dev.of_node)
96*49a51df4SHector Martin return -ENODEV;
97*49a51df4SHector Martin
98*49a51df4SHector Martin rtc = devm_kzalloc(&pdev->dev, sizeof(*rtc), GFP_KERNEL);
99*49a51df4SHector Martin if (!rtc)
100*49a51df4SHector Martin return -ENOMEM;
101*49a51df4SHector Martin
102*49a51df4SHector Martin rtc->dev = &pdev->dev;
103*49a51df4SHector Martin rtc->smc = smc;
104*49a51df4SHector Martin
105*49a51df4SHector Martin rtc->rtc_offset = devm_nvmem_cell_get(&pdev->dev, "rtc_offset");
106*49a51df4SHector Martin if (IS_ERR(rtc->rtc_offset))
107*49a51df4SHector Martin return dev_err_probe(&pdev->dev, PTR_ERR(rtc->rtc_offset),
108*49a51df4SHector Martin "Failed to get rtc_offset NVMEM cell\n");
109*49a51df4SHector Martin
110*49a51df4SHector Martin rtc->rtc_dev = devm_rtc_allocate_device(&pdev->dev);
111*49a51df4SHector Martin if (IS_ERR(rtc->rtc_dev))
112*49a51df4SHector Martin return PTR_ERR(rtc->rtc_dev);
113*49a51df4SHector Martin
114*49a51df4SHector Martin rtc->rtc_dev->ops = &macsmc_rtc_ops;
115*49a51df4SHector Martin rtc->rtc_dev->range_min = S64_MIN >> (RTC_SEC_SHIFT + (64 - RTC_BITS));
116*49a51df4SHector Martin rtc->rtc_dev->range_max = S64_MAX >> (RTC_SEC_SHIFT + (64 - RTC_BITS));
117*49a51df4SHector Martin
118*49a51df4SHector Martin platform_set_drvdata(pdev, rtc);
119*49a51df4SHector Martin
120*49a51df4SHector Martin return devm_rtc_register_device(rtc->rtc_dev);
121*49a51df4SHector Martin }
122*49a51df4SHector Martin
123*49a51df4SHector Martin static const struct of_device_id macsmc_rtc_of_table[] = {
124*49a51df4SHector Martin { .compatible = "apple,smc-rtc", },
125*49a51df4SHector Martin {}
126*49a51df4SHector Martin };
127*49a51df4SHector Martin MODULE_DEVICE_TABLE(of, macsmc_rtc_of_table);
128*49a51df4SHector Martin
129*49a51df4SHector Martin static struct platform_driver macsmc_rtc_driver = {
130*49a51df4SHector Martin .driver = {
131*49a51df4SHector Martin .name = "macsmc-rtc",
132*49a51df4SHector Martin .of_match_table = macsmc_rtc_of_table,
133*49a51df4SHector Martin },
134*49a51df4SHector Martin .probe = macsmc_rtc_probe,
135*49a51df4SHector Martin };
136*49a51df4SHector Martin module_platform_driver(macsmc_rtc_driver);
137*49a51df4SHector Martin
138*49a51df4SHector Martin MODULE_LICENSE("Dual MIT/GPL");
139*49a51df4SHector Martin MODULE_DESCRIPTION("Apple SMC RTC driver");
140*49a51df4SHector Martin MODULE_AUTHOR("Hector Martin <marcan@marcan.st>");
141