1
0
Fork 0
remarkable-linux/drivers/power/supply/max77818_battery.c

1412 lines
36 KiB
C

/*
* Fuel gauge driver for Maxim 77818
* Note that Maxim 77818 is mfd and this is its subdevice.
*
* Copyright (C) 2011 Samsung Electronics
* MyungJoo Ham <myungjoo.ham@samsung.com>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* It is based on max17042_battery driver.
*/
#include <linux/acpi.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/slab.h>
#include <linux/i2c.h>
#include <linux/delay.h>
#include <linux/interrupt.h>
#include <linux/pm.h>
#include <linux/mod_devicetable.h>
#include <linux/power_supply.h>
#include <linux/power/max17042_battery.h>
#include <linux/of.h>
#include <linux/regmap.h>
#include <linux/mfd/max77818/max77818.h>
#include <linux/platform_device.h>
/* Status register bits */
#define STATUS_POR_BIT (1 << 1)
#define STATUS_BST_BIT (1 << 3)
#define STATUS_VMN_BIT (1 << 8)
#define STATUS_TMN_BIT (1 << 9)
#define STATUS_SMN_BIT (1 << 10)
#define STATUS_BI_BIT (1 << 11)
#define STATUS_VMX_BIT (1 << 12)
#define STATUS_TMX_BIT (1 << 13)
#define STATUS_SMX_BIT (1 << 14)
#define STATUS_BR_BIT (1 << 15)
/* Interrupt mask bits */
#define CONFIG_ALRT_BIT_ENBL (1 << 2)
#define STATUS_INTR_SOCMIN_BIT (1 << 10)
#define STATUS_INTR_SOCMAX_BIT (1 << 14)
/* Config register bits */
#define CONFIG_FGCC_BIT (1 << 11)
/* Config2 register bits */
#define CONFIG2_LDMDL (1 << 5)
#define VFSOC0_LOCK 0x0000
#define VFSOC0_UNLOCK 0x0080
#define MODEL_UNLOCK1 0X0059
#define MODEL_UNLOCK2 0X00C4
#define MODEL_LOCK1 0X0000
#define MODEL_LOCK2 0X0000
#define dQ_ACC_DIV 0x4
#define dP_ACC_100 0x1900
#define dP_ACC_200 0x3200
#define MAX77818_VMAX_TOLERANCE 50 /* 50 mV */
#define SYNC_SET_FLAG(flag, lock) ( \
{ \
mutex_lock(lock); \
flag = true; \
mutex_unlock(lock); \
} \
)
#define SYNC_CLEAR_FLAG(flag, lock) ( \
{ \
mutex_lock(lock); \
flag = false; \
mutex_unlock(lock); \
} \
)
#define SYNC_GET_FLAG(flag, lock) ( \
{ \
bool state; \
mutex_lock(lock); \
state = flag; \
mutex_unlock(lock); \
state; \
} \
)
/* Parameter to be given from u-boot after doing update
* in order to verify that all custom FG parameters
* are configured according to DT */
static char config_update_param[255] = "verify";
module_param_string(config_update,
config_update_param,
sizeof(config_update_param),
0644);
MODULE_PARM_DESC(config_update,
"Optional parameter indicating the following config update mode:"
"- complete FG update (config_update=\"complete\")"
"- partial update (config_update=\"partial\")"
"- verification against current DT (config_update=\"verify\")"
"\n"
"By default, the verify mode is set."
"\n"
"Scripts may set the config_update param in order to re-config "
"the device if required, depending on versioning scheme or other "
"means of determining if a complete or partial update is required.");
/* Parameter to be given from command line in order to tune the delay introduced after
* clearing the FGCC bit before forwarding requests to the charger driver */
static int post_fgcc_change_delay_us = 100000;
module_param(post_fgcc_change_delay_us, int, 0644);
MODULE_PARM_DESC(post_fgcc_change_delay_us,
"Debug parameter used to tune the post FGCC change delay introduced "
"to let the charger/fuelgauge take back charging control before doing "
"any other configuration changes on either");
struct max77818_chip {
struct device *dev;
int irq;
struct regmap *regmap;
struct power_supply *battery;
struct max17042_platform_data *pdata;
struct work_struct work;
bool init_complete;
struct power_supply *charger;
struct mutex lock;
};
static enum power_supply_property max77818_battery_props[] = {
POWER_SUPPLY_PROP_STATUS,
POWER_SUPPLY_PROP_PRESENT,
POWER_SUPPLY_PROP_CYCLE_COUNT,
POWER_SUPPLY_PROP_VOLTAGE_MAX,
POWER_SUPPLY_PROP_VOLTAGE_MIN,
POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN,
POWER_SUPPLY_PROP_VOLTAGE_NOW,
POWER_SUPPLY_PROP_VOLTAGE_AVG,
POWER_SUPPLY_PROP_VOLTAGE_OCV,
POWER_SUPPLY_PROP_CAPACITY,
POWER_SUPPLY_PROP_CAPACITY_ALERT_MIN,
POWER_SUPPLY_PROP_CAPACITY_ALERT_MAX,
POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN,
POWER_SUPPLY_PROP_CHARGE_FULL,
POWER_SUPPLY_PROP_CHARGE_NOW,
POWER_SUPPLY_PROP_CHARGE_COUNTER,
POWER_SUPPLY_PROP_TEMP,
POWER_SUPPLY_PROP_TEMP_ALERT_MIN,
POWER_SUPPLY_PROP_TEMP_ALERT_MAX,
POWER_SUPPLY_PROP_HEALTH,
POWER_SUPPLY_PROP_CURRENT_NOW,
POWER_SUPPLY_PROP_CURRENT_AVG,
POWER_SUPPLY_PROP_TIME_TO_EMPTY_NOW,
POWER_SUPPLY_PROP_TIME_TO_FULL_NOW,
POWER_SUPPLY_PROP_CHARGER_MODE,
};
struct max77818_of_property {
const char *property_name;
u8 register_addr;
int (*reg_write_op)(struct regmap *map,
unsigned int reg,
unsigned int value);
bool is_learned_value;
};
static bool max77818_do_complete_update(struct max77818_chip *chip)
{
if (strncmp(config_update_param, "complete", 8) == 0) {
dev_dbg(chip->dev, "config_update='complete'\n");
return true;
}
else {
return false;
}
}
static bool max77818_do_partial_update(struct max77818_chip *chip)
{
if (strncmp(config_update_param, "partial", 7) == 0) {
dev_dbg(chip->dev, "config_update='partial'\n");
return true;
}
else {
return false;
}
}
static bool max77818_do_param_verification(struct max77818_chip *chip)
{
if (strncmp(config_update_param, "verify", 6) == 0) {
dev_dbg(chip->dev, "config_update='verify'\n");
return true;
}
else {
return false;
}
}
static int max77818_get_temperature(struct max77818_chip *chip, int *temp)
{
struct regmap *map = chip->regmap;
u32 data;
int ret;
ret = regmap_read(map, MAX17042_TEMP, &data);
if (ret < 0)
return ret;
*temp = sign_extend32(data, 15);
/* The value is converted into deci-centigrade scale */
/* Units of LSB = 1 / 256 degree Celsius */
*temp = *temp * 10 / 256;
return 0;
}
static int max77818_get_status(struct max77818_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 max77818_get_battery_health(struct max77818_chip *chip, int *health)
{
int temp, vavg, vbatt, ret;
u32 val;
ret = regmap_read(chip->regmap, MAX17042_AvgVCELL, &val);
if (ret < 0)
goto health_error;
/* bits [0-3] unused */
vavg = val * 625 / 8;
/* Convert to millivolts */
vavg /= 1000;
ret = regmap_read(chip->regmap, MAX17042_VCELL, &val);
if (ret < 0)
goto health_error;
/* bits [0-3] unused */
vbatt = val * 625 / 8;
/* Convert to millivolts */
vbatt /= 1000;
if (vavg < chip->pdata->vmin) {
*health = POWER_SUPPLY_HEALTH_DEAD;
goto out;
}
if (vbatt > chip->pdata->vmax + MAX77818_VMAX_TOLERANCE) {
*health = POWER_SUPPLY_HEALTH_OVERVOLTAGE;
goto out;
}
ret = max77818_get_temperature(chip, &temp);
if (ret < 0)
goto health_error;
if (temp < chip->pdata->temp_min) {
*health = POWER_SUPPLY_HEALTH_COLD;
goto out;
}
if (temp > chip->pdata->temp_max) {
*health = POWER_SUPPLY_HEALTH_OVERHEAT;
goto out;
}
*health = POWER_SUPPLY_HEALTH_GOOD;
out:
return 0;
health_error:
return ret;
}
static int max77818_set_fgcc_mode(struct max77818_chip *chip, bool enabled, bool *cur_mode)
{
unsigned int read_data;
int ret;
if (cur_mode) {
ret = regmap_read(chip->regmap, MAX17042_CONFIG, &read_data);
if (ret) {
dev_err(chip->dev, "Failed to read CONFIG register\n");
return ret;
}
*cur_mode = (read_data & CONFIG_FGCC_BIT);
}
dev_dbg(chip->dev, "Turning %s FGCC\n", enabled ? "on" : "off");
ret = regmap_update_bits(chip->regmap,
MAX17042_CONFIG,
CONFIG_FGCC_BIT,
enabled ? CONFIG_FGCC_BIT : 0x0000);
if (ret) {
dev_err(chip->dev,
"Failed to %s FGCC bit in CONFIG register\n",
enabled ? "set" : "clear");
return ret;
}
dev_dbg(chip->dev, "Waiting %d us after FGCC mode change..\n", post_fgcc_change_delay_us);
usleep_range(post_fgcc_change_delay_us, post_fgcc_change_delay_us + 100000);
return 0;
}
static int max77818_set_charger_mode(struct max77818_chip *chip,
const union power_supply_propval *val)
{
bool restore_state;
int ret;
if (!chip->charger) {
return -ENODEV;
}
mutex_lock(&chip->lock);
dev_dbg(chip->dev, "Clearing FGCC mode\n");
ret = max77818_set_fgcc_mode(chip, false, &restore_state);
if (ret) {
dev_err(chip->dev,
"Failed to clear FGCC bit in CONFIG register\n");
goto out;
}
dev_dbg(chip->dev,
"Trying to set charger mode (%d) through charger driver\n",
val->intval);
ret = power_supply_set_property(chip->charger,
POWER_SUPPLY_PROP_CHARGER_MODE,
val);
if (ret) {
dev_err(chip->dev,
"Failed to forward charger mode to charger driver\n");
goto out;
}
if (restore_state) {
dev_dbg(chip->dev,
"Restoring FGCC mode\n");
ret = max77818_set_fgcc_mode(chip, true, NULL);
if (ret) {
dev_err(chip->dev,
"Failed to set FGCC bit in CONFIG register\n");
goto out;
}
}
else {
dev_dbg(chip->dev,
"Leaving FGCC bit as it were (OFF)\n");
}
out:
mutex_unlock(&chip->lock);
return ret;
}
static int max77818_get_charger_mode(struct max77818_chip *chip,
int *charger_mode)
{
bool restore_state;
union power_supply_propval val;
int ret;
if (!chip->charger)
return -ENODEV;
mutex_lock(&chip->lock);
dev_dbg(chip->dev,
"Clearing FGCC mode\n");
ret = max77818_set_fgcc_mode(chip, false, &restore_state);
if (ret) {
dev_err(chip->dev,
"Failed to clear FGCC bit in CONFIG register\n");
goto out;
}
dev_dbg(chip->dev,
"Trying to read charger mode through charger driver\n");
ret = power_supply_get_property(chip->charger,
POWER_SUPPLY_PROP_CHARGER_MODE,
&val);
if (ret) {
dev_err(chip->dev,
"Failed to read charger mode from charger driver\n");
goto out;
}
*charger_mode = val.intval;
if (restore_state) {
dev_dbg(chip->dev,
"Restoring FGCC mode\n");
ret = max77818_set_fgcc_mode(chip, true, NULL);
if (ret) {
dev_err(chip->dev,
"Failed to set FGCC bit in CONFIG register\n");
goto out;
}
}
else {
dev_dbg(chip->dev,
"Leaving FGCC bit as it were (OFF)\n");
}
out:
mutex_unlock(&chip->lock);
return ret;
}
static int max77818_get_property(struct power_supply *psy,
enum power_supply_property psp,
union power_supply_propval *val)
{
struct max77818_chip *chip = power_supply_get_drvdata(psy);
struct regmap *map = chip->regmap;
int temp;
int ret;
u32 data;
u64 data64;
if (!SYNC_GET_FLAG(chip->init_complete, &chip->lock))
return -EAGAIN;
switch (psp) {
case POWER_SUPPLY_PROP_STATUS:
ret = max77818_get_status(chip, &val->intval);
if (ret < 0)
return ret;
break;
case POWER_SUPPLY_PROP_PRESENT:
/*
* MAX17042_STATUS_BattAbsent bit is not working for some
* reason unknown yet. We are working around the issue here by
* reading temperature register, in which a negative value
* indicates absence of battery.
*/
ret = max77818_get_temperature(chip, &temp);
if (ret < 0)
return ret;
if (temp < 0)
val->intval = 0;
else
val->intval = 1;
break;
case POWER_SUPPLY_PROP_CYCLE_COUNT:
ret = regmap_read(map, MAX17042_Cycles, &data);
if (ret < 0)
return ret;
val->intval = data;
break;
case POWER_SUPPLY_PROP_VOLTAGE_MAX:
ret = regmap_read(map, MAX17042_MinMaxVolt, &data);
if (ret < 0)
return ret;
val->intval = data >> 8;
val->intval *= 20000; /* Units of LSB = 20mV */
break;
case POWER_SUPPLY_PROP_VOLTAGE_MIN:
ret = regmap_read(map, MAX17042_MinMaxVolt, &data);
if (ret < 0)
return ret;
val->intval = (data & 0xff) * 20000; /* Units of 20mV */
break;
case POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN:
ret = regmap_read(map, MAX17047_V_empty, &data);
if (ret < 0)
return ret;
val->intval = data >> 7;
val->intval *= 10000; /* Units of LSB = 10mV */
break;
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
ret = regmap_read(map, MAX17042_VCELL, &data);
if (ret < 0)
return ret;
val->intval = data * 625 / 8;
break;
case POWER_SUPPLY_PROP_VOLTAGE_AVG:
ret = regmap_read(map, MAX17042_AvgVCELL, &data);
if (ret < 0)
return ret;
val->intval = data * 625 / 8;
break;
case POWER_SUPPLY_PROP_VOLTAGE_OCV:
ret = regmap_read(map, MAX17042_OCVInternal, &data);
if (ret < 0)
return ret;
val->intval = data * 625 / 8;
break;
case POWER_SUPPLY_PROP_CAPACITY:
ret = regmap_read(map, MAX17042_RepSOC, &data);
if (ret < 0)
return ret;
val->intval = data >> 8;
break;
case POWER_SUPPLY_PROP_CAPACITY_ALERT_MIN:
ret = regmap_read(map, MAX17042_SALRT_Th, &data);
if (ret < 0)
return ret;
val->intval = data & 0xff;
break;
case POWER_SUPPLY_PROP_CAPACITY_ALERT_MAX:
ret = regmap_read(map, MAX17042_SALRT_Th, &data);
if (ret < 0)
return ret;
val->intval = data >> 8;
break;
case POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN:
ret = regmap_read(map, MAX17042_DesignCap, &data);
if (ret < 0)
return ret;
data64 = data * 5000000ll;
do_div(data64, chip->pdata->r_sns);
val->intval = data64;
break;
case POWER_SUPPLY_PROP_CHARGE_FULL:
ret = regmap_read(map, MAX17042_FullCAP, &data);
if (ret < 0)
return ret;
data64 = data * 5000000ll;
do_div(data64, chip->pdata->r_sns);
val->intval = data64;
break;
case POWER_SUPPLY_PROP_CHARGE_NOW:
ret = regmap_read(map, MAX17042_RepCap, &data);
if (ret < 0)
return ret;
data64 = data * 5000000ll;
do_div(data64, chip->pdata->r_sns);
val->intval = data64;
break;
case POWER_SUPPLY_PROP_CHARGE_COUNTER:
ret = regmap_read(map, MAX17042_QH, &data);
if (ret < 0)
return ret;
val->intval = data * 1000 / 2;
break;
case POWER_SUPPLY_PROP_TEMP:
ret = max77818_get_temperature(chip, &val->intval);
if (ret < 0)
return ret;
break;
case POWER_SUPPLY_PROP_TEMP_ALERT_MIN:
ret = regmap_read(map, MAX17042_TALRT_Th, &data);
if (ret < 0)
return ret;
/* LSB is Alert Minimum. In deci-centigrade */
val->intval = sign_extend32(data & 0xff, 7) * 10;
break;
case POWER_SUPPLY_PROP_TEMP_ALERT_MAX:
ret = regmap_read(map, MAX17042_TALRT_Th, &data);
if (ret < 0)
return ret;
/* MSB is Alert Maximum. In deci-centigrade */
val->intval = sign_extend32(data >> 8, 7) * 10;
break;
case POWER_SUPPLY_PROP_HEALTH:
ret = max77818_get_battery_health(chip, &val->intval);
if (ret < 0)
return ret;
break;
case POWER_SUPPLY_PROP_CURRENT_NOW:
ret = regmap_read(map, MAX17042_Current, &data);
if (ret < 0)
return ret;
val->intval = sign_extend32(data, 15);
val->intval *= 1562500 / chip->pdata->r_sns;
break;
case POWER_SUPPLY_PROP_CURRENT_AVG:
ret = regmap_read(map, MAX17042_AvgCurrent, &data);
if (ret < 0)
return ret;
val->intval = sign_extend32(data, 15);
val->intval *= 1562500 / chip->pdata->r_sns;
break;
case POWER_SUPPLY_PROP_TIME_TO_EMPTY_NOW:
ret = regmap_read(map, MAX17042_TTE, &data);
val->intval = data * 5625 / 1000;
break;
case POWER_SUPPLY_PROP_TIME_TO_FULL_NOW:
ret = regmap_read(map, MAX77818_TTF, &data);
val->intval = data * 5625 / 1000;
break;
case POWER_SUPPLY_PROP_CHARGER_MODE:
ret = max77818_get_charger_mode(chip, &val->intval);
if (ret)
return ret;
break;
default:
return -EINVAL;
}
return 0;
}
static int max77818_set_property(struct power_supply *psy,
enum power_supply_property psp,
const union power_supply_propval *val)
{
struct max77818_chip *chip = power_supply_get_drvdata(psy);
struct regmap *map = chip->regmap;
int ret = 0;
u32 data;
int8_t temp;
switch (psp) {
case POWER_SUPPLY_PROP_TEMP_ALERT_MIN:
ret = regmap_read(map, MAX17042_TALRT_Th, &data);
if (ret < 0)
return ret;
/* Input in deci-centigrade, convert to centigrade */
temp = val->intval / 10;
/* force min < max */
if (temp >= (int8_t)(data >> 8))
temp = (int8_t)(data >> 8) - 1;
/* Write both MAX and MIN ALERT */
data = (data & 0xff00) + temp;
ret = regmap_write(map, MAX17042_TALRT_Th, data);
break;
case POWER_SUPPLY_PROP_TEMP_ALERT_MAX:
ret = regmap_read(map, MAX17042_TALRT_Th, &data);
if (ret < 0)
return ret;
/* Input in Deci-Centigrade, convert to centigrade */
temp = val->intval / 10;
/* force max > min */
if (temp <= (int8_t)(data & 0xff))
temp = (int8_t)(data & 0xff) + 1;
/* Write both MAX and MIN ALERT */
data = (data & 0xff) + (temp << 8);
ret = regmap_write(map, MAX17042_TALRT_Th, data);
break;
case POWER_SUPPLY_PROP_CAPACITY_ALERT_MIN:
if (val->intval < 0 || val->intval > 100) {
ret = -EINVAL;
break;
}
ret = regmap_update_bits(map, MAX17042_SALRT_Th,
0xff, val->intval);
break;
case POWER_SUPPLY_PROP_CAPACITY_ALERT_MAX:
if ((val->intval < 0 || val->intval > 100) &&
val->intval != 255) {
ret = -EINVAL;
break;
}
ret = regmap_update_bits(map, MAX17042_SALRT_Th,
0xff00, val->intval << 8);
break;
case POWER_SUPPLY_PROP_CHARGER_MODE:
if ((val->intval < 0) || (val->intval > 2)) {
ret = -EINVAL;
break;
}
ret = max77818_set_charger_mode(chip, val);
break;
default:
ret = -EINVAL;
}
return ret;
}
static int max77818_property_is_writeable(struct power_supply *psy,
enum power_supply_property psp)
{
int ret;
switch (psp) {
case POWER_SUPPLY_PROP_TEMP_ALERT_MIN:
case POWER_SUPPLY_PROP_TEMP_ALERT_MAX:
case POWER_SUPPLY_PROP_CAPACITY_ALERT_MIN:
case POWER_SUPPLY_PROP_CAPACITY_ALERT_MAX:
case POWER_SUPPLY_PROP_CHARGER_MODE:
ret = 1;
break;
default:
ret = 0;
}
return ret;
}
static void max77818_external_power_changed(struct power_supply *psy)
{
power_supply_changed(psy);
}
static int max77818_write_verify_reg(struct regmap *map, unsigned int reg, unsigned int value)
{
int retries = 8;
u32 read_value;
int ret;
do {
ret = regmap_write(map, reg, value);
regmap_read(map, reg, &read_value);
if (read_value != value) {
ret = -EIO;
retries--;
}
} while (retries && read_value != value);
if (ret < 0)
pr_err("%s: err %d\n", __func__, ret);
return ret;
}
static inline void max10742_unlock_model(struct max77818_chip *chip)
{
struct regmap *map = chip->regmap;
regmap_write(map, MAX17042_MLOCKReg1, MODEL_UNLOCK1);
regmap_write(map, MAX17042_MLOCKReg2, MODEL_UNLOCK2);
}
static inline void max10742_lock_model(struct max77818_chip *chip)
{
struct regmap *map = chip->regmap;
regmap_write(map, MAX17042_MLOCKReg1, MODEL_LOCK1);
regmap_write(map, MAX17042_MLOCKReg2, MODEL_LOCK2);
}
static inline void max77818_write_model_data(struct max77818_chip *chip,
u16 *data, int count)
{
struct regmap *map = chip->regmap;
int i;
for (i = 0; i < count; i++)
regmap_write(map, MAX17042_MODELChrTbl + i, data[i]);
}
static inline void max77818_read_model_data(struct max77818_chip *chip,
u16 *data, int count)
{
struct regmap *map = chip->regmap;
u32 tmp;
int i;
for (i = 0; i < count; i++) {
regmap_read(map, MAX17042_MODELChrTbl + i, &tmp);
data[i] = (u16)tmp;
}
}
static inline int max77818_model_data_compare(struct max77818_chip *chip,
u16 *data1, u16 *data2, int size)
{
int i;
if (memcmp(data1, data2, size)) {
dev_err(chip->dev, "%s compare failed\n", __func__);
for (i = 0; i < size; i++)
dev_info(chip->dev, "0x%x, 0x%x", data1[i], data2[i]);
dev_info(chip->dev, "\n");
return -EINVAL;
}
return 0;
}
static int max77818_init_model(struct max77818_chip *chip)
{
struct device_node *np = chip->dev->of_node;
u16 *data, *rdata;
int count;
int ret;
count = of_property_count_u16_elems(np, "maxim,cell-model-data");
if (count < 0 || count != MAX17042_CHARACTERIZATION_DATA_SIZE) {
dev_err(chip->dev, "invalid or missing maxim,cell-model-data: %d\n",
count);
return -EINVAL;
}
data = kcalloc(count, sizeof(*data), GFP_KERNEL);
if (!data) {
dev_err(chip->dev, "failed to kcalloc for data\n");
return -ENOMEM;
}
/* Read cell model data */
of_property_read_u16_array(np, "maxim,cell-model-data", data, count);
rdata = kcalloc(count, sizeof(*rdata), GFP_KERNEL);
if (!rdata) {
dev_err(chip->dev, "failed to kcalloc for rdata\n");
ret = -ENOMEM;
goto free_data;
}
max10742_unlock_model(chip);
max77818_write_model_data(chip, data, count);
max77818_read_model_data(chip, rdata, count);
ret = max77818_model_data_compare(chip, data, rdata, count);
if (ret)
dev_err(chip->dev, "model data compare failed: %d\n", ret);
max10742_lock_model(chip);
kfree(rdata);
free_data:
kfree(data);
return ret;
}
static int max77818_verify_model_lock(struct max77818_chip *chip)
{
int count = MAX17042_CHARACTERIZATION_DATA_SIZE;
u16 *data;
int ret = 0;
int i;
data = kcalloc(count, sizeof(*data), GFP_KERNEL);
if (!data)
return -ENOMEM;
max77818_read_model_data(chip, data, count);
for (i = 0; i < count; i++) {
if (data[i]) {
ret = -EINVAL;
break;
}
}
kfree(data);
return ret;
}
static int max77818_model_loading(struct max77818_chip *chip)
{
unsigned long timeout = jiffies + msecs_to_jiffies(400);
struct regmap *map = chip->regmap;
u32 config2;
/* Setting LdMdl bit */
regmap_read(map, MAX77818_Config2, &config2);
regmap_write(map, MAX77818_Config2, config2 | CONFIG2_LDMDL);
/* Poll LdMdl bit to be 0 under 400ms */
do {
regmap_read(map, MAX77818_Config2, &config2);
if ((config2 & CONFIG2_LDMDL) == 0)
break;
if (time_after(jiffies, timeout))
break;
} while (1);
regmap_read(map, MAX77818_Config2, &config2);
return (config2 & CONFIG2_LDMDL) ? -ETIMEDOUT : 0;
}
static inline int max77818_read_param(struct max77818_chip *chip,
const char *param,
u16 *value)
{
struct device_node *np = chip->dev->of_node;
int ret;
u16 read_value = 0;
ret = of_property_read_u16(np, param, &read_value);
if (!ret)
*value = read_value;
return ret;
}
static int max77818_read_param_and_verify(struct max77818_chip *chip,
struct max77818_of_property *prop)
{
u16 read_param = 0;
int read_cur_value = 0;
bool write_read_param = false;
int ret;
ret = max77818_read_param(chip, prop->property_name, &read_param);
if(!ret) {
dev_dbg(chip->dev, "Verifying '%s' (reg 0x%02x) = 0x%04x\n",
prop->property_name,
prop->register_addr,
read_param);
ret = regmap_read(chip->regmap,
prop->register_addr,
&read_cur_value);
if(ret) {
dev_warn(chip->dev,
"Failed to read '%s' from reg 0x%02x)\n",
prop->property_name,
prop->register_addr);
write_read_param = true;
}
if ((read_param != read_cur_value) || write_read_param) {
dev_dbg(chip->dev,
"Read '%s' (reg 0x%02x) = 0x%04x from device, "
"expected 0x%04x\n",
prop->property_name,
prop->register_addr,
read_cur_value,
read_param);
write_read_param = true;
}
if (write_read_param) {
dev_dbg(chip->dev, "Writing '%s' (reg 0x%02x): 0x%04x\n",
prop->property_name,
prop->register_addr,
read_param);
ret = prop->reg_write_op(chip->regmap,
prop->register_addr,
read_param);
}
}
else if (ret == -EINVAL)
dev_warn(chip->dev,
"'%s' property not given in DT, using default value\n",
prop->property_name);
else
dev_warn(chip->dev,
"Failed to read '%s' param from DT, check value\n",
prop->property_name);
return ret;
}
static int max77818_read_param_and_write(struct max77818_chip *chip,
struct max77818_of_property *prop)
{
u16 read_param;
int ret;
ret = max77818_read_param(chip, prop->property_name, &read_param);
if(!ret) {
dev_dbg(chip->dev, "Writing '%s' (reg 0x%02x): 0x%04x\n",
prop->property_name,
prop->register_addr,
read_param);
ret = prop->reg_write_op(chip->regmap,
prop->register_addr,
read_param);
if(ret) {
dev_warn(chip->dev,
"Failed to write '%s' property read from DT\n",
prop->property_name);
}
}
else if (ret == -EINVAL)
dev_warn(chip->dev,
"'%s' property not given in DT, using default value\n",
prop->property_name);
else
dev_warn(chip->dev,
"Failed to read '%s' param from DT, check value\n",
prop->property_name);
return ret;
}
static void max77818_unlock_extra_config_registers(struct max77818_chip *chip)
{
struct regmap *map = chip->regmap;
u32 value;
int ret;
ret = regmap_write(map, MAX17042_VFSOC0Enable, VFSOC0_UNLOCK);
if (ret)
dev_warn(chip->dev,
"Failed to write VFSOC0Enable to unlock: %d\n", ret);
ret = regmap_read(map, MAX17042_VFSOC, &value);
if (ret)
dev_warn(chip->dev, "Failed to read VFSOC: %d\n", ret);
max77818_write_verify_reg(map, MAX17042_VFSOC0, value);
}
static void max77818_lock_extra_config_registers(struct max77818_chip *chip)
{
struct regmap *map = chip->regmap;
int ret;
ret = regmap_write(map, MAX17042_VFSOC0Enable, VFSOC0_LOCK);
if (ret)
dev_warn(chip->dev,
"Failed to write VFSOC0Enable to lock: %d\n", ret);
}
static struct max77818_of_property max77818_relax_cfg =
{"maxim,relax-cfg", MAX17042_RelaxCFG, regmap_write};
static struct max77818_of_property max77818_custom_param_list [] = {
{ "maxim,learn-cfg", MAX17042_LearnCFG, regmap_write },
{ "maxim,config", MAX17042_CONFIG, regmap_write },
{ "maxim,config2", MAX77818_Config2, regmap_write },
{ "maxim,full-soc-threshold", MAX17047_FullSOCThr, regmap_write },
/* learned value, skipped during verify/write operation at boot */
{ "maxim,fullcaprep", MAX17042_FullCAP0, max77818_write_verify_reg, true },
{ "maxim,design-cap", MAX17042_DesignCap, regmap_write },
/* learned values, skipped during verify/write operation at boot */
{ "maxim,dpacc", MAX17042_dPacc, max77818_write_verify_reg, true },
{ "maxim,dqacc", MAX17042_dQacc, max77818_write_verify_reg, true },
{ "maxim,fullcapnom", MAX17042_FullCAPNom, max77818_write_verify_reg, true },
{ "maxim,misc-cfg", MAX17042_MiscCFG, regmap_write },
{ "maxim,v-empty", MAX17047_V_empty, regmap_write },
{ "maxim,qresidual00", MAX17047_QRTbl00, max77818_write_verify_reg },
{ "maxim,qresidual10", MAX17047_QRTbl10, max77818_write_verify_reg },
{ "maxim,qresidual20", MAX17047_QRTbl20, max77818_write_verify_reg },
{ "maxim,qresidual30", MAX17047_QRTbl30, max77818_write_verify_reg },
/* learned value, skipped during verify/write operation at boot */
{ "maxim,rcomp0", MAX17042_RCOMP0, max77818_write_verify_reg, true },
{ "maxim,tempco", MAX17042_TempCo, max77818_write_verify_reg },
{ "maxim,ichg-term", MAX17042_ICHGTerm, regmap_write },
{ "maxim,filter-cfg", MAX17042_FilterCFG, regmap_write },
/* learned value, skipped during verify/write operation at boot */
{ "maxim,iavg-empty", MAX17042_LAvg_empty, regmap_write, true },
{ "maxim,tgain", MAX17042_TGAIN, regmap_write },
{ "maxim,toff", MAx17042_TOFF, regmap_write },
{ "maxim,tcurve", MAX77818_TCURVE, regmap_write },
{ "maxim,talrt-th", MAX17042_TALRT_Th, regmap_write },
{ "maxim,talrt-th2", MAX77818_TALRT_Th2, regmap_write },
{ "maxim,jeita-curr", MAX77818_JEITA_Curr, regmap_write },
{ "maxim,jeita-volt", MAX77818_JEITA_Volt, regmap_write },
{ "maxim,chargestate0", MAX77818_ChargeState0, regmap_write },
{ "maxim,chargestate1", MAX77818_ChargeState1, regmap_write },
{ "maxim,chargestate2", MAX77818_ChargeState2, regmap_write },
{ "maxim,chargestate3", MAX77818_ChargeState3, regmap_write },
{ "maxim,chargestate4", MAX77818_ChargeState4, regmap_write },
{ "maxim,chargestate5", MAX77818_ChargeState5, regmap_write },
{ "maxim,chargestate6", MAX77818_ChargeState6, regmap_write },
{ "maxim,chargestate7", MAX77818_ChargeState7, regmap_write },
/* The order of the following ones should be respected */
{ "maxim,at-rate", MAX17042_AtRate, regmap_write },
{ "maxim,smart-chg-cfg", MAX77818_SmartChgCfg, regmap_write },
{ "maxim,convgcfg", MAX77818_ConvgCfg, regmap_write },
};
static void max77818_verify_custom_params(struct max77818_chip *chip)
{
int i;
dev_dbg(chip->dev, "Verifying custom params\n");
max77818_read_param_and_verify(chip, &max77818_relax_cfg);
max77818_unlock_extra_config_registers(chip);
for(i = 0; i < ARRAY_SIZE(max77818_custom_param_list); i++) {
if (max77818_custom_param_list[i].is_learned_value) {
dev_dbg(chip->dev,
"Skipping 'learned' value '%s'\n",
max77818_custom_param_list[i].property_name);
continue;
}
max77818_read_param_and_verify(chip,
&max77818_custom_param_list[i]);
}
max77818_lock_extra_config_registers(chip);
SYNC_SET_FLAG(chip->init_complete, &chip->lock);
}
static void max77818_write_custom_params(struct max77818_chip *chip)
{
struct regmap *map = chip->regmap;
int ret, i;
dev_dbg(chip->dev, "Writing custom params\n");
ret = regmap_write(map, MAX17042_RepCap, 0);
if (ret)
dev_warn(chip->dev, "Failed to write RepCap: %d\n", ret);
max77818_read_param_and_write(chip, &max77818_relax_cfg);
max77818_unlock_extra_config_registers(chip);
for(i = 0; i < ARRAY_SIZE(max77818_custom_param_list); i++) {
max77818_read_param_and_write(chip,
&max77818_custom_param_list[i]);
}
max77818_lock_extra_config_registers(chip);
}
static int max77818_init_chip(struct max77818_chip *chip)
{
struct regmap *map = chip->regmap;
int ret;
/* Write cell characterization data */
ret = max77818_init_model(chip);
if (ret) {
dev_err(chip->dev, "init model failed: %d\n", ret);
return ret;
}
ret = max77818_verify_model_lock(chip);
if (ret) {
dev_err(chip->dev, "lock verify failed: %d\n", ret);
return ret;
}
/* Write custom parameters from devcie tree */
max77818_write_custom_params(chip);
/* Initiate model loading for MAX77818 */
ret = max77818_model_loading(chip);
if (ret) {
dev_err(chip->dev, "initiate model loading failed: %d\n", ret);
return ret;
}
/* Wait 500 ms for SOC to be calculated from the new parameters */
msleep(500);
/* Init complete, Clear the POR bit */
regmap_update_bits(map, MAX17042_STATUS, STATUS_POR_BIT, 0x0);
return 0;
}
static irqreturn_t max77818_thread_handler(int id, void *dev)
{
struct max77818_chip *chip = dev;
u32 val;
regmap_read(chip->regmap, MAX17042_STATUS, &val);
if (val & STATUS_INTR_SOCMIN_BIT) {
dev_info(chip->dev, "MIN SOC alert\n");
sysfs_notify(&chip->battery->dev.kobj, NULL,
"capacity_alert_min");
}
if (val & STATUS_INTR_SOCMAX_BIT) {
dev_info(chip->dev, "MAX SOC alert\n");
sysfs_notify(&chip->battery->dev.kobj, NULL,
"capacity_alert_max");
}
power_supply_changed(chip->battery);
return IRQ_HANDLED;
}
static void max77818_init_worker(struct work_struct *work)
{
struct max77818_chip *chip = container_of(work, struct max77818_chip,
work);
int ret;
dev_dbg(chip->dev, "Doing complete re-config\n");
ret = max77818_init_chip(chip);
if (ret) {
dev_err(chip->dev, "failed to init chip: %d\n", ret);
return;
}
SYNC_SET_FLAG(chip->init_complete, &chip->lock);
}
static struct max17042_platform_data *
max77818_get_pdata(struct max77818_chip *chip)
{
struct max17042_platform_data *pdata;
struct device *dev = chip->dev;
struct device_node *np = dev->of_node;
pdata = devm_kzalloc(dev, sizeof(*pdata), GFP_KERNEL);
if (!pdata)
return NULL;
if (of_property_read_u32(np, "maxim,rsns-microohm", &pdata->r_sns))
pdata->r_sns = MAX17042_DEFAULT_SNS_RESISTOR;
if (of_property_read_s32(np, "maxim,cold-temp", &pdata->temp_min))
pdata->temp_min = INT_MIN;
if (of_property_read_s32(np, "maxim,over-heat-temp", &pdata->temp_max))
pdata->temp_max = INT_MAX;
if (of_property_read_s32(np, "maxim,dead-volt", &pdata->vmin))
pdata->vmin = INT_MIN;
if (of_property_read_s32(np, "maxim,over-volt", &pdata->vmax))
pdata->vmax = INT_MAX;
return pdata;
}
static const struct regmap_config max77818_regmap_config = {
.reg_bits = 8,
.val_bits = 16,
.val_format_endian = REGMAP_ENDIAN_NATIVE,
};
static const struct power_supply_desc max77818_psy_desc = {
.name = "max77818_battery",
.type = POWER_SUPPLY_TYPE_BATTERY,
.get_property = max77818_get_property,
.set_property = max77818_set_property,
.property_is_writeable = max77818_property_is_writeable,
.external_power_changed = max77818_external_power_changed,
.properties = max77818_battery_props,
.num_properties = ARRAY_SIZE(max77818_battery_props),
};
static int max77818_probe(struct platform_device *pdev)
{
struct max77818_dev *max77818 = dev_get_drvdata(pdev->dev.parent);
struct power_supply_config psy_cfg = {};
struct device *dev = &pdev->dev;
struct max77818_chip *chip;
u32 val;
int ret;
chip = devm_kzalloc(dev, sizeof(*chip), GFP_KERNEL);
if (!chip)
return -ENOMEM;
mutex_init(&chip->lock);
chip->dev = dev;
chip->regmap = max77818->regmap_fg;
chip->pdata = max77818_get_pdata(chip);
if (!chip->pdata) {
dev_err(dev, "no platform data provided\n");
return -EINVAL;
}
platform_set_drvdata(pdev, chip);
psy_cfg.drv_data = chip;
psy_cfg.of_node = dev->of_node;
SYNC_SET_FLAG(chip->init_complete, &chip->lock);
chip->battery = devm_power_supply_register(dev, &max77818_psy_desc,
&psy_cfg);
if (IS_ERR(chip->battery)) {
ret = PTR_ERR(chip->battery);
dev_err(dev, "failed to regiser supply: %d \n", ret);
return ret;
}
dev_dbg(dev, "Trying to reference charger (expecting charger as"
"first supply in given list of supplies)\n");
if (chip->battery->num_supplies > 0) {
chip->charger = power_supply_get_by_name(
chip->battery->supplied_from[0]);
if (!chip->charger) {
dev_warn(dev, "SBA: Failed to reference charger, "
"OTG/charger IRQ handling will not"
"be available - verify DT config\n");
}
}
/* Disable max SOC alert and set min SOC alert as 10% by default */
regmap_write(chip->regmap, MAX17042_SALRT_Th, (0xff << 8) | 0x0a );
chip->irq = regmap_irq_get_virq(max77818->irqc_intsrc, MAX77818_FG_INT);
if (chip->irq <= 0) {
dev_err(dev, "failed to get virq: %d\n", chip->irq);
return -ENODEV;
}
ret = devm_request_threaded_irq(dev, chip->irq, NULL,
max77818_thread_handler, 0,
chip->battery->desc->name,
chip);
if (ret) {
dev_err(dev, "failed to request irq: %d\n", ret);
return ret;
}
/* Read the POR bit set when the device boots from a total power loss.
* If this bit is set, the device is given its initial config read from
* DT.
*
* If the POR is not set, but the module parameter 'config_update' is
* set to 'complete' when booting (by image update scripts typically),
* a complete re-config of the FG device is performed in order to apply
* battery model parameters in addition to custom params read
* from updated DT.
*
* If the POR is not set, but the module parameter 'config_update' is
* set to 'partial' when booting (by image update scripts typically),
* a partial re-config of the custom params is performed in order to apply
* only custom parameters read from updated DT.
*/
regmap_read(chip->regmap, MAX17042_STATUS, &val);
if ((val & STATUS_POR_BIT) || max77818_do_complete_update(chip)) {
INIT_WORK(&chip->work, max77818_init_worker);
schedule_work(&chip->work);
} else if (max77818_do_partial_update(chip)) {
max77818_write_custom_params(chip);
} else if (max77818_do_param_verification(chip)) {
max77818_verify_custom_params(chip);
}
else {
dev_dbg(chip->dev, "No config change\n");
SYNC_SET_FLAG(chip->init_complete, &chip->lock);
}
return 0;
}
static struct platform_driver max77818_fg_driver = {
.driver = {
.name = MAX77818_FUELGAUGE_NAME,
.owner = THIS_MODULE,
},
.probe = max77818_probe,
};
module_platform_driver(max77818_fg_driver);
MODULE_DESCRIPTION("MAX77818 Fuel Gauge");
MODULE_LICENSE("GPL");