123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Copyright (c) 2016-2020 The Linux Foundation. All rights reserved.
- * Copyright (c) 2022-2023, Qualcomm Innovation Center, Inc. All rights reserved.
- */
- #define pr_fmt(fmt) "SMB1355: %s: " fmt, __func__
- #include <linux/device.h>
- #include <linux/regmap.h>
- #include <linux/interrupt.h>
- #include <linux/module.h>
- #include <linux/iio/consumer.h>
- #include <linux/iio/iio.h>
- #include <linux/platform_device.h>
- #include <linux/of.h>
- #include <linux/of_device.h>
- #include <linux/of_irq.h>
- #include <linux/regulator/driver.h>
- #include <linux/regulator/machine.h>
- #include <linux/regulator/of_regulator.h>
- #include <linux/power_supply.h>
- #include <linux/qti_power_supply.h>
- #include <linux/workqueue.h>
- #include <linux/pmic-voter.h>
- #include <linux/string.h>
- #include <dt-bindings/iio/qti_power_supply_iio.h>
- /* SMB1355 registers, different than mentioned in smb-reg.h */
- #define REVID_BASE 0x0100
- #define I2C_SS_DIG_BASE 0x0E00
- #define CHGR_BASE 0x1000
- #define ANA2_BASE 0x1100
- #define BATIF_BASE 0x1200
- #define USBIN_BASE 0x1300
- #define ANA1_BASE 0x1400
- #define MISC_BASE 0x1600
- #define REVID_MFG_ID_SPARE_REG (REVID_BASE + 0xFF)
- #define I2C_SS_DIG_PMIC_SID_REG (I2C_SS_DIG_BASE + 0x45)
- #define PMIC_SID_MASK GENMASK(3, 0)
- #define PMIC_SID0_BIT BIT(0)
- #define BATTERY_STATUS_2_REG (CHGR_BASE + 0x0B)
- #define DISABLE_CHARGING_BIT BIT(3)
- #define BATTERY_STATUS_3_REG (CHGR_BASE + 0x0C)
- #define BATT_GT_PRE_TO_FAST_BIT BIT(4)
- #define ENABLE_CHARGING_BIT BIT(3)
- #define CHGR_CHARGING_ENABLE_CMD_REG (CHGR_BASE + 0x42)
- #define CHARGING_ENABLE_CMD_BIT BIT(0)
- #define CHGR_CFG2_REG (CHGR_BASE + 0x51)
- #define CHG_EN_SRC_BIT BIT(7)
- #define CHG_EN_POLARITY_BIT BIT(6)
- #define CFG_REG (CHGR_BASE + 0x53)
- #define CHG_OPTION_PIN_TRIM_BIT BIT(7)
- #define BATN_SNS_CFG_BIT BIT(4)
- #define CFG_TAPER_DIS_AFVC_BIT BIT(3)
- #define BATFET_SHUTDOWN_CFG_BIT BIT(2)
- #define VDISCHG_EN_CFG_BIT BIT(1)
- #define VCHG_EN_CFG_BIT BIT(0)
- #define FAST_CHARGE_CURRENT_CFG_REG (CHGR_BASE + 0x61)
- #define FAST_CHARGE_CURRENT_SETTING_MASK GENMASK(7, 0)
- #define CHGR_BATTOV_CFG_REG (CHGR_BASE + 0x70)
- #define BATTOV_SETTING_MASK GENMASK(7, 0)
- #define CHGR_PRE_TO_FAST_THRESHOLD_CFG_REG (CHGR_BASE + 0x74)
- #define PRE_TO_FAST_CHARGE_THRESHOLD_MASK GENMASK(2, 0)
- #define ANA2_TR_SBQ_ICL_1X_REF_OFFSET_REG (ANA2_BASE + 0xF5)
- #define TR_SBQ_ICL_1X_REF_OFFSET GENMASK(4, 0)
- #define POWER_MODE_HICCUP_CFG (BATIF_BASE + 0x72)
- #define MAX_HICCUP_DUETO_BATDIS_MASK GENMASK(5, 2)
- #define HICCUP_TIMEOUT_CFG_MASK GENMASK(1, 0)
- #define BATIF_CFG_SMISC_BATID_REG (BATIF_BASE + 0x73)
- #define CFG_SMISC_RBIAS_EXT_CTRL_BIT BIT(2)
- #define SMB2CHG_BATIF_ENG_SMISC_DIETEMP (BATIF_BASE + 0xC0)
- #define TDIE_COMPARATOR_THRESHOLD GENMASK(5, 0)
- #define DIE_LOW_RANGE_BASE_DEGC 34
- #define DIE_LOW_RANGE_DELTA 16
- #define DIE_LOW_RANGE_MAX_DEGC 97
- #define DIE_LOW_RANGE_SHIFT 4
- #define BATIF_ENG_SCMISC_SPARE1_REG (BATIF_BASE + 0xC2)
- #define EXT_BIAS_PIN_BIT BIT(2)
- #define DIE_TEMP_COMP_HYST_BIT BIT(1)
- #define ANA1_ENG_SREFGEN_CFG2_REG (ANA1_BASE + 0xC1)
- #define VALLEY_COMPARATOR_EN_BIT BIT(0)
- #define TEMP_COMP_STATUS_REG (MISC_BASE + 0x07)
- #define TEMP_RST_HOT_BIT BIT(2)
- #define TEMP_UB_HOT_BIT BIT(1)
- #define TEMP_LB_HOT_BIT BIT(0)
- #define SKIN_TEMP_SHIFT 4
- #define MISC_RT_STS_REG (MISC_BASE + 0x10)
- #define HARD_ILIMIT_RT_STS_BIT BIT(5)
- #define BANDGAP_ENABLE_REG (MISC_BASE + 0x42)
- #define BANDGAP_ENABLE_CMD_BIT BIT(0)
- #define BARK_BITE_WDOG_PET_REG (MISC_BASE + 0x43)
- #define BARK_BITE_WDOG_PET_BIT BIT(0)
- #define CLOCK_REQUEST_REG (MISC_BASE + 0x44)
- #define CLOCK_REQUEST_CMD_BIT BIT(0)
- #define WD_CFG_REG (MISC_BASE + 0x51)
- #define WATCHDOG_TRIGGER_AFP_EN_BIT BIT(7)
- #define BARK_WDOG_INT_EN_BIT BIT(6)
- #define BITE_WDOG_INT_EN_BIT BIT(5)
- #define WDOG_IRQ_SFT_BIT BIT(2)
- #define WDOG_TIMER_EN_ON_PLUGIN_BIT BIT(1)
- #define WDOG_TIMER_EN_BIT BIT(0)
- #define MISC_CUST_SDCDC_CLK_CFG_REG (MISC_BASE + 0xA0)
- #define SWITCHER_CLK_FREQ_MASK GENMASK(3, 0)
- #define MISC_CUST_SDCDC_ILIMIT_CFG_REG (MISC_BASE + 0xA1)
- #define LS_VALLEY_THRESH_PCT_BIT BIT(3)
- #define PCL_LIMIT_MASK GENMASK(1, 0)
- #define SNARL_BARK_BITE_WD_CFG_REG (MISC_BASE + 0x53)
- #define BITE_WDOG_DISABLE_CHARGING_CFG_BIT BIT(7)
- #define SNARL_WDOG_TIMEOUT_MASK GENMASK(6, 4)
- #define BARK_WDOG_TIMEOUT_MASK GENMASK(3, 2)
- #define BITE_WDOG_TIMEOUT_MASK GENMASK(1, 0)
- #define MISC_THERMREG_SRC_CFG_REG (MISC_BASE + 0x70)
- #define BYP_THERM_CHG_CURR_ADJUST_BIT BIT(2)
- #define THERMREG_SKIN_CMP_SRC_EN_BIT BIT(1)
- #define THERMREG_DIE_CMP_SRC_EN_BIT BIT(0)
- #define MISC_CHGR_TRIM_OPTIONS_REG (MISC_BASE + 0x55)
- #define CMD_RBIAS_EN_BIT BIT(2)
- #define MISC_ENG_SDCDC_RESERVE1_REG (MISC_BASE + 0xC4)
- #define MINOFF_TIME_MASK BIT(6)
- #define MISC_ENG_SDCDC_CFG8_REG (MISC_BASE + 0xC7)
- #define DEAD_TIME_MASK GENMASK(2, 0)
- #define DEAD_TIME_32NS 0x4
- #define MISC_ENG_SDCDC_INPUT_CURRENT_CFG1_REG (MISC_BASE + 0xC8)
- #define PROLONG_ISENSE_MASK GENMASK(7, 6)
- #define PROLONG_ISENSEM_SHIFT 6
- #define SAMPLE_HOLD_DELAY_MASK GENMASK(5, 2)
- #define SAMPLE_HOLD_DELAY_SHIFT 2
- #define DISABLE_ILIMIT_BIT BIT(0)
- #define MISC_ENG_SDCDC_INPUT_CURRENT_CFG2_REG (MISC_BASE + 0xC9)
- #define INPUT_CURRENT_LIMIT_SOURCE_BIT BIT(7)
- #define TC_ISENSE_AMPLIFIER_MASK GENMASK(6, 4)
- #define TC_ISENSE_AMPLIFIER_SHIFT 4
- #define HS_II_CORRECTION_MASK GENMASK(3, 0)
- #define MISC_ENG_SDCDC_RESERVE3_REG (MISC_BASE + 0xCB)
- #define VDDCAP_SHORT_DISABLE_TRISTATE_BIT BIT(7)
- #define PCL_SHUTDOWN_BUCK_BIT BIT(6)
- #define ISENSE_TC_CORRECTION_BIT BIT(5)
- #define II_SOURCE_BIT BIT(4)
- #define SCALE_SLOPE_COMP_MASK GENMASK(3, 0)
- #define USBIN_CURRENT_LIMIT_CFG_REG (USBIN_BASE + 0x70)
- #define USB_TR_SCPATH_ICL_1X_GAIN_REG (USBIN_BASE + 0xF2)
- #define TR_SCPATH_ICL_1X_GAIN_MASK GENMASK(5, 0)
- #define IS_USBIN(mode) \
- ((mode == QTI_POWER_SUPPLY_PL_USBIN_USBIN) \
- || (mode == QTI_POWER_SUPPLY_PL_USBIN_USBIN_EXT))
- #define PARALLEL_ENABLE_VOTER "PARALLEL_ENABLE_VOTER"
- struct smb_chg_param {
- const char *name;
- u16 reg;
- int min_u;
- int max_u;
- int step_u;
- };
- struct smb_params {
- struct smb_chg_param fcc;
- struct smb_chg_param ov;
- struct smb_chg_param usb_icl;
- };
- static struct smb_params v1_params = {
- .fcc = {
- .name = "fast charge current",
- .reg = FAST_CHARGE_CURRENT_CFG_REG,
- .min_u = 0,
- .max_u = 6000000,
- .step_u = 25000,
- },
- .ov = {
- .name = "battery over voltage",
- .reg = CHGR_BATTOV_CFG_REG,
- .min_u = 2450000,
- .max_u = 5000000,
- .step_u = 10000,
- },
- .usb_icl = {
- .name = "usb input current limit",
- .reg = USBIN_CURRENT_LIMIT_CFG_REG,
- .min_u = 100000,
- .max_u = 5000000,
- .step_u = 30000,
- },
- };
- struct smb_irq_info {
- const char *name;
- const irq_handler_t handler;
- const bool wake;
- int irq;
- };
- struct smb_dt_props {
- bool disable_ctm;
- int pl_mode;
- int pl_batfet_mode;
- bool hw_die_temp_mitigation;
- u32 die_temp_threshold;
- };
- struct smb1355 {
- struct device *dev;
- char *name;
- struct regmap *regmap;
- int max_fcc;
- struct smb_dt_props dt;
- struct smb_params param;
- struct mutex write_lock;
- struct mutex suspend_lock;
- struct power_supply *parallel_psy;
- struct iio_dev *indio_dev;
- struct iio_chan_spec *iio_chan;
- int d_health;
- int c_health;
- int c_charger_temp_max;
- int die_temp_deciDegC;
- int suspended_usb_icl;
- int charge_type;
- int vbatt_uv;
- int fcc_ua;
- bool exit_die_temp;
- struct delayed_work die_temp_work;
- bool disabled;
- bool suspended;
- bool charging_enabled;
- bool pin_status;
- struct votable *irq_disable_votable;
- struct votable *fcc_votable;
- struct votable *fv_votable;
- };
- enum {
- CONNECTOR_TEMP = 0,
- DIE_TEMP,
- };
- struct smb1355_iio_channel {
- const char *datasheet_name;
- int channel_num;
- enum iio_chan_type type;
- long info_mask;
- };
- #define PL_IIO_CHAN(_name, _num, _type, _mask) \
- { \
- .datasheet_name = _name, \
- .channel_num = _num, \
- .type = _type, \
- .info_mask = _mask, \
- },
- #define PL_CHAN_ENERGY(_name, _num) \
- PL_IIO_CHAN(_name, _num, IIO_ENERGY, \
- BIT(IIO_CHAN_INFO_PROCESSED))
- #define PL_CHAN_INDEX(_name, _num) \
- PL_IIO_CHAN(_name, _num, IIO_INDEX, \
- BIT(IIO_CHAN_INFO_PROCESSED))
- #define PL_CHAN_TEMP(_name, _num) \
- PL_IIO_CHAN(_name, _num, IIO_TEMP, \
- BIT(IIO_CHAN_INFO_PROCESSED))
- #define PL_CHAN_VOLT(_name, _num) \
- PL_IIO_CHAN(_name, _num, IIO_VOLTAGE, \
- BIT(IIO_CHAN_INFO_PROCESSED))
- #define PL_CHAN_CURR(_name, _num) \
- PL_IIO_CHAN(_name, _num, IIO_CURRENT, \
- BIT(IIO_CHAN_INFO_PROCESSED))
- static const struct smb1355_iio_channel smb1355_iio_channels[] = {
- PL_CHAN_ENERGY("charge_type", PSY_IIO_CHARGE_TYPE)
- PL_CHAN_INDEX("online", PSY_IIO_ONLINE)
- PL_CHAN_INDEX("charging_enabled", PSY_IIO_CHARGING_ENABLED)
- PL_CHAN_INDEX("pin_enabled", PSY_IIO_PIN_ENABLED)
- PL_CHAN_INDEX("input_suspend", PSY_IIO_INPUT_SUSPEND)
- PL_CHAN_TEMP("charger_temp", PSY_IIO_CHARGER_TEMP)
- PL_CHAN_TEMP("charger_temp_max", PSY_IIO_CHARGER_TEMP_MAX)
- PL_CHAN_VOLT("voltage_max", PSY_IIO_VOLTAGE_MAX)
- PL_CHAN_CURR("constant_charge_current_max",
- PSY_IIO_CONSTANT_CHARGE_CURRENT_MAX)
- PL_CHAN_INDEX("parallel_mode", PSY_IIO_PARALLEL_MODE)
- PL_CHAN_INDEX("connector_health", PSY_IIO_CONNECTOR_HEALTH)
- PL_CHAN_INDEX("parallel_batfet_mode", PSY_IIO_PARALLEL_BATFET_MODE)
- PL_CHAN_CURR("parallel_fcc_max", PSY_IIO_PARALLEL_FCC_MAX)
- PL_CHAN_CURR("input_current_limited", PSY_IIO_INPUT_CURRENT_LIMITED)
- PL_CHAN_CURR("min_icl", PSY_IIO_MIN_ICL)
- PL_CHAN_CURR("current_max", PSY_IIO_CURRENT_MAX)
- PL_CHAN_INDEX("set_ship_mode", PSY_IIO_SET_SHIP_MODE)
- PL_CHAN_INDEX("die_health", PSY_IIO_DIE_HEALTH)
- };
- static bool is_secure(struct smb1355 *chip, int addr)
- {
- if (addr == CLOCK_REQUEST_REG || addr == I2C_SS_DIG_PMIC_SID_REG)
- return true;
- /* assume everything above 0xA0 is secure */
- return (addr & 0xFF) >= 0xA0;
- }
- static bool is_voter_available(struct smb1355 *chip)
- {
- if (!chip->fcc_votable) {
- chip->fcc_votable = find_votable("FCC");
- if (!chip->fcc_votable) {
- pr_debug("Couldn't find FCC votable\n");
- return false;
- }
- }
- if (!chip->fv_votable) {
- chip->fv_votable = find_votable("FV");
- if (!chip->fv_votable) {
- pr_debug("Couldn't find FV votable\n");
- return false;
- }
- }
- return true;
- }
- static int smb1355_read(struct smb1355 *chip, u16 addr, u8 *val)
- {
- unsigned int temp;
- int rc;
- rc = regmap_read(chip->regmap, addr, &temp);
- if (rc >= 0)
- *val = (u8)temp;
- return rc;
- }
- static int smb1355_masked_force_write(struct smb1355 *chip, u16 addr, u8 mask,
- u8 val)
- {
- int rc;
- mutex_lock(&chip->write_lock);
- if (is_secure(chip, addr)) {
- rc = regmap_write(chip->regmap, (addr & 0xFF00) | 0xD0, 0xA5);
- if (rc < 0)
- goto unlock;
- }
- rc = regmap_write_bits(chip->regmap, addr, mask, val);
- unlock:
- mutex_unlock(&chip->write_lock);
- return rc;
- }
- static int smb1355_masked_write(struct smb1355 *chip, u16 addr, u8 mask, u8 val)
- {
- int rc;
- mutex_lock(&chip->write_lock);
- if (is_secure(chip, addr)) {
- rc = regmap_write(chip->regmap, (addr & 0xFF00) | 0xD0, 0xA5);
- if (rc < 0)
- goto unlock;
- }
- rc = regmap_update_bits(chip->regmap, addr, mask, val);
- unlock:
- mutex_unlock(&chip->write_lock);
- return rc;
- }
- static int smb1355_write(struct smb1355 *chip, u16 addr, u8 val)
- {
- int rc;
- mutex_lock(&chip->write_lock);
- if (is_secure(chip, addr)) {
- rc = regmap_write(chip->regmap, (addr & ~(0xFF)) | 0xD0, 0xA5);
- if (rc < 0)
- goto unlock;
- }
- rc = regmap_write(chip->regmap, addr, val);
- unlock:
- mutex_unlock(&chip->write_lock);
- return rc;
- }
- static int smb1355_set_charge_param(struct smb1355 *chip,
- struct smb_chg_param *param, int val_u)
- {
- int rc;
- u8 val_raw;
- if (val_u > param->max_u || val_u < param->min_u) {
- pr_err("%s: %d is out of range [%d, %d]\n",
- param->name, val_u, param->min_u, param->max_u);
- return -EINVAL;
- }
- val_raw = (val_u - param->min_u) / param->step_u;
- rc = smb1355_write(chip, param->reg, val_raw);
- if (rc < 0) {
- pr_err("%s: Couldn't write 0x%02x to 0x%04x rc=%d\n",
- param->name, val_raw, param->reg, rc);
- return rc;
- }
- return rc;
- }
- static int smb1355_get_charge_param(struct smb1355 *chip,
- struct smb_chg_param *param, int *val_u)
- {
- int rc;
- u8 val_raw;
- rc = smb1355_read(chip, param->reg, &val_raw);
- if (rc < 0) {
- pr_err("%s: Couldn't read from 0x%04x rc=%d\n",
- param->name, param->reg, rc);
- return rc;
- }
- *val_u = val_raw * param->step_u + param->min_u;
- return rc;
- }
- #define UB_COMP_OFFSET_DEGC 34
- #define DIE_TEMP_MEAS_PERIOD_MS 10000
- static void die_temp_work(struct work_struct *work)
- {
- struct smb1355 *chip = container_of(work, struct smb1355,
- die_temp_work.work);
- int rc, i;
- u8 temp_stat;
- for (i = 0; i < BIT(5); i++) {
- rc = smb1355_masked_write(chip, SMB2CHG_BATIF_ENG_SMISC_DIETEMP,
- TDIE_COMPARATOR_THRESHOLD, i);
- if (rc < 0) {
- pr_err("Couldn't set temp comp threshold rc=%d\n", rc);
- continue;
- }
- if (chip->exit_die_temp)
- return;
- /* wait for the comparator output to deglitch */
- msleep(100);
- rc = smb1355_read(chip, TEMP_COMP_STATUS_REG, &temp_stat);
- if (rc < 0) {
- pr_err("Couldn't read temp comp status rc=%d\n", rc);
- continue;
- }
- if (!(temp_stat & TEMP_UB_HOT_BIT)) {
- /* found the temp */
- break;
- }
- }
- chip->die_temp_deciDegC = 10 * (i + UB_COMP_OFFSET_DEGC);
- schedule_delayed_work(&chip->die_temp_work,
- msecs_to_jiffies(DIE_TEMP_MEAS_PERIOD_MS));
- }
- static int smb1355_get_prop_input_current_limited(struct smb1355 *chip,
- int *val)
- {
- int rc;
- u8 stat = 0;
- rc = smb1355_read(chip, MISC_RT_STS_REG, &stat);
- if (rc < 0)
- pr_err("Couldn't read SMB1355_BATTERY_STATUS_3 rc=%d\n", rc);
- *val = !!(stat & HARD_ILIMIT_RT_STS_BIT);
- return 0;
- }
- static irqreturn_t smb1355_handle_chg_state_change(int irq, void *data)
- {
- struct smb1355 *chip = data;
- if (chip->parallel_psy)
- power_supply_changed(chip->parallel_psy);
- return IRQ_HANDLED;
- }
- static irqreturn_t smb1355_handle_wdog_bark(int irq, void *data)
- {
- struct smb1355 *chip = data;
- int rc;
- rc = smb1355_write(chip, BARK_BITE_WDOG_PET_REG,
- BARK_BITE_WDOG_PET_BIT);
- if (rc < 0)
- pr_err("Couldn't pet the dog rc=%d\n", rc);
- return IRQ_HANDLED;
- }
- static irqreturn_t smb1355_handle_temperature_change(int irq, void *data)
- {
- struct smb1355 *chip = data;
- if (chip->parallel_psy)
- power_supply_changed(chip->parallel_psy);
- return IRQ_HANDLED;
- }
- static int smb1355_determine_initial_status(struct smb1355 *chip)
- {
- smb1355_handle_temperature_change(0, chip);
- return 0;
- }
- #define DEFAULT_DIE_TEMP_LOW_THRESHOLD 90
- static int smb1355_parse_dt(struct smb1355 *chip)
- {
- struct device_node *node = chip->dev->of_node;
- int rc = 0;
- if (!node) {
- pr_err("device tree node missing\n");
- return -EINVAL;
- }
- chip->dt.disable_ctm =
- of_property_read_bool(node, "qcom,disable-ctm");
- /*
- * If parallel-mode property is not present default
- * parallel configuration is USBMID-USBMID.
- */
- rc = of_property_read_u32(node,
- "qcom,parallel-mode", &chip->dt.pl_mode);
- if (rc < 0)
- chip->dt.pl_mode = QTI_POWER_SUPPLY_PL_USBMID_USBMID;
- /*
- * If stacked-batfet property is not present default
- * configuration is NON-STACKED-BATFET.
- */
- chip->dt.pl_batfet_mode = QTI_POWER_SUPPLY_PL_NON_STACKED_BATFET;
- if (of_property_read_bool(node, "qcom,stacked-batfet"))
- chip->dt.pl_batfet_mode = QTI_POWER_SUPPLY_PL_STACKED_BATFET;
- chip->dt.hw_die_temp_mitigation = of_property_read_bool(node,
- "qcom,hw-die-temp-mitigation");
- chip->dt.die_temp_threshold = DEFAULT_DIE_TEMP_LOW_THRESHOLD;
- of_property_read_u32(node, "qcom,die-temp-threshold-degc",
- &chip->dt.die_temp_threshold);
- if (chip->dt.die_temp_threshold > DIE_LOW_RANGE_MAX_DEGC)
- chip->dt.die_temp_threshold = DIE_LOW_RANGE_MAX_DEGC;
- return 0;
- }
- static int smb1355_get_prop_batt_charge_type(struct smb1355 *chip,
- int *val)
- {
- int rc;
- u8 stat;
- rc = smb1355_read(chip, BATTERY_STATUS_3_REG, &stat);
- if (rc < 0) {
- pr_err("Couldn't read SMB1355_BATTERY_STATUS_3 rc=%d\n", rc);
- return rc;
- }
- if (stat & ENABLE_CHARGING_BIT) {
- if (stat & BATT_GT_PRE_TO_FAST_BIT)
- *val = POWER_SUPPLY_CHARGE_TYPE_FAST;
- else
- *val = POWER_SUPPLY_CHARGE_TYPE_TRICKLE;
- } else {
- *val = POWER_SUPPLY_CHARGE_TYPE_NONE;
- }
- return rc;
- }
- static int smb1355_get_prop_health(struct smb1355 *chip, int type)
- {
- u8 temp;
- int rc, shift;
- /* Connector-temp uses skin-temp configuration */
- shift = (type == CONNECTOR_TEMP) ? SKIN_TEMP_SHIFT : 0;
- rc = smb1355_read(chip, TEMP_COMP_STATUS_REG, &temp);
- if (rc < 0) {
- pr_err("Couldn't read comp stat reg rc = %d\n", rc);
- return POWER_SUPPLY_HEALTH_UNKNOWN;
- }
- if (temp & (TEMP_RST_HOT_BIT << shift))
- return POWER_SUPPLY_HEALTH_OVERHEAT;
- if (temp & (TEMP_UB_HOT_BIT << shift))
- return POWER_SUPPLY_HEALTH_HOT;
- if (temp & (TEMP_LB_HOT_BIT << shift))
- return POWER_SUPPLY_HEALTH_WARM;
- return POWER_SUPPLY_HEALTH_COOL;
- }
- static int smb1355_get_prop_voltage_max(struct smb1355 *chip, int *val)
- {
- int rc = 0;
- mutex_lock(&chip->suspend_lock);
- if (chip->suspended) {
- *val = chip->vbatt_uv;
- goto done;
- }
- rc = smb1355_get_charge_param(chip, &chip->param.ov, val);
- if (rc < 0)
- pr_err("failed to read vbatt rc=%d\n", rc);
- else
- chip->vbatt_uv = *val;
- done:
- mutex_unlock(&chip->suspend_lock);
- return rc;
- }
- static int smb1355_get_prop_constant_charge_current_max(struct smb1355 *chip,
- int *val)
- {
- int rc = 0;
- mutex_lock(&chip->suspend_lock);
- if (chip->suspended) {
- *val = chip->fcc_ua;
- goto done;
- }
- rc = smb1355_get_charge_param(chip, &chip->param.fcc, val);
- if (rc < 0)
- pr_err("failed to read fcc rc=%d\n", rc);
- else
- chip->fcc_ua = *val;
- done:
- mutex_unlock(&chip->suspend_lock);
- return rc;
- }
- static int smb1355_get_prop_health_value(struct smb1355 *chip,
- int *val, int type)
- {
- mutex_lock(&chip->suspend_lock);
- if (chip->suspended) {
- *val = (type == DIE_TEMP) ? chip->d_health :
- chip->c_health;
- } else {
- *val = smb1355_get_prop_health(chip, (type == DIE_TEMP) ?
- DIE_TEMP : CONNECTOR_TEMP);
- if (type == DIE_TEMP)
- chip->d_health = *val;
- else
- chip->c_health = *val;
- }
- mutex_unlock(&chip->suspend_lock);
- return 0;
- }
- static int smb1355_get_prop_online(struct smb1355 *chip, int *val)
- {
- int rc = 0;
- u8 stat;
- mutex_lock(&chip->suspend_lock);
- if (chip->suspended) {
- *val = chip->charging_enabled;
- goto done;
- }
- rc = smb1355_read(chip, BATTERY_STATUS_3_REG, &stat);
- if (rc < 0) {
- pr_err("failed to read BATTERY_STATUS_3_REG %d\n", rc);
- } else {
- *val = (bool)(stat & ENABLE_CHARGING_BIT);
- chip->charging_enabled = *val;
- }
- done:
- mutex_unlock(&chip->suspend_lock);
- return rc;
- }
- static int smb1355_get_prop_pin_enabled(struct smb1355 *chip, int *val)
- {
- int rc = 0;
- u8 stat;
- mutex_lock(&chip->suspend_lock);
- if (chip->suspended) {
- *val = chip->pin_status;
- goto done;
- }
- rc = smb1355_read(chip, BATTERY_STATUS_2_REG, &stat);
- if (rc < 0) {
- pr_err("failed to read BATTERY_STATUS_2_REG %d\n", rc);
- } else {
- *val = !(stat & DISABLE_CHARGING_BIT);
- chip->pin_status = *val;
- }
- done:
- mutex_unlock(&chip->suspend_lock);
- return rc;
- }
- static int smb1355_get_prop_charge_type(struct smb1355 *chip, int *val)
- {
- int rc = 0;
- /*
- * In case of system suspend we should not allow
- * register reads and writes to the device as it
- * leads to i2c transaction failures.
- */
- mutex_lock(&chip->suspend_lock);
- if (chip->suspended) {
- *val = chip->charge_type;
- goto done;
- }
- rc = smb1355_get_prop_batt_charge_type(chip, val);
- if (rc < 0)
- pr_err("failed to read batt_charge_type %d\n", rc);
- else
- chip->charge_type = *val;
- done:
- mutex_unlock(&chip->suspend_lock);
- return rc;
- }
- #define MIN_PARALLEL_ICL_UA 250000
- #define SUSPEND_CURRENT_UA 2000
- static int smb1355_set_parallel_charging(struct smb1355 *chip, bool disable)
- {
- int rc;
- if (chip->disabled == disable)
- return 0;
- if (IS_USBIN(chip->dt.pl_mode)) {
- /*
- * Initialize ICL configuration to minimum value while
- * depending upon the set icl configuration method to properly
- * configure the ICL value. At the same time, cache the value
- * of ICL to be reported as 2mA.
- */
- chip->suspended_usb_icl = SUSPEND_CURRENT_UA;
- smb1355_set_charge_param(chip,
- &chip->param.usb_icl, MIN_PARALLEL_ICL_UA);
- }
- rc = smb1355_masked_write(chip, WD_CFG_REG, WDOG_TIMER_EN_BIT,
- disable ? 0 : WDOG_TIMER_EN_BIT);
- if (rc < 0) {
- pr_err("Couldn't %s watchdog rc=%d\n",
- disable ? "disable" : "enable", rc);
- disable = true;
- }
- /*
- * Configure charge enable for high polarity and
- * When disabling charging set it to cmd register control(cmd bit=0)
- * When enabling charging set it to pin control
- */
- rc = smb1355_masked_write(chip, CHGR_CFG2_REG,
- CHG_EN_POLARITY_BIT | CHG_EN_SRC_BIT,
- disable ? 0 : CHG_EN_SRC_BIT);
- if (rc < 0) {
- pr_err("Couldn't configure charge enable source rc=%d\n", rc);
- disable = true;
- }
- chip->die_temp_deciDegC = -EINVAL;
- /* Only enable temperature measurement for s/w based mitigation */
- if (!chip->dt.hw_die_temp_mitigation) {
- if (disable) {
- chip->exit_die_temp = true;
- cancel_delayed_work_sync(&chip->die_temp_work);
- } else {
- /* start the work to measure temperature */
- chip->exit_die_temp = false;
- schedule_delayed_work(&chip->die_temp_work, 0);
- }
- }
- if (chip->irq_disable_votable)
- vote(chip->irq_disable_votable, PARALLEL_ENABLE_VOTER,
- disable, 0);
- rc = smb1355_masked_write(chip, BANDGAP_ENABLE_REG,
- BANDGAP_ENABLE_CMD_BIT,
- disable ? 0 : BANDGAP_ENABLE_CMD_BIT);
- if (rc < 0) {
- pr_err("Couldn't configure bandgap enable rc=%d\n", rc);
- return rc;
- }
- chip->disabled = disable;
- return 0;
- }
- static int smb1355_set_current_max(struct smb1355 *chip, int curr)
- {
- int rc = 0;
- if (!IS_USBIN(chip->dt.pl_mode))
- return 0;
- if ((curr / 1000) < 100) {
- /* disable parallel path (ICL < 100mA) */
- rc = smb1355_set_parallel_charging(chip, true);
- } else {
- rc = smb1355_set_parallel_charging(chip, false);
- if (rc < 0)
- return rc;
- rc = smb1355_set_charge_param(chip,
- &chip->param.usb_icl, curr);
- chip->suspended_usb_icl = 0;
- }
- return rc;
- }
- static int smb1355_clk_request(struct smb1355 *chip, bool enable)
- {
- int rc;
- rc = smb1355_masked_force_write(chip, CLOCK_REQUEST_REG,
- CLOCK_REQUEST_CMD_BIT,
- enable ? CLOCK_REQUEST_CMD_BIT : 0);
- if (rc < 0)
- pr_err("Couldn't %s clock rc=%d\n",
- enable ? "enable" : "disable", rc);
- return rc;
- }
- static int smb1355_parallel_get_prop(struct power_supply *psy,
- enum power_supply_property prop,
- union power_supply_propval *pval)
- {
- struct smb1355 *chip = power_supply_get_drvdata(psy);
- switch (prop) {
- case POWER_SUPPLY_PROP_MODEL_NAME:
- pval->strval = chip->name;
- break;
- default:
- pr_err_ratelimited("parallel psy get prop %d not supported\n",
- prop);
- return -EINVAL;
- }
- return 0;
- }
- /*****************************
- * PARALLEL PSY REGISTRATION *
- *****************************/
- static enum power_supply_property smb1355_parallel_props[] = {
- POWER_SUPPLY_PROP_MODEL_NAME,
- };
- static struct power_supply_desc parallel_psy_desc = {
- .name = "parallel",
- .type = POWER_SUPPLY_TYPE_MAINS,
- .properties = smb1355_parallel_props,
- .num_properties = ARRAY_SIZE(smb1355_parallel_props),
- .get_property = smb1355_parallel_get_prop,
- };
- static int smb1355_init_parallel_psy(struct smb1355 *chip)
- {
- struct power_supply_config parallel_cfg = {};
- parallel_cfg.drv_data = chip;
- parallel_cfg.of_node = chip->dev->of_node;
- /* change to smb1355's property list */
- parallel_psy_desc.properties = smb1355_parallel_props;
- parallel_psy_desc.num_properties = ARRAY_SIZE(smb1355_parallel_props);
- chip->parallel_psy = devm_power_supply_register(chip->dev,
- ¶llel_psy_desc,
- ¶llel_cfg);
- if (IS_ERR(chip->parallel_psy)) {
- pr_err("Couldn't register parallel power supply\n");
- return PTR_ERR(chip->parallel_psy);
- }
- return 0;
- }
- static int smb1355_iio_read_raw(struct iio_dev *indio_dev,
- struct iio_chan_spec const *chan,
- int *val1, int *val2, long mask)
- {
- struct smb1355 *chip = iio_priv(indio_dev);
- int rc = 0;
- *val1 = 0;
- switch (chan->channel) {
- case PSY_IIO_CHARGE_TYPE:
- rc = smb1355_get_prop_charge_type(chip, val1);
- break;
- case PSY_IIO_CHARGING_ENABLED:
- case PSY_IIO_ONLINE:
- rc = smb1355_get_prop_online(chip, val1);
- break;
- case PSY_IIO_PIN_ENABLED:
- rc = smb1355_get_prop_pin_enabled(chip, val1);
- break;
- case PSY_IIO_CHARGER_TEMP:
- *val1 = chip->die_temp_deciDegC;
- break;
- case PSY_IIO_CHARGER_TEMP_MAX:
- /*
- * In case of h/w controlled die_temp mitigation,
- * die_temp/die_temp_max can not be reported as this
- * requires run time manipulation of DIE_TEMP low
- * threshold which will interfere with h/w mitigation
- * scheme.
- */
- if (chip->dt.hw_die_temp_mitigation)
- *val1 = -EINVAL;
- else
- *val1 = chip->c_charger_temp_max;
- break;
- case PSY_IIO_INPUT_SUSPEND:
- *val1 = chip->disabled;
- break;
- case PSY_IIO_VOLTAGE_MAX:
- rc = smb1355_get_prop_voltage_max(chip, val1);
- break;
- case PSY_IIO_CONSTANT_CHARGE_CURRENT_MAX:
- rc = smb1355_get_prop_constant_charge_current_max(chip, val1);
- break;
- case PSY_IIO_PARALLEL_MODE:
- *val1 = chip->dt.pl_mode;
- break;
- case PSY_IIO_CONNECTOR_HEALTH:
- if (chip->c_health == -EINVAL)
- rc = smb1355_get_prop_health_value(chip, val1,
- CONNECTOR_TEMP);
- else
- *val1 = chip->c_health;
- break;
- case PSY_IIO_DIE_HEALTH:
- rc = smb1355_get_prop_health_value(chip, val1, DIE_TEMP);
- break;
- case PSY_IIO_PARALLEL_BATFET_MODE:
- *val1 = chip->dt.pl_batfet_mode;
- break;
- case PSY_IIO_INPUT_CURRENT_LIMITED:
- if (IS_USBIN(chip->dt.pl_mode))
- rc = smb1355_get_prop_input_current_limited(
- chip, val1);
- else
- *val1 = 0;
- break;
- case PSY_IIO_CURRENT_MAX:
- if (IS_USBIN(chip->dt.pl_mode)) {
- /* Report cached ICL until its configured correctly */
- if (chip->suspended_usb_icl)
- *val1 = chip->suspended_usb_icl;
- else
- rc = smb1355_get_charge_param(chip,
- &chip->param.usb_icl, val1);
- } else {
- *val1 = 0;
- }
- break;
- case PSY_IIO_MIN_ICL:
- *val1 = MIN_PARALLEL_ICL_UA;
- break;
- case PSY_IIO_PARALLEL_FCC_MAX:
- *val1 = chip->max_fcc;
- break;
- case PSY_IIO_SET_SHIP_MODE:
- /* Not in ship mode as long as device is active */
- *val1 = 0;
- break;
- default:
- pr_err_ratelimited("SMB1355 IIO channel %x not supported\n",
- chan->channel);
- return -EINVAL;
- }
- if (rc < 0) {
- pr_debug("Couldn't read channel %x rc = %d\n",
- chan->channel, rc);
- return -ENODATA;
- }
- return IIO_VAL_INT;
- }
- static int smb1355_iio_write_raw(struct iio_dev *indio_dev,
- struct iio_chan_spec const *chan,
- int val1, int val2, long mask)
- {
- struct smb1355 *chip = iio_priv(indio_dev);
- int rc = 0;
- mutex_lock(&chip->suspend_lock);
- if (chip->suspended) {
- pr_debug("SMB1355 IIO write channel %d\n",
- chan->channel);
- goto done;
- }
- switch (chan->channel) {
- case PSY_IIO_INPUT_SUSPEND:
- rc = smb1355_set_parallel_charging(chip, (bool)val1);
- break;
- case PSY_IIO_CURRENT_MAX:
- rc = smb1355_set_current_max(chip, val1);
- break;
- case PSY_IIO_VOLTAGE_MAX:
- rc = smb1355_set_charge_param(chip, &chip->param.ov,
- val1);
- break;
- case PSY_IIO_CONSTANT_CHARGE_CURRENT_MAX:
- rc = smb1355_set_charge_param(chip, &chip->param.fcc,
- val1);
- break;
- case PSY_IIO_CONNECTOR_HEALTH:
- chip->c_health = val1;
- power_supply_changed(chip->parallel_psy);
- break;
- case PSY_IIO_CHARGER_TEMP_MAX:
- chip->c_charger_temp_max = val1;
- break;
- case PSY_IIO_SET_SHIP_MODE:
- if (!val1)
- break;
- rc = smb1355_clk_request(chip, false);
- break;
- default:
- rc = -EINVAL;
- }
- if (rc < 0)
- pr_debug("Couldn't write to channel %x rc = %d\n",
- chan->channel, rc);
- done:
- mutex_unlock(&chip->suspend_lock);
- return rc;
- }
- static int smb1355_iio_fwnode_xlate(struct iio_dev *indio_dev,
- const struct fwnode_reference_args *iiospec)
- {
- struct smb1355 *chip = iio_priv(indio_dev);
- struct iio_chan_spec *iio_chan = chip->iio_chan;
- int i;
- for (i = 0; i < ARRAY_SIZE(smb1355_iio_channels);
- i++, iio_chan++)
- if (iio_chan->channel == iiospec->args[0])
- return i;
- return -EINVAL;
- }
- static const struct iio_info smb1355_iio_info = {
- .read_raw = smb1355_iio_read_raw,
- .write_raw = smb1355_iio_write_raw,
- .fwnode_xlate = smb1355_iio_fwnode_xlate,
- };
- static int smb1355_init_iio_psy(struct smb1355 *chip)
- {
- struct iio_dev *indio_dev = chip->indio_dev;
- struct iio_chan_spec *chan;
- int smb1355_num_iio_channels = ARRAY_SIZE(smb1355_iio_channels);
- int rc, i;
- chip->iio_chan = devm_kcalloc(chip->dev,
- smb1355_num_iio_channels,
- sizeof(*chip->iio_chan),
- GFP_KERNEL);
- if (!chip->iio_chan)
- return -ENOMEM;
- indio_dev->info = &smb1355_iio_info;
- indio_dev->dev.parent = chip->dev;
- indio_dev->dev.of_node = chip->dev->of_node;
- indio_dev->modes = INDIO_DIRECT_MODE;
- indio_dev->channels = chip->iio_chan;
- indio_dev->num_channels = smb1355_num_iio_channels;
- indio_dev->name = "smb1355-charger";
- for (i = 0; i < smb1355_num_iio_channels; i++) {
- chan = &chip->iio_chan[i];
- chan->address = i;
- chan->channel = smb1355_iio_channels[i].channel_num;
- chan->type = smb1355_iio_channels[i].type;
- chan->datasheet_name =
- smb1355_iio_channels[i].datasheet_name;
- chan->extend_name =
- smb1355_iio_channels[i].datasheet_name;
- chan->info_mask_separate =
- smb1355_iio_channels[i].info_mask;
- }
- rc = devm_iio_device_register(chip->dev, indio_dev);
- if (rc) {
- pr_err("Failed to register SMB1355 Parallel IIO device, rc=%d\n",
- rc);
- return rc;
- }
- return rc;
- }
- /***************************
- * HARDWARE INITIALIZATION *
- ***************************/
- #define MFG_ID_SMB1354 0x01
- #define MFG_ID_SMB1355 0xFF
- #define SMB1354_MAX_PARALLEL_FCC_UA 2500000
- #define SMB1355_MAX_PARALLEL_FCC_UA 6000000
- static int smb1355_detect_version(struct smb1355 *chip)
- {
- int rc;
- u8 val;
- rc = smb1355_read(chip, REVID_MFG_ID_SPARE_REG, &val);
- if (rc < 0) {
- pr_err("Unable to read REVID rc=%d\n", rc);
- return rc;
- }
- switch (val) {
- case MFG_ID_SMB1354:
- chip->name = "smb1354";
- chip->max_fcc = SMB1354_MAX_PARALLEL_FCC_UA;
- break;
- case MFG_ID_SMB1355:
- chip->name = "smb1355";
- chip->max_fcc = SMB1355_MAX_PARALLEL_FCC_UA;
- break;
- default:
- pr_err("Invalid value of REVID val=%d\n", val);
- return -EINVAL;
- }
- return rc;
- }
- static int smb1355_tskin_sensor_config(struct smb1355 *chip)
- {
- int rc;
- if (chip->dt.disable_ctm) {
- /*
- * the TSKIN sensor with external resistor needs a bias,
- * disable it here.
- */
- rc = smb1355_masked_write(chip, BATIF_ENG_SCMISC_SPARE1_REG,
- EXT_BIAS_PIN_BIT, 0);
- if (rc < 0) {
- pr_err("Couldn't enable ext bias pin path rc=%d\n",
- rc);
- return rc;
- }
- rc = smb1355_masked_write(chip, BATIF_CFG_SMISC_BATID_REG,
- CFG_SMISC_RBIAS_EXT_CTRL_BIT, 0);
- if (rc < 0) {
- pr_err("Couldn't set BATIF_CFG_SMISC_BATID rc=%d\n",
- rc);
- return rc;
- }
- rc = smb1355_masked_write(chip, MISC_CHGR_TRIM_OPTIONS_REG,
- CMD_RBIAS_EN_BIT, 0);
- if (rc < 0) {
- pr_err("Couldn't set MISC_CHGR_TRIM_OPTIONS rc=%d\n",
- rc);
- return rc;
- }
- /* disable skin temperature comparator source */
- rc = smb1355_masked_write(chip, MISC_THERMREG_SRC_CFG_REG,
- THERMREG_SKIN_CMP_SRC_EN_BIT, 0);
- if (rc < 0) {
- pr_err("Couldn't set Skin temp comparator src rc=%d\n",
- rc);
- return rc;
- }
- } else {
- /*
- * the TSKIN sensor with external resistor needs a bias,
- * enable it here.
- */
- rc = smb1355_masked_write(chip, BATIF_ENG_SCMISC_SPARE1_REG,
- EXT_BIAS_PIN_BIT, EXT_BIAS_PIN_BIT);
- if (rc < 0) {
- pr_err("Couldn't enable ext bias pin path rc=%d\n",
- rc);
- return rc;
- }
- rc = smb1355_masked_write(chip, BATIF_CFG_SMISC_BATID_REG,
- CFG_SMISC_RBIAS_EXT_CTRL_BIT,
- CFG_SMISC_RBIAS_EXT_CTRL_BIT);
- if (rc < 0) {
- pr_err("Couldn't set BATIF_CFG_SMISC_BATID rc=%d\n",
- rc);
- return rc;
- }
- rc = smb1355_masked_write(chip, MISC_CHGR_TRIM_OPTIONS_REG,
- CMD_RBIAS_EN_BIT,
- CMD_RBIAS_EN_BIT);
- if (rc < 0) {
- pr_err("Couldn't set MISC_CHGR_TRIM_OPTIONS rc=%d\n",
- rc);
- return rc;
- }
- /* Enable skin temperature comparator source */
- rc = smb1355_masked_write(chip, MISC_THERMREG_SRC_CFG_REG,
- THERMREG_SKIN_CMP_SRC_EN_BIT,
- THERMREG_SKIN_CMP_SRC_EN_BIT);
- if (rc < 0) {
- pr_err("Couldn't set Skin temp comparator src rc=%d\n",
- rc);
- return rc;
- }
- }
- return rc;
- }
- static int smb1355_init_hw(struct smb1355 *chip)
- {
- int rc;
- u8 val, range;
- /* request clock always on */
- rc = smb1355_clk_request(chip, true);
- if (rc < 0)
- return rc;
- /* Change to let SMB1355 only respond to address 0x0C */
- rc = smb1355_masked_write(chip, I2C_SS_DIG_PMIC_SID_REG,
- PMIC_SID_MASK, PMIC_SID0_BIT);
- if (rc < 0) {
- pr_err("Couldn't configure the I2C_SS_DIG_PMIC_SID_REG rc=%d\n",
- rc);
- return rc;
- }
- /* disable charging when watchdog bites & set bite-timeout to 8secs */
- val = BITE_WDOG_DISABLE_CHARGING_CFG_BIT | 0x3;
- rc = smb1355_masked_write(chip, SNARL_BARK_BITE_WD_CFG_REG,
- BITE_WDOG_DISABLE_CHARGING_CFG_BIT |
- BITE_WDOG_TIMEOUT_MASK, val);
- if (rc < 0) {
- pr_err("Couldn't configure the watchdog bite rc=%d\n", rc);
- return rc;
- }
- /* enable watchdog bark and bite interrupts, and disable the watchdog */
- rc = smb1355_masked_write(chip, WD_CFG_REG, WDOG_TIMER_EN_BIT
- | WDOG_TIMER_EN_ON_PLUGIN_BIT | BITE_WDOG_INT_EN_BIT
- | BARK_WDOG_INT_EN_BIT,
- BITE_WDOG_INT_EN_BIT | BARK_WDOG_INT_EN_BIT);
- if (rc < 0) {
- pr_err("Couldn't configure the watchdog rc=%d\n", rc);
- return rc;
- }
- /*
- * Disable command based SMB1355 enablement and disable parallel
- * charging path by switching to command based mode.
- */
- rc = smb1355_masked_write(chip, CHGR_CHARGING_ENABLE_CMD_REG,
- CHARGING_ENABLE_CMD_BIT, 0);
- if (rc < 0) {
- pr_err("Coudln't configure command bit, rc=%d\n", rc);
- return rc;
- }
- rc = smb1355_set_parallel_charging(chip, true);
- if (rc < 0) {
- pr_err("Couldn't disable parallel path rc=%d\n", rc);
- return rc;
- }
- /* initialize FCC to 0 */
- rc = smb1355_set_charge_param(chip, &chip->param.fcc, 0);
- if (rc < 0) {
- pr_err("Couldn't set 0 FCC rc=%d\n", rc);
- return rc;
- }
- /* HICCUP setting, unlimited retry with 250ms interval */
- rc = smb1355_masked_write(chip, POWER_MODE_HICCUP_CFG,
- HICCUP_TIMEOUT_CFG_MASK | MAX_HICCUP_DUETO_BATDIS_MASK,
- 0);
- if (rc < 0) {
- pr_err("Couldn't set HICCUP interval rc=%d\n",
- rc);
- return rc;
- }
- /* enable parallel current sensing */
- rc = smb1355_masked_write(chip, CFG_REG,
- VCHG_EN_CFG_BIT, VCHG_EN_CFG_BIT);
- if (rc < 0) {
- pr_err("Couldn't enable parallel current sensing rc=%d\n",
- rc);
- return rc;
- }
- /* set Pre-to-Fast Charging Threshold 2.6V */
- rc = smb1355_masked_write(chip, CHGR_PRE_TO_FAST_THRESHOLD_CFG_REG,
- PRE_TO_FAST_CHARGE_THRESHOLD_MASK, 0);
- if (rc < 0) {
- pr_err("Couldn't set PRE_TO_FAST_CHARGE_THRESHOLD rc=%d\n",
- rc);
- return rc;
- }
- /* Extend min-offtime same as blanking time */
- rc = smb1355_masked_write(chip, MISC_ENG_SDCDC_RESERVE1_REG,
- MINOFF_TIME_MASK, 0);
- if (rc < 0) {
- pr_err("Couldn't set MINOFF_TIME rc=%d\n", rc);
- return rc;
- }
- /* Set dead-time to 32ns */
- rc = smb1355_masked_write(chip, MISC_ENG_SDCDC_CFG8_REG,
- DEAD_TIME_MASK, DEAD_TIME_32NS);
- if (rc < 0) {
- pr_err("Couldn't set DEAD_TIME to 32ns rc=%d\n", rc);
- return rc;
- }
- /* Configure DIE temp Low threshold */
- if (chip->dt.hw_die_temp_mitigation) {
- range = (chip->dt.die_temp_threshold - DIE_LOW_RANGE_BASE_DEGC)
- / (DIE_LOW_RANGE_DELTA);
- val = (chip->dt.die_temp_threshold
- - ((range * DIE_LOW_RANGE_DELTA)
- + DIE_LOW_RANGE_BASE_DEGC))
- % DIE_LOW_RANGE_DELTA;
- rc = smb1355_masked_write(chip, SMB2CHG_BATIF_ENG_SMISC_DIETEMP,
- TDIE_COMPARATOR_THRESHOLD,
- (range << DIE_LOW_RANGE_SHIFT) | val);
- if (rc < 0) {
- pr_err("Couldn't set temp comp threshold rc=%d\n", rc);
- return rc;
- }
- }
- /*
- * Enable thermal Die temperature comparator source and
- * enable hardware controlled current adjustment for die temp
- * if charger is configured in h/w controlled die temp mitigation.
- */
- val = THERMREG_DIE_CMP_SRC_EN_BIT;
- if (!chip->dt.hw_die_temp_mitigation)
- val |= BYP_THERM_CHG_CURR_ADJUST_BIT;
- rc = smb1355_masked_write(chip, MISC_THERMREG_SRC_CFG_REG,
- THERMREG_DIE_CMP_SRC_EN_BIT | BYP_THERM_CHG_CURR_ADJUST_BIT,
- val);
- if (rc < 0) {
- pr_err("Couldn't set Skin temperature comparator src rc=%d\n",
- rc);
- return rc;
- }
- /*
- * Disable hysterisis for die temperature. This is so that sw can run
- * stepping scheme quickly
- */
- val = chip->dt.hw_die_temp_mitigation ? DIE_TEMP_COMP_HYST_BIT : 0;
- rc = smb1355_masked_write(chip, BATIF_ENG_SCMISC_SPARE1_REG,
- DIE_TEMP_COMP_HYST_BIT, val);
- if (rc < 0) {
- pr_err("Couldn't disable hyst. for die rc=%d\n", rc);
- return rc;
- }
- /* Enable valley current comparator all the time */
- rc = smb1355_masked_write(chip, ANA1_ENG_SREFGEN_CFG2_REG,
- VALLEY_COMPARATOR_EN_BIT, VALLEY_COMPARATOR_EN_BIT);
- if (rc < 0) {
- pr_err("Couldn't enable valley current comparator rc=%d\n", rc);
- return rc;
- }
- /* Set LS_VALLEY threshold to 85% */
- rc = smb1355_masked_write(chip, MISC_CUST_SDCDC_ILIMIT_CFG_REG,
- LS_VALLEY_THRESH_PCT_BIT, LS_VALLEY_THRESH_PCT_BIT);
- if (rc < 0) {
- pr_err("Couldn't set LS valley threshold to 85pc rc=%d\n", rc);
- return rc;
- }
- /* For SMB1354, set PCL to 8.6 A */
- if (!strcmp(chip->name, "smb1354")) {
- rc = smb1355_masked_write(chip, MISC_CUST_SDCDC_ILIMIT_CFG_REG,
- PCL_LIMIT_MASK, PCL_LIMIT_MASK);
- if (rc < 0) {
- pr_err("Couldn't set PCL limit to 8.6A rc=%d\n", rc);
- return rc;
- }
- }
- rc = smb1355_tskin_sensor_config(chip);
- if (rc < 0) {
- pr_err("Couldn't configure tskin regs rc=%d\n", rc);
- return rc;
- }
- /* USBIN-USBIN configuration */
- if (IS_USBIN(chip->dt.pl_mode)) {
- /* set swicther clock frequency to 700kHz */
- rc = smb1355_masked_write(chip, MISC_CUST_SDCDC_CLK_CFG_REG,
- SWITCHER_CLK_FREQ_MASK, 0x03);
- if (rc < 0) {
- pr_err("Couldn't set MISC_CUST_SDCDC_CLK_CFG rc=%d\n",
- rc);
- return rc;
- }
- /*
- * configure compensation for input current limit (ICL) loop
- * accuracy, scale slope compensation using 30k resistor.
- */
- rc = smb1355_masked_write(chip, MISC_ENG_SDCDC_RESERVE3_REG,
- II_SOURCE_BIT | SCALE_SLOPE_COMP_MASK,
- II_SOURCE_BIT);
- if (rc < 0) {
- pr_err("Couldn't set MISC_ENG_SDCDC_RESERVE3_REG rc=%d\n",
- rc);
- return rc;
- }
- /* configuration to improve ICL accuracy */
- rc = smb1355_masked_write(chip,
- MISC_ENG_SDCDC_INPUT_CURRENT_CFG1_REG,
- PROLONG_ISENSE_MASK | SAMPLE_HOLD_DELAY_MASK,
- ((uint8_t)0x0C << SAMPLE_HOLD_DELAY_SHIFT));
- if (rc < 0) {
- pr_err("Couldn't set MISC_ENG_SDCDC_INPUT_CURRENT_CFG1_REG rc=%d\n",
- rc);
- return rc;
- }
- rc = smb1355_masked_write(chip,
- MISC_ENG_SDCDC_INPUT_CURRENT_CFG2_REG,
- INPUT_CURRENT_LIMIT_SOURCE_BIT
- | HS_II_CORRECTION_MASK,
- INPUT_CURRENT_LIMIT_SOURCE_BIT | 0xC);
- if (rc < 0) {
- pr_err("Couldn't set MISC_ENG_SDCDC_INPUT_CURRENT_CFG2_REG rc=%d\n",
- rc);
- return rc;
- }
- /* configure DAC offset */
- rc = smb1355_masked_write(chip,
- ANA2_TR_SBQ_ICL_1X_REF_OFFSET_REG,
- TR_SBQ_ICL_1X_REF_OFFSET, 0x00);
- if (rc < 0) {
- pr_err("Couldn't set ANA2_TR_SBQ_ICL_1X_REF_OFFSET_REG rc=%d\n",
- rc);
- return rc;
- }
- /* configure DAC gain */
- rc = smb1355_masked_write(chip, USB_TR_SCPATH_ICL_1X_GAIN_REG,
- TR_SCPATH_ICL_1X_GAIN_MASK, 0x22);
- if (rc < 0) {
- pr_err("Couldn't set USB_TR_SCPATH_ICL_1X_GAIN_REG rc=%d\n",
- rc);
- return rc;
- }
- }
- return 0;
- }
- /**************************
- * INTERRUPT REGISTRATION *
- **************************/
- static struct smb_irq_info smb1355_irqs[] = {
- [0] = {
- .name = "wdog-bark",
- .handler = smb1355_handle_wdog_bark,
- .wake = true,
- },
- [1] = {
- .name = "chg-state-change",
- .handler = smb1355_handle_chg_state_change,
- .wake = true,
- },
- [2] = {
- .name = "temperature-change",
- .handler = smb1355_handle_temperature_change,
- },
- };
- static int smb1355_get_irq_index_byname(const char *irq_name)
- {
- int i;
- for (i = 0; i < ARRAY_SIZE(smb1355_irqs); i++) {
- if (strcmp(smb1355_irqs[i].name, irq_name) == 0)
- return i;
- }
- return -ENOENT;
- }
- static int smb1355_request_interrupt(struct smb1355 *chip,
- struct device_node *node,
- const char *irq_name)
- {
- int rc = 0, irq, irq_index;
- 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 = smb1355_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 (!smb1355_irqs[irq_index].handler)
- return 0;
- rc = devm_request_threaded_irq(chip->dev, irq, NULL,
- smb1355_irqs[irq_index].handler,
- IRQF_ONESHOT, irq_name, chip);
- if (rc < 0) {
- pr_err("Couldn't request irq %d rc=%d\n", irq, rc);
- return rc;
- }
- smb1355_irqs[irq_index].irq = irq;
- if (smb1355_irqs[irq_index].wake)
- enable_irq_wake(irq);
- return rc;
- }
- static int smb1355_request_interrupts(struct smb1355 *chip)
- {
- struct device_node *node = chip->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 = smb1355_request_interrupt(chip, child, name);
- if (rc < 0) {
- pr_err("Couldn't request interrupt %s rc=%d\n",
- name, rc);
- return rc;
- }
- }
- }
- return rc;
- }
- static int smb1355_irq_disable_callback(struct votable *votable, void *data,
- int disable, const char *client)
- {
- int i;
- for (i = 0; i < ARRAY_SIZE(smb1355_irqs); i++) {
- if (smb1355_irqs[i].irq) {
- if (disable)
- disable_irq(smb1355_irqs[i].irq);
- else
- enable_irq(smb1355_irqs[i].irq);
- }
- }
- return 0;
- }
- /*********
- * PROBE *
- *********/
- static const struct of_device_id match_table[] = {
- {
- .compatible = "qcom,smb1355",
- },
- { },
- };
- static int smb1355_probe(struct platform_device *pdev)
- {
- struct smb1355 *chip;
- const struct of_device_id *id;
- struct iio_dev *indio_dev;
- int rc = 0;
- indio_dev = devm_iio_device_alloc(&pdev->dev, sizeof(*chip));
- if (!indio_dev)
- return -ENOMEM;
- chip = iio_priv(indio_dev);
- chip->indio_dev = indio_dev;
- chip->dev = &pdev->dev;
- chip->param = v1_params;
- chip->c_health = -EINVAL;
- chip->d_health = -EINVAL;
- chip->c_charger_temp_max = -EINVAL;
- mutex_init(&chip->write_lock);
- mutex_init(&chip->suspend_lock);
- INIT_DELAYED_WORK(&chip->die_temp_work, die_temp_work);
- chip->disabled = false;
- chip->die_temp_deciDegC = -EINVAL;
- chip->regmap = dev_get_regmap(chip->dev->parent, NULL);
- if (!chip->regmap) {
- pr_err("parent regmap is missing\n");
- return -EINVAL;
- }
- id = of_match_device(of_match_ptr(match_table), chip->dev);
- if (!id) {
- pr_err("Couldn't find a matching device\n");
- return -ENODEV;
- }
- rc = smb1355_detect_version(chip);
- if (rc < 0) {
- pr_err("Couldn't detect SMB1355/1354 chip type rc=%d\n", rc);
- goto cleanup;
- }
- platform_set_drvdata(pdev, chip);
- rc = smb1355_parse_dt(chip);
- if (rc < 0) {
- pr_err("Couldn't parse device tree rc=%d\n", rc);
- goto cleanup;
- }
- rc = smb1355_init_hw(chip);
- if (rc < 0) {
- pr_err("Couldn't initialize hardware rc=%d\n", rc);
- goto cleanup;
- }
- rc = smb1355_init_parallel_psy(chip);
- if (rc < 0) {
- pr_err("Couldn't initialize parallel psy rc=%d\n", rc);
- goto cleanup;
- }
- rc = smb1355_init_iio_psy(chip);
- if (rc < 0) {
- pr_err("Couldn't initialize parallel IIO device rc=%d\n", rc);
- goto cleanup;
- }
- rc = smb1355_determine_initial_status(chip);
- if (rc < 0) {
- pr_err("Couldn't determine initial status rc=%d\n",
- rc);
- goto cleanup;
- }
- rc = smb1355_request_interrupts(chip);
- if (rc < 0) {
- pr_err("Couldn't request interrupts rc=%d\n", rc);
- goto cleanup;
- }
- chip->irq_disable_votable = create_votable("SMB1355_IRQ_DISABLE",
- VOTE_SET_ANY, smb1355_irq_disable_callback, chip);
- if (IS_ERR(chip->irq_disable_votable)) {
- rc = PTR_ERR(chip->irq_disable_votable);
- goto cleanup;
- }
- /* keep IRQ's disabled until parallel is enabled */
- vote(chip->irq_disable_votable, PARALLEL_ENABLE_VOTER, true, 0);
- pr_info("%s probed successfully pl_mode=%s batfet_mode=%s\n",
- chip->name,
- IS_USBIN(chip->dt.pl_mode) ? "USBIN-USBIN" : "USBMID-USBMID",
- (chip->dt.pl_batfet_mode == QTI_POWER_SUPPLY_PL_STACKED_BATFET)
- ? "STACKED_BATFET" : "NON-STACKED_BATFET");
- return rc;
- cleanup:
- platform_set_drvdata(pdev, NULL);
- return rc;
- }
- static int smb1355_remove(struct platform_device *pdev)
- {
- platform_set_drvdata(pdev, NULL);
- return 0;
- }
- static void smb1355_shutdown(struct platform_device *pdev)
- {
- struct smb1355 *chip = platform_get_drvdata(pdev);
- int rc;
- /* disable parallel charging path */
- rc = smb1355_set_parallel_charging(chip, true);
- if (rc < 0)
- pr_err("Couldn't disable parallel path rc=%d\n", rc);
- smb1355_clk_request(chip, false);
- }
- #ifdef CONFIG_PM_SLEEP
- static int smb1355_suspend(struct device *dev)
- {
- struct smb1355 *chip = dev_get_drvdata(dev);
- cancel_delayed_work_sync(&chip->die_temp_work);
- mutex_lock(&chip->suspend_lock);
- chip->suspended = true;
- mutex_unlock(&chip->suspend_lock);
- return 0;
- }
- static int smb1355_resume(struct device *dev)
- {
- struct smb1355 *chip = dev_get_drvdata(dev);
- mutex_lock(&chip->suspend_lock);
- chip->suspended = false;
- mutex_unlock(&chip->suspend_lock);
- /*
- * During suspend i2c failures are fixed by reporting cached
- * chip state, to report correct values we need to invoke
- * callbacks for the fcc and fv votables. To avoid excessive
- * invokes to callbacks invoke only when smb1355 is enabled.
- */
- if (is_voter_available(chip) && chip->charging_enabled) {
- rerun_election(chip->fcc_votable);
- rerun_election(chip->fv_votable);
- }
- return 0;
- }
- #endif
- static SIMPLE_DEV_PM_OPS(smb1355_pm_ops, smb1355_suspend, smb1355_resume);
- static struct platform_driver smb1355_driver = {
- .driver = {
- .name = "qcom,smb1355-charger",
- .pm = &smb1355_pm_ops,
- .of_match_table = match_table,
- },
- .probe = smb1355_probe,
- .remove = smb1355_remove,
- .shutdown = smb1355_shutdown,
- };
- module_platform_driver(smb1355_driver);
- MODULE_DESCRIPTION("QPNP SMB1355 Charger Driver");
- MODULE_LICENSE("GPL");
|