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.
 
 
 
kernel_samsung_sm7125/drivers/battery/common/sec_battery_thermal.c

856 lines
36 KiB

/*
* sec_battery_thermal.c
* Samsung Mobile Battery Driver
*
* Copyright (C) 2019 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 "sec_battery.h"
#define GENERATE_CHG_MODE_STRING(STRING) #STRING,
#if defined(CONFIG_PREVENT_USB_CONN_OVERHEAT)
extern int muic_set_hiccup_mode(int on_off);
#endif
char *sec_bat_thermal_zone[] = {
"COLD",
"COOL3",
"COOL2",
"COOL1",
"NORMAL",
"WARM",
"OVERHEAT",
"OVERHEATLIMIT",
};
static const char *CHARGE_MODE_STRING[] = {
FOREACH_CHARGE_MODE(GENERATE_CHG_MODE_STRING)
};
//#if defined(CONFIG_DUAL_BATTERY)
int sec_bat_get_high_priority_temp(struct sec_battery_info *battery)
{
int standard_temp = 250;
int priority_temp = battery->temperature;
if (is_wireless_type(battery->cable_type) && battery->temperature >= 400)
priority_temp -= 5;
if((battery->temperature > standard_temp) && (battery->sub_bat_temp > standard_temp)) {
if(battery->temperature < battery->sub_bat_temp)
priority_temp = battery->sub_bat_temp;
} else {
if(battery->temperature > battery->sub_bat_temp)
priority_temp = battery->sub_bat_temp;
}
pr_info("%s priority_temp = %d\n", __func__, priority_temp);
return priority_temp;
}
//#endif
void sec_bat_check_mix_temp(struct sec_battery_info *battery)
{
int temperature = battery->pdata->blkt_temp_check_type ? battery->blkt_temp : battery->temperature;
int chg_temp;
int input_current = 0;
if (battery->pdata->temp_check_type == SEC_BATTERY_TEMP_CHECK_NONE ||
battery->pdata->chg_temp_check_type == SEC_BATTERY_TEMP_CHECK_NONE)
return;
#if defined(CONFIG_DIRECT_CHARGING) && defined(CONFIG_PDIC_NOTIFIER)
if (is_pd_apdo_wire_type(battery->wire_status) && battery->pd_list.now_isApdo)
chg_temp = battery->dchg_temp;
else
chg_temp = battery->chg_temp;
#else
chg_temp = battery->chg_temp;
#endif
#if defined(CONFIG_DUAL_BATTERY)
temperature = sec_bat_get_high_priority_temp(battery);
#endif
if (battery->siop_level >= 100 && !battery->lcd_status && is_not_wireless_type(battery->cable_type)) {
if ((!battery->mix_limit && (temperature >= battery->pdata->mix_high_temp) &&
(chg_temp >= battery->pdata->mix_high_chg_temp)) ||
(battery->mix_limit && (temperature > battery->pdata->mix_high_temp_recovery))) {
int max_input_current = battery->pdata->full_check_current_1st + 50;
/* input current = float voltage * (topoff_current_1st + 50mA(margin)) / (vbus_level * 0.9) */
input_current = ((battery->pdata->chg_float_voltage / battery->pdata->chg_float_voltage_conv) *
max_input_current) / (battery->input_voltage * 90) / 10;
if (input_current > max_input_current)
input_current = max_input_current;
battery->mix_limit = true;
/* skip other heating control */
sec_bat_set_current_event(battery, SEC_BAT_CURRENT_EVENT_SKIP_HEATING_CONTROL,
SEC_BAT_CURRENT_EVENT_SKIP_HEATING_CONTROL);
sec_vote(battery->input_vote, VOTER_MIX_LIMIT, true, input_current);
#if defined(CONFIG_WIRELESS_TX_MODE)
if (battery->wc_tx_enable) {
pr_info("%s @Tx_Mode enter mix_temp_limit, TX mode should turn off\n", __func__);
sec_bat_set_tx_event(battery, BATT_TX_EVENT_WIRELESS_TX_HIGH_TEMP,
BATT_TX_EVENT_WIRELESS_TX_HIGH_TEMP);
battery->tx_retry_case |= SEC_BAT_TX_RETRY_MIX_TEMP;
sec_wireless_set_tx_enable(battery, false);
}
#endif
} else if (battery->mix_limit) {
battery->mix_limit = false;
sec_vote(battery->input_vote, VOTER_MIX_LIMIT, false, 0);
if (battery->tx_retry_case & SEC_BAT_TX_RETRY_MIX_TEMP) {
pr_info("%s @Tx_Mode recovery mix_temp_limit, TX mode should be retried\n", __func__);
if ((battery->tx_retry_case & ~SEC_BAT_TX_RETRY_MIX_TEMP) == 0)
sec_bat_set_tx_event(battery, BATT_TX_EVENT_WIRELESS_TX_RETRY,
BATT_TX_EVENT_WIRELESS_TX_RETRY);
battery->tx_retry_case &= ~SEC_BAT_TX_RETRY_MIX_TEMP;
}
}
pr_info("%s: mix_limit(%d), temp(%d), chg_temp(%d), input_current(%d)\n",
__func__, battery->mix_limit, temperature, chg_temp, input_current);
} else {
if (battery->mix_limit) {
battery->mix_limit = false;
sec_vote(battery->input_vote, VOTER_MIX_LIMIT, false, 0);
}
}
}
static int sec_bat_get_temp_by_temp_control_source(struct sec_battery_info *battery, int tcs)
{
switch (tcs) {
case TEMP_CONTROL_SOURCE_CHG_THM:
return battery->chg_temp;
case TEMP_CONTROL_SOURCE_USB_THM:
return battery->usb_temp;
case TEMP_CONTROL_SOURCE_WPC_THM:
return battery->wpc_temp;
case TEMP_CONTROL_SOURCE_NONE:
case TEMP_CONTROL_SOURCE_BAT_THM:
default:
return battery->temperature;
}
}
void sec_bat_check_wpc_temp(struct sec_battery_info *battery)
{
int input_current = INT_MAX, charging_current = INT_MAX;
if (battery->pdata->wpc_temp_check_type == SEC_BATTERY_TEMP_CHECK_NONE)
return;
if (is_wireless_type(battery->cable_type)) {
union power_supply_propval value = {0, };
int wpc_vout_level = WIRELESS_VOUT_10V;
mutex_lock(&battery->voutlock);
/* get vout level */
psy_do_property(battery->pdata->wireless_charger_name, get,
POWER_SUPPLY_EXT_PROP_WIRELESS_RX_VOUT, value);
if(is_hv_wireless_type(battery->cable_type) &&
value.intval == WIRELESS_VOUT_5_5V_STEP &&
battery->wpc_vout_level != WIRELESS_VOUT_5_5V_STEP) {
pr_info("%s: real vout was not 10V\n", __func__);
battery->wpc_vout_level = WIRELESS_VOUT_5_5V_STEP;
}
if (battery->siop_level >= 100 && !battery->lcd_status) {
int temp_val = sec_bat_get_temp_by_temp_control_source(battery,
battery->pdata->wpc_temp_control_source);
if ((!battery->chg_limit && temp_val >= battery->pdata->wpc_high_temp) ||
(battery->chg_limit && temp_val > battery->pdata->wpc_high_temp_recovery)) {
battery->chg_limit = true;
if(input_current > battery->pdata->wpc_input_limit_current) {
if (battery->cable_type == SEC_BATTERY_CABLE_WIRELESS_TX &&
battery->pdata->wpc_input_limit_by_tx_check)
input_current = battery->pdata->wpc_input_limit_current_by_tx;
else
input_current = battery->pdata->wpc_input_limit_current;
}
if(charging_current > battery->pdata->wpc_charging_limit_current)
charging_current = battery->pdata->wpc_charging_limit_current;
wpc_vout_level = WIRELESS_VOUT_5_5V_STEP;
} else if (battery->chg_limit) {
battery->chg_limit = false;
}
} else {
if ((is_hv_wireless_type(battery->cable_type) &&
battery->cable_type != SEC_BATTERY_CABLE_WIRELESS_HV_VEHICLE) ||
battery->cable_type == SEC_BATTERY_CABLE_PREPARE_WIRELESS_HV ||
battery->cable_type == SEC_BATTERY_CABLE_PREPARE_WIRELESS_20) {
int temp_val = sec_bat_get_temp_by_temp_control_source(battery,
battery->pdata->wpc_temp_lcd_on_control_source);
if ((!battery->chg_limit &&
temp_val >= battery->pdata->wpc_lcd_on_high_temp) ||
(battery->chg_limit &&
temp_val > battery->pdata->wpc_lcd_on_high_temp_rec)) {
if(input_current > battery->pdata->wpc_lcd_on_input_limit_current)
input_current = battery->pdata->wpc_lcd_on_input_limit_current;
if(charging_current > battery->pdata->wpc_charging_limit_current)
charging_current = battery->pdata->wpc_charging_limit_current;
battery->chg_limit = true;
wpc_vout_level = (battery->capacity < 95) ?
WIRELESS_VOUT_5_5V_STEP : WIRELESS_VOUT_10V;
} else if (battery->chg_limit) {
battery->chg_limit = false;
}
} else if (battery->chg_limit) {
battery->chg_limit = false;
}
}
if (is_hv_wireless_type(battery->cable_type)) {
#if defined(CONFIG_ISDB_CHARGING_CONTROL)
if ((battery->current_event & SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING) ||
(battery->current_event & SEC_BAT_CURRENT_EVENT_ISDB))
#else
if (battery->current_event & SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING)
#endif
wpc_vout_level = WIRELESS_VOUT_5_5V_STEP;
if (wpc_vout_level != battery->wpc_vout_level) {
battery->wpc_vout_level = wpc_vout_level;
if (battery->current_event & SEC_BAT_CURRENT_EVENT_WPC_VOUT_LOCK) {
pr_info("%s: block to set wpc vout level(%d) because otg on\n",
__func__, wpc_vout_level);
} else {
value.intval = wpc_vout_level;
psy_do_property(battery->pdata->wireless_charger_name, set,
POWER_SUPPLY_PROP_INPUT_VOLTAGE_REGULATION, value);
pr_info("%s: change vout level(%d)",
__func__, battery->wpc_vout_level);
sec_vote(battery->input_vote, VOTER_AICL, false, 0);
}
} else if (battery->wpc_vout_level == WIRELESS_VOUT_10V && !battery->chg_limit)
/* reset aicl current to recover current for unexpected aicl during */
/* before vout boosting completion */
sec_vote(battery->input_vote, VOTER_AICL, false, 0);
}
mutex_unlock(&battery->voutlock);
pr_info("%s: vout_level(%d), chg_limit(%d)\n",
__func__, battery->wpc_vout_level, battery->chg_limit);
if (battery->chg_limit) {
if ((battery->siop_level >= 100 && !battery->lcd_status) &&
(input_current > battery->pdata->wpc_input_limit_current))
input_current = battery->pdata->wpc_input_limit_current;
else if ((battery->siop_level < 100 || battery->lcd_status) &&
(input_current > battery->pdata->wpc_lcd_on_input_limit_current))
input_current = battery->pdata->wpc_lcd_on_input_limit_current;
}
if (input_current != INT_MAX) {
sec_vote(battery->input_vote, VOTER_CHG_TEMP, true, input_current);
sec_vote(battery->fcc_vote, VOTER_CHG_TEMP, true, charging_current);
} else {
sec_vote(battery->fcc_vote, VOTER_CHG_TEMP, false, 0);
sec_vote(battery->input_vote, VOTER_CHG_TEMP, false, 0);
}
}
}
void sec_bat_check_tx_temperature(struct sec_battery_info *battery)
{
if (battery->wc_tx_enable) {
if(battery->temperature >= battery->pdata->tx_high_threshold) {
pr_info("@Tx_Mode : %s: Battery temperature is too high. Tx mode should turn off\n", __func__);
/* set tx event */
sec_bat_set_tx_event(battery, BATT_TX_EVENT_WIRELESS_TX_HIGH_TEMP, BATT_TX_EVENT_WIRELESS_TX_HIGH_TEMP);
battery->tx_retry_case |= SEC_BAT_TX_RETRY_HIGH_TEMP;
sec_wireless_set_tx_enable(battery, false);
} else if (battery->temperature <= battery->pdata->tx_low_threshold) {
pr_info("@Tx_Mode : %s: Battery temperature is too low. Tx mode should turn off\n", __func__);
/* set tx event */
sec_bat_set_tx_event(battery, BATT_TX_EVENT_WIRELESS_TX_LOW_TEMP, BATT_TX_EVENT_WIRELESS_TX_LOW_TEMP);
battery->tx_retry_case |= SEC_BAT_TX_RETRY_LOW_TEMP;
sec_wireless_set_tx_enable(battery, false);
}
} else if (battery->tx_retry_case & SEC_BAT_TX_RETRY_HIGH_TEMP) {
if (battery->temperature <= battery->pdata->tx_high_recovery) {
pr_info("@Tx_Mode : %s: Battery temperature goes to normal(High). Retry TX mode\n", __func__);
battery->tx_retry_case &= ~SEC_BAT_TX_RETRY_HIGH_TEMP;
if (!battery->tx_retry_case)
sec_bat_set_tx_event(battery, BATT_TX_EVENT_WIRELESS_TX_RETRY, BATT_TX_EVENT_WIRELESS_TX_RETRY);
}
} else if (battery->tx_retry_case & SEC_BAT_TX_RETRY_LOW_TEMP) {
if (battery->temperature >= battery->pdata->tx_low_recovery) {
pr_info("@Tx_Mode : %s: Battery temperature goes to normal(Low). Retry TX mode\n", __func__);
battery->tx_retry_case &= ~SEC_BAT_TX_RETRY_LOW_TEMP;
if (!battery->tx_retry_case)
sec_bat_set_tx_event(battery, BATT_TX_EVENT_WIRELESS_TX_RETRY, BATT_TX_EVENT_WIRELESS_TX_RETRY);
}
}
}
#if defined(CONFIG_DIRECT_CHARGING) && defined(CONFIG_PDIC_NOTIFIER)
void sec_bat_check_direct_chg_temp(struct sec_battery_info *battery)
{
int input_current = 0, charging_current = 0;
if (battery->pdata->dchg_temp_check_type == SEC_BATTERY_TEMP_CHECK_NONE)
return;
if (battery->siop_level >= 100 && !battery->lcd_status) {
if (!battery->chg_limit && battery->pd_list.now_isApdo &&
(battery->dchg_temp >= battery->pdata->dchg_high_temp)) {
input_current = battery->pdata->dchg_input_limit_current;
charging_current = battery->pdata->dchg_charging_limit_current;
battery->chg_limit = true;
sec_vote(battery->fcc_vote, VOTER_CHG_TEMP, battery->chg_limit, charging_current);
sec_vote(battery->input_vote, VOTER_CHG_TEMP, battery->chg_limit, input_current);
} else if (!battery->chg_limit && (!battery->pd_list.now_isApdo) &&
(battery->chg_temp >= battery->pdata->chg_high_temp)) {
if (battery->input_voltage == SEC_INPUT_VOLTAGE_5V) {
input_current = battery->pdata->default_input_current;
charging_current = battery->pdata->default_charging_current;
} else {
input_current = battery->pdata->chg_input_limit_current;
charging_current = battery->pdata->chg_charging_limit_current;
}
battery->chg_limit = true;
sec_vote(battery->fcc_vote, VOTER_CHG_TEMP, battery->chg_limit, charging_current);
sec_vote(battery->input_vote, VOTER_CHG_TEMP, battery->chg_limit, input_current);
} else if (battery->chg_limit) {
if (((battery->dchg_temp <= battery->pdata->dchg_high_temp_recovery) &&
battery->pd_list.now_isApdo) || ((battery->chg_temp <= battery->pdata->chg_high_temp_recovery) &&
(!battery->pd_list.now_isApdo))) {
battery->chg_limit = false;
sec_vote(battery->fcc_vote, VOTER_CHG_TEMP, battery->chg_limit, charging_current);
sec_vote(battery->input_vote, VOTER_CHG_TEMP, battery->chg_limit, input_current);
} else {
if (battery->pd_list.now_isApdo) {
input_current = battery->pdata->dchg_input_limit_current;
charging_current = battery->pdata->dchg_charging_limit_current;
} else {
if (battery->input_voltage == SEC_INPUT_VOLTAGE_5V) {
input_current = battery->pdata->default_input_current;
charging_current = battery->pdata->default_charging_current;
} else {
input_current = battery->pdata->chg_input_limit_current;
charging_current = battery->pdata->chg_charging_limit_current;
}
}
battery->chg_limit = true;
sec_vote(battery->fcc_vote, VOTER_CHG_TEMP, battery->chg_limit, charging_current);
sec_vote(battery->input_vote, VOTER_CHG_TEMP, battery->chg_limit, input_current);
}
}
pr_info("%s: cable_type(%d), chg_limit(%d) vbus_by_siop(%d)\n", __func__,
battery->cable_type, battery->chg_limit, battery->vbus_chg_by_siop);
}
}
#endif
#if defined(CONFIG_PDIC_NOTIFIER)
void sec_bat_check_pdic_temp(struct sec_battery_info *battery)
{
int input_current, charging_current;
if (battery->pdata->chg_temp_check_type == SEC_BATTERY_TEMP_CHECK_NONE)
return;
if (battery->pdic_ps_rdy && battery->siop_level >= 100 && !battery->lcd_status) {
if ((!battery->chg_limit && (battery->chg_temp >= battery->pdata->chg_high_temp)) ||
(battery->chg_limit && (battery->chg_temp >= battery->pdata->chg_high_temp_recovery))) {
input_current =
(battery->pdata->chg_input_limit_current * SEC_INPUT_VOLTAGE_9V) / battery->input_voltage;
charging_current = battery->pdata->chg_charging_limit_current;
sec_vote(battery->fcc_vote, VOTER_PDIC_TEMP, true, charging_current);
sec_vote(battery->input_vote, VOTER_PDIC_TEMP, true, input_current);
battery->chg_limit = true;
} else if (battery->chg_limit && battery->chg_temp <= battery->pdata->chg_high_temp_recovery) {
sec_vote(battery->fcc_vote, VOTER_PDIC_TEMP, false, 0);
sec_vote(battery->input_vote, VOTER_PDIC_TEMP, false, 0);
battery->chg_limit = false;
}
pr_info("%s: cable_type(%d), chg_limit(%d)\n", __func__,
battery->cable_type, battery->chg_limit);
}
}
#endif
void sec_bat_check_afc_temp(struct sec_battery_info *battery)
{
int input_current = 0, charging_current = 0;
if (battery->pdata->chg_temp_check_type == SEC_BATTERY_TEMP_CHECK_NONE)
return;
#if defined(CONFIG_SUPPORT_HV_CTRL)
if (battery->siop_level >= 100 && !battery->lcd_status) {
if (!battery->chg_limit && is_hv_wire_type(battery->cable_type) &&
(battery->chg_temp >= battery->pdata->chg_high_temp)) {
input_current = battery->pdata->chg_input_limit_current;
charging_current = battery->pdata->chg_charging_limit_current;
battery->chg_limit = true;
sec_vote(battery->fcc_vote, VOTER_CHG_TEMP, battery->chg_limit, charging_current);
sec_vote(battery->input_vote, VOTER_CHG_TEMP, battery->chg_limit, input_current);
} else if (!battery->chg_limit && battery->max_charge_power >= (battery->pdata->pd_charging_charge_power - 500) &&
(battery->chg_temp >= battery->pdata->chg_high_temp)) {
input_current = battery->pdata->default_input_current;
charging_current = battery->pdata->default_charging_current;
battery->chg_limit = true;
sec_vote(battery->fcc_vote, VOTER_CHG_TEMP, battery->chg_limit, charging_current);
sec_vote(battery->input_vote, VOTER_CHG_TEMP, battery->chg_limit, input_current);
} else if (battery->chg_limit && is_hv_wire_type(battery->cable_type)) {
if (battery->chg_temp <= battery->pdata->chg_high_temp_recovery) {
battery->chg_limit = false;
sec_vote(battery->fcc_vote, VOTER_CHG_TEMP, battery->chg_limit, charging_current);
sec_vote(battery->input_vote, VOTER_CHG_TEMP, battery->chg_limit, input_current);
}
} else if (battery->chg_limit && battery->max_charge_power >= (battery->pdata->pd_charging_charge_power - 500)) {
if (battery->chg_temp <= battery->pdata->chg_high_temp_recovery) {
battery->chg_limit = false;
sec_vote(battery->fcc_vote, VOTER_CHG_TEMP, battery->chg_limit, charging_current);
sec_vote(battery->input_vote, VOTER_CHG_TEMP, battery->chg_limit, input_current);
}
}
pr_info("%s: cable_type(%d), chg_limit(%d) vbus_by_siop(%d)\n", __func__,
battery->cable_type, battery->chg_limit, battery->vbus_chg_by_siop);
}
#else
if ((!battery->chg_limit && is_hv_wire_type(battery->cable_type) &&
(battery->chg_temp >= battery->pdata->chg_high_temp)) ||
(battery->chg_limit && is_hv_wire_type(battery->cable_type) &&
(battery->chg_temp > battery->pdata->chg_high_temp_recovery))) {
if (!battery->chg_limit) {
input_current = battery->pdata->chg_input_limit_current;
charging_current = battery->pdata->chg_charging_limit_current;
sec_vote(battery->fcc_vote, VOTER_AFC_TEMP, true, charging_current);
sec_vote(battery->input_vote, VOTER_AFC_TEMP, true, input_current);
battery->chg_limit = true;
}
} else if (battery->chg_limit && is_hv_wire_type(battery->cable_type) &&
(battery->chg_temp <= battery->pdata->chg_high_temp_recovery)) {
sec_vote(battery->fcc_vote, VOTER_AFC_TEMP, false, 0);
sec_vote(battery->input_vote, VOTER_AFC_TEMP, false, 0);
battery->chg_limit = false;
}
#endif
}
void sec_bat_set_threshold(struct sec_battery_info *battery)
{
if (is_wireless_type(battery->cable_type) || battery->cable_type == SEC_BATTERY_CABLE_WIRELESS_FAKE) {
battery->cold_cool3_thresh = battery->pdata->wireless_cold_cool3_thresh;
battery->cool3_cool2_thresh = battery->pdata->wireless_cool3_cool2_thresh;
battery->cool2_cool1_thresh = battery->pdata->wireless_cool2_cool1_thresh;
battery->cool1_normal_thresh = battery->pdata->wireless_cool1_normal_thresh;
battery->normal_warm_thresh = battery->pdata->wireless_normal_warm_thresh;
battery->warm_overheat_thresh = battery->pdata->wireless_warm_overheat_thresh;
} else {
battery->cold_cool3_thresh = battery->pdata->wire_cold_cool3_thresh;
battery->cool3_cool2_thresh = battery->pdata->wire_cool3_cool2_thresh;
battery->cool2_cool1_thresh = battery->pdata->wire_cool2_cool1_thresh;
battery->cool1_normal_thresh = battery->pdata->wire_cool1_normal_thresh;
battery->normal_warm_thresh = battery->pdata->wire_normal_warm_thresh;
battery->warm_overheat_thresh = battery->pdata->wire_warm_overheat_thresh;
}
}
bool sec_usb_thm_overheatlimit(struct sec_battery_info *battery)
{
#if defined(CONFIG_PREVENT_USB_CONN_OVERHEAT)
int gap = 0;
int bat_thm = battery->temperature;
#endif
if (battery->pdata->usb_temp_check_type == SEC_BATTERY_TEMP_CHECK_NONE) {
pr_err("%s: USB_THM, Invalid Temp Check Type, usb_thm <- bat_thm\n", __func__);
battery->usb_temp = battery->temperature;
}
if (battery->usb_thm_status == USB_THM_NORMAL) {
#if defined(CONFIG_PREVENT_USB_CONN_OVERHEAT)
#if defined(CONFIG_DUAL_BATTERY)
/* select low temp thermistor */
if (battery->temperature > battery->sub_bat_temp)
bat_thm = battery->sub_bat_temp;
#endif
if (battery->usb_temp > bat_thm)
gap = battery->usb_temp - bat_thm;
#endif
if (battery->usb_temp >= battery->overheatlimit_threshold) {
pr_info("%s: Usb Temp over than %d (usb_thm : %d)\n", __func__,
battery->overheatlimit_threshold, battery->usb_temp);
battery->usb_thm_status = USB_THM_OVERHEATLIMIT;
return true;
#if defined(CONFIG_PREVENT_USB_CONN_OVERHEAT)
} else if ((battery->usb_temp >= battery->usb_protection_temp) &&
(gap >= battery->temp_gap_bat_usb)) {
pr_info("%s: Temp gap between Usb temp and Bat temp : %d\n", __func__, gap);
#if defined(CONFIG_BATTERY_CISD)
if (gap > battery->cisd.data[CISD_DATA_USB_OVERHEAT_ALONE_PER_DAY])
battery->cisd.data[CISD_DATA_USB_OVERHEAT_ALONE_PER_DAY] = gap;
#endif
battery->usb_thm_status = USB_THM_GAP_OVER;
return true;
#endif
} else {
battery->usb_thm_status = USB_THM_NORMAL;
return false;
}
} else if (battery->usb_thm_status == USB_THM_OVERHEATLIMIT) {
if (battery->usb_temp <= battery->overheatlimit_recovery) {
battery->usb_thm_status = USB_THM_NORMAL;
return false;
} else {
return true;
}
#if defined(CONFIG_PREVENT_USB_CONN_OVERHEAT)
} else if (battery->usb_thm_status == USB_THM_GAP_OVER) {
if (battery->usb_temp < battery->usb_protection_temp) {
battery->usb_thm_status = USB_THM_NORMAL;
return false;
} else {
return true;
}
#endif
} else {
battery->usb_thm_status = USB_THM_NORMAL;
return false;
}
return false;
}
#define THERMAL_HYSTERESIS_2 19
#define THERMAL_HYSTERESIS_5 49
void sec_bat_thermal_check(struct sec_battery_info *battery)
{
int bat_thm = battery->temperature;
int pre_thermal_zone = battery->thermal_zone;
int voter_swelling_status = SEC_BAT_CHG_MODE_CHARGING;
#if defined(CONFIG_DUAL_BATTERY)
bat_thm = sec_bat_get_high_priority_temp(battery);
#endif
pr_err("%s: co_c3: %d, c3_c2: %d, c2_c1: %d, c1_no: %d, no_wa: %d, wa_ov: %d, tz(%s)\n", __func__,
battery->cold_cool3_thresh, battery->cool3_cool2_thresh, battery->cool2_cool1_thresh,
battery->cool1_normal_thresh, battery->normal_warm_thresh, battery->warm_overheat_thresh,
sec_bat_thermal_zone[battery->thermal_zone]);
if (battery->status == POWER_SUPPLY_STATUS_DISCHARGING ||
battery->skip_swelling) {
battery->health_change = false;
pr_debug("%s: DISCHARGING or 15 test mode. stop thermal check\n", __func__);
battery->thermal_zone = BAT_THERMAL_NORMAL;
sec_vote(battery->topoff_vote, VOTER_SWELLING, false, 0);
sec_vote(battery->fcc_vote, VOTER_SWELLING, false, 0);
sec_vote(battery->fv_vote, VOTER_SWELLING, false, 0);
sec_vote(battery->chgen_vote, VOTER_SWELLING, false, 0);
sec_bat_set_current_event(battery, 0, SEC_BAT_CURRENT_EVENT_SWELLING_MODE);
sec_bat_set_threshold(battery);
return;
}
if (battery->pdata->temp_check_type == SEC_BATTERY_TEMP_CHECK_NONE) {
pr_err("%s: BAT_THM, Invalid Temp Check Type\n", __func__);
return;
} else {
/* COLD - COOL3 - COOL2 - COOL1 - NORMAL - WARM - OVERHEAT - OVERHEATLIMIT*/
if (sec_usb_thm_overheatlimit(battery)) {
battery->thermal_zone = BAT_THERMAL_OVERHEATLIMIT;
} else if (bat_thm >= battery->normal_warm_thresh) {
if (bat_thm >= battery->warm_overheat_thresh) {
battery->thermal_zone = BAT_THERMAL_OVERHEAT;
} else {
battery->thermal_zone = BAT_THERMAL_WARM;
}
} else if (bat_thm <= battery->cool1_normal_thresh) {
if (bat_thm <= battery->cold_cool3_thresh) {
battery->thermal_zone = BAT_THERMAL_COLD;
} else if (bat_thm <= battery->cool3_cool2_thresh) {
battery->thermal_zone = BAT_THERMAL_COOL3;
} else if (bat_thm <= battery->cool2_cool1_thresh) {
battery->thermal_zone = BAT_THERMAL_COOL2;
} else {
battery->thermal_zone = BAT_THERMAL_COOL1;
}
} else {
battery->thermal_zone = BAT_THERMAL_NORMAL;
}
}
if (pre_thermal_zone != battery->thermal_zone) {
battery->bat_thm_count++;
if (battery->bat_thm_count < battery->pdata->temp_check_count) {
pr_info("%s : bat_thm_count %d/%d\n", __func__,
battery->bat_thm_count, battery->pdata->temp_check_count);
battery->thermal_zone = pre_thermal_zone;
return;
}
pr_info("%s: thermal zone update (%s -> %s), bat_thm(%d), usb_thm(%d)\n", __func__,
sec_bat_thermal_zone[pre_thermal_zone],
sec_bat_thermal_zone[battery->thermal_zone], bat_thm, battery->usb_temp);
battery->health_change = true;
battery->bat_thm_count = 0;
pr_info("%s : SAFETY TIME RESET!\n", __func__);
battery->expired_time = battery->pdata->expired_time;
battery->prev_safety_time = 0;
sec_bat_set_threshold(battery);
sec_bat_set_current_event(battery, 0, SEC_BAT_CURRENT_EVENT_SWELLING_MODE);
switch (battery->thermal_zone) {
case BAT_THERMAL_OVERHEATLIMIT:
battery->warm_overheat_thresh -= THERMAL_HYSTERESIS_2;
battery->normal_warm_thresh -= THERMAL_HYSTERESIS_2;
if (battery->usb_thm_status == USB_THM_OVERHEATLIMIT) {
pr_info("%s: USB_THM_OVERHEATLIMIT\n", __func__);
#if defined(CONFIG_BATTERY_CISD)
battery->cisd.data[CISD_DATA_USB_OVERHEAT_CHARGING]++;
battery->cisd.data[CISD_DATA_USB_OVERHEAT_CHARGING_PER_DAY]++;
#endif
#if defined(CONFIG_PREVENT_USB_CONN_OVERHEAT)
} else if (battery->usb_thm_status == USB_THM_GAP_OVER) {
pr_info("%s: USB_THM_GAP_OVER : %d\n", __func__, gap);
#if defined(CONFIG_BATTERY_CISD)
battery->cisd.data[CISD_DATA_USB_OVERHEAT_RAPID_CHANGE]++;
battery->cisd.data[CISD_DATA_USB_OVERHEAT_RAPID_CHANGE_PER_DAY]++;
#endif
#endif
}
battery->health = POWER_SUPPLY_HEALTH_OVERHEATLIMIT;
sec_bat_set_charging_status(battery, POWER_SUPPLY_STATUS_NOT_CHARGING);
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_BUCK_OFF);
#if defined(CONFIG_BATTERY_CISD)
battery->cisd.data[CISD_DATA_UNSAFETY_TEMPERATURE]++;
battery->cisd.data[CISD_DATA_UNSAFE_TEMPERATURE_PER_DAY]++;
#endif
#if defined(CONFIG_PREVENT_USB_CONN_OVERHEAT)
if (lpcharge) {
if (is_hv_afc_wire_type(battery->cable_type) && !battery->vbus_limit) {
#if defined(CONFIG_MUIC_HV) || defined(CONFIG_SUPPORT_HV_CTRL)
battery->vbus_chg_by_siop = SEC_INPUT_VOLTAGE_0V;
muic_afc_request_voltage(AFC_REQUEST_CHARGER, SEC_INPUT_VOLTAGE_0V);
#endif
battery->vbus_limit = true;
pr_info("%s: Set AFC TA to 0V\n", __func__);
} else if (is_pd_wire_type(battery->cable_type)) {
select_pdo(1);
pr_info("%s: Set PD TA to PDO 0\n", __func__);
}
} else if (is_pd_wire_type(battery->cable_type)) {
select_pdo(1);
muic_set_hiccup_mode(1);
} else {
muic_set_hiccup_mode(1);
}
#endif
break;
case BAT_THERMAL_OVERHEAT:
battery->usb_thm_status = USB_THM_NORMAL;
battery->warm_overheat_thresh -= THERMAL_HYSTERESIS_2;
battery->normal_warm_thresh -= THERMAL_HYSTERESIS_2;
if (battery->voltage_now > battery->pdata->high_temp_float)
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_BUCK_OFF);
else
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_CHARGING_OFF);
battery->health = POWER_SUPPLY_HEALTH_OVERHEAT;
sec_bat_set_charging_status(battery, POWER_SUPPLY_STATUS_NOT_CHARGING);
#if defined(CONFIG_BATTERY_CISD)
battery->cisd.data[CISD_DATA_UNSAFETY_TEMPERATURE]++;
battery->cisd.data[CISD_DATA_UNSAFE_TEMPERATURE_PER_DAY]++;
#endif
sec_bat_set_current_event(battery, SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING,
SEC_BAT_CURRENT_EVENT_SWELLING_MODE);
break;
case BAT_THERMAL_WARM:
battery->usb_thm_status = USB_THM_NORMAL;
battery->normal_warm_thresh -= THERMAL_HYSTERESIS_2;
if (is_wireless_type(battery->cable_type) ||
battery->cable_type == SEC_BATTERY_CABLE_WIRELESS_FAKE) {
sec_vote(battery->fcc_vote, VOTER_SWELLING, true, battery->pdata->wireless_warm_current);
#if defined(CONFIG_BATTERY_CISD)
battery->cisd.data[CISD_DATA_WC_HIGH_TEMP_SWELLING]++;
battery->cisd.data[CISD_DATA_WC_HIGH_TEMP_SWELLING_PER_DAY]++;
#endif
} else {
sec_vote(battery->fcc_vote, VOTER_SWELLING, true, battery->pdata->wire_warm_current);
#if defined(CONFIG_BATTERY_CISD)
battery->cisd.data[CISD_DATA_HIGH_TEMP_SWELLING]++;
battery->cisd.data[CISD_DATA_HIGH_TEMP_SWELLING_PER_DAY]++;
#endif
}
sec_vote(battery->topoff_vote, VOTER_SWELLING, true, battery->pdata->high_temp_topoff);
if (battery->voltage_now > battery->pdata->high_temp_float) {
if (battery->wc_tx_enable &&
(is_hv_wire_type(battery->cable_type) || is_pd_wire_type(battery->cable_type)))
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_CHARGING_OFF);
else
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_BUCK_OFF);
} else if (battery->voltage_now > battery->pdata->swelling_high_rechg_voltage) {
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_CHARGING_OFF);
} else {
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_CHARGING);
sec_vote(battery->fv_vote, VOTER_SWELLING, true, battery->pdata->high_temp_float);
}
battery->health = POWER_SUPPLY_HEALTH_GOOD;
sec_bat_set_current_event(battery, SEC_BAT_CURRENT_EVENT_HIGH_TEMP_SWELLING,
SEC_BAT_CURRENT_EVENT_SWELLING_MODE);
break;
case BAT_THERMAL_COOL1:
battery->usb_thm_status = USB_THM_NORMAL;
battery->cool1_normal_thresh += THERMAL_HYSTERESIS_2;
if (is_wireless_type(battery->cable_type) ||
battery->cable_type == SEC_BATTERY_CABLE_WIRELESS_FAKE) {
sec_vote(battery->fcc_vote, VOTER_SWELLING, true, battery->pdata->wireless_cool1_current);
} else {
sec_vote(battery->fcc_vote, VOTER_SWELLING, true, battery->pdata->wire_cool1_current);
}
sec_vote(battery->fv_vote, VOTER_SWELLING, true, battery->pdata->low_temp_float);
sec_vote(battery->topoff_vote, VOTER_SWELLING, true, battery->pdata->low_temp_topoff);
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_CHARGING);
battery->health = POWER_SUPPLY_HEALTH_GOOD;
sec_bat_set_current_event(battery, SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_COOL1,
SEC_BAT_CURRENT_EVENT_SWELLING_MODE);
break;
case BAT_THERMAL_COOL2:
battery->usb_thm_status = USB_THM_NORMAL;
battery->cool2_cool1_thresh += THERMAL_HYSTERESIS_2;
battery->cool1_normal_thresh += THERMAL_HYSTERESIS_2;
if (is_wireless_type(battery->cable_type) ||
battery->cable_type == SEC_BATTERY_CABLE_WIRELESS_FAKE) {
sec_vote(battery->fcc_vote, VOTER_SWELLING, true, battery->pdata->wireless_cool2_current);
} else {
sec_vote(battery->fcc_vote, VOTER_SWELLING, true, battery->pdata->wire_cool2_current);
}
sec_vote(battery->fv_vote, VOTER_SWELLING, true, battery->pdata->low_temp_float);
sec_vote(battery->topoff_vote, VOTER_SWELLING, true, battery->pdata->low_temp_topoff);
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_CHARGING);
battery->health = POWER_SUPPLY_HEALTH_GOOD;
sec_bat_set_current_event(battery, SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_COOL2,
SEC_BAT_CURRENT_EVENT_SWELLING_MODE);
break;
case BAT_THERMAL_COOL3:
battery->usb_thm_status = USB_THM_NORMAL;
battery->cool3_cool2_thresh += THERMAL_HYSTERESIS_2;
battery->cool2_cool1_thresh += THERMAL_HYSTERESIS_2;
battery->cool1_normal_thresh += THERMAL_HYSTERESIS_2;
if (is_wireless_type(battery->cable_type) ||
battery->cable_type == SEC_BATTERY_CABLE_WIRELESS_FAKE) {
sec_vote(battery->fcc_vote, VOTER_SWELLING, true, battery->pdata->wireless_cool3_current);
} else {
sec_vote(battery->fcc_vote, VOTER_SWELLING, true, battery->pdata->wire_cool3_current);
}
sec_vote(battery->fv_vote, VOTER_SWELLING, true, battery->pdata->low_temp_float);
sec_vote(battery->topoff_vote, VOTER_SWELLING, true, battery->pdata->low_temp_topoff);
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_CHARGING);
battery->health = POWER_SUPPLY_HEALTH_GOOD;
sec_bat_set_current_event(battery, SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_COOL3,
SEC_BAT_CURRENT_EVENT_SWELLING_MODE);
break;
case BAT_THERMAL_COLD:
battery->usb_thm_status = USB_THM_NORMAL;
battery->cold_cool3_thresh += THERMAL_HYSTERESIS_2;
battery->cool3_cool2_thresh += THERMAL_HYSTERESIS_2;
battery->cool2_cool1_thresh += THERMAL_HYSTERESIS_2;
battery->cool1_normal_thresh += THERMAL_HYSTERESIS_2;
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_CHARGING_OFF);
battery->health = POWER_SUPPLY_HEALTH_COLD;
sec_bat_set_charging_status(battery, POWER_SUPPLY_STATUS_NOT_CHARGING);
#if defined(CONFIG_BATTERY_CISD)
battery->cisd.data[CISD_DATA_UNSAFETY_TEMPERATURE]++;
battery->cisd.data[CISD_DATA_UNSAFE_TEMPERATURE_PER_DAY]++;
#endif
sec_bat_set_current_event(battery, SEC_BAT_CURRENT_EVENT_LOW_TEMP_SWELLING_COOL3,
SEC_BAT_CURRENT_EVENT_SWELLING_MODE);
break;
case BAT_THERMAL_NORMAL:
default:
battery->usb_thm_status = USB_THM_NORMAL;
sec_vote(battery->fcc_vote, VOTER_SWELLING, false, 0);
sec_vote(battery->fv_vote, VOTER_SWELLING, false, 0);
sec_vote(battery->topoff_vote, VOTER_SWELLING, false, 0);
sec_vote(battery->chgen_vote, VOTER_SWELLING, false, 0);
battery->health = POWER_SUPPLY_HEALTH_GOOD;
break;
}
if ((battery->thermal_zone >= BAT_THERMAL_COOL3) && (battery->thermal_zone <= BAT_THERMAL_WARM)) {
#if defined(CONFIG_ENABLE_FULL_BY_SOC)
if ((battery->capacity >= 100) || (battery->status == POWER_SUPPLY_STATUS_FULL))
sec_bat_set_charging_status(battery, POWER_SUPPLY_STATUS_FULL);
else
sec_bat_set_charging_status(battery, POWER_SUPPLY_STATUS_CHARGING);
#else
if (battery->status != POWER_SUPPLY_STATUS_FULL)
sec_bat_set_charging_status(battery, POWER_SUPPLY_STATUS_CHARGING);
#endif
}
} else { /* pre_thermal_zone == battery->thermal_zone */
battery->health_change = false;
switch (battery->thermal_zone) {
case BAT_THERMAL_OVERHEAT:
/* does not need buck control at low temperatures */
if (battery->health == POWER_SUPPLY_HEALTH_OVERHEAT) {
if (get_sec_voter_status(battery->chgen_vote, VOTER_SWELLING, &voter_swelling_status) < 0)
pr_err("%s: INVALID VOTER ID\n", __func__);
pr_info("%s: voter_swelling_status: %s\n", __func__, CHARGE_MODE_STRING[voter_swelling_status]);
if ((voter_swelling_status == SEC_BAT_CHG_MODE_BUCK_OFF) &&
(battery->voltage_now < (battery->pdata->high_temp_float - battery->pdata->buck_recovery_margin))) {
pr_info("%s: Vnow(%dmV) < %dmV has dropped enough to get buck on mode in overheat conditions\n", __func__,
battery->voltage_now, (battery->pdata->high_temp_float - battery->pdata->buck_recovery_margin));
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_CHARGING_OFF);
}
}
break;
case BAT_THERMAL_WARM:
if (battery->health == POWER_SUPPLY_HEALTH_GOOD) {
if (get_sec_voter_status(battery->chgen_vote, VOTER_SWELLING, &voter_swelling_status) < 0)
pr_err("%s: INVALID VOTER ID\n", __func__);
pr_info("%s: voter_swelling_status: %s\n", __func__, CHARGE_MODE_STRING[voter_swelling_status]);
if (voter_swelling_status == SEC_BAT_CHG_MODE_CHARGING) {
if (sec_bat_check_full(battery, battery->pdata->full_check_type)) {
pr_info("%s: battery thermal zone WARM. Full charged.\n", __func__);
sec_vote(battery->chgen_vote, VOTER_SWELLING, true,
SEC_BAT_CHG_MODE_CHARGING_OFF);
}
} else if ((voter_swelling_status == SEC_BAT_CHG_MODE_CHARGING_OFF ||
voter_swelling_status == SEC_BAT_CHG_MODE_BUCK_OFF) &&
(battery->voltage_now <= battery->pdata->swelling_high_rechg_voltage)) {
pr_info("%s: thermal zone WARM. charging recovery. Vnow: %d\n",
__func__, battery->voltage_now);
battery->expired_time = battery->pdata->expired_time;
battery->prev_safety_time = 0;
sec_vote(battery->fv_vote, VOTER_SWELLING, true,
battery->pdata->high_temp_float);
sec_vote(battery->chgen_vote, VOTER_FULL_CHARGE, false, 0);
sec_vote(battery->chgen_vote, VOTER_SWELLING, true,
SEC_BAT_CHG_MODE_CHARGING);
} else if ((voter_swelling_status == SEC_BAT_CHG_MODE_BUCK_OFF) &&
(battery->voltage_now < (battery->pdata->high_temp_float - battery->pdata->buck_recovery_margin))) {
pr_info("%s: Vnow(%dmV) < %dmV has dropped enough to get buck on mode in high temperature swelling protection\n", __func__,
battery->voltage_now, (battery->pdata->high_temp_float - battery->pdata->buck_recovery_margin));
sec_vote(battery->chgen_vote, VOTER_SWELLING, true, SEC_BAT_CHG_MODE_CHARGING_OFF);
}
}
break;
default:
break;
}
}
return;
}