power: supply: max17042_battery: Add support for the STATUS property
authorHans de Goede <hdegoede@redhat.com>
Fri, 14 Apr 2017 18:32:51 +0000 (20:32 +0200)
committerSebastian Reichel <sre@kernel.org>
Mon, 1 May 2017 10:37:29 +0000 (12:37 +0200)
Userspace prefers the driver having a status property over having to guess
itself. Specifically this will properly make the GNOME3 UI (and likely
others) properly show discharging / charging / full status, instead
of always showing discharging as status.

Note that in the case there is no charger driver supplying the max17042,
then a status of unknown will get returned. At least upower treats
this the same as not having a status attribute, so in this case nothing
changes from a userspace pov.

Signed-off-by: Hans de Goede <hdegoede@redhat.com>
Reviewed-by: Krzysztof Kozlowski <krzk@kernel.org>
Signed-off-by: Sebastian Reichel <sebastian.reichel@collabora.co.uk>
drivers/power/supply/max17042_battery.c
include/linux/power/max17042_battery.h

index f0ff6e880ff219791bf5676fe01527158b2de3ac..62efe7eeb3f8d11112485c94787f845de14f8cc9 100644 (file)
@@ -76,6 +76,7 @@ struct max17042_chip {
 };
 
 static enum power_supply_property max17042_battery_props[] = {
+       POWER_SUPPLY_PROP_STATUS,
        POWER_SUPPLY_PROP_PRESENT,
        POWER_SUPPLY_PROP_CYCLE_COUNT,
        POWER_SUPPLY_PROP_VOLTAGE_MAX,
@@ -113,6 +114,46 @@ static int max17042_get_temperature(struct max17042_chip *chip, int *temp)
        return 0;
 }
 
+static int max17042_get_status(struct max17042_chip *chip, int *status)
+{
+       int ret, charge_full, charge_now;
+
+       ret = power_supply_am_i_supplied(chip->battery);
+       if (ret < 0) {
+               *status = POWER_SUPPLY_STATUS_UNKNOWN;
+               return 0;
+       }
+       if (ret == 0) {
+               *status = POWER_SUPPLY_STATUS_DISCHARGING;
+               return 0;
+       }
+
+       /*
+        * The MAX170xx has builtin end-of-charge detection and will update
+        * FullCAP to match RepCap when it detects end of charging.
+        *
+        * When this cycle the battery gets charged to a higher (calculated)
+        * capacity then the previous cycle then FullCAP will get updated
+        * contineously once end-of-charge detection kicks in, so allow the
+        * 2 to differ a bit.
+        */
+
+       ret = regmap_read(chip->regmap, MAX17042_FullCAP, &charge_full);
+       if (ret < 0)
+               return ret;
+
+       ret = regmap_read(chip->regmap, MAX17042_RepCap, &charge_now);
+       if (ret < 0)
+               return ret;
+
+       if ((charge_full - charge_now) <= MAX17042_FULL_THRESHOLD)
+               *status = POWER_SUPPLY_STATUS_FULL;
+       else
+               *status = POWER_SUPPLY_STATUS_CHARGING;
+
+       return 0;
+}
+
 static int max17042_get_battery_health(struct max17042_chip *chip, int *health)
 {
        int temp, vavg, vbatt, ret;
@@ -182,6 +223,11 @@ static int max17042_get_property(struct power_supply *psy,
                return -EAGAIN;
 
        switch (psp) {
+       case POWER_SUPPLY_PROP_STATUS:
+               ret = max17042_get_status(chip, &val->intval);
+               if (ret < 0)
+                       return ret;
+               break;
        case POWER_SUPPLY_PROP_PRESENT:
                ret = regmap_read(map, MAX17042_STATUS, &data);
                if (ret < 0)
index 3489fb0f909906018969bbaa8f28d87ffb5a2331..a7ed29baf44a6ce6c4ee01ba381655884090d6f2 100644 (file)
@@ -31,6 +31,9 @@
 #define MAX17042_DEFAULT_TEMP_MIN      (0)    /* For sys without temp sensor */
 #define MAX17042_DEFAULT_TEMP_MAX      (700)  /* 70 degrees Celcius */
 
+/* Consider RepCap which is less then 10 units below FullCAP full */
+#define MAX17042_FULL_THRESHOLD                10
+
 #define MAX17042_CHARACTERIZATION_DATA_SIZE 48
 
 enum max17042_register {