power: supply: ab8500: Standardize temp res lookup

The lookup from battery temperature to internal resistance was
using its own format. Rewrite this to use the table inside
struct power_supply_battery_info:s resist_table.

The supplied resistance table has to be rewritten to express
the resistance in percent of the factory resistance as a
side effect.

We can then rely on the library function
power_supply_temp2resist_simple() to interpolate the internal
resistance percent from the temperature.

Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
Signed-off-by: Sebastian Reichel <sebastian.reichel@collabora.com>
This commit is contained in:
Linus Walleij 2021-11-20 16:53:25 +01:00 committed by Sebastian Reichel
parent bc6e028714
commit 67acb291f3
3 changed files with 38 additions and 55 deletions

View File

@ -379,8 +379,6 @@ struct ab8500_maxim_parameters {
* @r_to_t_tbl: table containing resistance to temp points * @r_to_t_tbl: table containing resistance to temp points
* @n_v_cap_tbl_elements: number of elements in v_to_cap_tbl * @n_v_cap_tbl_elements: number of elements in v_to_cap_tbl
* @v_to_cap_tbl: Voltage to capacity (in %) table * @v_to_cap_tbl: Voltage to capacity (in %) table
* @n_batres_tbl_elements number of elements in the batres_tbl
* @batres_tbl battery internal resistance vs temperature table
*/ */
struct ab8500_battery_type { struct ab8500_battery_type {
int resis_high; int resis_high;
@ -397,8 +395,6 @@ struct ab8500_battery_type {
const struct ab8500_res_to_temp *r_to_t_tbl; const struct ab8500_res_to_temp *r_to_t_tbl;
int n_v_cap_tbl_elements; int n_v_cap_tbl_elements;
const struct ab8500_v_to_cap *v_to_cap_tbl; const struct ab8500_v_to_cap *v_to_cap_tbl;
int n_batres_tbl_elements;
const struct batres_vs_temp *batres_tbl;
}; };
/** /**
@ -502,17 +498,6 @@ struct res_to_temp {
int resist; int resist;
}; };
/**
* struct batres_vs_temp - defines one point in a temp vs battery internal
* resistance curve.
* @temp: battery pack temperature in Celsius
* @resist: battery internal reistance in mOhm
*/
struct batres_vs_temp {
int temp;
int resist;
};
/* Forward declaration */ /* Forward declaration */
struct ab8500_fg; struct ab8500_fg;

View File

@ -67,16 +67,17 @@ static const struct ab8500_res_to_temp temp_tbl[] = {
/* /*
* Note that the batres_vs_temp table must be strictly sorted by falling * Note that the batres_vs_temp table must be strictly sorted by falling
* temperature values to work. * temperature values to work. Factory resistance is 300 mOhm and the
* resistance values to the right are percentages of 300 mOhm.
*/ */
static const struct batres_vs_temp temp_to_batres_tbl_thermistor[] = { static struct power_supply_resistance_temp_table temp_to_batres_tbl_thermistor[] = {
{ 40, 120}, { .temp = 40, .resistance = 40 /* 120 mOhm */ },
{ 30, 135}, { .temp = 30, .resistance = 45 /* 135 mOhm */ },
{ 20, 165}, { .temp = 20, .resistance = 55 /* 165 mOhm */ },
{ 10, 230}, { .temp = 10, .resistance = 77 /* 230 mOhm */ },
{ 00, 325}, { .temp = 00, .resistance = 108 /* 325 mOhm */ },
{-10, 445}, { .temp = -10, .resistance = 158 /* 445 mOhm */ },
{-20, 595}, { .temp = -20, .resistance = 198 /* 595 mOhm */ },
}; };
/* Default battery type for reference designs is the unknown type */ /* Default battery type for reference designs is the unknown type */
@ -95,8 +96,6 @@ static struct ab8500_battery_type bat_type_thermistor_unknown = {
.r_to_t_tbl = temp_tbl, .r_to_t_tbl = temp_tbl,
.n_v_cap_tbl_elements = ARRAY_SIZE(cap_tbl), .n_v_cap_tbl_elements = ARRAY_SIZE(cap_tbl),
.v_to_cap_tbl = cap_tbl, .v_to_cap_tbl = cap_tbl,
.n_batres_tbl_elements = ARRAY_SIZE(temp_to_batres_tbl_thermistor),
.batres_tbl = temp_to_batres_tbl_thermistor,
}; };
static const struct ab8500_bm_capacity_levels cap_levels = { static const struct ab8500_bm_capacity_levels cap_levels = {
@ -209,8 +208,16 @@ int ab8500_bm_of_probe(struct power_supply *psy,
/* Charging stops when we drop below this current */ /* Charging stops when we drop below this current */
bi->charge_term_current_ua = 200000; bi->charge_term_current_ua = 200000;
if (bi->factory_internal_resistance_uohm < 0) /*
* Internal resistance and factory resistance are tightly coupled
* so both MUST be defined or we fall back to defaults.
*/
if ((bi->factory_internal_resistance_uohm < 0) ||
!bi->resist_table) {
bi->factory_internal_resistance_uohm = 300000; bi->factory_internal_resistance_uohm = 300000;
bi->resist_table = temp_to_batres_tbl_thermistor;
bi->resist_table_size = ARRAY_SIZE(temp_to_batres_tbl_thermistor);
}
if (bi->temp_min == INT_MIN) if (bi->temp_min == INT_MIN)
bi->temp_min = AB8500_TEMP_UNDER; bi->temp_min = AB8500_TEMP_UNDER;

View File

@ -901,44 +901,35 @@ static int ab8500_fg_uncomp_volt_to_capacity(struct ab8500_fg *di)
* @di: pointer to the ab8500_fg structure * @di: pointer to the ab8500_fg structure
* *
* Returns battery inner resistance added with the fuel gauge resistor value * Returns battery inner resistance added with the fuel gauge resistor value
* to get the total resistance in the whole link from gnd to bat+ node. * to get the total resistance in the whole link from gnd to bat+ node
* in milliohm.
*/ */
static int ab8500_fg_battery_resistance(struct ab8500_fg *di) static int ab8500_fg_battery_resistance(struct ab8500_fg *di)
{ {
int i, tbl_size; struct power_supply_battery_info *bi = &di->bm->bi;
const struct batres_vs_temp *tbl; int resistance_percent = 0;
int resist = 0; int resistance;
tbl = di->bm->bat_type->batres_tbl; resistance_percent = power_supply_temp2resist_simple(bi->resist_table,
tbl_size = di->bm->bat_type->n_batres_tbl_elements; bi->resist_table_size,
di->bat_temp / 10);
for (i = 0; i < tbl_size; ++i) { /*
if (di->bat_temp / 10 > tbl[i].temp) * We get a percentage of factory resistance here so first get
break; * the factory resistance in milliohms then calculate how much
} * resistance we have at this temperature.
*/
if ((i > 0) && (i < tbl_size)) { resistance = (bi->factory_internal_resistance_uohm / 1000);
resist = fixp_linear_interpolate( resistance = resistance * resistance_percent / 100;
tbl[i].temp,
tbl[i].resist,
tbl[i-1].temp,
tbl[i-1].resist,
di->bat_temp / 10);
} else if (i == 0) {
resist = tbl[0].resist;
} else {
resist = tbl[tbl_size - 1].resist;
}
dev_dbg(di->dev, "%s Temp: %d battery internal resistance: %d" dev_dbg(di->dev, "%s Temp: %d battery internal resistance: %d"
" fg resistance %d, total: %d (mOhm)\n", " fg resistance %d, total: %d (mOhm)\n",
__func__, di->bat_temp, resist, di->bm->fg_res / 10, __func__, di->bat_temp, resistance, di->bm->fg_res / 10,
(di->bm->fg_res / 10) + resist); (di->bm->fg_res / 10) + resistance);
/* fg_res variable is in 0.1mOhm */ /* fg_res variable is in 0.1mOhm */
resist += di->bm->fg_res / 10; resistance += di->bm->fg_res / 10;
return resist; return resistance;
} }
/** /**