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> 98174f150SMatt Helsley #include <linux/module.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; 208174f150SMatt Helsley 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 47*6301cb95SThomas Gleixner /* prevent accounting of that task to load */ 48*6301cb95SThomas Gleixner current->flags |= PF_FREEZING; 49*6301cb95SThomas Gleixner 508174f150SMatt Helsley for (;;) { 518174f150SMatt Helsley set_current_state(TASK_UNINTERRUPTIBLE); 528174f150SMatt Helsley if (!frozen(current)) 538174f150SMatt Helsley break; 548174f150SMatt Helsley schedule(); 558174f150SMatt Helsley } 56*6301cb95SThomas Gleixner 57*6301cb95SThomas Gleixner /* Remove the accounting blocker */ 58*6301cb95SThomas Gleixner current->flags &= ~PF_FREEZING; 59*6301cb95SThomas Gleixner 608174f150SMatt Helsley pr_debug("%s left refrigerator\n", current->comm); 618174f150SMatt Helsley __set_current_state(save); 628174f150SMatt Helsley } 638174f150SMatt Helsley EXPORT_SYMBOL(refrigerator); 648174f150SMatt Helsley 658174f150SMatt Helsley static void fake_signal_wake_up(struct task_struct *p) 668174f150SMatt Helsley { 678174f150SMatt Helsley unsigned long flags; 688174f150SMatt Helsley 698174f150SMatt Helsley spin_lock_irqsave(&p->sighand->siglock, flags); 708174f150SMatt Helsley signal_wake_up(p, 0); 718174f150SMatt Helsley spin_unlock_irqrestore(&p->sighand->siglock, flags); 728174f150SMatt Helsley } 738174f150SMatt Helsley 748174f150SMatt Helsley /** 758174f150SMatt Helsley * freeze_task - send a freeze request to given task 768174f150SMatt Helsley * @p: task to send the request to 778174f150SMatt Helsley * @sig_only: if set, the request will only be sent if the task has the 788174f150SMatt Helsley * PF_FREEZER_NOSIG flag unset 798174f150SMatt Helsley * Return value: 'false', if @sig_only is set and the task has 808174f150SMatt Helsley * PF_FREEZER_NOSIG set or the task is frozen, 'true', otherwise 818174f150SMatt Helsley * 828174f150SMatt Helsley * The freeze request is sent by setting the tasks's TIF_FREEZE flag and 838174f150SMatt Helsley * either sending a fake signal to it or waking it up, depending on whether 848174f150SMatt Helsley * or not it has PF_FREEZER_NOSIG set. If @sig_only is set and the task 858174f150SMatt Helsley * has PF_FREEZER_NOSIG set (ie. it is a typical kernel thread), its 868174f150SMatt Helsley * TIF_FREEZE flag will not be set. 878174f150SMatt Helsley */ 888174f150SMatt Helsley bool freeze_task(struct task_struct *p, bool sig_only) 898174f150SMatt Helsley { 908174f150SMatt Helsley /* 918174f150SMatt Helsley * We first check if the task is freezing and next if it has already 928174f150SMatt Helsley * been frozen to avoid the race with frozen_process() which first marks 938174f150SMatt Helsley * the task as frozen and next clears its TIF_FREEZE. 948174f150SMatt Helsley */ 958174f150SMatt Helsley if (!freezing(p)) { 968174f150SMatt Helsley rmb(); 978174f150SMatt Helsley if (frozen(p)) 988174f150SMatt Helsley return false; 998174f150SMatt Helsley 1008174f150SMatt Helsley if (!sig_only || should_send_signal(p)) 1018174f150SMatt Helsley set_freeze_flag(p); 1028174f150SMatt Helsley else 1038174f150SMatt Helsley return false; 1048174f150SMatt Helsley } 1058174f150SMatt Helsley 1068174f150SMatt Helsley if (should_send_signal(p)) { 1078174f150SMatt Helsley if (!signal_pending(p)) 1088174f150SMatt Helsley fake_signal_wake_up(p); 1098174f150SMatt Helsley } else if (sig_only) { 1108174f150SMatt Helsley return false; 1118174f150SMatt Helsley } else { 1128174f150SMatt Helsley wake_up_state(p, TASK_INTERRUPTIBLE); 1138174f150SMatt Helsley } 1148174f150SMatt Helsley 1158174f150SMatt Helsley return true; 1168174f150SMatt Helsley } 1178174f150SMatt Helsley 1188174f150SMatt Helsley void cancel_freezing(struct task_struct *p) 1198174f150SMatt Helsley { 1208174f150SMatt Helsley unsigned long flags; 1218174f150SMatt Helsley 1228174f150SMatt Helsley if (freezing(p)) { 1238174f150SMatt Helsley pr_debug(" clean up: %s\n", p->comm); 1248174f150SMatt Helsley clear_freeze_flag(p); 1258174f150SMatt Helsley spin_lock_irqsave(&p->sighand->siglock, flags); 1268174f150SMatt Helsley recalc_sigpending_and_wake(p); 1278174f150SMatt Helsley spin_unlock_irqrestore(&p->sighand->siglock, flags); 1288174f150SMatt Helsley } 1298174f150SMatt Helsley } 130dc52ddc0SMatt Helsley 13100c2e63cSLi Zefan static int __thaw_process(struct task_struct *p) 132dc52ddc0SMatt Helsley { 133dc52ddc0SMatt Helsley if (frozen(p)) { 134dc52ddc0SMatt Helsley p->flags &= ~PF_FROZEN; 135dc52ddc0SMatt Helsley return 1; 136dc52ddc0SMatt Helsley } 137dc52ddc0SMatt Helsley clear_freeze_flag(p); 138dc52ddc0SMatt Helsley return 0; 139dc52ddc0SMatt Helsley } 140dc52ddc0SMatt Helsley 14100c2e63cSLi Zefan /* 14200c2e63cSLi Zefan * Wake up a frozen process 14300c2e63cSLi Zefan * 14400c2e63cSLi Zefan * task_lock() is needed to prevent the race with refrigerator() which may 14500c2e63cSLi Zefan * occur if the freezing of tasks fails. Namely, without the lock, if the 14600c2e63cSLi Zefan * freezing of tasks failed, thaw_tasks() might have run before a task in 14700c2e63cSLi Zefan * refrigerator() could call frozen_process(), in which case the task would be 14800c2e63cSLi Zefan * frozen and no one would thaw it. 14900c2e63cSLi Zefan */ 150dc52ddc0SMatt Helsley int thaw_process(struct task_struct *p) 151dc52ddc0SMatt Helsley { 152dc52ddc0SMatt Helsley task_lock(p); 153dc52ddc0SMatt Helsley if (__thaw_process(p) == 1) { 154dc52ddc0SMatt Helsley task_unlock(p); 155dc52ddc0SMatt Helsley wake_up_process(p); 156dc52ddc0SMatt Helsley return 1; 157dc52ddc0SMatt Helsley } 158dc52ddc0SMatt Helsley task_unlock(p); 159dc52ddc0SMatt Helsley return 0; 160dc52ddc0SMatt Helsley } 161dc52ddc0SMatt Helsley EXPORT_SYMBOL(thaw_process); 162