]> www.infradead.org Git - users/hch/dma-mapping.git/commitdiff
PM / core: fix deferred probe breaking suspend resume order
authorFeng Kan <fkan@apm.com>
Tue, 10 Apr 2018 23:57:06 +0000 (16:57 -0700)
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>
Tue, 24 Apr 2018 10:18:25 +0000 (12:18 +0200)
When bridge and its endpoint is enumerated the devices are added to the
dpm list. Afterward, the bridge defers probe when IOMMU is not ready.
This causes the bridge to be moved to the end of the dpm list when
deferred probe kicks in. The order of the dpm list for bridge and
endpoint is reversed.

Add reordering code to move the bridge and its children and consumers to
the end of the pm list so the order for suspend and resume is not altered.
The code also move device and its children and consumers to the tail of
device_kset list if it is registered.

Signed-off-by: Toan Le <toanle@apm.com>
Signed-off-by: Feng Kan <fkan@apm.com>
Reviewed-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
drivers/base/base.h
drivers/base/core.c
drivers/base/dd.c

index d800de650fa56bc264b8465621e12c829b01d60d..a75c3025fb7886abca1349a3774d419dcd76d5a0 100644 (file)
@@ -161,3 +161,6 @@ extern void device_links_driver_cleanup(struct device *dev);
 extern void device_links_no_driver(struct device *dev);
 extern bool device_links_busy(struct device *dev);
 extern void device_links_unbind_consumers(struct device *dev);
+
+/* device pm support */
+void device_pm_move_to_tail(struct device *dev);
index b610816eb887f939a0f5952ffcd4424d0a46cf05..ad7b50897bcc703341ee7157baf743a225a3b4b5 100644 (file)
@@ -144,6 +144,26 @@ static int device_reorder_to_tail(struct device *dev, void *not_used)
        return 0;
 }
 
+/**
+ * device_pm_move_to_tail - Move set of devices to the end of device lists
+ * @dev: Device to move
+ *
+ * This is a device_reorder_to_tail() wrapper taking the requisite locks.
+ *
+ * It moves the @dev along with all of its children and all of its consumers
+ * to the ends of the device_kset and dpm_list, recursively.
+ */
+void device_pm_move_to_tail(struct device *dev)
+{
+       int idx;
+
+       idx = device_links_read_lock();
+       device_pm_lock();
+       device_reorder_to_tail(dev, NULL);
+       device_pm_unlock();
+       device_links_read_unlock(idx);
+}
+
 /**
  * device_link_add - Create a link between two devices.
  * @consumer: Consumer end of the link.
index c9f54089429bde006fad1135017d49a0422fefa5..10454fe5448259755e4b4e7df05231fe55ae6592 100644 (file)
@@ -122,9 +122,7 @@ static void deferred_probe_work_func(struct work_struct *work)
                 * the list is a good order for suspend but deferred
                 * probe makes that very unsafe.
                 */
-               device_pm_lock();
-               device_pm_move_last(dev);
-               device_pm_unlock();
+               device_pm_move_to_tail(dev);
 
                dev_dbg(dev, "Retrying from deferred list\n");
                if (initcall_debug && !initcalls_done)