// SPDX-License-Identifier: GPL-2.0-only OR MIT /* * Apple SMC RTC driver * Copyright The Asahi Linux Contributors */ #include #include #include #include #include #include #include #include /* 48-bit RTC */ #define RTC_BYTES 6 #define RTC_BITS (8 * RTC_BYTES) /* 32768 Hz clock */ #define RTC_SEC_SHIFT 15 struct macsmc_rtc { struct device *dev; struct apple_smc *smc; struct rtc_device *rtc_dev; struct nvmem_cell *rtc_offset; }; static int macsmc_rtc_get_time(struct device *dev, struct rtc_time *tm) { struct macsmc_rtc *rtc = dev_get_drvdata(dev); u64 ctr = 0, off = 0; time64_t now; void *p_off; size_t len; int ret; ret = apple_smc_read(rtc->smc, SMC_KEY(CLKM), &ctr, RTC_BYTES); if (ret < 0) return ret; if (ret != RTC_BYTES) return -EIO; p_off = nvmem_cell_read(rtc->rtc_offset, &len); if (IS_ERR(p_off)) return PTR_ERR(p_off); if (len < RTC_BYTES) { kfree(p_off); return -EIO; } memcpy(&off, p_off, RTC_BYTES); kfree(p_off); /* Sign extend from 48 to 64 bits, then arithmetic shift right 15 bits to get seconds */ now = sign_extend64(ctr + off, RTC_BITS - 1) >> RTC_SEC_SHIFT; rtc_time64_to_tm(now, tm); return ret; } static int macsmc_rtc_set_time(struct device *dev, struct rtc_time *tm) { struct macsmc_rtc *rtc = dev_get_drvdata(dev); u64 ctr = 0, off = 0; int ret; ret = apple_smc_read(rtc->smc, SMC_KEY(CLKM), &ctr, RTC_BYTES); if (ret < 0) return ret; if (ret != RTC_BYTES) return -EIO; /* This sets the offset such that the set second begins now */ off = (rtc_tm_to_time64(tm) << RTC_SEC_SHIFT) - ctr; return nvmem_cell_write(rtc->rtc_offset, &off, RTC_BYTES); } static const struct rtc_class_ops macsmc_rtc_ops = { .read_time = macsmc_rtc_get_time, .set_time = macsmc_rtc_set_time, }; static int macsmc_rtc_probe(struct platform_device *pdev) { struct apple_smc *smc = dev_get_drvdata(pdev->dev.parent); struct macsmc_rtc *rtc; /* * MFD will probe this device even without a node in the device tree, * thus bail out early if the SMC on the current machines does not * support RTC and has no node in the device tree. */ if (!pdev->dev.of_node) return -ENODEV; rtc = devm_kzalloc(&pdev->dev, sizeof(*rtc), GFP_KERNEL); if (!rtc) return -ENOMEM; rtc->dev = &pdev->dev; rtc->smc = smc; rtc->rtc_offset = devm_nvmem_cell_get(&pdev->dev, "rtc_offset"); if (IS_ERR(rtc->rtc_offset)) return dev_err_probe(&pdev->dev, PTR_ERR(rtc->rtc_offset), "Failed to get rtc_offset NVMEM cell\n"); rtc->rtc_dev = devm_rtc_allocate_device(&pdev->dev); if (IS_ERR(rtc->rtc_dev)) return PTR_ERR(rtc->rtc_dev); rtc->rtc_dev->ops = &macsmc_rtc_ops; rtc->rtc_dev->range_min = S64_MIN >> (RTC_SEC_SHIFT + (64 - RTC_BITS)); rtc->rtc_dev->range_max = S64_MAX >> (RTC_SEC_SHIFT + (64 - RTC_BITS)); platform_set_drvdata(pdev, rtc); return devm_rtc_register_device(rtc->rtc_dev); } static const struct of_device_id macsmc_rtc_of_table[] = { { .compatible = "apple,smc-rtc", }, {} }; MODULE_DEVICE_TABLE(of, macsmc_rtc_of_table); static struct platform_driver macsmc_rtc_driver = { .driver = { .name = "macsmc-rtc", .of_match_table = macsmc_rtc_of_table, }, .probe = macsmc_rtc_probe, }; module_platform_driver(macsmc_rtc_driver); MODULE_LICENSE("Dual MIT/GPL"); MODULE_DESCRIPTION("Apple SMC RTC driver"); MODULE_AUTHOR("Hector Martin ");