2008-10-19 07:27:21 +04:00
/*
* cgroup_freezer . c - control group freezer subsystem
*
* Copyright IBM Corporation , 2007
*
* Author : Cedric Le Goater < clg @ fr . ibm . com >
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of version 2.1 of the GNU Lesser General Public License
* as published by the Free Software Foundation .
*
* This program is distributed in the hope that it would be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE .
*/
# include <linux/module.h>
# include <linux/cgroup.h>
# include <linux/fs.h>
# include <linux/uaccess.h>
# include <linux/freezer.h>
# include <linux/seq_file.h>
enum freezer_state {
2008-10-19 07:27:23 +04:00
CGROUP_THAWED = 0 ,
CGROUP_FREEZING ,
CGROUP_FROZEN ,
2008-10-19 07:27:21 +04:00
} ;
struct freezer {
struct cgroup_subsys_state css ;
enum freezer_state state ;
spinlock_t lock ; /* protects _writes_ to state */
} ;
static inline struct freezer * cgroup_freezer (
struct cgroup * cgroup )
{
return container_of (
cgroup_subsys_state ( cgroup , freezer_subsys_id ) ,
struct freezer , css ) ;
}
static inline struct freezer * task_freezer ( struct task_struct * task )
{
return container_of ( task_subsys_state ( task , freezer_subsys_id ) ,
struct freezer , css ) ;
}
int cgroup_frozen ( struct task_struct * task )
{
struct freezer * freezer ;
enum freezer_state state ;
task_lock ( task ) ;
freezer = task_freezer ( task ) ;
state = freezer - > state ;
task_unlock ( task ) ;
2008-10-19 07:27:23 +04:00
return state = = CGROUP_FROZEN ;
2008-10-19 07:27:21 +04:00
}
/*
* cgroups_write_string ( ) limits the size of freezer state strings to
* CGROUP_LOCAL_BUFFER_SIZE
*/
static const char * freezer_state_strs [ ] = {
2008-10-19 07:27:23 +04:00
" THAWED " ,
2008-10-19 07:27:21 +04:00
" FREEZING " ,
" FROZEN " ,
} ;
/*
* State diagram
* Transitions are caused by userspace writes to the freezer . state file .
* The values in parenthesis are state labels . The rest are edge labels .
*
2008-10-19 07:27:23 +04:00
* ( THAWED ) - - FROZEN - - > ( FREEZING ) - - FROZEN - - > ( FROZEN )
* ^ ^ | |
* | \ _______THAWED_______ / |
* \ __________________________THAWED____________ /
2008-10-19 07:27:21 +04:00
*/
struct cgroup_subsys freezer_subsys ;
/* Locks taken and their ordering
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* css_set_lock
* cgroup_mutex ( AKA cgroup_lock )
* task - > alloc_lock ( AKA task_lock )
* freezer - > lock
* task - > sighand - > siglock
*
* cgroup code forces css_set_lock to be taken before task - > alloc_lock
*
* freezer_create ( ) , freezer_destroy ( ) :
* cgroup_mutex [ by cgroup core ]
*
* can_attach ( ) :
* cgroup_mutex
*
* cgroup_frozen ( ) :
* task - > alloc_lock ( to get task ' s cgroup )
*
* freezer_fork ( ) ( preserving fork ( ) performance means can ' t take cgroup_mutex ) :
* task - > alloc_lock ( to get task ' s cgroup )
* freezer - > lock
* sighand - > siglock ( if the cgroup is freezing )
*
* freezer_read ( ) :
* cgroup_mutex
* freezer - > lock
* read_lock css_set_lock ( cgroup iterator start )
*
* freezer_write ( ) ( freeze ) :
* cgroup_mutex
* freezer - > lock
* read_lock css_set_lock ( cgroup iterator start )
* sighand - > siglock
*
* freezer_write ( ) ( unfreeze ) :
* cgroup_mutex
* freezer - > lock
* read_lock css_set_lock ( cgroup iterator start )
* task - > alloc_lock ( to prevent races with freeze_task ( ) )
* sighand - > siglock
*/
static struct cgroup_subsys_state * freezer_create ( struct cgroup_subsys * ss ,
struct cgroup * cgroup )
{
struct freezer * freezer ;
freezer = kzalloc ( sizeof ( struct freezer ) , GFP_KERNEL ) ;
if ( ! freezer )
return ERR_PTR ( - ENOMEM ) ;
spin_lock_init ( & freezer - > lock ) ;
2008-10-19 07:27:23 +04:00
freezer - > state = CGROUP_THAWED ;
2008-10-19 07:27:21 +04:00
return & freezer - > css ;
}
static void freezer_destroy ( struct cgroup_subsys * ss ,
struct cgroup * cgroup )
{
kfree ( cgroup_freezer ( cgroup ) ) ;
}
2008-10-19 07:27:22 +04:00
/* Task is frozen or will freeze immediately when next it gets woken */
static bool is_task_frozen_enough ( struct task_struct * task )
{
return frozen ( task ) | |
( task_is_stopped_or_traced ( task ) & & freezing ( task ) ) ;
}
2008-10-19 07:27:21 +04:00
2008-10-19 07:27:22 +04:00
/*
* The call to cgroup_lock ( ) in the freezer . state write method prevents
* a write to that file racing against an attach , and hence the
* can_attach ( ) result will remain valid until the attach completes .
*/
2008-10-19 07:27:21 +04:00
static int freezer_can_attach ( struct cgroup_subsys * ss ,
struct cgroup * new_cgroup ,
2009-09-24 02:56:31 +04:00
struct task_struct * task , bool threadgroup )
2008-10-19 07:27:21 +04:00
{
struct freezer * freezer ;
2008-10-19 07:27:22 +04:00
2008-10-30 00:00:52 +03:00
/*
* Anything frozen can ' t move or be moved to / from .
*
* Since orig_freezer - > state = = FROZEN means that @ task has been
* frozen , so it ' s sufficient to check the latter condition .
*/
2008-10-19 07:27:22 +04:00
if ( is_task_frozen_enough ( task ) )
return - EBUSY ;
2008-10-19 07:27:21 +04:00
freezer = cgroup_freezer ( new_cgroup ) ;
2008-10-19 07:27:23 +04:00
if ( freezer - > state = = CGROUP_FROZEN )
2008-10-19 07:27:22 +04:00
return - EBUSY ;
2009-09-24 02:56:31 +04:00
if ( threadgroup ) {
struct task_struct * c ;
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( c , & task - > thread_group , thread_group ) {
if ( is_task_frozen_enough ( c ) ) {
rcu_read_unlock ( ) ;
return - EBUSY ;
}
}
rcu_read_unlock ( ) ;
}
2008-10-30 00:00:52 +03:00
return 0 ;
2008-10-19 07:27:21 +04:00
}
static void freezer_fork ( struct cgroup_subsys * ss , struct task_struct * task )
{
struct freezer * freezer ;
2008-11-13 00:26:49 +03:00
/*
* No lock is needed , since the task isn ' t on tasklist yet ,
* so it can ' t be moved to another cgroup , which means the
* freezer won ' t be removed and will be valid during this
* function call .
*/
2008-10-19 07:27:21 +04:00
freezer = task_freezer ( task ) ;
2008-11-13 00:26:50 +03:00
/*
* The root cgroup is non - freezable , so we can skip the
* following check .
*/
if ( ! freezer - > css . cgroup - > parent )
return ;
2008-10-19 07:27:21 +04:00
spin_lock_irq ( & freezer - > lock ) ;
2008-10-30 00:00:51 +03:00
BUG_ON ( freezer - > state = = CGROUP_FROZEN ) ;
2008-10-19 07:27:23 +04:00
/* Locking avoids race with FREEZING -> THAWED transitions. */
if ( freezer - > state = = CGROUP_FREEZING )
2008-10-19 07:27:21 +04:00
freeze_task ( task , true ) ;
spin_unlock_irq ( & freezer - > lock ) ;
}
/*
* caller must hold freezer - > lock
*/
2008-10-19 07:27:24 +04:00
static void update_freezer_state ( struct cgroup * cgroup ,
struct freezer * freezer )
2008-10-19 07:27:21 +04:00
{
struct cgroup_iter it ;
struct task_struct * task ;
unsigned int nfrozen = 0 , ntotal = 0 ;
cgroup_iter_start ( cgroup , & it ) ;
while ( ( task = cgroup_iter_next ( cgroup , & it ) ) ) {
ntotal + + ;
2008-10-19 07:27:22 +04:00
if ( is_task_frozen_enough ( task ) )
2008-10-19 07:27:21 +04:00
nfrozen + + ;
}
/*
* Transition to FROZEN when no new tasks can be added ensures
* that we never exist in the FROZEN state while there are unfrozen
* tasks .
*/
if ( nfrozen = = ntotal )
2008-10-19 07:27:23 +04:00
freezer - > state = CGROUP_FROZEN ;
2008-10-19 07:27:24 +04:00
else if ( nfrozen > 0 )
freezer - > state = CGROUP_FREEZING ;
else
freezer - > state = CGROUP_THAWED ;
2008-10-19 07:27:21 +04:00
cgroup_iter_end ( cgroup , & it ) ;
}
static int freezer_read ( struct cgroup * cgroup , struct cftype * cft ,
struct seq_file * m )
{
struct freezer * freezer ;
enum freezer_state state ;
if ( ! cgroup_lock_live_group ( cgroup ) )
return - ENODEV ;
freezer = cgroup_freezer ( cgroup ) ;
spin_lock_irq ( & freezer - > lock ) ;
state = freezer - > state ;
2008-10-19 07:27:23 +04:00
if ( state = = CGROUP_FREEZING ) {
2008-10-19 07:27:21 +04:00
/* We change from FREEZING to FROZEN lazily if the cgroup was
* only partially frozen when we exitted write . */
2008-10-19 07:27:24 +04:00
update_freezer_state ( cgroup , freezer ) ;
2008-10-19 07:27:21 +04:00
state = freezer - > state ;
}
spin_unlock_irq ( & freezer - > lock ) ;
cgroup_unlock ( ) ;
seq_puts ( m , freezer_state_strs [ state ] ) ;
seq_putc ( m , ' \n ' ) ;
return 0 ;
}
static int try_to_freeze_cgroup ( struct cgroup * cgroup , struct freezer * freezer )
{
struct cgroup_iter it ;
struct task_struct * task ;
unsigned int num_cant_freeze_now = 0 ;
2008-10-19 07:27:23 +04:00
freezer - > state = CGROUP_FREEZING ;
2008-10-19 07:27:21 +04:00
cgroup_iter_start ( cgroup , & it ) ;
while ( ( task = cgroup_iter_next ( cgroup , & it ) ) ) {
if ( ! freeze_task ( task , true ) )
continue ;
2008-10-19 07:27:22 +04:00
if ( is_task_frozen_enough ( task ) )
2008-10-19 07:27:21 +04:00
continue ;
if ( ! freezing ( task ) & & ! freezer_should_skip ( task ) )
num_cant_freeze_now + + ;
}
cgroup_iter_end ( cgroup , & it ) ;
return num_cant_freeze_now ? - EBUSY : 0 ;
}
2008-10-30 00:00:53 +03:00
static void unfreeze_cgroup ( struct cgroup * cgroup , struct freezer * freezer )
2008-10-19 07:27:21 +04:00
{
struct cgroup_iter it ;
struct task_struct * task ;
cgroup_iter_start ( cgroup , & it ) ;
while ( ( task = cgroup_iter_next ( cgroup , & it ) ) ) {
2008-10-30 00:00:53 +03:00
thaw_process ( task ) ;
2008-10-19 07:27:21 +04:00
}
cgroup_iter_end ( cgroup , & it ) ;
2008-10-30 00:00:53 +03:00
freezer - > state = CGROUP_THAWED ;
2008-10-19 07:27:21 +04:00
}
static int freezer_change_state ( struct cgroup * cgroup ,
enum freezer_state goal_state )
{
struct freezer * freezer ;
int retval = 0 ;
freezer = cgroup_freezer ( cgroup ) ;
2008-10-30 00:00:54 +03:00
2008-10-19 07:27:21 +04:00
spin_lock_irq ( & freezer - > lock ) ;
2008-10-30 00:00:54 +03:00
2008-10-19 07:27:24 +04:00
update_freezer_state ( cgroup , freezer ) ;
2008-10-19 07:27:21 +04:00
if ( goal_state = = freezer - > state )
goto out ;
2008-10-30 00:00:54 +03:00
switch ( goal_state ) {
2008-10-19 07:27:23 +04:00
case CGROUP_THAWED :
2008-10-30 00:00:54 +03:00
unfreeze_cgroup ( cgroup , freezer ) ;
2008-10-19 07:27:21 +04:00
break ;
2008-10-19 07:27:23 +04:00
case CGROUP_FROZEN :
2008-10-30 00:00:54 +03:00
retval = try_to_freeze_cgroup ( cgroup , freezer ) ;
2008-10-19 07:27:21 +04:00
break ;
default :
2008-10-30 00:00:54 +03:00
BUG ( ) ;
2008-10-19 07:27:21 +04:00
}
out :
spin_unlock_irq ( & freezer - > lock ) ;
return retval ;
}
static int freezer_write ( struct cgroup * cgroup ,
struct cftype * cft ,
const char * buffer )
{
int retval ;
enum freezer_state goal_state ;
2008-10-19 07:27:23 +04:00
if ( strcmp ( buffer , freezer_state_strs [ CGROUP_THAWED ] ) = = 0 )
goal_state = CGROUP_THAWED ;
else if ( strcmp ( buffer , freezer_state_strs [ CGROUP_FROZEN ] ) = = 0 )
goal_state = CGROUP_FROZEN ;
2008-10-19 07:27:21 +04:00
else
2008-11-13 00:26:50 +03:00
return - EINVAL ;
2008-10-19 07:27:21 +04:00
if ( ! cgroup_lock_live_group ( cgroup ) )
return - ENODEV ;
retval = freezer_change_state ( cgroup , goal_state ) ;
cgroup_unlock ( ) ;
return retval ;
}
static struct cftype files [ ] = {
{
. name = " state " ,
. read_seq_string = freezer_read ,
. write_string = freezer_write ,
} ,
} ;
static int freezer_populate ( struct cgroup_subsys * ss , struct cgroup * cgroup )
{
2008-11-13 00:26:50 +03:00
if ( ! cgroup - > parent )
return 0 ;
2008-10-19 07:27:21 +04:00
return cgroup_add_files ( cgroup , ss , files , ARRAY_SIZE ( files ) ) ;
}
struct cgroup_subsys freezer_subsys = {
. name = " freezer " ,
. create = freezer_create ,
. destroy = freezer_destroy ,
. populate = freezer_populate ,
. subsys_id = freezer_subsys_id ,
. can_attach = freezer_can_attach ,
. attach = NULL ,
. fork = freezer_fork ,
. exit = NULL ,
} ;