diff options
author | Krzysztof Kozlowski <k.kozlowski@samsung.com> | 2015-03-12 03:44:13 -0400 |
---|---|---|
committer | Sebastian Reichel <sre@kernel.org> | 2015-03-13 18:15:52 -0400 |
commit | 03e81acce5568c7105dc5bef6984c8b0edfe8d45 (patch) | |
tree | 5a23f5c2245e9d4f170e644e2050638c225b51a1 /drivers/power | |
parent | 1a352462b5377ac68f5955d674b3460c7bac52a3 (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.c | 17 |
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 | */ |
350 | struct power_supply *power_supply_get_by_name(const char *name) | 350 | struct 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 | } |
357 | EXPORT_SYMBOL_GPL(power_supply_get_by_name); | 363 | EXPORT_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 | } |
372 | EXPORT_SYMBOL_GPL(power_supply_put); | 379 | EXPORT_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 | } |
409 | EXPORT_SYMBOL_GPL(power_supply_get_by_phandle); | 422 | EXPORT_SYMBOL_GPL(power_supply_get_by_phandle); |
410 | #endif /* CONFIG_OF */ | 423 | #endif /* CONFIG_OF */ |