1d2912cb1SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only 218cb6368SRenaud Cerrato /* 3afc505bfSHugo Villeneuve * An I2C and SPI driver for the NXP PCF2127/29/31 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 * 11afc505bfSHugo Villeneuve * PCF2131 support 12afc505bfSHugo Villeneuve * Author: Hugo Villeneuve <hvilleneuve@dimonoff.com> 13afc505bfSHugo Villeneuve * 1418cb6368SRenaud Cerrato * based on the other drivers in this same directory. 1518cb6368SRenaud Cerrato * 16afc505bfSHugo Villeneuve * Datasheets: https://www.nxp.com/docs/en/data-sheet/PCF2127.pdf 17afc505bfSHugo Villeneuve * https://www.nxp.com/docs/en/data-sheet/PCF2131DS.pdf 1818cb6368SRenaud Cerrato */ 1918cb6368SRenaud Cerrato 2018cb6368SRenaud Cerrato #include <linux/i2c.h> 219408ec1aSAkinobu Mita #include <linux/spi/spi.h> 2218cb6368SRenaud Cerrato #include <linux/bcd.h> 2318cb6368SRenaud Cerrato #include <linux/rtc.h> 2418cb6368SRenaud Cerrato #include <linux/slab.h> 2518cb6368SRenaud Cerrato #include <linux/module.h> 2618cb6368SRenaud Cerrato #include <linux/of.h> 278a914bacSLiam Beguin #include <linux/of_irq.h> 28fd28ceb4SHugo Villeneuve #include <linux/of_device.h> 29907b3262SAkinobu Mita #include <linux/regmap.h> 300e735eaaSBruno Thomsen #include <linux/watchdog.h> 3118cb6368SRenaud Cerrato 32bbfe3a7aSBruno Thomsen /* Control register 1 */ 33bbfe3a7aSBruno Thomsen #define PCF2127_REG_CTRL1 0x00 34b9ac079aSPhilipp Rosenberger #define PCF2127_BIT_CTRL1_POR_OVRD BIT(3) 3503623b4bSBruno Thomsen #define PCF2127_BIT_CTRL1_TSF1 BIT(4) 36*3d715ebaSHugo Villeneuve #define PCF2127_BIT_CTRL1_STOP BIT(5) 37bbfe3a7aSBruno Thomsen /* Control register 2 */ 38bbfe3a7aSBruno Thomsen #define PCF2127_REG_CTRL2 0x01 398a914bacSLiam Beguin #define PCF2127_BIT_CTRL2_AIE BIT(1) 4003623b4bSBruno Thomsen #define PCF2127_BIT_CTRL2_TSIE BIT(2) 418a914bacSLiam Beguin #define PCF2127_BIT_CTRL2_AF BIT(4) 4203623b4bSBruno Thomsen #define PCF2127_BIT_CTRL2_TSF2 BIT(5) 4327006416SAlexandre Belloni #define PCF2127_BIT_CTRL2_WDTF BIT(6) 44bbfe3a7aSBruno Thomsen /* Control register 3 */ 45bbfe3a7aSBruno Thomsen #define PCF2127_REG_CTRL3 0x02 4603623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BLIE BIT(0) 4703623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BIE BIT(1) 48bbfe3a7aSBruno Thomsen #define PCF2127_BIT_CTRL3_BLF BIT(2) 4903623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BF BIT(3) 5003623b4bSBruno Thomsen #define PCF2127_BIT_CTRL3_BTSE BIT(4) 51bbfe3a7aSBruno Thomsen /* Time and date registers */ 526211aceeSHugo Villeneuve #define PCF2127_REG_TIME_BASE 0x03 53bbfe3a7aSBruno Thomsen #define PCF2127_BIT_SC_OSF BIT(7) 548a914bacSLiam Beguin /* Alarm registers */ 557c6f0db4SHugo Villeneuve #define PCF2127_REG_ALARM_BASE 0x0A 5627006416SAlexandre Belloni #define PCF2127_BIT_ALARM_AE BIT(7) 5715f57b3eSPhilipp Rosenberger /* CLKOUT control register */ 5815f57b3eSPhilipp Rosenberger #define PCF2127_REG_CLKOUT 0x0f 5915f57b3eSPhilipp Rosenberger #define PCF2127_BIT_CLKOUT_OTPR BIT(5) 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 67420cc9e8SHugo Villeneuve /* Tamper timestamp1 registers */ 68420cc9e8SHugo Villeneuve #define PCF2127_REG_TS1_BASE 0x12 6903623b4bSBruno Thomsen #define PCF2127_BIT_TS_CTRL_TSOFF BIT(6) 7003623b4bSBruno Thomsen #define PCF2127_BIT_TS_CTRL_TSM BIT(7) 71bbfe3a7aSBruno Thomsen /* 72bbfe3a7aSBruno Thomsen * RAM registers 73bbfe3a7aSBruno Thomsen * PCF2127 has 512 bytes general-purpose static RAM (SRAM) that is 74bbfe3a7aSBruno Thomsen * battery backed and can survive a power outage. 75afc505bfSHugo Villeneuve * PCF2129/31 doesn't have this feature. 76bbfe3a7aSBruno Thomsen */ 77bbfe3a7aSBruno Thomsen #define PCF2127_REG_RAM_ADDR_MSB 0x1A 78bbfe3a7aSBruno Thomsen #define PCF2127_REG_RAM_WRT_CMD 0x1C 79bbfe3a7aSBruno Thomsen #define PCF2127_REG_RAM_RD_CMD 0x1D 80f97cfddcSUwe Kleine-König 810e735eaaSBruno Thomsen /* Watchdog timer value constants */ 820e735eaaSBruno Thomsen #define PCF2127_WD_VAL_STOP 0 830e735eaaSBruno Thomsen #define PCF2127_WD_VAL_MIN 2 840e735eaaSBruno Thomsen #define PCF2127_WD_VAL_MAX 255 850e735eaaSBruno Thomsen #define PCF2127_WD_VAL_DEFAULT 60 86653ebd75SAndrea Scian 872f861984SMian Yousaf Kaukab /* Mask for currently enabled interrupts */ 882f861984SMian Yousaf Kaukab #define PCF2127_CTRL1_IRQ_MASK (PCF2127_BIT_CTRL1_TSF1) 892f861984SMian Yousaf Kaukab #define PCF2127_CTRL2_IRQ_MASK ( \ 902f861984SMian Yousaf Kaukab PCF2127_BIT_CTRL2_AF | \ 912f861984SMian Yousaf Kaukab PCF2127_BIT_CTRL2_WDTF | \ 922f861984SMian Yousaf Kaukab PCF2127_BIT_CTRL2_TSF2) 932f861984SMian Yousaf Kaukab 94afc505bfSHugo Villeneuve #define PCF2127_MAX_TS_SUPPORTED 4 95afc505bfSHugo Villeneuve 96afc505bfSHugo Villeneuve /* Control register 4 */ 97afc505bfSHugo Villeneuve #define PCF2131_REG_CTRL4 0x03 98afc505bfSHugo Villeneuve #define PCF2131_BIT_CTRL4_TSF4 BIT(4) 99afc505bfSHugo Villeneuve #define PCF2131_BIT_CTRL4_TSF3 BIT(5) 100afc505bfSHugo Villeneuve #define PCF2131_BIT_CTRL4_TSF2 BIT(6) 101afc505bfSHugo Villeneuve #define PCF2131_BIT_CTRL4_TSF1 BIT(7) 102afc505bfSHugo Villeneuve /* Control register 5 */ 103afc505bfSHugo Villeneuve #define PCF2131_REG_CTRL5 0x04 104afc505bfSHugo Villeneuve #define PCF2131_BIT_CTRL5_TSIE4 BIT(4) 105afc505bfSHugo Villeneuve #define PCF2131_BIT_CTRL5_TSIE3 BIT(5) 106afc505bfSHugo Villeneuve #define PCF2131_BIT_CTRL5_TSIE2 BIT(6) 107afc505bfSHugo Villeneuve #define PCF2131_BIT_CTRL5_TSIE1 BIT(7) 108afc505bfSHugo Villeneuve /* Software reset register */ 109afc505bfSHugo Villeneuve #define PCF2131_REG_SR_RESET 0x05 110afc505bfSHugo Villeneuve #define PCF2131_SR_RESET_READ_PATTERN (BIT(2) | BIT(5)) 111afc505bfSHugo Villeneuve #define PCF2131_SR_RESET_CPR_CMD (PCF2131_SR_RESET_READ_PATTERN | BIT(7)) 112afc505bfSHugo Villeneuve /* Time and date registers */ 113afc505bfSHugo Villeneuve #define PCF2131_REG_TIME_BASE 0x07 114afc505bfSHugo Villeneuve /* Alarm registers */ 115afc505bfSHugo Villeneuve #define PCF2131_REG_ALARM_BASE 0x0E 116afc505bfSHugo Villeneuve /* CLKOUT control register */ 117afc505bfSHugo Villeneuve #define PCF2131_REG_CLKOUT 0x13 118afc505bfSHugo Villeneuve /* Watchdog registers */ 119afc505bfSHugo Villeneuve #define PCF2131_REG_WD_CTL 0x35 120afc505bfSHugo Villeneuve #define PCF2131_REG_WD_VAL 0x36 121afc505bfSHugo Villeneuve /* Tamper timestamp1 registers */ 122afc505bfSHugo Villeneuve #define PCF2131_REG_TS1_BASE 0x14 123afc505bfSHugo Villeneuve /* Tamper timestamp2 registers */ 124afc505bfSHugo Villeneuve #define PCF2131_REG_TS2_BASE 0x1B 125afc505bfSHugo Villeneuve /* Tamper timestamp3 registers */ 126afc505bfSHugo Villeneuve #define PCF2131_REG_TS3_BASE 0x22 127afc505bfSHugo Villeneuve /* Tamper timestamp4 registers */ 128afc505bfSHugo Villeneuve #define PCF2131_REG_TS4_BASE 0x29 129afc505bfSHugo Villeneuve /* Interrupt mask registers */ 130afc505bfSHugo Villeneuve #define PCF2131_REG_INT_A_MASK1 0x31 131afc505bfSHugo Villeneuve #define PCF2131_REG_INT_A_MASK2 0x32 132afc505bfSHugo Villeneuve #define PCF2131_REG_INT_B_MASK1 0x33 133afc505bfSHugo Villeneuve #define PCF2131_REG_INT_B_MASK2 0x34 134afc505bfSHugo Villeneuve #define PCF2131_BIT_INT_BLIE BIT(0) 135afc505bfSHugo Villeneuve #define PCF2131_BIT_INT_BIE BIT(1) 136afc505bfSHugo Villeneuve #define PCF2131_BIT_INT_AIE BIT(2) 137afc505bfSHugo Villeneuve #define PCF2131_BIT_INT_WD_CD BIT(3) 138afc505bfSHugo Villeneuve #define PCF2131_BIT_INT_SI BIT(4) 139afc505bfSHugo Villeneuve #define PCF2131_BIT_INT_MI BIT(5) 140afc505bfSHugo Villeneuve #define PCF2131_CTRL2_IRQ_MASK ( \ 141afc505bfSHugo Villeneuve PCF2127_BIT_CTRL2_AF | \ 142afc505bfSHugo Villeneuve PCF2127_BIT_CTRL2_WDTF) 143afc505bfSHugo Villeneuve #define PCF2131_CTRL4_IRQ_MASK ( \ 144afc505bfSHugo Villeneuve PCF2131_BIT_CTRL4_TSF4 | \ 145afc505bfSHugo Villeneuve PCF2131_BIT_CTRL4_TSF3 | \ 146afc505bfSHugo Villeneuve PCF2131_BIT_CTRL4_TSF2 | \ 147afc505bfSHugo Villeneuve PCF2131_BIT_CTRL4_TSF1) 148420cc9e8SHugo Villeneuve 149fd28ceb4SHugo Villeneuve enum pcf21xx_type { 150fd28ceb4SHugo Villeneuve PCF2127, 151fd28ceb4SHugo Villeneuve PCF2129, 152afc505bfSHugo Villeneuve PCF2131, 153fd28ceb4SHugo Villeneuve PCF21XX_LAST_ID 154fd28ceb4SHugo Villeneuve }; 155fd28ceb4SHugo Villeneuve 156420cc9e8SHugo Villeneuve struct pcf21xx_ts_config { 157420cc9e8SHugo Villeneuve u8 reg_base; /* Base register to read timestamp values. */ 158420cc9e8SHugo Villeneuve 159420cc9e8SHugo Villeneuve /* 160420cc9e8SHugo Villeneuve * If the TS input pin is driven to GND, an interrupt can be generated 161420cc9e8SHugo Villeneuve * (supported by all variants). 162420cc9e8SHugo Villeneuve */ 163420cc9e8SHugo Villeneuve u8 gnd_detect_reg; /* Interrupt control register address. */ 164420cc9e8SHugo Villeneuve u8 gnd_detect_bit; /* Interrupt bit. */ 165420cc9e8SHugo Villeneuve 166420cc9e8SHugo Villeneuve /* 167420cc9e8SHugo Villeneuve * If the TS input pin is driven to an intermediate level between GND 168420cc9e8SHugo Villeneuve * and supply, an interrupt can be generated (optional feature depending 169420cc9e8SHugo Villeneuve * on variant). 170420cc9e8SHugo Villeneuve */ 171420cc9e8SHugo Villeneuve u8 inter_detect_reg; /* Interrupt control register address. */ 172420cc9e8SHugo Villeneuve u8 inter_detect_bit; /* Interrupt bit. */ 173420cc9e8SHugo Villeneuve 174420cc9e8SHugo Villeneuve u8 ie_reg; /* Interrupt enable control register. */ 175420cc9e8SHugo Villeneuve u8 ie_bit; /* Interrupt enable bit. */ 176420cc9e8SHugo Villeneuve }; 177420cc9e8SHugo Villeneuve 178fd28ceb4SHugo Villeneuve struct pcf21xx_config { 179fd28ceb4SHugo Villeneuve int type; /* IC variant */ 180fd28ceb4SHugo Villeneuve int max_register; 181fd28ceb4SHugo Villeneuve unsigned int has_nvmem:1; 182fd28ceb4SHugo Villeneuve unsigned int has_bit_wd_ctl_cd0:1; 183e1849b8fSHugo Villeneuve unsigned int has_int_a_b:1; /* PCF2131 supports two interrupt outputs. */ 1846211aceeSHugo Villeneuve u8 reg_time_base; /* Time/date base register. */ 1857c6f0db4SHugo Villeneuve u8 regs_alarm_base; /* Alarm function base registers. */ 1866b57ec29SHugo Villeneuve u8 reg_wd_ctl; /* Watchdog control register. */ 1876b57ec29SHugo Villeneuve u8 reg_wd_val; /* Watchdog value register. */ 188fc16599eSHugo Villeneuve u8 reg_clkout; /* Clkout register. */ 189420cc9e8SHugo Villeneuve unsigned int ts_count; 190420cc9e8SHugo Villeneuve struct pcf21xx_ts_config ts[PCF2127_MAX_TS_SUPPORTED]; 191420cc9e8SHugo Villeneuve struct attribute_group attribute_group; 192fd28ceb4SHugo Villeneuve }; 193fd28ceb4SHugo Villeneuve 19418cb6368SRenaud Cerrato struct pcf2127 { 19518cb6368SRenaud Cerrato struct rtc_device *rtc; 1960e735eaaSBruno Thomsen struct watchdog_device wdd; 197907b3262SAkinobu Mita struct regmap *regmap; 198fd28ceb4SHugo Villeneuve const struct pcf21xx_config *cfg; 1992f861984SMian Yousaf Kaukab bool irq_enabled; 200420cc9e8SHugo Villeneuve time64_t ts[PCF2127_MAX_TS_SUPPORTED]; /* Timestamp values. */ 201420cc9e8SHugo Villeneuve bool ts_valid[PCF2127_MAX_TS_SUPPORTED]; /* Timestamp valid indication. */ 20218cb6368SRenaud Cerrato }; 20318cb6368SRenaud Cerrato 20418cb6368SRenaud Cerrato /* 20518cb6368SRenaud Cerrato * In the routines that deal directly with the pcf2127 hardware, we use 20618cb6368SRenaud Cerrato * rtc_time -- month 0-11, hour 0-23, yr = calendar year-epoch. 20718cb6368SRenaud Cerrato */ 208907b3262SAkinobu Mita static int pcf2127_rtc_read_time(struct device *dev, struct rtc_time *tm) 20918cb6368SRenaud Cerrato { 210907b3262SAkinobu Mita struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 21131f077c3SHugo Villeneuve unsigned char buf[7]; 212907b3262SAkinobu Mita int ret; 21318cb6368SRenaud Cerrato 2147f43020eSBruno Thomsen /* 2157f43020eSBruno Thomsen * Avoid reading CTRL2 register as it causes WD_VAL register 2167f43020eSBruno Thomsen * value to reset to 0 which means watchdog is stopped. 2177f43020eSBruno Thomsen */ 2186211aceeSHugo Villeneuve ret = regmap_bulk_read(pcf2127->regmap, pcf2127->cfg->reg_time_base, 2196211aceeSHugo Villeneuve buf, sizeof(buf)); 220907b3262SAkinobu Mita if (ret) { 221907b3262SAkinobu Mita dev_err(dev, "%s: read error\n", __func__); 222907b3262SAkinobu Mita return ret; 22318cb6368SRenaud Cerrato } 22418cb6368SRenaud Cerrato 225bbfe3a7aSBruno Thomsen /* Clock integrity is not guaranteed when OSF flag is set. */ 22631f077c3SHugo Villeneuve if (buf[0] & PCF2127_BIT_SC_OSF) { 227653ebd75SAndrea Scian /* 228653ebd75SAndrea Scian * no need clear the flag here, 229653ebd75SAndrea Scian * it will be cleared once the new date is saved 230653ebd75SAndrea Scian */ 231907b3262SAkinobu Mita dev_warn(dev, 232653ebd75SAndrea Scian "oscillator stop detected, date/time is not reliable\n"); 233653ebd75SAndrea Scian return -EINVAL; 23418cb6368SRenaud Cerrato } 23518cb6368SRenaud Cerrato 236907b3262SAkinobu Mita dev_dbg(dev, 23731f077c3SHugo Villeneuve "%s: raw data is sec=%02x, min=%02x, hr=%02x, " 23818cb6368SRenaud Cerrato "mday=%02x, wday=%02x, mon=%02x, year=%02x\n", 23931f077c3SHugo Villeneuve __func__, buf[0], buf[1], buf[2], buf[3], buf[4], buf[5], buf[6]); 24018cb6368SRenaud Cerrato 24131f077c3SHugo Villeneuve tm->tm_sec = bcd2bin(buf[0] & 0x7F); 24231f077c3SHugo Villeneuve tm->tm_min = bcd2bin(buf[1] & 0x7F); 2430476b6c8SHugo Villeneuve tm->tm_hour = bcd2bin(buf[2] & 0x3F); 24431f077c3SHugo Villeneuve tm->tm_mday = bcd2bin(buf[3] & 0x3F); 24531f077c3SHugo Villeneuve tm->tm_wday = buf[4] & 0x07; 2460476b6c8SHugo Villeneuve tm->tm_mon = bcd2bin(buf[5] & 0x1F) - 1; 24731f077c3SHugo Villeneuve tm->tm_year = bcd2bin(buf[6]); 248b139bb5cSAlexandre Belloni tm->tm_year += 100; 24918cb6368SRenaud Cerrato 250907b3262SAkinobu Mita dev_dbg(dev, "%s: tm is secs=%d, mins=%d, hours=%d, " 25118cb6368SRenaud Cerrato "mday=%d, mon=%d, year=%d, wday=%d\n", 25218cb6368SRenaud Cerrato __func__, 25318cb6368SRenaud Cerrato tm->tm_sec, tm->tm_min, tm->tm_hour, 25418cb6368SRenaud Cerrato tm->tm_mday, tm->tm_mon, tm->tm_year, tm->tm_wday); 25518cb6368SRenaud Cerrato 25622652ba7SAlexandre Belloni return 0; 25718cb6368SRenaud Cerrato } 25818cb6368SRenaud Cerrato 259907b3262SAkinobu Mita static int pcf2127_rtc_set_time(struct device *dev, struct rtc_time *tm) 26018cb6368SRenaud Cerrato { 261907b3262SAkinobu Mita struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 262907b3262SAkinobu Mita unsigned char buf[7]; 26318cb6368SRenaud Cerrato int i = 0, err; 26418cb6368SRenaud Cerrato 265907b3262SAkinobu Mita dev_dbg(dev, "%s: secs=%d, mins=%d, hours=%d, " 26618cb6368SRenaud Cerrato "mday=%d, mon=%d, year=%d, wday=%d\n", 26718cb6368SRenaud Cerrato __func__, 26818cb6368SRenaud Cerrato tm->tm_sec, tm->tm_min, tm->tm_hour, 26918cb6368SRenaud Cerrato tm->tm_mday, tm->tm_mon, tm->tm_year, tm->tm_wday); 27018cb6368SRenaud Cerrato 27118cb6368SRenaud Cerrato /* hours, minutes and seconds */ 272653ebd75SAndrea Scian buf[i++] = bin2bcd(tm->tm_sec); /* this will also clear OSF flag */ 27318cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_min); 27418cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_hour); 27518cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_mday); 27618cb6368SRenaud Cerrato buf[i++] = tm->tm_wday & 0x07; 27718cb6368SRenaud Cerrato 27818cb6368SRenaud Cerrato /* month, 1 - 12 */ 27918cb6368SRenaud Cerrato buf[i++] = bin2bcd(tm->tm_mon + 1); 28018cb6368SRenaud Cerrato 28118cb6368SRenaud Cerrato /* year */ 282b139bb5cSAlexandre Belloni buf[i++] = bin2bcd(tm->tm_year - 100); 28318cb6368SRenaud Cerrato 284*3d715ebaSHugo Villeneuve /* Write access to time registers: 285*3d715ebaSHugo Villeneuve * PCF2127/29: no special action required. 286*3d715ebaSHugo Villeneuve * PCF2131: requires setting the STOP and CPR bits. STOP bit needs to 287*3d715ebaSHugo Villeneuve * be cleared after time registers are updated. 288*3d715ebaSHugo Villeneuve */ 289*3d715ebaSHugo Villeneuve if (pcf2127->cfg->type == PCF2131) { 290*3d715ebaSHugo Villeneuve err = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL1, 291*3d715ebaSHugo Villeneuve PCF2127_BIT_CTRL1_STOP, 292*3d715ebaSHugo Villeneuve PCF2127_BIT_CTRL1_STOP); 293*3d715ebaSHugo Villeneuve if (err) { 294*3d715ebaSHugo Villeneuve dev_dbg(dev, "setting STOP bit failed\n"); 295*3d715ebaSHugo Villeneuve return err; 296*3d715ebaSHugo Villeneuve } 297*3d715ebaSHugo Villeneuve 298*3d715ebaSHugo Villeneuve err = regmap_write(pcf2127->regmap, PCF2131_REG_SR_RESET, 299*3d715ebaSHugo Villeneuve PCF2131_SR_RESET_CPR_CMD); 300*3d715ebaSHugo Villeneuve if (err) { 301*3d715ebaSHugo Villeneuve dev_dbg(dev, "sending CPR cmd failed\n"); 302*3d715ebaSHugo Villeneuve return err; 303*3d715ebaSHugo Villeneuve } 304*3d715ebaSHugo Villeneuve } 305*3d715ebaSHugo Villeneuve 306*3d715ebaSHugo Villeneuve /* write time register's data */ 3076211aceeSHugo Villeneuve err = regmap_bulk_write(pcf2127->regmap, pcf2127->cfg->reg_time_base, buf, i); 308907b3262SAkinobu Mita if (err) { 3093d740c64SHugo Villeneuve dev_dbg(dev, "%s: err=%d", __func__, err); 310907b3262SAkinobu Mita return err; 31118cb6368SRenaud Cerrato } 31218cb6368SRenaud Cerrato 313*3d715ebaSHugo Villeneuve if (pcf2127->cfg->type == PCF2131) { 314*3d715ebaSHugo Villeneuve /* Clear STOP bit (PCF2131 only) after write is completed. */ 315*3d715ebaSHugo Villeneuve err = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL1, 316*3d715ebaSHugo Villeneuve PCF2127_BIT_CTRL1_STOP, 0); 317*3d715ebaSHugo Villeneuve if (err) { 318*3d715ebaSHugo Villeneuve dev_dbg(dev, "clearing STOP bit failed\n"); 319*3d715ebaSHugo Villeneuve return err; 320*3d715ebaSHugo Villeneuve } 321*3d715ebaSHugo Villeneuve } 322*3d715ebaSHugo Villeneuve 32318cb6368SRenaud Cerrato return 0; 32418cb6368SRenaud Cerrato } 32518cb6368SRenaud Cerrato 32618cb6368SRenaud Cerrato static int pcf2127_rtc_ioctl(struct device *dev, 32718cb6368SRenaud Cerrato unsigned int cmd, unsigned long arg) 32818cb6368SRenaud Cerrato { 329907b3262SAkinobu Mita struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 3307d65cf8cSAlexandre Belloni int val, touser = 0; 331f97cfddcSUwe Kleine-König int ret; 33218cb6368SRenaud Cerrato 33318cb6368SRenaud Cerrato switch (cmd) { 33418cb6368SRenaud Cerrato case RTC_VL_READ: 3357d65cf8cSAlexandre Belloni ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL3, &val); 336907b3262SAkinobu Mita if (ret) 337f97cfddcSUwe Kleine-König return ret; 33818cb6368SRenaud Cerrato 3397d65cf8cSAlexandre Belloni if (val & PCF2127_BIT_CTRL3_BLF) 3407d65cf8cSAlexandre Belloni touser |= RTC_VL_BACKUP_LOW; 3417d65cf8cSAlexandre Belloni 3427d65cf8cSAlexandre Belloni if (val & PCF2127_BIT_CTRL3_BF) 3437d65cf8cSAlexandre Belloni touser |= RTC_VL_BACKUP_SWITCH; 344f97cfddcSUwe Kleine-König 345af427311SAlexandre Belloni return put_user(touser, (unsigned int __user *)arg); 3467d65cf8cSAlexandre Belloni 3477d65cf8cSAlexandre Belloni case RTC_VL_CLR: 3487d65cf8cSAlexandre Belloni return regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL3, 3497d65cf8cSAlexandre Belloni PCF2127_BIT_CTRL3_BF, 0); 3507d65cf8cSAlexandre Belloni 35118cb6368SRenaud Cerrato default: 35218cb6368SRenaud Cerrato return -ENOIOCTLCMD; 35318cb6368SRenaud Cerrato } 35418cb6368SRenaud Cerrato } 35518cb6368SRenaud Cerrato 356d6c3029fSUwe Kleine-König static int pcf2127_nvmem_read(void *priv, unsigned int offset, 357d6c3029fSUwe Kleine-König void *val, size_t bytes) 358d6c3029fSUwe Kleine-König { 359d6c3029fSUwe Kleine-König struct pcf2127 *pcf2127 = priv; 360d6c3029fSUwe Kleine-König int ret; 361d6c3029fSUwe Kleine-König unsigned char offsetbuf[] = { offset >> 8, offset }; 362d6c3029fSUwe Kleine-König 363bbfe3a7aSBruno Thomsen ret = regmap_bulk_write(pcf2127->regmap, PCF2127_REG_RAM_ADDR_MSB, 364d6c3029fSUwe Kleine-König offsetbuf, 2); 365d6c3029fSUwe Kleine-König if (ret) 366d6c3029fSUwe Kleine-König return ret; 367d6c3029fSUwe Kleine-König 368ba1c30bfSDan Carpenter return regmap_bulk_read(pcf2127->regmap, PCF2127_REG_RAM_RD_CMD, 369d6c3029fSUwe Kleine-König val, bytes); 370d6c3029fSUwe Kleine-König } 371d6c3029fSUwe Kleine-König 372d6c3029fSUwe Kleine-König static int pcf2127_nvmem_write(void *priv, unsigned int offset, 373d6c3029fSUwe Kleine-König void *val, size_t bytes) 374d6c3029fSUwe Kleine-König { 375d6c3029fSUwe Kleine-König struct pcf2127 *pcf2127 = priv; 376d6c3029fSUwe Kleine-König int ret; 377d6c3029fSUwe Kleine-König unsigned char offsetbuf[] = { offset >> 8, offset }; 378d6c3029fSUwe Kleine-König 379bbfe3a7aSBruno Thomsen ret = regmap_bulk_write(pcf2127->regmap, PCF2127_REG_RAM_ADDR_MSB, 380d6c3029fSUwe Kleine-König offsetbuf, 2); 381d6c3029fSUwe Kleine-König if (ret) 382d6c3029fSUwe Kleine-König return ret; 383d6c3029fSUwe Kleine-König 384ba1c30bfSDan Carpenter return regmap_bulk_write(pcf2127->regmap, PCF2127_REG_RAM_WRT_CMD, 385d6c3029fSUwe Kleine-König val, bytes); 386d6c3029fSUwe Kleine-König } 387d6c3029fSUwe Kleine-König 3880e735eaaSBruno Thomsen /* watchdog driver */ 3890e735eaaSBruno Thomsen 3900e735eaaSBruno Thomsen static int pcf2127_wdt_ping(struct watchdog_device *wdd) 3910e735eaaSBruno Thomsen { 3920e735eaaSBruno Thomsen struct pcf2127 *pcf2127 = watchdog_get_drvdata(wdd); 3930e735eaaSBruno Thomsen 3946b57ec29SHugo Villeneuve return regmap_write(pcf2127->regmap, pcf2127->cfg->reg_wd_val, wdd->timeout); 3950e735eaaSBruno Thomsen } 3960e735eaaSBruno Thomsen 3970e735eaaSBruno Thomsen /* 3980e735eaaSBruno Thomsen * Restart watchdog timer if feature is active. 3990e735eaaSBruno Thomsen * 4000e735eaaSBruno Thomsen * Note: Reading CTRL2 register causes watchdog to stop which is unfortunate, 4010e735eaaSBruno Thomsen * since register also contain control/status flags for other features. 4020e735eaaSBruno Thomsen * Always call this function after reading CTRL2 register. 4030e735eaaSBruno Thomsen */ 4040e735eaaSBruno Thomsen static int pcf2127_wdt_active_ping(struct watchdog_device *wdd) 4050e735eaaSBruno Thomsen { 4060e735eaaSBruno Thomsen int ret = 0; 4070e735eaaSBruno Thomsen 4080e735eaaSBruno Thomsen if (watchdog_active(wdd)) { 4090e735eaaSBruno Thomsen ret = pcf2127_wdt_ping(wdd); 4100e735eaaSBruno Thomsen if (ret) 4110e735eaaSBruno Thomsen dev_err(wdd->parent, 4120e735eaaSBruno Thomsen "%s: watchdog restart failed, ret=%d\n", 4130e735eaaSBruno Thomsen __func__, ret); 4140e735eaaSBruno Thomsen } 4150e735eaaSBruno Thomsen 4160e735eaaSBruno Thomsen return ret; 4170e735eaaSBruno Thomsen } 4180e735eaaSBruno Thomsen 4190e735eaaSBruno Thomsen static int pcf2127_wdt_start(struct watchdog_device *wdd) 4200e735eaaSBruno Thomsen { 4210e735eaaSBruno Thomsen return pcf2127_wdt_ping(wdd); 4220e735eaaSBruno Thomsen } 4230e735eaaSBruno Thomsen 4240e735eaaSBruno Thomsen static int pcf2127_wdt_stop(struct watchdog_device *wdd) 4250e735eaaSBruno Thomsen { 4260e735eaaSBruno Thomsen struct pcf2127 *pcf2127 = watchdog_get_drvdata(wdd); 4270e735eaaSBruno Thomsen 4286b57ec29SHugo Villeneuve return regmap_write(pcf2127->regmap, pcf2127->cfg->reg_wd_val, 4290e735eaaSBruno Thomsen PCF2127_WD_VAL_STOP); 4300e735eaaSBruno Thomsen } 4310e735eaaSBruno Thomsen 4320e735eaaSBruno Thomsen static int pcf2127_wdt_set_timeout(struct watchdog_device *wdd, 4330e735eaaSBruno Thomsen unsigned int new_timeout) 4340e735eaaSBruno Thomsen { 4350e735eaaSBruno Thomsen dev_dbg(wdd->parent, "new watchdog timeout: %is (old: %is)\n", 4360e735eaaSBruno Thomsen new_timeout, wdd->timeout); 4370e735eaaSBruno Thomsen 4380e735eaaSBruno Thomsen wdd->timeout = new_timeout; 4390e735eaaSBruno Thomsen 4400e735eaaSBruno Thomsen return pcf2127_wdt_active_ping(wdd); 4410e735eaaSBruno Thomsen } 4420e735eaaSBruno Thomsen 4430e735eaaSBruno Thomsen static const struct watchdog_info pcf2127_wdt_info = { 4440e735eaaSBruno Thomsen .identity = "NXP PCF2127/PCF2129 Watchdog", 4450e735eaaSBruno Thomsen .options = WDIOF_KEEPALIVEPING | WDIOF_SETTIMEOUT, 4460e735eaaSBruno Thomsen }; 4470e735eaaSBruno Thomsen 4480e735eaaSBruno Thomsen static const struct watchdog_ops pcf2127_watchdog_ops = { 4490e735eaaSBruno Thomsen .owner = THIS_MODULE, 4500e735eaaSBruno Thomsen .start = pcf2127_wdt_start, 4510e735eaaSBruno Thomsen .stop = pcf2127_wdt_stop, 4520e735eaaSBruno Thomsen .ping = pcf2127_wdt_ping, 4530e735eaaSBruno Thomsen .set_timeout = pcf2127_wdt_set_timeout, 4540e735eaaSBruno Thomsen }; 4550e735eaaSBruno Thomsen 4565d78533aSUwe Kleine-König static int pcf2127_watchdog_init(struct device *dev, struct pcf2127 *pcf2127) 4575d78533aSUwe Kleine-König { 4585d78533aSUwe Kleine-König u32 wdd_timeout; 4595d78533aSUwe Kleine-König int ret; 4605d78533aSUwe Kleine-König 46171ac1345SUwe Kleine-König if (!IS_ENABLED(CONFIG_WATCHDOG) || 46271ac1345SUwe Kleine-König !device_property_read_bool(dev, "reset-source")) 4635d78533aSUwe Kleine-König return 0; 4645d78533aSUwe Kleine-König 4655d78533aSUwe Kleine-König pcf2127->wdd.parent = dev; 4665d78533aSUwe Kleine-König pcf2127->wdd.info = &pcf2127_wdt_info; 4675d78533aSUwe Kleine-König pcf2127->wdd.ops = &pcf2127_watchdog_ops; 4685d78533aSUwe Kleine-König pcf2127->wdd.min_timeout = PCF2127_WD_VAL_MIN; 4695d78533aSUwe Kleine-König pcf2127->wdd.max_timeout = PCF2127_WD_VAL_MAX; 4705d78533aSUwe Kleine-König pcf2127->wdd.timeout = PCF2127_WD_VAL_DEFAULT; 4715d78533aSUwe Kleine-König pcf2127->wdd.min_hw_heartbeat_ms = 500; 4725d78533aSUwe Kleine-König pcf2127->wdd.status = WATCHDOG_NOWAYOUT_INIT_STATUS; 4735d78533aSUwe Kleine-König 4745d78533aSUwe Kleine-König watchdog_set_drvdata(&pcf2127->wdd, pcf2127); 4755d78533aSUwe Kleine-König 4765d78533aSUwe Kleine-König /* Test if watchdog timer is started by bootloader */ 4776b57ec29SHugo Villeneuve ret = regmap_read(pcf2127->regmap, pcf2127->cfg->reg_wd_val, &wdd_timeout); 4785d78533aSUwe Kleine-König if (ret) 4795d78533aSUwe Kleine-König return ret; 4805d78533aSUwe Kleine-König 4815d78533aSUwe Kleine-König if (wdd_timeout) 4825d78533aSUwe Kleine-König set_bit(WDOG_HW_RUNNING, &pcf2127->wdd.status); 4835d78533aSUwe Kleine-König 4845d78533aSUwe Kleine-König return devm_watchdog_register_device(dev, &pcf2127->wdd); 4855d78533aSUwe Kleine-König } 4865d78533aSUwe Kleine-König 4878a914bacSLiam Beguin /* Alarm */ 4888a914bacSLiam Beguin static int pcf2127_rtc_read_alarm(struct device *dev, struct rtc_wkalrm *alrm) 4898a914bacSLiam Beguin { 4908a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 49173ce0530SHugo Villeneuve u8 buf[5]; 49273ce0530SHugo Villeneuve unsigned int ctrl2; 4938a914bacSLiam Beguin int ret; 4948a914bacSLiam Beguin 4958a914bacSLiam Beguin ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL2, &ctrl2); 4968a914bacSLiam Beguin if (ret) 4978a914bacSLiam Beguin return ret; 4988a914bacSLiam Beguin 4998a914bacSLiam Beguin ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 5008a914bacSLiam Beguin if (ret) 5018a914bacSLiam Beguin return ret; 5028a914bacSLiam Beguin 5037c6f0db4SHugo Villeneuve ret = regmap_bulk_read(pcf2127->regmap, pcf2127->cfg->regs_alarm_base, 5047c6f0db4SHugo Villeneuve buf, sizeof(buf)); 5058a914bacSLiam Beguin if (ret) 5068a914bacSLiam Beguin return ret; 5078a914bacSLiam Beguin 5088a914bacSLiam Beguin alrm->enabled = ctrl2 & PCF2127_BIT_CTRL2_AIE; 5098a914bacSLiam Beguin alrm->pending = ctrl2 & PCF2127_BIT_CTRL2_AF; 5108a914bacSLiam Beguin 5118a914bacSLiam Beguin alrm->time.tm_sec = bcd2bin(buf[0] & 0x7F); 5128a914bacSLiam Beguin alrm->time.tm_min = bcd2bin(buf[1] & 0x7F); 5138a914bacSLiam Beguin alrm->time.tm_hour = bcd2bin(buf[2] & 0x3F); 5148a914bacSLiam Beguin alrm->time.tm_mday = bcd2bin(buf[3] & 0x3F); 5158a914bacSLiam Beguin 5168a914bacSLiam Beguin return 0; 5178a914bacSLiam Beguin } 5188a914bacSLiam Beguin 5198a914bacSLiam Beguin static int pcf2127_rtc_alarm_irq_enable(struct device *dev, u32 enable) 5208a914bacSLiam Beguin { 5218a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 5228a914bacSLiam Beguin int ret; 5238a914bacSLiam Beguin 5248a914bacSLiam Beguin ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL2, 5258a914bacSLiam Beguin PCF2127_BIT_CTRL2_AIE, 5268a914bacSLiam Beguin enable ? PCF2127_BIT_CTRL2_AIE : 0); 5278a914bacSLiam Beguin if (ret) 5288a914bacSLiam Beguin return ret; 5298a914bacSLiam Beguin 5308a914bacSLiam Beguin return pcf2127_wdt_active_ping(&pcf2127->wdd); 5318a914bacSLiam Beguin } 5328a914bacSLiam Beguin 5338a914bacSLiam Beguin static int pcf2127_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alrm) 5348a914bacSLiam Beguin { 5358a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 5368a914bacSLiam Beguin uint8_t buf[5]; 5378a914bacSLiam Beguin int ret; 5388a914bacSLiam Beguin 5398a914bacSLiam Beguin ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL2, 5408a914bacSLiam Beguin PCF2127_BIT_CTRL2_AF, 0); 5418a914bacSLiam Beguin if (ret) 5428a914bacSLiam Beguin return ret; 5438a914bacSLiam Beguin 5448a914bacSLiam Beguin ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 5458a914bacSLiam Beguin if (ret) 5468a914bacSLiam Beguin return ret; 5478a914bacSLiam Beguin 5488a914bacSLiam Beguin buf[0] = bin2bcd(alrm->time.tm_sec); 5498a914bacSLiam Beguin buf[1] = bin2bcd(alrm->time.tm_min); 5508a914bacSLiam Beguin buf[2] = bin2bcd(alrm->time.tm_hour); 5518a914bacSLiam Beguin buf[3] = bin2bcd(alrm->time.tm_mday); 55227006416SAlexandre Belloni buf[4] = PCF2127_BIT_ALARM_AE; /* Do not match on week day */ 5538a914bacSLiam Beguin 5547c6f0db4SHugo Villeneuve ret = regmap_bulk_write(pcf2127->regmap, pcf2127->cfg->regs_alarm_base, 5557c6f0db4SHugo Villeneuve buf, sizeof(buf)); 5568a914bacSLiam Beguin if (ret) 5578a914bacSLiam Beguin return ret; 5588a914bacSLiam Beguin 5598a914bacSLiam Beguin return pcf2127_rtc_alarm_irq_enable(dev, alrm->enabled); 5608a914bacSLiam Beguin } 5618a914bacSLiam Beguin 5622f861984SMian Yousaf Kaukab /* 563420cc9e8SHugo Villeneuve * This function reads one timestamp function data, caller is responsible for 564420cc9e8SHugo Villeneuve * calling pcf2127_wdt_active_ping() 5652f861984SMian Yousaf Kaukab */ 566420cc9e8SHugo Villeneuve static int pcf2127_rtc_ts_read(struct device *dev, time64_t *ts, 567420cc9e8SHugo Villeneuve int ts_id) 5682f861984SMian Yousaf Kaukab { 5692f861984SMian Yousaf Kaukab struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 5702f861984SMian Yousaf Kaukab struct rtc_time tm; 5712f861984SMian Yousaf Kaukab int ret; 572720fb4b8SHugo Villeneuve unsigned char data[7]; 5732f861984SMian Yousaf Kaukab 574420cc9e8SHugo Villeneuve ret = regmap_bulk_read(pcf2127->regmap, pcf2127->cfg->ts[ts_id].reg_base, 575420cc9e8SHugo Villeneuve data, sizeof(data)); 5762f861984SMian Yousaf Kaukab if (ret) { 5772f861984SMian Yousaf Kaukab dev_err(dev, "%s: read error ret=%d\n", __func__, ret); 5782f861984SMian Yousaf Kaukab return ret; 5792f861984SMian Yousaf Kaukab } 5802f861984SMian Yousaf Kaukab 5812f861984SMian Yousaf Kaukab dev_dbg(dev, 582720fb4b8SHugo Villeneuve "%s: raw data is ts_sc=%02x, ts_mn=%02x, ts_hr=%02x, ts_dm=%02x, ts_mo=%02x, ts_yr=%02x\n", 583720fb4b8SHugo Villeneuve __func__, data[1], data[2], data[3], data[4], data[5], data[6]); 5842f861984SMian Yousaf Kaukab 585720fb4b8SHugo Villeneuve tm.tm_sec = bcd2bin(data[1] & 0x7F); 586720fb4b8SHugo Villeneuve tm.tm_min = bcd2bin(data[2] & 0x7F); 587720fb4b8SHugo Villeneuve tm.tm_hour = bcd2bin(data[3] & 0x3F); 588720fb4b8SHugo Villeneuve tm.tm_mday = bcd2bin(data[4] & 0x3F); 5892f861984SMian Yousaf Kaukab /* TS_MO register (month) value range: 1-12 */ 590720fb4b8SHugo Villeneuve tm.tm_mon = bcd2bin(data[5] & 0x1F) - 1; 591720fb4b8SHugo Villeneuve tm.tm_year = bcd2bin(data[6]); 5922f861984SMian Yousaf Kaukab if (tm.tm_year < 70) 5932f861984SMian Yousaf Kaukab tm.tm_year += 100; /* assume we are in 1970...2069 */ 5942f861984SMian Yousaf Kaukab 5952f861984SMian Yousaf Kaukab ret = rtc_valid_tm(&tm); 5962f861984SMian Yousaf Kaukab if (ret) { 5972f861984SMian Yousaf Kaukab dev_err(dev, "Invalid timestamp. ret=%d\n", ret); 5982f861984SMian Yousaf Kaukab return ret; 5992f861984SMian Yousaf Kaukab } 6002f861984SMian Yousaf Kaukab 6012f861984SMian Yousaf Kaukab *ts = rtc_tm_to_time64(&tm); 6022f861984SMian Yousaf Kaukab return 0; 6032f861984SMian Yousaf Kaukab }; 6042f861984SMian Yousaf Kaukab 605420cc9e8SHugo Villeneuve static void pcf2127_rtc_ts_snapshot(struct device *dev, int ts_id) 6062f861984SMian Yousaf Kaukab { 6072f861984SMian Yousaf Kaukab struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 6082f861984SMian Yousaf Kaukab int ret; 6092f861984SMian Yousaf Kaukab 610420cc9e8SHugo Villeneuve if (ts_id >= pcf2127->cfg->ts_count) 6112f861984SMian Yousaf Kaukab return; 6122f861984SMian Yousaf Kaukab 613420cc9e8SHugo Villeneuve /* Let userspace read the first timestamp */ 614420cc9e8SHugo Villeneuve if (pcf2127->ts_valid[ts_id]) 615420cc9e8SHugo Villeneuve return; 616420cc9e8SHugo Villeneuve 617420cc9e8SHugo Villeneuve ret = pcf2127_rtc_ts_read(dev, &pcf2127->ts[ts_id], ts_id); 6182f861984SMian Yousaf Kaukab if (!ret) 619420cc9e8SHugo Villeneuve pcf2127->ts_valid[ts_id] = true; 6202f861984SMian Yousaf Kaukab } 6212f861984SMian Yousaf Kaukab 6228a914bacSLiam Beguin static irqreturn_t pcf2127_rtc_irq(int irq, void *dev) 6238a914bacSLiam Beguin { 6248a914bacSLiam Beguin struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 625afc505bfSHugo Villeneuve unsigned int ctrl2; 6268a914bacSLiam Beguin int ret = 0; 6278a914bacSLiam Beguin 628afc505bfSHugo Villeneuve ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL2, &ctrl2); 6292f861984SMian Yousaf Kaukab if (ret) 6302f861984SMian Yousaf Kaukab return IRQ_NONE; 6312f861984SMian Yousaf Kaukab 632afc505bfSHugo Villeneuve if (pcf2127->cfg->ts_count == 1) { 633afc505bfSHugo Villeneuve /* PCF2127/29 */ 634afc505bfSHugo Villeneuve unsigned int ctrl1; 635afc505bfSHugo Villeneuve 636afc505bfSHugo Villeneuve ret = regmap_read(pcf2127->regmap, PCF2127_REG_CTRL1, &ctrl1); 6378a914bacSLiam Beguin if (ret) 6388a914bacSLiam Beguin return IRQ_NONE; 6398a914bacSLiam Beguin 6402f861984SMian Yousaf Kaukab if (!(ctrl1 & PCF2127_CTRL1_IRQ_MASK || ctrl2 & PCF2127_CTRL2_IRQ_MASK)) 64127006416SAlexandre Belloni return IRQ_NONE; 64227006416SAlexandre Belloni 6432f861984SMian Yousaf Kaukab if (ctrl1 & PCF2127_BIT_CTRL1_TSF1 || ctrl2 & PCF2127_BIT_CTRL2_TSF2) 644420cc9e8SHugo Villeneuve pcf2127_rtc_ts_snapshot(dev, 0); 6458a914bacSLiam Beguin 6462f861984SMian Yousaf Kaukab if (ctrl1 & PCF2127_CTRL1_IRQ_MASK) 6472f861984SMian Yousaf Kaukab regmap_write(pcf2127->regmap, PCF2127_REG_CTRL1, 6482f861984SMian Yousaf Kaukab ctrl1 & ~PCF2127_CTRL1_IRQ_MASK); 6492f861984SMian Yousaf Kaukab 6502f861984SMian Yousaf Kaukab if (ctrl2 & PCF2127_CTRL2_IRQ_MASK) 6512f861984SMian Yousaf Kaukab regmap_write(pcf2127->regmap, PCF2127_REG_CTRL2, 6522f861984SMian Yousaf Kaukab ctrl2 & ~PCF2127_CTRL2_IRQ_MASK); 653afc505bfSHugo Villeneuve } else { 654afc505bfSHugo Villeneuve /* PCF2131. */ 655afc505bfSHugo Villeneuve unsigned int ctrl4; 656afc505bfSHugo Villeneuve 657afc505bfSHugo Villeneuve ret = regmap_read(pcf2127->regmap, PCF2131_REG_CTRL4, &ctrl4); 658afc505bfSHugo Villeneuve if (ret) 659afc505bfSHugo Villeneuve return IRQ_NONE; 660afc505bfSHugo Villeneuve 661afc505bfSHugo Villeneuve if (!(ctrl4 & PCF2131_CTRL4_IRQ_MASK || ctrl2 & PCF2131_CTRL2_IRQ_MASK)) 662afc505bfSHugo Villeneuve return IRQ_NONE; 663afc505bfSHugo Villeneuve 664afc505bfSHugo Villeneuve if (ctrl4 & PCF2131_CTRL4_IRQ_MASK) { 665afc505bfSHugo Villeneuve int i; 666afc505bfSHugo Villeneuve int tsf_bit = PCF2131_BIT_CTRL4_TSF1; /* Start at bit 7. */ 667afc505bfSHugo Villeneuve 668afc505bfSHugo Villeneuve for (i = 0; i < pcf2127->cfg->ts_count; i++) { 669afc505bfSHugo Villeneuve if (ctrl4 & tsf_bit) 670afc505bfSHugo Villeneuve pcf2127_rtc_ts_snapshot(dev, i); 671afc505bfSHugo Villeneuve 672afc505bfSHugo Villeneuve tsf_bit = tsf_bit >> 1; 673afc505bfSHugo Villeneuve } 674afc505bfSHugo Villeneuve 675afc505bfSHugo Villeneuve regmap_write(pcf2127->regmap, PCF2131_REG_CTRL4, 676afc505bfSHugo Villeneuve ctrl4 & ~PCF2131_CTRL4_IRQ_MASK); 677afc505bfSHugo Villeneuve } 678afc505bfSHugo Villeneuve 679afc505bfSHugo Villeneuve if (ctrl2 & PCF2131_CTRL2_IRQ_MASK) 680afc505bfSHugo Villeneuve regmap_write(pcf2127->regmap, PCF2127_REG_CTRL2, 681afc505bfSHugo Villeneuve ctrl2 & ~PCF2131_CTRL2_IRQ_MASK); 682afc505bfSHugo Villeneuve } 6832f861984SMian Yousaf Kaukab 6842f861984SMian Yousaf Kaukab if (ctrl2 & PCF2127_BIT_CTRL2_AF) 6858a914bacSLiam Beguin rtc_update_irq(pcf2127->rtc, 1, RTC_IRQF | RTC_AF); 6868a914bacSLiam Beguin 68727006416SAlexandre Belloni pcf2127_wdt_active_ping(&pcf2127->wdd); 6888a914bacSLiam Beguin 6898a914bacSLiam Beguin return IRQ_HANDLED; 6908a914bacSLiam Beguin } 6918a914bacSLiam Beguin 69225cbe9c8SAlexandre Belloni static const struct rtc_class_ops pcf2127_rtc_ops = { 6938a914bacSLiam Beguin .ioctl = pcf2127_rtc_ioctl, 6948a914bacSLiam Beguin .read_time = pcf2127_rtc_read_time, 6958a914bacSLiam Beguin .set_time = pcf2127_rtc_set_time, 6968a914bacSLiam Beguin .read_alarm = pcf2127_rtc_read_alarm, 6978a914bacSLiam Beguin .set_alarm = pcf2127_rtc_set_alarm, 6988a914bacSLiam Beguin .alarm_irq_enable = pcf2127_rtc_alarm_irq_enable, 6998a914bacSLiam Beguin }; 7008a914bacSLiam Beguin 70103623b4bSBruno Thomsen /* sysfs interface */ 70203623b4bSBruno Thomsen 703420cc9e8SHugo Villeneuve static ssize_t timestamp_store(struct device *dev, 70403623b4bSBruno Thomsen struct device_attribute *attr, 705420cc9e8SHugo Villeneuve const char *buf, size_t count, int ts_id) 70603623b4bSBruno Thomsen { 70703623b4bSBruno Thomsen struct pcf2127 *pcf2127 = dev_get_drvdata(dev->parent); 70803623b4bSBruno Thomsen int ret; 70903623b4bSBruno Thomsen 710420cc9e8SHugo Villeneuve if (ts_id >= pcf2127->cfg->ts_count) 711420cc9e8SHugo Villeneuve return 0; 712420cc9e8SHugo Villeneuve 7132f861984SMian Yousaf Kaukab if (pcf2127->irq_enabled) { 714420cc9e8SHugo Villeneuve pcf2127->ts_valid[ts_id] = false; 7152f861984SMian Yousaf Kaukab } else { 716420cc9e8SHugo Villeneuve /* Always clear GND interrupt bit. */ 717420cc9e8SHugo Villeneuve ret = regmap_update_bits(pcf2127->regmap, 718420cc9e8SHugo Villeneuve pcf2127->cfg->ts[ts_id].gnd_detect_reg, 719420cc9e8SHugo Villeneuve pcf2127->cfg->ts[ts_id].gnd_detect_bit, 720420cc9e8SHugo Villeneuve 0); 721420cc9e8SHugo Villeneuve 72203623b4bSBruno Thomsen if (ret) { 723420cc9e8SHugo Villeneuve dev_err(dev, "%s: update TS gnd detect ret=%d\n", __func__, ret); 72403623b4bSBruno Thomsen return ret; 72503623b4bSBruno Thomsen } 72603623b4bSBruno Thomsen 727420cc9e8SHugo Villeneuve if (pcf2127->cfg->ts[ts_id].inter_detect_bit) { 728420cc9e8SHugo Villeneuve /* Clear intermediate level interrupt bit if supported. */ 729420cc9e8SHugo Villeneuve ret = regmap_update_bits(pcf2127->regmap, 730420cc9e8SHugo Villeneuve pcf2127->cfg->ts[ts_id].inter_detect_reg, 731420cc9e8SHugo Villeneuve pcf2127->cfg->ts[ts_id].inter_detect_bit, 732420cc9e8SHugo Villeneuve 0); 73303623b4bSBruno Thomsen if (ret) { 734420cc9e8SHugo Villeneuve dev_err(dev, "%s: update TS intermediate level detect ret=%d\n", 735420cc9e8SHugo Villeneuve __func__, ret); 73603623b4bSBruno Thomsen return ret; 73703623b4bSBruno Thomsen } 738420cc9e8SHugo Villeneuve } 73903623b4bSBruno Thomsen 74003623b4bSBruno Thomsen ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 74103623b4bSBruno Thomsen if (ret) 74203623b4bSBruno Thomsen return ret; 7432f861984SMian Yousaf Kaukab } 74403623b4bSBruno Thomsen 74503623b4bSBruno Thomsen return count; 746420cc9e8SHugo Villeneuve } 747420cc9e8SHugo Villeneuve 748420cc9e8SHugo Villeneuve static ssize_t timestamp0_store(struct device *dev, 749420cc9e8SHugo Villeneuve struct device_attribute *attr, 750420cc9e8SHugo Villeneuve const char *buf, size_t count) 751420cc9e8SHugo Villeneuve { 752420cc9e8SHugo Villeneuve return timestamp_store(dev, attr, buf, count, 0); 75303623b4bSBruno Thomsen }; 75403623b4bSBruno Thomsen 755afc505bfSHugo Villeneuve static ssize_t timestamp1_store(struct device *dev, 756afc505bfSHugo Villeneuve struct device_attribute *attr, 757afc505bfSHugo Villeneuve const char *buf, size_t count) 758afc505bfSHugo Villeneuve { 759afc505bfSHugo Villeneuve return timestamp_store(dev, attr, buf, count, 1); 760afc505bfSHugo Villeneuve }; 761afc505bfSHugo Villeneuve 762afc505bfSHugo Villeneuve static ssize_t timestamp2_store(struct device *dev, 763afc505bfSHugo Villeneuve struct device_attribute *attr, 764afc505bfSHugo Villeneuve const char *buf, size_t count) 765afc505bfSHugo Villeneuve { 766afc505bfSHugo Villeneuve return timestamp_store(dev, attr, buf, count, 2); 767afc505bfSHugo Villeneuve }; 768afc505bfSHugo Villeneuve 769afc505bfSHugo Villeneuve static ssize_t timestamp3_store(struct device *dev, 770afc505bfSHugo Villeneuve struct device_attribute *attr, 771afc505bfSHugo Villeneuve const char *buf, size_t count) 772afc505bfSHugo Villeneuve { 773afc505bfSHugo Villeneuve return timestamp_store(dev, attr, buf, count, 3); 774afc505bfSHugo Villeneuve }; 775afc505bfSHugo Villeneuve 776420cc9e8SHugo Villeneuve static ssize_t timestamp_show(struct device *dev, 777420cc9e8SHugo Villeneuve struct device_attribute *attr, char *buf, 778420cc9e8SHugo Villeneuve int ts_id) 77903623b4bSBruno Thomsen { 78003623b4bSBruno Thomsen struct pcf2127 *pcf2127 = dev_get_drvdata(dev->parent); 78103623b4bSBruno Thomsen int ret; 7822f861984SMian Yousaf Kaukab time64_t ts; 78303623b4bSBruno Thomsen 784420cc9e8SHugo Villeneuve if (ts_id >= pcf2127->cfg->ts_count) 785420cc9e8SHugo Villeneuve return 0; 786420cc9e8SHugo Villeneuve 7872f861984SMian Yousaf Kaukab if (pcf2127->irq_enabled) { 788420cc9e8SHugo Villeneuve if (!pcf2127->ts_valid[ts_id]) 7892f861984SMian Yousaf Kaukab return 0; 790420cc9e8SHugo Villeneuve ts = pcf2127->ts[ts_id]; 7912f861984SMian Yousaf Kaukab } else { 792420cc9e8SHugo Villeneuve u8 valid_low = 0; 793420cc9e8SHugo Villeneuve u8 valid_inter = 0; 794420cc9e8SHugo Villeneuve unsigned int ctrl; 795420cc9e8SHugo Villeneuve 796420cc9e8SHugo Villeneuve /* Check if TS input pin is driven to GND, supported by all 797420cc9e8SHugo Villeneuve * variants. 798420cc9e8SHugo Villeneuve */ 799420cc9e8SHugo Villeneuve ret = regmap_read(pcf2127->regmap, 800420cc9e8SHugo Villeneuve pcf2127->cfg->ts[ts_id].gnd_detect_reg, 801420cc9e8SHugo Villeneuve &ctrl); 8022f861984SMian Yousaf Kaukab if (ret) 8032f861984SMian Yousaf Kaukab return 0; 80403623b4bSBruno Thomsen 805420cc9e8SHugo Villeneuve valid_low = ctrl & pcf2127->cfg->ts[ts_id].gnd_detect_bit; 806420cc9e8SHugo Villeneuve 807420cc9e8SHugo Villeneuve if (pcf2127->cfg->ts[ts_id].inter_detect_bit) { 808420cc9e8SHugo Villeneuve /* Check if TS input pin is driven to intermediate level 809420cc9e8SHugo Villeneuve * between GND and supply, if supported by variant. 810420cc9e8SHugo Villeneuve */ 811420cc9e8SHugo Villeneuve ret = regmap_read(pcf2127->regmap, 812420cc9e8SHugo Villeneuve pcf2127->cfg->ts[ts_id].inter_detect_reg, 813420cc9e8SHugo Villeneuve &ctrl); 8142f861984SMian Yousaf Kaukab if (ret) 8152f861984SMian Yousaf Kaukab return 0; 8162f861984SMian Yousaf Kaukab 817420cc9e8SHugo Villeneuve valid_inter = ctrl & pcf2127->cfg->ts[ts_id].inter_detect_bit; 818420cc9e8SHugo Villeneuve } 819420cc9e8SHugo Villeneuve 820420cc9e8SHugo Villeneuve if (!valid_low && !valid_inter) 8212f861984SMian Yousaf Kaukab return 0; 8222f861984SMian Yousaf Kaukab 823420cc9e8SHugo Villeneuve ret = pcf2127_rtc_ts_read(dev->parent, &ts, ts_id); 8242f861984SMian Yousaf Kaukab if (ret) 8252f861984SMian Yousaf Kaukab return 0; 82603623b4bSBruno Thomsen 82703623b4bSBruno Thomsen ret = pcf2127_wdt_active_ping(&pcf2127->wdd); 82803623b4bSBruno Thomsen if (ret) 82903623b4bSBruno Thomsen return ret; 8302f861984SMian Yousaf Kaukab } 8312f861984SMian Yousaf Kaukab return sprintf(buf, "%llu\n", (unsigned long long)ts); 832420cc9e8SHugo Villeneuve } 833420cc9e8SHugo Villeneuve 834420cc9e8SHugo Villeneuve static ssize_t timestamp0_show(struct device *dev, 835420cc9e8SHugo Villeneuve struct device_attribute *attr, char *buf) 836420cc9e8SHugo Villeneuve { 837420cc9e8SHugo Villeneuve return timestamp_show(dev, attr, buf, 0); 83803623b4bSBruno Thomsen }; 83903623b4bSBruno Thomsen 840afc505bfSHugo Villeneuve static ssize_t timestamp1_show(struct device *dev, 841afc505bfSHugo Villeneuve struct device_attribute *attr, char *buf) 842afc505bfSHugo Villeneuve { 843afc505bfSHugo Villeneuve return timestamp_show(dev, attr, buf, 1); 844afc505bfSHugo Villeneuve }; 845afc505bfSHugo Villeneuve 846afc505bfSHugo Villeneuve static ssize_t timestamp2_show(struct device *dev, 847afc505bfSHugo Villeneuve struct device_attribute *attr, char *buf) 848afc505bfSHugo Villeneuve { 849afc505bfSHugo Villeneuve return timestamp_show(dev, attr, buf, 2); 850afc505bfSHugo Villeneuve }; 851afc505bfSHugo Villeneuve 852afc505bfSHugo Villeneuve static ssize_t timestamp3_show(struct device *dev, 853afc505bfSHugo Villeneuve struct device_attribute *attr, char *buf) 854afc505bfSHugo Villeneuve { 855afc505bfSHugo Villeneuve return timestamp_show(dev, attr, buf, 3); 856afc505bfSHugo Villeneuve }; 857afc505bfSHugo Villeneuve 85803623b4bSBruno Thomsen static DEVICE_ATTR_RW(timestamp0); 859afc505bfSHugo Villeneuve static DEVICE_ATTR_RW(timestamp1); 860afc505bfSHugo Villeneuve static DEVICE_ATTR_RW(timestamp2); 861afc505bfSHugo Villeneuve static DEVICE_ATTR_RW(timestamp3); 86203623b4bSBruno Thomsen 86303623b4bSBruno Thomsen static struct attribute *pcf2127_attrs[] = { 86403623b4bSBruno Thomsen &dev_attr_timestamp0.attr, 86503623b4bSBruno Thomsen NULL 86603623b4bSBruno Thomsen }; 86703623b4bSBruno Thomsen 868afc505bfSHugo Villeneuve static struct attribute *pcf2131_attrs[] = { 869afc505bfSHugo Villeneuve &dev_attr_timestamp0.attr, 870afc505bfSHugo Villeneuve &dev_attr_timestamp1.attr, 871afc505bfSHugo Villeneuve &dev_attr_timestamp2.attr, 872afc505bfSHugo Villeneuve &dev_attr_timestamp3.attr, 873afc505bfSHugo Villeneuve NULL 874afc505bfSHugo Villeneuve }; 875afc505bfSHugo Villeneuve 876fd28ceb4SHugo Villeneuve static struct pcf21xx_config pcf21xx_cfg[] = { 877fd28ceb4SHugo Villeneuve [PCF2127] = { 878fd28ceb4SHugo Villeneuve .type = PCF2127, 879fd28ceb4SHugo Villeneuve .max_register = 0x1d, 880fd28ceb4SHugo Villeneuve .has_nvmem = 1, 881fd28ceb4SHugo Villeneuve .has_bit_wd_ctl_cd0 = 1, 882e1849b8fSHugo Villeneuve .has_int_a_b = 0, 8836211aceeSHugo Villeneuve .reg_time_base = PCF2127_REG_TIME_BASE, 8847c6f0db4SHugo Villeneuve .regs_alarm_base = PCF2127_REG_ALARM_BASE, 8856b57ec29SHugo Villeneuve .reg_wd_ctl = PCF2127_REG_WD_CTL, 8866b57ec29SHugo Villeneuve .reg_wd_val = PCF2127_REG_WD_VAL, 887fc16599eSHugo Villeneuve .reg_clkout = PCF2127_REG_CLKOUT, 888420cc9e8SHugo Villeneuve .ts_count = 1, 889420cc9e8SHugo Villeneuve .ts[0] = { 890420cc9e8SHugo Villeneuve .reg_base = PCF2127_REG_TS1_BASE, 891420cc9e8SHugo Villeneuve .gnd_detect_reg = PCF2127_REG_CTRL1, 892420cc9e8SHugo Villeneuve .gnd_detect_bit = PCF2127_BIT_CTRL1_TSF1, 893420cc9e8SHugo Villeneuve .inter_detect_reg = PCF2127_REG_CTRL2, 894420cc9e8SHugo Villeneuve .inter_detect_bit = PCF2127_BIT_CTRL2_TSF2, 895420cc9e8SHugo Villeneuve .ie_reg = PCF2127_REG_CTRL2, 896420cc9e8SHugo Villeneuve .ie_bit = PCF2127_BIT_CTRL2_TSIE, 897420cc9e8SHugo Villeneuve }, 898420cc9e8SHugo Villeneuve .attribute_group = { 899420cc9e8SHugo Villeneuve .attrs = pcf2127_attrs, 900420cc9e8SHugo Villeneuve }, 901fd28ceb4SHugo Villeneuve }, 902fd28ceb4SHugo Villeneuve [PCF2129] = { 903fd28ceb4SHugo Villeneuve .type = PCF2129, 904fd28ceb4SHugo Villeneuve .max_register = 0x19, 905fd28ceb4SHugo Villeneuve .has_nvmem = 0, 906fd28ceb4SHugo Villeneuve .has_bit_wd_ctl_cd0 = 0, 907e1849b8fSHugo Villeneuve .has_int_a_b = 0, 9086211aceeSHugo Villeneuve .reg_time_base = PCF2127_REG_TIME_BASE, 9097c6f0db4SHugo Villeneuve .regs_alarm_base = PCF2127_REG_ALARM_BASE, 9106b57ec29SHugo Villeneuve .reg_wd_ctl = PCF2127_REG_WD_CTL, 9116b57ec29SHugo Villeneuve .reg_wd_val = PCF2127_REG_WD_VAL, 912fc16599eSHugo Villeneuve .reg_clkout = PCF2127_REG_CLKOUT, 913420cc9e8SHugo Villeneuve .ts_count = 1, 914420cc9e8SHugo Villeneuve .ts[0] = { 915420cc9e8SHugo Villeneuve .reg_base = PCF2127_REG_TS1_BASE, 916420cc9e8SHugo Villeneuve .gnd_detect_reg = PCF2127_REG_CTRL1, 917420cc9e8SHugo Villeneuve .gnd_detect_bit = PCF2127_BIT_CTRL1_TSF1, 918420cc9e8SHugo Villeneuve .inter_detect_reg = PCF2127_REG_CTRL2, 919420cc9e8SHugo Villeneuve .inter_detect_bit = PCF2127_BIT_CTRL2_TSF2, 920420cc9e8SHugo Villeneuve .ie_reg = PCF2127_REG_CTRL2, 921420cc9e8SHugo Villeneuve .ie_bit = PCF2127_BIT_CTRL2_TSIE, 922420cc9e8SHugo Villeneuve }, 923420cc9e8SHugo Villeneuve .attribute_group = { 924420cc9e8SHugo Villeneuve .attrs = pcf2127_attrs, 925420cc9e8SHugo Villeneuve }, 926fd28ceb4SHugo Villeneuve }, 927afc505bfSHugo Villeneuve [PCF2131] = { 928afc505bfSHugo Villeneuve .type = PCF2131, 929afc505bfSHugo Villeneuve .max_register = 0x36, 930afc505bfSHugo Villeneuve .has_nvmem = 0, 931afc505bfSHugo Villeneuve .has_bit_wd_ctl_cd0 = 0, 932e1849b8fSHugo Villeneuve .has_int_a_b = 1, 933afc505bfSHugo Villeneuve .reg_time_base = PCF2131_REG_TIME_BASE, 934afc505bfSHugo Villeneuve .regs_alarm_base = PCF2131_REG_ALARM_BASE, 935afc505bfSHugo Villeneuve .reg_wd_ctl = PCF2131_REG_WD_CTL, 936afc505bfSHugo Villeneuve .reg_wd_val = PCF2131_REG_WD_VAL, 937afc505bfSHugo Villeneuve .reg_clkout = PCF2131_REG_CLKOUT, 938afc505bfSHugo Villeneuve .ts_count = 4, 939afc505bfSHugo Villeneuve .ts[0] = { 940afc505bfSHugo Villeneuve .reg_base = PCF2131_REG_TS1_BASE, 941afc505bfSHugo Villeneuve .gnd_detect_reg = PCF2131_REG_CTRL4, 942afc505bfSHugo Villeneuve .gnd_detect_bit = PCF2131_BIT_CTRL4_TSF1, 943afc505bfSHugo Villeneuve .inter_detect_bit = 0, 944afc505bfSHugo Villeneuve .ie_reg = PCF2131_REG_CTRL5, 945afc505bfSHugo Villeneuve .ie_bit = PCF2131_BIT_CTRL5_TSIE1, 946afc505bfSHugo Villeneuve }, 947afc505bfSHugo Villeneuve .ts[1] = { 948afc505bfSHugo Villeneuve .reg_base = PCF2131_REG_TS2_BASE, 949afc505bfSHugo Villeneuve .gnd_detect_reg = PCF2131_REG_CTRL4, 950afc505bfSHugo Villeneuve .gnd_detect_bit = PCF2131_BIT_CTRL4_TSF2, 951afc505bfSHugo Villeneuve .inter_detect_bit = 0, 952afc505bfSHugo Villeneuve .ie_reg = PCF2131_REG_CTRL5, 953afc505bfSHugo Villeneuve .ie_bit = PCF2131_BIT_CTRL5_TSIE2, 954afc505bfSHugo Villeneuve }, 955afc505bfSHugo Villeneuve .ts[2] = { 956afc505bfSHugo Villeneuve .reg_base = PCF2131_REG_TS3_BASE, 957afc505bfSHugo Villeneuve .gnd_detect_reg = PCF2131_REG_CTRL4, 958afc505bfSHugo Villeneuve .gnd_detect_bit = PCF2131_BIT_CTRL4_TSF3, 959afc505bfSHugo Villeneuve .inter_detect_bit = 0, 960afc505bfSHugo Villeneuve .ie_reg = PCF2131_REG_CTRL5, 961afc505bfSHugo Villeneuve .ie_bit = PCF2131_BIT_CTRL5_TSIE3, 962afc505bfSHugo Villeneuve }, 963afc505bfSHugo Villeneuve .ts[3] = { 964afc505bfSHugo Villeneuve .reg_base = PCF2131_REG_TS4_BASE, 965afc505bfSHugo Villeneuve .gnd_detect_reg = PCF2131_REG_CTRL4, 966afc505bfSHugo Villeneuve .gnd_detect_bit = PCF2131_BIT_CTRL4_TSF4, 967afc505bfSHugo Villeneuve .inter_detect_bit = 0, 968afc505bfSHugo Villeneuve .ie_reg = PCF2131_REG_CTRL5, 969afc505bfSHugo Villeneuve .ie_bit = PCF2131_BIT_CTRL5_TSIE4, 970afc505bfSHugo Villeneuve }, 971afc505bfSHugo Villeneuve .attribute_group = { 972afc505bfSHugo Villeneuve .attrs = pcf2131_attrs, 973afc505bfSHugo Villeneuve }, 974afc505bfSHugo Villeneuve }, 975fd28ceb4SHugo Villeneuve }; 976fd28ceb4SHugo Villeneuve 977420cc9e8SHugo Villeneuve /* 978420cc9e8SHugo Villeneuve * Enable timestamp function and corresponding interrupt(s). 979420cc9e8SHugo Villeneuve */ 980420cc9e8SHugo Villeneuve static int pcf2127_enable_ts(struct device *dev, int ts_id) 981420cc9e8SHugo Villeneuve { 982420cc9e8SHugo Villeneuve struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 983420cc9e8SHugo Villeneuve int ret; 984420cc9e8SHugo Villeneuve 985420cc9e8SHugo Villeneuve if (ts_id >= pcf2127->cfg->ts_count) { 986420cc9e8SHugo Villeneuve dev_err(dev, "%s: invalid tamper detection ID (%d)\n", 987420cc9e8SHugo Villeneuve __func__, ts_id); 988420cc9e8SHugo Villeneuve return -EINVAL; 989420cc9e8SHugo Villeneuve } 990420cc9e8SHugo Villeneuve 991420cc9e8SHugo Villeneuve /* Enable timestamp function. */ 992420cc9e8SHugo Villeneuve ret = regmap_update_bits(pcf2127->regmap, 993420cc9e8SHugo Villeneuve pcf2127->cfg->ts[ts_id].reg_base, 994420cc9e8SHugo Villeneuve PCF2127_BIT_TS_CTRL_TSOFF | 995420cc9e8SHugo Villeneuve PCF2127_BIT_TS_CTRL_TSM, 996420cc9e8SHugo Villeneuve PCF2127_BIT_TS_CTRL_TSM); 997420cc9e8SHugo Villeneuve if (ret) { 998420cc9e8SHugo Villeneuve dev_err(dev, "%s: tamper detection config (ts%d_ctrl) failed\n", 999420cc9e8SHugo Villeneuve __func__, ts_id); 1000420cc9e8SHugo Villeneuve return ret; 1001420cc9e8SHugo Villeneuve } 1002420cc9e8SHugo Villeneuve 1003420cc9e8SHugo Villeneuve /* TS input pin driven to GND detection is supported by all variants. 1004420cc9e8SHugo Villeneuve * Make sure that interrupt bit is defined. 1005420cc9e8SHugo Villeneuve */ 1006420cc9e8SHugo Villeneuve if (pcf2127->cfg->ts[ts_id].gnd_detect_bit == 0) { 1007420cc9e8SHugo Villeneuve dev_err(dev, "%s: tamper detection to GND configuration invalid\n", 1008420cc9e8SHugo Villeneuve __func__); 1009420cc9e8SHugo Villeneuve return ret; 1010420cc9e8SHugo Villeneuve } 1011420cc9e8SHugo Villeneuve 1012420cc9e8SHugo Villeneuve /* 1013420cc9e8SHugo Villeneuve * Enable interrupt generation when TSF timestamp flag is set. 1014420cc9e8SHugo Villeneuve * Interrupt signals are open-drain outputs and can be left floating if 1015420cc9e8SHugo Villeneuve * unused. 1016420cc9e8SHugo Villeneuve */ 1017420cc9e8SHugo Villeneuve ret = regmap_update_bits(pcf2127->regmap, pcf2127->cfg->ts[ts_id].ie_reg, 1018420cc9e8SHugo Villeneuve pcf2127->cfg->ts[ts_id].ie_bit, 1019420cc9e8SHugo Villeneuve pcf2127->cfg->ts[ts_id].ie_bit); 1020420cc9e8SHugo Villeneuve if (ret) { 1021420cc9e8SHugo Villeneuve dev_err(dev, "%s: tamper detection TSIE%d config failed\n", 1022420cc9e8SHugo Villeneuve __func__, ts_id); 1023420cc9e8SHugo Villeneuve return ret; 1024420cc9e8SHugo Villeneuve } 1025420cc9e8SHugo Villeneuve 1026420cc9e8SHugo Villeneuve return ret; 1027420cc9e8SHugo Villeneuve } 1028420cc9e8SHugo Villeneuve 1029e1849b8fSHugo Villeneuve /* Route all interrupt sources to INT A pin. */ 1030e1849b8fSHugo Villeneuve static int pcf2127_configure_interrupt_pins(struct device *dev) 1031e1849b8fSHugo Villeneuve { 1032e1849b8fSHugo Villeneuve struct pcf2127 *pcf2127 = dev_get_drvdata(dev); 1033e1849b8fSHugo Villeneuve int ret; 1034e1849b8fSHugo Villeneuve 1035e1849b8fSHugo Villeneuve /* Mask bits need to be cleared to enable corresponding 1036e1849b8fSHugo Villeneuve * interrupt source. 1037e1849b8fSHugo Villeneuve */ 1038e1849b8fSHugo Villeneuve ret = regmap_write(pcf2127->regmap, 1039e1849b8fSHugo Villeneuve PCF2131_REG_INT_A_MASK1, 0); 1040e1849b8fSHugo Villeneuve if (ret) 1041e1849b8fSHugo Villeneuve return ret; 1042e1849b8fSHugo Villeneuve 1043e1849b8fSHugo Villeneuve ret = regmap_write(pcf2127->regmap, 1044e1849b8fSHugo Villeneuve PCF2131_REG_INT_A_MASK2, 0); 1045e1849b8fSHugo Villeneuve if (ret) 1046e1849b8fSHugo Villeneuve return ret; 1047e1849b8fSHugo Villeneuve 1048e1849b8fSHugo Villeneuve return ret; 1049e1849b8fSHugo Villeneuve } 1050e1849b8fSHugo Villeneuve 1051907b3262SAkinobu Mita static int pcf2127_probe(struct device *dev, struct regmap *regmap, 1052fd28ceb4SHugo Villeneuve int alarm_irq, const char *name, const struct pcf21xx_config *config) 105318cb6368SRenaud Cerrato { 105418cb6368SRenaud Cerrato struct pcf2127 *pcf2127; 1055d6c3029fSUwe Kleine-König int ret = 0; 105615f57b3eSPhilipp Rosenberger unsigned int val; 105718cb6368SRenaud Cerrato 1058907b3262SAkinobu Mita dev_dbg(dev, "%s\n", __func__); 105918cb6368SRenaud Cerrato 1060907b3262SAkinobu Mita pcf2127 = devm_kzalloc(dev, sizeof(*pcf2127), GFP_KERNEL); 106118cb6368SRenaud Cerrato if (!pcf2127) 106218cb6368SRenaud Cerrato return -ENOMEM; 106318cb6368SRenaud Cerrato 1064907b3262SAkinobu Mita pcf2127->regmap = regmap; 1065fd28ceb4SHugo Villeneuve pcf2127->cfg = config; 106618cb6368SRenaud Cerrato 1067907b3262SAkinobu Mita dev_set_drvdata(dev, pcf2127); 1068907b3262SAkinobu Mita 1069e788771cSBruno Thomsen pcf2127->rtc = devm_rtc_allocate_device(dev); 1070d6c3029fSUwe Kleine-König if (IS_ERR(pcf2127->rtc)) 1071d6c3029fSUwe Kleine-König return PTR_ERR(pcf2127->rtc); 107218cb6368SRenaud Cerrato 1073e788771cSBruno Thomsen pcf2127->rtc->ops = &pcf2127_rtc_ops; 1074b139bb5cSAlexandre Belloni pcf2127->rtc->range_min = RTC_TIMESTAMP_BEGIN_2000; 1075b139bb5cSAlexandre Belloni pcf2127->rtc->range_max = RTC_TIMESTAMP_END_2099; 1076b139bb5cSAlexandre Belloni pcf2127->rtc->set_start_time = true; /* Sets actual start to 1970 */ 1077bda10273SAlexandre Belloni set_bit(RTC_FEATURE_ALARM_RES_2S, pcf2127->rtc->features); 1078689fafd5SAlexandre Belloni clear_bit(RTC_FEATURE_UPDATE_INTERRUPT, pcf2127->rtc->features); 107925cbe9c8SAlexandre Belloni clear_bit(RTC_FEATURE_ALARM, pcf2127->rtc->features); 1080e788771cSBruno Thomsen 108135425bafSBiwen Li if (alarm_irq > 0) { 1082d4785b46SHugo Villeneuve unsigned long flags; 1083d4785b46SHugo Villeneuve 1084d4785b46SHugo Villeneuve /* 1085d4785b46SHugo Villeneuve * If flags = 0, devm_request_threaded_irq() will use IRQ flags 1086d4785b46SHugo Villeneuve * obtained from device tree. 1087d4785b46SHugo Villeneuve */ 1088d4785b46SHugo Villeneuve if (dev_fwnode(dev)) 1089d4785b46SHugo Villeneuve flags = 0; 1090d4785b46SHugo Villeneuve else 1091d4785b46SHugo Villeneuve flags = IRQF_TRIGGER_LOW; 1092d4785b46SHugo Villeneuve 109327006416SAlexandre Belloni ret = devm_request_threaded_irq(dev, alarm_irq, NULL, 109427006416SAlexandre Belloni pcf2127_rtc_irq, 1095d4785b46SHugo Villeneuve flags | IRQF_ONESHOT, 10968a914bacSLiam Beguin dev_name(dev), dev); 10978a914bacSLiam Beguin if (ret) { 10988a914bacSLiam Beguin dev_err(dev, "failed to request alarm irq\n"); 10998a914bacSLiam Beguin return ret; 11008a914bacSLiam Beguin } 11012f861984SMian Yousaf Kaukab pcf2127->irq_enabled = true; 11028a914bacSLiam Beguin } 11038a914bacSLiam Beguin 110435425bafSBiwen Li if (alarm_irq > 0 || device_property_read_bool(dev, "wakeup-source")) { 11058a914bacSLiam Beguin device_init_wakeup(dev, true); 110625cbe9c8SAlexandre Belloni set_bit(RTC_FEATURE_ALARM, pcf2127->rtc->features); 11078a914bacSLiam Beguin } 11088a914bacSLiam Beguin 1109e1849b8fSHugo Villeneuve if (pcf2127->cfg->has_int_a_b) { 1110e1849b8fSHugo Villeneuve /* Configure int A/B pins, independently of alarm_irq. */ 1111e1849b8fSHugo Villeneuve ret = pcf2127_configure_interrupt_pins(dev); 1112e1849b8fSHugo Villeneuve if (ret) { 1113e1849b8fSHugo Villeneuve dev_err(dev, "failed to configure interrupt pins\n"); 1114e1849b8fSHugo Villeneuve return ret; 1115e1849b8fSHugo Villeneuve } 1116e1849b8fSHugo Villeneuve } 1117e1849b8fSHugo Villeneuve 1118fd28ceb4SHugo Villeneuve if (pcf2127->cfg->has_nvmem) { 1119d6c3029fSUwe Kleine-König struct nvmem_config nvmem_cfg = { 1120d6c3029fSUwe Kleine-König .priv = pcf2127, 1121d6c3029fSUwe Kleine-König .reg_read = pcf2127_nvmem_read, 1122d6c3029fSUwe Kleine-König .reg_write = pcf2127_nvmem_write, 1123d6c3029fSUwe Kleine-König .size = 512, 1124d6c3029fSUwe Kleine-König }; 1125d6c3029fSUwe Kleine-König 11263a905c2dSBartosz Golaszewski ret = devm_rtc_nvmem_register(pcf2127->rtc, &nvmem_cfg); 1127d6c3029fSUwe Kleine-König } 1128d6c3029fSUwe Kleine-König 11290e735eaaSBruno Thomsen /* 1130b9ac079aSPhilipp Rosenberger * The "Power-On Reset Override" facility prevents the RTC to do a reset 1131b9ac079aSPhilipp Rosenberger * after power on. For normal operation the PORO must be disabled. 1132b9ac079aSPhilipp Rosenberger */ 1133b9ac079aSPhilipp Rosenberger regmap_clear_bits(pcf2127->regmap, PCF2127_REG_CTRL1, 1134b9ac079aSPhilipp Rosenberger PCF2127_BIT_CTRL1_POR_OVRD); 1135b9ac079aSPhilipp Rosenberger 1136fc16599eSHugo Villeneuve ret = regmap_read(pcf2127->regmap, pcf2127->cfg->reg_clkout, &val); 113715f57b3eSPhilipp Rosenberger if (ret < 0) 113815f57b3eSPhilipp Rosenberger return ret; 113915f57b3eSPhilipp Rosenberger 114015f57b3eSPhilipp Rosenberger if (!(val & PCF2127_BIT_CLKOUT_OTPR)) { 1141fc16599eSHugo Villeneuve ret = regmap_set_bits(pcf2127->regmap, pcf2127->cfg->reg_clkout, 114215f57b3eSPhilipp Rosenberger PCF2127_BIT_CLKOUT_OTPR); 114315f57b3eSPhilipp Rosenberger if (ret < 0) 114415f57b3eSPhilipp Rosenberger return ret; 114515f57b3eSPhilipp Rosenberger 114615f57b3eSPhilipp Rosenberger msleep(100); 114715f57b3eSPhilipp Rosenberger } 114815f57b3eSPhilipp Rosenberger 1149b9ac079aSPhilipp Rosenberger /* 11500e735eaaSBruno Thomsen * Watchdog timer enabled and reset pin /RST activated when timed out. 11510e735eaaSBruno Thomsen * Select 1Hz clock source for watchdog timer. 11520e735eaaSBruno Thomsen * Note: Countdown timer disabled and not available. 1153afc505bfSHugo Villeneuve * For pca2129, pcf2129 and pcf2131, only bit[7] is for Symbol WD_CD 11542843d565SBiwen Li * of register watchdg_tim_ctl. The bit[6] is labeled 11552843d565SBiwen Li * as T. Bits labeled as T must always be written with 11562843d565SBiwen Li * logic 0. 11570e735eaaSBruno Thomsen */ 11586b57ec29SHugo Villeneuve ret = regmap_update_bits(pcf2127->regmap, pcf2127->cfg->reg_wd_ctl, 11590e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_CD1 | 11600e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_CD0 | 11610e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_TF1 | 11620e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_TF0, 11630e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_CD1 | 1164fd28ceb4SHugo Villeneuve (pcf2127->cfg->has_bit_wd_ctl_cd0 ? PCF2127_BIT_WD_CTL_CD0 : 0) | 11650e735eaaSBruno Thomsen PCF2127_BIT_WD_CTL_TF1); 11660e735eaaSBruno Thomsen if (ret) { 11670e735eaaSBruno Thomsen dev_err(dev, "%s: watchdog config (wd_ctl) failed\n", __func__); 11680e735eaaSBruno Thomsen return ret; 11690e735eaaSBruno Thomsen } 11700e735eaaSBruno Thomsen 11715d78533aSUwe Kleine-König pcf2127_watchdog_init(dev, pcf2127); 11720e735eaaSBruno Thomsen 117303623b4bSBruno Thomsen /* 117403623b4bSBruno Thomsen * Disable battery low/switch-over timestamp and interrupts. 117503623b4bSBruno Thomsen * Clear battery interrupt flags which can block new trigger events. 117603623b4bSBruno Thomsen * Note: This is the default chip behaviour but added to ensure 117703623b4bSBruno Thomsen * correct tamper timestamp and interrupt function. 117803623b4bSBruno Thomsen */ 117903623b4bSBruno Thomsen ret = regmap_update_bits(pcf2127->regmap, PCF2127_REG_CTRL3, 118003623b4bSBruno Thomsen PCF2127_BIT_CTRL3_BTSE | 118103623b4bSBruno Thomsen PCF2127_BIT_CTRL3_BIE | 118203623b4bSBruno Thomsen PCF2127_BIT_CTRL3_BLIE, 0); 118303623b4bSBruno Thomsen if (ret) { 118403623b4bSBruno Thomsen dev_err(dev, "%s: interrupt config (ctrl3) failed\n", 118503623b4bSBruno Thomsen __func__); 118603623b4bSBruno Thomsen return ret; 118703623b4bSBruno Thomsen } 118803623b4bSBruno Thomsen 118903623b4bSBruno Thomsen /* 1190420cc9e8SHugo Villeneuve * Enable timestamp functions 1 to 4. 119103623b4bSBruno Thomsen */ 1192420cc9e8SHugo Villeneuve for (int i = 0; i < pcf2127->cfg->ts_count; i++) { 1193420cc9e8SHugo Villeneuve ret = pcf2127_enable_ts(dev, i); 1194420cc9e8SHugo Villeneuve if (ret) 119503623b4bSBruno Thomsen return ret; 119603623b4bSBruno Thomsen } 119703623b4bSBruno Thomsen 1198420cc9e8SHugo Villeneuve ret = rtc_add_group(pcf2127->rtc, &pcf2127->cfg->attribute_group); 119903623b4bSBruno Thomsen if (ret) { 120003623b4bSBruno Thomsen dev_err(dev, "%s: tamper sysfs registering failed\n", 120103623b4bSBruno Thomsen __func__); 120203623b4bSBruno Thomsen return ret; 120303623b4bSBruno Thomsen } 120403623b4bSBruno Thomsen 1205fdcfd854SBartosz Golaszewski return devm_rtc_register_device(pcf2127->rtc); 120618cb6368SRenaud Cerrato } 120718cb6368SRenaud Cerrato 120818cb6368SRenaud Cerrato #ifdef CONFIG_OF 120918cb6368SRenaud Cerrato static const struct of_device_id pcf2127_of_match[] = { 1210fd28ceb4SHugo Villeneuve { .compatible = "nxp,pcf2127", .data = &pcf21xx_cfg[PCF2127] }, 1211fd28ceb4SHugo Villeneuve { .compatible = "nxp,pcf2129", .data = &pcf21xx_cfg[PCF2129] }, 1212fd28ceb4SHugo Villeneuve { .compatible = "nxp,pca2129", .data = &pcf21xx_cfg[PCF2129] }, 1213afc505bfSHugo Villeneuve { .compatible = "nxp,pcf2131", .data = &pcf21xx_cfg[PCF2131] }, 121418cb6368SRenaud Cerrato {} 121518cb6368SRenaud Cerrato }; 121618cb6368SRenaud Cerrato MODULE_DEVICE_TABLE(of, pcf2127_of_match); 121718cb6368SRenaud Cerrato #endif 121818cb6368SRenaud Cerrato 12199408ec1aSAkinobu Mita #if IS_ENABLED(CONFIG_I2C) 12209408ec1aSAkinobu Mita 1221907b3262SAkinobu Mita static int pcf2127_i2c_write(void *context, const void *data, size_t count) 1222907b3262SAkinobu Mita { 1223907b3262SAkinobu Mita struct device *dev = context; 1224907b3262SAkinobu Mita struct i2c_client *client = to_i2c_client(dev); 1225907b3262SAkinobu Mita int ret; 1226907b3262SAkinobu Mita 1227907b3262SAkinobu Mita ret = i2c_master_send(client, data, count); 1228907b3262SAkinobu Mita if (ret != count) 1229907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 1230907b3262SAkinobu Mita 1231907b3262SAkinobu Mita return 0; 1232907b3262SAkinobu Mita } 1233907b3262SAkinobu Mita 1234907b3262SAkinobu Mita static int pcf2127_i2c_gather_write(void *context, 1235907b3262SAkinobu Mita const void *reg, size_t reg_size, 1236907b3262SAkinobu Mita const void *val, size_t val_size) 1237907b3262SAkinobu Mita { 1238907b3262SAkinobu Mita struct device *dev = context; 1239907b3262SAkinobu Mita struct i2c_client *client = to_i2c_client(dev); 1240907b3262SAkinobu Mita int ret; 1241907b3262SAkinobu Mita void *buf; 1242907b3262SAkinobu Mita 1243907b3262SAkinobu Mita if (WARN_ON(reg_size != 1)) 1244907b3262SAkinobu Mita return -EINVAL; 1245907b3262SAkinobu Mita 1246907b3262SAkinobu Mita buf = kmalloc(val_size + 1, GFP_KERNEL); 1247907b3262SAkinobu Mita if (!buf) 1248907b3262SAkinobu Mita return -ENOMEM; 1249907b3262SAkinobu Mita 1250907b3262SAkinobu Mita memcpy(buf, reg, 1); 1251907b3262SAkinobu Mita memcpy(buf + 1, val, val_size); 1252907b3262SAkinobu Mita 1253907b3262SAkinobu Mita ret = i2c_master_send(client, buf, val_size + 1); 12549bde0afbSXulin Sun 12559bde0afbSXulin Sun kfree(buf); 12569bde0afbSXulin Sun 1257907b3262SAkinobu Mita if (ret != val_size + 1) 1258907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 1259907b3262SAkinobu Mita 1260907b3262SAkinobu Mita return 0; 1261907b3262SAkinobu Mita } 1262907b3262SAkinobu Mita 1263907b3262SAkinobu Mita static int pcf2127_i2c_read(void *context, const void *reg, size_t reg_size, 1264907b3262SAkinobu Mita void *val, size_t val_size) 1265907b3262SAkinobu Mita { 1266907b3262SAkinobu Mita struct device *dev = context; 1267907b3262SAkinobu Mita struct i2c_client *client = to_i2c_client(dev); 1268907b3262SAkinobu Mita int ret; 1269907b3262SAkinobu Mita 1270907b3262SAkinobu Mita if (WARN_ON(reg_size != 1)) 1271907b3262SAkinobu Mita return -EINVAL; 1272907b3262SAkinobu Mita 1273907b3262SAkinobu Mita ret = i2c_master_send(client, reg, 1); 1274907b3262SAkinobu Mita if (ret != 1) 1275907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 1276907b3262SAkinobu Mita 1277907b3262SAkinobu Mita ret = i2c_master_recv(client, val, val_size); 1278907b3262SAkinobu Mita if (ret != val_size) 1279907b3262SAkinobu Mita return ret < 0 ? ret : -EIO; 1280907b3262SAkinobu Mita 1281907b3262SAkinobu Mita return 0; 1282907b3262SAkinobu Mita } 1283907b3262SAkinobu Mita 1284907b3262SAkinobu Mita /* 1285907b3262SAkinobu Mita * The reason we need this custom regmap_bus instead of using regmap_init_i2c() 1286907b3262SAkinobu Mita * is that the STOP condition is required between set register address and 1287907b3262SAkinobu Mita * read register data when reading from registers. 1288907b3262SAkinobu Mita */ 1289907b3262SAkinobu Mita static const struct regmap_bus pcf2127_i2c_regmap = { 1290907b3262SAkinobu Mita .write = pcf2127_i2c_write, 1291907b3262SAkinobu Mita .gather_write = pcf2127_i2c_gather_write, 1292907b3262SAkinobu Mita .read = pcf2127_i2c_read, 129318cb6368SRenaud Cerrato }; 129418cb6368SRenaud Cerrato 1295907b3262SAkinobu Mita static struct i2c_driver pcf2127_i2c_driver; 1296907b3262SAkinobu Mita 12975418e595SUwe Kleine-König static const struct i2c_device_id pcf2127_i2c_id[] = { 1298fd28ceb4SHugo Villeneuve { "pcf2127", PCF2127 }, 1299fd28ceb4SHugo Villeneuve { "pcf2129", PCF2129 }, 1300fd28ceb4SHugo Villeneuve { "pca2129", PCF2129 }, 1301afc505bfSHugo Villeneuve { "pcf2131", PCF2131 }, 13025418e595SUwe Kleine-König { } 13035418e595SUwe Kleine-König }; 13045418e595SUwe Kleine-König MODULE_DEVICE_TABLE(i2c, pcf2127_i2c_id); 13055418e595SUwe Kleine-König 13065418e595SUwe Kleine-König static int pcf2127_i2c_probe(struct i2c_client *client) 1307907b3262SAkinobu Mita { 1308907b3262SAkinobu Mita struct regmap *regmap; 1309fd28ceb4SHugo Villeneuve static struct regmap_config config = { 1310907b3262SAkinobu Mita .reg_bits = 8, 1311907b3262SAkinobu Mita .val_bits = 8, 1312907b3262SAkinobu Mita }; 1313fd28ceb4SHugo Villeneuve const struct pcf21xx_config *variant; 1314907b3262SAkinobu Mita 1315907b3262SAkinobu Mita if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) 1316907b3262SAkinobu Mita return -ENODEV; 1317907b3262SAkinobu Mita 1318fd28ceb4SHugo Villeneuve if (client->dev.of_node) { 1319fd28ceb4SHugo Villeneuve variant = of_device_get_match_data(&client->dev); 1320fd28ceb4SHugo Villeneuve if (!variant) 1321fd28ceb4SHugo Villeneuve return -ENODEV; 1322fd28ceb4SHugo Villeneuve } else { 1323fd28ceb4SHugo Villeneuve enum pcf21xx_type type = 1324fd28ceb4SHugo Villeneuve i2c_match_id(pcf2127_i2c_id, client)->driver_data; 1325fd28ceb4SHugo Villeneuve 1326fd28ceb4SHugo Villeneuve if (type >= PCF21XX_LAST_ID) 1327fd28ceb4SHugo Villeneuve return -ENODEV; 1328fd28ceb4SHugo Villeneuve variant = &pcf21xx_cfg[type]; 1329fd28ceb4SHugo Villeneuve } 1330fd28ceb4SHugo Villeneuve 1331fd28ceb4SHugo Villeneuve config.max_register = variant->max_register, 1332fd28ceb4SHugo Villeneuve 1333907b3262SAkinobu Mita regmap = devm_regmap_init(&client->dev, &pcf2127_i2c_regmap, 1334907b3262SAkinobu Mita &client->dev, &config); 1335907b3262SAkinobu Mita if (IS_ERR(regmap)) { 1336907b3262SAkinobu Mita dev_err(&client->dev, "%s: regmap allocation failed: %ld\n", 1337907b3262SAkinobu Mita __func__, PTR_ERR(regmap)); 1338907b3262SAkinobu Mita return PTR_ERR(regmap); 1339907b3262SAkinobu Mita } 1340907b3262SAkinobu Mita 134127006416SAlexandre Belloni return pcf2127_probe(&client->dev, regmap, client->irq, 1342fd28ceb4SHugo Villeneuve pcf2127_i2c_driver.driver.name, variant); 1343907b3262SAkinobu Mita } 1344907b3262SAkinobu Mita 1345907b3262SAkinobu Mita static struct i2c_driver pcf2127_i2c_driver = { 1346907b3262SAkinobu Mita .driver = { 1347907b3262SAkinobu Mita .name = "rtc-pcf2127-i2c", 1348907b3262SAkinobu Mita .of_match_table = of_match_ptr(pcf2127_of_match), 1349907b3262SAkinobu Mita }, 135031b0cecbSUwe Kleine-König .probe = pcf2127_i2c_probe, 1351907b3262SAkinobu Mita .id_table = pcf2127_i2c_id, 1352907b3262SAkinobu Mita }; 13539408ec1aSAkinobu Mita 13549408ec1aSAkinobu Mita static int pcf2127_i2c_register_driver(void) 13559408ec1aSAkinobu Mita { 13569408ec1aSAkinobu Mita return i2c_add_driver(&pcf2127_i2c_driver); 13579408ec1aSAkinobu Mita } 13589408ec1aSAkinobu Mita 13599408ec1aSAkinobu Mita static void pcf2127_i2c_unregister_driver(void) 13609408ec1aSAkinobu Mita { 13619408ec1aSAkinobu Mita i2c_del_driver(&pcf2127_i2c_driver); 13629408ec1aSAkinobu Mita } 13639408ec1aSAkinobu Mita 13649408ec1aSAkinobu Mita #else 13659408ec1aSAkinobu Mita 13669408ec1aSAkinobu Mita static int pcf2127_i2c_register_driver(void) 13679408ec1aSAkinobu Mita { 13689408ec1aSAkinobu Mita return 0; 13699408ec1aSAkinobu Mita } 13709408ec1aSAkinobu Mita 13719408ec1aSAkinobu Mita static void pcf2127_i2c_unregister_driver(void) 13729408ec1aSAkinobu Mita { 13739408ec1aSAkinobu Mita } 13749408ec1aSAkinobu Mita 13759408ec1aSAkinobu Mita #endif 13769408ec1aSAkinobu Mita 13779408ec1aSAkinobu Mita #if IS_ENABLED(CONFIG_SPI_MASTER) 13789408ec1aSAkinobu Mita 13799408ec1aSAkinobu Mita static struct spi_driver pcf2127_spi_driver; 1380fd28ceb4SHugo Villeneuve static const struct spi_device_id pcf2127_spi_id[]; 13819408ec1aSAkinobu Mita 13829408ec1aSAkinobu Mita static int pcf2127_spi_probe(struct spi_device *spi) 13839408ec1aSAkinobu Mita { 1384fd28ceb4SHugo Villeneuve static struct regmap_config config = { 13859408ec1aSAkinobu Mita .reg_bits = 8, 13869408ec1aSAkinobu Mita .val_bits = 8, 13879408ec1aSAkinobu Mita .read_flag_mask = 0xa0, 13889408ec1aSAkinobu Mita .write_flag_mask = 0x20, 13899408ec1aSAkinobu Mita }; 13909408ec1aSAkinobu Mita struct regmap *regmap; 1391fd28ceb4SHugo Villeneuve const struct pcf21xx_config *variant; 1392fd28ceb4SHugo Villeneuve 1393fd28ceb4SHugo Villeneuve if (spi->dev.of_node) { 1394fd28ceb4SHugo Villeneuve variant = of_device_get_match_data(&spi->dev); 1395fd28ceb4SHugo Villeneuve if (!variant) 1396fd28ceb4SHugo Villeneuve return -ENODEV; 1397fd28ceb4SHugo Villeneuve } else { 1398fd28ceb4SHugo Villeneuve enum pcf21xx_type type = spi_get_device_id(spi)->driver_data; 1399fd28ceb4SHugo Villeneuve 1400fd28ceb4SHugo Villeneuve if (type >= PCF21XX_LAST_ID) 1401fd28ceb4SHugo Villeneuve return -ENODEV; 1402fd28ceb4SHugo Villeneuve variant = &pcf21xx_cfg[type]; 1403fd28ceb4SHugo Villeneuve } 1404fd28ceb4SHugo Villeneuve 1405fd28ceb4SHugo Villeneuve config.max_register = variant->max_register, 14069408ec1aSAkinobu Mita 14079408ec1aSAkinobu Mita regmap = devm_regmap_init_spi(spi, &config); 14089408ec1aSAkinobu Mita if (IS_ERR(regmap)) { 14099408ec1aSAkinobu Mita dev_err(&spi->dev, "%s: regmap allocation failed: %ld\n", 14109408ec1aSAkinobu Mita __func__, PTR_ERR(regmap)); 14119408ec1aSAkinobu Mita return PTR_ERR(regmap); 14129408ec1aSAkinobu Mita } 14139408ec1aSAkinobu Mita 141427006416SAlexandre Belloni return pcf2127_probe(&spi->dev, regmap, spi->irq, 141527006416SAlexandre Belloni pcf2127_spi_driver.driver.name, 1416fd28ceb4SHugo Villeneuve variant); 14179408ec1aSAkinobu Mita } 14189408ec1aSAkinobu Mita 14199408ec1aSAkinobu Mita static const struct spi_device_id pcf2127_spi_id[] = { 1420fd28ceb4SHugo Villeneuve { "pcf2127", PCF2127 }, 1421fd28ceb4SHugo Villeneuve { "pcf2129", PCF2129 }, 1422fd28ceb4SHugo Villeneuve { "pca2129", PCF2129 }, 1423afc505bfSHugo Villeneuve { "pcf2131", PCF2131 }, 14249408ec1aSAkinobu Mita { } 14259408ec1aSAkinobu Mita }; 14269408ec1aSAkinobu Mita MODULE_DEVICE_TABLE(spi, pcf2127_spi_id); 14279408ec1aSAkinobu Mita 14289408ec1aSAkinobu Mita static struct spi_driver pcf2127_spi_driver = { 14299408ec1aSAkinobu Mita .driver = { 14309408ec1aSAkinobu Mita .name = "rtc-pcf2127-spi", 14319408ec1aSAkinobu Mita .of_match_table = of_match_ptr(pcf2127_of_match), 14329408ec1aSAkinobu Mita }, 14339408ec1aSAkinobu Mita .probe = pcf2127_spi_probe, 14349408ec1aSAkinobu Mita .id_table = pcf2127_spi_id, 14359408ec1aSAkinobu Mita }; 14369408ec1aSAkinobu Mita 14379408ec1aSAkinobu Mita static int pcf2127_spi_register_driver(void) 14389408ec1aSAkinobu Mita { 14399408ec1aSAkinobu Mita return spi_register_driver(&pcf2127_spi_driver); 14409408ec1aSAkinobu Mita } 14419408ec1aSAkinobu Mita 14429408ec1aSAkinobu Mita static void pcf2127_spi_unregister_driver(void) 14439408ec1aSAkinobu Mita { 14449408ec1aSAkinobu Mita spi_unregister_driver(&pcf2127_spi_driver); 14459408ec1aSAkinobu Mita } 14469408ec1aSAkinobu Mita 14479408ec1aSAkinobu Mita #else 14489408ec1aSAkinobu Mita 14499408ec1aSAkinobu Mita static int pcf2127_spi_register_driver(void) 14509408ec1aSAkinobu Mita { 14519408ec1aSAkinobu Mita return 0; 14529408ec1aSAkinobu Mita } 14539408ec1aSAkinobu Mita 14549408ec1aSAkinobu Mita static void pcf2127_spi_unregister_driver(void) 14559408ec1aSAkinobu Mita { 14569408ec1aSAkinobu Mita } 14579408ec1aSAkinobu Mita 14589408ec1aSAkinobu Mita #endif 14599408ec1aSAkinobu Mita 14609408ec1aSAkinobu Mita static int __init pcf2127_init(void) 14619408ec1aSAkinobu Mita { 14629408ec1aSAkinobu Mita int ret; 14639408ec1aSAkinobu Mita 14649408ec1aSAkinobu Mita ret = pcf2127_i2c_register_driver(); 14659408ec1aSAkinobu Mita if (ret) { 14669408ec1aSAkinobu Mita pr_err("Failed to register pcf2127 i2c driver: %d\n", ret); 14679408ec1aSAkinobu Mita return ret; 14689408ec1aSAkinobu Mita } 14699408ec1aSAkinobu Mita 14709408ec1aSAkinobu Mita ret = pcf2127_spi_register_driver(); 14719408ec1aSAkinobu Mita if (ret) { 14729408ec1aSAkinobu Mita pr_err("Failed to register pcf2127 spi driver: %d\n", ret); 14739408ec1aSAkinobu Mita pcf2127_i2c_unregister_driver(); 14749408ec1aSAkinobu Mita } 14759408ec1aSAkinobu Mita 14769408ec1aSAkinobu Mita return ret; 14779408ec1aSAkinobu Mita } 14789408ec1aSAkinobu Mita module_init(pcf2127_init) 14799408ec1aSAkinobu Mita 14809408ec1aSAkinobu Mita static void __exit pcf2127_exit(void) 14819408ec1aSAkinobu Mita { 14829408ec1aSAkinobu Mita pcf2127_spi_unregister_driver(); 14839408ec1aSAkinobu Mita pcf2127_i2c_unregister_driver(); 14849408ec1aSAkinobu Mita } 14859408ec1aSAkinobu Mita module_exit(pcf2127_exit) 148618cb6368SRenaud Cerrato 148718cb6368SRenaud Cerrato MODULE_AUTHOR("Renaud Cerrato <r.cerrato@til-technologies.fr>"); 1488afc505bfSHugo Villeneuve MODULE_DESCRIPTION("NXP PCF2127/29/31 RTC driver"); 14894d8318bcSUwe Kleine-König MODULE_LICENSE("GPL v2"); 1490