| /* SPDX-License-Identifier: GPL-2.0 */ |
| /* |
| * P9221 Wireless Charger Driver chip revision specific functions. |
| * |
| * Copyright (C) 2020 Google, LLC |
| * |
| */ |
| |
| #include <linux/device.h> |
| #include <linux/pm.h> |
| #include <linux/gpio/driver.h> |
| #include <linux/interrupt.h> |
| #include <linux/i2c.h> |
| #include <linux/module.h> |
| #include <linux/slab.h> |
| #include <linux/delay.h> |
| #include <linux/kernel.h> |
| #include <linux/alarmtimer.h> |
| #include "p9221_charger.h" |
| #include "google_bms.h" |
| |
| #define P9XXX_NUM_GPIOS 16 |
| #define P9XXX_MIN_GPIO 0 |
| #define P9XXX_MAX_GPIO 15 |
| #define P9XXX_GPIO_CPOUT_EN 1 |
| #define P9412_GPIO_CPOUT21_EN 2 |
| #define P9XXX_GPIO_CPOUT_CTL_EN 3 |
| #define P9XXX_GPIO_DC_SW_EN 4 |
| #define P9XXX_GPIO_ONLINE_SPOOF 5 |
| #define P9XXX_GPIO_RTX_STATE 6 |
| #define P9XXX_GPIO_RTX_SUPP 7 |
| #define P9XXX_GPIO_VBUS_EN 15 |
| |
| /* Simple Chip Specific Accessors */ |
| /* |
| * chip_get_rx_ilim |
| * |
| * Get the receive current limit (mA). |
| */ |
| static int p9221_chip_get_rx_ilim(struct p9221_charger_data *chgr, u32 *ma) |
| { |
| int ret; |
| u8 val; |
| |
| ret = chgr->reg_read_8(chgr, P9221R5_ILIM_SET_REG, &val); |
| if (ret) |
| return ret; |
| |
| *ma = ((val * 100) + 200); |
| return 0; |
| } |
| |
| static int p9412_chip_get_rx_ilim(struct p9221_charger_data *chgr, u32 *ma) |
| { |
| int ret; |
| u8 val; |
| |
| ret = chgr->reg_read_8(chgr, P9221R5_ILIM_SET_REG, &val); |
| if (ret) |
| return ret; |
| |
| *ma = (val + 1) * 100; |
| return 0; |
| } |
| |
| static int ra9530_chip_get_rx_ilim(struct p9221_charger_data *chgr, u32 *ma) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, RA9530_ILIM_REG, &val); |
| if (ret) |
| return ret; |
| |
| *ma = val; |
| return 0; |
| } |
| |
| static int p9222_chip_get_rx_ilim(struct p9221_charger_data *chgr, u32 *ma) |
| { |
| int ret; |
| u8 val; |
| |
| ret = chgr->reg_read_8(chgr, P9222RE_ILIM_SET_REG, &val); |
| if (ret) |
| return ret; |
| |
| *ma = ((val * 100) + 100); |
| return 0; |
| } |
| |
| /* |
| * chip_set_rx_ilim |
| * |
| * Set the receive current limit (mA). |
| */ |
| static int p9221_chip_set_rx_ilim(struct p9221_charger_data *chgr, u32 ma) |
| { |
| u8 val; |
| |
| /* uA -> 0.1A, offset 0.2A */ |
| if (ma < P9221_RX_ILIM_MIN_MA) |
| return -EINVAL; |
| |
| if (ma > P9221_RX_ILIM_MAX_MA) |
| ma = P9221_RX_ILIM_MAX_MA; |
| |
| val = (ma / 100) - 2; |
| return chgr->reg_write_8(chgr, P9221R5_ILIM_SET_REG, val); |
| } |
| |
| static int p9412_chip_set_rx_ilim(struct p9221_charger_data *chgr, u32 ma) |
| { |
| u8 val; |
| |
| if (ma > P9412_RX_ILIM_MAX_MA) |
| return -EINVAL; |
| |
| val = (ma / 100) - 1; |
| return chgr->reg_write_8(chgr, P9221R5_ILIM_SET_REG, val); |
| } |
| |
| static int ra9530_chip_set_rx_ilim(struct p9221_charger_data *chgr, u32 ma) |
| { |
| u16 val; |
| |
| if (ma > RA9530_RX_ILIM_MAX_MA) |
| ma = RA9530_RX_ILIM_MAX_MA; |
| |
| val = ma; |
| return chgr->reg_write_16(chgr, RA9530_ILIM_REG, val); |
| } |
| |
| static int p9222_chip_set_rx_ilim(struct p9221_charger_data *chgr, u32 ma) |
| { |
| u8 val; |
| |
| if (ma < P9222_RX_ILIM_MIN_MA) |
| return -EINVAL; |
| |
| if (ma > P9222_RX_ILIM_MAX_MA) |
| ma = P9222_RX_ILIM_MAX_MA; |
| |
| val = (ma / 100) - 1; |
| return chgr->reg_write_8(chgr, P9222RE_ILIM_SET_REG, val); |
| } |
| |
| /* |
| * chip_get_tx_ilim |
| * |
| * Get the transmit current limit (mA). |
| */ |
| static int p9221_chip_get_tx_ilim(struct p9221_charger_data *chgr, u32 *ma) |
| { |
| return -ENOTSUPP; |
| } |
| |
| static int p9382_chip_get_tx_ilim(struct p9221_charger_data *chgr, u32 *ma) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9382A_ILIM_SET_REG, &val); |
| if (ret) |
| return ret; |
| |
| *ma = (u32) val; |
| return 0; |
| } |
| |
| static int p9412_chip_get_tx_ilim(struct p9221_charger_data *chgr, u32 *ma) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9412_TX_I_API_LIM_REG, &val); |
| if (ret) |
| return ret; |
| |
| *ma = (u32) val; |
| return 0; |
| } |
| |
| /* |
| * chip_set_tx_ilim |
| * |
| * Set the transmit current limit (mA). |
| */ |
| static int p9221_chip_set_tx_ilim(struct p9221_charger_data *chgr, u32 ma) |
| { |
| return -ENOTSUPP; |
| } |
| |
| static int p9382_chip_set_tx_ilim(struct p9221_charger_data *chgr, u32 ma) |
| { |
| if ((ma < 0) || (ma > P9382A_RTX_ICL_MAX_MA)) |
| return -EINVAL; |
| |
| return chgr->reg_write_16(chgr, P9382A_ILIM_SET_REG, (u16) ma); |
| } |
| |
| static int p9412_chip_set_tx_ilim(struct p9221_charger_data *chgr, u32 ma) |
| { |
| if ((ma < 0) || (ma > P9382A_RTX_ICL_MAX_MA)) |
| return -EINVAL; |
| |
| return chgr->reg_write_16(chgr, P9412_TX_I_API_LIM_REG, (u16) ma); |
| } |
| |
| /* |
| * chip_get_die_temp |
| * |
| * Get the chip temperature (milli-C). |
| */ |
| static int p9221_chip_get_die_temp(struct p9221_charger_data *chgr, int *mc) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9221R5_DIE_TEMP_ADC_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mc = (val * 10 / 107 - 247) * 10; |
| return 0; |
| } |
| |
| static int p9412_chip_get_die_temp(struct p9221_charger_data *chgr, int *mc) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9412_DIE_TEMP_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mc = P9221_C_TO_MILLIC(val); |
| return 0; |
| } |
| |
| static int p9222_chip_get_die_temp(struct p9221_charger_data *chgr, int *mc) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9222RE_DIE_TEMP_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mc = P9221_C_TO_MILLIC(val); |
| return 0; |
| } |
| |
| /* |
| * chipeget_iout |
| * |
| * get measure of current out (mA). |
| */ |
| static int p9xxx_chip_get_iout(struct p9221_charger_data *chgr, u32 *ma) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9221R5_IOUT_REG, &val); |
| if (ret) |
| return ret; |
| |
| *ma = val; |
| return 0; |
| } |
| |
| static int p9222_chip_get_iout(struct p9221_charger_data *chgr, u32 *ma) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9222RE_IOUT_REG, &val); |
| if (ret) |
| return ret; |
| |
| *ma = val; |
| return 0; |
| } |
| |
| /* |
| * chip_get_vout |
| * |
| * get voltage out (mV). |
| */ |
| static int p9xxx_chip_get_vout(struct p9221_charger_data *chgr, u32 *mv) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9221R5_VOUT_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mv = val; |
| return 0; |
| } |
| |
| static int p9222_chip_get_vout(struct p9221_charger_data *chgr, u32 *mv) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9222RE_VOUT_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mv = val; |
| return 0; |
| } |
| |
| /* |
| * chip_get_vrect |
| * |
| * Get rectified voltage out (mV). |
| */ |
| static int p9xxx_chip_get_vrect(struct p9221_charger_data *chgr, u32 *mv) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9221R5_VRECT_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mv = val; |
| return 0; |
| } |
| |
| static int p9222_chip_get_vrect(struct p9221_charger_data *chgr, u32 *mv) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9222RE_VRECT_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mv = val; |
| return 0; |
| } |
| |
| /* |
| * chip_get_op_freq |
| * |
| * Get operating frequency (KHz). |
| */ |
| static int p9xxx_chip_get_op_freq(struct p9221_charger_data *chgr, u32 *khz) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9221R5_OP_FREQ_REG, &val); |
| if (ret) |
| return ret; |
| |
| *khz = (u32) val; |
| return 0; |
| } |
| |
| static int p9222_chip_get_op_freq(struct p9221_charger_data *chgr, u32 *khz) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9222RE_OP_FREQ_REG, &val); |
| if (ret) |
| return ret; |
| |
| *khz = (u32) val; |
| return 0; |
| } |
| static int p9xxx_chip_get_ping_freq(struct p9221_charger_data *chgr, u32 *khz) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9221R5_TX_PINGFREQ_REG, &val); |
| if (ret) |
| return ret; |
| |
| *khz = (u32) val; |
| return 0; |
| } |
| |
| static int p9222_chip_get_ping_freq(struct p9221_charger_data *chgr, u32 *khz) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9222RE_TX_PINGFREQ_REG, &val); |
| if (ret) |
| return ret; |
| |
| *khz = (u32) val; |
| return 0; |
| } |
| |
| /* |
| * chip_get_op_duty |
| * |
| * Get operating duty (%). |
| */ |
| static int p9221_chip_get_op_duty(struct p9221_charger_data *chgr, u32 *duty) |
| { |
| return -ENOTSUPP; |
| } |
| |
| static int p9xxx_chip_get_op_duty(struct p9221_charger_data *chgr, u32 *duty) |
| { |
| int ret; |
| u8 val; |
| |
| ret = chgr->reg_read_8(chgr, P9XXX_OP_DUTY_REG, &val); |
| if (ret) |
| return ret; |
| |
| *duty = (u32) val*100/256; |
| return 0; |
| } |
| |
| static int ra9530_chip_get_op_duty(struct p9221_charger_data *chgr, u32 *duty) |
| { |
| int ret; |
| u8 val; |
| |
| ret = chgr->reg_read_8(chgr, P9XXX_OP_DUTY_REG, &val); |
| if (ret) |
| return ret; |
| |
| *duty = (u32) val*50/255; |
| return 0; |
| } |
| |
| /* |
| * chip_get_op_bridge |
| * |
| * Get operating on half or full bridge. |
| */ |
| static int p9221_chip_get_op_bridge(struct p9221_charger_data *chgr, u8 *hf) |
| { |
| return -ENOTSUPP; |
| } |
| |
| static int ra9530_chip_get_op_bridge(struct p9221_charger_data *chgr, u8 *hf) |
| { |
| int ret; |
| u8 val; |
| |
| ret = chgr->reg_read_8(chgr, RA9530_TX_FB_HB_REG, &val); |
| if (ret) |
| return ret; |
| |
| *hf = val; |
| return 0; |
| } |
| |
| /* |
| * chip_get_tx_pwr |
| * |
| * Get TX power (mW). |
| */ |
| static int p9221_chip_get_tx_pwr(struct p9221_charger_data *chgr, u16 *pwr) |
| { |
| return -ENOTSUPP; |
| } |
| |
| static int p9xxx_chip_get_tx_pwr(struct p9221_charger_data *chgr, u16 *pwr) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9XXX_TX_CUR_PWR_REG, &val); |
| if (ret) |
| return ret; |
| |
| *pwr = val; |
| return 0; |
| } |
| |
| /* |
| * chip_get_rx_pwr |
| * |
| * Get RX power (mW). |
| */ |
| static int p9221_chip_get_rx_pwr(struct p9221_charger_data *chgr, u16 *pwr) |
| { |
| return -ENOTSUPP; |
| } |
| |
| static int p9xxx_chip_get_rx_pwr(struct p9221_charger_data *chgr, u16 *pwr) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9XXX_RX_CUR_PWR_REG, &val); |
| if (ret) |
| return ret; |
| |
| *pwr = val; |
| return 0; |
| } |
| |
| /* |
| * chip_get_vcpout |
| * |
| * Get CPout voltage(mV) |
| */ |
| static int p9412_chip_get_vcpout(struct p9221_charger_data *chgr, u32 *mv) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9412_VCPOUT_VOL_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mv = (u32) val; |
| return 0; |
| } |
| static int p9xxx_chip_get_vcpout(struct p9221_charger_data *chgr, u32 *mv) |
| { |
| return -ENOTSUPP; |
| } |
| |
| |
| /* |
| * chip_get_vout_max |
| * |
| * Get the maximum output voltage (mV). |
| */ |
| static int p9221_chip_get_vout_max(struct p9221_charger_data *chgr, u32 *mv) |
| { |
| return -ENOTSUPP; |
| } |
| |
| static int p9832_chip_get_vout_max(struct p9221_charger_data *chgr, u32 *mv) |
| { |
| int ret; |
| u8 val; |
| |
| ret = chgr->reg_read_8(chgr, P9221R5_VOUT_SET_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mv = val * 100; /* 100 mV units */ |
| return 0; |
| } |
| |
| static int p9412_chip_get_vout_max(struct p9221_charger_data *chgr, u32 *mv) |
| { |
| int ret; |
| u16 val; |
| |
| ret = chgr->reg_read_16(chgr, P9412_VOUT_SET_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mv = val * 10; /* 10 mV units */ |
| return 0; |
| } |
| |
| static int p9222_chip_get_vout_max(struct p9221_charger_data *chgr, u32 *mv) |
| { |
| int ret; |
| u8 val; |
| |
| ret = chgr->reg_read_8(chgr, P9222RE_VOUT_SET_REG, &val); |
| if (ret) |
| return ret; |
| |
| *mv = val * 100 + 3500; /* Vout = value * 100mV + 3500mV */ |
| return 0; |
| } |
| |
| /* |
| * chip_set_vout_max |
| * |
| * Set the maximum output voltage (mV). |
| */ |
| static int p9221_chip_set_vout_max(struct p9221_charger_data *chgr, u32 mv) |
| { |
| return -ENOTSUPP; |
| } |
| |
| static int p9832_chip_set_vout_max(struct p9221_charger_data *chgr, u32 mv) |
| { |
| int ret; |
| |
| if (mv < P9221_VOUT_SET_MIN_MV || mv > chgr->pdata->max_vout_mv) |
| return -EINVAL; |
| |
| ret = chgr->reg_write_8(chgr, P9221R5_VOUT_SET_REG, mv / 100); |
| return ret; |
| } |
| |
| static int p9412_chip_set_vout_max(struct p9221_charger_data *chgr, u32 mv) |
| { |
| int ret; |
| |
| if (mv < P9412_VOUT_SET_MIN_MV || mv > chgr->pdata->max_vout_mv) |
| return -EINVAL; |
| |
| dev_dbg(&chgr->client->dev, "%s: vout setting to: %u, caller: %pS\n", |
| __func__, mv, __builtin_return_address(2)); |
| |
| ret = chgr->reg_write_16(chgr, P9412_VOUT_SET_REG, mv / 10); |
| return ret; |
| } |
| |
| static int ra9530_chip_set_vout_max(struct p9221_charger_data *chgr, u32 mv) |
| { |
| int ret; |
| |
| if (mv < P9412_VOUT_SET_MIN_MV || mv > chgr->pdata->max_vout_mv) |
| return -EINVAL; |
| |
| dev_dbg(&chgr->client->dev, "%s: vout setting to: %u, caller: %pS\n", |
| __func__, mv, __builtin_return_address(2)); |
| |
| ret = chgr->reg_write_16(chgr, P9412_VOUT_SET_REG, mv / 10); |
| |
| if (chgr->chg_mode_votable && chgr->chg_mode_off) { |
| gvotable_cast_long_vote(chgr->chg_mode_votable, |
| P9221_WLC_VOTER, |
| 0, false); |
| chgr->chg_mode_off = false; |
| } |
| |
| return ret; |
| } |
| |
| static int p9222_chip_set_vout_max(struct p9221_charger_data *chgr, u32 mv) |
| { |
| int ret; |
| |
| if (mv < P9222_VOUT_SET_MIN_MV || mv > P9222_VOUT_SET_MAX_MV) |
| return -EINVAL; |
| |
| ret = chgr->reg_write_8(chgr, P9222RE_VOUT_SET_REG, (mv - 3500) / 100); |
| return ret; |
| } |
| |
| /* |
| * chip_get_tx_epp_guarpwr |
| * |
| * get EPP Tx GuarPwr (W) |
| */ |
| static int p9xxx_get_tx_epp_guarpwr(struct p9221_charger_data *chgr, u32 *tx_pwr) |
| { |
| u8 val8; |
| int ret; |
| |
| ret = chgr->reg_read_8(chgr, chgr->reg_epp_tx_guarpwr_addr, &val8); |
| if (ret) |
| return ret; |
| |
| *tx_pwr = P9412_HW_TO_MW(val8); |
| |
| return 0; |
| } |
| |
| /* system mode register */ |
| static int p9221_chip_get_sys_mode(struct p9221_charger_data *chgr, u8 *mode) |
| { |
| int ret; |
| u8 val8; |
| |
| ret = chgr->reg_read_8(chgr, P9221R5_SYSTEM_MODE_REG, &val8); |
| if (ret) |
| return ret; |
| |
| /* map to p9412 values */ |
| if (val8 & P9221R5_MODE_WPCMODE) |
| *mode = P9XXX_SYS_OP_MODE_WPC_BASIC; |
| else if (val8 & P9382A_MODE_TXMODE) |
| *mode = P9XXX_SYS_OP_MODE_TX_MODE; |
| else if (val8 & P9221R5_MODE_EXTENDED) |
| *mode = P9XXX_SYS_OP_MODE_WPC_EXTD; |
| else |
| *mode = P9XXX_SYS_OP_MODE_AC_MISSING; |
| return 0; |
| } |
| |
| static int p9222_chip_get_sys_mode(struct p9221_charger_data *chgr, u8 *mode) |
| { |
| int ret; |
| u8 val8; |
| |
| ret = chgr->reg_read_8(chgr, P9222RE_SYSTEM_MODE_REG, &val8); |
| if (ret) |
| return ret; |
| |
| /* map to p9412 values */ |
| if (val8 & P9222_SYS_OP_MODE_WPC_BASIC) |
| *mode = P9XXX_SYS_OP_MODE_WPC_BASIC; |
| else if (val8 & P9222_SYS_OP_MODE_WPC_EXTD) |
| *mode = P9XXX_SYS_OP_MODE_WPC_EXTD; |
| else |
| *mode = P9XXX_SYS_OP_MODE_AC_MISSING; |
| |
| return 0; |
| } |
| |
| static int p9412_chip_get_sys_mode(struct p9221_charger_data *chgr, u8 *mode) |
| { |
| return chgr->reg_read_8(chgr, P9221R5_SYSTEM_MODE_REG, mode); |
| } |
| |
| /* These are more involved than just chip access */ |
| |
| static int p9382_wait_for_mode(struct p9221_charger_data *chgr, u8 mode, u8 mask) |
| { |
| int loops, ret; |
| uint8_t sys_mode; |
| |
| /* 20 * 50 = 1 sec */ |
| for (loops = 20 ; loops ; loops--) { |
| ret = chgr->reg_read_8(chgr, P9221R5_SYSTEM_MODE_REG, |
| &sys_mode); |
| if (ret < 0) { |
| dev_err(&chgr->client->dev, |
| "cannot read system_mode (%d)", ret); |
| return -EIO; |
| } |
| |
| if ((sys_mode & mask) == mode) |
| return 0; |
| |
| msleep(50); |
| } |
| |
| return -ETIMEDOUT; |
| } |
| |
| /* get the data buf for receive */ |
| static int p9221_get_data_buf(struct p9221_charger_data *chgr, |
| u8 data[], size_t len) |
| { |
| if (!len || len > P9221R5_DATA_RECV_BUF_SIZE) |
| return -EINVAL; |
| |
| return chgr->reg_read_n(chgr, P9221R5_DATA_RECV_BUF_START, data, len); |
| } |
| |
| static int p9222_get_data_buf(struct p9221_charger_data *chgr, |
| u8 data[], size_t len) |
| { |
| if (!len || len > P9222RE_DATA_BUF_SIZE) |
| return -EINVAL; |
| |
| return chgr->reg_read_n(chgr, P9222RE_DATA_BUF_START, data, len); |
| } |
| |
| static int p9382_get_data_buf(struct p9221_charger_data *chgr, |
| u8 data[], size_t len) |
| { |
| if (!len || len > P9221R5_DATA_RECV_BUF_SIZE) |
| return -EINVAL; |
| |
| return chgr->reg_read_n(chgr, P9382A_DATA_RECV_BUF_START, data, len); |
| } |
| |
| static int p9412_get_data_buf(struct p9221_charger_data *chgr, |
| u8 data[], size_t len) |
| { |
| if (!len || len > P9412_DATA_BUF_SIZE) |
| return -EINVAL; |
| |
| return chgr->reg_read_n(chgr, P9412_DATA_BUF_START, data, len); |
| } |
| |
| static int ra9530_get_data_buf(struct p9221_charger_data *chgr, |
| u8 data[], size_t len) |
| { |
| if (!len || len > RA9530_DATA_BUF_SIZE) |
| return -EINVAL; |
| |
| return chgr->reg_read_n(chgr, RA9530_DATA_BUF_READ_START, data, len); |
| |
| } |
| |
| /* set the data buf for send */ |
| static int p9221_set_data_buf(struct p9221_charger_data *chgr, |
| const u8 data[], size_t len) |
| { |
| if (!len || len > P9221R5_DATA_SEND_BUF_SIZE) |
| return -EINVAL; |
| |
| return chgr->reg_write_n(chgr, P9221R5_DATA_SEND_BUF_START, data, len); |
| } |
| |
| static int p9222_set_data_buf(struct p9221_charger_data *chgr, |
| const u8 data[], size_t len) |
| { |
| if (!len || len > P9222RE_DATA_BUF_SIZE) |
| return -EINVAL; |
| |
| return chgr->reg_write_n(chgr, P9222RE_DATA_BUF_START, data, len); |
| } |
| |
| static int p9382_set_data_buf(struct p9221_charger_data *chgr, |
| const u8 data[], size_t len) |
| { |
| if (!len || len > P9221R5_DATA_SEND_BUF_SIZE) |
| return -EINVAL; |
| |
| return chgr->reg_write_n(chgr, P9382A_DATA_SEND_BUF_START, data, len); |
| } |
| |
| static int p9412_set_data_buf(struct p9221_charger_data *chgr, |
| const u8 data[], size_t len) |
| { |
| if (!len || len > P9412_DATA_BUF_SIZE) |
| return -EINVAL; |
| |
| return chgr->reg_write_n(chgr, P9412_DATA_BUF_START, data, len); |
| } |
| |
| static int ra9530_set_data_buf(struct p9221_charger_data *chgr, |
| const u8 data[], size_t len) |
| { |
| if (!len || len > RA9530_DATA_BUF_SIZE) |
| return -EINVAL; |
| |
| return chgr->reg_write_n(chgr, RA9530_DATA_BUF_WRITE_START, data, len); |
| } |
| /* receive size */ |
| static int p9221_get_cc_recv_size(struct p9221_charger_data *chgr, size_t *len) |
| { |
| int ret; |
| u8 len8; |
| |
| ret = chgr->reg_read_8(chgr, P9221R5_COM_CHAN_RECV_SIZE_REG, &len8); |
| if (!ret) |
| *len = len8; |
| return ret; |
| } |
| |
| static int p9222_get_cc_recv_size(struct p9221_charger_data *chgr, size_t *len) |
| { |
| int ret; |
| u8 len8; |
| |
| ret = chgr->reg_read_8(chgr, P9222RE_COM_CHAN_RECV_SIZE_REG, &len8); |
| if (ret == 0) |
| *len = len8; |
| return ret; |
| } |
| |
| static int p9412_get_cc_recv_size(struct p9221_charger_data *chgr, size_t *len) |
| { |
| int ret; |
| u16 len16; |
| |
| ret = chgr->reg_read_16(chgr, P9412_COM_CHAN_RECV_SIZE_REG, &len16); |
| if (!ret) |
| *len = len16; |
| return ret; |
| } |
| |
| static int ra9530_get_cc_recv_size(struct p9221_charger_data *chgr, size_t *len) |
| { |
| int ret; |
| u16 len16; |
| |
| ret = chgr->reg_read_16(chgr, RA9530_CC_READ_SIZE_REG, &len16); |
| if (!ret) |
| *len = len16; |
| return ret; |
| } |
| |
| /* send size */ |
| static int p9221_set_cc_send_size(struct p9221_charger_data *chgr, size_t len) |
| { |
| return chgr->reg_write_8(chgr, P9221R5_COM_CHAN_SEND_SIZE_REG, len); |
| } |
| |
| static int p9222_set_cc_send_size(struct p9221_charger_data *chgr, size_t len) |
| { |
| int ret; |
| |
| /* set packet type(BiDi) */ |
| ret = chgr->reg_write_8(chgr, chgr->reg_packet_type_addr, |
| BIDI_COM_PACKET_TYPE); |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "Failed to write packet type %d\n", ret); |
| return ret; |
| } |
| |
| return chgr->reg_write_16(chgr, P9222RE_COM_CHAN_SEND_SIZE_REG, len); |
| } |
| |
| static int p9382_set_cc_send_size(struct p9221_charger_data *chgr, size_t len) |
| { |
| int ret; |
| |
| /* set packet type to 0x100 */ |
| ret = chgr->reg_write_8(chgr, P9382A_COM_PACKET_TYPE_ADDR, |
| BIDI_COM_PACKET_TYPE); |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "Failed to write packet type %d\n", ret); |
| return ret; |
| } |
| |
| return chgr->reg_write_16(chgr, P9382A_COM_CHAN_SEND_SIZE_REG, len); |
| } |
| |
| static int p9412_set_cc_send_size(struct p9221_charger_data *chgr, size_t len) |
| { |
| int ret; |
| |
| /* set as ADT type(Authentication) */ |
| if (chgr->auth_type != 0) { |
| ret = chgr->reg_write_8(chgr, P9412_COM_PACKET_TYPE_ADDR, |
| (P9412_ADT_TYPE_AUTH << 3) | |
| (chgr->tx_len >> 8)); |
| ret |= chgr->reg_write_8(chgr, P9412_COM_PACKET_TYPE_ADDR + 1, |
| chgr->tx_len & 0xFF); |
| chgr->auth_type = 0; |
| } else { |
| /* set packet type(BiDi 0x98) to 0x800 */ |
| ret = chgr->reg_write_8(chgr, P9412_COM_PACKET_TYPE_ADDR, |
| BIDI_COM_PACKET_TYPE); |
| } |
| |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "Failed to write packet type %d\n", ret); |
| return ret; |
| } |
| |
| return chgr->reg_write_16(chgr, P9412_COM_CHAN_SEND_SIZE_REG, len); |
| } |
| |
| static int ra9530_set_cc_send_size(struct p9221_charger_data *chgr, size_t len) |
| { |
| int ret; |
| |
| /* set as ADT type(Authentication) */ |
| if (chgr->auth_type != 0) { |
| ret = chgr->reg_write_8(chgr, RA9530_CC_WRITE_TYPE_REG, |
| (P9412_ADT_TYPE_AUTH << RA9530_CC_TYPE_SHIFT) | |
| (chgr->tx_len >> 8)); |
| ret |= chgr->reg_write_8(chgr, RA9530_CC_WRITE_TYPE_REG + 1, |
| chgr->tx_len & 0xFF); |
| chgr->auth_type = 0; |
| } else { |
| /* set packet type(BiDi 0x98) to 0x800 */ |
| ret = chgr->reg_write_8(chgr, RA9530_CC_WRITE_TYPE_REG, |
| (RA9530_BIDI_COM_PACKET_TYPE << RA9530_CC_TYPE_SHIFT) | |
| (chgr->tx_len >> 8)); |
| ret |= chgr->reg_write_8(chgr, RA9530_CC_WRITE_TYPE_REG + 1, |
| chgr->tx_len & 0xFF); |
| } |
| |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "Failed to write packet type %d\n", ret); |
| return ret; |
| } |
| |
| return chgr->reg_write_16(chgr, RA9530_CC_WRITE_SIZE_REG, len); |
| } |
| |
| /* get align x */ |
| static int p9221_get_align_x(struct p9221_charger_data *chgr, u8 *x) |
| { |
| return chgr->reg_read_8(chgr, P9221R5_ALIGN_X_ADC_REG, x); |
| } |
| |
| static int p9412_get_align_x(struct p9221_charger_data *chgr, u8 *x) |
| { |
| return chgr->reg_read_8(chgr, P9412_ALIGN_X_REG, x); |
| } |
| |
| static int ra9530_get_align_x(struct p9221_charger_data *chgr, u8 *x) |
| { |
| // Not support |
| *x = 0; |
| return 0; |
| } |
| |
| /* get align y */ |
| static int p9221_get_align_y(struct p9221_charger_data *chgr, u8 *y) |
| { |
| return chgr->reg_read_8(chgr, P9221R5_ALIGN_Y_ADC_REG, y); |
| } |
| |
| static int p9412_get_align_y(struct p9221_charger_data *chgr, u8 *y) |
| { |
| return chgr->reg_read_8(chgr, P9412_ALIGN_Y_REG, y); |
| } |
| |
| static int ra9530_get_align_y(struct p9221_charger_data *chgr, u8 *y) |
| { |
| // Not support |
| *y = 0; |
| return 0; |
| } |
| |
| /* Simple Chip Specific Logic Functions */ |
| |
| /* Disable/Enable transmit mode. */ |
| static int p9221_chip_tx_mode(struct p9221_charger_data *chgr, bool enable) |
| { |
| logbuffer_log(chgr->rtx_log, "%s(%d)", __func__, enable); |
| return -ENOTSUPP; |
| } |
| |
| static int p9382_chip_tx_mode(struct p9221_charger_data *chgr, bool enable) |
| { |
| int ret; |
| u16 rev; |
| |
| logbuffer_log(chgr->rtx_log, "%s(%d)", __func__, enable); |
| if (enable) { |
| /* check FW revision */ |
| ret = chgr->reg_read_16(chgr, |
| P9221_OTP_FW_MINOR_REV_REG, &rev); |
| if (ret) |
| return ret; |
| |
| if (rev >= P9382A_FW_REV_25) { |
| /* write 0x0003 to 0x69 after rev 25 */ |
| ret = chgr->reg_write_16(chgr, |
| P9382A_TRX_ENABLE_REG, |
| P9382A_TX_INHIBIT); |
| } else { |
| /* write 0x0000 to 0x34 */ |
| ret = chgr->reg_write_16(chgr, |
| P9382A_STATUS_REG, 0); |
| } |
| |
| /* check 0x4C reads back as 0x04 */ |
| ret = p9382_wait_for_mode(chgr, P9382A_MODE_TXMODE, P9382A_MODE_MASK); |
| } else { |
| /* Write 0x80 to 0x4E, check 0x4C reads back as 0x0000 */ |
| ret = chgr->chip_set_cmd(chgr, P9221R5_COM_RENEGOTIATE); |
| if (ret == 0) { |
| ret = p9382_wait_for_mode(chgr, 0, P9382A_MODE_MASK); |
| if (ret < 0) |
| pr_err("cannot exit rTX mode (%d)\n", ret); |
| } |
| } |
| |
| return ret; |
| } |
| |
| static int p9412_chip_tx_apbst_enable(struct p9221_charger_data *chgr) |
| { |
| const u8 val8 = chgr->pdata->hw_ocp_det ? P9412_APBSTPING_7V : 0; |
| int ret; |
| |
| if (!chgr->pdata->apbst_en) |
| return 0; |
| |
| ret = chgr->reg_write_8(chgr, P9412_APBSTPING_REG, val8); |
| ret |= chgr->reg_write_8(chgr, P9412_APBSTCONTROL_REG, P9412_APBSTPING_7V); |
| logbuffer_log(chgr->rtx_log, "configure Ext-Boost Vout to %s.(%d)", |
| val8 > 0 ? "7V" : "5V", ret); |
| |
| if (!chgr->pdata->hw_ocp_det) |
| return ret; |
| /* Set ping phase current limit to 1.2A */ |
| ret |= chgr->reg_write_8(chgr, P9412_PLIM_REG, P9412_PLIM_1200MA); |
| logbuffer_log(chgr->rtx_log, "write %#02x to %#02x", P9412_PLIM_1200MA, P9412_PLIM_REG); |
| |
| return ret; |
| } |
| |
| static void rtx_current_limit_opt(struct p9221_charger_data *chgr) |
| { |
| int ret; |
| |
| /* Set API limit to 1.35A */ |
| ret = chgr->reg_write_16(chgr, P9412_I_API_Limit, chgr->rtx_api_limit); |
| logbuffer_log(chgr->rtx_log, "set api limit to %dMA", chgr->rtx_api_limit); |
| /* Set API Hyst to 0.8A */ |
| ret |= chgr->reg_write_8(chgr, P9412_I_API_Hys, P9412_I_API_Hys_08); |
| |
| if (ret < 0) |
| logbuffer_log(chgr->rtx_log, "fail to set RTx current limit, ret=%d\n", ret); |
| } |
| |
| static int p9412_chip_tx_mode(struct p9221_charger_data *chgr, bool enable) |
| { |
| int ret; |
| |
| logbuffer_log(chgr->rtx_log, "%s(%d)", __func__, enable); |
| |
| if (!enable) { |
| ret = chgr->chip_set_cmd(chgr, P9412_CMD_TXMODE_EXIT); |
| if (ret == 0) { |
| ret = p9382_wait_for_mode(chgr, 0, P9412_MODE_MASK); |
| if (ret < 0) |
| pr_err("cannot exit rTX mode (%d)", ret); |
| } |
| if (chgr->pdata->apbst_en) { |
| ret = chgr->reg_write_8(chgr, P9412_APBSTPING_REG, 0); |
| logbuffer_log(chgr->rtx_log, |
| "configure Ext-Boost back to 5V.(%d)", ret); |
| } |
| return ret; |
| } |
| |
| ret = p9412_chip_tx_apbst_enable(chgr); |
| if (ret < 0) |
| return ret; |
| |
| ret = chgr->reg_write_8(chgr, P9412_TX_CMD_REG, P9412_TX_CMD_TX_MODE_EN); |
| if (ret) { |
| logbuffer_log(chgr->rtx_log, |
| "tx_cmd_reg write failed (%d)", ret); |
| return ret; |
| } |
| ret = p9382_wait_for_mode(chgr, P9XXX_SYS_OP_MODE_TX_MODE, P9412_MODE_MASK); |
| if (ret) { |
| logbuffer_log(chgr->rtx_log, |
| "error waiting for tx_mode (%d)", ret); |
| return ret; |
| } |
| |
| ret = chgr->reg_write_16(chgr, P9412_TXOCP_REG, P9412_TXOCP_1400MA); |
| logbuffer_log(chgr->rtx_log, "configure TX OCP to %dMA", P9412_TXOCP_1400MA); |
| if (ret < 0) |
| return ret; |
| if (chgr->pdata->hw_ocp_det) { |
| rtx_current_limit_opt(chgr); |
| /* Set Frequency low limit to 120kHz */ |
| ret = chgr->reg_write_16(chgr, P9412_MIN_FREQ_PER, P9412_MIN_FREQ_PER_120); |
| if (ret < 0) |
| logbuffer_log(chgr->rtx_log, "fail to set frequency low limit, ret=%d\n", ret); |
| } |
| |
| /* Set Foreign Object Detection Threshold to 1600mW */ |
| ret = chgr->reg_write_16(chgr, P9412_TX_FOD_THRSH_REG, P9412_TX_FOD_THRSH_1600); |
| if (ret < 0) |
| logbuffer_log(chgr->rtx_log, "fail to set RTxFOD threshold, ret=%d\n", ret); |
| |
| if (chgr->pdata->apbst_en && !chgr->pdata->hw_ocp_det) |
| mod_delayed_work(system_wq, &chgr->chk_rtx_ocp_work, 0); |
| |
| return 0; |
| } |
| |
| static int ra9530_chip_tx_mode(struct p9221_charger_data *chgr, bool enable) |
| { |
| int ret = 0; |
| u16 val; |
| |
| dev_dbg(&chgr->client->dev, "%s(%d)\n", __func__, enable); |
| |
| if (!enable) { |
| ret = chgr->chip_set_cmd(chgr, P9412_CMD_TXMODE_EXIT); |
| if (ret == 0) { |
| ret = p9382_wait_for_mode(chgr, 0, RA9530_MODE_MASK); |
| if (ret < 0) |
| pr_err("cannot exit rTX mode (%d)", ret); |
| } |
| logbuffer_log(chgr->rtx_log, "rtx mode=0"); |
| return ret; |
| } |
| |
| /* Set ping phase current limit to 0.9A */ |
| ret = chgr->reg_write_16(chgr, RA9530_PLIM_REG, chgr->ra9530_rtx_plim); |
| logbuffer_log(chgr->rtx_log, "write %#02x to %#02x", chgr->ra9530_rtx_plim, RA9530_PLIM_REG); |
| |
| if (ret < 0) |
| return ret; |
| |
| ret = chgr->reg_write_8(chgr, P9412_TX_CMD_REG, P9412_TX_CMD_TX_MODE_EN); |
| if (ret) { |
| logbuffer_log(chgr->rtx_log, |
| "tx_cmd_reg write failed (%d)", ret); |
| return ret; |
| } |
| ret = p9382_wait_for_mode(chgr, P9XXX_SYS_OP_MODE_TX_MODE, RA9530_MODE_MASK); |
| if (ret) { |
| logbuffer_log(chgr->rtx_log, |
| "error waiting for tx_mode (%d)", ret); |
| return ret; |
| } |
| ret = chgr->reg_write_16(chgr, P9412_TXOCP_REG, chgr->rtx_ocp); |
| logbuffer_log(chgr->rtx_log, "configure TX OCP to %dMA", chgr->rtx_ocp); |
| if (ret < 0) |
| return ret; |
| |
| rtx_current_limit_opt(chgr); |
| /* |
| * Set FB Frequency low limit to 120kHz |
| * val = (clock/freq)-1 in number of 60MHz clock cycles |
| */ |
| val = chgr->rtx_fb_freq_low_limit; |
| ret = chgr->reg_write_16(chgr, RA9530_FB_MIN_FREQ_REG, val); |
| logbuffer_log(chgr->rtx_log, "set FB freq min: write %#02x to %#02x", |
| val, RA9530_FB_MIN_FREQ_REG); |
| if (ret < 0) |
| logbuffer_log(chgr->rtx_log, "min FB freq fail, ret=%d\n", ret); |
| |
| /* Set HB Frequency low limit to 105kHz */ |
| val = chgr->rtx_hb_freq_low_limit; |
| ret = chgr->reg_write_16(chgr, RA9530_HB_MIN_FREQ_REG, val); |
| logbuffer_log(chgr->rtx_log, "set HB freq min: write %#02x to %#02x", |
| val, RA9530_HB_MIN_FREQ_REG); |
| if (ret < 0) |
| logbuffer_log(chgr->rtx_log, "min HB freq fail, ret=%d\n", ret); |
| |
| val = chgr->rtx_hb_ping_freq; |
| ret = chgr->reg_write_16(chgr, RA9530_HB_PING_FREQ_REG, val); |
| logbuffer_log(chgr->rtx_log, "set HB ping freq: write %#02x to %#02x", |
| val, RA9530_HB_PING_FREQ_REG); |
| if (ret < 0) |
| logbuffer_log(chgr->rtx_log, "HB ping freq fail, ret=%d\n", ret); |
| |
| /* Set Foreign Object Detection Threshold to 1600mW */ |
| ret = chgr->reg_write_16(chgr, P9412_TX_FOD_THRSH_REG, chgr->rtx_fod_thrsh); |
| logbuffer_log(chgr->rtx_log, "set RTxFOD threshold : %dMW", chgr->rtx_fod_thrsh); |
| if (ret < 0) |
| logbuffer_log(chgr->rtx_log, "RTxFOD fail, ret=%d\n", ret); |
| |
| /* TODO: b/289877860 check if this is needed in RA9530 */ |
| //if (chgr->pdata->apbst_en && !chgr->pdata->hw_ocp_det) |
| // mod_delayed_work(system_wq, &chgr->chk_rtx_ocp_work, 0); |
| |
| logbuffer_log(chgr->rtx_log, "rtx mode=1"); |
| |
| return 0; |
| } |
| |
| static int p9222_chip_set_cmd_reg(struct p9221_charger_data *chgr, u16 cmd) |
| { |
| u16 cur_cmd = 0; |
| int retry; |
| int ret; |
| |
| for (retry = 0; retry < P9221_COM_CHAN_RETRIES; retry++) { |
| ret = chgr->reg_read_16(chgr, P9222RE_COM_REG, &cur_cmd); |
| if (ret == 0 && cur_cmd == 0) |
| break; |
| msleep(25); |
| } |
| |
| if (retry >= P9221_COM_CHAN_RETRIES) { |
| dev_err(&chgr->client->dev, |
| "Failed to wait for cmd free %02x\n", cur_cmd); |
| return -EBUSY; |
| } |
| |
| ret = chgr->reg_write_16(chgr, P9222RE_COM_REG, (u16)cmd); |
| if (ret) |
| dev_err(&chgr->client->dev, |
| "Failed to set cmd reg %02x: %d\n", (u16)cmd, ret); |
| |
| return ret; |
| } |
| |
| static int p9xxx_chip_set_cmd_reg(struct p9221_charger_data *chgr, u16 cmd) |
| { |
| u16 cur_cmd = 0; |
| int retry; |
| int ret; |
| |
| for (retry = 0; retry < P9221_COM_CHAN_RETRIES; retry++) { |
| ret = chgr->reg_read_16(chgr, P9221_COM_REG, &cur_cmd); |
| if (ret == 0 && cur_cmd == 0) |
| break; |
| msleep(25); |
| } |
| |
| if (retry >= P9221_COM_CHAN_RETRIES) { |
| dev_err(&chgr->client->dev, |
| "Failed to wait for cmd free %02x\n", cur_cmd); |
| return -EBUSY; |
| } |
| |
| ret = chgr->reg_write_16(chgr, P9221_COM_REG, cmd); |
| if (ret) |
| dev_err(&chgr->client->dev, |
| "Failed to set cmd reg %02x: %d\n", cmd, ret); |
| |
| return ret; |
| } |
| |
| /* ccreset */ |
| static int p9221_send_ccreset(struct p9221_charger_data *chgr) |
| { |
| int ret; |
| |
| dev_info(&chgr->client->dev, "%s CC reset\n", |
| chgr->is_mfg_google? "Send" : "Ignore"); |
| |
| if (chgr->is_mfg_google == false) |
| return 0; |
| |
| mutex_lock(&chgr->cmd_lock); |
| |
| ret = chgr->reg_write_8(chgr, P9221R5_COM_CHAN_RESET_REG, |
| P9221R5_COM_CHAN_CCRESET); |
| if (ret == 0) |
| ret = chgr->chip_set_cmd(chgr, P9221R5_COM_CCACTIVATE); |
| |
| mutex_unlock(&chgr->cmd_lock); |
| return ret; |
| } |
| |
| static int p9412_send_ccreset(struct p9221_charger_data *chgr) |
| { |
| int ret = 0; |
| |
| dev_info(&chgr->client->dev, "%s CC reset\n", |
| chgr->is_mfg_google? "Send" : "Ignore"); |
| |
| if (chgr->is_mfg_google == false) |
| return 0; |
| |
| mutex_lock(&chgr->cmd_lock); |
| |
| if (chgr->cc_reset_pending) { |
| mutex_unlock(&chgr->cmd_lock); |
| goto error_done; |
| } |
| |
| chgr->cc_reset_pending = true; |
| |
| ret = chgr->reg_write_8(chgr, P9412_COM_PACKET_TYPE_ADDR, |
| CHANNEL_RESET_PACKET_TYPE); |
| ret |= chgr->reg_write_8(chgr, P9412_COM_CHAN_SEND_SIZE_REG, 0); |
| if (ret == 0) |
| ret = chgr->chip_set_cmd(chgr, P9412_COM_CCACTIVATE); |
| |
| mutex_unlock(&chgr->cmd_lock); |
| |
| if (ret < 0) { |
| chgr->cc_reset_pending = false; |
| goto error_done; |
| } |
| |
| /* ccreset needs 100ms, so set the timeout to 500ms */ |
| wait_event_timeout(chgr->ccreset_wq, chgr->cc_reset_pending == false, |
| msecs_to_jiffies(500)); |
| |
| if (chgr->cc_reset_pending) { |
| /* TODO: offline or command failed? */ |
| chgr->cc_reset_pending = false; |
| ret = -ETIMEDOUT; |
| } |
| |
| error_done: |
| if (ret != 0) |
| dev_err(&chgr->client->dev, "Error sending CC reset (%d)\n", |
| ret); |
| return ret; |
| } |
| |
| static int ra9530_send_ccreset(struct p9221_charger_data *chgr) |
| { |
| int ret = 0; |
| |
| dev_info(&chgr->client->dev, "%s CC reset\n", |
| chgr->is_mfg_google? "Send" : "Ignore"); |
| |
| if (chgr->is_mfg_google == false) |
| return 0; |
| |
| mutex_lock(&chgr->cmd_lock); |
| |
| if (chgr->cc_reset_pending) { |
| mutex_unlock(&chgr->cmd_lock); |
| goto error_done; |
| } |
| |
| chgr->cc_reset_pending = true; |
| |
| ret = chgr->reg_write_8(chgr, RA9530_CC_WRITE_TYPE_REG, |
| CHANNEL_RESET_PACKET_TYPE); |
| ret |= chgr->reg_write_8(chgr, RA9530_CC_WRITE_SIZE_REG, 0); |
| if (ret == 0) |
| ret = chgr->chip_set_cmd(chgr, P9412_COM_CCACTIVATE); |
| |
| mutex_unlock(&chgr->cmd_lock); |
| |
| if (ret < 0) { |
| chgr->cc_reset_pending = false; |
| goto error_done; |
| } |
| |
| /* ccreset needs 100ms, so set the timeout to 500ms */ |
| wait_event_timeout(chgr->ccreset_wq, chgr->cc_reset_pending == false, |
| msecs_to_jiffies(500)); |
| |
| if (chgr->cc_reset_pending) { |
| /* TODO: offline or command failed? */ |
| chgr->cc_reset_pending = false; |
| ret = -ETIMEDOUT; |
| } |
| |
| error_done: |
| if (ret != 0) |
| dev_err(&chgr->client->dev, "Error sending CC reset (%d)\n", |
| ret); |
| return ret; |
| } |
| |
| /* send eop */ |
| /* send multiple times to make sure it works */ |
| #define EOP_RESTART_COUNT 2 |
| #define P9222_EOP_REPEAT_COUNT 4 |
| static int p9222_send_repeat_eop(struct p9221_charger_data *chgr, u8 reason) |
| { |
| int count, ret = 0; |
| u8 val, ept_reason; |
| |
| for (count = 0; count < P9222_EOP_REPEAT_COUNT; count++) { |
| ept_reason = (count < EOP_RESTART_COUNT) ? P9221_EOP_RESTART_POWER : reason; |
| |
| ret = chgr->reg_write_8(chgr, P9222RE_EPT_REG, ept_reason); |
| if (ret == 0) |
| ret = chgr->chip_set_cmd(chgr, P9221R5_COM_SENDEPT); |
| if (ret < 0) { |
| dev_err(&chgr->client->dev, "fail send eop_%d (%d)\n", count, ret); |
| return ret; |
| } |
| mdelay(500); |
| |
| /* Check Tx is offline due to EPT command works */ |
| ret = chgr->reg_read_8(chgr, P9221_STATUS_REG, &val); |
| if (ret < 0) { |
| dev_info(&chgr->client->dev, |
| "WLC chip offline, count=%d, ret=%d\n", count, ret); |
| break; |
| } |
| } |
| |
| dev_info(&chgr->client->dev, |
| "send 3xEOP command success(reason=%02x)\n", reason); |
| |
| return 0; |
| } |
| |
| static int p9221_send_eop(struct p9221_charger_data *chgr, u8 reason) |
| { |
| int ret; |
| |
| dev_info(&chgr->client->dev, "Send EOP reason=%d\n", reason); |
| |
| mutex_lock(&chgr->cmd_lock); |
| |
| ret = chgr->reg_write_8(chgr, P9221R5_EPT_REG, reason); |
| if (ret == 0) |
| ret = chgr->chip_set_cmd(chgr, P9221R5_COM_SENDEPT); |
| |
| mutex_unlock(&chgr->cmd_lock); |
| return ret; |
| } |
| /* send eop */ |
| static int p9222_send_eop(struct p9221_charger_data *chgr, u8 reason) |
| { |
| int ret; |
| |
| dev_info(&chgr->client->dev, "Send P9222 EOP reason=%d\n", reason); |
| |
| mutex_lock(&chgr->cmd_lock); |
| |
| if (chgr->pdata->disable_repeat_eop) { |
| ret = chgr->reg_write_8(chgr, P9222RE_EPT_REG, reason); |
| if (ret == 0) |
| ret = chgr->chip_set_cmd(chgr, P9221R5_COM_SENDEPT); |
| } else { |
| ret = p9222_send_repeat_eop(chgr, reason); |
| } |
| |
| mutex_unlock(&chgr->cmd_lock); |
| |
| return ret; |
| } |
| |
| /* 3 times to make sure it works */ |
| static int p9412_send_3eop(struct p9221_charger_data *chgr, u8 reason) |
| { |
| int count, ret = 0; |
| u8 val; |
| |
| for (count = 0; count < 3; count++) { |
| /* Command P9412 to send EPT */ |
| ret = chgr->reg_write_8(chgr, P9221R5_EPT_REG, reason); |
| if (ret == 0) |
| ret = chgr->chip_set_cmd(chgr, P9221R5_COM_SENDEPT); |
| if (ret < 0) { |
| dev_err(&chgr->client->dev, "fail send eop%d (%d)\n", |
| count, ret); |
| return ret; |
| } else |
| dev_info(&chgr->client->dev, "send eop command success\n"); |
| |
| mdelay(500); |
| |
| ret = chgr->reg_read_8(chgr, P9221_STATUS_REG, &val); |
| if (ret < 0) |
| return ret; |
| } |
| |
| return ret; |
| } |
| |
| static int p9412_send_eop(struct p9221_charger_data *chgr, u8 reason) |
| { |
| int ret = 0; |
| |
| dev_info(&chgr->client->dev, "Send EOP reason=%d\n", reason); |
| |
| mutex_lock(&chgr->cmd_lock); |
| |
| if (reason == P9221_EOP_REVERT_TO_BPP) { |
| ret = chgr->reg_write_8(chgr, P9221R5_EPT_REG, reason); |
| if (ret == 0) |
| ret = chgr->chip_set_cmd(chgr, P9221R5_COM_SENDEPT); |
| } else { |
| ret = p9412_send_3eop(chgr, reason); |
| } |
| |
| mutex_unlock(&chgr->cmd_lock); |
| |
| return ret; |
| } |
| |
| /* RTx mode: send Tx ID */ |
| static int p9221_send_txid(struct p9221_charger_data *chgr) |
| { |
| return -ENOTSUPP; |
| } |
| |
| static int p9xxx_send_txid(struct p9221_charger_data *chgr) |
| { |
| int ret; |
| |
| mutex_lock(&chgr->cmd_lock); |
| |
| /* write packet type to 0x100 */ |
| ret = chgr->reg_write_8(chgr, chgr->reg_packet_type_addr, |
| PROPRIETARY_PACKET_TYPE); |
| if (ret) |
| goto error; |
| |
| memset(chgr->tx_buf, 0, chgr->tx_buf_size); |
| |
| /* write 0x4F as header to 0x104 */ |
| chgr->tx_buf[0] = FAST_SERIAL_ID_HEADER; |
| chgr->tx_len = FAST_SERIAL_ID_SIZE; |
| |
| /* TODO: write txid to bit(23, 0) */ |
| memset(&chgr->tx_buf[1], 0x12, FAST_SERIAL_ID_SIZE - 1); |
| |
| /* write phone type to bit(23, 17) */ |
| chgr->tx_buf[3] = chgr->pdata->phone_type << 1; |
| |
| /* write accessory type to bit(31, 24) */ |
| chgr->tx_buf[4] = TX_ACCESSORY_TYPE; |
| |
| ret |= p9xxx_chip_set_pp_buf(chgr, chgr->tx_buf, chgr->tx_len + 1); |
| if (ret) { |
| dev_err(&chgr->client->dev, "Failed to load tx %d\n", ret); |
| goto error; |
| } |
| |
| /* send packet */ |
| ret = chgr->chip_set_cmd(chgr, P9221R5_COM_CCACTIVATE); |
| if (ret) |
| dev_err(&chgr->client->dev, "Failed to send txid %d\n", ret); |
| else |
| chgr->last_capacity = -1; |
| |
| error: |
| mutex_unlock(&chgr->cmd_lock); |
| return ret; |
| } |
| |
| /* RTx mode: send csp in Tx mode */ |
| static int p9221_send_csp_in_txmode(struct p9221_charger_data *chgr, u8 stat) |
| { |
| return -ENOTSUPP; |
| } |
| static int p9xxx_send_csp_in_txmode(struct p9221_charger_data *chgr, u8 stat) |
| { |
| int ret = 0; |
| u16 status_reg; |
| |
| ret = chgr->reg_read_16(chgr, P9221_STATUS_REG, &status_reg); |
| if ((ret != 0) || (status_reg & chgr->ints.rx_connected_bit) == 0) |
| return -EINVAL; |
| |
| dev_info(&chgr->client->dev, "Send Tx soc=%d\n", stat); |
| /* write packet type to 0x100 */ |
| ret = chgr->reg_write_8(chgr, |
| chgr->reg_packet_type_addr, |
| PROPRIETARY_PACKET_TYPE); |
| |
| if (ret != 0) |
| return ret; |
| |
| memset(chgr->tx_buf, 0, P9221R5_DATA_SEND_BUF_SIZE); |
| |
| chgr->tx_len = CHARGE_STATUS_PACKET_SIZE; |
| /* write 0x48 to 0x104 */ |
| chgr->tx_buf[0] = CHARGE_STATUS_PACKET_HEADER; |
| /* sype: power control 0x08 */ |
| chgr->tx_buf[1] = PP_TYPE_POWER_CONTROL; |
| /* subtype: state of charge report 0x10 */ |
| chgr->tx_buf[2] = PP_SUBTYPE_SOC; |
| /* soc: allow for 0.5% to fill up 0-200 -> 0-100% */ |
| chgr->tx_buf[3] = stat * 2; |
| /* crc-8 */ |
| chgr->tx_buf[4] = p9221_crc8(&chgr->tx_buf[1], chgr->tx_len - 1, |
| CRC8_INIT_VALUE); |
| |
| ret = p9xxx_chip_set_pp_buf(chgr, chgr->tx_buf, chgr->tx_len + 1); |
| |
| ret |= chgr->chip_set_cmd(chgr, P9221R5_COM_CCACTIVATE); |
| |
| memset(chgr->tx_buf, 0, P9221R5_DATA_SEND_BUF_SIZE); |
| chgr->tx_len = 0; |
| |
| return ret; |
| } |
| |
| /* renegotiate power from charger->pdata->epp_rp_value */ |
| static int p9221_chip_renegotiate_pwr(struct p9221_charger_data *chgr) |
| { |
| int ret; |
| int val8 = P9412_MW_TO_HW(chgr->pdata->epp_rp_value); |
| |
| /* units 0.5 W*/ |
| ret = chgr->reg_write_8(chgr, |
| P9221R5_EPP_REQ_NEGOTIATED_POWER_REG, val8); |
| if (ret < 0) |
| dev_err(&chgr->client->dev, |
| "cannot write to EPP_NEG_POWER=%d (%d)\n", |
| val8, ret); |
| return ret; |
| } |
| |
| static int p9222_chip_renegotiate_pwr(struct p9221_charger_data *chgr) |
| { |
| int ret = 0, guar_pwr_mw, cnt; |
| u8 val8, rp8; |
| |
| if (chgr->check_rp != RP_CHECKING) |
| return ret; |
| |
| ret = chgr->reg_read_8(chgr, P9222RE_EPP_TX_GUARANTEED_POWER_REG, &val8); |
| if (ret) |
| return ret; |
| guar_pwr_mw = P9412_HW_TO_MW(val8); |
| |
| /* write renegotiated power to 11W(>10W) or 10W(<=10W) */ |
| if (chgr->pdata->epp_rp_low_value != -1 && guar_pwr_mw <= P9222_NEG_POWER_10W) |
| rp8 = P9412_MW_TO_HW(chgr->pdata->epp_rp_low_value); |
| else |
| rp8 = P9412_MW_TO_HW(chgr->pdata->epp_rp_value); |
| |
| /* |
| * The neg_pwr write window is 200ms to 340ms, write every 20ms to make |
| * sure it works |
| */ |
| for (cnt = 0; cnt < 7 ; cnt++) { |
| /* units 0.5 W */ |
| ret = chgr->reg_write_8(chgr, P9222RE_EPP_REQ_NEGOTIATED_POWER_REG, rp8); |
| ret |= chgr->reg_write_8(chgr, P9222RE_EPP_REQ_MAXIMUM_POWER_REG, rp8); |
| |
| usleep_range(20 * USEC_PER_MSEC, 22 * USEC_PER_MSEC); |
| if (!chgr->online) |
| return -ENODEV; |
| } |
| if (ret == 0) |
| logbuffer_log(chgr->log, "read neg_pwr=0x%x, write neg_pwr=0x%x(guar_pwr=%dW)", |
| val8, rp8, guar_pwr_mw/1000); |
| |
| chgr->check_rp = RP_DONE; |
| |
| return ret; |
| } |
| |
| static int p9412_chip_renegotiate_pwr(struct p9221_charger_data *chgr) |
| { |
| int ret; |
| u8 val8; |
| int try; |
| int guar_pwr_mw; /* power provided by charger */ |
| int cur_pwr_mw; /* power currently supplied */ |
| int tgt_pwr_mw; /* power to be requested */ |
| int cfg_pwr_mw = chgr->pdata->epp_rp_value; |
| |
| ret = chgr->chip_get_sys_mode(chgr, &val8); |
| if (ret) |
| goto out; |
| |
| if (val8 != P9XXX_SYS_OP_MODE_WPC_EXTD) |
| return 0; |
| |
| logbuffer_log(chgr->log, "%s: WPC renegotiation", __func__); |
| |
| /* |
| * Compare the current power to the available power o |
| * determine if renegotiation is worthwhile. |
| */ |
| ret = chgr->reg_read_8(chgr, P9221R5_EPP_TX_GUARANTEED_POWER_REG, |
| &val8); |
| if (ret) |
| goto out; |
| guar_pwr_mw = P9412_HW_TO_MW(val8); |
| |
| ret = chgr->reg_read_8(chgr, P9221R5_EPP_CUR_NEGOTIATED_POWER_REG, |
| &val8); |
| if (ret) |
| goto out; |
| cur_pwr_mw = P9412_HW_TO_MW(val8); |
| |
| tgt_pwr_mw = min(cfg_pwr_mw, guar_pwr_mw); |
| logbuffer_log(chgr->log, "%s: tgt pwr = %d cur pwr = %d mW", |
| __func__, tgt_pwr_mw, cur_pwr_mw); |
| |
| if (cur_pwr_mw >= tgt_pwr_mw) { |
| ret = -EAGAIN; |
| logbuffer_log(chgr->log, "%s: no extra power available", |
| __func__); |
| goto out; |
| } |
| |
| /* Set the voltage to maximum value as defined in device-tree. */ |
| ret = chgr->chip_set_vout_max(chgr, chgr->pdata->epp_vout_mv); |
| |
| val8 = P9412_MW_TO_HW(tgt_pwr_mw); |
| ret = chgr->reg_write_8(chgr, |
| P9221R5_EPP_REQ_NEGOTIATED_POWER_REG, |
| val8); |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "cannot write to EPP_NEG_POWER=%d (%d)\n", val8, ret); |
| goto out; |
| } |
| |
| val8 = P9412_MW_TO_HW(tgt_pwr_mw); |
| ret = chgr->reg_write_8(chgr, |
| P9221R5_EPP_REQ_MAXIMUM_POWER_REG, |
| val8); |
| if (ret < 0) |
| dev_err(&chgr->client->dev, |
| "cannot write to EPP_MAX_POWER=%d (%d)\n", |
| chgr->pdata->epp_rp_value, ret); |
| |
| ret = chgr->chip_set_cmd(chgr, P9221_COM_RENEGOTIATE); |
| if (ret < 0) |
| dev_err(&chgr->client->dev, |
| "cannot write to sys_cmd =%d (%d)\n", |
| chgr->pdata->epp_rp_value, ret); |
| |
| /* Wait for renegotiation to complete. */ |
| ret = -ETIME; |
| for (try = 0; try < P9412_RN_MAX_POLL_ATTEMPTS; try++) { |
| msleep(P9412_RN_DELAY_MS); |
| ret = chgr->reg_read_8(chgr, |
| P9221R5_EPP_RENEGOTIATION_REG, |
| &val8); |
| if (val8 & P9412_RN_STATUS_DONE) { |
| ret = 0; |
| break; |
| } else if (val8 & P9412_RN_STATUS_ERROR) { |
| ret = -EIO; |
| break; |
| } |
| } |
| logbuffer_log(chgr->log, "%s: status = 0x%02x (tries = %d)", |
| __func__, val8, try); |
| out: |
| return ret; |
| } |
| static void p9222_check_neg_power(struct p9221_charger_data *chgr) |
| { |
| u32 vout_mv; |
| int ret; |
| |
| chgr->dc_icl_epp_neg = chgr->pdata->epp_icl > 0 ? |
| chgr->pdata->epp_icl : P9221_DC_ICL_EPP_UA; |
| |
| /* For EPP but Vout < 9V */ |
| ret = chgr->chip_get_vout_max(chgr, &vout_mv); |
| if (ret == 0 && vout_mv > 0 && vout_mv < 9000) { |
| chgr->dc_icl_epp_neg = P9221_DC_ICL_BPP_UA; |
| dev_info(&chgr->client->dev, "EPP less than 10W,use dc_icl=%dmA\n", |
| chgr->dc_icl_epp_neg); |
| } |
| |
| /* Based on mfg code to set EPP DC_ICL */ |
| if (chgr->mfg == P9221_PTMC_EPP_TX_2767 && chgr->pdata->tx_2767_icl > 0) { |
| chgr->dc_icl_epp_neg = chgr->pdata->tx_2767_icl; |
| dev_info(&chgr->client->dev, "EPP TX(%d), use dc_icl=%dmA\n", |
| chgr->mfg, chgr->dc_icl_epp_neg); |
| } |
| } |
| /* Read EPP_CUR_NEGOTIATED_POWER_REG to configure DC_ICL for EPP */ |
| static void p9xxx_check_neg_power(struct p9221_charger_data *chgr) |
| { |
| int ret; |
| u8 np8; |
| |
| chgr->dc_icl_epp_neg = P9221_DC_ICL_EPP_UA; |
| |
| if (chgr->chip_id < P9382A_CHIP_ID) |
| return; |
| |
| if (chgr->is_mfg_google) { |
| chgr->dc_icl_epp_neg = P9XXX_DC_ICL_EPP_1000; |
| dev_info(&chgr->client->dev, |
| "mfg code=%02x, use dc_icl=%dmA\n", |
| WLC_MFG_GOOGLE, P9XXX_DC_ICL_EPP_1000); |
| return; |
| } |
| ret = chgr->reg_read_8(chgr, P9221R5_EPP_CUR_NEGOTIATED_POWER_REG, &np8); |
| if (ret) |
| dev_err(&chgr->client->dev, |
| "Could not read Tx neg power: %d\n", ret); |
| else if (np8 < P9XXX_NEG_POWER_10W) { |
| /* |
| * base on firmware 17 |
| * Vout is 5V when Tx<10W, use BPP ICL |
| */ |
| chgr->dc_icl_epp_neg = chgr->low_neg_pwr_icl; |
| dev_info(&chgr->client->dev, |
| "EPP less than 10W,use dc_icl=%dmA,np=%02x\n", |
| chgr->low_neg_pwr_icl / 1000, np8); |
| } else if (np8 < P9XXX_NEG_POWER_11W) { |
| chgr->dc_icl_epp_neg = P9XXX_DC_ICL_EPP_1000; |
| dev_info(&chgr->client->dev, |
| "Use dc_icl=%dmA,np=%02x\n", |
| chgr->dc_icl_epp_neg/1000, np8); |
| } |
| } |
| |
| static int p9221_capdiv_en(struct p9221_charger_data *chgr, u8 mode) |
| { |
| return -ENOTSUPP; |
| } |
| |
| static int p9412_capdiv_en(struct p9221_charger_data *chgr, u8 mode) |
| { |
| const u8 mask = mode; |
| int ret, loops; |
| u8 cdmode; |
| |
| /* TODO: it probably needs a lock around this */ |
| |
| ret = chgr->reg_read_8(chgr, P9412_CDMODE_STS_REG, &cdmode); |
| if (ret < 0) |
| return ret; |
| |
| /* If the results are as expected, no changes needed */ |
| if ((cdmode & mask) == mask) |
| return ret; |
| |
| ret = chgr->reg_write_8(chgr, P9412_CDMODE_REQ_REG, mask); |
| if (ret < 0) |
| return -EIO; |
| |
| ret = chgr->chip_set_cmd(chgr, INIT_CAP_DIV_CMD); |
| if (ret < 0) |
| return -EIO; |
| |
| msleep(50); |
| |
| /* verify the change to Cap Divider mode */ |
| for (loops = 30 ; loops ; loops--) { |
| |
| ret = chgr->reg_read_8(chgr, P9412_CDMODE_STS_REG, &cdmode); |
| if (ret < 0) |
| return ret; |
| |
| if ((cdmode & mask) == mask) |
| break; |
| |
| msleep(100); |
| } |
| |
| return ((cdmode & mask) == mask) ? 0 : -ETIMEDOUT; |
| } |
| |
| static int ra9530_capdiv_en(struct p9221_charger_data *chgr, u8 mode) |
| { |
| /* Capdiv mode isn't supported, but we should return success */ |
| return 0; |
| } |
| |
| static int p9221_prop_mode_enable(struct p9221_charger_data *chgr, int req_pwr) |
| { |
| return -ENOTSUPP; |
| } |
| |
| #define MAX77759_CHGR_MODE_ALL_OFF 0 |
| /* b/202795383 remove load current before enable P9412 CD mode */ |
| static int p9412_prop_mode_capdiv_enable(struct p9221_charger_data *chgr) |
| { |
| int ret, i; |
| |
| |
| /* TODO: need to become a fake GPIO in the max77759 charger */ |
| if (!chgr->chg_mode_votable) |
| chgr->chg_mode_votable = |
| gvotable_election_get_handle(GBMS_MODE_VOTABLE); |
| if (chgr->chg_mode_votable) |
| gvotable_cast_long_vote(chgr->chg_mode_votable, |
| P9221_WLC_VOTER, |
| MAX77759_CHGR_MODE_ALL_OFF, true); |
| |
| /* total 2 seconds wait and early exit when WLC offline */ |
| for (i = 0; i < 20; i += 1) { |
| usleep_range(100 * USEC_PER_MSEC, 120 * USEC_PER_MSEC); |
| if (!chgr->online) { |
| ret = -ENODEV; |
| goto error_exit; |
| } |
| } |
| |
| ret = p9412_capdiv_en(chgr, CDMODE_CAP_DIV_MODE); |
| |
| /* total 1 seconds wait and early exit when WLC offline */ |
| for (i = 0; i < 10; i += 1) { |
| usleep_range(100 * USEC_PER_MSEC, 120 * USEC_PER_MSEC); |
| if (!chgr->online) { |
| ret = -ENODEV; |
| goto error_exit; |
| } |
| } |
| |
| error_exit: |
| if (chgr->chg_mode_votable) |
| gvotable_cast_long_vote(chgr->chg_mode_votable, |
| P9221_WLC_VOTER, |
| MAX77759_CHGR_MODE_ALL_OFF, false); |
| |
| return ret; |
| } |
| |
| static int p9412_prop_mode_enable(struct p9221_charger_data *chgr, int req_pwr) |
| { |
| const int req_pwr_val = req_pwr * 2 / 1000; |
| int ret = 0, loops, i, txpwr_mw; |
| u8 val8, cdmode, txpwr, pwr_stp, mode_sts, err_sts, prop_cur_pwr, prop_req_pwr; |
| u32 val = 0; |
| |
| if (p9xxx_is_capdiv_en(chgr)) |
| goto err_exit; |
| |
| ret = chgr->chip_get_sys_mode(chgr, &val8); |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "PROP_MODE: cannot get sys mode\n"); |
| return 0; |
| } |
| |
| if (val8 == P9XXX_SYS_OP_MODE_PROPRIETARY) { |
| ret = chgr->reg_read_8(chgr, P9412_PROP_TX_POTEN_PWR_REG, &txpwr); |
| txpwr_mw = txpwr * 1000 / 2; |
| if (ret != 0 || txpwr_mw < chgr->pdata->hpp_neg_pwr) { |
| dev_info(&chgr->client->dev, |
| "PROP_MODE: power=%dmW not supported\n", txpwr_mw); |
| chgr->prop_mode_en = false; |
| goto err_exit; |
| } |
| |
| goto enable_capdiv; |
| } |
| |
| /* |
| * Step 0: clear data type buffer: |
| * write 0 to 0x800 and 0x801 |
| */ |
| ret = chgr->reg_write_n(chgr, |
| P9412_COM_PACKET_TYPE_ADDR, |
| &val, 4); |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "Failed to clear data type buffer: %d\n", ret); |
| goto err_exit; |
| } |
| |
| /* |
| * Step 1: clear all interrupts: |
| * write 0xFFFF to 0x3A then write 0x20 to 0x4E |
| */ |
| mutex_lock(&chgr->cmd_lock); |
| |
| ret = chgr->reg_write_16(chgr, |
| P9221R5_INT_CLEAR_REG, P9XXX_INT_CLEAR_MASK); |
| if (ret == 0) |
| ret = chgr->chip_set_cmd(chgr, P9221_COM_CLEAR_INT_MASK); |
| if (ret) { |
| dev_err(&chgr->client->dev, "Failed to reset INT: %d\n", ret); |
| mutex_unlock(&chgr->cmd_lock); |
| goto err_exit; |
| } |
| |
| mutex_unlock(&chgr->cmd_lock); |
| |
| msleep(50); |
| |
| enable_capdiv: |
| if (chgr->pdata->has_sw_ramp) { |
| dev_dbg(&chgr->client->dev, "%s: voter=%s, icl=%d\n", |
| __func__, P9221_HPP_VOTER, P9XXX_CDMODE_ENABLE_ICL_UA); |
| ret = p9xxx_sw_ramp_icl(chgr, P9XXX_CDMODE_ENABLE_ICL_UA); |
| if (ret < 0) |
| return ret; |
| |
| ret = gvotable_cast_int_vote(chgr->dc_icl_votable, P9221_HPP_VOTER, |
| P9XXX_CDMODE_ENABLE_ICL_UA, true); |
| if (ret == 0) |
| ret = gvotable_cast_int_vote(chgr->dc_icl_votable, |
| P9221_RAMP_VOTER, 0, false); |
| if (ret < 0) |
| dev_err(&chgr->client->dev, "%s: cannot setup sw ramp (%d)\n", |
| __func__, ret); |
| |
| } |
| |
| /* |
| * Step 2: enable Cap Divider configuration: |
| * write 0x02 to 0x101 then write 0x40 to 0x4E |
| */ |
| ret = p9412_prop_mode_capdiv_enable(chgr); |
| if (ret) { |
| dev_err(&chgr->client->dev, "PROP_MODE: fail to enable Cap Div mode\n"); |
| goto err_exit; |
| } |
| |
| for (i = 0; i < 30; i += 1) { |
| usleep_range(100 * USEC_PER_MSEC, 120 * USEC_PER_MSEC); |
| if (!chgr->online) |
| goto err_exit; |
| } |
| |
| /* |
| * Step 3: Enable Proprietary Mode: write 0x01 to 0x4F (0x4E bit8) |
| */ |
| if (val8 == P9XXX_SYS_OP_MODE_PROPRIETARY) |
| goto request_pwr; |
| |
| ret = chgr->chip_set_cmd(chgr, PROP_MODE_EN_CMD); |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "PROP_MODE: fail to send PROP_MODE_EN_CMD\n"); |
| goto err_exit; |
| } |
| |
| msleep(50); |
| |
| /* |
| * Step 4: wait for PropModeStat interrupt, register 0x37[4] |
| */ |
| /* 60 * 50 = 3 secs */ |
| for (loops = 60 ; loops ; loops--) { |
| if (!chgr->online) { |
| chgr->prop_mode_en = false; |
| ret = -ENODEV; |
| break; |
| } |
| if (chgr->prop_mode_en) { |
| dev_info(&chgr->client->dev, |
| "PROP_MODE: Proprietary Mode Enabled\n"); |
| break; |
| } |
| msleep(50); |
| } |
| if (!chgr->prop_mode_en) |
| goto err_exit; |
| |
| request_pwr: |
| /* |
| * Step 5: Read TX potential power register (0xC4) |
| * [TX max power capability] in 0.5W units |
| */ |
| ret = chgr->reg_read_8(chgr, P9412_PROP_TX_POTEN_PWR_REG, &txpwr); |
| txpwr_mw = txpwr * 1000 / 2; |
| if ((ret == 0) && (txpwr_mw >= chgr->pdata->hpp_neg_pwr)) { |
| dev_info(&chgr->client->dev, |
| "PROP_MODE: Tx potential power=%dmW\n", txpwr_mw); |
| } else { |
| chgr->prop_mode_en = false; |
| goto err_exit; |
| } |
| |
| /* |
| * Step 6: Request xx W Neg power by writing 0xC5, |
| * then write 0x02 to 0x4F |
| */ |
| ret = chgr->reg_write_8(chgr, P9412_PROP_REQ_PWR_REG, req_pwr_val); |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "PROP_MODE: fail to write pwr req register\n"); |
| chgr->prop_mode_en = false; |
| goto err_exit; |
| } |
| dev_info(&chgr->client->dev, "request power=%dmW\n", req_pwr_val * 1000 / 2); |
| |
| /* Request power from TX based on PropReqPwr (0xC5) */ |
| ret = chgr->chip_set_cmd(chgr, PROP_REQ_PWR_CMD); |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "PROP_MODE: fail to send PROP_REQ_PWR_CMD\n"); |
| chgr->prop_mode_en = false; |
| goto err_exit; |
| } |
| |
| for (i = 0; i < 30; i += 1) { |
| usleep_range(100 * USEC_PER_MSEC, 120 * USEC_PER_MSEC); |
| if (!chgr->online) { |
| chgr->prop_mode_en = false; |
| ret = -ENODEV; |
| dev_err(&chgr->client->dev, |
| "PROP_MODE: charger went offline after requesting prop mode\n"); |
| } |
| } |
| |
| err_exit: |
| if (ret == -ENODEV) |
| return ret; |
| |
| /* check status */ |
| ret = chgr->chip_get_sys_mode(chgr, &val8); |
| ret |= chgr->reg_read_8(chgr, P9412_PROP_CURR_PWR_REG, &prop_cur_pwr); |
| ret |= chgr->reg_read_8(chgr, P9412_PROP_MODE_PWR_STEP_REG, &pwr_stp); |
| ret |= chgr->reg_read_8(chgr, P9412_PROP_MODE_STATUS_REG, &mode_sts); |
| ret |= chgr->reg_read_8(chgr, P9412_PROP_MODE_ERR_STS_REG, &err_sts); |
| ret |= chgr->reg_read_8(chgr, P9412_CDMODE_STS_REG, &cdmode); |
| ret |= chgr->reg_read_8(chgr, P9412_PROP_REQ_PWR_REG, &prop_req_pwr); |
| |
| pr_debug("%s PROP_MODE: en=%d,sys_mode=%02x,mode_sts=%02x,err_sts=%02x," |
| "cdmode=%02x,pwr_stp=%02x,req_pwr=%02x,prop_cur_pwr=%02x,txpwr=%dmW", |
| __func__, chgr->prop_mode_en, val8, mode_sts, err_sts, |
| cdmode, pwr_stp, prop_req_pwr, prop_cur_pwr, txpwr_mw); |
| |
| if (!ret) { |
| dev_info(&chgr->client->dev, |
| "PROP_MODE: en=%d,sys_mode=%02x,mode_sts=%02x," |
| "err_sts=%02x,cdmode=%02x,pwr_stp=%02x," |
| "req_pwr=%02x,prop_cur_pwr=%02x", |
| chgr->prop_mode_en, val8, mode_sts, err_sts, |
| cdmode, pwr_stp, prop_req_pwr, prop_cur_pwr); |
| } |
| |
| |
| if (!chgr->prop_mode_en) { |
| int rc; |
| |
| rc = gvotable_cast_int_vote(chgr->dc_icl_votable, P9221_HPP_VOTER, |
| P9XXX_CDMODE_ENABLE_ICL_UA, false); |
| if (rc <0) |
| dev_err(&chgr->client->dev, "%s: cannot remove HPP voter (%d)\n", |
| __func__, ret); |
| } |
| |
| return chgr->prop_mode_en; |
| } |
| |
| static int ra9530_prop_mode_enable(struct p9221_charger_data *chgr, int req_pwr) |
| { |
| const int req_pwr_val = req_pwr * 2 / 1000; |
| int ret, loops, i, max_wait_time, loop_cnt, txpwr_mw; |
| u8 val8, cdmode, txpwr, pwr_stp, mode_sts, err_sts, prop_cur_pwr, prop_req_pwr; |
| |
| ret = chgr->chip_get_sys_mode(chgr, &val8); |
| if (ret) { |
| dev_err(&chgr->client->dev, "PROP_MODE: cannot get sys mode\n"); |
| return 0; |
| } |
| |
| /* wait for PropModeStat interrupt/wait Vout raise to request power */ |
| max_wait_time = chgr->de_wait_prop_irq_ms > 0 ? |
| chgr->de_wait_prop_irq_ms : chgr->pdata->wait_prop_irq_ms; |
| |
| /* |
| * clear all interrupts: |
| * write 0xFFFF to 0x3A then write 0x20 to 0x4E |
| */ |
| mutex_lock(&chgr->cmd_lock); |
| |
| ret = chgr->reg_write_16(chgr, P9221R5_INT_CLEAR_REG, P9XXX_INT_CLEAR_MASK); |
| if (ret == 0) |
| ret = chgr->chip_set_cmd(chgr, P9221_COM_CLEAR_INT_MASK); |
| if (ret) { |
| dev_err(&chgr->client->dev, "Failed to reset INT: %d\n", ret); |
| mutex_unlock(&chgr->cmd_lock); |
| goto err_exit; |
| } |
| |
| mutex_unlock(&chgr->cmd_lock); |
| |
| msleep(50); |
| |
| if (chgr->pdata->has_sw_ramp) { |
| dev_dbg(&chgr->client->dev, "%s: voter=%s, icl=%d\n", |
| __func__, P9221_HPP_VOTER, P9XXX_PROPMODE_ENABLE_ICL_UA); |
| ret = p9xxx_sw_ramp_icl(chgr, P9XXX_PROPMODE_ENABLE_ICL_UA); |
| if (ret < 0) |
| return ret; |
| |
| ret = gvotable_cast_int_vote(chgr->dc_icl_votable, P9221_HPP_VOTER, |
| P9XXX_PROPMODE_ENABLE_ICL_UA, true); |
| if (ret == 0) |
| ret = gvotable_cast_int_vote(chgr->dc_icl_votable, |
| P9221_RAMP_VOTER, 0, false); |
| if (ret < 0) |
| dev_err(&chgr->client->dev, "%s: cannot setup sw ramp (%d)\n", |
| __func__, ret); |
| } |
| |
| if (!chgr->chg_mode_votable) |
| chgr->chg_mode_votable = |
| gvotable_election_get_handle(GBMS_MODE_VOTABLE); |
| |
| if (chgr->chg_mode_votable) { |
| gvotable_cast_long_vote(chgr->chg_mode_votable, |
| P9221_WLC_VOTER, |
| MAX77759_CHGR_MODE_ALL_OFF, true); |
| chgr->chg_mode_off = true; |
| } |
| |
| ret = p9xxx_chip_set_cmfet_reg(chgr, chgr->wlc_dc_comcap); |
| if (ret) |
| dev_warn(&chgr->client->dev, "%s: cannot set COM Cap (%d)\n", __func__, ret); |
| |
| /* |
| * Enable Proprietary Mode: write 0x01 to 0x4F (0x4E bit8) |
| */ |
| if (val8 == P9XXX_SYS_OP_MODE_PROPRIETARY) { |
| chgr->prop_mode_en = true; |
| goto request_pwr; |
| } |
| |
| ret = chgr->chip_set_cmd(chgr, PROP_MODE_EN_CMD); |
| if (ret) { |
| dev_err(&chgr->client->dev, "PROP_MODE: fail to send PROP_MODE_EN_CMD\n"); |
| goto err_exit; |
| } |
| |
| msleep(50); |
| |
| /* |
| * wait for PropModeStat interrupt, register 0x37[4] for 60 * 50 = 3 secs |
| */ |
| loop_cnt = max_wait_time / 50; |
| for (loops = loop_cnt ; loops ; loops--) { |
| if (!chgr->online) { |
| chgr->prop_mode_en = false; |
| ret = -ENODEV; |
| break; |
| } |
| if (chgr->prop_mode_en) { |
| dev_info(&chgr->client->dev, "PROP_MODE: Proprietary Mode Enabled\n"); |
| break; |
| } |
| msleep(50); |
| } |
| if (!chgr->prop_mode_en) |
| goto err_exit; |
| |
| request_pwr: |
| |
| /* total 2 seconds wait and early exit when WLC offline */ |
| for (i = 0; i < 20; i += 1) { |
| usleep_range(100 * USEC_PER_MSEC, 120 * USEC_PER_MSEC); |
| if (!chgr->online) { |
| chgr->prop_mode_en = false; |
| ret = -ENODEV; |
| goto err_exit; |
| } |
| } |
| |
| /* |
| * Read TX potential power register (0xC4) |
| * [TX max power capability] in 0.5W units |
| */ |
| ret = chgr->reg_read_8(chgr, P9412_PROP_TX_POTEN_PWR_REG, &txpwr); |
| txpwr_mw = txpwr * 1000 / 2; |
| if ((ret != 0) || (txpwr_mw < chgr->pdata->hpp_neg_pwr)) { |
| chgr->prop_mode_en = false; |
| goto err_exit; |
| } |
| dev_info(&chgr->client->dev, "PROP_MODE: Tx potential power=%dmW\n", txpwr_mw); |
| |
| /* |
| * Request xx W Neg power by writing 0xC5 |
| */ |
| ret = chgr->reg_write_8(chgr, P9412_PROP_REQ_PWR_REG, req_pwr_val); |
| if (ret) { |
| dev_err(&chgr->client->dev, "PROP_MODE: fail to write pwr req register\n"); |
| chgr->prop_mode_en = false; |
| goto err_exit; |
| } |
| dev_info(&chgr->client->dev, "request power=%dmW\n", req_pwr_val * 1000 / 2); |
| |
| /* Set power step to 3W */ |
| ret = chgr->reg_write_8(chgr, P9412_PROP_MODE_PWR_STEP_REG, RA9530_PROP_MODE_PWR_STEP); |
| if (ret) { |
| dev_err(&chgr->client->dev, "PROP_MODE: fail to write pwr step register\n"); |
| chgr->prop_mode_en = false; |
| goto err_exit; |
| } |
| |
| chgr->negotiation_complete = false; |
| /* Request power from TX based on PropReqPwr (0xC5) by writing 0x02 to 0x4F */ |
| ret = chgr->chip_set_cmd(chgr, PROP_REQ_PWR_CMD); |
| if (ret) { |
| dev_err(&chgr->client->dev, |
| "PROP_MODE: fail to send PROP_REQ_PWR_CMD\n"); |
| chgr->prop_mode_en = false; |
| goto err_exit; |
| } |
| |
| loop_cnt = max_wait_time / 50; |
| for (loops = loop_cnt ; loops ; loops--) { |
| usleep_range(50 * USEC_PER_MSEC, 60 * USEC_PER_MSEC); |
| ret = chgr->reg_read_8(chgr, P9412_PROP_CURR_PWR_REG, &prop_cur_pwr); |
| if (chgr->negotiation_complete && ret == 0 && prop_cur_pwr == req_pwr_val) |
| break; |
| if (!chgr->online) |
| chgr->prop_mode_en = false; |
| } |
| if (ret == 0 && prop_cur_pwr < req_pwr_val) { |
| dev_err(&chgr->client->dev, "%s: curr_pwr=%d\n", __func__, prop_cur_pwr); |
| chgr->negotiation_complete = false; |
| } |
| |
| |
| err_exit: |
| if (ret == -ENODEV) |
| return ret; |
| |
| /* check status */ |
| ret = chgr->chip_get_sys_mode(chgr, &val8); |
| ret |= chgr->reg_read_8(chgr, P9412_PROP_CURR_PWR_REG, &prop_cur_pwr); |
| ret |= chgr->reg_read_8(chgr, P9412_PROP_MODE_PWR_STEP_REG, &pwr_stp); |
| ret |= chgr->reg_read_8(chgr, P9412_PROP_MODE_STATUS_REG, &mode_sts); |
| ret |= chgr->reg_read_8(chgr, P9412_PROP_MODE_ERR_STS_REG, &err_sts); |
| ret |= chgr->reg_read_8(chgr, P9412_CDMODE_STS_REG, &cdmode); |
| ret |= chgr->reg_read_8(chgr, P9412_PROP_REQ_PWR_REG, &prop_req_pwr); |
| |
| dev_dbg(&chgr->client->dev, "%s PROP_MODE: en=%d,sys_mode=%02x,mode_sts=%02x,err_sts=%02x," |
| "cdmode=%02x,pwr_stp=%02x,req_pwr=%02x,prop_cur_pwr=%02x,txpwr=%dmW", |
| __func__, chgr->prop_mode_en, val8, mode_sts, err_sts, |
| cdmode, pwr_stp, prop_req_pwr, prop_cur_pwr, txpwr_mw); |
| |
| if (!ret) { |
| dev_info(&chgr->client->dev, |
| "PROP_MODE: en=%d,sys_mode=%02x,mode_sts=%02x," |
| "err_sts=%02x,cdmode=%02x,pwr_stp=%02x," |
| "req_pwr=%02x,prop_cur_pwr=%02x", |
| chgr->prop_mode_en, val8, mode_sts, err_sts, |
| cdmode, pwr_stp, prop_req_pwr, prop_cur_pwr); |
| } |
| |
| if (!chgr->prop_mode_en || !chgr->negotiation_complete) { |
| int rc; |
| |
| dev_dbg(&chgr->client->dev, "%s prop_mode_en=%d,negotiation_complete=%d\n", |
| __func__, chgr->prop_mode_en, chgr->negotiation_complete); |
| |
| rc = gvotable_cast_int_vote(chgr->dc_icl_votable, P9221_HPP_VOTER, |
| P9XXX_PROPMODE_ENABLE_ICL_UA, false); |
| if (rc <0) |
| dev_err(&chgr->client->dev, "%s: cannot remove HPP voter (%d)\n", |
| __func__, ret); |
| |
| if (chgr->chg_mode_votable) { |
| gvotable_cast_long_vote(chgr->chg_mode_votable, |
| P9221_WLC_VOTER, |
| MAX77759_CHGR_MODE_ALL_OFF, false); |
| chgr->chg_mode_off = false; |
| } |
| } |
| |
| return chgr->prop_mode_en && chgr->negotiation_complete; |
| } |
| |
| static bool p9xxx_is_vout_on(struct p9221_charger_data *chgr) |
| { |
| int ret; |
| u8 val = 0; |
| |
| ret = chgr->reg_read_8(chgr, P9221_STATUS_REG, &val); |
| if (ret < 0) { |
| dev_err(&chgr->client->dev, "can't get vout state, ret=%d\n", ret); |
| return false; |
| } |
| |
| return val & P9221_STAT_VOUT; |
| } |
| |
| static bool p9xxx_chip_is_calibrated(struct p9221_charger_data *chgr) |
| { |
| return true; |
| } |
| |
| static bool p9412_chip_is_calibrated(struct p9221_charger_data *chgr) |
| { |
| u8 val; |
| int ret; |
| |
| ret = chgr->reg_read_8(chgr, P9412_EPP_CAL_STATE_REG, &val); |
| |
| if (ret) |
| return false; |
| |
| dev_dbg(&chgr->client->dev, "EPP_CAL_STATE_REG=%02x\n", val); |
| |
| return (val & P9412_EPP_CAL_STATE_MASK) == 0; |
| } |
| |
| static bool ra9530_chip_is_calibrated(struct p9221_charger_data *chgr) |
| { |
| u8 val; |
| int ret; |
| |
| if (!p9221_is_epp(chgr)) |
| return chgr->chip_is_vout_on(chgr); |
| |
| ret = chgr->reg_read_8(chgr, P9412_EPP_CAL_STATE_REG, &val); |
| |
| if (ret) |
| return false; |
| |
| dev_dbg(&chgr->client->dev, "EPP_CAL_STATE_REG=%02x\n", val); |
| |
| if (chgr->fw_rev < RA9530_FW_REV_22) |
| return chgr->chip_is_vout_on(chgr) && (val & RA9530_V21_EPP_CAL_STATE_MASK) == 0; |
| |
| return chgr->chip_is_vout_on(chgr) && |
| (val & RA9530_EPP_CAL_STATE_MASK) == RA9530_EPP_CALIBRATED_STATE; |
| } |
| |
| static void p9xxx_chip_set_cmfet(struct p9221_charger_data *chgr) |
| { |
| |
| } |
| |
| static void ra9530_chip_set_cmfet(struct p9221_charger_data *chgr) |
| { |
| u32 vout_mv, val = chgr->pdata->gpp_cmfet; |
| int ret; |
| |
| if (chgr->pdata->gpp_cmfet <= 0) |
| return; |
| |
| ret = chgr->chip_get_vout_max(chgr, &vout_mv); |
| if (ret == 0 && vout_mv < 8000) |
| val = 0; |
| |
| if (!p9221_is_epp(chgr) || chgr->mfg != WLC_MFG_GOOGLE) |
| val = 0; |
| |
| pr_info("vout_mv=%d, is_epp=%d, mfg=%d cmfet=%d\n", |
| vout_mv, p9221_is_epp(chgr), chgr->mfg, val); |
| |
| ret = p9xxx_chip_set_cmfet_reg(chgr, val); |
| if (ret < 0) |
| pr_err("fail to write cmfet_reg, ret=%d\n", ret); |
| } |
| |
| static int p9xxx_chip_set_bpp_icl(struct p9221_charger_data *chgr) |
| { |
| const int default_icl = chgr->pdata->bpp_icl > 0 ? |
| chgr->pdata->bpp_icl : P9221_DC_ICL_BPP_UA; |
| |
| return default_icl; |
| } |
| |
| static int p9222_chip_set_bpp_icl(struct p9221_charger_data *chgr) |
| { |
| const int default_icl = chgr->pdata->bpp_icl > 0 ? |
| chgr->pdata->bpp_icl : P9221_DC_ICL_BPP_UA; |
| |
| if (chgr->pdata->freq_109_icl > 0 && is_ping_freq_fixed_at(chgr, 109)) |
| return chgr->pdata->freq_109_icl; |
| |
| return default_icl; |
| } |
| |
| static int ra9530_chip_set_bpp_icl(struct p9221_charger_data *chgr) |
| { |
| u32 vout_mv; |
| int ret, icl = P9221_DC_ICL_BPP_UA, ramp_icl; |
| |
| if (p9221_is_epp(chgr)) |
| return icl; |
| |
| if (chgr->pdata->bpp_icl == 0 && chgr->pdata->bpp_lv_icl == 0) |
| return icl; |
| |
| ret = chgr->chip_get_vout_max(chgr, &vout_mv); |
| if (ret < 0) |
| return icl; |
| |
| mutex_lock(&chgr->icl_lock); |
| chgr->icl_ramp_ua = P9221_DC_ICL_BPP_RAMP_DEFAULT_UA; |
| |
| if (vout_mv > 6000) { |
| icl = chgr->pdata->bpp_icl ? chgr->pdata->bpp_icl : icl; |
| ramp_icl = chgr->pdata->bpp_icl_ramp_ua; |
| } else { |
| icl = chgr->pdata->bpp_lv_icl ? chgr->pdata->bpp_lv_icl : icl; |
| ramp_icl = chgr->pdata->bpp_lv_icl_ramp_ua; |
| } |
| |
| chgr->pdata->icl_ramp_delay_ms = ramp_icl ? P9221_DC_ICL_BPP_RAMP_DELAY_DEFAULT_MS : -1; |
| chgr->icl_ramp_ua = ramp_icl ? ramp_icl : chgr->icl_ramp_ua; |
| |
| logbuffer_log(chgr->log, "bpp_icl=%d, ramp_icl=%d, ramp_en=%d\n", |
| icl, chgr->icl_ramp_ua, chgr->pdata->icl_ramp_delay_ms > 0); |
| |
| mutex_unlock(&chgr->icl_lock); |
| |
| return icl; |
| } |
| |
| static void p9xxx_magsafe_optimized(struct p9221_charger_data *chgr) |
| { |
| |
| } |
| |
| static void p9222_magsafe_optimized(struct p9221_charger_data *chgr) |
| { |
| int ret; |
| |
| if (!chgr->pdata->magsafe_optimized) |
| return; |
| |
| /* check BPP mode && ping frequency == 128k */ |
| if (p9221_is_epp(chgr) || !is_ping_freq_fixed_at(chgr, 128)) |
| return; |
| |
| /* set MOT(minimum on-time) to 60% */ |
| ret = p9xxx_chip_set_mot_reg(chgr, P9222_MOT_60PCT); |
| if (ret < 0) |
| dev_err(&chgr->client->dev, "Fail to set MOT, ret=%d\n", ret); |
| |
| /* set Vout to 6.4V */ |
| ret = chgr->chip_set_vout_max(chgr, 6400); |
| if (ret < 0) |
| dev_err(&chgr->client->dev, "Fail to set Vout, ret=%d\n", ret); |
| } |
| |
| static void p9xxx_chip_set_ovp(struct p9221_charger_data *chgr, u8 mode) |
| { |
| |
| } |
| |
| static void ra9530_chip_set_ovp(struct p9221_charger_data *chgr, u8 mode) |
| { |
| int ret = 0; |
| |
| switch (mode) { |
| case OVSET_BPP: |
| chgr->wlc_ovp = RA9530_OVSET_13V; |
| break; |
| case OVSET_EPP: |
| chgr->wlc_ovp = RA9530_OVSET_16_7V; |
| break; |
| case OVSET_HPP: |
| chgr->wlc_ovp = RA9530_OVSET_24_7V; |
| break; |
| default: |
| break; |
| } |
| |
| if (chgr->wlc_ovp) |
| ret = chgr->reg_write_8(chgr, RA9530_OVSET_REG, chgr->wlc_ovp); |
| if (ret) |
| dev_warn(&chgr->client->dev, "fail to set ovp\n"); |
| } |
| |
| void p9221_chip_init_interrupt_bits(struct p9221_charger_data *chgr, u16 chip_id) |
| { |
| chgr->ints.mode_changed_bit = P9221R5_STAT_MODECHANGED; |
| chgr->ints.vrecton_bit = P9221R5_STAT_VRECTON; |
| chgr->ints.vout_changed_bit = P9221R5_STAT_VOUTCHANGED; |
| |
| switch (chip_id) { |
| case P9412_CHIP_ID: |
| chgr->ints.over_curr_bit = P9412_STAT_OVC; |
| chgr->ints.over_volt_bit = 0; /* TODO: b/181191668*/ |
| chgr->ints.over_temp_bit = P9412_STAT_OVT; |
| chgr->ints.over_uv_bit = 0; |
| chgr->ints.cc_send_busy_bit = P9412_STAT_CCSENDBUSY; |
| chgr->ints.cc_data_rcvd_bit = P9412_STAT_CCDATARCVD; |
| chgr->ints.pp_rcvd_bit = P9412_STAT_PPRCVD; |
| chgr->ints.cc_error_bit = P9412_STAT_CCERROR; |
| chgr->ints.cc_reset_bit = 0; |
| chgr->ints.propmode_stat_bit = P9412_PROP_MODE_STAT_INT; |
| chgr->ints.cdmode_change_bit = P9412_CDMODE_CHANGE_INT; |
| chgr->ints.cdmode_err_bit = P9412_CDMODE_ERROR_INT; |
| chgr->ints.extended_mode_bit = 0; |
| |
| chgr->ints.hard_ocp_bit = P9412_STAT_OVC; |
| chgr->ints.tx_conflict_bit = P9412_STAT_TXCONFLICT; |
| chgr->ints.csp_bit = P9412_STAT_CSP; |
| chgr->ints.rx_connected_bit = P9412_STAT_RXCONNECTED; |
| chgr->ints.tx_fod_bit = 0; |
| chgr->ints.tx_underpower_bit = 0; |
| chgr->ints.tx_uvlo_bit = 0; |
| chgr->ints.pppsent_bit = P9412_STAT_PPPSENT; |
| chgr->ints.ocp_ping_bit = P9412_STAT_OCP_PING; |
| break; |
| case RA9530_CHIP_ID: |
| // Mostly Same as 9412 |
| chgr->ints.over_curr_bit = P9412_STAT_OVC; |
| chgr->ints.over_volt_bit = P9412_STAT_OVV; // TODO: check if 9412 issue still occur in HPP |
| chgr->ints.over_temp_bit = P9412_STAT_OVT; |
| chgr->ints.over_uv_bit = 0; |
| chgr->ints.cc_send_busy_bit = P9412_STAT_CCSENDBUSY; |
| chgr->ints.cc_data_rcvd_bit = P9412_STAT_CCDATARCVD; |
| chgr->ints.pp_rcvd_bit = P9412_STAT_PPRCVD; |
| chgr->ints.cc_error_bit = P9412_STAT_CCERROR; |
| chgr->ints.cc_reset_bit = 0; |
| chgr->ints.propmode_stat_bit = P9412_PROP_MODE_STAT_INT; |
| chgr->ints.cdmode_change_bit = 0; // No capdiv mode |
| chgr->ints.cdmode_err_bit = 0; // No capdiv mode |
| chgr->ints.extended_mode_bit = 0; |
| |
| chgr->ints.hard_ocp_bit = P9412_STAT_OVC; |
| chgr->ints.tx_conflict_bit = P9412_STAT_TXCONFLICT; |
| chgr->ints.csp_bit = P9412_STAT_CSP; |
| chgr->ints.rx_connected_bit = P9412_STAT_RXCONNECTED; |
| chgr->ints.tx_fod_bit = 0; |
| chgr->ints.tx_underpower_bit = 0; |
| chgr->ints.tx_uvlo_bit = 0; |
| chgr->ints.pppsent_bit = P9412_STAT_PPPSENT; |
| chgr->ints.ocp_ping_bit = P9412_STAT_OCP_PING; |
| break; |
| case P9382A_CHIP_ID: |
| chgr->ints.over_curr_bit = P9221R5_STAT_OVC; |
| chgr->ints.over_volt_bit = P9221R5_STAT_OVV; |
| chgr->ints.over_temp_bit = P9221R5_STAT_OVT; |
| chgr->ints.over_uv_bit = P9221R5_STAT_UV; |
| chgr->ints.cc_send_busy_bit = P9221R5_STAT_CCSENDBUSY; |
| chgr->ints.cc_data_rcvd_bit = P9221R5_STAT_CCDATARCVD; |
| chgr->ints.pp_rcvd_bit = P9221R5_STAT_PPRCVD; |
| chgr->ints.cc_error_bit = P9221R5_STAT_CCERROR; |
| chgr->ints.cc_reset_bit = P9221R5_STAT_CCRESET; |
| chgr->ints.propmode_stat_bit = 0; |
| chgr->ints.cdmode_change_bit = 0; |
| chgr->ints.cdmode_err_bit = 0; |
| chgr->ints.extended_mode_bit = 0; |
| |
| chgr->ints.hard_ocp_bit = P9382_STAT_HARD_OCP; |
| chgr->ints.tx_conflict_bit = P9382_STAT_TXCONFLICT; |
| chgr->ints.csp_bit = P9382_STAT_CSP; |
| chgr->ints.rx_connected_bit = P9382_STAT_RXCONNECTED; |
| chgr->ints.tx_fod_bit = P9382_STAT_TXFOD; |
| chgr->ints.tx_underpower_bit = P9382_STAT_TXUNDERPOWER; |
| chgr->ints.tx_uvlo_bit = P9382_STAT_TXUVLO; |
| chgr->ints.pppsent_bit = P9221R5_STAT_CCSENDBUSY; |
| chgr->ints.ocp_ping_bit = 0; |
| break; |
| case P9222_CHIP_ID: |
| chgr->ints.over_curr_bit = P9222_STAT_OVC; |
| chgr->ints.over_volt_bit = P9222_STAT_OVV; |
| chgr->ints.over_temp_bit = P9222_STAT_OVT; |
| chgr->ints.over_uv_bit = 0; |
| chgr->ints.cc_send_busy_bit = P9221R5_STAT_CCSENDBUSY; |
| chgr->ints.cc_data_rcvd_bit = P9221R5_STAT_CCDATARCVD; |
| chgr->ints.pp_rcvd_bit = P9222_STAT_PPRCVD; |
| chgr->ints.cc_error_bit = P9222_STAT_CCERROR; |
| chgr->ints.cc_reset_bit = 0; |
| chgr->ints.propmode_stat_bit = 0; |
| chgr->ints.cdmode_change_bit = 0; |
| chgr->ints.cdmode_err_bit = 0; |
| chgr->ints.extended_mode_bit = P9222_EXTENDED_MODE; |
| |
| chgr->ints.hard_ocp_bit = 0; |
| chgr->ints.tx_conflict_bit = 0; |
| chgr->ints.csp_bit = 0; |
| chgr->ints.rx_connected_bit = 0; |
| chgr->ints.tx_fod_bit = 0; |
| chgr->ints.tx_underpower_bit = 0; |
| chgr->ints.tx_uvlo_bit = 0; |
| chgr->ints.pppsent_bit = 0; |
| chgr->ints.ocp_ping_bit = 0; |
| break; |
| default: |
| chgr->ints.over_curr_bit = P9221R5_STAT_OVC; |
| chgr->ints.over_volt_bit = P9221R5_STAT_OVV; |
| chgr->ints.over_temp_bit = P9221R5_STAT_OVT; |
| chgr->ints.over_uv_bit = P9221R5_STAT_UV; |
| chgr->ints.cc_send_busy_bit = P9221R5_STAT_CCSENDBUSY; |
| chgr->ints.cc_data_rcvd_bit = P9221R5_STAT_CCDATARCVD; |
| chgr->ints.pp_rcvd_bit = P9221R5_STAT_PPRCVD; |
| chgr->ints.cc_error_bit = P9221R5_STAT_CCERROR; |
| chgr->ints.cc_reset_bit = P9221R5_STAT_CCRESET; |
| chgr->ints.propmode_stat_bit = 0; |
| chgr->ints.cdmode_change_bit = 0; |
| chgr->ints.cdmode_err_bit = 0; |
| chgr->ints.extended_mode_bit = 0; |
| |
| chgr->ints.hard_ocp_bit = 0; |
| chgr->ints.tx_conflict_bit = 0; |
| chgr->ints.csp_bit = 0; |
| chgr->ints.rx_connected_bit = 0; |
| chgr->ints.tx_fod_bit = 0; |
| chgr->ints.tx_underpower_bit = 0; |
| chgr->ints.tx_uvlo_bit = 0; |
| chgr->ints.pppsent_bit = 0; |
| chgr->ints.ocp_ping_bit = 0; |
| break; |
| } |
| |
| |
| chgr->ints.stat_limit_mask = (chgr->ints.over_uv_bit | |
| chgr->ints.over_temp_bit | |
| chgr->ints.over_volt_bit | |
| chgr->ints.over_curr_bit); |
| chgr->ints.stat_cc_mask = (chgr->ints.cc_send_busy_bit | |
| chgr->ints.cc_reset_bit | |
| chgr->ints.pp_rcvd_bit | |
| chgr->ints.cc_error_bit | |
| chgr->ints.cc_data_rcvd_bit); |
| chgr->ints.prop_mode_mask = (chgr->ints.propmode_stat_bit | |
| chgr->ints.cdmode_change_bit | |
| chgr->ints.cdmode_err_bit); |
| chgr->ints.stat_rtx_mask = (chgr->ints.stat_limit_mask | |
| chgr->ints.stat_cc_mask | |
| chgr->ints.mode_changed_bit | |
| chgr->ints.vout_changed_bit | |
| chgr->ints.hard_ocp_bit | |
| chgr->ints.tx_conflict_bit | |
| chgr->ints.csp_bit | |
| chgr->ints.rx_connected_bit | |
| chgr->ints.tx_fod_bit | |
| chgr->ints.tx_underpower_bit | |
| chgr->ints.tx_uvlo_bit | |
| chgr->ints.pppsent_bit | |
| chgr->ints.ocp_ping_bit); |
| } |
| |
| void p9221_chip_init_params(struct p9221_charger_data *chgr, u16 chip_id) |
| { |
| chgr->wlc_ocp = P9221R5_ILIM_MAX_UA; |
| chgr->low_neg_pwr_icl = P9221_DC_ICL_BPP_UA; |
| |
| switch (chip_id) { |
| case P9412_CHIP_ID: |
| chgr->reg_tx_id_addr = P9412_PROP_TX_ID_REG; |
| chgr->reg_tx_mfg_code_addr = P9382_EPP_TX_MFG_CODE_REG; |
| chgr->reg_packet_type_addr = P9412_COM_PACKET_TYPE_ADDR; |
| chgr->reg_set_pp_buf_addr = P9412_PP_SEND_BUF_START; |
| chgr->reg_get_pp_buf_addr = P9412_PP_RECV_BUF_START; |
| chgr->set_cmd_ccactivate_bit = P9412_COM_CCACTIVATE; |
| chgr->reg_set_fod_addr = P9221R5_FOD_REG; |
| chgr->reg_q_factor_addr = P9221R5_EPP_Q_FACTOR_REG; |
| chgr->reg_rf_value_addr = 0; |
| chgr->reg_csp_addr = P9221R5_CHARGE_STAT_REG; |
| chgr->reg_light_load_addr = 0; |
| chgr->reg_mot_addr = P9412_MOT_REG; |
| chgr->reg_cmfet_addr = P9412_CMFET_L_REG; |
| chgr->reg_hivout_cmfet_addr = P9412_HIVOUT_CMFET_REG; |
| chgr->reg_epp_tx_guarpwr_addr = P9221R5_EPP_TX_GUARANTEED_POWER_REG; |
| chgr->rtx_api_limit = P9412_I_API_Limit_1350MA; |
| chgr->reg_freq_limit_addr = 0; |
| chgr->reg_ask_mod_fet_addr = 0; |
| chgr->wlc_dc_comcap = P9412_CMFET_2_COMM; |
| chgr->wlc_dd_comcap = P9412_CMFET_ENABLE_ALL; |
| chgr->wlc_default_comcap = P9412_CMFET_DEFAULT; |
| chgr->wlc_disable_comcap = P9412_CMFET_DISABLE_ALL; |
| break; |
| case RA9530_CHIP_ID: |
| chgr->reg_tx_id_addr = P9412_PROP_TX_ID_REG; |
| chgr->reg_tx_mfg_code_addr = P9382_EPP_TX_MFG_CODE_REG; |
| chgr->reg_packet_type_addr = P9412_COM_PACKET_TYPE_ADDR; |
| chgr->reg_set_pp_buf_addr = P9412_PP_SEND_BUF_START; |
| chgr->reg_get_pp_buf_addr = P9412_PP_RECV_BUF_START; |
| chgr->set_cmd_ccactivate_bit = P9412_COM_CCACTIVATE; |
| chgr->reg_set_fod_addr = P9221R5_FOD_REG; |
| chgr->reg_q_factor_addr = P9221R5_EPP_Q_FACTOR_REG; |
| chgr->reg_rf_value_addr = RA9530_EPP_RF_REG; |
| chgr->reg_csp_addr = P9221R5_CHARGE_STAT_REG; |
| chgr->reg_light_load_addr = 0; |
| chgr->reg_mot_addr = 0; |
| chgr->reg_cmfet_addr = RA9530_CMFET_REG; |
| chgr->reg_epp_tx_guarpwr_addr = P9221R5_EPP_TX_GUARANTEED_POWER_REG; |
| chgr->rtx_api_limit = RA9530_I_API_Limit_1350MA; |
| chgr->rtx_ocp = RA9530_TXOCP_1400MA; |
| chgr->rtx_fod_thrsh = RA9530_TX_FOD_THRSH_1600; |
| chgr->ra9530_rtx_plim = RA9530_PLIM_900MA; |
| chgr->wlc_ocp = RA9530_ILIM_MAX_UA; |
| chgr->wlc_ovp = RA9530_OVSET_13V; |
| chgr->low_neg_pwr_icl = RA9530_ILIM_500UA; |
| chgr->det_off_debounce = 1000; |
| chgr->reg_freq_limit_addr = 0; |
| chgr->reg_ask_mod_fet_addr = 0; |
| chgr->wlc_dc_comcap = RA9530_CMFET_COM_A_B; |
| chgr->wlc_dd_comcap = RA9530_CMFET_ENABLE_ALL; |
| chgr->wlc_default_comcap = RA9530_CMFET_COM_1_2; |
| chgr->wlc_disable_comcap = RA9530_CMFET_DISABLE_ALL; |
| chgr->rtx_fb_freq_low_limit = RA9530_FREQ_PER_120; |
| chgr->rtx_hb_freq_low_limit = RA9530_FREQ_PER_105; |
| chgr->rtx_hb_ping_freq = RA9530_FREQ_PER_120; |
| break; |
| case P9382A_CHIP_ID: |
| chgr->reg_tx_id_addr = P9382_PROP_TX_ID_REG; |
| chgr->reg_tx_mfg_code_addr = P9382_EPP_TX_MFG_CODE_REG; |
| chgr->reg_packet_type_addr = P9382A_COM_PACKET_TYPE_ADDR; |
| chgr->reg_set_pp_buf_addr = P9382A_DATA_SEND_BUF_START; |
| chgr->reg_get_pp_buf_addr = P9382A_DATA_RECV_BUF_START; |
| chgr->set_cmd_ccactivate_bit = P9221R5_COM_CCACTIVATE; |
| chgr->reg_set_fod_addr = P9221R5_FOD_REG; |
| chgr->reg_q_factor_addr = P9221R5_EPP_Q_FACTOR_REG; |
| chgr->reg_rf_value_addr = 0; |
| chgr->reg_csp_addr = P9221R5_CHARGE_STAT_REG; |
| chgr->reg_light_load_addr = 0; |
| chgr->reg_mot_addr = 0; |
| chgr->reg_cmfet_addr = 0; |
| chgr->reg_epp_tx_guarpwr_addr = P9221R5_EPP_TX_GUARANTEED_POWER_REG; |
| chgr->reg_freq_limit_addr = 0; |
| chgr->reg_ask_mod_fet_addr = 0; |
| break; |
| case P9222_CHIP_ID: |
| chgr->reg_tx_id_addr = P9222RE_PROP_TX_ID_REG; |
| chgr->reg_tx_mfg_code_addr = P9222RE_TX_MFG_CODE_REG; |
| chgr->reg_packet_type_addr = P9222RE_COM_PACKET_TYPE_ADDR; |
| chgr->reg_set_pp_buf_addr = P9222RE_PP_SEND_BUF_START; |
| chgr->reg_get_pp_buf_addr = P9222RE_PP_RECV_BUF_START; |
| chgr->set_cmd_ccactivate_bit = P9222RE_COM_CCACTIVATE; |
| chgr->reg_set_fod_addr = P9222RE_FOD_REG; |
| chgr->reg_q_factor_addr = P9222RE_EPP_Q_FACTOR_REG; |
| chgr->reg_rf_value_addr = P9222RE_RESONANCE_FREQ_REG; |
| chgr->reg_csp_addr = P9222RE_CHARGE_STAT_REG; |
| chgr->reg_light_load_addr = P9222_RX_CALIBRATION_LIGHT_LOAD; |
| chgr->reg_mot_addr = P9222_MOT_REG; |
| chgr->reg_cmfet_addr = 0; |
| chgr->reg_epp_tx_guarpwr_addr = P9222RE_EPP_TX_GUARANTEED_POWER_REG; |
| chgr->reg_freq_limit_addr = P9222RE_FREQ_LIMIT_REG; |
| chgr->reg_ask_mod_fet_addr = P9222RE_ASK_MOD_REG; |
| break; |
| default: |
| chgr->reg_tx_id_addr = P9221R5_PROP_TX_ID_REG; |
| chgr->reg_tx_mfg_code_addr = P9221R5_EPP_TX_MFG_CODE_REG; |
| chgr->reg_packet_type_addr = 0; |
| chgr->reg_set_pp_buf_addr = P9221R5_DATA_SEND_BUF_START; |
| chgr->reg_get_pp_buf_addr = P9221R5_DATA_RECV_BUF_START; |
| chgr->set_cmd_ccactivate_bit = P9221R5_COM_CCACTIVATE; |
| chgr->reg_set_fod_addr = P9221R5_FOD_REG; |
| chgr->reg_q_factor_addr = P9221R5_EPP_Q_FACTOR_REG; |
| chgr->reg_rf_value_addr = 0; |
| chgr->reg_csp_addr = P9221R5_CHARGE_STAT_REG; |
| chgr->reg_light_load_addr = 0; |
| chgr->reg_mot_addr = 0; |
| chgr->reg_cmfet_addr = 0; |
| chgr->reg_epp_tx_guarpwr_addr = P9221R5_EPP_TX_GUARANTEED_POWER_REG; |
| chgr->reg_freq_limit_addr = 0; |
| chgr->reg_ask_mod_fet_addr = 0; |
| break; |
| } |
| } |
| |
| int p9221_chip_init_funcs(struct p9221_charger_data *chgr, u16 chip_id) |
| { |
| chgr->chip_get_iout = p9xxx_chip_get_iout; |
| chgr->chip_get_vout = p9xxx_chip_get_vout; |
| chgr->chip_set_cmd = p9xxx_chip_set_cmd_reg; |
| chgr->chip_get_op_freq = p9xxx_chip_get_op_freq; |
| chgr->chip_get_vrect = p9xxx_chip_get_vrect; |
| chgr->chip_get_vcpout = p9xxx_chip_get_vcpout; |
| chgr->chip_get_tx_epp_guarpwr = p9xxx_get_tx_epp_guarpwr; |
| chgr->chip_get_op_duty = p9221_chip_get_op_duty; |
| chgr->chip_get_op_bridge = p9221_chip_get_op_bridge; |
| chgr->chip_get_tx_pwr = p9221_chip_get_tx_pwr; |
| chgr->chip_get_rx_pwr = p9221_chip_get_rx_pwr; |
| chgr->chip_is_calibrated = p9xxx_chip_is_calibrated; |
| chgr->chip_set_cmfet = p9xxx_chip_set_cmfet; |
| chgr->chip_set_bpp_icl = p9xxx_chip_set_bpp_icl; |
| chgr->chip_get_ping_freq = p9xxx_chip_get_ping_freq; |
| chgr->chip_magsafe_optimized = p9xxx_magsafe_optimized; |
| chgr->chip_is_vout_on = p9xxx_is_vout_on; |
| chgr->chip_set_ovp = p9xxx_chip_set_ovp; |
| |
| switch (chip_id) { |
| case P9412_CHIP_ID: |
| chgr->rx_buf_size = P9412_DATA_BUF_SIZE; |
| chgr->tx_buf_size = P9412_DATA_BUF_SIZE; |
| |
| chgr->chip_get_rx_ilim = p9412_chip_get_rx_ilim; |
| chgr->chip_set_rx_ilim = p9412_chip_set_rx_ilim; |
| chgr->chip_get_tx_ilim = p9412_chip_get_tx_ilim; |
| chgr->chip_set_tx_ilim = p9412_chip_set_tx_ilim; |
| chgr->chip_get_die_temp = p9412_chip_get_die_temp; |
| chgr->chip_get_vout_max = p9412_chip_get_vout_max; |
| chgr->chip_set_vout_max = p9412_chip_set_vout_max; |
| chgr->chip_tx_mode_en = p9412_chip_tx_mode; |
| chgr->chip_get_data_buf = p9412_get_data_buf; |
| chgr->chip_set_data_buf = p9412_set_data_buf; |
| chgr->chip_get_cc_recv_size = p9412_get_cc_recv_size; |
| chgr->chip_set_cc_send_size = p9412_set_cc_send_size; |
| chgr->chip_get_align_x = p9412_get_align_x; |
| chgr->chip_get_align_y = p9412_get_align_y; |
| chgr->chip_send_ccreset = p9412_send_ccreset; |
| chgr->chip_send_eop = p9412_send_eop; |
| chgr->chip_get_sys_mode = p9412_chip_get_sys_mode; |
| chgr->chip_renegotiate_pwr = p9412_chip_renegotiate_pwr; |
| chgr->chip_prop_mode_en = p9412_prop_mode_enable; |
| chgr->chip_check_neg_power = p9xxx_check_neg_power; |
| chgr->chip_send_txid = p9xxx_send_txid; |
| chgr->chip_send_csp_in_txmode = p9xxx_send_csp_in_txmode; |
| chgr->chip_capdiv_en = p9412_capdiv_en; |
| chgr->chip_get_vcpout = p9412_chip_get_vcpout; |
| chgr->chip_get_op_duty = p9xxx_chip_get_op_duty; |
| chgr->chip_get_tx_pwr = p9xxx_chip_get_tx_pwr; |
| chgr->chip_get_rx_pwr = p9xxx_chip_get_rx_pwr; |
| chgr->chip_is_calibrated = p9412_chip_is_calibrated; |
| break; |
| case RA9530_CHIP_ID: |
| chgr->rx_buf_size = RA9530_DATA_BUF_SIZE; |
| chgr->tx_buf_size = RA9530_DATA_BUF_SIZE; |
| chgr->chip_get_rx_ilim = ra9530_chip_get_rx_ilim; |
| chgr->chip_set_rx_ilim = ra9530_chip_set_rx_ilim; |
| |
| chgr->chip_get_tx_ilim = p9412_chip_get_tx_ilim; |
| chgr->chip_set_tx_ilim = p9412_chip_set_tx_ilim; |
| chgr->chip_get_die_temp = p9412_chip_get_die_temp; |
| chgr->chip_get_vout_max = p9412_chip_get_vout_max; |
| chgr->chip_set_vout_max = ra9530_chip_set_vout_max; |
| chgr->chip_tx_mode_en = ra9530_chip_tx_mode; |
| chgr->chip_get_data_buf = ra9530_get_data_buf; |
| chgr->chip_set_data_buf = ra9530_set_data_buf; |
| chgr->chip_get_cc_recv_size = ra9530_get_cc_recv_size; |
| chgr->chip_set_cc_send_size = ra9530_set_cc_send_size; |
| chgr->chip_get_align_x = ra9530_get_align_x; |
| chgr->chip_get_align_y = ra9530_get_align_y; |
| chgr->chip_send_ccreset = ra9530_send_ccreset; |
| chgr->chip_send_eop = p9412_send_eop; |
| chgr->chip_get_sys_mode = p9412_chip_get_sys_mode; |
| chgr->chip_renegotiate_pwr = p9412_chip_renegotiate_pwr; |
| chgr->chip_prop_mode_en = ra9530_prop_mode_enable; |
| chgr->chip_check_neg_power = p9xxx_check_neg_power; |
| chgr->chip_send_txid = p9xxx_send_txid; |
| chgr->chip_send_csp_in_txmode = p9xxx_send_csp_in_txmode; |
| chgr->chip_capdiv_en = ra9530_capdiv_en; |
| chgr->chip_get_op_duty = ra9530_chip_get_op_duty; |
| chgr->chip_get_op_bridge = ra9530_chip_get_op_bridge; |
| chgr->chip_get_tx_pwr = p9xxx_chip_get_tx_pwr; |
| chgr->chip_get_rx_pwr = p9xxx_chip_get_rx_pwr; |
| chgr->chip_is_calibrated = ra9530_chip_is_calibrated; |
| chgr->chip_set_cmfet = ra9530_chip_set_cmfet; |
| chgr->chip_set_bpp_icl = ra9530_chip_set_bpp_icl; |
| chgr->chip_set_ovp = ra9530_chip_set_ovp; |
| break; |
| case P9382A_CHIP_ID: |
| chgr->rx_buf_size = P9221R5_DATA_RECV_BUF_SIZE; |
| chgr->tx_buf_size = P9221R5_DATA_SEND_BUF_SIZE; |
| |
| chgr->chip_get_rx_ilim = p9221_chip_get_rx_ilim; |
| chgr->chip_set_rx_ilim = p9221_chip_set_rx_ilim; |
| chgr->chip_get_tx_ilim = p9382_chip_get_tx_ilim; |
| chgr->chip_set_tx_ilim = p9382_chip_set_tx_ilim; |
| chgr->chip_get_die_temp = p9221_chip_get_die_temp; |
| chgr->chip_get_vout_max = p9832_chip_get_vout_max; |
| chgr->chip_set_vout_max = p9832_chip_set_vout_max; |
| chgr->chip_tx_mode_en = p9382_chip_tx_mode; |
| chgr->chip_get_data_buf = p9382_get_data_buf; |
| chgr->chip_set_data_buf = p9382_set_data_buf; |
| chgr->chip_get_cc_recv_size = p9221_get_cc_recv_size; |
| chgr->chip_set_cc_send_size = p9382_set_cc_send_size; |
| chgr->chip_get_align_x = p9221_get_align_x; |
| chgr->chip_get_align_y = p9221_get_align_y; |
| chgr->chip_send_ccreset = p9221_send_ccreset; |
| chgr->chip_send_eop = p9221_send_eop; |
| chgr->chip_get_sys_mode = p9221_chip_get_sys_mode; |
| chgr->chip_renegotiate_pwr = p9221_chip_renegotiate_pwr; |
| chgr->chip_prop_mode_en = p9221_prop_mode_enable; |
| chgr->chip_check_neg_power = p9xxx_check_neg_power; |
| chgr->chip_send_txid = p9xxx_send_txid; |
| chgr->chip_send_csp_in_txmode = p9xxx_send_csp_in_txmode; |
| chgr->chip_capdiv_en = p9221_capdiv_en; |
| chgr->chip_get_op_duty = p9xxx_chip_get_op_duty; |
| break; |
| case P9222_CHIP_ID: |
| chgr->chip_get_iout = p9222_chip_get_iout; |
| chgr->chip_get_vout = p9222_chip_get_vout; |
| chgr->chip_get_op_freq = p9222_chip_get_op_freq; |
| chgr->chip_get_vrect = p9222_chip_get_vrect; |
| chgr->chip_set_cmd = p9222_chip_set_cmd_reg; |
| |
| chgr->rx_buf_size = P9221R5_DATA_RECV_BUF_SIZE; |
| chgr->tx_buf_size = P9221R5_DATA_SEND_BUF_SIZE; |
| |
| chgr->chip_get_rx_ilim = p9222_chip_get_rx_ilim; |
| chgr->chip_set_rx_ilim = p9222_chip_set_rx_ilim; |
| chgr->chip_get_tx_ilim = p9221_chip_get_tx_ilim; |
| chgr->chip_set_tx_ilim = p9221_chip_set_tx_ilim; |
| chgr->chip_get_die_temp = p9222_chip_get_die_temp; |
| chgr->chip_get_vout_max = p9222_chip_get_vout_max; |
| chgr->chip_set_vout_max = p9222_chip_set_vout_max; |
| chgr->chip_tx_mode_en = p9221_chip_tx_mode; |
| chgr->chip_get_data_buf = p9222_get_data_buf; |
| chgr->chip_set_data_buf = p9222_set_data_buf; |
| chgr->chip_get_cc_recv_size = p9222_get_cc_recv_size; |
| chgr->chip_set_cc_send_size = p9222_set_cc_send_size; |
| chgr->chip_get_align_x = p9221_get_align_x; |
| chgr->chip_get_align_y = p9221_get_align_y; |
| chgr->chip_send_ccreset = p9221_send_ccreset; |
| chgr->chip_send_eop = p9222_send_eop; |
| chgr->chip_get_sys_mode = p9222_chip_get_sys_mode; |
| chgr->chip_renegotiate_pwr = p9222_chip_renegotiate_pwr; |
| chgr->chip_prop_mode_en = p9221_prop_mode_enable; |
| chgr->chip_check_neg_power = p9222_check_neg_power; |
| chgr->chip_send_txid = p9221_send_txid; |
| chgr->chip_send_csp_in_txmode = p9221_send_csp_in_txmode; |
| chgr->chip_capdiv_en = p9221_capdiv_en; |
| chgr->chip_get_ping_freq = p9222_chip_get_ping_freq; |
| chgr->chip_magsafe_optimized = p9222_magsafe_optimized; |
| chgr->chip_set_bpp_icl = p9222_chip_set_bpp_icl; |
| break; |
| default: |
| chgr->rx_buf_size = P9221R5_DATA_RECV_BUF_SIZE; |
| chgr->tx_buf_size = P9221R5_DATA_SEND_BUF_SIZE; |
| |
| chgr->chip_get_rx_ilim = p9221_chip_get_rx_ilim; |
| chgr->chip_set_rx_ilim = p9221_chip_set_rx_ilim; |
| chgr->chip_get_tx_ilim = p9221_chip_get_tx_ilim; |
| chgr->chip_set_tx_ilim = p9221_chip_set_tx_ilim; |
| chgr->chip_get_die_temp = p9221_chip_get_die_temp; |
| chgr->chip_get_vout_max = p9221_chip_get_vout_max; |
| chgr->chip_set_vout_max = p9221_chip_set_vout_max; |
| chgr->chip_tx_mode_en = p9221_chip_tx_mode; |
| chgr->chip_set_data_buf = p9221_set_data_buf; |
| chgr->chip_get_data_buf = p9221_get_data_buf; |
| chgr->chip_get_cc_recv_size = p9221_get_cc_recv_size; |
| chgr->chip_set_cc_send_size = p9221_set_cc_send_size; |
| chgr->chip_get_align_x = p9221_get_align_x; |
| chgr->chip_get_align_y = p9221_get_align_y; |
| chgr->chip_send_ccreset = p9221_send_ccreset; |
| chgr->chip_send_eop = p9221_send_eop; |
| chgr->chip_get_sys_mode = p9221_chip_get_sys_mode; |
| chgr->chip_renegotiate_pwr = p9221_chip_renegotiate_pwr; |
| chgr->chip_prop_mode_en = p9221_prop_mode_enable; |
| chgr->chip_check_neg_power = p9xxx_check_neg_power; |
| chgr->chip_send_txid = p9221_send_txid; |
| chgr->chip_send_csp_in_txmode = p9221_send_csp_in_txmode; |
| chgr->chip_capdiv_en = p9221_capdiv_en; |
| break; |
| } |
| |
| chgr->rx_buf = devm_kzalloc(chgr->dev, chgr->rx_buf_size, GFP_KERNEL); |
| if (chgr->rx_buf == NULL) |
| return -ENOMEM; |
| |
| chgr->tx_buf = devm_kzalloc(chgr->dev, chgr->tx_buf_size, GFP_KERNEL); |
| if (chgr->tx_buf == NULL) |
| return -ENOMEM; |
| |
| return 0; |
| } |
| |
| #if IS_ENABLED(CONFIG_GPIOLIB) |
| int p9xxx_gpio_set_value(struct p9221_charger_data *chgr, int gpio, int value) |
| { |
| if (gpio <= 0) |
| return -EINVAL; |
| |
| logbuffer_log(chgr->log, "%s: set gpio %d to %d", __func__, gpio, value); |
| gpio_set_value_cansleep(gpio, value); |
| |
| return 0; |
| } |
| |
| static int p9xxx_gpio_get_direction(struct gpio_chip *chip, |
| unsigned int offset) |
| { |
| return GPIOF_DIR_OUT; |
| } |
| |
| static int p9xxx_gpio_get(struct gpio_chip *chip, unsigned int offset) |
| { |
| //struct p9221_charger_data *charger = gpiochip_get_data(chip); |
| int value = 0; |
| |
| switch (offset) { |
| case P9XXX_GPIO_CPOUT_EN: |
| break; |
| case P9412_GPIO_CPOUT21_EN: |
| // read cap divider |
| break; |
| case P9XXX_GPIO_CPOUT_CTL_EN: |
| break; |
| default: |
| break; |
| } |
| |
| return value; |
| } |
| |
| #define P9412_BPP_VOUT_DFLT 5000 |
| #define P9412_BPP_WLC_OTG_VOUT 5200 |
| static void p9xxx_gpio_set(struct gpio_chip *chip, unsigned int offset, int value) |
| { |
| struct p9221_charger_data *charger = gpiochip_get_data(chip); |
| int ret = 0, vout_mv; |
| int prev_state; |
| u8 mode; |
| |
| switch (offset) { |
| case P9XXX_GPIO_CPOUT_EN: |
| /* take offline (if online) and set/reset QI_EN_L */ |
| ret = gvotable_cast_bool_vote(charger->wlc_disable_votable, |
| CPOUT_EN_VOTER, !value); |
| break; |
| case P9412_GPIO_CPOUT21_EN: |
| /* TODO: no-op for FW38+ */ |
| mode = !!value ? CDMODE_CAP_DIV_MODE : 0; |
| ret = p9412_capdiv_en(charger, mode); |
| break; |
| case P9XXX_GPIO_CPOUT_CTL_EN: |
| if (p9221_is_epp(charger)) |
| break; |
| /* No need if DD is triggered */ |
| if (charger->trigger_power_mitigation) |
| break; |
| /* If Vout is already > 5.2V, no change is required */ |
| ret = charger->chip_get_vout_max(charger, &vout_mv); |
| if (ret == 0 && vout_mv > P9412_BPP_WLC_OTG_VOUT) |
| break; |
| /* b/174068520: set vout to 5.2 for BPP_WLC_RX+OTG */ |
| if (value) |
| ret = charger->chip_set_vout_max(charger, P9412_BPP_WLC_OTG_VOUT); |
| else if (value == 0) |
| ret = charger->chip_set_vout_max(charger, P9412_BPP_VOUT_DFLT); |
| break; |
| case P9XXX_GPIO_VBUS_EN: |
| if (charger->pdata->wlc_en < 0) |
| break; |
| value = (!!value) ^ charger->pdata->wlc_en_act_low; |
| gpio_direction_output(charger->pdata->wlc_en, value); |
| break; |
| case P9XXX_GPIO_DC_SW_EN: |
| ret = p9xxx_gpio_set_value(charger, charger->pdata->dc_switch_gpio, value); |
| break; |
| case P9XXX_GPIO_ONLINE_SPOOF: |
| mutex_lock(&charger->irq_det_lock); |
| if (charger->pdata->irq_det_gpio >= 0 && |
| value && charger->online && !charger->online_spoof) { |
| if (charger->pdata->ldo_en_gpio > 0) { |
| gpio_set_value_cansleep(charger->pdata->ldo_en_gpio, 1); |
| logbuffer_prlog(charger->log, |
| "online_spoof=1 ldo_en=1 online=%d", charger->online); |
| } else { |
| logbuffer_prlog(charger->log, "online_spoof=1"); |
| } |
| enable_irq(charger->pdata->irq_det_int); |
| enable_irq_wake(charger->pdata->irq_det_int); |
| charger->online_spoof = true; |
| cancel_delayed_work(&charger->stop_online_spoof_work); |
| charger->det_status = gpio_get_value_cansleep(charger->pdata->irq_det_gpio); |
| /* Prevent device removal while spoofing is enabled */ |
| if (charger->det_status == 1) { |
| __pm_stay_awake(charger->det_status_ws); |
| mod_delayed_work(system_wq, &charger->change_det_status_work, |
| msecs_to_jiffies(DET_READY_DEBOUNCE_MS)); |
| } |
| } |
| mutex_unlock(&charger->irq_det_lock); |
| if (!value && charger->pdata->ldo_en_gpio > 0) { |
| gpio_set_value_cansleep(charger->pdata->ldo_en_gpio, 0); |
| logbuffer_prlog(charger->log, "pxxx_gpio online_spoof=0 ldo_en_gpio=0"); |
| } |
| pr_debug("%s: GPIO offset=%d value=%d charger->det_status:%d online=%d", |
| __func__, offset, value, charger->det_status, charger->online); |
| break; |
| case P9XXX_GPIO_RTX_STATE: |
| if (!charger->pdata->has_rtx_gpio) { |
| ret = -ENOTSUPP; |
| break; |
| } |
| mutex_lock(&charger->rtx_gpio_lock); |
| prev_state = charger->rtx_gpio_state; |
| |
| if (prev_state == RTX_RETRY) { |
| mutex_unlock(&charger->rtx_gpio_lock); |
| break; |
| } |
| |
| if (value == 0 && prev_state == RTX_READY && charger->ben_state == RTX_BEN_ON) |
| charger->rtx_gpio_state = RTX_RETRY; |
| else if (value == 1) |
| charger->rtx_gpio_state = RTX_READY; |
| |
| mutex_unlock(&charger->rtx_gpio_lock); |
| break; |
| case P9XXX_GPIO_RTX_SUPP: |
| if (!charger->pdata->has_rtx_gpio) { |
| ret = -ENOTSUPP; |
| break; |
| } |
| mutex_lock(&charger->rtx_gpio_lock); |
| |
| if (value == 0) { |
| charger->rtx_gpio_state = RTX_NOT_SUPP; |
| charger->rtx_err |= RTX_CHRG_NOTSUP_BIT; |
| schedule_work(&charger->uevent_work); |
| } else if (value == 1 && charger->rtx_err & RTX_CHRG_NOTSUP_BIT) { |
| charger->rtx_err &= ~RTX_CHRG_NOTSUP_BIT; |
| schedule_work(&charger->uevent_work); |
| } |
| mutex_unlock(&charger->rtx_gpio_lock); |
| break; |
| default: |
| ret = -EINVAL; |
| break; |
| } |
| |
| pr_debug("%s: GPIO offset=%d value=%d ret:%d\n", __func__, |
| offset, value, ret); |
| |
| if (ret < 0) |
| dev_err(&charger->client->dev, "GPIO%d: value=%d ret:%d\n", |
| offset, value, ret); |
| } |
| |
| void p9xxx_gpio_init(struct p9221_charger_data *charger) |
| { |
| charger->gpio.owner = THIS_MODULE; |
| charger->gpio.label = "p9xxx_gpio"; |
| charger->gpio.get_direction = p9xxx_gpio_get_direction; |
| charger->gpio.get = p9xxx_gpio_get; |
| charger->gpio.set = p9xxx_gpio_set; |
| charger->gpio.base = -1; |
| charger->gpio.ngpio = P9XXX_NUM_GPIOS; |
| charger->gpio.can_sleep = true; |
| } |
| #endif |