3290 lines
82 KiB
C
Executable File
3290 lines
82 KiB
C
Executable File
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (c) 2018-2021 The Linux Foundation. All rights reserved.
|
|
* Copyright (c) 2022-2024, Qualcomm Innovation Center, Inc. All rights reserved.
|
|
*/
|
|
|
|
#include <linux/debugfs.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/device.h>
|
|
#include <linux/module.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/regmap.h>
|
|
#include <linux/power_supply.h>
|
|
#include <linux/of.h>
|
|
#include <linux/of_device.h>
|
|
#include <linux/of_irq.h>
|
|
#include <linux/log2.h>
|
|
#include <linux/regulator/driver.h>
|
|
#include <linux/regulator/of_regulator.h>
|
|
#include <linux/regulator/machine.h>
|
|
#include <linux/iio/consumer.h>
|
|
#include <linux/pmic-voter.h>
|
|
#include <linux/usb/typec.h>
|
|
#include "smb5-reg.h"
|
|
#include "smb5-lib.h"
|
|
#include "smb5-iio.h"
|
|
#include "schgm-flash.h"
|
|
|
|
static struct smb_params smb5_pmi632_params = {
|
|
.fcc = {
|
|
.name = "fast charge current",
|
|
.reg = CHGR_FAST_CHARGE_CURRENT_CFG_REG,
|
|
.min_u = 0,
|
|
.max_u = 3000000,
|
|
.step_u = 50000,
|
|
},
|
|
.fv = {
|
|
.name = "float voltage",
|
|
.reg = CHGR_FLOAT_VOLTAGE_CFG_REG,
|
|
.min_u = 3600000,
|
|
.max_u = 4800000,
|
|
.step_u = 10000,
|
|
},
|
|
.usb_icl = {
|
|
.name = "usb input current limit",
|
|
.reg = USBIN_CURRENT_LIMIT_CFG_REG,
|
|
.min_u = 0,
|
|
.max_u = 3000000,
|
|
.step_u = 50000,
|
|
},
|
|
.icl_max_stat = {
|
|
.name = "dcdc icl max status",
|
|
.reg = ICL_MAX_STATUS_REG,
|
|
.min_u = 0,
|
|
.max_u = 3000000,
|
|
.step_u = 50000,
|
|
},
|
|
.icl_stat = {
|
|
.name = "input current limit status",
|
|
.reg = ICL_STATUS_REG,
|
|
.min_u = 0,
|
|
.max_u = 3000000,
|
|
.step_u = 50000,
|
|
},
|
|
.otg_cl = {
|
|
.name = "usb otg current limit",
|
|
.reg = DCDC_OTG_CURRENT_LIMIT_CFG_REG,
|
|
.min_u = 500000,
|
|
.max_u = 1000000,
|
|
.step_u = 250000,
|
|
},
|
|
.jeita_cc_comp_hot = {
|
|
.name = "jeita fcc reduction",
|
|
.reg = JEITA_CCCOMP_CFG_HOT_REG,
|
|
.min_u = 0,
|
|
.max_u = 1575000,
|
|
.step_u = 25000,
|
|
},
|
|
.jeita_cc_comp_cold = {
|
|
.name = "jeita fcc reduction",
|
|
.reg = JEITA_CCCOMP_CFG_COLD_REG,
|
|
.min_u = 0,
|
|
.max_u = 1575000,
|
|
.step_u = 25000,
|
|
},
|
|
.freq_switcher = {
|
|
.name = "switching frequency",
|
|
.reg = DCDC_FSW_SEL_REG,
|
|
.min_u = 600,
|
|
.max_u = 1200,
|
|
.step_u = 400,
|
|
.set_proc = smblib_set_chg_freq,
|
|
},
|
|
.aicl_5v_threshold = {
|
|
.name = "AICL 5V threshold",
|
|
.reg = USBIN_5V_AICL_THRESHOLD_REG,
|
|
.min_u = 4000,
|
|
.max_u = 4700,
|
|
.step_u = 100,
|
|
},
|
|
.aicl_cont_threshold = {
|
|
.name = "AICL CONT threshold",
|
|
.reg = USBIN_CONT_AICL_THRESHOLD_REG,
|
|
.min_u = 4000,
|
|
.max_u = 8800,
|
|
.step_u = 100,
|
|
.get_proc = smblib_get_aicl_cont_threshold,
|
|
.set_proc = smblib_set_aicl_cont_threshold,
|
|
},
|
|
};
|
|
|
|
static struct smb_params smb5_pm8150b_params = {
|
|
.fcc = {
|
|
.name = "fast charge current",
|
|
.reg = CHGR_FAST_CHARGE_CURRENT_CFG_REG,
|
|
.min_u = 0,
|
|
.max_u = 8000000,
|
|
.step_u = 50000,
|
|
},
|
|
.fv = {
|
|
.name = "float voltage",
|
|
.reg = CHGR_FLOAT_VOLTAGE_CFG_REG,
|
|
.min_u = 3600000,
|
|
.max_u = 4790000,
|
|
.step_u = 10000,
|
|
},
|
|
.usb_icl = {
|
|
.name = "usb input current limit",
|
|
.reg = USBIN_CURRENT_LIMIT_CFG_REG,
|
|
.min_u = 0,
|
|
.max_u = 5000000,
|
|
.step_u = 50000,
|
|
},
|
|
.icl_max_stat = {
|
|
.name = "dcdc icl max status",
|
|
.reg = ICL_MAX_STATUS_REG,
|
|
.min_u = 0,
|
|
.max_u = 5000000,
|
|
.step_u = 50000,
|
|
},
|
|
.icl_stat = {
|
|
.name = "aicl icl status",
|
|
.reg = AICL_ICL_STATUS_REG,
|
|
.min_u = 0,
|
|
.max_u = 5000000,
|
|
.step_u = 50000,
|
|
},
|
|
.otg_cl = {
|
|
.name = "usb otg current limit",
|
|
.reg = DCDC_OTG_CURRENT_LIMIT_CFG_REG,
|
|
.min_u = 500000,
|
|
.max_u = 3000000,
|
|
.step_u = 500000,
|
|
},
|
|
.dc_icl = {
|
|
.name = "DC input current limit",
|
|
.reg = DCDC_CFG_REF_MAX_PSNS_REG,
|
|
.min_u = 0,
|
|
.max_u = DCIN_ICL_MAX_UA,
|
|
.step_u = 50000,
|
|
},
|
|
.jeita_cc_comp_hot = {
|
|
.name = "jeita fcc reduction",
|
|
.reg = JEITA_CCCOMP_CFG_HOT_REG,
|
|
.min_u = 0,
|
|
.max_u = 8000000,
|
|
.step_u = 25000,
|
|
.set_proc = NULL,
|
|
},
|
|
.jeita_cc_comp_cold = {
|
|
.name = "jeita fcc reduction",
|
|
.reg = JEITA_CCCOMP_CFG_COLD_REG,
|
|
.min_u = 0,
|
|
.max_u = 8000000,
|
|
.step_u = 25000,
|
|
.set_proc = NULL,
|
|
},
|
|
.freq_switcher = {
|
|
.name = "switching frequency",
|
|
.reg = DCDC_FSW_SEL_REG,
|
|
.min_u = 600,
|
|
.max_u = 1200,
|
|
.step_u = 400,
|
|
.set_proc = smblib_set_chg_freq,
|
|
},
|
|
.aicl_5v_threshold = {
|
|
.name = "AICL 5V threshold",
|
|
.reg = USBIN_5V_AICL_THRESHOLD_REG,
|
|
.min_u = 4000,
|
|
.max_u = 4700,
|
|
.step_u = 100,
|
|
},
|
|
.aicl_cont_threshold = {
|
|
.name = "AICL CONT threshold",
|
|
.reg = USBIN_CONT_AICL_THRESHOLD_REG,
|
|
.min_u = 4000,
|
|
.max_u = 11800,
|
|
.step_u = 100,
|
|
.get_proc = smblib_get_aicl_cont_threshold,
|
|
.set_proc = smblib_set_aicl_cont_threshold,
|
|
},
|
|
};
|
|
|
|
struct smb_dt_props {
|
|
int usb_icl_ua;
|
|
enum float_options float_option;
|
|
int chg_inhibit_thr_mv;
|
|
bool no_battery;
|
|
bool hvdcp_disable;
|
|
bool hvdcp_autonomous;
|
|
bool adc_based_aicl;
|
|
int sec_charger_config;
|
|
int auto_recharge_soc;
|
|
int auto_recharge_vbat_mv;
|
|
int wd_bark_time;
|
|
int wd_snarl_time_cfg;
|
|
int batt_profile_fcc_ua;
|
|
int batt_profile_fv_uv;
|
|
int term_current_src;
|
|
int term_current_thresh_hi_ma;
|
|
int term_current_thresh_lo_ma;
|
|
};
|
|
|
|
struct smb5 {
|
|
struct smb_charger chg;
|
|
struct dentry *dfs_root;
|
|
struct smb_dt_props dt;
|
|
unsigned int nchannels;
|
|
struct iio_channel *iio_chans;
|
|
struct iio_chan_spec *iio_chan_ids;
|
|
};
|
|
|
|
static int __debug_mask;
|
|
|
|
static ssize_t pd_disabled_show(struct device *dev, struct device_attribute
|
|
*attr, char *buf)
|
|
{
|
|
struct smb5 *chip = dev_get_drvdata(dev);
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
return scnprintf(buf, PAGE_SIZE, "%d\n", chg->pd_disabled);
|
|
}
|
|
|
|
static ssize_t pd_disabled_store(struct device *dev, struct device_attribute
|
|
*attr, const char *buf, size_t count)
|
|
{
|
|
int val;
|
|
struct smb5 *chip = dev_get_drvdata(dev);
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
if (kstrtos32(buf, 0, &val))
|
|
return -EINVAL;
|
|
|
|
chg->pd_disabled = val;
|
|
|
|
return count;
|
|
}
|
|
static DEVICE_ATTR_RW(pd_disabled);
|
|
|
|
static ssize_t weak_chg_icl_ua_show(struct device *dev, struct device_attribute
|
|
*attr, char *buf)
|
|
{
|
|
struct smb5 *chip = dev_get_drvdata(dev);
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
return scnprintf(buf, PAGE_SIZE, "%d\n", chg->weak_chg_icl_ua);
|
|
}
|
|
|
|
static ssize_t weak_chg_icl_ua_store(struct device *dev, struct device_attribute
|
|
*attr, const char *buf, size_t count)
|
|
{
|
|
int val;
|
|
struct smb5 *chip = dev_get_drvdata(dev);
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
if (kstrtos32(buf, 0, &val))
|
|
return -EINVAL;
|
|
|
|
chg->weak_chg_icl_ua = val;
|
|
|
|
return count;
|
|
}
|
|
static DEVICE_ATTR_RW(weak_chg_icl_ua);
|
|
|
|
static struct attribute *smb5_attrs[] = {
|
|
&dev_attr_pd_disabled.attr,
|
|
&dev_attr_weak_chg_icl_ua.attr,
|
|
NULL,
|
|
};
|
|
ATTRIBUTE_GROUPS(smb5);
|
|
|
|
enum {
|
|
BAT_THERM = 0,
|
|
MISC_THERM,
|
|
CONN_THERM,
|
|
SMB_THERM,
|
|
};
|
|
|
|
static const struct clamp_config clamp_levels[] = {
|
|
{ {0x11C6, 0x11F9, 0x13F1}, {0x60, 0x2E, 0x90} },
|
|
{ {0x11C6, 0x11F9, 0x13F1}, {0x60, 0x2B, 0x9C} },
|
|
};
|
|
|
|
#define PMI632_MAX_ICL_UA 3000000
|
|
#define PM6150_MAX_FCC_UA 3000000
|
|
static int smb5_chg_config_init(struct smb5 *chip)
|
|
{
|
|
struct smb_charger *chg = &chip->chg;
|
|
struct device_node *node = chg->dev->of_node;
|
|
int subtype = (u8)(unsigned long)of_device_get_match_data(chg->dev);
|
|
|
|
switch (subtype) {
|
|
case PM8150B:
|
|
chip->chg.chg_param.smb_version = PM8150B;
|
|
chg->param = smb5_pm8150b_params;
|
|
chg->name = "pm8150b_charger";
|
|
chg->wa_flags |= CHG_TERMINATION_WA;
|
|
break;
|
|
case PM7250B:
|
|
chip->chg.chg_param.smb_version = PM7250B;
|
|
chg->param = smb5_pm8150b_params;
|
|
chg->name = "pm7250b_charger";
|
|
chg->wa_flags |= CHG_TERMINATION_WA;
|
|
chg->uusb_moisture_protection_capable = true;
|
|
break;
|
|
case PM6150:
|
|
chip->chg.chg_param.smb_version = PM6150;
|
|
chg->param = smb5_pm8150b_params;
|
|
chg->name = "pm6150_charger";
|
|
chg->wa_flags |= SW_THERM_REGULATION_WA | CHG_TERMINATION_WA;
|
|
chg->uusb_moisture_protection_capable = true;
|
|
chg->main_fcc_max = PM6150_MAX_FCC_UA;
|
|
break;
|
|
case PMI632:
|
|
chip->chg.chg_param.smb_version = PMI632;
|
|
chg->wa_flags |= WEAK_ADAPTER_WA | USBIN_OV_WA
|
|
| CHG_TERMINATION_WA | USBIN_ADC_WA
|
|
| SKIP_MISC_PBS_IRQ_WA;
|
|
chg->param = smb5_pmi632_params;
|
|
chg->use_extcon = true;
|
|
chg->name = "pmi632_charger";
|
|
/* PMI632 does not support PD */
|
|
chg->pd_not_supported = true;
|
|
chg->lpd_disabled = true;
|
|
chg->uusb_moisture_protection_enabled = true;
|
|
chg->hw_max_icl_ua =
|
|
(chip->dt.usb_icl_ua > 0) ? chip->dt.usb_icl_ua
|
|
: PMI632_MAX_ICL_UA;
|
|
break;
|
|
default:
|
|
pr_err("PMIC subtype %d not supported\n", subtype);
|
|
return -EINVAL;
|
|
}
|
|
|
|
chg->chg_freq.freq_5V = 600;
|
|
chg->chg_freq.freq_6V_8V = 800;
|
|
chg->chg_freq.freq_9V = 1050;
|
|
chg->chg_freq.freq_12V = 1200;
|
|
chg->chg_freq.freq_removal = 1050;
|
|
chg->chg_freq.freq_below_otg_threshold = 800;
|
|
chg->chg_freq.freq_above_otg_threshold = 800;
|
|
|
|
if (of_property_read_bool(node, "qcom,disable-sw-thermal-regulation"))
|
|
chg->wa_flags &= ~SW_THERM_REGULATION_WA;
|
|
|
|
if (of_property_read_bool(node, "qcom,disable-fcc-restriction"))
|
|
chg->main_fcc_max = -EINVAL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define PULL_NO_PULL 0
|
|
#define PULL_30K 30
|
|
#define PULL_100K 100
|
|
#define PULL_400K 400
|
|
static int get_valid_pullup(int pull_up)
|
|
{
|
|
/* pull up can only be 0/30K/100K/400K) */
|
|
switch (pull_up) {
|
|
case PULL_NO_PULL:
|
|
return INTERNAL_PULL_NO_PULL;
|
|
case PULL_30K:
|
|
return INTERNAL_PULL_30K_PULL;
|
|
case PULL_100K:
|
|
return INTERNAL_PULL_100K_PULL;
|
|
case PULL_400K:
|
|
return INTERNAL_PULL_400K_PULL;
|
|
default:
|
|
return INTERNAL_PULL_100K_PULL;
|
|
}
|
|
}
|
|
|
|
#define INTERNAL_PULL_UP_MASK 0x3
|
|
static int smb5_configure_internal_pull(struct smb_charger *chg, int type,
|
|
int pull)
|
|
{
|
|
int rc;
|
|
int shift = type * 2;
|
|
u8 mask = INTERNAL_PULL_UP_MASK << shift;
|
|
u8 val = pull << shift;
|
|
|
|
rc = smblib_masked_write(chg, BATIF_ADC_INTERNAL_PULL_UP_REG,
|
|
mask, val);
|
|
if (rc < 0)
|
|
dev_err(chg->dev,
|
|
"Couldn't configure ADC pull-up reg rc=%d\n", rc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
#define MICRO_1P5A 1500000
|
|
#define MICRO_P1A 100000
|
|
#define MICRO_1PA 1000000
|
|
#define MICRO_3PA 3000000
|
|
#define MICRO_4PA 4000000
|
|
#define OTG_DEFAULT_DEGLITCH_TIME_MS 50
|
|
#define DEFAULT_WD_BARK_TIME 64
|
|
#define DEFAULT_WD_SNARL_TIME_8S 0x07
|
|
#define DEFAULT_FCC_STEP_SIZE_UA 100000
|
|
#define DEFAULT_FCC_STEP_UPDATE_DELAY_MS 1000
|
|
static int smb5_parse_dt_misc(struct smb5 *chip, struct device_node *node)
|
|
{
|
|
int rc = 0, byte_len;
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
of_property_read_u32(node, "qcom,sec-charger-config",
|
|
&chip->dt.sec_charger_config);
|
|
chg->sec_cp_present =
|
|
chip->dt.sec_charger_config ==
|
|
QTI_POWER_SUPPLY_CHARGER_SEC_CP ||
|
|
chip->dt.sec_charger_config ==
|
|
QTI_POWER_SUPPLY_CHARGER_SEC_CP_PL;
|
|
|
|
chg->sec_pl_present =
|
|
chip->dt.sec_charger_config ==
|
|
QTI_POWER_SUPPLY_CHARGER_SEC_PL ||
|
|
chip->dt.sec_charger_config ==
|
|
QTI_POWER_SUPPLY_CHARGER_SEC_CP_PL;
|
|
|
|
chg->step_chg_enabled = of_property_read_bool(node,
|
|
"qcom,step-charging-enable");
|
|
|
|
chg->typec_legacy_use_rp_icl = of_property_read_bool(node,
|
|
"qcom,typec-legacy-rp-icl");
|
|
|
|
chg->sw_jeita_enabled = of_property_read_bool(node,
|
|
"qcom,sw-jeita-enable");
|
|
|
|
chg->jeita_arb_enable = of_property_read_bool(node,
|
|
"qcom,jeita-arb-enable");
|
|
|
|
chg->pd_not_supported = chg->pd_not_supported ||
|
|
of_property_read_bool(node, "qcom,usb-pd-disable");
|
|
|
|
chg->lpd_disabled = chg->lpd_disabled ||
|
|
of_property_read_bool(node, "qcom,lpd-disable");
|
|
|
|
rc = of_property_read_u32(node, "qcom,wd-bark-time-secs",
|
|
&chip->dt.wd_bark_time);
|
|
if (rc < 0 || chip->dt.wd_bark_time < MIN_WD_BARK_TIME)
|
|
chip->dt.wd_bark_time = DEFAULT_WD_BARK_TIME;
|
|
|
|
rc = of_property_read_u32(node, "qcom,wd-snarl-time-config",
|
|
&chip->dt.wd_snarl_time_cfg);
|
|
if (rc < 0)
|
|
chip->dt.wd_snarl_time_cfg = DEFAULT_WD_SNARL_TIME_8S;
|
|
|
|
chip->dt.no_battery = of_property_read_bool(node,
|
|
"qcom,batteryless-platform");
|
|
|
|
if (of_find_property(node, "qcom,thermal-mitigation", &byte_len)) {
|
|
chg->thermal_mitigation = devm_kzalloc(chg->dev, byte_len,
|
|
GFP_KERNEL);
|
|
|
|
if (chg->thermal_mitigation == NULL)
|
|
return -ENOMEM;
|
|
|
|
chg->thermal_levels = byte_len / sizeof(u32);
|
|
rc = of_property_read_u32_array(node,
|
|
"qcom,thermal-mitigation",
|
|
chg->thermal_mitigation,
|
|
chg->thermal_levels);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't read threm limits rc = %d\n", rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
rc = of_property_read_u32(node, "qcom,charger-temp-max",
|
|
&chg->charger_temp_max);
|
|
if (rc < 0)
|
|
chg->charger_temp_max = -EINVAL;
|
|
|
|
rc = of_property_read_u32(node, "qcom,smb-temp-max",
|
|
&chg->smb_temp_max);
|
|
if (rc < 0)
|
|
chg->smb_temp_max = -EINVAL;
|
|
|
|
rc = of_property_read_u32(node, "qcom,float-option",
|
|
&chip->dt.float_option);
|
|
if (!rc && (chip->dt.float_option < 0 || chip->dt.float_option > 4)) {
|
|
pr_err("qcom,float-option is out of range [0, 4]\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
chip->dt.hvdcp_disable = of_property_read_bool(node,
|
|
"qcom,hvdcp-disable");
|
|
chg->hvdcp_disable = chip->dt.hvdcp_disable;
|
|
|
|
chip->dt.hvdcp_autonomous = of_property_read_bool(node,
|
|
"qcom,hvdcp-autonomous-enable");
|
|
|
|
chip->dt.auto_recharge_soc = -EINVAL;
|
|
rc = of_property_read_u32(node, "qcom,auto-recharge-soc",
|
|
&chip->dt.auto_recharge_soc);
|
|
if (!rc && (chip->dt.auto_recharge_soc < 0 ||
|
|
chip->dt.auto_recharge_soc > 100)) {
|
|
pr_err("qcom,auto-recharge-soc is incorrect\n");
|
|
return -EINVAL;
|
|
}
|
|
chg->auto_recharge_soc = chip->dt.auto_recharge_soc;
|
|
|
|
chg->suspend_input_on_debug_batt = of_property_read_bool(node,
|
|
"qcom,suspend-input-on-debug-batt");
|
|
|
|
chg->fake_chg_status_on_debug_batt = of_property_read_bool(node,
|
|
"qcom,fake-chg-status-on-debug-batt");
|
|
|
|
rc = of_property_read_u32(node, "qcom,otg-deglitch-time-ms",
|
|
&chg->otg_delay_ms);
|
|
if (rc < 0)
|
|
chg->otg_delay_ms = OTG_DEFAULT_DEGLITCH_TIME_MS;
|
|
|
|
chg->fcc_stepper_enable = of_property_read_bool(node,
|
|
"qcom,fcc-stepping-enable");
|
|
|
|
if (chg->uusb_moisture_protection_capable)
|
|
chg->uusb_moisture_protection_enabled =
|
|
of_property_read_bool(node,
|
|
"qcom,uusb-moisture-protection-enable");
|
|
|
|
chg->hw_die_temp_mitigation = of_property_read_bool(node,
|
|
"qcom,hw-die-temp-mitigation");
|
|
|
|
chg->hw_connector_mitigation = of_property_read_bool(node,
|
|
"qcom,hw-connector-mitigation");
|
|
|
|
chg->hw_skin_temp_mitigation = of_property_read_bool(node,
|
|
"qcom,hw-skin-temp-mitigation");
|
|
|
|
chg->en_skin_therm_mitigation = of_property_read_bool(node,
|
|
"qcom,en-skin-therm-mitigation");
|
|
|
|
chg->connector_pull_up = -EINVAL;
|
|
of_property_read_u32(node, "qcom,connector-internal-pull-kohm",
|
|
&chg->connector_pull_up);
|
|
|
|
chg->disable_suspend_on_collapse = of_property_read_bool(node,
|
|
"qcom,disable-suspend-on-collapse");
|
|
chg->smb_pull_up = -EINVAL;
|
|
of_property_read_u32(node, "qcom,smb-internal-pull-kohm",
|
|
&chg->smb_pull_up);
|
|
|
|
chip->dt.adc_based_aicl = of_property_read_bool(node,
|
|
"qcom,adc-based-aicl");
|
|
|
|
of_property_read_u32(node, "qcom,fcc-step-delay-ms",
|
|
&chg->chg_param.fcc_step_delay_ms);
|
|
if (chg->chg_param.fcc_step_delay_ms <= 0)
|
|
chg->chg_param.fcc_step_delay_ms =
|
|
DEFAULT_FCC_STEP_UPDATE_DELAY_MS;
|
|
|
|
of_property_read_u32(node, "qcom,fcc-step-size-ua",
|
|
&chg->chg_param.fcc_step_size_ua);
|
|
if (chg->chg_param.fcc_step_size_ua <= 0)
|
|
chg->chg_param.fcc_step_size_ua = DEFAULT_FCC_STEP_SIZE_UA;
|
|
|
|
/*
|
|
* If property is present parallel charging with CP is disabled
|
|
* with HVDCP3 adapter.
|
|
*/
|
|
chg->hvdcp3_standalone_config = of_property_read_bool(node,
|
|
"qcom,hvdcp3-standalone-config");
|
|
|
|
of_property_read_u32(node, "qcom,hvdcp3-max-icl-ua",
|
|
&chg->chg_param.hvdcp3_max_icl_ua);
|
|
if (chg->chg_param.hvdcp3_max_icl_ua <= 0)
|
|
chg->chg_param.hvdcp3_max_icl_ua = MICRO_3PA;
|
|
|
|
of_property_read_u32(node, "qcom,hvdcp2-max-icl-ua",
|
|
&chg->chg_param.hvdcp2_max_icl_ua);
|
|
if (chg->chg_param.hvdcp2_max_icl_ua <= 0)
|
|
chg->chg_param.hvdcp2_max_icl_ua = MICRO_3PA;
|
|
|
|
/* Used only in Adapter CV mode of operation */
|
|
of_property_read_u32(node, "qcom,qc4-max-icl-ua",
|
|
&chg->chg_param.qc4_max_icl_ua);
|
|
if (chg->chg_param.qc4_max_icl_ua <= 0)
|
|
chg->chg_param.qc4_max_icl_ua = MICRO_4PA;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_parse_dt_adc_channels(struct smb_charger *chg)
|
|
{
|
|
int rc = 0;
|
|
|
|
rc = smblib_get_iio_channel(chg, "mid_voltage", &chg->iio.mid_chan);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smblib_get_iio_channel(chg, "usb_in_voltage",
|
|
&chg->iio.usbin_v_chan);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smblib_get_iio_channel(chg, "chg_temp", &chg->iio.temp_chan);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smblib_get_iio_channel(chg, "usb_in_current",
|
|
&chg->iio.usbin_i_chan);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smblib_get_iio_channel(chg, "sbux_res", &chg->iio.sbux_chan);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smblib_get_iio_channel(chg, "vph_voltage", &chg->iio.vph_v_chan);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smblib_get_iio_channel(chg, "die_temp", &chg->iio.die_temp_chan);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smblib_get_iio_channel(chg, "conn_temp",
|
|
&chg->iio.connector_temp_chan);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smblib_get_iio_channel(chg, "skin_temp", &chg->iio.skin_temp_chan);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smblib_get_iio_channel(chg, "smb_temp", &chg->iio.smb_temp_chan);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_parse_dt_currents(struct smb5 *chip, struct device_node *node)
|
|
{
|
|
int rc = 0, tmp;
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
rc = of_property_read_u32(node,
|
|
"qcom,fcc-max-ua", &chip->dt.batt_profile_fcc_ua);
|
|
if (rc < 0)
|
|
chip->dt.batt_profile_fcc_ua = -EINVAL;
|
|
|
|
rc = of_property_read_u32(node,
|
|
"qcom,usb-icl-ua", &chip->dt.usb_icl_ua);
|
|
if (rc < 0)
|
|
chip->dt.usb_icl_ua = -EINVAL;
|
|
chg->dcp_icl_ua = chip->dt.usb_icl_ua;
|
|
|
|
rc = of_property_read_u32(node,
|
|
"qcom,otg-cl-ua", &chg->otg_cl_ua);
|
|
if (rc < 0)
|
|
chg->otg_cl_ua =
|
|
(chip->chg.chg_param.smb_version == PMI632) ?
|
|
MICRO_1PA : MICRO_3PA;
|
|
|
|
rc = of_property_read_u32(node, "qcom,chg-term-src",
|
|
&chip->dt.term_current_src);
|
|
if (rc < 0)
|
|
chip->dt.term_current_src = ITERM_SRC_UNSPECIFIED;
|
|
|
|
if (chip->dt.term_current_src == ITERM_SRC_ADC)
|
|
rc = of_property_read_u32(node, "qcom,chg-term-base-current-ma",
|
|
&chip->dt.term_current_thresh_lo_ma);
|
|
|
|
rc = of_property_read_u32(node, "qcom,chg-term-current-ma",
|
|
&chip->dt.term_current_thresh_hi_ma);
|
|
|
|
chg->wls_icl_ua = DCIN_ICL_MAX_UA;
|
|
rc = of_property_read_u32(node, "qcom,wls-current-max-ua",
|
|
&tmp);
|
|
if (!rc && tmp < DCIN_ICL_MAX_UA)
|
|
chg->wls_icl_ua = tmp;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_parse_dt_voltages(struct smb5 *chip, struct device_node *node)
|
|
{
|
|
int rc = 0;
|
|
|
|
rc = of_property_read_u32(node,
|
|
"qcom,fv-max-uv", &chip->dt.batt_profile_fv_uv);
|
|
if (rc < 0)
|
|
chip->dt.batt_profile_fv_uv = -EINVAL;
|
|
|
|
rc = of_property_read_u32(node, "qcom,chg-inhibit-threshold-mv",
|
|
&chip->dt.chg_inhibit_thr_mv);
|
|
if (!rc && (chip->dt.chg_inhibit_thr_mv < 0 ||
|
|
chip->dt.chg_inhibit_thr_mv > 300)) {
|
|
pr_err("qcom,chg-inhibit-threshold-mv is incorrect\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
chip->dt.auto_recharge_vbat_mv = -EINVAL;
|
|
rc = of_property_read_u32(node, "qcom,auto-recharge-vbat-mv",
|
|
&chip->dt.auto_recharge_vbat_mv);
|
|
if (!rc && (chip->dt.auto_recharge_vbat_mv < 0)) {
|
|
pr_err("qcom,auto-recharge-vbat-mv is incorrect\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_parse_sdam(struct smb5 *chip, struct device_node *node)
|
|
{
|
|
struct device_node *child;
|
|
struct smb_charger *chg = &chip->chg;
|
|
struct property *prop;
|
|
const char *name;
|
|
int rc;
|
|
u32 base;
|
|
u8 type;
|
|
|
|
for_each_available_child_of_node(node, child) {
|
|
of_property_for_each_string(child, "reg", prop, name) {
|
|
rc = of_property_read_u32(child, "reg", &base);
|
|
if (rc < 0) {
|
|
pr_err("Failed to read base rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = smblib_read(chg, base + PERPH_TYPE_OFFSET, &type);
|
|
if (rc < 0) {
|
|
pr_err("Failed to read type rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
switch (type) {
|
|
case SDAM_TYPE:
|
|
chg->sdam_base = base;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!chg->sdam_base)
|
|
pr_debug("SDAM node not defined\n");
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_parse_dt(struct smb5 *chip)
|
|
{
|
|
struct smb_charger *chg = &chip->chg;
|
|
struct device_node *node = chg->dev->of_node;
|
|
int rc = 0;
|
|
|
|
if (!node) {
|
|
pr_err("device tree node missing\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = smb5_parse_dt_voltages(chip, node);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smb5_parse_dt_currents(chip, node);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smb5_parse_dt_adc_channels(chg);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smb5_parse_dt_misc(chip, node);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smb5_parse_sdam(chip, node);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int smb5_set_prop_comp_clamp_level(struct smb_charger *chg,
|
|
int val)
|
|
{
|
|
int rc = 0, i;
|
|
struct clamp_config clamp_config;
|
|
enum comp_clamp_levels level;
|
|
|
|
level = val;
|
|
if (level >= MAX_CLAMP_LEVEL) {
|
|
pr_err("Invalid comp clamp level=%d\n", val);
|
|
return -EINVAL;
|
|
}
|
|
|
|
for (i = 0; i < ARRAY_SIZE(clamp_config.reg); i++) {
|
|
rc = smblib_write(chg, clamp_levels[level].reg[i],
|
|
clamp_levels[level].val[i]);
|
|
if (rc < 0)
|
|
dev_err(chg->dev,
|
|
"Failed to configure comp clamp settings for reg=0x%04x rc=%d\n",
|
|
clamp_levels[level].reg[i], rc);
|
|
}
|
|
|
|
chg->comp_clamp_level = val;
|
|
|
|
return rc;
|
|
}
|
|
|
|
/************************
|
|
* USB PSY REGISTRATION *
|
|
************************/
|
|
static enum power_supply_property smb5_usb_props[] = {
|
|
POWER_SUPPLY_PROP_PRESENT,
|
|
POWER_SUPPLY_PROP_ONLINE,
|
|
POWER_SUPPLY_PROP_VOLTAGE_NOW,
|
|
POWER_SUPPLY_PROP_CURRENT_NOW,
|
|
POWER_SUPPLY_PROP_CURRENT_MAX,
|
|
POWER_SUPPLY_PROP_TYPE,
|
|
POWER_SUPPLY_PROP_VOLTAGE_MAX,
|
|
POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN,
|
|
POWER_SUPPLY_PROP_SCOPE,
|
|
POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT,
|
|
POWER_SUPPLY_PROP_POWER_NOW,
|
|
};
|
|
|
|
static int smb5_usb_get_prop(struct power_supply *psy,
|
|
enum power_supply_property psp,
|
|
union power_supply_propval *val)
|
|
{
|
|
struct smb5 *chip = power_supply_get_drvdata(psy);
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc = 0;
|
|
|
|
val->intval = 0;
|
|
|
|
switch (psp) {
|
|
case POWER_SUPPLY_PROP_PRESENT:
|
|
rc = smblib_get_prop_usb_present(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_ONLINE:
|
|
rc = smblib_get_usb_online(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN:
|
|
rc = smblib_get_prop_usb_voltage_max_design(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_VOLTAGE_MAX:
|
|
rc = smblib_get_prop_usb_voltage_max(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
|
rc = smblib_get_prop_usb_voltage_now(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CURRENT_NOW:
|
|
rc = smblib_get_prop_usb_current_now(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CURRENT_MAX:
|
|
rc = smblib_get_prop_input_current_max(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_TYPE:
|
|
val->intval = POWER_SUPPLY_TYPE_USB_PD;
|
|
break;
|
|
case POWER_SUPPLY_PROP_SCOPE:
|
|
rc = smblib_get_prop_scope(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
|
|
/* USB uses this to set SDP current */
|
|
val->intval = get_client_vote(chg->usb_icl_votable,
|
|
USB_PSY_VOTER);
|
|
break;
|
|
case POWER_SUPPLY_PROP_POWER_NOW:
|
|
/* Show power rating for QC3+ adapter. */
|
|
if (chg->real_charger_type == QTI_POWER_SUPPLY_TYPE_USB_HVDCP_3P5)
|
|
val->intval = chg->qc3p5_detected_mw;
|
|
else
|
|
rc = -ENODATA;
|
|
break;
|
|
default:
|
|
pr_err("get prop %d is not supported in usb\n", psp);
|
|
rc = -EINVAL;
|
|
break;
|
|
}
|
|
|
|
if (rc < 0) {
|
|
pr_debug("Couldn't get prop %d rc = %d\n", psp, rc);
|
|
return -ENODATA;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_usb_set_prop(struct power_supply *psy,
|
|
enum power_supply_property psp,
|
|
const union power_supply_propval *val)
|
|
{
|
|
struct smb5 *chip = power_supply_get_drvdata(psy);
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc = 0;
|
|
|
|
switch (psp) {
|
|
case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
|
|
rc = smblib_set_prop_sdp_current_max(chg, val->intval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_POWER_NOW:
|
|
chg->qc3p5_detected_mw = val->intval;
|
|
break;
|
|
default:
|
|
pr_err("Set prop %d is not supported in usb psy\n",
|
|
psp);
|
|
rc = -EINVAL;
|
|
break;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_usb_prop_is_writeable(struct power_supply *psy,
|
|
enum power_supply_property psp)
|
|
{
|
|
switch (psp) {
|
|
case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
|
|
case POWER_SUPPLY_PROP_POWER_NOW:
|
|
return 1;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct power_supply_desc usb_psy_desc = {
|
|
.name = "usb",
|
|
.type = POWER_SUPPLY_TYPE_USB_PD,
|
|
.properties = smb5_usb_props,
|
|
.num_properties = ARRAY_SIZE(smb5_usb_props),
|
|
.get_property = smb5_usb_get_prop,
|
|
.set_property = smb5_usb_set_prop,
|
|
.property_is_writeable = smb5_usb_prop_is_writeable,
|
|
};
|
|
|
|
static int smb5_init_usb_psy(struct smb5 *chip)
|
|
{
|
|
struct power_supply_config usb_cfg = {};
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
usb_cfg.drv_data = chip;
|
|
usb_cfg.of_node = chg->dev->of_node;
|
|
chg->usb_psy = devm_power_supply_register(chg->dev,
|
|
&usb_psy_desc,
|
|
&usb_cfg);
|
|
if (IS_ERR(chg->usb_psy)) {
|
|
pr_err("Couldn't register USB power supply\n");
|
|
return PTR_ERR(chg->usb_psy);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/********************************
|
|
* USB PC_PORT PSY REGISTRATION *
|
|
********************************/
|
|
static enum power_supply_property smb5_usb_port_props[] = {
|
|
POWER_SUPPLY_PROP_TYPE,
|
|
POWER_SUPPLY_PROP_ONLINE,
|
|
POWER_SUPPLY_PROP_VOLTAGE_MAX,
|
|
POWER_SUPPLY_PROP_CURRENT_MAX,
|
|
};
|
|
|
|
static int smb5_usb_port_get_prop(struct power_supply *psy,
|
|
enum power_supply_property psp,
|
|
union power_supply_propval *val)
|
|
{
|
|
struct smb5 *chip = power_supply_get_drvdata(psy);
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc = 0;
|
|
|
|
switch (psp) {
|
|
case POWER_SUPPLY_PROP_TYPE:
|
|
val->intval = POWER_SUPPLY_TYPE_USB;
|
|
break;
|
|
case POWER_SUPPLY_PROP_ONLINE:
|
|
rc = smblib_get_prop_usb_online(chg, val);
|
|
if (!val->intval)
|
|
break;
|
|
|
|
if (((chg->typec_mode ==
|
|
QTI_POWER_SUPPLY_TYPEC_SOURCE_DEFAULT) ||
|
|
(chg->connector_type ==
|
|
QTI_POWER_SUPPLY_CONNECTOR_MICRO_USB))
|
|
&& (chg->real_charger_type == POWER_SUPPLY_TYPE_USB))
|
|
val->intval = 1;
|
|
else
|
|
val->intval = 0;
|
|
break;
|
|
case POWER_SUPPLY_PROP_VOLTAGE_MAX:
|
|
val->intval = 5000000;
|
|
break;
|
|
case POWER_SUPPLY_PROP_CURRENT_MAX:
|
|
rc = smblib_get_prop_input_current_settled(chg, val);
|
|
break;
|
|
default:
|
|
pr_err_ratelimited("Get prop %d is not supported in pc_port\n",
|
|
psp);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (rc < 0) {
|
|
pr_debug("Couldn't get prop %d rc = %d\n", psp, rc);
|
|
return -ENODATA;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_usb_port_set_prop(struct power_supply *psy,
|
|
enum power_supply_property psp,
|
|
const union power_supply_propval *val)
|
|
{
|
|
int rc = 0;
|
|
|
|
switch (psp) {
|
|
default:
|
|
pr_err_ratelimited("Set prop %d is not supported in pc_port\n",
|
|
psp);
|
|
rc = -EINVAL;
|
|
break;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static const struct power_supply_desc usb_port_psy_desc = {
|
|
.name = "pc_port",
|
|
.type = POWER_SUPPLY_TYPE_USB,
|
|
.properties = smb5_usb_port_props,
|
|
.num_properties = ARRAY_SIZE(smb5_usb_port_props),
|
|
.get_property = smb5_usb_port_get_prop,
|
|
.set_property = smb5_usb_port_set_prop,
|
|
};
|
|
|
|
static int smb5_init_usb_port_psy(struct smb5 *chip)
|
|
{
|
|
struct power_supply_config usb_port_cfg = {};
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
usb_port_cfg.drv_data = chip;
|
|
usb_port_cfg.of_node = chg->dev->of_node;
|
|
chg->usb_port_psy = devm_power_supply_register(chg->dev,
|
|
&usb_port_psy_desc,
|
|
&usb_port_cfg);
|
|
if (IS_ERR(chg->usb_port_psy)) {
|
|
pr_err("Couldn't register USB pc_port power supply\n");
|
|
return PTR_ERR(chg->usb_port_psy);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*************************
|
|
* DC PSY REGISTRATION *
|
|
*************************/
|
|
|
|
static enum power_supply_property smb5_dc_props[] = {
|
|
POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT,
|
|
POWER_SUPPLY_PROP_PRESENT,
|
|
POWER_SUPPLY_PROP_ONLINE,
|
|
POWER_SUPPLY_PROP_VOLTAGE_NOW,
|
|
POWER_SUPPLY_PROP_CURRENT_MAX,
|
|
POWER_SUPPLY_PROP_VOLTAGE_MAX,
|
|
};
|
|
|
|
static int smb5_dc_get_prop(struct power_supply *psy,
|
|
enum power_supply_property psp,
|
|
union power_supply_propval *val)
|
|
{
|
|
struct smb5 *chip = power_supply_get_drvdata(psy);
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc = 0;
|
|
|
|
switch (psp) {
|
|
case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
|
|
/* For DC, INPUT_CURRENT_LIMIT equates to INPUT_SUSPEND */
|
|
val->intval = get_effective_result(chg->dc_suspend_votable);
|
|
break;
|
|
case POWER_SUPPLY_PROP_PRESENT:
|
|
rc = smblib_get_prop_dc_present(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_ONLINE:
|
|
rc = smblib_get_prop_dc_online(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
|
rc = smblib_get_prop_dc_voltage_now(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CURRENT_MAX:
|
|
rc = smblib_get_prop_dc_current_max(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_VOLTAGE_MAX:
|
|
rc = smblib_get_prop_dc_voltage_max(chg, val);
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
if (rc < 0) {
|
|
pr_debug("Couldn't get prop %d rc = %d\n", psp, rc);
|
|
return -ENODATA;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_dc_set_prop(struct power_supply *psy,
|
|
enum power_supply_property psp,
|
|
const union power_supply_propval *val)
|
|
{
|
|
struct smb5 *chip = power_supply_get_drvdata(psy);
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc = 0;
|
|
|
|
switch (psp) {
|
|
case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
|
|
rc = vote(chg->dc_suspend_votable, WBC_VOTER,
|
|
(bool)val->intval, 0);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CURRENT_MAX:
|
|
rc = smblib_set_prop_dc_current_max(chg, val);
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_dc_prop_is_writeable(struct power_supply *psy,
|
|
enum power_supply_property psp)
|
|
{
|
|
switch (psp) {
|
|
case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
|
|
case POWER_SUPPLY_PROP_CURRENT_MAX:
|
|
return 1;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct power_supply_desc dc_psy_desc = {
|
|
.name = "dc",
|
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
|
.properties = smb5_dc_props,
|
|
.num_properties = ARRAY_SIZE(smb5_dc_props),
|
|
.get_property = smb5_dc_get_prop,
|
|
.set_property = smb5_dc_set_prop,
|
|
.property_is_writeable = smb5_dc_prop_is_writeable,
|
|
};
|
|
|
|
static int smb5_init_dc_psy(struct smb5 *chip)
|
|
{
|
|
struct power_supply_config dc_cfg = {};
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
dc_cfg.drv_data = chip;
|
|
dc_cfg.of_node = chg->dev->of_node;
|
|
chg->dc_psy = devm_power_supply_register(chg->dev,
|
|
&dc_psy_desc,
|
|
&dc_cfg);
|
|
if (IS_ERR(chg->dc_psy)) {
|
|
pr_err("Couldn't register USB power supply\n");
|
|
return PTR_ERR(chg->dc_psy);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*************************
|
|
* BATT PSY REGISTRATION *
|
|
*************************/
|
|
static enum power_supply_property smb5_batt_props[] = {
|
|
POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT,
|
|
POWER_SUPPLY_PROP_STATUS,
|
|
POWER_SUPPLY_PROP_HEALTH,
|
|
POWER_SUPPLY_PROP_PRESENT,
|
|
POWER_SUPPLY_PROP_CHARGE_TYPE,
|
|
POWER_SUPPLY_PROP_CAPACITY,
|
|
POWER_SUPPLY_PROP_VOLTAGE_NOW,
|
|
POWER_SUPPLY_PROP_VOLTAGE_MAX,
|
|
POWER_SUPPLY_PROP_CURRENT_NOW,
|
|
POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX,
|
|
POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT,
|
|
POWER_SUPPLY_PROP_CHARGE_TERM_CURRENT,
|
|
POWER_SUPPLY_PROP_TEMP,
|
|
POWER_SUPPLY_PROP_TECHNOLOGY,
|
|
POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT_MAX,
|
|
POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT,
|
|
POWER_SUPPLY_PROP_CHARGE_COUNTER,
|
|
POWER_SUPPLY_PROP_CYCLE_COUNT,
|
|
POWER_SUPPLY_PROP_CHARGE_FULL,
|
|
POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN,
|
|
POWER_SUPPLY_PROP_TIME_TO_FULL_NOW,
|
|
};
|
|
|
|
#define DEBUG_ACCESSORY_TEMP_DECIDEGC 250
|
|
static int smb5_batt_get_prop(struct power_supply *psy,
|
|
enum power_supply_property psp,
|
|
union power_supply_propval *pval)
|
|
{
|
|
struct smb_charger *chg = power_supply_get_drvdata(psy);
|
|
int rc = 0;
|
|
|
|
switch (psp) {
|
|
case POWER_SUPPLY_PROP_STATUS:
|
|
rc = smblib_get_prop_batt_status(chg, pval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_HEALTH:
|
|
rc = smblib_get_prop_batt_health(chg, pval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_PRESENT:
|
|
rc = smblib_get_prop_batt_present(chg, pval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
|
|
rc = smblib_get_prop_input_suspend(chg, pval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CHARGE_TYPE:
|
|
rc = smblib_get_prop_batt_charge_type(chg, pval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CAPACITY:
|
|
rc = smblib_get_prop_batt_capacity(chg, pval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT:
|
|
rc = smblib_get_prop_system_temp_level(chg, pval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT_MAX:
|
|
rc = smblib_get_prop_system_temp_level_max(chg, pval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
|
rc = smblib_get_prop_from_bms(chg,
|
|
SMB5_QG_VOLTAGE_NOW, &pval->intval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_VOLTAGE_MAX:
|
|
pval->intval = get_client_vote(chg->fv_votable,
|
|
QNOVO_VOTER);
|
|
if (pval->intval < 0)
|
|
pval->intval = get_client_vote(chg->fv_votable,
|
|
BATT_PROFILE_VOTER);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CURRENT_NOW:
|
|
rc = smblib_get_batt_current_now(chg, pval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX:
|
|
pval->intval = get_client_vote(chg->fcc_votable,
|
|
BATT_PROFILE_VOTER);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT:
|
|
pval->intval = get_effective_result(chg->fcc_votable);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CHARGE_TERM_CURRENT:
|
|
rc = smblib_get_prop_batt_iterm(chg, pval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_TEMP:
|
|
if (chg->typec_mode ==
|
|
QTI_POWER_SUPPLY_TYPEC_SINK_DEBUG_ACCESSORY)
|
|
pval->intval = DEBUG_ACCESSORY_TEMP_DECIDEGC;
|
|
else
|
|
rc = smblib_get_prop_from_bms(chg,
|
|
SMB5_QG_TEMP, &pval->intval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
|
pval->intval = POWER_SUPPLY_TECHNOLOGY_LION;
|
|
break;
|
|
case POWER_SUPPLY_PROP_CHARGE_COUNTER:
|
|
rc = smblib_get_prop_from_bms(chg,
|
|
SMB5_QG_CHARGE_COUNTER, &pval->intval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CYCLE_COUNT:
|
|
rc = smblib_get_prop_from_bms(chg,
|
|
SMB5_QG_CYCLE_COUNT, &pval->intval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CHARGE_FULL:
|
|
rc = smblib_get_prop_from_bms(chg,
|
|
SMB5_QG_CHARGE_FULL, &pval->intval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN:
|
|
rc = smblib_get_prop_from_bms(chg,
|
|
SMB5_QG_CHARGE_FULL_DESIGN, &pval->intval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_TIME_TO_FULL_NOW:
|
|
rc = smblib_get_prop_from_bms(chg,
|
|
SMB5_QG_TIME_TO_FULL_NOW, &pval->intval);
|
|
break;
|
|
default:
|
|
pr_err("batt power supply prop %d not supported\n", psp);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (rc < 0) {
|
|
pr_debug("Couldn't get prop %d rc = %d\n", psp, rc);
|
|
return -ENODATA;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_batt_set_prop(struct power_supply *psy,
|
|
enum power_supply_property prop,
|
|
const union power_supply_propval *val)
|
|
{
|
|
int rc = 0;
|
|
struct smb_charger *chg = power_supply_get_drvdata(psy);
|
|
|
|
switch (prop) {
|
|
case POWER_SUPPLY_PROP_STATUS:
|
|
rc = smblib_set_prop_batt_status(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
|
|
rc = smblib_set_prop_input_suspend(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT:
|
|
rc = smblib_set_prop_system_temp_level(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CAPACITY:
|
|
rc = smblib_set_prop_batt_capacity(chg, val);
|
|
break;
|
|
case POWER_SUPPLY_PROP_VOLTAGE_MAX:
|
|
chg->batt_profile_fv_uv = val->intval;
|
|
vote(chg->fv_votable, BATT_PROFILE_VOTER, true, val->intval);
|
|
break;
|
|
case POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX:
|
|
chg->batt_profile_fcc_ua = val->intval;
|
|
vote(chg->fcc_votable, BATT_PROFILE_VOTER, true, val->intval);
|
|
break;
|
|
default:
|
|
rc = -EINVAL;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_batt_prop_is_writeable(struct power_supply *psy,
|
|
enum power_supply_property psp)
|
|
{
|
|
switch (psp) {
|
|
case POWER_SUPPLY_PROP_STATUS:
|
|
case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
|
|
case POWER_SUPPLY_PROP_CAPACITY:
|
|
return 1;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct power_supply_desc batt_psy_desc = {
|
|
.name = "battery",
|
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
|
.properties = smb5_batt_props,
|
|
.num_properties = ARRAY_SIZE(smb5_batt_props),
|
|
.get_property = smb5_batt_get_prop,
|
|
.set_property = smb5_batt_set_prop,
|
|
.property_is_writeable = smb5_batt_prop_is_writeable,
|
|
};
|
|
|
|
static int smb5_init_batt_psy(struct smb5 *chip)
|
|
{
|
|
struct power_supply_config batt_cfg = {};
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc = 0;
|
|
|
|
batt_cfg.drv_data = chg;
|
|
batt_cfg.of_node = chg->dev->of_node;
|
|
chg->batt_psy = devm_power_supply_register(chg->dev,
|
|
&batt_psy_desc,
|
|
&batt_cfg);
|
|
if (IS_ERR(chg->batt_psy)) {
|
|
pr_err("Couldn't register battery power supply\n");
|
|
return PTR_ERR(chg->batt_psy);
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
/******************************
|
|
* VBUS REGULATOR REGISTRATION *
|
|
******************************/
|
|
|
|
static const struct regulator_ops smb5_vbus_reg_ops = {
|
|
.enable = smblib_vbus_regulator_enable,
|
|
.disable = smblib_vbus_regulator_disable,
|
|
.is_enabled = smblib_vbus_regulator_is_enabled,
|
|
};
|
|
|
|
static int smb5_init_vbus_regulator(struct smb5 *chip)
|
|
{
|
|
struct smb_charger *chg = &chip->chg;
|
|
struct regulator_config cfg = {};
|
|
int rc = 0;
|
|
|
|
chg->vbus_vreg = devm_kzalloc(chg->dev, sizeof(*chg->vbus_vreg),
|
|
GFP_KERNEL);
|
|
if (!chg->vbus_vreg)
|
|
return -ENOMEM;
|
|
|
|
cfg.dev = chg->dev;
|
|
cfg.driver_data = chip;
|
|
|
|
chg->vbus_vreg->rdesc.owner = THIS_MODULE;
|
|
chg->vbus_vreg->rdesc.type = REGULATOR_VOLTAGE;
|
|
chg->vbus_vreg->rdesc.ops = &smb5_vbus_reg_ops;
|
|
chg->vbus_vreg->rdesc.of_match = "qcom,smb5-vbus";
|
|
chg->vbus_vreg->rdesc.name = "qcom,smb5-vbus";
|
|
|
|
chg->vbus_vreg->rdev = devm_regulator_register(chg->dev,
|
|
&chg->vbus_vreg->rdesc, &cfg);
|
|
if (IS_ERR(chg->vbus_vreg->rdev)) {
|
|
rc = PTR_ERR(chg->vbus_vreg->rdev);
|
|
chg->vbus_vreg->rdev = NULL;
|
|
if (rc != -EPROBE_DEFER)
|
|
pr_err("Couldn't register VBUS regulator rc=%d\n", rc);
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
/******************************
|
|
* VCONN REGULATOR REGISTRATION *
|
|
******************************/
|
|
|
|
static const struct regulator_ops smb5_vconn_reg_ops = {
|
|
.enable = smblib_vconn_regulator_enable,
|
|
.disable = smblib_vconn_regulator_disable,
|
|
.is_enabled = smblib_vconn_regulator_is_enabled,
|
|
};
|
|
|
|
static int smb5_init_vconn_regulator(struct smb5 *chip)
|
|
{
|
|
struct smb_charger *chg = &chip->chg;
|
|
struct regulator_config cfg = {};
|
|
int rc = 0;
|
|
|
|
if (chg->connector_type == QTI_POWER_SUPPLY_CONNECTOR_MICRO_USB)
|
|
return 0;
|
|
|
|
chg->vconn_vreg = devm_kzalloc(chg->dev, sizeof(*chg->vconn_vreg),
|
|
GFP_KERNEL);
|
|
if (!chg->vconn_vreg)
|
|
return -ENOMEM;
|
|
|
|
cfg.dev = chg->dev;
|
|
cfg.driver_data = chip;
|
|
|
|
chg->vconn_vreg->rdesc.owner = THIS_MODULE;
|
|
chg->vconn_vreg->rdesc.type = REGULATOR_VOLTAGE;
|
|
chg->vconn_vreg->rdesc.ops = &smb5_vconn_reg_ops;
|
|
chg->vconn_vreg->rdesc.of_match = "qcom,smb5-vconn";
|
|
chg->vconn_vreg->rdesc.name = "qcom,smb5-vconn";
|
|
|
|
chg->vconn_vreg->rdev = devm_regulator_register(chg->dev,
|
|
&chg->vconn_vreg->rdesc, &cfg);
|
|
if (IS_ERR(chg->vconn_vreg->rdev)) {
|
|
rc = PTR_ERR(chg->vconn_vreg->rdev);
|
|
chg->vconn_vreg->rdev = NULL;
|
|
if (rc != -EPROBE_DEFER)
|
|
pr_err("Couldn't register VCONN regulator rc=%d\n", rc);
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
/***************************
|
|
* HARDWARE INITIALIZATION *
|
|
***************************/
|
|
static int smb5_configure_typec(struct smb_charger *chg)
|
|
{
|
|
int rc, val;
|
|
u8 value = 0;
|
|
|
|
rc = smblib_read(chg, LEGACY_CABLE_STATUS_REG, &value);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't read Legacy status rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* Across reboot, standard typeC cables get detected as legacy
|
|
* cables due to VBUS attachment prior to CC attach/detach.
|
|
* Reset the legacy detection logic by enabling/disabling the typeC mode.
|
|
*/
|
|
|
|
if (value & TYPEC_LEGACY_CABLE_STATUS_BIT) {
|
|
val = QTI_POWER_SUPPLY_TYPEC_PR_NONE;
|
|
rc = smblib_set_prop_typec_power_role(chg, val);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't disable TYPEC rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/* delay before enabling typeC */
|
|
msleep(50);
|
|
|
|
val = QTI_POWER_SUPPLY_TYPEC_PR_DUAL;
|
|
rc = smblib_set_prop_typec_power_role(chg, val);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't enable TYPEC rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/*delay after enabling typeC*/
|
|
msleep(100);
|
|
}
|
|
|
|
smblib_apsd_enable(chg, true);
|
|
|
|
rc = smblib_read(chg, TYPE_C_SNK_STATUS_REG, &value);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "failed to read TYPE_C_SNK_STATUS_REG rc=%d\n",
|
|
rc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
if (!(value & SNK_DAM_MASK)) {
|
|
rc = smblib_masked_write(chg, TYPE_C_CFG_REG,
|
|
BC1P2_START_ON_CC_BIT, 0);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "failed to write TYPE_C_CFG_REG rc=%d\n",
|
|
rc);
|
|
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
/* Use simple write to clear interrupts */
|
|
rc = smblib_write(chg, TYPE_C_INTERRUPT_EN_CFG_1_REG, 0);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure Type-C interrupts rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
value = chg->lpd_disabled ? 0 : TYPEC_WATER_DETECTION_INT_EN_BIT;
|
|
/* Use simple write to enable only required interrupts */
|
|
rc = smblib_write(chg, TYPE_C_INTERRUPT_EN_CFG_2_REG,
|
|
TYPEC_SRC_BATT_HPWR_INT_EN_BIT | value);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure Type-C interrupts rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/* enable try.snk and clear force sink for DRP mode */
|
|
rc = smblib_masked_write(chg, TYPE_C_MODE_CFG_REG,
|
|
EN_TRY_SNK_BIT | EN_SNK_ONLY_BIT,
|
|
EN_TRY_SNK_BIT);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure TYPE_C_MODE_CFG_REG rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
chg->typec_try_mode |= EN_TRY_SNK_BIT;
|
|
|
|
/* For PD capable targets configure VCONN for software control */
|
|
if (!chg->pd_not_supported) {
|
|
rc = smblib_masked_write(chg, TYPE_C_VCONN_CONTROL_REG,
|
|
VCONN_EN_SRC_BIT | VCONN_EN_VALUE_BIT,
|
|
VCONN_EN_SRC_BIT);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure VCONN for SW control rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
if (chg->chg_param.smb_version != PMI632) {
|
|
/*
|
|
* Enable detection of unoriented debug
|
|
* accessory in source mode.
|
|
*/
|
|
rc = smblib_masked_write(chg, DEBUG_ACCESS_SRC_CFG_REG,
|
|
EN_UNORIENTED_DEBUG_ACCESS_SRC_BIT,
|
|
EN_UNORIENTED_DEBUG_ACCESS_SRC_BIT);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure TYPE_C_DEBUG_ACCESS_SRC_CFG_REG rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = smblib_masked_write(chg, USBIN_LOAD_CFG_REG,
|
|
USBIN_IN_COLLAPSE_GF_SEL_MASK |
|
|
USBIN_AICL_STEP_TIMING_SEL_MASK,
|
|
0);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't set USBIN_LOAD_CFG_REG rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
/* Set CC threshold to 1.6 V in source mode */
|
|
rc = smblib_masked_write(chg, TYPE_C_EXIT_STATE_CFG_REG,
|
|
SEL_SRC_UPPER_REF_BIT, SEL_SRC_UPPER_REF_BIT);
|
|
if (rc < 0)
|
|
dev_err(chg->dev,
|
|
"Couldn't configure CC threshold voltage rc=%d\n", rc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_configure_micro_usb(struct smb_charger *chg)
|
|
{
|
|
int rc;
|
|
|
|
/* For micro USB connector, use extcon by default */
|
|
chg->use_extcon = true;
|
|
chg->pd_not_supported = true;
|
|
|
|
rc = smblib_masked_write(chg, TYPE_C_INTERRUPT_EN_CFG_2_REG,
|
|
MICRO_USB_STATE_CHANGE_INT_EN_BIT,
|
|
MICRO_USB_STATE_CHANGE_INT_EN_BIT);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure Type-C interrupts rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
if (chg->uusb_moisture_protection_enabled) {
|
|
/* Enable moisture detection interrupt */
|
|
rc = smblib_masked_write(chg, TYPE_C_INTERRUPT_EN_CFG_2_REG,
|
|
TYPEC_WATER_DETECTION_INT_EN_BIT,
|
|
TYPEC_WATER_DETECTION_INT_EN_BIT);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't enable moisture detection interrupt rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
/* Enable uUSB factory mode */
|
|
rc = smblib_masked_write(chg, TYPEC_U_USB_CFG_REG,
|
|
EN_MICRO_USB_FACTORY_MODE_BIT,
|
|
EN_MICRO_USB_FACTORY_MODE_BIT);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't enable uUSB factory mode c=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
/* Disable periodic monitoring of CC_ID pin */
|
|
rc = smblib_write(chg,
|
|
((chg->chg_param.smb_version == PMI632) ?
|
|
PMI632_TYPEC_U_USB_WATER_PROTECTION_CFG_REG :
|
|
TYPEC_U_USB_WATER_PROTECTION_CFG_REG), 0);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't disable periodic monitoring of CC_ID rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
/* Enable HVDCP detection and authentication */
|
|
if (!chg->hvdcp_disable)
|
|
smblib_hvdcp_detect_enable(chg, true);
|
|
|
|
return rc;
|
|
}
|
|
|
|
#define RAW_ITERM(iterm_ma, max_range) \
|
|
div_s64((int64_t)iterm_ma * ADC_CHG_ITERM_MASK, max_range)
|
|
static int smb5_configure_iterm_thresholds_adc(struct smb5 *chip)
|
|
{
|
|
u8 *buf;
|
|
int rc = 0;
|
|
s16 raw_hi_thresh, raw_lo_thresh, max_limit_ma;
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
if (chip->chg.chg_param.smb_version == PMI632)
|
|
max_limit_ma = ITERM_LIMITS_PMI632_MA;
|
|
else
|
|
max_limit_ma = ITERM_LIMITS_PM8150B_MA;
|
|
|
|
if (chip->dt.term_current_thresh_hi_ma < (-1 * max_limit_ma)
|
|
|| chip->dt.term_current_thresh_hi_ma > max_limit_ma
|
|
|| chip->dt.term_current_thresh_lo_ma < (-1 * max_limit_ma)
|
|
|| chip->dt.term_current_thresh_lo_ma > max_limit_ma) {
|
|
dev_err(chg->dev, "ITERM threshold out of range rc=%d\n", rc);
|
|
return -EINVAL;
|
|
}
|
|
|
|
/*
|
|
* Conversion:
|
|
* raw (A) = (term_current * ADC_CHG_ITERM_MASK) / max_limit_ma
|
|
* Note: raw needs to be converted to big-endian format.
|
|
*/
|
|
|
|
if (chip->dt.term_current_thresh_hi_ma) {
|
|
raw_hi_thresh = RAW_ITERM(chip->dt.term_current_thresh_hi_ma,
|
|
max_limit_ma);
|
|
raw_hi_thresh = sign_extend32(raw_hi_thresh, 15);
|
|
buf = (u8 *)&raw_hi_thresh;
|
|
raw_hi_thresh = buf[1] | (buf[0] << 8);
|
|
|
|
rc = smblib_batch_write(chg, CHGR_ADC_ITERM_UP_THD_MSB_REG,
|
|
(u8 *)&raw_hi_thresh, 2);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure ITERM threshold HIGH rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
if (chip->dt.term_current_thresh_lo_ma) {
|
|
raw_lo_thresh = RAW_ITERM(chip->dt.term_current_thresh_lo_ma,
|
|
max_limit_ma);
|
|
raw_lo_thresh = sign_extend32(raw_lo_thresh, 15);
|
|
buf = (u8 *)&raw_lo_thresh;
|
|
raw_lo_thresh = buf[1] | (buf[0] << 8);
|
|
|
|
rc = smblib_batch_write(chg, CHGR_ADC_ITERM_LO_THD_MSB_REG,
|
|
(u8 *)&raw_lo_thresh, 2);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure ITERM threshold LOW rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_configure_iterm_thresholds(struct smb5 *chip)
|
|
{
|
|
int rc = 0;
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
switch (chip->dt.term_current_src) {
|
|
case ITERM_SRC_ADC:
|
|
if (chip->chg.chg_param.smb_version == PM8150B) {
|
|
rc = smblib_masked_write(chg, CHGR_ADC_TERM_CFG_REG,
|
|
TERM_BASED_ON_SYNC_CONV_OR_SAMPLE_CNT,
|
|
TERM_BASED_ON_SAMPLE_CNT);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure ADC_ITERM_CFG rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
rc = smb5_configure_iterm_thresholds_adc(chip);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_configure_mitigation(struct smb_charger *chg)
|
|
{
|
|
int rc;
|
|
u8 chan = 0, src_cfg = 0;
|
|
|
|
if (!chg->hw_die_temp_mitigation && !chg->hw_connector_mitigation &&
|
|
!chg->hw_skin_temp_mitigation) {
|
|
src_cfg = THERMREG_SW_ICL_ADJUST_BIT;
|
|
} else {
|
|
if (chg->hw_die_temp_mitigation) {
|
|
chan = DIE_TEMP_CHANNEL_EN_BIT;
|
|
src_cfg = THERMREG_DIE_ADC_SRC_EN_BIT
|
|
| THERMREG_DIE_CMP_SRC_EN_BIT;
|
|
}
|
|
|
|
if (chg->hw_connector_mitigation) {
|
|
chan |= CONN_THM_CHANNEL_EN_BIT;
|
|
src_cfg |= THERMREG_CONNECTOR_ADC_SRC_EN_BIT;
|
|
}
|
|
|
|
if (chg->hw_skin_temp_mitigation) {
|
|
chan |= MISC_THM_CHANNEL_EN_BIT;
|
|
src_cfg |= THERMREG_SKIN_ADC_SRC_EN_BIT;
|
|
}
|
|
|
|
rc = smblib_masked_write(chg, BATIF_ADC_CHANNEL_EN_REG,
|
|
CONN_THM_CHANNEL_EN_BIT | DIE_TEMP_CHANNEL_EN_BIT |
|
|
MISC_THM_CHANNEL_EN_BIT, chan);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't enable ADC channel rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
rc = smblib_masked_write(chg, MISC_THERMREG_SRC_CFG_REG,
|
|
THERMREG_SW_ICL_ADJUST_BIT | THERMREG_DIE_ADC_SRC_EN_BIT |
|
|
THERMREG_DIE_CMP_SRC_EN_BIT | THERMREG_SKIN_ADC_SRC_EN_BIT |
|
|
SKIN_ADC_CFG_BIT | THERMREG_CONNECTOR_ADC_SRC_EN_BIT, src_cfg);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure THERM_SRC reg rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_init_dc_peripheral(struct smb_charger *chg)
|
|
{
|
|
int rc = 0;
|
|
|
|
/* PMI632 does not have DC peripheral */
|
|
if (chg->chg_param.smb_version == PMI632)
|
|
return 0;
|
|
|
|
/* Set DCIN ICL to 100 mA */
|
|
rc = smblib_set_charge_param(chg, &chg->param.dc_icl, DCIN_ICL_MIN_UA);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't set dc_icl rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/* Disable DC Input missing poller function */
|
|
rc = smblib_masked_write(chg, DCIN_LOAD_CFG_REG,
|
|
INPUT_MISS_POLL_EN_BIT, 0);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't disable DC Input missing poller rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_configure_recharging(struct smb5 *chip)
|
|
{
|
|
int rc = 0;
|
|
struct smb_charger *chg = &chip->chg;
|
|
int val;
|
|
/* Configure VBATT-based or automatic recharging */
|
|
|
|
rc = smblib_masked_write(chg, CHGR_CFG2_REG, RECHG_MASK,
|
|
(chip->dt.auto_recharge_vbat_mv != -EINVAL) ?
|
|
VBAT_BASED_RECHG_BIT : 0);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure VBAT-rechg CHG_CFG2_REG rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
/* program the auto-recharge VBAT threshold */
|
|
if (chip->dt.auto_recharge_vbat_mv != -EINVAL) {
|
|
u32 temp = VBAT_TO_VRAW_ADC(chip->dt.auto_recharge_vbat_mv);
|
|
|
|
temp = ((temp & 0xFF00) >> 8) | ((temp & 0xFF) << 8);
|
|
rc = smblib_batch_write(chg,
|
|
CHGR_ADC_RECHARGE_THRESHOLD_MSB_REG, (u8 *)&temp, 2);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure ADC_RECHARGE_THRESHOLD REG rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
/* Program the sample count for VBAT based recharge to 3 */
|
|
rc = smblib_masked_write(chg, CHGR_NO_SAMPLE_TERM_RCHG_CFG_REG,
|
|
NO_OF_SAMPLE_FOR_RCHG,
|
|
2 << NO_OF_SAMPLE_FOR_RCHG_SHIFT);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure CHGR_NO_SAMPLE_FOR_TERM_RCHG_CFG rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
rc = smblib_masked_write(chg, CHGR_CFG2_REG, RECHG_MASK,
|
|
(chip->dt.auto_recharge_soc != -EINVAL) ?
|
|
SOC_BASED_RECHG_BIT : VBAT_BASED_RECHG_BIT);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure SOC-rechg CHG_CFG2_REG rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
/* program the auto-recharge threshold */
|
|
if (chip->dt.auto_recharge_soc != -EINVAL) {
|
|
val = chip->dt.auto_recharge_soc;
|
|
rc = smblib_set_prop_rechg_soc_thresh(chg, val);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure CHG_RCHG_SOC_REG rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
/* Program the sample count for SOC based recharge to 1 */
|
|
rc = smblib_masked_write(chg, CHGR_NO_SAMPLE_TERM_RCHG_CFG_REG,
|
|
NO_OF_SAMPLE_FOR_RCHG, 0);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure CHGR_NO_SAMPLE_FOR_TERM_RCHG_CFG rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_configure_float_charger(struct smb5 *chip)
|
|
{
|
|
int rc = 0;
|
|
u8 val = 0;
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
/* configure float charger options */
|
|
switch (chip->dt.float_option) {
|
|
case FLOAT_SDP:
|
|
val = FORCE_FLOAT_SDP_CFG_BIT;
|
|
break;
|
|
case DISABLE_CHARGING:
|
|
val = FLOAT_DIS_CHGING_CFG_BIT;
|
|
break;
|
|
case SUSPEND_INPUT:
|
|
val = SUSPEND_FLOAT_CFG_BIT;
|
|
break;
|
|
case FLOAT_DCP:
|
|
default:
|
|
val = 0;
|
|
break;
|
|
}
|
|
|
|
chg->float_cfg = val;
|
|
/* Update float charger setting and set DCD timeout 300ms */
|
|
rc = smblib_masked_write(chg, USBIN_OPTIONS_2_CFG_REG,
|
|
FLOAT_OPTIONS_MASK | DCD_TIMEOUT_SEL_BIT, val);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't change float charger setting rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int smb5_init_connector_type(struct smb_charger *chg)
|
|
{
|
|
int rc, type = 0;
|
|
u8 val = 0;
|
|
|
|
/*
|
|
* PMI632 can have the connector type defined by a dedicated register
|
|
* PMI632_TYPEC_MICRO_USB_MODE_REG or by a common TYPEC_U_USB_CFG_REG.
|
|
*/
|
|
if (chg->chg_param.smb_version == PMI632) {
|
|
rc = smblib_read(chg, PMI632_TYPEC_MICRO_USB_MODE_REG, &val);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't read USB mode rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
type = !!(val & MICRO_USB_MODE_ONLY_BIT);
|
|
}
|
|
|
|
/*
|
|
* If PMI632_TYPEC_MICRO_USB_MODE_REG is not set and for all non-PMI632
|
|
* check the connector type using TYPEC_U_USB_CFG_REG.
|
|
*/
|
|
if (!type) {
|
|
rc = smblib_read(chg, TYPEC_U_USB_CFG_REG, &val);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't read U_USB config rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
type = !!(val & EN_MICRO_USB_MODE_BIT);
|
|
}
|
|
|
|
pr_debug("Connector type=%s\n", type ? "Micro USB" : "TypeC");
|
|
|
|
if (type) {
|
|
chg->connector_type = QTI_POWER_SUPPLY_CONNECTOR_MICRO_USB;
|
|
rc = smb5_configure_micro_usb(chg);
|
|
} else {
|
|
chg->connector_type = QTI_POWER_SUPPLY_CONNECTOR_TYPEC;
|
|
rc = smb5_configure_typec(chg);
|
|
}
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure TypeC/micro-USB mode rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* PMI632 based hw init:
|
|
* - Rerun APSD to ensure proper charger detection if device
|
|
* boots with charger connected.
|
|
* - Initialize flash module for PMI632
|
|
*/
|
|
if (chg->chg_param.smb_version == PMI632) {
|
|
schgm_flash_init(chg);
|
|
smblib_rerun_apsd_if_required(chg);
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static int smb5_init_hw(struct smb5 *chip)
|
|
{
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc;
|
|
u8 val = 0, mask = 0, buf[2] = {0};
|
|
|
|
if (chip->dt.no_battery)
|
|
chg->fake_capacity = 50;
|
|
|
|
if (chg->sdam_base) {
|
|
rc = smblib_write(chg,
|
|
chg->sdam_base + SDAM_QC_DET_STATUS_REG, 0);
|
|
if (rc < 0)
|
|
pr_err("Couldn't clear SDAM QC status rc=%d\n", rc);
|
|
|
|
rc = smblib_batch_write(chg,
|
|
chg->sdam_base + SDAM_QC_ADC_LSB_REG, buf, 2);
|
|
if (rc < 0)
|
|
pr_err("Couldn't clear SDAM ADC status rc=%d\n", rc);
|
|
}
|
|
|
|
if (chip->dt.batt_profile_fcc_ua < 0)
|
|
smblib_get_charge_param(chg, &chg->param.fcc,
|
|
&chg->batt_profile_fcc_ua);
|
|
|
|
if (chip->dt.batt_profile_fv_uv < 0)
|
|
smblib_get_charge_param(chg, &chg->param.fv,
|
|
&chg->batt_profile_fv_uv);
|
|
|
|
smblib_get_charge_param(chg, &chg->param.usb_icl,
|
|
&chg->default_icl_ua);
|
|
smblib_get_charge_param(chg, &chg->param.aicl_5v_threshold,
|
|
&chg->default_aicl_5v_threshold_mv);
|
|
chg->aicl_5v_threshold_mv = chg->default_aicl_5v_threshold_mv;
|
|
smblib_get_charge_param(chg, &chg->param.aicl_cont_threshold,
|
|
&chg->default_aicl_cont_threshold_mv);
|
|
chg->aicl_cont_threshold_mv = chg->default_aicl_cont_threshold_mv;
|
|
|
|
if (chg->charger_temp_max == -EINVAL) {
|
|
rc = smblib_get_thermal_threshold(chg,
|
|
DIE_REG_H_THRESHOLD_MSB_REG,
|
|
&chg->charger_temp_max);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't get charger_temp_max rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* If SW thermal regulation WA is active then all the HW temperature
|
|
* comparators need to be disabled to prevent HW thermal regulation,
|
|
* apart from DIE_TEMP analog comparator for SHDN regulation.
|
|
*/
|
|
if (chg->wa_flags & SW_THERM_REGULATION_WA) {
|
|
rc = smblib_write(chg, MISC_THERMREG_SRC_CFG_REG,
|
|
THERMREG_SW_ICL_ADJUST_BIT
|
|
| THERMREG_DIE_CMP_SRC_EN_BIT);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't disable HW thermal regulation rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
} else {
|
|
/* configure temperature mitigation */
|
|
rc = smb5_configure_mitigation(chg);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure mitigation rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
/* Set HVDCP autonomous mode per DT option */
|
|
smblib_hvdcp_hw_inov_enable(chg, chip->dt.hvdcp_autonomous);
|
|
|
|
/* Enable HVDCP authentication algorithm for non-PD designs */
|
|
if (chg->pd_not_supported)
|
|
smblib_hvdcp_detect_enable(chg, true);
|
|
|
|
/* Disable HVDCP and authentication algorithm if specified in DT */
|
|
if (chg->hvdcp_disable)
|
|
smblib_hvdcp_detect_enable(chg, false);
|
|
|
|
rc = smb5_init_connector_type(chg);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure connector type rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
/* Use ICL results from HW */
|
|
rc = smblib_icl_override(chg, HW_AUTO_MODE);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't disable ICL override rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/* set OTG current limit */
|
|
rc = smblib_set_charge_param(chg, &chg->param.otg_cl, chg->otg_cl_ua);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't set otg current limit rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/* vote 0mA on usb_icl for non battery platforms */
|
|
vote(chg->usb_icl_votable,
|
|
DEFAULT_VOTER, chip->dt.no_battery, 0);
|
|
vote(chg->dc_suspend_votable,
|
|
DEFAULT_VOTER, chip->dt.no_battery, 0);
|
|
vote(chg->fcc_votable, HW_LIMIT_VOTER,
|
|
chip->dt.batt_profile_fcc_ua > 0, chip->dt.batt_profile_fcc_ua);
|
|
vote(chg->fv_votable, HW_LIMIT_VOTER,
|
|
chip->dt.batt_profile_fv_uv > 0, chip->dt.batt_profile_fv_uv);
|
|
vote(chg->fcc_votable,
|
|
BATT_PROFILE_VOTER, chg->batt_profile_fcc_ua > 0,
|
|
chg->batt_profile_fcc_ua);
|
|
vote(chg->fv_votable,
|
|
BATT_PROFILE_VOTER, chg->batt_profile_fv_uv > 0,
|
|
chg->batt_profile_fv_uv);
|
|
|
|
/* Some h/w limit maximum supported ICL */
|
|
vote(chg->usb_icl_votable, HW_LIMIT_VOTER,
|
|
chg->hw_max_icl_ua > 0, chg->hw_max_icl_ua);
|
|
|
|
/* Initialize DC peripheral configurations */
|
|
rc = smb5_init_dc_peripheral(chg);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
/*
|
|
* AICL configuration: enable aicl and aicl rerun and based on DT
|
|
* configuration enable/disable ADB based AICL and Suspend on collapse.
|
|
*/
|
|
mask = USBIN_AICL_PERIODIC_RERUN_EN_BIT | USBIN_AICL_ADC_EN_BIT
|
|
| USBIN_AICL_EN_BIT | SUSPEND_ON_COLLAPSE_USBIN_BIT;
|
|
val = USBIN_AICL_PERIODIC_RERUN_EN_BIT | USBIN_AICL_EN_BIT;
|
|
if (!chg->disable_suspend_on_collapse)
|
|
val |= SUSPEND_ON_COLLAPSE_USBIN_BIT;
|
|
if (chip->dt.adc_based_aicl)
|
|
val |= USBIN_AICL_ADC_EN_BIT;
|
|
|
|
rc = smblib_masked_write(chg, USBIN_AICL_OPTIONS_CFG_REG,
|
|
mask, val);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't config AICL rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = smblib_write(chg, AICL_RERUN_TIME_CFG_REG,
|
|
AICL_RERUN_TIME_12S_VAL);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure AICL rerun interval rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/* enable the charging path */
|
|
rc = vote(chg->chg_disable_votable, DEFAULT_VOTER, false, 0);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't enable charging rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/* configure VBUS for software control */
|
|
rc = smblib_masked_write(chg, DCDC_OTG_CFG_REG, OTG_EN_SRC_CFG_BIT, 0);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure VBUS for SW control rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
val = (ilog2(chip->dt.wd_bark_time / 16) << BARK_WDOG_TIMEOUT_SHIFT)
|
|
& BARK_WDOG_TIMEOUT_MASK;
|
|
val |= (BITE_WDOG_TIMEOUT_8S | BITE_WDOG_DISABLE_CHARGING_CFG_BIT);
|
|
val |= (chip->dt.wd_snarl_time_cfg << SNARL_WDOG_TIMEOUT_SHIFT)
|
|
& SNARL_WDOG_TIMEOUT_MASK;
|
|
|
|
rc = smblib_masked_write(chg, SNARL_BARK_BITE_WD_CFG_REG,
|
|
BITE_WDOG_DISABLE_CHARGING_CFG_BIT |
|
|
SNARL_WDOG_TIMEOUT_MASK | BARK_WDOG_TIMEOUT_MASK |
|
|
BITE_WDOG_TIMEOUT_MASK,
|
|
val);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't configue WD config rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/* enable WD BARK and enable it on plugin */
|
|
val = WDOG_TIMER_EN_ON_PLUGIN_BIT | BARK_WDOG_INT_EN_BIT;
|
|
rc = smblib_masked_write(chg, WD_CFG_REG,
|
|
WATCHDOG_TRIGGER_AFP_EN_BIT |
|
|
WDOG_TIMER_EN_ON_PLUGIN_BIT |
|
|
BARK_WDOG_INT_EN_BIT, val);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't configue WD config rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
/* set termination current threshold values */
|
|
rc = smb5_configure_iterm_thresholds(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't configure ITERM thresholds rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = smb5_configure_float_charger(chip);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
switch (chip->dt.chg_inhibit_thr_mv) {
|
|
case 50:
|
|
rc = smblib_masked_write(chg, CHARGE_INHIBIT_THRESHOLD_CFG_REG,
|
|
CHARGE_INHIBIT_THRESHOLD_MASK,
|
|
INHIBIT_ANALOG_VFLT_MINUS_50MV);
|
|
break;
|
|
case 100:
|
|
rc = smblib_masked_write(chg, CHARGE_INHIBIT_THRESHOLD_CFG_REG,
|
|
CHARGE_INHIBIT_THRESHOLD_MASK,
|
|
INHIBIT_ANALOG_VFLT_MINUS_100MV);
|
|
break;
|
|
case 200:
|
|
rc = smblib_masked_write(chg, CHARGE_INHIBIT_THRESHOLD_CFG_REG,
|
|
CHARGE_INHIBIT_THRESHOLD_MASK,
|
|
INHIBIT_ANALOG_VFLT_MINUS_200MV);
|
|
break;
|
|
case 300:
|
|
rc = smblib_masked_write(chg, CHARGE_INHIBIT_THRESHOLD_CFG_REG,
|
|
CHARGE_INHIBIT_THRESHOLD_MASK,
|
|
INHIBIT_ANALOG_VFLT_MINUS_300MV);
|
|
break;
|
|
case 0:
|
|
rc = smblib_masked_write(chg, CHGR_CFG2_REG,
|
|
CHARGER_INHIBIT_BIT, 0);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure charge inhibit threshold rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = smblib_write(chg, CHGR_FAST_CHARGE_SAFETY_TIMER_CFG_REG,
|
|
FAST_CHARGE_SAFETY_TIMER_768_MIN);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't set CHGR_FAST_CHARGE_SAFETY_TIMER_CFG_REG rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = smb5_configure_recharging(chip);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smblib_disable_hw_jeita(chg, true);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't set hw jeita rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = smblib_masked_write(chg, DCDC_ENG_SDCDC_CFG5_REG,
|
|
ENG_SDCDC_BAT_HPWR_MASK, BOOST_MODE_THRESH_3P6_V);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure DCDC_ENG_SDCDC_CFG5 rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
if (chg->connector_pull_up != -EINVAL) {
|
|
rc = smb5_configure_internal_pull(chg, CONN_THERM,
|
|
get_valid_pullup(chg->connector_pull_up));
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure CONN_THERM pull-up rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
if (chg->smb_pull_up != -EINVAL) {
|
|
rc = smb5_configure_internal_pull(chg, SMB_THERM,
|
|
get_valid_pullup(chg->smb_pull_up));
|
|
if (rc < 0) {
|
|
dev_err(chg->dev,
|
|
"Couldn't configure SMB pull-up rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_post_init(struct smb5 *chip)
|
|
{
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc, val;
|
|
|
|
/*
|
|
* In case the usb path is suspended, we would have missed disabling
|
|
* the icl change interrupt because the interrupt could have been
|
|
* not requested
|
|
*/
|
|
rerun_election(chg->usb_icl_votable);
|
|
|
|
/* configure power role for dual-role */
|
|
val = QTI_POWER_SUPPLY_TYPEC_PR_DUAL;
|
|
rc = smblib_set_prop_typec_power_role(chg, val);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "Couldn't configure DRP role rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
rerun_election(chg->temp_change_irq_disable_votable);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/****************************
|
|
* DETERMINE INITIAL STATUS *
|
|
****************************/
|
|
|
|
static int smb5_determine_initial_status(struct smb5 *chip)
|
|
{
|
|
struct smb_irq_data irq_data = {chip, "determine-initial-status"};
|
|
struct smb_charger *chg = &chip->chg;
|
|
union power_supply_propval val;
|
|
int rc;
|
|
|
|
rc = smblib_get_prop_usb_present(chg, &val);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't get usb present rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
chg->early_usb_attach = val.intval;
|
|
|
|
if (chg->iio_chan_list_qg)
|
|
smblib_config_charger_on_debug_battery(chg);
|
|
|
|
smb5_usb_plugin_irq_handler(0, &irq_data);
|
|
smb5_dc_plugin_irq_handler(0, &irq_data);
|
|
smb5_typec_attach_detach_irq_handler(0, &irq_data);
|
|
smb5_typec_state_change_irq_handler(0, &irq_data);
|
|
smb5_usb_source_change_irq_handler(0, &irq_data);
|
|
smb5_chg_state_change_irq_handler(0, &irq_data);
|
|
smb5_icl_change_irq_handler(0, &irq_data);
|
|
smb5_batt_temp_changed_irq_handler(0, &irq_data);
|
|
smb5_wdog_bark_irq_handler(0, &irq_data);
|
|
smb5_typec_or_rid_detection_change_irq_handler(0, &irq_data);
|
|
smb5_wdog_snarl_irq_handler(0, &irq_data);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**************************
|
|
* INTERRUPT REGISTRATION *
|
|
**************************/
|
|
|
|
static struct smb_irq_info smb5_irqs[] = {
|
|
/* CHARGER IRQs */
|
|
[CHGR_ERROR_IRQ] = {
|
|
.name = "chgr-error",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
[CHG_STATE_CHANGE_IRQ] = {
|
|
.name = "chg-state-change",
|
|
.handler = smb5_chg_state_change_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[STEP_CHG_STATE_CHANGE_IRQ] = {
|
|
.name = "step-chg-state-change",
|
|
},
|
|
[STEP_CHG_SOC_UPDATE_FAIL_IRQ] = {
|
|
.name = "step-chg-soc-update-fail",
|
|
},
|
|
[STEP_CHG_SOC_UPDATE_REQ_IRQ] = {
|
|
.name = "step-chg-soc-update-req",
|
|
},
|
|
[FG_FVCAL_QUALIFIED_IRQ] = {
|
|
.name = "fg-fvcal-qualified",
|
|
},
|
|
[VPH_ALARM_IRQ] = {
|
|
.name = "vph-alarm",
|
|
},
|
|
[VPH_DROP_PRECHG_IRQ] = {
|
|
.name = "vph-drop-prechg",
|
|
},
|
|
/* DCDC IRQs */
|
|
[OTG_FAIL_IRQ] = {
|
|
.name = "otg-fail",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
[OTG_OC_DISABLE_SW_IRQ] = {
|
|
.name = "otg-oc-disable-sw",
|
|
},
|
|
[OTG_OC_HICCUP_IRQ] = {
|
|
.name = "otg-oc-hiccup",
|
|
},
|
|
[BSM_ACTIVE_IRQ] = {
|
|
.name = "bsm-active",
|
|
},
|
|
[HIGH_DUTY_CYCLE_IRQ] = {
|
|
.name = "high-duty-cycle",
|
|
.handler = smb5_high_duty_cycle_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[INPUT_CURRENT_LIMITING_IRQ] = {
|
|
.name = "input-current-limiting",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
[CONCURRENT_MODE_DISABLE_IRQ] = {
|
|
.name = "concurrent-mode-disable",
|
|
},
|
|
[SWITCHER_POWER_OK_IRQ] = {
|
|
.name = "switcher-power-ok",
|
|
.handler = smb5_switcher_power_ok_irq_handler,
|
|
},
|
|
/* BATTERY IRQs */
|
|
[BAT_TEMP_IRQ] = {
|
|
.name = "bat-temp",
|
|
.handler = smb5_batt_temp_changed_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[ALL_CHNL_CONV_DONE_IRQ] = {
|
|
.name = "all-chnl-conv-done",
|
|
},
|
|
[BAT_OV_IRQ] = {
|
|
.name = "bat-ov",
|
|
.handler = smb5_batt_psy_changed_irq_handler,
|
|
},
|
|
[BAT_LOW_IRQ] = {
|
|
.name = "bat-low",
|
|
.handler = smb5_batt_psy_changed_irq_handler,
|
|
},
|
|
[BAT_THERM_OR_ID_MISSING_IRQ] = {
|
|
.name = "bat-therm-or-id-missing",
|
|
.handler = smb5_batt_psy_changed_irq_handler,
|
|
},
|
|
[BAT_TERMINAL_MISSING_IRQ] = {
|
|
.name = "bat-terminal-missing",
|
|
.handler = smb5_batt_psy_changed_irq_handler,
|
|
},
|
|
[BUCK_OC_IRQ] = {
|
|
.name = "buck-oc",
|
|
},
|
|
[VPH_OV_IRQ] = {
|
|
.name = "vph-ov",
|
|
},
|
|
/* USB INPUT IRQs */
|
|
[USBIN_COLLAPSE_IRQ] = {
|
|
.name = "usbin-collapse",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
[USBIN_VASHDN_IRQ] = {
|
|
.name = "usbin-vashdn",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
[USBIN_UV_IRQ] = {
|
|
.name = "usbin-uv",
|
|
.handler = smb5_usbin_uv_irq_handler,
|
|
.wake = true,
|
|
.storm_data = {true, 3000, 5},
|
|
},
|
|
[USBIN_OV_IRQ] = {
|
|
.name = "usbin-ov",
|
|
.handler = smb5_usbin_ov_irq_handler,
|
|
},
|
|
[USBIN_PLUGIN_IRQ] = {
|
|
.name = "usbin-plugin",
|
|
.handler = smb5_usb_plugin_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[USBIN_REVI_CHANGE_IRQ] = {
|
|
.name = "usbin-revi-change",
|
|
},
|
|
[USBIN_SRC_CHANGE_IRQ] = {
|
|
.name = "usbin-src-change",
|
|
.handler = smb5_usb_source_change_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[USBIN_ICL_CHANGE_IRQ] = {
|
|
.name = "usbin-icl-change",
|
|
.handler = smb5_icl_change_irq_handler,
|
|
.wake = true,
|
|
},
|
|
/* DC INPUT IRQs */
|
|
[DCIN_VASHDN_IRQ] = {
|
|
.name = "dcin-vashdn",
|
|
},
|
|
[DCIN_UV_IRQ] = {
|
|
.name = "dcin-uv",
|
|
.handler = smb5_dcin_uv_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[DCIN_OV_IRQ] = {
|
|
.name = "dcin-ov",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
[DCIN_PLUGIN_IRQ] = {
|
|
.name = "dcin-plugin",
|
|
.handler = smb5_dc_plugin_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[DCIN_REVI_IRQ] = {
|
|
.name = "dcin-revi",
|
|
},
|
|
[DCIN_PON_IRQ] = {
|
|
.name = "dcin-pon",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
[DCIN_EN_IRQ] = {
|
|
.name = "dcin-en",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
/* TYPEC IRQs */
|
|
[TYPEC_OR_RID_DETECTION_CHANGE_IRQ] = {
|
|
.name = "typec-or-rid-detect-change",
|
|
.handler = smb5_typec_or_rid_detection_change_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[TYPEC_VPD_DETECT_IRQ] = {
|
|
.name = "typec-vpd-detect",
|
|
},
|
|
[TYPEC_CC_STATE_CHANGE_IRQ] = {
|
|
.name = "typec-cc-state-change",
|
|
.handler = smb5_typec_state_change_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[TYPEC_VCONN_OC_IRQ] = {
|
|
.name = "typec-vconn-oc",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
[TYPEC_VBUS_CHANGE_IRQ] = {
|
|
.name = "typec-vbus-change",
|
|
},
|
|
[TYPEC_ATTACH_DETACH_IRQ] = {
|
|
.name = "typec-attach-detach",
|
|
.handler = smb5_typec_attach_detach_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[TYPEC_LEGACY_CABLE_DETECT_IRQ] = {
|
|
.name = "typec-legacy-cable-detect",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
[TYPEC_TRY_SNK_SRC_DETECT_IRQ] = {
|
|
.name = "typec-try-snk-src-detect",
|
|
},
|
|
/* MISCELLANEOUS IRQs */
|
|
[WDOG_SNARL_IRQ] = {
|
|
.name = "wdog-snarl",
|
|
.handler = smb5_wdog_snarl_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[WDOG_BARK_IRQ] = {
|
|
.name = "wdog-bark",
|
|
.handler = smb5_wdog_bark_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[AICL_FAIL_IRQ] = {
|
|
.name = "aicl-fail",
|
|
},
|
|
[AICL_DONE_IRQ] = {
|
|
.name = "aicl-done",
|
|
.handler = smb5_default_irq_handler,
|
|
},
|
|
[SMB_EN_IRQ] = {
|
|
.name = "smb-en",
|
|
.handler = smb5_smb_en_irq_handler,
|
|
},
|
|
[IMP_TRIGGER_IRQ] = {
|
|
.name = "imp-trigger",
|
|
},
|
|
/*
|
|
* triggered when DIE or SKIN or CONNECTOR temperature across
|
|
* either of the _REG_L, _REG_H, _RST, or _SHDN thresholds
|
|
*/
|
|
[TEMP_CHANGE_IRQ] = {
|
|
.name = "temp-change",
|
|
.handler = smb5_temp_change_irq_handler,
|
|
.wake = true,
|
|
},
|
|
[TEMP_CHANGE_SMB_IRQ] = {
|
|
.name = "temp-change-smb",
|
|
},
|
|
/* FLASH */
|
|
[VREG_OK_IRQ] = {
|
|
.name = "vreg-ok",
|
|
},
|
|
[ILIM_S2_IRQ] = {
|
|
.name = "ilim2-s2",
|
|
.handler = smb5_schgm_flash_ilim2_irq_handler,
|
|
},
|
|
[ILIM_S1_IRQ] = {
|
|
.name = "ilim1-s1",
|
|
},
|
|
[VOUT_DOWN_IRQ] = {
|
|
.name = "vout-down",
|
|
},
|
|
[VOUT_UP_IRQ] = {
|
|
.name = "vout-up",
|
|
},
|
|
[FLASH_STATE_CHANGE_IRQ] = {
|
|
.name = "flash-state-change",
|
|
.handler = smb5_schgm_flash_state_change_irq_handler,
|
|
},
|
|
[TORCH_REQ_IRQ] = {
|
|
.name = "torch-req",
|
|
},
|
|
[FLASH_EN_IRQ] = {
|
|
.name = "flash-en",
|
|
},
|
|
/* SDAM */
|
|
[SDAM_STS_IRQ] = {
|
|
.name = "sdam-sts",
|
|
.handler = smb5_sdam_sts_change_irq_handler,
|
|
},
|
|
};
|
|
|
|
static int smb5_get_irq_index_byname(const char *irq_name)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(smb5_irqs); i++) {
|
|
if (strcmp(smb5_irqs[i].name, irq_name) == 0)
|
|
return i;
|
|
}
|
|
|
|
return -ENOENT;
|
|
}
|
|
|
|
static int smb5_request_interrupt(struct smb5 *chip,
|
|
struct device_node *node, const char *irq_name)
|
|
{
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc, irq, irq_index;
|
|
struct smb_irq_data *irq_data;
|
|
|
|
irq = of_irq_get_byname(node, irq_name);
|
|
if (irq < 0) {
|
|
pr_err("Couldn't get irq %s byname\n", irq_name);
|
|
return irq;
|
|
}
|
|
|
|
irq_index = smb5_get_irq_index_byname(irq_name);
|
|
if (irq_index < 0) {
|
|
pr_err("%s is not a defined irq\n", irq_name);
|
|
return irq_index;
|
|
}
|
|
|
|
if (!smb5_irqs[irq_index].handler)
|
|
return 0;
|
|
|
|
irq_data = devm_kzalloc(chg->dev, sizeof(*irq_data), GFP_KERNEL);
|
|
if (!irq_data)
|
|
return -ENOMEM;
|
|
|
|
irq_data->parent_data = chip;
|
|
irq_data->name = irq_name;
|
|
irq_data->storm_data = smb5_irqs[irq_index].storm_data;
|
|
mutex_init(&irq_data->storm_data.storm_lock);
|
|
|
|
smb5_irqs[irq_index].enabled = true;
|
|
rc = devm_request_threaded_irq(chg->dev, irq, NULL,
|
|
smb5_irqs[irq_index].handler,
|
|
IRQF_ONESHOT, irq_name, irq_data);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't request irq %d\n", irq);
|
|
return rc;
|
|
}
|
|
|
|
smb5_irqs[irq_index].irq = irq;
|
|
smb5_irqs[irq_index].irq_data = irq_data;
|
|
if (smb5_irqs[irq_index].wake)
|
|
enable_irq_wake(irq);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_request_interrupts(struct smb5 *chip)
|
|
{
|
|
struct smb_charger *chg = &chip->chg;
|
|
struct device_node *node = chg->dev->of_node;
|
|
struct device_node *child;
|
|
int rc = 0;
|
|
const char *name;
|
|
struct property *prop;
|
|
|
|
for_each_available_child_of_node(node, child) {
|
|
of_property_for_each_string(child, "interrupt-names",
|
|
prop, name) {
|
|
rc = smb5_request_interrupt(chip, child, name);
|
|
if (rc < 0)
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
vote(chg->limited_irq_disable_votable, CHARGER_TYPE_VOTER, true, 0);
|
|
vote(chg->hdc_irq_disable_votable, CHARGER_TYPE_VOTER, true, 0);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static void smb5_free_interrupts(struct smb_charger *chg)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(smb5_irqs); i++) {
|
|
if (smb5_irqs[i].irq > 0)
|
|
if (smb5_irqs[i].wake)
|
|
disable_irq_wake(smb5_irqs[i].irq);
|
|
}
|
|
}
|
|
|
|
static void smb5_disable_interrupts(struct smb_charger *chg)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(smb5_irqs); i++) {
|
|
if (smb5_irqs[i].irq > 0)
|
|
disable_irq(smb5_irqs[i].irq);
|
|
}
|
|
}
|
|
|
|
#if defined(CONFIG_DEBUG_FS)
|
|
|
|
static int force_batt_psy_update_write(void *data, u64 val)
|
|
{
|
|
struct smb_charger *chg = data;
|
|
|
|
power_supply_changed(chg->batt_psy);
|
|
return 0;
|
|
}
|
|
DEFINE_DEBUGFS_ATTRIBUTE(force_batt_psy_update_ops, NULL,
|
|
force_batt_psy_update_write, "0x%02llx\n");
|
|
|
|
static int force_usb_psy_update_write(void *data, u64 val)
|
|
{
|
|
struct smb_charger *chg = data;
|
|
|
|
power_supply_changed(chg->usb_psy);
|
|
return 0;
|
|
}
|
|
DEFINE_DEBUGFS_ATTRIBUTE(force_usb_psy_update_ops, NULL,
|
|
force_usb_psy_update_write, "0x%02llx\n");
|
|
|
|
static int force_dc_psy_update_write(void *data, u64 val)
|
|
{
|
|
struct smb_charger *chg = data;
|
|
|
|
power_supply_changed(chg->dc_psy);
|
|
return 0;
|
|
}
|
|
DEFINE_DEBUGFS_ATTRIBUTE(force_dc_psy_update_ops, NULL,
|
|
force_dc_psy_update_write, "0x%02llx\n");
|
|
|
|
static void smb5_create_debugfs(struct smb5 *chip)
|
|
{
|
|
struct dentry *file;
|
|
|
|
chip->dfs_root = debugfs_create_dir("charger", NULL);
|
|
if (IS_ERR_OR_NULL(chip->dfs_root)) {
|
|
pr_err("Couldn't create charger debugfs rc=%ld\n",
|
|
(long)chip->dfs_root);
|
|
return;
|
|
}
|
|
|
|
file = debugfs_create_file("force_batt_psy_update", 0600,
|
|
chip->dfs_root, chip, &force_batt_psy_update_ops);
|
|
if (IS_ERR_OR_NULL(file))
|
|
pr_err("Couldn't create force_batt_psy_update file rc=%ld\n",
|
|
(long)file);
|
|
|
|
file = debugfs_create_file("force_usb_psy_update", 0600,
|
|
chip->dfs_root, chip, &force_usb_psy_update_ops);
|
|
if (IS_ERR_OR_NULL(file))
|
|
pr_err("Couldn't create force_usb_psy_update file rc=%ld\n",
|
|
(long)file);
|
|
|
|
file = debugfs_create_file("force_dc_psy_update", 0600,
|
|
chip->dfs_root, chip, &force_dc_psy_update_ops);
|
|
if (IS_ERR_OR_NULL(file))
|
|
pr_err("Couldn't create force_dc_psy_update file rc=%ld\n",
|
|
(long)file);
|
|
|
|
debugfs_create_u32("debug_mask", 0600, chip->dfs_root,
|
|
&__debug_mask);
|
|
}
|
|
|
|
#else
|
|
|
|
static void smb5_create_debugfs(struct smb5 *chip)
|
|
{}
|
|
|
|
#endif
|
|
|
|
static int smb5_show_charger_status(struct smb5 *chip)
|
|
{
|
|
struct smb_charger *chg = &chip->chg;
|
|
union power_supply_propval val;
|
|
int usb_present, batt_present, batt_health, batt_charge_type;
|
|
int rc;
|
|
|
|
rc = smblib_get_prop_usb_present(chg, &val);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't get usb present rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
usb_present = val.intval;
|
|
|
|
rc = smblib_get_prop_batt_present(chg, &val);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't get batt present rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
batt_present = val.intval;
|
|
|
|
rc = smblib_get_prop_batt_health(chg, &val);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't get batt health rc=%d\n", rc);
|
|
val.intval = POWER_SUPPLY_HEALTH_UNKNOWN;
|
|
}
|
|
batt_health = val.intval;
|
|
|
|
rc = smblib_get_prop_batt_charge_type(chg, &val);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't get batt charge type rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
batt_charge_type = val.intval;
|
|
|
|
pr_info("SMB5 status - usb:present=%d type=%d batt:present = %d health = %d charge = %d\n",
|
|
usb_present, chg->real_charger_type,
|
|
batt_present, batt_health, batt_charge_type);
|
|
return rc;
|
|
}
|
|
|
|
/*********************************
|
|
* TYPEC CLASS REGISTRATION *
|
|
**********************************/
|
|
|
|
static int smb5_init_typec_class(struct smb5 *chip)
|
|
{
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc = 0;
|
|
|
|
mutex_init(&chg->typec_lock);
|
|
|
|
/* Register typec class for only non-PD TypeC and uUSB designs */
|
|
if (!chg->pd_not_supported)
|
|
return rc;
|
|
|
|
chg->typec_caps.type = TYPEC_PORT_DRP;
|
|
chg->typec_caps.data = TYPEC_PORT_DRD;
|
|
chg->typec_partner_desc.usb_pd = false;
|
|
chg->typec_partner_desc.accessory = TYPEC_ACCESSORY_NONE;
|
|
chg->typec_caps.revision = 0x0130;
|
|
|
|
chg->typec_port = typec_register_port(chg->dev, &chg->typec_caps);
|
|
if (IS_ERR(chg->typec_port)) {
|
|
rc = PTR_ERR(chg->typec_port);
|
|
pr_err("failed to register typec_port rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_fwnode_xlate(struct iio_dev *indio_dev,
|
|
const struct fwnode_reference_args *iiospec)
|
|
{
|
|
struct smb5 *iio_chip = iio_priv(indio_dev);
|
|
struct iio_chan_spec *iio_chan = iio_chip->iio_chan_ids;
|
|
int i;
|
|
|
|
for (i = 0; i < iio_chip->nchannels; i++, iio_chan++)
|
|
if (iio_chan->channel == iiospec->args[0])
|
|
return i;
|
|
|
|
return -EINVAL;
|
|
}
|
|
|
|
static int smb5_read_raw(struct iio_dev *indio_dev,
|
|
struct iio_chan_spec const *chan, int *val, int *val2,
|
|
long mask)
|
|
{
|
|
struct smb5 *iio_chip = iio_priv(indio_dev);
|
|
struct smb_charger *chg = &iio_chip->chg;
|
|
|
|
return smb5_iio_get_prop(chg, chan->channel, val);
|
|
}
|
|
|
|
static int smb5_write_raw(struct iio_dev *indio_dev,
|
|
struct iio_chan_spec const *chan, int val, int val2,
|
|
long mask)
|
|
{
|
|
struct smb5 *iio_chip = iio_priv(indio_dev);
|
|
struct smb_charger *chg = &iio_chip->chg;
|
|
|
|
return smb5_iio_set_prop(chg, chan->channel, val);
|
|
}
|
|
|
|
static const struct iio_info smb5_iio_info = {
|
|
.read_raw = smb5_read_raw,
|
|
.write_raw = smb5_write_raw,
|
|
.fwnode_xlate = smb5_fwnode_xlate,
|
|
};
|
|
|
|
static int smb5_direct_iio_read(struct device *dev, int iio_chan_no, int *val)
|
|
{
|
|
struct smb5 *chip = dev_get_drvdata(dev);
|
|
struct smb_charger *chg = &chip->chg;
|
|
int rc;
|
|
|
|
rc = smb5_iio_get_prop(chg, iio_chan_no, val);
|
|
|
|
return (rc < 0) ? rc : 0;
|
|
}
|
|
|
|
static int smb5_direct_iio_write(struct device *dev, int iio_chan_no, int val)
|
|
{
|
|
struct smb5 *chip = dev_get_drvdata(dev);
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
return smb5_iio_set_prop(chg, iio_chan_no, val);
|
|
}
|
|
|
|
static int smb5_iio_init(struct smb5 *chip, struct platform_device *pdev,
|
|
struct iio_dev *indio_dev)
|
|
{
|
|
struct iio_chan_spec *iio_chan;
|
|
int i, rc;
|
|
|
|
for (i = 0; i < chip->nchannels; i++) {
|
|
chip->iio_chans[i].indio_dev = indio_dev;
|
|
iio_chan = &chip->iio_chan_ids[i];
|
|
chip->iio_chans[i].channel = iio_chan;
|
|
|
|
iio_chan->channel = smb5_chans_pmic[i].channel_num;
|
|
iio_chan->datasheet_name = smb5_chans_pmic[i].datasheet_name;
|
|
iio_chan->extend_name = smb5_chans_pmic[i].datasheet_name;
|
|
iio_chan->info_mask_separate = smb5_chans_pmic[i].info_mask;
|
|
iio_chan->type = smb5_chans_pmic[i].type;
|
|
iio_chan->address = i;
|
|
}
|
|
|
|
indio_dev->dev.parent = &pdev->dev;
|
|
indio_dev->dev.of_node = pdev->dev.of_node;
|
|
indio_dev->name = "qpnp-smb5";
|
|
indio_dev->modes = INDIO_DIRECT_MODE;
|
|
indio_dev->channels = chip->iio_chan_ids;
|
|
indio_dev->num_channels = chip->nchannels;
|
|
|
|
rc = devm_iio_device_register(&pdev->dev, indio_dev);
|
|
if (rc)
|
|
pr_err("iio device register failed rc=%d\n", rc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
int smb5_extcon_init(struct smb_charger *chg)
|
|
{
|
|
int rc;
|
|
|
|
/* extcon registration */
|
|
chg->extcon = devm_extcon_dev_allocate(chg->dev, smblib_extcon_cable);
|
|
if (IS_ERR(chg->extcon)) {
|
|
rc = PTR_ERR(chg->extcon);
|
|
dev_err(chg->dev, "failed to allocate extcon device rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = devm_extcon_dev_register(chg->dev, chg->extcon);
|
|
if (rc < 0) {
|
|
dev_err(chg->dev, "failed to register extcon device rc=%d\n",
|
|
rc);
|
|
return rc;
|
|
}
|
|
|
|
/* Support reporting polarity and speed via properties */
|
|
rc = extcon_set_property_capability(chg->extcon,
|
|
EXTCON_USB, EXTCON_PROP_USB_TYPEC_POLARITY);
|
|
rc |= extcon_set_property_capability(chg->extcon,
|
|
EXTCON_USB, EXTCON_PROP_USB_SS);
|
|
rc |= extcon_set_property_capability(chg->extcon,
|
|
EXTCON_USB_HOST, EXTCON_PROP_USB_TYPEC_POLARITY);
|
|
rc |= extcon_set_property_capability(chg->extcon,
|
|
EXTCON_USB_HOST, EXTCON_PROP_USB_SS);
|
|
if (rc < 0)
|
|
dev_err(chg->dev,
|
|
"failed to configure extcon capabilities\n");
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_probe(struct platform_device *pdev)
|
|
{
|
|
struct smb5 *chip;
|
|
struct iio_dev *indio_dev;
|
|
struct smb_charger *chg;
|
|
int rc = 0;
|
|
|
|
indio_dev = devm_iio_device_alloc(&pdev->dev, sizeof(*chip));
|
|
if (!indio_dev)
|
|
return -ENOMEM;
|
|
|
|
indio_dev->info = &smb5_iio_info;
|
|
chip = iio_priv(indio_dev);
|
|
chip->nchannels = ARRAY_SIZE(smb5_chans_pmic);
|
|
|
|
chip->iio_chans = devm_kcalloc(&pdev->dev, chip->nchannels,
|
|
sizeof(*chip->iio_chans), GFP_KERNEL);
|
|
if (!chip->iio_chans)
|
|
return -ENOMEM;
|
|
|
|
chip->iio_chan_ids = devm_kcalloc(&pdev->dev, chip->nchannels,
|
|
sizeof(*chip->iio_chan_ids), GFP_KERNEL);
|
|
if (!chip->iio_chan_ids)
|
|
return -ENOMEM;
|
|
|
|
chg = &chip->chg;
|
|
chg->iio_chans = chip->iio_chans;
|
|
chg->iio_chan_list_qg = NULL;
|
|
chg->dev = &pdev->dev;
|
|
chg->debug_mask = &__debug_mask;
|
|
chg->pd_disabled = 0;
|
|
chg->weak_chg_icl_ua = 500000;
|
|
chg->mode = PARALLEL_MASTER;
|
|
chg->irq_info = smb5_irqs;
|
|
chg->die_health = -EINVAL;
|
|
chg->connector_health = -EINVAL;
|
|
chg->otg_present = false;
|
|
chg->main_fcc_max = -EINVAL;
|
|
mutex_init(&chg->adc_lock);
|
|
|
|
chg->regmap = dev_get_regmap(chg->dev->parent, NULL);
|
|
if (!chg->regmap) {
|
|
pr_err("parent regmap is missing\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = smb5_iio_init(chip, pdev, indio_dev);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
rc = smb5_chg_config_init(chip);
|
|
if (rc < 0) {
|
|
if (rc != -EPROBE_DEFER)
|
|
pr_err("Couldn't setup chg_config rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = smb5_parse_dt(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't parse device tree rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
if (alarmtimer_get_rtcdev())
|
|
alarm_init(&chg->lpd_recheck_timer, ALARM_REALTIME,
|
|
smblib_lpd_recheck_timer);
|
|
else
|
|
return -EPROBE_DEFER;
|
|
|
|
/* set driver data before resources request it */
|
|
platform_set_drvdata(pdev, chip);
|
|
|
|
chg->chg_param.iio_read = smb5_direct_iio_read;
|
|
chg->chg_param.iio_write = smb5_direct_iio_write;
|
|
|
|
rc = smblib_init(chg);
|
|
if (rc < 0) {
|
|
pr_err("Smblib_init failed rc=%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = smb5_extcon_init(chg);
|
|
if (rc < 0)
|
|
goto cleanup;
|
|
|
|
rc = smb5_init_hw(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't initialize hardware rc=%d\n", rc);
|
|
goto cleanup;
|
|
}
|
|
|
|
/*
|
|
* VBUS regulator enablement/disablement for host mode is handled
|
|
* by USB-PD driver only. For micro-USB and non-PD typeC designs,
|
|
* the VBUS regulator is enabled/disabled by the smb driver itself
|
|
* before sending extcon notifications.
|
|
* Hence, register vbus and vconn regulators for PD supported designs
|
|
* only.
|
|
*/
|
|
if (!chg->pd_not_supported) {
|
|
rc = smb5_init_vbus_regulator(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't initialize vbus regulator rc=%d\n",
|
|
rc);
|
|
goto cleanup;
|
|
}
|
|
|
|
rc = smb5_init_vconn_regulator(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't initialize vconn regulator rc=%d\n",
|
|
rc);
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
switch (chg->chg_param.smb_version) {
|
|
case PM8150B:
|
|
case PM6150:
|
|
case PM7250B:
|
|
rc = smb5_init_dc_psy(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't initialize dc psy rc=%d\n", rc);
|
|
goto cleanup;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
rc = smb5_init_usb_psy(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't initialize usb psy rc=%d\n", rc);
|
|
goto cleanup;
|
|
}
|
|
|
|
rc = smb5_init_usb_port_psy(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't initialize usb pc_port psy rc=%d\n", rc);
|
|
goto cleanup;
|
|
}
|
|
|
|
rc = smb5_init_batt_psy(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't initialize batt psy rc=%d\n", rc);
|
|
goto cleanup;
|
|
}
|
|
|
|
rc = smb5_init_typec_class(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't initialize typec class rc=%d\n", rc);
|
|
goto cleanup;
|
|
}
|
|
|
|
rc = smb5_determine_initial_status(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't determine initial status rc=%d\n",
|
|
rc);
|
|
goto cleanup;
|
|
}
|
|
|
|
rc = smb5_request_interrupts(chip);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't request interrupts rc=%d\n", rc);
|
|
goto cleanup;
|
|
}
|
|
|
|
rc = smb5_post_init(chip);
|
|
if (rc < 0) {
|
|
pr_err("Failed in post init rc=%d\n", rc);
|
|
goto free_irq;
|
|
}
|
|
|
|
smb5_create_debugfs(chip);
|
|
|
|
rc = sysfs_create_groups(&chg->dev->kobj, smb5_groups);
|
|
if (rc < 0) {
|
|
pr_err("Couldn't create sysfs files rc=%d\n", rc);
|
|
goto free_irq;
|
|
}
|
|
|
|
rc = smb5_show_charger_status(chip);
|
|
if (rc < 0) {
|
|
pr_err("Failed in getting charger status rc=%d\n", rc);
|
|
goto free_irq;
|
|
}
|
|
|
|
device_init_wakeup(chg->dev, true);
|
|
|
|
pr_info("QPNP SMB5 probed successfully\n");
|
|
|
|
return rc;
|
|
|
|
free_irq:
|
|
smb5_free_interrupts(chg);
|
|
cleanup:
|
|
smblib_deinit(chg);
|
|
platform_set_drvdata(pdev, NULL);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int smb5_remove(struct platform_device *pdev)
|
|
{
|
|
struct smb5 *chip = platform_get_drvdata(pdev);
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
/* force enable APSD */
|
|
smblib_masked_write(chg, USBIN_OPTIONS_1_CFG_REG,
|
|
BC1P2_SRC_DETECT_BIT, BC1P2_SRC_DETECT_BIT);
|
|
|
|
smb5_free_interrupts(chg);
|
|
smblib_deinit(chg);
|
|
sysfs_remove_groups(&chg->dev->kobj, smb5_groups);
|
|
platform_set_drvdata(pdev, NULL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void smb5_shutdown(struct platform_device *pdev)
|
|
{
|
|
struct smb5 *chip = platform_get_drvdata(pdev);
|
|
struct smb_charger *chg = &chip->chg;
|
|
|
|
/* disable all interrupts */
|
|
smb5_disable_interrupts(chg);
|
|
|
|
/* disable the SMB_EN configuration */
|
|
smblib_masked_write(chg, MISC_SMB_EN_CMD_REG, EN_CP_CMD_BIT, 0);
|
|
|
|
/* configure power role for UFP */
|
|
if (chg->connector_type == QTI_POWER_SUPPLY_CONNECTOR_TYPEC)
|
|
smblib_masked_write(chg, TYPE_C_MODE_CFG_REG,
|
|
TYPEC_POWER_ROLE_CMD_MASK, EN_SNK_ONLY_BIT);
|
|
|
|
/* force enable and rerun APSD */
|
|
smblib_apsd_enable(chg, true);
|
|
smblib_hvdcp_exit_config(chg);
|
|
}
|
|
|
|
static const struct of_device_id match_table[] = {
|
|
{
|
|
.compatible = "qcom,pm8150-smb5",
|
|
.data = (void *)PM8150B,
|
|
},
|
|
{
|
|
.compatible = "qcom,pm7250b-smb5",
|
|
.data = (void *)PM7250B,
|
|
},
|
|
{
|
|
.compatible = "qcom,pm6150-smb5",
|
|
.data = (void *)PM6150,
|
|
},
|
|
{
|
|
.compatible = "qcom,pmi632-smb5",
|
|
.data = (void *)PMI632,
|
|
},
|
|
{ },
|
|
};
|
|
|
|
static struct platform_driver smb5_driver = {
|
|
.driver = {
|
|
.name = "qcom,qpnp-smb5",
|
|
.of_match_table = match_table,
|
|
},
|
|
.probe = smb5_probe,
|
|
.remove = smb5_remove,
|
|
.shutdown = smb5_shutdown,
|
|
};
|
|
module_platform_driver(smb5_driver);
|
|
|
|
MODULE_DESCRIPTION("QPNP SMB5 Charger Driver");
|
|
MODULE_LICENSE("GPL");
|