2006-07-14 00:24:36 -07: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>
2017-02-04 01:20:53 +01:00
# include <linux/sched/task.h>
2017-02-05 11:48:36 +01:00
# include <linux/sched/cputime.h>
2006-07-14 00:24:36 -07:00
# include <linux/slab.h>
2009-09-18 23:55:55 +04:00
# include <linux/taskstats.h>
2006-07-14 00:24:36 -07:00
# include <linux/time.h>
# include <linux/sysctl.h>
# include <linux/delayacct.h>
2011-07-11 15:28:14 -04:00
# include <linux/module.h>
2006-07-14 00:24:36 -07:00
2006-07-30 03:03:11 -07:00
int delayacct_on __read_mostly = 1 ; /* Delay accounting turned on/off */
2011-07-11 15:28:14 -04:00
EXPORT_SYMBOL_GPL ( delayacct_on ) ;
2006-12-06 20:33:20 -08:00
struct kmem_cache * delayacct_cache ;
2006-07-14 00:24:36 -07:00
2006-07-30 03:03:11 -07:00
static int __init delayacct_setup_disable ( char * str )
2006-07-14 00:24:36 -07:00
{
2006-07-30 03:03:11 -07:00
delayacct_on = 0 ;
2006-07-14 00:24:36 -07:00
return 1 ;
}
2006-07-30 03:03:11 -07:00
__setup ( " nodelayacct " , delayacct_setup_disable ) ;
2006-07-14 00:24:36 -07:00
void delayacct_init ( void )
{
2016-01-14 15:18:21 -08:00
delayacct_cache = KMEM_CACHE ( task_delay_info , SLAB_PANIC | SLAB_ACCOUNT ) ;
2006-07-14 00:24:36 -07:00
delayacct_tsk_init ( & init_task ) ;
}
void __delayacct_tsk_init ( struct task_struct * tsk )
{
2006-12-06 20:33:17 -08:00
tsk - > delays = kmem_cache_zalloc ( delayacct_cache , GFP_KERNEL ) ;
2006-07-14 00:24:36 -07:00
if ( tsk - > delays )
spin_lock_init ( & tsk - > delays - > lock ) ;
}
/*
2014-07-16 21:04:35 +00:00
* Finish delay accounting for a statistic using its timestamps ( @ start ) ,
* accumalator ( @ total ) and @ count
2006-07-14 00:24:36 -07:00
*/
2014-07-16 21:04:35 +00:00
static void delayacct_end ( u64 * start , u64 * total , u32 * count )
2006-07-14 00:24:36 -07:00
{
2014-07-16 21:04:35 +00:00
s64 ns = ktime_get_ns ( ) - * start ;
2006-11-05 23:52:10 -08:00
unsigned long flags ;
2006-07-14 00:24:36 -07:00
2014-07-16 21:04:35 +00:00
if ( ns > 0 ) {
spin_lock_irqsave ( & current - > delays - > lock , flags ) ;
* total + = ns ;
( * count ) + + ;
spin_unlock_irqrestore ( & current - > delays - > lock , flags ) ;
}
2006-07-14 00:24:36 -07:00
}
2006-07-14 00:24:37 -07:00
void __delayacct_blkio_start ( void )
{
2014-07-16 21:04:35 +00:00
current - > delays - > blkio_start = ktime_get_ns ( ) ;
2006-07-14 00:24:37 -07:00
}
void __delayacct_blkio_end ( void )
{
if ( current - > delays - > flags & DELAYACCT_PF_SWAPIN )
/* Swapin block I/O */
delayacct_end ( & current - > delays - > blkio_start ,
& current - > delays - > swapin_delay ,
& current - > delays - > swapin_count ) ;
else /* Other block I/O */
delayacct_end ( & current - > delays - > blkio_start ,
& current - > delays - > blkio_delay ,
& current - > delays - > blkio_count ) ;
}
2006-07-14 00:24:41 -07:00
int __delayacct_add_tsk ( struct taskstats * d , struct task_struct * tsk )
{
2017-01-31 04:09:29 +01:00
u64 utime , stime , stimescaled , utimescaled ;
2014-07-16 21:04:37 +00:00
unsigned long long t2 , t3 ;
unsigned long flags , t1 ;
s64 tmp ;
2006-07-14 00:24:41 -07:00
2017-01-31 04:09:29 +01:00
task_cputime ( tsk , & utime , & stime ) ;
2014-07-16 21:04:37 +00:00
tmp = ( s64 ) d - > cpu_run_real_total ;
2017-01-31 04:09:29 +01:00
tmp + = utime + stime ;
2006-07-14 00:24:41 -07:00
d - > cpu_run_real_total = ( tmp < ( s64 ) d - > cpu_run_real_total ) ? 0 : tmp ;
2017-01-31 04:09:29 +01:00
task_cputime_scaled ( tsk , & utimescaled , & stimescaled ) ;
2014-07-16 21:04:37 +00:00
tmp = ( s64 ) d - > cpu_scaled_run_real_total ;
2017-01-31 04:09:29 +01:00
tmp + = utimescaled + stimescaled ;
2007-10-18 03:06:34 -07:00
d - > cpu_scaled_run_real_total =
( tmp < ( s64 ) d - > cpu_scaled_run_real_total ) ? 0 : tmp ;
2006-07-14 00:24:41 -07:00
/*
* No locking available for sched_info ( and too expensive to add one )
* Mitigate by taking snapshot of values
*/
2007-10-15 17:00:12 +02:00
t1 = tsk - > sched_info . pcount ;
2006-07-14 00:24:41 -07:00
t2 = tsk - > sched_info . run_delay ;
2008-12-16 23:41:22 -08:00
t3 = tsk - > se . sum_exec_runtime ;
2006-07-14 00:24:41 -07:00
d - > cpu_count + = t1 ;
2007-07-09 18:52:00 +02:00
tmp = ( s64 ) d - > cpu_delay_total + t2 ;
2006-07-14 00:24:41 -07:00
d - > cpu_delay_total = ( tmp < ( s64 ) d - > cpu_delay_total ) ? 0 : tmp ;
2007-07-09 18:52:00 +02:00
tmp = ( s64 ) d - > cpu_run_virtual_total + t3 ;
2006-07-14 00:24:41 -07:00
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-05 23:52:10 -08:00
spin_lock_irqsave ( & tsk - > delays - > lock , flags ) ;
2006-07-14 00:24:41 -07: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 ;
2008-07-25 01:48:53 -07:00
tmp = d - > freepages_delay_total + tsk - > delays - > freepages_delay ;
d - > freepages_delay_total = ( tmp < d - > freepages_delay_total ) ? 0 : tmp ;
2006-07-14 00:24:41 -07:00
d - > blkio_count + = tsk - > delays - > blkio_count ;
d - > swapin_count + = tsk - > delays - > swapin_count ;
2008-07-25 01:48:53 -07:00
d - > freepages_count + = tsk - > delays - > freepages_count ;
2006-11-05 23:52:10 -08:00
spin_unlock_irqrestore ( & tsk - > delays - > lock , flags ) ;
2006-07-14 00:24:41 -07:00
return 0 ;
}
2006-07-14 00:24:43 -07:00
__u64 __delayacct_blkio_ticks ( struct task_struct * tsk )
{
__u64 ret ;
2006-11-05 23:52:10 -08:00
unsigned long flags ;
2006-07-14 00:24:43 -07:00
2006-11-05 23:52:10 -08:00
spin_lock_irqsave ( & tsk - > delays - > lock , flags ) ;
2006-07-14 00:24:43 -07:00
ret = nsec_to_clock_t ( tsk - > delays - > blkio_delay +
tsk - > delays - > swapin_delay ) ;
2006-11-05 23:52:10 -08:00
spin_unlock_irqrestore ( & tsk - > delays - > lock , flags ) ;
2006-07-14 00:24:43 -07:00
return ret ;
}
2008-07-25 01:48:52 -07:00
void __delayacct_freepages_start ( void )
{
2014-07-16 21:04:35 +00:00
current - > delays - > freepages_start = ktime_get_ns ( ) ;
2008-07-25 01:48:52 -07:00
}
void __delayacct_freepages_end ( void )
{
delayacct_end ( & current - > delays - > freepages_start ,
& current - > delays - > freepages_delay ,
& current - > delays - > freepages_count ) ;
}