blob: d6481803110a976501fe7396353519900fd7528d [file] [log] [blame]
/*
* sec_step_charging.c
* Samsung Mobile Battery Driver
*
* Copyright (C) 2018 Samsung Electronics
*
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/
#include "include/sec_battery.h"
#define STEP_CHARGING_CONDITION_VOLTAGE 0x01
#define STEP_CHARGING_CONDITION_SOC 0x02
#define STEP_CHARGING_CONDITION_CHARGE_POWER 0x04
#define STEP_CHARGING_CONDITION_ONLINE 0x08
#define STEP_CHARGING_CONDITION_CURRENT_NOW 0x10
#define STEP_CHARGING_CONDITION_FLOAT_VOLTAGE 0x20
#define STEP_CHARGING_CONDITION_INPUT_CURRENT 0x40
#define STEP_CHARGING_CONDITION_SOC_INIT_ONLY 0x80 /* use this to consider SOC to decide starting step only */
#define STEP_CHARGING_CONDITION_DC_INIT (STEP_CHARGING_CONDITION_VOLTAGE | STEP_CHARGING_CONDITION_SOC | STEP_CHARGING_CONDITION_SOC_INIT_ONLY)
#define DIRECT_CHARGING_FLOAT_VOLTAGE_MARGIN 20
#define DIRECT_CHARGING_FORCE_SOC_MARGIN 10
void sec_bat_reset_step_charging(struct sec_battery_info *battery)
{
pr_info("%s\n", __func__);
battery->step_charging_status = -1;
#if defined(CONFIG_DIRECT_CHARGING)
battery->dc_float_voltage_set = false;
#endif
}
void sec_bat_exit_step_charging(struct sec_battery_info *battery)
{
battery->pdata->charging_current[battery->cable_type].fast_charging_current =
battery->pdata->step_charging_current[battery->step_charging_step-1];
if ((battery->step_charging_type & STEP_CHARGING_CONDITION_FLOAT_VOLTAGE) &&
(battery->swelling_mode == SWELLING_MODE_NONE)) {
union power_supply_propval val;
pr_info("%s : float voltage = %d \n", __func__,
battery->pdata->step_charging_float_voltage[battery->step_charging_step-1]);
val.intval = battery->pdata->step_charging_float_voltage[battery->step_charging_step-1];
psy_do_property(battery->pdata->charger_name, set,
POWER_SUPPLY_PROP_VOLTAGE_MAX, val);
}
sec_bat_reset_step_charging(battery);
}
/*
* true: step is changed
* false: not changed
*/
bool sec_bat_check_step_charging(struct sec_battery_info *battery)
{
int i = 0, value = 0, step_condition = 0;
static int curr_cnt = 0;
pr_info("%s\n", __func__);
#if defined(CONFIG_SEC_FACTORY)
return false;
#endif
#if defined(CONFIG_ENG_BATTERY_CONCEPT)
if(battery->test_charge_current)
return false;
if(battery->test_step_condition <= 100)
battery->pdata->step_charging_condition[0] = battery->test_step_condition;
#endif
if (!battery->step_charging_type)
return false;
if (battery->step_charging_type & STEP_CHARGING_CONDITION_ONLINE) {
#if defined(CONFIG_DIRECT_CHARGING)
if (is_pd_apdo_wire_type(battery->cable_type) &&
!((battery->current_event & SEC_BAT_CURRENT_EVENT_DC_ERR) &&
(battery->ta_alert_mode == OCP_NONE)))
return false;
if ((is_pd_apdo_wire_type(battery->cable_type) || is_pd_apdo_wire_type(battery->wire_status)) &&
(battery->pdic_info.sink_status.rp_currentlvl == RP_CURRENT_LEVEL3)) {
pr_info("%s: This cable type should be checked in dc step check\n", __func__);
return false;
}
#endif
if (!is_hv_wire_type(battery->cable_type) && !(battery->cable_type == SEC_BATTERY_CABLE_PDIC) &&
!(battery->pdic_info.sink_status.rp_currentlvl == RP_CURRENT_LEVEL3))
return false;
}
if (battery->step_charging_type & STEP_CHARGING_CONDITION_CHARGE_POWER) {
if (battery->max_charge_power < battery->step_charging_charge_power) {
/* In case of max_charge_power falling by AICL during step-charging ongoing */
sec_bat_exit_step_charging(battery);
return false;
}
}
if (battery->step_charging_status < 0)
i = 0;
else
i = battery->step_charging_status;
step_condition = battery->pdata->step_charging_condition[i];
if (battery->step_charging_type & STEP_CHARGING_CONDITION_VOLTAGE) {
value = battery->voltage_avg;
} else if (battery->step_charging_type & STEP_CHARGING_CONDITION_SOC) {
value = battery->capacity;
if (battery->siop_level < 100 || battery->lcd_status) {
step_condition = battery->pdata->step_charging_condition[i] + 15;
curr_cnt = 0;
}
} else {
return false;
}
while(i < battery->step_charging_step - 1) {
if (value < step_condition) {
break;
}
i++;
if ((battery->step_charging_type & STEP_CHARGING_CONDITION_SOC) &&
(battery->siop_level < 100 || battery->lcd_status))
step_condition = battery->pdata->step_charging_condition[i] + 15;
else
step_condition = battery->pdata->step_charging_condition[i];
if(battery->step_charging_status != -1)
break;
}
if (i != battery->step_charging_status) {
/* this is only for no consuming current */
if ((battery->step_charging_type & STEP_CHARGING_CONDITION_CURRENT_NOW) &&
(battery->siop_level >= 100 && !battery->lcd_status) &&
battery->step_charging_status >= 0) {
int condition_curr;
condition_curr = max(battery->current_avg, battery->current_now);
if(condition_curr < battery->pdata->step_charging_condition_curr[battery->step_charging_status]) {
curr_cnt++;
pr_info("%s : cnt = %d, current avg(%d)mA < current condition(%d)mA\n", __func__,
curr_cnt, condition_curr, battery->pdata->step_charging_condition_curr[battery->step_charging_status]);
if(curr_cnt < 3)
return false;
} else {
pr_info("%s : clear count, current avg(%d)mA >= current condition(%d)mA or"
" < 0mA this log is for debug\n", __func__,
condition_curr, battery->pdata->step_charging_condition_curr[battery->step_charging_status]);
curr_cnt = 0;
return false;
}
}
pr_info("%s : prev=%d, new=%d, value=%d, current=%d, curr_cnt=%d\n", __func__,
battery->step_charging_status, i, value, battery->pdata->step_charging_current[i], curr_cnt);
battery->pdata->charging_current[battery->cable_type].fast_charging_current = battery->pdata->step_charging_current[i];
battery->step_charging_status = i;
if ((battery->step_charging_type & STEP_CHARGING_CONDITION_FLOAT_VOLTAGE) &&
(battery->swelling_mode == SWELLING_MODE_NONE)) {
union power_supply_propval val;
pr_info("%s : float voltage = %d \n", __func__, battery->pdata->step_charging_float_voltage[i]);
val.intval = battery->pdata->step_charging_float_voltage[i];
psy_do_property(battery->pdata->charger_name, set,
POWER_SUPPLY_PROP_VOLTAGE_MAX, val);
}
return true;
}
return false;
}
#if defined(CONFIG_DIRECT_CHARGING)
bool sec_bat_check_dc_step_charging(struct sec_battery_info *battery)
{
int i, value;
int step = -1, step_vol = -1, step_input = -1, step_soc = -1, soc_condition = 0;
bool force_change_step = false;
union power_supply_propval val;
if (!battery->dc_step_chg_type)
return false;
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_CHARGE_POWER)
if (battery->charge_power < battery->dc_step_chg_charge_power)
return false;
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_ONLINE) {
if (!is_pd_apdo_wire_type(battery->cable_type))
return false;
}
if (battery->current_event & SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING ||
battery->current_event & SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING ||
battery->current_event & SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_2ND ||
battery->current_event & SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_3RD ||
battery->current_event & SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_4TH ||
battery->current_event & SEC_BAT_CURRENT_EVENT_HV_DISABLE ||
((battery->current_event & SEC_BAT_CURRENT_EVENT_DC_ERR) &&
(battery->ta_alert_mode == OCP_NONE)) ||
battery->current_event & SEC_BAT_CURRENT_EVENT_SIOP_LIMIT) {
if (battery->step_charging_status >= 0)
sec_bat_reset_step_charging(battery);
return false;
}
if (battery->step_charging_status < 0)
i = 0;
else
i = battery->step_charging_status;
if (!(battery->dc_step_chg_type & STEP_CHARGING_CONDITION_DC_INIT)) {
pr_info("%s : cond_vol and cond_soc are both empty\n", __func__);
return false;
}
/* this is only for step enter condition and do not use STEP_CHARGING_CONDITION_SOC at the same time */
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_SOC_INIT_ONLY) {
if (battery->step_charging_status < 0) {
step_soc = i;
value = battery->capacity;
while(step_soc < battery->dc_step_chg_step - 1) {
soc_condition = battery->pdata->dc_step_chg_cond_soc[battery->pdata->age_step][step_soc];
if (value < soc_condition)
break;
step_soc++;
}
if ((step_soc < step) || (step < 0))
step = step_soc;
pr_info("%s : set initial step(%d) by soc\n", __func__, step_soc);
goto check_dc_step_change;
} else
step_soc = battery->dc_step_chg_step - 1;
}
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_SOC) {
step_soc = i;
value = battery->capacity;
while(step_soc < battery->dc_step_chg_step - 1) {
soc_condition = battery->pdata->dc_step_chg_cond_soc[battery->pdata->age_step][step_soc];
if (battery->step_charging_status >= 0 &&
(battery->siop_level < 100 || battery->lcd_status)) {
soc_condition += DIRECT_CHARGING_FORCE_SOC_MARGIN;
force_change_step = true;
}
if (value < soc_condition)
break;
step_soc++;
if (battery->step_charging_status >= 0)
break;
}
if ((step_soc < step) || (step < 0))
step = step_soc;
if (battery->step_charging_status < 0) {
pr_info("%s : set initial step(%d) by soc\n", __func__, step_soc);
goto check_dc_step_change;
}
if (force_change_step) {
pr_info("%s : force check step(%d) by soc\n", __func__, step_soc);
step_vol = step_input = step_soc;
battery->dc_step_chg_iin_cnt = battery->pdata->dc_step_chg_iin_check_cnt;
goto check_dc_step_change;
}
} else
step_soc = battery->dc_step_chg_step - 1;
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_VOLTAGE) {
step_vol = i;
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_FLOAT_VOLTAGE)
value = battery->voltage_now + DIRECT_CHARGING_FLOAT_VOLTAGE_MARGIN;
else
value = battery->voltage_avg;
while(step_vol < battery->dc_step_chg_step - 1) {
if (value < battery->pdata->dc_step_chg_cond_vol[step_vol])
break;
step_vol++;
if (battery->step_charging_status >= 0)
break;
}
if ((step_vol < step) || (step < 0))
step = step_vol;
if (battery->step_charging_status < 0) {
pr_info("%s : set initial step(%d) by vol\n", __func__, step_vol);
goto check_dc_step_change;
}
} else
step_vol = battery->dc_step_chg_step - 1;
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_INPUT_CURRENT) {
step_input = i;
psy_do_property(battery->pdata->charger_name, get,
POWER_SUPPLY_EXT_PROP_DIRECT_CHARGER_MODE, val);
if (val.intval != SEC_DIRECT_CHG_MODE_DIRECT_ON) {
pr_info("%s : dc no charging status = %d \n", __func__, val.intval);
battery->dc_step_chg_iin_cnt = 0;
return false;
} else if (battery->siop_level >= 100 && !battery->lcd_status) {
val.intval = SEC_BATTERY_IIN_MA;
psy_do_property(battery->pdata->charger_name, get,
POWER_SUPPLY_EXT_PROP_MEASURE_INPUT, val);
value = val.intval;
while(step_input < battery->dc_step_chg_step - 1) {
if (value > battery->pdata->dc_step_chg_cond_iin[step_input])
break;
step_input++;
if (battery->step_charging_status >= 0) {
battery->dc_step_chg_iin_cnt++;
break;
} else {
battery->dc_step_chg_iin_cnt = 0;
}
}
} else {
/* Do not check input current when lcd is on or siop is not 100 since there might be quite big system current */
step_input = battery->dc_step_chg_step - 1;
}
if ((step_input < step) || (step < 0))
step = step_input;
} else
step_input = battery->dc_step_chg_step - 1;
check_dc_step_change:
pr_info("%s : curr_step(%d), step_vol(%d), step_soc(%d), step_input(%d), curr_cnt(%d/%d)\n",
__func__, step, step_vol, step_soc, step_input,
battery->dc_step_chg_iin_cnt, battery->pdata->dc_step_chg_iin_check_cnt);
if (battery->step_charging_status < 0 ||
(step != battery->step_charging_status && step == min(min(step_vol, step_soc), step_input))) {
if ((battery->dc_step_chg_type & STEP_CHARGING_CONDITION_INPUT_CURRENT) &&
(battery->step_charging_status >= 0)) {
if (battery->dc_step_chg_iin_cnt < battery->pdata->dc_step_chg_iin_check_cnt &&
(battery->siop_level >= 100 && !battery->lcd_status)) {
pr_info("%s : keep step(%d), curr_cnt(%d/%d)\n",
__func__, battery->step_charging_status,
battery->dc_step_chg_iin_cnt, battery->pdata->dc_step_chg_iin_check_cnt);
return false;
}
}
pr_info("%s : cable(%d), soc(%d), step changed(%d->%d), current(%dmA)\n",
__func__, battery->cable_type, battery->capacity,
battery->step_charging_status, step, battery->pdata->dc_step_chg_val_iout[battery->pdata->age_step][step]);
/* set charging current */
battery->pdata->charging_current[battery->cable_type].fast_charging_current = battery->pdata->dc_step_chg_val_iout[battery->pdata->age_step][step];
if ((battery->dc_step_chg_type & STEP_CHARGING_CONDITION_FLOAT_VOLTAGE) &&
(battery->swelling_mode == SWELLING_MODE_NONE)) {
if (battery->step_charging_status < 0) {
pr_info("%s : step float voltage = %d \n", __func__, battery->pdata->dc_step_chg_val_vfloat[battery->pdata->age_step][step]);
val.intval = battery->pdata->dc_step_chg_val_vfloat[battery->pdata->age_step][step];
psy_do_property(battery->pdata->charger_name, set,
POWER_SUPPLY_EXT_PROP_DIRECT_VOLTAGE_MAX, val);
}
battery->dc_float_voltage_set = true;
}
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_INPUT_CURRENT) {
if (battery->step_charging_status < 0 && !(battery->current_event & SEC_BAT_CURRENT_EVENT_SWELLING_MODE)) {
pr_info("%s : step input current = %d \n", __func__, battery->pdata->dc_step_chg_val_iout[battery->pdata->age_step][step] / 2);
val.intval = battery->pdata->dc_step_chg_val_iout[battery->pdata->age_step][step] / 2;
psy_do_property(battery->pdata->charger_name, set,
POWER_SUPPLY_EXT_PROP_DIRECT_CURRENT_MAX, val);
}
}
battery->step_charging_status = step;
battery->dc_step_chg_iin_cnt = 0;
return true;
} else {
battery->dc_step_chg_iin_cnt = 0;
}
return false;
}
int sec_dc_step_charging_dt(struct sec_battery_info *battery, struct device *dev)
{
struct device_node *np = dev->of_node;
int ret = 0, len = 0;
sec_battery_platform_data_t *pdata = battery->pdata;
unsigned int i = 0, j = 0, dc_step_chg_type = 0;
const u32 *p;
char str[128] = {0,};
u32 *soc_cond_temp, *dc_step_chg_val_vfloat_temp, *dc_step_chg_val_iout_temp;
ret = of_property_read_u32(np, "battery,dc_step_chg_type",
&battery->dc_step_chg_type);
pr_err("%s: dc_step_chg_type 0x%x\n", __func__, battery->dc_step_chg_type);
if (ret) {
pr_err("%s: dc_step_chg_type is Empty\n", __func__);
battery->dc_step_chg_type = 0;
return -1;
}
ret = of_property_read_u32(np, "battery,dc_step_chg_charge_power",
&battery->dc_step_chg_charge_power);
if (ret) {
pr_err("%s: dc_step_chg_charge_power is Empty\n", __func__);
battery->dc_step_chg_charge_power = 20000;
}
ret = of_property_read_u32(np, "battery,dc_step_chg_step",
&battery->dc_step_chg_step);
if (ret) {
pr_err("%s: dc_step_chg_step is Empty\n", __func__);
battery->dc_step_chg_step = 0;
goto dc_step_charging_dt_error;
} else {
pr_err("%s: dc_step_chg_step is %d\n",
__func__, battery->dc_step_chg_step);
}
dc_step_chg_type = battery->dc_step_chg_type;
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_VOLTAGE) {
p = of_get_property(np, "battery,dc_step_chg_cond_vol", &len);
if (!p) {
pr_err("%s: dc_step_chg_cond_vol is Empty, type(0x%X->0x%X)\n",
__func__, battery->dc_step_chg_type,
battery->dc_step_chg_type & ~STEP_CHARGING_CONDITION_VOLTAGE);
battery->dc_step_chg_type &= ~STEP_CHARGING_CONDITION_VOLTAGE;
} else {
len = len / sizeof(u32);
if (len != battery->dc_step_chg_step) {
/* [dchg] TODO: do some error handling */
pr_err("%s: len of dc_step_chg_cond_vol is not matched, len(%d/%d)\n",
__func__, len, battery->dc_step_chg_step);
}
pdata->dc_step_chg_cond_vol = kzalloc(sizeof(u32) * len, GFP_KERNEL);
ret = of_property_read_u32_array(np, "battery,dc_step_chg_cond_vol",
pdata->dc_step_chg_cond_vol, len);
if (ret) {
pr_info("%s : dc_step_chg_cond_vol read fail\n", __func__);
battery->dc_step_chg_type &= ~STEP_CHARGING_CONDITION_VOLTAGE;
}
}
}
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_SOC ||
battery->dc_step_chg_type & STEP_CHARGING_CONDITION_SOC_INIT_ONLY) {
p = of_get_property(np, "battery,dc_step_chg_cond_soc", &len);
if (!p) {
pr_err("%s: dc_step_chg_cond_soc is Empty, type(0x%X->0x%x)\n",
__func__, battery->dc_step_chg_type,
battery->dc_step_chg_type & ~(STEP_CHARGING_CONDITION_SOC | STEP_CHARGING_CONDITION_SOC_INIT_ONLY));
battery->dc_step_chg_type &= ~(STEP_CHARGING_CONDITION_SOC | STEP_CHARGING_CONDITION_SOC_INIT_ONLY);
} else {
len = len / sizeof(u32);
if (len/battery->pdata->num_age_step != battery->dc_step_chg_step) {
/* [dchg] TODO: do some error handling */
pr_err("%s: len of dc_step_charging_cond_soc is not matched, len(%d/%d)\n",
__func__, len, battery->dc_step_chg_step);
}
pr_info("%s step(%d) * age_step(%d) = %d \n", __func__, battery->dc_step_chg_step, battery->pdata->num_age_step, len);
/* get dt to buff */
soc_cond_temp = kzalloc(sizeof(u32) * battery->dc_step_chg_step * battery->pdata->num_age_step , GFP_KERNEL);
ret = of_property_read_u32_array(np, "battery,dc_step_chg_cond_soc",
soc_cond_temp, battery->dc_step_chg_step * battery->pdata->num_age_step);
/* copy buff to 2d arr */
pdata->dc_step_chg_cond_soc = kzalloc(sizeof(u32 *) * battery->pdata->num_age_step , GFP_KERNEL);
for( i = 0; i < battery->pdata->num_age_step; i++) {
pdata->dc_step_chg_cond_soc[i] = kzalloc(sizeof(u32) * battery->dc_step_chg_step , GFP_KERNEL);
for( j = 0; j < battery->dc_step_chg_step; j++) {
pdata->dc_step_chg_cond_soc[i][j] = soc_cond_temp[i*battery->dc_step_chg_step + j];
}
}
/* if there are only 1 dimentional array of value, get the same value */
if(battery->dc_step_chg_step * battery->pdata->num_age_step != len) {
ret = of_property_read_u32_array(np, "battery,dc_step_chg_cond_soc",
*pdata->dc_step_chg_cond_soc, battery->dc_step_chg_step);
for( i = 1; i < battery->pdata->num_age_step; i++) {
for( j = 0; j < battery->dc_step_chg_step; j++){
pdata->dc_step_chg_cond_soc[i][j] = pdata->dc_step_chg_cond_soc[0][j];
}
}
}
//////// debug log ////
for( i = 0; i < battery->pdata->num_age_step; i++) {
for( j = 0; j < battery->dc_step_chg_step; j++) {
pr_info("%s soc arr = %d ", __func__, pdata->dc_step_chg_cond_soc[i][j]);
}
pr_info("\n");
}
if (ret) {
pr_info("%s : dc_step_chg_cond_soc read fail\n", __func__);
battery->dc_step_chg_type &= ~STEP_CHARGING_CONDITION_SOC;
}
kfree(soc_cond_temp);
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_SOC &&
battery->dc_step_chg_type & STEP_CHARGING_CONDITION_SOC_INIT_ONLY) {
pr_info("%s : do not set SOC and SOC_INIT_ONLY at the same time\n", __func__);
battery->dc_step_chg_type &= ~STEP_CHARGING_CONDITION_SOC;
}
}
}
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_INPUT_CURRENT) {
p = of_get_property(np, "battery,dc_step_chg_cond_iin", &len);
if (!p) {
pr_err("%s: dc_step_chg_cond_iin is Empty, type(0x%X->0x%x)\n",
__func__, battery->dc_step_chg_type,
battery->dc_step_chg_type & ~STEP_CHARGING_CONDITION_INPUT_CURRENT);
battery->dc_step_chg_type &= ~STEP_CHARGING_CONDITION_INPUT_CURRENT;
} else {
len = len / sizeof(u32);
if (len != battery->dc_step_chg_step) {
/* [dchg] TODO: do some error handling */
pr_err("%s: len of dc_step_chg_cond_iin is not matched, len(%d/%d)\n",
__func__, len, battery->dc_step_chg_step);
}
pdata->dc_step_chg_cond_iin = kzalloc(sizeof(u32) * len, GFP_KERNEL);
ret = of_property_read_u32_array(np, "battery,dc_step_chg_cond_iin",
pdata->dc_step_chg_cond_iin, len);
if (ret) {
pr_info("%s : dc_step_chg_cond_iin read fail\n", __func__);
battery->dc_step_chg_type &= ~STEP_CHARGING_CONDITION_INPUT_CURRENT;
}
ret = of_property_read_u32(np, "battery,dc_step_chg_iin_check_cnt",
&battery->pdata->dc_step_chg_iin_check_cnt);
if (ret) {
pr_err("%s: dc_step_chg_iin_check_cnt is Empty\n", __func__);
battery->pdata->dc_step_chg_iin_check_cnt = 2;
} else {
pr_err("%s: dc_step_chg_iin_check_cnt is %d\n",
__func__, battery->pdata->dc_step_chg_iin_check_cnt);
}
}
}
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_FLOAT_VOLTAGE) {
p = of_get_property(np, "battery,dc_step_chg_val_vfloat", &len);
if (!p) {
pr_err("%s: dc_step_chg_val_vfloat is Empty, type(0x%X->0x%x)\n",
__func__, battery->dc_step_chg_type,
battery->dc_step_chg_type & ~STEP_CHARGING_CONDITION_FLOAT_VOLTAGE);
battery->dc_step_chg_type &= ~STEP_CHARGING_CONDITION_FLOAT_VOLTAGE;
} else {
len = len / sizeof(u32);
if (len/battery->pdata->num_age_step != battery->dc_step_chg_step) {
/* [dchg] TODO: do some error handling */
pr_err("%s: len of dc_step_chg_val_vfloat is not matched, len(%d/%d)\n",
__func__, len, battery->dc_step_chg_step);
}
pr_info("%s step(%d) * age_step(%d) = %d \n", __func__, battery->dc_step_chg_step, battery->pdata->num_age_step, len); // debug
dc_step_chg_val_vfloat_temp = kzalloc(sizeof(u32) * battery->dc_step_chg_step * battery->pdata->num_age_step , GFP_KERNEL);
ret = of_property_read_u32_array(np, "battery,dc_step_chg_val_vfloat",
dc_step_chg_val_vfloat_temp, battery->dc_step_chg_step * battery->pdata->num_age_step);
/* copy buff to 2d arr */
pdata->dc_step_chg_val_vfloat = kzalloc(sizeof(u32 *) * battery->pdata->num_age_step , GFP_KERNEL);
for( i = 0; i < battery->pdata->num_age_step; i++) {
pdata->dc_step_chg_val_vfloat[i] = kzalloc(sizeof(u32) * battery->dc_step_chg_step , GFP_KERNEL);
for( j = 0; j < battery->dc_step_chg_step; j++) {
pdata->dc_step_chg_val_vfloat[i][j] = dc_step_chg_val_vfloat_temp[i*battery->dc_step_chg_step + j];
}
}
/* if there are only 1 dimentional array of value, get the same value */
if(battery->dc_step_chg_step * battery->pdata->num_age_step != len) {
ret = of_property_read_u32_array(np, "battery,dc_step_chg_val_vfloat",
*pdata->dc_step_chg_val_vfloat, battery->dc_step_chg_step);
for( i = 1; i < battery->pdata->num_age_step; i++) {
for( j = 0; j < battery->dc_step_chg_step; j++){
pdata->dc_step_chg_val_vfloat[i][j] = pdata->dc_step_chg_val_vfloat[0][j];
}
}
}
//////// debug log ////
for( i = 0; i < battery->pdata->num_age_step; i++) {
for( j = 0; j < battery->dc_step_chg_step; j++) {
pr_info("%s vfloat arr = %d ", __func__, pdata->dc_step_chg_val_vfloat[i][j]);
}
pr_info("\n");
}
if (ret) {
pr_info("%s : dc_step_chg_val_vfloat read fail\n", __func__);
battery->dc_step_chg_type &= ~STEP_CHARGING_CONDITION_FLOAT_VOLTAGE;
}
kfree(dc_step_chg_val_vfloat_temp);
}
}
p = of_get_property(np, "battery,dc_step_chg_val_iout", &len);
if (!p) {
pr_err("%s: dc_step_chg_val_iout is Empty\n", __func__);
battery->dc_step_chg_type = 0;
return -1;
} else {
len = len / sizeof(u32);
if (len/battery->pdata->num_age_step != battery->dc_step_chg_step) {
/* [dchg] TODO: do some error handling */
pr_err("%s: len of dc_step_chg_val_iout is not matched, len(%d/%d)\n",
__func__, len, battery->dc_step_chg_step);
}
pr_info("%s step(%d) * age_step(%d) = %d \n", __func__, battery->dc_step_chg_step, battery->pdata->num_age_step, len); // debug
dc_step_chg_val_iout_temp = kzalloc(sizeof(u32) * battery->dc_step_chg_step * battery->pdata->num_age_step , GFP_KERNEL);
ret = of_property_read_u32_array(np, "battery,dc_step_chg_val_iout",
dc_step_chg_val_iout_temp, battery->dc_step_chg_step * battery->pdata->num_age_step);
/* copy buff to 2d arr */
pdata->dc_step_chg_val_iout = kzalloc(sizeof(u32 *) * battery->pdata->num_age_step , GFP_KERNEL);
for( i = 0; i < battery->pdata->num_age_step; i++) {
pdata->dc_step_chg_val_iout[i] = kzalloc(sizeof(u32) * battery->dc_step_chg_step , GFP_KERNEL);
for( j = 0; j < battery->dc_step_chg_step; j++) {
pdata->dc_step_chg_val_iout[i][j] = dc_step_chg_val_iout_temp[i*battery->dc_step_chg_step + j];
}
}
/* if there are only 1 dimentional array of value, get the same value */
if(battery->dc_step_chg_step * battery->pdata->num_age_step != len) {
ret = of_property_read_u32_array(np, "battery,dc_step_chg_val_iout",
*pdata->dc_step_chg_val_iout, battery->dc_step_chg_step);
for( i = 1; i < battery->pdata->num_age_step; i++) {
for( j = 0; j < battery->dc_step_chg_step; j++){
pdata->dc_step_chg_val_iout[i][j] = pdata->dc_step_chg_val_iout[0][j];
}
}
}
//////// debug log ////
for( i = 0; i < battery->pdata->num_age_step; i++) {
for( j = 0; j < battery->dc_step_chg_step; j++) {
pr_info("%s iout arr = %d ", __func__, pdata->dc_step_chg_val_iout[i][j]);
}
pr_info("\n");
}
if (ret) {
pr_info("%s : dc_step_chg_val_iout read fail\n", __func__);
}
kfree(dc_step_chg_val_iout_temp);
}
if (battery->dc_step_chg_type != dc_step_chg_type)
pr_err("%s : dc_step_chg_type is changed, type(0x%X->0x%x)\n",
__func__, dc_step_chg_type, battery->dc_step_chg_type);
// print dc step charging information
for (i = 0; i < battery->dc_step_chg_step; i++) {
memset(str, 0x0, sizeof(str));
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_VOLTAGE)
sprintf(str + strlen(str), "cond_vol: %dmV, ", pdata->dc_step_chg_cond_vol[i]);
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_SOC || battery->dc_step_chg_type & STEP_CHARGING_CONDITION_SOC_INIT_ONLY)
sprintf(str + strlen(str), "cond_soc: %d%%, ", pdata->dc_step_chg_cond_soc[battery->pdata->age_step][i]);
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_INPUT_CURRENT)
sprintf(str + strlen(str), "cond_iin: %dmA, ", pdata->dc_step_chg_cond_iin[i]);
if (battery->dc_step_chg_type & STEP_CHARGING_CONDITION_FLOAT_VOLTAGE)
sprintf(str + strlen(str), "vfloat: %dmV, ", pdata->dc_step_chg_val_vfloat[battery->pdata->age_step][i]);
sprintf(str + strlen(str), "iout: %dmA,", pdata->dc_step_chg_val_iout[battery->pdata->age_step][i]);
pr_info("%s : step [%d] %s\n", __func__, i, str);
}
return 0;
dc_step_charging_dt_error:
return -1;
}
#endif
#if defined(CONFIG_BATTERY_AGE_FORECAST)
void sec_bat_set_aging_info_step_charging(struct sec_battery_info *battery)
{
#if defined(CONFIG_DIRECT_CHARGING)
union power_supply_propval val;
int i = 0;
#endif
if (battery->step_charging_type) {
if (battery->step_charging_type & STEP_CHARGING_CONDITION_FLOAT_VOLTAGE)
battery->pdata->step_charging_float_voltage[battery->step_charging_step-1] = battery->pdata->chg_float_voltage;
battery->pdata->step_charging_condition[0] =
battery->pdata->age_data[battery->pdata->age_step].step_charging_condition;
dev_info(battery->dev, "%s: float_v(%d), step_conditon(%d)\n",
__func__, battery->pdata->step_charging_float_voltage[battery->step_charging_step-1],
battery->pdata->step_charging_condition[0]);
}
#if defined(CONFIG_DIRECT_CHARGING)
for (i = 0; i < battery->dc_step_chg_step; i++) {
if (battery->pdata->dc_step_chg_val_vfloat[battery->pdata->age_step][i] > battery->pdata->chg_float_voltage)
battery->pdata->dc_step_chg_val_vfloat[battery->pdata->age_step][i] = battery->pdata->chg_float_voltage;
battery->pdata->dc_step_chg_cond_vol[i] = battery->pdata->dc_step_chg_val_vfloat[battery->pdata->age_step][i];
if (i < battery->dc_step_chg_step - 1)
battery->pdata->dc_step_chg_cond_iin[i] = battery->pdata->dc_step_chg_val_iout[battery->pdata->age_step][i+1] / 2;
}
for (i = 0; i < battery->dc_step_chg_step; i++)
dev_info(battery->dev, "%s: cond_vol: %dmV, vfloat: %dmV, cond_iin: %dmA, iout: %dmA\n", __func__,
battery->pdata->dc_step_chg_cond_vol[i], battery->pdata->dc_step_chg_val_vfloat[battery->pdata->age_step][i],
battery->pdata->dc_step_chg_cond_iin[i], battery->pdata->dc_step_chg_val_iout[battery->pdata->age_step][i]);
val.intval = battery->pdata->dc_step_chg_val_vfloat[battery->pdata->age_step][battery->dc_step_chg_step-1];
psy_do_property(battery->pdata->charger_name, set,
POWER_SUPPLY_EXT_PROP_DIRECT_FLOAT_MAX, val);
sec_bat_reset_step_charging(battery);
sec_bat_check_dc_step_charging(battery);
#endif
}
#endif
void sec_step_charging_init(struct sec_battery_info *battery, struct device *dev)
{
struct device_node *np = dev->of_node;
int ret, len;
sec_battery_platform_data_t *pdata = battery->pdata;
unsigned int i;
const u32 *p;
ret = of_property_read_u32(np, "battery,step_charging_type",
&battery->step_charging_type);
pr_err("%s: step_charging_type 0x%x\n", __func__, battery->step_charging_type);
if (ret) {
pr_err("%s: step_charging_type is Empty\n", __func__);
battery->step_charging_type = 0;
return;
}
ret = of_property_read_u32(np, "battery,step_charging_charge_power",
&battery->step_charging_charge_power);
if (ret) {
pr_err("%s: step_charging_charge_power is Empty\n", __func__);
battery->step_charging_charge_power = 20000;
}
p = of_get_property(np, "battery,step_charging_condition", &len);
if (!p) {
battery->step_charging_step = 0;
} else {
len = len / sizeof(u32);
battery->step_charging_step = len;
pdata->step_charging_condition = kzalloc(sizeof(u32) * len, GFP_KERNEL);
ret = of_property_read_u32_array(np, "battery,step_charging_condition",
pdata->step_charging_condition, len);
if (ret) {
pr_info("%s : step_charging_condition read fail\n", __func__);
battery->step_charging_step = 0;
}
pdata->step_charging_condition_curr = kzalloc(sizeof(u32) * len, GFP_KERNEL);
ret = of_property_read_u32_array(np, "battery,step_charging_condition_curr",
pdata->step_charging_condition_curr, len);
if (ret) {
pr_info("%s : step_charging_condition_curr read fail\n", __func__);
battery->step_charging_step = 0;
}
if (len > 0) {
pdata->step_charging_float_voltage = kzalloc(sizeof(u32) * len, GFP_KERNEL);
ret = of_property_read_u32_array(np, "battery,step_charging_float_voltage",
pdata->step_charging_float_voltage, len);
if (ret) {
pr_info("%s : step_charging_float_voltage read fail\n", __func__);
} else {
for (i = 0; i < len; i++) {
pr_info("%s : step condition(%d), float voltage(%d)\n",
__func__, pdata->step_charging_condition[i],
pdata->step_charging_float_voltage[i]);
}
}
pdata->step_charging_current = kzalloc(sizeof(u32) * len, GFP_KERNEL);
ret = of_property_read_u32_array(np, "battery,step_charging_current",
pdata->step_charging_current, len);
if (ret) {
pr_info("%s : step_charging_current read fail\n", __func__);
battery->step_charging_step = 0;
} else {
battery->step_charging_status = -1;
for (i = 0; i < len; i++) {
pr_info("%s : step condition(%d), current(%d)\n",
__func__, pdata->step_charging_condition[i],
pdata->step_charging_current[i]);
}
}
}
}
#if defined(CONFIG_DIRECT_CHARGING)
sec_dc_step_charging_dt(battery, dev);
#endif
}