| /* |
| * drivers/base/power/common.c - Common device power management code. |
| * |
| * Copyright (C) 2011 Rafael J. Wysocki <rjw@sisk.pl>, Renesas Electronics Corp. |
| * |
| * This file is released under the GPLv2. |
| */ |
| |
| #include <linux/init.h> |
| #include <linux/kernel.h> |
| #include <linux/device.h> |
| #include <linux/export.h> |
| #include <linux/slab.h> |
| #include <linux/pm_clock.h> |
| |
| /** |
| * dev_pm_get_subsys_data - Create or refcount power.subsys_data for device. |
| * @dev: Device to handle. |
| * |
| * If power.subsys_data is NULL, point it to a new object, otherwise increment |
| * its reference counter. Return 1 if a new object has been created, otherwise |
| * return 0 or error code. |
| */ |
| int dev_pm_get_subsys_data(struct device *dev) |
| { |
| struct pm_subsys_data *psd; |
| int ret = 0; |
| |
| psd = kzalloc(sizeof(*psd), GFP_KERNEL); |
| if (!psd) |
| return -ENOMEM; |
| |
| spin_lock_irq(&dev->power.lock); |
| |
| if (dev->power.subsys_data) { |
| dev->power.subsys_data->refcount++; |
| } else { |
| spin_lock_init(&psd->lock); |
| psd->refcount = 1; |
| dev->power.subsys_data = psd; |
| pm_clk_init(dev); |
| psd = NULL; |
| ret = 1; |
| } |
| |
| spin_unlock_irq(&dev->power.lock); |
| |
| /* kfree() verifies that its argument is nonzero. */ |
| kfree(psd); |
| |
| return ret; |
| } |
| EXPORT_SYMBOL_GPL(dev_pm_get_subsys_data); |
| |
| /** |
| * dev_pm_put_subsys_data - Drop reference to power.subsys_data. |
| * @dev: Device to handle. |
| * |
| * If the reference counter of power.subsys_data is zero after dropping the |
| * reference, power.subsys_data is removed. Return 1 if that happens or 0 |
| * otherwise. |
| */ |
| int dev_pm_put_subsys_data(struct device *dev) |
| { |
| struct pm_subsys_data *psd; |
| int ret = 0; |
| |
| spin_lock_irq(&dev->power.lock); |
| |
| psd = dev_to_psd(dev); |
| if (!psd) { |
| ret = -EINVAL; |
| goto out; |
| } |
| |
| if (--psd->refcount == 0) { |
| dev->power.subsys_data = NULL; |
| kfree(psd); |
| ret = 1; |
| } |
| |
| out: |
| spin_unlock_irq(&dev->power.lock); |
| |
| return ret; |
| } |
| EXPORT_SYMBOL_GPL(dev_pm_put_subsys_data); |