blob: 5de42f46b43fb455a301ad64463b478af238f6fe [file] [log] [blame]
Harry Yang4b7db0f2017-11-27 10:50:44 -08001/* Copyright (c) 2018 The Linux Foundation. All rights reserved.
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
13#include <linux/debugfs.h>
14#include <linux/delay.h>
15#include <linux/device.h>
16#include <linux/module.h>
17#include <linux/platform_device.h>
18#include <linux/regmap.h>
19#include <linux/power_supply.h>
20#include <linux/of.h>
21#include <linux/of_irq.h>
22#include <linux/log2.h>
23#include <linux/qpnp/qpnp-revid.h>
24#include <linux/regulator/driver.h>
25#include <linux/regulator/of_regulator.h>
26#include <linux/regulator/machine.h>
27#include <linux/pmic-voter.h>
28#include "smb5-reg.h"
29#include "smb5-lib.h"
Ashay Jaiswal09feab82018-02-12 12:33:18 +053030#include "schgm-flash.h"
Harry Yang4b7db0f2017-11-27 10:50:44 -080031
Ashay Jaiswala9e10912018-02-02 14:03:35 +053032static struct smb_params smb5_pmi632_params = {
33 .fcc = {
34 .name = "fast charge current",
35 .reg = CHGR_FAST_CHARGE_CURRENT_CFG_REG,
36 .min_u = 0,
37 .max_u = 3000000,
38 .step_u = 50000,
39 },
40 .fv = {
41 .name = "float voltage",
42 .reg = CHGR_FLOAT_VOLTAGE_CFG_REG,
43 .min_u = 3600000,
44 .max_u = 4800000,
45 .step_u = 10000,
46 },
47 .usb_icl = {
48 .name = "usb input current limit",
49 .reg = USBIN_CURRENT_LIMIT_CFG_REG,
50 .min_u = 0,
51 .max_u = 3000000,
52 .step_u = 50000,
53 },
54 .icl_stat = {
55 .name = "input current limit status",
56 .reg = AICL_ICL_STATUS_REG,
57 .min_u = 0,
58 .max_u = 3000000,
59 .step_u = 50000,
60 },
61 .otg_cl = {
62 .name = "usb otg current limit",
63 .reg = DCDC_OTG_CURRENT_LIMIT_CFG_REG,
64 .min_u = 500000,
65 .max_u = 1000000,
66 .step_u = 250000,
67 },
68 .jeita_cc_comp_hot = {
69 .name = "jeita fcc reduction",
70 .reg = JEITA_CCCOMP_CFG_HOT_REG,
71 .min_u = 0,
72 .max_u = 1575000,
73 .step_u = 25000,
74 },
75 .jeita_cc_comp_cold = {
76 .name = "jeita fcc reduction",
77 .reg = JEITA_CCCOMP_CFG_COLD_REG,
78 .min_u = 0,
79 .max_u = 1575000,
80 .step_u = 25000,
81 },
82 .freq_switcher = {
83 .name = "switching frequency",
84 .reg = DCDC_FSW_SEL_REG,
85 .min_u = 600,
86 .max_u = 1200,
87 .step_u = 400,
88 .set_proc = smblib_set_chg_freq,
89 },
90};
91
92static struct smb_params smb5_pmi855_params = {
Harry Yang4b7db0f2017-11-27 10:50:44 -080093 .fcc = {
94 .name = "fast charge current",
95 .reg = CHGR_FAST_CHARGE_CURRENT_CFG_REG,
96 .min_u = 0,
97 .max_u = 8000000,
98 .step_u = 25000,
99 },
100 .fv = {
101 .name = "float voltage",
102 .reg = CHGR_FLOAT_VOLTAGE_CFG_REG,
103 .min_u = 3600000,
104 .max_u = 4790000,
105 .step_u = 10000,
106 },
107 .usb_icl = {
108 .name = "usb input current limit",
109 .reg = USBIN_CURRENT_LIMIT_CFG_REG,
110 .min_u = 0,
111 .max_u = 5000000,
112 .step_u = 50000,
113 },
114 .icl_stat = {
115 .name = "input current limit status",
116 .reg = AICL_ICL_STATUS_REG,
117 .min_u = 0,
118 .max_u = 5000000,
119 .step_u = 50000,
120 },
121 .otg_cl = {
122 .name = "usb otg current limit",
123 .reg = DCDC_OTG_CURRENT_LIMIT_CFG_REG,
124 .min_u = 500000,
125 .max_u = 3000000,
126 .step_u = 500000,
127 },
128 .jeita_cc_comp_hot = {
129 .name = "jeita fcc reduction",
130 .reg = JEITA_CCCOMP_CFG_HOT_REG,
131 .min_u = 0,
132 .max_u = 8000000,
133 .step_u = 25000,
134 .set_proc = NULL,
135 },
136 .jeita_cc_comp_cold = {
137 .name = "jeita fcc reduction",
138 .reg = JEITA_CCCOMP_CFG_COLD_REG,
139 .min_u = 0,
140 .max_u = 8000000,
141 .step_u = 25000,
142 .set_proc = NULL,
143 },
144 .freq_switcher = {
145 .name = "switching frequency",
146 .reg = DCDC_FSW_SEL_REG,
147 .min_u = 1200,
148 .max_u = 2400,
149 .step_u = 400,
150 .set_proc = NULL,
151 },
152};
153
154struct smb_dt_props {
155 int usb_icl_ua;
156 struct device_node *revid_dev_node;
157 enum float_options float_option;
158 int chg_inhibit_thr_mv;
159 bool no_battery;
160 bool hvdcp_disable;
Anirudh Ghayal1380d312018-02-23 00:01:43 +0530161 int auto_recharge_soc;
162 int auto_recharge_vbat_mv;
Harry Yang4b7db0f2017-11-27 10:50:44 -0800163 int wd_bark_time;
164 int batt_profile_fcc_ua;
165 int batt_profile_fv_uv;
166};
167
168struct smb5 {
169 struct smb_charger chg;
170 struct dentry *dfs_root;
171 struct smb_dt_props dt;
172};
173
174static int __debug_mask;
175module_param_named(
176 debug_mask, __debug_mask, int, 0600
177);
178
179static int __weak_chg_icl_ua = 500000;
180module_param_named(
181 weak_chg_icl_ua, __weak_chg_icl_ua, int, 0600
182);
183
Ashay Jaiswala9e10912018-02-02 14:03:35 +0530184#define PMI632_MAX_ICL_UA 3000000
185static int smb5_chg_config_init(struct smb5 *chip)
186{
187 struct smb_charger *chg = &chip->chg;
188 struct pmic_revid_data *pmic_rev_id;
189 struct device_node *revid_dev_node;
190 int rc = 0;
191
192 revid_dev_node = of_parse_phandle(chip->chg.dev->of_node,
193 "qcom,pmic-revid", 0);
194 if (!revid_dev_node) {
195 pr_err("Missing qcom,pmic-revid property\n");
196 return -EINVAL;
197 }
198
199 pmic_rev_id = get_revid_data(revid_dev_node);
200 if (IS_ERR_OR_NULL(pmic_rev_id)) {
201 /*
202 * the revid peripheral must be registered, any failure
203 * here only indicates that the rev-id module has not
204 * probed yet.
205 */
206 rc = -EPROBE_DEFER;
207 goto out;
208 }
209
210 switch (pmic_rev_id->pmic_subtype) {
211 case PM855B_SUBTYPE:
212 chip->chg.smb_version = PM855B_SUBTYPE;
213 chg->param = smb5_pmi855_params;
214 chg->name = "pm855b_charger";
215 break;
216 case PMI632_SUBTYPE:
217 chip->chg.smb_version = PMI632_SUBTYPE;
218 chg->param = smb5_pmi632_params;
219 chg->use_extcon = true;
220 chg->name = "pmi632_charger";
221 chg->hw_max_icl_ua =
222 (chip->dt.usb_icl_ua > 0) ? chip->dt.usb_icl_ua
223 : PMI632_MAX_ICL_UA;
224 chg->chg_freq.freq_5V = 600;
225 chg->chg_freq.freq_6V_8V = 800;
226 chg->chg_freq.freq_9V = 1050;
227 chg->chg_freq.freq_removal = 1050;
228 chg->chg_freq.freq_below_otg_threshold = 800;
229 chg->chg_freq.freq_above_otg_threshold = 800;
230 break;
231 default:
232 pr_err("PMIC subtype %d not supported\n",
233 pmic_rev_id->pmic_subtype);
234 rc = -EINVAL;
235 }
236
237out:
238 of_node_put(revid_dev_node);
239 return rc;
240}
241
Harry Yang4b7db0f2017-11-27 10:50:44 -0800242#define MICRO_1P5A 1500000
243#define MICRO_P1A 100000
244#define OTG_DEFAULT_DEGLITCH_TIME_MS 50
245#define MIN_WD_BARK_TIME 16
246#define DEFAULT_WD_BARK_TIME 64
247#define BITE_WDOG_TIMEOUT_8S 0x3
248#define BARK_WDOG_TIMEOUT_MASK GENMASK(3, 2)
249#define BARK_WDOG_TIMEOUT_SHIFT 2
250static int smb5_parse_dt(struct smb5 *chip)
251{
252 struct smb_charger *chg = &chip->chg;
253 struct device_node *node = chg->dev->of_node;
254 int rc, byte_len;
255
256 if (!node) {
257 pr_err("device tree node missing\n");
258 return -EINVAL;
259 }
260
261 chg->step_chg_enabled = of_property_read_bool(node,
262 "qcom,step-charging-enable");
263
264 chg->sw_jeita_enabled = of_property_read_bool(node,
265 "qcom,sw-jeita-enable");
266
267 rc = of_property_read_u32(node, "qcom,wd-bark-time-secs",
268 &chip->dt.wd_bark_time);
269 if (rc < 0 || chip->dt.wd_bark_time < MIN_WD_BARK_TIME)
270 chip->dt.wd_bark_time = DEFAULT_WD_BARK_TIME;
271
272 chip->dt.no_battery = of_property_read_bool(node,
273 "qcom,batteryless-platform");
274
275 rc = of_property_read_u32(node,
276 "qcom,fcc-max-ua", &chip->dt.batt_profile_fcc_ua);
277 if (rc < 0)
278 chip->dt.batt_profile_fcc_ua = -EINVAL;
279
280 rc = of_property_read_u32(node,
281 "qcom,fv-max-uv", &chip->dt.batt_profile_fv_uv);
282 if (rc < 0)
283 chip->dt.batt_profile_fv_uv = -EINVAL;
284
285 rc = of_property_read_u32(node,
286 "qcom,usb-icl-ua", &chip->dt.usb_icl_ua);
287 if (rc < 0)
288 chip->dt.usb_icl_ua = -EINVAL;
289
290 rc = of_property_read_u32(node,
291 "qcom,otg-cl-ua", &chg->otg_cl_ua);
292 if (rc < 0)
293 chg->otg_cl_ua = MICRO_1P5A;
294
295 if (of_find_property(node, "qcom,thermal-mitigation", &byte_len)) {
296 chg->thermal_mitigation = devm_kzalloc(chg->dev, byte_len,
297 GFP_KERNEL);
298
299 if (chg->thermal_mitigation == NULL)
300 return -ENOMEM;
301
302 chg->thermal_levels = byte_len / sizeof(u32);
303 rc = of_property_read_u32_array(node,
304 "qcom,thermal-mitigation",
305 chg->thermal_mitigation,
306 chg->thermal_levels);
307 if (rc < 0) {
308 dev_err(chg->dev,
309 "Couldn't read threm limits rc = %d\n", rc);
310 return rc;
311 }
312 }
313
314 rc = of_property_read_u32(node, "qcom,float-option",
315 &chip->dt.float_option);
316 if (!rc && (chip->dt.float_option < 0 || chip->dt.float_option > 4)) {
317 pr_err("qcom,float-option is out of range [0, 4]\n");
318 return -EINVAL;
319 }
320
321 chip->dt.hvdcp_disable = of_property_read_bool(node,
322 "qcom,hvdcp-disable");
323
324
325 rc = of_property_read_u32(node, "qcom,chg-inhibit-threshold-mv",
326 &chip->dt.chg_inhibit_thr_mv);
327 if (!rc && (chip->dt.chg_inhibit_thr_mv < 0 ||
328 chip->dt.chg_inhibit_thr_mv > 300)) {
329 pr_err("qcom,chg-inhibit-threshold-mv is incorrect\n");
330 return -EINVAL;
331 }
332
Anirudh Ghayal1380d312018-02-23 00:01:43 +0530333 chip->dt.auto_recharge_soc = -EINVAL;
334 rc = of_property_read_u32(node, "qcom,auto-recharge-soc",
335 &chip->dt.auto_recharge_soc);
336 if (!rc && (chip->dt.auto_recharge_soc < 0 ||
337 chip->dt.auto_recharge_soc > 100)) {
338 pr_err("qcom,auto-recharge-soc is incorrect\n");
339 return -EINVAL;
340 }
341 chg->auto_recharge_soc = chip->dt.auto_recharge_soc;
342
343 chip->dt.auto_recharge_vbat_mv = -EINVAL;
344 rc = of_property_read_u32(node, "qcom,auto-recharge-vbat-mv",
345 &chip->dt.auto_recharge_vbat_mv);
346 if (!rc && (chip->dt.auto_recharge_vbat_mv < 0)) {
347 pr_err("qcom,auto-recharge-vbat-mv is incorrect\n");
348 return -EINVAL;
349 }
Harry Yang4b7db0f2017-11-27 10:50:44 -0800350
Harry Yang4b7db0f2017-11-27 10:50:44 -0800351 chg->dcp_icl_ua = chip->dt.usb_icl_ua;
352
353 chg->suspend_input_on_debug_batt = of_property_read_bool(node,
354 "qcom,suspend-input-on-debug-batt");
355
356 rc = of_property_read_u32(node, "qcom,otg-deglitch-time-ms",
357 &chg->otg_delay_ms);
358 if (rc < 0)
359 chg->otg_delay_ms = OTG_DEFAULT_DEGLITCH_TIME_MS;
360
361 return 0;
362}
363
364/************************
365 * USB PSY REGISTRATION *
366 ************************/
367static enum power_supply_property smb5_usb_props[] = {
368 POWER_SUPPLY_PROP_PRESENT,
369 POWER_SUPPLY_PROP_ONLINE,
370 POWER_SUPPLY_PROP_PD_CURRENT_MAX,
371 POWER_SUPPLY_PROP_CURRENT_MAX,
372 POWER_SUPPLY_PROP_TYPE,
373 POWER_SUPPLY_PROP_TYPEC_MODE,
374 POWER_SUPPLY_PROP_TYPEC_POWER_ROLE,
375 POWER_SUPPLY_PROP_TYPEC_CC_ORIENTATION,
376 POWER_SUPPLY_PROP_PD_ALLOWED,
377 POWER_SUPPLY_PROP_PD_ACTIVE,
378 POWER_SUPPLY_PROP_INPUT_CURRENT_SETTLED,
379 POWER_SUPPLY_PROP_INPUT_CURRENT_NOW,
380 POWER_SUPPLY_PROP_BOOST_CURRENT,
381 POWER_SUPPLY_PROP_PE_START,
382 POWER_SUPPLY_PROP_CTM_CURRENT_MAX,
383 POWER_SUPPLY_PROP_HW_CURRENT_MAX,
384 POWER_SUPPLY_PROP_REAL_TYPE,
385 POWER_SUPPLY_PROP_PR_SWAP,
386 POWER_SUPPLY_PROP_PD_VOLTAGE_MAX,
387 POWER_SUPPLY_PROP_PD_VOLTAGE_MIN,
388 POWER_SUPPLY_PROP_SDP_CURRENT_MAX,
Ashay Jaiswala9e10912018-02-02 14:03:35 +0530389 POWER_SUPPLY_PROP_CONNECTOR_TYPE,
Ashay Jaiswal1f71b412017-10-31 14:33:27 +0530390 POWER_SUPPLY_PROP_VOLTAGE_MAX,
Ashay Jaiswal09feab82018-02-12 12:33:18 +0530391 POWER_SUPPLY_PROP_SCOPE,
Harry Yang4b7db0f2017-11-27 10:50:44 -0800392};
393
394static int smb5_usb_get_prop(struct power_supply *psy,
395 enum power_supply_property psp,
396 union power_supply_propval *val)
397{
398 struct smb5 *chip = power_supply_get_drvdata(psy);
399 struct smb_charger *chg = &chip->chg;
Ashay Jaiswal09feab82018-02-12 12:33:18 +0530400 union power_supply_propval pval;
Harry Yang4b7db0f2017-11-27 10:50:44 -0800401 int rc = 0;
402
403 switch (psp) {
404 case POWER_SUPPLY_PROP_PRESENT:
405 rc = smblib_get_prop_usb_present(chg, val);
406 break;
407 case POWER_SUPPLY_PROP_ONLINE:
408 rc = smblib_get_prop_usb_online(chg, val);
409 if (!val->intval)
410 break;
411
Ashay Jaiswala9e10912018-02-02 14:03:35 +0530412 if (((chg->typec_mode == POWER_SUPPLY_TYPEC_SOURCE_DEFAULT) ||
413 (chg->connector_type == POWER_SUPPLY_CONNECTOR_MICRO_USB))
414 && (chg->real_charger_type == POWER_SUPPLY_TYPE_USB))
Harry Yang4b7db0f2017-11-27 10:50:44 -0800415 val->intval = 0;
416 else
417 val->intval = 1;
Ashay Jaiswala9e10912018-02-02 14:03:35 +0530418
Harry Yang4b7db0f2017-11-27 10:50:44 -0800419 if (chg->real_charger_type == POWER_SUPPLY_TYPE_UNKNOWN)
420 val->intval = 0;
421 break;
422 case POWER_SUPPLY_PROP_VOLTAGE_MAX:
423 rc = smblib_get_prop_usb_voltage_max(chg, val);
424 break;
425 case POWER_SUPPLY_PROP_PD_CURRENT_MAX:
426 val->intval = get_client_vote(chg->usb_icl_votable, PD_VOTER);
427 break;
428 case POWER_SUPPLY_PROP_CURRENT_MAX:
429 rc = smblib_get_prop_input_current_settled(chg, val);
430 break;
431 case POWER_SUPPLY_PROP_TYPE:
432 val->intval = POWER_SUPPLY_TYPE_USB_PD;
433 break;
434 case POWER_SUPPLY_PROP_REAL_TYPE:
435 val->intval = chg->real_charger_type;
436 break;
437 case POWER_SUPPLY_PROP_TYPEC_MODE:
Ashay Jaiswala9e10912018-02-02 14:03:35 +0530438 if (chg->connector_type == POWER_SUPPLY_CONNECTOR_MICRO_USB)
Harry Yang4b7db0f2017-11-27 10:50:44 -0800439 val->intval = POWER_SUPPLY_TYPEC_NONE;
440 else
441 val->intval = chg->typec_mode;
442 break;
443 case POWER_SUPPLY_PROP_TYPEC_POWER_ROLE:
Ashay Jaiswala9e10912018-02-02 14:03:35 +0530444 if (chg->connector_type == POWER_SUPPLY_CONNECTOR_MICRO_USB)
Harry Yang4b7db0f2017-11-27 10:50:44 -0800445 val->intval = POWER_SUPPLY_TYPEC_PR_NONE;
446 else
447 rc = smblib_get_prop_typec_power_role(chg, val);
448 break;
449 case POWER_SUPPLY_PROP_TYPEC_CC_ORIENTATION:
Ashay Jaiswala9e10912018-02-02 14:03:35 +0530450 if (chg->connector_type == POWER_SUPPLY_CONNECTOR_MICRO_USB)
Harry Yang4b7db0f2017-11-27 10:50:44 -0800451 val->intval = 0;
452 else
453 rc = smblib_get_prop_typec_cc_orientation(chg, val);
454 break;
455 case POWER_SUPPLY_PROP_PD_ALLOWED:
456 rc = smblib_get_prop_pd_allowed(chg, val);
457 break;
458 case POWER_SUPPLY_PROP_PD_ACTIVE:
459 val->intval = chg->pd_active;
460 break;
461 case POWER_SUPPLY_PROP_INPUT_CURRENT_SETTLED:
462 rc = smblib_get_prop_input_current_settled(chg, val);
463 break;
464 case POWER_SUPPLY_PROP_BOOST_CURRENT:
465 val->intval = chg->boost_current_ua;
466 break;
467 case POWER_SUPPLY_PROP_PD_IN_HARD_RESET:
468 rc = smblib_get_prop_pd_in_hard_reset(chg, val);
469 break;
470 case POWER_SUPPLY_PROP_PD_USB_SUSPEND_SUPPORTED:
471 val->intval = chg->system_suspend_supported;
472 break;
473 case POWER_SUPPLY_PROP_PE_START:
474 rc = smblib_get_pe_start(chg, val);
475 break;
476 case POWER_SUPPLY_PROP_CTM_CURRENT_MAX:
477 val->intval = get_client_vote(chg->usb_icl_votable, CTM_VOTER);
478 break;
479 case POWER_SUPPLY_PROP_HW_CURRENT_MAX:
480 rc = smblib_get_charge_current(chg, &val->intval);
481 break;
482 case POWER_SUPPLY_PROP_PR_SWAP:
483 rc = smblib_get_prop_pr_swap_in_progress(chg, val);
484 break;
485 case POWER_SUPPLY_PROP_PD_VOLTAGE_MAX:
486 val->intval = chg->voltage_max_uv;
487 break;
488 case POWER_SUPPLY_PROP_PD_VOLTAGE_MIN:
489 val->intval = chg->voltage_min_uv;
490 break;
491 case POWER_SUPPLY_PROP_SDP_CURRENT_MAX:
492 val->intval = get_client_vote(chg->usb_icl_votable,
493 USB_PSY_VOTER);
494 break;
Ashay Jaiswala9e10912018-02-02 14:03:35 +0530495 case POWER_SUPPLY_PROP_CONNECTOR_TYPE:
496 val->intval = chg->connector_type;
497 break;
Ashay Jaiswal09feab82018-02-12 12:33:18 +0530498 case POWER_SUPPLY_PROP_SCOPE:
499 val->intval = POWER_SUPPLY_SCOPE_UNKNOWN;
500 rc = smblib_get_prop_usb_present(chg, &pval);
501 if (rc < 0)
502 break;
503 val->intval = pval.intval ? POWER_SUPPLY_SCOPE_DEVICE
504 : chg->otg_present ? POWER_SUPPLY_SCOPE_SYSTEM
505 : POWER_SUPPLY_SCOPE_UNKNOWN;
506 break;
Harry Yang4b7db0f2017-11-27 10:50:44 -0800507 default:
508 pr_err("get prop %d is not supported in usb\n", psp);
509 rc = -EINVAL;
510 break;
511 }
512
513 if (rc < 0) {
514 pr_debug("Couldn't get prop %d rc = %d\n", psp, rc);
515 return -ENODATA;
516 }
517
518 return 0;
519}
520
521static int smb5_usb_set_prop(struct power_supply *psy,
522 enum power_supply_property psp,
523 const union power_supply_propval *val)
524{
525 struct smb5 *chip = power_supply_get_drvdata(psy);
526 struct smb_charger *chg = &chip->chg;
527 int rc = 0;
528
529 mutex_lock(&chg->lock);
530 if (!chg->typec_present) {
531 rc = -EINVAL;
532 goto unlock;
533 }
534
535 switch (psp) {
536 case POWER_SUPPLY_PROP_PD_CURRENT_MAX:
537 rc = smblib_set_prop_pd_current_max(chg, val);
538 break;
539 case POWER_SUPPLY_PROP_TYPEC_POWER_ROLE:
540 rc = smblib_set_prop_typec_power_role(chg, val);
541 break;
542 case POWER_SUPPLY_PROP_PD_ACTIVE:
543 rc = smblib_set_prop_pd_active(chg, val);
544 break;
545 case POWER_SUPPLY_PROP_PD_IN_HARD_RESET:
546 rc = smblib_set_prop_pd_in_hard_reset(chg, val);
547 break;
548 case POWER_SUPPLY_PROP_PD_USB_SUSPEND_SUPPORTED:
549 chg->system_suspend_supported = val->intval;
550 break;
551 case POWER_SUPPLY_PROP_BOOST_CURRENT:
552 rc = smblib_set_prop_boost_current(chg, val);
553 break;
554 case POWER_SUPPLY_PROP_CTM_CURRENT_MAX:
555 rc = vote(chg->usb_icl_votable, CTM_VOTER,
556 val->intval >= 0, val->intval);
557 break;
558 case POWER_SUPPLY_PROP_PR_SWAP:
559 rc = smblib_set_prop_pr_swap_in_progress(chg, val);
560 break;
561 case POWER_SUPPLY_PROP_PD_VOLTAGE_MAX:
562 rc = smblib_set_prop_pd_voltage_max(chg, val);
563 break;
564 case POWER_SUPPLY_PROP_PD_VOLTAGE_MIN:
565 rc = smblib_set_prop_pd_voltage_min(chg, val);
566 break;
567 case POWER_SUPPLY_PROP_SDP_CURRENT_MAX:
568 rc = smblib_set_prop_sdp_current_max(chg, val);
569 break;
570 default:
571 pr_err("set prop %d is not supported\n", psp);
572 rc = -EINVAL;
573 break;
574 }
575
576unlock:
577 mutex_unlock(&chg->lock);
578 return rc;
579}
580
581static int smb5_usb_prop_is_writeable(struct power_supply *psy,
582 enum power_supply_property psp)
583{
584 switch (psp) {
585 case POWER_SUPPLY_PROP_CTM_CURRENT_MAX:
586 return 1;
587 default:
588 break;
589 }
590
591 return 0;
592}
593
594static const struct power_supply_desc usb_psy_desc = {
595 .name = "usb",
596 .type = POWER_SUPPLY_TYPE_USB_PD,
597 .properties = smb5_usb_props,
598 .num_properties = ARRAY_SIZE(smb5_usb_props),
599 .get_property = smb5_usb_get_prop,
600 .set_property = smb5_usb_set_prop,
601 .property_is_writeable = smb5_usb_prop_is_writeable,
602};
603
604static int smb5_init_usb_psy(struct smb5 *chip)
605{
606 struct power_supply_config usb_cfg = {};
607 struct smb_charger *chg = &chip->chg;
608
609 usb_cfg.drv_data = chip;
610 usb_cfg.of_node = chg->dev->of_node;
611 chg->usb_psy = devm_power_supply_register(chg->dev,
612 &usb_psy_desc,
613 &usb_cfg);
614 if (IS_ERR(chg->usb_psy)) {
615 pr_err("Couldn't register USB power supply\n");
616 return PTR_ERR(chg->usb_psy);
617 }
618
619 return 0;
620}
621
622/********************************
623 * USB PC_PORT PSY REGISTRATION *
624 ********************************/
625static enum power_supply_property smb5_usb_port_props[] = {
626 POWER_SUPPLY_PROP_TYPE,
627 POWER_SUPPLY_PROP_ONLINE,
628 POWER_SUPPLY_PROP_VOLTAGE_MAX,
629 POWER_SUPPLY_PROP_CURRENT_MAX,
630};
631
632static int smb5_usb_port_get_prop(struct power_supply *psy,
633 enum power_supply_property psp,
634 union power_supply_propval *val)
635{
636 struct smb5 *chip = power_supply_get_drvdata(psy);
637 struct smb_charger *chg = &chip->chg;
638 int rc = 0;
639
640 switch (psp) {
641 case POWER_SUPPLY_PROP_TYPE:
642 val->intval = POWER_SUPPLY_TYPE_USB;
643 break;
644 case POWER_SUPPLY_PROP_ONLINE:
645 rc = smblib_get_prop_usb_online(chg, val);
646 if (!val->intval)
647 break;
648
Ashay Jaiswala9e10912018-02-02 14:03:35 +0530649 if (((chg->typec_mode == POWER_SUPPLY_TYPEC_SOURCE_DEFAULT) ||
650 (chg->connector_type == POWER_SUPPLY_CONNECTOR_MICRO_USB))
651 && (chg->real_charger_type == POWER_SUPPLY_TYPE_USB))
Harry Yang4b7db0f2017-11-27 10:50:44 -0800652 val->intval = 1;
653 else
654 val->intval = 0;
655 break;
656 case POWER_SUPPLY_PROP_VOLTAGE_MAX:
657 val->intval = 5000000;
658 break;
659 case POWER_SUPPLY_PROP_CURRENT_MAX:
660 rc = smblib_get_prop_input_current_settled(chg, val);
661 break;
662 default:
663 pr_err_ratelimited("Get prop %d is not supported in pc_port\n",
664 psp);
665 return -EINVAL;
666 }
667
668 if (rc < 0) {
669 pr_debug("Couldn't get prop %d rc = %d\n", psp, rc);
670 return -ENODATA;
671 }
672
673 return 0;
674}
675
676static int smb5_usb_port_set_prop(struct power_supply *psy,
677 enum power_supply_property psp,
678 const union power_supply_propval *val)
679{
680 int rc = 0;
681
682 switch (psp) {
683 default:
684 pr_err_ratelimited("Set prop %d is not supported in pc_port\n",
685 psp);
686 rc = -EINVAL;
687 break;
688 }
689
690 return rc;
691}
692
693static const struct power_supply_desc usb_port_psy_desc = {
694 .name = "pc_port",
695 .type = POWER_SUPPLY_TYPE_USB,
696 .properties = smb5_usb_port_props,
697 .num_properties = ARRAY_SIZE(smb5_usb_port_props),
698 .get_property = smb5_usb_port_get_prop,
699 .set_property = smb5_usb_port_set_prop,
700};
701
702static int smb5_init_usb_port_psy(struct smb5 *chip)
703{
704 struct power_supply_config usb_port_cfg = {};
705 struct smb_charger *chg = &chip->chg;
706
707 usb_port_cfg.drv_data = chip;
708 usb_port_cfg.of_node = chg->dev->of_node;
709 chg->usb_port_psy = devm_power_supply_register(chg->dev,
710 &usb_port_psy_desc,
711 &usb_port_cfg);
712 if (IS_ERR(chg->usb_port_psy)) {
713 pr_err("Couldn't register USB pc_port power supply\n");
714 return PTR_ERR(chg->usb_port_psy);
715 }
716
717 return 0;
718}
719
720/*****************************
721 * USB MAIN PSY REGISTRATION *
722 *****************************/
723
724static enum power_supply_property smb5_usb_main_props[] = {
725 POWER_SUPPLY_PROP_VOLTAGE_MAX,
726 POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX,
727 POWER_SUPPLY_PROP_TYPE,
728 POWER_SUPPLY_PROP_INPUT_CURRENT_SETTLED,
729 POWER_SUPPLY_PROP_INPUT_VOLTAGE_SETTLED,
730 POWER_SUPPLY_PROP_FCC_DELTA,
731 POWER_SUPPLY_PROP_CURRENT_MAX,
Ashay Jaiswal09feab82018-02-12 12:33:18 +0530732 POWER_SUPPLY_PROP_FLASH_ACTIVE,
733 POWER_SUPPLY_PROP_FLASH_TRIGGER,
Harry Yang4b7db0f2017-11-27 10:50:44 -0800734};
735
736static int smb5_usb_main_get_prop(struct power_supply *psy,
737 enum power_supply_property psp,
738 union power_supply_propval *val)
739{
740 struct smb5 *chip = power_supply_get_drvdata(psy);
741 struct smb_charger *chg = &chip->chg;
742 int rc = 0;
743
744 switch (psp) {
745 case POWER_SUPPLY_PROP_VOLTAGE_MAX:
746 rc = smblib_get_charge_param(chg, &chg->param.fv, &val->intval);
747 break;
748 case POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX:
749 rc = smblib_get_charge_param(chg, &chg->param.fcc,
750 &val->intval);
751 break;
752 case POWER_SUPPLY_PROP_TYPE:
753 val->intval = POWER_SUPPLY_TYPE_MAIN;
754 break;
755 case POWER_SUPPLY_PROP_INPUT_CURRENT_SETTLED:
756 rc = smblib_get_prop_input_current_settled(chg, val);
757 break;
758 case POWER_SUPPLY_PROP_INPUT_VOLTAGE_SETTLED:
759 rc = smblib_get_prop_input_voltage_settled(chg, val);
760 break;
761 case POWER_SUPPLY_PROP_FCC_DELTA:
762 rc = smblib_get_prop_fcc_delta(chg, val);
763 break;
764 case POWER_SUPPLY_PROP_CURRENT_MAX:
765 rc = smblib_get_icl_current(chg, &val->intval);
766 break;
Ashay Jaiswal09feab82018-02-12 12:33:18 +0530767 case POWER_SUPPLY_PROP_FLASH_ACTIVE:
768 val->intval = chg->flash_active;
769 break;
770 case POWER_SUPPLY_PROP_FLASH_TRIGGER:
771 rc = schgm_flash_get_vreg_ok(chg, &val->intval);
772 break;
Harry Yang4b7db0f2017-11-27 10:50:44 -0800773 default:
774 pr_debug("get prop %d is not supported in usb-main\n", psp);
775 rc = -EINVAL;
776 break;
777 }
778 if (rc < 0) {
779 pr_debug("Couldn't get prop %d rc = %d\n", psp, rc);
780 return -ENODATA;
781 }
782
783 return 0;
784}
785
786static int smb5_usb_main_set_prop(struct power_supply *psy,
787 enum power_supply_property psp,
788 const union power_supply_propval *val)
789{
790 struct smb5 *chip = power_supply_get_drvdata(psy);
791 struct smb_charger *chg = &chip->chg;
792 int rc = 0;
793
794 switch (psp) {
795 case POWER_SUPPLY_PROP_VOLTAGE_MAX:
796 rc = smblib_set_charge_param(chg, &chg->param.fv, val->intval);
797 break;
798 case POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX:
799 rc = smblib_set_charge_param(chg, &chg->param.fcc, val->intval);
800 break;
801 case POWER_SUPPLY_PROP_CURRENT_MAX:
802 rc = smblib_set_icl_current(chg, val->intval);
803 break;
Ashay Jaiswal09feab82018-02-12 12:33:18 +0530804 case POWER_SUPPLY_PROP_FLASH_ACTIVE:
805 chg->flash_active = val->intval;
806 break;
Harry Yang4b7db0f2017-11-27 10:50:44 -0800807 default:
808 pr_err("set prop %d is not supported\n", psp);
809 rc = -EINVAL;
810 break;
811 }
812
813 return rc;
814}
815
816static const struct power_supply_desc usb_main_psy_desc = {
817 .name = "main",
818 .type = POWER_SUPPLY_TYPE_MAIN,
819 .properties = smb5_usb_main_props,
820 .num_properties = ARRAY_SIZE(smb5_usb_main_props),
821 .get_property = smb5_usb_main_get_prop,
822 .set_property = smb5_usb_main_set_prop,
823};
824
825static int smb5_init_usb_main_psy(struct smb5 *chip)
826{
827 struct power_supply_config usb_main_cfg = {};
828 struct smb_charger *chg = &chip->chg;
829
830 usb_main_cfg.drv_data = chip;
831 usb_main_cfg.of_node = chg->dev->of_node;
832 chg->usb_main_psy = devm_power_supply_register(chg->dev,
833 &usb_main_psy_desc,
834 &usb_main_cfg);
835 if (IS_ERR(chg->usb_main_psy)) {
836 pr_err("Couldn't register USB main power supply\n");
837 return PTR_ERR(chg->usb_main_psy);
838 }
839
840 return 0;
841}
842
843/*************************
844 * DC PSY REGISTRATION *
845 *************************/
846
847static enum power_supply_property smb5_dc_props[] = {
848 POWER_SUPPLY_PROP_INPUT_SUSPEND,
849 POWER_SUPPLY_PROP_PRESENT,
850 POWER_SUPPLY_PROP_ONLINE,
851 POWER_SUPPLY_PROP_REAL_TYPE,
852};
853
854static int smb5_dc_get_prop(struct power_supply *psy,
855 enum power_supply_property psp,
856 union power_supply_propval *val)
857{
858 struct smb5 *chip = power_supply_get_drvdata(psy);
859 struct smb_charger *chg = &chip->chg;
860 int rc = 0;
861
862 switch (psp) {
863 case POWER_SUPPLY_PROP_INPUT_SUSPEND:
864 val->intval = get_effective_result(chg->dc_suspend_votable);
865 break;
866 case POWER_SUPPLY_PROP_PRESENT:
867 rc = smblib_get_prop_dc_present(chg, val);
868 break;
869 case POWER_SUPPLY_PROP_ONLINE:
870 rc = smblib_get_prop_dc_online(chg, val);
871 break;
872 case POWER_SUPPLY_PROP_REAL_TYPE:
873 val->intval = POWER_SUPPLY_TYPE_WIPOWER;
874 break;
875 default:
876 return -EINVAL;
877 }
878 if (rc < 0) {
879 pr_debug("Couldn't get prop %d rc = %d\n", psp, rc);
880 return -ENODATA;
881 }
882 return 0;
883}
884
885static int smb5_dc_set_prop(struct power_supply *psy,
886 enum power_supply_property psp,
887 const union power_supply_propval *val)
888{
889 struct smb5 *chip = power_supply_get_drvdata(psy);
890 struct smb_charger *chg = &chip->chg;
891 int rc = 0;
892
893 switch (psp) {
894 case POWER_SUPPLY_PROP_INPUT_SUSPEND:
895 rc = vote(chg->dc_suspend_votable, WBC_VOTER,
896 (bool)val->intval, 0);
897 break;
898 default:
899 return -EINVAL;
900 }
901
902 return rc;
903}
904
905static int smb5_dc_prop_is_writeable(struct power_supply *psy,
906 enum power_supply_property psp)
907{
908 int rc;
909
910 switch (psp) {
911 default:
912 rc = 0;
913 break;
914 }
915
916 return rc;
917}
918
919static const struct power_supply_desc dc_psy_desc = {
920 .name = "dc",
921 .type = POWER_SUPPLY_TYPE_WIRELESS,
922 .properties = smb5_dc_props,
923 .num_properties = ARRAY_SIZE(smb5_dc_props),
924 .get_property = smb5_dc_get_prop,
925 .set_property = smb5_dc_set_prop,
926 .property_is_writeable = smb5_dc_prop_is_writeable,
927};
928
929static int smb5_init_dc_psy(struct smb5 *chip)
930{
931 struct power_supply_config dc_cfg = {};
932 struct smb_charger *chg = &chip->chg;
933
934 dc_cfg.drv_data = chip;
935 dc_cfg.of_node = chg->dev->of_node;
936 chg->dc_psy = devm_power_supply_register(chg->dev,
937 &dc_psy_desc,
938 &dc_cfg);
939 if (IS_ERR(chg->dc_psy)) {
940 pr_err("Couldn't register USB power supply\n");
941 return PTR_ERR(chg->dc_psy);
942 }
943
944 return 0;
945}
946
947/*************************
948 * BATT PSY REGISTRATION *
949 *************************/
950static enum power_supply_property smb5_batt_props[] = {
951 POWER_SUPPLY_PROP_INPUT_SUSPEND,
Anirudh Ghayaldfdb8412018-04-21 10:37:18 +0530952 POWER_SUPPLY_PROP_BATTERY_CHARGING_ENABLED,
Harry Yang4b7db0f2017-11-27 10:50:44 -0800953 POWER_SUPPLY_PROP_STATUS,
954 POWER_SUPPLY_PROP_HEALTH,
955 POWER_SUPPLY_PROP_PRESENT,
956 POWER_SUPPLY_PROP_CHARGE_TYPE,
957 POWER_SUPPLY_PROP_CAPACITY,
958 POWER_SUPPLY_PROP_INPUT_CURRENT_LIMITED,
959 POWER_SUPPLY_PROP_VOLTAGE_NOW,
960 POWER_SUPPLY_PROP_VOLTAGE_MAX,
961 POWER_SUPPLY_PROP_CURRENT_NOW,
962 POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX,
963 POWER_SUPPLY_PROP_TEMP,
964 POWER_SUPPLY_PROP_TECHNOLOGY,
965 POWER_SUPPLY_PROP_STEP_CHARGING_ENABLED,
966 POWER_SUPPLY_PROP_SW_JEITA_ENABLED,
967 POWER_SUPPLY_PROP_CHARGE_DONE,
968 POWER_SUPPLY_PROP_PARALLEL_DISABLE,
969 POWER_SUPPLY_PROP_SET_SHIP_MODE,
970 POWER_SUPPLY_PROP_DIE_HEALTH,
971 POWER_SUPPLY_PROP_RERUN_AICL,
972 POWER_SUPPLY_PROP_DP_DM,
973 POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT_MAX,
974 POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT,
975 POWER_SUPPLY_PROP_CHARGE_COUNTER,
Subbaraman Narayanamurthyc4feb892018-05-07 18:43:45 -0700976 POWER_SUPPLY_PROP_CYCLE_COUNT,
Anirudh Ghayal1380d312018-02-23 00:01:43 +0530977 POWER_SUPPLY_PROP_RECHARGE_SOC,
Harry Yang4b7db0f2017-11-27 10:50:44 -0800978};
979
980static int smb5_batt_get_prop(struct power_supply *psy,
981 enum power_supply_property psp,
982 union power_supply_propval *val)
983{
984 struct smb_charger *chg = power_supply_get_drvdata(psy);
985 int rc = 0;
986
987 switch (psp) {
988 case POWER_SUPPLY_PROP_STATUS:
989 rc = smblib_get_prop_batt_status(chg, val);
990 break;
991 case POWER_SUPPLY_PROP_HEALTH:
992 rc = smblib_get_prop_batt_health(chg, val);
993 break;
994 case POWER_SUPPLY_PROP_PRESENT:
995 rc = smblib_get_prop_batt_present(chg, val);
996 break;
997 case POWER_SUPPLY_PROP_INPUT_SUSPEND:
998 rc = smblib_get_prop_input_suspend(chg, val);
999 break;
Anirudh Ghayaldfdb8412018-04-21 10:37:18 +05301000 case POWER_SUPPLY_PROP_BATTERY_CHARGING_ENABLED:
1001 val->intval = !get_client_vote(chg->chg_disable_votable,
1002 USER_VOTER);
1003 break;
Harry Yang4b7db0f2017-11-27 10:50:44 -08001004 case POWER_SUPPLY_PROP_CHARGE_TYPE:
1005 rc = smblib_get_prop_batt_charge_type(chg, val);
1006 break;
1007 case POWER_SUPPLY_PROP_CAPACITY:
1008 rc = smblib_get_prop_batt_capacity(chg, val);
1009 break;
1010 case POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT:
1011 rc = smblib_get_prop_system_temp_level(chg, val);
1012 break;
1013 case POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT_MAX:
1014 rc = smblib_get_prop_system_temp_level_max(chg, val);
1015 break;
1016 case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMITED:
1017 rc = smblib_get_prop_input_current_limited(chg, val);
1018 break;
1019 case POWER_SUPPLY_PROP_STEP_CHARGING_ENABLED:
1020 val->intval = chg->step_chg_enabled;
1021 break;
1022 case POWER_SUPPLY_PROP_SW_JEITA_ENABLED:
1023 val->intval = chg->sw_jeita_enabled;
1024 break;
1025 case POWER_SUPPLY_PROP_VOLTAGE_NOW:
1026 rc = smblib_get_prop_batt_voltage_now(chg, val);
1027 break;
1028 case POWER_SUPPLY_PROP_VOLTAGE_MAX:
1029 val->intval = get_client_vote(chg->fv_votable,
1030 BATT_PROFILE_VOTER);
1031 break;
1032 case POWER_SUPPLY_PROP_CURRENT_NOW:
1033 rc = smblib_get_prop_batt_current_now(chg, val);
1034 break;
1035 case POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX:
1036 val->intval = get_client_vote(chg->fcc_votable,
1037 BATT_PROFILE_VOTER);
1038 break;
1039 case POWER_SUPPLY_PROP_TEMP:
1040 rc = smblib_get_prop_batt_temp(chg, val);
1041 break;
1042 case POWER_SUPPLY_PROP_TECHNOLOGY:
1043 val->intval = POWER_SUPPLY_TECHNOLOGY_LION;
1044 break;
1045 case POWER_SUPPLY_PROP_CHARGE_DONE:
1046 rc = smblib_get_prop_batt_charge_done(chg, val);
1047 break;
1048 case POWER_SUPPLY_PROP_PARALLEL_DISABLE:
1049 val->intval = get_client_vote(chg->pl_disable_votable,
1050 USER_VOTER);
1051 break;
1052 case POWER_SUPPLY_PROP_SET_SHIP_MODE:
1053 /* Not in ship mode as long as device is active */
1054 val->intval = 0;
1055 break;
1056 case POWER_SUPPLY_PROP_DIE_HEALTH:
1057 if (chg->die_health == -EINVAL)
1058 rc = smblib_get_prop_die_health(chg, val);
1059 else
1060 val->intval = chg->die_health;
1061 break;
1062 case POWER_SUPPLY_PROP_DP_DM:
1063 val->intval = chg->pulse_cnt;
1064 break;
1065 case POWER_SUPPLY_PROP_RERUN_AICL:
1066 val->intval = 0;
1067 break;
1068 case POWER_SUPPLY_PROP_CHARGE_COUNTER:
1069 rc = smblib_get_prop_batt_charge_counter(chg, val);
1070 break;
Subbaraman Narayanamurthyc4feb892018-05-07 18:43:45 -07001071 case POWER_SUPPLY_PROP_CYCLE_COUNT:
1072 rc = smblib_get_prop_batt_cycle_count(chg, val);
1073 break;
Anirudh Ghayal1380d312018-02-23 00:01:43 +05301074 case POWER_SUPPLY_PROP_RECHARGE_SOC:
1075 val->intval = chg->auto_recharge_soc;
1076 break;
Harry Yang4b7db0f2017-11-27 10:50:44 -08001077 default:
1078 pr_err("batt power supply prop %d not supported\n", psp);
1079 return -EINVAL;
1080 }
1081
1082 if (rc < 0) {
1083 pr_debug("Couldn't get prop %d rc = %d\n", psp, rc);
1084 return -ENODATA;
1085 }
1086
1087 return 0;
1088}
1089
1090static int smb5_batt_set_prop(struct power_supply *psy,
1091 enum power_supply_property prop,
1092 const union power_supply_propval *val)
1093{
1094 int rc = 0;
1095 struct smb_charger *chg = power_supply_get_drvdata(psy);
Ashay Jaiswala55b7ed2018-03-23 17:23:48 +05301096 bool enable;
Harry Yang4b7db0f2017-11-27 10:50:44 -08001097
1098 switch (prop) {
1099 case POWER_SUPPLY_PROP_STATUS:
1100 rc = smblib_set_prop_batt_status(chg, val);
1101 break;
1102 case POWER_SUPPLY_PROP_INPUT_SUSPEND:
1103 rc = smblib_set_prop_input_suspend(chg, val);
1104 break;
Anirudh Ghayaldfdb8412018-04-21 10:37:18 +05301105 case POWER_SUPPLY_PROP_BATTERY_CHARGING_ENABLED:
1106 vote(chg->chg_disable_votable, USER_VOTER, !val->intval, 0);
1107 break;
Harry Yang4b7db0f2017-11-27 10:50:44 -08001108 case POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT:
1109 rc = smblib_set_prop_system_temp_level(chg, val);
1110 break;
1111 case POWER_SUPPLY_PROP_CAPACITY:
1112 rc = smblib_set_prop_batt_capacity(chg, val);
1113 break;
1114 case POWER_SUPPLY_PROP_PARALLEL_DISABLE:
1115 vote(chg->pl_disable_votable, USER_VOTER, (bool)val->intval, 0);
1116 break;
1117 case POWER_SUPPLY_PROP_VOLTAGE_MAX:
1118 chg->batt_profile_fv_uv = val->intval;
1119 vote(chg->fv_votable, BATT_PROFILE_VOTER, true, val->intval);
1120 break;
1121 case POWER_SUPPLY_PROP_STEP_CHARGING_ENABLED:
Ashay Jaiswala55b7ed2018-03-23 17:23:48 +05301122 enable = !!val->intval || chg->sw_jeita_enabled;
1123 rc = smblib_configure_wdog(chg, enable);
1124 if (rc == 0)
1125 chg->step_chg_enabled = !!val->intval;
Harry Yang4b7db0f2017-11-27 10:50:44 -08001126 break;
1127 case POWER_SUPPLY_PROP_SW_JEITA_ENABLED:
1128 if (chg->sw_jeita_enabled != (!!val->intval)) {
1129 rc = smblib_disable_hw_jeita(chg, !!val->intval);
Ashay Jaiswala55b7ed2018-03-23 17:23:48 +05301130 enable = !!val->intval || chg->step_chg_enabled;
1131 rc |= smblib_configure_wdog(chg, enable);
Harry Yang4b7db0f2017-11-27 10:50:44 -08001132 if (rc == 0)
1133 chg->sw_jeita_enabled = !!val->intval;
1134 }
1135 break;
1136 case POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX:
1137 chg->batt_profile_fcc_ua = val->intval;
1138 vote(chg->fcc_votable, BATT_PROFILE_VOTER, true, val->intval);
1139 break;
1140 case POWER_SUPPLY_PROP_SET_SHIP_MODE:
1141 /* Not in ship mode as long as the device is active */
1142 if (!val->intval)
1143 break;
1144 if (chg->pl.psy)
1145 power_supply_set_property(chg->pl.psy,
1146 POWER_SUPPLY_PROP_SET_SHIP_MODE, val);
1147 rc = smblib_set_prop_ship_mode(chg, val);
1148 break;
1149 case POWER_SUPPLY_PROP_RERUN_AICL:
1150 rc = smblib_rerun_aicl(chg);
1151 break;
1152 case POWER_SUPPLY_PROP_DP_DM:
1153 rc = smblib_dp_dm(chg, val->intval);
1154 break;
1155 case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMITED:
1156 rc = smblib_set_prop_input_current_limited(chg, val);
1157 break;
1158 case POWER_SUPPLY_PROP_DIE_HEALTH:
1159 chg->die_health = val->intval;
1160 power_supply_changed(chg->batt_psy);
1161 break;
Anirudh Ghayaldfdb8412018-04-21 10:37:18 +05301162 case POWER_SUPPLY_PROP_RECHARGE_SOC:
1163 if (chg->smb_version == PMI632_SUBTYPE) {
1164 /* toggle charging to force recharge */
1165 vote(chg->chg_disable_votable, FORCE_RECHARGE_VOTER,
1166 true, 0);
1167 /* charge disable delay */
1168 msleep(50);
1169 vote(chg->chg_disable_votable, FORCE_RECHARGE_VOTER,
1170 false, 0);
1171 }
1172 break;
Harry Yang4b7db0f2017-11-27 10:50:44 -08001173 default:
1174 rc = -EINVAL;
1175 }
1176
1177 return rc;
1178}
1179
1180static int smb5_batt_prop_is_writeable(struct power_supply *psy,
1181 enum power_supply_property psp)
1182{
1183 switch (psp) {
1184 case POWER_SUPPLY_PROP_STATUS:
1185 case POWER_SUPPLY_PROP_INPUT_SUSPEND:
1186 case POWER_SUPPLY_PROP_SYSTEM_TEMP_LEVEL:
1187 case POWER_SUPPLY_PROP_CAPACITY:
1188 case POWER_SUPPLY_PROP_PARALLEL_DISABLE:
1189 case POWER_SUPPLY_PROP_DP_DM:
1190 case POWER_SUPPLY_PROP_RERUN_AICL:
1191 case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMITED:
1192 case POWER_SUPPLY_PROP_STEP_CHARGING_ENABLED:
1193 case POWER_SUPPLY_PROP_SW_JEITA_ENABLED:
1194 case POWER_SUPPLY_PROP_DIE_HEALTH:
Anirudh Ghayaldfdb8412018-04-21 10:37:18 +05301195 case POWER_SUPPLY_PROP_BATTERY_CHARGING_ENABLED:
Harry Yang4b7db0f2017-11-27 10:50:44 -08001196 return 1;
1197 default:
1198 break;
1199 }
1200
1201 return 0;
1202}
1203
1204static const struct power_supply_desc batt_psy_desc = {
1205 .name = "battery",
1206 .type = POWER_SUPPLY_TYPE_BATTERY,
1207 .properties = smb5_batt_props,
1208 .num_properties = ARRAY_SIZE(smb5_batt_props),
1209 .get_property = smb5_batt_get_prop,
1210 .set_property = smb5_batt_set_prop,
1211 .property_is_writeable = smb5_batt_prop_is_writeable,
1212};
1213
1214static int smb5_init_batt_psy(struct smb5 *chip)
1215{
1216 struct power_supply_config batt_cfg = {};
1217 struct smb_charger *chg = &chip->chg;
1218 int rc = 0;
1219
1220 batt_cfg.drv_data = chg;
1221 batt_cfg.of_node = chg->dev->of_node;
1222 chg->batt_psy = devm_power_supply_register(chg->dev,
1223 &batt_psy_desc,
1224 &batt_cfg);
1225 if (IS_ERR(chg->batt_psy)) {
1226 pr_err("Couldn't register battery power supply\n");
1227 return PTR_ERR(chg->batt_psy);
1228 }
1229
1230 return rc;
1231}
1232
1233/******************************
1234 * VBUS REGULATOR REGISTRATION *
1235 ******************************/
1236
1237static struct regulator_ops smb5_vbus_reg_ops = {
1238 .enable = smblib_vbus_regulator_enable,
1239 .disable = smblib_vbus_regulator_disable,
1240 .is_enabled = smblib_vbus_regulator_is_enabled,
1241};
1242
1243static int smb5_init_vbus_regulator(struct smb5 *chip)
1244{
1245 struct smb_charger *chg = &chip->chg;
1246 struct regulator_config cfg = {};
1247 int rc = 0;
1248
1249 chg->vbus_vreg = devm_kzalloc(chg->dev, sizeof(*chg->vbus_vreg),
1250 GFP_KERNEL);
1251 if (!chg->vbus_vreg)
1252 return -ENOMEM;
1253
1254 cfg.dev = chg->dev;
1255 cfg.driver_data = chip;
1256
1257 chg->vbus_vreg->rdesc.owner = THIS_MODULE;
1258 chg->vbus_vreg->rdesc.type = REGULATOR_VOLTAGE;
1259 chg->vbus_vreg->rdesc.ops = &smb5_vbus_reg_ops;
1260 chg->vbus_vreg->rdesc.of_match = "qcom,smb5-vbus";
1261 chg->vbus_vreg->rdesc.name = "qcom,smb5-vbus";
1262
1263 chg->vbus_vreg->rdev = devm_regulator_register(chg->dev,
1264 &chg->vbus_vreg->rdesc, &cfg);
1265 if (IS_ERR(chg->vbus_vreg->rdev)) {
1266 rc = PTR_ERR(chg->vbus_vreg->rdev);
1267 chg->vbus_vreg->rdev = NULL;
1268 if (rc != -EPROBE_DEFER)
1269 pr_err("Couldn't register VBUS regulator rc=%d\n", rc);
1270 }
1271
1272 return rc;
1273}
1274
1275/******************************
1276 * VCONN REGULATOR REGISTRATION *
1277 ******************************/
1278
1279static struct regulator_ops smb5_vconn_reg_ops = {
1280 .enable = smblib_vconn_regulator_enable,
1281 .disable = smblib_vconn_regulator_disable,
1282 .is_enabled = smblib_vconn_regulator_is_enabled,
1283};
1284
1285static int smb5_init_vconn_regulator(struct smb5 *chip)
1286{
1287 struct smb_charger *chg = &chip->chg;
1288 struct regulator_config cfg = {};
1289 int rc = 0;
1290
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301291 if (chg->connector_type == POWER_SUPPLY_CONNECTOR_MICRO_USB)
Harry Yang4b7db0f2017-11-27 10:50:44 -08001292 return 0;
1293
1294 chg->vconn_vreg = devm_kzalloc(chg->dev, sizeof(*chg->vconn_vreg),
1295 GFP_KERNEL);
1296 if (!chg->vconn_vreg)
1297 return -ENOMEM;
1298
1299 cfg.dev = chg->dev;
1300 cfg.driver_data = chip;
1301
1302 chg->vconn_vreg->rdesc.owner = THIS_MODULE;
1303 chg->vconn_vreg->rdesc.type = REGULATOR_VOLTAGE;
1304 chg->vconn_vreg->rdesc.ops = &smb5_vconn_reg_ops;
1305 chg->vconn_vreg->rdesc.of_match = "qcom,smb5-vconn";
1306 chg->vconn_vreg->rdesc.name = "qcom,smb5-vconn";
1307
1308 chg->vconn_vreg->rdev = devm_regulator_register(chg->dev,
1309 &chg->vconn_vreg->rdesc, &cfg);
1310 if (IS_ERR(chg->vconn_vreg->rdev)) {
1311 rc = PTR_ERR(chg->vconn_vreg->rdev);
1312 chg->vconn_vreg->rdev = NULL;
1313 if (rc != -EPROBE_DEFER)
1314 pr_err("Couldn't register VCONN regulator rc=%d\n", rc);
1315 }
1316
1317 return rc;
1318}
1319
1320/***************************
1321 * HARDWARE INITIALIZATION *
1322 ***************************/
1323static int smb5_configure_typec(struct smb_charger *chg)
1324{
1325 int rc;
1326
1327 rc = smblib_write(chg, TYPE_C_INTERRUPT_EN_CFG_1_REG,
1328 TYPEC_CCOUT_DETACH_INT_EN_BIT |
1329 TYPEC_CCOUT_ATTACH_INT_EN_BIT);
1330 if (rc < 0) {
1331 dev_err(chg->dev,
1332 "Couldn't configure Type-C interrupts rc=%d\n", rc);
1333 return rc;
1334 }
1335
1336 rc = smblib_write(chg, TYPE_C_INTERRUPT_EN_CFG_2_REG,
Harry Yang4b7db0f2017-11-27 10:50:44 -08001337 TYPEC_WATER_DETECTION_INT_EN_BIT);
1338 if (rc < 0) {
1339 dev_err(chg->dev,
1340 "Couldn't configure Type-C interrupts rc=%d\n", rc);
1341 return rc;
1342 }
1343
1344 /* configure VCONN for software control */
1345 rc = smblib_masked_write(chg, TYPE_C_VCONN_CONTROL_REG,
1346 VCONN_EN_SRC_BIT | VCONN_EN_VALUE_BIT,
1347 VCONN_EN_SRC_BIT);
1348 if (rc < 0) {
1349 dev_err(chg->dev,
1350 "Couldn't configure VCONN for SW control rc=%d\n", rc);
1351 return rc;
1352 }
1353
1354 return rc;
1355}
1356
1357static int smb5_configure_micro_usb(struct smb_charger *chg)
1358{
1359 int rc;
1360
1361 /* configure micro USB mode */
1362 rc = smblib_masked_write(chg, TYPEC_U_USB_CFG_REG,
1363 EN_MICRO_USB_MODE_BIT, EN_MICRO_USB_MODE_BIT);
1364 if (rc < 0) {
1365 dev_err(chg->dev, "Couldn't enable micro USB mode rc=%d\n", rc);
1366 return rc;
1367 }
1368
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301369 rc = smblib_masked_write(chg, TYPE_C_INTERRUPT_EN_CFG_2_REG,
1370 MICRO_USB_STATE_CHANGE_INT_EN_BIT,
1371 MICRO_USB_STATE_CHANGE_INT_EN_BIT);
1372 if (rc < 0) {
1373 dev_err(chg->dev,
1374 "Couldn't configure Type-C interrupts rc=%d\n", rc);
1375 return rc;
1376 }
1377
Harry Yang4b7db0f2017-11-27 10:50:44 -08001378 return rc;
1379}
1380
1381static int smb5_init_hw(struct smb5 *chip)
1382{
1383 struct smb_charger *chg = &chip->chg;
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301384 int rc, type = 0;
1385 u8 val = 0;
Harry Yang4b7db0f2017-11-27 10:50:44 -08001386
1387 if (chip->dt.no_battery)
1388 chg->fake_capacity = 50;
1389
1390 if (chip->dt.batt_profile_fcc_ua < 0)
1391 smblib_get_charge_param(chg, &chg->param.fcc,
1392 &chg->batt_profile_fcc_ua);
1393
1394 if (chip->dt.batt_profile_fv_uv < 0)
1395 smblib_get_charge_param(chg, &chg->param.fv,
1396 &chg->batt_profile_fv_uv);
1397
1398 smblib_get_charge_param(chg, &chg->param.usb_icl,
1399 &chg->default_icl_ua);
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301400
1401 /* Use SW based VBUS control, disable HW autonomous mode */
1402 /* TODO: auth can be enabled through vote based on APSD flow */
1403 rc = smblib_masked_write(chg, USBIN_OPTIONS_1_CFG_REG,
1404 HVDCP_AUTH_ALG_EN_CFG_BIT | HVDCP_AUTONOMOUS_MODE_EN_CFG_BIT,
1405 HVDCP_AUTH_ALG_EN_CFG_BIT);
1406 if (rc < 0) {
1407 dev_err(chg->dev, "Couldn't configure HVDCP rc=%d\n", rc);
1408 return rc;
1409 }
1410
1411 /*
1412 * PMI632 can have the connector type defined by a dedicated register
1413 * TYPEC_MICRO_USB_MODE_REG or by a common TYPEC_U_USB_CFG_REG.
1414 */
1415 if (chg->smb_version == PMI632_SUBTYPE) {
1416 rc = smblib_read(chg, TYPEC_MICRO_USB_MODE_REG, &val);
1417 if (rc < 0) {
1418 dev_err(chg->dev, "Couldn't read USB mode rc=%d\n", rc);
1419 return rc;
1420 }
1421 type = !!(val & MICRO_USB_MODE_ONLY_BIT);
1422 }
1423
1424 /*
1425 * If TYPEC_MICRO_USB_MODE_REG is not set and for all non-PMI632
1426 * check the connector type using TYPEC_U_USB_CFG_REG.
1427 */
1428 if (!type) {
1429 rc = smblib_read(chg, TYPEC_U_USB_CFG_REG, &val);
1430 if (rc < 0) {
1431 dev_err(chg->dev, "Couldn't read U_USB config rc=%d\n",
1432 rc);
1433 return rc;
1434 }
1435
1436 type = !!(val & EN_MICRO_USB_MODE_BIT);
1437 }
1438
1439 chg->connector_type = type ? POWER_SUPPLY_CONNECTOR_MICRO_USB
1440 : POWER_SUPPLY_CONNECTOR_TYPEC;
1441 pr_debug("Connector type=%s\n", type ? "Micro USB" : "TypeC");
1442
Ashay Jaiswal09feab82018-02-12 12:33:18 +05301443 /*
1444 * PMI632 based hw init:
1445 * - Initialize flash module for PMI632
1446 */
1447 if (chg->smb_version == PMI632_SUBTYPE)
1448 schgm_flash_init(chg);
1449
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301450 smblib_rerun_apsd_if_required(chg);
Harry Yang4b7db0f2017-11-27 10:50:44 -08001451
1452 /* vote 0mA on usb_icl for non battery platforms */
1453 vote(chg->usb_icl_votable,
1454 DEFAULT_VOTER, chip->dt.no_battery, 0);
1455 vote(chg->dc_suspend_votable,
1456 DEFAULT_VOTER, chip->dt.no_battery, 0);
1457 vote(chg->fcc_votable, HW_LIMIT_VOTER,
1458 chip->dt.batt_profile_fcc_ua > 0, chip->dt.batt_profile_fcc_ua);
1459 vote(chg->fv_votable, HW_LIMIT_VOTER,
1460 chip->dt.batt_profile_fv_uv > 0, chip->dt.batt_profile_fv_uv);
1461 vote(chg->fcc_votable,
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301462 BATT_PROFILE_VOTER, chg->batt_profile_fcc_ua > 0,
1463 chg->batt_profile_fcc_ua);
Harry Yang4b7db0f2017-11-27 10:50:44 -08001464 vote(chg->fv_votable,
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301465 BATT_PROFILE_VOTER, chg->batt_profile_fv_uv > 0,
1466 chg->batt_profile_fv_uv);
Harry Yang4b7db0f2017-11-27 10:50:44 -08001467 vote(chg->pd_disallowed_votable_indirect, CC_DETACHED_VOTER,
1468 true, 0);
1469 vote(chg->pd_disallowed_votable_indirect, APSD_VOTER,
1470 true, 0);
1471 vote(chg->pd_disallowed_votable_indirect, MICRO_USB_VOTER,
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301472 chg->connector_type == POWER_SUPPLY_CONNECTOR_MICRO_USB, 0);
1473
1474 /* Some h/w limit maximum supported ICL */
1475 vote(chg->usb_icl_votable, HW_LIMIT_VOTER,
1476 chg->hw_max_icl_ua > 0, chg->hw_max_icl_ua);
Harry Yang4b7db0f2017-11-27 10:50:44 -08001477
1478 /*
1479 * AICL configuration:
1480 * start from min and AICL ADC disable
1481 */
1482 rc = smblib_masked_write(chg, USBIN_AICL_OPTIONS_CFG_REG,
1483 USBIN_AICL_ADC_EN_BIT, 0);
1484 if (rc < 0) {
1485 dev_err(chg->dev, "Couldn't configure AICL rc=%d\n", rc);
1486 return rc;
1487 }
1488
1489 /* enable the charging path */
1490 rc = vote(chg->chg_disable_votable, DEFAULT_VOTER, false, 0);
1491 if (rc < 0) {
1492 dev_err(chg->dev, "Couldn't enable charging rc=%d\n", rc);
1493 return rc;
1494 }
1495
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301496 if (chg->connector_type == POWER_SUPPLY_CONNECTOR_MICRO_USB)
Harry Yang4b7db0f2017-11-27 10:50:44 -08001497 rc = smb5_configure_micro_usb(chg);
1498 else
1499 rc = smb5_configure_typec(chg);
1500 if (rc < 0) {
1501 dev_err(chg->dev,
1502 "Couldn't configure TypeC/micro-USB mode rc=%d\n", rc);
1503 return rc;
1504 }
1505
1506 /* configure VBUS for software control */
1507 rc = smblib_masked_write(chg, DCDC_OTG_CFG_REG, OTG_EN_SRC_CFG_BIT, 0);
1508 if (rc < 0) {
1509 dev_err(chg->dev,
1510 "Couldn't configure VBUS for SW control rc=%d\n", rc);
1511 return rc;
1512 }
1513
Ashay Jaiswala55b7ed2018-03-23 17:23:48 +05301514 /*
1515 * configure the one time watchdong periodic interval and
1516 * disable "watchdog bite disable charging".
1517 */
Harry Yang4b7db0f2017-11-27 10:50:44 -08001518 val = (ilog2(chip->dt.wd_bark_time / 16) << BARK_WDOG_TIMEOUT_SHIFT)
1519 & BARK_WDOG_TIMEOUT_MASK;
1520 val |= BITE_WDOG_TIMEOUT_8S;
1521 rc = smblib_masked_write(chg, SNARL_BARK_BITE_WD_CFG_REG,
1522 BITE_WDOG_DISABLE_CHARGING_CFG_BIT |
1523 BARK_WDOG_TIMEOUT_MASK | BITE_WDOG_TIMEOUT_MASK,
1524 val);
1525 if (rc < 0) {
1526 pr_err("Couldn't configue WD config rc=%d\n", rc);
1527 return rc;
1528 }
1529
Harry Yang4b7db0f2017-11-27 10:50:44 -08001530 /* configure float charger options */
1531 switch (chip->dt.float_option) {
1532 case FLOAT_DCP:
1533 rc = smblib_masked_write(chg, USBIN_OPTIONS_2_CFG_REG,
1534 FLOAT_OPTIONS_MASK, 0);
1535 break;
1536 case FLOAT_SDP:
1537 rc = smblib_masked_write(chg, USBIN_OPTIONS_2_CFG_REG,
1538 FLOAT_OPTIONS_MASK, FORCE_FLOAT_SDP_CFG_BIT);
1539 break;
1540 case DISABLE_CHARGING:
1541 rc = smblib_masked_write(chg, USBIN_OPTIONS_2_CFG_REG,
1542 FLOAT_OPTIONS_MASK, FLOAT_DIS_CHGING_CFG_BIT);
1543 break;
1544 case SUSPEND_INPUT:
1545 rc = smblib_masked_write(chg, USBIN_OPTIONS_2_CFG_REG,
1546 FLOAT_OPTIONS_MASK, SUSPEND_FLOAT_CFG_BIT);
1547 break;
1548 default:
1549 rc = 0;
1550 break;
1551 }
1552
1553 if (rc < 0) {
1554 dev_err(chg->dev, "Couldn't configure float charger options rc=%d\n",
1555 rc);
1556 return rc;
1557 }
1558
1559 rc = smblib_read(chg, USBIN_OPTIONS_2_CFG_REG, &chg->float_cfg);
1560 if (rc < 0) {
1561 dev_err(chg->dev, "Couldn't read float charger options rc=%d\n",
1562 rc);
1563 return rc;
1564 }
1565
1566 switch (chip->dt.chg_inhibit_thr_mv) {
1567 case 50:
1568 rc = smblib_masked_write(chg, CHARGE_INHIBIT_THRESHOLD_CFG_REG,
1569 CHARGE_INHIBIT_THRESHOLD_MASK,
1570 INHIBIT_ANALOG_VFLT_MINUS_50MV);
1571 break;
1572 case 100:
1573 rc = smblib_masked_write(chg, CHARGE_INHIBIT_THRESHOLD_CFG_REG,
1574 CHARGE_INHIBIT_THRESHOLD_MASK,
1575 INHIBIT_ANALOG_VFLT_MINUS_100MV);
1576 break;
1577 case 200:
1578 rc = smblib_masked_write(chg, CHARGE_INHIBIT_THRESHOLD_CFG_REG,
1579 CHARGE_INHIBIT_THRESHOLD_MASK,
1580 INHIBIT_ANALOG_VFLT_MINUS_200MV);
1581 break;
1582 case 300:
1583 rc = smblib_masked_write(chg, CHARGE_INHIBIT_THRESHOLD_CFG_REG,
1584 CHARGE_INHIBIT_THRESHOLD_MASK,
1585 INHIBIT_ANALOG_VFLT_MINUS_300MV);
1586 break;
1587 case 0:
1588 rc = smblib_masked_write(chg, CHGR_CFG2_REG,
1589 CHARGER_INHIBIT_BIT, 0);
1590 default:
1591 break;
1592 }
1593
1594 if (rc < 0) {
1595 dev_err(chg->dev, "Couldn't configure charge inhibit threshold rc=%d\n",
1596 rc);
1597 return rc;
1598 }
1599
Anirudh Ghayal1380d312018-02-23 00:01:43 +05301600 rc = smblib_masked_write(chg, CHGR_CFG2_REG, RECHG_MASK,
1601 (chip->dt.auto_recharge_vbat_mv != -EINVAL) ?
1602 VBAT_BASED_RECHG_BIT : 0);
Harry Yang4b7db0f2017-11-27 10:50:44 -08001603 if (rc < 0) {
Anirudh Ghayal1380d312018-02-23 00:01:43 +05301604 dev_err(chg->dev, "Couldn't configure VBAT-rechg CHG_CFG2_REG rc=%d\n",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001605 rc);
1606 return rc;
1607 }
1608
Anirudh Ghayal1380d312018-02-23 00:01:43 +05301609 /* program the auto-recharge VBAT threshold */
1610 if (chip->dt.auto_recharge_vbat_mv != -EINVAL) {
1611 u32 temp = VBAT_TO_VRAW_ADC(chip->dt.auto_recharge_vbat_mv);
1612
1613 temp = ((temp & 0xFF00) >> 8) | ((temp & 0xFF) << 8);
1614 rc = smblib_batch_write(chg,
1615 CHGR_ADC_RECHARGE_THRESHOLD_MSB_REG, (u8 *)&temp, 2);
1616 if (rc < 0) {
1617 dev_err(chg->dev, "Couldn't configure ADC_RECHARGE_THRESHOLD REG rc=%d\n",
1618 rc);
1619 return rc;
1620 }
1621 /* Program the sample count for VBAT based recharge to 3 */
1622 rc = smblib_masked_write(chg, CHGR_NO_SAMPLE_TERM_RCHG_CFG_REG,
1623 NO_OF_SAMPLE_FOR_RCHG,
1624 2 << NO_OF_SAMPLE_FOR_RCHG_SHIFT);
1625 if (rc < 0) {
1626 dev_err(chg->dev, "Couldn't configure CHGR_NO_SAMPLE_FOR_TERM_RCHG_CFG rc=%d\n",
1627 rc);
1628 return rc;
1629 }
1630 }
1631
1632 rc = smblib_masked_write(chg, CHGR_CFG2_REG, RECHG_MASK,
1633 (chip->dt.auto_recharge_soc != -EINVAL) ?
1634 SOC_BASED_RECHG_BIT : VBAT_BASED_RECHG_BIT);
1635 if (rc < 0) {
1636 dev_err(chg->dev, "Couldn't configure SOC-rechg CHG_CFG2_REG rc=%d\n",
1637 rc);
1638 return rc;
1639 }
1640
1641 /* program the auto-recharge threshold */
1642 if (chip->dt.auto_recharge_soc != -EINVAL) {
1643 rc = smblib_write(chg, CHARGE_RCHG_SOC_THRESHOLD_CFG_REG,
1644 (chip->dt.auto_recharge_soc * 255) / 100);
1645 if (rc < 0) {
1646 dev_err(chg->dev, "Couldn't configure CHG_RCHG_SOC_REG rc=%d\n",
1647 rc);
1648 return rc;
1649 }
1650 /* Program the sample count for SOC based recharge to 1 */
1651 rc = smblib_masked_write(chg, CHGR_NO_SAMPLE_TERM_RCHG_CFG_REG,
1652 NO_OF_SAMPLE_FOR_RCHG, 0);
1653 if (rc < 0) {
1654 dev_err(chg->dev, "Couldn't configure CHGR_NO_SAMPLE_FOR_TERM_RCHG_CFG rc=%d\n",
1655 rc);
1656 return rc;
1657 }
1658 }
1659
Harry Yang4b7db0f2017-11-27 10:50:44 -08001660 if (chg->sw_jeita_enabled) {
1661 rc = smblib_disable_hw_jeita(chg, true);
1662 if (rc < 0) {
1663 dev_err(chg->dev, "Couldn't set hw jeita rc=%d\n", rc);
1664 return rc;
1665 }
1666 }
1667
Ashay Jaiswala55b7ed2018-03-23 17:23:48 +05301668 rc = smblib_configure_wdog(chg,
1669 chg->step_chg_enabled || chg->sw_jeita_enabled);
1670 if (rc < 0) {
1671 dev_err(chg->dev, "Couldn't configure watchdog rc=%d\n",
1672 rc);
1673 return rc;
1674 }
1675
Harry Yang4b7db0f2017-11-27 10:50:44 -08001676 return rc;
1677}
1678
1679static int smb5_post_init(struct smb5 *chip)
1680{
1681 struct smb_charger *chg = &chip->chg;
1682 union power_supply_propval pval;
1683 int rc;
1684
1685 /*
1686 * In case the usb path is suspended, we would have missed disabling
1687 * the icl change interrupt because the interrupt could have been
1688 * not requested
1689 */
1690 rerun_election(chg->usb_icl_votable);
1691
1692 /* configure power role for dual-role */
1693 pval.intval = POWER_SUPPLY_TYPEC_PR_DUAL;
1694 rc = smblib_set_prop_typec_power_role(chg, &pval);
1695 if (rc < 0) {
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301696 dev_err(chg->dev, "Couldn't configure DRP role rc=%d\n",
1697 rc);
Harry Yang4b7db0f2017-11-27 10:50:44 -08001698 return rc;
1699 }
1700
1701 rerun_election(chg->usb_irq_enable_votable);
1702
1703 return 0;
1704}
1705
1706/****************************
1707 * DETERMINE INITIAL STATUS *
1708 ****************************/
1709
1710static int smb5_determine_initial_status(struct smb5 *chip)
1711{
1712 struct smb_irq_data irq_data = {chip, "determine-initial-status"};
1713 struct smb_charger *chg = &chip->chg;
1714
1715 if (chg->bms_psy)
1716 smblib_suspend_on_debug_battery(chg);
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301717
Harry Yang4b7db0f2017-11-27 10:50:44 -08001718 usb_plugin_irq_handler(0, &irq_data);
1719 typec_state_change_irq_handler(0, &irq_data);
1720 usb_source_change_irq_handler(0, &irq_data);
1721 chg_state_change_irq_handler(0, &irq_data);
1722 icl_change_irq_handler(0, &irq_data);
1723 batt_temp_changed_irq_handler(0, &irq_data);
1724 wdog_bark_irq_handler(0, &irq_data);
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301725 typec_or_rid_detection_change_irq_handler(0, &irq_data);
Harry Yang4b7db0f2017-11-27 10:50:44 -08001726
1727 return 0;
1728}
1729
1730/**************************
1731 * INTERRUPT REGISTRATION *
1732 **************************/
1733
1734static struct smb_irq_info smb5_irqs[] = {
1735 /* CHARGER IRQs */
1736 [CHGR_ERROR_IRQ] = {
1737 .name = "chgr-error",
1738 .handler = default_irq_handler,
1739 },
1740 [CHG_STATE_CHANGE_IRQ] = {
1741 .name = "chg-state-change",
1742 .handler = chg_state_change_irq_handler,
Ashay Jaiswal48015c32018-03-07 18:25:43 +05301743 .wake = true,
Harry Yang4b7db0f2017-11-27 10:50:44 -08001744 },
1745 [STEP_CHG_STATE_CHANGE_IRQ] = {
1746 .name = "step-chg-state-change",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001747 },
1748 [STEP_CHG_SOC_UPDATE_FAIL_IRQ] = {
1749 .name = "step-chg-soc-update-fail",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001750 },
1751 [STEP_CHG_SOC_UPDATE_REQ_IRQ] = {
1752 .name = "step-chg-soc-update-req",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001753 },
1754 [FG_FVCAL_QUALIFIED_IRQ] = {
1755 .name = "fg-fvcal-qualified",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001756 },
1757 [VPH_ALARM_IRQ] = {
1758 .name = "vph-alarm",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001759 },
1760 [VPH_DROP_PRECHG_IRQ] = {
1761 .name = "vph-drop-prechg",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001762 },
1763 /* DCDC IRQs */
1764 [OTG_FAIL_IRQ] = {
1765 .name = "otg-fail",
1766 .handler = default_irq_handler,
1767 },
1768 [OTG_OC_DISABLE_SW_IRQ] = {
1769 .name = "otg-oc-disable-sw",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001770 },
1771 [OTG_OC_HICCUP_IRQ] = {
1772 .name = "otg-oc-hiccup",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001773 },
1774 [BSM_ACTIVE_IRQ] = {
1775 .name = "bsm-active",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001776 },
1777 [HIGH_DUTY_CYCLE_IRQ] = {
1778 .name = "high-duty-cycle",
1779 .handler = high_duty_cycle_irq_handler,
Ashay Jaiswal48015c32018-03-07 18:25:43 +05301780 .wake = true,
Harry Yang4b7db0f2017-11-27 10:50:44 -08001781 },
1782 [INPUT_CURRENT_LIMITING_IRQ] = {
1783 .name = "input-current-limiting",
1784 .handler = default_irq_handler,
1785 },
1786 [CONCURRENT_MODE_DISABLE_IRQ] = {
1787 .name = "concurrent-mode-disable",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001788 },
1789 [SWITCHER_POWER_OK_IRQ] = {
1790 .name = "switcher-power-ok",
1791 .handler = switcher_power_ok_irq_handler,
1792 },
1793 /* BATTERY IRQs */
1794 [BAT_TEMP_IRQ] = {
1795 .name = "bat-temp",
1796 .handler = batt_temp_changed_irq_handler,
Ashay Jaiswal48015c32018-03-07 18:25:43 +05301797 .wake = true,
Harry Yang4b7db0f2017-11-27 10:50:44 -08001798 },
1799 [ALL_CHNL_CONV_DONE_IRQ] = {
1800 .name = "all-chnl-conv-done",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001801 },
1802 [BAT_OV_IRQ] = {
1803 .name = "bat-ov",
1804 .handler = batt_psy_changed_irq_handler,
1805 },
1806 [BAT_LOW_IRQ] = {
1807 .name = "bat-low",
1808 .handler = batt_psy_changed_irq_handler,
1809 },
1810 [BAT_THERM_OR_ID_MISSING_IRQ] = {
1811 .name = "bat-therm-or-id-missing",
1812 .handler = batt_psy_changed_irq_handler,
1813 },
1814 [BAT_TERMINAL_MISSING_IRQ] = {
1815 .name = "bat-terminal-missing",
1816 .handler = batt_psy_changed_irq_handler,
1817 },
1818 [BUCK_OC_IRQ] = {
1819 .name = "buck-oc",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001820 },
1821 [VPH_OV_IRQ] = {
1822 .name = "vph-ov",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001823 },
1824 /* USB INPUT IRQs */
1825 [USBIN_COLLAPSE_IRQ] = {
1826 .name = "usbin-collapse",
1827 .handler = default_irq_handler,
1828 },
1829 [USBIN_VASHDN_IRQ] = {
1830 .name = "usbin-vashdn",
1831 .handler = default_irq_handler,
1832 },
1833 [USBIN_UV_IRQ] = {
1834 .name = "usbin-uv",
1835 .handler = usbin_uv_irq_handler,
1836 },
1837 [USBIN_OV_IRQ] = {
1838 .name = "usbin-ov",
1839 .handler = default_irq_handler,
1840 },
1841 [USBIN_PLUGIN_IRQ] = {
1842 .name = "usbin-plugin",
1843 .handler = usb_plugin_irq_handler,
Ashay Jaiswal48015c32018-03-07 18:25:43 +05301844 .wake = true,
Harry Yang4b7db0f2017-11-27 10:50:44 -08001845 },
1846 [USBIN_REVI_CHANGE_IRQ] = {
1847 .name = "usbin-revi-change",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001848 },
1849 [USBIN_SRC_CHANGE_IRQ] = {
1850 .name = "usbin-src-change",
1851 .handler = usb_source_change_irq_handler,
Ashay Jaiswal48015c32018-03-07 18:25:43 +05301852 .wake = true,
Harry Yang4b7db0f2017-11-27 10:50:44 -08001853 },
1854 [USBIN_ICL_CHANGE_IRQ] = {
1855 .name = "usbin-icl-change",
1856 .handler = icl_change_irq_handler,
Ashay Jaiswal48015c32018-03-07 18:25:43 +05301857 .wake = true,
Harry Yang4b7db0f2017-11-27 10:50:44 -08001858 },
1859 /* DC INPUT IRQs */
1860 [DCIN_VASHDN_IRQ] = {
1861 .name = "dcin-vashdn",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001862 },
1863 [DCIN_UV_IRQ] = {
1864 .name = "dcin-uv",
1865 .handler = default_irq_handler,
1866 },
1867 [DCIN_OV_IRQ] = {
1868 .name = "dcin-ov",
1869 .handler = default_irq_handler,
1870 },
1871 [DCIN_PLUGIN_IRQ] = {
1872 .name = "dcin-plugin",
1873 .handler = dc_plugin_irq_handler,
1874 .wake = true,
1875 },
1876 [DCIN_REVI_IRQ] = {
1877 .name = "dcin-revi",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001878 },
1879 [DCIN_PON_IRQ] = {
1880 .name = "dcin-pon",
1881 .handler = default_irq_handler,
1882 },
1883 [DCIN_EN_IRQ] = {
1884 .name = "dcin-en",
1885 .handler = default_irq_handler,
1886 },
1887 /* TYPEC IRQs */
1888 [TYPEC_OR_RID_DETECTION_CHANGE_IRQ] = {
1889 .name = "typec-or-rid-detect-change",
Ashay Jaiswala9e10912018-02-02 14:03:35 +05301890 .handler = typec_or_rid_detection_change_irq_handler,
Ashay Jaiswal48015c32018-03-07 18:25:43 +05301891 .wake = true,
Harry Yang4b7db0f2017-11-27 10:50:44 -08001892 },
1893 [TYPEC_VPD_DETECT_IRQ] = {
1894 .name = "typec-vpd-detect",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001895 },
1896 [TYPEC_CC_STATE_CHANGE_IRQ] = {
1897 .name = "typec-cc-state-change",
1898 .handler = typec_state_change_irq_handler,
Ashay Jaiswal48015c32018-03-07 18:25:43 +05301899 .wake = true,
Harry Yang4b7db0f2017-11-27 10:50:44 -08001900 },
1901 [TYPEC_VCONN_OC_IRQ] = {
1902 .name = "typec-vconn-oc",
1903 .handler = default_irq_handler,
1904 },
1905 [TYPEC_VBUS_CHANGE_IRQ] = {
1906 .name = "typec-vbus-change",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001907 },
1908 [TYPEC_ATTACH_DETACH_IRQ] = {
1909 .name = "typec-attach-detach",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001910 },
1911 [TYPEC_LEGACY_CABLE_DETECT_IRQ] = {
1912 .name = "typec-legacy-cable-detect",
1913 .handler = default_irq_handler,
1914 },
1915 [TYPEC_TRY_SNK_SRC_DETECT_IRQ] = {
1916 .name = "typec-try-snk-src-detect",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001917 },
1918 /* MISCELLANEOUS IRQs */
1919 [WDOG_SNARL_IRQ] = {
1920 .name = "wdog-snarl",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001921 },
1922 [WDOG_BARK_IRQ] = {
1923 .name = "wdog-bark",
1924 .handler = wdog_bark_irq_handler,
Ashay Jaiswala55b7ed2018-03-23 17:23:48 +05301925 .wake = true,
Harry Yang4b7db0f2017-11-27 10:50:44 -08001926 },
1927 [AICL_FAIL_IRQ] = {
1928 .name = "aicl-fail",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001929 },
1930 [AICL_DONE_IRQ] = {
1931 .name = "aicl-done",
1932 .handler = default_irq_handler,
1933 },
1934 [SMB_EN_IRQ] = {
1935 .name = "smb-en",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001936 },
1937 [IMP_TRIGGER_IRQ] = {
1938 .name = "imp-trigger",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001939 },
1940 [TEMP_CHANGE_IRQ] = {
1941 .name = "temp-change",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001942 },
1943 [TEMP_CHANGE_SMB_IRQ] = {
1944 .name = "temp-change-smb",
Harry Yang4b7db0f2017-11-27 10:50:44 -08001945 },
Ashay Jaiswal09feab82018-02-12 12:33:18 +05301946 /* FLASH */
1947 [VREG_OK_IRQ] = {
1948 .name = "vreg-ok",
Ashay Jaiswal09feab82018-02-12 12:33:18 +05301949 },
1950 [ILIM_S2_IRQ] = {
1951 .name = "ilim2-s2",
1952 .handler = schgm_flash_ilim2_irq_handler,
1953 },
1954 [ILIM_S1_IRQ] = {
1955 .name = "ilim1-s1",
Ashay Jaiswal09feab82018-02-12 12:33:18 +05301956 },
1957 [VOUT_DOWN_IRQ] = {
1958 .name = "vout-down",
Ashay Jaiswal09feab82018-02-12 12:33:18 +05301959 },
1960 [VOUT_UP_IRQ] = {
1961 .name = "vout-up",
Ashay Jaiswal09feab82018-02-12 12:33:18 +05301962 },
1963 [FLASH_STATE_CHANGE_IRQ] = {
1964 .name = "flash-state-change",
1965 .handler = schgm_flash_state_change_irq_handler,
1966 },
1967 [TORCH_REQ_IRQ] = {
1968 .name = "torch-req",
Ashay Jaiswal09feab82018-02-12 12:33:18 +05301969 },
1970 [FLASH_EN_IRQ] = {
1971 .name = "flash-en",
Ashay Jaiswal09feab82018-02-12 12:33:18 +05301972 },
Harry Yang4b7db0f2017-11-27 10:50:44 -08001973};
1974
1975static int smb5_get_irq_index_byname(const char *irq_name)
1976{
1977 int i;
1978
1979 for (i = 0; i < ARRAY_SIZE(smb5_irqs); i++) {
1980 if (strcmp(smb5_irqs[i].name, irq_name) == 0)
1981 return i;
1982 }
1983
1984 return -ENOENT;
1985}
1986
1987static int smb5_request_interrupt(struct smb5 *chip,
1988 struct device_node *node, const char *irq_name)
1989{
1990 struct smb_charger *chg = &chip->chg;
1991 int rc, irq, irq_index;
1992 struct smb_irq_data *irq_data;
1993
1994 irq = of_irq_get_byname(node, irq_name);
1995 if (irq < 0) {
1996 pr_err("Couldn't get irq %s byname\n", irq_name);
1997 return irq;
1998 }
1999
2000 irq_index = smb5_get_irq_index_byname(irq_name);
2001 if (irq_index < 0) {
2002 pr_err("%s is not a defined irq\n", irq_name);
2003 return irq_index;
2004 }
2005
2006 if (!smb5_irqs[irq_index].handler)
2007 return 0;
2008
2009 irq_data = devm_kzalloc(chg->dev, sizeof(*irq_data), GFP_KERNEL);
2010 if (!irq_data)
2011 return -ENOMEM;
2012
2013 irq_data->parent_data = chip;
2014 irq_data->name = irq_name;
2015 irq_data->storm_data = smb5_irqs[irq_index].storm_data;
2016 mutex_init(&irq_data->storm_data.storm_lock);
2017
2018 rc = devm_request_threaded_irq(chg->dev, irq, NULL,
2019 smb5_irqs[irq_index].handler,
2020 IRQF_ONESHOT, irq_name, irq_data);
2021 if (rc < 0) {
2022 pr_err("Couldn't request irq %d\n", irq);
2023 return rc;
2024 }
2025
2026 smb5_irqs[irq_index].irq = irq;
2027 smb5_irqs[irq_index].irq_data = irq_data;
2028 if (smb5_irqs[irq_index].wake)
2029 enable_irq_wake(irq);
2030
2031 return rc;
2032}
2033
2034static int smb5_request_interrupts(struct smb5 *chip)
2035{
2036 struct smb_charger *chg = &chip->chg;
2037 struct device_node *node = chg->dev->of_node;
2038 struct device_node *child;
2039 int rc = 0;
2040 const char *name;
2041 struct property *prop;
2042
2043 for_each_available_child_of_node(node, child) {
2044 of_property_for_each_string(child, "interrupt-names",
2045 prop, name) {
2046 rc = smb5_request_interrupt(chip, child, name);
2047 if (rc < 0)
2048 return rc;
2049 }
2050 }
2051 if (chg->irq_info[USBIN_ICL_CHANGE_IRQ].irq)
2052 chg->usb_icl_change_irq_enabled = true;
2053
2054 return rc;
2055}
2056
2057static void smb5_free_interrupts(struct smb_charger *chg)
2058{
2059 int i;
2060
2061 for (i = 0; i < ARRAY_SIZE(smb5_irqs); i++) {
2062 if (smb5_irqs[i].irq > 0) {
2063 if (smb5_irqs[i].wake)
2064 disable_irq_wake(smb5_irqs[i].irq);
2065
2066 devm_free_irq(chg->dev, smb5_irqs[i].irq,
2067 smb5_irqs[i].irq_data);
2068 }
2069 }
2070}
2071
2072static void smb5_disable_interrupts(struct smb_charger *chg)
2073{
2074 int i;
2075
2076 for (i = 0; i < ARRAY_SIZE(smb5_irqs); i++) {
2077 if (smb5_irqs[i].irq > 0)
2078 disable_irq(smb5_irqs[i].irq);
2079 }
2080}
2081
2082#if defined(CONFIG_DEBUG_FS)
2083
2084static int force_batt_psy_update_write(void *data, u64 val)
2085{
2086 struct smb_charger *chg = data;
2087
2088 power_supply_changed(chg->batt_psy);
2089 return 0;
2090}
2091DEFINE_SIMPLE_ATTRIBUTE(force_batt_psy_update_ops, NULL,
2092 force_batt_psy_update_write, "0x%02llx\n");
2093
2094static int force_usb_psy_update_write(void *data, u64 val)
2095{
2096 struct smb_charger *chg = data;
2097
2098 power_supply_changed(chg->usb_psy);
2099 return 0;
2100}
2101DEFINE_SIMPLE_ATTRIBUTE(force_usb_psy_update_ops, NULL,
2102 force_usb_psy_update_write, "0x%02llx\n");
2103
2104static int force_dc_psy_update_write(void *data, u64 val)
2105{
2106 struct smb_charger *chg = data;
2107
2108 power_supply_changed(chg->dc_psy);
2109 return 0;
2110}
2111DEFINE_SIMPLE_ATTRIBUTE(force_dc_psy_update_ops, NULL,
2112 force_dc_psy_update_write, "0x%02llx\n");
2113
2114static void smb5_create_debugfs(struct smb5 *chip)
2115{
2116 struct dentry *file;
2117
2118 chip->dfs_root = debugfs_create_dir("charger", NULL);
2119 if (IS_ERR_OR_NULL(chip->dfs_root)) {
2120 pr_err("Couldn't create charger debugfs rc=%ld\n",
2121 (long)chip->dfs_root);
2122 return;
2123 }
2124
2125 file = debugfs_create_file("force_batt_psy_update", 0600,
2126 chip->dfs_root, chip, &force_batt_psy_update_ops);
2127 if (IS_ERR_OR_NULL(file))
2128 pr_err("Couldn't create force_batt_psy_update file rc=%ld\n",
2129 (long)file);
2130
2131 file = debugfs_create_file("force_usb_psy_update", 0600,
2132 chip->dfs_root, chip, &force_usb_psy_update_ops);
2133 if (IS_ERR_OR_NULL(file))
2134 pr_err("Couldn't create force_usb_psy_update file rc=%ld\n",
2135 (long)file);
2136
2137 file = debugfs_create_file("force_dc_psy_update", 0600,
2138 chip->dfs_root, chip, &force_dc_psy_update_ops);
2139 if (IS_ERR_OR_NULL(file))
2140 pr_err("Couldn't create force_dc_psy_update file rc=%ld\n",
2141 (long)file);
2142}
2143
2144#else
2145
2146static void smb5_create_debugfs(struct smb5 *chip)
2147{}
2148
2149#endif
2150
2151static int smb5_show_charger_status(struct smb5 *chip)
2152{
2153 struct smb_charger *chg = &chip->chg;
2154 union power_supply_propval val;
2155 int usb_present, batt_present, batt_health, batt_charge_type;
2156 int rc;
2157
2158 rc = smblib_get_prop_usb_present(chg, &val);
2159 if (rc < 0) {
2160 pr_err("Couldn't get usb present rc=%d\n", rc);
2161 return rc;
2162 }
2163 usb_present = val.intval;
2164
2165 rc = smblib_get_prop_batt_present(chg, &val);
2166 if (rc < 0) {
2167 pr_err("Couldn't get batt present rc=%d\n", rc);
2168 return rc;
2169 }
2170 batt_present = val.intval;
2171
2172 rc = smblib_get_prop_batt_health(chg, &val);
2173 if (rc < 0) {
2174 pr_err("Couldn't get batt health rc=%d\n", rc);
2175 val.intval = POWER_SUPPLY_HEALTH_UNKNOWN;
2176 }
2177 batt_health = val.intval;
2178
2179 rc = smblib_get_prop_batt_charge_type(chg, &val);
2180 if (rc < 0) {
2181 pr_err("Couldn't get batt charge type rc=%d\n", rc);
2182 return rc;
2183 }
2184 batt_charge_type = val.intval;
2185
2186 pr_info("SMB5 status - usb:present=%d type=%d batt:present = %d health = %d charge = %d\n",
2187 usb_present, chg->real_charger_type,
2188 batt_present, batt_health, batt_charge_type);
2189 return rc;
2190}
2191
2192static int smb5_probe(struct platform_device *pdev)
2193{
2194 struct smb5 *chip;
2195 struct smb_charger *chg;
2196 int rc = 0;
2197
2198 chip = devm_kzalloc(&pdev->dev, sizeof(*chip), GFP_KERNEL);
2199 if (!chip)
2200 return -ENOMEM;
2201
2202 chg = &chip->chg;
2203 chg->dev = &pdev->dev;
Harry Yang4b7db0f2017-11-27 10:50:44 -08002204 chg->debug_mask = &__debug_mask;
2205 chg->weak_chg_icl_ua = &__weak_chg_icl_ua;
2206 chg->mode = PARALLEL_MASTER;
2207 chg->irq_info = smb5_irqs;
2208 chg->die_health = -EINVAL;
Ashay Jaiswala9e10912018-02-02 14:03:35 +05302209 chg->otg_present = false;
Harry Yang4b7db0f2017-11-27 10:50:44 -08002210
2211 chg->regmap = dev_get_regmap(chg->dev->parent, NULL);
2212 if (!chg->regmap) {
2213 pr_err("parent regmap is missing\n");
2214 return -EINVAL;
2215 }
2216
2217 rc = smb5_parse_dt(chip);
2218 if (rc < 0) {
2219 pr_err("Couldn't parse device tree rc=%d\n", rc);
Ashay Jaiswala9e10912018-02-02 14:03:35 +05302220 return rc;
2221 }
2222
2223 rc = smb5_chg_config_init(chip);
2224 if (rc < 0) {
2225 if (rc != -EPROBE_DEFER)
2226 pr_err("Couldn't setup chg_config rc=%d\n", rc);
2227 return rc;
Harry Yang4b7db0f2017-11-27 10:50:44 -08002228 }
2229
2230 rc = smblib_init(chg);
2231 if (rc < 0) {
2232 pr_err("Smblib_init failed rc=%d\n", rc);
Ashay Jaiswala9e10912018-02-02 14:03:35 +05302233 return rc;
Harry Yang4b7db0f2017-11-27 10:50:44 -08002234 }
2235
2236 /* set driver data before resources request it */
2237 platform_set_drvdata(pdev, chip);
2238
2239 rc = smb5_init_vbus_regulator(chip);
2240 if (rc < 0) {
2241 pr_err("Couldn't initialize vbus regulator rc=%d\n",
2242 rc);
2243 goto cleanup;
2244 }
2245
2246 rc = smb5_init_vconn_regulator(chip);
2247 if (rc < 0) {
2248 pr_err("Couldn't initialize vconn regulator rc=%d\n",
2249 rc);
2250 goto cleanup;
2251 }
2252
2253 /* extcon registration */
2254 chg->extcon = devm_extcon_dev_allocate(chg->dev, smblib_extcon_cable);
2255 if (IS_ERR(chg->extcon)) {
2256 rc = PTR_ERR(chg->extcon);
2257 dev_err(chg->dev, "failed to allocate extcon device rc=%d\n",
2258 rc);
2259 goto cleanup;
2260 }
2261
2262 rc = devm_extcon_dev_register(chg->dev, chg->extcon);
2263 if (rc < 0) {
2264 dev_err(chg->dev, "failed to register extcon device rc=%d\n",
2265 rc);
2266 goto cleanup;
2267 }
2268
2269 rc = smb5_init_hw(chip);
2270 if (rc < 0) {
2271 pr_err("Couldn't initialize hardware rc=%d\n", rc);
2272 goto cleanup;
2273 }
2274
Ashay Jaiswala9e10912018-02-02 14:03:35 +05302275 if (chg->smb_version == PM855B_SUBTYPE) {
2276 rc = smb5_init_dc_psy(chip);
2277 if (rc < 0) {
2278 pr_err("Couldn't initialize dc psy rc=%d\n", rc);
2279 goto cleanup;
2280 }
Harry Yang4b7db0f2017-11-27 10:50:44 -08002281 }
2282
2283 rc = smb5_init_usb_psy(chip);
2284 if (rc < 0) {
2285 pr_err("Couldn't initialize usb psy rc=%d\n", rc);
2286 goto cleanup;
2287 }
2288
2289 rc = smb5_init_usb_main_psy(chip);
2290 if (rc < 0) {
2291 pr_err("Couldn't initialize usb main psy rc=%d\n", rc);
2292 goto cleanup;
2293 }
2294
2295 rc = smb5_init_usb_port_psy(chip);
2296 if (rc < 0) {
2297 pr_err("Couldn't initialize usb pc_port psy rc=%d\n", rc);
2298 goto cleanup;
2299 }
2300
2301 rc = smb5_init_batt_psy(chip);
2302 if (rc < 0) {
2303 pr_err("Couldn't initialize batt psy rc=%d\n", rc);
2304 goto cleanup;
2305 }
2306
2307 rc = smb5_determine_initial_status(chip);
2308 if (rc < 0) {
2309 pr_err("Couldn't determine initial status rc=%d\n",
2310 rc);
2311 goto cleanup;
2312 }
2313
2314 rc = smb5_request_interrupts(chip);
2315 if (rc < 0) {
2316 pr_err("Couldn't request interrupts rc=%d\n", rc);
2317 goto cleanup;
2318 }
2319
2320 rc = smb5_post_init(chip);
2321 if (rc < 0) {
2322 pr_err("Failed in post init rc=%d\n", rc);
Ashay Jaiswala9e10912018-02-02 14:03:35 +05302323 goto free_irq;
Harry Yang4b7db0f2017-11-27 10:50:44 -08002324 }
2325
2326 smb5_create_debugfs(chip);
2327
2328 rc = smb5_show_charger_status(chip);
2329 if (rc < 0) {
2330 pr_err("Failed in getting charger status rc=%d\n", rc);
Ashay Jaiswala9e10912018-02-02 14:03:35 +05302331 goto free_irq;
Harry Yang4b7db0f2017-11-27 10:50:44 -08002332 }
2333
2334 device_init_wakeup(chg->dev, true);
2335
2336 pr_info("QPNP SMB5 probed successfully\n");
2337
2338 return rc;
2339
Ashay Jaiswala9e10912018-02-02 14:03:35 +05302340free_irq:
Harry Yang4b7db0f2017-11-27 10:50:44 -08002341 smb5_free_interrupts(chg);
Ashay Jaiswala9e10912018-02-02 14:03:35 +05302342cleanup:
Harry Yang4b7db0f2017-11-27 10:50:44 -08002343 smblib_deinit(chg);
2344 platform_set_drvdata(pdev, NULL);
2345
2346 return rc;
2347}
2348
2349static int smb5_remove(struct platform_device *pdev)
2350{
2351 struct smb5 *chip = platform_get_drvdata(pdev);
2352 struct smb_charger *chg = &chip->chg;
2353
2354 smb5_free_interrupts(chg);
2355 smblib_deinit(chg);
2356 platform_set_drvdata(pdev, NULL);
2357 return 0;
2358}
2359
2360static void smb5_shutdown(struct platform_device *pdev)
2361{
2362 struct smb5 *chip = platform_get_drvdata(pdev);
2363 struct smb_charger *chg = &chip->chg;
2364
2365 /* disable all interrupts */
2366 smb5_disable_interrupts(chg);
2367
2368 /* configure power role for UFP */
Ashay Jaiswala9e10912018-02-02 14:03:35 +05302369 if (chg->connector_type == POWER_SUPPLY_CONNECTOR_TYPEC)
Harry Yang4b7db0f2017-11-27 10:50:44 -08002370 smblib_masked_write(chg, TYPE_C_MODE_CFG_REG,
2371 TYPEC_POWER_ROLE_CMD_MASK, EN_SNK_ONLY_BIT);
2372
2373 /* force HVDCP to 5V */
2374 smblib_masked_write(chg, USBIN_OPTIONS_1_CFG_REG,
2375 HVDCP_AUTONOMOUS_MODE_EN_CFG_BIT, 0);
2376 smblib_write(chg, CMD_HVDCP_2_REG, FORCE_5V_BIT);
2377
2378 /* force enable APSD */
2379 smblib_masked_write(chg, USBIN_OPTIONS_1_CFG_REG,
2380 BC1P2_SRC_DETECT_BIT, BC1P2_SRC_DETECT_BIT);
2381}
2382
2383static const struct of_device_id match_table[] = {
2384 { .compatible = "qcom,qpnp-smb5", },
2385 { },
2386};
2387
2388static struct platform_driver smb5_driver = {
2389 .driver = {
2390 .name = "qcom,qpnp-smb5",
2391 .owner = THIS_MODULE,
2392 .of_match_table = match_table,
2393 },
2394 .probe = smb5_probe,
2395 .remove = smb5_remove,
2396 .shutdown = smb5_shutdown,
2397};
2398module_platform_driver(smb5_driver);
2399
2400MODULE_DESCRIPTION("QPNP SMB5 Charger Driver");
2401MODULE_LICENSE("GPL v2");