| /* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved. |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License version 2 and |
| * only version 2 as published by the Free Software Foundation. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| */ |
| #define pr_fmt(fmt) "%s: " fmt, __func__ |
| |
| #include <linux/module.h> |
| #include <linux/moduleparam.h> |
| #include <linux/platform_device.h> |
| #include <linux/errno.h> |
| #include <linux/device.h> |
| #include <linux/power_supply.h> |
| #include <linux/delay.h> |
| #include <linux/slab.h> |
| |
| #define BCL_DEV_NAME "battery_current_limit" |
| #define BCL_NAME_LENGTH 20 |
| /* |
| * Default BCL poll interval 1000 msec |
| */ |
| #define BCL_POLL_INTERVAL 1000 |
| /* |
| * Mininum BCL poll interval 10 msec |
| */ |
| #define MIN_BCL_POLL_INTERVAL 10 |
| |
| static const char bcl_type[] = "bcl"; |
| |
| /* |
| * Battery Current Limit Enable or Not |
| */ |
| enum bcl_device_mode { |
| BCL_DEVICE_DISABLED = 0, |
| BCL_DEVICE_ENABLED, |
| }; |
| |
| /* |
| * Battery Current Limit IBat Imax Threshold Mode |
| */ |
| enum bcl_ibat_imax_threshold_mode { |
| BCL_IBAT_IMAX_THRESHOLD_DISABLED = 0, |
| BCL_IBAT_IMAX_THRESHOLD_ENABLED, |
| }; |
| |
| /* |
| * Battery Current Limit Ibat Imax Trip Type (High and Low Threshold) |
| */ |
| enum bcl_ibat_imax_threshold_type { |
| BCL_IBAT_IMAX_THRESHOLD_TYPE_LOW = 0, |
| BCL_IBAT_IMAX_THRESHOLD_TYPE_HIGH, |
| BCL_IBAT_IMAX_THRESHOLD_TYPE_MAX, |
| }; |
| |
| /** |
| * BCL control block |
| * |
| */ |
| struct bcl_context { |
| /* BCL device */ |
| struct device *dev; |
| |
| /* BCL related config parameter */ |
| /* BCL mode enable or not */ |
| enum bcl_device_mode bcl_mode; |
| /* BCL Ibat/IMax Threshold Activate or Not */ |
| enum bcl_ibat_imax_threshold_mode |
| bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_MAX]; |
| /* BCL Ibat/IMax Threshold value in milli Amp */ |
| int bcl_threshold_value_ma[BCL_IBAT_IMAX_THRESHOLD_TYPE_MAX]; |
| /* BCL Type */ |
| char bcl_type[BCL_NAME_LENGTH]; |
| /* BCL poll in usec */ |
| int bcl_poll_interval_msec; |
| |
| /* BCL realtime value based on poll */ |
| /* BCL realtime ibat in milli Amp*/ |
| int bcl_ibat_ma; |
| /* BCL realtime calculated imax in milli Amp*/ |
| int bcl_imax_ma; |
| /* BCL realtime calculated ocv in uV*/ |
| int bcl_ocv_uv; |
| /* BCL realtime vbat in mV*/ |
| int bcl_vbat_mv; |
| /* BCL realtime rbat in mOhms*/ |
| int bcl_rbat; |
| /* BCL period poll delay work structure */ |
| struct delayed_work bcl_imax_work; |
| |
| }; |
| |
| static struct bcl_context *gbcl; |
| |
| /* |
| * BCL imax calculation and trigger notification to user space |
| * if imax cross threshold |
| */ |
| static void bcl_calculate_imax_trigger(void) |
| { |
| int ibatt_ua, vbatt_uv; |
| int imax_ma; |
| int ibatt_ma, vbatt_mv; |
| int imax_low_threshold; |
| int imax_high_threshold; |
| bool threshold_cross = false; |
| union power_supply_propval ret = {0,}; |
| static struct power_supply *psy; |
| |
| if (!gbcl) { |
| pr_err("called before initialization\n"); |
| return; |
| } |
| |
| if (psy == NULL) { |
| psy = power_supply_get_by_name("battery"); |
| if (psy == NULL) { |
| pr_err("failed to get ps battery\n"); |
| return; |
| } |
| } |
| |
| if (psy->get_property(psy, POWER_SUPPLY_PROP_CURRENT_NOW, &ret)) |
| return; |
| ibatt_ua = ret.intval; |
| |
| if (psy->get_property(psy, POWER_SUPPLY_PROP_VOLTAGE_NOW, &ret)) |
| return; |
| vbatt_uv = ret.intval; |
| |
| if (psy->get_property(psy, POWER_SUPPLY_PROP_CURRENT_MAX, &ret)) |
| return; |
| imax_ma = ret.intval/1000; |
| |
| ibatt_ma = ibatt_ua/1000; |
| vbatt_mv = vbatt_uv/1000; |
| |
| gbcl->bcl_ibat_ma = ibatt_ma; |
| gbcl->bcl_imax_ma = imax_ma; |
| gbcl->bcl_vbat_mv = vbatt_mv; |
| |
| pr_debug("ibatt %d, imax %d, vbatt %d\n", ibatt_ma, imax_ma, vbatt_mv); |
| if (gbcl->bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_HIGH] |
| == BCL_IBAT_IMAX_THRESHOLD_ENABLED) { |
| imax_high_threshold = |
| imax_ma - gbcl->bcl_threshold_value_ma |
| [BCL_IBAT_IMAX_THRESHOLD_TYPE_HIGH]; |
| if (ibatt_ma >= imax_high_threshold) |
| threshold_cross = true; |
| } |
| |
| if (gbcl->bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_LOW] |
| == BCL_IBAT_IMAX_THRESHOLD_ENABLED) { |
| imax_low_threshold = |
| imax_ma - gbcl->bcl_threshold_value_ma |
| [BCL_IBAT_IMAX_THRESHOLD_TYPE_LOW]; |
| if (ibatt_ma <= imax_low_threshold) |
| threshold_cross = true; |
| } |
| |
| if (threshold_cross) { |
| sysfs_notify(&gbcl->dev->kobj, |
| NULL, "type"); |
| } |
| } |
| |
| /* |
| * BCL imax work |
| */ |
| static void bcl_imax_work(struct work_struct *work) |
| { |
| struct bcl_context *bcl = container_of(work, |
| struct bcl_context, bcl_imax_work.work); |
| |
| if (gbcl->bcl_mode == BCL_DEVICE_ENABLED) { |
| bcl_calculate_imax_trigger(); |
| /* restart the delay work for caculating imax */ |
| schedule_delayed_work(&bcl->bcl_imax_work, |
| msecs_to_jiffies(bcl->bcl_poll_interval_msec)); |
| } |
| } |
| |
| /* |
| * Set BCL mode |
| */ |
| static void bcl_mode_set(enum bcl_device_mode mode) |
| { |
| if (!gbcl) |
| return; |
| |
| if (gbcl->bcl_mode == mode) |
| return; |
| |
| if (gbcl->bcl_mode == BCL_DEVICE_DISABLED |
| && mode == BCL_DEVICE_ENABLED) { |
| gbcl->bcl_mode = mode; |
| bcl_imax_work(&(gbcl->bcl_imax_work.work)); |
| return; |
| } else if (gbcl->bcl_mode == BCL_DEVICE_ENABLED |
| && mode == BCL_DEVICE_DISABLED) { |
| gbcl->bcl_mode = mode; |
| cancel_delayed_work_sync(&(gbcl->bcl_imax_work)); |
| return; |
| } |
| |
| return; |
| } |
| |
| #define show_bcl(name, variable, format) \ |
| static ssize_t \ |
| name##_show(struct device *dev, struct device_attribute *attr, char *buf) \ |
| { \ |
| if (gbcl) \ |
| return snprintf(buf, PAGE_SIZE, format, gbcl->variable); \ |
| else \ |
| return -EPERM; \ |
| } |
| |
| show_bcl(type, bcl_type, "%s\n") |
| show_bcl(ibat, bcl_ibat_ma, "%d\n") |
| show_bcl(imax, bcl_imax_ma, "%d\n") |
| show_bcl(vbat, bcl_vbat_mv, "%d\n") |
| show_bcl(rbat, bcl_rbat, "%d\n") |
| show_bcl(ocv, bcl_ocv_uv, "%d\n") |
| show_bcl(poll_interval, bcl_poll_interval_msec, "%d\n") |
| |
| static ssize_t |
| mode_show(struct device *dev, struct device_attribute *attr, char *buf) |
| { |
| if (!gbcl) |
| return -EPERM; |
| |
| return snprintf(buf, PAGE_SIZE, "%s\n", |
| gbcl->bcl_mode == BCL_DEVICE_ENABLED ? "enabled" |
| : "disabled"); |
| } |
| |
| static ssize_t |
| mode_store(struct device *dev, struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| if (!gbcl) |
| return -EPERM; |
| |
| if (!strncmp(buf, "enabled", 7)) |
| bcl_mode_set(BCL_DEVICE_ENABLED); |
| else if (!strncmp(buf, "disabled", 8)) |
| bcl_mode_set(BCL_DEVICE_DISABLED); |
| else |
| return -EINVAL; |
| |
| return count; |
| } |
| |
| static ssize_t |
| ibat_imax_low_threshold_mode_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| if (!gbcl) |
| return -EPERM; |
| |
| return snprintf(buf, PAGE_SIZE, "%s\n", |
| gbcl->bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_LOW] |
| == BCL_IBAT_IMAX_THRESHOLD_ENABLED ? "enabled" : "disabled"); |
| } |
| |
| static ssize_t |
| ibat_imax_low_threshold_mode_store(struct device *dev, |
| struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| if (!gbcl) |
| return -EPERM; |
| |
| if (!strncmp(buf, "enabled", 7)) |
| gbcl->bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_LOW] |
| = BCL_IBAT_IMAX_THRESHOLD_ENABLED; |
| else if (!strncmp(buf, "disabled", 8)) |
| gbcl->bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_LOW] |
| = BCL_IBAT_IMAX_THRESHOLD_DISABLED; |
| else |
| return -EINVAL; |
| |
| return count; |
| } |
| |
| static ssize_t |
| ibat_imax_low_threshold_value_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| if (!gbcl) |
| return -EPERM; |
| |
| return snprintf(buf, PAGE_SIZE, "%d\n", |
| gbcl->bcl_threshold_value_ma[BCL_IBAT_IMAX_THRESHOLD_TYPE_LOW]); |
| } |
| |
| static ssize_t |
| ibat_imax_low_threshold_value_store(struct device *dev, |
| struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| int value; |
| |
| if (!gbcl) |
| return -EPERM; |
| |
| if (!sscanf(buf, "%d", &value)) |
| return -EINVAL; |
| |
| if (value < 0) |
| return -EINVAL; |
| |
| gbcl->bcl_threshold_value_ma[BCL_IBAT_IMAX_THRESHOLD_TYPE_LOW] |
| = value; |
| |
| return count; |
| } |
| |
| static ssize_t |
| ibat_imax_high_threshold_mode_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| if (!gbcl) |
| return -EPERM; |
| |
| return snprintf(buf, PAGE_SIZE, "%s\n", |
| gbcl->bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_HIGH] |
| == BCL_IBAT_IMAX_THRESHOLD_ENABLED ? "enabled" : "disabled"); |
| } |
| |
| static ssize_t |
| ibat_imax_high_threshold_mode_store(struct device *dev, |
| struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| if (!gbcl) |
| return -EPERM; |
| |
| if (!strncmp(buf, "enabled", 7)) |
| gbcl->bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_HIGH] |
| = BCL_IBAT_IMAX_THRESHOLD_ENABLED; |
| else if (!strncmp(buf, "disabled", 8)) |
| gbcl->bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_HIGH] |
| = BCL_IBAT_IMAX_THRESHOLD_DISABLED; |
| else |
| return -EINVAL; |
| |
| return count; |
| } |
| |
| static ssize_t |
| ibat_imax_high_threshold_value_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| if (!gbcl) |
| return -EPERM; |
| |
| return snprintf(buf, PAGE_SIZE, "%d\n", |
| gbcl->bcl_threshold_value_ma[BCL_IBAT_IMAX_THRESHOLD_TYPE_HIGH]); |
| } |
| |
| static ssize_t |
| ibat_imax_high_threshold_value_store(struct device *dev, |
| struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| int value; |
| |
| if (!gbcl) |
| return -EPERM; |
| |
| if (!sscanf(buf, "%d", &value)) |
| return -EINVAL; |
| |
| if (value < 0) |
| return -EINVAL; |
| |
| gbcl->bcl_threshold_value_ma[BCL_IBAT_IMAX_THRESHOLD_TYPE_HIGH] |
| = value; |
| |
| return count; |
| } |
| |
| static ssize_t |
| poll_interval_store(struct device *dev, |
| struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| int value; |
| |
| if (!gbcl) |
| return -EPERM; |
| |
| if (!sscanf(buf, "%d", &value)) |
| return -EINVAL; |
| |
| if (value < MIN_BCL_POLL_INTERVAL) |
| return -EINVAL; |
| |
| gbcl->bcl_poll_interval_msec = value; |
| |
| return count; |
| } |
| |
| /* |
| * BCL device attributes |
| */ |
| static struct device_attribute bcl_dev_attr[] = { |
| __ATTR(type, 0444, type_show, NULL), |
| __ATTR(ibat, 0444, ibat_show, NULL), |
| __ATTR(vbat, 0444, vbat_show, NULL), |
| __ATTR(rbat, 0444, rbat_show, NULL), |
| __ATTR(ocv, 0444, ocv_show, NULL), |
| __ATTR(imax, 0444, imax_show, NULL), |
| __ATTR(mode, 0644, mode_show, mode_store), |
| __ATTR(poll_interval, 0644, |
| poll_interval_show, poll_interval_store), |
| __ATTR(ibat_imax_low_threshold_mode, 0644, |
| ibat_imax_low_threshold_mode_show, |
| ibat_imax_low_threshold_mode_store), |
| __ATTR(ibat_imax_high_threshold_mode, 0644, |
| ibat_imax_high_threshold_mode_show, |
| ibat_imax_high_threshold_mode_store), |
| __ATTR(ibat_imax_low_threshold_value, 0644, |
| ibat_imax_low_threshold_value_show, |
| ibat_imax_low_threshold_value_store), |
| __ATTR(ibat_imax_high_threshold_value, 0644, |
| ibat_imax_high_threshold_value_show, |
| ibat_imax_high_threshold_value_store) |
| }; |
| |
| static int create_bcl_sysfs(struct bcl_context *bcl) |
| { |
| int result = 0; |
| int num_attr = sizeof(bcl_dev_attr)/sizeof(struct device_attribute); |
| int i; |
| |
| for (i = 0; i < num_attr; i++) { |
| result = device_create_file(bcl->dev, &bcl_dev_attr[i]); |
| if (result < 0) |
| return result; |
| } |
| |
| return 0; |
| } |
| |
| static void remove_bcl_sysfs(struct bcl_context *bcl) |
| { |
| int num_attr = sizeof(bcl_dev_attr)/sizeof(struct device_attribute); |
| int i; |
| |
| for (i = 0; i < num_attr; i++) |
| device_remove_file(bcl->dev, &bcl_dev_attr[i]); |
| |
| return; |
| } |
| |
| static int __devinit bcl_probe(struct platform_device *pdev) |
| { |
| struct bcl_context *bcl; |
| int ret = 0; |
| |
| bcl = kzalloc(sizeof(struct bcl_context), GFP_KERNEL); |
| |
| if (!bcl) { |
| pr_err("Cannot allocate bcl_context\n"); |
| return -ENOMEM; |
| } |
| |
| gbcl = bcl; |
| |
| /* For BCL */ |
| /* Init default BCL params */ |
| bcl->dev = &pdev->dev; |
| bcl->bcl_mode = BCL_DEVICE_DISABLED; |
| bcl->bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_LOW] |
| = BCL_IBAT_IMAX_THRESHOLD_DISABLED; |
| bcl->bcl_threshold_mode[BCL_IBAT_IMAX_THRESHOLD_TYPE_HIGH] |
| = BCL_IBAT_IMAX_THRESHOLD_DISABLED; |
| bcl->bcl_threshold_value_ma[BCL_IBAT_IMAX_THRESHOLD_TYPE_LOW] = 0; |
| bcl->bcl_threshold_value_ma[BCL_IBAT_IMAX_THRESHOLD_TYPE_HIGH] = 0; |
| snprintf(bcl->bcl_type, BCL_NAME_LENGTH, "%s", bcl_type); |
| bcl->bcl_poll_interval_msec = BCL_POLL_INTERVAL; |
| ret = create_bcl_sysfs(bcl); |
| if (ret < 0) { |
| pr_err("Cannot create bcl sysfs\n"); |
| kfree(bcl); |
| return ret; |
| } |
| platform_set_drvdata(pdev, bcl); |
| INIT_DELAYED_WORK(&bcl->bcl_imax_work, bcl_imax_work); |
| |
| return 0; |
| } |
| |
| static int __devexit bcl_remove(struct platform_device *pdev) |
| { |
| remove_bcl_sysfs(gbcl); |
| kfree(gbcl); |
| gbcl = NULL; |
| platform_set_drvdata(pdev, NULL); |
| return 0; |
| } |
| |
| static struct of_device_id bcl_match_table[] = { |
| {.compatible = "qcom,bcl"}, |
| {}, |
| }; |
| |
| static struct platform_driver bcl_driver = { |
| .probe = bcl_probe, |
| .remove = __devexit_p(bcl_remove), |
| .driver = { |
| .name = BCL_DEV_NAME, |
| .owner = THIS_MODULE, |
| .of_match_table = bcl_match_table, |
| }, |
| }; |
| |
| static int __init bcl_init(void) |
| { |
| return platform_driver_register(&bcl_driver); |
| } |
| |
| static void __exit bcl_exit(void) |
| { |
| platform_driver_unregister(&bcl_driver); |
| } |
| |
| late_initcall(bcl_init); |
| module_exit(bcl_exit); |
| |
| MODULE_LICENSE("GPL v2"); |
| MODULE_DESCRIPTION("battery current limit driver"); |
| MODULE_ALIAS("platform:" BCL_DEV_NAME); |