Freezing isn't exactly the most latency sensitive operation and there's no reason to burn cpu cycles and power waiting for it to complete. msleep(10) instead of yield(). This should improve reliability of emergency hibernation. [rjw: Modified the comment next to the msleep(10).] Signed-off-by: Tejun Heo <tj@kernel.org> Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
		
			
				
	
	
		
			167 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			167 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * drivers/power/process.c - Functions for starting/stopping processes on 
 | |
|  *                           suspend transitions.
 | |
|  *
 | |
|  * Originally from swsusp.
 | |
|  */
 | |
| 
 | |
| 
 | |
| #undef DEBUG
 | |
| 
 | |
| #include <linux/interrupt.h>
 | |
| #include <linux/oom.h>
 | |
| #include <linux/suspend.h>
 | |
| #include <linux/module.h>
 | |
| #include <linux/syscalls.h>
 | |
| #include <linux/freezer.h>
 | |
| #include <linux/delay.h>
 | |
| 
 | |
| /* 
 | |
|  * Timeout for stopping processes
 | |
|  */
 | |
| #define TIMEOUT	(20 * HZ)
 | |
| 
 | |
| static inline int freezeable(struct task_struct * p)
 | |
| {
 | |
| 	if ((p == current) ||
 | |
| 	    (p->flags & PF_NOFREEZE) ||
 | |
| 	    (p->exit_state != 0))
 | |
| 		return 0;
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static int try_to_freeze_tasks(bool sig_only)
 | |
| {
 | |
| 	struct task_struct *g, *p;
 | |
| 	unsigned long end_time;
 | |
| 	unsigned int todo;
 | |
| 	struct timeval start, end;
 | |
| 	u64 elapsed_csecs64;
 | |
| 	unsigned int elapsed_csecs;
 | |
| 
 | |
| 	do_gettimeofday(&start);
 | |
| 
 | |
| 	end_time = jiffies + TIMEOUT;
 | |
| 	while (true) {
 | |
| 		todo = 0;
 | |
| 		read_lock(&tasklist_lock);
 | |
| 		do_each_thread(g, p) {
 | |
| 			if (frozen(p) || !freezeable(p))
 | |
| 				continue;
 | |
| 
 | |
| 			if (!freeze_task(p, sig_only))
 | |
| 				continue;
 | |
| 
 | |
| 			/*
 | |
| 			 * 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.
 | |
| 			 */
 | |
| 			if (!task_is_stopped_or_traced(p) &&
 | |
| 			    !freezer_should_skip(p))
 | |
| 				todo++;
 | |
| 		} while_each_thread(g, p);
 | |
| 		read_unlock(&tasklist_lock);
 | |
| 		if (!todo || time_after(jiffies, end_time))
 | |
| 			break;
 | |
| 
 | |
| 		/*
 | |
| 		 * 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) {
 | |
| 		/* This does not unfreeze processes that are already frozen
 | |
| 		 * (we have slightly ugly calling convention in that respect,
 | |
| 		 * and caller must call thaw_processes() if something fails),
 | |
| 		 * but it cleans up leftover PF_FREEZE requests.
 | |
| 		 */
 | |
| 		printk("\n");
 | |
| 		printk(KERN_ERR "Freezing of tasks failed after %d.%02d seconds "
 | |
| 				"(%d tasks refusing to freeze):\n",
 | |
| 				elapsed_csecs / 100, elapsed_csecs % 100, todo);
 | |
| 		show_state();
 | |
| 		read_lock(&tasklist_lock);
 | |
| 		do_each_thread(g, p) {
 | |
| 			task_lock(p);
 | |
| 			if (freezing(p) && !freezer_should_skip(p))
 | |
| 				printk(KERN_ERR " %s\n", p->comm);
 | |
| 			cancel_freezing(p);
 | |
| 			task_unlock(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 - tell processes to enter the refrigerator
 | |
|  */
 | |
| int freeze_processes(void)
 | |
| {
 | |
| 	int error;
 | |
| 
 | |
| 	printk("Freezing user space processes ... ");
 | |
| 	error = try_to_freeze_tasks(true);
 | |
| 	if (error)
 | |
| 		goto Exit;
 | |
| 	printk("done.\n");
 | |
| 
 | |
| 	printk("Freezing remaining freezable tasks ... ");
 | |
| 	error = try_to_freeze_tasks(false);
 | |
| 	if (error)
 | |
| 		goto Exit;
 | |
| 	printk("done.");
 | |
| 
 | |
| 	oom_killer_disable();
 | |
|  Exit:
 | |
| 	BUG_ON(in_atomic());
 | |
| 	printk("\n");
 | |
| 
 | |
| 	return error;
 | |
| }
 | |
| 
 | |
| static void thaw_tasks(bool nosig_only)
 | |
| {
 | |
| 	struct task_struct *g, *p;
 | |
| 
 | |
| 	read_lock(&tasklist_lock);
 | |
| 	do_each_thread(g, p) {
 | |
| 		if (!freezeable(p))
 | |
| 			continue;
 | |
| 
 | |
| 		if (nosig_only && should_send_signal(p))
 | |
| 			continue;
 | |
| 
 | |
| 		if (cgroup_frozen(p))
 | |
| 			continue;
 | |
| 
 | |
| 		thaw_process(p);
 | |
| 	} while_each_thread(g, p);
 | |
| 	read_unlock(&tasklist_lock);
 | |
| }
 | |
| 
 | |
| void thaw_processes(void)
 | |
| {
 | |
| 	oom_killer_enable();
 | |
| 
 | |
| 	printk("Restarting tasks ... ");
 | |
| 	thaw_tasks(true);
 | |
| 	thaw_tasks(false);
 | |
| 	schedule();
 | |
| 	printk("done.\n");
 | |
| }
 | |
| 
 |