Source-Changes-HG archive

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

[src/trunk]: src/sys/dev/i2c - No need to use I2C_F_POLL here.



details:   https://anonhg.NetBSD.org/src/rev/261e6502c0a4
branches:  trunk
changeset: 466513:261e6502c0a4
user:      thorpej <thorpej%NetBSD.org@localhost>
date:      Mon Dec 23 21:24:59 2019 +0000

description:
- No need to use I2C_F_POLL here.
- Refactor register read / write functions, splitting out i2c bus
  acquire / release everywhere.  This fixes what was almost certainly
  a deadlock in the FDT regulator section of the code (acquire bus ->
  read register, which again acquires bus -> locking against myself).

diffstat:

 sys/dev/i2c/tps65217pmic.c |  118 ++++++++++++++++++++++++++++++--------------
 1 files changed, 79 insertions(+), 39 deletions(-)

diffs (264 lines):

diff -r 7e494f70a2b9 -r 261e6502c0a4 sys/dev/i2c/tps65217pmic.c
--- a/sys/dev/i2c/tps65217pmic.c        Mon Dec 23 20:49:09 2019 +0000
+++ b/sys/dev/i2c/tps65217pmic.c        Mon Dec 23 21:24:59 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: tps65217pmic.c,v 1.14 2019/11/03 22:55:34 jmcneill Exp $ */
+/*     $NetBSD: tps65217pmic.c,v 1.15 2019/12/23 21:24:59 thorpej Exp $ */
 
 /*-
  * Copyright (c) 2013 The NetBSD Foundation, Inc.
@@ -37,7 +37,7 @@
 #include "opt_fdt.h"
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: tps65217pmic.c,v 1.14 2019/11/03 22:55:34 jmcneill Exp $");
+__KERNEL_RCSID(0, "$NetBSD: tps65217pmic.c,v 1.15 2019/12/23 21:24:59 thorpej Exp $");
 
 #include <sys/param.h>
 #include <sys/systm.h>
@@ -146,12 +146,14 @@
        bool is_xadj;                   /* voltage is adjusted externally */
 
        uint16_t current_voltage;       /* in mV */
-
 };
 
 static int tps65217pmic_match(device_t, cfdata_t, void *);
 static void tps65217pmic_attach(device_t, device_t, void *);
 
+static int tps65217pmic_i2c_lock(struct tps65217pmic_softc *);
+static void tps65217pmic_i2c_unlock(struct tps65217pmic_softc *);
+
 static uint8_t tps65217pmic_reg_read(struct tps65217pmic_softc *, uint8_t);
 static void tps65217pmic_reg_write(struct tps65217pmic_softc *, uint8_t,
     uint8_t);
@@ -398,11 +400,19 @@
        intrmask = TPS65217PMIC_INT_USBM | TPS65217PMIC_INT_ACM |
            TPS65217PMIC_INT_PBM;
 
+       if (tps65217pmic_i2c_lock(sc) != 0) {
+               aprint_error_dev(sc->sc_dev,
+                   "failed to initialize power monitor\n");
+               return;
+       }
+
        status = tps65217pmic_reg_read(sc, TPS65217PMIC_STATUS);
        ppath = tps65217pmic_reg_read(sc, TPS65217PMIC_PPATH);
        /* acknowledge and disregard whatever interrupt was generated earlier */
        intr = tps65217pmic_reg_read(sc, TPS65217PMIC_INT);
 
+       tps65217pmic_i2c_unlock(sc);
+
        sc->sc_usbstatus = status & TPS65217PMIC_STATUS_USBPWR;
        sc->sc_acstatus = status & TPS65217PMIC_STATUS_ACPWR;
        sc->sc_usbenabled = ppath & TPS65217PMIC_PPATH_USB_EN;
@@ -430,7 +440,14 @@
 
        mutex_enter(&sc->sc_lock);
 
+       if (tps65217pmic_i2c_lock(sc) != 0) {
+               device_printf(sc->sc_dev,
+                   "WARNING: unable to perform power monitor task.\n");
+               return;
+       }
        status = tps65217pmic_reg_read(sc, TPS65217PMIC_STATUS);
+       tps65217pmic_i2c_unlock(sc);
+
        usbstatus = status & TPS65217PMIC_STATUS_USBPWR;
        acstatus = status & TPS65217PMIC_STATUS_ACPWR;
 
@@ -510,11 +527,19 @@
                return;
        }
 
+       if (tps65217pmic_i2c_lock(sc) != 0) {
+               device_printf(sc->sc_dev,
+                   "WARNING: unable to configure LED\n");
+               return;
+       }
+
        tps65217pmic_reg_write(sc, TPS65217PMIC_WLEDCTRL1, val);
        tps65217pmic_reg_write(sc, TPS65217PMIC_WLEDCTRL2,
            (brightness - 1) & TPS65217PMIC_WLEDCTRL2_DUTY);
        val |= TPS65217PMIC_WLEDCTRL1_ISINK_EN;
        tps65217pmic_reg_write(sc, TPS65217PMIC_WLEDCTRL1, val);
+
+       tps65217pmic_i2c_unlock(sc);
 }
 
 static void
@@ -523,10 +548,18 @@
        int i;
        struct tps_reg_param *c_reg;
 
+       if (tps65217pmic_i2c_lock(sc) != 0) {
+               device_printf(sc->sc_dev,
+                   "WARNING: unable to refresh regulators\n");
+               return;
+       }
+
        for (i = 0; i < NTPS_REG; i++) {
                c_reg = &tps_regulators[i];
                tps65217pmic_regulator_read_config(sc, c_reg);
        }
+
+       tps65217pmic_i2c_unlock(sc);
 }
 
 /* Get version and revision of the chip. */
@@ -535,8 +568,16 @@
 {
        uint8_t chipid;
 
+       if (tps65217pmic_i2c_lock(sc) != 0) {
+               device_printf(sc->sc_dev,
+                   "WARNING: unable to get chip ID\n");
+               return;
+       }
+
        chipid = tps65217pmic_reg_read(sc, TPS65217PMIC_CHIPID);
 
+       tps65217pmic_i2c_unlock(sc);
+
        sc->sc_version = chipid & TPS65217PMIC_CHIPID_VER_MASK;
        sc->sc_revision = chipid & TPS65217PMIC_CHIPID_REV_MASK;
 }
@@ -694,32 +735,45 @@
        aprint_normal("\n");
 }
 
+static int
+tps65217pmic_i2c_lock(struct tps65217pmic_softc *sc)
+{
+       int error;
+
+       error = iic_acquire_bus(sc->sc_tag, 0);
+       if (error) {
+               device_printf(sc->sc_dev,
+                   "unable to acquire i2c bus, error %d\n", error);
+       }
+       return error;
+}
+
+static void
+tps65217pmic_i2c_unlock(struct tps65217pmic_softc *sc)
+{
+       iic_release_bus(sc->sc_tag, 0);
+}
+
 static uint8_t
 tps65217pmic_reg_read(struct tps65217pmic_softc *sc, uint8_t reg)
 {
        uint8_t wbuf[2];
        uint8_t rv;
 
-       if (iic_acquire_bus(sc->sc_tag, I2C_F_POLL) != 0) {
-               aprint_error_dev(sc->sc_dev, "cannot acquire bus for read\n");
-               return 0;
-       }
-
        wbuf[0] = reg;
 
        if (iic_exec(sc->sc_tag, I2C_OP_READ_WITH_STOP, sc->sc_addr, wbuf,
-           1, &rv, 1, I2C_F_POLL)) {
+           1, &rv, 1, 0)) {
                aprint_error_dev(sc->sc_dev, "cannot execute operation\n");
-               iic_release_bus(sc->sc_tag, I2C_F_POLL);
+               iic_release_bus(sc->sc_tag, 0);
                return 0;
        }
-       iic_release_bus(sc->sc_tag, I2C_F_POLL);
 
        return rv;
 }
 
 static void
-tps65217pmic_reg_write_unlocked(struct tps65217pmic_softc *sc,
+tps65217pmic_reg_write(struct tps65217pmic_softc *sc,
     uint8_t reg, uint8_t data)
 {
        uint8_t wbuf[2];
@@ -728,40 +782,26 @@
        wbuf[1] = data;
 
        if (iic_exec(sc->sc_tag, I2C_OP_WRITE_WITH_STOP, sc->sc_addr, NULL, 0,
-           wbuf, 2, I2C_F_POLL)) {
+           wbuf, 2, 0)) {
                aprint_error_dev(sc->sc_dev, "cannot execute I2C write\n");
        }
 }
 
-static void __unused
-tps65217pmic_reg_write(struct tps65217pmic_softc *sc, uint8_t reg, uint8_t data)
-{
-
-       if (iic_acquire_bus(sc->sc_tag, I2C_F_POLL) != 0) {
-               aprint_error_dev(sc->sc_dev, "cannot acquire bus for write\n");
-               return;
-       }
-
-       tps65217pmic_reg_write_unlocked(sc, reg, data);
-
-       iic_release_bus(sc->sc_tag, I2C_F_POLL);
-}
-
 static void
 tps65217pmic_reg_write_l2(struct tps65217pmic_softc *sc,
     uint8_t reg, uint8_t data)
 {
        uint8_t regpw = reg ^ TPS65217PMIC_PASSWORD_XOR;
-       if (iic_acquire_bus(sc->sc_tag, I2C_F_POLL) != 0) {
-               aprint_error_dev(sc->sc_dev, "cannot acquire bus for write\n");
+
+       if (tps65217pmic_i2c_lock(sc))
                return;
-       }
 
-       tps65217pmic_reg_write_unlocked(sc, TPS65217PMIC_PASSWORD, regpw);
-       tps65217pmic_reg_write_unlocked(sc, reg, data);
-       tps65217pmic_reg_write_unlocked(sc, TPS65217PMIC_PASSWORD, regpw);
-       tps65217pmic_reg_write_unlocked(sc, reg, data);
-       iic_release_bus(sc->sc_tag, I2C_F_POLL);
+       tps65217pmic_reg_write(sc, TPS65217PMIC_PASSWORD, regpw);
+       tps65217pmic_reg_write(sc, reg, data);
+       tps65217pmic_reg_write(sc, TPS65217PMIC_PASSWORD, regpw);
+       tps65217pmic_reg_write(sc, reg, data);
+
+       tps65217pmic_i2c_unlock(sc);
 }
 
 static void
@@ -946,7 +986,7 @@
        uint8_t val;
        int error;
 
-       error = iic_acquire_bus(pmic_sc->sc_tag, I2C_F_POLL);
+       error = tps65217pmic_i2c_lock(pmic_sc);
        if (error != 0)
                return error;
 
@@ -959,7 +999,7 @@
 
        regulator->is_enabled = enable;
 
-       iic_release_bus(pmic_sc->sc_tag, I2C_F_POLL);
+       tps65217pmic_i2c_unlock(pmic_sc);
 
        return 0;
 }
@@ -972,13 +1012,13 @@
        struct tps_reg_param *regulator = sc->sc_param;
        int error;
 
-       error = iic_acquire_bus(pmic_sc->sc_tag, I2C_F_POLL);
+       error = tps65217pmic_i2c_lock(pmic_sc);
        if (error != 0)
                return error;
 
        error = tps65217pmic_set_volt(pmic_sc->sc_dev, regulator->name, min_uvol / 1000);
 
-       iic_release_bus(pmic_sc->sc_tag, I2C_F_POLL);
+       tps65217pmic_i2c_unlock(pmic_sc);
 
        return error;
 }



Home | Main Index | Thread Index | Old Index