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>
# 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>
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
2006-12-06 20:34:37 -08:00
# define FREEZER_KERNEL_THREADS 0
# define FREEZER_USER_SPACE 1
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 ;
}
2007-05-23 13:57:29 -07:00
/*
* freezing is complete , mark current process as frozen
*/
static inline void frozen_process ( void )
{
if ( ! unlikely ( current - > flags & PF_NOFREEZE ) ) {
current - > flags | = PF_FROZEN ;
wmb ( ) ;
}
2007-07-19 01:47:33 -07:00
clear_freeze_flag ( current ) ;
2007-05-23 13:57:29 -07:00
}
2005-04-16 15:20:36 -07:00
/* Refrigerator is place where frozen processes are stored :-). */
2005-06-24 23:13:50 -07:00
void refrigerator ( void )
2005-04-16 15:20:36 -07:00
{
/* Hmm, should we be allowed to suspend when there are realtime
processes around ? */
long save ;
2007-05-23 13:57:24 -07:00
task_lock ( current ) ;
if ( freezing ( current ) ) {
2007-05-23 13:57:29 -07:00
frozen_process ( ) ;
2007-05-23 13:57:24 -07:00
task_unlock ( current ) ;
} else {
task_unlock ( current ) ;
return ;
}
2005-04-16 15:20:36 -07:00
save = current - > state ;
pr_debug ( " %s entered refrigerator \n " , current - > comm ) ;
spin_lock_irq ( & current - > sighand - > siglock ) ;
recalc_sigpending ( ) ; /* We sent fake signal, clean it up */
spin_unlock_irq ( & current - > sighand - > siglock ) ;
2007-05-06 14:50:40 -07:00
for ( ; ; ) {
set_current_state ( TASK_UNINTERRUPTIBLE ) ;
if ( ! frozen ( current ) )
break ;
2005-04-16 15:20:36 -07:00
schedule ( ) ;
2005-09-03 15:56:53 -07:00
}
2005-04-16 15:20:36 -07:00
pr_debug ( " %s left refrigerator \n " , current - > comm ) ;
current - > state = save ;
}
2007-07-19 01:47:33 -07:00
static void freeze_task ( struct task_struct * p )
2006-03-23 03:00:04 -08:00
{
unsigned long flags ;
if ( ! freezing ( p ) ) {
2006-12-13 00:34:30 -08:00
rmb ( ) ;
if ( ! frozen ( p ) ) {
2007-07-19 01:47:33 -07:00
set_freeze_flag ( p ) ;
2006-12-13 00:34:30 -08:00
if ( p - > state = = TASK_STOPPED )
force_sig_specific ( SIGSTOP , p ) ;
spin_lock_irqsave ( & p - > sighand - > siglock , flags ) ;
signal_wake_up ( p , p - > state = = TASK_STOPPED ) ;
spin_unlock_irqrestore ( & p - > sighand - > siglock , flags ) ;
}
2006-03-23 03:00:04 -08:00
}
}
2006-08-05 12:13:42 -07:00
static void cancel_freezing ( struct task_struct * p )
{
unsigned long flags ;
if ( freezing ( p ) ) {
pr_debug ( " clean up: %s \n " , p - > comm ) ;
2007-07-19 01:47:33 -07:00
clear_freeze_flag ( p ) ;
2006-08-05 12:13:42 -07:00
spin_lock_irqsave ( & p - > sighand - > siglock , flags ) ;
2007-05-23 13:57:44 -07:00
recalc_sigpending_and_wake ( p ) ;
2006-08-05 12:13:42 -07:00
spin_unlock_irqrestore ( & p - > sighand - > siglock , flags ) ;
}
}
2006-12-06 20:34:40 -08:00
static unsigned int try_to_freeze_tasks ( int freeze_user_space )
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 ;
2005-06-24 23:13:50 -07:00
2006-12-06 20:34:40 -08:00
end_time = jiffies + TIMEOUT ;
2005-04-16 15:20:36 -07:00
do {
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
2007-07-19 01:47:33 -07:00
if ( freeze_user_space ) {
if ( p - > state = = TASK_TRACED & &
frozen ( p - > parent ) ) {
cancel_freezing ( p ) ;
continue ;
}
/*
* Kernel threads should not have TIF_FREEZE set
* at this point , so we must ensure that either
* p - > mm is not NULL * and * PF_BORROWED_MM is
* unset , or TIF_FRREZE is left unset .
* The task_lock ( ) is necessary to prevent races
* with exit_mm ( ) or use_mm ( ) / unuse_mm ( ) from
* occuring .
*/
task_lock ( p ) ;
if ( ! p - > mm | | ( p - > flags & PF_BORROWED_MM ) ) {
task_unlock ( p ) ;
continue ;
}
freeze_task ( p ) ;
task_unlock ( p ) ;
} else {
freeze_task ( p ) ;
2006-08-05 12:13:42 -07:00
}
2007-05-23 13:57:25 -07:00
if ( ! freezer_should_skip ( p ) )
todo + + ;
2005-04-16 15:20:36 -07:00
} while_each_thread ( g , p ) ;
read_unlock ( & tasklist_lock ) ;
yield ( ) ; /* Yield is okay here */
2006-12-06 20:34:40 -08:00
if ( todo & & time_after ( jiffies , end_time ) )
2005-09-03 15:57:05 -07:00
break ;
2006-12-06 20:34:40 -08:00
} while ( todo ) ;
2005-06-24 23:13:50 -07:00
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-07-19 01:47:33 -07:00
printk ( KERN_ERR " Freezing of %s timed out after %d seconds "
2006-12-06 20:34:40 -08:00
" (%d tasks refusing to freeze): \n " ,
2007-07-19 01:47:33 -07:00
freeze_user_space ? " user space " : " tasks " ,
2006-12-06 20:34:40 -08:00
TIMEOUT / HZ , todo ) ;
2007-07-19 01:47:26 -07:00
show_state ( ) ;
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 ) )
2006-12-06 20:34:26 -08:00
printk ( KERN_ERR " %s \n " , p - > comm ) ;
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 ) ;
}
2006-12-06 20:34:40 -08:00
return todo ;
}
/**
* freeze_processes - tell processes to enter the refrigerator
*
* Returns 0 on success , or the number of processes that didn ' t freeze ,
* although they were told to .
*/
int freeze_processes ( void )
{
unsigned int nr_unfrozen ;
printk ( " Stopping tasks ... " ) ;
nr_unfrozen = try_to_freeze_tasks ( FREEZER_USER_SPACE ) ;
if ( nr_unfrozen )
return nr_unfrozen ;
sys_sync ( ) ;
nr_unfrozen = try_to_freeze_tasks ( FREEZER_KERNEL_THREADS ) ;
if ( nr_unfrozen )
return nr_unfrozen ;
2006-12-06 20:34:26 -08:00
printk ( " done. \n " ) ;
2005-04-16 15:20:36 -07:00
BUG_ON ( in_atomic ( ) ) ;
return 0 ;
}
2006-12-06 20:34:37 -08:00
static void thaw_tasks ( int thaw_user_space )
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
2007-07-19 01:47:33 -07:00
if ( ! p - > mm = = thaw_user_space )
2006-12-06 20:34:37 -08:00
continue ;
2005-04-16 15:20:36 -07:00
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 )
{
printk ( " Restarting tasks ... " ) ;
thaw_tasks ( FREEZER_KERNEL_THREADS ) ;
thaw_tasks ( FREEZER_USER_SPACE ) ;
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
}
EXPORT_SYMBOL ( refrigerator ) ;