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 18*55f2503cSPingfan Liu /* indicate whether PM freezing is in effect, protected by 19*55f2503cSPingfan Liu * system_transition_mutex 20*55f2503cSPingfan Liu */ 21a3201227STejun Heo bool pm_freezing; 22a3201227STejun Heo bool pm_nosig_freezing; 23a3201227STejun Heo 2485fbd722STejun Heo /* 2585fbd722STejun Heo * Temporary export for the deadlock workaround in ata_scsi_hotplug(). 2685fbd722STejun Heo * Remove once the hack becomes unnecessary. 2785fbd722STejun Heo */ 2885fbd722STejun Heo EXPORT_SYMBOL_GPL(pm_freezing); 2985fbd722STejun Heo 300c9af092STejun Heo /* protects freezing and frozen transitions */ 310c9af092STejun Heo static DEFINE_SPINLOCK(freezer_lock); 328174f150SMatt Helsley 33a3201227STejun Heo /** 34a3201227STejun Heo * freezing_slow_path - slow path for testing whether a task needs to be frozen 35a3201227STejun Heo * @p: task to be tested 36a3201227STejun Heo * 37a3201227STejun Heo * This function is called by freezing() if system_freezing_cnt isn't zero 38a3201227STejun Heo * and tests whether @p needs to enter and stay in frozen state. Can be 39a3201227STejun Heo * called under any context. The freezers are responsible for ensuring the 40a3201227STejun Heo * target tasks see the updated state. 41a3201227STejun Heo */ 42a3201227STejun Heo bool freezing_slow_path(struct task_struct *p) 43a3201227STejun Heo { 442b44c4dbSColin Cross if (p->flags & (PF_NOFREEZE | PF_SUSPEND_TASK)) 45a3201227STejun Heo return false; 46a3201227STejun Heo 47a34c80a7SMichal Hocko if (test_tsk_thread_flag(p, TIF_MEMDIE)) 4851fae6daSCong Wang return false; 4951fae6daSCong Wang 50a3201227STejun Heo if (pm_nosig_freezing || cgroup_freezing(p)) 51a3201227STejun Heo return true; 52a3201227STejun Heo 5334b087e4STejun Heo if (pm_freezing && !(p->flags & PF_KTHREAD)) 54a3201227STejun Heo return true; 55a3201227STejun Heo 56a3201227STejun Heo return false; 57a3201227STejun Heo } 58a3201227STejun Heo EXPORT_SYMBOL(freezing_slow_path); 59a3201227STejun Heo 608174f150SMatt Helsley /* Refrigerator is place where frozen processes are stored :-). */ 618a32c441STejun Heo bool __refrigerator(bool check_kthr_stop) 628174f150SMatt Helsley { 638174f150SMatt Helsley /* Hmm, should we be allowed to suspend when there are realtime 648174f150SMatt Helsley processes around? */ 65a0acae0eSTejun Heo bool was_frozen = false; 665ece3eaeSTejun Heo long save = current->state; 678174f150SMatt Helsley 688174f150SMatt Helsley pr_debug("%s entered refrigerator\n", current->comm); 698174f150SMatt Helsley 708174f150SMatt Helsley for (;;) { 718174f150SMatt Helsley set_current_state(TASK_UNINTERRUPTIBLE); 725ece3eaeSTejun Heo 735ece3eaeSTejun Heo spin_lock_irq(&freezer_lock); 745ece3eaeSTejun Heo current->flags |= PF_FROZEN; 756907483bSTejun Heo if (!freezing(current) || 768a32c441STejun Heo (check_kthr_stop && kthread_should_stop())) 775ece3eaeSTejun Heo current->flags &= ~PF_FROZEN; 785ece3eaeSTejun Heo spin_unlock_irq(&freezer_lock); 795ece3eaeSTejun Heo 805ece3eaeSTejun Heo if (!(current->flags & PF_FROZEN)) 818174f150SMatt Helsley break; 82a0acae0eSTejun Heo was_frozen = true; 838174f150SMatt Helsley schedule(); 848174f150SMatt Helsley } 856301cb95SThomas Gleixner 868174f150SMatt Helsley pr_debug("%s left refrigerator\n", current->comm); 8750fb4f7fSTejun Heo 8850fb4f7fSTejun Heo /* 8950fb4f7fSTejun Heo * Restore saved task state before returning. The mb'd version 9050fb4f7fSTejun Heo * needs to be used; otherwise, it might silently break 9150fb4f7fSTejun Heo * synchronization which depends on ordered task state change. 9250fb4f7fSTejun Heo */ 9350fb4f7fSTejun Heo set_current_state(save); 94a0acae0eSTejun Heo 95a0acae0eSTejun Heo return was_frozen; 968174f150SMatt Helsley } 97a0acae0eSTejun Heo EXPORT_SYMBOL(__refrigerator); 988174f150SMatt Helsley 998174f150SMatt Helsley static void fake_signal_wake_up(struct task_struct *p) 1008174f150SMatt Helsley { 1018174f150SMatt Helsley unsigned long flags; 1028174f150SMatt Helsley 10337ad8acaSTejun Heo if (lock_task_sighand(p, &flags)) { 104d6cc7685STejun Heo signal_wake_up(p, 0); 10537ad8acaSTejun Heo unlock_task_sighand(p, &flags); 10637ad8acaSTejun Heo } 1078174f150SMatt Helsley } 1088174f150SMatt Helsley 1098174f150SMatt Helsley /** 1108174f150SMatt Helsley * freeze_task - send a freeze request to given task 1118174f150SMatt Helsley * @p: task to send the request to 1128174f150SMatt Helsley * 11337f08be1SMarcos Paulo de Souza * If @p is freezing, the freeze request is sent either by sending a fake 11437f08be1SMarcos Paulo de Souza * signal (if it's not a kernel thread) or waking it up (if it's a kernel 11537f08be1SMarcos Paulo de Souza * thread). 116839e3407STejun Heo * 117839e3407STejun Heo * RETURNS: 118839e3407STejun Heo * %false, if @p is not freezing or already frozen; %true, otherwise 1198174f150SMatt Helsley */ 120839e3407STejun Heo bool freeze_task(struct task_struct *p) 1218174f150SMatt Helsley { 1220c9af092STejun Heo unsigned long flags; 1238174f150SMatt Helsley 124613f5d13SColin Cross /* 125613f5d13SColin Cross * This check can race with freezer_do_not_count, but worst case that 126613f5d13SColin Cross * will result in an extra wakeup being sent to the task. It does not 127613f5d13SColin Cross * race with freezer_count(), the barriers in freezer_count() and 128613f5d13SColin Cross * freezer_should_skip() ensure that either freezer_count() sees 129613f5d13SColin Cross * freezing == true in try_to_freeze() and freezes, or 130613f5d13SColin Cross * freezer_should_skip() sees !PF_FREEZE_SKIP and freezes the task 131613f5d13SColin Cross * normally. 132613f5d13SColin Cross */ 133613f5d13SColin Cross if (freezer_should_skip(p)) 134613f5d13SColin Cross return false; 135613f5d13SColin Cross 1360c9af092STejun Heo spin_lock_irqsave(&freezer_lock, flags); 137a3201227STejun Heo if (!freezing(p) || frozen(p)) { 138a3201227STejun Heo spin_unlock_irqrestore(&freezer_lock, flags); 139a3201227STejun Heo return false; 140a3201227STejun Heo } 1418174f150SMatt Helsley 1425d8f72b5SOleg Nesterov if (!(p->flags & PF_KTHREAD)) 1438174f150SMatt Helsley fake_signal_wake_up(p); 1445d8f72b5SOleg Nesterov else 1458174f150SMatt Helsley wake_up_state(p, TASK_INTERRUPTIBLE); 146a3201227STejun Heo 1470c9af092STejun Heo spin_unlock_irqrestore(&freezer_lock, flags); 148a3201227STejun Heo return true; 1498174f150SMatt Helsley } 1508174f150SMatt Helsley 151a5be2d0dSTejun Heo void __thaw_task(struct task_struct *p) 152dc52ddc0SMatt Helsley { 1530c9af092STejun Heo unsigned long flags; 154a5be2d0dSTejun Heo 1550c9af092STejun Heo spin_lock_irqsave(&freezer_lock, flags); 15634b087e4STejun Heo if (frozen(p)) 1576907483bSTejun Heo wake_up_process(p); 1580c9af092STejun Heo spin_unlock_irqrestore(&freezer_lock, flags); 159dc52ddc0SMatt Helsley } 16096ee6d85STejun Heo 16196ee6d85STejun Heo /** 16234b087e4STejun Heo * set_freezable - make %current freezable 16396ee6d85STejun Heo * 16496ee6d85STejun Heo * Mark %current freezable and enter refrigerator if necessary. 16596ee6d85STejun Heo */ 16634b087e4STejun Heo bool set_freezable(void) 16796ee6d85STejun Heo { 16896ee6d85STejun Heo might_sleep(); 16996ee6d85STejun Heo 17096ee6d85STejun Heo /* 17196ee6d85STejun Heo * Modify flags while holding freezer_lock. This ensures the 17296ee6d85STejun Heo * freezer notices that we aren't frozen yet or the freezing 17396ee6d85STejun Heo * condition is visible to try_to_freeze() below. 17496ee6d85STejun Heo */ 17596ee6d85STejun Heo spin_lock_irq(&freezer_lock); 17696ee6d85STejun Heo current->flags &= ~PF_NOFREEZE; 17796ee6d85STejun Heo spin_unlock_irq(&freezer_lock); 17896ee6d85STejun Heo 17996ee6d85STejun Heo return try_to_freeze(); 18096ee6d85STejun Heo } 18134b087e4STejun Heo EXPORT_SYMBOL(set_freezable); 182