Commit b1c0f99b authored by Rafael J. Wysocki's avatar Rafael J. Wysocki

ACPI / PM: Expose current status of ACPI power resources

Since ACPI power resources are going to be used more extensively on
new hardware platforms, it becomes necessary for user space (powertop
in particular) to observe some properties of those resources for
diagnostics purposes.

For this reason, expose the current status of each ACPI power
resource to user space via sysfs by adding a new resource_in_use
attribute to the sysfs directory representing the given power
resource.
Signed-off-by: default avatarRafael J. Wysocki <rafael.j.wysocki@intel.com>
Acked-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 836aedb1
What: /sys/devices/.../resource_in_use
Date: January 2013
Contact: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Description:
The /sys/devices/.../resource_in_use attribute is only present
for device objects representing ACPI power resources.
If present, it contains a number (0 or 1) representing the
current status of the given power resource (0 means that the
resource is not in use and therefore it has been turned off).
This attribute is read-only.
...@@ -87,6 +87,12 @@ static DEFINE_MUTEX(power_resource_list_lock); ...@@ -87,6 +87,12 @@ static DEFINE_MUTEX(power_resource_list_lock);
Power Resource Management Power Resource Management
-------------------------------------------------------------------------- */ -------------------------------------------------------------------------- */
static inline
struct acpi_power_resource *to_power_resource(struct acpi_device *device)
{
return container_of(device, struct acpi_power_resource, device);
}
static struct acpi_power_resource *acpi_power_get_context(acpi_handle handle) static struct acpi_power_resource *acpi_power_get_context(acpi_handle handle)
{ {
struct acpi_device *device; struct acpi_device *device;
...@@ -94,7 +100,7 @@ static struct acpi_power_resource *acpi_power_get_context(acpi_handle handle) ...@@ -94,7 +100,7 @@ static struct acpi_power_resource *acpi_power_get_context(acpi_handle handle)
if (acpi_bus_get_device(handle, &device)) if (acpi_bus_get_device(handle, &device))
return NULL; return NULL;
return container_of(device, struct acpi_power_resource, device); return to_power_resource(device);
} }
static int acpi_power_resources_list_add(acpi_handle handle, static int acpi_power_resources_list_add(acpi_handle handle,
...@@ -678,6 +684,21 @@ static void acpi_release_power_resource(struct device *dev) ...@@ -678,6 +684,21 @@ static void acpi_release_power_resource(struct device *dev)
kfree(resource); kfree(resource);
} }
static ssize_t acpi_power_in_use_show(struct device *dev,
struct device_attribute *attr,
char *buf) {
struct acpi_power_resource *resource;
resource = to_power_resource(to_acpi_device(dev));
return sprintf(buf, "%u\n", !!resource->ref_count);
}
static DEVICE_ATTR(resource_in_use, 0444, acpi_power_in_use_show, NULL);
static void acpi_power_sysfs_remove(struct acpi_device *device)
{
device_remove_file(&device->dev, &dev_attr_resource_in_use);
}
int acpi_add_power_resource(acpi_handle handle) int acpi_add_power_resource(acpi_handle handle)
{ {
struct acpi_power_resource *resource; struct acpi_power_resource *resource;
...@@ -725,6 +746,9 @@ int acpi_add_power_resource(acpi_handle handle) ...@@ -725,6 +746,9 @@ int acpi_add_power_resource(acpi_handle handle)
if (result) if (result)
goto err; goto err;
if (!device_create_file(&device->dev, &dev_attr_resource_in_use))
device->remove = acpi_power_sysfs_remove;
mutex_lock(&power_resource_list_lock); mutex_lock(&power_resource_list_lock);
list_add(&resource->list_node, &acpi_power_resource_list); list_add(&resource->list_node, &acpi_power_resource_list);
mutex_unlock(&power_resource_list_lock); mutex_unlock(&power_resource_list_lock);
......
...@@ -791,6 +791,9 @@ static void acpi_device_unregister(struct acpi_device *device) ...@@ -791,6 +791,9 @@ static void acpi_device_unregister(struct acpi_device *device)
acpi_power_add_remove_device(device, false); acpi_power_add_remove_device(device, false);
acpi_device_remove_files(device); acpi_device_remove_files(device);
if (device->remove)
device->remove(device);
device_del(&device->dev); device_del(&device->dev);
/* /*
* Drop the reference counts of all power resources the device depends * Drop the reference counts of all power resources the device depends
......
...@@ -280,6 +280,7 @@ struct acpi_device { ...@@ -280,6 +280,7 @@ struct acpi_device {
struct mutex physical_node_lock; struct mutex physical_node_lock;
DECLARE_BITMAP(physical_node_id_bitmap, ACPI_MAX_PHYSICAL_NODE); DECLARE_BITMAP(physical_node_id_bitmap, ACPI_MAX_PHYSICAL_NODE);
struct list_head power_dependent; struct list_head power_dependent;
void (*remove)(struct acpi_device *);
}; };
static inline void *acpi_driver_data(struct acpi_device *d) static inline void *acpi_driver_data(struct acpi_device *d)
......
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