Skip to content

Commit

Permalink
PM: sleep: Pause cpuidle later and resume it earlier during system tr…
Browse files Browse the repository at this point in the history
…ansitions

Commit 8651f97 ("PM / cpuidle: System resume hang fix with
cpuidle") that introduced cpuidle pausing during system suspend
did that to work around a platform firmware issue causing systems
to hang during resume if CPUs were allowed to enter idle states
in the system suspend and resume code paths.

However, pausing cpuidle before the last phase of suspending
devices is the source of an otherwise arbitrary difference between
the suspend-to-idle path and other system suspend variants, so it is
cleaner to do that later, before taking secondary CPUs offline (it
is still safer to take secondary CPUs offline with cpuidle paused,
though).

Modify the code accordingly, but in order to avoid code duplication,
introduce new wrapper functions, pm_sleep_disable_secondary_cpus()
and pm_sleep_enable_secondary_cpus(), to combine cpuidle_pause()
and cpuidle_resume(), respectively, with the handling of secondary
CPUs during system-wide transitions to sleep states.

Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Reviewed-by: Ulf Hansson <ulf.hansson@linaro.org>
Tested-by: Ulf Hansson <ulf.hansson@linaro.org>
  • Loading branch information
Rafael J. Wysocki committed Oct 26, 2021
1 parent 8d89835 commit 23f62d7
Show file tree
Hide file tree
Showing 4 changed files with 24 additions and 20 deletions.
8 changes: 1 addition & 7 deletions drivers/base/power/main.c
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,6 @@
#include <linux/suspend.h>
#include <trace/events/power.h>
#include <linux/cpufreq.h>
#include <linux/cpuidle.h>
#include <linux/devfreq.h>
#include <linux/timer.h>

Expand Down Expand Up @@ -879,7 +878,6 @@ void dpm_resume_early(pm_message_t state)
void dpm_resume_start(pm_message_t state)
{
dpm_resume_noirq(state);
cpuidle_resume();
dpm_resume_early(state);
}
EXPORT_SYMBOL_GPL(dpm_resume_start);
Expand Down Expand Up @@ -1519,13 +1517,9 @@ int dpm_suspend_end(pm_message_t state)
if (error)
goto out;

cpuidle_pause();

error = dpm_suspend_noirq(state);
if (error) {
cpuidle_resume();
if (error)
dpm_resume_early(resume_event(state));
}

out:
dpm_show_time(starttime, state, error, "end");
Expand Down
12 changes: 7 additions & 5 deletions kernel/power/hibernate.c
Original file line number Diff line number Diff line change
Expand Up @@ -300,7 +300,7 @@ static int create_image(int platform_mode)
if (error || hibernation_test(TEST_PLATFORM))
goto Platform_finish;

error = suspend_disable_secondary_cpus();
error = pm_sleep_disable_secondary_cpus();
if (error || hibernation_test(TEST_CPUS))
goto Enable_cpus;

Expand Down Expand Up @@ -342,7 +342,7 @@ static int create_image(int platform_mode)
local_irq_enable();

Enable_cpus:
suspend_enable_secondary_cpus();
pm_sleep_enable_secondary_cpus();

/* Allow architectures to do nosmt-specific post-resume dances */
if (!in_suspend)
Expand Down Expand Up @@ -466,6 +466,8 @@ static int resume_target_kernel(bool platform_mode)
if (error)
goto Cleanup;

cpuidle_pause();

error = hibernate_resume_nonboot_cpu_disable();
if (error)
goto Enable_cpus;
Expand Down Expand Up @@ -509,7 +511,7 @@ static int resume_target_kernel(bool platform_mode)
local_irq_enable();

Enable_cpus:
suspend_enable_secondary_cpus();
pm_sleep_enable_secondary_cpus();

Cleanup:
platform_restore_cleanup(platform_mode);
Expand Down Expand Up @@ -587,7 +589,7 @@ int hibernation_platform_enter(void)
if (error)
goto Platform_finish;

error = suspend_disable_secondary_cpus();
error = pm_sleep_disable_secondary_cpus();
if (error)
goto Enable_cpus;

Expand All @@ -609,7 +611,7 @@ int hibernation_platform_enter(void)
local_irq_enable();

Enable_cpus:
suspend_enable_secondary_cpus();
pm_sleep_enable_secondary_cpus();

Platform_finish:
hibernation_ops->finish();
Expand Down
14 changes: 14 additions & 0 deletions kernel/power/power.h
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@
#include <linux/utsname.h>
#include <linux/freezer.h>
#include <linux/compiler.h>
#include <linux/cpu.h>
#include <linux/cpuidle.h>

struct swsusp_info {
struct new_utsname uts;
Expand Down Expand Up @@ -310,3 +312,15 @@ extern int pm_wake_lock(const char *buf);
extern int pm_wake_unlock(const char *buf);

#endif /* !CONFIG_PM_WAKELOCKS */

static inline int pm_sleep_disable_secondary_cpus(void)
{
cpuidle_pause();
return suspend_disable_secondary_cpus();
}

static inline void pm_sleep_enable_secondary_cpus(void)
{
suspend_enable_secondary_cpus();
cpuidle_resume();
}
10 changes: 2 additions & 8 deletions kernel/power/suspend.c
Original file line number Diff line number Diff line change
Expand Up @@ -403,9 +403,6 @@ static int suspend_enter(suspend_state_t state, bool *wakeup)
if (error)
goto Devices_early_resume;

if (state != PM_SUSPEND_TO_IDLE)
cpuidle_pause();

error = dpm_suspend_noirq(PMSG_SUSPEND);
if (error) {
pr_err("noirq suspend of devices failed\n");
Expand All @@ -423,7 +420,7 @@ static int suspend_enter(suspend_state_t state, bool *wakeup)
goto Platform_wake;
}

error = suspend_disable_secondary_cpus();
error = pm_sleep_disable_secondary_cpus();
if (error || suspend_test(TEST_CPUS))
goto Enable_cpus;

Expand Down Expand Up @@ -453,16 +450,13 @@ static int suspend_enter(suspend_state_t state, bool *wakeup)
BUG_ON(irqs_disabled());

Enable_cpus:
suspend_enable_secondary_cpus();
pm_sleep_enable_secondary_cpus();

Platform_wake:
platform_resume_noirq(state);
dpm_resume_noirq(PMSG_RESUME);

Platform_early_resume:
if (state != PM_SUSPEND_TO_IDLE)
cpuidle_resume();

platform_resume_early(state);

Devices_early_resume:
Expand Down

0 comments on commit 23f62d7

Please sign in to comment.