2006-07-14 11:24:36 +04:00
/* delayacct.c - per-task delay accounting
*
* Copyright ( C ) Shailabh Nagar , IBM Corp . 2006
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* 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 . See
* the GNU General Public License for more details .
*/
# include <linux/sched.h>
# include <linux/slab.h>
# include <linux/time.h>
# include <linux/sysctl.h>
# include <linux/delayacct.h>
2006-07-30 14:03:11 +04:00
int delayacct_on __read_mostly = 1 ; /* Delay accounting turned on/off */
2006-12-07 07:33:20 +03:00
struct kmem_cache * delayacct_cache ;
2006-07-14 11:24:36 +04:00
2006-07-30 14:03:11 +04:00
static int __init delayacct_setup_disable ( char * str )
2006-07-14 11:24:36 +04:00
{
2006-07-30 14:03:11 +04:00
delayacct_on = 0 ;
2006-07-14 11:24:36 +04:00
return 1 ;
}
2006-07-30 14:03:11 +04:00
__setup ( " nodelayacct " , delayacct_setup_disable ) ;
2006-07-14 11:24:36 +04:00
void delayacct_init ( void )
{
delayacct_cache = kmem_cache_create ( " delayacct_cache " ,
sizeof ( struct task_delay_info ) ,
0 ,
SLAB_PANIC ,
NULL , NULL ) ;
delayacct_tsk_init ( & init_task ) ;
}
void __delayacct_tsk_init ( struct task_struct * tsk )
{
2006-12-07 07:33:17 +03:00
tsk - > delays = kmem_cache_zalloc ( delayacct_cache , GFP_KERNEL ) ;
2006-07-14 11:24:36 +04:00
if ( tsk - > delays )
spin_lock_init ( & tsk - > delays - > lock ) ;
}
/*
* Start accounting for a delay statistic using
* its starting timestamp ( @ start )
*/
static inline void delayacct_start ( struct timespec * start )
{
do_posix_clock_monotonic_gettime ( start ) ;
}
/*
* Finish delay accounting for a statistic using
* its timestamps ( @ start , @ end ) , accumalator ( @ total ) and @ count
*/
static void delayacct_end ( struct timespec * start , struct timespec * end ,
u64 * total , u32 * count )
{
struct timespec ts ;
s64 ns ;
2006-11-06 10:52:10 +03:00
unsigned long flags ;
2006-07-14 11:24:36 +04:00
do_posix_clock_monotonic_gettime ( end ) ;
ts = timespec_sub ( * end , * start ) ;
ns = timespec_to_ns ( & ts ) ;
if ( ns < 0 )
return ;
2006-11-06 10:52:10 +03:00
spin_lock_irqsave ( & current - > delays - > lock , flags ) ;
2006-07-14 11:24:36 +04:00
* total + = ns ;
( * count ) + + ;
2006-11-06 10:52:10 +03:00
spin_unlock_irqrestore ( & current - > delays - > lock , flags ) ;
2006-07-14 11:24:36 +04:00
}
2006-07-14 11:24:37 +04:00
void __delayacct_blkio_start ( void )
{
delayacct_start ( & current - > delays - > blkio_start ) ;
}
void __delayacct_blkio_end ( void )
{
if ( current - > delays - > flags & DELAYACCT_PF_SWAPIN )
/* Swapin block I/O */
delayacct_end ( & current - > delays - > blkio_start ,
& current - > delays - > blkio_end ,
& current - > delays - > swapin_delay ,
& current - > delays - > swapin_count ) ;
else /* Other block I/O */
delayacct_end ( & current - > delays - > blkio_start ,
& current - > delays - > blkio_end ,
& current - > delays - > blkio_delay ,
& current - > delays - > blkio_count ) ;
}
2006-07-14 11:24:41 +04:00
int __delayacct_add_tsk ( struct taskstats * d , struct task_struct * tsk )
{
s64 tmp ;
struct timespec ts ;
unsigned long t1 , t2 , t3 ;
2006-11-06 10:52:10 +03:00
unsigned long flags ;
2006-07-14 11:24:41 +04:00
/* Though tsk->delays accessed later, early exit avoids
* unnecessary returning of other data
*/
if ( ! tsk - > delays )
goto done ;
tmp = ( s64 ) d - > cpu_run_real_total ;
cputime_to_timespec ( tsk - > utime + tsk - > stime , & ts ) ;
tmp + = timespec_to_ns ( & ts ) ;
d - > cpu_run_real_total = ( tmp < ( s64 ) d - > cpu_run_real_total ) ? 0 : tmp ;
/*
* No locking available for sched_info ( and too expensive to add one )
* Mitigate by taking snapshot of values
*/
t1 = tsk - > sched_info . pcnt ;
t2 = tsk - > sched_info . run_delay ;
t3 = tsk - > sched_info . cpu_time ;
d - > cpu_count + = t1 ;
jiffies_to_timespec ( t2 , & ts ) ;
tmp = ( s64 ) d - > cpu_delay_total + timespec_to_ns ( & ts ) ;
d - > cpu_delay_total = ( tmp < ( s64 ) d - > cpu_delay_total ) ? 0 : tmp ;
tmp = ( s64 ) d - > cpu_run_virtual_total + ( s64 ) jiffies_to_usecs ( t3 ) * 1000 ;
d - > cpu_run_virtual_total =
( tmp < ( s64 ) d - > cpu_run_virtual_total ) ? 0 : tmp ;
/* zero XXX_total, non-zero XXX_count implies XXX stat overflowed */
2006-11-06 10:52:10 +03:00
spin_lock_irqsave ( & tsk - > delays - > lock , flags ) ;
2006-07-14 11:24:41 +04:00
tmp = d - > blkio_delay_total + tsk - > delays - > blkio_delay ;
d - > blkio_delay_total = ( tmp < d - > blkio_delay_total ) ? 0 : tmp ;
tmp = d - > swapin_delay_total + tsk - > delays - > swapin_delay ;
d - > swapin_delay_total = ( tmp < d - > swapin_delay_total ) ? 0 : tmp ;
d - > blkio_count + = tsk - > delays - > blkio_count ;
d - > swapin_count + = tsk - > delays - > swapin_count ;
2006-11-06 10:52:10 +03:00
spin_unlock_irqrestore ( & tsk - > delays - > lock , flags ) ;
2006-07-14 11:24:41 +04:00
done :
return 0 ;
}
2006-07-14 11:24:43 +04:00
__u64 __delayacct_blkio_ticks ( struct task_struct * tsk )
{
__u64 ret ;
2006-11-06 10:52:10 +03:00
unsigned long flags ;
2006-07-14 11:24:43 +04:00
2006-11-06 10:52:10 +03:00
spin_lock_irqsave ( & tsk - > delays - > lock , flags ) ;
2006-07-14 11:24:43 +04:00
ret = nsec_to_clock_t ( tsk - > delays - > blkio_delay +
tsk - > delays - > swapin_delay ) ;
2006-11-06 10:52:10 +03:00
spin_unlock_irqrestore ( & tsk - > delays - > lock , flags ) ;
2006-07-14 11:24:43 +04:00
return ret ;
}