|
@@ -51,6 +51,7 @@
|
|
|
#define BQ27x00_REG_LMD 0x12 /* Last measured discharge */
|
|
|
#define BQ27x00_REG_CYCT 0x2A /* Cycle count total */
|
|
|
#define BQ27x00_REG_AE 0x22 /* Available energy */
|
|
|
+#define BQ27x00_POWER_AVG 0x24
|
|
|
|
|
|
#define BQ27000_REG_RSOC 0x0B /* Relative State-of-Charge */
|
|
|
#define BQ27000_REG_ILMD 0x76 /* Initial last measured discharge */
|
|
@@ -66,8 +67,10 @@
|
|
|
#define BQ27500_FLAG_SOCF BIT(1) /* State-of-Charge threshold final */
|
|
|
#define BQ27500_FLAG_SOC1 BIT(2) /* State-of-Charge threshold 1 */
|
|
|
#define BQ27500_FLAG_FC BIT(9)
|
|
|
+#define BQ27500_FLAG_OTC BIT(15)
|
|
|
|
|
|
#define BQ27000_RS 20 /* Resistor sense */
|
|
|
+#define BQ27x00_POWER_CONSTANT (256 * 29200 / 1000)
|
|
|
|
|
|
struct bq27x00_device_info;
|
|
|
struct bq27x00_access_methods {
|
|
@@ -86,6 +89,8 @@ struct bq27x00_reg_cache {
|
|
|
int capacity;
|
|
|
int energy;
|
|
|
int flags;
|
|
|
+ int power_avg;
|
|
|
+ int health;
|
|
|
};
|
|
|
|
|
|
struct bq27x00_device_info {
|
|
@@ -123,6 +128,8 @@ static enum power_supply_property bq27x00_battery_props[] = {
|
|
|
POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN,
|
|
|
POWER_SUPPLY_PROP_CYCLE_COUNT,
|
|
|
POWER_SUPPLY_PROP_ENERGY_NOW,
|
|
|
+ POWER_SUPPLY_PROP_POWER_AVG,
|
|
|
+ POWER_SUPPLY_PROP_HEALTH,
|
|
|
};
|
|
|
|
|
|
static unsigned int poll_interval = 360;
|
|
@@ -306,6 +313,60 @@ static int bq27x00_battery_read_time(struct bq27x00_device_info *di, u8 reg)
|
|
|
return tval * 60;
|
|
|
}
|
|
|
|
|
|
+/*
|
|
|
+ * Read a power avg register.
|
|
|
+ * Return < 0 if something fails.
|
|
|
+ */
|
|
|
+static int bq27x00_battery_read_pwr_avg(struct bq27x00_device_info *di, u8 reg)
|
|
|
+{
|
|
|
+ int tval;
|
|
|
+
|
|
|
+ tval = bq27x00_read(di, reg, false);
|
|
|
+ if (tval < 0) {
|
|
|
+ dev_err(di->dev, "error reading power avg rgister %02x: %d\n",
|
|
|
+ reg, tval);
|
|
|
+ return tval;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (di->chip == BQ27500)
|
|
|
+ return tval;
|
|
|
+ else
|
|
|
+ return (tval * BQ27x00_POWER_CONSTANT) / BQ27000_RS;
|
|
|
+}
|
|
|
+
|
|
|
+/*
|
|
|
+ * Read flag register.
|
|
|
+ * Return < 0 if something fails.
|
|
|
+ */
|
|
|
+static int bq27x00_battery_read_health(struct bq27x00_device_info *di)
|
|
|
+{
|
|
|
+ int tval;
|
|
|
+
|
|
|
+ tval = bq27x00_read(di, BQ27x00_REG_FLAGS, false);
|
|
|
+ if (tval < 0) {
|
|
|
+ dev_err(di->dev, "error reading flag register:%d\n", tval);
|
|
|
+ return tval;
|
|
|
+ }
|
|
|
+
|
|
|
+ if ((di->chip == BQ27500)) {
|
|
|
+ if (tval & BQ27500_FLAG_SOCF)
|
|
|
+ tval = POWER_SUPPLY_HEALTH_DEAD;
|
|
|
+ else if (tval & BQ27500_FLAG_OTC)
|
|
|
+ tval = POWER_SUPPLY_HEALTH_OVERHEAT;
|
|
|
+ else
|
|
|
+ tval = POWER_SUPPLY_HEALTH_GOOD;
|
|
|
+ return tval;
|
|
|
+ } else {
|
|
|
+ if (tval & BQ27000_FLAG_EDV1)
|
|
|
+ tval = POWER_SUPPLY_HEALTH_DEAD;
|
|
|
+ else
|
|
|
+ tval = POWER_SUPPLY_HEALTH_GOOD;
|
|
|
+ return tval;
|
|
|
+ }
|
|
|
+
|
|
|
+ return -1;
|
|
|
+}
|
|
|
+
|
|
|
static void bq27x00_update(struct bq27x00_device_info *di)
|
|
|
{
|
|
|
struct bq27x00_reg_cache cache = {0, };
|
|
@@ -321,6 +382,7 @@ static void bq27x00_update(struct bq27x00_device_info *di)
|
|
|
cache.time_to_empty_avg = -ENODATA;
|
|
|
cache.time_to_full = -ENODATA;
|
|
|
cache.charge_full = -ENODATA;
|
|
|
+ cache.health = -ENODATA;
|
|
|
} else {
|
|
|
cache.capacity = bq27x00_battery_read_rsoc(di);
|
|
|
cache.energy = bq27x00_battery_read_energy(di);
|
|
@@ -328,9 +390,12 @@ static void bq27x00_update(struct bq27x00_device_info *di)
|
|
|
cache.time_to_empty_avg = bq27x00_battery_read_time(di, BQ27x00_REG_TTECP);
|
|
|
cache.time_to_full = bq27x00_battery_read_time(di, BQ27x00_REG_TTF);
|
|
|
cache.charge_full = bq27x00_battery_read_lmd(di);
|
|
|
+ cache.health = bq27x00_battery_read_health(di);
|
|
|
}
|
|
|
cache.temperature = bq27x00_battery_read_temperature(di);
|
|
|
cache.cycle_count = bq27x00_battery_read_cyct(di);
|
|
|
+ cache.power_avg =
|
|
|
+ bq27x00_battery_read_pwr_avg(di, BQ27x00_POWER_AVG);
|
|
|
|
|
|
/* We only have to read charge design full once */
|
|
|
if (di->charge_design_full <= 0)
|
|
@@ -550,6 +615,12 @@ static int bq27x00_battery_get_property(struct power_supply *psy,
|
|
|
case POWER_SUPPLY_PROP_ENERGY_NOW:
|
|
|
ret = bq27x00_simple_value(di->cache.energy, val);
|
|
|
break;
|
|
|
+ case POWER_SUPPLY_PROP_POWER_AVG:
|
|
|
+ ret = bq27x00_simple_value(di->cache.power_avg, val);
|
|
|
+ break;
|
|
|
+ case POWER_SUPPLY_PROP_HEALTH:
|
|
|
+ ret = bq27x00_simple_value(di->cache.health, val);
|
|
|
+ break;
|
|
|
default:
|
|
|
return -EINVAL;
|
|
|
}
|