Commit cecbf8d5 authored by Hans de Goede's avatar Hans de Goede Committed by Sebastian Reichel

power: axp20x_usb: Add support for usb power-supply on axp22x pmics

The usb power-supply on the axp22x pmics is mostly identical to the
one on the axp20x pmics. One significant difference is that it cannot
measure / monitor the usb voltage / current.
Signed-off-by: default avatarHans de Goede <hdegoede@redhat.com>
Acked-by: default avatarRob Herring <robh@kernel.org>
Signed-off-by: default avatarSebastian Reichel <sre@kernel.org>
parent 5630b433
AXP20x USB power supply AXP20x USB power supply
Required Properties: Required Properties:
-compatible: "x-powers,axp202-usb-power-supply" -compatible: One of: "x-powers,axp202-usb-power-supply"
"x-powers,axp221-usb-power-supply"
This node is a subnode of the axp20x PMIC. This node is a subnode of the axp20x PMIC.
......
...@@ -42,6 +42,7 @@ ...@@ -42,6 +42,7 @@
#define AXP20X_VBUS_MON_VBUS_VALID BIT(3) #define AXP20X_VBUS_MON_VBUS_VALID BIT(3)
struct axp20x_usb_power { struct axp20x_usb_power {
struct device_node *np;
struct regmap *regmap; struct regmap *regmap;
struct power_supply *supply; struct power_supply *supply;
}; };
...@@ -85,7 +86,12 @@ static int axp20x_usb_power_get_property(struct power_supply *psy, ...@@ -85,7 +86,12 @@ static int axp20x_usb_power_get_property(struct power_supply *psy,
switch (v & AXP20X_VBUS_CLIMIT_MASK) { switch (v & AXP20X_VBUS_CLIMIT_MASK) {
case AXP20X_VBUC_CLIMIT_100mA: case AXP20X_VBUC_CLIMIT_100mA:
val->intval = 100000; if (of_device_is_compatible(power->np,
"x-powers,axp202-usb-power-supply")) {
val->intval = 100000;
} else {
val->intval = -1; /* No 100mA limit */
}
break; break;
case AXP20X_VBUC_CLIMIT_500mA: case AXP20X_VBUC_CLIMIT_500mA:
val->intval = 500000; val->intval = 500000;
...@@ -122,16 +128,19 @@ static int axp20x_usb_power_get_property(struct power_supply *psy, ...@@ -122,16 +128,19 @@ static int axp20x_usb_power_get_property(struct power_supply *psy,
break; break;
} }
ret = regmap_read(power->regmap, AXP20X_USB_OTG_STATUS, &v); val->intval = POWER_SUPPLY_HEALTH_GOOD;
if (ret)
return ret;
if (!(v & AXP20X_USB_STATUS_VBUS_VALID)) { if (of_device_is_compatible(power->np,
val->intval = POWER_SUPPLY_HEALTH_UNSPEC_FAILURE; "x-powers,axp202-usb-power-supply")) {
break; ret = regmap_read(power->regmap,
} AXP20X_USB_OTG_STATUS, &v);
if (ret)
return ret;
val->intval = POWER_SUPPLY_HEALTH_GOOD; if (!(v & AXP20X_USB_STATUS_VBUS_VALID))
val->intval =
POWER_SUPPLY_HEALTH_UNSPEC_FAILURE;
}
break; break;
case POWER_SUPPLY_PROP_PRESENT: case POWER_SUPPLY_PROP_PRESENT:
val->intval = !!(input & AXP20X_PWR_STATUS_VBUS_PRESENT); val->intval = !!(input & AXP20X_PWR_STATUS_VBUS_PRESENT);
...@@ -156,6 +165,14 @@ static enum power_supply_property axp20x_usb_power_properties[] = { ...@@ -156,6 +165,14 @@ static enum power_supply_property axp20x_usb_power_properties[] = {
POWER_SUPPLY_PROP_CURRENT_NOW, POWER_SUPPLY_PROP_CURRENT_NOW,
}; };
static enum power_supply_property axp22x_usb_power_properties[] = {
POWER_SUPPLY_PROP_HEALTH,
POWER_SUPPLY_PROP_PRESENT,
POWER_SUPPLY_PROP_ONLINE,
POWER_SUPPLY_PROP_VOLTAGE_MIN,
POWER_SUPPLY_PROP_CURRENT_MAX,
};
static const struct power_supply_desc axp20x_usb_power_desc = { static const struct power_supply_desc axp20x_usb_power_desc = {
.name = "axp20x-usb", .name = "axp20x-usb",
.type = POWER_SUPPLY_TYPE_USB, .type = POWER_SUPPLY_TYPE_USB,
...@@ -164,13 +181,25 @@ static const struct power_supply_desc axp20x_usb_power_desc = { ...@@ -164,13 +181,25 @@ static const struct power_supply_desc axp20x_usb_power_desc = {
.get_property = axp20x_usb_power_get_property, .get_property = axp20x_usb_power_get_property,
}; };
static const struct power_supply_desc axp22x_usb_power_desc = {
.name = "axp20x-usb",
.type = POWER_SUPPLY_TYPE_USB,
.properties = axp22x_usb_power_properties,
.num_properties = ARRAY_SIZE(axp22x_usb_power_properties),
.get_property = axp20x_usb_power_get_property,
};
static int axp20x_usb_power_probe(struct platform_device *pdev) static int axp20x_usb_power_probe(struct platform_device *pdev)
{ {
struct axp20x_dev *axp20x = dev_get_drvdata(pdev->dev.parent); struct axp20x_dev *axp20x = dev_get_drvdata(pdev->dev.parent);
struct power_supply_config psy_cfg = {}; struct power_supply_config psy_cfg = {};
struct axp20x_usb_power *power; struct axp20x_usb_power *power;
static const char * const irq_names[] = { "VBUS_PLUGIN", static const char * const axp20x_irq_names[] = { "VBUS_PLUGIN",
"VBUS_REMOVAL", "VBUS_VALID", "VBUS_NOT_VALID" }; "VBUS_REMOVAL", "VBUS_VALID", "VBUS_NOT_VALID", NULL };
static const char * const axp22x_irq_names[] = {
"VBUS_PLUGIN", "VBUS_REMOVAL", NULL };
static const char * const *irq_names;
const struct power_supply_desc *usb_power_desc;
int i, irq, ret; int i, irq, ret;
if (!of_device_is_available(pdev->dev.of_node)) if (!of_device_is_available(pdev->dev.of_node))
...@@ -185,31 +214,47 @@ static int axp20x_usb_power_probe(struct platform_device *pdev) ...@@ -185,31 +214,47 @@ static int axp20x_usb_power_probe(struct platform_device *pdev)
if (!power) if (!power)
return -ENOMEM; return -ENOMEM;
power->np = pdev->dev.of_node;
power->regmap = axp20x->regmap; power->regmap = axp20x->regmap;
/* Enable vbus valid checking */ if (of_device_is_compatible(power->np,
ret = regmap_update_bits(power->regmap, AXP20X_VBUS_MON, "x-powers,axp202-usb-power-supply")) {
AXP20X_VBUS_MON_VBUS_VALID, AXP20X_VBUS_MON_VBUS_VALID); /* Enable vbus valid checking */
if (ret) ret = regmap_update_bits(power->regmap, AXP20X_VBUS_MON,
return ret; AXP20X_VBUS_MON_VBUS_VALID,
AXP20X_VBUS_MON_VBUS_VALID);
if (ret)
return ret;
/* Enable vbus voltage and current measurement */ /* Enable vbus voltage and current measurement */
ret = regmap_update_bits(power->regmap, AXP20X_ADC_EN1, ret = regmap_update_bits(power->regmap, AXP20X_ADC_EN1,
AXP20X_ADC_EN1_VBUS_CURR | AXP20X_ADC_EN1_VBUS_VOLT, AXP20X_ADC_EN1_VBUS_CURR | AXP20X_ADC_EN1_VBUS_VOLT,
AXP20X_ADC_EN1_VBUS_CURR | AXP20X_ADC_EN1_VBUS_VOLT); AXP20X_ADC_EN1_VBUS_CURR | AXP20X_ADC_EN1_VBUS_VOLT);
if (ret) if (ret)
return ret; return ret;
usb_power_desc = &axp20x_usb_power_desc;
irq_names = axp20x_irq_names;
} else if (of_device_is_compatible(power->np,
"x-powers,axp221-usb-power-supply")) {
usb_power_desc = &axp22x_usb_power_desc;
irq_names = axp22x_irq_names;
} else {
dev_err(&pdev->dev, "Unsupported AXP variant: %ld\n",
axp20x->variant);
return -EINVAL;
}
psy_cfg.of_node = pdev->dev.of_node; psy_cfg.of_node = pdev->dev.of_node;
psy_cfg.drv_data = power; psy_cfg.drv_data = power;
power->supply = devm_power_supply_register(&pdev->dev, power->supply = devm_power_supply_register(&pdev->dev, usb_power_desc,
&axp20x_usb_power_desc, &psy_cfg); &psy_cfg);
if (IS_ERR(power->supply)) if (IS_ERR(power->supply))
return PTR_ERR(power->supply); return PTR_ERR(power->supply);
/* Request irqs after registering, as irqs may trigger immediately */ /* Request irqs after registering, as irqs may trigger immediately */
for (i = 0; i < ARRAY_SIZE(irq_names); i++) { for (i = 0; irq_names[i]; i++) {
irq = platform_get_irq_byname(pdev, irq_names[i]); irq = platform_get_irq_byname(pdev, irq_names[i]);
if (irq < 0) { if (irq < 0) {
dev_warn(&pdev->dev, "No IRQ for %s: %d\n", dev_warn(&pdev->dev, "No IRQ for %s: %d\n",
...@@ -229,6 +274,7 @@ static int axp20x_usb_power_probe(struct platform_device *pdev) ...@@ -229,6 +274,7 @@ static int axp20x_usb_power_probe(struct platform_device *pdev)
static const struct of_device_id axp20x_usb_power_match[] = { static const struct of_device_id axp20x_usb_power_match[] = {
{ .compatible = "x-powers,axp202-usb-power-supply" }, { .compatible = "x-powers,axp202-usb-power-supply" },
{ .compatible = "x-powers,axp221-usb-power-supply" },
{ } { }
}; };
MODULE_DEVICE_TABLE(of, axp20x_usb_power_match); MODULE_DEVICE_TABLE(of, axp20x_usb_power_match);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment