0001
0002
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
0014
0015 #include <linux/io.h>
0016 #include <linux/module.h>
0017 #include <linux/irq.h>
0018 #include <linux/interrupt.h>
0019 #include <linux/rtc.h>
0020 #include <linux/slab.h>
0021 #include <linux/workqueue.h>
0022 #include <linux/platform_device.h>
0023 #include <linux/power/charger-manager.h>
0024 #include <linux/regulator/consumer.h>
0025 #include <linux/sysfs.h>
0026 #include <linux/of.h>
0027 #include <linux/thermal.h>
0028
0029 static struct {
0030 const char *name;
0031 u64 extcon_type;
0032 } extcon_mapping[] = {
0033
0034 { "USB", EXTCON_USB },
0035 { "USB-HOST", EXTCON_USB_HOST },
0036 { "SDP", EXTCON_CHG_USB_SDP },
0037 { "DCP", EXTCON_CHG_USB_DCP },
0038 { "CDP", EXTCON_CHG_USB_CDP },
0039 { "ACA", EXTCON_CHG_USB_ACA },
0040 { "FAST-CHARGER", EXTCON_CHG_USB_FAST },
0041 { "SLOW-CHARGER", EXTCON_CHG_USB_SLOW },
0042 { "WPT", EXTCON_CHG_WPT },
0043 { "PD", EXTCON_CHG_USB_PD },
0044 { "DOCK", EXTCON_DOCK },
0045 { "JIG", EXTCON_JIG },
0046 { "MECHANICAL", EXTCON_MECHANICAL },
0047
0048 { "TA", EXTCON_CHG_USB_SDP },
0049 { "CHARGE-DOWNSTREAM", EXTCON_CHG_USB_CDP },
0050 };
0051
0052
0053
0054
0055
0056 #define CM_DEFAULT_RECHARGE_TEMP_DIFF 50
0057 #define CM_DEFAULT_CHARGE_TEMP_MAX 500
0058
0059
0060
0061
0062
0063
0064 #define CM_JIFFIES_SMALL (2)
0065
0066
0067 #define CM_MIN_VALID(x, y) x = (((y > 0) && ((x) > (y))) ? (y) : (x))
0068
0069
0070
0071
0072
0073 #define CM_RTC_SMALL (2)
0074
0075 static LIST_HEAD(cm_list);
0076 static DEFINE_MUTEX(cm_list_mtx);
0077
0078
0079 static struct alarm *cm_timer;
0080
0081 static bool cm_suspended;
0082 static bool cm_timer_set;
0083 static unsigned long cm_suspend_duration_ms;
0084
0085
0086 static unsigned long polling_jiffy = ULONG_MAX;
0087 static unsigned long next_polling;
0088 static struct workqueue_struct *cm_wq;
0089 static struct delayed_work cm_monitor_work;
0090
0091
0092
0093
0094
0095 static bool is_batt_present(struct charger_manager *cm)
0096 {
0097 union power_supply_propval val;
0098 struct power_supply *psy;
0099 bool present = false;
0100 int i, ret;
0101
0102 switch (cm->desc->battery_present) {
0103 case CM_BATTERY_PRESENT:
0104 present = true;
0105 break;
0106 case CM_NO_BATTERY:
0107 break;
0108 case CM_FUEL_GAUGE:
0109 psy = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
0110 if (!psy)
0111 break;
0112
0113 ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_PRESENT,
0114 &val);
0115 if (ret == 0 && val.intval)
0116 present = true;
0117 power_supply_put(psy);
0118 break;
0119 case CM_CHARGER_STAT:
0120 for (i = 0; cm->desc->psy_charger_stat[i]; i++) {
0121 psy = power_supply_get_by_name(
0122 cm->desc->psy_charger_stat[i]);
0123 if (!psy) {
0124 dev_err(cm->dev, "Cannot find power supply \"%s\"\n",
0125 cm->desc->psy_charger_stat[i]);
0126 continue;
0127 }
0128
0129 ret = power_supply_get_property(psy,
0130 POWER_SUPPLY_PROP_PRESENT, &val);
0131 power_supply_put(psy);
0132 if (ret == 0 && val.intval) {
0133 present = true;
0134 break;
0135 }
0136 }
0137 break;
0138 }
0139
0140 return present;
0141 }
0142
0143
0144
0145
0146
0147
0148
0149
0150
0151 static bool is_ext_pwr_online(struct charger_manager *cm)
0152 {
0153 union power_supply_propval val;
0154 struct power_supply *psy;
0155 bool online = false;
0156 int i, ret;
0157
0158
0159 for (i = 0; cm->desc->psy_charger_stat[i]; i++) {
0160 psy = power_supply_get_by_name(cm->desc->psy_charger_stat[i]);
0161 if (!psy) {
0162 dev_err(cm->dev, "Cannot find power supply \"%s\"\n",
0163 cm->desc->psy_charger_stat[i]);
0164 continue;
0165 }
0166
0167 ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_ONLINE,
0168 &val);
0169 power_supply_put(psy);
0170 if (ret == 0 && val.intval) {
0171 online = true;
0172 break;
0173 }
0174 }
0175
0176 return online;
0177 }
0178
0179
0180
0181
0182
0183
0184
0185
0186
0187 static int get_batt_uV(struct charger_manager *cm, int *uV)
0188 {
0189 union power_supply_propval val;
0190 struct power_supply *fuel_gauge;
0191 int ret;
0192
0193 fuel_gauge = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
0194 if (!fuel_gauge)
0195 return -ENODEV;
0196
0197 ret = power_supply_get_property(fuel_gauge,
0198 POWER_SUPPLY_PROP_VOLTAGE_NOW, &val);
0199 power_supply_put(fuel_gauge);
0200 if (ret)
0201 return ret;
0202
0203 *uV = val.intval;
0204 return 0;
0205 }
0206
0207
0208
0209
0210
0211 static bool is_charging(struct charger_manager *cm)
0212 {
0213 int i, ret;
0214 bool charging = false;
0215 struct power_supply *psy;
0216 union power_supply_propval val;
0217
0218
0219 if (!is_batt_present(cm))
0220 return false;
0221
0222
0223 for (i = 0; cm->desc->psy_charger_stat[i]; i++) {
0224
0225 if (cm->emergency_stop)
0226 continue;
0227 if (!cm->charger_enabled)
0228 continue;
0229
0230 psy = power_supply_get_by_name(cm->desc->psy_charger_stat[i]);
0231 if (!psy) {
0232 dev_err(cm->dev, "Cannot find power supply \"%s\"\n",
0233 cm->desc->psy_charger_stat[i]);
0234 continue;
0235 }
0236
0237
0238 ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_ONLINE,
0239 &val);
0240 if (ret) {
0241 dev_warn(cm->dev, "Cannot read ONLINE value from %s\n",
0242 cm->desc->psy_charger_stat[i]);
0243 power_supply_put(psy);
0244 continue;
0245 }
0246 if (val.intval == 0) {
0247 power_supply_put(psy);
0248 continue;
0249 }
0250
0251
0252
0253
0254
0255 ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_STATUS,
0256 &val);
0257 power_supply_put(psy);
0258 if (ret) {
0259 dev_warn(cm->dev, "Cannot read STATUS value from %s\n",
0260 cm->desc->psy_charger_stat[i]);
0261 continue;
0262 }
0263 if (val.intval == POWER_SUPPLY_STATUS_FULL ||
0264 val.intval == POWER_SUPPLY_STATUS_DISCHARGING ||
0265 val.intval == POWER_SUPPLY_STATUS_NOT_CHARGING)
0266 continue;
0267
0268
0269 charging = true;
0270 break;
0271 }
0272
0273 return charging;
0274 }
0275
0276
0277
0278
0279
0280 static bool is_full_charged(struct charger_manager *cm)
0281 {
0282 struct charger_desc *desc = cm->desc;
0283 union power_supply_propval val;
0284 struct power_supply *fuel_gauge;
0285 bool is_full = false;
0286 int ret = 0;
0287 int uV;
0288
0289
0290 if (!is_batt_present(cm))
0291 return false;
0292
0293 fuel_gauge = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
0294 if (!fuel_gauge)
0295 return false;
0296
0297
0298 if (desc->fullbatt_uV > 0) {
0299 ret = get_batt_uV(cm, &uV);
0300 if (!ret) {
0301
0302 if (cm->battery_status == POWER_SUPPLY_STATUS_FULL
0303 && desc->fullbatt_vchkdrop_uV)
0304 uV += desc->fullbatt_vchkdrop_uV;
0305 if (uV >= desc->fullbatt_uV)
0306 return true;
0307 }
0308 }
0309
0310 if (desc->fullbatt_full_capacity > 0) {
0311 val.intval = 0;
0312
0313
0314 ret = power_supply_get_property(fuel_gauge,
0315 POWER_SUPPLY_PROP_CHARGE_FULL, &val);
0316 if (!ret && val.intval > desc->fullbatt_full_capacity) {
0317 is_full = true;
0318 goto out;
0319 }
0320 }
0321
0322
0323 if (desc->fullbatt_soc > 0) {
0324 val.intval = 0;
0325
0326 ret = power_supply_get_property(fuel_gauge,
0327 POWER_SUPPLY_PROP_CAPACITY, &val);
0328 if (!ret && val.intval >= desc->fullbatt_soc) {
0329 is_full = true;
0330 goto out;
0331 }
0332 }
0333
0334 out:
0335 power_supply_put(fuel_gauge);
0336 return is_full;
0337 }
0338
0339
0340
0341
0342
0343 static bool is_polling_required(struct charger_manager *cm)
0344 {
0345 switch (cm->desc->polling_mode) {
0346 case CM_POLL_DISABLE:
0347 return false;
0348 case CM_POLL_ALWAYS:
0349 return true;
0350 case CM_POLL_EXTERNAL_POWER_ONLY:
0351 return is_ext_pwr_online(cm);
0352 case CM_POLL_CHARGING_ONLY:
0353 return is_charging(cm);
0354 default:
0355 dev_warn(cm->dev, "Incorrect polling_mode (%d)\n",
0356 cm->desc->polling_mode);
0357 }
0358
0359 return false;
0360 }
0361
0362
0363
0364
0365
0366
0367
0368
0369
0370
0371
0372 static int try_charger_enable(struct charger_manager *cm, bool enable)
0373 {
0374 int err = 0, i;
0375 struct charger_desc *desc = cm->desc;
0376
0377
0378 if (enable == cm->charger_enabled)
0379 return 0;
0380
0381 if (enable) {
0382 if (cm->emergency_stop)
0383 return -EAGAIN;
0384
0385
0386
0387
0388
0389 cm->charging_start_time = ktime_to_ms(ktime_get());
0390 cm->charging_end_time = 0;
0391
0392 for (i = 0 ; i < desc->num_charger_regulators ; i++) {
0393 if (desc->charger_regulators[i].externally_control)
0394 continue;
0395
0396 err = regulator_enable(desc->charger_regulators[i].consumer);
0397 if (err < 0) {
0398 dev_warn(cm->dev, "Cannot enable %s regulator\n",
0399 desc->charger_regulators[i].regulator_name);
0400 }
0401 }
0402 } else {
0403
0404
0405
0406
0407 cm->charging_start_time = 0;
0408 cm->charging_end_time = ktime_to_ms(ktime_get());
0409
0410 for (i = 0 ; i < desc->num_charger_regulators ; i++) {
0411 if (desc->charger_regulators[i].externally_control)
0412 continue;
0413
0414 err = regulator_disable(desc->charger_regulators[i].consumer);
0415 if (err < 0) {
0416 dev_warn(cm->dev, "Cannot disable %s regulator\n",
0417 desc->charger_regulators[i].regulator_name);
0418 }
0419 }
0420
0421
0422
0423
0424
0425 for (i = 0; i < desc->num_charger_regulators; i++) {
0426 if (regulator_is_enabled(
0427 desc->charger_regulators[i].consumer)) {
0428 regulator_force_disable(
0429 desc->charger_regulators[i].consumer);
0430 dev_warn(cm->dev, "Disable regulator(%s) forcibly\n",
0431 desc->charger_regulators[i].regulator_name);
0432 }
0433 }
0434 }
0435
0436 if (!err)
0437 cm->charger_enabled = enable;
0438
0439 return err;
0440 }
0441
0442
0443
0444
0445
0446
0447
0448
0449
0450
0451
0452 static int check_charging_duration(struct charger_manager *cm)
0453 {
0454 struct charger_desc *desc = cm->desc;
0455 u64 curr = ktime_to_ms(ktime_get());
0456 u64 duration;
0457 int ret = false;
0458
0459 if (!desc->charging_max_duration_ms &&
0460 !desc->discharging_max_duration_ms)
0461 return ret;
0462
0463 if (cm->charger_enabled) {
0464 duration = curr - cm->charging_start_time;
0465
0466 if (duration > desc->charging_max_duration_ms) {
0467 dev_info(cm->dev, "Charging duration exceed %ums\n",
0468 desc->charging_max_duration_ms);
0469 ret = true;
0470 }
0471 } else if (cm->battery_status == POWER_SUPPLY_STATUS_NOT_CHARGING) {
0472 duration = curr - cm->charging_end_time;
0473
0474 if (duration > desc->discharging_max_duration_ms) {
0475 dev_info(cm->dev, "Discharging duration exceed %ums\n",
0476 desc->discharging_max_duration_ms);
0477 ret = true;
0478 }
0479 }
0480
0481 return ret;
0482 }
0483
0484 static int cm_get_battery_temperature_by_psy(struct charger_manager *cm,
0485 int *temp)
0486 {
0487 struct power_supply *fuel_gauge;
0488 int ret;
0489
0490 fuel_gauge = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
0491 if (!fuel_gauge)
0492 return -ENODEV;
0493
0494 ret = power_supply_get_property(fuel_gauge,
0495 POWER_SUPPLY_PROP_TEMP,
0496 (union power_supply_propval *)temp);
0497 power_supply_put(fuel_gauge);
0498
0499 return ret;
0500 }
0501
0502 static int cm_get_battery_temperature(struct charger_manager *cm,
0503 int *temp)
0504 {
0505 int ret;
0506
0507 if (!cm->desc->measure_battery_temp)
0508 return -ENODEV;
0509
0510 #ifdef CONFIG_THERMAL
0511 if (cm->tzd_batt) {
0512 ret = thermal_zone_get_temp(cm->tzd_batt, temp);
0513 if (!ret)
0514
0515 *temp /= 100;
0516 } else
0517 #endif
0518 {
0519
0520 ret = cm_get_battery_temperature_by_psy(cm, temp);
0521 }
0522
0523 return ret;
0524 }
0525
0526 static int cm_check_thermal_status(struct charger_manager *cm)
0527 {
0528 struct charger_desc *desc = cm->desc;
0529 int temp, upper_limit, lower_limit;
0530 int ret = 0;
0531
0532 ret = cm_get_battery_temperature(cm, &temp);
0533 if (ret) {
0534
0535
0536
0537
0538
0539 dev_err(cm->dev, "Failed to get battery temperature\n");
0540 return 0;
0541 }
0542
0543 upper_limit = desc->temp_max;
0544 lower_limit = desc->temp_min;
0545
0546 if (cm->emergency_stop) {
0547 upper_limit -= desc->temp_diff;
0548 lower_limit += desc->temp_diff;
0549 }
0550
0551 if (temp > upper_limit)
0552 ret = CM_BATT_OVERHEAT;
0553 else if (temp < lower_limit)
0554 ret = CM_BATT_COLD;
0555 else
0556 ret = CM_BATT_OK;
0557
0558 cm->emergency_stop = ret;
0559
0560 return ret;
0561 }
0562
0563
0564
0565
0566
0567 static int cm_get_target_status(struct charger_manager *cm)
0568 {
0569 if (!is_ext_pwr_online(cm))
0570 return POWER_SUPPLY_STATUS_DISCHARGING;
0571
0572 if (cm_check_thermal_status(cm)) {
0573
0574 if (check_charging_duration(cm))
0575 goto charging_ok;
0576 return POWER_SUPPLY_STATUS_NOT_CHARGING;
0577 }
0578
0579 switch (cm->battery_status) {
0580 case POWER_SUPPLY_STATUS_CHARGING:
0581
0582 if (check_charging_duration(cm))
0583 return POWER_SUPPLY_STATUS_FULL;
0584 fallthrough;
0585 case POWER_SUPPLY_STATUS_FULL:
0586 if (is_full_charged(cm))
0587 return POWER_SUPPLY_STATUS_FULL;
0588 fallthrough;
0589 default:
0590 break;
0591 }
0592
0593 charging_ok:
0594
0595 return POWER_SUPPLY_STATUS_CHARGING;
0596 }
0597
0598
0599
0600
0601
0602
0603
0604
0605 static bool _cm_monitor(struct charger_manager *cm)
0606 {
0607 int target;
0608
0609 target = cm_get_target_status(cm);
0610
0611 try_charger_enable(cm, (target == POWER_SUPPLY_STATUS_CHARGING));
0612
0613 if (cm->battery_status != target) {
0614 cm->battery_status = target;
0615 power_supply_changed(cm->charger_psy);
0616 }
0617
0618 return (cm->battery_status == POWER_SUPPLY_STATUS_NOT_CHARGING);
0619 }
0620
0621
0622
0623
0624
0625
0626
0627 static bool cm_monitor(void)
0628 {
0629 bool stop = false;
0630 struct charger_manager *cm;
0631
0632 mutex_lock(&cm_list_mtx);
0633
0634 list_for_each_entry(cm, &cm_list, entry) {
0635 if (_cm_monitor(cm))
0636 stop = true;
0637 }
0638
0639 mutex_unlock(&cm_list_mtx);
0640
0641 return stop;
0642 }
0643
0644
0645
0646
0647
0648 static void _setup_polling(struct work_struct *work)
0649 {
0650 unsigned long min = ULONG_MAX;
0651 struct charger_manager *cm;
0652 bool keep_polling = false;
0653 unsigned long _next_polling;
0654
0655 mutex_lock(&cm_list_mtx);
0656
0657 list_for_each_entry(cm, &cm_list, entry) {
0658 if (is_polling_required(cm) && cm->desc->polling_interval_ms) {
0659 keep_polling = true;
0660
0661 if (min > cm->desc->polling_interval_ms)
0662 min = cm->desc->polling_interval_ms;
0663 }
0664 }
0665
0666 polling_jiffy = msecs_to_jiffies(min);
0667 if (polling_jiffy <= CM_JIFFIES_SMALL)
0668 polling_jiffy = CM_JIFFIES_SMALL + 1;
0669
0670 if (!keep_polling)
0671 polling_jiffy = ULONG_MAX;
0672 if (polling_jiffy == ULONG_MAX)
0673 goto out;
0674
0675 WARN(cm_wq == NULL, "charger-manager: workqueue not initialized"
0676 ". try it later. %s\n", __func__);
0677
0678
0679
0680
0681
0682
0683
0684 _next_polling = jiffies + polling_jiffy;
0685
0686 if (time_before(_next_polling, next_polling)) {
0687 mod_delayed_work(cm_wq, &cm_monitor_work, polling_jiffy);
0688 next_polling = _next_polling;
0689 } else {
0690 if (queue_delayed_work(cm_wq, &cm_monitor_work, polling_jiffy))
0691 next_polling = _next_polling;
0692 }
0693 out:
0694 mutex_unlock(&cm_list_mtx);
0695 }
0696 static DECLARE_WORK(setup_polling, _setup_polling);
0697
0698
0699
0700
0701
0702
0703
0704
0705 static void cm_monitor_poller(struct work_struct *work)
0706 {
0707 cm_monitor();
0708 schedule_work(&setup_polling);
0709 }
0710
0711 static int charger_get_property(struct power_supply *psy,
0712 enum power_supply_property psp,
0713 union power_supply_propval *val)
0714 {
0715 struct charger_manager *cm = power_supply_get_drvdata(psy);
0716 struct charger_desc *desc = cm->desc;
0717 struct power_supply *fuel_gauge = NULL;
0718 int ret = 0;
0719 int uV;
0720
0721 switch (psp) {
0722 case POWER_SUPPLY_PROP_STATUS:
0723 val->intval = cm->battery_status;
0724 break;
0725 case POWER_SUPPLY_PROP_HEALTH:
0726 if (cm->emergency_stop == CM_BATT_OVERHEAT)
0727 val->intval = POWER_SUPPLY_HEALTH_OVERHEAT;
0728 else if (cm->emergency_stop == CM_BATT_COLD)
0729 val->intval = POWER_SUPPLY_HEALTH_COLD;
0730 else
0731 val->intval = POWER_SUPPLY_HEALTH_GOOD;
0732 break;
0733 case POWER_SUPPLY_PROP_PRESENT:
0734 if (is_batt_present(cm))
0735 val->intval = 1;
0736 else
0737 val->intval = 0;
0738 break;
0739 case POWER_SUPPLY_PROP_VOLTAGE_NOW:
0740 ret = get_batt_uV(cm, &val->intval);
0741 break;
0742 case POWER_SUPPLY_PROP_CURRENT_NOW:
0743 fuel_gauge = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
0744 if (!fuel_gauge) {
0745 ret = -ENODEV;
0746 break;
0747 }
0748 ret = power_supply_get_property(fuel_gauge,
0749 POWER_SUPPLY_PROP_CURRENT_NOW, val);
0750 break;
0751 case POWER_SUPPLY_PROP_TEMP:
0752 return cm_get_battery_temperature(cm, &val->intval);
0753 case POWER_SUPPLY_PROP_CAPACITY:
0754 if (!is_batt_present(cm)) {
0755
0756 val->intval = 100;
0757 break;
0758 }
0759
0760 fuel_gauge = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
0761 if (!fuel_gauge) {
0762 ret = -ENODEV;
0763 break;
0764 }
0765
0766 ret = power_supply_get_property(fuel_gauge,
0767 POWER_SUPPLY_PROP_CAPACITY, val);
0768 if (ret)
0769 break;
0770
0771 if (val->intval > 100) {
0772 val->intval = 100;
0773 break;
0774 }
0775 if (val->intval < 0)
0776 val->intval = 0;
0777
0778
0779 if (is_charging(cm))
0780 break;
0781
0782
0783
0784
0785
0786 ret = get_batt_uV(cm, &uV);
0787 if (ret) {
0788
0789 ret = 0;
0790 break;
0791 }
0792
0793 if (desc->fullbatt_uV > 0 && uV >= desc->fullbatt_uV &&
0794 !is_charging(cm)) {
0795 val->intval = 100;
0796 break;
0797 }
0798
0799 break;
0800 case POWER_SUPPLY_PROP_ONLINE:
0801 if (is_ext_pwr_online(cm))
0802 val->intval = 1;
0803 else
0804 val->intval = 0;
0805 break;
0806 case POWER_SUPPLY_PROP_CHARGE_FULL:
0807 case POWER_SUPPLY_PROP_CHARGE_NOW:
0808 fuel_gauge = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
0809 if (!fuel_gauge) {
0810 ret = -ENODEV;
0811 break;
0812 }
0813 ret = power_supply_get_property(fuel_gauge, psp, val);
0814 break;
0815 default:
0816 return -EINVAL;
0817 }
0818 if (fuel_gauge)
0819 power_supply_put(fuel_gauge);
0820 return ret;
0821 }
0822
0823 #define NUM_CHARGER_PSY_OPTIONAL (4)
0824 static enum power_supply_property default_charger_props[] = {
0825
0826 POWER_SUPPLY_PROP_STATUS,
0827 POWER_SUPPLY_PROP_HEALTH,
0828 POWER_SUPPLY_PROP_PRESENT,
0829 POWER_SUPPLY_PROP_VOLTAGE_NOW,
0830 POWER_SUPPLY_PROP_CAPACITY,
0831 POWER_SUPPLY_PROP_ONLINE,
0832
0833
0834
0835
0836
0837
0838
0839 };
0840
0841 static const struct power_supply_desc psy_default = {
0842 .name = "battery",
0843 .type = POWER_SUPPLY_TYPE_BATTERY,
0844 .properties = default_charger_props,
0845 .num_properties = ARRAY_SIZE(default_charger_props),
0846 .get_property = charger_get_property,
0847 .no_thermal = true,
0848 };
0849
0850
0851
0852
0853
0854
0855
0856
0857
0858
0859
0860 static bool cm_setup_timer(void)
0861 {
0862 struct charger_manager *cm;
0863 unsigned int wakeup_ms = UINT_MAX;
0864 int timer_req = 0;
0865
0866 if (time_after(next_polling, jiffies))
0867 CM_MIN_VALID(wakeup_ms,
0868 jiffies_to_msecs(next_polling - jiffies));
0869
0870 mutex_lock(&cm_list_mtx);
0871 list_for_each_entry(cm, &cm_list, entry) {
0872
0873 if (!is_polling_required(cm) && !cm->emergency_stop)
0874 continue;
0875 timer_req++;
0876 if (cm->desc->polling_interval_ms == 0)
0877 continue;
0878 CM_MIN_VALID(wakeup_ms, cm->desc->polling_interval_ms);
0879 }
0880 mutex_unlock(&cm_list_mtx);
0881
0882 if (timer_req && cm_timer) {
0883 ktime_t now, add;
0884
0885
0886
0887
0888
0889 if (wakeup_ms == UINT_MAX ||
0890 wakeup_ms < CM_RTC_SMALL * MSEC_PER_SEC)
0891 wakeup_ms = 2 * CM_RTC_SMALL * MSEC_PER_SEC;
0892
0893 pr_info("Charger Manager wakeup timer: %u ms\n", wakeup_ms);
0894
0895 now = ktime_get_boottime();
0896 add = ktime_set(wakeup_ms / MSEC_PER_SEC,
0897 (wakeup_ms % MSEC_PER_SEC) * NSEC_PER_MSEC);
0898 alarm_start(cm_timer, ktime_add(now, add));
0899
0900 cm_suspend_duration_ms = wakeup_ms;
0901
0902 return true;
0903 }
0904 return false;
0905 }
0906
0907
0908
0909
0910
0911
0912
0913 static void charger_extcon_work(struct work_struct *work)
0914 {
0915 struct charger_cable *cable =
0916 container_of(work, struct charger_cable, wq);
0917 int ret;
0918
0919 if (cable->attached && cable->min_uA != 0 && cable->max_uA != 0) {
0920 ret = regulator_set_current_limit(cable->charger->consumer,
0921 cable->min_uA, cable->max_uA);
0922 if (ret < 0) {
0923 pr_err("Cannot set current limit of %s (%s)\n",
0924 cable->charger->regulator_name, cable->name);
0925 return;
0926 }
0927
0928 pr_info("Set current limit of %s : %duA ~ %duA\n",
0929 cable->charger->regulator_name,
0930 cable->min_uA, cable->max_uA);
0931 }
0932
0933 cancel_delayed_work(&cm_monitor_work);
0934 queue_delayed_work(cm_wq, &cm_monitor_work, 0);
0935 }
0936
0937
0938
0939
0940
0941
0942
0943
0944
0945 static int charger_extcon_notifier(struct notifier_block *self,
0946 unsigned long event, void *ptr)
0947 {
0948 struct charger_cable *cable =
0949 container_of(self, struct charger_cable, nb);
0950
0951
0952
0953
0954
0955 cable->attached = event;
0956
0957
0958
0959
0960
0961 schedule_work(&cable->wq);
0962
0963 return NOTIFY_DONE;
0964 }
0965
0966
0967
0968
0969
0970
0971
0972
0973 static int charger_extcon_init(struct charger_manager *cm,
0974 struct charger_cable *cable)
0975 {
0976 int ret, i;
0977 u64 extcon_type = EXTCON_NONE;
0978
0979
0980
0981
0982
0983
0984 INIT_WORK(&cable->wq, charger_extcon_work);
0985 cable->nb.notifier_call = charger_extcon_notifier;
0986
0987 cable->extcon_dev = extcon_get_extcon_dev(cable->extcon_name);
0988 if (IS_ERR(cable->extcon_dev)) {
0989 pr_err("Cannot find extcon_dev for %s (cable: %s)\n",
0990 cable->extcon_name, cable->name);
0991 return PTR_ERR(cable->extcon_dev);
0992 }
0993
0994 for (i = 0; i < ARRAY_SIZE(extcon_mapping); i++) {
0995 if (!strcmp(cable->name, extcon_mapping[i].name)) {
0996 extcon_type = extcon_mapping[i].extcon_type;
0997 break;
0998 }
0999 }
1000 if (extcon_type == EXTCON_NONE) {
1001 pr_err("Cannot find cable for type %s", cable->name);
1002 return -EINVAL;
1003 }
1004
1005 cable->extcon_type = extcon_type;
1006
1007 ret = devm_extcon_register_notifier(cm->dev, cable->extcon_dev,
1008 cable->extcon_type, &cable->nb);
1009 if (ret < 0) {
1010 pr_err("Cannot register extcon_dev for %s (cable: %s)\n",
1011 cable->extcon_name, cable->name);
1012 return ret;
1013 }
1014
1015 return 0;
1016 }
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028 static int charger_manager_register_extcon(struct charger_manager *cm)
1029 {
1030 struct charger_desc *desc = cm->desc;
1031 struct charger_regulator *charger;
1032 unsigned long event;
1033 int ret;
1034 int i;
1035 int j;
1036
1037 for (i = 0; i < desc->num_charger_regulators; i++) {
1038 charger = &desc->charger_regulators[i];
1039
1040 charger->consumer = regulator_get(cm->dev,
1041 charger->regulator_name);
1042 if (IS_ERR(charger->consumer)) {
1043 dev_err(cm->dev, "Cannot find charger(%s)\n",
1044 charger->regulator_name);
1045 return PTR_ERR(charger->consumer);
1046 }
1047 charger->cm = cm;
1048
1049 for (j = 0; j < charger->num_cables; j++) {
1050 struct charger_cable *cable = &charger->cables[j];
1051
1052 ret = charger_extcon_init(cm, cable);
1053 if (ret < 0) {
1054 dev_err(cm->dev, "Cannot initialize charger(%s)\n",
1055 charger->regulator_name);
1056 return ret;
1057 }
1058 cable->charger = charger;
1059 cable->cm = cm;
1060
1061 event = extcon_get_state(cable->extcon_dev,
1062 cable->extcon_type);
1063 charger_extcon_notifier(&cable->nb,
1064 event, NULL);
1065 }
1066 }
1067
1068 return 0;
1069 }
1070
1071
1072 static ssize_t charger_name_show(struct device *dev,
1073 struct device_attribute *attr, char *buf)
1074 {
1075 struct charger_regulator *charger
1076 = container_of(attr, struct charger_regulator, attr_name);
1077
1078 return sprintf(buf, "%s\n", charger->regulator_name);
1079 }
1080
1081 static ssize_t charger_state_show(struct device *dev,
1082 struct device_attribute *attr, char *buf)
1083 {
1084 struct charger_regulator *charger
1085 = container_of(attr, struct charger_regulator, attr_state);
1086 int state = 0;
1087
1088 if (!charger->externally_control)
1089 state = regulator_is_enabled(charger->consumer);
1090
1091 return sprintf(buf, "%s\n", state ? "enabled" : "disabled");
1092 }
1093
1094 static ssize_t charger_externally_control_show(struct device *dev,
1095 struct device_attribute *attr, char *buf)
1096 {
1097 struct charger_regulator *charger = container_of(attr,
1098 struct charger_regulator, attr_externally_control);
1099
1100 return sprintf(buf, "%d\n", charger->externally_control);
1101 }
1102
1103 static ssize_t charger_externally_control_store(struct device *dev,
1104 struct device_attribute *attr, const char *buf,
1105 size_t count)
1106 {
1107 struct charger_regulator *charger
1108 = container_of(attr, struct charger_regulator,
1109 attr_externally_control);
1110 struct charger_manager *cm = charger->cm;
1111 struct charger_desc *desc = cm->desc;
1112 int i;
1113 int ret;
1114 int externally_control;
1115 int chargers_externally_control = 1;
1116
1117 ret = sscanf(buf, "%d", &externally_control);
1118 if (ret == 0) {
1119 ret = -EINVAL;
1120 return ret;
1121 }
1122
1123 if (!externally_control) {
1124 charger->externally_control = 0;
1125 return count;
1126 }
1127
1128 for (i = 0; i < desc->num_charger_regulators; i++) {
1129 if (&desc->charger_regulators[i] != charger &&
1130 !desc->charger_regulators[i].externally_control) {
1131
1132
1133
1134
1135 chargers_externally_control = 0;
1136 break;
1137 }
1138 }
1139
1140 if (!chargers_externally_control) {
1141 if (cm->charger_enabled) {
1142 try_charger_enable(charger->cm, false);
1143 charger->externally_control = externally_control;
1144 try_charger_enable(charger->cm, true);
1145 } else {
1146 charger->externally_control = externally_control;
1147 }
1148 } else {
1149 dev_warn(cm->dev,
1150 "'%s' regulator should be controlled in charger-manager because charger-manager must need at least one charger for charging\n",
1151 charger->regulator_name);
1152 }
1153
1154 return count;
1155 }
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170 static int charger_manager_prepare_sysfs(struct charger_manager *cm)
1171 {
1172 struct charger_desc *desc = cm->desc;
1173 struct charger_regulator *charger;
1174 int chargers_externally_control = 1;
1175 char *name;
1176 int i;
1177
1178
1179 for (i = 0; i < desc->num_charger_regulators; i++) {
1180 charger = &desc->charger_regulators[i];
1181
1182 name = devm_kasprintf(cm->dev, GFP_KERNEL, "charger.%d", i);
1183 if (!name)
1184 return -ENOMEM;
1185
1186 charger->attrs[0] = &charger->attr_name.attr;
1187 charger->attrs[1] = &charger->attr_state.attr;
1188 charger->attrs[2] = &charger->attr_externally_control.attr;
1189 charger->attrs[3] = NULL;
1190
1191 charger->attr_grp.name = name;
1192 charger->attr_grp.attrs = charger->attrs;
1193 desc->sysfs_groups[i] = &charger->attr_grp;
1194
1195 sysfs_attr_init(&charger->attr_name.attr);
1196 charger->attr_name.attr.name = "name";
1197 charger->attr_name.attr.mode = 0444;
1198 charger->attr_name.show = charger_name_show;
1199
1200 sysfs_attr_init(&charger->attr_state.attr);
1201 charger->attr_state.attr.name = "state";
1202 charger->attr_state.attr.mode = 0444;
1203 charger->attr_state.show = charger_state_show;
1204
1205 sysfs_attr_init(&charger->attr_externally_control.attr);
1206 charger->attr_externally_control.attr.name
1207 = "externally_control";
1208 charger->attr_externally_control.attr.mode = 0644;
1209 charger->attr_externally_control.show
1210 = charger_externally_control_show;
1211 charger->attr_externally_control.store
1212 = charger_externally_control_store;
1213
1214 if (!desc->charger_regulators[i].externally_control ||
1215 !chargers_externally_control)
1216 chargers_externally_control = 0;
1217
1218 dev_info(cm->dev, "'%s' regulator's externally_control is %d\n",
1219 charger->regulator_name, charger->externally_control);
1220 }
1221
1222 if (chargers_externally_control) {
1223 dev_err(cm->dev, "Cannot register regulator because charger-manager must need at least one charger for charging battery\n");
1224 return -EINVAL;
1225 }
1226
1227 return 0;
1228 }
1229
1230 static int cm_init_thermal_data(struct charger_manager *cm,
1231 struct power_supply *fuel_gauge,
1232 enum power_supply_property *properties,
1233 size_t *num_properties)
1234 {
1235 struct charger_desc *desc = cm->desc;
1236 union power_supply_propval val;
1237 int ret;
1238
1239
1240 ret = power_supply_get_property(fuel_gauge,
1241 POWER_SUPPLY_PROP_TEMP, &val);
1242
1243 if (!ret) {
1244 properties[*num_properties] = POWER_SUPPLY_PROP_TEMP;
1245 (*num_properties)++;
1246 cm->desc->measure_battery_temp = true;
1247 }
1248 #ifdef CONFIG_THERMAL
1249 if (ret && desc->thermal_zone) {
1250 cm->tzd_batt =
1251 thermal_zone_get_zone_by_name(desc->thermal_zone);
1252 if (IS_ERR(cm->tzd_batt))
1253 return PTR_ERR(cm->tzd_batt);
1254
1255
1256 properties[*num_properties] = POWER_SUPPLY_PROP_TEMP;
1257 (*num_properties)++;
1258 cm->desc->measure_battery_temp = true;
1259 ret = 0;
1260 }
1261 #endif
1262 if (cm->desc->measure_battery_temp) {
1263
1264 if (!desc->temp_max)
1265 desc->temp_max = CM_DEFAULT_CHARGE_TEMP_MAX;
1266 if (!desc->temp_diff)
1267 desc->temp_diff = CM_DEFAULT_RECHARGE_TEMP_DIFF;
1268 }
1269
1270 return ret;
1271 }
1272
1273 static const struct of_device_id charger_manager_match[] = {
1274 {
1275 .compatible = "charger-manager",
1276 },
1277 {},
1278 };
1279 MODULE_DEVICE_TABLE(of, charger_manager_match);
1280
1281 static struct charger_desc *of_cm_parse_desc(struct device *dev)
1282 {
1283 struct charger_desc *desc;
1284 struct device_node *np = dev->of_node;
1285 u32 poll_mode = CM_POLL_DISABLE;
1286 u32 battery_stat = CM_NO_BATTERY;
1287 int num_chgs = 0;
1288
1289 desc = devm_kzalloc(dev, sizeof(*desc), GFP_KERNEL);
1290 if (!desc)
1291 return ERR_PTR(-ENOMEM);
1292
1293 of_property_read_string(np, "cm-name", &desc->psy_name);
1294
1295 of_property_read_u32(np, "cm-poll-mode", &poll_mode);
1296 desc->polling_mode = poll_mode;
1297
1298 of_property_read_u32(np, "cm-poll-interval",
1299 &desc->polling_interval_ms);
1300
1301 of_property_read_u32(np, "cm-fullbatt-vchkdrop-volt",
1302 &desc->fullbatt_vchkdrop_uV);
1303 of_property_read_u32(np, "cm-fullbatt-voltage", &desc->fullbatt_uV);
1304 of_property_read_u32(np, "cm-fullbatt-soc", &desc->fullbatt_soc);
1305 of_property_read_u32(np, "cm-fullbatt-capacity",
1306 &desc->fullbatt_full_capacity);
1307
1308 of_property_read_u32(np, "cm-battery-stat", &battery_stat);
1309 desc->battery_present = battery_stat;
1310
1311
1312 num_chgs = of_property_count_strings(np, "cm-chargers");
1313 if (num_chgs > 0) {
1314 int i;
1315
1316
1317 desc->psy_charger_stat = devm_kcalloc(dev,
1318 num_chgs + 1,
1319 sizeof(char *),
1320 GFP_KERNEL);
1321 if (!desc->psy_charger_stat)
1322 return ERR_PTR(-ENOMEM);
1323
1324 for (i = 0; i < num_chgs; i++)
1325 of_property_read_string_index(np, "cm-chargers",
1326 i, &desc->psy_charger_stat[i]);
1327 }
1328
1329 of_property_read_string(np, "cm-fuel-gauge", &desc->psy_fuel_gauge);
1330
1331 of_property_read_string(np, "cm-thermal-zone", &desc->thermal_zone);
1332
1333 of_property_read_u32(np, "cm-battery-cold", &desc->temp_min);
1334 if (of_get_property(np, "cm-battery-cold-in-minus", NULL))
1335 desc->temp_min *= -1;
1336 of_property_read_u32(np, "cm-battery-hot", &desc->temp_max);
1337 of_property_read_u32(np, "cm-battery-temp-diff", &desc->temp_diff);
1338
1339 of_property_read_u32(np, "cm-charging-max",
1340 &desc->charging_max_duration_ms);
1341 of_property_read_u32(np, "cm-discharging-max",
1342 &desc->discharging_max_duration_ms);
1343
1344
1345 desc->num_charger_regulators = of_get_child_count(np);
1346 if (desc->num_charger_regulators) {
1347 struct charger_regulator *chg_regs;
1348 struct device_node *child;
1349
1350 chg_regs = devm_kcalloc(dev,
1351 desc->num_charger_regulators,
1352 sizeof(*chg_regs),
1353 GFP_KERNEL);
1354 if (!chg_regs)
1355 return ERR_PTR(-ENOMEM);
1356
1357 desc->charger_regulators = chg_regs;
1358
1359 desc->sysfs_groups = devm_kcalloc(dev,
1360 desc->num_charger_regulators + 1,
1361 sizeof(*desc->sysfs_groups),
1362 GFP_KERNEL);
1363 if (!desc->sysfs_groups)
1364 return ERR_PTR(-ENOMEM);
1365
1366 for_each_child_of_node(np, child) {
1367 struct charger_cable *cables;
1368 struct device_node *_child;
1369
1370 of_property_read_string(child, "cm-regulator-name",
1371 &chg_regs->regulator_name);
1372
1373
1374 chg_regs->num_cables = of_get_child_count(child);
1375 if (chg_regs->num_cables) {
1376 cables = devm_kcalloc(dev,
1377 chg_regs->num_cables,
1378 sizeof(*cables),
1379 GFP_KERNEL);
1380 if (!cables) {
1381 of_node_put(child);
1382 return ERR_PTR(-ENOMEM);
1383 }
1384
1385 chg_regs->cables = cables;
1386
1387 for_each_child_of_node(child, _child) {
1388 of_property_read_string(_child,
1389 "cm-cable-name", &cables->name);
1390 of_property_read_string(_child,
1391 "cm-cable-extcon",
1392 &cables->extcon_name);
1393 of_property_read_u32(_child,
1394 "cm-cable-min",
1395 &cables->min_uA);
1396 of_property_read_u32(_child,
1397 "cm-cable-max",
1398 &cables->max_uA);
1399 cables++;
1400 }
1401 }
1402 chg_regs++;
1403 }
1404 }
1405 return desc;
1406 }
1407
1408 static inline struct charger_desc *cm_get_drv_data(struct platform_device *pdev)
1409 {
1410 if (pdev->dev.of_node)
1411 return of_cm_parse_desc(&pdev->dev);
1412 return dev_get_platdata(&pdev->dev);
1413 }
1414
1415 static enum alarmtimer_restart cm_timer_func(struct alarm *alarm, ktime_t now)
1416 {
1417 cm_timer_set = false;
1418 return ALARMTIMER_NORESTART;
1419 }
1420
1421 static int charger_manager_probe(struct platform_device *pdev)
1422 {
1423 struct charger_desc *desc = cm_get_drv_data(pdev);
1424 struct charger_manager *cm;
1425 int ret, i = 0;
1426 union power_supply_propval val;
1427 struct power_supply *fuel_gauge;
1428 enum power_supply_property *properties;
1429 size_t num_properties;
1430 struct power_supply_config psy_cfg = {};
1431
1432 if (IS_ERR(desc)) {
1433 dev_err(&pdev->dev, "No platform data (desc) found\n");
1434 return PTR_ERR(desc);
1435 }
1436
1437 cm = devm_kzalloc(&pdev->dev, sizeof(*cm), GFP_KERNEL);
1438 if (!cm)
1439 return -ENOMEM;
1440
1441
1442 cm->dev = &pdev->dev;
1443 cm->desc = desc;
1444 psy_cfg.drv_data = cm;
1445
1446
1447 if (alarmtimer_get_rtcdev()) {
1448 cm_timer = devm_kzalloc(cm->dev, sizeof(*cm_timer), GFP_KERNEL);
1449 if (!cm_timer)
1450 return -ENOMEM;
1451 alarm_init(cm_timer, ALARM_BOOTTIME, cm_timer_func);
1452 }
1453
1454
1455
1456
1457
1458 if (desc->fullbatt_uV == 0) {
1459 dev_info(&pdev->dev, "Ignoring full-battery voltage threshold as it is not supplied\n");
1460 }
1461 if (!desc->fullbatt_vchkdrop_uV) {
1462 dev_info(&pdev->dev, "Disabling full-battery voltage drop checking mechanism as it is not supplied\n");
1463 desc->fullbatt_vchkdrop_uV = 0;
1464 }
1465 if (desc->fullbatt_soc == 0) {
1466 dev_info(&pdev->dev, "Ignoring full-battery soc(state of charge) threshold as it is not supplied\n");
1467 }
1468 if (desc->fullbatt_full_capacity == 0) {
1469 dev_info(&pdev->dev, "Ignoring full-battery full capacity threshold as it is not supplied\n");
1470 }
1471
1472 if (!desc->charger_regulators || desc->num_charger_regulators < 1) {
1473 dev_err(&pdev->dev, "charger_regulators undefined\n");
1474 return -EINVAL;
1475 }
1476
1477 if (!desc->psy_charger_stat || !desc->psy_charger_stat[0]) {
1478 dev_err(&pdev->dev, "No power supply defined\n");
1479 return -EINVAL;
1480 }
1481
1482 if (!desc->psy_fuel_gauge) {
1483 dev_err(&pdev->dev, "No fuel gauge power supply defined\n");
1484 return -EINVAL;
1485 }
1486
1487
1488 for (i = 0; desc->psy_charger_stat[i]; i++) {
1489 struct power_supply *psy;
1490
1491 psy = power_supply_get_by_name(desc->psy_charger_stat[i]);
1492 if (!psy) {
1493 dev_err(&pdev->dev, "Cannot find power supply \"%s\"\n",
1494 desc->psy_charger_stat[i]);
1495 return -ENODEV;
1496 }
1497 power_supply_put(psy);
1498 }
1499
1500 if (cm->desc->polling_mode != CM_POLL_DISABLE &&
1501 (desc->polling_interval_ms == 0 ||
1502 msecs_to_jiffies(desc->polling_interval_ms) <= CM_JIFFIES_SMALL)) {
1503 dev_err(&pdev->dev, "polling_interval_ms is too small\n");
1504 return -EINVAL;
1505 }
1506
1507 if (!desc->charging_max_duration_ms ||
1508 !desc->discharging_max_duration_ms) {
1509 dev_info(&pdev->dev, "Cannot limit charging duration checking mechanism to prevent overcharge/overheat and control discharging duration\n");
1510 desc->charging_max_duration_ms = 0;
1511 desc->discharging_max_duration_ms = 0;
1512 }
1513
1514 platform_set_drvdata(pdev, cm);
1515
1516 memcpy(&cm->charger_psy_desc, &psy_default, sizeof(psy_default));
1517
1518 if (!desc->psy_name)
1519 strncpy(cm->psy_name_buf, psy_default.name, PSY_NAME_MAX);
1520 else
1521 strncpy(cm->psy_name_buf, desc->psy_name, PSY_NAME_MAX);
1522 cm->charger_psy_desc.name = cm->psy_name_buf;
1523
1524
1525 properties = devm_kcalloc(&pdev->dev,
1526 ARRAY_SIZE(default_charger_props) +
1527 NUM_CHARGER_PSY_OPTIONAL,
1528 sizeof(*properties), GFP_KERNEL);
1529 if (!properties)
1530 return -ENOMEM;
1531
1532 memcpy(properties, default_charger_props,
1533 sizeof(enum power_supply_property) *
1534 ARRAY_SIZE(default_charger_props));
1535 num_properties = ARRAY_SIZE(default_charger_props);
1536
1537
1538 fuel_gauge = power_supply_get_by_name(desc->psy_fuel_gauge);
1539 if (!fuel_gauge) {
1540 dev_err(&pdev->dev, "Cannot find power supply \"%s\"\n",
1541 desc->psy_fuel_gauge);
1542 return -ENODEV;
1543 }
1544 if (!power_supply_get_property(fuel_gauge,
1545 POWER_SUPPLY_PROP_CHARGE_FULL, &val)) {
1546 properties[num_properties] =
1547 POWER_SUPPLY_PROP_CHARGE_FULL;
1548 num_properties++;
1549 }
1550 if (!power_supply_get_property(fuel_gauge,
1551 POWER_SUPPLY_PROP_CHARGE_NOW, &val)) {
1552 properties[num_properties] =
1553 POWER_SUPPLY_PROP_CHARGE_NOW;
1554 num_properties++;
1555 }
1556 if (!power_supply_get_property(fuel_gauge,
1557 POWER_SUPPLY_PROP_CURRENT_NOW,
1558 &val)) {
1559 properties[num_properties] =
1560 POWER_SUPPLY_PROP_CURRENT_NOW;
1561 num_properties++;
1562 }
1563
1564 ret = cm_init_thermal_data(cm, fuel_gauge, properties, &num_properties);
1565 if (ret) {
1566 dev_err(&pdev->dev, "Failed to initialize thermal data\n");
1567 cm->desc->measure_battery_temp = false;
1568 }
1569 power_supply_put(fuel_gauge);
1570
1571 cm->charger_psy_desc.properties = properties;
1572 cm->charger_psy_desc.num_properties = num_properties;
1573
1574
1575 ret = charger_manager_prepare_sysfs(cm);
1576 if (ret < 0) {
1577 dev_err(&pdev->dev,
1578 "Cannot prepare sysfs entry of regulators\n");
1579 return ret;
1580 }
1581 psy_cfg.attr_grp = desc->sysfs_groups;
1582
1583 cm->charger_psy = power_supply_register(&pdev->dev,
1584 &cm->charger_psy_desc,
1585 &psy_cfg);
1586 if (IS_ERR(cm->charger_psy)) {
1587 dev_err(&pdev->dev, "Cannot register charger-manager with name \"%s\"\n",
1588 cm->charger_psy_desc.name);
1589 return PTR_ERR(cm->charger_psy);
1590 }
1591
1592
1593 ret = charger_manager_register_extcon(cm);
1594 if (ret < 0) {
1595 dev_err(&pdev->dev, "Cannot initialize extcon device\n");
1596 goto err_reg_extcon;
1597 }
1598
1599
1600 mutex_lock(&cm_list_mtx);
1601 list_add(&cm->entry, &cm_list);
1602 mutex_unlock(&cm_list_mtx);
1603
1604
1605
1606
1607
1608 device_init_wakeup(&pdev->dev, true);
1609 device_set_wakeup_capable(&pdev->dev, false);
1610
1611
1612
1613
1614
1615
1616 cm_monitor();
1617
1618 schedule_work(&setup_polling);
1619
1620 return 0;
1621
1622 err_reg_extcon:
1623 for (i = 0; i < desc->num_charger_regulators; i++)
1624 regulator_put(desc->charger_regulators[i].consumer);
1625
1626 power_supply_unregister(cm->charger_psy);
1627
1628 return ret;
1629 }
1630
1631 static int charger_manager_remove(struct platform_device *pdev)
1632 {
1633 struct charger_manager *cm = platform_get_drvdata(pdev);
1634 struct charger_desc *desc = cm->desc;
1635 int i = 0;
1636
1637
1638 mutex_lock(&cm_list_mtx);
1639 list_del(&cm->entry);
1640 mutex_unlock(&cm_list_mtx);
1641
1642 cancel_work_sync(&setup_polling);
1643 cancel_delayed_work_sync(&cm_monitor_work);
1644
1645 for (i = 0 ; i < desc->num_charger_regulators ; i++)
1646 regulator_put(desc->charger_regulators[i].consumer);
1647
1648 power_supply_unregister(cm->charger_psy);
1649
1650 try_charger_enable(cm, false);
1651
1652 return 0;
1653 }
1654
1655 static const struct platform_device_id charger_manager_id[] = {
1656 { "charger-manager", 0 },
1657 { },
1658 };
1659 MODULE_DEVICE_TABLE(platform, charger_manager_id);
1660
1661 static int cm_suspend_noirq(struct device *dev)
1662 {
1663 if (device_may_wakeup(dev)) {
1664 device_set_wakeup_capable(dev, false);
1665 return -EAGAIN;
1666 }
1667
1668 return 0;
1669 }
1670
1671 static bool cm_need_to_awake(void)
1672 {
1673 struct charger_manager *cm;
1674
1675 if (cm_timer)
1676 return false;
1677
1678 mutex_lock(&cm_list_mtx);
1679 list_for_each_entry(cm, &cm_list, entry) {
1680 if (is_charging(cm)) {
1681 mutex_unlock(&cm_list_mtx);
1682 return true;
1683 }
1684 }
1685 mutex_unlock(&cm_list_mtx);
1686
1687 return false;
1688 }
1689
1690 static int cm_suspend_prepare(struct device *dev)
1691 {
1692 if (cm_need_to_awake())
1693 return -EBUSY;
1694
1695 if (!cm_suspended)
1696 cm_suspended = true;
1697
1698 cm_timer_set = cm_setup_timer();
1699
1700 if (cm_timer_set) {
1701 cancel_work_sync(&setup_polling);
1702 cancel_delayed_work_sync(&cm_monitor_work);
1703 }
1704
1705 return 0;
1706 }
1707
1708 static void cm_suspend_complete(struct device *dev)
1709 {
1710 struct charger_manager *cm = dev_get_drvdata(dev);
1711
1712 if (cm_suspended)
1713 cm_suspended = false;
1714
1715 if (cm_timer_set) {
1716 ktime_t remain;
1717
1718 alarm_cancel(cm_timer);
1719 cm_timer_set = false;
1720 remain = alarm_expires_remaining(cm_timer);
1721 cm_suspend_duration_ms -= ktime_to_ms(remain);
1722 schedule_work(&setup_polling);
1723 }
1724
1725 _cm_monitor(cm);
1726
1727 device_set_wakeup_capable(cm->dev, false);
1728 }
1729
1730 static const struct dev_pm_ops charger_manager_pm = {
1731 .prepare = cm_suspend_prepare,
1732 .suspend_noirq = cm_suspend_noirq,
1733 .complete = cm_suspend_complete,
1734 };
1735
1736 static struct platform_driver charger_manager_driver = {
1737 .driver = {
1738 .name = "charger-manager",
1739 .pm = &charger_manager_pm,
1740 .of_match_table = charger_manager_match,
1741 },
1742 .probe = charger_manager_probe,
1743 .remove = charger_manager_remove,
1744 .id_table = charger_manager_id,
1745 };
1746
1747 static int __init charger_manager_init(void)
1748 {
1749 cm_wq = create_freezable_workqueue("charger_manager");
1750 if (unlikely(!cm_wq))
1751 return -ENOMEM;
1752
1753 INIT_DELAYED_WORK(&cm_monitor_work, cm_monitor_poller);
1754
1755 return platform_driver_register(&charger_manager_driver);
1756 }
1757 late_initcall(charger_manager_init);
1758
1759 static void __exit charger_manager_cleanup(void)
1760 {
1761 destroy_workqueue(cm_wq);
1762 cm_wq = NULL;
1763
1764 platform_driver_unregister(&charger_manager_driver);
1765 }
1766 module_exit(charger_manager_cleanup);
1767
1768 MODULE_AUTHOR("MyungJoo Ham <myungjoo.ham@samsung.com>");
1769 MODULE_DESCRIPTION("Charger Manager");
1770 MODULE_LICENSE("GPL");