18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * TI LP8788 MFD - rtc driver 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright 2012 Texas Instruments 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * Author: Milo(Woogyom) Kim <milo.kim@ti.com> 88c2ecf20Sopenharmony_ci */ 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ci#include <linux/err.h> 118c2ecf20Sopenharmony_ci#include <linux/irqdomain.h> 128c2ecf20Sopenharmony_ci#include <linux/mfd/lp8788.h> 138c2ecf20Sopenharmony_ci#include <linux/module.h> 148c2ecf20Sopenharmony_ci#include <linux/platform_device.h> 158c2ecf20Sopenharmony_ci#include <linux/rtc.h> 168c2ecf20Sopenharmony_ci#include <linux/slab.h> 178c2ecf20Sopenharmony_ci 188c2ecf20Sopenharmony_ci/* register address */ 198c2ecf20Sopenharmony_ci#define LP8788_INTEN_3 0x05 208c2ecf20Sopenharmony_ci#define LP8788_RTC_UNLOCK 0x64 218c2ecf20Sopenharmony_ci#define LP8788_RTC_SEC 0x70 228c2ecf20Sopenharmony_ci#define LP8788_ALM1_SEC 0x77 238c2ecf20Sopenharmony_ci#define LP8788_ALM1_EN 0x7D 248c2ecf20Sopenharmony_ci#define LP8788_ALM2_SEC 0x7E 258c2ecf20Sopenharmony_ci#define LP8788_ALM2_EN 0x84 268c2ecf20Sopenharmony_ci 278c2ecf20Sopenharmony_ci/* mask/shift bits */ 288c2ecf20Sopenharmony_ci#define LP8788_INT_RTC_ALM1_M BIT(1) /* Addr 05h */ 298c2ecf20Sopenharmony_ci#define LP8788_INT_RTC_ALM1_S 1 308c2ecf20Sopenharmony_ci#define LP8788_INT_RTC_ALM2_M BIT(2) /* Addr 05h */ 318c2ecf20Sopenharmony_ci#define LP8788_INT_RTC_ALM2_S 2 328c2ecf20Sopenharmony_ci#define LP8788_ALM_EN_M BIT(7) /* Addr 7Dh or 84h */ 338c2ecf20Sopenharmony_ci#define LP8788_ALM_EN_S 7 348c2ecf20Sopenharmony_ci 358c2ecf20Sopenharmony_ci#define DEFAULT_ALARM_SEL LP8788_ALARM_1 368c2ecf20Sopenharmony_ci#define LP8788_MONTH_OFFSET 1 378c2ecf20Sopenharmony_ci#define LP8788_BASE_YEAR 2000 388c2ecf20Sopenharmony_ci#define MAX_WDAY_BITS 7 398c2ecf20Sopenharmony_ci#define LP8788_WDAY_SET 1 408c2ecf20Sopenharmony_ci#define RTC_UNLOCK 0x1 418c2ecf20Sopenharmony_ci#define RTC_LATCH 0x2 428c2ecf20Sopenharmony_ci#define ALARM_IRQ_FLAG (RTC_IRQF | RTC_AF) 438c2ecf20Sopenharmony_ci 448c2ecf20Sopenharmony_cienum lp8788_time { 458c2ecf20Sopenharmony_ci LPTIME_SEC, 468c2ecf20Sopenharmony_ci LPTIME_MIN, 478c2ecf20Sopenharmony_ci LPTIME_HOUR, 488c2ecf20Sopenharmony_ci LPTIME_MDAY, 498c2ecf20Sopenharmony_ci LPTIME_MON, 508c2ecf20Sopenharmony_ci LPTIME_YEAR, 518c2ecf20Sopenharmony_ci LPTIME_WDAY, 528c2ecf20Sopenharmony_ci LPTIME_MAX, 538c2ecf20Sopenharmony_ci}; 548c2ecf20Sopenharmony_ci 558c2ecf20Sopenharmony_cistruct lp8788_rtc { 568c2ecf20Sopenharmony_ci struct lp8788 *lp; 578c2ecf20Sopenharmony_ci struct rtc_device *rdev; 588c2ecf20Sopenharmony_ci enum lp8788_alarm_sel alarm; 598c2ecf20Sopenharmony_ci int irq; 608c2ecf20Sopenharmony_ci}; 618c2ecf20Sopenharmony_ci 628c2ecf20Sopenharmony_cistatic const u8 addr_alarm_sec[LP8788_ALARM_MAX] = { 638c2ecf20Sopenharmony_ci LP8788_ALM1_SEC, 648c2ecf20Sopenharmony_ci LP8788_ALM2_SEC, 658c2ecf20Sopenharmony_ci}; 668c2ecf20Sopenharmony_ci 678c2ecf20Sopenharmony_cistatic const u8 addr_alarm_en[LP8788_ALARM_MAX] = { 688c2ecf20Sopenharmony_ci LP8788_ALM1_EN, 698c2ecf20Sopenharmony_ci LP8788_ALM2_EN, 708c2ecf20Sopenharmony_ci}; 718c2ecf20Sopenharmony_ci 728c2ecf20Sopenharmony_cistatic const u8 mask_alarm_en[LP8788_ALARM_MAX] = { 738c2ecf20Sopenharmony_ci LP8788_INT_RTC_ALM1_M, 748c2ecf20Sopenharmony_ci LP8788_INT_RTC_ALM2_M, 758c2ecf20Sopenharmony_ci}; 768c2ecf20Sopenharmony_ci 778c2ecf20Sopenharmony_cistatic const u8 shift_alarm_en[LP8788_ALARM_MAX] = { 788c2ecf20Sopenharmony_ci LP8788_INT_RTC_ALM1_S, 798c2ecf20Sopenharmony_ci LP8788_INT_RTC_ALM2_S, 808c2ecf20Sopenharmony_ci}; 818c2ecf20Sopenharmony_ci 828c2ecf20Sopenharmony_cistatic int _to_tm_wday(u8 lp8788_wday) 838c2ecf20Sopenharmony_ci{ 848c2ecf20Sopenharmony_ci int i; 858c2ecf20Sopenharmony_ci 868c2ecf20Sopenharmony_ci if (lp8788_wday == 0) 878c2ecf20Sopenharmony_ci return 0; 888c2ecf20Sopenharmony_ci 898c2ecf20Sopenharmony_ci /* lookup defined weekday from read register value */ 908c2ecf20Sopenharmony_ci for (i = 0; i < MAX_WDAY_BITS; i++) { 918c2ecf20Sopenharmony_ci if ((lp8788_wday >> i) == LP8788_WDAY_SET) 928c2ecf20Sopenharmony_ci break; 938c2ecf20Sopenharmony_ci } 948c2ecf20Sopenharmony_ci 958c2ecf20Sopenharmony_ci return i + 1; 968c2ecf20Sopenharmony_ci} 978c2ecf20Sopenharmony_ci 988c2ecf20Sopenharmony_cistatic inline int _to_lp8788_wday(int tm_wday) 998c2ecf20Sopenharmony_ci{ 1008c2ecf20Sopenharmony_ci return LP8788_WDAY_SET << (tm_wday - 1); 1018c2ecf20Sopenharmony_ci} 1028c2ecf20Sopenharmony_ci 1038c2ecf20Sopenharmony_cistatic void lp8788_rtc_unlock(struct lp8788 *lp) 1048c2ecf20Sopenharmony_ci{ 1058c2ecf20Sopenharmony_ci lp8788_write_byte(lp, LP8788_RTC_UNLOCK, RTC_UNLOCK); 1068c2ecf20Sopenharmony_ci lp8788_write_byte(lp, LP8788_RTC_UNLOCK, RTC_LATCH); 1078c2ecf20Sopenharmony_ci} 1088c2ecf20Sopenharmony_ci 1098c2ecf20Sopenharmony_cistatic int lp8788_rtc_read_time(struct device *dev, struct rtc_time *tm) 1108c2ecf20Sopenharmony_ci{ 1118c2ecf20Sopenharmony_ci struct lp8788_rtc *rtc = dev_get_drvdata(dev); 1128c2ecf20Sopenharmony_ci struct lp8788 *lp = rtc->lp; 1138c2ecf20Sopenharmony_ci u8 data[LPTIME_MAX]; 1148c2ecf20Sopenharmony_ci int ret; 1158c2ecf20Sopenharmony_ci 1168c2ecf20Sopenharmony_ci lp8788_rtc_unlock(lp); 1178c2ecf20Sopenharmony_ci 1188c2ecf20Sopenharmony_ci ret = lp8788_read_multi_bytes(lp, LP8788_RTC_SEC, data, LPTIME_MAX); 1198c2ecf20Sopenharmony_ci if (ret) 1208c2ecf20Sopenharmony_ci return ret; 1218c2ecf20Sopenharmony_ci 1228c2ecf20Sopenharmony_ci tm->tm_sec = data[LPTIME_SEC]; 1238c2ecf20Sopenharmony_ci tm->tm_min = data[LPTIME_MIN]; 1248c2ecf20Sopenharmony_ci tm->tm_hour = data[LPTIME_HOUR]; 1258c2ecf20Sopenharmony_ci tm->tm_mday = data[LPTIME_MDAY]; 1268c2ecf20Sopenharmony_ci tm->tm_mon = data[LPTIME_MON] - LP8788_MONTH_OFFSET; 1278c2ecf20Sopenharmony_ci tm->tm_year = data[LPTIME_YEAR] + LP8788_BASE_YEAR - 1900; 1288c2ecf20Sopenharmony_ci tm->tm_wday = _to_tm_wday(data[LPTIME_WDAY]); 1298c2ecf20Sopenharmony_ci 1308c2ecf20Sopenharmony_ci return 0; 1318c2ecf20Sopenharmony_ci} 1328c2ecf20Sopenharmony_ci 1338c2ecf20Sopenharmony_cistatic int lp8788_rtc_set_time(struct device *dev, struct rtc_time *tm) 1348c2ecf20Sopenharmony_ci{ 1358c2ecf20Sopenharmony_ci struct lp8788_rtc *rtc = dev_get_drvdata(dev); 1368c2ecf20Sopenharmony_ci struct lp8788 *lp = rtc->lp; 1378c2ecf20Sopenharmony_ci u8 data[LPTIME_MAX - 1]; 1388c2ecf20Sopenharmony_ci int ret, i, year; 1398c2ecf20Sopenharmony_ci 1408c2ecf20Sopenharmony_ci year = tm->tm_year + 1900 - LP8788_BASE_YEAR; 1418c2ecf20Sopenharmony_ci if (year < 0) { 1428c2ecf20Sopenharmony_ci dev_err(lp->dev, "invalid year: %d\n", year); 1438c2ecf20Sopenharmony_ci return -EINVAL; 1448c2ecf20Sopenharmony_ci } 1458c2ecf20Sopenharmony_ci 1468c2ecf20Sopenharmony_ci /* because rtc weekday is a readonly register, do not update */ 1478c2ecf20Sopenharmony_ci data[LPTIME_SEC] = tm->tm_sec; 1488c2ecf20Sopenharmony_ci data[LPTIME_MIN] = tm->tm_min; 1498c2ecf20Sopenharmony_ci data[LPTIME_HOUR] = tm->tm_hour; 1508c2ecf20Sopenharmony_ci data[LPTIME_MDAY] = tm->tm_mday; 1518c2ecf20Sopenharmony_ci data[LPTIME_MON] = tm->tm_mon + LP8788_MONTH_OFFSET; 1528c2ecf20Sopenharmony_ci data[LPTIME_YEAR] = year; 1538c2ecf20Sopenharmony_ci 1548c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(data); i++) { 1558c2ecf20Sopenharmony_ci ret = lp8788_write_byte(lp, LP8788_RTC_SEC + i, data[i]); 1568c2ecf20Sopenharmony_ci if (ret) 1578c2ecf20Sopenharmony_ci return ret; 1588c2ecf20Sopenharmony_ci } 1598c2ecf20Sopenharmony_ci 1608c2ecf20Sopenharmony_ci return 0; 1618c2ecf20Sopenharmony_ci} 1628c2ecf20Sopenharmony_ci 1638c2ecf20Sopenharmony_cistatic int lp8788_read_alarm(struct device *dev, struct rtc_wkalrm *alarm) 1648c2ecf20Sopenharmony_ci{ 1658c2ecf20Sopenharmony_ci struct lp8788_rtc *rtc = dev_get_drvdata(dev); 1668c2ecf20Sopenharmony_ci struct lp8788 *lp = rtc->lp; 1678c2ecf20Sopenharmony_ci struct rtc_time *tm = &alarm->time; 1688c2ecf20Sopenharmony_ci u8 addr, data[LPTIME_MAX]; 1698c2ecf20Sopenharmony_ci int ret; 1708c2ecf20Sopenharmony_ci 1718c2ecf20Sopenharmony_ci addr = addr_alarm_sec[rtc->alarm]; 1728c2ecf20Sopenharmony_ci ret = lp8788_read_multi_bytes(lp, addr, data, LPTIME_MAX); 1738c2ecf20Sopenharmony_ci if (ret) 1748c2ecf20Sopenharmony_ci return ret; 1758c2ecf20Sopenharmony_ci 1768c2ecf20Sopenharmony_ci tm->tm_sec = data[LPTIME_SEC]; 1778c2ecf20Sopenharmony_ci tm->tm_min = data[LPTIME_MIN]; 1788c2ecf20Sopenharmony_ci tm->tm_hour = data[LPTIME_HOUR]; 1798c2ecf20Sopenharmony_ci tm->tm_mday = data[LPTIME_MDAY]; 1808c2ecf20Sopenharmony_ci tm->tm_mon = data[LPTIME_MON] - LP8788_MONTH_OFFSET; 1818c2ecf20Sopenharmony_ci tm->tm_year = data[LPTIME_YEAR] + LP8788_BASE_YEAR - 1900; 1828c2ecf20Sopenharmony_ci tm->tm_wday = _to_tm_wday(data[LPTIME_WDAY]); 1838c2ecf20Sopenharmony_ci alarm->enabled = data[LPTIME_WDAY] & LP8788_ALM_EN_M; 1848c2ecf20Sopenharmony_ci 1858c2ecf20Sopenharmony_ci return 0; 1868c2ecf20Sopenharmony_ci} 1878c2ecf20Sopenharmony_ci 1888c2ecf20Sopenharmony_cistatic int lp8788_set_alarm(struct device *dev, struct rtc_wkalrm *alarm) 1898c2ecf20Sopenharmony_ci{ 1908c2ecf20Sopenharmony_ci struct lp8788_rtc *rtc = dev_get_drvdata(dev); 1918c2ecf20Sopenharmony_ci struct lp8788 *lp = rtc->lp; 1928c2ecf20Sopenharmony_ci struct rtc_time *tm = &alarm->time; 1938c2ecf20Sopenharmony_ci u8 addr, data[LPTIME_MAX]; 1948c2ecf20Sopenharmony_ci int ret, i, year; 1958c2ecf20Sopenharmony_ci 1968c2ecf20Sopenharmony_ci year = tm->tm_year + 1900 - LP8788_BASE_YEAR; 1978c2ecf20Sopenharmony_ci if (year < 0) { 1988c2ecf20Sopenharmony_ci dev_err(lp->dev, "invalid year: %d\n", year); 1998c2ecf20Sopenharmony_ci return -EINVAL; 2008c2ecf20Sopenharmony_ci } 2018c2ecf20Sopenharmony_ci 2028c2ecf20Sopenharmony_ci data[LPTIME_SEC] = tm->tm_sec; 2038c2ecf20Sopenharmony_ci data[LPTIME_MIN] = tm->tm_min; 2048c2ecf20Sopenharmony_ci data[LPTIME_HOUR] = tm->tm_hour; 2058c2ecf20Sopenharmony_ci data[LPTIME_MDAY] = tm->tm_mday; 2068c2ecf20Sopenharmony_ci data[LPTIME_MON] = tm->tm_mon + LP8788_MONTH_OFFSET; 2078c2ecf20Sopenharmony_ci data[LPTIME_YEAR] = year; 2088c2ecf20Sopenharmony_ci data[LPTIME_WDAY] = _to_lp8788_wday(tm->tm_wday); 2098c2ecf20Sopenharmony_ci 2108c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(data); i++) { 2118c2ecf20Sopenharmony_ci addr = addr_alarm_sec[rtc->alarm] + i; 2128c2ecf20Sopenharmony_ci ret = lp8788_write_byte(lp, addr, data[i]); 2138c2ecf20Sopenharmony_ci if (ret) 2148c2ecf20Sopenharmony_ci return ret; 2158c2ecf20Sopenharmony_ci } 2168c2ecf20Sopenharmony_ci 2178c2ecf20Sopenharmony_ci alarm->enabled = 1; 2188c2ecf20Sopenharmony_ci addr = addr_alarm_en[rtc->alarm]; 2198c2ecf20Sopenharmony_ci 2208c2ecf20Sopenharmony_ci return lp8788_update_bits(lp, addr, LP8788_ALM_EN_M, 2218c2ecf20Sopenharmony_ci alarm->enabled << LP8788_ALM_EN_S); 2228c2ecf20Sopenharmony_ci} 2238c2ecf20Sopenharmony_ci 2248c2ecf20Sopenharmony_cistatic int lp8788_alarm_irq_enable(struct device *dev, unsigned int enable) 2258c2ecf20Sopenharmony_ci{ 2268c2ecf20Sopenharmony_ci struct lp8788_rtc *rtc = dev_get_drvdata(dev); 2278c2ecf20Sopenharmony_ci struct lp8788 *lp = rtc->lp; 2288c2ecf20Sopenharmony_ci u8 mask, shift; 2298c2ecf20Sopenharmony_ci 2308c2ecf20Sopenharmony_ci if (!rtc->irq) 2318c2ecf20Sopenharmony_ci return -EIO; 2328c2ecf20Sopenharmony_ci 2338c2ecf20Sopenharmony_ci mask = mask_alarm_en[rtc->alarm]; 2348c2ecf20Sopenharmony_ci shift = shift_alarm_en[rtc->alarm]; 2358c2ecf20Sopenharmony_ci 2368c2ecf20Sopenharmony_ci return lp8788_update_bits(lp, LP8788_INTEN_3, mask, enable << shift); 2378c2ecf20Sopenharmony_ci} 2388c2ecf20Sopenharmony_ci 2398c2ecf20Sopenharmony_cistatic const struct rtc_class_ops lp8788_rtc_ops = { 2408c2ecf20Sopenharmony_ci .read_time = lp8788_rtc_read_time, 2418c2ecf20Sopenharmony_ci .set_time = lp8788_rtc_set_time, 2428c2ecf20Sopenharmony_ci .read_alarm = lp8788_read_alarm, 2438c2ecf20Sopenharmony_ci .set_alarm = lp8788_set_alarm, 2448c2ecf20Sopenharmony_ci .alarm_irq_enable = lp8788_alarm_irq_enable, 2458c2ecf20Sopenharmony_ci}; 2468c2ecf20Sopenharmony_ci 2478c2ecf20Sopenharmony_cistatic irqreturn_t lp8788_alarm_irq_handler(int irq, void *ptr) 2488c2ecf20Sopenharmony_ci{ 2498c2ecf20Sopenharmony_ci struct lp8788_rtc *rtc = ptr; 2508c2ecf20Sopenharmony_ci 2518c2ecf20Sopenharmony_ci rtc_update_irq(rtc->rdev, 1, ALARM_IRQ_FLAG); 2528c2ecf20Sopenharmony_ci return IRQ_HANDLED; 2538c2ecf20Sopenharmony_ci} 2548c2ecf20Sopenharmony_ci 2558c2ecf20Sopenharmony_cistatic int lp8788_alarm_irq_register(struct platform_device *pdev, 2568c2ecf20Sopenharmony_ci struct lp8788_rtc *rtc) 2578c2ecf20Sopenharmony_ci{ 2588c2ecf20Sopenharmony_ci struct resource *r; 2598c2ecf20Sopenharmony_ci struct lp8788 *lp = rtc->lp; 2608c2ecf20Sopenharmony_ci struct irq_domain *irqdm = lp->irqdm; 2618c2ecf20Sopenharmony_ci int irq; 2628c2ecf20Sopenharmony_ci 2638c2ecf20Sopenharmony_ci rtc->irq = 0; 2648c2ecf20Sopenharmony_ci 2658c2ecf20Sopenharmony_ci /* even the alarm IRQ number is not specified, rtc time should work */ 2668c2ecf20Sopenharmony_ci r = platform_get_resource_byname(pdev, IORESOURCE_IRQ, LP8788_ALM_IRQ); 2678c2ecf20Sopenharmony_ci if (!r) 2688c2ecf20Sopenharmony_ci return 0; 2698c2ecf20Sopenharmony_ci 2708c2ecf20Sopenharmony_ci if (rtc->alarm == LP8788_ALARM_1) 2718c2ecf20Sopenharmony_ci irq = r->start; 2728c2ecf20Sopenharmony_ci else 2738c2ecf20Sopenharmony_ci irq = r->end; 2748c2ecf20Sopenharmony_ci 2758c2ecf20Sopenharmony_ci rtc->irq = irq_create_mapping(irqdm, irq); 2768c2ecf20Sopenharmony_ci 2778c2ecf20Sopenharmony_ci return devm_request_threaded_irq(&pdev->dev, rtc->irq, NULL, 2788c2ecf20Sopenharmony_ci lp8788_alarm_irq_handler, 2798c2ecf20Sopenharmony_ci 0, LP8788_ALM_IRQ, rtc); 2808c2ecf20Sopenharmony_ci} 2818c2ecf20Sopenharmony_ci 2828c2ecf20Sopenharmony_cistatic int lp8788_rtc_probe(struct platform_device *pdev) 2838c2ecf20Sopenharmony_ci{ 2848c2ecf20Sopenharmony_ci struct lp8788 *lp = dev_get_drvdata(pdev->dev.parent); 2858c2ecf20Sopenharmony_ci struct lp8788_rtc *rtc; 2868c2ecf20Sopenharmony_ci struct device *dev = &pdev->dev; 2878c2ecf20Sopenharmony_ci 2888c2ecf20Sopenharmony_ci rtc = devm_kzalloc(dev, sizeof(struct lp8788_rtc), GFP_KERNEL); 2898c2ecf20Sopenharmony_ci if (!rtc) 2908c2ecf20Sopenharmony_ci return -ENOMEM; 2918c2ecf20Sopenharmony_ci 2928c2ecf20Sopenharmony_ci rtc->lp = lp; 2938c2ecf20Sopenharmony_ci rtc->alarm = lp->pdata ? lp->pdata->alarm_sel : DEFAULT_ALARM_SEL; 2948c2ecf20Sopenharmony_ci platform_set_drvdata(pdev, rtc); 2958c2ecf20Sopenharmony_ci 2968c2ecf20Sopenharmony_ci device_init_wakeup(dev, 1); 2978c2ecf20Sopenharmony_ci 2988c2ecf20Sopenharmony_ci rtc->rdev = devm_rtc_device_register(dev, "lp8788_rtc", 2998c2ecf20Sopenharmony_ci &lp8788_rtc_ops, THIS_MODULE); 3008c2ecf20Sopenharmony_ci if (IS_ERR(rtc->rdev)) { 3018c2ecf20Sopenharmony_ci dev_err(dev, "can not register rtc device\n"); 3028c2ecf20Sopenharmony_ci return PTR_ERR(rtc->rdev); 3038c2ecf20Sopenharmony_ci } 3048c2ecf20Sopenharmony_ci 3058c2ecf20Sopenharmony_ci if (lp8788_alarm_irq_register(pdev, rtc)) 3068c2ecf20Sopenharmony_ci dev_warn(lp->dev, "no rtc irq handler\n"); 3078c2ecf20Sopenharmony_ci 3088c2ecf20Sopenharmony_ci return 0; 3098c2ecf20Sopenharmony_ci} 3108c2ecf20Sopenharmony_ci 3118c2ecf20Sopenharmony_cistatic struct platform_driver lp8788_rtc_driver = { 3128c2ecf20Sopenharmony_ci .probe = lp8788_rtc_probe, 3138c2ecf20Sopenharmony_ci .driver = { 3148c2ecf20Sopenharmony_ci .name = LP8788_DEV_RTC, 3158c2ecf20Sopenharmony_ci }, 3168c2ecf20Sopenharmony_ci}; 3178c2ecf20Sopenharmony_cimodule_platform_driver(lp8788_rtc_driver); 3188c2ecf20Sopenharmony_ci 3198c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Texas Instruments LP8788 RTC Driver"); 3208c2ecf20Sopenharmony_ciMODULE_AUTHOR("Milo Kim"); 3218c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 3228c2ecf20Sopenharmony_ciMODULE_ALIAS("platform:lp8788-rtc"); 323