Skip to content
Snippets Groups Projects
process.c 4.32 KiB
Newer Older
  • Learn to ignore specific revisions
  • Linus Torvalds's avatar
    Linus Torvalds committed
    /*
     * drivers/power/process.c - Functions for starting/stopping processes on 
     *                           suspend transitions.
     *
     * Originally from swsusp.
     */
    
    
    #undef DEBUG
    
    #include <linux/interrupt.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #include <linux/suspend.h>
    #include <linux/module.h>
    
    #include <linux/syscalls.h>
    
    #include <linux/delay.h>
    
    #include <linux/workqueue.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    /* 
     * Timeout for stopping processes
     */
    
    #define TIMEOUT	(20 * HZ)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    
    static int try_to_freeze_tasks(bool user_only)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	struct task_struct *g, *p;
    
    	unsigned long end_time;
    	unsigned int todo;
    
    	bool wq_busy = false;
    
    	struct timeval start, end;
    
    	u64 elapsed_csecs64;
    
    	unsigned int elapsed_csecs;
    
    
    	do_gettimeofday(&start);
    
    	end_time = jiffies + TIMEOUT;
    
    		freeze_workqueues_begin();
    
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		read_lock(&tasklist_lock);
    		do_each_thread(g, p) {
    
    			if (p == current || !freeze_task(p))
    
    			/*
    			 * Now that we've done set_freeze_flag, don't
    			 * perturb a task in TASK_STOPPED or TASK_TRACED.
    			 * It is "frozen enough".  If the task does wake
    			 * up, it will immediately call try_to_freeze.
    
    			 *
    			 * Because freeze_task() goes through p's
    			 * scheduler lock after setting TIF_FREEZE, it's
    			 * guaranteed that either we see TASK_RUNNING or
    			 * try_to_stop() after schedule() in ptrace/signal
    			 * stop sees TIF_FREEZE.
    
    			 */
    			if (!task_is_stopped_or_traced(p) &&
    			    !freezer_should_skip(p))
    
    				todo++;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		} while_each_thread(g, p);
    		read_unlock(&tasklist_lock);
    
    		if (!user_only) {
    
    			wq_busy = freeze_workqueues_busy();
    			todo += wq_busy;
    		}
    
    
    		if (!todo || time_after(jiffies, end_time))
    
    		/*
    		 * We need to retry, but first give the freezing tasks some
    		 * time to enter the regrigerator.
    		 */
    		msleep(10);
    	}
    
    	do_gettimeofday(&end);
    	elapsed_csecs64 = timeval_to_ns(&end) - timeval_to_ns(&start);
    	do_div(elapsed_csecs64, NSEC_PER_SEC / 100);
    	elapsed_csecs = elapsed_csecs64;
    
    
    	if (todo) {
    
    		printk(KERN_ERR "Freezing of tasks %s after %d.%02d seconds "
    
    		       "(%d tasks refusing to freeze, wq_busy=%d):\n",
    
    		       wakeup ? "aborted" : "failed",
    
    		       elapsed_csecs / 100, elapsed_csecs % 100,
    		       todo - wq_busy, wq_busy);
    
    
    		read_lock(&tasklist_lock);
    
    		do_each_thread(g, p) {
    
    			if (!wakeup && !freezer_should_skip(p) &&
    
    			    p != current && freezing(p) && !frozen(p))
    
    		} while_each_thread(g, p);
    
    		read_unlock(&tasklist_lock);
    
    	} else {
    		printk("(elapsed %d.%02d seconds) ", elapsed_csecs / 100,
    			elapsed_csecs % 100);
    
    	return todo ? -EBUSY : 0;
    
     * freeze_processes - Signal user space processes to enter the refrigerator.
    
     *
     * On success, returns 0.  On failure, -errno and system is fully thawed.
    
     */
    int freeze_processes(void)
    {
    
    	if (!pm_freezing)
    		atomic_inc(&system_freezing_cnt);
    
    
    	printk("Freezing user space processes ... ");
    
    	error = try_to_freeze_tasks(true);
    
    	if (!error) {
    		printk("done.");
    		oom_killer_disable();
    	}
    	printk("\n");
    	BUG_ON(in_atomic());
    
    
    	if (error)
    		thaw_processes();
    
    	return error;
    }
    
    /**
     * freeze_kernel_threads - Make freezable kernel threads go to the refrigerator.
    
     *
     * On success, returns 0.  On failure, -errno and system is fully thawed.
    
     */
    int freeze_kernel_threads(void)
    {
    	int error;
    
    	printk("Freezing remaining freezable tasks ... ");
    
    	error = try_to_freeze_tasks(false);
    
    	printk("\n");
    
    	if (error)
    		thaw_processes();
    
    	return error;
    
    void thaw_processes(void)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    	struct task_struct *g, *p;
    
    
    	if (pm_freezing)
    		atomic_dec(&system_freezing_cnt);
    	pm_freezing = false;
    	pm_nosig_freezing = false;
    
    
    	oom_killer_enable();
    
    	printk("Restarting tasks ... ");
    
    	thaw_workqueues();
    
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	read_lock(&tasklist_lock);
    
    	do_each_thread(g, p) {
    
    	} while_each_thread(g, p);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	read_unlock(&tasklist_lock);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	schedule();
    
    void thaw_kernel_threads(void)
    {
    	struct task_struct *g, *p;
    
    	pm_nosig_freezing = false;
    	printk("Restarting kernel threads ... ");
    
    	thaw_workqueues();
    
    	read_lock(&tasklist_lock);
    	do_each_thread(g, p) {
    		if (p->flags & (PF_KTHREAD | PF_WQ_WORKER))
    			__thaw_task(p);
    	} while_each_thread(g, p);
    	read_unlock(&tasklist_lock);
    
    	schedule();
    	printk("done.\n");
    }