2017-06-13 17:18:03 -04:00
/*
* Debug controller
*
* WARNING : This controller is for cgroup core debugging only .
* Its interfaces are unstable and subject to changes at any time .
*/
2017-06-13 17:18:02 -04:00
# include <linux/ctype.h>
# include <linux/mm.h>
# include <linux/slab.h>
# include "cgroup-internal.h"
static struct cgroup_subsys_state *
debug_css_alloc ( struct cgroup_subsys_state * parent_css )
{
struct cgroup_subsys_state * css = kzalloc ( sizeof ( * css ) , GFP_KERNEL ) ;
if ( ! css )
return ERR_PTR ( - ENOMEM ) ;
return css ;
}
static void debug_css_free ( struct cgroup_subsys_state * css )
{
kfree ( css ) ;
}
/*
* debug_taskcount_read - return the number of tasks in a cgroup .
* @ cgrp : the cgroup in question
*/
static u64 debug_taskcount_read ( struct cgroup_subsys_state * css ,
struct cftype * cft )
{
return cgroup_task_count ( css - > cgroup ) ;
}
2017-06-13 17:18:04 -04:00
static int current_css_set_read ( struct seq_file * seq , void * v )
2017-06-13 17:18:02 -04:00
{
2017-06-13 17:18:04 -04:00
struct css_set * cset ;
struct cgroup_subsys * ss ;
struct cgroup_subsys_state * css ;
int i , refcnt ;
mutex_lock ( & cgroup_mutex ) ;
spin_lock_irq ( & css_set_lock ) ;
rcu_read_lock ( ) ;
cset = rcu_dereference ( current - > cgroups ) ;
refcnt = refcount_read ( & cset - > refcount ) ;
seq_printf ( seq , " css_set %pK %d " , cset , refcnt ) ;
if ( refcnt > cset - > nr_tasks )
seq_printf ( seq , " +%d " , refcnt - cset - > nr_tasks ) ;
seq_puts ( seq , " \n " ) ;
/*
* Print the css ' es stored in the current css_set .
*/
for_each_subsys ( ss , i ) {
css = cset - > subsys [ ss - > id ] ;
if ( ! css )
continue ;
seq_printf ( seq , " %2d: %-4s \t - %lx[%d] \n " , ss - > id , ss - > name ,
( unsigned long ) css , css - > id ) ;
}
rcu_read_unlock ( ) ;
spin_unlock_irq ( & css_set_lock ) ;
mutex_unlock ( & cgroup_mutex ) ;
return 0 ;
2017-06-13 17:18:02 -04:00
}
static u64 current_css_set_refcount_read ( struct cgroup_subsys_state * css ,
struct cftype * cft )
{
u64 count ;
rcu_read_lock ( ) ;
count = refcount_read ( & task_css_set ( current ) - > refcount ) ;
rcu_read_unlock ( ) ;
return count ;
}
static int current_css_set_cg_links_read ( struct seq_file * seq , void * v )
{
struct cgrp_cset_link * link ;
struct css_set * cset ;
char * name_buf ;
name_buf = kmalloc ( NAME_MAX + 1 , GFP_KERNEL ) ;
if ( ! name_buf )
return - ENOMEM ;
spin_lock_irq ( & css_set_lock ) ;
rcu_read_lock ( ) ;
cset = rcu_dereference ( current - > cgroups ) ;
list_for_each_entry ( link , & cset - > cgrp_links , cgrp_link ) {
struct cgroup * c = link - > cgrp ;
cgroup_name ( c , name_buf , NAME_MAX + 1 ) ;
seq_printf ( seq , " Root %d group %s \n " ,
c - > root - > hierarchy_id , name_buf ) ;
}
rcu_read_unlock ( ) ;
spin_unlock_irq ( & css_set_lock ) ;
kfree ( name_buf ) ;
return 0 ;
}
# define MAX_TASKS_SHOWN_PER_CSS 25
static int cgroup_css_links_read ( struct seq_file * seq , void * v )
{
struct cgroup_subsys_state * css = seq_css ( seq ) ;
struct cgrp_cset_link * link ;
2017-06-13 17:18:04 -04:00
int dead_cnt = 0 , extra_refs = 0 ;
2017-06-13 17:18:02 -04:00
spin_lock_irq ( & css_set_lock ) ;
list_for_each_entry ( link , & css - > cgroup - > cset_links , cset_link ) {
struct css_set * cset = link - > cset ;
struct task_struct * task ;
int count = 0 ;
2017-06-13 17:18:04 -04:00
int refcnt = refcount_read ( & cset - > refcount ) ;
2017-06-13 17:18:02 -04:00
2017-06-13 17:18:04 -04:00
seq_printf ( seq , " %d " , refcnt ) ;
if ( refcnt - cset - > nr_tasks > 0 ) {
int extra = refcnt - cset - > nr_tasks ;
seq_printf ( seq , " +%d " , extra ) ;
/*
* Take out the one additional reference in
* init_css_set .
*/
if ( cset = = & init_css_set )
extra - - ;
extra_refs + = extra ;
}
seq_puts ( seq , " \n " ) ;
2017-06-13 17:18:02 -04:00
list_for_each_entry ( task , & cset - > tasks , cg_list ) {
2017-06-13 17:18:04 -04:00
if ( count + + < = MAX_TASKS_SHOWN_PER_CSS )
seq_printf ( seq , " task %d \n " ,
task_pid_vnr ( task ) ) ;
2017-06-13 17:18:02 -04:00
}
list_for_each_entry ( task , & cset - > mg_tasks , cg_list ) {
2017-06-13 17:18:04 -04:00
if ( count + + < = MAX_TASKS_SHOWN_PER_CSS )
seq_printf ( seq , " task %d \n " ,
task_pid_vnr ( task ) ) ;
2017-06-13 17:18:02 -04:00
}
2017-06-13 17:18:04 -04:00
/* show # of overflowed tasks */
if ( count > MAX_TASKS_SHOWN_PER_CSS )
seq_printf ( seq , " ... (%d) \n " ,
count - MAX_TASKS_SHOWN_PER_CSS ) ;
if ( cset - > dead ) {
seq_puts ( seq , " [dead] \n " ) ;
dead_cnt + + ;
}
WARN_ON ( count ! = cset - > nr_tasks ) ;
2017-06-13 17:18:02 -04:00
}
spin_unlock_irq ( & css_set_lock ) ;
2017-06-13 17:18:04 -04:00
if ( ! dead_cnt & & ! extra_refs )
return 0 ;
seq_puts ( seq , " \n " ) ;
if ( extra_refs )
seq_printf ( seq , " extra references = %d \n " , extra_refs ) ;
if ( dead_cnt )
seq_printf ( seq , " dead css_sets = %d \n " , dead_cnt ) ;
return 0 ;
}
static int cgroup_subsys_states_read ( struct seq_file * seq , void * v )
{
struct cgroup * cgrp = seq_css ( seq ) - > cgroup ;
struct cgroup_subsys * ss ;
struct cgroup_subsys_state * css ;
char pbuf [ 16 ] ;
int i ;
mutex_lock ( & cgroup_mutex ) ;
for_each_subsys ( ss , i ) {
css = rcu_dereference_check ( cgrp - > subsys [ ss - > id ] , true ) ;
if ( ! css )
continue ;
pbuf [ 0 ] = ' \0 ' ;
/* Show the parent CSS if applicable*/
if ( css - > parent )
snprintf ( pbuf , sizeof ( pbuf ) - 1 , " P=%d " ,
css - > parent - > id ) ;
seq_printf ( seq , " %2d: %-4s \t - %lx[%d] %d%s \n " , ss - > id , ss - > name ,
( unsigned long ) css , css - > id ,
atomic_read ( & css - > online_cnt ) , pbuf ) ;
}
mutex_unlock ( & cgroup_mutex ) ;
return 0 ;
}
static int cgroup_masks_read ( struct seq_file * seq , void * v )
{
struct cgroup * cgrp = seq_css ( seq ) - > cgroup ;
struct cgroup_subsys * ss ;
int i , j ;
struct {
u16 * mask ;
char * name ;
} mask_list [ ] = {
{ & cgrp - > subtree_control , " subtree_control " } ,
{ & cgrp - > subtree_ss_mask , " subtree_ss_mask " } ,
} ;
mutex_lock ( & cgroup_mutex ) ;
for ( i = 0 ; i < ARRAY_SIZE ( mask_list ) ; i + + ) {
u16 mask = * mask_list [ i ] . mask ;
bool first = true ;
seq_printf ( seq , " %-17s: " , mask_list [ i ] . name ) ;
for_each_subsys ( ss , j ) {
if ( ! ( mask & ( 1 < < ss - > id ) ) )
continue ;
if ( ! first )
seq_puts ( seq , " , " ) ;
seq_puts ( seq , ss - > name ) ;
first = false ;
}
seq_putc ( seq , ' \n ' ) ;
}
mutex_unlock ( & cgroup_mutex ) ;
2017-06-13 17:18:02 -04:00
return 0 ;
}
static u64 releasable_read ( struct cgroup_subsys_state * css , struct cftype * cft )
{
return ( ! cgroup_is_populated ( css - > cgroup ) & &
! css_has_online_children ( & css - > cgroup - > self ) ) ;
}
2017-06-14 16:01:32 -04:00
static struct cftype debug_legacy_files [ ] = {
2017-06-13 17:18:02 -04:00
{
. name = " taskcount " ,
. read_u64 = debug_taskcount_read ,
} ,
{
. name = " current_css_set " ,
2017-06-13 17:18:04 -04:00
. seq_show = current_css_set_read ,
. flags = CFTYPE_ONLY_ON_ROOT ,
2017-06-13 17:18:02 -04:00
} ,
{
. name = " current_css_set_refcount " ,
. read_u64 = current_css_set_refcount_read ,
2017-06-13 17:18:04 -04:00
. flags = CFTYPE_ONLY_ON_ROOT ,
2017-06-13 17:18:02 -04:00
} ,
{
. name = " current_css_set_cg_links " ,
. seq_show = current_css_set_cg_links_read ,
2017-06-13 17:18:04 -04:00
. flags = CFTYPE_ONLY_ON_ROOT ,
2017-06-13 17:18:02 -04:00
} ,
{
. name = " cgroup_css_links " ,
. seq_show = cgroup_css_links_read ,
} ,
2017-06-13 17:18:04 -04:00
{
. name = " cgroup_subsys_states " ,
. seq_show = cgroup_subsys_states_read ,
} ,
{
. name = " cgroup_masks " ,
. seq_show = cgroup_masks_read ,
} ,
2017-06-13 17:18:02 -04:00
{
. name = " releasable " ,
. read_u64 = releasable_read ,
} ,
{ } /* terminate */
} ;
2017-06-14 16:01:32 -04:00
static struct cftype debug_files [ ] = {
{
. name = " taskcount " ,
. read_u64 = debug_taskcount_read ,
} ,
{
. name = " current_css_set " ,
. seq_show = current_css_set_read ,
. flags = CFTYPE_ONLY_ON_ROOT ,
} ,
{
. name = " current_css_set_refcount " ,
. read_u64 = current_css_set_refcount_read ,
. flags = CFTYPE_ONLY_ON_ROOT ,
} ,
{
. name = " current_css_set_cg_links " ,
. seq_show = current_css_set_cg_links_read ,
. flags = CFTYPE_ONLY_ON_ROOT ,
} ,
{
. name = " css_links " ,
. seq_show = cgroup_css_links_read ,
} ,
{
. name = " csses " ,
. seq_show = cgroup_subsys_states_read ,
} ,
{
. name = " masks " ,
. seq_show = cgroup_masks_read ,
} ,
{ } /* terminate */
} ;
2017-06-13 17:18:02 -04:00
struct cgroup_subsys debug_cgrp_subsys = {
2017-06-13 17:18:04 -04:00
. css_alloc = debug_css_alloc ,
. css_free = debug_css_free ,
2017-06-14 16:01:32 -04:00
. legacy_cftypes = debug_legacy_files ,
2017-06-13 17:18:02 -04:00
} ;
2017-06-14 16:01:32 -04:00
/*
* On v2 , debug is an implicit controller enabled by " cgroup_debug " boot
* parameter .
*/
static int __init enable_cgroup_debug ( char * str )
{
debug_cgrp_subsys . dfl_cftypes = debug_files ;
debug_cgrp_subsys . implicit_on_dfl = true ;
return 1 ;
}
__setup ( " cgroup_debug " , enable_cgroup_debug ) ;