ACPI / PM: Introduce concept of a _PR0 dependent device
authorMika Westerberg <mika.westerberg@linux.intel.com>
Tue, 25 Jun 2019 10:29:41 +0000 (13:29 +0300)
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>
Thu, 27 Jun 2019 10:31:57 +0000 (12:31 +0200)
If there are shared power resources between otherwise unrelated devices
turning them on causes the other devices sharing them to be powered up
as well. In case of PCI devices go into D0uninitialized state meaning
that if they were configured to trigger wake that configuration is lost
at this point.

For this reason introduce a concept of "_PR0 dependent device" that can
be added to any ACPI device that has power resources. The dependent
device will be included in a list of dependent devices for all power
resources returned by the ACPI device's _PR0 (assuming it has one).
Whenever a power resource having dependent devices is turned physically
on (its _ON method is called) we runtime resume all of them to allow
their driver or in case of PCI the PCI core to re-initialize the device
and its wake configuration.

This adds two functions that can be used to add and remove these
dependent devices. Note the dependent device does not necessary need
share power resources so this functionality can be used to add "software
dependencies" as well if needed.

Signed-off-by: Mika Westerberg <mika.westerberg@linux.intel.com>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
drivers/acpi/power.c
include/acpi/acpi_bus.h

index a916417b9e70f69b4d022196eb25ba372ad781e6..fe1e7bc91a5ea24009c541d0387d2f62d18ea7f7 100644 (file)
@@ -42,6 +42,11 @@ ACPI_MODULE_NAME("power");
 #define ACPI_POWER_RESOURCE_STATE_ON   0x01
 #define ACPI_POWER_RESOURCE_STATE_UNKNOWN 0xFF
 
+struct acpi_power_dependent_device {
+       struct device *dev;
+       struct list_head node;
+};
+
 struct acpi_power_resource {
        struct acpi_device device;
        struct list_head list_node;
@@ -51,6 +56,7 @@ struct acpi_power_resource {
        unsigned int ref_count;
        bool wakeup_enabled;
        struct mutex resource_lock;
+       struct list_head dependents;
 };
 
 struct acpi_power_resource_entry {
@@ -232,8 +238,121 @@ static int acpi_power_get_list_state(struct list_head *list, int *state)
        return 0;
 }
 
+static int
+acpi_power_resource_add_dependent(struct acpi_power_resource *resource,
+                                 struct device *dev)
+{
+       struct acpi_power_dependent_device *dep;
+       int ret = 0;
+
+       mutex_lock(&resource->resource_lock);
+       list_for_each_entry(dep, &resource->dependents, node) {
+               /* Only add it once */
+               if (dep->dev == dev)
+                       goto unlock;
+       }
+
+       dep = kzalloc(sizeof(*dep), GFP_KERNEL);
+       if (!dep) {
+               ret = -ENOMEM;
+               goto unlock;
+       }
+
+       dep->dev = dev;
+       list_add_tail(&dep->node, &resource->dependents);
+       dev_dbg(dev, "added power dependency to [%s]\n", resource->name);
+
+unlock:
+       mutex_unlock(&resource->resource_lock);
+       return ret;
+}
+
+static void
+acpi_power_resource_remove_dependent(struct acpi_power_resource *resource,
+                                    struct device *dev)
+{
+       struct acpi_power_dependent_device *dep;
+
+       mutex_lock(&resource->resource_lock);
+       list_for_each_entry(dep, &resource->dependents, node) {
+               if (dep->dev == dev) {
+                       list_del(&dep->node);
+                       kfree(dep);
+                       dev_dbg(dev, "removed power dependency to [%s]\n",
+                               resource->name);
+                       break;
+               }
+       }
+       mutex_unlock(&resource->resource_lock);
+}
+
+/**
+ * acpi_device_power_add_dependent - Add dependent device of this ACPI device
+ * @adev: ACPI device pointer
+ * @dev: Dependent device
+ *
+ * If @adev has non-empty _PR0 the @dev is added as dependent device to all
+ * power resources returned by it. This means that whenever these power
+ * resources are turned _ON the dependent devices get runtime resumed. This
+ * is needed for devices such as PCI to allow its driver to re-initialize
+ * it after it went to D0uninitialized.
+ *
+ * If @adev does not have _PR0 this does nothing.
+ *
+ * Returns %0 in case of success and negative errno otherwise.
+ */
+int acpi_device_power_add_dependent(struct acpi_device *adev,
+                                   struct device *dev)
+{
+       struct acpi_power_resource_entry *entry;
+       struct list_head *resources;
+       int ret;
+
+       if (!adev->flags.power_manageable)
+               return 0;
+
+       resources = &adev->power.states[ACPI_STATE_D0].resources;
+       list_for_each_entry(entry, resources, node) {
+               ret = acpi_power_resource_add_dependent(entry->resource, dev);
+               if (ret)
+                       goto err;
+       }
+
+       return 0;
+
+err:
+       list_for_each_entry(entry, resources, node)
+               acpi_power_resource_remove_dependent(entry->resource, dev);
+
+       return ret;
+}
+
+/**
+ * acpi_device_power_remove_dependent - Remove dependent device
+ * @adev: ACPI device pointer
+ * @dev: Dependent device
+ *
+ * Does the opposite of acpi_device_power_add_dependent() and removes the
+ * dependent device if it is found. Can be called to @adev that does not
+ * have _PR0 as well.
+ */
+void acpi_device_power_remove_dependent(struct acpi_device *adev,
+                                       struct device *dev)
+{
+       struct acpi_power_resource_entry *entry;
+       struct list_head *resources;
+
+       if (!adev->flags.power_manageable)
+               return;
+
+       resources = &adev->power.states[ACPI_STATE_D0].resources;
+       list_for_each_entry_reverse(entry, resources, node)
+               acpi_power_resource_remove_dependent(entry->resource, dev);
+}
+
 static int __acpi_power_on(struct acpi_power_resource *resource)
 {
+       struct acpi_power_dependent_device *dep;
        acpi_status status = AE_OK;
 
        status = acpi_evaluate_object(resource->device.handle, "_ON", NULL, NULL);
@@ -243,6 +362,21 @@ static int __acpi_power_on(struct acpi_power_resource *resource)
        ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Power resource [%s] turned on\n",
                          resource->name));
 
+       /*
+        * If there are other dependents on this power resource we need to
+        * resume them now so that their drivers can re-initialize the
+        * hardware properly after it went back to D0.
+        */
+       if (list_empty(&resource->dependents) ||
+           list_is_singular(&resource->dependents))
+               return 0;
+
+       list_for_each_entry(dep, &resource->dependents, node) {
+               dev_dbg(dep->dev, "runtime resuming because [%s] turned on\n",
+                       resource->name);
+               pm_request_resume(dep->dev);
+       }
+
        return 0;
 }
 
@@ -810,6 +944,7 @@ int acpi_add_power_resource(acpi_handle handle)
                                ACPI_STA_DEFAULT);
        mutex_init(&resource->resource_lock);
        INIT_LIST_HEAD(&resource->list_node);
+       INIT_LIST_HEAD(&resource->dependents);
        resource->name = device->pnp.bus_id;
        strcpy(acpi_device_name(device), ACPI_POWER_DEVICE_NAME);
        strcpy(acpi_device_class(device), ACPI_POWER_CLASS);
index 31b6c87d6240d6e7b9d6e70339775e76780b929a..4752ff0a9d9b55f8737d4111fc43470aaafaab2b 100644 (file)
@@ -513,6 +513,10 @@ int acpi_device_fix_up_power(struct acpi_device *device);
 int acpi_bus_update_power(acpi_handle handle, int *state_p);
 int acpi_device_update_power(struct acpi_device *device, int *state_p);
 bool acpi_bus_power_manageable(acpi_handle handle);
+int acpi_device_power_add_dependent(struct acpi_device *adev,
+                                   struct device *dev);
+void acpi_device_power_remove_dependent(struct acpi_device *adev,
+                                       struct device *dev);
 
 #ifdef CONFIG_PM
 bool acpi_bus_can_wakeup(acpi_handle handle);