From nobody Sun Feb 8 17:46:45 2026 Received: from mail.andi.de1.cc (mail.andi.de1.cc [178.238.236.174]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 3CE1F149C6F; Fri, 2 Jan 2026 10:16:09 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=178.238.236.174 ARC-Seal: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1767348972; cv=none; b=WvvCL9vRNe6cIYTHWA1asVU2NJV6650a/wdTgy6Iu5EuASYUJRUAUQQaoqcAUn4/LJipQA2G5szwxSclj//mq/ClT+v4sLbv8SXjksjOoPnptcuiJb/HnefS5K3sK5r9H4639ZJ8MjKIor8R4jHNDXdBMWki4xxhDXCEJAodHZQ= ARC-Message-Signature: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1767348972; c=relaxed/simple; bh=LVcGh/6SNJstE5IufFNTQ//Nfrp5aD/qOQSUQBUrNuU=; h=From:Date:Subject:MIME-Version:Content-Type:Message-Id:References: In-Reply-To:To:Cc; b=dozr7cSv5gnqD6CuXDTlyvgzH5viiaT/78yaqWvYvGcUSRbLVp/5/WZDaavz81wa0ZMfNtOAQxe33fVT6XtwwXMUF6ZnWyPaLfxKc5XkG38/ut6dsXkFjYMOW+hRG9ksT3p7g0jxRN22jkvA+gzGGC+/+ikNv+cGHC/Q8GhcbbE= ARC-Authentication-Results: i=1; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=kemnade.info; spf=pass smtp.mailfrom=kemnade.info; dkim=pass (2048-bit key) header.d=kemnade.info header.i=@kemnade.info header.b=67CcigvT; arc=none smtp.client-ip=178.238.236.174 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=kemnade.info Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=kemnade.info Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=kemnade.info header.i=@kemnade.info header.b="67CcigvT" DKIM-Signature: v=1; a=rsa-sha256; q=dns/txt; c=relaxed/relaxed; d=kemnade.info; s=20220719; h=Cc:In-Reply-To:References:From:Sender:Reply-To: Content-ID:Content-Description:Resent-Date:Resent-From:Resent-Sender: Resent-To:Resent-Cc:Resent-Message-ID; bh=noCo3TWyz/2Co4e4aZ+iblmN+RDw5lq2fxKmTHRklOU=; b=67CcigvTDoMIzH59A+zoOOrvUu ifUOctskrnNbsBjOVK0BP4CH88DbmaDLwhtDkQL6BXiYvWes0Kh4ne05822o8aenYghmLRPnFo15t QQXHZcUCsoJCriwCpU783SZ0j1TUld6AyKwYTaAZ28xnseFRt2L16PhBBDz8c7S3sfWis4y41e1Vv APRtD1QCSpYnXhLTD6WdOSLyuecRDCUVawXIKiZt+ssG6+JpyZQEo+r1SSBnqQI88DEdcrD89ldI6 mudPefVXLEuNfuFNEL5XbyDItzC/6DVBz1/fA5oEFGAvxE/AG7YY9uB58W9CZCpPYHVjgFgBLes8F uGw6m/Yg==; From: Andreas Kemnade Date: Fri, 02 Jan 2026 11:13:56 +0100 Subject: [PATCH v3 1/2] regulator: dt-bindings: Document TI TPS65185 Precedence: bulk X-Mailing-List: linux-kernel@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Type: text/plain; charset="utf-8" Content-Transfer-Encoding: quoted-printable Message-Id: <20260102-tps65185-submit-v3-1-23bda35772f2@kemnade.info> References: <20260102-tps65185-submit-v3-0-23bda35772f2@kemnade.info> In-Reply-To: <20260102-tps65185-submit-v3-0-23bda35772f2@kemnade.info> To: Liam Girdwood , Mark Brown , Rob Herring , Krzysztof Kozlowski , Conor Dooley , Guenter Roeck Cc: linux-kernel@vger.kernel.org, devicetree@vger.kernel.org, linux-hwmon@vger.kernel.org, Andreas Kemnade , Krzysztof Kozlowski X-Mailer: b4 0.15-dev-a6db3 X-Developer-Signature: v=1; a=openpgp-sha256; l=3262; i=andreas@kemnade.info; h=from:subject:message-id; bh=LVcGh/6SNJstE5IufFNTQ//Nfrp5aD/qOQSUQBUrNuU=; b=owGbwMvMwCUm/rzkS6lq2x3G02pJDJnhs+7YnW1VV38/O9mlaBbzrMLAUJ/cv8bfWad93fSD6 U3LO37BjlIWBjEuBlkxRZZf1gpun1Se5QZPjbCHmcPKBDKEgYtTACYSepfhn9GNo84b6/9yBb3V FVH+9GPT+2aNxKfTfrAe2Xd/RZqECjPDH86afyfZ4gtP6SoFKn/R/qRcFXnR4dFJWbWDz5enCSu pMgEA X-Developer-Key: i=andreas@kemnade.info; a=openpgp; fpr=EEC0DB858E66C0DA70620AC07DBD6AC74DE29324 Document the TPS65185. GPIO names are same as in the datasheet except for the PWRUP pad which is described as "enable". That pin is optional because the rising edge corresponds to setting one register bit and falling edge to another register bit. Reviewed-by: Krzysztof Kozlowski Signed-off-by: Andreas Kemnade --- .../devicetree/bindings/regulator/ti,tps65185.yaml | 96 ++++++++++++++++++= ++++ 1 file changed, 96 insertions(+) diff --git a/Documentation/devicetree/bindings/regulator/ti,tps65185.yaml b= /Documentation/devicetree/bindings/regulator/ti,tps65185.yaml new file mode 100644 index 000000000000..af0f638b80bc --- /dev/null +++ b/Documentation/devicetree/bindings/regulator/ti,tps65185.yaml @@ -0,0 +1,96 @@ +# SPDX-License-Identifier: GPL-2.0-only OR BSD-2-Clause +%YAML 1.2 +--- +$id: http://devicetree.org/schemas/regulator/ti,tps65185.yaml# +$schema: http://devicetree.org/meta-schemas/core.yaml# + +title: TI TPS65185 Power Management Integrated Circuit + +maintainers: + - Andreas Kemnade + +description: + TPS65185 is a Power Management IC to provide Power for EPDs with one 3.3V + switch, 2 symmetric LDOs behind 2 DC/DC converters, and one unsymmetric + regulator for a compensation voltage. + +properties: + compatible: + const: ti,tps65185 + + reg: + maxItems: 1 + + enable-gpios: + description: + PWRUP pin + maxItems: 1 + + pwr-good-gpios: + maxItems: 1 + + vcom-ctrl-gpios: + maxItems: 1 + + wakeup-gpios: + maxItems: 1 + + vin-supply: true + + interrupts: + maxItems: 1 + + regulators: + type: object + additionalProperties: false + patternProperties: + "^(vcom|vposneg|v3p3)$": + unevaluatedProperties: false + type: object + $ref: /schemas/regulator/regulator.yaml + +required: + - compatible + - reg + - pwr-good-gpios + - vin-supply + +additionalProperties: false + +examples: + - | + #include + #include + i2c { + #address-cells =3D <1>; + #size-cells =3D <0>; + + pmic@18 { + compatible =3D "ti,tps65185"; + reg =3D <0x18>; + pinctrl-names =3D "default"; + pinctrl-0 =3D <&pinctrl_tps65185_gpio>; + pwr-good-gpios =3D <&gpio2 7 GPIO_ACTIVE_HIGH>; + vcom-ctrl-gpios =3D <&gpio2 9 GPIO_ACTIVE_HIGH>; + enable-gpios =3D <&gpio2 8 GPIO_ACTIVE_HIGH>; + wakeup-gpios =3D <&gpio2 5 GPIO_ACTIVE_HIGH>; + vin-supply =3D <&epdc_pmic_supply>; + interrupts-extended =3D <&gpio2 0 IRQ_TYPE_LEVEL_LOW>; + + regulators { + vcom { + regulator-name =3D "vcom"; + }; + + vposneg { + regulator-name =3D "vposneg"; + regulator-min-microvolt =3D <15000000>; + regulator-max-microvolt =3D <15000000>; + }; + + v3p3 { + regulator-name =3D "v3p3"; + }; + }; + }; + }; --=20 2.47.3 From nobody Sun Feb 8 17:46:45 2026 Received: from mail.andi.de1.cc (mail.andi.de1.cc [178.238.236.174]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 3CE701A9FBA; Fri, 2 Jan 2026 10:16:08 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=178.238.236.174 ARC-Seal: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1767348973; cv=none; b=BC9bubFvTVMFukjY5LvE2x1c6c7yXHhxC4Aj1LRjmb69jgXMKQU/uqjdXWSN0e30OzDCtXqOAaac6TFl5TQ3T9q79Z5MItAZmYlzmEQisq4r6RLd+YHgNa2hv7clstRSkV6JxYT6asw3+PBS1SAe9r+zZXSUTNR+rnontEBVZpo= ARC-Message-Signature: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1767348973; c=relaxed/simple; bh=YHIG2teSNRD9jK8S8mBpJW0Iq87DILrFYdZNZXIRgu8=; h=From:Date:Subject:MIME-Version:Content-Type:Message-Id:References: In-Reply-To:To:Cc; b=erJ/ijvxrLpdzxsJEE7/jzNeW0IRSZ6qxdkdHGJhLi7IZegNo9vqf49GhB6s2I6I0Y0NMmsKNWHrxKN6IQ55XzpCFmxqCKkoO/9zrkOJAWaxm2iVlOzTQop6f/BV5KLMIOLqnwi2xDXhvrBsX97Uc7KtkNu8covdmNRbs1l6jo4= ARC-Authentication-Results: i=1; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=kemnade.info; spf=pass smtp.mailfrom=kemnade.info; dkim=pass (2048-bit key) header.d=kemnade.info header.i=@kemnade.info header.b=pal3i1EJ; arc=none smtp.client-ip=178.238.236.174 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=kemnade.info Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=kemnade.info Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=kemnade.info header.i=@kemnade.info header.b="pal3i1EJ" DKIM-Signature: v=1; a=rsa-sha256; q=dns/txt; c=relaxed/relaxed; d=kemnade.info; s=20220719; h=Cc:In-Reply-To:References:From:Sender:Reply-To: Content-ID:Content-Description:Resent-Date:Resent-From:Resent-Sender: Resent-To:Resent-Cc:Resent-Message-ID; bh=xvP6YGWZdJ3cpp1y+FwX/0J+boMg67bhQxhgAQBiUdM=; b=pal3i1EJGIT8yylthUuWp7X/UH Tlz1Qo+NWVz82D02SFxstX2xx8SeLFSxuZFgcaY6BuSqzJlp6vk2ccoZeykCOf+WQlqKUlcjUGPqM P15reuRCsnL3phYayd2B7W7uxMSanWzAg2ksmhae1ZEdH0gB9h0aFbLgKN90Z3mX96vsp0D4RWZRQ 0ChVghNL3fLRkneovQ51zk3EMX0sw3MuAlab83GWC3UrAHMnxnvIi2/QhfxjVGtJZ5XiB18TgB4MZ +qx685YKB9kuvgiPqUPEnInpTFljQa/VrwbLp0SCwU4Ultku26u7kMmkE417EmHWoWDmH4DjmGTXK ITtlLbRQ==; From: Andreas Kemnade Date: Fri, 02 Jan 2026 11:13:57 +0100 Subject: [PATCH v3 2/2] regulator: Add TPS65185 driver Precedence: bulk X-Mailing-List: linux-kernel@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Type: text/plain; charset="utf-8" Content-Transfer-Encoding: quoted-printable Message-Id: <20260102-tps65185-submit-v3-2-23bda35772f2@kemnade.info> References: <20260102-tps65185-submit-v3-0-23bda35772f2@kemnade.info> In-Reply-To: <20260102-tps65185-submit-v3-0-23bda35772f2@kemnade.info> To: Liam Girdwood , Mark Brown , Rob Herring , Krzysztof Kozlowski , Conor Dooley , Guenter Roeck Cc: linux-kernel@vger.kernel.org, devicetree@vger.kernel.org, linux-hwmon@vger.kernel.org, Andreas Kemnade , Josua Mayer X-Mailer: b4 0.15-dev-a6db3 X-Developer-Signature: v=1; a=openpgp-sha256; l=15691; i=andreas@kemnade.info; h=from:subject:message-id; bh=YHIG2teSNRD9jK8S8mBpJW0Iq87DILrFYdZNZXIRgu8=; b=owGbwMvMwCUm/rzkS6lq2x3G02pJDJnhs+6kmy38r79WJ7r5Pbt5SWP/BUG7PwUBkww3XpOzy bl/VSCno5SFQYyLQVZMkeWXtYLbJ5VnucFTI+xh5rAygQxh4OIUgIkoqzIytFUoZgQkxHFv6Jm0 4V+Q8tZp+gt4/b5xb5ZKeiOge9PgB8NfCXW5t5+PcwQX7euZ9lZ6+o2d049cs3906aHecd89kzc 0cQMA X-Developer-Key: i=andreas@kemnade.info; a=openpgp; fpr=EEC0DB858E66C0DA70620AC07DBD6AC74DE29324 Add a driver for the TPS65185 regulator. Implement handling of the various gpio pins. Because the PWRUP (=3Denable) pin functionality can be achieved by just using two bits instead, just ensure that it is set to a stable value. Implement the pair of symmetric LDOs as a single regulator because they share a single voltage set register. As the VCOM regulator sits behind that machinery, just define that one as a supply. For simplicity, just add the temperature sensor (depending on external NTC) directly. There is a mechanism to measure some kick-back voltage during a defined EPD operation, to calibrate the VCOM voltage setting and store that non-volatile in the chip to be the power up default setup. That is not implemented yet in the driver, but that also means that there is a non-factory default value in these registers after power-up. Tested-by: Josua Mayer Signed-off-by: Andreas Kemnade --- drivers/regulator/Kconfig | 11 ++ drivers/regulator/Makefile | 1 + drivers/regulator/tps65185.c | 454 +++++++++++++++++++++++++++++++++++++++= ++++ 3 files changed, 466 insertions(+) diff --git a/drivers/regulator/Kconfig b/drivers/regulator/Kconfig index d2335276cce5..5c539782e48d 100644 --- a/drivers/regulator/Kconfig +++ b/drivers/regulator/Kconfig @@ -1690,6 +1690,17 @@ config REGULATOR_TPS65132 This driver supports TPS65132 single inductor - dual output power supply specifically designed for display panels. =20 +config REGULATOR_TPS65185 + tristate "TI TPS65185 EPD regulator" + depends on I2C + select REGMAP_I2C + help + This driver supports the TPS65185 voltage regulator chip + which is used to provide power to Electronic Paper Displays + so it is found in E-Book readers. + If HWWON is enabled, it also provides temperature measurement. + + config REGULATOR_TPS65217 tristate "TI TPS65217 Power regulators" depends on MFD_TPS65217 diff --git a/drivers/regulator/Makefile b/drivers/regulator/Makefile index 1beba1493241..240de94cd432 100644 --- a/drivers/regulator/Makefile +++ b/drivers/regulator/Makefile @@ -192,6 +192,7 @@ obj-$(CONFIG_REGULATOR_TPS65023) +=3D tps65023-regulato= r.o obj-$(CONFIG_REGULATOR_TPS6507X) +=3D tps6507x-regulator.o obj-$(CONFIG_REGULATOR_TPS65086) +=3D tps65086-regulator.o obj-$(CONFIG_REGULATOR_TPS65090) +=3D tps65090-regulator.o +obj-$(CONFIG_REGULATOR_TPS65185) +=3D tps65185.o obj-$(CONFIG_REGULATOR_TPS65217) +=3D tps65217-regulator.o obj-$(CONFIG_REGULATOR_TPS65218) +=3D tps65218-regulator.o obj-$(CONFIG_REGULATOR_TPS65219) +=3D tps65219-regulator.o diff --git a/drivers/regulator/tps65185.c b/drivers/regulator/tps65185.c new file mode 100644 index 000000000000..3286c9ab33d0 --- /dev/null +++ b/drivers/regulator/tps65185.c @@ -0,0 +1,454 @@ +// SPDX-License-Identifier: GPL-2.0-only +// Copyright (C) 2025 Andreas Kemnade + +/* Datasheet: https://www.ti.com/lit/gpn/tps65185 */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define TPS65185_REG_TMST_VALUE 0 +#define TPS65185_REG_ENABLE 1 +#define TPS65185_REG_VADJ 2 +#define TPS65185_REG_VCOM1 3 +#define TPS65185_REG_VCOM2 4 +#define TPS65185_REG_INT_EN1 5 +#define TPS65185_REG_INT_EN2 6 +#define TPS65185_REG_INT1 7 +#define TPS65185_REG_INT2 8 +#define TPS65185_REG_TMST1 0xd +#define TPS65185_REG_TMST2 0xe +#define TPS65185_REG_PG 0xf +#define TPS65185_REG_REVID 0x10 + +#define TPS65185_READ_THERM BIT(7) +#define TPS65185_CONV_END BIT(5) + +#define TPS65185_ENABLE_ACTIVE BIT(7) +#define TPS65185_ENABLE_STANDBY BIT(6) + +#define PGOOD_TIMEOUT_MSECS 200 + +struct tps65185_data { + struct device *dev; + struct regmap *regmap; + struct gpio_desc *pgood_gpio; + struct gpio_desc *pwrup_gpio; + struct gpio_desc *vcom_ctrl_gpio; + struct gpio_desc *wakeup_gpio; + struct completion pgood_completion; + int pgood_irq; + struct completion tmst_completion; +}; + +static const struct hwmon_channel_info *tps65185_info[] =3D { + HWMON_CHANNEL_INFO(chip, HWMON_C_REGISTER_TZ), + HWMON_CHANNEL_INFO(temp, HWMON_T_INPUT), + NULL +}; + +static int tps65185_hwmon_read(struct device *dev, enum hwmon_sensor_types= type, + u32 attr, int channel, long *temp) +{ + struct tps65185_data *data =3D dev_get_drvdata(dev); + unsigned int val; + int ret; + + reinit_completion(&data->tmst_completion); + /* start acquisition */ + regmap_update_bits(data->regmap, TPS65185_REG_TMST1, + TPS65185_READ_THERM, TPS65185_READ_THERM); + wait_for_completion_timeout(&data->tmst_completion, + msecs_to_jiffies(PGOOD_TIMEOUT_MSECS)); + ret =3D regmap_read(data->regmap, TPS65185_REG_TMST1, &val); + if (!(val & TPS65185_CONV_END)) + return -ETIMEDOUT; + + ret =3D regmap_read(data->regmap, TPS65185_REG_TMST_VALUE, &val); + if (ret) + return ret; + + *temp =3D (s8)val * 1000; + + return 0; +} + +static umode_t tps65185_hwmon_is_visible(const void *data, + enum hwmon_sensor_types type, + u32 attr, int channel) +{ + return 0444; +} + +static const struct hwmon_ops tps65185_hwmon_ops =3D { + .is_visible =3D tps65185_hwmon_is_visible, + .read =3D tps65185_hwmon_read, +}; + +static const struct hwmon_chip_info tps65185_chip_info =3D { + .ops =3D &tps65185_hwmon_ops, + .info =3D tps65185_info, +}; + +static bool tps65185_volatile_reg(struct device *dev, unsigned int reg) +{ + switch (reg) { + case TPS65185_REG_TMST_VALUE: + case TPS65185_REG_ENABLE: + case TPS65185_REG_VCOM2: + case TPS65185_REG_INT1: + case TPS65185_REG_INT2: + case TPS65185_REG_TMST1: + return true; + default: + return false; + } +} + +static const struct regmap_config regmap_config =3D { + .reg_bits =3D 8, + .val_bits =3D 8, + .max_register =3D 0x10, + .cache_type =3D REGCACHE_MAPLE, + .volatile_reg =3D tps65185_volatile_reg, +}; + +static const struct regulator_ops tps65185_v3p3ops =3D { + .list_voltage =3D regulator_list_voltage_linear, + .enable =3D regulator_enable_regmap, + .disable =3D regulator_disable_regmap, + .is_enabled =3D regulator_is_enabled_regmap, +}; + +static int tps65185_check_powergood(struct regulator_dev *rdev) +{ + struct tps65185_data *data =3D rdev_get_drvdata(rdev); + + return gpiod_get_value_cansleep(data->pgood_gpio); +} + +static int tps65185_vposneg_get_voltage_sel(struct regulator_dev *rdev) +{ + int ret; + + ret =3D regulator_get_voltage_sel_regmap(rdev); + if (ret < 0) + return ret; + + /* highest value is lowest voltage */ + return 6 - ret; +} + +static int tps65185_vposneg_set_voltage_sel(struct regulator_dev *rdev, un= signed int selector) +{ + return regulator_set_voltage_sel_regmap(rdev, 6 - selector); +} + +static irqreturn_t pgood_handler(int irq, void *dev_id) +{ + struct tps65185_data *data =3D dev_id; + + complete(&data->pgood_completion); + + return IRQ_HANDLED; +} + +static int tps65185_vposneg_enable(struct regulator_dev *rdev) +{ + struct tps65185_data *data =3D rdev_get_drvdata(rdev); + int ret; + + reinit_completion(&data->pgood_completion); + if (data->pwrup_gpio) + ret =3D gpiod_set_value_cansleep(data->pwrup_gpio, 1); + else + ret =3D regmap_update_bits(data->regmap, TPS65185_REG_ENABLE, + TPS65185_ENABLE_ACTIVE, + TPS65185_ENABLE_ACTIVE); + + if (ret) + return ret; + + dev_dbg(data->dev, "turning on..."); + wait_for_completion_timeout(&data->pgood_completion, + msecs_to_jiffies(PGOOD_TIMEOUT_MSECS)); + dev_dbg(data->dev, "turned on"); + if (gpiod_get_value_cansleep(data->pgood_gpio) !=3D 1) + return -ETIMEDOUT; + + return 0; +} + +static int tps65185_vposneg_disable(struct regulator_dev *rdev) +{ + struct tps65185_data *data =3D rdev_get_drvdata(rdev); + int ret; + + if (data->pwrup_gpio) + ret =3D gpiod_set_value_cansleep(data->pwrup_gpio, 0); + else + ret =3D regmap_update_bits(data->regmap, TPS65185_REG_ENABLE, + TPS65185_ENABLE_STANDBY, + TPS65185_ENABLE_STANDBY); + + return ret; +} + +static int tps65185_vcom_set_voltage_sel(struct regulator_dev *rdev, unsig= ned int selector) +{ + struct tps65185_data *data =3D rdev_get_drvdata(rdev); + int ret; + + ret =3D regmap_update_bits(data->regmap, TPS65185_REG_VCOM2, BIT(0), sele= ctor >> 8); + if (ret < 0) + return ret; + + return regmap_write(data->regmap, TPS65185_REG_VCOM1, selector & 0xFF); +} + +static int tps65185_vcom_get_voltage_sel(struct regulator_dev *rdev) +{ + struct tps65185_data *data =3D rdev_get_drvdata(rdev); + int ret; + unsigned int sel, sel2; + + ret =3D regmap_read(data->regmap, TPS65185_REG_VCOM1, &sel); + if (ret < 0) + return ret; + + ret =3D regmap_read(data->regmap, TPS65185_REG_VCOM2, &sel2); + if (ret < 0) + return ret; + + if (sel2 & BIT(0)) + sel |=3D 0x100; + + return sel; +} + +static const struct regulator_ops tps65185_vcom_ops =3D { + .list_voltage =3D regulator_list_voltage_linear, + .map_voltage =3D regulator_map_voltage_linear, + .set_voltage_sel =3D tps65185_vcom_set_voltage_sel, + .get_voltage_sel =3D tps65185_vcom_get_voltage_sel, +}; + +static const struct regulator_ops tps65185_vposneg_ops =3D { + .list_voltage =3D regulator_list_voltage_linear, + .map_voltage =3D regulator_map_voltage_linear, + .enable =3D tps65185_vposneg_enable, + .disable =3D tps65185_vposneg_disable, + .is_enabled =3D tps65185_check_powergood, + .set_voltage_sel =3D tps65185_vposneg_set_voltage_sel, + .get_voltage_sel =3D tps65185_vposneg_get_voltage_sel, +}; + +static const struct regulator_desc regulators[] =3D { + { + .name =3D "v3p3", + .of_match =3D of_match_ptr("v3p3"), + .regulators_node =3D of_match_ptr("regulators"), + .id =3D 0, + .ops =3D &tps65185_v3p3ops, + .type =3D REGULATOR_VOLTAGE, + .owner =3D THIS_MODULE, + .enable_reg =3D TPS65185_REG_ENABLE, + .enable_mask =3D BIT(5), + .n_voltages =3D 1, + .min_uV =3D 3300000, + }, + { + .name =3D "vposneg", + .of_match =3D of_match_ptr("vposneg"), + .regulators_node =3D of_match_ptr("regulators"), + .id =3D 1, + .ops =3D &tps65185_vposneg_ops, + .type =3D REGULATOR_VOLTAGE, + .owner =3D THIS_MODULE, + .n_voltages =3D 4, + .vsel_reg =3D TPS65185_REG_VADJ, + .vsel_mask =3D 0x7, + .min_uV =3D 14250000, + .uV_step =3D 250000, + } +}; + +static const struct regulator_desc vcom_regulator_desc =3D { + .name =3D "vcom", + .of_match =3D of_match_ptr("vcom"), + .regulators_node =3D of_match_ptr("regulators"), + .supply_name =3D "vposneg", + .id =3D 2, + .ops =3D &tps65185_vcom_ops, + .type =3D REGULATOR_VOLTAGE, + .owner =3D THIS_MODULE, + .n_voltages =3D 511, + .min_uV =3D 0, + .uV_step =3D 10000, +}; + +static irqreturn_t tps65185_irq_thread(int irq, void *dev_id) +{ + struct tps65185_data *data =3D dev_id; + unsigned int int_status_1, int_status_2; + int ret; + + /* read both status to have irq cleared */ + ret =3D regmap_read(data->regmap, TPS65185_REG_INT1, &int_status_1); + if (ret) + return IRQ_NONE; + + ret =3D regmap_read(data->regmap, TPS65185_REG_INT2, &int_status_2); + if (ret) + return IRQ_NONE; + + if (int_status_2 & BIT(0)) + complete(&data->tmst_completion); + + dev_dbg(data->dev, "irq status %02x %02x\n", int_status_1, int_status_2); + + if (int_status_1 || int_status_2) + return IRQ_HANDLED; + + return IRQ_NONE; +} + +static int tps65185_probe(struct i2c_client *client) +{ + struct tps65185_data *data; + struct regulator_config config =3D { }; + struct regulator_dev *rdev; + int ret =3D 0; + int i; + + data =3D devm_kzalloc(&client->dev, sizeof(*data), GFP_KERNEL); + data->regmap =3D devm_regmap_init_i2c(client, ®map_config); + if (IS_ERR(data->regmap)) + return dev_err_probe(&client->dev, PTR_ERR(data->regmap), + "failed to allocate regmap!\n"); + + data->pgood_gpio =3D devm_gpiod_get(&client->dev, "pwr-good", GPIOD_IN); + if (IS_ERR(data->pgood_gpio)) + return dev_err_probe(&client->dev, + PTR_ERR(data->pgood_gpio), + "failed to get power good gpio\n"); + + data->pgood_irq =3D gpiod_to_irq(data->pgood_gpio); + if (data->pgood_irq < 0) + return data->pgood_irq; + + data->pwrup_gpio =3D devm_gpiod_get_optional(&client->dev, "enable", GPIO= D_OUT_LOW); + if (IS_ERR(data->pwrup_gpio)) + return dev_err_probe(&client->dev, PTR_ERR(data->pwrup_gpio), + "failed to get pwrup gpio\n"); + + data->wakeup_gpio =3D devm_gpiod_get_optional(&client->dev, "wakeup", GPI= OD_OUT_HIGH); + if (IS_ERR(data->wakeup_gpio)) + return dev_err_probe(&client->dev, + PTR_ERR(data->wakeup_gpio), + "failed to get wakeup gpio\n"); + + data->vcom_ctrl_gpio =3D devm_gpiod_get_optional(&client->dev, "vcom-ctrl= ", GPIOD_OUT_LOW); + if (IS_ERR(data->vcom_ctrl_gpio)) + return dev_err_probe(&client->dev, + PTR_ERR(data->vcom_ctrl_gpio), + "failed to get vcm ctrl gpio\n"); + + ret =3D devm_regulator_get_enable(&client->dev, "vin"); + if (ret) + return dev_err_probe(&client->dev, ret, + "failed to get vin regulator\n"); + + data->dev =3D &client->dev; + i2c_set_clientdata(client, data); + + init_completion(&data->pgood_completion); + init_completion(&data->tmst_completion); + + ret =3D devm_request_threaded_irq(&client->dev, data->pgood_irq, NULL, + pgood_handler, + IRQF_TRIGGER_RISING | IRQF_ONESHOT, + "PGOOD", data); + if (ret) + return dev_err_probe(&client->dev, ret, + "failed to request power good irq\n"); + + if (client->irq) { + ret =3D devm_request_threaded_irq(&client->dev, client->irq, + NULL, tps65185_irq_thread, + IRQF_TRIGGER_LOW | IRQF_ONESHOT, + "tps65185", data); + if (ret) + return dev_err_probe(&client->dev, ret, + "failed to request irq\n"); + } + + ret =3D regmap_update_bits(data->regmap, TPS65185_REG_INT_EN2, BIT(0), BI= T(0)); + if (ret) + return dev_err_probe(&client->dev, ret, + "failed to enable temp irq\n"); + + config.driver_data =3D data; + config.dev =3D &client->dev; + config.regmap =3D data->regmap; + + for (i =3D 0; i < ARRAY_SIZE(regulators); i++) { + rdev =3D devm_regulator_register(&client->dev, ®ulators[i], + &config); + if (IS_ERR(rdev)) + return dev_err_probe(&client->dev, PTR_ERR(rdev), + "failed to register %s regulator\n", + regulators[i].name); + } + + config.ena_gpiod =3D data->vcom_ctrl_gpio; + rdev =3D devm_regulator_register(&client->dev, &vcom_regulator_desc, &con= fig); + if (IS_ERR(rdev)) + return dev_err_probe(&client->dev, PTR_ERR(rdev), + "failed to register vcom regulator\n"); + + if (IS_REACHABLE(CONFIG_HWMON)) { + struct device *hwmon_dev; + + hwmon_dev =3D devm_hwmon_device_register_with_info(&client->dev, "tps651= 85", data, + &tps65185_chip_info, NULL); + if (IS_ERR(hwmon_dev)) + dev_notice(&client->dev, "failed to register hwmon\n"); + } + + return 0; +} + +static const struct of_device_id tps65185_dt_ids[] =3D { + { + .compatible =3D "ti,tps65185", + }, { + /* sentinel */ + } +}; +MODULE_DEVICE_TABLE(of, tps65185_dt_ids); + +static struct i2c_driver tps65185_i2c_driver =3D { + .driver =3D { + .name =3D "tps65185", + .of_match_table =3D tps65185_dt_ids, + }, + .probe =3D tps65185_probe, +}; + +module_i2c_driver(tps65185_i2c_driver); + +/* Module information */ +MODULE_DESCRIPTION("TPS65185 regulator driver"); +MODULE_LICENSE("GPL"); + --=20 2.47.3