18174f150SMatt Helsley /* 28174f150SMatt Helsley * kernel/freezer.c - Function to freeze a process 38174f150SMatt Helsley * 48174f150SMatt Helsley * Originally from kernel/power/process.c 58174f150SMatt Helsley */ 68174f150SMatt Helsley 78174f150SMatt Helsley #include <linux/interrupt.h> 88174f150SMatt Helsley #include <linux/suspend.h> 99984de1aSPaul Gortmaker #include <linux/export.h> 108174f150SMatt Helsley #include <linux/syscalls.h> 118174f150SMatt Helsley #include <linux/freezer.h> 128174f150SMatt Helsley 138174f150SMatt Helsley /* 148174f150SMatt Helsley * freezing is complete, mark current process as frozen 158174f150SMatt Helsley */ 168174f150SMatt Helsley static inline void frozen_process(void) 178174f150SMatt Helsley { 188174f150SMatt Helsley if (!unlikely(current->flags & PF_NOFREEZE)) { 198174f150SMatt Helsley current->flags |= PF_FROZEN; 20ee940d8dSMike Frysinger smp_wmb(); 218174f150SMatt Helsley } 228174f150SMatt Helsley clear_freeze_flag(current); 238174f150SMatt Helsley } 248174f150SMatt Helsley 258174f150SMatt Helsley /* Refrigerator is place where frozen processes are stored :-). */ 26*a0acae0eSTejun Heo bool __refrigerator(void) 278174f150SMatt Helsley { 288174f150SMatt Helsley /* Hmm, should we be allowed to suspend when there are realtime 298174f150SMatt Helsley processes around? */ 30*a0acae0eSTejun Heo bool was_frozen = false; 318174f150SMatt Helsley long save; 328174f150SMatt Helsley 338174f150SMatt Helsley task_lock(current); 348174f150SMatt Helsley if (freezing(current)) { 358174f150SMatt Helsley frozen_process(); 368174f150SMatt Helsley task_unlock(current); 378174f150SMatt Helsley } else { 388174f150SMatt Helsley task_unlock(current); 39*a0acae0eSTejun Heo return was_frozen; 408174f150SMatt Helsley } 418174f150SMatt Helsley save = current->state; 428174f150SMatt Helsley pr_debug("%s entered refrigerator\n", current->comm); 438174f150SMatt Helsley 448174f150SMatt Helsley spin_lock_irq(¤t->sighand->siglock); 458174f150SMatt Helsley recalc_sigpending(); /* We sent fake signal, clean it up */ 468174f150SMatt Helsley spin_unlock_irq(¤t->sighand->siglock); 478174f150SMatt Helsley 486301cb95SThomas Gleixner /* prevent accounting of that task to load */ 496301cb95SThomas Gleixner current->flags |= PF_FREEZING; 506301cb95SThomas Gleixner 518174f150SMatt Helsley for (;;) { 528174f150SMatt Helsley set_current_state(TASK_UNINTERRUPTIBLE); 538174f150SMatt Helsley if (!frozen(current)) 548174f150SMatt Helsley break; 55*a0acae0eSTejun Heo was_frozen = true; 568174f150SMatt Helsley schedule(); 578174f150SMatt Helsley } 586301cb95SThomas Gleixner 596301cb95SThomas Gleixner /* Remove the accounting blocker */ 606301cb95SThomas Gleixner current->flags &= ~PF_FREEZING; 616301cb95SThomas Gleixner 628174f150SMatt Helsley pr_debug("%s left refrigerator\n", current->comm); 6350fb4f7fSTejun Heo 6450fb4f7fSTejun Heo /* 6550fb4f7fSTejun Heo * Restore saved task state before returning. The mb'd version 6650fb4f7fSTejun Heo * needs to be used; otherwise, it might silently break 6750fb4f7fSTejun Heo * synchronization which depends on ordered task state change. 6850fb4f7fSTejun Heo */ 6950fb4f7fSTejun Heo set_current_state(save); 70*a0acae0eSTejun Heo 71*a0acae0eSTejun Heo return was_frozen; 728174f150SMatt Helsley } 73*a0acae0eSTejun Heo EXPORT_SYMBOL(__refrigerator); 748174f150SMatt Helsley 758174f150SMatt Helsley static void fake_signal_wake_up(struct task_struct *p) 768174f150SMatt Helsley { 778174f150SMatt Helsley unsigned long flags; 788174f150SMatt Helsley 798174f150SMatt Helsley spin_lock_irqsave(&p->sighand->siglock, flags); 80d6cc7685STejun Heo signal_wake_up(p, 0); 818174f150SMatt Helsley spin_unlock_irqrestore(&p->sighand->siglock, flags); 828174f150SMatt Helsley } 838174f150SMatt Helsley 848174f150SMatt Helsley /** 858174f150SMatt Helsley * freeze_task - send a freeze request to given task 868174f150SMatt Helsley * @p: task to send the request to 878174f150SMatt Helsley * @sig_only: if set, the request will only be sent if the task has the 888174f150SMatt Helsley * PF_FREEZER_NOSIG flag unset 898174f150SMatt Helsley * Return value: 'false', if @sig_only is set and the task has 908174f150SMatt Helsley * PF_FREEZER_NOSIG set or the task is frozen, 'true', otherwise 918174f150SMatt Helsley * 928174f150SMatt Helsley * The freeze request is sent by setting the tasks's TIF_FREEZE flag and 938174f150SMatt Helsley * either sending a fake signal to it or waking it up, depending on whether 948174f150SMatt Helsley * or not it has PF_FREEZER_NOSIG set. If @sig_only is set and the task 958174f150SMatt Helsley * has PF_FREEZER_NOSIG set (ie. it is a typical kernel thread), its 968174f150SMatt Helsley * TIF_FREEZE flag will not be set. 978174f150SMatt Helsley */ 988174f150SMatt Helsley bool freeze_task(struct task_struct *p, bool sig_only) 998174f150SMatt Helsley { 1008174f150SMatt Helsley /* 1018174f150SMatt Helsley * We first check if the task is freezing and next if it has already 1028174f150SMatt Helsley * been frozen to avoid the race with frozen_process() which first marks 1038174f150SMatt Helsley * the task as frozen and next clears its TIF_FREEZE. 1048174f150SMatt Helsley */ 1058174f150SMatt Helsley if (!freezing(p)) { 106ee940d8dSMike Frysinger smp_rmb(); 1078174f150SMatt Helsley if (frozen(p)) 1088174f150SMatt Helsley return false; 1098174f150SMatt Helsley 1108174f150SMatt Helsley if (!sig_only || should_send_signal(p)) 1118174f150SMatt Helsley set_freeze_flag(p); 1128174f150SMatt Helsley else 1138174f150SMatt Helsley return false; 1148174f150SMatt Helsley } 1158174f150SMatt Helsley 1168174f150SMatt Helsley if (should_send_signal(p)) { 1178174f150SMatt Helsley fake_signal_wake_up(p); 1188cfe400cSTejun Heo /* 1198cfe400cSTejun Heo * fake_signal_wake_up() goes through p's scheduler 1208cfe400cSTejun Heo * lock and guarantees that TASK_STOPPED/TRACED -> 1218cfe400cSTejun Heo * TASK_RUNNING transition can't race with task state 1228cfe400cSTejun Heo * testing in try_to_freeze_tasks(). 1238cfe400cSTejun Heo */ 1248174f150SMatt Helsley } else if (sig_only) { 1258174f150SMatt Helsley return false; 1268174f150SMatt Helsley } else { 1278174f150SMatt Helsley wake_up_state(p, TASK_INTERRUPTIBLE); 1288174f150SMatt Helsley } 1298174f150SMatt Helsley 1308174f150SMatt Helsley return true; 1318174f150SMatt Helsley } 1328174f150SMatt Helsley 1338174f150SMatt Helsley void cancel_freezing(struct task_struct *p) 1348174f150SMatt Helsley { 1358174f150SMatt Helsley unsigned long flags; 1368174f150SMatt Helsley 1378174f150SMatt Helsley if (freezing(p)) { 1388174f150SMatt Helsley pr_debug(" clean up: %s\n", p->comm); 1398174f150SMatt Helsley clear_freeze_flag(p); 1408174f150SMatt Helsley spin_lock_irqsave(&p->sighand->siglock, flags); 1418174f150SMatt Helsley recalc_sigpending_and_wake(p); 1428174f150SMatt Helsley spin_unlock_irqrestore(&p->sighand->siglock, flags); 1438174f150SMatt Helsley } 1448174f150SMatt Helsley } 145dc52ddc0SMatt Helsley 14600c2e63cSLi Zefan static int __thaw_process(struct task_struct *p) 147dc52ddc0SMatt Helsley { 148dc52ddc0SMatt Helsley if (frozen(p)) { 149dc52ddc0SMatt Helsley p->flags &= ~PF_FROZEN; 150dc52ddc0SMatt Helsley return 1; 151dc52ddc0SMatt Helsley } 152dc52ddc0SMatt Helsley clear_freeze_flag(p); 153dc52ddc0SMatt Helsley return 0; 154dc52ddc0SMatt Helsley } 155dc52ddc0SMatt Helsley 15600c2e63cSLi Zefan /* 15700c2e63cSLi Zefan * Wake up a frozen process 15800c2e63cSLi Zefan * 15900c2e63cSLi Zefan * task_lock() is needed to prevent the race with refrigerator() which may 16000c2e63cSLi Zefan * occur if the freezing of tasks fails. Namely, without the lock, if the 16100c2e63cSLi Zefan * freezing of tasks failed, thaw_tasks() might have run before a task in 16200c2e63cSLi Zefan * refrigerator() could call frozen_process(), in which case the task would be 16300c2e63cSLi Zefan * frozen and no one would thaw it. 16400c2e63cSLi Zefan */ 165dc52ddc0SMatt Helsley int thaw_process(struct task_struct *p) 166dc52ddc0SMatt Helsley { 167dc52ddc0SMatt Helsley task_lock(p); 168dc52ddc0SMatt Helsley if (__thaw_process(p) == 1) { 169dc52ddc0SMatt Helsley task_unlock(p); 170dc52ddc0SMatt Helsley wake_up_process(p); 171dc52ddc0SMatt Helsley return 1; 172dc52ddc0SMatt Helsley } 173dc52ddc0SMatt Helsley task_unlock(p); 174dc52ddc0SMatt Helsley return 0; 175dc52ddc0SMatt Helsley } 176dc52ddc0SMatt Helsley EXPORT_SYMBOL(thaw_process); 177