| /* |
| * Google Battery Management System |
| * |
| * Copyright (C) 2018 Google Inc. |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| */ |
| |
| #define gbms_owner(p) ((p)->owner_name ? (p)->owner_name : "google_bms") |
| |
| #define gbms_info(p, fmt, ...) \ |
| pr_info("%s: " fmt, gbms_owner(p), ##__VA_ARGS__) |
| #define gbms_warn(p, fmt, ...) \ |
| pr_warn("%s: " fmt, gbms_owner(p), ##__VA_ARGS__) |
| #define gbms_err(p, fmt, ...) \ |
| pr_err("%s: " fmt, gbms_owner(p), ##__VA_ARGS__) |
| |
| #include <linux/kernel.h> |
| #include <linux/printk.h> |
| #include <linux/module.h> |
| #include <linux/slab.h> |
| #include <linux/of.h> |
| #include <linux/regmap.h> |
| |
| #include "google_psy.h" |
| #include "google_bms.h" |
| |
| /* sync from google/logbuffer.c */ |
| #define LOG_BUFFER_ENTRY_SIZE 256 |
| |
| #define GBMS_DEFAULT_FV_UV_RESOLUTION 25000 |
| #define GBMS_DEFAULT_FV_UV_MARGIN_DPCT 1020 |
| #define GBMS_DEFAULT_FV_DC_RATIO 20 |
| #define GBMS_DEFAULT_CV_DEBOUNCE_CNT 3 |
| #define GBMS_DEFAULT_CV_UPDATE_INTERVAL 2000 |
| #define GBMS_DEFAULT_CV_TIER_OV_CNT 10 |
| #define GBMS_DEFAULT_CV_TIER_SWITCH_CNT 3 |
| #define GBMS_DEFAULT_CV_OTV_MARGIN 0 |
| |
| /* same as POWER_SUPPLY_CHARGE_TYPE_TEXT */ |
| static const char *psy_chgt_str[] = { |
| [POWER_SUPPLY_CHARGE_TYPE_UNKNOWN] = "Unknown", |
| [POWER_SUPPLY_CHARGE_TYPE_NONE] = "N/A", |
| [POWER_SUPPLY_CHARGE_TYPE_TRICKLE] = "Trickle", |
| [POWER_SUPPLY_CHARGE_TYPE_FAST] = "Fast", |
| [POWER_SUPPLY_CHARGE_TYPE_STANDARD] = "Standard", |
| [POWER_SUPPLY_CHARGE_TYPE_ADAPTIVE] = "Adaptive", |
| [POWER_SUPPLY_CHARGE_TYPE_CUSTOM] = "Custom", |
| [POWER_SUPPLY_CHARGE_TYPE_LONGLIFE] = "Long Life", |
| [POWER_SUPPLY_CHARGE_TYPE_TAPER] = "Taper", |
| }; |
| |
| const char *gbms_chg_type_s(int cgh_type) |
| { |
| if (cgh_type < 0 || cgh_type > ARRAY_SIZE(psy_chgt_str)) |
| return "<err>"; |
| return psy_chgt_str[cgh_type]; |
| } |
| EXPORT_SYMBOL_GPL(gbms_chg_type_s); |
| |
| /* same as POWER_SUPPLY_STATUS_TEXT */ |
| static const char *psy_chgs_str[] = { |
| "Unknown", "Charging", "Discharging", "Not Charging", "Full" |
| }; |
| |
| const char *gbms_chg_status_s(int chg_status) |
| { |
| if (chg_status < 0 || chg_status > ARRAY_SIZE(psy_chgs_str)) |
| return "<err>"; |
| return psy_chgs_str[chg_status]; |
| } |
| EXPORT_SYMBOL_GPL(gbms_chg_status_s); |
| |
| |
| const char *gbms_chg_ev_adapter_s(int adapter) |
| { |
| static char *chg_ev_adapter_type_str[] = { |
| FOREACH_CHG_EV_ADAPTER(CHG_EV_ADAPTER_STRING) |
| }; |
| |
| if (adapter < 0 || adapter > ARRAY_SIZE(chg_ev_adapter_type_str)) |
| return "<err>"; |
| return chg_ev_adapter_type_str[adapter]; |
| } |
| EXPORT_SYMBOL_GPL(gbms_chg_ev_adapter_s); |
| |
| static const char *gbms_get_code(const int index) |
| { |
| const static char *codes[] = {"n", "s", "d", "l", "v", "vo", "p", "f", |
| "t", "dl", "st", "tc", "r", "w", "rs", |
| "n", "ny", "h", "hp", "ha"}; |
| const int len = ARRAY_SIZE(codes); |
| |
| return (index >= 0 && index < len) ? codes[index] : "?"; |
| } |
| |
| /* convert C rates to current. Caller can account for tolerances reducing |
| * battery_capacity. fv_uv_resolution is used to create discrete steps. |
| * NOTE: the call covert C rates to chanrge currents IN PLACE, ie you cannot |
| * call this twice. |
| */ |
| void gbms_init_chg_table(struct gbms_chg_profile *profile, |
| struct device_node *node, u32 capacity_ma) |
| { |
| u32 ccm; |
| int vi, ti, ret; |
| const int fv_uv_step = profile->fv_uv_resolution; |
| u32 cccm_array_size = (profile->temp_nb_limits - 1) |
| * profile->volt_nb_limits; |
| |
| profile->capacity_ma = capacity_ma; |
| |
| ret = of_property_read_u32_array(node, "google,chg-cc-limits", |
| profile->cccm_limits, |
| cccm_array_size); |
| if (ret < 0) |
| pr_warn("unable to get default cccm_limits.\n"); |
| |
| /* chg-battery-capacity is in mAh, chg-cc-limits relative to 100 */ |
| for (ti = 0; ti < profile->temp_nb_limits - 1; ti++) { |
| for (vi = 0; vi < profile->volt_nb_limits; vi++) { |
| ccm = GBMS_CCCM_LIMITS(profile, ti, vi); |
| ccm *= capacity_ma * 10; |
| |
| /* round to the nearest resolution */ |
| if (fv_uv_step) |
| ccm = DIV_ROUND_CLOSEST(ccm, fv_uv_step) |
| * fv_uv_step; |
| |
| GBMS_CCCM_LIMITS_SET(profile, ti, vi) = ccm; |
| } |
| } |
| } |
| EXPORT_SYMBOL_GPL(gbms_init_chg_table); |
| |
| /* configure standard device charge profile properties */ |
| static int gbms_read_cccm_limits(struct gbms_chg_profile *profile, |
| struct device_node *node) |
| { |
| int ret; |
| |
| profile->temp_nb_limits = |
| of_property_count_elems_of_size(node, "google,chg-temp-limits", |
| sizeof(u32)); |
| if (profile->temp_nb_limits <= 0) { |
| ret = profile->temp_nb_limits; |
| gbms_err(profile, "cannot read chg-temp-limits, ret=%d\n", ret); |
| return ret; |
| } |
| if (profile->temp_nb_limits > GBMS_CHG_TEMP_NB_LIMITS_MAX) { |
| gbms_err(profile, "chg-temp-nb-limits exceeds driver max: %d\n", |
| GBMS_CHG_TEMP_NB_LIMITS_MAX); |
| return -EINVAL; |
| } |
| ret = of_property_read_u32_array(node, "google,chg-temp-limits", |
| (u32 *)profile->temp_limits, |
| profile->temp_nb_limits); |
| if (ret < 0) { |
| gbms_err(profile, "cannot read chg-temp-limits table, ret=%d\n", |
| ret); |
| return ret; |
| } |
| |
| profile->volt_nb_limits = |
| of_property_count_elems_of_size(node, "google,chg-cv-limits", |
| sizeof(u32)); |
| if (profile->volt_nb_limits <= 0) { |
| ret = profile->volt_nb_limits; |
| gbms_err(profile, "cannot read chg-cv-limits, ret=%d\n", ret); |
| return ret; |
| } |
| if (profile->volt_nb_limits > GBMS_CHG_VOLT_NB_LIMITS_MAX) { |
| gbms_err(profile, "chg-cv-nb-limits exceeds driver max: %d\n", |
| GBMS_CHG_VOLT_NB_LIMITS_MAX); |
| return -EINVAL; |
| } |
| ret = of_property_read_u32_array(node, "google,chg-cv-limits", |
| (u32 *)profile->volt_limits, |
| profile->volt_nb_limits); |
| if (ret < 0) { |
| gbms_err(profile, "cannot read chg-cv-limits table, ret=%d\n", |
| ret); |
| return ret; |
| } |
| |
| memset(profile->topoff_limits, 0, sizeof(profile->topoff_limits)); |
| profile->topoff_nb_limits = |
| of_property_count_elems_of_size(node, "google,chg-topoff-limits", |
| sizeof(u32)); |
| if (profile->topoff_nb_limits > 0) { |
| if (profile->topoff_nb_limits > GBMS_CHG_TOPOFF_NB_LIMITS_MAX) { |
| gbms_err(profile, "chg-topoff-nb-limits exceeds driver max:%d\n", |
| GBMS_CHG_TOPOFF_NB_LIMITS_MAX); |
| return -EINVAL; |
| } |
| ret = of_property_read_u32_array(node, "google,chg-topoff-limits", |
| (u32 *)profile->topoff_limits, |
| profile->topoff_nb_limits); |
| if (ret < 0) { |
| gbms_err(profile, "cannot read chg-topoff-limits table, ret=%d\n", |
| ret); |
| return ret; |
| } |
| gbms_info(profile, "dynamic topoff enabled\n"); |
| } |
| |
| return 0; |
| } |
| |
| int gbms_read_aacr_limits(struct gbms_chg_profile *profile, |
| struct device_node *node) |
| { |
| int ret = 0, cycle_nb_limits = 0, fade10_nb_limits = 0; |
| |
| if (!profile || !node) |
| return -ENODEV; |
| |
| ret = of_property_count_elems_of_size(node, |
| "google,aacr-ref-cycles", sizeof(u32)); |
| if (ret < 0) |
| goto no_data; |
| |
| cycle_nb_limits = ret; |
| |
| ret = of_property_count_elems_of_size(node, |
| "google,aacr-ref-fade10", sizeof(u32)); |
| if (ret < 0) |
| goto no_data; |
| |
| fade10_nb_limits = ret; |
| |
| if (cycle_nb_limits != fade10_nb_limits || |
| cycle_nb_limits > GBMS_AACR_DATA_MAX || |
| cycle_nb_limits == 0) { |
| gbms_warn(profile, "aacr not enable, cycle_nb:%d, fade10_nb:%d, max:%d", |
| cycle_nb_limits, fade10_nb_limits, GBMS_AACR_DATA_MAX); |
| profile->aacr_nb_limits = 0; |
| return -ERANGE; |
| } |
| |
| ret = of_property_read_u32_array(node, "google,aacr-ref-cycles", |
| (u32 *)profile->reference_cycles, cycle_nb_limits); |
| if (ret < 0) |
| return ret; |
| |
| ret = of_property_read_u32_array(node, "google,aacr-ref-fade10", |
| (u32 *)profile->reference_fade10, fade10_nb_limits); |
| if (ret < 0) |
| return ret; |
| |
| profile->aacr_nb_limits = cycle_nb_limits; |
| |
| return 0; |
| |
| no_data: |
| profile->aacr_nb_limits = 0; |
| return ret; |
| } |
| EXPORT_SYMBOL_GPL(gbms_read_aacr_limits); |
| |
| /* return the pct amount of capacity fade at cycles or negative if not enabled */ |
| int gbms_aacr_fade10(const struct gbms_chg_profile *profile, int cycles) |
| { |
| int cycle_s = 0, fade_s = 0; |
| int idx, cycle_f, fade_f; |
| |
| if (profile->aacr_nb_limits == 0 || cycles < 0) |
| return -EINVAL; |
| |
| for (idx = 0; idx < profile->aacr_nb_limits - 1; idx++) |
| if (cycles < profile->reference_cycles[idx]) |
| break; |
| |
| /* Interpolation */ |
| cycle_f = profile->reference_cycles[idx]; |
| fade_f = profile->reference_fade10[idx]; |
| if (idx > 0) { |
| cycle_s = profile->reference_cycles[idx - 1]; |
| fade_s = profile->reference_fade10[idx - 1]; |
| } |
| |
| return (cycles - cycle_s) * (fade_f - fade_s) / (cycle_f - cycle_s) + fade_s; |
| } |
| EXPORT_SYMBOL_GPL(gbms_aacr_fade10); |
| |
| int gbms_init_chg_profile_internal(struct gbms_chg_profile *profile, |
| struct device_node *node, |
| const char *owner_name) |
| { |
| int ret, vi; |
| u32 cccm_array_size, mem_size; |
| |
| profile->owner_name = owner_name; |
| |
| ret = gbms_read_cccm_limits(profile, node); |
| if (ret < 0) |
| return ret; |
| |
| cccm_array_size = (profile->temp_nb_limits - 1) |
| * profile->volt_nb_limits; |
| mem_size = sizeof(s32) * cccm_array_size; |
| |
| profile->cccm_limits = kzalloc(mem_size, GFP_KERNEL); |
| if (!profile->cccm_limits) |
| return -ENOMEM; |
| |
| /* load C rates into profile->cccm_limits */ |
| ret = of_property_read_u32_array(node, "google,chg-cc-limits", |
| profile->cccm_limits, |
| cccm_array_size); |
| if (ret < 0) { |
| gbms_err(profile, "cannot read chg-cc-limits table, ret=%d\n", |
| ret); |
| kfree(profile->cccm_limits); |
| profile->cccm_limits = 0; |
| return -EINVAL; |
| } |
| |
| /* for irdrop compensation in taper step */ |
| ret = of_property_read_u32(node, "google,fv-uv-resolution", |
| &profile->fv_uv_resolution); |
| if (ret < 0) |
| profile->fv_uv_resolution = GBMS_DEFAULT_FV_UV_RESOLUTION; |
| |
| /* how close to tier voltage is close enough */ |
| ret = of_property_read_u32(node, "google,cv-range-accuracy", |
| &profile->cv_range_accuracy); |
| if (ret < 0) |
| profile->cv_range_accuracy = profile->fv_uv_resolution / 2; |
| |
| /* IEEE1725, default to 1020, cap irdrop offset */ |
| ret = of_property_read_u32(node, "google,fv-uv-margin-dpct", |
| &profile->fv_uv_margin_dpct); |
| if (ret < 0) |
| profile->fv_uv_margin_dpct = GBMS_DEFAULT_FV_UV_MARGIN_DPCT; |
| |
| ret = of_property_read_u32(node, "google,fv-dc-ratio", |
| &profile->fv_dc_ratio); |
| if (ret < 0) |
| profile->fv_dc_ratio = GBMS_DEFAULT_FV_DC_RATIO; |
| |
| /* debounce tier switch */ |
| ret = of_property_read_u32(node, "google,cv-debounce-cnt", |
| &profile->cv_debounce_cnt); |
| if (ret < 0) |
| profile->cv_debounce_cnt = GBMS_DEFAULT_CV_DEBOUNCE_CNT; |
| |
| /* how fast to poll in taper */ |
| ret = of_property_read_u32(node, "google,cv-update-interval", |
| &profile->cv_update_interval); |
| if (ret < 0) |
| profile->cv_update_interval = GBMS_DEFAULT_CV_UPDATE_INTERVAL; |
| |
| /* tier over voltage penalty */ |
| ret = of_property_read_u32(node, "google,cv-tier-ov-cnt", |
| &profile->cv_tier_ov_cnt); |
| if (ret < 0) |
| profile->cv_tier_ov_cnt = GBMS_DEFAULT_CV_TIER_OV_CNT; |
| |
| /* how many samples under next tier to wait before switching */ |
| ret = of_property_read_u32(node, "google,cv-tier-switch-cnt", |
| &profile->cv_tier_switch_cnt); |
| if (ret < 0) |
| profile->cv_tier_switch_cnt = GBMS_DEFAULT_CV_TIER_SWITCH_CNT; |
| |
| /* allow being "a little" over tier voltage, experimental */ |
| ret = of_property_read_u32(node, "google,cv-otv-margin", |
| &profile->cv_otv_margin); |
| if (ret < 0) |
| profile->cv_otv_margin = GBMS_DEFAULT_CV_OTV_MARGIN; |
| |
| /* sanity on voltages (should warn?) */ |
| for (vi = 0; vi < profile->volt_nb_limits; vi++) |
| profile->volt_limits[vi] = profile->volt_limits[vi] / |
| profile->fv_uv_resolution * profile->fv_uv_resolution; |
| |
| return 0; |
| } |
| EXPORT_SYMBOL_GPL(gbms_init_chg_profile_internal); |
| |
| void gbms_free_chg_profile(struct gbms_chg_profile *profile) |
| { |
| kfree(profile->cccm_limits); |
| profile->cccm_limits = 0; |
| } |
| EXPORT_SYMBOL_GPL(gbms_free_chg_profile); |
| |
| /* NOTE: I should really pass the scale */ |
| void gbms_dump_raw_profile(char *buff, size_t len, const struct gbms_chg_profile *profile, int scale) |
| { |
| const int tscale = (scale == 1) ? 1 : 10; |
| int ti, vi, count = 0; |
| |
| count += scnprintf(buff + count, len - count, "Profile constant charge limits:\n"); |
| count += scnprintf(buff + count, len - count, "|T \\ V"); |
| for (vi = 0; vi < profile->volt_nb_limits; vi++) { |
| count += scnprintf(buff + count, len - count, " %4d", |
| profile->volt_limits[vi] / scale); |
| } |
| count += scnprintf(buff + count, len - count, "\n"); |
| |
| for (ti = 0; ti < profile->temp_nb_limits - 1; ti++) { |
| count += scnprintf(buff + count, len - count, "|%2d:%2d", |
| profile->temp_limits[ti] / tscale, |
| profile->temp_limits[ti + 1] / tscale); |
| for (vi = 0; vi < profile->volt_nb_limits; vi++) { |
| count += scnprintf(buff + count, len - count, " %4d", |
| GBMS_CCCM_LIMITS(profile, ti, vi) |
| / scale); |
| } |
| count += scnprintf(buff + count, len - count, "\n"); |
| } |
| } |
| EXPORT_SYMBOL_GPL(gbms_dump_raw_profile); |
| |
| /* |
| * When charging in DC, the fv_max will be (FV + cc_max * fv_dc_ratio). |
| * DC can be detected by cc_ua non-zero. |
| * The gap between Vchg and Vbat is caused by the ibat and the impedance. |
| * The fv_dc_ratio is used to simulate the impedance to get the proper fv, so |
| * the vbat will not over the otv threshold. |
| */ |
| int gbms_msc_round_fv_uv(const struct gbms_chg_profile *profile, |
| int vtier, int fv_uv, int cc_ua) |
| { |
| int result, fv_uv_orig = fv_uv; |
| const unsigned int fv_uv_max = (vtier / 1000) * profile->fv_uv_margin_dpct; |
| const unsigned int dc_fv_uv_max = vtier + (cc_ua / 1000) * profile->fv_dc_ratio; |
| const unsigned int last_fv = profile->volt_limits[profile->volt_nb_limits - 1]; |
| unsigned int fv_max; |
| |
| if (cc_ua == 0) |
| fv_max = fv_uv_max; |
| else if (dc_fv_uv_max >= last_fv) |
| fv_max = last_fv - profile->fv_uv_resolution; |
| else |
| fv_max = dc_fv_uv_max; |
| |
| if (fv_max != 0 && fv_uv > fv_max) |
| fv_uv = fv_max; |
| |
| fv_uv += profile->fv_uv_resolution / 2; |
| result = fv_uv - (fv_uv % profile->fv_uv_resolution); |
| |
| if (fv_max != 0) |
| gbms_info(profile, "MSC_ROUND: fv_uv=%d vtier=%d fv_uv_max=%d -> %d\n", |
| fv_uv_orig, vtier, fv_max, result); |
| |
| return result; |
| } |
| EXPORT_SYMBOL_GPL(gbms_msc_round_fv_uv); |
| |
| /* charge profile idx based on the battery temperature |
| * TODO: return -1 when temperature is lower than profile->temp_limits[0] or |
| * higher than profile->temp_limits[profile->temp_nb_limits - 1] |
| */ |
| int gbms_msc_temp_idx(const struct gbms_chg_profile *profile, int temp) |
| { |
| int temp_idx = 0; |
| |
| /* |
| * needs to limit under table size after the last ++ |
| * ex. temp_nb_limits=7 make 6 temp range from 0 to 5 |
| * so we need to limit in temp_nb_limits - 2 |
| */ |
| while (temp_idx < profile->temp_nb_limits - 2 && |
| temp >= profile->temp_limits[temp_idx + 1]) |
| temp_idx++; |
| |
| return temp_idx; |
| } |
| EXPORT_SYMBOL_GPL(gbms_msc_temp_idx); |
| |
| /* Compute the step index given the battery voltage |
| * When selecting an index need to make sure that headroom for the tier voltage |
| * will allow to send to the battery _at least_ next tier max FCC current and |
| * well over charge termination current. |
| */ |
| int gbms_msc_voltage_idx(const struct gbms_chg_profile *profile, int vbatt) |
| { |
| int vbatt_idx = 0; |
| |
| while (vbatt_idx < profile->volt_nb_limits - 1 && |
| vbatt > profile->volt_limits[vbatt_idx]) |
| vbatt_idx++; |
| |
| /* assumes that 3 times the hardware resolution is ok |
| * TODO: make it configurable? tune? |
| */ |
| if (vbatt_idx != profile->volt_nb_limits - 1) { |
| const int vt = profile->volt_limits[vbatt_idx]; |
| const int headr = profile->fv_uv_resolution * 3; |
| |
| if ((vt - vbatt) < headr) |
| vbatt_idx += 1; |
| } |
| |
| return vbatt_idx; |
| } |
| EXPORT_SYMBOL_GPL(gbms_msc_voltage_idx); |
| |
| uint8_t gbms_gen_chg_flags(int chg_status, int chg_type) |
| { |
| uint8_t flags = 0; |
| |
| if (chg_status != POWER_SUPPLY_STATUS_DISCHARGING) { |
| flags |= GBMS_CS_FLAG_BUCK_EN; |
| |
| /* FULL makes sense only when charging is enabled */ |
| if (chg_status == POWER_SUPPLY_STATUS_FULL) |
| flags |= GBMS_CS_FLAG_DONE; |
| } |
| if (chg_type == POWER_SUPPLY_CHARGE_TYPE_FAST) |
| flags |= GBMS_CS_FLAG_CC; |
| if (chg_type == POWER_SUPPLY_CHARGE_TYPE_TAPER) |
| flags |= GBMS_CS_FLAG_CV; |
| |
| return flags; |
| } |
| EXPORT_SYMBOL_GPL(gbms_gen_chg_flags); |
| |
| static int gbms_gen_state(union gbms_charger_state *chg_state, |
| struct power_supply *chg_psy) |
| { |
| int vchrg, chg_type, chg_status, ioerr; |
| |
| /* TODO: if (chg_drv->chg_mode == CHG_DRV_MODE_NOIRDROP) vchrg = 0; */ |
| /* Battery needs to know charger voltage and state to run the irdrop |
| * compensation code, can disable here sending a 0 vchgr |
| */ |
| vchrg = GPSY_GET_PROP(chg_psy, POWER_SUPPLY_PROP_VOLTAGE_NOW); |
| chg_type = GPSY_GET_PROP(chg_psy, POWER_SUPPLY_PROP_CHARGE_TYPE); |
| chg_status = GPSY_GET_INT_PROP(chg_psy, POWER_SUPPLY_PROP_STATUS, |
| &ioerr); |
| if (vchrg < 0 || chg_type < 0 || ioerr < 0) { |
| pr_err("MSC_CHG error vchrg=%d chg_type=%d chg_status=%d\n", |
| vchrg, chg_type, chg_status); |
| return -EINVAL; |
| } |
| |
| chg_state->f.chg_status = chg_status; |
| chg_state->f.chg_type = chg_type; |
| chg_state->f.flags = gbms_gen_chg_flags(chg_state->f.chg_status, |
| chg_state->f.chg_type); |
| chg_state->f.vchrg = vchrg / 1000; /* vchrg is in uA, f.vchrg us mA */ |
| |
| return 0; |
| } |
| |
| /* read or generate charge state */ |
| int gbms_read_charger_state(union gbms_charger_state *chg_state, |
| struct power_supply *chg_psy) |
| { |
| int64_t val; |
| int ret; |
| |
| val = GPSY_GET_INT64_PROP(chg_psy, GBMS_PROP_CHARGE_CHARGER_STATE, |
| &ret); |
| if (ret == 0) { |
| chg_state->v = val; |
| } else if (ret == -EAGAIN) { |
| return ret; |
| } else { |
| int ichg; |
| |
| ret = gbms_gen_state(chg_state, chg_psy); |
| if (ret < 0) |
| return ret; |
| |
| ichg = GPSY_GET_PROP(chg_psy, POWER_SUPPLY_PROP_CURRENT_NOW); |
| if (ichg > 0) |
| chg_state->f.icl = ichg / 1000; |
| |
| pr_info("MSC_CHG chg_state=%lx [0x%x:%d:%d:%d] ichg=%d\n", |
| (unsigned long)chg_state->v, |
| chg_state->f.flags, |
| chg_state->f.chg_type, |
| chg_state->f.chg_status, |
| chg_state->f.vchrg, |
| ichg); |
| } |
| |
| return 0; |
| } |
| EXPORT_SYMBOL_GPL(gbms_read_charger_state); |
| |
| /* ------------------------------------------------------------------------- */ |
| |
| /* convert cycle counts array to string */ |
| int gbms_cycle_count_cstr_bc(char *buf, size_t size, |
| const u16 *ccount, int bcnt) |
| { |
| int len = 0, i; |
| |
| for (i = 0; i < bcnt; i++) |
| len += scnprintf(buf + len, size - len, "%d ", ccount[i]); |
| buf[len - 1] = '\n'; |
| |
| return len; |
| } |
| EXPORT_SYMBOL_GPL(gbms_cycle_count_cstr_bc); |
| |
| /* parse the result of gbms_cycle_count_cstr_bc() back to array */ |
| int gbms_cycle_count_sscan_bc(u16 *ccount, int bcnt, const char *buff) |
| { |
| int i, val[10]; |
| |
| /* sscanf has 10 fixed conversions */ |
| if (bcnt != 10) |
| return -ERANGE; |
| |
| if (sscanf(buff, "%d %d %d %d %d %d %d %d %d %d", |
| &val[0], &val[1], &val[2], &val[3], &val[4], |
| &val[5], &val[6], &val[7], &val[8], &val[9]) |
| != bcnt) |
| return -EINVAL; |
| |
| for (i = 0; i < bcnt ; i++) |
| if (val[i] >= 0 && val[i] < U16_MAX) |
| ccount[i] = val[i]; |
| |
| return 0; |
| } |
| EXPORT_SYMBOL_GPL(gbms_cycle_count_sscan_bc); |
| |
| /* ------------------------------------------------------------------------- */ |
| |
| #define gbms_desc_from_psy(psy) \ |
| container_of(psy->desc, struct gbms_desc, psy_dsc) |
| |
| int gbms_set_property(struct power_supply *psy, enum gbms_property psp, |
| const union gbms_propval *val) |
| { |
| struct gbms_desc *dsc; |
| int ret; |
| |
| if (!psy) |
| return -EINVAL; |
| |
| dsc = gbms_desc_from_psy(psy); |
| if (dsc->set_property) { |
| const bool writable = (dsc->property_is_writeable) ? |
| dsc->property_is_writeable(psy, psp) : |
| false; |
| if (writable) { |
| ret = dsc->set_property(psy, psp, val); |
| if (ret == 0) |
| return 0; |
| } |
| } |
| |
| if (!dsc->forward) |
| return -ENODEV; |
| |
| pr_debug("set %d for '%s' to %d\n", psp, psy->desc->name, |
| val->prop.intval); |
| |
| ret = power_supply_set_property(psy, psp, &val->prop); |
| if (ret < 0) { |
| pr_err("failed to psp=%d for '%s', ret=%d\n", |
| psp, psy->desc->name, ret); |
| } |
| |
| return ret; |
| } |
| EXPORT_SYMBOL_GPL(gbms_set_property); |
| |
| int gbms_get_property(struct power_supply *psy, enum gbms_property psp, |
| union gbms_propval *val) |
| { |
| struct gbms_desc *dsc; |
| int ret; |
| |
| if (!psy) |
| return -EINVAL; |
| |
| dsc = gbms_desc_from_psy(psy); |
| if (dsc->get_property) { |
| ret = dsc->get_property(psy, psp, val); |
| if (ret == 0) |
| return 0; |
| } |
| |
| if (!dsc->forward) |
| return -ENODEV; |
| |
| ret = power_supply_get_property(psy, psp, &val->prop); |
| if (ret < 0) |
| pr_err("failed to get psp=%d from '%s', ret=%d\n", |
| psp, psy->desc->name, ret); |
| return ret; |
| } |
| EXPORT_SYMBOL_GPL(gbms_get_property); |
| |
| void gbms_logbuffer_prlog(struct logbuffer *log, int level, int debug_no_logbuffer, |
| int debug_printk_prlog, const char *f, ...) |
| { |
| va_list args; |
| |
| va_start(args, f); |
| |
| if (!debug_no_logbuffer) |
| logbuffer_vlog(log, f, args); |
| |
| if (level <= debug_printk_prlog) |
| vprintk(f, args); |
| |
| va_end(args); |
| } |
| EXPORT_SYMBOL_GPL(gbms_logbuffer_prlog); |
| |
| void gbms_logbuffer_devlog(struct logbuffer *log, struct device *dev, int level, |
| int debug_no_logbuffer, int debug_printk_prlog, |
| const char *f, ...) |
| { |
| struct va_format vaf; |
| va_list args; |
| |
| va_start(args, f); |
| |
| vaf.fmt = f; |
| vaf.va = &args; |
| |
| if (!debug_no_logbuffer) |
| logbuffer_vlog(log, f, args); |
| |
| if (level <= debug_printk_prlog) |
| dev_printk_emit(level, dev, "%s %s: %pV", |
| dev_driver_string(dev), |
| dev_name(dev), &vaf); |
| |
| va_end(args); |
| } |
| EXPORT_SYMBOL_GPL(gbms_logbuffer_devlog); |
| |
| bool chg_state_is_disconnected(const union gbms_charger_state *chg_state) |
| { |
| return ((chg_state->f.flags & GBMS_CS_FLAG_BUCK_EN) == 0) && |
| (chg_state->f.chg_status == POWER_SUPPLY_STATUS_DISCHARGING || |
| chg_state->f.chg_status == POWER_SUPPLY_STATUS_UNKNOWN); |
| } |
| EXPORT_SYMBOL_GPL(chg_state_is_disconnected); |
| |
| |
| /* Tier stats common routines */ |
| void gbms_tier_stats_init(struct gbms_ce_tier_stats *stats, int8_t idx) |
| { |
| stats->vtier_idx = idx; |
| stats->temp_idx = -1; |
| stats->soc_in = -1; |
| } |
| EXPORT_SYMBOL_GPL(gbms_tier_stats_init); |
| |
| /* call holding stats_lock */ |
| void gbms_chg_stats_tier(struct gbms_ce_tier_stats *tier, |
| int msc_state, |
| ktime_t elap) |
| { |
| if (msc_state < 0 || msc_state >= MSC_STATES_COUNT) |
| return; |
| |
| tier->msc_cnt[msc_state] += 1; |
| tier->msc_elap[msc_state] += elap; |
| } |
| EXPORT_SYMBOL_GPL(gbms_chg_stats_tier); |
| |
| void gbms_stats_update_tier(int temp_idx, int ibatt_ma, int temp, ktime_t elap, |
| int cc, union gbms_charger_state *chg_state, |
| enum gbms_msc_states_t msc_state, int soc_in, |
| struct gbms_ce_tier_stats *tier) |
| { |
| const uint16_t icl_settled = chg_state->f.icl; |
| |
| /* |
| * book time to previous msc_state for this tier, there is an |
| * interesting wrinkle here since some tiers (health, full, etc) |
| * might be entered and exited multiple times. |
| */ |
| gbms_chg_stats_tier(tier, msc_state, elap); |
| tier->sample_count += 1; |
| |
| if (tier->soc_in == -1) { |
| tier->temp_idx = temp_idx; |
| |
| tier->temp_in = temp; |
| tier->temp_min = temp; |
| tier->temp_max = temp; |
| |
| tier->ibatt_min = ibatt_ma; |
| tier->ibatt_max = ibatt_ma; |
| |
| tier->icl_min = icl_settled; |
| tier->icl_max = icl_settled; |
| |
| tier->soc_in = soc_in; |
| tier->cc_in = cc; |
| tier->cc_total = 0; |
| return; |
| } |
| |
| /* crossed temperature tier */ |
| if (temp_idx != tier->temp_idx) |
| tier->temp_idx = -1; |
| |
| if (chg_state->f.chg_type == POWER_SUPPLY_CHARGE_TYPE_FAST) { |
| tier->time_fast += elap; |
| } else if (chg_state->f.chg_type == POWER_SUPPLY_CHARGE_TYPE_TAPER) { |
| tier->time_taper += elap; |
| } else { |
| tier->time_other += elap; |
| } |
| |
| /* |
| * averages: temp < 100. icl_settled < 3000, sum(ibatt) |
| * is bound to battery capacity, elap in seconds, sums |
| * are stored in an s64. For icl_settled I need a tier |
| * to last for more than ~97M years. |
| */ |
| if (temp < tier->temp_min) |
| tier->temp_min = temp; |
| if (temp > tier->temp_max) |
| tier->temp_max = temp; |
| tier->temp_sum += temp * elap; |
| |
| if (icl_settled < tier->icl_min) |
| tier->icl_min = icl_settled; |
| if (icl_settled > tier->icl_max) |
| tier->icl_max = icl_settled; |
| tier->icl_sum += icl_settled * elap; |
| |
| if (ibatt_ma < tier->ibatt_min) |
| tier->ibatt_min = ibatt_ma; |
| if (ibatt_ma > tier->ibatt_max) |
| tier->ibatt_max = ibatt_ma; |
| tier->ibatt_sum += ibatt_ma * elap; |
| |
| tier->cc_total = cc - tier->cc_in; |
| } |
| EXPORT_SYMBOL_GPL(gbms_stats_update_tier); |
| |
| /* Log only when elap != 0 */ |
| int gbms_tier_stats_cstr(char *buff, int size, |
| const struct gbms_ce_tier_stats *tier_stat, |
| bool verbose) |
| { |
| const int soc_in = tier_stat->soc_in >> 8; |
| const long elap = tier_stat->time_fast + tier_stat->time_taper + |
| tier_stat->time_other; |
| |
| long temp_avg, ibatt_avg, icl_avg; |
| int j, len = 0; |
| |
| if (elap) { |
| temp_avg = tier_stat->temp_sum / elap; |
| ibatt_avg = tier_stat->ibatt_sum / elap; |
| icl_avg = tier_stat->icl_sum / elap; |
| } else { |
| temp_avg = 0; |
| ibatt_avg = 0; |
| icl_avg = 0; |
| } |
| |
| len += scnprintf(&buff[len], size - len, "\n%d%c ", |
| tier_stat->vtier_idx, |
| (verbose) ? ':' : ','); |
| |
| len += scnprintf(&buff[len], size - len, |
| "%d.%d,%d,%d, %d,%d,%d, %d,%ld,%d, %d,%ld,%d, %d,%ld,%d", |
| soc_in, |
| tier_stat->soc_in & 0xff, |
| tier_stat->cc_in, |
| tier_stat->temp_in, |
| tier_stat->time_fast, |
| tier_stat->time_taper, |
| tier_stat->time_other, |
| tier_stat->temp_min, |
| temp_avg, |
| tier_stat->temp_max, |
| tier_stat->ibatt_min, |
| ibatt_avg, |
| tier_stat->ibatt_max, |
| tier_stat->icl_min, |
| icl_avg, |
| tier_stat->icl_max); |
| |
| if (!verbose || !elap) |
| return len; |
| |
| /* time spent in every multi step charging state */ |
| len += scnprintf(&buff[len], size - len, "\n%d:", |
| tier_stat->vtier_idx); |
| |
| for (j = 0; j < MSC_STATES_COUNT; j++) |
| len += scnprintf(&buff[len], size - len, " %s=%d", |
| gbms_get_code(j), tier_stat->msc_elap[j]); |
| |
| /* count spent in each step charging state */ |
| len += scnprintf(&buff[len], size - len, "\n%d:", |
| tier_stat->vtier_idx); |
| |
| for (j = 0; j < MSC_STATES_COUNT; j++) |
| len += scnprintf(&buff[len], size - len, " %s=%d", |
| gbms_get_code(j), tier_stat->msc_cnt[j]); |
| |
| return len; |
| } |
| EXPORT_SYMBOL_GPL(gbms_tier_stats_cstr); |
| |
| void gbms_log_cstr_handler(struct logbuffer *log, char *buf, int len) |
| { |
| int i, j = 0; |
| char tmp[LOG_BUFFER_ENTRY_SIZE]; |
| |
| buf[len] = '\n'; |
| for (i = 0; i <= len; i++) { |
| if (buf[i] == '\n') { |
| tmp[j] = '\0'; |
| /* skip first blank line */ |
| if (i != 0) |
| logbuffer_log(log, "%s", tmp); |
| j = 0; |
| } else if (j >= LOG_BUFFER_ENTRY_SIZE - 1) { |
| tmp[j] = '\0'; |
| logbuffer_log(log, "%s", tmp); |
| i--; |
| j = 0; |
| } else { |
| tmp[j] = buf[i]; |
| j++; |
| } |
| } |
| } |
| EXPORT_SYMBOL_GPL(gbms_log_cstr_handler); |
| |