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 * 13cee2cc21SAkinobu Mita * Datasheet: http://cache.nxp.com/documents/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> 24907b3262SAkinobu Mita #include <linux/regmap.h> 250e735eaaSBruno Thomsen #include <linux/watchdog.h> 2618cb6368SRenaud Cerrato 27bbfe3a7aSBruno Thomsen /* Control register 1 */ 28bbfe3a7aSBruno Thomsen #define PCF2127_REG_CTRL1 0x00 2903623b4bSBruno Thomsen #define PCF2127_BIT_CTRL1_TSF1 BIT(4) 30bbfe3a7aSBruno Thomsen /* Control register 2 */ 31bbfe3a7aSBruno Thomsen #define PCF2127_REG_CTRL2 0x01 328a914bacSLiam Beguin #define PCF2127_BIT_CTRL2_AIE BIT(1) 3303623b4bSBruno Thomsen #define PCF2127_BIT_CTRL2_TSIE BIT(2) 348a914bacSLiam Beguin #define PCF2127_BIT_CTRL2_AF BIT(4) 3503623b4bSBruno Thomsen #define PCF2127_BIT_CTRL2_TSF2 BIT(5) 3627006416SAlexandre Belloni #define PCF2127_BIT_CTRL2_WDTF BIT(6) 37bbfe3a7aSBruno Thomsen /* Control register 3 */ 38bbfe3a7aSBruno Thomsen #define PCF2127_REG_CTRL3 0x02 3903623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BLIE BIT(0) 4003623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BIE BIT(1) 41bbfe3a7aSBruno Thomsen #define PCF2127_BIT_CTRL3_BLF BIT(2) 4203623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BF BIT(3) 4303623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BTSE BIT(4) 44bbfe3a7aSBruno Thomsen /* Time and date registers */ 45bbfe3a7aSBruno Thomsen #define PCF2127_REG_SC 0x03 46bbfe3a7aSBruno Thomsen #define PCF2127_BIT_SC_OSF BIT(7) 47bbfe3a7aSBruno Thomsen #define PCF2127_REG_MN 0x04 48bbfe3a7aSBruno Thomsen #define PCF2127_REG_HR 0x05 49bbfe3a7aSBruno Thomsen #define PCF2127_REG_DM 0x06 50bbfe3a7aSBruno Thomsen #define PCF2127_REG_DW 0x07 51bbfe3a7aSBruno Thomsen #define PCF2127_REG_MO 0x08 52bbfe3a7aSBruno Thomsen #define PCF2127_REG_YR 0x09 538a914bacSLiam Beguin /* Alarm registers */ 548a914bacSLiam Beguin #define PCF2127_REG_ALARM_SC 0x0A 558a914bacSLiam Beguin #define PCF2127_REG_ALARM_MN 0x0B 568a914bacSLiam Beguin #define PCF2127_REG_ALARM_HR 0x0C 578a914bacSLiam Beguin #define PCF2127_REG_ALARM_DM 0x0D 588a914bacSLiam Beguin #define PCF2127_REG_ALARM_DW 0x0E 5927006416SAlexandre Belloni #define PCF2127_BIT_ALARM_AE BIT(7) 600e735eaaSBruno Thomsen /* Watchdog registers */ 610e735eaaSBruno Thomsen #define PCF2127_REG_WD_CTL 0x10 620e735eaaSBruno Thomsen #define PCF2127_BIT_WD_CTL_TF0 BIT(0) 630e735eaaSBruno Thomsen #define PCF2127_BIT_WD_CTL_TF1 BIT(1) 640e735eaaSBruno Thomsen #define PCF2127_BIT_WD_CTL_CD0 BIT(6) 650e735eaaSBruno Thomsen #define PCF2127_BIT_WD_CTL_CD1 BIT(7) 660e735eaaSBruno Thomsen #define PCF2127_REG_WD_VAL 0x11 6703623b4bSBruno Thomsen /* Tamper timestamp registers */ 6803623b4bSBruno Thomsen #define PCF2127_REG_TS_CTRL 0x12 6903623b4bSBruno Thomsen #define PCF2127_BIT_TS_CTRL_TSOFF BIT(6) 7003623b4bSBruno Thomsen #define PCF2127_BIT_TS_CTRL_TSM BIT(7) 7103623b4bSBruno Thomsen #define PCF2127_REG_TS_SC 0x13 7203623b4bSBruno Thomsen #define PCF2127_REG_TS_MN 0x14 7303623b4bSBruno Thomsen #define PCF2127_REG_TS_HR 0x15 7403623b4bSBruno Thomsen #define PCF2127_REG_TS_DM 0x16 7503623b4bSBruno Thomsen #define PCF2127_REG_TS_MO 0x17 7603623b4bSBruno Thomsen #define PCF2127_REG_TS_YR 0x18 77bbfe3a7aSBruno Thomsen /* 78bbfe3a7aSBruno Thomsen * RAM registers 79bbfe3a7aSBruno Thomsen * PCF2127 has 512 bytes general-purpose static RAM (SRAM) that is 80bbfe3a7aSBruno Thomsen * battery backed and can survive a power outage. 81bbfe3a7aSBruno Thomsen * PCF2129 doesn't have this feature. 82bbfe3a7aSBruno Thomsen */ 83bbfe3a7aSBruno Thomsen #define PCF2127_REG_RAM_ADDR_MSB 0x1A 84bbfe3a7aSBruno Thomsen #define PCF2127_REG_RAM_WRT_CMD 0x1C 85bbfe3a7aSBruno Thomsen #define PCF2127_REG_RAM_RD_CMD 0x1D 86f97cfddcSUwe Kleine-König 870e735eaaSBruno Thomsen /* Watchdog timer value constants */ 880e735eaaSBruno Thomsen #define PCF2127_WD_VAL_STOP 0 890e735eaaSBruno Thomsen #define PCF2127_WD_VAL_MIN 2 900e735eaaSBruno Thomsen #define PCF2127_WD_VAL_MAX 255 910e735eaaSBruno Thomsen #define PCF2127_WD_VAL_DEFAULT 60 92653ebd75SAndrea Scian 9318cb6368SRenaud Cerrato struct pcf2127 { 9418cb6368SRenaud Cerrato struct rtc_device *rtc; 950e735eaaSBruno Thomsen struct watchdog_device wdd; 96907b3262SAkinobu Mita struct regmap *regmap; 9718cb6368SRenaud Cerrato }; 9818cb6368SRenaud Cerrato 9918cb6368SRenaud Cerrato /* 10018cb6368SRenaud Cerrato * In the routines that deal directly with the pcf2127 hardware, we use 10118cb6368SRenaud Cerrato * rtc_time -- month 0-11, hour 0-23, yr = calendar year-epoch. 10218cb6368SRenaud Cerrato */ 103907b3262SAkinobu Mita static int pcf2127_rtc_read_time(struct device *dev, struct rtc_time *tm) 10418cb6368SRenaud Cerrato { 105907b3262SAkinobu Mita struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 106907b3262SAkinobu Mita unsigned char buf[10]; 107907b3262SAkinobu Mita int ret; 10818cb6368SRenaud Cerrato 1097f43020eSBruno Thomsen /* 1107f43020eSBruno Thomsen * Avoid reading CTRL2 register as it causes WD_VAL register 1117f43020eSBruno Thomsen * value to reset to 0 which means watchdog is stopped. 1127f43020eSBruno Thomsen */ 1137f43020eSBruno Thomsen ret = regmap_bulk_read(pcf2127->regmap, PCF2127_REG_CTRL3, 1147f43020eSBruno Thomsen (buf + PCF2127_REG_CTRL3), 1157f43020eSBruno Thomsen ARRAY_SIZE(buf) - PCF2127_REG_CTRL3); 116907b3262SAkinobu Mita if (ret) { 117907b3262SAkinobu Mita dev_err(dev, "%s: read error\n", __func__); 118907b3262SAkinobu Mita return ret; 11918cb6368SRenaud Cerrato } 12018cb6368SRenaud Cerrato 121bbfe3a7aSBruno Thomsen if (buf[PCF2127_REG_CTRL3] & PCF2127_BIT_CTRL3_BLF) 122907b3262SAkinobu Mita dev_info(dev, 123653ebd75SAndrea Scian "low voltage detected, check/replace RTC battery.\n"); 124653ebd75SAndrea Scian 125bbfe3a7aSBruno Thomsen /* Clock integrity is not guaranteed when OSF flag is set. */ 126bbfe3a7aSBruno Thomsen if (buf[PCF2127_REG_SC] & PCF2127_BIT_SC_OSF) { 127653ebd75SAndrea Scian /* 128653ebd75SAndrea Scian * no need clear the flag here, 129653ebd75SAndrea Scian * it will be cleared once the new date is saved 130653ebd75SAndrea Scian */ 131907b3262SAkinobu Mita dev_warn(dev, 132653ebd75SAndrea Scian "oscillator stop detected, date/time is not reliable\n"); 133653ebd75SAndrea Scian return -EINVAL; 13418cb6368SRenaud Cerrato } 13518cb6368SRenaud Cerrato 136907b3262SAkinobu Mita dev_dbg(dev, 1377f43020eSBruno Thomsen "%s: raw data is cr3=%02x, sec=%02x, min=%02x, hr=%02x, " 13818cb6368SRenaud Cerrato "mday=%02x, wday=%02x, mon=%02x, year=%02x\n", 1397f43020eSBruno Thomsen __func__, buf[PCF2127_REG_CTRL3], buf[PCF2127_REG_SC], 1407f43020eSBruno Thomsen buf[PCF2127_REG_MN], buf[PCF2127_REG_HR], 1417f43020eSBruno Thomsen buf[PCF2127_REG_DM], buf[PCF2127_REG_DW], 1427f43020eSBruno Thomsen buf[PCF2127_REG_MO], buf[PCF2127_REG_YR]); 14318cb6368SRenaud Cerrato 14418cb6368SRenaud Cerrato tm->tm_sec = bcd2bin(buf[PCF2127_REG_SC] & 0x7F); 14518cb6368SRenaud Cerrato tm->tm_min = bcd2bin(buf[PCF2127_REG_MN] & 0x7F); 14618cb6368SRenaud Cerrato tm->tm_hour = bcd2bin(buf[PCF2127_REG_HR] & 0x3F); /* rtc hr 0-23 */ 14718cb6368SRenaud Cerrato tm->tm_mday = bcd2bin(buf[PCF2127_REG_DM] & 0x3F); 14818cb6368SRenaud Cerrato tm->tm_wday = buf[PCF2127_REG_DW] & 0x07; 14918cb6368SRenaud Cerrato tm->tm_mon = bcd2bin(buf[PCF2127_REG_MO] & 0x1F) - 1; /* rtc mn 1-12 */ 15018cb6368SRenaud Cerrato tm->tm_year = bcd2bin(buf[PCF2127_REG_YR]); 151b139bb5cSAlexandre Belloni tm->tm_year += 100; 15218cb6368SRenaud Cerrato 153907b3262SAkinobu Mita dev_dbg(dev, "%s: tm is secs=%d, mins=%d, hours=%d, " 15418cb6368SRenaud Cerrato "mday=%d, mon=%d, year=%d, wday=%d\n", 15518cb6368SRenaud Cerrato __func__, 15618cb6368SRenaud Cerrato tm->tm_sec, tm->tm_min, tm->tm_hour, 15718cb6368SRenaud Cerrato tm->tm_mday, tm->tm_mon, tm->tm_year, tm->tm_wday); 15818cb6368SRenaud Cerrato 15922652ba7SAlexandre Belloni return 0; 16018cb6368SRenaud Cerrato } 16118cb6368SRenaud Cerrato 162907b3262SAkinobu Mita static int pcf2127_rtc_set_time(struct device *dev, struct rtc_time *tm) 16318cb6368SRenaud Cerrato { 164907b3262SAkinobu Mita struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 165907b3262SAkinobu Mita unsigned char buf[7]; 16618cb6368SRenaud Cerrato int i = 0, err; 16718cb6368SRenaud Cerrato 168907b3262SAkinobu Mita dev_dbg(dev, "%s: secs=%d, mins=%d, hours=%d, " 16918cb6368SRenaud Cerrato "mday=%d, mon=%d, year=%d, wday=%d\n", 17018cb6368SRenaud Cerrato __func__, 17118cb6368SRenaud Cerrato tm->tm_sec, tm->tm_min, tm->tm_hour, 17218cb6368SRenaud Cerrato tm->tm_mday, tm->tm_mon, tm->tm_year, tm->tm_wday); 17318cb6368SRenaud Cerrato 17418cb6368SRenaud Cerrato /* hours, minutes and seconds */ 175653ebd75SAndrea Scian buf[i++] = bin2bcd(tm->tm_sec); /* this will also clear OSF flag */ 17618cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_min); 17718cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_hour); 17818cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_mday); 17918cb6368SRenaud Cerrato buf[i++] = tm->tm_wday & 0x07; 18018cb6368SRenaud Cerrato 18118cb6368SRenaud Cerrato /* month, 1 - 12 */ 18218cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_mon + 1); 18318cb6368SRenaud Cerrato 18418cb6368SRenaud Cerrato /* year */ 185b139bb5cSAlexandre Belloni buf[i++] = bin2bcd(tm->tm_year - 100); 18618cb6368SRenaud Cerrato 18718cb6368SRenaud Cerrato /* write register's data */ 188907b3262SAkinobu Mita err = regmap_bulk_write(pcf2127->regmap, PCF2127_REG_SC, buf, i); 189907b3262SAkinobu Mita if (err) { 190907b3262SAkinobu Mita dev_err(dev, 19118cb6368SRenaud Cerrato "%s: err=%d", __func__, err); 192907b3262SAkinobu Mita return err; 19318cb6368SRenaud Cerrato } 19418cb6368SRenaud Cerrato 19518cb6368SRenaud Cerrato return 0; 19618cb6368SRenaud Cerrato } 19718cb6368SRenaud Cerrato 19818cb6368SRenaud Cerrato static int pcf2127_rtc_ioctl(struct device *dev, 19918cb6368SRenaud Cerrato unsigned int cmd, unsigned long arg) 20018cb6368SRenaud Cerrato { 201907b3262SAkinobu Mita struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 2027d65cf8cSAlexandre Belloni int val, touser = 0; 203f97cfddcSUwe Kleine-König int ret; 20418cb6368SRenaud Cerrato 20518cb6368SRenaud Cerrato switch (cmd) { 20618cb6368SRenaud Cerrato case RTC_VL_READ: 2077d65cf8cSAlexandre Belloni ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL3, &val); 208907b3262SAkinobu Mita if (ret) 209f97cfddcSUwe Kleine-König return ret; 21018cb6368SRenaud Cerrato 2117d65cf8cSAlexandre Belloni if (val & PCF2127_BIT_CTRL3_BLF) 2127d65cf8cSAlexandre Belloni touser |= RTC_VL_BACKUP_LOW; 2137d65cf8cSAlexandre Belloni 2147d65cf8cSAlexandre Belloni if (val & PCF2127_BIT_CTRL3_BF) 2157d65cf8cSAlexandre Belloni touser |= RTC_VL_BACKUP_SWITCH; 216f97cfddcSUwe Kleine-König 217af427311SAlexandre Belloni return put_user(touser, (unsigned int __user *)arg); 2187d65cf8cSAlexandre Belloni 2197d65cf8cSAlexandre Belloni case RTC_VL_CLR: 2207d65cf8cSAlexandre Belloni return regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL3, 2217d65cf8cSAlexandre Belloni PCF2127_BIT_CTRL3_BF, 0); 2227d65cf8cSAlexandre Belloni 22318cb6368SRenaud Cerrato default: 22418cb6368SRenaud Cerrato return -ENOIOCTLCMD; 22518cb6368SRenaud Cerrato } 22618cb6368SRenaud Cerrato } 22718cb6368SRenaud Cerrato 22818cb6368SRenaud Cerrato static const struct rtc_class_ops pcf2127_rtc_ops = { 22918cb6368SRenaud Cerrato .ioctl = pcf2127_rtc_ioctl, 23018cb6368SRenaud Cerrato .read_time = pcf2127_rtc_read_time, 23118cb6368SRenaud Cerrato .set_time = pcf2127_rtc_set_time, 23218cb6368SRenaud Cerrato }; 23318cb6368SRenaud Cerrato 234d6c3029fSUwe Kleine-König static int pcf2127_nvmem_read(void *priv, unsigned int offset, 235d6c3029fSUwe Kleine-König void *val, size_t bytes) 236d6c3029fSUwe Kleine-König { 237d6c3029fSUwe Kleine-König struct pcf2127 *pcf2127 = priv; 238d6c3029fSUwe Kleine-König int ret; 239d6c3029fSUwe Kleine-König unsigned char offsetbuf[] = { offset >> 8, offset }; 240d6c3029fSUwe Kleine-König 241bbfe3a7aSBruno Thomsen ret = regmap_bulk_write(pcf2127->regmap, PCF2127_REG_RAM_ADDR_MSB, 242d6c3029fSUwe Kleine-König offsetbuf, 2); 243d6c3029fSUwe Kleine-König if (ret) 244d6c3029fSUwe Kleine-König return ret; 245d6c3029fSUwe Kleine-König 246bbfe3a7aSBruno Thomsen ret = regmap_bulk_read(pcf2127->regmap, PCF2127_REG_RAM_RD_CMD, 247d6c3029fSUwe Kleine-König val, bytes); 248d6c3029fSUwe Kleine-König 249d6c3029fSUwe Kleine-König return ret ?: bytes; 250d6c3029fSUwe Kleine-König } 251d6c3029fSUwe Kleine-König 252d6c3029fSUwe Kleine-König static int pcf2127_nvmem_write(void *priv, unsigned int offset, 253d6c3029fSUwe Kleine-König void *val, size_t bytes) 254d6c3029fSUwe Kleine-König { 255d6c3029fSUwe Kleine-König struct pcf2127 *pcf2127 = priv; 256d6c3029fSUwe Kleine-König int ret; 257d6c3029fSUwe Kleine-König unsigned char offsetbuf[] = { offset >> 8, offset }; 258d6c3029fSUwe Kleine-König 259bbfe3a7aSBruno Thomsen ret = regmap_bulk_write(pcf2127->regmap, PCF2127_REG_RAM_ADDR_MSB, 260d6c3029fSUwe Kleine-König offsetbuf, 2); 261d6c3029fSUwe Kleine-König if (ret) 262d6c3029fSUwe Kleine-König return ret; 263d6c3029fSUwe Kleine-König 264bbfe3a7aSBruno Thomsen ret = regmap_bulk_write(pcf2127->regmap, PCF2127_REG_RAM_WRT_CMD, 265d6c3029fSUwe Kleine-König val, bytes); 266d6c3029fSUwe Kleine-König 267d6c3029fSUwe Kleine-König return ret ?: bytes; 268d6c3029fSUwe Kleine-König } 269d6c3029fSUwe Kleine-König 2700e735eaaSBruno Thomsen /* watchdog driver */ 2710e735eaaSBruno Thomsen 2720e735eaaSBruno Thomsen static int pcf2127_wdt_ping(struct watchdog_device *wdd) 2730e735eaaSBruno Thomsen { 2740e735eaaSBruno Thomsen struct pcf2127 *pcf2127 = watchdog_get_drvdata(wdd); 2750e735eaaSBruno Thomsen 2760e735eaaSBruno Thomsen return regmap_write(pcf2127->regmap, PCF2127_REG_WD_VAL, wdd->timeout); 2770e735eaaSBruno Thomsen } 2780e735eaaSBruno Thomsen 2790e735eaaSBruno Thomsen /* 2800e735eaaSBruno Thomsen * Restart watchdog timer if feature is active. 2810e735eaaSBruno Thomsen * 2820e735eaaSBruno Thomsen * Note: Reading CTRL2 register causes watchdog to stop which is unfortunate, 2830e735eaaSBruno Thomsen * since register also contain control/status flags for other features. 2840e735eaaSBruno Thomsen * Always call this function after reading CTRL2 register. 2850e735eaaSBruno Thomsen */ 2860e735eaaSBruno Thomsen static int pcf2127_wdt_active_ping(struct watchdog_device *wdd) 2870e735eaaSBruno Thomsen { 2880e735eaaSBruno Thomsen int ret = 0; 2890e735eaaSBruno Thomsen 2900e735eaaSBruno Thomsen if (watchdog_active(wdd)) { 2910e735eaaSBruno Thomsen ret = pcf2127_wdt_ping(wdd); 2920e735eaaSBruno Thomsen if (ret) 2930e735eaaSBruno Thomsen dev_err(wdd->parent, 2940e735eaaSBruno Thomsen "%s: watchdog restart failed, ret=%d\n", 2950e735eaaSBruno Thomsen __func__, ret); 2960e735eaaSBruno Thomsen } 2970e735eaaSBruno Thomsen 2980e735eaaSBruno Thomsen return ret; 2990e735eaaSBruno Thomsen } 3000e735eaaSBruno Thomsen 3010e735eaaSBruno Thomsen static int pcf2127_wdt_start(struct watchdog_device *wdd) 3020e735eaaSBruno Thomsen { 3030e735eaaSBruno Thomsen return pcf2127_wdt_ping(wdd); 3040e735eaaSBruno Thomsen } 3050e735eaaSBruno Thomsen 3060e735eaaSBruno Thomsen static int pcf2127_wdt_stop(struct watchdog_device *wdd) 3070e735eaaSBruno Thomsen { 3080e735eaaSBruno Thomsen struct pcf2127 *pcf2127 = watchdog_get_drvdata(wdd); 3090e735eaaSBruno Thomsen 3100e735eaaSBruno Thomsen return regmap_write(pcf2127->regmap, PCF2127_REG_WD_VAL, 3110e735eaaSBruno Thomsen PCF2127_WD_VAL_STOP); 3120e735eaaSBruno Thomsen } 3130e735eaaSBruno Thomsen 3140e735eaaSBruno Thomsen static int pcf2127_wdt_set_timeout(struct watchdog_device *wdd, 3150e735eaaSBruno Thomsen unsigned int new_timeout) 3160e735eaaSBruno Thomsen { 3170e735eaaSBruno Thomsen dev_dbg(wdd->parent, "new watchdog timeout: %is (old: %is)\n", 3180e735eaaSBruno Thomsen new_timeout, wdd->timeout); 3190e735eaaSBruno Thomsen 3200e735eaaSBruno Thomsen wdd->timeout = new_timeout; 3210e735eaaSBruno Thomsen 3220e735eaaSBruno Thomsen return pcf2127_wdt_active_ping(wdd); 3230e735eaaSBruno Thomsen } 3240e735eaaSBruno Thomsen 3250e735eaaSBruno Thomsen static const struct watchdog_info pcf2127_wdt_info = { 3260e735eaaSBruno Thomsen .identity = "NXP PCF2127/PCF2129 Watchdog", 3270e735eaaSBruno Thomsen .options = WDIOF_KEEPALIVEPING | WDIOF_SETTIMEOUT, 3280e735eaaSBruno Thomsen }; 3290e735eaaSBruno Thomsen 3300e735eaaSBruno Thomsen static const struct watchdog_ops pcf2127_watchdog_ops = { 3310e735eaaSBruno Thomsen .owner = THIS_MODULE, 3320e735eaaSBruno Thomsen .start = pcf2127_wdt_start, 3330e735eaaSBruno Thomsen .stop = pcf2127_wdt_stop, 3340e735eaaSBruno Thomsen .ping = pcf2127_wdt_ping, 3350e735eaaSBruno Thomsen .set_timeout = pcf2127_wdt_set_timeout, 3360e735eaaSBruno Thomsen }; 3370e735eaaSBruno Thomsen 338*5d78533aSUwe Kleine-König static int pcf2127_watchdog_init(struct device *dev, struct pcf2127 *pcf2127) 339*5d78533aSUwe Kleine-König { 340*5d78533aSUwe Kleine-König u32 wdd_timeout; 341*5d78533aSUwe Kleine-König int ret; 342*5d78533aSUwe Kleine-König 343*5d78533aSUwe Kleine-König if (!IS_ENABLED(CONFIG_WATCHDOG)) 344*5d78533aSUwe Kleine-König return 0; 345*5d78533aSUwe Kleine-König 346*5d78533aSUwe Kleine-König pcf2127->wdd.parent = dev; 347*5d78533aSUwe Kleine-König pcf2127->wdd.info = &pcf2127_wdt_info; 348*5d78533aSUwe Kleine-König pcf2127->wdd.ops = &pcf2127_watchdog_ops; 349*5d78533aSUwe Kleine-König pcf2127->wdd.min_timeout = PCF2127_WD_VAL_MIN; 350*5d78533aSUwe Kleine-König pcf2127->wdd.max_timeout = PCF2127_WD_VAL_MAX; 351*5d78533aSUwe Kleine-König pcf2127->wdd.timeout = PCF2127_WD_VAL_DEFAULT; 352*5d78533aSUwe Kleine-König pcf2127->wdd.min_hw_heartbeat_ms = 500; 353*5d78533aSUwe Kleine-König pcf2127->wdd.status = WATCHDOG_NOWAYOUT_INIT_STATUS; 354*5d78533aSUwe Kleine-König 355*5d78533aSUwe Kleine-König watchdog_set_drvdata(&pcf2127->wdd, pcf2127); 356*5d78533aSUwe Kleine-König 357*5d78533aSUwe Kleine-König /* Test if watchdog timer is started by bootloader */ 358*5d78533aSUwe Kleine-König ret = regmap_read(pcf2127->regmap, PCF2127_REG_WD_VAL, &wdd_timeout); 359*5d78533aSUwe Kleine-König if (ret) 360*5d78533aSUwe Kleine-König return ret; 361*5d78533aSUwe Kleine-König 362*5d78533aSUwe Kleine-König if (wdd_timeout) 363*5d78533aSUwe Kleine-König set_bit(WDOG_HW_RUNNING, &pcf2127->wdd.status); 364*5d78533aSUwe Kleine-König 365*5d78533aSUwe Kleine-König return devm_watchdog_register_device(dev, &pcf2127->wdd); 366*5d78533aSUwe Kleine-König } 367*5d78533aSUwe 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); 3728a914bacSLiam Beguin unsigned int buf[5], ctrl2; 3738a914bacSLiam Beguin int ret; 3748a914bacSLiam Beguin 3758a914bacSLiam Beguin ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL2, &ctrl2); 3768a914bacSLiam Beguin if (ret) 3778a914bacSLiam Beguin return ret; 3788a914bacSLiam Beguin 3798a914bacSLiam Beguin ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 3808a914bacSLiam Beguin if (ret) 3818a914bacSLiam Beguin return ret; 3828a914bacSLiam Beguin 3838a914bacSLiam Beguin ret = regmap_bulk_read(pcf2127->regmap, PCF2127_REG_ALARM_SC, buf, 3848a914bacSLiam Beguin sizeof(buf)); 3858a914bacSLiam Beguin if (ret) 3868a914bacSLiam Beguin return ret; 3878a914bacSLiam Beguin 3888a914bacSLiam Beguin alrm->enabled = ctrl2 & PCF2127_BIT_CTRL2_AIE; 3898a914bacSLiam Beguin alrm->pending = ctrl2 & PCF2127_BIT_CTRL2_AF; 3908a914bacSLiam Beguin 3918a914bacSLiam Beguin alrm->time.tm_sec = bcd2bin(buf[0] & 0x7F); 3928a914bacSLiam Beguin alrm->time.tm_min = bcd2bin(buf[1] & 0x7F); 3938a914bacSLiam Beguin alrm->time.tm_hour = bcd2bin(buf[2] & 0x3F); 3948a914bacSLiam Beguin alrm->time.tm_mday = bcd2bin(buf[3] & 0x3F); 3958a914bacSLiam Beguin 3968a914bacSLiam Beguin return 0; 3978a914bacSLiam Beguin } 3988a914bacSLiam Beguin 3998a914bacSLiam Beguin static int pcf2127_rtc_alarm_irq_enable(struct device *dev, u32 enable) 4008a914bacSLiam Beguin { 4018a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 4028a914bacSLiam Beguin int ret; 4038a914bacSLiam Beguin 4048a914bacSLiam Beguin ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL2, 4058a914bacSLiam Beguin PCF2127_BIT_CTRL2_AIE, 4068a914bacSLiam Beguin enable ? PCF2127_BIT_CTRL2_AIE : 0); 4078a914bacSLiam Beguin if (ret) 4088a914bacSLiam Beguin return ret; 4098a914bacSLiam Beguin 4108a914bacSLiam Beguin return pcf2127_wdt_active_ping(&pcf2127->wdd); 4118a914bacSLiam Beguin } 4128a914bacSLiam Beguin 4138a914bacSLiam Beguin static int pcf2127_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alrm) 4148a914bacSLiam Beguin { 4158a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 4168a914bacSLiam Beguin uint8_t buf[5]; 4178a914bacSLiam Beguin int ret; 4188a914bacSLiam Beguin 4198a914bacSLiam Beguin ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL2, 4208a914bacSLiam Beguin PCF2127_BIT_CTRL2_AF, 0); 4218a914bacSLiam Beguin if (ret) 4228a914bacSLiam Beguin return ret; 4238a914bacSLiam Beguin 4248a914bacSLiam Beguin ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 4258a914bacSLiam Beguin if (ret) 4268a914bacSLiam Beguin return ret; 4278a914bacSLiam Beguin 4288a914bacSLiam Beguin buf[0] = bin2bcd(alrm->time.tm_sec); 4298a914bacSLiam Beguin buf[1] = bin2bcd(alrm->time.tm_min); 4308a914bacSLiam Beguin buf[2] = bin2bcd(alrm->time.tm_hour); 4318a914bacSLiam Beguin buf[3] = bin2bcd(alrm->time.tm_mday); 43227006416SAlexandre Belloni buf[4] = PCF2127_BIT_ALARM_AE; /* Do not match on week day */ 4338a914bacSLiam Beguin 4348a914bacSLiam Beguin ret = regmap_bulk_write(pcf2127->regmap, PCF2127_REG_ALARM_SC, buf, 4358a914bacSLiam Beguin sizeof(buf)); 4368a914bacSLiam Beguin if (ret) 4378a914bacSLiam Beguin return ret; 4388a914bacSLiam Beguin 4398a914bacSLiam Beguin return pcf2127_rtc_alarm_irq_enable(dev, alrm->enabled); 4408a914bacSLiam Beguin } 4418a914bacSLiam Beguin 4428a914bacSLiam Beguin static irqreturn_t pcf2127_rtc_irq(int irq, void *dev) 4438a914bacSLiam Beguin { 4448a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 4458a914bacSLiam Beguin unsigned int ctrl2 = 0; 4468a914bacSLiam Beguin int ret = 0; 4478a914bacSLiam Beguin 4488a914bacSLiam Beguin ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL2, &ctrl2); 4498a914bacSLiam Beguin if (ret) 4508a914bacSLiam Beguin return IRQ_NONE; 4518a914bacSLiam Beguin 45227006416SAlexandre Belloni if (!(ctrl2 & PCF2127_BIT_CTRL2_AF)) 45327006416SAlexandre Belloni return IRQ_NONE; 45427006416SAlexandre Belloni 4558a914bacSLiam Beguin regmap_write(pcf2127->regmap, PCF2127_REG_CTRL2, 45627006416SAlexandre Belloni ctrl2 & ~(PCF2127_BIT_CTRL2_AF | PCF2127_BIT_CTRL2_WDTF)); 4578a914bacSLiam Beguin 4588a914bacSLiam Beguin rtc_update_irq(pcf2127->rtc, 1, RTC_IRQF | RTC_AF); 4598a914bacSLiam Beguin 46027006416SAlexandre Belloni pcf2127_wdt_active_ping(&pcf2127->wdd); 4618a914bacSLiam Beguin 4628a914bacSLiam Beguin return IRQ_HANDLED; 4638a914bacSLiam Beguin } 4648a914bacSLiam Beguin 4658a914bacSLiam Beguin static const struct rtc_class_ops pcf2127_rtc_alrm_ops = { 4668a914bacSLiam Beguin .ioctl = pcf2127_rtc_ioctl, 4678a914bacSLiam Beguin .read_time = pcf2127_rtc_read_time, 4688a914bacSLiam Beguin .set_time = pcf2127_rtc_set_time, 4698a914bacSLiam Beguin .read_alarm = pcf2127_rtc_read_alarm, 4708a914bacSLiam Beguin .set_alarm = pcf2127_rtc_set_alarm, 4718a914bacSLiam Beguin .alarm_irq_enable = pcf2127_rtc_alarm_irq_enable, 4728a914bacSLiam Beguin }; 4738a914bacSLiam Beguin 47403623b4bSBruno Thomsen /* sysfs interface */ 47503623b4bSBruno Thomsen 47603623b4bSBruno Thomsen static ssize_t timestamp0_store(struct device *dev, 47703623b4bSBruno Thomsen struct device_attribute *attr, 47803623b4bSBruno Thomsen const char *buf, size_t count) 47903623b4bSBruno Thomsen { 48003623b4bSBruno Thomsen struct pcf2127 *pcf2127 = dev_get_drvdata(dev->parent); 48103623b4bSBruno Thomsen int ret; 48203623b4bSBruno Thomsen 48303623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL1, 48403623b4bSBruno Thomsen PCF2127_BIT_CTRL1_TSF1, 0); 48503623b4bSBruno Thomsen if (ret) { 48603623b4bSBruno Thomsen dev_err(dev, "%s: update ctrl1 ret=%d\n", __func__, ret); 48703623b4bSBruno Thomsen return ret; 48803623b4bSBruno Thomsen } 48903623b4bSBruno Thomsen 49003623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL2, 49103623b4bSBruno Thomsen PCF2127_BIT_CTRL2_TSF2, 0); 49203623b4bSBruno Thomsen if (ret) { 49303623b4bSBruno Thomsen dev_err(dev, "%s: update ctrl2 ret=%d\n", __func__, ret); 49403623b4bSBruno Thomsen return ret; 49503623b4bSBruno Thomsen } 49603623b4bSBruno Thomsen 49703623b4bSBruno Thomsen ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 49803623b4bSBruno Thomsen if (ret) 49903623b4bSBruno Thomsen return ret; 50003623b4bSBruno Thomsen 50103623b4bSBruno Thomsen return count; 50203623b4bSBruno Thomsen }; 50303623b4bSBruno Thomsen 50403623b4bSBruno Thomsen static ssize_t timestamp0_show(struct device *dev, 50503623b4bSBruno Thomsen struct device_attribute *attr, char *buf) 50603623b4bSBruno Thomsen { 50703623b4bSBruno Thomsen struct pcf2127 *pcf2127 = dev_get_drvdata(dev->parent); 50803623b4bSBruno Thomsen struct rtc_time tm; 50903623b4bSBruno Thomsen int ret; 51003623b4bSBruno Thomsen unsigned char data[25]; 51103623b4bSBruno Thomsen 51203623b4bSBruno Thomsen ret = regmap_bulk_read(pcf2127->regmap, PCF2127_REG_CTRL1, data, 51303623b4bSBruno Thomsen sizeof(data)); 51403623b4bSBruno Thomsen if (ret) { 51503623b4bSBruno Thomsen dev_err(dev, "%s: read error ret=%d\n", __func__, ret); 51603623b4bSBruno Thomsen return ret; 51703623b4bSBruno Thomsen } 51803623b4bSBruno Thomsen 51903623b4bSBruno Thomsen dev_dbg(dev, 52003623b4bSBruno Thomsen "%s: raw data is cr1=%02x, cr2=%02x, cr3=%02x, ts_sc=%02x, " 52103623b4bSBruno Thomsen "ts_mn=%02x, ts_hr=%02x, ts_dm=%02x, ts_mo=%02x, ts_yr=%02x\n", 52203623b4bSBruno Thomsen __func__, data[PCF2127_REG_CTRL1], data[PCF2127_REG_CTRL2], 52303623b4bSBruno Thomsen data[PCF2127_REG_CTRL3], data[PCF2127_REG_TS_SC], 52403623b4bSBruno Thomsen data[PCF2127_REG_TS_MN], data[PCF2127_REG_TS_HR], 52503623b4bSBruno Thomsen data[PCF2127_REG_TS_DM], data[PCF2127_REG_TS_MO], 52603623b4bSBruno Thomsen data[PCF2127_REG_TS_YR]); 52703623b4bSBruno Thomsen 52803623b4bSBruno Thomsen ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 52903623b4bSBruno Thomsen if (ret) 53003623b4bSBruno Thomsen return ret; 53103623b4bSBruno Thomsen 53203623b4bSBruno Thomsen if (!(data[PCF2127_REG_CTRL1] & PCF2127_BIT_CTRL1_TSF1) && 53303623b4bSBruno Thomsen !(data[PCF2127_REG_CTRL2] & PCF2127_BIT_CTRL2_TSF2)) 53403623b4bSBruno Thomsen return 0; 53503623b4bSBruno Thomsen 53603623b4bSBruno Thomsen tm.tm_sec = bcd2bin(data[PCF2127_REG_TS_SC] & 0x7F); 53703623b4bSBruno Thomsen tm.tm_min = bcd2bin(data[PCF2127_REG_TS_MN] & 0x7F); 53803623b4bSBruno Thomsen tm.tm_hour = bcd2bin(data[PCF2127_REG_TS_HR] & 0x3F); 53903623b4bSBruno Thomsen tm.tm_mday = bcd2bin(data[PCF2127_REG_TS_DM] & 0x3F); 54003623b4bSBruno Thomsen /* TS_MO register (month) value range: 1-12 */ 54103623b4bSBruno Thomsen tm.tm_mon = bcd2bin(data[PCF2127_REG_TS_MO] & 0x1F) - 1; 54203623b4bSBruno Thomsen tm.tm_year = bcd2bin(data[PCF2127_REG_TS_YR]); 54303623b4bSBruno Thomsen if (tm.tm_year < 70) 54403623b4bSBruno Thomsen tm.tm_year += 100; /* assume we are in 1970...2069 */ 54503623b4bSBruno Thomsen 54603623b4bSBruno Thomsen ret = rtc_valid_tm(&tm); 54703623b4bSBruno Thomsen if (ret) 54803623b4bSBruno Thomsen return ret; 54903623b4bSBruno Thomsen 55003623b4bSBruno Thomsen return sprintf(buf, "%llu\n", 55103623b4bSBruno Thomsen (unsigned long long)rtc_tm_to_time64(&tm)); 55203623b4bSBruno Thomsen }; 55303623b4bSBruno Thomsen 55403623b4bSBruno Thomsen static DEVICE_ATTR_RW(timestamp0); 55503623b4bSBruno Thomsen 55603623b4bSBruno Thomsen static struct attribute *pcf2127_attrs[] = { 55703623b4bSBruno Thomsen &dev_attr_timestamp0.attr, 55803623b4bSBruno Thomsen NULL 55903623b4bSBruno Thomsen }; 56003623b4bSBruno Thomsen 56103623b4bSBruno Thomsen static const struct attribute_group pcf2127_attr_group = { 56203623b4bSBruno Thomsen .attrs = pcf2127_attrs, 56303623b4bSBruno Thomsen }; 56403623b4bSBruno Thomsen 565907b3262SAkinobu Mita static int pcf2127_probe(struct device *dev, struct regmap *regmap, 56627006416SAlexandre Belloni int alarm_irq, const char *name, bool has_nvmem) 56718cb6368SRenaud Cerrato { 56818cb6368SRenaud Cerrato struct pcf2127 *pcf2127; 569d6c3029fSUwe Kleine-König int ret = 0; 57018cb6368SRenaud Cerrato 571907b3262SAkinobu Mita dev_dbg(dev, "%s\n", __func__); 57218cb6368SRenaud Cerrato 573907b3262SAkinobu Mita pcf2127 = devm_kzalloc(dev, sizeof(*pcf2127), GFP_KERNEL); 57418cb6368SRenaud Cerrato if (!pcf2127) 57518cb6368SRenaud Cerrato return -ENOMEM; 57618cb6368SRenaud Cerrato 577907b3262SAkinobu Mita pcf2127->regmap = regmap; 57818cb6368SRenaud Cerrato 579907b3262SAkinobu Mita dev_set_drvdata(dev, pcf2127); 580907b3262SAkinobu Mita 581e788771cSBruno Thomsen pcf2127->rtc = devm_rtc_allocate_device(dev); 582d6c3029fSUwe Kleine-König if (IS_ERR(pcf2127->rtc)) 583d6c3029fSUwe Kleine-König return PTR_ERR(pcf2127->rtc); 58418cb6368SRenaud Cerrato 585e788771cSBruno Thomsen pcf2127->rtc->ops = &pcf2127_rtc_ops; 586b139bb5cSAlexandre Belloni pcf2127->rtc->range_min = RTC_TIMESTAMP_BEGIN_2000; 587b139bb5cSAlexandre Belloni pcf2127->rtc->range_max = RTC_TIMESTAMP_END_2099; 588b139bb5cSAlexandre Belloni pcf2127->rtc->set_start_time = true; /* Sets actual start to 1970 */ 58927006416SAlexandre Belloni pcf2127->rtc->uie_unsupported = 1; 590e788771cSBruno Thomsen 59135425bafSBiwen Li if (alarm_irq > 0) { 59227006416SAlexandre Belloni ret = devm_request_threaded_irq(dev, alarm_irq, NULL, 59327006416SAlexandre Belloni pcf2127_rtc_irq, 5948a914bacSLiam Beguin IRQF_TRIGGER_LOW | IRQF_ONESHOT, 5958a914bacSLiam Beguin dev_name(dev), dev); 5968a914bacSLiam Beguin if (ret) { 5978a914bacSLiam Beguin dev_err(dev, "failed to request alarm irq\n"); 5988a914bacSLiam Beguin return ret; 5998a914bacSLiam Beguin } 6008a914bacSLiam Beguin } 6018a914bacSLiam Beguin 60235425bafSBiwen Li if (alarm_irq > 0 || device_property_read_bool(dev, "wakeup-source")) { 6038a914bacSLiam Beguin device_init_wakeup(dev, true); 6048a914bacSLiam Beguin pcf2127->rtc->ops = &pcf2127_rtc_alrm_ops; 6058a914bacSLiam Beguin } 6068a914bacSLiam Beguin 607d6c3029fSUwe Kleine-König if (has_nvmem) { 608d6c3029fSUwe Kleine-König struct nvmem_config nvmem_cfg = { 609d6c3029fSUwe Kleine-König .priv = pcf2127, 610d6c3029fSUwe Kleine-König .reg_read = pcf2127_nvmem_read, 611d6c3029fSUwe Kleine-König .reg_write = pcf2127_nvmem_write, 612d6c3029fSUwe Kleine-König .size = 512, 613d6c3029fSUwe Kleine-König }; 614d6c3029fSUwe Kleine-König 615d6c3029fSUwe Kleine-König ret = rtc_nvmem_register(pcf2127->rtc, &nvmem_cfg); 616d6c3029fSUwe Kleine-König } 617d6c3029fSUwe Kleine-König 6180e735eaaSBruno Thomsen /* 6190e735eaaSBruno Thomsen * Watchdog timer enabled and reset pin /RST activated when timed out. 6200e735eaaSBruno Thomsen * Select 1Hz clock source for watchdog timer. 6210e735eaaSBruno Thomsen * Note: Countdown timer disabled and not available. 6220e735eaaSBruno Thomsen */ 6230e735eaaSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_WD_CTL, 6240e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_CD1 | 6250e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_CD0 | 6260e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_TF1 | 6270e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_TF0, 6280e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_CD1 | 6290e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_CD0 | 6300e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_TF1); 6310e735eaaSBruno Thomsen if (ret) { 6320e735eaaSBruno Thomsen dev_err(dev, "%s: watchdog config (wd_ctl) failed\n", __func__); 6330e735eaaSBruno Thomsen return ret; 6340e735eaaSBruno Thomsen } 6350e735eaaSBruno Thomsen 636*5d78533aSUwe Kleine-König pcf2127_watchdog_init(dev, pcf2127); 6370e735eaaSBruno Thomsen 63803623b4bSBruno Thomsen /* 63903623b4bSBruno Thomsen * Disable battery low/switch-over timestamp and interrupts. 64003623b4bSBruno Thomsen * Clear battery interrupt flags which can block new trigger events. 64103623b4bSBruno Thomsen * Note: This is the default chip behaviour but added to ensure 64203623b4bSBruno Thomsen * correct tamper timestamp and interrupt function. 64303623b4bSBruno Thomsen */ 64403623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL3, 64503623b4bSBruno Thomsen PCF2127_BIT_CTRL3_BTSE | 64603623b4bSBruno Thomsen PCF2127_BIT_CTRL3_BIE | 64703623b4bSBruno Thomsen PCF2127_BIT_CTRL3_BLIE, 0); 64803623b4bSBruno Thomsen if (ret) { 64903623b4bSBruno Thomsen dev_err(dev, "%s: interrupt config (ctrl3) failed\n", 65003623b4bSBruno Thomsen __func__); 65103623b4bSBruno Thomsen return ret; 65203623b4bSBruno Thomsen } 65303623b4bSBruno Thomsen 65403623b4bSBruno Thomsen /* 65503623b4bSBruno Thomsen * Enable timestamp function and store timestamp of first trigger 65603623b4bSBruno Thomsen * event until TSF1 and TFS2 interrupt flags are cleared. 65703623b4bSBruno Thomsen */ 65803623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_TS_CTRL, 65903623b4bSBruno Thomsen PCF2127_BIT_TS_CTRL_TSOFF | 66003623b4bSBruno Thomsen PCF2127_BIT_TS_CTRL_TSM, 66103623b4bSBruno Thomsen PCF2127_BIT_TS_CTRL_TSM); 66203623b4bSBruno Thomsen if (ret) { 66303623b4bSBruno Thomsen dev_err(dev, "%s: tamper detection config (ts_ctrl) failed\n", 66403623b4bSBruno Thomsen __func__); 66503623b4bSBruno Thomsen return ret; 66603623b4bSBruno Thomsen } 66703623b4bSBruno Thomsen 66803623b4bSBruno Thomsen /* 66903623b4bSBruno Thomsen * Enable interrupt generation when TSF1 or TSF2 timestamp flags 67003623b4bSBruno Thomsen * are set. Interrupt signal is an open-drain output and can be 67103623b4bSBruno Thomsen * left floating if unused. 67203623b4bSBruno Thomsen */ 67303623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL2, 67403623b4bSBruno Thomsen PCF2127_BIT_CTRL2_TSIE, 67503623b4bSBruno Thomsen PCF2127_BIT_CTRL2_TSIE); 67603623b4bSBruno Thomsen if (ret) { 67703623b4bSBruno Thomsen dev_err(dev, "%s: tamper detection config (ctrl2) failed\n", 67803623b4bSBruno Thomsen __func__); 67903623b4bSBruno Thomsen return ret; 68003623b4bSBruno Thomsen } 68103623b4bSBruno Thomsen 68203623b4bSBruno Thomsen ret = rtc_add_group(pcf2127->rtc, &pcf2127_attr_group); 68303623b4bSBruno Thomsen if (ret) { 68403623b4bSBruno Thomsen dev_err(dev, "%s: tamper sysfs registering failed\n", 68503623b4bSBruno Thomsen __func__); 68603623b4bSBruno Thomsen return ret; 68703623b4bSBruno Thomsen } 68803623b4bSBruno Thomsen 689e788771cSBruno Thomsen return rtc_register_device(pcf2127->rtc); 69018cb6368SRenaud Cerrato } 69118cb6368SRenaud Cerrato 69218cb6368SRenaud Cerrato #ifdef CONFIG_OF 69318cb6368SRenaud Cerrato static const struct of_device_id pcf2127_of_match[] = { 69418cb6368SRenaud Cerrato { .compatible = "nxp,pcf2127" }, 695cee2cc21SAkinobu Mita { .compatible = "nxp,pcf2129" }, 696985b30dbSLiam Beguin { .compatible = "nxp,pca2129" }, 69718cb6368SRenaud Cerrato {} 69818cb6368SRenaud Cerrato }; 69918cb6368SRenaud Cerrato MODULE_DEVICE_TABLE(of, pcf2127_of_match); 70018cb6368SRenaud Cerrato #endif 70118cb6368SRenaud Cerrato 7029408ec1aSAkinobu Mita #if IS_ENABLED(CONFIG_I2C) 7039408ec1aSAkinobu Mita 704907b3262SAkinobu Mita static int pcf2127_i2c_write(void *context, const void *data, size_t count) 705907b3262SAkinobu Mita { 706907b3262SAkinobu Mita struct device *dev = context; 707907b3262SAkinobu Mita struct i2c_client *client = to_i2c_client(dev); 708907b3262SAkinobu Mita int ret; 709907b3262SAkinobu Mita 710907b3262SAkinobu Mita ret = i2c_master_send(client, data, count); 711907b3262SAkinobu Mita if (ret != count) 712907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 713907b3262SAkinobu Mita 714907b3262SAkinobu Mita return 0; 715907b3262SAkinobu Mita } 716907b3262SAkinobu Mita 717907b3262SAkinobu Mita static int pcf2127_i2c_gather_write(void *context, 718907b3262SAkinobu Mita const void *reg, size_t reg_size, 719907b3262SAkinobu Mita const void *val, size_t val_size) 720907b3262SAkinobu Mita { 721907b3262SAkinobu Mita struct device *dev = context; 722907b3262SAkinobu Mita struct i2c_client *client = to_i2c_client(dev); 723907b3262SAkinobu Mita int ret; 724907b3262SAkinobu Mita void *buf; 725907b3262SAkinobu Mita 726907b3262SAkinobu Mita if (WARN_ON(reg_size != 1)) 727907b3262SAkinobu Mita return -EINVAL; 728907b3262SAkinobu Mita 729907b3262SAkinobu Mita buf = kmalloc(val_size + 1, GFP_KERNEL); 730907b3262SAkinobu Mita if (!buf) 731907b3262SAkinobu Mita return -ENOMEM; 732907b3262SAkinobu Mita 733907b3262SAkinobu Mita memcpy(buf, reg, 1); 734907b3262SAkinobu Mita memcpy(buf + 1, val, val_size); 735907b3262SAkinobu Mita 736907b3262SAkinobu Mita ret = i2c_master_send(client, buf, val_size + 1); 7379bde0afbSXulin Sun 7389bde0afbSXulin Sun kfree(buf); 7399bde0afbSXulin Sun 740907b3262SAkinobu Mita if (ret != val_size + 1) 741907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 742907b3262SAkinobu Mita 743907b3262SAkinobu Mita return 0; 744907b3262SAkinobu Mita } 745907b3262SAkinobu Mita 746907b3262SAkinobu Mita static int pcf2127_i2c_read(void *context, const void *reg, size_t reg_size, 747907b3262SAkinobu Mita void *val, size_t val_size) 748907b3262SAkinobu Mita { 749907b3262SAkinobu Mita struct device *dev = context; 750907b3262SAkinobu Mita struct i2c_client *client = to_i2c_client(dev); 751907b3262SAkinobu Mita int ret; 752907b3262SAkinobu Mita 753907b3262SAkinobu Mita if (WARN_ON(reg_size != 1)) 754907b3262SAkinobu Mita return -EINVAL; 755907b3262SAkinobu Mita 756907b3262SAkinobu Mita ret = i2c_master_send(client, reg, 1); 757907b3262SAkinobu Mita if (ret != 1) 758907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 759907b3262SAkinobu Mita 760907b3262SAkinobu Mita ret = i2c_master_recv(client, val, val_size); 761907b3262SAkinobu Mita if (ret != val_size) 762907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 763907b3262SAkinobu Mita 764907b3262SAkinobu Mita return 0; 765907b3262SAkinobu Mita } 766907b3262SAkinobu Mita 767907b3262SAkinobu Mita /* 768907b3262SAkinobu Mita * The reason we need this custom regmap_bus instead of using regmap_init_i2c() 769907b3262SAkinobu Mita * is that the STOP condition is required between set register address and 770907b3262SAkinobu Mita * read register data when reading from registers. 771907b3262SAkinobu Mita */ 772907b3262SAkinobu Mita static const struct regmap_bus pcf2127_i2c_regmap = { 773907b3262SAkinobu Mita .write = pcf2127_i2c_write, 774907b3262SAkinobu Mita .gather_write = pcf2127_i2c_gather_write, 775907b3262SAkinobu Mita .read = pcf2127_i2c_read, 77618cb6368SRenaud Cerrato }; 77718cb6368SRenaud Cerrato 778907b3262SAkinobu Mita static struct i2c_driver pcf2127_i2c_driver; 779907b3262SAkinobu Mita 780907b3262SAkinobu Mita static int pcf2127_i2c_probe(struct i2c_client *client, 781907b3262SAkinobu Mita const struct i2c_device_id *id) 782907b3262SAkinobu Mita { 783907b3262SAkinobu Mita struct regmap *regmap; 784907b3262SAkinobu Mita static const struct regmap_config config = { 785907b3262SAkinobu Mita .reg_bits = 8, 786907b3262SAkinobu Mita .val_bits = 8, 787040e6dc0SAlexandre Belloni .max_register = 0x1d, 788907b3262SAkinobu Mita }; 789907b3262SAkinobu Mita 790907b3262SAkinobu Mita if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) 791907b3262SAkinobu Mita return -ENODEV; 792907b3262SAkinobu Mita 793907b3262SAkinobu Mita regmap = devm_regmap_init(&client->dev, &pcf2127_i2c_regmap, 794907b3262SAkinobu Mita &client->dev, &config); 795907b3262SAkinobu Mita if (IS_ERR(regmap)) { 796907b3262SAkinobu Mita dev_err(&client->dev, "%s: regmap allocation failed: %ld\n", 797907b3262SAkinobu Mita __func__, PTR_ERR(regmap)); 798907b3262SAkinobu Mita return PTR_ERR(regmap); 799907b3262SAkinobu Mita } 800907b3262SAkinobu Mita 80127006416SAlexandre Belloni return pcf2127_probe(&client->dev, regmap, client->irq, 802d6c3029fSUwe Kleine-König pcf2127_i2c_driver.driver.name, id->driver_data); 803907b3262SAkinobu Mita } 804907b3262SAkinobu Mita 805907b3262SAkinobu Mita static const struct i2c_device_id pcf2127_i2c_id[] = { 806d6c3029fSUwe Kleine-König { "pcf2127", 1 }, 807cee2cc21SAkinobu Mita { "pcf2129", 0 }, 808985b30dbSLiam Beguin { "pca2129", 0 }, 809907b3262SAkinobu Mita { } 810907b3262SAkinobu Mita }; 811907b3262SAkinobu Mita MODULE_DEVICE_TABLE(i2c, pcf2127_i2c_id); 812907b3262SAkinobu Mita 813907b3262SAkinobu Mita static struct i2c_driver pcf2127_i2c_driver = { 814907b3262SAkinobu Mita .driver = { 815907b3262SAkinobu Mita .name = "rtc-pcf2127-i2c", 816907b3262SAkinobu Mita .of_match_table = of_match_ptr(pcf2127_of_match), 817907b3262SAkinobu Mita }, 818907b3262SAkinobu Mita .probe = pcf2127_i2c_probe, 819907b3262SAkinobu Mita .id_table = pcf2127_i2c_id, 820907b3262SAkinobu Mita }; 8219408ec1aSAkinobu Mita 8229408ec1aSAkinobu Mita static int pcf2127_i2c_register_driver(void) 8239408ec1aSAkinobu Mita { 8249408ec1aSAkinobu Mita return i2c_add_driver(&pcf2127_i2c_driver); 8259408ec1aSAkinobu Mita } 8269408ec1aSAkinobu Mita 8279408ec1aSAkinobu Mita static void pcf2127_i2c_unregister_driver(void) 8289408ec1aSAkinobu Mita { 8299408ec1aSAkinobu Mita i2c_del_driver(&pcf2127_i2c_driver); 8309408ec1aSAkinobu Mita } 8319408ec1aSAkinobu Mita 8329408ec1aSAkinobu Mita #else 8339408ec1aSAkinobu Mita 8349408ec1aSAkinobu Mita static int pcf2127_i2c_register_driver(void) 8359408ec1aSAkinobu Mita { 8369408ec1aSAkinobu Mita return 0; 8379408ec1aSAkinobu Mita } 8389408ec1aSAkinobu Mita 8399408ec1aSAkinobu Mita static void pcf2127_i2c_unregister_driver(void) 8409408ec1aSAkinobu Mita { 8419408ec1aSAkinobu Mita } 8429408ec1aSAkinobu Mita 8439408ec1aSAkinobu Mita #endif 8449408ec1aSAkinobu Mita 8459408ec1aSAkinobu Mita #if IS_ENABLED(CONFIG_SPI_MASTER) 8469408ec1aSAkinobu Mita 8479408ec1aSAkinobu Mita static struct spi_driver pcf2127_spi_driver; 8489408ec1aSAkinobu Mita 8499408ec1aSAkinobu Mita static int pcf2127_spi_probe(struct spi_device *spi) 8509408ec1aSAkinobu Mita { 8519408ec1aSAkinobu Mita static const struct regmap_config config = { 8529408ec1aSAkinobu Mita .reg_bits = 8, 8539408ec1aSAkinobu Mita .val_bits = 8, 8549408ec1aSAkinobu Mita .read_flag_mask = 0xa0, 8559408ec1aSAkinobu Mita .write_flag_mask = 0x20, 856040e6dc0SAlexandre Belloni .max_register = 0x1d, 8579408ec1aSAkinobu Mita }; 8589408ec1aSAkinobu Mita struct regmap *regmap; 8599408ec1aSAkinobu Mita 8609408ec1aSAkinobu Mita regmap = devm_regmap_init_spi(spi, &config); 8619408ec1aSAkinobu Mita if (IS_ERR(regmap)) { 8629408ec1aSAkinobu Mita dev_err(&spi->dev, "%s: regmap allocation failed: %ld\n", 8639408ec1aSAkinobu Mita __func__, PTR_ERR(regmap)); 8649408ec1aSAkinobu Mita return PTR_ERR(regmap); 8659408ec1aSAkinobu Mita } 8669408ec1aSAkinobu Mita 86727006416SAlexandre Belloni return pcf2127_probe(&spi->dev, regmap, spi->irq, 86827006416SAlexandre Belloni pcf2127_spi_driver.driver.name, 869d6c3029fSUwe Kleine-König spi_get_device_id(spi)->driver_data); 8709408ec1aSAkinobu Mita } 8719408ec1aSAkinobu Mita 8729408ec1aSAkinobu Mita static const struct spi_device_id pcf2127_spi_id[] = { 873d6c3029fSUwe Kleine-König { "pcf2127", 1 }, 874cee2cc21SAkinobu Mita { "pcf2129", 0 }, 875985b30dbSLiam Beguin { "pca2129", 0 }, 8769408ec1aSAkinobu Mita { } 8779408ec1aSAkinobu Mita }; 8789408ec1aSAkinobu Mita MODULE_DEVICE_TABLE(spi, pcf2127_spi_id); 8799408ec1aSAkinobu Mita 8809408ec1aSAkinobu Mita static struct spi_driver pcf2127_spi_driver = { 8819408ec1aSAkinobu Mita .driver = { 8829408ec1aSAkinobu Mita .name = "rtc-pcf2127-spi", 8839408ec1aSAkinobu Mita .of_match_table = of_match_ptr(pcf2127_of_match), 8849408ec1aSAkinobu Mita }, 8859408ec1aSAkinobu Mita .probe = pcf2127_spi_probe, 8869408ec1aSAkinobu Mita .id_table = pcf2127_spi_id, 8879408ec1aSAkinobu Mita }; 8889408ec1aSAkinobu Mita 8899408ec1aSAkinobu Mita static int pcf2127_spi_register_driver(void) 8909408ec1aSAkinobu Mita { 8919408ec1aSAkinobu Mita return spi_register_driver(&pcf2127_spi_driver); 8929408ec1aSAkinobu Mita } 8939408ec1aSAkinobu Mita 8949408ec1aSAkinobu Mita static void pcf2127_spi_unregister_driver(void) 8959408ec1aSAkinobu Mita { 8969408ec1aSAkinobu Mita spi_unregister_driver(&pcf2127_spi_driver); 8979408ec1aSAkinobu Mita } 8989408ec1aSAkinobu Mita 8999408ec1aSAkinobu Mita #else 9009408ec1aSAkinobu Mita 9019408ec1aSAkinobu Mita static int pcf2127_spi_register_driver(void) 9029408ec1aSAkinobu Mita { 9039408ec1aSAkinobu Mita return 0; 9049408ec1aSAkinobu Mita } 9059408ec1aSAkinobu Mita 9069408ec1aSAkinobu Mita static void pcf2127_spi_unregister_driver(void) 9079408ec1aSAkinobu Mita { 9089408ec1aSAkinobu Mita } 9099408ec1aSAkinobu Mita 9109408ec1aSAkinobu Mita #endif 9119408ec1aSAkinobu Mita 9129408ec1aSAkinobu Mita static int __init pcf2127_init(void) 9139408ec1aSAkinobu Mita { 9149408ec1aSAkinobu Mita int ret; 9159408ec1aSAkinobu Mita 9169408ec1aSAkinobu Mita ret = pcf2127_i2c_register_driver(); 9179408ec1aSAkinobu Mita if (ret) { 9189408ec1aSAkinobu Mita pr_err("Failed to register pcf2127 i2c driver: %d\n", ret); 9199408ec1aSAkinobu Mita return ret; 9209408ec1aSAkinobu Mita } 9219408ec1aSAkinobu Mita 9229408ec1aSAkinobu Mita ret = pcf2127_spi_register_driver(); 9239408ec1aSAkinobu Mita if (ret) { 9249408ec1aSAkinobu Mita pr_err("Failed to register pcf2127 spi driver: %d\n", ret); 9259408ec1aSAkinobu Mita pcf2127_i2c_unregister_driver(); 9269408ec1aSAkinobu Mita } 9279408ec1aSAkinobu Mita 9289408ec1aSAkinobu Mita return ret; 9299408ec1aSAkinobu Mita } 9309408ec1aSAkinobu Mita module_init(pcf2127_init) 9319408ec1aSAkinobu Mita 9329408ec1aSAkinobu Mita static void __exit pcf2127_exit(void) 9339408ec1aSAkinobu Mita { 9349408ec1aSAkinobu Mita pcf2127_spi_unregister_driver(); 9359408ec1aSAkinobu Mita pcf2127_i2c_unregister_driver(); 9369408ec1aSAkinobu Mita } 9379408ec1aSAkinobu Mita module_exit(pcf2127_exit) 93818cb6368SRenaud Cerrato 93918cb6368SRenaud Cerrato MODULE_AUTHOR("Renaud Cerrato <r.cerrato@til-technologies.fr>"); 940cee2cc21SAkinobu Mita MODULE_DESCRIPTION("NXP PCF2127/29 RTC driver"); 9414d8318bcSUwe Kleine-König MODULE_LICENSE("GPL v2"); 942