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> 128a32c441STejun Heo #include <linux/kthread.h> 138174f150SMatt Helsley 14a3201227STejun Heo /* total number of freezing conditions in effect */ 15a3201227STejun Heo atomic_t system_freezing_cnt = ATOMIC_INIT(0); 16a3201227STejun Heo EXPORT_SYMBOL(system_freezing_cnt); 17a3201227STejun Heo 18a3201227STejun Heo /* indicate whether PM freezing is in effect, protected by pm_mutex */ 19a3201227STejun Heo bool pm_freezing; 20a3201227STejun Heo bool pm_nosig_freezing; 21a3201227STejun Heo 220c9af092STejun Heo /* protects freezing and frozen transitions */ 230c9af092STejun Heo static DEFINE_SPINLOCK(freezer_lock); 248174f150SMatt Helsley 25a3201227STejun Heo /** 26a3201227STejun Heo * freezing_slow_path - slow path for testing whether a task needs to be frozen 27a3201227STejun Heo * @p: task to be tested 28a3201227STejun Heo * 29a3201227STejun Heo * This function is called by freezing() if system_freezing_cnt isn't zero 30a3201227STejun Heo * and tests whether @p needs to enter and stay in frozen state. Can be 31a3201227STejun Heo * called under any context. The freezers are responsible for ensuring the 32a3201227STejun Heo * target tasks see the updated state. 33a3201227STejun Heo */ 34a3201227STejun Heo bool freezing_slow_path(struct task_struct *p) 35a3201227STejun Heo { 36a3201227STejun Heo if (p->flags & PF_NOFREEZE) 37a3201227STejun Heo return false; 38a3201227STejun Heo 39a3201227STejun Heo if (pm_nosig_freezing || cgroup_freezing(p)) 40a3201227STejun Heo return true; 41a3201227STejun Heo 4234b087e4STejun Heo if (pm_freezing && !(p->flags & PF_KTHREAD)) 43a3201227STejun Heo return true; 44a3201227STejun Heo 45a3201227STejun Heo return false; 46a3201227STejun Heo } 47a3201227STejun Heo EXPORT_SYMBOL(freezing_slow_path); 48a3201227STejun Heo 498174f150SMatt Helsley /* Refrigerator is place where frozen processes are stored :-). */ 508a32c441STejun Heo bool __refrigerator(bool check_kthr_stop) 518174f150SMatt Helsley { 528174f150SMatt Helsley /* Hmm, should we be allowed to suspend when there are realtime 538174f150SMatt Helsley processes around? */ 54a0acae0eSTejun Heo bool was_frozen = false; 555ece3eaeSTejun Heo long save = current->state; 568174f150SMatt Helsley 578174f150SMatt Helsley pr_debug("%s entered refrigerator\n", current->comm); 588174f150SMatt Helsley 598174f150SMatt Helsley for (;;) { 608174f150SMatt Helsley set_current_state(TASK_UNINTERRUPTIBLE); 615ece3eaeSTejun Heo 625ece3eaeSTejun Heo spin_lock_irq(&freezer_lock); 635ece3eaeSTejun Heo current->flags |= PF_FROZEN; 646907483bSTejun Heo if (!freezing(current) || 658a32c441STejun Heo (check_kthr_stop && kthread_should_stop())) 665ece3eaeSTejun Heo current->flags &= ~PF_FROZEN; 675ece3eaeSTejun Heo spin_unlock_irq(&freezer_lock); 685ece3eaeSTejun Heo 695ece3eaeSTejun Heo if (!(current->flags & PF_FROZEN)) 708174f150SMatt Helsley break; 71a0acae0eSTejun Heo was_frozen = true; 728174f150SMatt Helsley schedule(); 738174f150SMatt Helsley } 746301cb95SThomas Gleixner 758174f150SMatt Helsley pr_debug("%s left refrigerator\n", current->comm); 7650fb4f7fSTejun Heo 7750fb4f7fSTejun Heo /* 7850fb4f7fSTejun Heo * Restore saved task state before returning. The mb'd version 7950fb4f7fSTejun Heo * needs to be used; otherwise, it might silently break 8050fb4f7fSTejun Heo * synchronization which depends on ordered task state change. 8150fb4f7fSTejun Heo */ 8250fb4f7fSTejun Heo set_current_state(save); 83a0acae0eSTejun Heo 84a0acae0eSTejun Heo return was_frozen; 858174f150SMatt Helsley } 86a0acae0eSTejun Heo EXPORT_SYMBOL(__refrigerator); 878174f150SMatt Helsley 888174f150SMatt Helsley static void fake_signal_wake_up(struct task_struct *p) 898174f150SMatt Helsley { 908174f150SMatt Helsley unsigned long flags; 918174f150SMatt Helsley 9237ad8acaSTejun Heo if (lock_task_sighand(p, &flags)) { 93d6cc7685STejun Heo signal_wake_up(p, 0); 9437ad8acaSTejun Heo unlock_task_sighand(p, &flags); 9537ad8acaSTejun Heo } 968174f150SMatt Helsley } 978174f150SMatt Helsley 988174f150SMatt Helsley /** 998174f150SMatt Helsley * freeze_task - send a freeze request to given task 1008174f150SMatt Helsley * @p: task to send the request to 1018174f150SMatt Helsley * 10237f08be1SMarcos Paulo de Souza * If @p is freezing, the freeze request is sent either by sending a fake 10337f08be1SMarcos Paulo de Souza * signal (if it's not a kernel thread) or waking it up (if it's a kernel 10437f08be1SMarcos Paulo de Souza * thread). 105839e3407STejun Heo * 106839e3407STejun Heo * RETURNS: 107839e3407STejun Heo * %false, if @p is not freezing or already frozen; %true, otherwise 1088174f150SMatt Helsley */ 109839e3407STejun Heo bool freeze_task(struct task_struct *p) 1108174f150SMatt Helsley { 1110c9af092STejun Heo unsigned long flags; 1128174f150SMatt Helsley 1130c9af092STejun Heo spin_lock_irqsave(&freezer_lock, flags); 114a3201227STejun Heo if (!freezing(p) || frozen(p)) { 115a3201227STejun Heo spin_unlock_irqrestore(&freezer_lock, flags); 116a3201227STejun Heo return false; 117a3201227STejun Heo } 1188174f150SMatt Helsley 119*5d8f72b5SOleg Nesterov if (!(p->flags & PF_KTHREAD)) 1208174f150SMatt Helsley fake_signal_wake_up(p); 121*5d8f72b5SOleg Nesterov else 1228174f150SMatt Helsley wake_up_state(p, TASK_INTERRUPTIBLE); 123a3201227STejun Heo 1240c9af092STejun Heo spin_unlock_irqrestore(&freezer_lock, flags); 125a3201227STejun Heo return true; 1268174f150SMatt Helsley } 1278174f150SMatt Helsley 128a5be2d0dSTejun Heo void __thaw_task(struct task_struct *p) 129dc52ddc0SMatt Helsley { 1300c9af092STejun Heo unsigned long flags; 131a5be2d0dSTejun Heo 1326907483bSTejun Heo /* 1336907483bSTejun Heo * Clear freezing and kick @p if FROZEN. Clearing is guaranteed to 1346907483bSTejun Heo * be visible to @p as waking up implies wmb. Waking up inside 1356907483bSTejun Heo * freezer_lock also prevents wakeups from leaking outside 1366907483bSTejun Heo * refrigerator. 1376907483bSTejun Heo */ 1380c9af092STejun Heo spin_lock_irqsave(&freezer_lock, flags); 13934b087e4STejun Heo if (frozen(p)) 1406907483bSTejun Heo wake_up_process(p); 1410c9af092STejun Heo spin_unlock_irqrestore(&freezer_lock, flags); 142dc52ddc0SMatt Helsley } 14396ee6d85STejun Heo 14496ee6d85STejun Heo /** 14534b087e4STejun Heo * set_freezable - make %current freezable 14696ee6d85STejun Heo * 14796ee6d85STejun Heo * Mark %current freezable and enter refrigerator if necessary. 14896ee6d85STejun Heo */ 14934b087e4STejun Heo bool set_freezable(void) 15096ee6d85STejun Heo { 15196ee6d85STejun Heo might_sleep(); 15296ee6d85STejun Heo 15396ee6d85STejun Heo /* 15496ee6d85STejun Heo * Modify flags while holding freezer_lock. This ensures the 15596ee6d85STejun Heo * freezer notices that we aren't frozen yet or the freezing 15696ee6d85STejun Heo * condition is visible to try_to_freeze() below. 15796ee6d85STejun Heo */ 15896ee6d85STejun Heo spin_lock_irq(&freezer_lock); 15996ee6d85STejun Heo current->flags &= ~PF_NOFREEZE; 16096ee6d85STejun Heo spin_unlock_irq(&freezer_lock); 16196ee6d85STejun Heo 16296ee6d85STejun Heo return try_to_freeze(); 16396ee6d85STejun Heo } 16434b087e4STejun Heo EXPORT_SYMBOL(set_freezable); 165