smb347-charger: Fix battery status reporting logic for charger faults
This patch checks for charger status register for determining the battery charging status and reports Discharing/Charging/Not Charging/Full accordingly. This patch also adds the interrupt support for Safety Timer Expiration. This interrupt is helpful in debugging the cause for charger fault. Signed-off-by: Ramakrishna Pallala <ramakrishna.pallala@intel.com> Acked-by: Mika Westerberg <mika.westerberg@linux.intel.com> Signed-off-by: Anton Vorontsov <anton.vorontsov@linaro.org>
This commit is contained in:
parent
ec60ea5cd6
commit
1502cfe19b
|
@ -80,6 +80,7 @@
|
||||||
#define CFG_FAULT_IRQ_DCIN_UV BIT(2)
|
#define CFG_FAULT_IRQ_DCIN_UV BIT(2)
|
||||||
#define CFG_STATUS_IRQ 0x0d
|
#define CFG_STATUS_IRQ 0x0d
|
||||||
#define CFG_STATUS_IRQ_TERMINATION_OR_TAPER BIT(4)
|
#define CFG_STATUS_IRQ_TERMINATION_OR_TAPER BIT(4)
|
||||||
|
#define CFG_STATUS_IRQ_CHARGE_TIMEOUT BIT(7)
|
||||||
#define CFG_ADDRESS 0x0e
|
#define CFG_ADDRESS 0x0e
|
||||||
|
|
||||||
/* Command registers */
|
/* Command registers */
|
||||||
|
@ -96,6 +97,9 @@
|
||||||
#define IRQSTAT_C_TERMINATION_STAT BIT(0)
|
#define IRQSTAT_C_TERMINATION_STAT BIT(0)
|
||||||
#define IRQSTAT_C_TERMINATION_IRQ BIT(1)
|
#define IRQSTAT_C_TERMINATION_IRQ BIT(1)
|
||||||
#define IRQSTAT_C_TAPER_IRQ BIT(3)
|
#define IRQSTAT_C_TAPER_IRQ BIT(3)
|
||||||
|
#define IRQSTAT_D 0x38
|
||||||
|
#define IRQSTAT_D_CHARGE_TIMEOUT_STAT BIT(2)
|
||||||
|
#define IRQSTAT_D_CHARGE_TIMEOUT_IRQ BIT(3)
|
||||||
#define IRQSTAT_E 0x39
|
#define IRQSTAT_E 0x39
|
||||||
#define IRQSTAT_E_USBIN_UV_STAT BIT(0)
|
#define IRQSTAT_E_USBIN_UV_STAT BIT(0)
|
||||||
#define IRQSTAT_E_USBIN_UV_IRQ BIT(1)
|
#define IRQSTAT_E_USBIN_UV_IRQ BIT(1)
|
||||||
|
@ -109,8 +113,10 @@
|
||||||
#define STAT_B 0x3c
|
#define STAT_B 0x3c
|
||||||
#define STAT_C 0x3d
|
#define STAT_C 0x3d
|
||||||
#define STAT_C_CHG_ENABLED BIT(0)
|
#define STAT_C_CHG_ENABLED BIT(0)
|
||||||
|
#define STAT_C_HOLDOFF_STAT BIT(3)
|
||||||
#define STAT_C_CHG_MASK 0x06
|
#define STAT_C_CHG_MASK 0x06
|
||||||
#define STAT_C_CHG_SHIFT 1
|
#define STAT_C_CHG_SHIFT 1
|
||||||
|
#define STAT_C_CHG_TERM BIT(5)
|
||||||
#define STAT_C_CHARGER_ERROR BIT(6)
|
#define STAT_C_CHARGER_ERROR BIT(6)
|
||||||
#define STAT_E 0x3f
|
#define STAT_E 0x3f
|
||||||
|
|
||||||
|
@ -701,7 +707,7 @@ fail:
|
||||||
static irqreturn_t smb347_interrupt(int irq, void *data)
|
static irqreturn_t smb347_interrupt(int irq, void *data)
|
||||||
{
|
{
|
||||||
struct smb347_charger *smb = data;
|
struct smb347_charger *smb = data;
|
||||||
unsigned int stat_c, irqstat_e, irqstat_c;
|
unsigned int stat_c, irqstat_c, irqstat_d, irqstat_e;
|
||||||
bool handled = false;
|
bool handled = false;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -717,6 +723,12 @@ static irqreturn_t smb347_interrupt(int irq, void *data)
|
||||||
return IRQ_NONE;
|
return IRQ_NONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ret = regmap_read(smb->regmap, IRQSTAT_D, &irqstat_d);
|
||||||
|
if (ret < 0) {
|
||||||
|
dev_warn(smb->dev, "reading IRQSTAT_D failed\n");
|
||||||
|
return IRQ_NONE;
|
||||||
|
}
|
||||||
|
|
||||||
ret = regmap_read(smb->regmap, IRQSTAT_E, &irqstat_e);
|
ret = regmap_read(smb->regmap, IRQSTAT_E, &irqstat_e);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
dev_warn(smb->dev, "reading IRQSTAT_E failed\n");
|
dev_warn(smb->dev, "reading IRQSTAT_E failed\n");
|
||||||
|
@ -724,13 +736,11 @@ static irqreturn_t smb347_interrupt(int irq, void *data)
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If we get charger error we report the error back to user and
|
* If we get charger error we report the error back to user.
|
||||||
* disable charging.
|
* If the error is recovered charging will resume again.
|
||||||
*/
|
*/
|
||||||
if (stat_c & STAT_C_CHARGER_ERROR) {
|
if (stat_c & STAT_C_CHARGER_ERROR) {
|
||||||
dev_err(smb->dev, "error in charger, disabling charging\n");
|
dev_err(smb->dev, "charging stopped due to charger error\n");
|
||||||
|
|
||||||
smb347_charging_disable(smb);
|
|
||||||
power_supply_changed(&smb->battery);
|
power_supply_changed(&smb->battery);
|
||||||
handled = true;
|
handled = true;
|
||||||
}
|
}
|
||||||
|
@ -743,6 +753,20 @@ static irqreturn_t smb347_interrupt(int irq, void *data)
|
||||||
if (irqstat_c & (IRQSTAT_C_TERMINATION_IRQ | IRQSTAT_C_TAPER_IRQ)) {
|
if (irqstat_c & (IRQSTAT_C_TERMINATION_IRQ | IRQSTAT_C_TAPER_IRQ)) {
|
||||||
if (irqstat_c & IRQSTAT_C_TERMINATION_STAT)
|
if (irqstat_c & IRQSTAT_C_TERMINATION_STAT)
|
||||||
power_supply_changed(&smb->battery);
|
power_supply_changed(&smb->battery);
|
||||||
|
dev_dbg(smb->dev, "going to HW maintenance mode\n");
|
||||||
|
handled = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If we got a charger timeout INT that means the charge
|
||||||
|
* full is not detected with in charge timeout value.
|
||||||
|
*/
|
||||||
|
if (irqstat_d & IRQSTAT_D_CHARGE_TIMEOUT_IRQ) {
|
||||||
|
dev_dbg(smb->dev, "total Charge Timeout INT received\n");
|
||||||
|
|
||||||
|
if (irqstat_d & IRQSTAT_D_CHARGE_TIMEOUT_STAT)
|
||||||
|
dev_warn(smb->dev, "charging stopped due to timeout\n");
|
||||||
|
power_supply_changed(&smb->battery);
|
||||||
handled = true;
|
handled = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -776,6 +800,7 @@ static int smb347_irq_set(struct smb347_charger *smb, bool enable)
|
||||||
* Enable/disable interrupts for:
|
* Enable/disable interrupts for:
|
||||||
* - under voltage
|
* - under voltage
|
||||||
* - termination current reached
|
* - termination current reached
|
||||||
|
* - charger timeout
|
||||||
* - charger error
|
* - charger error
|
||||||
*/
|
*/
|
||||||
ret = regmap_update_bits(smb->regmap, CFG_FAULT_IRQ, 0xff,
|
ret = regmap_update_bits(smb->regmap, CFG_FAULT_IRQ, 0xff,
|
||||||
|
@ -784,7 +809,8 @@ static int smb347_irq_set(struct smb347_charger *smb, bool enable)
|
||||||
goto fail;
|
goto fail;
|
||||||
|
|
||||||
ret = regmap_update_bits(smb->regmap, CFG_STATUS_IRQ, 0xff,
|
ret = regmap_update_bits(smb->regmap, CFG_STATUS_IRQ, 0xff,
|
||||||
enable ? CFG_STATUS_IRQ_TERMINATION_OR_TAPER : 0);
|
enable ? (CFG_STATUS_IRQ_TERMINATION_OR_TAPER |
|
||||||
|
CFG_STATUS_IRQ_CHARGE_TIMEOUT) : 0);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
|
|
||||||
|
@ -988,6 +1014,51 @@ static enum power_supply_property smb347_usb_properties[] = {
|
||||||
POWER_SUPPLY_PROP_CONSTANT_CHARGE_VOLTAGE,
|
POWER_SUPPLY_PROP_CONSTANT_CHARGE_VOLTAGE,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static int smb347_get_charging_status(struct smb347_charger *smb)
|
||||||
|
{
|
||||||
|
int ret, status;
|
||||||
|
unsigned int val;
|
||||||
|
|
||||||
|
if (!smb347_is_ps_online(smb))
|
||||||
|
return POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
|
|
||||||
|
ret = regmap_read(smb->regmap, STAT_C, &val);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
if ((val & STAT_C_CHARGER_ERROR) ||
|
||||||
|
(val & STAT_C_HOLDOFF_STAT)) {
|
||||||
|
/*
|
||||||
|
* set to NOT CHARGING upon charger error
|
||||||
|
* or charging has stopped.
|
||||||
|
*/
|
||||||
|
status = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
||||||
|
} else {
|
||||||
|
if ((val & STAT_C_CHG_MASK) >> STAT_C_CHG_SHIFT) {
|
||||||
|
/*
|
||||||
|
* set to charging if battery is in pre-charge,
|
||||||
|
* fast charge or taper charging mode.
|
||||||
|
*/
|
||||||
|
status = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
|
} else if (val & STAT_C_CHG_TERM) {
|
||||||
|
/*
|
||||||
|
* set the status to FULL if battery is not in pre
|
||||||
|
* charge, fast charge or taper charging mode AND
|
||||||
|
* charging is terminated at least once.
|
||||||
|
*/
|
||||||
|
status = POWER_SUPPLY_STATUS_FULL;
|
||||||
|
} else {
|
||||||
|
/*
|
||||||
|
* in this case no charger error or termination
|
||||||
|
* occured but charging is not in progress!!!
|
||||||
|
*/
|
||||||
|
status = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
static int smb347_battery_get_property(struct power_supply *psy,
|
static int smb347_battery_get_property(struct power_supply *psy,
|
||||||
enum power_supply_property prop,
|
enum power_supply_property prop,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
|
@ -1003,14 +1074,10 @@ static int smb347_battery_get_property(struct power_supply *psy,
|
||||||
|
|
||||||
switch (prop) {
|
switch (prop) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
if (!smb347_is_ps_online(smb)) {
|
ret = smb347_get_charging_status(smb);
|
||||||
val->intval = POWER_SUPPLY_STATUS_DISCHARGING;
|
if (ret < 0)
|
||||||
break;
|
return ret;
|
||||||
}
|
val->intval = ret;
|
||||||
if (smb347_charging_status(smb))
|
|
||||||
val->intval = POWER_SUPPLY_STATUS_CHARGING;
|
|
||||||
else
|
|
||||||
val->intval = POWER_SUPPLY_STATUS_FULL;
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case POWER_SUPPLY_PROP_CHARGE_TYPE:
|
case POWER_SUPPLY_PROP_CHARGE_TYPE:
|
||||||
|
|
Loading…
Reference in New Issue