| /* SPDX-License-Identifier: GPL-2.0 */ |
| /* |
| * PCA9468 Direct Charger PPS Integration |
| * |
| * Copyright (C) 2021 Google, LLC |
| * |
| */ |
| |
| |
| #include <linux/err.h> |
| #include <linux/init.h> |
| #include <linux/version.h> |
| #include <linux/delay.h> |
| #include <linux/dev_printk.h> |
| #include <linux/of_device.h> |
| #include <linux/regmap.h> |
| |
| #include "pca9468_regs.h" |
| #include "pca9468_charger.h" |
| |
| /* Logging ----------------------------------------------------------------- */ |
| |
| int debug_printk_prlog = LOGLEVEL_INFO; |
| int debug_no_logbuffer = 0; |
| |
| /* DC PPS integration ------------------------------------------------------ */ |
| |
| static void p9468_chg_stats_set_apdo(struct p9468_chg_stats *chg_data, u32 apdo); |
| |
| static struct device_node *pca9468_find_config(struct device_node * node) |
| { |
| struct device_node *temp; |
| |
| if (!node) |
| return node; |
| temp = of_parse_phandle(node, "pca9468,google_cpm", 0); |
| if (temp) |
| node = temp; |
| return node; |
| } |
| |
| int pca9468_probe_pps(struct pca9468_charger *pca9468_chg) |
| { |
| const char *tmp_name = NULL; |
| bool pps_available = false; |
| struct device_node *node; |
| int ret; |
| |
| node = pca9468_find_config(pca9468_chg->dev->of_node); |
| if (!node) |
| return -ENODEV; |
| |
| ret = of_property_read_u32(node, "google,tcpm-power-supply", |
| &pca9468_chg->tcpm_phandle); |
| if (ret < 0) |
| pr_warn("pca9468: pca,tcpm-power-supply not defined\n"); |
| else |
| pps_available |= true; |
| |
| ret = of_property_read_string(node, "google,wlc_dc-power-supply", |
| &tmp_name); |
| if (ret < 0) |
| pr_warn("pca9468: google,wlc_dc-power-supply not defined\n"); |
| if (ret == 0) { |
| pca9468_chg->wlc_psy_name = |
| devm_kstrdup(pca9468_chg->dev, tmp_name, GFP_KERNEL); |
| if (!pca9468_chg->wlc_psy_name) |
| return -ENOMEM; |
| |
| pps_available |= true; |
| } |
| |
| return pps_available ? 0 : -ENODEV; |
| } |
| |
| /* ------------------------------------------------------------------------ */ |
| |
| /* switch PDO if needed */ |
| int pca9468_request_pdo(struct pca9468_charger *pca9468) |
| { |
| int ret = 0; |
| |
| pr_debug("%s: ta_objpos=%u, ta_vol=%u, ta_cur=%u\n", __func__, |
| pca9468->ta_objpos, pca9468->ta_vol, pca9468->ta_cur); |
| |
| /* |
| * the reference implementation call pps_request_pdo() twice with a |
| * 100 ms delay between the calls when the function returns -EBUSY: |
| * |
| * ret = pps_request_pdo(&pca9468->pps_data, pca9468->ta_objpos, |
| * pca9468->ta_vol, pca9468->ta_cur, |
| * pca9468->pd); |
| * |
| * The wrapper in google_dc_pps route the calls to the tcpm engine |
| * via tcpm_update_sink_capabilities(). The sync capabilities are |
| * in pps_data, ->ta_objpos select the (A)PDO index, ->ta_vol and |
| * ->ta_cur are the desired TA voltage and current. |
| * |
| * this is now handled by pps_update_adapter() |
| * |
| * TODO: verify the timing and make sure that there are no races that |
| * cause the targets |
| */ |
| |
| return ret; |
| } |
| |
| int pca9468_usbpd_setup(struct pca9468_charger *pca9468) |
| { |
| struct power_supply *tcpm_psy; |
| bool online; |
| int ret = 0; |
| |
| if (pca9468->pd != NULL) |
| goto check_online; |
| |
| if (pca9468->tcpm_psy_name) { |
| tcpm_psy = power_supply_get_by_name(pca9468->tcpm_psy_name); |
| if (!tcpm_psy) |
| return -ENODEV; |
| |
| pca9468->pd = tcpm_psy; |
| } else if (pca9468->tcpm_phandle) { |
| struct device_node *node; |
| |
| node = pca9468_find_config(pca9468->dev->of_node); |
| if (!node) |
| return -ENODEV; |
| tcpm_psy = pps_get_tcpm_psy(node, 2); |
| if (IS_ERR(tcpm_psy)) |
| return PTR_ERR(tcpm_psy); |
| if (!tcpm_psy) { |
| pca9468->tcpm_phandle = 0; |
| return -ENODEV; |
| } |
| |
| pr_err("%s: TCPM name is %s\n", __func__, |
| pps_name(tcpm_psy)); |
| pca9468->tcpm_psy_name = tcpm_psy->desc->name; |
| pca9468->pd = tcpm_psy; |
| } else { |
| pr_err("%s: TCPM DC not defined\n", __func__); |
| return -ENODEV; |
| } |
| |
| /* not needed if tcpm-power-supply is not there */ |
| ret = pps_init(&pca9468->pps_data, pca9468->dev, tcpm_psy, "pca-pps"); |
| if (ret == 0) { |
| pps_set_logbuffer(&pca9468->pps_data, pca9468->log); |
| pps_init_state(&pca9468->pps_data); |
| } |
| |
| check_online: |
| online = pps_prog_check_online(&pca9468->pps_data, pca9468->pd); |
| if (!online) |
| return -ENODEV; |
| |
| return ret; |
| } |
| |
| /* call holding mutex_unlock(&pca9468->lock); */ |
| int pca9468_send_pd_message(struct pca9468_charger *pca9468, |
| unsigned int msg_type) |
| { |
| struct pd_pps_data *pps_data = &pca9468->pps_data; |
| struct power_supply *tcpm_psy = pca9468->pd; |
| bool online; |
| int pps_ui; |
| int ret; |
| |
| if (!tcpm_psy || (pca9468->charging_state == DC_STATE_NO_CHARGING && |
| msg_type == PD_MSG_REQUEST_APDO) || !pca9468->mains_online) { |
| pr_debug("%s: failure tcpm_psy_ok=%d charging_state=%u online=%d", |
| __func__, tcpm_psy != 0, pca9468->charging_state, |
| pca9468->mains_online); |
| return -EINVAL; |
| } |
| |
| /* false when offline (0) or not in prog (1) mode */ |
| online = pps_prog_check_online(&pca9468->pps_data, tcpm_psy); |
| if (!online) { |
| pr_debug("%s: not online", __func__); |
| return -EINVAL; |
| } |
| |
| /* turn off PPS/PROG, revert to PD */ |
| if (msg_type == MSG_REQUEST_FIXED_PDO) { |
| ret = pps_prog_offline(&pca9468->pps_data, tcpm_psy); |
| pr_debug("%s: requesting offline ret=%d\n", __func__, ret); |
| /* TODO: reset state? */ |
| return ret; |
| } |
| |
| pr_debug("%s: tcpm_psy_ok=%d pd_online=%d pps_stage=%d charging_state=%u", |
| __func__, tcpm_psy != 0, pps_data->pd_online, |
| pps_data->stage, pca9468->charging_state); |
| |
| if (pca9468->pps_data.stage == PPS_ACTIVE) { |
| |
| /* not sure I need to do this */ |
| ret = pca9468_request_pdo(pca9468); |
| if (ret == 0) { |
| const int pre_out_uv = pps_data->out_uv; |
| const int pre_out_ua = pps_data->op_ua; |
| |
| pr_debug("%s: ta_vol=%u, ta_cur=%u, ta_objpos=%u\n", |
| __func__, pca9468->ta_vol, pca9468->ta_cur, |
| pca9468->ta_objpos); |
| |
| pps_ui = pps_update_adapter(&pca9468->pps_data, |
| pca9468->ta_vol, |
| pca9468->ta_cur, |
| tcpm_psy); |
| pr_debug("%s: out_uv=%d %d->%d, out_ua=%d %d->%d (%d)\n", |
| __func__, |
| pps_data->out_uv, pre_out_uv, pca9468->ta_vol, |
| pps_data->op_ua, pre_out_ua, pca9468->ta_cur, |
| pps_ui); |
| |
| if (pps_ui == 0) |
| pps_ui = PCA9468_PDMSG_WAIT_T; |
| if (pps_ui < 0) |
| pps_ui = PCA9468_PDMSG_RETRY_T; |
| } else { |
| pr_debug("%s: request_pdo failed ret=%d\n", |
| __func__, ret); |
| pps_ui = PCA9468_PDMSG_RETRY_T; |
| } |
| |
| } else { |
| ret = pps_keep_alive(pps_data, tcpm_psy); |
| if (ret == 0) |
| pps_ui = PD_T_PPS_TIMEOUT; |
| |
| pr_debug("%s: keep alive ret=%d\n", __func__, ret); |
| } |
| |
| if (((pca9468->charging_state == DC_STATE_NO_CHARGING) && |
| (msg_type == PD_MSG_REQUEST_APDO)) || |
| (pca9468->mains_online == false)) { |
| |
| /* |
| * Vbus reset might occour even when PD comms is successful. |
| * Check again. |
| */ |
| pps_ui = -EINVAL; |
| } |
| |
| /* PPS_Work: will reschedule */ |
| pr_debug("%s: pps_ui = %d\n", __func__, pps_ui); |
| if (pps_ui > 0) |
| mod_delayed_work(system_wq, &pca9468->pps_work, |
| msecs_to_jiffies(pps_ui)); |
| |
| return pps_ui; |
| } |
| |
| /* |
| * Get the max current/voltage/power of APDO from the CC/PD driver. |
| * |
| * Initialize &pca9468->ta_max_vol, &pca9468->ta_max_cur, &pca9468->ta_max_pwr |
| * initialize pca9468->pps_data and &pca9468->ta_objpos also |
| * |
| * call holding mutex_unlock(&pca9468->lock); |
| */ |
| int pca9468_get_apdo_max_power(struct pca9468_charger *pca9468, |
| unsigned int ta_max_vol, |
| unsigned int ta_max_cur) |
| { |
| int ret; |
| |
| /* limits */ |
| pca9468->ta_objpos = 0; /* if !=0 will return the ca */ |
| pca9468->ta_max_vol = ta_max_vol; |
| pca9468->ta_max_cur = ta_max_cur; |
| |
| /* check the phandle */ |
| ret = pca9468_usbpd_setup(pca9468); |
| if (ret != 0) { |
| dev_err(pca9468->dev, "cannot find TCPM %d\n", ret); |
| pca9468->pd = NULL; |
| return ret; |
| } |
| |
| /* technically already in pda_data since check online does it */ |
| ret = pps_get_src_cap(&pca9468->pps_data, pca9468->pd); |
| if (ret < 0) |
| return ret; |
| |
| ret = pps_get_apdo_max_power(&pca9468->pps_data, &pca9468->ta_objpos, |
| &pca9468->ta_max_vol, &pca9468->ta_max_cur, |
| &pca9468->ta_max_pwr); |
| if (ret < 0) { |
| pr_err("cannot determine the apdo max power ret = %d\n", ret); |
| return ret; |
| } |
| |
| pr_debug("%s: APDO pos=%u max_v=%u max_c=%u max_pwr=%lu\n", __func__, |
| pca9468->ta_objpos, pca9468->ta_max_vol, pca9468->ta_max_cur, |
| pca9468->ta_max_pwr); |
| |
| p9468_chg_stats_set_apdo(&pca9468->chg_data, |
| pca9468->pps_data.src_caps[pca9468->ta_objpos - 1]); |
| |
| return 0; |
| } |
| |
| /* WLC_DC ---------------------------------------------------------------- */ |
| /* call holding mutex_unlock(&pca9468->lock); */ |
| struct power_supply *pca9468_get_rx_psy(struct pca9468_charger *pca9468) |
| { |
| if (!pca9468->wlc_psy) { |
| const char *wlc_psy_name = pca9468->wlc_psy_name ? : "wireless"; |
| |
| pca9468->wlc_psy = power_supply_get_by_name(wlc_psy_name); |
| if (!pca9468->wlc_psy) { |
| dev_err(pca9468->dev, "%s Cannot find %s power supply\n", |
| __func__, wlc_psy_name); |
| } |
| } |
| |
| return pca9468->wlc_psy; |
| } |
| |
| /* call holding mutex_unlock(&pca9468->lock); */ |
| int pca9468_send_rx_voltage(struct pca9468_charger *pca9468, |
| unsigned int msg_type) |
| { |
| union power_supply_propval pro_val; |
| struct power_supply *wlc_psy; |
| int ret = -EINVAL; |
| |
| /* Vbus reset happened in the previous PD communication */ |
| if (pca9468->mains_online == false) |
| goto out; |
| |
| wlc_psy = pca9468_get_rx_psy(pca9468); |
| if (!wlc_psy) { |
| ret = -ENODEV; |
| goto out; |
| } |
| |
| /* turn off PPS/PROG, revert to PD */ |
| if (msg_type == MSG_REQUEST_FIXED_PDO) { |
| union power_supply_propval online; |
| int ret; |
| |
| ret = power_supply_get_property(wlc_psy, POWER_SUPPLY_PROP_ONLINE, &online); |
| if (ret == 0 && online.intval == PPS_PSY_PROG_ONLINE) { |
| unsigned int val; |
| |
| /* |
| * Make sure the pca is in stby mode before setting |
| * online back to PPS_PSY_FIXED_ONLINE. |
| */ |
| ret = regmap_read(pca9468->regmap, PCA9468_REG_START_CTRL, &val); |
| if (ret < 0 || !(val & PCA9468_STANDBY_FORCED)) { |
| dev_err(pca9468->dev, |
| "Device not in stby val=%x (%d)\n", |
| val, ret); |
| |
| return -EINVAL; |
| } |
| |
| pro_val.intval = PPS_PSY_FIXED_ONLINE; |
| ret = power_supply_set_property(wlc_psy, POWER_SUPPLY_PROP_ONLINE, |
| &pro_val); |
| pr_debug("%s: online=%d->%d ret=%d\n", __func__, |
| online.intval, pro_val.intval, ret); |
| } else if (ret < 0) { |
| pr_err("%s: online=%d ret=%d\n", __func__, |
| online.intval, ret); |
| } |
| |
| /* TODO: reset state? */ |
| |
| /* done if don't have alternate voltage */ |
| if (!pca9468->ta_vol) |
| return ret; |
| } |
| |
| pro_val.intval = pca9468->ta_vol; |
| ret = power_supply_set_property(wlc_psy, POWER_SUPPLY_PROP_VOLTAGE_NOW, |
| &pro_val); |
| if (ret < 0) |
| dev_err(pca9468->dev, "Cannot set RX voltage to %d (%d)\n", |
| pro_val.intval, ret); |
| |
| /* Vbus reset might happen, check the charging state again */ |
| if (pca9468->mains_online == false) { |
| pr_warn("%s: mains offline\n", __func__); |
| ret = -EINVAL; |
| } |
| |
| logbuffer_prlog(pca9468, LOGLEVEL_DEBUG, "WLCDC: online=%d ta_vol=%d (%d)", |
| pca9468->mains_online, pca9468->ta_vol, ret); |
| |
| out: |
| return ret; |
| } |
| |
| /* |
| * Get the max current/voltage/power of RXIC from the WCRX driver |
| * Initialize &pca9468->ta_max_vol, &pca9468->ta_max_cur, &pca9468->ta_max_pwr |
| * call holding mutex_unlock(&pca9468->lock); |
| */ |
| int pca9468_get_rx_max_power(struct pca9468_charger *pca9468) |
| { |
| union power_supply_propval pro_val; |
| struct power_supply *wlc_psy; |
| int ret = 0; |
| |
| wlc_psy = pca9468_get_rx_psy(pca9468); |
| if (!wlc_psy) { |
| dev_err(pca9468->dev, "Cannot find wireless power supply\n"); |
| return -ENODEV; |
| } |
| |
| /* Get the maximum voltage */ |
| ret = power_supply_get_property(wlc_psy, POWER_SUPPLY_PROP_VOLTAGE_MAX, |
| &pro_val); |
| if (ret < 0) { |
| dev_err(pca9468->dev, "%s Cannot get the maximum RX voltage (%d)\n", |
| __func__, ret); |
| return ret; |
| } |
| |
| /* RX IC cannot support the request maximum voltage */ |
| if (pca9468->ta_max_vol > pro_val.intval) { |
| dev_err(pca9468->dev, "%s max %d cannot support ta_max %d voltage\n", |
| __func__, pro_val.intval, pca9468->ta_max_vol); |
| return -EINVAL; |
| } |
| |
| pca9468->ta_max_vol = pro_val.intval; |
| |
| /* Get the maximum current */ |
| ret = power_supply_get_property(wlc_psy, POWER_SUPPLY_PROP_CURRENT_MAX, |
| &pro_val); |
| if (ret < 0) { |
| dev_err(pca9468->dev, "%s Cannot get the maximum RX current (%d)\n", |
| __func__, ret); |
| return ret; |
| } |
| |
| pca9468->ta_max_cur = pro_val.intval; |
| pca9468->ta_max_pwr = (pca9468->ta_max_vol / 1000) * |
| (pca9468->ta_max_cur / 1000); |
| |
| logbuffer_prlog(pca9468, LOGLEVEL_INFO, "WLCDC: max_cur=%d max_pwr=%ld", |
| pca9468->ta_max_cur, pca9468->ta_max_pwr); |
| return 0; |
| } |
| |
| /* called from start_direct_charging(), negative will abort */ |
| int pca9468_set_ta_type(struct pca9468_charger *pca9468, int pps_index) |
| { |
| if (pps_index == PPS_INDEX_TCPM) { |
| int ret; |
| |
| ret = pca9468_usbpd_setup(pca9468); |
| if (ret != 0) { |
| dev_err(pca9468->dev, "Cannot find the TA %d\n", ret); |
| return ret; |
| } |
| |
| pca9468->ta_type = TA_TYPE_USBPD; |
| pca9468->chg_mode = CHG_2TO1_DC_MODE; |
| } else if (pps_index == PPS_INDEX_WLC) { |
| struct power_supply *wlc_psy; |
| |
| wlc_psy = pca9468_get_rx_psy(pca9468); |
| if (!wlc_psy) { |
| dev_err(pca9468->dev, "Cannot find wireless power supply\n"); |
| return -ENODEV; |
| } |
| |
| pca9468->ta_type = TA_TYPE_WIRELESS; |
| pca9468->chg_mode = CHG_4TO1_DC_MODE; |
| } else { |
| pca9468->ta_type = TA_TYPE_UNKNOWN; |
| pca9468->chg_mode = 0; |
| return -EINVAL; |
| } |
| |
| return 0; |
| } |
| |
| |
| /* GBMS integration ------------------------------------------------------ */ |
| |
| int pca9468_get_charge_type(struct pca9468_charger *pca9468) |
| { |
| int ret, sts; |
| |
| if (!pca9468->mains_online) |
| return POWER_SUPPLY_CHARGE_TYPE_NONE; |
| |
| /* |
| * HW will reports PCA9468_BIT_IIN_LOOP_STS (CC) or |
| * PCA9468_BIT_VFLT_LOOP_STS (CV) or inactive (i.e. openloop). |
| */ |
| ret = regmap_read(pca9468->regmap, PCA9468_REG_STS_A, &sts); |
| if (ret < 0) |
| return ret; |
| |
| pr_debug("%s: sts_a=%0x2 VFLT=%d IIN=%d charging_state=%d\n", |
| __func__, sts, !!(sts & PCA9468_BIT_VFLT_LOOP_STS), |
| !!(sts & PCA9468_BIT_IIN_LOOP_STS), |
| pca9468->charging_state); |
| |
| /* Use SW state for now */ |
| switch (pca9468->charging_state) { |
| case DC_STATE_ADJUST_CC: |
| case DC_STATE_CC_MODE: |
| case DC_STATE_ADJUST_TAVOL: |
| case DC_STATE_ADJUST_TACUR: |
| return POWER_SUPPLY_CHARGE_TYPE_FAST; |
| case DC_STATE_START_CV: |
| case DC_STATE_CV_MODE: |
| return POWER_SUPPLY_CHARGE_TYPE_TAPER; |
| case DC_STATE_CHECK_ACTIVE: /* in preset */ |
| case DC_STATE_CHARGING_DONE: |
| break; |
| } |
| |
| return POWER_SUPPLY_CHARGE_TYPE_NONE; |
| } |
| |
| #define PCA9468_NOT_CHARGING \ |
| (PCA9468_BIT_SHUTDOWN_STATE_STS | PCA9468_BIT_STANDBY_STATE_STS) |
| #define PCA9468_ANY_CHARGING_LOOP \ |
| (PCA9468_BIT_CHG_LOOP_STS | PCA9468_BIT_IIN_LOOP_STS | \ |
| PCA9468_BIT_VFLT_LOOP_STS) |
| |
| int pca9468_get_status(struct pca9468_charger *pca9468) |
| { |
| u8 val[8]; |
| int ret; |
| |
| ret = regmap_bulk_read(pca9468->regmap, PCA9468_REG_INT1_STS, |
| &val[PCA9468_REG_INT1_STS], 5); |
| if (ret < 0) { |
| pr_debug("%s: ioerr=%d", __func__, ret); |
| return POWER_SUPPLY_STATUS_UNKNOWN; |
| } |
| |
| pr_debug("%s: int1_sts=0x%x,sts_a=0x%x,sts_b=0x%x,sts_c=0x%x,sts_d=0x%x\n", |
| __func__, val[3], val[4], val[5], val[6], val[7]); |
| |
| if ((val[PCA9468_REG_STS_B] & PCA9468_BIT_ACTIVE_STATE_STS) == 0 || |
| (val[PCA9468_REG_INT1_STS] & PCA9468_BIT_V_OK_STS) == 0) { |
| const bool online = pca9468->mains_online; |
| |
| /* no disconnect during charger transition */ |
| return online ? POWER_SUPPLY_STATUS_NOT_CHARGING : |
| POWER_SUPPLY_STATUS_DISCHARGING; |
| } |
| |
| /* Use SW state (for now) */ |
| switch (pca9468->charging_state) { |
| case DC_STATE_NO_CHARGING: |
| case DC_STATE_CHECK_VBAT: |
| case DC_STATE_PRESET_DC: |
| case DC_STATE_CHECK_ACTIVE: |
| return POWER_SUPPLY_STATUS_NOT_CHARGING; |
| case DC_STATE_ADJUST_CC: |
| case DC_STATE_CC_MODE: |
| case DC_STATE_START_CV: |
| case DC_STATE_CV_MODE: |
| return POWER_SUPPLY_STATUS_CHARGING; |
| /* cpm will need to stop it */ |
| case DC_STATE_CHARGING_DONE: |
| return POWER_SUPPLY_STATUS_CHARGING; |
| default: |
| break; |
| } |
| |
| return POWER_SUPPLY_STATUS_UNKNOWN; |
| } |
| |
| #define PCA9468_PRESENT_MASK \ |
| (PCA9468_BIT_ACTIVE_STATE_STS | PCA9468_BIT_STANDBY_STATE_STS) |
| |
| int pca9468_is_present(struct pca9468_charger *pca9468) |
| { |
| int sts = 0; |
| |
| regmap_read(pca9468->regmap, PCA9468_REG_STS_B, &sts); |
| return !!(sts & PCA9468_PRESENT_MASK); |
| } |
| |
| int pca9468_get_chg_chgr_state(struct pca9468_charger *pca9468, |
| union gbms_charger_state *chg_state) |
| { |
| int vchrg; |
| |
| chg_state->v = 0; |
| chg_state->f.chg_status = pca9468_get_status(pca9468); |
| chg_state->f.chg_type = pca9468_get_charge_type(pca9468); |
| chg_state->f.flags = gbms_gen_chg_flags(chg_state->f.chg_status, |
| chg_state->f.chg_type); |
| chg_state->f.flags |= GBMS_CS_FLAG_NOCOMP; |
| |
| vchrg = pca9468_read_adc(pca9468, ADCCH_VBAT); |
| if (vchrg > 0) |
| chg_state->f.vchrg = vchrg / 1000; |
| |
| if (chg_state->f.chg_status != POWER_SUPPLY_STATUS_DISCHARGING) { |
| int rc; |
| |
| rc = pca9468_input_current_limit(pca9468); |
| if (rc > 0) |
| chg_state->f.icl = rc / 1000; |
| } |
| |
| return 0; |
| } |
| |
| /* ------------------------------------------------------------------------ */ |
| |
| /* call holding (&pca9468->lock); */ |
| void p9468_chg_stats_init(struct p9468_chg_stats *chg_data) |
| { |
| memset(chg_data, 0, sizeof(*chg_data)); |
| chg_data->adapter_capabilities[0] |= P9468_CHGS_VER; |
| } |
| |
| static void p9468_chg_stats_set_apdo(struct p9468_chg_stats *chg_data, u32 apdo) |
| { |
| chg_data->adapter_capabilities[1] = apdo; |
| } |
| |
| /* call holding (&pca9468->lock); */ |
| int p9468_chg_stats_update(struct p9468_chg_stats *chg_data, |
| const struct pca9468_charger *pca9468) |
| { |
| switch (pca9468->charging_state) { |
| case DC_STATE_NO_CHARGING: |
| chg_data->nc_count++; |
| break; |
| case DC_STATE_CHECK_VBAT: |
| case DC_STATE_PRESET_DC: |
| chg_data->pre_count++; |
| break; |
| case DC_STATE_CHECK_ACTIVE: |
| chg_data->ca_count++; |
| break; |
| case DC_STATE_ADJUST_CC: |
| case DC_STATE_CC_MODE: |
| chg_data->cc_count++; |
| break; |
| case DC_STATE_START_CV: |
| case DC_STATE_CV_MODE: |
| chg_data->cv_count++; |
| break; |
| case DC_STATE_ADJUST_TAVOL: |
| case DC_STATE_ADJUST_TACUR: |
| chg_data->adj_count++; |
| break; |
| case DC_STATE_CHARGING_DONE: |
| chg_data->receiver_state[0] |= P9468_CHGS_F_DONE; |
| break; |
| default: break; |
| } |
| |
| return 0; |
| } |
| |
| void p9468_chg_stats_dump(const struct pca9468_charger *pca9468) |
| { |
| const struct p9468_chg_stats *chg_data = &pca9468->chg_data; |
| |
| logbuffer_prlog(pca9468, LOGLEVEL_INFO, |
| "N: ovc=%d,ovc_ibatt=%d,ovc_delta=%d rcp=%d,stby=%d", |
| chg_data->ovc_count, |
| chg_data->ovc_max_ibatt, chg_data->ovc_max_delta, |
| chg_data->rcp_count, chg_data->stby_count); |
| logbuffer_prlog(pca9468, LOGLEVEL_INFO, |
| "C: nc=%d,pre=%d,ca=%d,cc=%d,cv=%d,adj=%d\n", |
| chg_data->nc_count, chg_data->pre_count, |
| chg_data->ca_count, chg_data->cc_count, |
| chg_data->cv_count, chg_data->adj_count); |
| } |
| |
| int p9468_chg_stats_done(struct p9468_chg_stats *chg_data, |
| const struct pca9468_charger *pca9468) |
| { |
| /* AC[0] version */ |
| /* AC[1] is APDO */ |
| /* RS[0][0:8] flags */ |
| if (chg_data->stby_count) |
| p9468_chg_stats_update_flags(chg_data, P9468_CHGS_F_STBY); |
| chg_data->receiver_state[0] = (chg_data->pre_count & 0xff) << |
| P9468_CHGS_PRE_SHIFT; |
| chg_data->receiver_state[0] |= (chg_data->rcp_count & 0xff) << |
| P9468_CHGS_RCPC_SHIFT; |
| chg_data->receiver_state[0] |= (chg_data->nc_count & 0xff) << |
| P9468_CHGS_NC_SHIFT; |
| /* RS[1] counters */ |
| chg_data->receiver_state[1] = (chg_data->ovc_count & 0xffff) << |
| P9468_CHGS_OVCC_SHIFT; |
| chg_data->receiver_state[1] |= (chg_data->adj_count & 0xffff) << |
| P9468_CHGS_ADJ_SHIFT; |
| /* RS[2] counters */ |
| chg_data->receiver_state[2] = (chg_data->adj_count & 0xffff) << |
| P9468_CHGS_ADJ_SHIFT; |
| chg_data->receiver_state[2] |= (chg_data->adj_count & 0xffff) << |
| P9468_CHGS_ADJ_SHIFT; |
| /* RS[3] counters */ |
| chg_data->receiver_state[3] = (chg_data->cc_count & 0xffff) << |
| P9468_CHGS_CC_SHIFT; |
| chg_data->receiver_state[3] |= (chg_data->cv_count & 0xffff) << |
| P9468_CHGS_CV_SHIFT; |
| /* RS[4] counters */ |
| chg_data->receiver_state[1] = (chg_data->ca_count & 0xff) << |
| P9468_CHGS_CA_SHIFT; |
| |
| chg_data->valid = true; |
| |
| return 0; |
| } |