aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorKrzysztof Kozlowski <k.kozlowski@samsung.com>2015-02-20 08:32:24 -0500
committerSebastian Reichel <sre@kernel.org>2015-02-25 16:18:18 -0500
commita2c1d531854c4319610f1d83351213b47a633969 (patch)
tree70d4f9b5bd25c39dd11ccb80af24b379baaa5a6c
parentf852ec461e24504690445e7d281cbe806df5ccef (diff)
power_supply: ipaq_micro_battery: Check return values in probe
The return values of create_singlethread_workqueue() and power_supply_register() calls were not checked and even on error probe() function returned 0. 1. If allocation of workqueue failed (returning NULL) then further accesses could lead to NULL pointer dereference. The queue_delayed_work() expects workqueue to be non-NULL. 2. If registration of power supply failed then during unbind the driver tried to unregister power supply which was not actually registered. This could lead to memory corruption because power_supply_unregister() unconditionally cleans up given power supply. Signed-off-by: Krzysztof Kozlowski <k.kozlowski@samsung.com> Fixes: 00a588f9d27f ("power: add driver for battery reading on iPaq h3xxx") Cc: <stable@vger.kernel.org> Signed-off-by: Sebastian Reichel <sre@kernel.org>
-rw-r--r--drivers/power/ipaq_micro_battery.c21
1 files changed, 19 insertions, 2 deletions
diff --git a/drivers/power/ipaq_micro_battery.c b/drivers/power/ipaq_micro_battery.c
index 698cf1636bb8..96b15e003f3f 100644
--- a/drivers/power/ipaq_micro_battery.c
+++ b/drivers/power/ipaq_micro_battery.c
@@ -226,6 +226,7 @@ static struct power_supply micro_ac_power = {
226static int micro_batt_probe(struct platform_device *pdev) 226static int micro_batt_probe(struct platform_device *pdev)
227{ 227{
228 struct micro_battery *mb; 228 struct micro_battery *mb;
229 int ret;
229 230
230 mb = devm_kzalloc(&pdev->dev, sizeof(*mb), GFP_KERNEL); 231 mb = devm_kzalloc(&pdev->dev, sizeof(*mb), GFP_KERNEL);
231 if (!mb) 232 if (!mb)
@@ -233,14 +234,30 @@ static int micro_batt_probe(struct platform_device *pdev)
233 234
234 mb->micro = dev_get_drvdata(pdev->dev.parent); 235 mb->micro = dev_get_drvdata(pdev->dev.parent);
235 mb->wq = create_singlethread_workqueue("ipaq-battery-wq"); 236 mb->wq = create_singlethread_workqueue("ipaq-battery-wq");
237 if (!mb->wq)
238 return -ENOMEM;
239
236 INIT_DELAYED_WORK(&mb->update, micro_battery_work); 240 INIT_DELAYED_WORK(&mb->update, micro_battery_work);
237 platform_set_drvdata(pdev, mb); 241 platform_set_drvdata(pdev, mb);
238 queue_delayed_work(mb->wq, &mb->update, 1); 242 queue_delayed_work(mb->wq, &mb->update, 1);
239 power_supply_register(&pdev->dev, &micro_batt_power); 243
240 power_supply_register(&pdev->dev, &micro_ac_power); 244 ret = power_supply_register(&pdev->dev, &micro_batt_power);
245 if (ret < 0)
246 goto batt_err;
247
248 ret = power_supply_register(&pdev->dev, &micro_ac_power);
249 if (ret < 0)
250 goto ac_err;
241 251
242 dev_info(&pdev->dev, "iPAQ micro battery driver\n"); 252 dev_info(&pdev->dev, "iPAQ micro battery driver\n");
243 return 0; 253 return 0;
254
255ac_err:
256 power_supply_unregister(&micro_ac_power);
257batt_err:
258 cancel_delayed_work_sync(&mb->update);
259 destroy_workqueue(mb->wq);
260 return ret;
244} 261}
245 262
246static int micro_batt_remove(struct platform_device *pdev) 263static int micro_batt_remove(struct platform_device *pdev)