You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1266 lines
36 KiB
1266 lines
36 KiB
/* Copyright (c) 2017-2018 The Linux Foundation. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 and
|
|
* only version 2 as published by the Free Software Foundation.
|
|
*
|
|
* 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.
|
|
*/
|
|
#define pr_fmt(fmt) "QCOM-STEPCHG: %s: " fmt, __func__
|
|
|
|
#include <linux/delay.h>
|
|
#include <linux/module.h>
|
|
#include <linux/of.h>
|
|
#include <linux/of_batterydata.h>
|
|
#include <linux/power_supply.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/pmic-voter.h>
|
|
#include "step-chg-jeita.h"
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
#include "../../../battery_qc/include/sec_battery_qc.h"
|
|
#endif
|
|
|
|
#define STEP_CHG_VOTER "STEP_CHG_VOTER"
|
|
#define JEITA_VOTER "JEITA_VOTER"
|
|
|
|
#define is_between(left, right, value) \
|
|
(((left) >= (right) && (left) >= (value) \
|
|
&& (value) >= (right)) \
|
|
|| ((left) <= (right) && (left) <= (value) \
|
|
&& (value) <= (right)))
|
|
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC) && defined(CONFIG_ENG_BATTERY_CONCEPT)
|
|
int step_chg_test_low[3];
|
|
int step_chg_test_high[3];
|
|
int step_chg_test_val[3];
|
|
bool update_step;
|
|
#endif
|
|
|
|
struct step_chg_cfg {
|
|
struct step_chg_jeita_param param;
|
|
struct range_data fcc_cfg[MAX_STEP_CHG_ENTRIES];
|
|
};
|
|
|
|
struct jeita_fcc_cfg {
|
|
struct step_chg_jeita_param param;
|
|
struct range_data fcc_cfg[MAX_STEP_CHG_ENTRIES];
|
|
};
|
|
|
|
struct jeita_fv_cfg {
|
|
struct step_chg_jeita_param param;
|
|
struct range_data fv_cfg[MAX_STEP_CHG_ENTRIES];
|
|
};
|
|
|
|
struct step_chg_info {
|
|
struct device *dev;
|
|
ktime_t step_last_update_time;
|
|
ktime_t jeita_last_update_time;
|
|
bool step_chg_enable;
|
|
bool sw_jeita_enable;
|
|
bool jeita_arb_en;
|
|
bool config_is_read;
|
|
bool step_chg_cfg_valid;
|
|
bool sw_jeita_cfg_valid;
|
|
bool soc_based_step_chg;
|
|
bool ocv_based_step_chg;
|
|
bool vbat_avg_based_step_chg;
|
|
bool batt_missing;
|
|
bool taper_fcc;
|
|
int jeita_fcc_index;
|
|
int jeita_fv_index;
|
|
int step_index;
|
|
int get_config_retry_count;
|
|
|
|
struct step_chg_cfg *step_chg_config;
|
|
struct jeita_fcc_cfg *jeita_fcc_config;
|
|
struct jeita_fv_cfg *jeita_fv_config;
|
|
|
|
struct votable *fcc_votable;
|
|
struct votable *fv_votable;
|
|
struct votable *usb_icl_votable;
|
|
struct wakeup_source *step_chg_ws;
|
|
struct power_supply *batt_psy;
|
|
struct power_supply *bms_psy;
|
|
struct power_supply *usb_psy;
|
|
#if !defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
struct power_supply *main_psy;
|
|
#endif
|
|
struct power_supply *dc_psy;
|
|
struct delayed_work status_change_work;
|
|
struct delayed_work get_config_work;
|
|
struct notifier_block nb;
|
|
};
|
|
|
|
static struct step_chg_info *the_chip;
|
|
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
#if defined(CONFIG_SEC_FACTORY)
|
|
#define STEP_CHG_HYSTERISIS_DELAY_US 1000000 /* 1 sec */
|
|
#else
|
|
#define STEP_CHG_HYSTERISIS_DELAY_US 7000000 /* 7 secs */
|
|
#endif
|
|
#else /* !CONFIG_BATTERY_SAMSUNG_USING_QC */
|
|
#define STEP_CHG_HYSTERISIS_DELAY_US 5000000 /* 5 secs */
|
|
#endif
|
|
|
|
#define BATT_HOT_DECIDEGREE_MAX 600
|
|
#define GET_CONFIG_DELAY_MS 2000
|
|
#define GET_CONFIG_RETRY_COUNT 50
|
|
#define WAIT_BATT_ID_READY_MS 200
|
|
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
int swelling_index[SWELLING_TYPE_CNT][JEITA_MAX] = {
|
|
{-1, 0, 1, 2, 3}, /* X - COOL2 - COOL1 - NORMAL - WARM */
|
|
{0, 1, 2, 3, 4}, /* COOL3 - COOL2 - COOL1 - NORMAL - WARM */
|
|
};
|
|
|
|
int swelling_type;
|
|
static int step_max;
|
|
static int factory_step_current;
|
|
|
|
int get_factory_step_max(void)
|
|
{
|
|
return step_max;
|
|
}
|
|
int get_factory_step_current(void)
|
|
{
|
|
return factory_step_current;
|
|
}
|
|
#endif
|
|
|
|
static bool is_batt_available(struct step_chg_info *chip)
|
|
{
|
|
if (!chip->batt_psy)
|
|
chip->batt_psy = power_supply_get_by_name("battery");
|
|
|
|
if (!chip->batt_psy)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool is_bms_available(struct step_chg_info *chip)
|
|
{
|
|
if (!chip->bms_psy)
|
|
chip->bms_psy = power_supply_get_by_name("bms");
|
|
|
|
if (!chip->bms_psy)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool is_usb_available(struct step_chg_info *chip)
|
|
{
|
|
if (!chip->usb_psy)
|
|
chip->usb_psy = power_supply_get_by_name("usb");
|
|
|
|
if (!chip->usb_psy)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool is_input_present(struct step_chg_info *chip)
|
|
{
|
|
int rc = 0, input_present = 0;
|
|
union power_supply_propval pval = {0, };
|
|
|
|
if (!chip->usb_psy)
|
|
chip->usb_psy = power_supply_get_by_name("usb");
|
|
if (chip->usb_psy) {
|
|
rc = power_supply_get_property(chip->usb_psy,
|
|
POWER_SUPPLY_PROP_PRESENT, &pval);
|
|
if (rc < 0)
|
|
pr_err("Couldn't read USB Present status, rc=%d\n", rc);
|
|
else
|
|
input_present |= pval.intval;
|
|
}
|
|
|
|
if (!chip->dc_psy)
|
|
chip->dc_psy = power_supply_get_by_name("dc");
|
|
if (chip->dc_psy) {
|
|
rc = power_supply_get_property(chip->dc_psy,
|
|
POWER_SUPPLY_PROP_PRESENT, &pval);
|
|
if (rc < 0)
|
|
pr_err("Couldn't read DC Present status, rc=%d\n", rc);
|
|
else
|
|
input_present |= pval.intval;
|
|
}
|
|
|
|
if (input_present)
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
int read_range_data_from_node(struct device_node *node,
|
|
const char *prop_str, struct range_data *ranges,
|
|
u32 max_threshold, u32 max_value)
|
|
{
|
|
int rc = 0, i, length, per_tuple_length, tuples;
|
|
|
|
if (!node || !prop_str || !ranges) {
|
|
pr_err("Invalid parameters passed\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = of_property_count_elems_of_size(node, prop_str, sizeof(u32));
|
|
if (rc < 0) {
|
|
pr_err("Count %s failed, rc=%d\n", prop_str, rc);
|
|
return rc;
|
|
}
|
|
|
|
length = rc;
|
|
per_tuple_length = sizeof(struct range_data) / sizeof(u32);
|
|
if (length % per_tuple_length) {
|
|
pr_err("%s length (%d) should be multiple of %d\n",
|
|
prop_str, length, per_tuple_length);
|
|
return -EINVAL;
|
|
}
|
|
tuples = length / per_tuple_length;
|
|
|
|
if (tuples > MAX_STEP_CHG_ENTRIES) {
|
|
pr_err("too many entries(%d), only %d allowed\n",
|
|
tuples, MAX_STEP_CHG_ENTRIES);
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = of_property_read_u32_array(node, prop_str,
|
|
(u32 *)ranges, length);
|
|
if (rc) {
|
|
pr_err("Read %s failed, rc=%d", prop_str, rc);
|
|
return rc;
|
|
}
|
|
|
|
for (i = 0; i < tuples; i++) {
|
|
if (ranges[i].low_threshold >
|
|
ranges[i].high_threshold) {
|
|
pr_err("%s thresholds should be in ascendant ranges\n",
|
|
prop_str);
|
|
rc = -EINVAL;
|
|
goto clean;
|
|
}
|
|
|
|
if (i != 0) {
|
|
if (ranges[i - 1].high_threshold >
|
|
ranges[i].low_threshold) {
|
|
pr_err("%s thresholds should be in ascendant ranges\n",
|
|
prop_str);
|
|
rc = -EINVAL;
|
|
goto clean;
|
|
}
|
|
}
|
|
|
|
if (ranges[i].low_threshold > max_threshold)
|
|
ranges[i].low_threshold = max_threshold;
|
|
if (ranges[i].high_threshold > max_threshold)
|
|
ranges[i].high_threshold = max_threshold;
|
|
if (ranges[i].value > max_value)
|
|
ranges[i].value = max_value;
|
|
}
|
|
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
if (strcmp(prop_str, "qcom,jeita-fcc-ranges") == 0) {
|
|
swelling_type = tuples - DEFAULT_SWELLING_CNT;
|
|
|
|
if ((swelling_type < 0) || (swelling_type >= SWELLING_TYPE_CNT)) {
|
|
pr_info("%s: invalid swelling_type : %d\n", __func__, swelling_type);
|
|
swelling_type = 0;
|
|
}
|
|
|
|
pr_info("%s: swelling_type : %d\n", __func__, swelling_type);
|
|
}
|
|
if (strcmp(prop_str, "qcom,step-chg-ranges") == 0) {
|
|
step_max = tuples - 1;
|
|
factory_step_current = ranges[step_max].value;
|
|
pr_info("%s: factory step max: %d, current: %d\n", __func__, step_max, factory_step_current);
|
|
}
|
|
#endif
|
|
|
|
return rc;
|
|
clean:
|
|
memset(ranges, 0, tuples * sizeof(struct range_data));
|
|
return rc;
|
|
}
|
|
EXPORT_SYMBOL(read_range_data_from_node);
|
|
|
|
static int get_step_chg_jeita_setting_from_profile(struct step_chg_info *chip)
|
|
{
|
|
struct device_node *batt_node, *profile_node;
|
|
u32 max_fv_uv, max_fcc_ma;
|
|
const char *batt_type_str;
|
|
const __be32 *handle;
|
|
int batt_id_ohms, rc;
|
|
union power_supply_propval prop = {0, };
|
|
|
|
handle = of_get_property(chip->dev->of_node,
|
|
"qcom,battery-data", NULL);
|
|
if (!handle) {
|
|
pr_debug("ignore getting sw-jeita/step charging settings from profile\n");
|
|
return 0;
|
|
}
|
|
|
|
batt_node = of_find_node_by_phandle(be32_to_cpup(handle));
|
|
if (!batt_node) {
|
|
pr_err("Get battery data node failed\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!is_bms_available(chip))
|
|
return -ENODEV;
|
|
|
|
power_supply_get_property(chip->bms_psy,
|
|
POWER_SUPPLY_PROP_RESISTANCE_ID, &prop);
|
|
batt_id_ohms = prop.intval;
|
|
|
|
/* bms_psy has not yet read the batt_id */
|
|
if (batt_id_ohms < 0)
|
|
return -EBUSY;
|
|
|
|
profile_node = of_batterydata_get_best_profile(batt_node,
|
|
batt_id_ohms / 1000, NULL);
|
|
if (IS_ERR(profile_node))
|
|
return PTR_ERR(profile_node);
|
|
|
|
if (!profile_node) {
|
|
pr_err("Couldn't find profile\n");
|
|
return -ENODATA;
|
|
}
|
|
|
|
rc = of_property_read_string(profile_node, "qcom,battery-type",
|
|
&batt_type_str);
|
|
if (rc < 0) {
|
|
pr_err("battery type unavailable, rc:%d\n", rc);
|
|
return rc;
|
|
}
|
|
pr_debug("battery: %s detected, getting sw-jeita/step charging settings\n",
|
|
batt_type_str);
|
|
|
|
rc = of_property_read_u32(profile_node, "qcom,max-voltage-uv",
|
|
&max_fv_uv);
|
|
if (rc < 0) {
|
|
pr_err("max-voltage_uv reading failed, rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = of_property_read_u32(profile_node, "qcom,fastchg-current-ma",
|
|
&max_fcc_ma);
|
|
if (rc < 0) {
|
|
pr_err("max-fastchg-current-ma reading failed, rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
chip->taper_fcc = of_property_read_bool(profile_node, "qcom,taper-fcc");
|
|
|
|
chip->soc_based_step_chg =
|
|
of_property_read_bool(profile_node, "qcom,soc-based-step-chg");
|
|
if (chip->soc_based_step_chg) {
|
|
chip->step_chg_config->param.psy_prop =
|
|
POWER_SUPPLY_PROP_CAPACITY;
|
|
chip->step_chg_config->param.prop_name = "SOC";
|
|
chip->step_chg_config->param.hysteresis = 0;
|
|
}
|
|
|
|
chip->ocv_based_step_chg =
|
|
of_property_read_bool(profile_node, "qcom,ocv-based-step-chg");
|
|
if (chip->ocv_based_step_chg) {
|
|
chip->step_chg_config->param.psy_prop =
|
|
POWER_SUPPLY_PROP_VOLTAGE_OCV;
|
|
chip->step_chg_config->param.prop_name = "OCV";
|
|
chip->step_chg_config->param.hysteresis = 10000;
|
|
chip->step_chg_config->param.use_bms = true;
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
chip->step_chg_config->param.hysteresis = 0;
|
|
#endif
|
|
}
|
|
|
|
chip->vbat_avg_based_step_chg =
|
|
of_property_read_bool(profile_node,
|
|
"qcom,vbat-avg-based-step-chg");
|
|
if (chip->vbat_avg_based_step_chg) {
|
|
chip->step_chg_config->param.psy_prop =
|
|
POWER_SUPPLY_PROP_VOLTAGE_AVG;
|
|
chip->step_chg_config->param.prop_name = "VBAT_AVG";
|
|
chip->step_chg_config->param.hysteresis = 0;
|
|
chip->step_chg_config->param.use_bms = true;
|
|
}
|
|
|
|
chip->step_chg_cfg_valid = true;
|
|
rc = read_range_data_from_node(profile_node,
|
|
"qcom,step-chg-ranges",
|
|
chip->step_chg_config->fcc_cfg,
|
|
chip->soc_based_step_chg ? 100 : max_fv_uv,
|
|
max_fcc_ma * 1000);
|
|
if (rc < 0) {
|
|
pr_debug("Read qcom,step-chg-ranges failed from battery profile, rc=%d\n",
|
|
rc);
|
|
chip->step_chg_cfg_valid = false;
|
|
}
|
|
|
|
chip->sw_jeita_cfg_valid = true;
|
|
rc = read_range_data_from_node(profile_node,
|
|
"qcom,jeita-fcc-ranges",
|
|
chip->jeita_fcc_config->fcc_cfg,
|
|
BATT_HOT_DECIDEGREE_MAX, max_fcc_ma * 1000);
|
|
if (rc < 0) {
|
|
pr_debug("Read qcom,jeita-fcc-ranges failed from battery profile, rc=%d\n",
|
|
rc);
|
|
chip->sw_jeita_cfg_valid = false;
|
|
}
|
|
|
|
rc = read_range_data_from_node(profile_node,
|
|
"qcom,jeita-fv-ranges",
|
|
chip->jeita_fv_config->fv_cfg,
|
|
BATT_HOT_DECIDEGREE_MAX, max_fv_uv);
|
|
if (rc < 0) {
|
|
pr_debug("Read qcom,jeita-fv-ranges failed from battery profile, rc=%d\n",
|
|
rc);
|
|
chip->sw_jeita_cfg_valid = false;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC) && defined(CONFIG_ENG_BATTERY_CONCEPT)
|
|
void update_step_chg_data(int t[])
|
|
{
|
|
step_chg_test_low[0] = t[0] * 1000;
|
|
step_chg_test_high[0] = t[1] * 1000;
|
|
step_chg_test_val[0] = t[2] * 1000;
|
|
step_chg_test_low[1] = t[3] * 1000;
|
|
step_chg_test_high[1] = t[4] * 1000;
|
|
step_chg_test_val[1] = t[5] * 1000;
|
|
step_chg_test_low[2] = t[6] * 1000;
|
|
step_chg_test_high[2] = t[7] * 1000;
|
|
step_chg_test_val[2] = t[8] * 1000;
|
|
update_step = true;
|
|
}
|
|
#endif
|
|
|
|
static void get_config_work(struct work_struct *work)
|
|
{
|
|
struct step_chg_info *chip = container_of(work,
|
|
struct step_chg_info, get_config_work.work);
|
|
int i, rc;
|
|
|
|
chip->config_is_read = false;
|
|
rc = get_step_chg_jeita_setting_from_profile(chip);
|
|
|
|
if (rc < 0) {
|
|
if (rc == -ENODEV || rc == -EBUSY) {
|
|
if (chip->get_config_retry_count++
|
|
< GET_CONFIG_RETRY_COUNT) {
|
|
pr_debug("bms_psy is not ready, retry: %d\n",
|
|
chip->get_config_retry_count);
|
|
goto reschedule;
|
|
}
|
|
}
|
|
}
|
|
|
|
chip->config_is_read = true;
|
|
|
|
for (i = 0; i < MAX_STEP_CHG_ENTRIES; i++)
|
|
pr_debug("step-chg-cfg: %duV(SoC) ~ %duV(SoC), %duA\n",
|
|
chip->step_chg_config->fcc_cfg[i].low_threshold,
|
|
chip->step_chg_config->fcc_cfg[i].high_threshold,
|
|
chip->step_chg_config->fcc_cfg[i].value);
|
|
for (i = 0; i < MAX_STEP_CHG_ENTRIES; i++)
|
|
pr_debug("jeita-fcc-cfg: %ddecidegree ~ %ddecidegre, %duA\n",
|
|
chip->jeita_fcc_config->fcc_cfg[i].low_threshold,
|
|
chip->jeita_fcc_config->fcc_cfg[i].high_threshold,
|
|
chip->jeita_fcc_config->fcc_cfg[i].value);
|
|
for (i = 0; i < MAX_STEP_CHG_ENTRIES; i++)
|
|
pr_debug("jeita-fv-cfg: %ddecidegree ~ %ddecidegre, %duV\n",
|
|
chip->jeita_fv_config->fv_cfg[i].low_threshold,
|
|
chip->jeita_fv_config->fv_cfg[i].high_threshold,
|
|
chip->jeita_fv_config->fv_cfg[i].value);
|
|
|
|
return;
|
|
|
|
reschedule:
|
|
schedule_delayed_work(&chip->get_config_work,
|
|
msecs_to_jiffies(GET_CONFIG_DELAY_MS));
|
|
|
|
}
|
|
|
|
static int get_val(struct range_data *range, int hysteresis, int current_index,
|
|
int threshold,
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
int change_rechg, struct step_chg_info *chip,
|
|
#endif
|
|
int *new_index, int *val)
|
|
{
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
struct sec_battery_info *battery = get_sec_battery();
|
|
union power_supply_propval pval = {0, };
|
|
int rc = 0;
|
|
int rechg_vbat_cool;
|
|
#endif
|
|
int i;
|
|
|
|
*new_index = -EINVAL;
|
|
|
|
/*
|
|
* If the threshold is lesser than the minimum allowed range,
|
|
* return -ENODATA.
|
|
*/
|
|
if (threshold < range[0].low_threshold)
|
|
return -ENODATA;
|
|
|
|
/* First try to find the matching index without hysteresis */
|
|
for (i = 0; i < MAX_STEP_CHG_ENTRIES; i++) {
|
|
if (!range[i].high_threshold && !range[i].low_threshold) {
|
|
/* First invalid table entry; exit loop */
|
|
break;
|
|
}
|
|
|
|
if (is_between(range[i].low_threshold,
|
|
range[i].high_threshold, threshold)) {
|
|
*new_index = i;
|
|
*val = range[i].value;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* If nothing was found, the threshold exceeds the max range for sure
|
|
* as the other case where it is lesser than the min range is handled
|
|
* at the very beginning of this function. Therefore, clip it to the
|
|
* max allowed range value, which is the one corresponding to the last
|
|
* valid entry in the battery profile data array.
|
|
*/
|
|
if (*new_index == -EINVAL) {
|
|
if (i == 0) {
|
|
/* Battery profile data array is completely invalid */
|
|
return -ENODATA;
|
|
}
|
|
|
|
*new_index = (i - 1);
|
|
*val = range[*new_index].value;
|
|
}
|
|
|
|
/*
|
|
* If we don't have a current_index return this
|
|
* newfound value. There is no hysterisis from out of range
|
|
* to in range transition
|
|
*/
|
|
#if !defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
if (current_index == -EINVAL)
|
|
return 0;
|
|
#endif
|
|
/*
|
|
* Check for hysteresis if it in the neighbourhood
|
|
* of our current index.
|
|
*/
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
if (current_index != -EINVAL) {
|
|
#endif
|
|
if (*new_index == current_index + 1) {
|
|
if (threshold < range[*new_index].low_threshold + hysteresis) {
|
|
/*
|
|
* Stay in the current index, threshold is not higher
|
|
* by hysteresis amount
|
|
*/
|
|
*new_index = current_index;
|
|
*val = range[current_index].value;
|
|
}
|
|
} else if (*new_index == current_index - 1) {
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
if (hysteresis == 0) /* Hysteresis 0 is only for step-chg */
|
|
hysteresis = 100000;
|
|
#endif
|
|
if (threshold > range[*new_index].high_threshold - hysteresis) {
|
|
/*
|
|
* stay in the current index, threshold is not lower
|
|
* by hysteresis amount
|
|
*/
|
|
*new_index = current_index;
|
|
*val = range[current_index].value;
|
|
}
|
|
}
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
}
|
|
#endif
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
if (change_rechg && (*new_index != current_index)) { /* COOL3 - COOL2 - COOL1 - NORMAL - WARM */
|
|
sec_bat_set_current_event(0, SEC_BAT_CURRENT_EVENT_SWELLING_MODE);
|
|
rc = power_supply_get_property(chip->bms_psy,
|
|
POWER_SUPPLY_PROP_VOLTAGE_MAX, &pval);
|
|
if (rc < 0) {
|
|
pr_err("get recharge Vbat failed, rc=%d\n", rc);
|
|
rechg_vbat_cool = RECHG_COOL;
|
|
} else {
|
|
rechg_vbat_cool = pval.intval / 1000 - 150; //mv
|
|
}
|
|
|
|
if (*new_index == swelling_index[swelling_type][JEITA_COOL1]) {
|
|
pval.intval = rechg_vbat_cool;
|
|
sec_bat_set_current_event(
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING,
|
|
(SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING |
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_2ND |
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_3RD |
|
|
SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING));
|
|
} else if (*new_index == swelling_index[swelling_type][JEITA_COOL2]) {
|
|
pval.intval = rechg_vbat_cool;
|
|
sec_bat_set_current_event(
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_2ND,
|
|
(SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING |
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_2ND |
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_3RD |
|
|
SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING));
|
|
} else if (*new_index == swelling_index[swelling_type][JEITA_COOL3]) {
|
|
pval.intval = rechg_vbat_cool;
|
|
sec_bat_set_current_event(
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_3RD,
|
|
(SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING |
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_2ND |
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_3RD |
|
|
SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING));
|
|
} else if (*new_index == swelling_index[swelling_type][JEITA_WARM]) {
|
|
pval.intval = RECHG_WARM;
|
|
sec_bat_set_current_event(
|
|
SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING,
|
|
(SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING |
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_2ND |
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_3RD |
|
|
SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING));
|
|
} else { /* current_index == JEITA_NORMAL */
|
|
if (!is_usb_available(chip)) {
|
|
pval.intval = RECHG_NORMAL;
|
|
} else {
|
|
rc = power_supply_get_property(chip->usb_psy,
|
|
(enum power_supply_property)POWER_SUPPLY_EXT_FIXED_RECHARGE_VBAT, &pval);
|
|
if (rc < 0) {
|
|
pr_err("get recharge Vbat failed, rc=%d\n", rc);
|
|
pval.intval = RECHG_NORMAL;
|
|
}
|
|
}
|
|
sec_bat_set_current_event(0,
|
|
(SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING |
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_2ND |
|
|
SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_3RD |
|
|
SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING));
|
|
}
|
|
|
|
pr_info("%s: index = %d -> %d, rechg voltage = %d\n",
|
|
__func__, current_index, *new_index, pval.intval);
|
|
rc = power_supply_set_property(chip->batt_psy,
|
|
POWER_SUPPLY_PROP_RECHARGE_VBAT, &pval);
|
|
if (rc < 0)
|
|
pr_err("Set recharge Vbat failed, rc=%d\n", rc);
|
|
|
|
if (!battery) {
|
|
pr_info("%s: return index = %d -> %d\n",
|
|
__func__, *new_index, current_index);
|
|
*new_index = current_index;
|
|
}
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
#define TAPERED_STEP_CHG_FCC_REDUCTION_STEP_MA 50000 /* 50 mA */
|
|
static void taper_fcc_step_chg(struct step_chg_info *chip, int index,
|
|
int current_voltage)
|
|
{
|
|
u32 current_fcc, target_fcc;
|
|
|
|
if (index < 0) {
|
|
pr_err("Invalid STEP CHG index\n");
|
|
return;
|
|
}
|
|
|
|
current_fcc = get_effective_result(chip->fcc_votable);
|
|
target_fcc = chip->step_chg_config->fcc_cfg[index].value;
|
|
|
|
if (index == 0) {
|
|
vote(chip->fcc_votable, STEP_CHG_VOTER, true, target_fcc);
|
|
} else if (current_voltage >
|
|
(chip->step_chg_config->fcc_cfg[index - 1].high_threshold +
|
|
chip->step_chg_config->param.hysteresis)) {
|
|
/*
|
|
* Ramp down FCC in pre-configured steps till the current index
|
|
* FCC configuration is reached, whenever the step charging
|
|
* control parameter exceeds the high threshold of previous
|
|
* step charging index configuration.
|
|
*/
|
|
vote(chip->fcc_votable, STEP_CHG_VOTER, true, max(target_fcc,
|
|
current_fcc - TAPERED_STEP_CHG_FCC_REDUCTION_STEP_MA));
|
|
} else if ((current_fcc >
|
|
chip->step_chg_config->fcc_cfg[index - 1].value) &&
|
|
(current_voltage >
|
|
chip->step_chg_config->fcc_cfg[index - 1].low_threshold +
|
|
chip->step_chg_config->param.hysteresis)) {
|
|
/*
|
|
* In case the step charging index switch to the next higher
|
|
* index without FCCs saturation for the previous index, ramp
|
|
* down FCC till previous index FCC configuration is reached.
|
|
*/
|
|
vote(chip->fcc_votable, STEP_CHG_VOTER, true,
|
|
max(chip->step_chg_config->fcc_cfg[index - 1].value,
|
|
current_fcc - TAPERED_STEP_CHG_FCC_REDUCTION_STEP_MA));
|
|
}
|
|
}
|
|
|
|
static int handle_step_chg_config(struct step_chg_info *chip)
|
|
{
|
|
union power_supply_propval pval = {0, };
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
union power_supply_propval val = {0, };
|
|
struct sec_battery_info *battery = get_sec_battery();
|
|
#if defined(CONFIG_ENG_BATTERY_CONCEPT)
|
|
int i = 0;
|
|
#endif
|
|
#if defined(CONFIG_SEC_FACTORY)
|
|
static int count = 0;
|
|
#endif
|
|
#endif
|
|
int rc = 0, fcc_ua = 0, current_index;
|
|
u64 elapsed_us;
|
|
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
#if defined(CONFIG_ENG_BATTERY_CONCEPT)
|
|
if (update_step) {
|
|
for (i = 0; i < 3; i++) {
|
|
chip->step_chg_config->fcc_cfg[i].low_threshold = step_chg_test_low[i];
|
|
chip->step_chg_config->fcc_cfg[i].high_threshold = step_chg_test_high[i];
|
|
chip->step_chg_config->fcc_cfg[i].value = step_chg_test_val[i];
|
|
pr_info("update_step_chg_date, step %d : %d %d %d\n",
|
|
i + 1,
|
|
chip->step_chg_config->fcc_cfg[i].low_threshold,
|
|
chip->step_chg_config->fcc_cfg[i].high_threshold,
|
|
chip->step_chg_config->fcc_cfg[i].value);
|
|
}
|
|
update_step = false;
|
|
}
|
|
|
|
#endif
|
|
if (battery) {
|
|
if(is_slate_mode(battery))
|
|
chip->step_index = 0;
|
|
}
|
|
if (!is_input_present(chip)) {
|
|
chip->step_index = 0;
|
|
val.intval = chip->step_index;
|
|
rc = power_supply_set_property(chip->batt_psy,
|
|
(enum power_supply_property)POWER_SUPPLY_EXT_PROP_DIRECT_CHARGING_STEP, &val);
|
|
if (rc < 0)
|
|
pr_err("Set direct charging step failed, rc=%d\n", rc);
|
|
val.intval = -1;
|
|
rc = power_supply_set_property(chip->batt_psy,
|
|
(enum power_supply_property)POWER_SUPPLY_EXT_PROP_DIRECT_CHARGING_VOL, &val);
|
|
if (rc < 0)
|
|
pr_err("Set direct charging vol failed, rc=%d\n", rc);
|
|
}
|
|
#endif
|
|
elapsed_us = ktime_us_delta(ktime_get(), chip->step_last_update_time);
|
|
if (elapsed_us < STEP_CHG_HYSTERISIS_DELAY_US)
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
return 0;
|
|
#else
|
|
goto reschedule;
|
|
#endif
|
|
|
|
rc = power_supply_get_property(chip->batt_psy,
|
|
POWER_SUPPLY_PROP_STEP_CHARGING_ENABLED, &pval);
|
|
if (rc < 0)
|
|
chip->step_chg_enable = 0;
|
|
else
|
|
chip->step_chg_enable = pval.intval;
|
|
|
|
if (!chip->step_chg_enable || !chip->step_chg_cfg_valid) {
|
|
if (chip->fcc_votable)
|
|
vote(chip->fcc_votable, STEP_CHG_VOTER, false, 0);
|
|
goto update_time;
|
|
}
|
|
|
|
if (chip->step_chg_config->param.use_bms)
|
|
rc = power_supply_get_property(chip->bms_psy,
|
|
chip->step_chg_config->param.psy_prop, &pval);
|
|
else
|
|
rc = power_supply_get_property(chip->batt_psy,
|
|
chip->step_chg_config->param.psy_prop, &pval);
|
|
|
|
if (rc < 0) {
|
|
pr_err("Couldn't read %s property rc=%d\n",
|
|
chip->step_chg_config->param.prop_name, rc);
|
|
return rc;
|
|
}
|
|
|
|
current_index = chip->step_index;
|
|
rc = get_val(chip->step_chg_config->fcc_cfg,
|
|
chip->step_chg_config->param.hysteresis,
|
|
chip->step_index,
|
|
pval.intval,
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
NOT_CHANGE_RECHG, chip,
|
|
#endif
|
|
&chip->step_index,
|
|
&fcc_ua);
|
|
if (rc < 0) {
|
|
/* remove the vote if no step-based fcc is found */
|
|
if (chip->fcc_votable)
|
|
vote(chip->fcc_votable, STEP_CHG_VOTER, false, 0);
|
|
goto update_time;
|
|
}
|
|
|
|
/* Do not drop step-chg index, if input supply is present */
|
|
if (is_input_present(chip)) {
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
if (chip->step_index < current_index){
|
|
chip->step_index = current_index;
|
|
fcc_ua = chip->step_chg_config->fcc_cfg[current_index].value;
|
|
}
|
|
#else
|
|
if (chip->step_index < current_index)
|
|
chip->step_index = current_index;
|
|
#endif
|
|
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC) && defined(CONFIG_SEC_FACTORY)
|
|
count = 0;
|
|
#endif
|
|
} else {
|
|
chip->step_index = 0;
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
fcc_ua = chip->step_chg_config->fcc_cfg[chip->step_index].value;
|
|
#endif
|
|
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC) && defined(CONFIG_SEC_FACTORY)
|
|
pval.intval = -18;
|
|
count++;
|
|
pr_err("Error code: %d, count: %d\n", pval.intval, count);
|
|
if (count < 10) {
|
|
return 0;
|
|
} else {
|
|
pr_err("TEST FAIL: %d, count: %d\n", pval.intval, count);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
pr_info("voltage: %d, step_index: %d\n", pval.intval, chip->step_index);
|
|
val.intval = chip->step_index;
|
|
rc = power_supply_set_property(chip->batt_psy,
|
|
(enum power_supply_property)POWER_SUPPLY_EXT_PROP_DIRECT_CHARGING_STEP, &val);
|
|
if (rc < 0)
|
|
pr_err("Set direct charging step failed, rc=%d\n", rc);
|
|
rc = power_supply_set_property(chip->batt_psy,
|
|
(enum power_supply_property)POWER_SUPPLY_EXT_PROP_DIRECT_CHARGING_VOL, &pval);
|
|
if (rc < 0)
|
|
pr_err("Set direct charging vol failed, rc=%d\n", rc);
|
|
#endif
|
|
if (!chip->fcc_votable)
|
|
chip->fcc_votable = find_votable("FCC");
|
|
if (!chip->fcc_votable)
|
|
return -EINVAL;
|
|
|
|
if (chip->taper_fcc)
|
|
taper_fcc_step_chg(chip, chip->step_index, pval.intval);
|
|
else
|
|
vote(chip->fcc_votable, STEP_CHG_VOTER, true, fcc_ua);
|
|
|
|
pr_debug("%s = %d Step-FCC = %duA taper-fcc: %d\n",
|
|
chip->step_chg_config->param.prop_name, pval.intval,
|
|
get_client_vote(chip->fcc_votable, STEP_CHG_VOTER),
|
|
chip->taper_fcc);
|
|
|
|
update_time:
|
|
chip->step_last_update_time = ktime_get();
|
|
return 0;
|
|
#if !defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
reschedule:
|
|
/* reschedule 1000uS after the remaining time */
|
|
return (STEP_CHG_HYSTERISIS_DELAY_US - elapsed_us + 1000);
|
|
#endif
|
|
}
|
|
|
|
#define JEITA_SUSPEND_HYST_UV 50000
|
|
static int handle_jeita(struct step_chg_info *chip)
|
|
{
|
|
union power_supply_propval pval = {0, };
|
|
int rc = 0, fcc_ua = 0, fv_uv = 0;
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
int pre_fcc_index = 0, pre_fv_index = 0;
|
|
#endif
|
|
u64 elapsed_us;
|
|
|
|
rc = power_supply_get_property(chip->batt_psy,
|
|
POWER_SUPPLY_PROP_SW_JEITA_ENABLED, &pval);
|
|
if (rc < 0)
|
|
chip->sw_jeita_enable = 0;
|
|
else
|
|
chip->sw_jeita_enable = pval.intval;
|
|
|
|
if (!chip->sw_jeita_enable || !chip->sw_jeita_cfg_valid) {
|
|
if (chip->fcc_votable)
|
|
vote(chip->fcc_votable, JEITA_VOTER, false, 0);
|
|
if (chip->fv_votable)
|
|
vote(chip->fv_votable, JEITA_VOTER, false, 0);
|
|
if (chip->usb_icl_votable)
|
|
vote(chip->usb_icl_votable, JEITA_VOTER, false, 0);
|
|
return 0;
|
|
}
|
|
|
|
elapsed_us = ktime_us_delta(ktime_get(), chip->jeita_last_update_time);
|
|
if (elapsed_us < STEP_CHG_HYSTERISIS_DELAY_US)
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
return 0;
|
|
#else
|
|
goto reschedule;
|
|
#endif
|
|
|
|
if (chip->jeita_fcc_config->param.use_bms)
|
|
rc = power_supply_get_property(chip->bms_psy,
|
|
chip->jeita_fcc_config->param.psy_prop, &pval);
|
|
else
|
|
rc = power_supply_get_property(chip->batt_psy,
|
|
chip->jeita_fcc_config->param.psy_prop, &pval);
|
|
|
|
if (rc < 0) {
|
|
pr_err("Couldn't read %s property rc=%d\n",
|
|
chip->jeita_fcc_config->param.prop_name, rc);
|
|
return rc;
|
|
}
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
pre_fcc_index = chip->jeita_fcc_index;
|
|
#endif
|
|
rc = get_val(chip->jeita_fcc_config->fcc_cfg,
|
|
chip->jeita_fcc_config->param.hysteresis,
|
|
chip->jeita_fcc_index,
|
|
pval.intval,
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
CHANGE_RECHG, chip,
|
|
#endif
|
|
&chip->jeita_fcc_index,
|
|
&fcc_ua);
|
|
if (rc < 0)
|
|
fcc_ua = 0;
|
|
|
|
if (!chip->fcc_votable)
|
|
chip->fcc_votable = find_votable("FCC");
|
|
if (!chip->fcc_votable)
|
|
/* changing FCC is a must */
|
|
return -EINVAL;
|
|
|
|
vote(chip->fcc_votable, JEITA_VOTER, fcc_ua ? true : false, fcc_ua);
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
pre_fv_index = chip->jeita_fv_index;
|
|
#endif
|
|
rc = get_val(chip->jeita_fv_config->fv_cfg,
|
|
chip->jeita_fv_config->param.hysteresis,
|
|
chip->jeita_fv_index,
|
|
pval.intval,
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
NOT_CHANGE_RECHG, chip,
|
|
#endif
|
|
&chip->jeita_fv_index,
|
|
&fv_uv);
|
|
if (rc < 0)
|
|
fv_uv = 0;
|
|
|
|
chip->fv_votable = find_votable("FV");
|
|
if (!chip->fv_votable)
|
|
goto update_time;
|
|
|
|
if (!chip->usb_icl_votable)
|
|
chip->usb_icl_votable = find_votable("USB_ICL");
|
|
|
|
if (!chip->usb_icl_votable)
|
|
goto set_jeita_fv;
|
|
|
|
/*
|
|
* If JEITA float voltage is same as max-vfloat of battery then
|
|
* skip any further VBAT specific checks.
|
|
*/
|
|
rc = power_supply_get_property(chip->batt_psy,
|
|
POWER_SUPPLY_PROP_VOLTAGE_MAX, &pval);
|
|
if (rc || (pval.intval == fv_uv)) {
|
|
vote(chip->usb_icl_votable, JEITA_VOTER, false, 0);
|
|
goto set_jeita_fv;
|
|
}
|
|
|
|
/*
|
|
* Suspend USB input path if battery voltage is above
|
|
* JEITA VFLOAT threshold.
|
|
*/
|
|
if (chip->jeita_arb_en && fv_uv > 0) {
|
|
rc = power_supply_get_property(chip->batt_psy,
|
|
POWER_SUPPLY_PROP_VOLTAGE_NOW, &pval);
|
|
if (!rc && (pval.intval > fv_uv))
|
|
vote(chip->usb_icl_votable, JEITA_VOTER, true, 0);
|
|
else if (pval.intval < (fv_uv - JEITA_SUSPEND_HYST_UV))
|
|
vote(chip->usb_icl_votable, JEITA_VOTER, false, 0);
|
|
}
|
|
|
|
set_jeita_fv:
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
/*should notice smb5-lib enter swelling mode before change FV, to avoid vatt OVP*/
|
|
if(pre_fcc_index != chip->jeita_fcc_index || pre_fv_index != chip->jeita_fv_index){
|
|
pr_info("%s: fcc_index = %d -> %d fv_index = %d -> %d \n",
|
|
__func__, pre_fcc_index, chip->jeita_fcc_index, pre_fv_index, chip->jeita_fv_index);
|
|
pval.intval = chip->jeita_fcc_index;
|
|
rc = power_supply_set_property(chip->batt_psy,
|
|
(enum power_supply_property)POWER_SUPPLY_EXT_PROP_CHG_SWELLING_STATE, &pval);
|
|
}
|
|
#endif
|
|
vote(chip->fv_votable, JEITA_VOTER, fv_uv ? true : false, fv_uv);
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
/*start charge for warm swelling recovery*/
|
|
if(pre_fv_index == 1 && chip->jeita_fv_index == 0) {
|
|
pval.intval = 0;
|
|
rc = power_supply_set_property(chip->batt_psy,
|
|
POWER_SUPPLY_PROP_FORCE_RECHARGE, &pval);
|
|
}
|
|
#endif
|
|
update_time:
|
|
chip->jeita_last_update_time = ktime_get();
|
|
#if !defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
if (!chip->main_psy)
|
|
chip->main_psy = power_supply_get_by_name("main");
|
|
if (chip->main_psy)
|
|
power_supply_changed(chip->main_psy);
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
#if !defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
reschedule:
|
|
/* reschedule 1000uS after the remaining time */
|
|
return (STEP_CHG_HYSTERISIS_DELAY_US - elapsed_us + 1000);
|
|
#endif
|
|
|
|
}
|
|
|
|
static int handle_battery_insertion(struct step_chg_info *chip)
|
|
{
|
|
int rc;
|
|
union power_supply_propval pval = {0, };
|
|
|
|
rc = power_supply_get_property(chip->batt_psy,
|
|
POWER_SUPPLY_PROP_PRESENT, &pval);
|
|
if (rc < 0) {
|
|
pr_err("Get battery present status failed, rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
if (chip->batt_missing != (!pval.intval)) {
|
|
chip->batt_missing = !pval.intval;
|
|
pr_debug("battery %s detected\n",
|
|
chip->batt_missing ? "removal" : "insertion");
|
|
if (chip->batt_missing) {
|
|
chip->step_chg_cfg_valid = false;
|
|
chip->sw_jeita_cfg_valid = false;
|
|
chip->get_config_retry_count = 0;
|
|
} else {
|
|
/*
|
|
* Get config for the new inserted battery, delay
|
|
* to make sure BMS has read out the batt_id.
|
|
*/
|
|
schedule_delayed_work(&chip->get_config_work,
|
|
msecs_to_jiffies(WAIT_BATT_ID_READY_MS));
|
|
}
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static void status_change_work(struct work_struct *work)
|
|
{
|
|
struct step_chg_info *chip = container_of(work,
|
|
struct step_chg_info, status_change_work.work);
|
|
int rc = 0;
|
|
#if !defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
int reschedule_us;
|
|
int reschedule_jeita_work_us = 0;
|
|
int reschedule_step_work_us = 0;
|
|
#endif
|
|
union power_supply_propval prop = {0, };
|
|
|
|
if (!is_batt_available(chip) || !is_bms_available(chip))
|
|
goto exit_work;
|
|
|
|
handle_battery_insertion(chip);
|
|
|
|
if (!is_input_present(chip)) {
|
|
chip->jeita_fcc_index = -EINVAL;
|
|
} else {
|
|
/* skip elapsed_us debounce for handling battery temperature */
|
|
rc = handle_jeita(chip);
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
if (rc < 0)
|
|
#else
|
|
if (rc > 0)
|
|
reschedule_jeita_work_us = rc;
|
|
else if (rc < 0)
|
|
#endif
|
|
pr_err("Couldn't handle sw jeita rc = %d\n", rc);
|
|
}
|
|
|
|
rc = handle_step_chg_config(chip);
|
|
#if !defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
if (rc > 0)
|
|
reschedule_step_work_us = rc;
|
|
#endif
|
|
if (rc < 0)
|
|
pr_err("Couldn't handle step rc = %d\n", rc);
|
|
|
|
/* Remove stale votes on USB removal */
|
|
if (is_usb_available(chip)) {
|
|
prop.intval = 0;
|
|
power_supply_get_property(chip->usb_psy,
|
|
POWER_SUPPLY_PROP_PRESENT, &prop);
|
|
if (!prop.intval) {
|
|
if (chip->usb_icl_votable)
|
|
vote(chip->usb_icl_votable, JEITA_VOTER,
|
|
false, 0);
|
|
}
|
|
}
|
|
#if !defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
reschedule_us = min(reschedule_jeita_work_us, reschedule_step_work_us);
|
|
if (reschedule_us == 0)
|
|
goto exit_work;
|
|
else
|
|
schedule_delayed_work(&chip->status_change_work,
|
|
usecs_to_jiffies(reschedule_us));
|
|
return;
|
|
#endif
|
|
|
|
exit_work:
|
|
__pm_relax(chip->step_chg_ws);
|
|
}
|
|
|
|
static int step_chg_notifier_call(struct notifier_block *nb,
|
|
unsigned long ev, void *v)
|
|
{
|
|
struct power_supply *psy = v;
|
|
struct step_chg_info *chip = container_of(nb, struct step_chg_info, nb);
|
|
|
|
if (ev != PSY_EVENT_PROP_CHANGED)
|
|
return NOTIFY_OK;
|
|
|
|
if ((strcmp(psy->desc->name, "battery") == 0)
|
|
|| (strcmp(psy->desc->name, "usb") == 0)) {
|
|
__pm_stay_awake(chip->step_chg_ws);
|
|
schedule_delayed_work(&chip->status_change_work, 0);
|
|
}
|
|
|
|
if ((strcmp(psy->desc->name, "bms") == 0)) {
|
|
if (chip->bms_psy == NULL)
|
|
chip->bms_psy = psy;
|
|
if (!chip->config_is_read)
|
|
schedule_delayed_work(&chip->get_config_work, 0);
|
|
}
|
|
|
|
return NOTIFY_OK;
|
|
}
|
|
|
|
static int step_chg_register_notifier(struct step_chg_info *chip)
|
|
{
|
|
int rc;
|
|
|
|
chip->nb.notifier_call = step_chg_notifier_call;
|
|
rc = power_supply_reg_notifier(&chip->nb);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't register psy notifier rc = %d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int qcom_step_chg_init(struct device *dev,
|
|
bool step_chg_enable, bool sw_jeita_enable, bool jeita_arb_en)
|
|
{
|
|
int rc;
|
|
struct step_chg_info *chip;
|
|
|
|
if (the_chip) {
|
|
pr_err("Already initialized\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
chip = devm_kzalloc(dev, sizeof(*chip), GFP_KERNEL);
|
|
if (!chip)
|
|
return -ENOMEM;
|
|
|
|
chip->step_chg_ws = wakeup_source_register(dev, "qcom-step-chg");
|
|
if (!chip->step_chg_ws)
|
|
return -EINVAL;
|
|
|
|
chip->dev = dev;
|
|
chip->step_chg_enable = step_chg_enable;
|
|
chip->sw_jeita_enable = sw_jeita_enable;
|
|
chip->jeita_arb_en = jeita_arb_en;
|
|
chip->step_index = -EINVAL;
|
|
chip->jeita_fcc_index = -EINVAL;
|
|
chip->jeita_fv_index = -EINVAL;
|
|
|
|
chip->step_chg_config = devm_kzalloc(dev,
|
|
sizeof(struct step_chg_cfg), GFP_KERNEL);
|
|
if (!chip->step_chg_config)
|
|
return -ENOMEM;
|
|
|
|
chip->step_chg_config->param.psy_prop = POWER_SUPPLY_PROP_VOLTAGE_NOW;
|
|
chip->step_chg_config->param.prop_name = "VBATT";
|
|
chip->step_chg_config->param.hysteresis = 100000;
|
|
|
|
#if defined(CONFIG_BATTERY_SAMSUNG_USING_QC)
|
|
chip->step_chg_config->param.hysteresis = 0;
|
|
#if defined(CONFIG_ENG_BATTERY_CONCEPT)
|
|
update_step = false;
|
|
#endif
|
|
#endif
|
|
|
|
chip->jeita_fcc_config = devm_kzalloc(dev,
|
|
sizeof(struct jeita_fcc_cfg), GFP_KERNEL);
|
|
chip->jeita_fv_config = devm_kzalloc(dev,
|
|
sizeof(struct jeita_fv_cfg), GFP_KERNEL);
|
|
if (!chip->jeita_fcc_config || !chip->jeita_fv_config)
|
|
return -ENOMEM;
|
|
|
|
chip->jeita_fcc_config->param.psy_prop = POWER_SUPPLY_PROP_TEMP;
|
|
chip->jeita_fcc_config->param.prop_name = "BATT_TEMP";
|
|
chip->jeita_fcc_config->param.hysteresis = 10;
|
|
chip->jeita_fv_config->param.psy_prop = POWER_SUPPLY_PROP_TEMP;
|
|
chip->jeita_fv_config->param.prop_name = "BATT_TEMP";
|
|
chip->jeita_fv_config->param.hysteresis = 10;
|
|
|
|
INIT_DELAYED_WORK(&chip->status_change_work, status_change_work);
|
|
INIT_DELAYED_WORK(&chip->get_config_work, get_config_work);
|
|
|
|
rc = step_chg_register_notifier(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't register psy notifier rc = %d\n", rc);
|
|
goto release_wakeup_source;
|
|
}
|
|
|
|
schedule_delayed_work(&chip->get_config_work,
|
|
msecs_to_jiffies(GET_CONFIG_DELAY_MS));
|
|
|
|
the_chip = chip;
|
|
|
|
return 0;
|
|
|
|
release_wakeup_source:
|
|
wakeup_source_unregister(chip->step_chg_ws);
|
|
return rc;
|
|
}
|
|
|
|
void qcom_step_chg_deinit(void)
|
|
{
|
|
struct step_chg_info *chip = the_chip;
|
|
|
|
if (!chip)
|
|
return;
|
|
|
|
cancel_delayed_work_sync(&chip->status_change_work);
|
|
cancel_delayed_work_sync(&chip->get_config_work);
|
|
power_supply_unreg_notifier(&chip->nb);
|
|
wakeup_source_unregister(chip->step_chg_ws);
|
|
the_chip = NULL;
|
|
}
|
|
|