Commit 23783f81 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'pm-fixes-for-3.3-rc3' of git://git.kernel.org/pub/scm/linux/kernel/git/rafael/linux-pm

Power management fixes for 3.3-rc3

Three power management regression fixes, one for a recent regression introcuded
by the freezer changes during the 3.3 merge window and two for regressions
in cpuidle (resulting from PM QoS changes) and in the hibernate user space
interface, both introduced during the 3.2 development cycle.

They include:

* Two hibernate (s2disk) regression fixes from Srivatsa S. Bhat (for
 regressions introduced during the 3.3 merge window and during the 3.2
 development cycle).

* A cpuidle fix from Venki Pallipadi for a regression resulting from PM QoS
 changes during the 3.2 development cycle causing cpuidle to work incorrectly
 for CONFIG_PM unset.

* tag 'pm-fixes-for-3.3-rc3' of git://git.kernel.org/pub/scm/linux/kernel/git/rafael/linux-pm:
  PM / QoS: CPU C-state breakage with PM Qos change
  PM / Freezer: Thaw only kernel threads if freezing of kernel threads fails
  PM / Hibernate: Thaw kernel threads in SNAPSHOT_CREATE_IMAGE ioctl path
parents d9142025 d020283d
...@@ -110,7 +110,19 @@ static inline void pm_qos_remove_request(struct pm_qos_request *req) ...@@ -110,7 +110,19 @@ static inline void pm_qos_remove_request(struct pm_qos_request *req)
{ return; } { return; }
static inline int pm_qos_request(int pm_qos_class) static inline int pm_qos_request(int pm_qos_class)
{ return 0; } {
switch (pm_qos_class) {
case PM_QOS_CPU_DMA_LATENCY:
return PM_QOS_CPU_DMA_LAT_DEFAULT_VALUE;
case PM_QOS_NETWORK_LATENCY:
return PM_QOS_NETWORK_LAT_DEFAULT_VALUE;
case PM_QOS_NETWORK_THROUGHPUT:
return PM_QOS_NETWORK_THROUGHPUT_DEFAULT_VALUE;
default:
return PM_QOS_DEFAULT_VALUE;
}
}
static inline int pm_qos_add_notifier(int pm_qos_class, static inline int pm_qos_add_notifier(int pm_qos_class,
struct notifier_block *notifier) struct notifier_block *notifier)
{ return 0; } { return 0; }
......
...@@ -231,8 +231,28 @@ extern int pm_test_level; ...@@ -231,8 +231,28 @@ extern int pm_test_level;
#ifdef CONFIG_SUSPEND_FREEZER #ifdef CONFIG_SUSPEND_FREEZER
static inline int suspend_freeze_processes(void) static inline int suspend_freeze_processes(void)
{ {
int error = freeze_processes(); int error;
return error ? : freeze_kernel_threads();
error = freeze_processes();
/*
* freeze_processes() automatically thaws every task if freezing
* fails. So we need not do anything extra upon error.
*/
if (error)
goto Finish;
error = freeze_kernel_threads();
/*
* freeze_kernel_threads() thaws only kernel threads upon freezing
* failure. So we have to thaw the userspace tasks ourselves.
*/
if (error)
thaw_processes();
Finish:
return error;
} }
static inline void suspend_thaw_processes(void) static inline void suspend_thaw_processes(void)
......
...@@ -143,7 +143,10 @@ int freeze_processes(void) ...@@ -143,7 +143,10 @@ int freeze_processes(void)
/** /**
* freeze_kernel_threads - Make freezable kernel threads go to the refrigerator. * freeze_kernel_threads - Make freezable kernel threads go to the refrigerator.
* *
* On success, returns 0. On failure, -errno and system is fully thawed. * On success, returns 0. On failure, -errno and only the kernel threads are
* thawed, so as to give a chance to the caller to do additional cleanups
* (if any) before thawing the userspace tasks. So, it is the responsibility
* of the caller to thaw the userspace tasks, when the time is right.
*/ */
int freeze_kernel_threads(void) int freeze_kernel_threads(void)
{ {
...@@ -159,7 +162,7 @@ int freeze_kernel_threads(void) ...@@ -159,7 +162,7 @@ int freeze_kernel_threads(void)
BUG_ON(in_atomic()); BUG_ON(in_atomic());
if (error) if (error)
thaw_processes(); thaw_kernel_threads();
return error; return error;
} }
......
...@@ -249,13 +249,15 @@ static long snapshot_ioctl(struct file *filp, unsigned int cmd, ...@@ -249,13 +249,15 @@ static long snapshot_ioctl(struct file *filp, unsigned int cmd,
} }
pm_restore_gfp_mask(); pm_restore_gfp_mask();
error = hibernation_snapshot(data->platform_support); error = hibernation_snapshot(data->platform_support);
if (!error) { if (error) {
thaw_kernel_threads();
} else {
error = put_user(in_suspend, (int __user *)arg); error = put_user(in_suspend, (int __user *)arg);
if (!error && !freezer_test_done) if (!error && !freezer_test_done)
data->ready = 1; data->ready = 1;
if (freezer_test_done) { if (freezer_test_done) {
freezer_test_done = false; freezer_test_done = false;
thaw_processes(); thaw_kernel_threads();
} }
} }
break; break;
......
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