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 :-). */ 268174f150SMatt Helsley void refrigerator(void) 278174f150SMatt Helsley { 288174f150SMatt Helsley /* Hmm, should we be allowed to suspend when there are realtime 298174f150SMatt Helsley processes around? */ 308174f150SMatt Helsley long save; 318174f150SMatt Helsley 328174f150SMatt Helsley task_lock(current); 338174f150SMatt Helsley if (freezing(current)) { 348174f150SMatt Helsley frozen_process(); 358174f150SMatt Helsley task_unlock(current); 368174f150SMatt Helsley } else { 378174f150SMatt Helsley task_unlock(current); 388174f150SMatt Helsley return; 398174f150SMatt Helsley } 408174f150SMatt Helsley save = current->state; 418174f150SMatt Helsley pr_debug("%s entered refrigerator\n", current->comm); 428174f150SMatt Helsley 438174f150SMatt Helsley spin_lock_irq(¤t->sighand->siglock); 448174f150SMatt Helsley recalc_sigpending(); /* We sent fake signal, clean it up */ 458174f150SMatt Helsley spin_unlock_irq(¤t->sighand->siglock); 468174f150SMatt Helsley 476301cb95SThomas Gleixner /* prevent accounting of that task to load */ 486301cb95SThomas Gleixner current->flags |= PF_FREEZING; 496301cb95SThomas Gleixner 508174f150SMatt Helsley for (;;) { 518174f150SMatt Helsley set_current_state(TASK_UNINTERRUPTIBLE); 528174f150SMatt Helsley if (!frozen(current)) 538174f150SMatt Helsley break; 548174f150SMatt Helsley schedule(); 558174f150SMatt Helsley } 566301cb95SThomas Gleixner 576301cb95SThomas Gleixner /* Remove the accounting blocker */ 586301cb95SThomas Gleixner current->flags &= ~PF_FREEZING; 596301cb95SThomas Gleixner 608174f150SMatt Helsley pr_debug("%s left refrigerator\n", current->comm); 61*50fb4f7fSTejun Heo 62*50fb4f7fSTejun Heo /* 63*50fb4f7fSTejun Heo * Restore saved task state before returning. The mb'd version 64*50fb4f7fSTejun Heo * needs to be used; otherwise, it might silently break 65*50fb4f7fSTejun Heo * synchronization which depends on ordered task state change. 66*50fb4f7fSTejun Heo */ 67*50fb4f7fSTejun Heo set_current_state(save); 688174f150SMatt Helsley } 698174f150SMatt Helsley EXPORT_SYMBOL(refrigerator); 708174f150SMatt Helsley 718174f150SMatt Helsley static void fake_signal_wake_up(struct task_struct *p) 728174f150SMatt Helsley { 738174f150SMatt Helsley unsigned long flags; 748174f150SMatt Helsley 758174f150SMatt Helsley spin_lock_irqsave(&p->sighand->siglock, flags); 76d6cc7685STejun Heo signal_wake_up(p, 0); 778174f150SMatt Helsley spin_unlock_irqrestore(&p->sighand->siglock, flags); 788174f150SMatt Helsley } 798174f150SMatt Helsley 808174f150SMatt Helsley /** 818174f150SMatt Helsley * freeze_task - send a freeze request to given task 828174f150SMatt Helsley * @p: task to send the request to 838174f150SMatt Helsley * @sig_only: if set, the request will only be sent if the task has the 848174f150SMatt Helsley * PF_FREEZER_NOSIG flag unset 858174f150SMatt Helsley * Return value: 'false', if @sig_only is set and the task has 868174f150SMatt Helsley * PF_FREEZER_NOSIG set or the task is frozen, 'true', otherwise 878174f150SMatt Helsley * 888174f150SMatt Helsley * The freeze request is sent by setting the tasks's TIF_FREEZE flag and 898174f150SMatt Helsley * either sending a fake signal to it or waking it up, depending on whether 908174f150SMatt Helsley * or not it has PF_FREEZER_NOSIG set. If @sig_only is set and the task 918174f150SMatt Helsley * has PF_FREEZER_NOSIG set (ie. it is a typical kernel thread), its 928174f150SMatt Helsley * TIF_FREEZE flag will not be set. 938174f150SMatt Helsley */ 948174f150SMatt Helsley bool freeze_task(struct task_struct *p, bool sig_only) 958174f150SMatt Helsley { 968174f150SMatt Helsley /* 978174f150SMatt Helsley * We first check if the task is freezing and next if it has already 988174f150SMatt Helsley * been frozen to avoid the race with frozen_process() which first marks 998174f150SMatt Helsley * the task as frozen and next clears its TIF_FREEZE. 1008174f150SMatt Helsley */ 1018174f150SMatt Helsley if (!freezing(p)) { 102ee940d8dSMike Frysinger smp_rmb(); 1038174f150SMatt Helsley if (frozen(p)) 1048174f150SMatt Helsley return false; 1058174f150SMatt Helsley 1068174f150SMatt Helsley if (!sig_only || should_send_signal(p)) 1078174f150SMatt Helsley set_freeze_flag(p); 1088174f150SMatt Helsley else 1098174f150SMatt Helsley return false; 1108174f150SMatt Helsley } 1118174f150SMatt Helsley 1128174f150SMatt Helsley if (should_send_signal(p)) { 1138174f150SMatt Helsley fake_signal_wake_up(p); 1148cfe400cSTejun Heo /* 1158cfe400cSTejun Heo * fake_signal_wake_up() goes through p's scheduler 1168cfe400cSTejun Heo * lock and guarantees that TASK_STOPPED/TRACED -> 1178cfe400cSTejun Heo * TASK_RUNNING transition can't race with task state 1188cfe400cSTejun Heo * testing in try_to_freeze_tasks(). 1198cfe400cSTejun Heo */ 1208174f150SMatt Helsley } else if (sig_only) { 1218174f150SMatt Helsley return false; 1228174f150SMatt Helsley } else { 1238174f150SMatt Helsley wake_up_state(p, TASK_INTERRUPTIBLE); 1248174f150SMatt Helsley } 1258174f150SMatt Helsley 1268174f150SMatt Helsley return true; 1278174f150SMatt Helsley } 1288174f150SMatt Helsley 1298174f150SMatt Helsley void cancel_freezing(struct task_struct *p) 1308174f150SMatt Helsley { 1318174f150SMatt Helsley unsigned long flags; 1328174f150SMatt Helsley 1338174f150SMatt Helsley if (freezing(p)) { 1348174f150SMatt Helsley pr_debug(" clean up: %s\n", p->comm); 1358174f150SMatt Helsley clear_freeze_flag(p); 1368174f150SMatt Helsley spin_lock_irqsave(&p->sighand->siglock, flags); 1378174f150SMatt Helsley recalc_sigpending_and_wake(p); 1388174f150SMatt Helsley spin_unlock_irqrestore(&p->sighand->siglock, flags); 1398174f150SMatt Helsley } 1408174f150SMatt Helsley } 141dc52ddc0SMatt Helsley 14200c2e63cSLi Zefan static int __thaw_process(struct task_struct *p) 143dc52ddc0SMatt Helsley { 144dc52ddc0SMatt Helsley if (frozen(p)) { 145dc52ddc0SMatt Helsley p->flags &= ~PF_FROZEN; 146dc52ddc0SMatt Helsley return 1; 147dc52ddc0SMatt Helsley } 148dc52ddc0SMatt Helsley clear_freeze_flag(p); 149dc52ddc0SMatt Helsley return 0; 150dc52ddc0SMatt Helsley } 151dc52ddc0SMatt Helsley 15200c2e63cSLi Zefan /* 15300c2e63cSLi Zefan * Wake up a frozen process 15400c2e63cSLi Zefan * 15500c2e63cSLi Zefan * task_lock() is needed to prevent the race with refrigerator() which may 15600c2e63cSLi Zefan * occur if the freezing of tasks fails. Namely, without the lock, if the 15700c2e63cSLi Zefan * freezing of tasks failed, thaw_tasks() might have run before a task in 15800c2e63cSLi Zefan * refrigerator() could call frozen_process(), in which case the task would be 15900c2e63cSLi Zefan * frozen and no one would thaw it. 16000c2e63cSLi Zefan */ 161dc52ddc0SMatt Helsley int thaw_process(struct task_struct *p) 162dc52ddc0SMatt Helsley { 163dc52ddc0SMatt Helsley task_lock(p); 164dc52ddc0SMatt Helsley if (__thaw_process(p) == 1) { 165dc52ddc0SMatt Helsley task_unlock(p); 166dc52ddc0SMatt Helsley wake_up_process(p); 167dc52ddc0SMatt Helsley return 1; 168dc52ddc0SMatt Helsley } 169dc52ddc0SMatt Helsley task_unlock(p); 170dc52ddc0SMatt Helsley return 0; 171dc52ddc0SMatt Helsley } 172dc52ddc0SMatt Helsley EXPORT_SYMBOL(thaw_process); 173