#include <linux/cleanup.h>
#include <linux/devm-helpers.h>
#include <linux/gpio/consumer.h>
#include <linux/mutex.h>
#include <linux/power_supply.h>
#include <linux/workqueue.h>
#include "adc-battery-helper.h"
#define MOV_AVG_WINDOW_SIZE ADC_BAT_HELPER_MOV_AVG_WINDOW_SIZE
#define INIT_POLL_TIME (5 * HZ)
#define POLL_TIME (30 * HZ)
#define SETTLE_TIME (1 * HZ)
#define INIT_POLL_COUNT 30
#define CURR_HYST_UA 65000
#define LOW_BAT_UV 3700000
#define FULL_BAT_HYST_UV 38000
#define AMBIENT_TEMP_CELSIUS 25
static int adc_battery_helper_get_status(struct adc_battery_helper *help)
{
int full_uv =
help->psy->battery_info->constant_charge_voltage_max_uv - FULL_BAT_HYST_UV;
if (help->curr_ua > CURR_HYST_UA)
return POWER_SUPPLY_STATUS_CHARGING;
if (help->curr_ua < -CURR_HYST_UA)
return POWER_SUPPLY_STATUS_DISCHARGING;
if (help->supplied) {
bool full;
if (help->charge_finished)
full = gpiod_get_value_cansleep(help->charge_finished);
else
full = help->ocv_avg_uv > full_uv;
if (full)
return POWER_SUPPLY_STATUS_FULL;
}
return POWER_SUPPLY_STATUS_NOT_CHARGING;
}
static void adc_battery_helper_work(struct work_struct *work)
{
struct adc_battery_helper *help = container_of(work, struct adc_battery_helper,
work.work);
int i, curr_diff_ua, volt_diff_uv, res_mohm, ret, win_size;
struct device *dev = help->psy->dev.parent;
int volt_uv, prev_volt_uv = help->volt_uv;
int curr_ua, prev_curr_ua = help->curr_ua;
bool prev_supplied = help->supplied;
int prev_status = help->status;
guard(mutex)(&help->lock);
ret = help->get_voltage_and_current_now(help->psy, &volt_uv, &curr_ua);
if (ret)
goto out;
help->volt_uv = volt_uv;
help->curr_ua = curr_ua;
help->ocv_uv[help->ocv_avg_index] =
help->volt_uv - help->curr_ua * help->intern_res_avg_mohm / 1000;
dev_dbg(dev, "volt-now: %d, curr-now: %d, volt-ocv: %d\n",
help->volt_uv, help->curr_ua, help->ocv_uv[help->ocv_avg_index]);
help->ocv_avg_index = (help->ocv_avg_index + 1) % MOV_AVG_WINDOW_SIZE;
help->poll_count++;
help->ocv_avg_uv = 0;
win_size = min(help->poll_count, MOV_AVG_WINDOW_SIZE);
for (i = 0; i < win_size; i++)
help->ocv_avg_uv += help->ocv_uv[i];
help->ocv_avg_uv /= win_size;
help->supplied = power_supply_am_i_supplied(help->psy);
help->status = adc_battery_helper_get_status(help);
if (help->status == POWER_SUPPLY_STATUS_FULL)
help->capacity = 100;
else
help->capacity = power_supply_batinfo_ocv2cap(help->psy->battery_info,
help->ocv_avg_uv,
AMBIENT_TEMP_CELSIUS);
if (help->supplied != prev_supplied ||
help->volt_uv < LOW_BAT_UV ||
help->poll_count < 2)
goto out;
curr_diff_ua = abs(help->curr_ua - prev_curr_ua);
if (curr_diff_ua < CURR_HYST_UA)
goto out;
volt_diff_uv = abs(help->volt_uv - prev_volt_uv);
res_mohm = volt_diff_uv * 1000 / curr_diff_ua;
if ((res_mohm < (help->intern_res_avg_mohm * 2 / 3)) ||
(res_mohm > (help->intern_res_avg_mohm * 4 / 3))) {
dev_dbg(dev, "Ignoring outlier internal resistance %d mOhm\n", res_mohm);
goto out;
}
dev_dbg(dev, "Internal resistance %d mOhm\n", res_mohm);
help->intern_res_mohm[help->intern_res_avg_index] = res_mohm;
help->intern_res_avg_index = (help->intern_res_avg_index + 1) % MOV_AVG_WINDOW_SIZE;
help->intern_res_poll_count++;
help->intern_res_avg_mohm = 0;
win_size = min(help->intern_res_poll_count, MOV_AVG_WINDOW_SIZE);
for (i = 0; i < win_size; i++)
help->intern_res_avg_mohm += help->intern_res_mohm[i];
help->intern_res_avg_mohm /= win_size;
out:
queue_delayed_work(system_percpu_wq, &help->work,
(help->poll_count <= INIT_POLL_COUNT) ?
INIT_POLL_TIME : POLL_TIME);
if (help->status != prev_status)
power_supply_changed(help->psy);
}
const enum power_supply_property adc_battery_helper_properties[] = {
POWER_SUPPLY_PROP_STATUS,
POWER_SUPPLY_PROP_VOLTAGE_NOW,
POWER_SUPPLY_PROP_VOLTAGE_OCV,
POWER_SUPPLY_PROP_CURRENT_NOW,
POWER_SUPPLY_PROP_CAPACITY,
POWER_SUPPLY_PROP_PRESENT,
POWER_SUPPLY_PROP_SCOPE,
};
EXPORT_SYMBOL_GPL(adc_battery_helper_properties);
static_assert(ARRAY_SIZE(adc_battery_helper_properties) ==
ADC_HELPER_NUM_PROPERTIES);
int adc_battery_helper_get_property(struct power_supply *psy,
enum power_supply_property psp,
union power_supply_propval *val)
{
struct adc_battery_helper *help = power_supply_get_drvdata(psy);
int dummy, ret = 0;
guard(mutex)(&help->lock);
switch (psp) {
case POWER_SUPPLY_PROP_STATUS:
val->intval = help->status;
break;
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
ret = help->get_voltage_and_current_now(psy, &val->intval, &dummy);
break;
case POWER_SUPPLY_PROP_VOLTAGE_OCV:
val->intval = help->ocv_avg_uv;
break;
case POWER_SUPPLY_PROP_CURRENT_NOW:
ret = help->get_voltage_and_current_now(psy, &dummy, &val->intval);
break;
case POWER_SUPPLY_PROP_CAPACITY:
val->intval = help->capacity;
break;
case POWER_SUPPLY_PROP_PRESENT:
val->intval = 1;
break;
case POWER_SUPPLY_PROP_SCOPE:
val->intval = POWER_SUPPLY_SCOPE_SYSTEM;
break;
default:
return -EINVAL;
}
return ret;
}
EXPORT_SYMBOL_GPL(adc_battery_helper_get_property);
void adc_battery_helper_external_power_changed(struct power_supply *psy)
{
struct adc_battery_helper *help = power_supply_get_drvdata(psy);
dev_dbg(help->psy->dev.parent, "external power changed\n");
mod_delayed_work(system_percpu_wq, &help->work, SETTLE_TIME);
}
EXPORT_SYMBOL_GPL(adc_battery_helper_external_power_changed);
static void adc_battery_helper_start_work(struct adc_battery_helper *help)
{
help->poll_count = 0;
help->ocv_avg_index = 0;
queue_delayed_work(system_percpu_wq, &help->work, 0);
flush_delayed_work(&help->work);
}
int adc_battery_helper_init(struct adc_battery_helper *help, struct power_supply *psy,
adc_battery_helper_get_func get_voltage_and_current_now,
struct gpio_desc *charge_finished_gpio)
{
struct device *dev = psy->dev.parent;
int ret;
help->psy = psy;
help->get_voltage_and_current_now = get_voltage_and_current_now;
help->charge_finished = charge_finished_gpio;
ret = devm_mutex_init(dev, &help->lock);
if (ret)
return ret;
ret = devm_delayed_work_autocancel(dev, &help->work, adc_battery_helper_work);
if (ret)
return ret;
if (!help->psy->battery_info ||
help->psy->battery_info->factory_internal_resistance_uohm == -EINVAL ||
help->psy->battery_info->constant_charge_voltage_max_uv == -EINVAL ||
!psy->battery_info->ocv_table[0]) {
dev_err(dev, "error required properties are missing\n");
return -ENODEV;
}
help->intern_res_avg_mohm =
help->psy->battery_info->factory_internal_resistance_uohm / 1000;
help->intern_res_mohm[0] = help->intern_res_avg_mohm;
help->intern_res_avg_index = 1;
help->intern_res_poll_count = 1;
adc_battery_helper_start_work(help);
return 0;
}
EXPORT_SYMBOL_GPL(adc_battery_helper_init);
int adc_battery_helper_suspend(struct device *dev)
{
struct adc_battery_helper *help = dev_get_drvdata(dev);
cancel_delayed_work_sync(&help->work);
return 0;
}
EXPORT_SYMBOL_GPL(adc_battery_helper_suspend);
int adc_battery_helper_resume(struct device *dev)
{
struct adc_battery_helper *help = dev_get_drvdata(dev);
adc_battery_helper_start_work(help);
return 0;
}
EXPORT_SYMBOL_GPL(adc_battery_helper_resume);
MODULE_AUTHOR("Hans de Goede <hansg@kernel.org>");
MODULE_DESCRIPTION("ADC battery capacity estimation helper");
MODULE_LICENSE("GPL");