Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 | // SPDX-License-Identifier: GPL-2.0-only /* * Analog Devices DS1672 I2C RTC driver * * Copyright 2025 Gateworks Corporation. */ #include <dm.h> #include <i2c.h> #include <rtc.h> #include <dm/device_compat.h> /* Registers */ #define DS1672_REG_CNT_BASE 0 #define DS1672_REG_CONTROL 4 #define DS1672_REG_TRICKLE 5 #define DS1672_REG_CONTROL_EOSC 0x80 static int ds1672_read_time(struct udevice *dev, struct rtc_time *tm) { time64_t secs; u8 regs[4]; int ret; ret = dm_i2c_read(dev, DS1672_REG_CONTROL, regs, 1); if (ret) return ret; if (regs[0] & DS1672_REG_CONTROL_EOSC) { dev_err(dev, "Oscillator not enabled. Set time to enable.\n"); return -EINVAL; } ret = dm_i2c_read(dev, DS1672_REG_CNT_BASE, regs, 4); if (ret) return ret; dev_dbg(dev, "raw read: 0x%02x 0x%02x 0x%02x 0x%02x\n", regs[0], regs[1], regs[2], regs[3]); secs = ((unsigned long)regs[3] << 24) | (regs[2] << 16) | (regs[1] << 8) | regs[0]; rtc_to_tm(secs, tm); dev_dbg(dev, "read %lld %4d-%02d-%02d (wday=%d) %2d:%02d:%02d\n", secs, tm->tm_year, tm->tm_mon, tm->tm_mday, tm->tm_wday, tm->tm_hour, tm->tm_min, tm->tm_sec); return 0; } static int ds1672_set_time(struct udevice *dev, const struct rtc_time *tm) { time64_t secs = rtc_mktime(tm); u8 regs[5]; dev_dbg(dev, "set %4d-%02d-%02d (wday=%d) %2d:%02d:%02d %lld\n", tm->tm_year, tm->tm_mon, tm->tm_mday, tm->tm_wday, tm->tm_hour, tm->tm_min, tm->tm_sec, secs); if (tm->tm_year < 2000) { dev_err(dev, "year %d (before 2000) not supported\n", tm->tm_year); return -EINVAL; } regs[0] = secs & 0x000000ff; regs[1] = (secs & 0x0000ff00) >> 8; regs[2] = (secs & 0x00ff0000) >> 16; regs[3] = (secs & 0xff000000) >> 24; regs[4] = 0; /* set control reg to enable counting */ return dm_i2c_write(dev, DS1672_REG_CNT_BASE, regs, 5); } static int ds1672_reset(struct udevice *dev) { u8 regs[5] = { 0 }; return dm_i2c_write(dev, DS1672_REG_CNT_BASE, regs, 5); } static int ds1672_read8(struct udevice *dev, unsigned int reg) { return dm_i2c_reg_read(dev, reg); } static int ds1672_write8(struct udevice *dev, unsigned int reg, int val) { return dm_i2c_reg_write(dev, reg, val); } static const struct rtc_ops ds1672_rtc_ops = { .get = ds1672_read_time, .set = ds1672_set_time, .reset = ds1672_reset, .read8 = ds1672_read8, .write8 = ds1672_write8, }; static int ds1672_probe(struct udevice *dev) { i2c_set_chip_flags(dev, DM_I2C_CHIP_RD_ADDRESS | DM_I2C_CHIP_WR_ADDRESS); return 0; } static const struct udevice_id ds1672_of_id[] = { { .compatible = "dallas,ds1672" }, { } }; U_BOOT_DRIVER(rtc_max313xx) = { .name = "rtc-ds1672", .id = UCLASS_RTC, .probe = ds1672_probe, .of_match = ds1672_of_id, .ops = &ds1672_rtc_ops, }; |