]> git.ipfire.org Git - thirdparty/u-boot.git/commitdiff
power: regulator: Add a driver for the AXP PMIC drivevbus
authorSamuel Holland <samuel@sholland.org>
Tue, 31 Oct 2023 06:39:51 +0000 (01:39 -0500)
committerTom Rini <trini@konsulko.com>
Mon, 28 Apr 2025 18:45:43 +0000 (12:45 -0600)
AXP PMICs have a pin which can either report the USB VBUS state, or
driving a regulator that supplies USB VBUS. Add a regulator driver for
controlling this pin. The selection between input and output is done via
the x-powers,drive-vbus-en pin on the PMIC (parent) node.

Signed-off-by: Samuel Holland <samuel@sholland.org>
Signed-off-by: Andre Przywara <andre.przywara@arm.com>
Reviewed-by: Jernej Skrabec <jernej.skrabec@gmail.com>
drivers/power/pmic/axp.c
drivers/power/regulator/Kconfig
drivers/power/regulator/Makefile
drivers/power/regulator/axp_drivevbus.c [new file with mode: 0644]

index 521a39dd5664b4a06e11dbb2afaf1824bf1aa431..c300fd2bbc204d2365afd4341259be9b7e59e119 100644 (file)
@@ -51,6 +51,7 @@ static const struct pmic_child_info axp_pmic_child_info[] = {
        { "cldo",       "axp_regulator" },
        { "dc",         "axp_regulator" },
        { "dldo",       "axp_regulator" },
+       { "drivevbus",  "axp_drivevbus" },
        { "eldo",       "axp_regulator" },
        { "fldo",       "axp_regulator" },
        { "ldo",        "axp_regulator" },
index bec2d2d7d4908026225cd847154cdfbce564cc0f..95912ef5633bda3a4228adb72eeb341d23d81418 100644 (file)
@@ -57,6 +57,13 @@ config SPL_REGULATOR_AXP
          Enable support in SPL for the regulators (DCDCs, LDOs) in the
          X-Powers AXP152, AXP2xx, and AXP8xx PMICs.
 
+config REGULATOR_AXP_DRIVEVBUS
+       bool "Enable driver for X-Powers AXP PMIC drivevbus"
+       depends on DM_REGULATOR && PMIC_AXP
+       help
+         Enable support for sensing or driving the USB VBUS on
+         X-Powers AXP2xx and AXP8xx PMICs.
+
 config REGULATOR_AXP_USB_POWER
        bool "Enable driver for X-Powers AXP PMIC USB power supply"
        depends on DM_REGULATOR && PMIC_AXP
index 99affa235f3bb718163196b0f5633381fab5ec7a..0ee5d908a2acf87e9cee1968061f7343848eb2ad 100644 (file)
@@ -8,6 +8,7 @@ obj-$(CONFIG_$(PHASE_)DM_REGULATOR) += regulator-uclass.o
 obj-$(CONFIG_REGULATOR_ACT8846) += act8846.o
 obj-$(CONFIG_REGULATOR_AS3722) += as3722_regulator.o
 obj-$(CONFIG_$(PHASE_)REGULATOR_AXP) += axp_regulator.o
+obj-$(CONFIG_$(PHASE_)REGULATOR_AXP_DRIVEVBUS) += axp_drivevbus.o
 obj-$(CONFIG_$(PHASE_)REGULATOR_AXP_USB_POWER) += axp_usb_power.o
 obj-$(CONFIG_$(PHASE_)DM_REGULATOR_DA9063) += da9063.o
 obj-$(CONFIG_$(PHASE_)DM_REGULATOR_MAX77663) += max77663_regulator.o
diff --git a/drivers/power/regulator/axp_drivevbus.c b/drivers/power/regulator/axp_drivevbus.c
new file mode 100644 (file)
index 0000000..c463de8
--- /dev/null
@@ -0,0 +1,57 @@
+// SPDX-License-Identifier: GPL-2.0+
+
+#include <dm.h>
+#include <power/pmic.h>
+#include <power/regulator.h>
+
+#define AXP_VBUS_IPSOUT                        0x30
+#define AXP_VBUS_IPSOUT_DRIVEBUS               BIT(2)
+#define AXP_MISC_CTRL                  0x8f
+#define AXP_MISC_CTRL_N_VBUSEN_FUNC            BIT(4)
+
+static int axp_drivevbus_get_enable(struct udevice *dev)
+{
+       int ret;
+
+       ret = pmic_reg_read(dev->parent, AXP_VBUS_IPSOUT);
+       if (ret < 0)
+               return ret;
+
+       return !!(ret & AXP_VBUS_IPSOUT_DRIVEBUS);
+}
+
+static int axp_drivevbus_set_enable(struct udevice *dev, bool enable)
+{
+       return pmic_clrsetbits(dev->parent, AXP_VBUS_IPSOUT,
+                              AXP_VBUS_IPSOUT_DRIVEBUS,
+                              enable ? AXP_VBUS_IPSOUT_DRIVEBUS : 0);
+}
+
+static const struct dm_regulator_ops axp_drivevbus_ops = {
+       .get_enable             = axp_drivevbus_get_enable,
+       .set_enable             = axp_drivevbus_set_enable,
+};
+
+static int axp_drivevbus_probe(struct udevice *dev)
+{
+       struct dm_regulator_uclass_plat *uc_plat = dev_get_uclass_plat(dev);
+       int ret;
+
+       uc_plat->type = REGULATOR_TYPE_FIXED;
+
+       if (dev_read_bool(dev->parent, "x-powers,drive-vbus-en")) {
+               ret = pmic_clrsetbits(dev->parent, AXP_MISC_CTRL,
+                                     AXP_MISC_CTRL_N_VBUSEN_FUNC, 0);
+               if (ret)
+                       return ret;
+       }
+
+       return 0;
+}
+
+U_BOOT_DRIVER(axp_drivevbus) = {
+       .name           = "axp_drivevbus",
+       .id             = UCLASS_REGULATOR,
+       .probe          = axp_drivevbus_probe,
+       .ops            = &axp_drivevbus_ops,
+};