18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Motorola CPCAP PMIC RTC driver 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Based on cpcap-regulator.c from Motorola Linux kernel tree 68c2ecf20Sopenharmony_ci * Copyright (C) 2009 Motorola, Inc. 78c2ecf20Sopenharmony_ci * 88c2ecf20Sopenharmony_ci * Rewritten for mainline kernel 98c2ecf20Sopenharmony_ci * - use DT 108c2ecf20Sopenharmony_ci * - use regmap 118c2ecf20Sopenharmony_ci * - use standard interrupt framework 128c2ecf20Sopenharmony_ci * - use managed device resources 138c2ecf20Sopenharmony_ci * - remove custom "secure clock daemon" helpers 148c2ecf20Sopenharmony_ci * 158c2ecf20Sopenharmony_ci * Copyright (C) 2017 Sebastian Reichel <sre@kernel.org> 168c2ecf20Sopenharmony_ci */ 178c2ecf20Sopenharmony_ci#include <linux/kernel.h> 188c2ecf20Sopenharmony_ci#include <linux/module.h> 198c2ecf20Sopenharmony_ci#include <linux/mod_devicetable.h> 208c2ecf20Sopenharmony_ci#include <linux/init.h> 218c2ecf20Sopenharmony_ci#include <linux/device.h> 228c2ecf20Sopenharmony_ci#include <linux/platform_device.h> 238c2ecf20Sopenharmony_ci#include <linux/rtc.h> 248c2ecf20Sopenharmony_ci#include <linux/err.h> 258c2ecf20Sopenharmony_ci#include <linux/regmap.h> 268c2ecf20Sopenharmony_ci#include <linux/mfd/motorola-cpcap.h> 278c2ecf20Sopenharmony_ci#include <linux/slab.h> 288c2ecf20Sopenharmony_ci#include <linux/sched.h> 298c2ecf20Sopenharmony_ci 308c2ecf20Sopenharmony_ci#define SECS_PER_DAY 86400 318c2ecf20Sopenharmony_ci#define DAY_MASK 0x7FFF 328c2ecf20Sopenharmony_ci#define TOD1_MASK 0x00FF 338c2ecf20Sopenharmony_ci#define TOD2_MASK 0x01FF 348c2ecf20Sopenharmony_ci 358c2ecf20Sopenharmony_cistruct cpcap_time { 368c2ecf20Sopenharmony_ci int day; 378c2ecf20Sopenharmony_ci int tod1; 388c2ecf20Sopenharmony_ci int tod2; 398c2ecf20Sopenharmony_ci}; 408c2ecf20Sopenharmony_ci 418c2ecf20Sopenharmony_cistruct cpcap_rtc { 428c2ecf20Sopenharmony_ci struct regmap *regmap; 438c2ecf20Sopenharmony_ci struct rtc_device *rtc_dev; 448c2ecf20Sopenharmony_ci u16 vendor; 458c2ecf20Sopenharmony_ci int alarm_irq; 468c2ecf20Sopenharmony_ci bool alarm_enabled; 478c2ecf20Sopenharmony_ci int update_irq; 488c2ecf20Sopenharmony_ci bool update_enabled; 498c2ecf20Sopenharmony_ci}; 508c2ecf20Sopenharmony_ci 518c2ecf20Sopenharmony_cistatic void cpcap2rtc_time(struct rtc_time *rtc, struct cpcap_time *cpcap) 528c2ecf20Sopenharmony_ci{ 538c2ecf20Sopenharmony_ci unsigned long int tod; 548c2ecf20Sopenharmony_ci unsigned long int time; 558c2ecf20Sopenharmony_ci 568c2ecf20Sopenharmony_ci tod = (cpcap->tod1 & TOD1_MASK) | ((cpcap->tod2 & TOD2_MASK) << 8); 578c2ecf20Sopenharmony_ci time = tod + ((cpcap->day & DAY_MASK) * SECS_PER_DAY); 588c2ecf20Sopenharmony_ci 598c2ecf20Sopenharmony_ci rtc_time64_to_tm(time, rtc); 608c2ecf20Sopenharmony_ci} 618c2ecf20Sopenharmony_ci 628c2ecf20Sopenharmony_cistatic void rtc2cpcap_time(struct cpcap_time *cpcap, struct rtc_time *rtc) 638c2ecf20Sopenharmony_ci{ 648c2ecf20Sopenharmony_ci unsigned long time; 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_ci time = rtc_tm_to_time64(rtc); 678c2ecf20Sopenharmony_ci 688c2ecf20Sopenharmony_ci cpcap->day = time / SECS_PER_DAY; 698c2ecf20Sopenharmony_ci time %= SECS_PER_DAY; 708c2ecf20Sopenharmony_ci cpcap->tod2 = (time >> 8) & TOD2_MASK; 718c2ecf20Sopenharmony_ci cpcap->tod1 = time & TOD1_MASK; 728c2ecf20Sopenharmony_ci} 738c2ecf20Sopenharmony_ci 748c2ecf20Sopenharmony_cistatic int cpcap_rtc_alarm_irq_enable(struct device *dev, unsigned int enabled) 758c2ecf20Sopenharmony_ci{ 768c2ecf20Sopenharmony_ci struct cpcap_rtc *rtc = dev_get_drvdata(dev); 778c2ecf20Sopenharmony_ci 788c2ecf20Sopenharmony_ci if (rtc->alarm_enabled == enabled) 798c2ecf20Sopenharmony_ci return 0; 808c2ecf20Sopenharmony_ci 818c2ecf20Sopenharmony_ci if (enabled) 828c2ecf20Sopenharmony_ci enable_irq(rtc->alarm_irq); 838c2ecf20Sopenharmony_ci else 848c2ecf20Sopenharmony_ci disable_irq(rtc->alarm_irq); 858c2ecf20Sopenharmony_ci 868c2ecf20Sopenharmony_ci rtc->alarm_enabled = !!enabled; 878c2ecf20Sopenharmony_ci 888c2ecf20Sopenharmony_ci return 0; 898c2ecf20Sopenharmony_ci} 908c2ecf20Sopenharmony_ci 918c2ecf20Sopenharmony_cistatic int cpcap_rtc_read_time(struct device *dev, struct rtc_time *tm) 928c2ecf20Sopenharmony_ci{ 938c2ecf20Sopenharmony_ci struct cpcap_rtc *rtc; 948c2ecf20Sopenharmony_ci struct cpcap_time cpcap_tm; 958c2ecf20Sopenharmony_ci int temp_tod2; 968c2ecf20Sopenharmony_ci int ret; 978c2ecf20Sopenharmony_ci 988c2ecf20Sopenharmony_ci rtc = dev_get_drvdata(dev); 998c2ecf20Sopenharmony_ci 1008c2ecf20Sopenharmony_ci ret = regmap_read(rtc->regmap, CPCAP_REG_TOD2, &temp_tod2); 1018c2ecf20Sopenharmony_ci ret |= regmap_read(rtc->regmap, CPCAP_REG_DAY, &cpcap_tm.day); 1028c2ecf20Sopenharmony_ci ret |= regmap_read(rtc->regmap, CPCAP_REG_TOD1, &cpcap_tm.tod1); 1038c2ecf20Sopenharmony_ci ret |= regmap_read(rtc->regmap, CPCAP_REG_TOD2, &cpcap_tm.tod2); 1048c2ecf20Sopenharmony_ci 1058c2ecf20Sopenharmony_ci if (temp_tod2 > cpcap_tm.tod2) 1068c2ecf20Sopenharmony_ci ret |= regmap_read(rtc->regmap, CPCAP_REG_DAY, &cpcap_tm.day); 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci if (ret) { 1098c2ecf20Sopenharmony_ci dev_err(dev, "Failed to read time\n"); 1108c2ecf20Sopenharmony_ci return -EIO; 1118c2ecf20Sopenharmony_ci } 1128c2ecf20Sopenharmony_ci 1138c2ecf20Sopenharmony_ci cpcap2rtc_time(tm, &cpcap_tm); 1148c2ecf20Sopenharmony_ci 1158c2ecf20Sopenharmony_ci return 0; 1168c2ecf20Sopenharmony_ci} 1178c2ecf20Sopenharmony_ci 1188c2ecf20Sopenharmony_cistatic int cpcap_rtc_set_time(struct device *dev, struct rtc_time *tm) 1198c2ecf20Sopenharmony_ci{ 1208c2ecf20Sopenharmony_ci struct cpcap_rtc *rtc; 1218c2ecf20Sopenharmony_ci struct cpcap_time cpcap_tm; 1228c2ecf20Sopenharmony_ci int ret = 0; 1238c2ecf20Sopenharmony_ci 1248c2ecf20Sopenharmony_ci rtc = dev_get_drvdata(dev); 1258c2ecf20Sopenharmony_ci 1268c2ecf20Sopenharmony_ci rtc2cpcap_time(&cpcap_tm, tm); 1278c2ecf20Sopenharmony_ci 1288c2ecf20Sopenharmony_ci if (rtc->alarm_enabled) 1298c2ecf20Sopenharmony_ci disable_irq(rtc->alarm_irq); 1308c2ecf20Sopenharmony_ci if (rtc->update_enabled) 1318c2ecf20Sopenharmony_ci disable_irq(rtc->update_irq); 1328c2ecf20Sopenharmony_ci 1338c2ecf20Sopenharmony_ci if (rtc->vendor == CPCAP_VENDOR_ST) { 1348c2ecf20Sopenharmony_ci /* The TOD1 and TOD2 registers MUST be written in this order 1358c2ecf20Sopenharmony_ci * for the change to properly set. 1368c2ecf20Sopenharmony_ci */ 1378c2ecf20Sopenharmony_ci ret |= regmap_update_bits(rtc->regmap, CPCAP_REG_TOD1, 1388c2ecf20Sopenharmony_ci TOD1_MASK, cpcap_tm.tod1); 1398c2ecf20Sopenharmony_ci ret |= regmap_update_bits(rtc->regmap, CPCAP_REG_TOD2, 1408c2ecf20Sopenharmony_ci TOD2_MASK, cpcap_tm.tod2); 1418c2ecf20Sopenharmony_ci ret |= regmap_update_bits(rtc->regmap, CPCAP_REG_DAY, 1428c2ecf20Sopenharmony_ci DAY_MASK, cpcap_tm.day); 1438c2ecf20Sopenharmony_ci } else { 1448c2ecf20Sopenharmony_ci /* Clearing the upper lower 8 bits of the TOD guarantees that 1458c2ecf20Sopenharmony_ci * the upper half of TOD (TOD2) will not increment for 0xFF RTC 1468c2ecf20Sopenharmony_ci * ticks (255 seconds). During this time we can safely write 1478c2ecf20Sopenharmony_ci * to DAY, TOD2, then TOD1 (in that order) and expect RTC to be 1488c2ecf20Sopenharmony_ci * synchronized to the exact time requested upon the final write 1498c2ecf20Sopenharmony_ci * to TOD1. 1508c2ecf20Sopenharmony_ci */ 1518c2ecf20Sopenharmony_ci ret |= regmap_update_bits(rtc->regmap, CPCAP_REG_TOD1, 1528c2ecf20Sopenharmony_ci TOD1_MASK, 0); 1538c2ecf20Sopenharmony_ci ret |= regmap_update_bits(rtc->regmap, CPCAP_REG_DAY, 1548c2ecf20Sopenharmony_ci DAY_MASK, cpcap_tm.day); 1558c2ecf20Sopenharmony_ci ret |= regmap_update_bits(rtc->regmap, CPCAP_REG_TOD2, 1568c2ecf20Sopenharmony_ci TOD2_MASK, cpcap_tm.tod2); 1578c2ecf20Sopenharmony_ci ret |= regmap_update_bits(rtc->regmap, CPCAP_REG_TOD1, 1588c2ecf20Sopenharmony_ci TOD1_MASK, cpcap_tm.tod1); 1598c2ecf20Sopenharmony_ci } 1608c2ecf20Sopenharmony_ci 1618c2ecf20Sopenharmony_ci if (rtc->update_enabled) 1628c2ecf20Sopenharmony_ci enable_irq(rtc->update_irq); 1638c2ecf20Sopenharmony_ci if (rtc->alarm_enabled) 1648c2ecf20Sopenharmony_ci enable_irq(rtc->alarm_irq); 1658c2ecf20Sopenharmony_ci 1668c2ecf20Sopenharmony_ci return ret; 1678c2ecf20Sopenharmony_ci} 1688c2ecf20Sopenharmony_ci 1698c2ecf20Sopenharmony_cistatic int cpcap_rtc_read_alarm(struct device *dev, struct rtc_wkalrm *alrm) 1708c2ecf20Sopenharmony_ci{ 1718c2ecf20Sopenharmony_ci struct cpcap_rtc *rtc; 1728c2ecf20Sopenharmony_ci struct cpcap_time cpcap_tm; 1738c2ecf20Sopenharmony_ci int ret; 1748c2ecf20Sopenharmony_ci 1758c2ecf20Sopenharmony_ci rtc = dev_get_drvdata(dev); 1768c2ecf20Sopenharmony_ci 1778c2ecf20Sopenharmony_ci alrm->enabled = rtc->alarm_enabled; 1788c2ecf20Sopenharmony_ci 1798c2ecf20Sopenharmony_ci ret = regmap_read(rtc->regmap, CPCAP_REG_DAYA, &cpcap_tm.day); 1808c2ecf20Sopenharmony_ci ret |= regmap_read(rtc->regmap, CPCAP_REG_TODA2, &cpcap_tm.tod2); 1818c2ecf20Sopenharmony_ci ret |= regmap_read(rtc->regmap, CPCAP_REG_TODA1, &cpcap_tm.tod1); 1828c2ecf20Sopenharmony_ci 1838c2ecf20Sopenharmony_ci if (ret) { 1848c2ecf20Sopenharmony_ci dev_err(dev, "Failed to read time\n"); 1858c2ecf20Sopenharmony_ci return -EIO; 1868c2ecf20Sopenharmony_ci } 1878c2ecf20Sopenharmony_ci 1888c2ecf20Sopenharmony_ci cpcap2rtc_time(&alrm->time, &cpcap_tm); 1898c2ecf20Sopenharmony_ci return rtc_valid_tm(&alrm->time); 1908c2ecf20Sopenharmony_ci} 1918c2ecf20Sopenharmony_ci 1928c2ecf20Sopenharmony_cistatic int cpcap_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alrm) 1938c2ecf20Sopenharmony_ci{ 1948c2ecf20Sopenharmony_ci struct cpcap_rtc *rtc; 1958c2ecf20Sopenharmony_ci struct cpcap_time cpcap_tm; 1968c2ecf20Sopenharmony_ci int ret; 1978c2ecf20Sopenharmony_ci 1988c2ecf20Sopenharmony_ci rtc = dev_get_drvdata(dev); 1998c2ecf20Sopenharmony_ci 2008c2ecf20Sopenharmony_ci rtc2cpcap_time(&cpcap_tm, &alrm->time); 2018c2ecf20Sopenharmony_ci 2028c2ecf20Sopenharmony_ci if (rtc->alarm_enabled) 2038c2ecf20Sopenharmony_ci disable_irq(rtc->alarm_irq); 2048c2ecf20Sopenharmony_ci 2058c2ecf20Sopenharmony_ci ret = regmap_update_bits(rtc->regmap, CPCAP_REG_DAYA, DAY_MASK, 2068c2ecf20Sopenharmony_ci cpcap_tm.day); 2078c2ecf20Sopenharmony_ci ret |= regmap_update_bits(rtc->regmap, CPCAP_REG_TODA2, TOD2_MASK, 2088c2ecf20Sopenharmony_ci cpcap_tm.tod2); 2098c2ecf20Sopenharmony_ci ret |= regmap_update_bits(rtc->regmap, CPCAP_REG_TODA1, TOD1_MASK, 2108c2ecf20Sopenharmony_ci cpcap_tm.tod1); 2118c2ecf20Sopenharmony_ci 2128c2ecf20Sopenharmony_ci if (!ret) { 2138c2ecf20Sopenharmony_ci enable_irq(rtc->alarm_irq); 2148c2ecf20Sopenharmony_ci rtc->alarm_enabled = true; 2158c2ecf20Sopenharmony_ci } 2168c2ecf20Sopenharmony_ci 2178c2ecf20Sopenharmony_ci return ret; 2188c2ecf20Sopenharmony_ci} 2198c2ecf20Sopenharmony_ci 2208c2ecf20Sopenharmony_cistatic const struct rtc_class_ops cpcap_rtc_ops = { 2218c2ecf20Sopenharmony_ci .read_time = cpcap_rtc_read_time, 2228c2ecf20Sopenharmony_ci .set_time = cpcap_rtc_set_time, 2238c2ecf20Sopenharmony_ci .read_alarm = cpcap_rtc_read_alarm, 2248c2ecf20Sopenharmony_ci .set_alarm = cpcap_rtc_set_alarm, 2258c2ecf20Sopenharmony_ci .alarm_irq_enable = cpcap_rtc_alarm_irq_enable, 2268c2ecf20Sopenharmony_ci}; 2278c2ecf20Sopenharmony_ci 2288c2ecf20Sopenharmony_cistatic irqreturn_t cpcap_rtc_alarm_irq(int irq, void *data) 2298c2ecf20Sopenharmony_ci{ 2308c2ecf20Sopenharmony_ci struct cpcap_rtc *rtc = data; 2318c2ecf20Sopenharmony_ci 2328c2ecf20Sopenharmony_ci rtc_update_irq(rtc->rtc_dev, 1, RTC_AF | RTC_IRQF); 2338c2ecf20Sopenharmony_ci return IRQ_HANDLED; 2348c2ecf20Sopenharmony_ci} 2358c2ecf20Sopenharmony_ci 2368c2ecf20Sopenharmony_cistatic irqreturn_t cpcap_rtc_update_irq(int irq, void *data) 2378c2ecf20Sopenharmony_ci{ 2388c2ecf20Sopenharmony_ci struct cpcap_rtc *rtc = data; 2398c2ecf20Sopenharmony_ci 2408c2ecf20Sopenharmony_ci rtc_update_irq(rtc->rtc_dev, 1, RTC_UF | RTC_IRQF); 2418c2ecf20Sopenharmony_ci return IRQ_HANDLED; 2428c2ecf20Sopenharmony_ci} 2438c2ecf20Sopenharmony_ci 2448c2ecf20Sopenharmony_cistatic int cpcap_rtc_probe(struct platform_device *pdev) 2458c2ecf20Sopenharmony_ci{ 2468c2ecf20Sopenharmony_ci struct device *dev = &pdev->dev; 2478c2ecf20Sopenharmony_ci struct cpcap_rtc *rtc; 2488c2ecf20Sopenharmony_ci int err; 2498c2ecf20Sopenharmony_ci 2508c2ecf20Sopenharmony_ci rtc = devm_kzalloc(dev, sizeof(*rtc), GFP_KERNEL); 2518c2ecf20Sopenharmony_ci if (!rtc) 2528c2ecf20Sopenharmony_ci return -ENOMEM; 2538c2ecf20Sopenharmony_ci 2548c2ecf20Sopenharmony_ci rtc->regmap = dev_get_regmap(dev->parent, NULL); 2558c2ecf20Sopenharmony_ci if (!rtc->regmap) 2568c2ecf20Sopenharmony_ci return -ENODEV; 2578c2ecf20Sopenharmony_ci 2588c2ecf20Sopenharmony_ci platform_set_drvdata(pdev, rtc); 2598c2ecf20Sopenharmony_ci rtc->rtc_dev = devm_rtc_allocate_device(dev); 2608c2ecf20Sopenharmony_ci if (IS_ERR(rtc->rtc_dev)) 2618c2ecf20Sopenharmony_ci return PTR_ERR(rtc->rtc_dev); 2628c2ecf20Sopenharmony_ci 2638c2ecf20Sopenharmony_ci rtc->rtc_dev->ops = &cpcap_rtc_ops; 2648c2ecf20Sopenharmony_ci rtc->rtc_dev->range_max = (timeu64_t) (DAY_MASK + 1) * SECS_PER_DAY - 1; 2658c2ecf20Sopenharmony_ci 2668c2ecf20Sopenharmony_ci err = cpcap_get_vendor(dev, rtc->regmap, &rtc->vendor); 2678c2ecf20Sopenharmony_ci if (err) 2688c2ecf20Sopenharmony_ci return err; 2698c2ecf20Sopenharmony_ci 2708c2ecf20Sopenharmony_ci rtc->alarm_irq = platform_get_irq(pdev, 0); 2718c2ecf20Sopenharmony_ci err = devm_request_threaded_irq(dev, rtc->alarm_irq, NULL, 2728c2ecf20Sopenharmony_ci cpcap_rtc_alarm_irq, IRQF_TRIGGER_NONE, 2738c2ecf20Sopenharmony_ci "rtc_alarm", rtc); 2748c2ecf20Sopenharmony_ci if (err) { 2758c2ecf20Sopenharmony_ci dev_err(dev, "Could not request alarm irq: %d\n", err); 2768c2ecf20Sopenharmony_ci return err; 2778c2ecf20Sopenharmony_ci } 2788c2ecf20Sopenharmony_ci disable_irq(rtc->alarm_irq); 2798c2ecf20Sopenharmony_ci 2808c2ecf20Sopenharmony_ci /* Stock Android uses the 1 Hz interrupt for "secure clock daemon", 2818c2ecf20Sopenharmony_ci * which is not supported by the mainline kernel. The mainline kernel 2828c2ecf20Sopenharmony_ci * does not use the irq at the moment, but we explicitly request and 2838c2ecf20Sopenharmony_ci * disable it, so that its masked and does not wake up the processor 2848c2ecf20Sopenharmony_ci * every second. 2858c2ecf20Sopenharmony_ci */ 2868c2ecf20Sopenharmony_ci rtc->update_irq = platform_get_irq(pdev, 1); 2878c2ecf20Sopenharmony_ci err = devm_request_threaded_irq(dev, rtc->update_irq, NULL, 2888c2ecf20Sopenharmony_ci cpcap_rtc_update_irq, IRQF_TRIGGER_NONE, 2898c2ecf20Sopenharmony_ci "rtc_1hz", rtc); 2908c2ecf20Sopenharmony_ci if (err) { 2918c2ecf20Sopenharmony_ci dev_err(dev, "Could not request update irq: %d\n", err); 2928c2ecf20Sopenharmony_ci return err; 2938c2ecf20Sopenharmony_ci } 2948c2ecf20Sopenharmony_ci disable_irq(rtc->update_irq); 2958c2ecf20Sopenharmony_ci 2968c2ecf20Sopenharmony_ci err = device_init_wakeup(dev, 1); 2978c2ecf20Sopenharmony_ci if (err) { 2988c2ecf20Sopenharmony_ci dev_err(dev, "wakeup initialization failed (%d)\n", err); 2998c2ecf20Sopenharmony_ci /* ignore error and continue without wakeup support */ 3008c2ecf20Sopenharmony_ci } 3018c2ecf20Sopenharmony_ci 3028c2ecf20Sopenharmony_ci return rtc_register_device(rtc->rtc_dev); 3038c2ecf20Sopenharmony_ci} 3048c2ecf20Sopenharmony_ci 3058c2ecf20Sopenharmony_cistatic const struct of_device_id cpcap_rtc_of_match[] = { 3068c2ecf20Sopenharmony_ci { .compatible = "motorola,cpcap-rtc", }, 3078c2ecf20Sopenharmony_ci {}, 3088c2ecf20Sopenharmony_ci}; 3098c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, cpcap_rtc_of_match); 3108c2ecf20Sopenharmony_ci 3118c2ecf20Sopenharmony_cistatic struct platform_driver cpcap_rtc_driver = { 3128c2ecf20Sopenharmony_ci .probe = cpcap_rtc_probe, 3138c2ecf20Sopenharmony_ci .driver = { 3148c2ecf20Sopenharmony_ci .name = "cpcap-rtc", 3158c2ecf20Sopenharmony_ci .of_match_table = cpcap_rtc_of_match, 3168c2ecf20Sopenharmony_ci }, 3178c2ecf20Sopenharmony_ci}; 3188c2ecf20Sopenharmony_ci 3198c2ecf20Sopenharmony_cimodule_platform_driver(cpcap_rtc_driver); 3208c2ecf20Sopenharmony_ci 3218c2ecf20Sopenharmony_ciMODULE_ALIAS("platform:cpcap-rtc"); 3228c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("CPCAP RTC driver"); 3238c2ecf20Sopenharmony_ciMODULE_AUTHOR("Sebastian Reichel <sre@kernel.org>"); 3248c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 325