aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/power/process.c
diff options
context:
space:
mode:
authorRafael J. Wysocki <rjw@sisk.pl>2007-05-23 16:57:24 -0400
committerLinus Torvalds <torvalds@woody.linux-foundation.org>2007-05-23 23:14:10 -0400
commit33e1c288da62a6a5aa9077a6b7bfa690b1b02cf4 (patch)
tree68837483316db1fa08bcb2b444cabae75d31a5b2 /kernel/power/process.c
parent585a2858b970cb6e2e5ca4877eefd18b4dba8ed4 (diff)
freezer: close potential race between refrigerator and thaw_tasks
If the freezing of tasks fails and a task is preempted in refrigerator() before calling frozen_process(), then thaw_tasks() may run before this task is frozen. In that case the task will freeze and no one will thaw it. To fix this race we can call freezing(current) in refrigerator() along with frozen_process(current) under the task_lock() which also should be taken in the error path of try_to_freeze_tasks() as well as in thaw_process(). Moreover, if thaw_process() additionally clears TIF_FREEZE for tasks that are not frozen, we can be sure that all tasks are thawed and there are no pending "freeze" requests after thaw_tasks() has run. Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl> Acked-by: Pavel Machek <pavel@ucw.cz> Cc: Gautham R Shenoy <ego@in.ibm.com> Cc: Oleg Nesterov <oleg@tv-sign.ru> Cc: "Eric W. Biederman" <ebiederm@xmission.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'kernel/power/process.c')
-rw-r--r--kernel/power/process.c12
1 files changed, 11 insertions, 1 deletions
diff --git a/kernel/power/process.c b/kernel/power/process.c
index 088419387388..02e490e311eb 100644
--- a/kernel/power/process.c
+++ b/kernel/power/process.c
@@ -37,10 +37,18 @@ void refrigerator(void)
37 /* Hmm, should we be allowed to suspend when there are realtime 37 /* Hmm, should we be allowed to suspend when there are realtime
38 processes around? */ 38 processes around? */
39 long save; 39 long save;
40
41 task_lock(current);
42 if (freezing(current)) {
43 frozen_process(current);
44 task_unlock(current);
45 } else {
46 task_unlock(current);
47 return;
48 }
40 save = current->state; 49 save = current->state;
41 pr_debug("%s entered refrigerator\n", current->comm); 50 pr_debug("%s entered refrigerator\n", current->comm);
42 51
43 frozen_process(current);
44 spin_lock_irq(&current->sighand->siglock); 52 spin_lock_irq(&current->sighand->siglock);
45 recalc_sigpending(); /* We sent fake signal, clean it up */ 53 recalc_sigpending(); /* We sent fake signal, clean it up */
46 spin_unlock_irq(&current->sighand->siglock); 54 spin_unlock_irq(&current->sighand->siglock);
@@ -152,10 +160,12 @@ static unsigned int try_to_freeze_tasks(int freeze_user_space)
152 if (is_user_space(p) == !freeze_user_space) 160 if (is_user_space(p) == !freeze_user_space)
153 continue; 161 continue;
154 162
163 task_lock(p);
155 if (freezeable(p) && !frozen(p)) 164 if (freezeable(p) && !frozen(p))
156 printk(KERN_ERR " %s\n", p->comm); 165 printk(KERN_ERR " %s\n", p->comm);
157 166
158 cancel_freezing(p); 167 cancel_freezing(p);
168 task_unlock(p);
159 } while_each_thread(g, p); 169 } while_each_thread(g, p);
160 read_unlock(&tasklist_lock); 170 read_unlock(&tasklist_lock);
161 } 171 }