2007-07-09 18:52:00 +02:00
/*
* kernel / time / sched_debug . c
*
* Print the CFS rbtree
*
* Copyright ( C ) 2007 , Red Hat , Inc . , Ingo Molnar
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/proc_fs.h>
# include <linux/sched.h>
# include <linux/seq_file.h>
# include <linux/kallsyms.h>
# include <linux/utsname.h>
/*
* This allows printing both to / proc / sched_debug and
* to the console
*/
# define SEQ_printf(m, x...) \
do { \
if ( m ) \
seq_printf ( m , x ) ; \
else \
printk ( x ) ; \
} while ( 0 )
2007-10-15 17:00:08 +02:00
/*
* Ease the printing of nsec fields :
*/
static long long nsec_high ( long long nsec )
{
if ( nsec < 0 ) {
nsec = - nsec ;
do_div ( nsec , 1000000 ) ;
return - nsec ;
}
do_div ( nsec , 1000000 ) ;
return nsec ;
}
static unsigned long nsec_low ( long long nsec )
{
if ( nsec < 0 )
nsec = - nsec ;
return do_div ( nsec , 1000000 ) ;
}
# define SPLIT_NS(x) nsec_high(x), nsec_low(x)
2007-07-09 18:52:00 +02:00
static void
2007-08-09 11:16:51 +02:00
print_task ( struct seq_file * m , struct rq * rq , struct task_struct * p )
2007-07-09 18:52:00 +02:00
{
if ( rq - > curr = = p )
SEQ_printf ( m , " R " ) ;
else
SEQ_printf ( m , " " ) ;
2007-10-15 17:00:08 +02:00
SEQ_printf ( m , " %15s %5d %9Ld.%06ld %9Ld %5d " ,
2007-07-09 18:52:00 +02:00
p - > comm , p - > pid ,
2007-10-15 17:00:08 +02:00
SPLIT_NS ( p - > se . vruntime ) ,
2007-07-09 18:52:00 +02:00
( long long ) ( p - > nvcsw + p - > nivcsw ) ,
2007-08-06 04:26:59 +01:00
p - > prio ) ;
2007-08-02 17:41:40 +02:00
# ifdef CONFIG_SCHEDSTATS
2007-10-15 17:00:08 +02:00
SEQ_printf ( m , " %9Ld.%06ld %9Ld.%06ld %9Ld.%06ld \n " ,
2007-10-15 17:00:08 +02:00
SPLIT_NS ( p - > se . vruntime ) ,
SPLIT_NS ( p - > se . sum_exec_runtime ) ,
SPLIT_NS ( p - > se . sum_sleep_runtime ) ) ;
2007-08-02 17:41:40 +02:00
# else
2007-10-15 17:00:08 +02:00
SEQ_printf ( m , " %15Ld %15Ld %15Ld.%06ld %15Ld.%06ld %15Ld.%06ld \n " ,
0LL , 0LL , 0LL , 0L , 0LL , 0L , 0LL , 0L ) ;
2007-08-02 17:41:40 +02:00
# endif
2007-07-09 18:52:00 +02:00
}
2007-08-09 11:16:51 +02:00
static void print_rq ( struct seq_file * m , struct rq * rq , int rq_cpu )
2007-07-09 18:52:00 +02:00
{
struct task_struct * g , * p ;
SEQ_printf ( m ,
" \n runnable tasks: \n "
2007-10-15 17:00:08 +02:00
" task PID tree-key switches prio "
" exec-runtime sum-exec sum-sleep \n "
2007-10-15 17:00:08 +02:00
" ------------------------------------------------------ "
2007-10-15 17:00:08 +02:00
" ---------------------------------------------------- \n " ) ;
2007-07-09 18:52:00 +02:00
read_lock_irq ( & tasklist_lock ) ;
do_each_thread ( g , p ) {
if ( ! p - > se . on_rq | | task_cpu ( p ) ! = rq_cpu )
continue ;
2007-08-09 11:16:51 +02:00
print_task ( m , rq , p ) ;
2007-07-09 18:52:00 +02:00
} while_each_thread ( g , p ) ;
read_unlock_irq ( & tasklist_lock ) ;
}
2007-08-09 11:16:47 +02:00
void print_cfs_rq ( struct seq_file * m , int cpu , struct cfs_rq * cfs_rq )
2007-07-09 18:52:00 +02:00
{
2007-10-15 17:00:06 +02:00
s64 MIN_vruntime = - 1 , min_vruntime , max_vruntime = - 1 ,
spread , rq0_min_vruntime , spread0 ;
2007-10-15 17:00:05 +02:00
struct rq * rq = & per_cpu ( runqueues , cpu ) ;
struct sched_entity * last ;
unsigned long flags ;
2007-08-10 23:05:11 +02:00
SEQ_printf ( m , " \n cfs_rq \n " ) ;
2007-07-09 18:52:00 +02:00
2007-10-15 17:00:08 +02:00
SEQ_printf ( m , " .%-30s: %Ld.%06ld \n " , " exec_clock " ,
SPLIT_NS ( cfs_rq - > exec_clock ) ) ;
2007-10-15 17:00:05 +02:00
spin_lock_irqsave ( & rq - > lock , flags ) ;
if ( cfs_rq - > rb_leftmost )
MIN_vruntime = ( __pick_next_entity ( cfs_rq ) ) - > vruntime ;
last = __pick_last_entity ( cfs_rq ) ;
if ( last )
max_vruntime = last - > vruntime ;
2007-10-15 17:00:06 +02:00
min_vruntime = rq - > cfs . min_vruntime ;
rq0_min_vruntime = per_cpu ( runqueues , 0 ) . cfs . min_vruntime ;
2007-10-15 17:00:05 +02:00
spin_unlock_irqrestore ( & rq - > lock , flags ) ;
2007-10-15 17:00:08 +02:00
SEQ_printf ( m , " .%-30s: %Ld.%06ld \n " , " MIN_vruntime " ,
SPLIT_NS ( MIN_vruntime ) ) ;
SEQ_printf ( m , " .%-30s: %Ld.%06ld \n " , " min_vruntime " ,
SPLIT_NS ( min_vruntime ) ) ;
SEQ_printf ( m , " .%-30s: %Ld.%06ld \n " , " max_vruntime " ,
SPLIT_NS ( max_vruntime ) ) ;
2007-10-15 17:00:05 +02:00
spread = max_vruntime - MIN_vruntime ;
2007-10-15 17:00:08 +02:00
SEQ_printf ( m , " .%-30s: %Ld.%06ld \n " , " spread " ,
SPLIT_NS ( spread ) ) ;
2007-10-15 17:00:06 +02:00
spread0 = min_vruntime - rq0_min_vruntime ;
2007-10-15 17:00:08 +02:00
SEQ_printf ( m , " .%-30s: %Ld.%06ld \n " , " spread0 " ,
SPLIT_NS ( spread0 ) ) ;
2007-10-15 17:00:09 +02:00
SEQ_printf ( m , " .%-30s: %ld \n " , " nr_running " , cfs_rq - > nr_running ) ;
SEQ_printf ( m , " .%-30s: %ld \n " , " load " , cfs_rq - > load . weight ) ;
2007-07-09 18:52:00 +02:00
}
2007-08-09 11:16:51 +02:00
static void print_cpu ( struct seq_file * m , int cpu )
2007-07-09 18:52:00 +02:00
{
struct rq * rq = & per_cpu ( runqueues , cpu ) ;
# ifdef CONFIG_X86
{
unsigned int freq = cpu_khz ? : 1 ;
SEQ_printf ( m , " \n cpu#%d, %u.%03u MHz \n " ,
cpu , freq / 1000 , ( freq % 1000 ) ) ;
}
# else
SEQ_printf ( m , " \n cpu#%d \n " , cpu ) ;
# endif
# define P(x) \
SEQ_printf ( m , " .%-30s: %Ld \n " , # x , ( long long ) ( rq - > x ) )
2007-10-15 17:00:08 +02:00
# define PN(x) \
SEQ_printf ( m , " .%-30s: %Ld.%06ld \n " , # x , SPLIT_NS ( rq - > x ) )
2007-07-09 18:52:00 +02:00
P ( nr_running ) ;
SEQ_printf ( m , " .%-30s: %lu \n " , " load " ,
2007-10-15 17:00:06 +02:00
rq - > load . weight ) ;
2007-07-09 18:52:00 +02:00
P ( nr_switches ) ;
P ( nr_load_updates ) ;
P ( nr_uninterruptible ) ;
SEQ_printf ( m , " .%-30s: %lu \n " , " jiffies " , jiffies ) ;
2007-10-15 17:00:08 +02:00
PN ( next_balance ) ;
2007-07-09 18:52:00 +02:00
P ( curr - > pid ) ;
2007-10-15 17:00:08 +02:00
PN ( clock ) ;
PN ( idle_clock ) ;
PN ( prev_clock_raw ) ;
2007-07-09 18:52:00 +02:00
P ( clock_warps ) ;
P ( clock_overflows ) ;
2007-08-23 15:18:02 +02:00
P ( clock_deep_idle_events ) ;
2007-10-15 17:00:08 +02:00
PN ( clock_max_delta ) ;
2007-07-09 18:52:00 +02:00
P ( cpu_load [ 0 ] ) ;
P ( cpu_load [ 1 ] ) ;
P ( cpu_load [ 2 ] ) ;
P ( cpu_load [ 3 ] ) ;
P ( cpu_load [ 4 ] ) ;
# undef P
2007-10-15 17:00:08 +02:00
# undef PN
2007-07-09 18:52:00 +02:00
2007-08-09 11:16:47 +02:00
print_cfs_stats ( m , cpu ) ;
2007-07-09 18:52:00 +02:00
2007-08-09 11:16:51 +02:00
print_rq ( m , rq , cpu ) ;
2007-07-09 18:52:00 +02:00
}
static int sched_debug_show ( struct seq_file * m , void * v )
{
u64 now = ktime_to_ns ( ktime_get ( ) ) ;
int cpu ;
2007-08-02 17:41:40 +02:00
SEQ_printf ( m , " Sched Debug Version: v0.05-v20, %s %.*s \n " ,
2007-07-09 18:52:00 +02:00
init_utsname ( ) - > release ,
( int ) strcspn ( init_utsname ( ) - > version , " " ) ,
init_utsname ( ) - > version ) ;
2007-10-15 17:00:08 +02:00
SEQ_printf ( m , " now at %Lu.%06ld msecs \n " , SPLIT_NS ( now ) ) ;
2007-07-09 18:52:00 +02:00
for_each_online_cpu ( cpu )
2007-08-09 11:16:51 +02:00
print_cpu ( m , cpu ) ;
2007-07-09 18:52:00 +02:00
SEQ_printf ( m , " \n " ) ;
return 0 ;
}
2007-07-26 13:40:43 +02:00
static void sysrq_sched_debug_show ( void )
2007-07-09 18:52:00 +02:00
{
sched_debug_show ( NULL , NULL ) ;
}
2007-10-15 17:00:09 +02:00
# ifdef CONFIG_FAIR_USER_SCHED
static DEFINE_MUTEX ( root_user_share_mutex ) ;
static int
root_user_share_read_proc ( char * page , char * * start , off_t off , int count ,
int * eof , void * data )
{
int len ;
len = sprintf ( page , " %d \n " , init_task_grp_load ) ;
return len ;
}
static int
root_user_share_write_proc ( struct file * file , const char __user * buffer ,
unsigned long count , void * data )
{
unsigned long shares ;
char kbuf [ sizeof ( unsigned long ) + 1 ] ;
int rc = 0 ;
if ( copy_from_user ( kbuf , buffer , sizeof ( kbuf ) ) )
return - EFAULT ;
shares = simple_strtoul ( kbuf , NULL , 0 ) ;
if ( ! shares )
shares = NICE_0_LOAD ;
mutex_lock ( & root_user_share_mutex ) ;
init_task_grp_load = shares ;
rc = sched_group_set_shares ( & init_task_grp , shares ) ;
mutex_unlock ( & root_user_share_mutex ) ;
return ( rc < 0 ? rc : count ) ;
}
# endif /* CONFIG_FAIR_USER_SCHED */
2007-07-09 18:52:00 +02:00
static int sched_debug_open ( struct inode * inode , struct file * filp )
{
return single_open ( filp , sched_debug_show , NULL ) ;
}
static struct file_operations sched_debug_fops = {
. open = sched_debug_open ,
. read = seq_read ,
. llseek = seq_lseek ,
2007-07-31 00:38:50 -07:00
. release = single_release ,
2007-07-09 18:52:00 +02:00
} ;
static int __init init_sched_debug_procfs ( void )
{
struct proc_dir_entry * pe ;
pe = create_proc_entry ( " sched_debug " , 0644 , NULL ) ;
if ( ! pe )
return - ENOMEM ;
pe - > proc_fops = & sched_debug_fops ;
2007-10-15 17:00:09 +02:00
# ifdef CONFIG_FAIR_USER_SCHED
pe = create_proc_entry ( " root_user_share " , 0644 , NULL ) ;
if ( ! pe )
return - ENOMEM ;
pe - > read_proc = root_user_share_read_proc ;
pe - > write_proc = root_user_share_write_proc ;
# endif
2007-07-09 18:52:00 +02:00
return 0 ;
}
__initcall ( init_sched_debug_procfs ) ;
void proc_sched_show_task ( struct task_struct * p , struct seq_file * m )
{
unsigned long flags ;
int num_threads = 1 ;
rcu_read_lock ( ) ;
if ( lock_task_sighand ( p , & flags ) ) {
num_threads = atomic_read ( & p - > signal - > count ) ;
unlock_task_sighand ( p , & flags ) ;
}
rcu_read_unlock ( ) ;
SEQ_printf ( m , " %s (%d, #threads: %d) \n " , p - > comm , p - > pid , num_threads ) ;
SEQ_printf ( m , " ---------------------------------------------- \n " ) ;
# define P(F) \
SEQ_printf ( m , " %-25s:%20Ld \n " , # F , ( long long ) p - > F )
2007-10-15 17:00:08 +02:00
# define PN(F) \
SEQ_printf ( m , " %-25s:%14Ld.%06ld \n " , # F , SPLIT_NS ( ( long long ) p - > F ) )
2007-07-09 18:52:00 +02:00
2007-10-15 17:00:08 +02:00
PN ( se . exec_start ) ;
PN ( se . vruntime ) ;
PN ( se . sum_exec_runtime ) ;
2007-08-02 17:41:40 +02:00
# ifdef CONFIG_SCHEDSTATS
2007-10-15 17:00:08 +02:00
PN ( se . wait_start ) ;
PN ( se . sleep_start ) ;
PN ( se . block_start ) ;
PN ( se . sleep_max ) ;
PN ( se . block_max ) ;
PN ( se . exec_max ) ;
PN ( se . slice_max ) ;
PN ( se . wait_max ) ;
2007-08-02 17:41:40 +02:00
# endif
2007-07-09 18:52:00 +02:00
SEQ_printf ( m , " %-25s:%20Ld \n " ,
" nr_switches " , ( long long ) ( p - > nvcsw + p - > nivcsw ) ) ;
P ( se . load . weight ) ;
P ( policy ) ;
P ( prio ) ;
# undef P
2007-10-15 17:00:08 +02:00
# undef PN
2007-07-09 18:52:00 +02:00
{
u64 t0 , t1 ;
t0 = sched_clock ( ) ;
t1 = sched_clock ( ) ;
SEQ_printf ( m , " %-25s:%20Ld \n " ,
" clock-delta " , ( long long ) ( t1 - t0 ) ) ;
}
}
void proc_sched_set_task ( struct task_struct * p )
{
2007-08-02 17:41:40 +02:00
# ifdef CONFIG_SCHEDSTATS
2007-10-15 17:00:02 +02:00
p - > se . sleep_max = 0 ;
p - > se . block_max = 0 ;
p - > se . exec_max = 0 ;
2007-10-15 17:00:02 +02:00
p - > se . slice_max = 0 ;
2007-10-15 17:00:02 +02:00
p - > se . wait_max = 0 ;
2007-08-02 17:41:40 +02:00
# endif
2007-10-15 17:00:02 +02:00
p - > se . sum_exec_runtime = 0 ;
2007-09-05 14:32:49 +02:00
p - > se . prev_sum_exec_runtime = 0 ;
2007-07-09 18:52:00 +02:00
}