2006-09-30 23:28:55 -07:00
/*
* tsacct . c - System accounting over taskstats interface
*
* Copyright ( C ) Jay Lan , < jlan @ sgi . com >
*
*
* 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 will 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/kernel.h>
# include <linux/sched.h>
# include <linux/tsacct_kern.h>
# include <linux/acct.h>
2006-09-30 23:29:00 -07:00
# include <linux/jiffies.h>
2006-09-30 23:28:55 -07:00
/*
* fill in basic accounting fields
*/
void bacct_add_tsk ( struct taskstats * stats , struct task_struct * tsk )
{
2008-11-14 10:39:19 +11:00
const struct cred * tcred ;
2006-09-30 23:28:55 -07:00
struct timespec uptime , ts ;
2008-07-25 01:48:50 -07:00
u64 ac_etime ;
2006-09-30 23:28:55 -07:00
BUILD_BUG_ON ( TS_COMM_LEN < TASK_COMM_LEN ) ;
/* calculate task elapsed time in timespec */
do_posix_clock_monotonic_gettime ( & uptime ) ;
2006-10-28 10:38:50 -07:00
ts = timespec_sub ( uptime , tsk - > start_time ) ;
2008-07-25 01:48:50 -07:00
/* rebase elapsed time to usec (should never be negative) */
2006-09-30 23:28:55 -07:00
ac_etime = timespec_to_ns ( & ts ) ;
do_div ( ac_etime , NSEC_PER_USEC ) ;
stats - > ac_etime = ac_etime ;
2007-07-24 17:47:43 -07:00
stats - > ac_btime = get_seconds ( ) - ts . tv_sec ;
2006-09-30 23:28:55 -07:00
if ( thread_group_leader ( tsk ) ) {
stats - > ac_exitcode = tsk - > exit_code ;
if ( tsk - > flags & PF_FORKNOEXEC )
stats - > ac_flag | = AFORK ;
}
if ( tsk - > flags & PF_SUPERPRIV )
stats - > ac_flag | = ASU ;
if ( tsk - > flags & PF_DUMPCORE )
stats - > ac_flag | = ACORE ;
if ( tsk - > flags & PF_SIGNALED )
stats - > ac_flag | = AXSIG ;
stats - > ac_nice = task_nice ( tsk ) ;
stats - > ac_sched = tsk - > policy ;
stats - > ac_pid = tsk - > pid ;
2006-10-28 10:38:50 -07:00
rcu_read_lock ( ) ;
2008-11-14 10:39:19 +11:00
tcred = __task_cred ( tsk ) ;
stats - > ac_uid = tcred - > uid ;
stats - > ac_gid = tcred - > gid ;
2006-10-28 10:38:50 -07:00
stats - > ac_ppid = pid_alive ( tsk ) ?
rcu_dereference ( tsk - > real_parent ) - > tgid : 0 ;
rcu_read_unlock ( ) ;
2006-09-30 23:28:55 -07:00
stats - > ac_utime = cputime_to_msecs ( tsk - > utime ) * USEC_PER_MSEC ;
stats - > ac_stime = cputime_to_msecs ( tsk - > stime ) * USEC_PER_MSEC ;
2007-10-18 03:06:34 -07:00
stats - > ac_utimescaled =
cputime_to_msecs ( tsk - > utimescaled ) * USEC_PER_MSEC ;
stats - > ac_stimescaled =
cputime_to_msecs ( tsk - > stimescaled ) * USEC_PER_MSEC ;
2006-09-30 23:28:55 -07:00
stats - > ac_minflt = tsk - > min_flt ;
stats - > ac_majflt = tsk - > maj_flt ;
strncpy ( stats - > ac_comm , tsk - > comm , sizeof ( stats - > ac_comm ) ) ;
}
2006-09-30 23:28:58 -07:00
# ifdef CONFIG_TASK_XACCT
2006-09-30 23:29:00 -07:00
# define KB 1024
# define MB (1024*KB)
2006-09-30 23:28:58 -07:00
/*
* fill in extended accounting fields
*/
void xacct_add_tsk ( struct taskstats * stats , struct task_struct * p )
{
2006-10-29 22:46:43 -08:00
struct mm_struct * mm ;
2008-07-25 01:48:40 -07:00
/* convert pages-usec to Mbyte-usec */
stats - > coremem = p - > acct_rss_mem1 * PAGE_SIZE / MB ;
stats - > virtmem = p - > acct_vm_mem1 * PAGE_SIZE / MB ;
2006-10-29 22:46:43 -08:00
mm = get_task_mm ( p ) ;
if ( mm ) {
2006-09-30 23:29:00 -07:00
/* adjust to KB unit */
2009-01-06 14:40:29 -08:00
stats - > hiwater_rss = get_mm_hiwater_rss ( mm ) * PAGE_SIZE / KB ;
stats - > hiwater_vm = get_mm_hiwater_vm ( mm ) * PAGE_SIZE / KB ;
2006-10-29 22:46:43 -08:00
mmput ( mm ) ;
2006-09-30 23:28:58 -07:00
}
2008-07-28 00:48:12 +02:00
stats - > read_char = p - > ioac . rchar ;
stats - > write_char = p - > ioac . wchar ;
stats - > read_syscalls = p - > ioac . syscr ;
stats - > write_syscalls = p - > ioac . syscw ;
2006-12-10 02:19:53 -08:00
# ifdef CONFIG_TASK_IO_ACCOUNTING
2008-07-28 00:48:12 +02:00
stats - > read_bytes = p - > ioac . read_bytes ;
stats - > write_bytes = p - > ioac . write_bytes ;
stats - > cancelled_write_bytes = p - > ioac . cancelled_write_bytes ;
2006-12-10 02:19:53 -08:00
# else
stats - > read_bytes = 0 ;
stats - > write_bytes = 0 ;
stats - > cancelled_write_bytes = 0 ;
# endif
2006-09-30 23:28:58 -07:00
}
2006-09-30 23:29:00 -07:00
# undef KB
# undef MB
2006-09-30 23:28:59 -07:00
/**
* acct_update_integrals - update mm integral fields in task_struct
* @ tsk : task_struct for accounting
*/
void acct_update_integrals ( struct task_struct * tsk )
{
if ( likely ( tsk - > mm ) ) {
2008-07-25 01:48:40 -07:00
cputime_t time , dtime ;
struct timeval value ;
u64 delta ;
time = tsk - > stime + tsk - > utime ;
dtime = cputime_sub ( time , tsk - > acct_timexpd ) ;
jiffies_to_timeval ( cputime_to_jiffies ( dtime ) , & value ) ;
delta = value . tv_sec ;
delta = delta * USEC_PER_SEC + value . tv_usec ;
2006-09-30 23:28:59 -07:00
if ( delta = = 0 )
return ;
2008-07-25 01:48:40 -07:00
tsk - > acct_timexpd = time ;
2006-09-30 23:28:59 -07:00
tsk - > acct_rss_mem1 + = delta * get_mm_rss ( tsk - > mm ) ;
tsk - > acct_vm_mem1 + = delta * tsk - > mm - > total_vm ;
}
}
/**
* acct_clear_integrals - clear the mm integral fields in task_struct
* @ tsk : task_struct whose accounting fields are cleared
*/
void acct_clear_integrals ( struct task_struct * tsk )
{
2008-07-25 01:48:40 -07:00
tsk - > acct_timexpd = 0 ;
2006-09-30 23:28:59 -07:00
tsk - > acct_rss_mem1 = 0 ;
tsk - > acct_vm_mem1 = 0 ;
}
2006-09-30 23:28:58 -07:00
# endif