Commit 1c1f13a0 authored by Linus Walleij's avatar Linus Walleij Committed by Sebastian Reichel

power: supply: ab8500: Move to componentized binding

The driver has problems with the different components of
the charging code racing with each other to probe().

This results in all four subdrivers populating battery
information to ascertain that it is populated for their
own needs for example.

Fix this by using component probing and thus expressing
to the kernel that these are dependent components.
The probes can happen in any order and will only acquire
resources such as state container, regulators and
interrupts and initialize the data structures, but no
execution happens until the .bind() callback is called.

The charging driver is the main component and binds
first, then bind in order the three subcomponents:
ab8500-fg, ab8500-btemp and ab8500-chargalg.

Do some housekeeping while we are moving the code around.
Like use devm_* for IRQs so as to cut down on some
boilerplate.
Signed-off-by: default avatarLinus Walleij <linus.walleij@linaro.org>
Signed-off-by: default avatarSebastian Reichel <sebastian.reichel@collabora.com>
parent 94233f11
...@@ -730,4 +730,8 @@ int ab8500_bm_of_probe(struct device *dev, ...@@ -730,4 +730,8 @@ int ab8500_bm_of_probe(struct device *dev,
struct device_node *np, struct device_node *np,
struct abx500_bm_data *bm); struct abx500_bm_data *bm);
extern struct platform_driver ab8500_fg_driver;
extern struct platform_driver ab8500_btemp_driver;
extern struct platform_driver abx500_chargalg_driver;
#endif /* _AB8500_CHARGER_H_ */ #endif /* _AB8500_CHARGER_H_ */
...@@ -13,6 +13,7 @@ ...@@ -13,6 +13,7 @@
#include <linux/init.h> #include <linux/init.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/device.h> #include <linux/device.h>
#include <linux/component.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/slab.h> #include <linux/slab.h>
...@@ -932,26 +933,6 @@ static int __maybe_unused ab8500_btemp_suspend(struct device *dev) ...@@ -932,26 +933,6 @@ static int __maybe_unused ab8500_btemp_suspend(struct device *dev)
return 0; return 0;
} }
static int ab8500_btemp_remove(struct platform_device *pdev)
{
struct ab8500_btemp *di = platform_get_drvdata(pdev);
int i, irq;
/* Disable interrupts */
for (i = 0; i < ARRAY_SIZE(ab8500_btemp_irq); i++) {
irq = platform_get_irq_byname(pdev, ab8500_btemp_irq[i].name);
free_irq(irq, di);
}
/* Delete the work queue */
destroy_workqueue(di->btemp_wq);
flush_scheduled_work();
power_supply_unregister(di->btemp_psy);
return 0;
}
static char *supply_interface[] = { static char *supply_interface[] = {
"ab8500_chargalg", "ab8500_chargalg",
"ab8500_fg", "ab8500_fg",
...@@ -966,6 +947,40 @@ static const struct power_supply_desc ab8500_btemp_desc = { ...@@ -966,6 +947,40 @@ static const struct power_supply_desc ab8500_btemp_desc = {
.external_power_changed = ab8500_btemp_external_power_changed, .external_power_changed = ab8500_btemp_external_power_changed,
}; };
static int ab8500_btemp_bind(struct device *dev, struct device *master,
void *data)
{
struct ab8500_btemp *di = dev_get_drvdata(dev);
/* Create a work queue for the btemp */
di->btemp_wq =
alloc_workqueue("ab8500_btemp_wq", WQ_MEM_RECLAIM, 0);
if (di->btemp_wq == NULL) {
dev_err(dev, "failed to create work queue\n");
return -ENOMEM;
}
/* Kick off periodic temperature measurements */
ab8500_btemp_periodic(di, true);
return 0;
}
static void ab8500_btemp_unbind(struct device *dev, struct device *master,
void *data)
{
struct ab8500_btemp *di = dev_get_drvdata(dev);
/* Delete the work queue */
destroy_workqueue(di->btemp_wq);
flush_scheduled_work();
}
static const struct component_ops ab8500_btemp_component_ops = {
.bind = ab8500_btemp_bind,
.unbind = ab8500_btemp_unbind,
};
static int ab8500_btemp_probe(struct platform_device *pdev) static int ab8500_btemp_probe(struct platform_device *pdev)
{ {
struct device_node *np = pdev->dev.of_node; struct device_node *np = pdev->dev.of_node;
...@@ -1011,14 +1026,6 @@ static int ab8500_btemp_probe(struct platform_device *pdev) ...@@ -1011,14 +1026,6 @@ static int ab8500_btemp_probe(struct platform_device *pdev)
psy_cfg.num_supplicants = ARRAY_SIZE(supply_interface); psy_cfg.num_supplicants = ARRAY_SIZE(supply_interface);
psy_cfg.drv_data = di; psy_cfg.drv_data = di;
/* Create a work queue for the btemp */
di->btemp_wq =
alloc_workqueue("ab8500_btemp_wq", WQ_MEM_RECLAIM, 0);
if (di->btemp_wq == NULL) {
dev_err(dev, "failed to create work queue\n");
return -ENOMEM;
}
/* Init work for measuring temperature periodically */ /* Init work for measuring temperature periodically */
INIT_DEFERRABLE_WORK(&di->btemp_periodic_work, INIT_DEFERRABLE_WORK(&di->btemp_periodic_work,
ab8500_btemp_periodic_work); ab8500_btemp_periodic_work);
...@@ -1031,7 +1038,7 @@ static int ab8500_btemp_probe(struct platform_device *pdev) ...@@ -1031,7 +1038,7 @@ static int ab8500_btemp_probe(struct platform_device *pdev)
AB8500_BTEMP_HIGH_TH, &val); AB8500_BTEMP_HIGH_TH, &val);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "%s ab8500 read failed\n", __func__); dev_err(dev, "%s ab8500 read failed\n", __func__);
goto free_btemp_wq; return ret;
} }
switch (val) { switch (val) {
case BTEMP_HIGH_TH_57_0: case BTEMP_HIGH_TH_57_0:
...@@ -1050,30 +1057,28 @@ static int ab8500_btemp_probe(struct platform_device *pdev) ...@@ -1050,30 +1057,28 @@ static int ab8500_btemp_probe(struct platform_device *pdev)
} }
/* Register BTEMP power supply class */ /* Register BTEMP power supply class */
di->btemp_psy = power_supply_register(dev, &ab8500_btemp_desc, di->btemp_psy = devm_power_supply_register(dev, &ab8500_btemp_desc,
&psy_cfg); &psy_cfg);
if (IS_ERR(di->btemp_psy)) { if (IS_ERR(di->btemp_psy)) {
dev_err(dev, "failed to register BTEMP psy\n"); dev_err(dev, "failed to register BTEMP psy\n");
ret = PTR_ERR(di->btemp_psy); return PTR_ERR(di->btemp_psy);
goto free_btemp_wq;
} }
/* Register interrupts */ /* Register interrupts */
for (i = 0; i < ARRAY_SIZE(ab8500_btemp_irq); i++) { for (i = 0; i < ARRAY_SIZE(ab8500_btemp_irq); i++) {
irq = platform_get_irq_byname(pdev, ab8500_btemp_irq[i].name); irq = platform_get_irq_byname(pdev, ab8500_btemp_irq[i].name);
if (irq < 0) { if (irq < 0)
ret = irq; return irq;
goto free_irq;
}
ret = request_threaded_irq(irq, NULL, ab8500_btemp_irq[i].isr, ret = devm_request_threaded_irq(dev, irq, NULL,
ab8500_btemp_irq[i].isr,
IRQF_SHARED | IRQF_NO_SUSPEND | IRQF_ONESHOT, IRQF_SHARED | IRQF_NO_SUSPEND | IRQF_ONESHOT,
ab8500_btemp_irq[i].name, di); ab8500_btemp_irq[i].name, di);
if (ret) { if (ret) {
dev_err(dev, "failed to request %s IRQ %d: %d\n" dev_err(dev, "failed to request %s IRQ %d: %d\n"
, ab8500_btemp_irq[i].name, irq, ret); , ab8500_btemp_irq[i].name, irq, ret);
goto free_irq; return ret;
} }
dev_dbg(dev, "Requested %s IRQ %d: %d\n", dev_dbg(dev, "Requested %s IRQ %d: %d\n",
ab8500_btemp_irq[i].name, irq, ret); ab8500_btemp_irq[i].name, irq, ret);
...@@ -1081,23 +1086,16 @@ static int ab8500_btemp_probe(struct platform_device *pdev) ...@@ -1081,23 +1086,16 @@ static int ab8500_btemp_probe(struct platform_device *pdev)
platform_set_drvdata(pdev, di); platform_set_drvdata(pdev, di);
/* Kick off periodic temperature measurements */
ab8500_btemp_periodic(di, true);
list_add_tail(&di->node, &ab8500_btemp_list); list_add_tail(&di->node, &ab8500_btemp_list);
return ret; return component_add(dev, &ab8500_btemp_component_ops);
}
free_irq: static int ab8500_btemp_remove(struct platform_device *pdev)
/* We also have to free all successfully registered irqs */ {
for (i = i - 1; i >= 0; i--) { component_del(&pdev->dev, &ab8500_btemp_component_ops);
irq = platform_get_irq_byname(pdev, ab8500_btemp_irq[i].name);
free_irq(irq, di);
}
power_supply_unregister(di->btemp_psy); return 0;
free_btemp_wq:
destroy_workqueue(di->btemp_wq);
return ret;
} }
static SIMPLE_DEV_PM_OPS(ab8500_btemp_pm_ops, ab8500_btemp_suspend, ab8500_btemp_resume); static SIMPLE_DEV_PM_OPS(ab8500_btemp_pm_ops, ab8500_btemp_suspend, ab8500_btemp_resume);
...@@ -1107,7 +1105,7 @@ static const struct of_device_id ab8500_btemp_match[] = { ...@@ -1107,7 +1105,7 @@ static const struct of_device_id ab8500_btemp_match[] = {
{ }, { },
}; };
static struct platform_driver ab8500_btemp_driver = { struct platform_driver ab8500_btemp_driver = {
.probe = ab8500_btemp_probe, .probe = ab8500_btemp_probe,
.remove = ab8500_btemp_remove, .remove = ab8500_btemp_remove,
.driver = { .driver = {
...@@ -1116,20 +1114,6 @@ static struct platform_driver ab8500_btemp_driver = { ...@@ -1116,20 +1114,6 @@ static struct platform_driver ab8500_btemp_driver = {
.pm = &ab8500_btemp_pm_ops, .pm = &ab8500_btemp_pm_ops,
}, },
}; };
static int __init ab8500_btemp_init(void)
{
return platform_driver_register(&ab8500_btemp_driver);
}
static void __exit ab8500_btemp_exit(void)
{
platform_driver_unregister(&ab8500_btemp_driver);
}
device_initcall(ab8500_btemp_init);
module_exit(ab8500_btemp_exit);
MODULE_LICENSE("GPL v2"); MODULE_LICENSE("GPL v2");
MODULE_AUTHOR("Johan Palsson, Karl Komierowski, Arun R Murthy"); MODULE_AUTHOR("Johan Palsson, Karl Komierowski, Arun R Murthy");
MODULE_ALIAS("platform:ab8500-btemp"); MODULE_ALIAS("platform:ab8500-btemp");
......
...@@ -13,6 +13,7 @@ ...@@ -13,6 +13,7 @@
#include <linux/init.h> #include <linux/init.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/device.h> #include <linux/device.h>
#include <linux/component.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/notifier.h> #include <linux/notifier.h>
...@@ -3276,10 +3277,74 @@ static struct notifier_block charger_nb = { ...@@ -3276,10 +3277,74 @@ static struct notifier_block charger_nb = {
.notifier_call = ab8500_external_charger_prepare, .notifier_call = ab8500_external_charger_prepare,
}; };
static int ab8500_charger_remove(struct platform_device *pdev) static char *supply_interface[] = {
"ab8500_chargalg",
"ab8500_fg",
"ab8500_btemp",
};
static const struct power_supply_desc ab8500_ac_chg_desc = {
.name = "ab8500_ac",
.type = POWER_SUPPLY_TYPE_MAINS,
.properties = ab8500_charger_ac_props,
.num_properties = ARRAY_SIZE(ab8500_charger_ac_props),
.get_property = ab8500_charger_ac_get_property,
};
static const struct power_supply_desc ab8500_usb_chg_desc = {
.name = "ab8500_usb",
.type = POWER_SUPPLY_TYPE_USB,
.properties = ab8500_charger_usb_props,
.num_properties = ARRAY_SIZE(ab8500_charger_usb_props),
.get_property = ab8500_charger_usb_get_property,
};
static int ab8500_charger_bind(struct device *dev)
{ {
struct ab8500_charger *di = platform_get_drvdata(pdev); struct ab8500_charger *di = dev_get_drvdata(dev);
int i, irq, ret; int ch_stat;
int ret;
/* Create a work queue for the charger */
di->charger_wq = alloc_ordered_workqueue("ab8500_charger_wq",
WQ_MEM_RECLAIM);
if (di->charger_wq == NULL) {
dev_err(dev, "failed to create work queue\n");
return -ENOMEM;
}
ch_stat = ab8500_charger_detect_chargers(di, false);
if (ch_stat & AC_PW_CONN) {
if (is_ab8500(di->parent))
queue_delayed_work(di->charger_wq,
&di->ac_charger_attached_work,
HZ);
}
if (ch_stat & USB_PW_CONN) {
if (is_ab8500(di->parent))
queue_delayed_work(di->charger_wq,
&di->usb_charger_attached_work,
HZ);
di->vbus_detected = true;
di->vbus_detected_start = true;
queue_work(di->charger_wq,
&di->detect_usb_type_work);
}
ret = component_bind_all(dev, di);
if (ret) {
dev_err(dev, "can't bind component devices\n");
return ret;
}
return 0;
}
static void ab8500_charger_unbind(struct device *dev)
{
struct ab8500_charger *di = dev_get_drvdata(dev);
int ret;
/* Disable AC charging */ /* Disable AC charging */
ab8500_charger_ac_en(&di->ac_chg, false, 0, 0); ab8500_charger_ac_en(&di->ac_chg, false, 0, 0);
...@@ -3287,68 +3352,47 @@ static int ab8500_charger_remove(struct platform_device *pdev) ...@@ -3287,68 +3352,47 @@ static int ab8500_charger_remove(struct platform_device *pdev)
/* Disable USB charging */ /* Disable USB charging */
ab8500_charger_usb_en(&di->usb_chg, false, 0, 0); ab8500_charger_usb_en(&di->usb_chg, false, 0, 0);
/* Disable interrupts */
for (i = 0; i < ARRAY_SIZE(ab8500_charger_irq); i++) {
irq = platform_get_irq_byname(pdev, ab8500_charger_irq[i].name);
free_irq(irq, di);
}
/* Backup battery voltage and current disable */ /* Backup battery voltage and current disable */
ret = abx500_mask_and_set_register_interruptible(di->dev, ret = abx500_mask_and_set_register_interruptible(di->dev,
AB8500_RTC, AB8500_RTC_CTRL_REG, RTC_BUP_CH_ENA, 0); AB8500_RTC, AB8500_RTC_CTRL_REG, RTC_BUP_CH_ENA, 0);
if (ret < 0) if (ret < 0)
dev_err(di->dev, "%s mask and set failed\n", __func__); dev_err(di->dev, "%s mask and set failed\n", __func__);
usb_unregister_notifier(di->usb_phy, &di->nb);
usb_put_phy(di->usb_phy);
/* Delete the work queue */ /* Delete the work queue */
destroy_workqueue(di->charger_wq); destroy_workqueue(di->charger_wq);
/* Unregister external charger enable notifier */
if (!di->ac_chg.enabled)
blocking_notifier_chain_unregister(
&charger_notifier_list, &charger_nb);
flush_scheduled_work(); flush_scheduled_work();
if (di->usb_chg.enabled)
power_supply_unregister(di->usb_chg.psy);
if (di->ac_chg.enabled && !di->ac_chg.external)
power_supply_unregister(di->ac_chg.psy);
return 0; /* Unbind fg, btemp, algorithm */
component_unbind_all(dev, di);
} }
static char *supply_interface[] = { static const struct component_master_ops ab8500_charger_comp_ops = {
"ab8500_chargalg", .bind = ab8500_charger_bind,
"ab8500_fg", .unbind = ab8500_charger_unbind,
"ab8500_btemp",
}; };
static const struct power_supply_desc ab8500_ac_chg_desc = { static struct platform_driver *const ab8500_charger_component_drivers[] = {
.name = "ab8500_ac", &ab8500_fg_driver,
.type = POWER_SUPPLY_TYPE_MAINS, &ab8500_btemp_driver,
.properties = ab8500_charger_ac_props, &abx500_chargalg_driver,
.num_properties = ARRAY_SIZE(ab8500_charger_ac_props),
.get_property = ab8500_charger_ac_get_property,
}; };
static const struct power_supply_desc ab8500_usb_chg_desc = { static int ab8500_charger_compare_dev(struct device *dev, void *data)
.name = "ab8500_usb", {
.type = POWER_SUPPLY_TYPE_USB, return dev == data;
.properties = ab8500_charger_usb_props, }
.num_properties = ARRAY_SIZE(ab8500_charger_usb_props),
.get_property = ab8500_charger_usb_get_property,
};
static int ab8500_charger_probe(struct platform_device *pdev) static int ab8500_charger_probe(struct platform_device *pdev)
{ {
struct device_node *np = pdev->dev.of_node; struct device *dev = &pdev->dev;
struct device_node *np = dev->of_node;
struct component_match *match = NULL;
struct power_supply_config ac_psy_cfg = {}, usb_psy_cfg = {}; struct power_supply_config ac_psy_cfg = {}, usb_psy_cfg = {};
struct ab8500_charger *di; struct ab8500_charger *di;
int irq, i, charger_status, ret = 0, ch_stat; int charger_status;
struct device *dev = &pdev->dev; int i, irq;
int ret;
di = devm_kzalloc(dev, sizeof(*di), GFP_KERNEL); di = devm_kzalloc(dev, sizeof(*di), GFP_KERNEL);
if (!di) if (!di)
...@@ -3393,6 +3437,38 @@ static int ab8500_charger_probe(struct platform_device *pdev) ...@@ -3393,6 +3437,38 @@ static int ab8500_charger_probe(struct platform_device *pdev)
return ret; return ret;
} }
/*
* VDD ADC supply needs to be enabled from this driver when there
* is a charger connected to avoid erroneous BTEMP_HIGH/LOW
* interrupts during charging
*/
di->regu = devm_regulator_get(dev, "vddadc");
if (IS_ERR(di->regu)) {
ret = PTR_ERR(di->regu);
dev_err(dev, "failed to get vddadc regulator\n");
return ret;
}
/* Request interrupts */
for (i = 0; i < ARRAY_SIZE(ab8500_charger_irq); i++) {
irq = platform_get_irq_byname(pdev, ab8500_charger_irq[i].name);
if (irq < 0)
return irq;
ret = devm_request_threaded_irq(dev,
irq, NULL, ab8500_charger_irq[i].isr,
IRQF_SHARED | IRQF_NO_SUSPEND | IRQF_ONESHOT,
ab8500_charger_irq[i].name, di);
if (ret != 0) {
dev_err(dev, "failed to request %s IRQ %d: %d\n"
, ab8500_charger_irq[i].name, irq, ret);
return ret;
}
dev_dbg(dev, "Requested %s IRQ %d: %d\n",
ab8500_charger_irq[i].name, irq, ret);
}
/* initialize lock */ /* initialize lock */
spin_lock_init(&di->usb_state.usb_lock); spin_lock_init(&di->usb_state.usb_lock);
mutex_init(&di->usb_ipt_crnt_lock); mutex_init(&di->usb_ipt_crnt_lock);
...@@ -3422,11 +3498,6 @@ static int ab8500_charger_probe(struct platform_device *pdev) ...@@ -3422,11 +3498,6 @@ static int ab8500_charger_probe(struct platform_device *pdev)
di->ac_chg.enabled = di->bm->ac_enabled; di->ac_chg.enabled = di->bm->ac_enabled;
di->ac_chg.external = false; di->ac_chg.external = false;
/*notifier for external charger enabling*/
if (!di->ac_chg.enabled)
blocking_notifier_chain_register(
&charger_notifier_list, &charger_nb);
/* USB supply */ /* USB supply */
/* ux500_charger sub-class */ /* ux500_charger sub-class */
di->usb_chg.ops.enable = &ab8500_charger_usb_en; di->usb_chg.ops.enable = &ab8500_charger_usb_en;
...@@ -3442,14 +3513,6 @@ static int ab8500_charger_probe(struct platform_device *pdev) ...@@ -3442,14 +3513,6 @@ static int ab8500_charger_probe(struct platform_device *pdev)
di->usb_chg.external = false; di->usb_chg.external = false;
di->usb_state.usb_current = -1; di->usb_state.usb_current = -1;
/* Create a work queue for the charger */
di->charger_wq = alloc_ordered_workqueue("ab8500_charger_wq",
WQ_MEM_RECLAIM);
if (di->charger_wq == NULL) {
dev_err(dev, "failed to create work queue\n");
return -ENOMEM;
}
mutex_init(&di->charger_attached_mutex); mutex_init(&di->charger_attached_mutex);
/* Init work for HW failure check */ /* Init work for HW failure check */
...@@ -3500,63 +3563,36 @@ static int ab8500_charger_probe(struct platform_device *pdev) ...@@ -3500,63 +3563,36 @@ static int ab8500_charger_probe(struct platform_device *pdev)
INIT_WORK(&di->check_usb_thermal_prot_work, INIT_WORK(&di->check_usb_thermal_prot_work,
ab8500_charger_check_usb_thermal_prot_work); ab8500_charger_check_usb_thermal_prot_work);
/*
* VDD ADC supply needs to be enabled from this driver when there
* is a charger connected to avoid erroneous BTEMP_HIGH/LOW
* interrupts during charging
*/
di->regu = devm_regulator_get(dev, "vddadc");
if (IS_ERR(di->regu)) {
ret = PTR_ERR(di->regu);
dev_err(dev, "failed to get vddadc regulator\n");
goto free_charger_wq;
}
/* Initialize OVV, and other registers */ /* Initialize OVV, and other registers */
ret = ab8500_charger_init_hw_registers(di); ret = ab8500_charger_init_hw_registers(di);
if (ret) { if (ret) {
dev_err(dev, "failed to initialize ABB registers\n"); dev_err(dev, "failed to initialize ABB registers\n");
goto free_charger_wq; return ret;
} }
/* Register AC charger class */ /* Register AC charger class */
if (di->ac_chg.enabled) { if (di->ac_chg.enabled) {
di->ac_chg.psy = power_supply_register(dev, di->ac_chg.psy = devm_power_supply_register(dev,
&ab8500_ac_chg_desc, &ab8500_ac_chg_desc,
&ac_psy_cfg); &ac_psy_cfg);
if (IS_ERR(di->ac_chg.psy)) { if (IS_ERR(di->ac_chg.psy)) {
dev_err(dev, "failed to register AC charger\n"); dev_err(dev, "failed to register AC charger\n");
ret = PTR_ERR(di->ac_chg.psy); return PTR_ERR(di->ac_chg.psy);
goto free_charger_wq;
} }
} }
/* Register USB charger class */ /* Register USB charger class */
if (di->usb_chg.enabled) { if (di->usb_chg.enabled) {
di->usb_chg.psy = power_supply_register(dev, di->usb_chg.psy = devm_power_supply_register(dev,
&ab8500_usb_chg_desc, &ab8500_usb_chg_desc,
&usb_psy_cfg); &usb_psy_cfg);
if (IS_ERR(di->usb_chg.psy)) { if (IS_ERR(di->usb_chg.psy)) {
dev_err(dev, "failed to register USB charger\n"); dev_err(dev, "failed to register USB charger\n");
ret = PTR_ERR(di->usb_chg.psy); return PTR_ERR(di->usb_chg.psy);
goto free_ac;
} }
} }
di->usb_phy = usb_get_phy(USB_PHY_TYPE_USB2);
if (IS_ERR_OR_NULL(di->usb_phy)) {
dev_err(dev, "failed to get usb transceiver\n");
ret = -EINVAL;
goto free_usb;
}
di->nb.notifier_call = ab8500_charger_usb_notifier_call;
ret = usb_register_notifier(di->usb_phy, &di->nb);
if (ret) {
dev_err(dev, "failed to register usb notifier\n");
goto put_usb_phy;
}
/* Identify the connected charger types during startup */ /* Identify the connected charger types during startup */
charger_status = ab8500_charger_detect_chargers(di, true); charger_status = ab8500_charger_detect_chargers(di, true);
if (charger_status & AC_PW_CONN) { if (charger_status & AC_PW_CONN) {
...@@ -3566,78 +3602,86 @@ static int ab8500_charger_probe(struct platform_device *pdev) ...@@ -3566,78 +3602,86 @@ static int ab8500_charger_probe(struct platform_device *pdev)
sysfs_notify(&di->ac_chg.psy->dev.kobj, NULL, "present"); sysfs_notify(&di->ac_chg.psy->dev.kobj, NULL, "present");
} }
if (charger_status & USB_PW_CONN) { platform_set_drvdata(pdev, di);
di->vbus_detected = true;
di->vbus_detected_start = true;
queue_work(di->charger_wq,
&di->detect_usb_type_work);
}
/* Register interrupts */
for (i = 0; i < ARRAY_SIZE(ab8500_charger_irq); i++) {
irq = platform_get_irq_byname(pdev, ab8500_charger_irq[i].name);
if (irq < 0) {
ret = irq;
goto free_irq;
}
ret = request_threaded_irq(irq, NULL, ab8500_charger_irq[i].isr, /* Create something that will match the subdrivers when we bind */
IRQF_SHARED | IRQF_NO_SUSPEND | IRQF_ONESHOT, for (i = 0; i < ARRAY_SIZE(ab8500_charger_component_drivers); i++) {
ab8500_charger_irq[i].name, di); struct device_driver *drv = &ab8500_charger_component_drivers[i]->driver;
struct device *p = NULL, *d;
if (ret != 0) { while ((d = platform_find_device_by_driver(p, drv))) {
dev_err(dev, "failed to request %s IRQ %d: %d\n" put_device(p);
, ab8500_charger_irq[i].name, irq, ret); component_match_add(dev, &match,
goto free_irq; ab8500_charger_compare_dev, d);
p = d;
} }
dev_dbg(dev, "Requested %s IRQ %d: %d\n", put_device(p);
ab8500_charger_irq[i].name, irq, ret); }
if (!match) {
dev_err(dev, "no matching components\n");
return -ENODEV;
}
if (IS_ERR(match)) {
dev_err(dev, "could not create component match\n");
return PTR_ERR(match);
} }
platform_set_drvdata(pdev, di); /* Notifier for external charger enabling */
if (!di->ac_chg.enabled)
blocking_notifier_chain_register(
&charger_notifier_list, &charger_nb);
mutex_lock(&di->charger_attached_mutex);
ch_stat = ab8500_charger_detect_chargers(di, false); di->usb_phy = usb_get_phy(USB_PHY_TYPE_USB2);
if (IS_ERR_OR_NULL(di->usb_phy)) {
if ((ch_stat & AC_PW_CONN) == AC_PW_CONN) { dev_err(dev, "failed to get usb transceiver\n");
if (is_ab8500(di->parent)) ret = -EINVAL;
queue_delayed_work(di->charger_wq, goto out_charger_notifier;
&di->ac_charger_attached_work,
HZ);
} }
if ((ch_stat & USB_PW_CONN) == USB_PW_CONN) { di->nb.notifier_call = ab8500_charger_usb_notifier_call;
if (is_ab8500(di->parent)) ret = usb_register_notifier(di->usb_phy, &di->nb);
queue_delayed_work(di->charger_wq, if (ret) {
&di->usb_charger_attached_work, dev_err(dev, "failed to register usb notifier\n");
HZ); goto put_usb_phy;
} }
mutex_unlock(&di->charger_attached_mutex);
return ret; ret = component_master_add_with_match(&pdev->dev,
&ab8500_charger_comp_ops,
match);
if (ret) {
dev_err(dev, "failed to add component master\n");
goto free_notifier;
}
free_irq: return 0;
usb_unregister_notifier(di->usb_phy, &di->nb);
/* We also have to free all successfully registered irqs */ free_notifier:
for (i = i - 1; i >= 0; i--) { usb_unregister_notifier(di->usb_phy, &di->nb);
irq = platform_get_irq_byname(pdev, ab8500_charger_irq[i].name);
free_irq(irq, di);
}
put_usb_phy: put_usb_phy:
usb_put_phy(di->usb_phy); usb_put_phy(di->usb_phy);
free_usb: out_charger_notifier:
if (di->usb_chg.enabled) if (!di->ac_chg.enabled)
power_supply_unregister(di->usb_chg.psy); blocking_notifier_chain_unregister(
free_ac: &charger_notifier_list, &charger_nb);
if (di->ac_chg.enabled)
power_supply_unregister(di->ac_chg.psy);
free_charger_wq:
destroy_workqueue(di->charger_wq);
return ret; return ret;
} }
static int ab8500_charger_remove(struct platform_device *pdev)
{
struct ab8500_charger *di = platform_get_drvdata(pdev);
component_master_del(&pdev->dev, &ab8500_charger_comp_ops);
usb_unregister_notifier(di->usb_phy, &di->nb);
usb_put_phy(di->usb_phy);
if (!di->ac_chg.enabled)
blocking_notifier_chain_unregister(
&charger_notifier_list, &charger_nb);
return 0;
}
static SIMPLE_DEV_PM_OPS(ab8500_charger_pm_ops, ab8500_charger_suspend, ab8500_charger_resume); static SIMPLE_DEV_PM_OPS(ab8500_charger_pm_ops, ab8500_charger_suspend, ab8500_charger_resume);
static const struct of_device_id ab8500_charger_match[] = { static const struct of_device_id ab8500_charger_match[] = {
...@@ -3657,15 +3701,24 @@ static struct platform_driver ab8500_charger_driver = { ...@@ -3657,15 +3701,24 @@ static struct platform_driver ab8500_charger_driver = {
static int __init ab8500_charger_init(void) static int __init ab8500_charger_init(void)
{ {
int ret;
ret = platform_register_drivers(ab8500_charger_component_drivers,
ARRAY_SIZE(ab8500_charger_component_drivers));
if (ret)
return ret;
return platform_driver_register(&ab8500_charger_driver); return platform_driver_register(&ab8500_charger_driver);
} }
static void __exit ab8500_charger_exit(void) static void __exit ab8500_charger_exit(void)
{ {
platform_unregister_drivers(ab8500_charger_component_drivers,
ARRAY_SIZE(ab8500_charger_component_drivers));
platform_driver_unregister(&ab8500_charger_driver); platform_driver_unregister(&ab8500_charger_driver);
} }
subsys_initcall_sync(ab8500_charger_init); module_init(ab8500_charger_init);
module_exit(ab8500_charger_exit); module_exit(ab8500_charger_exit);
MODULE_LICENSE("GPL v2"); MODULE_LICENSE("GPL v2");
......
...@@ -17,6 +17,7 @@ ...@@ -17,6 +17,7 @@
#include <linux/init.h> #include <linux/init.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/component.h>
#include <linux/device.h> #include <linux/device.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
...@@ -2980,27 +2981,6 @@ static int __maybe_unused ab8500_fg_suspend(struct device *dev) ...@@ -2980,27 +2981,6 @@ static int __maybe_unused ab8500_fg_suspend(struct device *dev)
return 0; return 0;
} }
static int ab8500_fg_remove(struct platform_device *pdev)
{
int ret = 0;
struct ab8500_fg *di = platform_get_drvdata(pdev);
list_del(&di->node);
/* Disable coulomb counter */
ret = ab8500_fg_coulomb_counter(di, false);
if (ret)
dev_err(di->dev, "failed to disable coulomb counter\n");
destroy_workqueue(di->fg_wq);
ab8500_fg_sysfs_exit(di);
flush_scheduled_work();
ab8500_fg_sysfs_psy_remove_attrs(di);
power_supply_unregister(di->fg_psy);
return ret;
}
/* ab8500 fg driver interrupts and their respective isr */ /* ab8500 fg driver interrupts and their respective isr */
static struct ab8500_fg_interrupts ab8500_fg_irq[] = { static struct ab8500_fg_interrupts ab8500_fg_irq[] = {
{"NCONV_ACCU", ab8500_fg_cc_convend_handler}, {"NCONV_ACCU", ab8500_fg_cc_convend_handler},
...@@ -3024,11 +3004,50 @@ static const struct power_supply_desc ab8500_fg_desc = { ...@@ -3024,11 +3004,50 @@ static const struct power_supply_desc ab8500_fg_desc = {
.external_power_changed = ab8500_fg_external_power_changed, .external_power_changed = ab8500_fg_external_power_changed,
}; };
static int ab8500_fg_bind(struct device *dev, struct device *master,
void *data)
{
struct ab8500_fg *di = dev_get_drvdata(dev);
/* Create a work queue for running the FG algorithm */
di->fg_wq = alloc_ordered_workqueue("ab8500_fg_wq", WQ_MEM_RECLAIM);
if (di->fg_wq == NULL) {
dev_err(dev, "failed to create work queue\n");
return -ENOMEM;
}
/* Start the coulomb counter */
ab8500_fg_coulomb_counter(di, true);
/* Run the FG algorithm */
queue_delayed_work(di->fg_wq, &di->fg_periodic_work, 0);
return 0;
}
static void ab8500_fg_unbind(struct device *dev, struct device *master,
void *data)
{
struct ab8500_fg *di = dev_get_drvdata(dev);
int ret;
/* Disable coulomb counter */
ret = ab8500_fg_coulomb_counter(di, false);
if (ret)
dev_err(dev, "failed to disable coulomb counter\n");
destroy_workqueue(di->fg_wq);
flush_scheduled_work();
}
static const struct component_ops ab8500_fg_component_ops = {
.bind = ab8500_fg_bind,
.unbind = ab8500_fg_unbind,
};
static int ab8500_fg_probe(struct platform_device *pdev) static int ab8500_fg_probe(struct platform_device *pdev)
{ {
struct device_node *np = pdev->dev.of_node;
struct power_supply_config psy_cfg = {};
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct power_supply_config psy_cfg = {};
struct ab8500_fg *di; struct ab8500_fg *di;
int i, irq; int i, irq;
int ret = 0; int ret = 0;
...@@ -3074,13 +3093,6 @@ static int ab8500_fg_probe(struct platform_device *pdev) ...@@ -3074,13 +3093,6 @@ static int ab8500_fg_probe(struct platform_device *pdev)
ab8500_fg_charge_state_to(di, AB8500_FG_CHARGE_INIT); ab8500_fg_charge_state_to(di, AB8500_FG_CHARGE_INIT);
ab8500_fg_discharge_state_to(di, AB8500_FG_DISCHARGE_INIT); ab8500_fg_discharge_state_to(di, AB8500_FG_DISCHARGE_INIT);
/* Create a work queue for running the FG algorithm */
di->fg_wq = alloc_ordered_workqueue("ab8500_fg_wq", WQ_MEM_RECLAIM);
if (di->fg_wq == NULL) {
dev_err(dev, "failed to create work queue\n");
return -ENOMEM;
}
/* Init work for running the fg algorithm instantly */ /* Init work for running the fg algorithm instantly */
INIT_WORK(&di->fg_work, ab8500_fg_instant_work); INIT_WORK(&di->fg_work, ab8500_fg_instant_work);
...@@ -3113,7 +3125,7 @@ static int ab8500_fg_probe(struct platform_device *pdev) ...@@ -3113,7 +3125,7 @@ static int ab8500_fg_probe(struct platform_device *pdev)
ret = ab8500_fg_init_hw_registers(di); ret = ab8500_fg_init_hw_registers(di);
if (ret) { if (ret) {
dev_err(dev, "failed to initialize registers\n"); dev_err(dev, "failed to initialize registers\n");
goto free_inst_curr_wq; return ret;
} }
/* Consider battery unknown until we're informed otherwise */ /* Consider battery unknown until we're informed otherwise */
...@@ -3121,15 +3133,13 @@ static int ab8500_fg_probe(struct platform_device *pdev) ...@@ -3121,15 +3133,13 @@ static int ab8500_fg_probe(struct platform_device *pdev)
di->flags.batt_id_received = false; di->flags.batt_id_received = false;
/* Register FG power supply class */ /* Register FG power supply class */
di->fg_psy = power_supply_register(dev, &ab8500_fg_desc, &psy_cfg); di->fg_psy = devm_power_supply_register(dev, &ab8500_fg_desc, &psy_cfg);
if (IS_ERR(di->fg_psy)) { if (IS_ERR(di->fg_psy)) {
dev_err(dev, "failed to register FG psy\n"); dev_err(dev, "failed to register FG psy\n");
ret = PTR_ERR(di->fg_psy); return PTR_ERR(di->fg_psy);
goto free_inst_curr_wq;
} }
di->fg_samples = SEC_TO_SAMPLE(di->bm->fg_params->init_timer); di->fg_samples = SEC_TO_SAMPLE(di->bm->fg_params->init_timer);
ab8500_fg_coulomb_counter(di, true);
/* /*
* Initialize completion used to notify completion and start * Initialize completion used to notify completion and start
...@@ -3141,19 +3151,18 @@ static int ab8500_fg_probe(struct platform_device *pdev) ...@@ -3141,19 +3151,18 @@ static int ab8500_fg_probe(struct platform_device *pdev)
/* Register primary interrupt handlers */ /* Register primary interrupt handlers */
for (i = 0; i < ARRAY_SIZE(ab8500_fg_irq); i++) { for (i = 0; i < ARRAY_SIZE(ab8500_fg_irq); i++) {
irq = platform_get_irq_byname(pdev, ab8500_fg_irq[i].name); irq = platform_get_irq_byname(pdev, ab8500_fg_irq[i].name);
if (irq < 0) { if (irq < 0)
ret = irq; return irq;
goto free_irq;
}
ret = request_threaded_irq(irq, NULL, ab8500_fg_irq[i].isr, ret = devm_request_threaded_irq(dev, irq, NULL,
ab8500_fg_irq[i].isr,
IRQF_SHARED | IRQF_NO_SUSPEND | IRQF_ONESHOT, IRQF_SHARED | IRQF_NO_SUSPEND | IRQF_ONESHOT,
ab8500_fg_irq[i].name, di); ab8500_fg_irq[i].name, di);
if (ret != 0) { if (ret != 0) {
dev_err(dev, "failed to request %s IRQ %d: %d\n", dev_err(dev, "failed to request %s IRQ %d: %d\n",
ab8500_fg_irq[i].name, irq, ret); ab8500_fg_irq[i].name, irq, ret);
goto free_irq; return ret;
} }
dev_dbg(dev, "Requested %s IRQ %d: %d\n", dev_dbg(dev, "Requested %s IRQ %d: %d\n",
ab8500_fg_irq[i].name, irq, ret); ab8500_fg_irq[i].name, irq, ret);
...@@ -3168,14 +3177,14 @@ static int ab8500_fg_probe(struct platform_device *pdev) ...@@ -3168,14 +3177,14 @@ static int ab8500_fg_probe(struct platform_device *pdev)
ret = ab8500_fg_sysfs_init(di); ret = ab8500_fg_sysfs_init(di);
if (ret) { if (ret) {
dev_err(dev, "failed to create sysfs entry\n"); dev_err(dev, "failed to create sysfs entry\n");
goto free_irq; return ret;
} }
ret = ab8500_fg_sysfs_psy_create_attrs(di); ret = ab8500_fg_sysfs_psy_create_attrs(di);
if (ret) { if (ret) {
dev_err(dev, "failed to create FG psy\n"); dev_err(dev, "failed to create FG psy\n");
ab8500_fg_sysfs_exit(di); ab8500_fg_sysfs_exit(di);
goto free_irq; return ret;
} }
/* Calibrate the fg first time */ /* Calibrate the fg first time */
...@@ -3185,24 +3194,21 @@ static int ab8500_fg_probe(struct platform_device *pdev) ...@@ -3185,24 +3194,21 @@ static int ab8500_fg_probe(struct platform_device *pdev)
/* Use room temp as default value until we get an update from driver. */ /* Use room temp as default value until we get an update from driver. */
di->bat_temp = 210; di->bat_temp = 210;
/* Run the FG algorithm */
queue_delayed_work(di->fg_wq, &di->fg_periodic_work, 0);
list_add_tail(&di->node, &ab8500_fg_list); list_add_tail(&di->node, &ab8500_fg_list);
return ret; return component_add(dev, &ab8500_fg_component_ops);
}
free_irq: static int ab8500_fg_remove(struct platform_device *pdev)
/* We also have to free all registered irqs */ {
while (--i >= 0) { int ret = 0;
/* Last assignment of i from primary interrupt handlers */ struct ab8500_fg *di = platform_get_drvdata(pdev);
irq = platform_get_irq_byname(pdev, ab8500_fg_irq[i].name);
free_irq(irq, di); component_del(&pdev->dev, &ab8500_fg_component_ops);
} list_del(&di->node);
ab8500_fg_sysfs_exit(di);
ab8500_fg_sysfs_psy_remove_attrs(di);
power_supply_unregister(di->fg_psy);
free_inst_curr_wq:
destroy_workqueue(di->fg_wq);
return ret; return ret;
} }
...@@ -3213,7 +3219,7 @@ static const struct of_device_id ab8500_fg_match[] = { ...@@ -3213,7 +3219,7 @@ static const struct of_device_id ab8500_fg_match[] = {
{ }, { },
}; };
static struct platform_driver ab8500_fg_driver = { struct platform_driver ab8500_fg_driver = {
.probe = ab8500_fg_probe, .probe = ab8500_fg_probe,
.remove = ab8500_fg_remove, .remove = ab8500_fg_remove,
.driver = { .driver = {
...@@ -3222,20 +3228,6 @@ static struct platform_driver ab8500_fg_driver = { ...@@ -3222,20 +3228,6 @@ static struct platform_driver ab8500_fg_driver = {
.pm = &ab8500_fg_pm_ops, .pm = &ab8500_fg_pm_ops,
}, },
}; };
static int __init ab8500_fg_init(void)
{
return platform_driver_register(&ab8500_fg_driver);
}
static void __exit ab8500_fg_exit(void)
{
platform_driver_unregister(&ab8500_fg_driver);
}
subsys_initcall_sync(ab8500_fg_init);
module_exit(ab8500_fg_exit);
MODULE_LICENSE("GPL v2"); MODULE_LICENSE("GPL v2");
MODULE_AUTHOR("Johan Palsson, Karl Komierowski"); MODULE_AUTHOR("Johan Palsson, Karl Komierowski");
MODULE_ALIAS("platform:ab8500-fg"); MODULE_ALIAS("platform:ab8500-fg");
......
...@@ -15,6 +15,7 @@ ...@@ -15,6 +15,7 @@
#include <linux/init.h> #include <linux/init.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/device.h> #include <linux/device.h>
#include <linux/component.h>
#include <linux/hrtimer.h> #include <linux/hrtimer.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/delay.h> #include <linux/delay.h>
...@@ -1943,13 +1944,44 @@ static int __maybe_unused abx500_chargalg_suspend(struct device *dev) ...@@ -1943,13 +1944,44 @@ static int __maybe_unused abx500_chargalg_suspend(struct device *dev)
return 0; return 0;
} }
static int abx500_chargalg_remove(struct platform_device *pdev) static char *supply_interface[] = {
"ab8500_fg",
};
static const struct power_supply_desc abx500_chargalg_desc = {
.name = "abx500_chargalg",
.type = POWER_SUPPLY_TYPE_BATTERY,
.properties = abx500_chargalg_props,
.num_properties = ARRAY_SIZE(abx500_chargalg_props),
.get_property = abx500_chargalg_get_property,
.external_power_changed = abx500_chargalg_external_power_changed,
};
static int abx500_chargalg_bind(struct device *dev, struct device *master,
void *data)
{ {
struct abx500_chargalg *di = platform_get_drvdata(pdev); struct abx500_chargalg *di = dev_get_drvdata(dev);
/* sysfs interface to enable/disbale charging from user space */ /* Create a work queue for the chargalg */
abx500_chargalg_sysfs_exit(di); di->chargalg_wq = alloc_ordered_workqueue("abx500_chargalg_wq",
WQ_MEM_RECLAIM);
if (di->chargalg_wq == NULL) {
dev_err(di->dev, "failed to create work queue\n");
return -ENOMEM;
}
/* Run the charging algorithm */
queue_delayed_work(di->chargalg_wq, &di->chargalg_periodic_work, 0);
return 0;
}
static void abx500_chargalg_unbind(struct device *dev, struct device *master,
void *data)
{
struct abx500_chargalg *di = dev_get_drvdata(dev);
/* Stop all timers and work */
hrtimer_cancel(&di->safety_timer); hrtimer_cancel(&di->safety_timer);
hrtimer_cancel(&di->maintenance_timer); hrtimer_cancel(&di->maintenance_timer);
...@@ -1959,48 +1991,36 @@ static int abx500_chargalg_remove(struct platform_device *pdev) ...@@ -1959,48 +1991,36 @@ static int abx500_chargalg_remove(struct platform_device *pdev)
/* Delete the work queue */ /* Delete the work queue */
destroy_workqueue(di->chargalg_wq); destroy_workqueue(di->chargalg_wq);
flush_scheduled_work();
power_supply_unregister(di->chargalg_psy);
return 0;
} }
static char *supply_interface[] = { static const struct component_ops abx500_chargalg_component_ops = {
"ab8500_fg", .bind = abx500_chargalg_bind,
}; .unbind = abx500_chargalg_unbind,
static const struct power_supply_desc abx500_chargalg_desc = {
.name = "abx500_chargalg",
.type = POWER_SUPPLY_TYPE_BATTERY,
.properties = abx500_chargalg_props,
.num_properties = ARRAY_SIZE(abx500_chargalg_props),
.get_property = abx500_chargalg_get_property,
.external_power_changed = abx500_chargalg_external_power_changed,
}; };
static int abx500_chargalg_probe(struct platform_device *pdev) static int abx500_chargalg_probe(struct platform_device *pdev)
{ {
struct device_node *np = pdev->dev.of_node; struct device *dev = &pdev->dev;
struct device_node *np = dev->of_node;
struct power_supply_config psy_cfg = {}; struct power_supply_config psy_cfg = {};
struct abx500_chargalg *di; struct abx500_chargalg *di;
int ret = 0; int ret = 0;
di = devm_kzalloc(&pdev->dev, sizeof(*di), GFP_KERNEL); di = devm_kzalloc(dev, sizeof(*di), GFP_KERNEL);
if (!di) { if (!di)
dev_err(&pdev->dev, "%s no mem for ab8500_chargalg\n", __func__);
return -ENOMEM; return -ENOMEM;
}
di->bm = &ab8500_bm_data; di->bm = &ab8500_bm_data;
ret = ab8500_bm_of_probe(&pdev->dev, np, di->bm); ret = ab8500_bm_of_probe(dev, np, di->bm);
if (ret) { if (ret) {
dev_err(&pdev->dev, "failed to get battery information\n"); dev_err(dev, "failed to get battery information\n");
return ret; return ret;
} }
/* get device struct and parent */ /* get device struct and parent */
di->dev = &pdev->dev; di->dev = dev;
di->parent = dev_get_drvdata(pdev->dev.parent); di->parent = dev_get_drvdata(pdev->dev.parent);
psy_cfg.supplied_to = supply_interface; psy_cfg.supplied_to = supply_interface;
...@@ -2016,14 +2036,6 @@ static int abx500_chargalg_probe(struct platform_device *pdev) ...@@ -2016,14 +2036,6 @@ static int abx500_chargalg_probe(struct platform_device *pdev)
di->maintenance_timer.function = di->maintenance_timer.function =
abx500_chargalg_maintenance_timer_expired; abx500_chargalg_maintenance_timer_expired;
/* Create a work queue for the chargalg */
di->chargalg_wq = alloc_ordered_workqueue("abx500_chargalg_wq",
WQ_MEM_RECLAIM);
if (di->chargalg_wq == NULL) {
dev_err(di->dev, "failed to create work queue\n");
return -ENOMEM;
}
/* Init work for chargalg */ /* Init work for chargalg */
INIT_DEFERRABLE_WORK(&di->chargalg_periodic_work, INIT_DEFERRABLE_WORK(&di->chargalg_periodic_work,
abx500_chargalg_periodic_work); abx500_chargalg_periodic_work);
...@@ -2037,12 +2049,12 @@ static int abx500_chargalg_probe(struct platform_device *pdev) ...@@ -2037,12 +2049,12 @@ static int abx500_chargalg_probe(struct platform_device *pdev)
di->chg_info.prev_conn_chg = -1; di->chg_info.prev_conn_chg = -1;
/* Register chargalg power supply class */ /* Register chargalg power supply class */
di->chargalg_psy = power_supply_register(di->dev, &abx500_chargalg_desc, di->chargalg_psy = devm_power_supply_register(di->dev,
&abx500_chargalg_desc,
&psy_cfg); &psy_cfg);
if (IS_ERR(di->chargalg_psy)) { if (IS_ERR(di->chargalg_psy)) {
dev_err(di->dev, "failed to register chargalg psy\n"); dev_err(di->dev, "failed to register chargalg psy\n");
ret = PTR_ERR(di->chargalg_psy); return PTR_ERR(di->chargalg_psy);
goto free_chargalg_wq;
} }
platform_set_drvdata(pdev, di); platform_set_drvdata(pdev, di);
...@@ -2051,21 +2063,24 @@ static int abx500_chargalg_probe(struct platform_device *pdev) ...@@ -2051,21 +2063,24 @@ static int abx500_chargalg_probe(struct platform_device *pdev)
ret = abx500_chargalg_sysfs_init(di); ret = abx500_chargalg_sysfs_init(di);
if (ret) { if (ret) {
dev_err(di->dev, "failed to create sysfs entry\n"); dev_err(di->dev, "failed to create sysfs entry\n");
goto free_psy; return ret;
} }
di->curr_status.curr_step = CHARGALG_CURR_STEP_HIGH; di->curr_status.curr_step = CHARGALG_CURR_STEP_HIGH;
/* Run the charging algorithm */
queue_delayed_work(di->chargalg_wq, &di->chargalg_periodic_work, 0);
dev_info(di->dev, "probe success\n"); dev_info(di->dev, "probe success\n");
return ret; return component_add(dev, &abx500_chargalg_component_ops);
}
free_psy: static int abx500_chargalg_remove(struct platform_device *pdev)
power_supply_unregister(di->chargalg_psy); {
free_chargalg_wq: struct abx500_chargalg *di = platform_get_drvdata(pdev);
destroy_workqueue(di->chargalg_wq);
return ret; component_del(&pdev->dev, &abx500_chargalg_component_ops);
/* sysfs interface to enable/disable charging from user space */
abx500_chargalg_sysfs_exit(di);
return 0;
} }
static SIMPLE_DEV_PM_OPS(abx500_chargalg_pm_ops, abx500_chargalg_suspend, abx500_chargalg_resume); static SIMPLE_DEV_PM_OPS(abx500_chargalg_pm_ops, abx500_chargalg_suspend, abx500_chargalg_resume);
...@@ -2075,7 +2090,7 @@ static const struct of_device_id ab8500_chargalg_match[] = { ...@@ -2075,7 +2090,7 @@ static const struct of_device_id ab8500_chargalg_match[] = {
{ }, { },
}; };
static struct platform_driver abx500_chargalg_driver = { struct platform_driver abx500_chargalg_driver = {
.probe = abx500_chargalg_probe, .probe = abx500_chargalg_probe,
.remove = abx500_chargalg_remove, .remove = abx500_chargalg_remove,
.driver = { .driver = {
...@@ -2084,9 +2099,6 @@ static struct platform_driver abx500_chargalg_driver = { ...@@ -2084,9 +2099,6 @@ static struct platform_driver abx500_chargalg_driver = {
.pm = &abx500_chargalg_pm_ops, .pm = &abx500_chargalg_pm_ops,
}, },
}; };
module_platform_driver(abx500_chargalg_driver);
MODULE_LICENSE("GPL v2"); MODULE_LICENSE("GPL v2");
MODULE_AUTHOR("Johan Palsson, Karl Komierowski"); MODULE_AUTHOR("Johan Palsson, Karl Komierowski");
MODULE_ALIAS("platform:abx500-chargalg"); MODULE_ALIAS("platform:abx500-chargalg");
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment