123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856 |
- /*
- * sec_cisd.c
- * Samsung Mobile Battery Driver
- *
- * Copyright (C) 2012 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"
- #include "sec_cisd.h"
- #if IS_ENABLED(CONFIG_SEC_ABC)
- #include <linux/sti/abc_common.h>
- #endif
- const char *cisd_data_str[] = {
- "RESET_ALG", "ALG_INDEX", "FULL_CNT", "CAP_MAX", "CAP_MIN", "RECHARGING_CNT", "VALERT_CNT",
- "BATT_CYCLE", "WIRE_CNT", "WIRELESS_CNT", "HIGH_SWELLING_CNT", "LOW_SWELLING_CNT",
- "WC_HIGH_SWELLING_CNT", "SWELLING_FULL_CNT", "SWELLING_RECOVERY_CNT", "AICL_CNT", "BATT_THM_MAX",
- "BATT_THM_MIN", "CHG_THM_MAX", "CHG_THM_MIN", "WPC_THM_MAX", "WPC_THM_MIN", "USB_THM_MAX", "USB_THM_MIN",
- "CHG_BATT_THM_MAX", "CHG_BATT_THM_MIN", "CHG_CHG_THM_MAX", "CHG_CHG_THM_MIN", "CHG_WPC_THM_MAX",
- "CHG_WPC_THM_MIN", "CHG_USB_THM_MAX", "CHG_USB_THM_MIN", "USB_OVERHEAT_CHARGING", "UNSAFETY_VOLT",
- "UNSAFETY_TEMP", "SAFETY_TIMER", "VSYS_OVP", "VBAT_OVP", "USB_OVERHEAT_RAPID_CHANGE", "ASOC",
- "USB_OVERHEAT_ALONE", "CAP_NOM", "RC0"
- };
- EXPORT_SYMBOL(cisd_data_str);
- const char *cisd_data_str_d[] = {
- "FULL_CNT_D", "CAP_MAX_D", "CAP_MIN_D", "RECHARGING_CNT_D", "VALERT_CNT_D", "WIRE_CNT_D", "WIRELESS_CNT_D",
- "HIGH_SWELLING_CNT_D", "LOW_SWELLING_CNT_D", "WC_HIGH_SWELLING_CNT_D", "SWELLING_FULL_CNT_D",
- "SWELLING_RECOVERY_CNT_D", "AICL_CNT_D", "BATT_THM_MAX_D", "BATT_THM_MIN_D", "SUB_BATT_THM_MAX_D",
- "SUB_BATT_THM_MIN_D", "CHG_THM_MAX_D", "CHG_THM_MIN_D", "USB_THM_MAX_D", "USB_THM_MIN_D",
- "CHG_BATT_THM_MAX_D", "CHG_BATT_THM_MIN_D", "CHG_SUB_BATT_THM_MAX_D", "CHG_SUB_BATT_THM_MIN_D",
- "CHG_CHG_THM_MAX_D", "CHG_CHG_THM_MIN_D", "CHG_USB_THM_MAX_D", "CHG_USB_THM_MIN_D",
- "USB_OVERHEAT_CHARGING_D", "UNSAFETY_VOLT_D", "UNSAFETY_TEMP_D",
- "SAFETY_TIMER_D", "VSYS_OVP_D", "VBAT_OVP_D", "USB_OVERHEAT_RAPID_CHANGE_D", "BUCK_OFF_D",
- "USB_OVERHEAT_ALONE_D", "DROP_SENSOR_D", "CHG_TIME_D", "TOTAL_CHG_TIME_D"
- };
- EXPORT_SYMBOL(cisd_data_str_d);
- const char *cisd_cable_data_str[] = {"TA", "AFC", "AFC_FAIL", "QC", "QC_FAIL", "PD", "PD_HIGH", "HV_WC_20", "FPDO_DC"};
- EXPORT_SYMBOL(cisd_cable_data_str);
- const char *cisd_tx_data_str[] = {"ON", "OTHER", "GEAR", "PHONE", "BUDS"};
- EXPORT_SYMBOL(cisd_tx_data_str);
- #if IS_ENABLED(CONFIG_DUAL_BATTERY)
- const char *cisd_event_data_str[] = {"DC_ERR", "TA_OCP_DET", "TA_OCP_ON", "OVP_EVENT_POWER", "OVP_EVENT_SIGNAL", "OTG", "D2D", "MAIN_BAT_ERR", "SUB_BAT_ERR", "WA_ERR"};
- #else
- const char *cisd_event_data_str[] = {"DC_ERR", "TA_OCP_DET", "TA_OCP_ON", "OVP_EVENT_POWER", "OVP_EVENT_SIGNAL", "OTG", "D2D"};
- #endif
- EXPORT_SYMBOL(cisd_event_data_str);
- bool sec_bat_cisd_check(struct sec_battery_info *battery)
- {
- union power_supply_propval val = {0, };
- struct cisd *pcisd = &battery->cisd;
- bool ret = false;
- int voltage = battery->voltage_now;
- if (battery->factory_mode || battery->is_jig_on || battery->skip_cisd) {
- dev_info(battery->dev, "%s: No need to check in factory mode\n",
- __func__);
- return ret;
- }
- #if IS_ENABLED(CONFIG_DUAL_BATTERY)
- voltage = max(battery->voltage_avg_main, battery->voltage_avg_sub);
- #endif
- if ((battery->status == POWER_SUPPLY_STATUS_CHARGING) ||
- (battery->status == POWER_SUPPLY_STATUS_FULL)) {
- /* check abnormal vbat */
- pcisd->ab_vbat_check_count = voltage > pcisd->max_voltage_thr ?
- pcisd->ab_vbat_check_count + 1 : 0;
- if ((pcisd->ab_vbat_check_count >= pcisd->ab_vbat_max_count) &&
- !(pcisd->state & CISD_STATE_OVER_VOLTAGE)) {
- dev_info(battery->dev, "%s : [CISD] Battery Over Voltage Protection !! vbat(%d)mV\n",
- __func__, voltage);
- val.intval = true;
- psy_do_property("battery", set, POWER_SUPPLY_EXT_PROP_VBAT_OVP,
- val);
- pcisd->data[CISD_DATA_VBAT_OVP]++;
- pcisd->data[CISD_DATA_VBAT_OVP_PER_DAY]++;
- pcisd->state |= CISD_STATE_OVER_VOLTAGE;
- #if IS_ENABLED(CONFIG_SEC_ABC)
- #if IS_ENABLED(CONFIG_SEC_FACTORY)
- sec_abc_send_event("MODULE=battery@INFO=over_voltage");
- #else
- sec_abc_send_event("MODULE=battery@WARN=over_voltage");
- #endif
- #endif
- }
- if (battery->temperature > pcisd->data[CISD_DATA_CHG_BATT_TEMP_MAX])
- pcisd->data[CISD_DATA_CHG_BATT_TEMP_MAX] = battery->temperature;
- if (battery->temperature < pcisd->data[CISD_DATA_CHG_BATT_TEMP_MIN])
- pcisd->data[CISD_DATA_CHG_BATT_TEMP_MIN] = battery->temperature;
- if (battery->chg_temp > pcisd->data[CISD_DATA_CHG_CHG_TEMP_MAX])
- pcisd->data[CISD_DATA_CHG_CHG_TEMP_MAX] = battery->chg_temp;
- if (battery->chg_temp < pcisd->data[CISD_DATA_CHG_CHG_TEMP_MIN])
- pcisd->data[CISD_DATA_CHG_CHG_TEMP_MIN] = battery->chg_temp;
- if (battery->wpc_temp > pcisd->data[CISD_DATA_CHG_WPC_TEMP_MAX])
- pcisd->data[CISD_DATA_CHG_WPC_TEMP_MAX] = battery->wpc_temp;
- if (battery->wpc_temp < pcisd->data[CISD_DATA_CHG_WPC_TEMP_MIN])
- pcisd->data[CISD_DATA_CHG_WPC_TEMP_MIN] = battery->wpc_temp;
- if (battery->usb_temp > pcisd->data[CISD_DATA_CHG_USB_TEMP_MAX])
- pcisd->data[CISD_DATA_CHG_USB_TEMP_MAX] = battery->usb_temp;
- if (battery->usb_temp < pcisd->data[CISD_DATA_CHG_USB_TEMP_MIN])
- pcisd->data[CISD_DATA_CHG_USB_TEMP_MIN] = battery->usb_temp;
- if (battery->temperature > pcisd->data[CISD_DATA_CHG_BATT_TEMP_MAX_PER_DAY])
- pcisd->data[CISD_DATA_CHG_BATT_TEMP_MAX_PER_DAY] = battery->temperature;
- if (battery->temperature < pcisd->data[CISD_DATA_CHG_BATT_TEMP_MIN_PER_DAY])
- pcisd->data[CISD_DATA_CHG_BATT_TEMP_MIN_PER_DAY] = battery->temperature;
- if (battery->sub_bat_temp > pcisd->data[CISD_DATA_CHG_SUB_BATT_TEMP_MAX_PER_DAY])
- pcisd->data[CISD_DATA_CHG_SUB_BATT_TEMP_MAX_PER_DAY] = battery->sub_bat_temp;
- if (battery->sub_bat_temp < pcisd->data[CISD_DATA_CHG_SUB_BATT_TEMP_MIN_PER_DAY])
- pcisd->data[CISD_DATA_CHG_SUB_BATT_TEMP_MIN_PER_DAY] = battery->sub_bat_temp;
- if (battery->chg_temp > pcisd->data[CISD_DATA_CHG_CHG_TEMP_MAX_PER_DAY])
- pcisd->data[CISD_DATA_CHG_CHG_TEMP_MAX_PER_DAY] = battery->chg_temp;
- if (battery->chg_temp < pcisd->data[CISD_DATA_CHG_CHG_TEMP_MIN_PER_DAY])
- pcisd->data[CISD_DATA_CHG_CHG_TEMP_MIN_PER_DAY] = battery->chg_temp;
- if (battery->usb_temp > pcisd->data[CISD_DATA_CHG_USB_TEMP_MAX_PER_DAY])
- pcisd->data[CISD_DATA_CHG_USB_TEMP_MAX_PER_DAY] = battery->usb_temp;
- if (battery->usb_temp < pcisd->data[CISD_DATA_CHG_USB_TEMP_MIN_PER_DAY])
- pcisd->data[CISD_DATA_CHG_USB_TEMP_MIN_PER_DAY] = battery->usb_temp;
- if (battery->usb_temp > 800 && !battery->usb_overheat_check) {
- battery->cisd.data[CISD_DATA_USB_OVERHEAT_CHARGING]++;
- battery->cisd.data[CISD_DATA_USB_OVERHEAT_CHARGING_PER_DAY]++;
- battery->usb_overheat_check = true;
- }
- } else {
- /* discharging */
- if (battery->status == POWER_SUPPLY_STATUS_NOT_CHARGING) {
- /* check abnormal vbat */
- pcisd->ab_vbat_check_count = voltage > pcisd->max_voltage_thr ?
- pcisd->ab_vbat_check_count + 1 : 0;
- if ((pcisd->ab_vbat_check_count >= pcisd->ab_vbat_max_count) &&
- !(pcisd->state & CISD_STATE_OVER_VOLTAGE)) {
- pcisd->data[CISD_DATA_VBAT_OVP]++;
- pcisd->data[CISD_DATA_VBAT_OVP_PER_DAY]++;
- pcisd->state |= CISD_STATE_OVER_VOLTAGE;
- #if IS_ENABLED(CONFIG_SEC_ABC)
- #if IS_ENABLED(CONFIG_SEC_FACTORY)
- sec_abc_send_event("MODULE=battery@INFO=over_voltage");
- #else
- sec_abc_send_event("MODULE=battery@WARN=over_voltage");
- #endif
- #endif
- }
- }
- val.intval = SEC_BATTERY_CAPACITY_FULL;
- psy_do_property(battery->pdata->fuelgauge_name, get,
- POWER_SUPPLY_PROP_ENERGY_NOW, val);
- if (val.intval == -1) {
- dev_info(battery->dev, "%s: [CISD] FG I2C fail. skip cisd check\n", __func__);
- return ret;
- }
- if (val.intval > pcisd->data[CISD_DATA_CAP_MAX])
- pcisd->data[CISD_DATA_CAP_MAX] = val.intval;
- if (val.intval < pcisd->data[CISD_DATA_CAP_MIN])
- pcisd->data[CISD_DATA_CAP_MIN] = val.intval;
- if (val.intval > pcisd->data[CISD_DATA_CAP_MAX_PER_DAY])
- pcisd->data[CISD_DATA_CAP_MAX_PER_DAY] = val.intval;
- if (val.intval < pcisd->data[CISD_DATA_CAP_MIN_PER_DAY])
- pcisd->data[CISD_DATA_CAP_MIN_PER_DAY] = val.intval;
- val.intval = SEC_BATTERY_CAPACITY_AGEDCELL;
- psy_do_property(battery->pdata->fuelgauge_name, get,
- POWER_SUPPLY_PROP_ENERGY_NOW, val);
- if (val.intval == -1) {
- dev_info(battery->dev, "%s: [CISD] FG I2C fail. skip cisd check\n", __func__);
- return ret;
- }
- pcisd->data[CISD_DATA_CAP_NOM] = val.intval;
- dev_info(battery->dev, "%s: [CISD] CAP_NOM %dmAh\n", __func__, pcisd->data[CISD_DATA_CAP_NOM]);
- val.intval = SEC_BATTERY_CAPACITY_RC0;
- psy_do_property(battery->pdata->fuelgauge_name, get,
- POWER_SUPPLY_PROP_ENERGY_NOW, val);
- if (val.intval == -1) {
- dev_info(battery->dev, "%s: [CISD] FG I2C fail. skip cisd check\n", __func__);
- return ret;
- }
- pcisd->data[CISD_DATA_RC0] = val.intval;
- dev_info(battery->dev, "%s: [CISD] RC0 0x%x\n", __func__, pcisd->data[CISD_DATA_RC0]);
- }
- if (battery->temperature > pcisd->data[CISD_DATA_BATT_TEMP_MAX])
- pcisd->data[CISD_DATA_BATT_TEMP_MAX] = battery->temperature;
- if (battery->temperature < battery->cisd.data[CISD_DATA_BATT_TEMP_MIN])
- pcisd->data[CISD_DATA_BATT_TEMP_MIN] = battery->temperature;
- if (battery->chg_temp > pcisd->data[CISD_DATA_CHG_TEMP_MAX])
- pcisd->data[CISD_DATA_CHG_TEMP_MAX] = battery->chg_temp;
- if (battery->chg_temp < pcisd->data[CISD_DATA_CHG_TEMP_MIN])
- pcisd->data[CISD_DATA_CHG_TEMP_MIN] = battery->chg_temp;
- if (battery->wpc_temp > pcisd->data[CISD_DATA_WPC_TEMP_MAX])
- pcisd->data[CISD_DATA_WPC_TEMP_MAX] = battery->wpc_temp;
- if (battery->wpc_temp < battery->cisd.data[CISD_DATA_WPC_TEMP_MIN])
- pcisd->data[CISD_DATA_WPC_TEMP_MIN] = battery->wpc_temp;
- if (battery->usb_temp > pcisd->data[CISD_DATA_USB_TEMP_MAX])
- pcisd->data[CISD_DATA_USB_TEMP_MAX] = battery->usb_temp;
- if (battery->usb_temp < pcisd->data[CISD_DATA_USB_TEMP_MIN])
- pcisd->data[CISD_DATA_USB_TEMP_MIN] = battery->usb_temp;
- if (battery->temperature > pcisd->data[CISD_DATA_BATT_TEMP_MAX_PER_DAY])
- pcisd->data[CISD_DATA_BATT_TEMP_MAX_PER_DAY] = battery->temperature;
- if (battery->temperature < pcisd->data[CISD_DATA_BATT_TEMP_MIN_PER_DAY])
- pcisd->data[CISD_DATA_BATT_TEMP_MIN_PER_DAY] = battery->temperature;
- if (battery->sub_bat_temp > pcisd->data[CISD_DATA_SUB_BATT_TEMP_MAX_PER_DAY])
- pcisd->data[CISD_DATA_SUB_BATT_TEMP_MAX_PER_DAY] = battery->sub_bat_temp;
- if (battery->sub_bat_temp < pcisd->data[CISD_DATA_SUB_BATT_TEMP_MIN_PER_DAY])
- pcisd->data[CISD_DATA_SUB_BATT_TEMP_MIN_PER_DAY] = battery->sub_bat_temp;
- if (battery->chg_temp > pcisd->data[CISD_DATA_CHG_TEMP_MAX_PER_DAY])
- pcisd->data[CISD_DATA_CHG_TEMP_MAX_PER_DAY] = battery->chg_temp;
- if (battery->chg_temp < pcisd->data[CISD_DATA_CHG_TEMP_MIN_PER_DAY])
- pcisd->data[CISD_DATA_CHG_TEMP_MIN_PER_DAY] = battery->chg_temp;
- if (battery->usb_temp > pcisd->data[CISD_DATA_USB_TEMP_MAX_PER_DAY])
- pcisd->data[CISD_DATA_USB_TEMP_MAX_PER_DAY] = battery->usb_temp;
- if (battery->usb_temp < pcisd->data[CISD_DATA_USB_TEMP_MIN_PER_DAY])
- pcisd->data[CISD_DATA_USB_TEMP_MIN_PER_DAY] = battery->usb_temp;
- return ret;
- }
- EXPORT_SYMBOL(sec_bat_cisd_check);
- static irqreturn_t cisd_irq_thread(int irq, void *data)
- {
- struct cisd *pcisd = data;
- pr_info("%s: irq(%d)\n", __func__, irq);
- if (irq == pcisd->irq_ovp_power &&
- !gpio_get_value(pcisd->gpio_ovp_power))
- pcisd->event_data[EVENT_OVP_POWER]++;
- if (irq == pcisd->irq_ovp_signal &&
- !gpio_get_value(pcisd->gpio_ovp_signal))
- pcisd->event_data[EVENT_OVP_SIGNAL]++;
- return IRQ_HANDLED;
- }
- #ifdef CONFIG_OF
- static void sec_cisd_parse_dt(struct cisd *pcisd)
- {
- struct device_node *np;
- int ret = 0;
- np = of_find_node_by_name(NULL, "sec-cisd");
- if (!np) {
- pr_err("%s: np NULL\n", __func__);
- return;
- }
- ret = of_get_named_gpio(np, "ovp_power", 0);
- if (ret >= 0) {
- pcisd->gpio_ovp_power = ret;
- pr_info("%s: set ovp_power gpio(%d)\n", __func__, pcisd->gpio_ovp_power);
- pcisd->irq_ovp_power = gpio_to_irq(pcisd->gpio_ovp_power);
- ret = request_threaded_irq(pcisd->irq_ovp_power, NULL,
- cisd_irq_thread, IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
- "cisd-ovp-power", pcisd);
- if (ret < 0) {
- pr_err("%s: failed to request ovp_power irq(ret = %d)\n",
- __func__, ret);
- pcisd->irq_ovp_power = 0;
- } else
- pr_info("%s: set irq_ovp_power(%d)\n", __func__, pcisd->irq_ovp_power);
- } else
- pr_err("%s: failed to get ovp_power\n", __func__);
- ret = of_get_named_gpio(np, "ovp_signal", 0);
- if (ret >= 0) {
- pcisd->gpio_ovp_signal = ret;
- pr_info("%s: set ovp_signal gpio(%d)\n", __func__, pcisd->gpio_ovp_signal);
- pcisd->irq_ovp_signal = gpio_to_irq(pcisd->gpio_ovp_signal);
- ret = request_threaded_irq(pcisd->irq_ovp_signal, NULL,
- cisd_irq_thread, IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
- "cisd-ovp-signal", pcisd);
- if (ret < 0) {
- pr_err("%s: failed to request ovp_signal irq(ret = %d)\n",
- __func__, ret);
- pcisd->irq_ovp_signal = 0;
- } else
- pr_info("%s: set irq_ovp_signal(%d)\n", __func__, pcisd->irq_ovp_signal);
- } else
- pr_err("%s: failed to get ovp_signal\n", __func__);
- }
- #else
- static void sec_cisd_parse_dt(struct cisd *pcisd)
- {
- }
- #endif
- struct cisd *gcisd;
- EXPORT_SYMBOL(gcisd);
- void sec_battery_cisd_init(struct sec_battery_info *battery)
- {
- /* parse dt */
- sec_cisd_parse_dt(&battery->cisd);
- /* init cisd data */
- battery->cisd.state = CISD_STATE_NONE;
- battery->cisd.data[CISD_DATA_ALG_INDEX] = battery->pdata->cisd_alg_index;
- battery->cisd.data[CISD_DATA_FULL_COUNT] = 1;
- battery->cisd.data[CISD_DATA_BATT_TEMP_MAX] = -300;
- battery->cisd.data[CISD_DATA_CHG_TEMP_MAX] = -300;
- battery->cisd.data[CISD_DATA_WPC_TEMP_MAX] = -300;
- battery->cisd.data[CISD_DATA_USB_TEMP_MAX] = -300;
- battery->cisd.data[CISD_DATA_BATT_TEMP_MIN] = 1000;
- battery->cisd.data[CISD_DATA_CHG_TEMP_MIN] = 1000;
- battery->cisd.data[CISD_DATA_WPC_TEMP_MIN] = 1000;
- battery->cisd.data[CISD_DATA_USB_TEMP_MIN] = 1000;
- battery->cisd.data[CISD_DATA_CHG_BATT_TEMP_MAX] = -300;
- battery->cisd.data[CISD_DATA_CHG_CHG_TEMP_MAX] = -300;
- battery->cisd.data[CISD_DATA_CHG_WPC_TEMP_MAX] = -300;
- battery->cisd.data[CISD_DATA_CHG_USB_TEMP_MAX] = -300;
- battery->cisd.data[CISD_DATA_CHG_BATT_TEMP_MIN] = 1000;
- battery->cisd.data[CISD_DATA_CHG_CHG_TEMP_MIN] = 1000;
- battery->cisd.data[CISD_DATA_CHG_WPC_TEMP_MIN] = 1000;
- battery->cisd.data[CISD_DATA_CHG_USB_TEMP_MIN] = 1000;
- battery->cisd.data[CISD_DATA_CAP_MIN] = 0xFFFF;
- battery->cisd.data[CISD_DATA_ASOC] = 100;
- battery->cisd.data[CISD_DATA_FULL_COUNT_PER_DAY] = 1;
- battery->cisd.data[CISD_DATA_BATT_TEMP_MAX_PER_DAY] = -300;
- battery->cisd.data[CISD_DATA_SUB_BATT_TEMP_MAX_PER_DAY] = -300;
- battery->cisd.data[CISD_DATA_CHG_TEMP_MAX_PER_DAY] = -300;
- battery->cisd.data[CISD_DATA_USB_TEMP_MAX_PER_DAY] = -300;
- battery->cisd.data[CISD_DATA_BATT_TEMP_MIN_PER_DAY] = 1000;
- battery->cisd.data[CISD_DATA_SUB_BATT_TEMP_MIN_PER_DAY] = 1000;
- battery->cisd.data[CISD_DATA_CHG_TEMP_MIN_PER_DAY] = 1000;
- battery->cisd.data[CISD_DATA_USB_TEMP_MIN_PER_DAY] = 1000;
- battery->cisd.data[CISD_DATA_CHG_BATT_TEMP_MAX_PER_DAY] = -300;
- battery->cisd.data[CISD_DATA_CHG_SUB_BATT_TEMP_MAX_PER_DAY] = -300;
- battery->cisd.data[CISD_DATA_CHG_CHG_TEMP_MAX_PER_DAY] = -300;
- battery->cisd.data[CISD_DATA_CHG_USB_TEMP_MAX_PER_DAY] = -300;
- battery->cisd.data[CISD_DATA_CHG_BATT_TEMP_MIN_PER_DAY] = 1000;
- battery->cisd.data[CISD_DATA_CHG_SUB_BATT_TEMP_MIN_PER_DAY] = 1000;
- battery->cisd.data[CISD_DATA_CHG_CHG_TEMP_MIN_PER_DAY] = 1000;
- battery->cisd.data[CISD_DATA_CHG_USB_TEMP_MIN_PER_DAY] = 1000;
- battery->cisd.data[CISD_DATA_CHG_RETENTION_TIME_PER_DAY] = 0;
- battery->cisd.data[CISD_DATA_TOTAL_CHG_RETENTION_TIME_PER_DAY] = 0;
- battery->cisd.ab_vbat_max_count = 2; /* should be 2 */
- battery->cisd.ab_vbat_check_count = 0;
- battery->cisd.max_voltage_thr = battery->pdata->max_voltage_thr;
- /* set cisd pointer */
- gcisd = &battery->cisd;
- /* initialize pad data */
- mutex_init(&battery->cisd.padlock);
- mutex_init(&battery->cisd.powerlock);
- mutex_init(&battery->cisd.pdlock);
- init_cisd_pad_data(&battery->cisd);
- init_cisd_power_data(&battery->cisd);
- init_cisd_pd_data(&battery->cisd);
- }
- EXPORT_SYMBOL(sec_battery_cisd_init);
- static struct pad_data *create_pad_data(unsigned int pad_id, unsigned int pad_count)
- {
- struct pad_data *temp_data;
- temp_data = kzalloc(sizeof(struct pad_data), GFP_KERNEL);
- if (temp_data == NULL)
- return NULL;
- temp_data->id = pad_id;
- temp_data->count = pad_count;
- temp_data->prev = temp_data->next = NULL;
- return temp_data;
- }
- static struct pad_data *find_pad_data_by_id(struct cisd *cisd, unsigned int pad_id)
- {
- struct pad_data *temp_data = cisd->pad_array->next;
- if (cisd->pad_count <= 0 || temp_data == NULL)
- return NULL;
- while ((temp_data->id != pad_id) &&
- ((temp_data = temp_data->next) != NULL));
- return temp_data;
- }
- static void add_pad_data(struct cisd *cisd, unsigned int pad_id, unsigned int pad_count)
- {
- struct pad_data *temp_data = cisd->pad_array->next;
- struct pad_data *pad_data;
- if (pad_id >= MAX_PAD_ID)
- return;
- pad_data = create_pad_data(pad_id, pad_count);
- if (pad_data == NULL)
- return;
- pr_info("%s: id(0x%x), count(%d)\n", __func__, pad_id, pad_count);
- while (temp_data) {
- if (temp_data->id > pad_id) {
- temp_data->prev->next = pad_data;
- pad_data->prev = temp_data->prev;
- pad_data->next = temp_data;
- temp_data->prev = pad_data;
- cisd->pad_count++;
- return;
- }
- temp_data = temp_data->next;
- }
- pr_info("%s: failed to add pad_data(%d, %d)\n",
- __func__, pad_id, pad_count);
- kfree(pad_data);
- }
- void init_cisd_pad_data(struct cisd *cisd)
- {
- struct pad_data *temp_data = NULL;
- mutex_lock(&cisd->padlock);
- temp_data = cisd->pad_array;
- while (temp_data) {
- struct pad_data *next_data = temp_data->next;
- kfree(temp_data);
- temp_data = next_data;
- }
- /* create dummy data */
- cisd->pad_array = create_pad_data(0, 0);
- if (cisd->pad_array == NULL)
- goto err_create_dummy_data;
- temp_data = create_pad_data(MAX_PAD_ID, 0);
- if (temp_data == NULL) {
- kfree(cisd->pad_array);
- cisd->pad_array = NULL;
- goto err_create_dummy_data;
- }
- cisd->pad_count = 0;
- cisd->pad_array->next = temp_data;
- temp_data->prev = cisd->pad_array;
- err_create_dummy_data:
- mutex_unlock(&cisd->padlock);
- }
- EXPORT_SYMBOL(init_cisd_pad_data);
- void count_cisd_pad_data(struct cisd *cisd, unsigned int pad_id)
- {
- struct pad_data *pad_data;
- if (cisd->pad_array == NULL) {
- pr_info("%s: can't update the connected count of pad_id(0x%x) because of null\n",
- __func__, pad_id);
- return;
- }
- mutex_lock(&cisd->padlock);
- pad_data = find_pad_data_by_id(cisd, pad_id);
- if (pad_data != NULL)
- pad_data->count++;
- else
- add_pad_data(cisd, pad_id, 1);
- mutex_unlock(&cisd->padlock);
- }
- EXPORT_SYMBOL(count_cisd_pad_data);
- void set_cisd_pad_data(struct sec_battery_info *battery, const char *buf)
- {
- struct cisd *pcisd = &battery->cisd;
- unsigned int pad_total_count, pad_id, pad_count;
- struct pad_data *pad_data;
- int i, x;
- pr_info("%s: %s\n", __func__, buf);
- if (pcisd->pad_count > 0)
- init_cisd_pad_data(pcisd);
- if (pcisd->pad_array == NULL) {
- pr_info("%s: can't set the pad data because of null\n", __func__);
- return;
- }
- if (sscanf(buf, "%10u %n", &pad_total_count, &x) <= 0) {
- pr_info("%s: failed to read pad index\n", __func__);
- return;
- }
- buf += (size_t)x;
- pr_info("%s: add pad data(count: %d)\n", __func__, pad_total_count);
- for (i = 0; i < pad_total_count; i++) {
- if (sscanf(buf, "0x%02x:%10u %n", &pad_id, &pad_count, &x) != 2) {
- pr_info("%s: failed to read pad data(0x%x, %d, %d)!!!re-init pad data\n",
- __func__, pad_id, pad_count, x);
- init_cisd_pad_data(pcisd);
- break;
- }
- buf += (size_t)x;
- mutex_lock(&pcisd->padlock);
- pad_data = find_pad_data_by_id(pcisd, pad_id);
- if (pad_data != NULL)
- pad_data->count = pad_count;
- else
- add_pad_data(pcisd, pad_id, pad_count);
- mutex_unlock(&pcisd->padlock);
- }
- }
- EXPORT_SYMBOL(set_cisd_pad_data);
- static struct power_data *create_power_data(unsigned int power, unsigned int power_count)
- {
- struct power_data *temp_data;
- temp_data = kzalloc(sizeof(struct power_data), GFP_KERNEL);
- if (temp_data == NULL)
- return NULL;
- temp_data->power = power;
- temp_data->count = power_count;
- temp_data->prev = temp_data->next = NULL;
- return temp_data;
- }
- static struct power_data *find_data_by_power(struct cisd *cisd, unsigned int power)
- {
- struct power_data *temp_data = cisd->power_array->next;
- if (cisd->power_count <= 0 || temp_data == NULL)
- return NULL;
- while ((temp_data->power != power) &&
- ((temp_data = temp_data->next) != NULL));
- return temp_data;
- }
- static void add_power_data(struct cisd *cisd, unsigned int power, unsigned int power_count)
- {
- struct power_data *temp_data = cisd->power_array->next;
- struct power_data *power_data;
- power_data = create_power_data(power, power_count);
- if (power_data == NULL)
- return;
- pr_info("%s: power(%d), count(%d)\n", __func__, power, power_count);
- while (temp_data) {
- if (temp_data->power > power) {
- temp_data->prev->next = power_data;
- power_data->prev = temp_data->prev;
- power_data->next = temp_data;
- temp_data->prev = power_data;
- cisd->power_count++;
- return;
- }
- temp_data = temp_data->next;
- }
- pr_info("%s: failed to add pad_data(%d, %d)\n",
- __func__, power, power_count);
- kfree(power_data);
- }
- void init_cisd_power_data(struct cisd *cisd)
- {
- struct power_data *temp_data = NULL;
- mutex_lock(&cisd->powerlock);
- temp_data = cisd->power_array;
- while (temp_data) {
- struct power_data *next_data = temp_data->next;
- kfree(temp_data);
- temp_data = next_data;
- }
- /* create dummy data */
- cisd->power_array = create_power_data(0, 0);
- if (cisd->power_array == NULL)
- goto err_create_dummy_data;
- temp_data = create_power_data(MAX_CHARGER_POWER, 0);
- if (temp_data == NULL) {
- kfree(cisd->power_array);
- cisd->power_array = NULL;
- goto err_create_dummy_data;
- }
- cisd->power_count = 0;
- cisd->power_array->next = temp_data;
- temp_data->prev = cisd->power_array;
- err_create_dummy_data:
- mutex_unlock(&cisd->powerlock);
- }
- EXPORT_SYMBOL(init_cisd_power_data);
- #define FIND_MAX_POWER 45000
- #define FIND_POWER_STEP 10000
- #define POWER_MARGIN 1000
- void count_cisd_power_data(struct cisd *cisd, int power)
- {
- struct power_data *power_data;
- int power_index = 0;
- pr_info("%s: power value : %d\n", __func__, power);
- if (cisd->power_array == NULL || power < 15000) {
- pr_info("%s: can't update the connected count of power(%d) because of null\n",
- __func__, power);
- return;
- }
- power_index = FIND_MAX_POWER;
- while (power_index >= 14000) {
- if (power + POWER_MARGIN - power_index >= 0) {
- power_index /= 1000;
- break;
- }
- power_index -= FIND_POWER_STEP;
- }
- mutex_lock(&cisd->powerlock);
- power_data = find_data_by_power(cisd, power_index);
- if (power_data != NULL)
- power_data->count++;
- else
- add_power_data(cisd, power_index, 1);
- mutex_unlock(&cisd->powerlock);
- }
- EXPORT_SYMBOL(count_cisd_power_data);
- void set_cisd_power_data(struct sec_battery_info *battery, const char *buf)
- {
- struct cisd *pcisd = &battery->cisd;
- unsigned int power_total_count, power_id, power_count;
- struct power_data *power_data;
- int i, x;
- pr_info("%s: %s\n", __func__, buf);
- if (pcisd->power_count > 0)
- init_cisd_power_data(pcisd);
- if (pcisd->power_array == NULL) {
- pr_info("%s: can't set the power data because of null\n", __func__);
- return;
- }
- if (sscanf(buf, "%10u %n", &power_total_count, &x) <= 0)
- return;
- buf += (size_t)x;
- pr_info("%s: add power data(count: %d)\n", __func__, power_total_count);
- for (i = 0; i < power_total_count; i++) {
- if (sscanf(buf, "%10u:%10u %n", &power_id, &power_count, &x) != 2) {
- pr_info("%s: failed to read power data(%d, %d, %d)!!!re-init power data\n",
- __func__, power_id, power_count, x);
- init_cisd_power_data(pcisd);
- break;
- }
- buf += (size_t)x;
- mutex_lock(&pcisd->powerlock);
- power_data = find_data_by_power(pcisd, power_id);
- if (power_data != NULL)
- power_data->count = power_count;
- else
- add_power_data(pcisd, power_id, power_count);
- mutex_unlock(&pcisd->powerlock);
- }
- }
- EXPORT_SYMBOL(set_cisd_power_data);
- static struct pd_data *create_pd_data(unsigned short pid, unsigned int pd_count)
- {
- struct pd_data *temp_data;
- temp_data = kzalloc(sizeof(struct pd_data), GFP_KERNEL);
- if (temp_data == NULL)
- return NULL;
- temp_data->pid = pid;
- temp_data->count = pd_count;
- temp_data->prev = temp_data->next = NULL;
- return temp_data;
- }
- static struct pd_data *find_data_by_pd(struct cisd *cisd, unsigned short pid)
- {
- struct pd_data *temp_data = cisd->pd_array->next;
- if (cisd->pd_count <= 0 || temp_data == NULL)
- return NULL;
- while ((temp_data->pid != pid) &&
- ((temp_data = temp_data->next) != NULL))
- ;
- return temp_data;
- }
- static void add_pd_data(struct cisd *cisd, unsigned short pid, unsigned int pd_count)
- {
- struct pd_data *temp_data = cisd->pd_array->next;
- struct pd_data *pd_data;
- pd_data = create_pd_data(pid, pd_count);
- if (pd_data == NULL)
- return;
- pr_info("%s: pid(0x%04x), count(%d)\n", __func__, pid, pd_count);
- while (temp_data) {
- if (temp_data->pid > pid) {
- temp_data->prev->next = pd_data;
- pd_data->prev = temp_data->prev;
- pd_data->next = temp_data;
- temp_data->prev = pd_data;
- cisd->pd_count++;
- return;
- }
- temp_data = temp_data->next;
- }
- pr_info("%s: failed to add pd_data(0x%04x, %d)\n",
- __func__, pid, pd_count);
- kfree(pd_data);
- }
- void init_cisd_pd_data(struct cisd *cisd)
- {
- struct pd_data *temp_data = NULL;
- mutex_lock(&cisd->pdlock);
- temp_data = cisd->pd_array;
- while (temp_data) {
- struct pd_data *next_data = temp_data->next;
- kfree(temp_data);
- temp_data = next_data;
- }
- /* create dummy data */
- cisd->pd_array = create_pd_data(0, 0);
- if (cisd->pd_array == NULL)
- goto err_create_dummy_data;
- temp_data = create_pd_data(MAX_SS_PD_PID, 0);
- if (temp_data == NULL) {
- kfree(cisd->pd_array);
- cisd->pd_array = NULL;
- goto err_create_dummy_data;
- }
- cisd->pd_count = 0;
- cisd->pd_array->next = temp_data;
- temp_data->prev = cisd->pd_array;
- err_create_dummy_data:
- mutex_unlock(&cisd->pdlock);
- }
- EXPORT_SYMBOL(init_cisd_pd_data);
- void count_cisd_pd_data(unsigned short vid, unsigned short pid)
- {
- struct power_supply *psy = power_supply_get_by_name("battery");
- struct sec_battery_info *battery = power_supply_get_drvdata(psy);
- struct cisd *cisd = &battery->cisd;
- struct pd_data *pd_data;
- pr_info("%s: vid : 0x%04x, pid : 0x%04x\n", __func__, vid, pid);
- if (cisd->pd_array == NULL) {
- pr_info("%s: can't update count of pid(0x%04x) because of null\n",
- __func__, pid);
- return;
- }
- if ((vid != SS_PD_VID) || ((vid == SS_PD_VID)
- && (pid < MIN_SS_PD_PID || pid > MAX_SS_PD_PID))) {
- pr_info("%s: other pd ta(vid_0x%04x, pid_0x%04x). change pid to 0x0000\n",
- __func__, vid, pid);
- pid = 0x0;
- }
- mutex_lock(&cisd->pdlock);
- pd_data = find_data_by_pd(cisd, pid);
- if (pd_data != NULL)
- pd_data->count++;
- else
- add_pd_data(cisd, pid, 1);
- mutex_unlock(&cisd->pdlock);
- }
- EXPORT_SYMBOL(count_cisd_pd_data);
- void set_cisd_pd_data(struct sec_battery_info *battery, const char *buf)
- {
- struct cisd *pcisd = &battery->cisd;
- unsigned int pd_total_count, pid, pd_count;
- struct pd_data *pd_data;
- int i, x;
- pr_info("%s: %s\n", __func__, buf);
- if (pcisd->pd_count > 0)
- init_cisd_pd_data(pcisd);
- if (pcisd->pd_array == NULL) {
- pr_info("%s: can't set the pd data because of null\n", __func__);
- return;
- }
- if (sscanf(buf, "%10u %n", &pd_total_count, &x) <= 0)
- return;
- buf += (size_t)x;
- pr_info("%s: add pd data(count: %d)\n", __func__, pd_total_count);
- for (i = 0; i < pd_total_count; i++) {
- if (sscanf(buf, "0x%04x:%10u %n", &pid, &pd_count, &x) != 2) {
- pr_info("%s: failed to read pd data(0x%04x, %d, %d)!!!re-init pd data\n",
- __func__, pid, pd_count, x);
- init_cisd_pd_data(pcisd);
- break;
- }
- buf += (size_t)x;
- mutex_lock(&pcisd->pdlock);
- pd_data = find_data_by_pd(pcisd, pid);
- if (pd_data != NULL)
- pd_data->count = pd_count;
- else
- add_pd_data(pcisd, pid, pd_count);
- mutex_unlock(&pcisd->pdlock);
- }
- }
- EXPORT_SYMBOL(set_cisd_pd_data);
|