1d2912cb1SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only 218cb6368SRenaud Cerrato /* 3cee2cc21SAkinobu Mita * An I2C and SPI driver for the NXP PCF2127/29 RTC 418cb6368SRenaud Cerrato * Copyright 2013 Til-Technologies 518cb6368SRenaud Cerrato * 618cb6368SRenaud Cerrato * Author: Renaud Cerrato <r.cerrato@til-technologies.fr> 718cb6368SRenaud Cerrato * 80e735eaaSBruno Thomsen * Watchdog and tamper functions 90e735eaaSBruno Thomsen * Author: Bruno Thomsen <bruno.thomsen@gmail.com> 100e735eaaSBruno Thomsen * 1118cb6368SRenaud Cerrato * based on the other drivers in this same directory. 1218cb6368SRenaud Cerrato * 13836e9ea3SFabio Estevam * Datasheet: https://www.nxp.com/docs/en/data-sheet/PCF2127.pdf 1418cb6368SRenaud Cerrato */ 1518cb6368SRenaud Cerrato 1618cb6368SRenaud Cerrato #include <linux/i2c.h> 179408ec1aSAkinobu Mita #include <linux/spi/spi.h> 1818cb6368SRenaud Cerrato #include <linux/bcd.h> 1918cb6368SRenaud Cerrato #include <linux/rtc.h> 2018cb6368SRenaud Cerrato #include <linux/slab.h> 2118cb6368SRenaud Cerrato #include <linux/module.h> 2218cb6368SRenaud Cerrato #include <linux/of.h> 238a914bacSLiam Beguin #include <linux/of_irq.h> 24fd28ceb4SHugo Villeneuve #include <linux/of_device.h> 25907b3262SAkinobu Mita #include <linux/regmap.h> 260e735eaaSBruno Thomsen #include <linux/watchdog.h> 2718cb6368SRenaud Cerrato 28bbfe3a7aSBruno Thomsen /* Control register 1 */ 29bbfe3a7aSBruno Thomsen #define PCF2127_REG_CTRL1 0x00 30b9ac079aSPhilipp Rosenberger #define PCF2127_BIT_CTRL1_POR_OVRD BIT(3) 3103623b4bSBruno Thomsen #define PCF2127_BIT_CTRL1_TSF1 BIT(4) 32bbfe3a7aSBruno Thomsen /* Control register 2 */ 33bbfe3a7aSBruno Thomsen #define PCF2127_REG_CTRL2 0x01 348a914bacSLiam Beguin #define PCF2127_BIT_CTRL2_AIE BIT(1) 3503623b4bSBruno Thomsen #define PCF2127_BIT_CTRL2_TSIE BIT(2) 368a914bacSLiam Beguin #define PCF2127_BIT_CTRL2_AF BIT(4) 3703623b4bSBruno Thomsen #define PCF2127_BIT_CTRL2_TSF2 BIT(5) 3827006416SAlexandre Belloni #define PCF2127_BIT_CTRL2_WDTF BIT(6) 39bbfe3a7aSBruno Thomsen /* Control register 3 */ 40bbfe3a7aSBruno Thomsen #define PCF2127_REG_CTRL3 0x02 4103623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BLIE BIT(0) 4203623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BIE BIT(1) 43bbfe3a7aSBruno Thomsen #define PCF2127_BIT_CTRL3_BLF BIT(2) 4403623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BF BIT(3) 4503623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BTSE BIT(4) 46bbfe3a7aSBruno Thomsen /* Time and date registers */ 476211aceeSHugo Villeneuve #define PCF2127_REG_TIME_BASE 0x03 48bbfe3a7aSBruno Thomsen #define PCF2127_BIT_SC_OSF BIT(7) 498a914bacSLiam Beguin /* Alarm registers */ 507c6f0db4SHugo Villeneuve #define PCF2127_REG_ALARM_BASE 0x0A 5127006416SAlexandre Belloni #define PCF2127_BIT_ALARM_AE BIT(7) 5215f57b3eSPhilipp Rosenberger /* CLKOUT control register */ 5315f57b3eSPhilipp Rosenberger #define PCF2127_REG_CLKOUT 0x0f 5415f57b3eSPhilipp Rosenberger #define PCF2127_BIT_CLKOUT_OTPR BIT(5) 550e735eaaSBruno Thomsen /* Watchdog registers */ 560e735eaaSBruno Thomsen #define PCF2127_REG_WD_CTL 0x10 570e735eaaSBruno Thomsen #define PCF2127_BIT_WD_CTL_TF0 BIT(0) 580e735eaaSBruno Thomsen #define PCF2127_BIT_WD_CTL_TF1 BIT(1) 590e735eaaSBruno Thomsen #define PCF2127_BIT_WD_CTL_CD0 BIT(6) 600e735eaaSBruno Thomsen #define PCF2127_BIT_WD_CTL_CD1 BIT(7) 610e735eaaSBruno Thomsen #define PCF2127_REG_WD_VAL 0x11 6203623b4bSBruno Thomsen /* Tamper timestamp registers */ 6303623b4bSBruno Thomsen #define PCF2127_REG_TS_CTRL 0x12 6403623b4bSBruno Thomsen #define PCF2127_BIT_TS_CTRL_TSOFF BIT(6) 6503623b4bSBruno Thomsen #define PCF2127_BIT_TS_CTRL_TSM BIT(7) 66bbfe3a7aSBruno Thomsen /* 67bbfe3a7aSBruno Thomsen * RAM registers 68bbfe3a7aSBruno Thomsen * PCF2127 has 512 bytes general-purpose static RAM (SRAM) that is 69bbfe3a7aSBruno Thomsen * battery backed and can survive a power outage. 70bbfe3a7aSBruno Thomsen * PCF2129 doesn't have this feature. 71bbfe3a7aSBruno Thomsen */ 72bbfe3a7aSBruno Thomsen #define PCF2127_REG_RAM_ADDR_MSB 0x1A 73bbfe3a7aSBruno Thomsen #define PCF2127_REG_RAM_WRT_CMD 0x1C 74bbfe3a7aSBruno Thomsen #define PCF2127_REG_RAM_RD_CMD 0x1D 75f97cfddcSUwe Kleine-König 760e735eaaSBruno Thomsen /* Watchdog timer value constants */ 770e735eaaSBruno Thomsen #define PCF2127_WD_VAL_STOP 0 780e735eaaSBruno Thomsen #define PCF2127_WD_VAL_MIN 2 790e735eaaSBruno Thomsen #define PCF2127_WD_VAL_MAX 255 800e735eaaSBruno Thomsen #define PCF2127_WD_VAL_DEFAULT 60 81653ebd75SAndrea Scian 822f861984SMian Yousaf Kaukab /* Mask for currently enabled interrupts */ 832f861984SMian Yousaf Kaukab #define PCF2127_CTRL1_IRQ_MASK (PCF2127_BIT_CTRL1_TSF1) 842f861984SMian Yousaf Kaukab #define PCF2127_CTRL2_IRQ_MASK ( \ 852f861984SMian Yousaf Kaukab PCF2127_BIT_CTRL2_AF | \ 862f861984SMian Yousaf Kaukab PCF2127_BIT_CTRL2_WDTF | \ 872f861984SMian Yousaf Kaukab PCF2127_BIT_CTRL2_TSF2) 882f861984SMian Yousaf Kaukab 89fd28ceb4SHugo Villeneuve enum pcf21xx_type { 90fd28ceb4SHugo Villeneuve PCF2127, 91fd28ceb4SHugo Villeneuve PCF2129, 92fd28ceb4SHugo Villeneuve PCF21XX_LAST_ID 93fd28ceb4SHugo Villeneuve }; 94fd28ceb4SHugo Villeneuve 95fd28ceb4SHugo Villeneuve struct pcf21xx_config { 96fd28ceb4SHugo Villeneuve int type; /* IC variant */ 97fd28ceb4SHugo Villeneuve int max_register; 98fd28ceb4SHugo Villeneuve unsigned int has_nvmem:1; 99fd28ceb4SHugo Villeneuve unsigned int has_bit_wd_ctl_cd0:1; 1006211aceeSHugo Villeneuve u8 reg_time_base; /* Time/date base register. */ 1017c6f0db4SHugo Villeneuve u8 regs_alarm_base; /* Alarm function base registers. */ 1026b57ec29SHugo Villeneuve u8 reg_wd_ctl; /* Watchdog control register. */ 1036b57ec29SHugo Villeneuve u8 reg_wd_val; /* Watchdog value register. */ 104*fc16599eSHugo Villeneuve u8 reg_clkout; /* Clkout register. */ 105fd28ceb4SHugo Villeneuve }; 106fd28ceb4SHugo Villeneuve 10718cb6368SRenaud Cerrato struct pcf2127 { 10818cb6368SRenaud Cerrato struct rtc_device *rtc; 1090e735eaaSBruno Thomsen struct watchdog_device wdd; 110907b3262SAkinobu Mita struct regmap *regmap; 111fd28ceb4SHugo Villeneuve const struct pcf21xx_config *cfg; 1122f861984SMian Yousaf Kaukab time64_t ts; 1132f861984SMian Yousaf Kaukab bool ts_valid; 1142f861984SMian Yousaf Kaukab bool irq_enabled; 11518cb6368SRenaud Cerrato }; 11618cb6368SRenaud Cerrato 11718cb6368SRenaud Cerrato /* 11818cb6368SRenaud Cerrato * In the routines that deal directly with the pcf2127 hardware, we use 11918cb6368SRenaud Cerrato * rtc_time -- month 0-11, hour 0-23, yr = calendar year-epoch. 12018cb6368SRenaud Cerrato */ 121907b3262SAkinobu Mita static int pcf2127_rtc_read_time(struct device *dev, struct rtc_time *tm) 12218cb6368SRenaud Cerrato { 123907b3262SAkinobu Mita struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 12431f077c3SHugo Villeneuve unsigned char buf[7]; 125907b3262SAkinobu Mita int ret; 12618cb6368SRenaud Cerrato 1277f43020eSBruno Thomsen /* 1287f43020eSBruno Thomsen * Avoid reading CTRL2 register as it causes WD_VAL register 1297f43020eSBruno Thomsen * value to reset to 0 which means watchdog is stopped. 1307f43020eSBruno Thomsen */ 1316211aceeSHugo Villeneuve ret = regmap_bulk_read(pcf2127->regmap, pcf2127->cfg->reg_time_base, 1326211aceeSHugo Villeneuve buf, sizeof(buf)); 133907b3262SAkinobu Mita if (ret) { 134907b3262SAkinobu Mita dev_err(dev, "%s: read error\n", __func__); 135907b3262SAkinobu Mita return ret; 13618cb6368SRenaud Cerrato } 13718cb6368SRenaud Cerrato 138bbfe3a7aSBruno Thomsen /* Clock integrity is not guaranteed when OSF flag is set. */ 13931f077c3SHugo Villeneuve if (buf[0] & PCF2127_BIT_SC_OSF) { 140653ebd75SAndrea Scian /* 141653ebd75SAndrea Scian * no need clear the flag here, 142653ebd75SAndrea Scian * it will be cleared once the new date is saved 143653ebd75SAndrea Scian */ 144907b3262SAkinobu Mita dev_warn(dev, 145653ebd75SAndrea Scian "oscillator stop detected, date/time is not reliable\n"); 146653ebd75SAndrea Scian return -EINVAL; 14718cb6368SRenaud Cerrato } 14818cb6368SRenaud Cerrato 149907b3262SAkinobu Mita dev_dbg(dev, 15031f077c3SHugo Villeneuve "%s: raw data is sec=%02x, min=%02x, hr=%02x, " 15118cb6368SRenaud Cerrato "mday=%02x, wday=%02x, mon=%02x, year=%02x\n", 15231f077c3SHugo Villeneuve __func__, buf[0], buf[1], buf[2], buf[3], buf[4], buf[5], buf[6]); 15318cb6368SRenaud Cerrato 15431f077c3SHugo Villeneuve tm->tm_sec = bcd2bin(buf[0] & 0x7F); 15531f077c3SHugo Villeneuve tm->tm_min = bcd2bin(buf[1] & 0x7F); 1560476b6c8SHugo Villeneuve tm->tm_hour = bcd2bin(buf[2] & 0x3F); 15731f077c3SHugo Villeneuve tm->tm_mday = bcd2bin(buf[3] & 0x3F); 15831f077c3SHugo Villeneuve tm->tm_wday = buf[4] & 0x07; 1590476b6c8SHugo Villeneuve tm->tm_mon = bcd2bin(buf[5] & 0x1F) - 1; 16031f077c3SHugo Villeneuve tm->tm_year = bcd2bin(buf[6]); 161b139bb5cSAlexandre Belloni tm->tm_year += 100; 16218cb6368SRenaud Cerrato 163907b3262SAkinobu Mita dev_dbg(dev, "%s: tm is secs=%d, mins=%d, hours=%d, " 16418cb6368SRenaud Cerrato "mday=%d, mon=%d, year=%d, wday=%d\n", 16518cb6368SRenaud Cerrato __func__, 16618cb6368SRenaud Cerrato tm->tm_sec, tm->tm_min, tm->tm_hour, 16718cb6368SRenaud Cerrato tm->tm_mday, tm->tm_mon, tm->tm_year, tm->tm_wday); 16818cb6368SRenaud Cerrato 16922652ba7SAlexandre Belloni return 0; 17018cb6368SRenaud Cerrato } 17118cb6368SRenaud Cerrato 172907b3262SAkinobu Mita static int pcf2127_rtc_set_time(struct device *dev, struct rtc_time *tm) 17318cb6368SRenaud Cerrato { 174907b3262SAkinobu Mita struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 175907b3262SAkinobu Mita unsigned char buf[7]; 17618cb6368SRenaud Cerrato int i = 0, err; 17718cb6368SRenaud Cerrato 178907b3262SAkinobu Mita dev_dbg(dev, "%s: secs=%d, mins=%d, hours=%d, " 17918cb6368SRenaud Cerrato "mday=%d, mon=%d, year=%d, wday=%d\n", 18018cb6368SRenaud Cerrato __func__, 18118cb6368SRenaud Cerrato tm->tm_sec, tm->tm_min, tm->tm_hour, 18218cb6368SRenaud Cerrato tm->tm_mday, tm->tm_mon, tm->tm_year, tm->tm_wday); 18318cb6368SRenaud Cerrato 18418cb6368SRenaud Cerrato /* hours, minutes and seconds */ 185653ebd75SAndrea Scian buf[i++] = bin2bcd(tm->tm_sec); /* this will also clear OSF flag */ 18618cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_min); 18718cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_hour); 18818cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_mday); 18918cb6368SRenaud Cerrato buf[i++] = tm->tm_wday & 0x07; 19018cb6368SRenaud Cerrato 19118cb6368SRenaud Cerrato /* month, 1 - 12 */ 19218cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_mon + 1); 19318cb6368SRenaud Cerrato 19418cb6368SRenaud Cerrato /* year */ 195b139bb5cSAlexandre Belloni buf[i++] = bin2bcd(tm->tm_year - 100); 19618cb6368SRenaud Cerrato 19718cb6368SRenaud Cerrato /* write register's data */ 1986211aceeSHugo Villeneuve err = regmap_bulk_write(pcf2127->regmap, pcf2127->cfg->reg_time_base, buf, i); 199907b3262SAkinobu Mita if (err) { 2003d740c64SHugo Villeneuve dev_dbg(dev, "%s: err=%d", __func__, err); 201907b3262SAkinobu Mita return err; 20218cb6368SRenaud Cerrato } 20318cb6368SRenaud Cerrato 20418cb6368SRenaud Cerrato return 0; 20518cb6368SRenaud Cerrato } 20618cb6368SRenaud Cerrato 20718cb6368SRenaud Cerrato static int pcf2127_rtc_ioctl(struct device *dev, 20818cb6368SRenaud Cerrato unsigned int cmd, unsigned long arg) 20918cb6368SRenaud Cerrato { 210907b3262SAkinobu Mita struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 2117d65cf8cSAlexandre Belloni int val, touser = 0; 212f97cfddcSUwe Kleine-König int ret; 21318cb6368SRenaud Cerrato 21418cb6368SRenaud Cerrato switch (cmd) { 21518cb6368SRenaud Cerrato case RTC_VL_READ: 2167d65cf8cSAlexandre Belloni ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL3, &val); 217907b3262SAkinobu Mita if (ret) 218f97cfddcSUwe Kleine-König return ret; 21918cb6368SRenaud Cerrato 2207d65cf8cSAlexandre Belloni if (val & PCF2127_BIT_CTRL3_BLF) 2217d65cf8cSAlexandre Belloni touser |= RTC_VL_BACKUP_LOW; 2227d65cf8cSAlexandre Belloni 2237d65cf8cSAlexandre Belloni if (val & PCF2127_BIT_CTRL3_BF) 2247d65cf8cSAlexandre Belloni touser |= RTC_VL_BACKUP_SWITCH; 225f97cfddcSUwe Kleine-König 226af427311SAlexandre Belloni return put_user(touser, (unsigned int __user *)arg); 2277d65cf8cSAlexandre Belloni 2287d65cf8cSAlexandre Belloni case RTC_VL_CLR: 2297d65cf8cSAlexandre Belloni return regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL3, 2307d65cf8cSAlexandre Belloni PCF2127_BIT_CTRL3_BF, 0); 2317d65cf8cSAlexandre Belloni 23218cb6368SRenaud Cerrato default: 23318cb6368SRenaud Cerrato return -ENOIOCTLCMD; 23418cb6368SRenaud Cerrato } 23518cb6368SRenaud Cerrato } 23618cb6368SRenaud Cerrato 237d6c3029fSUwe Kleine-König static int pcf2127_nvmem_read(void *priv, unsigned int offset, 238d6c3029fSUwe Kleine-König void *val, size_t bytes) 239d6c3029fSUwe Kleine-König { 240d6c3029fSUwe Kleine-König struct pcf2127 *pcf2127 = priv; 241d6c3029fSUwe Kleine-König int ret; 242d6c3029fSUwe Kleine-König unsigned char offsetbuf[] = { offset >> 8, offset }; 243d6c3029fSUwe Kleine-König 244bbfe3a7aSBruno Thomsen ret = regmap_bulk_write(pcf2127->regmap, PCF2127_REG_RAM_ADDR_MSB, 245d6c3029fSUwe Kleine-König offsetbuf, 2); 246d6c3029fSUwe Kleine-König if (ret) 247d6c3029fSUwe Kleine-König return ret; 248d6c3029fSUwe Kleine-König 249ba1c30bfSDan Carpenter return regmap_bulk_read(pcf2127->regmap, PCF2127_REG_RAM_RD_CMD, 250d6c3029fSUwe Kleine-König val, bytes); 251d6c3029fSUwe Kleine-König } 252d6c3029fSUwe Kleine-König 253d6c3029fSUwe Kleine-König static int pcf2127_nvmem_write(void *priv, unsigned int offset, 254d6c3029fSUwe Kleine-König void *val, size_t bytes) 255d6c3029fSUwe Kleine-König { 256d6c3029fSUwe Kleine-König struct pcf2127 *pcf2127 = priv; 257d6c3029fSUwe Kleine-König int ret; 258d6c3029fSUwe Kleine-König unsigned char offsetbuf[] = { offset >> 8, offset }; 259d6c3029fSUwe Kleine-König 260bbfe3a7aSBruno Thomsen ret = regmap_bulk_write(pcf2127->regmap, PCF2127_REG_RAM_ADDR_MSB, 261d6c3029fSUwe Kleine-König offsetbuf, 2); 262d6c3029fSUwe Kleine-König if (ret) 263d6c3029fSUwe Kleine-König return ret; 264d6c3029fSUwe Kleine-König 265ba1c30bfSDan Carpenter return regmap_bulk_write(pcf2127->regmap, PCF2127_REG_RAM_WRT_CMD, 266d6c3029fSUwe Kleine-König val, bytes); 267d6c3029fSUwe Kleine-König } 268d6c3029fSUwe Kleine-König 2690e735eaaSBruno Thomsen /* watchdog driver */ 2700e735eaaSBruno Thomsen 2710e735eaaSBruno Thomsen static int pcf2127_wdt_ping(struct watchdog_device *wdd) 2720e735eaaSBruno Thomsen { 2730e735eaaSBruno Thomsen struct pcf2127 *pcf2127 = watchdog_get_drvdata(wdd); 2740e735eaaSBruno Thomsen 2756b57ec29SHugo Villeneuve return regmap_write(pcf2127->regmap, pcf2127->cfg->reg_wd_val, wdd->timeout); 2760e735eaaSBruno Thomsen } 2770e735eaaSBruno Thomsen 2780e735eaaSBruno Thomsen /* 2790e735eaaSBruno Thomsen * Restart watchdog timer if feature is active. 2800e735eaaSBruno Thomsen * 2810e735eaaSBruno Thomsen * Note: Reading CTRL2 register causes watchdog to stop which is unfortunate, 2820e735eaaSBruno Thomsen * since register also contain control/status flags for other features. 2830e735eaaSBruno Thomsen * Always call this function after reading CTRL2 register. 2840e735eaaSBruno Thomsen */ 2850e735eaaSBruno Thomsen static int pcf2127_wdt_active_ping(struct watchdog_device *wdd) 2860e735eaaSBruno Thomsen { 2870e735eaaSBruno Thomsen int ret = 0; 2880e735eaaSBruno Thomsen 2890e735eaaSBruno Thomsen if (watchdog_active(wdd)) { 2900e735eaaSBruno Thomsen ret = pcf2127_wdt_ping(wdd); 2910e735eaaSBruno Thomsen if (ret) 2920e735eaaSBruno Thomsen dev_err(wdd->parent, 2930e735eaaSBruno Thomsen "%s: watchdog restart failed, ret=%d\n", 2940e735eaaSBruno Thomsen __func__, ret); 2950e735eaaSBruno Thomsen } 2960e735eaaSBruno Thomsen 2970e735eaaSBruno Thomsen return ret; 2980e735eaaSBruno Thomsen } 2990e735eaaSBruno Thomsen 3000e735eaaSBruno Thomsen static int pcf2127_wdt_start(struct watchdog_device *wdd) 3010e735eaaSBruno Thomsen { 3020e735eaaSBruno Thomsen return pcf2127_wdt_ping(wdd); 3030e735eaaSBruno Thomsen } 3040e735eaaSBruno Thomsen 3050e735eaaSBruno Thomsen static int pcf2127_wdt_stop(struct watchdog_device *wdd) 3060e735eaaSBruno Thomsen { 3070e735eaaSBruno Thomsen struct pcf2127 *pcf2127 = watchdog_get_drvdata(wdd); 3080e735eaaSBruno Thomsen 3096b57ec29SHugo Villeneuve return regmap_write(pcf2127->regmap, pcf2127->cfg->reg_wd_val, 3100e735eaaSBruno Thomsen PCF2127_WD_VAL_STOP); 3110e735eaaSBruno Thomsen } 3120e735eaaSBruno Thomsen 3130e735eaaSBruno Thomsen static int pcf2127_wdt_set_timeout(struct watchdog_device *wdd, 3140e735eaaSBruno Thomsen unsigned int new_timeout) 3150e735eaaSBruno Thomsen { 3160e735eaaSBruno Thomsen dev_dbg(wdd->parent, "new watchdog timeout: %is (old: %is)\n", 3170e735eaaSBruno Thomsen new_timeout, wdd->timeout); 3180e735eaaSBruno Thomsen 3190e735eaaSBruno Thomsen wdd->timeout = new_timeout; 3200e735eaaSBruno Thomsen 3210e735eaaSBruno Thomsen return pcf2127_wdt_active_ping(wdd); 3220e735eaaSBruno Thomsen } 3230e735eaaSBruno Thomsen 3240e735eaaSBruno Thomsen static const struct watchdog_info pcf2127_wdt_info = { 3250e735eaaSBruno Thomsen .identity = "NXP PCF2127/PCF2129 Watchdog", 3260e735eaaSBruno Thomsen .options = WDIOF_KEEPALIVEPING | WDIOF_SETTIMEOUT, 3270e735eaaSBruno Thomsen }; 3280e735eaaSBruno Thomsen 3290e735eaaSBruno Thomsen static const struct watchdog_ops pcf2127_watchdog_ops = { 3300e735eaaSBruno Thomsen .owner = THIS_MODULE, 3310e735eaaSBruno Thomsen .start = pcf2127_wdt_start, 3320e735eaaSBruno Thomsen .stop = pcf2127_wdt_stop, 3330e735eaaSBruno Thomsen .ping = pcf2127_wdt_ping, 3340e735eaaSBruno Thomsen .set_timeout = pcf2127_wdt_set_timeout, 3350e735eaaSBruno Thomsen }; 3360e735eaaSBruno Thomsen 3375d78533aSUwe Kleine-König static int pcf2127_watchdog_init(struct device *dev, struct pcf2127 *pcf2127) 3385d78533aSUwe Kleine-König { 3395d78533aSUwe Kleine-König u32 wdd_timeout; 3405d78533aSUwe Kleine-König int ret; 3415d78533aSUwe Kleine-König 34271ac1345SUwe Kleine-König if (!IS_ENABLED(CONFIG_WATCHDOG) || 34371ac1345SUwe Kleine-König !device_property_read_bool(dev, "reset-source")) 3445d78533aSUwe Kleine-König return 0; 3455d78533aSUwe Kleine-König 3465d78533aSUwe Kleine-König pcf2127->wdd.parent = dev; 3475d78533aSUwe Kleine-König pcf2127->wdd.info = &pcf2127_wdt_info; 3485d78533aSUwe Kleine-König pcf2127->wdd.ops = &pcf2127_watchdog_ops; 3495d78533aSUwe Kleine-König pcf2127->wdd.min_timeout = PCF2127_WD_VAL_MIN; 3505d78533aSUwe Kleine-König pcf2127->wdd.max_timeout = PCF2127_WD_VAL_MAX; 3515d78533aSUwe Kleine-König pcf2127->wdd.timeout = PCF2127_WD_VAL_DEFAULT; 3525d78533aSUwe Kleine-König pcf2127->wdd.min_hw_heartbeat_ms = 500; 3535d78533aSUwe Kleine-König pcf2127->wdd.status = WATCHDOG_NOWAYOUT_INIT_STATUS; 3545d78533aSUwe Kleine-König 3555d78533aSUwe Kleine-König watchdog_set_drvdata(&pcf2127->wdd, pcf2127); 3565d78533aSUwe Kleine-König 3575d78533aSUwe Kleine-König /* Test if watchdog timer is started by bootloader */ 3586b57ec29SHugo Villeneuve ret = regmap_read(pcf2127->regmap, pcf2127->cfg->reg_wd_val, &wdd_timeout); 3595d78533aSUwe Kleine-König if (ret) 3605d78533aSUwe Kleine-König return ret; 3615d78533aSUwe Kleine-König 3625d78533aSUwe Kleine-König if (wdd_timeout) 3635d78533aSUwe Kleine-König set_bit(WDOG_HW_RUNNING, &pcf2127->wdd.status); 3645d78533aSUwe Kleine-König 3655d78533aSUwe Kleine-König return devm_watchdog_register_device(dev, &pcf2127->wdd); 3665d78533aSUwe Kleine-König } 3675d78533aSUwe Kleine-König 3688a914bacSLiam Beguin /* Alarm */ 3698a914bacSLiam Beguin static int pcf2127_rtc_read_alarm(struct device *dev, struct rtc_wkalrm *alrm) 3708a914bacSLiam Beguin { 3718a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 37273ce0530SHugo Villeneuve u8 buf[5]; 37373ce0530SHugo Villeneuve unsigned int ctrl2; 3748a914bacSLiam Beguin int ret; 3758a914bacSLiam Beguin 3768a914bacSLiam Beguin ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL2, &ctrl2); 3778a914bacSLiam Beguin if (ret) 3788a914bacSLiam Beguin return ret; 3798a914bacSLiam Beguin 3808a914bacSLiam Beguin ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 3818a914bacSLiam Beguin if (ret) 3828a914bacSLiam Beguin return ret; 3838a914bacSLiam Beguin 3847c6f0db4SHugo Villeneuve ret = regmap_bulk_read(pcf2127->regmap, pcf2127->cfg->regs_alarm_base, 3857c6f0db4SHugo Villeneuve buf, sizeof(buf)); 3868a914bacSLiam Beguin if (ret) 3878a914bacSLiam Beguin return ret; 3888a914bacSLiam Beguin 3898a914bacSLiam Beguin alrm->enabled = ctrl2 & PCF2127_BIT_CTRL2_AIE; 3908a914bacSLiam Beguin alrm->pending = ctrl2 & PCF2127_BIT_CTRL2_AF; 3918a914bacSLiam Beguin 3928a914bacSLiam Beguin alrm->time.tm_sec = bcd2bin(buf[0] & 0x7F); 3938a914bacSLiam Beguin alrm->time.tm_min = bcd2bin(buf[1] & 0x7F); 3948a914bacSLiam Beguin alrm->time.tm_hour = bcd2bin(buf[2] & 0x3F); 3958a914bacSLiam Beguin alrm->time.tm_mday = bcd2bin(buf[3] & 0x3F); 3968a914bacSLiam Beguin 3978a914bacSLiam Beguin return 0; 3988a914bacSLiam Beguin } 3998a914bacSLiam Beguin 4008a914bacSLiam Beguin static int pcf2127_rtc_alarm_irq_enable(struct device *dev, u32 enable) 4018a914bacSLiam Beguin { 4028a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 4038a914bacSLiam Beguin int ret; 4048a914bacSLiam Beguin 4058a914bacSLiam Beguin ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL2, 4068a914bacSLiam Beguin PCF2127_BIT_CTRL2_AIE, 4078a914bacSLiam Beguin enable ? PCF2127_BIT_CTRL2_AIE : 0); 4088a914bacSLiam Beguin if (ret) 4098a914bacSLiam Beguin return ret; 4108a914bacSLiam Beguin 4118a914bacSLiam Beguin return pcf2127_wdt_active_ping(&pcf2127->wdd); 4128a914bacSLiam Beguin } 4138a914bacSLiam Beguin 4148a914bacSLiam Beguin static int pcf2127_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alrm) 4158a914bacSLiam Beguin { 4168a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 4178a914bacSLiam Beguin uint8_t buf[5]; 4188a914bacSLiam Beguin int ret; 4198a914bacSLiam Beguin 4208a914bacSLiam Beguin ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL2, 4218a914bacSLiam Beguin PCF2127_BIT_CTRL2_AF, 0); 4228a914bacSLiam Beguin if (ret) 4238a914bacSLiam Beguin return ret; 4248a914bacSLiam Beguin 4258a914bacSLiam Beguin ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 4268a914bacSLiam Beguin if (ret) 4278a914bacSLiam Beguin return ret; 4288a914bacSLiam Beguin 4298a914bacSLiam Beguin buf[0] = bin2bcd(alrm->time.tm_sec); 4308a914bacSLiam Beguin buf[1] = bin2bcd(alrm->time.tm_min); 4318a914bacSLiam Beguin buf[2] = bin2bcd(alrm->time.tm_hour); 4328a914bacSLiam Beguin buf[3] = bin2bcd(alrm->time.tm_mday); 43327006416SAlexandre Belloni buf[4] = PCF2127_BIT_ALARM_AE; /* Do not match on week day */ 4348a914bacSLiam Beguin 4357c6f0db4SHugo Villeneuve ret = regmap_bulk_write(pcf2127->regmap, pcf2127->cfg->regs_alarm_base, 4367c6f0db4SHugo Villeneuve buf, sizeof(buf)); 4378a914bacSLiam Beguin if (ret) 4388a914bacSLiam Beguin return ret; 4398a914bacSLiam Beguin 4408a914bacSLiam Beguin return pcf2127_rtc_alarm_irq_enable(dev, alrm->enabled); 4418a914bacSLiam Beguin } 4428a914bacSLiam Beguin 4432f861984SMian Yousaf Kaukab /* 4442f861984SMian Yousaf Kaukab * This function reads ctrl2 register, caller is responsible for calling 4452f861984SMian Yousaf Kaukab * pcf2127_wdt_active_ping() 4462f861984SMian Yousaf Kaukab */ 4472f861984SMian Yousaf Kaukab static int pcf2127_rtc_ts_read(struct device *dev, time64_t *ts) 4482f861984SMian Yousaf Kaukab { 4492f861984SMian Yousaf Kaukab struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 4502f861984SMian Yousaf Kaukab struct rtc_time tm; 4512f861984SMian Yousaf Kaukab int ret; 452720fb4b8SHugo Villeneuve unsigned char data[7]; 4532f861984SMian Yousaf Kaukab 454720fb4b8SHugo Villeneuve ret = regmap_bulk_read(pcf2127->regmap, PCF2127_REG_TS_CTRL, data, 4552f861984SMian Yousaf Kaukab sizeof(data)); 4562f861984SMian Yousaf Kaukab if (ret) { 4572f861984SMian Yousaf Kaukab dev_err(dev, "%s: read error ret=%d\n", __func__, ret); 4582f861984SMian Yousaf Kaukab return ret; 4592f861984SMian Yousaf Kaukab } 4602f861984SMian Yousaf Kaukab 4612f861984SMian Yousaf Kaukab dev_dbg(dev, 462720fb4b8SHugo Villeneuve "%s: raw data is ts_sc=%02x, ts_mn=%02x, ts_hr=%02x, ts_dm=%02x, ts_mo=%02x, ts_yr=%02x\n", 463720fb4b8SHugo Villeneuve __func__, data[1], data[2], data[3], data[4], data[5], data[6]); 4642f861984SMian Yousaf Kaukab 465720fb4b8SHugo Villeneuve tm.tm_sec = bcd2bin(data[1] & 0x7F); 466720fb4b8SHugo Villeneuve tm.tm_min = bcd2bin(data[2] & 0x7F); 467720fb4b8SHugo Villeneuve tm.tm_hour = bcd2bin(data[3] & 0x3F); 468720fb4b8SHugo Villeneuve tm.tm_mday = bcd2bin(data[4] & 0x3F); 4692f861984SMian Yousaf Kaukab /* TS_MO register (month) value range: 1-12 */ 470720fb4b8SHugo Villeneuve tm.tm_mon = bcd2bin(data[5] & 0x1F) - 1; 471720fb4b8SHugo Villeneuve tm.tm_year = bcd2bin(data[6]); 4722f861984SMian Yousaf Kaukab if (tm.tm_year < 70) 4732f861984SMian Yousaf Kaukab tm.tm_year += 100; /* assume we are in 1970...2069 */ 4742f861984SMian Yousaf Kaukab 4752f861984SMian Yousaf Kaukab ret = rtc_valid_tm(&tm); 4762f861984SMian Yousaf Kaukab if (ret) { 4772f861984SMian Yousaf Kaukab dev_err(dev, "Invalid timestamp. ret=%d\n", ret); 4782f861984SMian Yousaf Kaukab return ret; 4792f861984SMian Yousaf Kaukab } 4802f861984SMian Yousaf Kaukab 4812f861984SMian Yousaf Kaukab *ts = rtc_tm_to_time64(&tm); 4822f861984SMian Yousaf Kaukab return 0; 4832f861984SMian Yousaf Kaukab }; 4842f861984SMian Yousaf Kaukab 4852f861984SMian Yousaf Kaukab static void pcf2127_rtc_ts_snapshot(struct device *dev) 4862f861984SMian Yousaf Kaukab { 4872f861984SMian Yousaf Kaukab struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 4882f861984SMian Yousaf Kaukab int ret; 4892f861984SMian Yousaf Kaukab 4902f861984SMian Yousaf Kaukab /* Let userspace read the first timestamp */ 4912f861984SMian Yousaf Kaukab if (pcf2127->ts_valid) 4922f861984SMian Yousaf Kaukab return; 4932f861984SMian Yousaf Kaukab 4942f861984SMian Yousaf Kaukab ret = pcf2127_rtc_ts_read(dev, &pcf2127->ts); 4952f861984SMian Yousaf Kaukab if (!ret) 4962f861984SMian Yousaf Kaukab pcf2127->ts_valid = true; 4972f861984SMian Yousaf Kaukab } 4982f861984SMian Yousaf Kaukab 4998a914bacSLiam Beguin static irqreturn_t pcf2127_rtc_irq(int irq, void *dev) 5008a914bacSLiam Beguin { 5018a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 5022f861984SMian Yousaf Kaukab unsigned int ctrl1, ctrl2; 5038a914bacSLiam Beguin int ret = 0; 5048a914bacSLiam Beguin 5052f861984SMian Yousaf Kaukab ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL1, &ctrl1); 5062f861984SMian Yousaf Kaukab if (ret) 5072f861984SMian Yousaf Kaukab return IRQ_NONE; 5082f861984SMian Yousaf Kaukab 5098a914bacSLiam Beguin ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL2, &ctrl2); 5108a914bacSLiam Beguin if (ret) 5118a914bacSLiam Beguin return IRQ_NONE; 5128a914bacSLiam Beguin 5132f861984SMian Yousaf Kaukab if (!(ctrl1 & PCF2127_CTRL1_IRQ_MASK || ctrl2 & PCF2127_CTRL2_IRQ_MASK)) 51427006416SAlexandre Belloni return IRQ_NONE; 51527006416SAlexandre Belloni 5162f861984SMian Yousaf Kaukab if (ctrl1 & PCF2127_BIT_CTRL1_TSF1 || ctrl2 & PCF2127_BIT_CTRL2_TSF2) 5172f861984SMian Yousaf Kaukab pcf2127_rtc_ts_snapshot(dev); 5188a914bacSLiam Beguin 5192f861984SMian Yousaf Kaukab if (ctrl1 & PCF2127_CTRL1_IRQ_MASK) 5202f861984SMian Yousaf Kaukab regmap_write(pcf2127->regmap, PCF2127_REG_CTRL1, 5212f861984SMian Yousaf Kaukab ctrl1 & ~PCF2127_CTRL1_IRQ_MASK); 5222f861984SMian Yousaf Kaukab 5232f861984SMian Yousaf Kaukab if (ctrl2 & PCF2127_CTRL2_IRQ_MASK) 5242f861984SMian Yousaf Kaukab regmap_write(pcf2127->regmap, PCF2127_REG_CTRL2, 5252f861984SMian Yousaf Kaukab ctrl2 & ~PCF2127_CTRL2_IRQ_MASK); 5262f861984SMian Yousaf Kaukab 5272f861984SMian Yousaf Kaukab if (ctrl2 & PCF2127_BIT_CTRL2_AF) 5288a914bacSLiam Beguin rtc_update_irq(pcf2127->rtc, 1, RTC_IRQF | RTC_AF); 5298a914bacSLiam Beguin 53027006416SAlexandre Belloni pcf2127_wdt_active_ping(&pcf2127->wdd); 5318a914bacSLiam Beguin 5328a914bacSLiam Beguin return IRQ_HANDLED; 5338a914bacSLiam Beguin } 5348a914bacSLiam Beguin 53525cbe9c8SAlexandre Belloni static const struct rtc_class_ops pcf2127_rtc_ops = { 5368a914bacSLiam Beguin .ioctl = pcf2127_rtc_ioctl, 5378a914bacSLiam Beguin .read_time = pcf2127_rtc_read_time, 5388a914bacSLiam Beguin .set_time = pcf2127_rtc_set_time, 5398a914bacSLiam Beguin .read_alarm = pcf2127_rtc_read_alarm, 5408a914bacSLiam Beguin .set_alarm = pcf2127_rtc_set_alarm, 5418a914bacSLiam Beguin .alarm_irq_enable = pcf2127_rtc_alarm_irq_enable, 5428a914bacSLiam Beguin }; 5438a914bacSLiam Beguin 54403623b4bSBruno Thomsen /* sysfs interface */ 54503623b4bSBruno Thomsen 54603623b4bSBruno Thomsen static ssize_t timestamp0_store(struct device *dev, 54703623b4bSBruno Thomsen struct device_attribute *attr, 54803623b4bSBruno Thomsen const char *buf, size_t count) 54903623b4bSBruno Thomsen { 55003623b4bSBruno Thomsen struct pcf2127 *pcf2127 = dev_get_drvdata(dev->parent); 55103623b4bSBruno Thomsen int ret; 55203623b4bSBruno Thomsen 5532f861984SMian Yousaf Kaukab if (pcf2127->irq_enabled) { 5542f861984SMian Yousaf Kaukab pcf2127->ts_valid = false; 5552f861984SMian Yousaf Kaukab } else { 55603623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL1, 55703623b4bSBruno Thomsen PCF2127_BIT_CTRL1_TSF1, 0); 55803623b4bSBruno Thomsen if (ret) { 55903623b4bSBruno Thomsen dev_err(dev, "%s: update ctrl1 ret=%d\n", __func__, ret); 56003623b4bSBruno Thomsen return ret; 56103623b4bSBruno Thomsen } 56203623b4bSBruno Thomsen 56303623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL2, 56403623b4bSBruno Thomsen PCF2127_BIT_CTRL2_TSF2, 0); 56503623b4bSBruno Thomsen if (ret) { 56603623b4bSBruno Thomsen dev_err(dev, "%s: update ctrl2 ret=%d\n", __func__, ret); 56703623b4bSBruno Thomsen return ret; 56803623b4bSBruno Thomsen } 56903623b4bSBruno Thomsen 57003623b4bSBruno Thomsen ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 57103623b4bSBruno Thomsen if (ret) 57203623b4bSBruno Thomsen return ret; 5732f861984SMian Yousaf Kaukab } 57403623b4bSBruno Thomsen 57503623b4bSBruno Thomsen return count; 57603623b4bSBruno Thomsen }; 57703623b4bSBruno Thomsen 57803623b4bSBruno Thomsen static ssize_t timestamp0_show(struct device *dev, 57903623b4bSBruno Thomsen struct device_attribute *attr, char *buf) 58003623b4bSBruno Thomsen { 58103623b4bSBruno Thomsen struct pcf2127 *pcf2127 = dev_get_drvdata(dev->parent); 5822f861984SMian Yousaf Kaukab unsigned int ctrl1, ctrl2; 58303623b4bSBruno Thomsen int ret; 5842f861984SMian Yousaf Kaukab time64_t ts; 58503623b4bSBruno Thomsen 5862f861984SMian Yousaf Kaukab if (pcf2127->irq_enabled) { 5872f861984SMian Yousaf Kaukab if (!pcf2127->ts_valid) 5882f861984SMian Yousaf Kaukab return 0; 5892f861984SMian Yousaf Kaukab ts = pcf2127->ts; 5902f861984SMian Yousaf Kaukab } else { 5912f861984SMian Yousaf Kaukab ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL1, &ctrl1); 5922f861984SMian Yousaf Kaukab if (ret) 5932f861984SMian Yousaf Kaukab return 0; 59403623b4bSBruno Thomsen 5952f861984SMian Yousaf Kaukab ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL2, &ctrl2); 5962f861984SMian Yousaf Kaukab if (ret) 5972f861984SMian Yousaf Kaukab return 0; 5982f861984SMian Yousaf Kaukab 5992f861984SMian Yousaf Kaukab if (!(ctrl1 & PCF2127_BIT_CTRL1_TSF1) && 6002f861984SMian Yousaf Kaukab !(ctrl2 & PCF2127_BIT_CTRL2_TSF2)) 6012f861984SMian Yousaf Kaukab return 0; 6022f861984SMian Yousaf Kaukab 6032f861984SMian Yousaf Kaukab ret = pcf2127_rtc_ts_read(dev->parent, &ts); 6042f861984SMian Yousaf Kaukab if (ret) 6052f861984SMian Yousaf Kaukab return 0; 60603623b4bSBruno Thomsen 60703623b4bSBruno Thomsen ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 60803623b4bSBruno Thomsen if (ret) 60903623b4bSBruno Thomsen return ret; 6102f861984SMian Yousaf Kaukab } 6112f861984SMian Yousaf Kaukab return sprintf(buf, "%llu\n", (unsigned long long)ts); 61203623b4bSBruno Thomsen }; 61303623b4bSBruno Thomsen 61403623b4bSBruno Thomsen static DEVICE_ATTR_RW(timestamp0); 61503623b4bSBruno Thomsen 61603623b4bSBruno Thomsen static struct attribute *pcf2127_attrs[] = { 61703623b4bSBruno Thomsen &dev_attr_timestamp0.attr, 61803623b4bSBruno Thomsen NULL 61903623b4bSBruno Thomsen }; 62003623b4bSBruno Thomsen 62103623b4bSBruno Thomsen static const struct attribute_group pcf2127_attr_group = { 62203623b4bSBruno Thomsen .attrs = pcf2127_attrs, 62303623b4bSBruno Thomsen }; 62403623b4bSBruno Thomsen 625fd28ceb4SHugo Villeneuve static struct pcf21xx_config pcf21xx_cfg[] = { 626fd28ceb4SHugo Villeneuve [PCF2127] = { 627fd28ceb4SHugo Villeneuve .type = PCF2127, 628fd28ceb4SHugo Villeneuve .max_register = 0x1d, 629fd28ceb4SHugo Villeneuve .has_nvmem = 1, 630fd28ceb4SHugo Villeneuve .has_bit_wd_ctl_cd0 = 1, 6316211aceeSHugo Villeneuve .reg_time_base = PCF2127_REG_TIME_BASE, 6327c6f0db4SHugo Villeneuve .regs_alarm_base = PCF2127_REG_ALARM_BASE, 6336b57ec29SHugo Villeneuve .reg_wd_ctl = PCF2127_REG_WD_CTL, 6346b57ec29SHugo Villeneuve .reg_wd_val = PCF2127_REG_WD_VAL, 635*fc16599eSHugo Villeneuve .reg_clkout = PCF2127_REG_CLKOUT, 636fd28ceb4SHugo Villeneuve }, 637fd28ceb4SHugo Villeneuve [PCF2129] = { 638fd28ceb4SHugo Villeneuve .type = PCF2129, 639fd28ceb4SHugo Villeneuve .max_register = 0x19, 640fd28ceb4SHugo Villeneuve .has_nvmem = 0, 641fd28ceb4SHugo Villeneuve .has_bit_wd_ctl_cd0 = 0, 6426211aceeSHugo Villeneuve .reg_time_base = PCF2127_REG_TIME_BASE, 6437c6f0db4SHugo Villeneuve .regs_alarm_base = PCF2127_REG_ALARM_BASE, 6446b57ec29SHugo Villeneuve .reg_wd_ctl = PCF2127_REG_WD_CTL, 6456b57ec29SHugo Villeneuve .reg_wd_val = PCF2127_REG_WD_VAL, 646*fc16599eSHugo Villeneuve .reg_clkout = PCF2127_REG_CLKOUT, 647fd28ceb4SHugo Villeneuve }, 648fd28ceb4SHugo Villeneuve }; 649fd28ceb4SHugo Villeneuve 650907b3262SAkinobu Mita static int pcf2127_probe(struct device *dev, struct regmap *regmap, 651fd28ceb4SHugo Villeneuve int alarm_irq, const char *name, const struct pcf21xx_config *config) 65218cb6368SRenaud Cerrato { 65318cb6368SRenaud Cerrato struct pcf2127 *pcf2127; 654d6c3029fSUwe Kleine-König int ret = 0; 65515f57b3eSPhilipp Rosenberger unsigned int val; 65618cb6368SRenaud Cerrato 657907b3262SAkinobu Mita dev_dbg(dev, "%s\n", __func__); 65818cb6368SRenaud Cerrato 659907b3262SAkinobu Mita pcf2127 = devm_kzalloc(dev, sizeof(*pcf2127), GFP_KERNEL); 66018cb6368SRenaud Cerrato if (!pcf2127) 66118cb6368SRenaud Cerrato return -ENOMEM; 66218cb6368SRenaud Cerrato 663907b3262SAkinobu Mita pcf2127->regmap = regmap; 664fd28ceb4SHugo Villeneuve pcf2127->cfg = config; 66518cb6368SRenaud Cerrato 666907b3262SAkinobu Mita dev_set_drvdata(dev, pcf2127); 667907b3262SAkinobu Mita 668e788771cSBruno Thomsen pcf2127->rtc = devm_rtc_allocate_device(dev); 669d6c3029fSUwe Kleine-König if (IS_ERR(pcf2127->rtc)) 670d6c3029fSUwe Kleine-König return PTR_ERR(pcf2127->rtc); 67118cb6368SRenaud Cerrato 672e788771cSBruno Thomsen pcf2127->rtc->ops = &pcf2127_rtc_ops; 673b139bb5cSAlexandre Belloni pcf2127->rtc->range_min = RTC_TIMESTAMP_BEGIN_2000; 674b139bb5cSAlexandre Belloni pcf2127->rtc->range_max = RTC_TIMESTAMP_END_2099; 675b139bb5cSAlexandre Belloni pcf2127->rtc->set_start_time = true; /* Sets actual start to 1970 */ 676bda10273SAlexandre Belloni set_bit(RTC_FEATURE_ALARM_RES_2S, pcf2127->rtc->features); 677689fafd5SAlexandre Belloni clear_bit(RTC_FEATURE_UPDATE_INTERRUPT, pcf2127->rtc->features); 67825cbe9c8SAlexandre Belloni clear_bit(RTC_FEATURE_ALARM, pcf2127->rtc->features); 679e788771cSBruno Thomsen 68035425bafSBiwen Li if (alarm_irq > 0) { 681d4785b46SHugo Villeneuve unsigned long flags; 682d4785b46SHugo Villeneuve 683d4785b46SHugo Villeneuve /* 684d4785b46SHugo Villeneuve * If flags = 0, devm_request_threaded_irq() will use IRQ flags 685d4785b46SHugo Villeneuve * obtained from device tree. 686d4785b46SHugo Villeneuve */ 687d4785b46SHugo Villeneuve if (dev_fwnode(dev)) 688d4785b46SHugo Villeneuve flags = 0; 689d4785b46SHugo Villeneuve else 690d4785b46SHugo Villeneuve flags = IRQF_TRIGGER_LOW; 691d4785b46SHugo Villeneuve 69227006416SAlexandre Belloni ret = devm_request_threaded_irq(dev, alarm_irq, NULL, 69327006416SAlexandre Belloni pcf2127_rtc_irq, 694d4785b46SHugo Villeneuve flags | IRQF_ONESHOT, 6958a914bacSLiam Beguin dev_name(dev), dev); 6968a914bacSLiam Beguin if (ret) { 6978a914bacSLiam Beguin dev_err(dev, "failed to request alarm irq\n"); 6988a914bacSLiam Beguin return ret; 6998a914bacSLiam Beguin } 7002f861984SMian Yousaf Kaukab pcf2127->irq_enabled = true; 7018a914bacSLiam Beguin } 7028a914bacSLiam Beguin 70335425bafSBiwen Li if (alarm_irq > 0 || device_property_read_bool(dev, "wakeup-source")) { 7048a914bacSLiam Beguin device_init_wakeup(dev, true); 70525cbe9c8SAlexandre Belloni set_bit(RTC_FEATURE_ALARM, pcf2127->rtc->features); 7068a914bacSLiam Beguin } 7078a914bacSLiam Beguin 708fd28ceb4SHugo Villeneuve if (pcf2127->cfg->has_nvmem) { 709d6c3029fSUwe Kleine-König struct nvmem_config nvmem_cfg = { 710d6c3029fSUwe Kleine-König .priv = pcf2127, 711d6c3029fSUwe Kleine-König .reg_read = pcf2127_nvmem_read, 712d6c3029fSUwe Kleine-König .reg_write = pcf2127_nvmem_write, 713d6c3029fSUwe Kleine-König .size = 512, 714d6c3029fSUwe Kleine-König }; 715d6c3029fSUwe Kleine-König 7163a905c2dSBartosz Golaszewski ret = devm_rtc_nvmem_register(pcf2127->rtc, &nvmem_cfg); 717d6c3029fSUwe Kleine-König } 718d6c3029fSUwe Kleine-König 7190e735eaaSBruno Thomsen /* 720b9ac079aSPhilipp Rosenberger * The "Power-On Reset Override" facility prevents the RTC to do a reset 721b9ac079aSPhilipp Rosenberger * after power on. For normal operation the PORO must be disabled. 722b9ac079aSPhilipp Rosenberger */ 723b9ac079aSPhilipp Rosenberger regmap_clear_bits(pcf2127->regmap, PCF2127_REG_CTRL1, 724b9ac079aSPhilipp Rosenberger PCF2127_BIT_CTRL1_POR_OVRD); 725b9ac079aSPhilipp Rosenberger 726*fc16599eSHugo Villeneuve ret = regmap_read(pcf2127->regmap, pcf2127->cfg->reg_clkout, &val); 72715f57b3eSPhilipp Rosenberger if (ret < 0) 72815f57b3eSPhilipp Rosenberger return ret; 72915f57b3eSPhilipp Rosenberger 73015f57b3eSPhilipp Rosenberger if (!(val & PCF2127_BIT_CLKOUT_OTPR)) { 731*fc16599eSHugo Villeneuve ret = regmap_set_bits(pcf2127->regmap, pcf2127->cfg->reg_clkout, 73215f57b3eSPhilipp Rosenberger PCF2127_BIT_CLKOUT_OTPR); 73315f57b3eSPhilipp Rosenberger if (ret < 0) 73415f57b3eSPhilipp Rosenberger return ret; 73515f57b3eSPhilipp Rosenberger 73615f57b3eSPhilipp Rosenberger msleep(100); 73715f57b3eSPhilipp Rosenberger } 73815f57b3eSPhilipp Rosenberger 739b9ac079aSPhilipp Rosenberger /* 7400e735eaaSBruno Thomsen * Watchdog timer enabled and reset pin /RST activated when timed out. 7410e735eaaSBruno Thomsen * Select 1Hz clock source for watchdog timer. 7420e735eaaSBruno Thomsen * Note: Countdown timer disabled and not available. 7432843d565SBiwen Li * For pca2129, pcf2129, only bit[7] is for Symbol WD_CD 7442843d565SBiwen Li * of register watchdg_tim_ctl. The bit[6] is labeled 7452843d565SBiwen Li * as T. Bits labeled as T must always be written with 7462843d565SBiwen Li * logic 0. 7470e735eaaSBruno Thomsen */ 7486b57ec29SHugo Villeneuve ret = regmap_update_bits(pcf2127->regmap, pcf2127->cfg->reg_wd_ctl, 7490e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_CD1 | 7500e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_CD0 | 7510e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_TF1 | 7520e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_TF0, 7530e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_CD1 | 754fd28ceb4SHugo Villeneuve (pcf2127->cfg->has_bit_wd_ctl_cd0 ? PCF2127_BIT_WD_CTL_CD0 : 0) | 7550e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_TF1); 7560e735eaaSBruno Thomsen if (ret) { 7570e735eaaSBruno Thomsen dev_err(dev, "%s: watchdog config (wd_ctl) failed\n", __func__); 7580e735eaaSBruno Thomsen return ret; 7590e735eaaSBruno Thomsen } 7600e735eaaSBruno Thomsen 7615d78533aSUwe Kleine-König pcf2127_watchdog_init(dev, pcf2127); 7620e735eaaSBruno Thomsen 76303623b4bSBruno Thomsen /* 76403623b4bSBruno Thomsen * Disable battery low/switch-over timestamp and interrupts. 76503623b4bSBruno Thomsen * Clear battery interrupt flags which can block new trigger events. 76603623b4bSBruno Thomsen * Note: This is the default chip behaviour but added to ensure 76703623b4bSBruno Thomsen * correct tamper timestamp and interrupt function. 76803623b4bSBruno Thomsen */ 76903623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL3, 77003623b4bSBruno Thomsen PCF2127_BIT_CTRL3_BTSE | 77103623b4bSBruno Thomsen PCF2127_BIT_CTRL3_BIE | 77203623b4bSBruno Thomsen PCF2127_BIT_CTRL3_BLIE, 0); 77303623b4bSBruno Thomsen if (ret) { 77403623b4bSBruno Thomsen dev_err(dev, "%s: interrupt config (ctrl3) failed\n", 77503623b4bSBruno Thomsen __func__); 77603623b4bSBruno Thomsen return ret; 77703623b4bSBruno Thomsen } 77803623b4bSBruno Thomsen 77903623b4bSBruno Thomsen /* 78003623b4bSBruno Thomsen * Enable timestamp function and store timestamp of first trigger 7817b69b54aSHugo Villeneuve * event until TSF1 and TSF2 interrupt flags are cleared. 78203623b4bSBruno Thomsen */ 78303623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_TS_CTRL, 78403623b4bSBruno Thomsen PCF2127_BIT_TS_CTRL_TSOFF | 78503623b4bSBruno Thomsen PCF2127_BIT_TS_CTRL_TSM, 78603623b4bSBruno Thomsen PCF2127_BIT_TS_CTRL_TSM); 78703623b4bSBruno Thomsen if (ret) { 78803623b4bSBruno Thomsen dev_err(dev, "%s: tamper detection config (ts_ctrl) failed\n", 78903623b4bSBruno Thomsen __func__); 79003623b4bSBruno Thomsen return ret; 79103623b4bSBruno Thomsen } 79203623b4bSBruno Thomsen 79303623b4bSBruno Thomsen /* 79403623b4bSBruno Thomsen * Enable interrupt generation when TSF1 or TSF2 timestamp flags 79503623b4bSBruno Thomsen * are set. Interrupt signal is an open-drain output and can be 79603623b4bSBruno Thomsen * left floating if unused. 79703623b4bSBruno Thomsen */ 79803623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL2, 79903623b4bSBruno Thomsen PCF2127_BIT_CTRL2_TSIE, 80003623b4bSBruno Thomsen PCF2127_BIT_CTRL2_TSIE); 80103623b4bSBruno Thomsen if (ret) { 80203623b4bSBruno Thomsen dev_err(dev, "%s: tamper detection config (ctrl2) failed\n", 80303623b4bSBruno Thomsen __func__); 80403623b4bSBruno Thomsen return ret; 80503623b4bSBruno Thomsen } 80603623b4bSBruno Thomsen 80703623b4bSBruno Thomsen ret = rtc_add_group(pcf2127->rtc, &pcf2127_attr_group); 80803623b4bSBruno Thomsen if (ret) { 80903623b4bSBruno Thomsen dev_err(dev, "%s: tamper sysfs registering failed\n", 81003623b4bSBruno Thomsen __func__); 81103623b4bSBruno Thomsen return ret; 81203623b4bSBruno Thomsen } 81303623b4bSBruno Thomsen 814fdcfd854SBartosz Golaszewski return devm_rtc_register_device(pcf2127->rtc); 81518cb6368SRenaud Cerrato } 81618cb6368SRenaud Cerrato 81718cb6368SRenaud Cerrato #ifdef CONFIG_OF 81818cb6368SRenaud Cerrato static const struct of_device_id pcf2127_of_match[] = { 819fd28ceb4SHugo Villeneuve { .compatible = "nxp,pcf2127", .data = &pcf21xx_cfg[PCF2127] }, 820fd28ceb4SHugo Villeneuve { .compatible = "nxp,pcf2129", .data = &pcf21xx_cfg[PCF2129] }, 821fd28ceb4SHugo Villeneuve { .compatible = "nxp,pca2129", .data = &pcf21xx_cfg[PCF2129] }, 82218cb6368SRenaud Cerrato {} 82318cb6368SRenaud Cerrato }; 82418cb6368SRenaud Cerrato MODULE_DEVICE_TABLE(of, pcf2127_of_match); 82518cb6368SRenaud Cerrato #endif 82618cb6368SRenaud Cerrato 8279408ec1aSAkinobu Mita #if IS_ENABLED(CONFIG_I2C) 8289408ec1aSAkinobu Mita 829907b3262SAkinobu Mita static int pcf2127_i2c_write(void *context, const void *data, size_t count) 830907b3262SAkinobu Mita { 831907b3262SAkinobu Mita struct device *dev = context; 832907b3262SAkinobu Mita struct i2c_client *client = to_i2c_client(dev); 833907b3262SAkinobu Mita int ret; 834907b3262SAkinobu Mita 835907b3262SAkinobu Mita ret = i2c_master_send(client, data, count); 836907b3262SAkinobu Mita if (ret != count) 837907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 838907b3262SAkinobu Mita 839907b3262SAkinobu Mita return 0; 840907b3262SAkinobu Mita } 841907b3262SAkinobu Mita 842907b3262SAkinobu Mita static int pcf2127_i2c_gather_write(void *context, 843907b3262SAkinobu Mita const void *reg, size_t reg_size, 844907b3262SAkinobu Mita const void *val, size_t val_size) 845907b3262SAkinobu Mita { 846907b3262SAkinobu Mita struct device *dev = context; 847907b3262SAkinobu Mita struct i2c_client *client = to_i2c_client(dev); 848907b3262SAkinobu Mita int ret; 849907b3262SAkinobu Mita void *buf; 850907b3262SAkinobu Mita 851907b3262SAkinobu Mita if (WARN_ON(reg_size != 1)) 852907b3262SAkinobu Mita return -EINVAL; 853907b3262SAkinobu Mita 854907b3262SAkinobu Mita buf = kmalloc(val_size + 1, GFP_KERNEL); 855907b3262SAkinobu Mita if (!buf) 856907b3262SAkinobu Mita return -ENOMEM; 857907b3262SAkinobu Mita 858907b3262SAkinobu Mita memcpy(buf, reg, 1); 859907b3262SAkinobu Mita memcpy(buf + 1, val, val_size); 860907b3262SAkinobu Mita 861907b3262SAkinobu Mita ret = i2c_master_send(client, buf, val_size + 1); 8629bde0afbSXulin Sun 8639bde0afbSXulin Sun kfree(buf); 8649bde0afbSXulin Sun 865907b3262SAkinobu Mita if (ret != val_size + 1) 866907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 867907b3262SAkinobu Mita 868907b3262SAkinobu Mita return 0; 869907b3262SAkinobu Mita } 870907b3262SAkinobu Mita 871907b3262SAkinobu Mita static int pcf2127_i2c_read(void *context, const void *reg, size_t reg_size, 872907b3262SAkinobu Mita void *val, size_t val_size) 873907b3262SAkinobu Mita { 874907b3262SAkinobu Mita struct device *dev = context; 875907b3262SAkinobu Mita struct i2c_client *client = to_i2c_client(dev); 876907b3262SAkinobu Mita int ret; 877907b3262SAkinobu Mita 878907b3262SAkinobu Mita if (WARN_ON(reg_size != 1)) 879907b3262SAkinobu Mita return -EINVAL; 880907b3262SAkinobu Mita 881907b3262SAkinobu Mita ret = i2c_master_send(client, reg, 1); 882907b3262SAkinobu Mita if (ret != 1) 883907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 884907b3262SAkinobu Mita 885907b3262SAkinobu Mita ret = i2c_master_recv(client, val, val_size); 886907b3262SAkinobu Mita if (ret != val_size) 887907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 888907b3262SAkinobu Mita 889907b3262SAkinobu Mita return 0; 890907b3262SAkinobu Mita } 891907b3262SAkinobu Mita 892907b3262SAkinobu Mita /* 893907b3262SAkinobu Mita * The reason we need this custom regmap_bus instead of using regmap_init_i2c() 894907b3262SAkinobu Mita * is that the STOP condition is required between set register address and 895907b3262SAkinobu Mita * read register data when reading from registers. 896907b3262SAkinobu Mita */ 897907b3262SAkinobu Mita static const struct regmap_bus pcf2127_i2c_regmap = { 898907b3262SAkinobu Mita .write = pcf2127_i2c_write, 899907b3262SAkinobu Mita .gather_write = pcf2127_i2c_gather_write, 900907b3262SAkinobu Mita .read = pcf2127_i2c_read, 90118cb6368SRenaud Cerrato }; 90218cb6368SRenaud Cerrato 903907b3262SAkinobu Mita static struct i2c_driver pcf2127_i2c_driver; 904907b3262SAkinobu Mita 9055418e595SUwe Kleine-König static const struct i2c_device_id pcf2127_i2c_id[] = { 906fd28ceb4SHugo Villeneuve { "pcf2127", PCF2127 }, 907fd28ceb4SHugo Villeneuve { "pcf2129", PCF2129 }, 908fd28ceb4SHugo Villeneuve { "pca2129", PCF2129 }, 9095418e595SUwe Kleine-König { } 9105418e595SUwe Kleine-König }; 9115418e595SUwe Kleine-König MODULE_DEVICE_TABLE(i2c, pcf2127_i2c_id); 9125418e595SUwe Kleine-König 9135418e595SUwe Kleine-König static int pcf2127_i2c_probe(struct i2c_client *client) 914907b3262SAkinobu Mita { 915907b3262SAkinobu Mita struct regmap *regmap; 916fd28ceb4SHugo Villeneuve static struct regmap_config config = { 917907b3262SAkinobu Mita .reg_bits = 8, 918907b3262SAkinobu Mita .val_bits = 8, 919907b3262SAkinobu Mita }; 920fd28ceb4SHugo Villeneuve const struct pcf21xx_config *variant; 921907b3262SAkinobu Mita 922907b3262SAkinobu Mita if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) 923907b3262SAkinobu Mita return -ENODEV; 924907b3262SAkinobu Mita 925fd28ceb4SHugo Villeneuve if (client->dev.of_node) { 926fd28ceb4SHugo Villeneuve variant = of_device_get_match_data(&client->dev); 927fd28ceb4SHugo Villeneuve if (!variant) 928fd28ceb4SHugo Villeneuve return -ENODEV; 929fd28ceb4SHugo Villeneuve } else { 930fd28ceb4SHugo Villeneuve enum pcf21xx_type type = 931fd28ceb4SHugo Villeneuve i2c_match_id(pcf2127_i2c_id, client)->driver_data; 932fd28ceb4SHugo Villeneuve 933fd28ceb4SHugo Villeneuve if (type >= PCF21XX_LAST_ID) 934fd28ceb4SHugo Villeneuve return -ENODEV; 935fd28ceb4SHugo Villeneuve variant = &pcf21xx_cfg[type]; 936fd28ceb4SHugo Villeneuve } 937fd28ceb4SHugo Villeneuve 938fd28ceb4SHugo Villeneuve config.max_register = variant->max_register, 939fd28ceb4SHugo Villeneuve 940907b3262SAkinobu Mita regmap = devm_regmap_init(&client->dev, &pcf2127_i2c_regmap, 941907b3262SAkinobu Mita &client->dev, &config); 942907b3262SAkinobu Mita if (IS_ERR(regmap)) { 943907b3262SAkinobu Mita dev_err(&client->dev, "%s: regmap allocation failed: %ld\n", 944907b3262SAkinobu Mita __func__, PTR_ERR(regmap)); 945907b3262SAkinobu Mita return PTR_ERR(regmap); 946907b3262SAkinobu Mita } 947907b3262SAkinobu Mita 94827006416SAlexandre Belloni return pcf2127_probe(&client->dev, regmap, client->irq, 949fd28ceb4SHugo Villeneuve pcf2127_i2c_driver.driver.name, variant); 950907b3262SAkinobu Mita } 951907b3262SAkinobu Mita 952907b3262SAkinobu Mita static struct i2c_driver pcf2127_i2c_driver = { 953907b3262SAkinobu Mita .driver = { 954907b3262SAkinobu Mita .name = "rtc-pcf2127-i2c", 955907b3262SAkinobu Mita .of_match_table = of_match_ptr(pcf2127_of_match), 956907b3262SAkinobu Mita }, 95731b0cecbSUwe Kleine-König .probe = pcf2127_i2c_probe, 958907b3262SAkinobu Mita .id_table = pcf2127_i2c_id, 959907b3262SAkinobu Mita }; 9609408ec1aSAkinobu Mita 9619408ec1aSAkinobu Mita static int pcf2127_i2c_register_driver(void) 9629408ec1aSAkinobu Mita { 9639408ec1aSAkinobu Mita return i2c_add_driver(&pcf2127_i2c_driver); 9649408ec1aSAkinobu Mita } 9659408ec1aSAkinobu Mita 9669408ec1aSAkinobu Mita static void pcf2127_i2c_unregister_driver(void) 9679408ec1aSAkinobu Mita { 9689408ec1aSAkinobu Mita i2c_del_driver(&pcf2127_i2c_driver); 9699408ec1aSAkinobu Mita } 9709408ec1aSAkinobu Mita 9719408ec1aSAkinobu Mita #else 9729408ec1aSAkinobu Mita 9739408ec1aSAkinobu Mita static int pcf2127_i2c_register_driver(void) 9749408ec1aSAkinobu Mita { 9759408ec1aSAkinobu Mita return 0; 9769408ec1aSAkinobu Mita } 9779408ec1aSAkinobu Mita 9789408ec1aSAkinobu Mita static void pcf2127_i2c_unregister_driver(void) 9799408ec1aSAkinobu Mita { 9809408ec1aSAkinobu Mita } 9819408ec1aSAkinobu Mita 9829408ec1aSAkinobu Mita #endif 9839408ec1aSAkinobu Mita 9849408ec1aSAkinobu Mita #if IS_ENABLED(CONFIG_SPI_MASTER) 9859408ec1aSAkinobu Mita 9869408ec1aSAkinobu Mita static struct spi_driver pcf2127_spi_driver; 987fd28ceb4SHugo Villeneuve static const struct spi_device_id pcf2127_spi_id[]; 9889408ec1aSAkinobu Mita 9899408ec1aSAkinobu Mita static int pcf2127_spi_probe(struct spi_device *spi) 9909408ec1aSAkinobu Mita { 991fd28ceb4SHugo Villeneuve static struct regmap_config config = { 9929408ec1aSAkinobu Mita .reg_bits = 8, 9939408ec1aSAkinobu Mita .val_bits = 8, 9949408ec1aSAkinobu Mita .read_flag_mask = 0xa0, 9959408ec1aSAkinobu Mita .write_flag_mask = 0x20, 9969408ec1aSAkinobu Mita }; 9979408ec1aSAkinobu Mita struct regmap *regmap; 998fd28ceb4SHugo Villeneuve const struct pcf21xx_config *variant; 999fd28ceb4SHugo Villeneuve 1000fd28ceb4SHugo Villeneuve if (spi->dev.of_node) { 1001fd28ceb4SHugo Villeneuve variant = of_device_get_match_data(&spi->dev); 1002fd28ceb4SHugo Villeneuve if (!variant) 1003fd28ceb4SHugo Villeneuve return -ENODEV; 1004fd28ceb4SHugo Villeneuve } else { 1005fd28ceb4SHugo Villeneuve enum pcf21xx_type type = spi_get_device_id(spi)->driver_data; 1006fd28ceb4SHugo Villeneuve 1007fd28ceb4SHugo Villeneuve if (type >= PCF21XX_LAST_ID) 1008fd28ceb4SHugo Villeneuve return -ENODEV; 1009fd28ceb4SHugo Villeneuve variant = &pcf21xx_cfg[type]; 1010fd28ceb4SHugo Villeneuve } 1011fd28ceb4SHugo Villeneuve 1012fd28ceb4SHugo Villeneuve config.max_register = variant->max_register, 10139408ec1aSAkinobu Mita 10149408ec1aSAkinobu Mita regmap = devm_regmap_init_spi(spi, &config); 10159408ec1aSAkinobu Mita if (IS_ERR(regmap)) { 10169408ec1aSAkinobu Mita dev_err(&spi->dev, "%s: regmap allocation failed: %ld\n", 10179408ec1aSAkinobu Mita __func__, PTR_ERR(regmap)); 10189408ec1aSAkinobu Mita return PTR_ERR(regmap); 10199408ec1aSAkinobu Mita } 10209408ec1aSAkinobu Mita 102127006416SAlexandre Belloni return pcf2127_probe(&spi->dev, regmap, spi->irq, 102227006416SAlexandre Belloni pcf2127_spi_driver.driver.name, 1023fd28ceb4SHugo Villeneuve variant); 10249408ec1aSAkinobu Mita } 10259408ec1aSAkinobu Mita 10269408ec1aSAkinobu Mita static const struct spi_device_id pcf2127_spi_id[] = { 1027fd28ceb4SHugo Villeneuve { "pcf2127", PCF2127 }, 1028fd28ceb4SHugo Villeneuve { "pcf2129", PCF2129 }, 1029fd28ceb4SHugo Villeneuve { "pca2129", PCF2129 }, 10309408ec1aSAkinobu Mita { } 10319408ec1aSAkinobu Mita }; 10329408ec1aSAkinobu Mita MODULE_DEVICE_TABLE(spi, pcf2127_spi_id); 10339408ec1aSAkinobu Mita 10349408ec1aSAkinobu Mita static struct spi_driver pcf2127_spi_driver = { 10359408ec1aSAkinobu Mita .driver = { 10369408ec1aSAkinobu Mita .name = "rtc-pcf2127-spi", 10379408ec1aSAkinobu Mita .of_match_table = of_match_ptr(pcf2127_of_match), 10389408ec1aSAkinobu Mita }, 10399408ec1aSAkinobu Mita .probe = pcf2127_spi_probe, 10409408ec1aSAkinobu Mita .id_table = pcf2127_spi_id, 10419408ec1aSAkinobu Mita }; 10429408ec1aSAkinobu Mita 10439408ec1aSAkinobu Mita static int pcf2127_spi_register_driver(void) 10449408ec1aSAkinobu Mita { 10459408ec1aSAkinobu Mita return spi_register_driver(&pcf2127_spi_driver); 10469408ec1aSAkinobu Mita } 10479408ec1aSAkinobu Mita 10489408ec1aSAkinobu Mita static void pcf2127_spi_unregister_driver(void) 10499408ec1aSAkinobu Mita { 10509408ec1aSAkinobu Mita spi_unregister_driver(&pcf2127_spi_driver); 10519408ec1aSAkinobu Mita } 10529408ec1aSAkinobu Mita 10539408ec1aSAkinobu Mita #else 10549408ec1aSAkinobu Mita 10559408ec1aSAkinobu Mita static int pcf2127_spi_register_driver(void) 10569408ec1aSAkinobu Mita { 10579408ec1aSAkinobu Mita return 0; 10589408ec1aSAkinobu Mita } 10599408ec1aSAkinobu Mita 10609408ec1aSAkinobu Mita static void pcf2127_spi_unregister_driver(void) 10619408ec1aSAkinobu Mita { 10629408ec1aSAkinobu Mita } 10639408ec1aSAkinobu Mita 10649408ec1aSAkinobu Mita #endif 10659408ec1aSAkinobu Mita 10669408ec1aSAkinobu Mita static int __init pcf2127_init(void) 10679408ec1aSAkinobu Mita { 10689408ec1aSAkinobu Mita int ret; 10699408ec1aSAkinobu Mita 10709408ec1aSAkinobu Mita ret = pcf2127_i2c_register_driver(); 10719408ec1aSAkinobu Mita if (ret) { 10729408ec1aSAkinobu Mita pr_err("Failed to register pcf2127 i2c driver: %d\n", ret); 10739408ec1aSAkinobu Mita return ret; 10749408ec1aSAkinobu Mita } 10759408ec1aSAkinobu Mita 10769408ec1aSAkinobu Mita ret = pcf2127_spi_register_driver(); 10779408ec1aSAkinobu Mita if (ret) { 10789408ec1aSAkinobu Mita pr_err("Failed to register pcf2127 spi driver: %d\n", ret); 10799408ec1aSAkinobu Mita pcf2127_i2c_unregister_driver(); 10809408ec1aSAkinobu Mita } 10819408ec1aSAkinobu Mita 10829408ec1aSAkinobu Mita return ret; 10839408ec1aSAkinobu Mita } 10849408ec1aSAkinobu Mita module_init(pcf2127_init) 10859408ec1aSAkinobu Mita 10869408ec1aSAkinobu Mita static void __exit pcf2127_exit(void) 10879408ec1aSAkinobu Mita { 10889408ec1aSAkinobu Mita pcf2127_spi_unregister_driver(); 10899408ec1aSAkinobu Mita pcf2127_i2c_unregister_driver(); 10909408ec1aSAkinobu Mita } 10919408ec1aSAkinobu Mita module_exit(pcf2127_exit) 109218cb6368SRenaud Cerrato 109318cb6368SRenaud Cerrato MODULE_AUTHOR("Renaud Cerrato <r.cerrato@til-technologies.fr>"); 1094cee2cc21SAkinobu Mita MODULE_DESCRIPTION("NXP PCF2127/29 RTC driver"); 10954d8318bcSUwe Kleine-König MODULE_LICENSE("GPL v2"); 1096