aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/power
diff options
context:
space:
mode:
authorKrzysztof Kozlowski <k.kozlowski@samsung.com>2015-03-12 03:44:13 -0400
committerSebastian Reichel <sre@kernel.org>2015-03-13 18:15:52 -0400
commit03e81acce5568c7105dc5bef6984c8b0edfe8d45 (patch)
tree5a23f5c2245e9d4f170e644e2050638c225b51a1 /drivers/power
parent1a352462b5377ac68f5955d674b3460c7bac52a3 (diff)
power_supply: Increment power supply use counter when obtaining references
Increment the power_supply.use_cnt usage counter on: - power_supply_get_by_phandle() - power_supply_get_by_name() and decrement it on power_supply_put() call. This helps tracking of valid usage of power supply instance by consumers. The usage counter itself also allows safe calling of power_supply_get_property-like functions even when driver unregisters this power supply. Signed-off-by: Krzysztof Kozlowski <k.kozlowski@samsung.com> Reviewed-by: Bartlomiej Zolnierkiewicz <b.zolnierkie@samsung.com> Signed-off-by: Sebastian Reichel <sre@kernel.org>
Diffstat (limited to 'drivers/power')
-rw-r--r--drivers/power/power_supply_core.c17
1 files changed, 15 insertions, 2 deletions
diff --git a/drivers/power/power_supply_core.c b/drivers/power/power_supply_core.c
index b4ec14683b7f..2ed4a4a6b3c5 100644
--- a/drivers/power/power_supply_core.c
+++ b/drivers/power/power_supply_core.c
@@ -349,10 +349,16 @@ static int power_supply_match_device_by_name(struct device *dev, const void *dat
349 */ 349 */
350struct power_supply *power_supply_get_by_name(const char *name) 350struct power_supply *power_supply_get_by_name(const char *name)
351{ 351{
352 struct power_supply *psy = NULL;
352 struct device *dev = class_find_device(power_supply_class, NULL, name, 353 struct device *dev = class_find_device(power_supply_class, NULL, name,
353 power_supply_match_device_by_name); 354 power_supply_match_device_by_name);
354 355
355 return dev ? dev_get_drvdata(dev) : NULL; 356 if (dev) {
357 psy = dev_get_drvdata(dev);
358 atomic_inc(&psy->use_cnt);
359 }
360
361 return psy;
356} 362}
357EXPORT_SYMBOL_GPL(power_supply_get_by_name); 363EXPORT_SYMBOL_GPL(power_supply_get_by_name);
358 364
@@ -367,6 +373,7 @@ void power_supply_put(struct power_supply *psy)
367{ 373{
368 might_sleep(); 374 might_sleep();
369 375
376 atomic_dec(&psy->use_cnt);
370 put_device(&psy->dev); 377 put_device(&psy->dev);
371} 378}
372EXPORT_SYMBOL_GPL(power_supply_put); 379EXPORT_SYMBOL_GPL(power_supply_put);
@@ -393,6 +400,7 @@ struct power_supply *power_supply_get_by_phandle(struct device_node *np,
393 const char *property) 400 const char *property)
394{ 401{
395 struct device_node *power_supply_np; 402 struct device_node *power_supply_np;
403 struct power_supply *psy = NULL;
396 struct device *dev; 404 struct device *dev;
397 405
398 power_supply_np = of_parse_phandle(np, property, 0); 406 power_supply_np = of_parse_phandle(np, property, 0);
@@ -404,7 +412,12 @@ struct power_supply *power_supply_get_by_phandle(struct device_node *np,
404 412
405 of_node_put(power_supply_np); 413 of_node_put(power_supply_np);
406 414
407 return dev ? dev_get_drvdata(dev) : NULL; 415 if (dev) {
416 psy = dev_get_drvdata(dev);
417 atomic_inc(&psy->use_cnt);
418 }
419
420 return psy;
408} 421}
409EXPORT_SYMBOL_GPL(power_supply_get_by_phandle); 422EXPORT_SYMBOL_GPL(power_supply_get_by_phandle);
410#endif /* CONFIG_OF */ 423#endif /* CONFIG_OF */