Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/dev/i2c Add RTC support



details:   https://anonhg.NetBSD.org/src/rev/0a2b75a533e5
branches:  trunk
changeset: 465009:0a2b75a533e5
user:      jmcneill <jmcneill%NetBSD.org@localhost>
date:      Fri Nov 01 09:49:55 2019 +0000

description:
Add RTC support

diffstat:

 sys/dev/i2c/twl4030.c |  116 ++++++++++++++++++++++++++++++++++++++++++++++++-
 1 files changed, 113 insertions(+), 3 deletions(-)

diffs (188 lines):

diff -r 7e3ca7777c1b -r 0a2b75a533e5 sys/dev/i2c/twl4030.c
--- a/sys/dev/i2c/twl4030.c     Fri Nov 01 09:49:21 2019 +0000
+++ b/sys/dev/i2c/twl4030.c     Fri Nov 01 09:49:55 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: twl4030.c,v 1.1 2019/10/30 21:38:28 jmcneill Exp $ */
+/* $NetBSD: twl4030.c,v 1.2 2019/11/01 09:49:55 jmcneill Exp $ */
 
 /*-
  * Copyright (c) 2019 Jared McNeill <jmcneill%invisible.ca@localhost>
@@ -27,7 +27,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: twl4030.c,v 1.1 2019/10/30 21:38:28 jmcneill Exp $");
+__KERNEL_RCSID(0, "$NetBSD: twl4030.c,v 1.2 2019/11/01 09:49:55 jmcneill Exp $");
 
 #include <sys/param.h>
 #include <sys/systm.h>
@@ -50,6 +50,12 @@
 #define        ADDR_AUX        0x02
 #define        ADDR_POWER      0x03
 
+/* INTBR registers */
+#define        IDCODE_7_0      0x85
+#define        IDCODE_15_8     0x86
+#define        IDCODE_23_16    0x87
+#define        IDCODE_31_24    0x88
+
 /* GPIO registers */
 #define        GPIOBASE                0x98
 #define        GPIODATAIN(pin)         (GPIOBASE + 0x00 + (pin) / 8)
@@ -60,6 +66,18 @@
 #define        GPIOPUPDCTR(pin)        (GPIOBASE + 0x13 + (n) / 4)
 #define         PUPD_BITS(pin)         __BITS((pin) % 4 + 1, (pin) % 4)
 
+/* POWER registers */
+#define        SECONDS_REG             0x1c
+#define        MINUTES_REG             0x1d
+#define        HOURS_REG               0x1e
+#define        DAYS_REG                0x1f
+#define        MONTHS_REG              0x20
+#define        YEARS_REG               0x21
+#define        WEEKS_REG               0x22
+#define        RTC_CTRL_REG            0x29
+#define         GET_TIME               __BIT(6)
+#define         STOP_RTC               __BIT(0)
+
 struct twl_softc {
        device_t        sc_dev;
        i2c_tag_t       sc_i2c;
@@ -67,6 +85,8 @@
        int             sc_phandle;
 
        int             sc_npins;
+
+       struct todr_chip_handle sc_todr;
 };
 
 struct twl_pin {
@@ -81,6 +101,7 @@
        { NULL,                         0 }
 };
 
+static const char * const rtc_compatible[] = { "ti,twl4030-rtc", NULL };
 static const char * const gpio_compatible[] = { "ti,twl4030-gpio", NULL };
 
 static uint8_t
@@ -118,6 +139,70 @@
 #define        INT_READ(sc, reg)       twl_read((sc), ADDR_INT, (reg), I2C_F_POLL)
 #define        INT_WRITE(sc, reg, val) twl_write((sc), ADDR_INT, (reg), (val), I2C_F_POLL)
 
+#define        POWER_READ(sc, reg)     twl_read((sc), ADDR_POWER, (reg), I2C_F_POLL)
+#define        POWER_WRITE(sc, reg, val) twl_write((sc), ADDR_POWER, (reg), (val), I2C_F_POLL)
+
+static void
+twl_rtc_enable(struct twl_softc *sc, bool onoff)
+{
+       uint8_t rtc_ctrl;
+
+       rtc_ctrl = POWER_READ(sc, RTC_CTRL_REG);
+       if (onoff)
+               rtc_ctrl |= STOP_RTC;   /* 1: RTC is running */
+       else
+               rtc_ctrl &= ~STOP_RTC;  /* 0: RTC is frozen */
+       POWER_WRITE(sc, RTC_CTRL_REG, rtc_ctrl);
+}
+
+static int
+twl_rtc_gettime(todr_chip_handle_t tch, struct clock_ymdhms *dt)
+{
+       struct twl_softc *sc = tch->cookie;
+       uint8_t seconds_reg, minutes_reg, hours_reg,
+           days_reg, months_reg, years_reg, weeks_reg;
+
+       iic_acquire_bus(sc->sc_i2c, I2C_F_POLL);
+       seconds_reg = POWER_READ(sc, SECONDS_REG);
+       minutes_reg = POWER_READ(sc, MINUTES_REG);
+       hours_reg = POWER_READ(sc, HOURS_REG);
+       days_reg = POWER_READ(sc, DAYS_REG);
+       months_reg = POWER_READ(sc, MONTHS_REG);
+       years_reg = POWER_READ(sc, YEARS_REG);
+       weeks_reg = POWER_READ(sc, WEEKS_REG);
+       iic_release_bus(sc->sc_i2c, I2C_F_POLL);
+
+       dt->dt_sec = bcdtobin(seconds_reg);
+       dt->dt_min = bcdtobin(minutes_reg);
+       dt->dt_hour = bcdtobin(hours_reg);
+       dt->dt_day = bcdtobin(days_reg);
+       dt->dt_mon = bcdtobin(months_reg);
+       dt->dt_year = bcdtobin(years_reg) + 2000;
+       dt->dt_wday = bcdtobin(weeks_reg);
+
+       return 0;
+}
+
+static int
+twl_rtc_settime(todr_chip_handle_t tch, struct clock_ymdhms *dt)
+{
+       struct twl_softc *sc = tch->cookie;
+
+       iic_acquire_bus(sc->sc_i2c, I2C_F_POLL);
+       twl_rtc_enable(sc, false);
+        POWER_WRITE(sc, SECONDS_REG, bintobcd(dt->dt_sec));
+        POWER_WRITE(sc, MINUTES_REG, bintobcd(dt->dt_min));
+        POWER_WRITE(sc, HOURS_REG, bintobcd(dt->dt_hour));
+        POWER_WRITE(sc, DAYS_REG, bintobcd(dt->dt_day));
+        POWER_WRITE(sc, MONTHS_REG, bintobcd(dt->dt_mon));
+        POWER_WRITE(sc, YEARS_REG, bintobcd(dt->dt_year % 100));
+        POWER_WRITE(sc, WEEKS_REG, bintobcd(dt->dt_wday));
+       twl_rtc_enable(sc, true);
+       iic_release_bus(sc->sc_i2c, I2C_F_POLL);
+
+       return 0;
+}
+
 static int
 twl_gpio_config(struct twl_softc *sc, int pin, int flags)
 {
@@ -235,6 +320,19 @@
 };
 
 static void
+twl_rtc_attach(struct twl_softc *sc, const int phandle)
+{
+       iic_acquire_bus(sc->sc_i2c, I2C_F_POLL);
+       twl_rtc_enable(sc, true);
+       iic_release_bus(sc->sc_i2c, I2C_F_POLL);
+
+       sc->sc_todr.todr_gettime_ymdhms = twl_rtc_gettime;
+       sc->sc_todr.todr_settime_ymdhms = twl_rtc_settime;
+       sc->sc_todr.cookie = sc;
+       fdtbus_todr_attach(sc->sc_dev, phandle, &sc->sc_todr);
+}
+
+static void
 twl_gpio_attach(struct twl_softc *sc, const int phandle)
 {
        fdtbus_register_gpio_controller(sc->sc_dev, phandle, &twl_gpio_funcs);
@@ -257,6 +355,7 @@
 {
        struct twl_softc * const sc = device_private(self);
        struct i2c_attach_args *ia = aux;
+       uint32_t idcode;
        int child;
 
        sc->sc_dev = self;
@@ -272,9 +371,20 @@
                if (of_match_compatible(child, gpio_compatible)) {
                        aprint_normal(", GPIO");
                        twl_gpio_attach(sc, child);
+               } else if (of_match_compatible(child, rtc_compatible)) {
+                       aprint_normal(", RTC");
+                       twl_rtc_attach(sc, child);
                }
        }
-       aprint_normal("\n");
+
+       I2C_LOCK(sc);
+       idcode = INT_READ(sc, IDCODE_7_0);
+       idcode |= (uint32_t)INT_READ(sc, IDCODE_15_8) << 8;
+       idcode |= (uint32_t)INT_READ(sc, IDCODE_23_16) << 16;
+       idcode |= (uint32_t)INT_READ(sc, IDCODE_31_24) << 24;
+       I2C_UNLOCK(sc);
+
+       aprint_normal(", IDCODE 0x%08x\n", idcode);
 }
 
 CFATTACH_DECL_NEW(twl, sizeof(struct twl_softc),



Home | Main Index | Thread Index | Old Index