PM / sleep: Check pm_wakeup_pending() in __device_suspend_noirq()
authorRafael J. Wysocki <rafael.j.wysocki@intel.com>
Fri, 21 Jul 2017 00:12:10 +0000 (02:12 +0200)
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>
Mon, 24 Jul 2017 21:53:46 +0000 (23:53 +0200)
Restore the pm_wakeup_pending() check in __device_suspend_noirq()
removed by commit eed4d47efe95 (ACPI / sleep: Ignore spurious SCI
wakeups from suspend-to-idle) as that allows the function to return
earlier if there's a wakeup event pending already (so that it may
spend less time on carrying out operations that will be reversed
shortly anyway) and rework the main suspend-to-idle loop to take
that optimization into account.

Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
drivers/base/power/main.c
kernel/power/suspend.c

index f64e5b819e5cc406f538d63bb617ed8d6143e3a3..ea1732ed7a9d9071db73784c7e37d505ed8dea0a 100644 (file)
@@ -1105,6 +1105,11 @@ static int __device_suspend_noirq(struct device *dev, pm_message_t state, bool a
        if (async_error)
                goto Complete;
 
+       if (pm_wakeup_pending()) {
+               async_error = -EBUSY;
+               goto Complete;
+       }
+
        if (dev->power.syscore || dev->power.direct_complete)
                goto Complete;
 
index 0c61713b6e5ce820bbe81e33fa68c1bb50744b98..5cf232795318122a2eeec6393456c4f5352a4cb1 100644 (file)
@@ -108,19 +108,32 @@ static void s2idle_loop(void)
 {
        pm_pr_dbg("suspend-to-idle\n");
 
-       while (!dpm_suspend_noirq(PMSG_SUSPEND)) {
+       for (;;) {
+               int error;
+
+               dpm_noirq_begin();
+
                /*
                 * Suspend-to-idle equals
                 * frozen processes + suspended devices + idle processors.
                 * Thus freeze_enter() should be called right after
                 * all devices have been suspended.
                 */
-               freeze_enter();
+               error = dpm_noirq_suspend_devices(PMSG_SUSPEND);
+               if (!error)
+                       freeze_enter();
+
+               dpm_noirq_resume_devices(PMSG_RESUME);
+               if (error && (error != -EBUSY || !pm_wakeup_pending())) {
+                       dpm_noirq_end();
+                       break;
+               }
 
                if (freeze_ops && freeze_ops->wake)
                        freeze_ops->wake();
 
-               dpm_resume_noirq(PMSG_RESUME);
+               dpm_noirq_end();
+
                if (freeze_ops && freeze_ops->sync)
                        freeze_ops->sync();