Message ID | 1471569215-6046-6-git-send-email-wenyou.yang@atmel.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Hi, On Fri, Aug 19, 2016 at 09:13:33AM +0800, Wenyou Yang wrote: > Add the power supply capacity level property, it corresponds to > POWER_SUPPLY_CAPACITY_LEVEL_*. > > It also utilizes the precision voltage detector function module > to catch the low battery voltage. > > Signed-off-by: Wenyou Yang <wenyou.yang@atmel.com> > --- > > Changes in v6: > - For 'lbo-gpios', use gpiod API instead of old gpio API to handle. > > Changes in v5: None > Changes in v4: > - Change devname of devm_request_irq() from "lbo-detect" to > "act8945a, lbo-detect". > > Changes in v3: None > Changes in v2: None > > drivers/power/supply/act8945a_charger.c | 75 +++++++++++++++++++++++++++++++++ > 1 file changed, 75 insertions(+) > > diff --git a/drivers/power/supply/act8945a_charger.c b/drivers/power/supply/act8945a_charger.c > index 2ff2f8d..de7ad48 100644 > --- a/drivers/power/supply/act8945a_charger.c > +++ b/drivers/power/supply/act8945a_charger.c > @@ -18,6 +18,7 @@ > #include <linux/platform_device.h> > #include <linux/power_supply.h> > #include <linux/regmap.h> > +#include <linux/gpio/consumer.h> > > static const char *act8945a_charger_model = "ACT8945A"; > static const char *act8945a_charger_manufacturer = "Active-semi"; > @@ -83,6 +84,7 @@ struct act8945a_charger { > struct work_struct work; > > bool init_done; > + struct gpio_desc *lbo_gpio; > }; > > static int act8945a_get_charger_state(struct regmap *regmap, int *val) > @@ -208,11 +210,67 @@ static int act8945a_get_battery_health(struct regmap *regmap, int *val) > return 0; > } > > +static int act8945a_get_capacity_level(struct act8945a_charger *charger, > + struct regmap *regmap, int *val) > +{ > + int ret; > + unsigned int status, state, config; > + int lbo_level = gpiod_get_value(charger->lbo_gpio); > + > + ret = regmap_read(regmap, ACT8945A_APCH_STATUS, &status); > + if (ret < 0) > + return ret; > + > + ret = regmap_read(regmap, ACT8945A_APCH_CFG, &config); > + if (ret < 0) > + return ret; > + > + ret = regmap_read(regmap, ACT8945A_APCH_STATE, &state); > + if (ret < 0) > + return ret; > + > + state &= APCH_STATE_CSTATE; > + state >>= APCH_STATE_CSTATE_SHIFT; > + > + switch (state) { > + case APCH_STATE_CSTATE_PRE: > + *val = POWER_SUPPLY_CAPACITY_LEVEL_LOW; > + break; > + case APCH_STATE_CSTATE_FAST: > + if (lbo_level) > + *val = POWER_SUPPLY_CAPACITY_LEVEL_HIGH; > + else > + *val = POWER_SUPPLY_CAPACITY_LEVEL_LOW; > + break; > + case APCH_STATE_CSTATE_EOC: > + if (status & APCH_STATUS_CHGDAT) > + *val = POWER_SUPPLY_CAPACITY_LEVEL_FULL; > + else > + *val = POWER_SUPPLY_CAPACITY_LEVEL_NORMAL; > + break; > + case APCH_STATE_CSTATE_DISABLED: > + default: > + if (config & APCH_CFG_SUSCHG) { > + *val = POWER_SUPPLY_CAPACITY_LEVEL_UNKNOWN; > + } else { > + *val = POWER_SUPPLY_CAPACITY_LEVEL_NORMAL; > + if (!(status & APCH_STATUS_INDAT)) { > + if (!lbo_level) > + *val = POWER_SUPPLY_CAPACITY_LEVEL_CRITICAL; > + } > + } > + break; > + } > + > + return 0; > +} > + > static enum power_supply_property act8945a_charger_props[] = { > POWER_SUPPLY_PROP_STATUS, > POWER_SUPPLY_PROP_CHARGE_TYPE, > POWER_SUPPLY_PROP_TECHNOLOGY, > POWER_SUPPLY_PROP_HEALTH, > + POWER_SUPPLY_PROP_CAPACITY_LEVEL, > POWER_SUPPLY_PROP_MODEL_NAME, > POWER_SUPPLY_PROP_MANUFACTURER > }; > @@ -238,6 +296,10 @@ static int act8945a_charger_get_property(struct power_supply *psy, > case POWER_SUPPLY_PROP_HEALTH: > ret = act8945a_get_battery_health(regmap, &val->intval); > break; > + case POWER_SUPPLY_PROP_CAPACITY_LEVEL: > + ret = act8945a_get_capacity_level(charger, > + regmap, &val->intval); > + break; > case POWER_SUPPLY_PROP_MODEL_NAME: > val->strval = act8945a_charger_model; > break; > @@ -352,6 +414,19 @@ static int act8945a_charger_config(struct device *dev, > if (tmp & APCH_CFG_SUSCHG) > value |= APCH_CFG_SUSCHG; > > + charger->lbo_gpio = devm_gpiod_get(dev, "active-semi,lbo", GPIOD_IN); > + if (IS_ERR(charger->lbo_gpio)) { > + dev_err(dev, "unable to claim gpio \"lbo\"\n"); > + charger->lbo_gpio = NULL; > + }; You need special handling for -EPROBE_DEFER. > + ret = devm_request_irq(dev, gpiod_to_irq(charger->lbo_gpio), > + act8945a_status_changed, > + (IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING), > + "act8945a_lbo_detect", charger); > + if (ret) > + dev_dbg(dev, "failed to request \"lbo\" pin IRQ\n"); Also everything in act8945a_charger_config is done with parent device as *dev. Thus all devm_*(dev, ...) calls are wrong. -- Sebastian
diff --git a/drivers/power/supply/act8945a_charger.c b/drivers/power/supply/act8945a_charger.c index 2ff2f8d..de7ad48 100644 --- a/drivers/power/supply/act8945a_charger.c +++ b/drivers/power/supply/act8945a_charger.c @@ -18,6 +18,7 @@ #include <linux/platform_device.h> #include <linux/power_supply.h> #include <linux/regmap.h> +#include <linux/gpio/consumer.h> static const char *act8945a_charger_model = "ACT8945A"; static const char *act8945a_charger_manufacturer = "Active-semi"; @@ -83,6 +84,7 @@ struct act8945a_charger { struct work_struct work; bool init_done; + struct gpio_desc *lbo_gpio; }; static int act8945a_get_charger_state(struct regmap *regmap, int *val) @@ -208,11 +210,67 @@ static int act8945a_get_battery_health(struct regmap *regmap, int *val) return 0; } +static int act8945a_get_capacity_level(struct act8945a_charger *charger, + struct regmap *regmap, int *val) +{ + int ret; + unsigned int status, state, config; + int lbo_level = gpiod_get_value(charger->lbo_gpio); + + ret = regmap_read(regmap, ACT8945A_APCH_STATUS, &status); + if (ret < 0) + return ret; + + ret = regmap_read(regmap, ACT8945A_APCH_CFG, &config); + if (ret < 0) + return ret; + + ret = regmap_read(regmap, ACT8945A_APCH_STATE, &state); + if (ret < 0) + return ret; + + state &= APCH_STATE_CSTATE; + state >>= APCH_STATE_CSTATE_SHIFT; + + switch (state) { + case APCH_STATE_CSTATE_PRE: + *val = POWER_SUPPLY_CAPACITY_LEVEL_LOW; + break; + case APCH_STATE_CSTATE_FAST: + if (lbo_level) + *val = POWER_SUPPLY_CAPACITY_LEVEL_HIGH; + else + *val = POWER_SUPPLY_CAPACITY_LEVEL_LOW; + break; + case APCH_STATE_CSTATE_EOC: + if (status & APCH_STATUS_CHGDAT) + *val = POWER_SUPPLY_CAPACITY_LEVEL_FULL; + else + *val = POWER_SUPPLY_CAPACITY_LEVEL_NORMAL; + break; + case APCH_STATE_CSTATE_DISABLED: + default: + if (config & APCH_CFG_SUSCHG) { + *val = POWER_SUPPLY_CAPACITY_LEVEL_UNKNOWN; + } else { + *val = POWER_SUPPLY_CAPACITY_LEVEL_NORMAL; + if (!(status & APCH_STATUS_INDAT)) { + if (!lbo_level) + *val = POWER_SUPPLY_CAPACITY_LEVEL_CRITICAL; + } + } + break; + } + + return 0; +} + static enum power_supply_property act8945a_charger_props[] = { POWER_SUPPLY_PROP_STATUS, POWER_SUPPLY_PROP_CHARGE_TYPE, POWER_SUPPLY_PROP_TECHNOLOGY, POWER_SUPPLY_PROP_HEALTH, + POWER_SUPPLY_PROP_CAPACITY_LEVEL, POWER_SUPPLY_PROP_MODEL_NAME, POWER_SUPPLY_PROP_MANUFACTURER }; @@ -238,6 +296,10 @@ static int act8945a_charger_get_property(struct power_supply *psy, case POWER_SUPPLY_PROP_HEALTH: ret = act8945a_get_battery_health(regmap, &val->intval); break; + case POWER_SUPPLY_PROP_CAPACITY_LEVEL: + ret = act8945a_get_capacity_level(charger, + regmap, &val->intval); + break; case POWER_SUPPLY_PROP_MODEL_NAME: val->strval = act8945a_charger_model; break; @@ -352,6 +414,19 @@ static int act8945a_charger_config(struct device *dev, if (tmp & APCH_CFG_SUSCHG) value |= APCH_CFG_SUSCHG; + charger->lbo_gpio = devm_gpiod_get(dev, "active-semi,lbo", GPIOD_IN); + if (IS_ERR(charger->lbo_gpio)) { + dev_err(dev, "unable to claim gpio \"lbo\"\n"); + charger->lbo_gpio = NULL; + }; + + ret = devm_request_irq(dev, gpiod_to_irq(charger->lbo_gpio), + act8945a_status_changed, + (IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING), + "act8945a_lbo_detect", charger); + if (ret) + dev_dbg(dev, "failed to request \"lbo\" pin IRQ\n"); + chglev_pin = of_get_named_gpio_flags(np, "active-semi,chglev-gpios", 0, &flags);
Add the power supply capacity level property, it corresponds to POWER_SUPPLY_CAPACITY_LEVEL_*. It also utilizes the precision voltage detector function module to catch the low battery voltage. Signed-off-by: Wenyou Yang <wenyou.yang@atmel.com> --- Changes in v6: - For 'lbo-gpios', use gpiod API instead of old gpio API to handle. Changes in v5: None Changes in v4: - Change devname of devm_request_irq() from "lbo-detect" to "act8945a, lbo-detect". Changes in v3: None Changes in v2: None drivers/power/supply/act8945a_charger.c | 75 +++++++++++++++++++++++++++++++++ 1 file changed, 75 insertions(+)