2006-12-07 04:34:23 +00:00
|
|
|
/* Freezer declarations */
|
|
|
|
|
2006-12-10 10:18:58 +00:00
|
|
|
#include <linux/sched.h>
|
|
|
|
|
2006-12-07 04:34:23 +00:00
|
|
|
#ifdef CONFIG_PM
|
|
|
|
/*
|
|
|
|
* Check if a process has been frozen
|
|
|
|
*/
|
|
|
|
static inline int frozen(struct task_struct *p)
|
|
|
|
{
|
|
|
|
return p->flags & PF_FROZEN;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check if there is a request to freeze a process
|
|
|
|
*/
|
|
|
|
static inline int freezing(struct task_struct *p)
|
|
|
|
{
|
2006-12-13 08:34:30 +00:00
|
|
|
return test_tsk_thread_flag(p, TIF_FREEZE);
|
2006-12-07 04:34:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Request that a process be frozen
|
|
|
|
*/
|
|
|
|
static inline void freeze(struct task_struct *p)
|
|
|
|
{
|
2006-12-13 08:34:30 +00:00
|
|
|
set_tsk_thread_flag(p, TIF_FREEZE);
|
2006-12-07 04:34:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Sometimes we may need to cancel the previous 'freeze' request
|
|
|
|
*/
|
|
|
|
static inline void do_not_freeze(struct task_struct *p)
|
|
|
|
{
|
2006-12-13 08:34:30 +00:00
|
|
|
clear_tsk_thread_flag(p, TIF_FREEZE);
|
2006-12-07 04:34:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Wake up a frozen process
|
2007-05-23 20:57:24 +00:00
|
|
|
*
|
|
|
|
* task_lock() is taken to prevent the race with refrigerator() which may
|
|
|
|
* occur if the freezing of tasks fails. Namely, without the lock, if the
|
|
|
|
* freezing of tasks failed, thaw_tasks() might have run before a task in
|
|
|
|
* refrigerator() could call frozen_process(), in which case the task would be
|
|
|
|
* frozen and no one would thaw it.
|
2006-12-07 04:34:23 +00:00
|
|
|
*/
|
|
|
|
static inline int thaw_process(struct task_struct *p)
|
|
|
|
{
|
2007-05-23 20:57:24 +00:00
|
|
|
task_lock(p);
|
2006-12-07 04:34:23 +00:00
|
|
|
if (frozen(p)) {
|
|
|
|
p->flags &= ~PF_FROZEN;
|
2007-05-23 20:57:24 +00:00
|
|
|
task_unlock(p);
|
2006-12-07 04:34:23 +00:00
|
|
|
wake_up_process(p);
|
|
|
|
return 1;
|
|
|
|
}
|
2007-05-23 20:57:24 +00:00
|
|
|
clear_tsk_thread_flag(p, TIF_FREEZE);
|
|
|
|
task_unlock(p);
|
2006-12-07 04:34:23 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* freezing is complete, mark process as frozen
|
|
|
|
*/
|
|
|
|
static inline void frozen_process(struct task_struct *p)
|
|
|
|
{
|
2006-12-13 08:34:30 +00:00
|
|
|
p->flags |= PF_FROZEN;
|
|
|
|
wmb();
|
|
|
|
clear_tsk_thread_flag(p, TIF_FREEZE);
|
2006-12-07 04:34:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
extern void refrigerator(void);
|
|
|
|
extern int freeze_processes(void);
|
2006-12-07 04:34:37 +00:00
|
|
|
extern void thaw_processes(void);
|
2006-12-07 04:34:23 +00:00
|
|
|
|
|
|
|
static inline int try_to_freeze(void)
|
|
|
|
{
|
|
|
|
if (freezing(current)) {
|
|
|
|
refrigerator();
|
|
|
|
return 1;
|
|
|
|
} else
|
|
|
|
return 0;
|
|
|
|
}
|
2006-12-07 04:34:28 +00:00
|
|
|
|
|
|
|
extern void thaw_some_processes(int all);
|
|
|
|
|
2006-12-07 04:34:23 +00:00
|
|
|
#else
|
|
|
|
static inline int frozen(struct task_struct *p) { return 0; }
|
|
|
|
static inline int freezing(struct task_struct *p) { return 0; }
|
|
|
|
static inline void freeze(struct task_struct *p) { BUG(); }
|
|
|
|
static inline int thaw_process(struct task_struct *p) { return 1; }
|
|
|
|
static inline void frozen_process(struct task_struct *p) { BUG(); }
|
|
|
|
|
|
|
|
static inline void refrigerator(void) {}
|
|
|
|
static inline int freeze_processes(void) { BUG(); return 0; }
|
|
|
|
static inline void thaw_processes(void) {}
|
|
|
|
|
|
|
|
static inline int try_to_freeze(void) { return 0; }
|
|
|
|
|
|
|
|
|
|
|
|
#endif
|