2005-04-16 15:20:36 -07:00
/*
* drivers / power / process . c - Functions for starting / stopping processes on
* suspend transitions .
*
* Originally from swsusp .
*/
# undef DEBUG
# include <linux/interrupt.h>
2009-09-21 17:03:09 -07:00
# include <linux/oom.h>
2005-04-16 15:20:36 -07:00
# include <linux/suspend.h>
# include <linux/module.h>
2006-03-23 03:00:04 -08:00
# include <linux/syscalls.h>
2006-12-06 20:34:23 -08:00
# include <linux/freezer.h>
2009-10-08 22:47:30 +02:00
# include <linux/delay.h>
2010-06-29 10:07:12 +02:00
# include <linux/workqueue.h>
2005-04-16 15:20:36 -07:00
/*
* Timeout for stopping processes
*/
2006-03-23 03:00:04 -08:00
# define TIMEOUT (20 * HZ)
2005-04-16 15:20:36 -07:00
static inline int freezeable ( struct task_struct * p )
{
2007-05-08 00:24:01 -07:00
if ( ( p = = current ) | |
2005-04-16 15:20:36 -07:00
( p - > flags & PF_NOFREEZE ) | |
2007-05-08 00:24:01 -07:00
( p - > exit_state ! = 0 ) )
2005-04-16 15:20:36 -07:00
return 0 ;
return 1 ;
}
2008-06-11 22:04:29 +02:00
static int try_to_freeze_tasks ( bool sig_only )
2005-04-16 15:20:36 -07:00
{
struct task_struct * g , * p ;
2006-12-06 20:34:40 -08:00
unsigned long end_time ;
unsigned int todo ;
2010-06-29 10:07:12 +02:00
bool wq_busy = false ;
2007-10-18 03:04:49 -07:00
struct timeval start , end ;
2008-07-23 21:28:44 -07:00
u64 elapsed_csecs64 ;
2007-10-18 03:04:49 -07:00
unsigned int elapsed_csecs ;
do_gettimeofday ( & start ) ;
2005-06-24 23:13:50 -07:00
2006-12-06 20:34:40 -08:00
end_time = jiffies + TIMEOUT ;
2010-06-29 10:07:12 +02:00
if ( ! sig_only )
freeze_workqueues_begin ( ) ;
2009-10-08 22:47:30 +02:00
while ( true ) {
2006-12-06 20:34:40 -08:00
todo = 0 ;
2005-04-16 15:20:36 -07:00
read_lock ( & tasklist_lock ) ;
do_each_thread ( g , p ) {
2007-07-19 01:47:33 -07:00
if ( frozen ( p ) | | ! freezeable ( p ) )
2005-04-16 15:20:36 -07:00
continue ;
2006-12-06 20:34:40 -08:00
2008-06-11 22:04:29 +02:00
if ( ! freeze_task ( p , sig_only ) )
2007-10-18 03:04:46 -07:00
continue ;
2008-03-03 20:22:05 -08:00
/*
* 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 ) )
2007-05-23 13:57:25 -07:00
todo + + ;
2005-04-16 15:20:36 -07:00
} while_each_thread ( g , p ) ;
read_unlock ( & tasklist_lock ) ;
2010-06-29 10:07:12 +02:00
if ( ! sig_only ) {
wq_busy = freeze_workqueues_busy ( ) ;
todo + = wq_busy ;
}
2009-10-08 22:47:30 +02:00
if ( ! todo | | time_after ( jiffies , end_time ) )
2005-09-03 15:57:05 -07:00
break ;
2009-10-08 22:47:30 +02:00
/*
* We need to retry , but first give the freezing tasks some
* time to enter the regrigerator .
*/
msleep ( 10 ) ;
}
2005-06-24 23:13:50 -07:00
2007-10-18 03:04:49 -07:00
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 ;
2005-09-03 15:57:05 -07:00
if ( todo ) {
2006-12-06 20:34:40 -08:00
/* 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 .
*/
2006-12-06 20:34:26 -08:00
printk ( " \n " ) ;
2007-10-18 03:04:49 -07:00
printk ( KERN_ERR " Freezing of tasks failed after %d.%02d seconds "
2010-06-29 10:07:12 +02:00
" (%d tasks refusing to freeze, wq_busy=%d): \n " ,
elapsed_csecs / 100 , elapsed_csecs % 100 ,
todo - wq_busy , wq_busy ) ;
thaw_workqueues ( ) ;
2005-09-03 15:57:05 -07:00
read_lock ( & tasklist_lock ) ;
2006-03-23 03:00:04 -08:00
do_each_thread ( g , p ) {
2007-05-23 13:57:24 -07:00
task_lock ( p ) ;
2007-07-19 01:47:33 -07:00
if ( freezing ( p ) & & ! freezer_should_skip ( p ) )
2010-03-10 22:59:13 +01:00
sched_show_task ( p ) ;
2006-08-05 12:13:42 -07:00
cancel_freezing ( p ) ;
2007-05-23 13:57:24 -07:00
task_unlock ( p ) ;
2006-03-23 03:00:04 -08:00
} while_each_thread ( g , p ) ;
2005-09-03 15:57:05 -07:00
read_unlock ( & tasklist_lock ) ;
2007-10-18 03:04:49 -07:00
} else {
printk ( " (elapsed %d.%02d seconds) " , elapsed_csecs / 100 ,
elapsed_csecs % 100 ) ;
2005-09-03 15:57:05 -07:00
}
2007-07-19 01:47:34 -07:00
return todo ? - EBUSY : 0 ;
2006-12-06 20:34:40 -08:00
}
/**
* freeze_processes - tell processes to enter the refrigerator
*/
int freeze_processes ( void )
{
2007-07-19 01:47:34 -07:00
int error ;
2006-12-06 20:34:40 -08:00
2007-10-18 03:04:48 -07:00
printk ( " Freezing user space processes ... " ) ;
2008-06-11 22:04:29 +02:00
error = try_to_freeze_tasks ( true ) ;
2007-07-19 01:47:34 -07:00
if ( error )
2007-10-18 03:04:48 -07:00
goto Exit ;
printk ( " done. \n " ) ;
2006-12-06 20:34:40 -08:00
2007-10-18 03:04:48 -07:00
printk ( " Freezing remaining freezable tasks ... " ) ;
2008-06-11 22:04:29 +02:00
error = try_to_freeze_tasks ( false ) ;
2007-07-19 01:47:34 -07:00
if ( error )
2007-10-18 03:04:48 -07:00
goto Exit ;
printk ( " done. " ) ;
2009-06-16 15:32:41 -07:00
oom_killer_disable ( ) ;
2007-10-18 03:04:48 -07:00
Exit :
2005-04-16 15:20:36 -07:00
BUG_ON ( in_atomic ( ) ) ;
2007-10-18 03:04:48 -07:00
printk ( " \n " ) ;
2009-06-16 15:32:41 -07:00
2007-10-18 03:04:48 -07:00
return error ;
2005-04-16 15:20:36 -07:00
}
2008-06-11 22:04:29 +02:00
static void thaw_tasks ( bool nosig_only )
2005-04-16 15:20:36 -07:00
{
struct task_struct * g , * p ;
read_lock ( & tasklist_lock ) ;
2006-12-06 20:34:37 -08:00
do_each_thread ( g , p ) {
if ( ! freezeable ( p ) )
continue ;
2006-12-06 20:34:28 -08:00
2008-06-11 22:04:29 +02:00
if ( nosig_only & & should_send_signal ( p ) )
2006-12-06 20:34:37 -08:00
continue ;
2005-04-16 15:20:36 -07:00
2010-03-26 23:51:44 +01:00
if ( cgroup_freezing_or_frozen ( p ) )
2008-10-18 20:27:22 -07:00
continue ;
2007-05-23 13:57:25 -07:00
thaw_process ( p ) ;
2006-12-06 20:34:37 -08:00
} while_each_thread ( g , p ) ;
2005-04-16 15:20:36 -07:00
read_unlock ( & tasklist_lock ) ;
2006-12-06 20:34:37 -08:00
}
void thaw_processes ( void )
{
2009-06-16 15:32:41 -07:00
oom_killer_enable ( ) ;
2006-12-06 20:34:37 -08:00
printk ( " Restarting tasks ... " ) ;
2010-06-29 10:07:12 +02:00
thaw_workqueues ( ) ;
2008-06-11 22:04:29 +02:00
thaw_tasks ( true ) ;
thaw_tasks ( false ) ;
2005-04-16 15:20:36 -07:00
schedule ( ) ;
2006-12-06 20:34:26 -08:00
printk ( " done. \n " ) ;
2005-04-16 15:20:36 -07:00
}