2019-05-27 08:55:06 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2006-09-30 23:28:55 -07:00
/*
* tsacct . c - System accounting over taskstats interface
*
* Copyright ( C ) Jay Lan , < jlan @ sgi . com >
*/
# include <linux/kernel.h>
2017-02-08 18:51:30 +01:00
# include <linux/sched/signal.h>
2017-02-08 18:51:29 +01:00
# include <linux/sched/mm.h>
2017-02-05 11:48:36 +01:00
# include <linux/sched/cputime.h>
2006-09-30 23:28:55 -07:00
# include <linux/tsacct_kern.h>
# include <linux/acct.h>
2006-09-30 23:29:00 -07:00
# include <linux/jiffies.h>
2010-03-05 13:41:39 -08:00
# include <linux/mm.h>
2006-09-30 23:28:55 -07:00
/*
* fill in basic accounting fields
*/
2012-02-07 17:56:49 -08:00
void bacct_add_tsk ( struct user_namespace * user_ns ,
struct pid_namespace * pid_ns ,
struct taskstats * stats , struct task_struct * tsk )
2006-09-30 23:28:55 -07:00
{
2008-11-14 10:39:19 +11:00
const struct cred * tcred ;
2017-01-31 04:09:30 +01:00
u64 utime , stime , utimescaled , stimescaled ;
2014-07-16 21:04:34 +00:00
u64 delta ;
2019-10-24 16:34:25 +02:00
time64_t btime ;
2006-09-30 23:28:55 -07:00
BUILD_BUG_ON ( TS_COMM_LEN < TASK_COMM_LEN ) ;
2014-07-16 21:04:34 +00:00
/* calculate task elapsed time in nsec */
delta = ktime_get_ns ( ) - tsk - > start_time ;
/* Convert to micro seconds */
do_div ( delta , NSEC_PER_USEC ) ;
stats - > ac_etime = delta ;
2019-10-24 16:34:25 +02:00
/* Convert to seconds for btime (note y2106 limit) */
btime = ktime_get_real_seconds ( ) - div_u64 ( delta , USEC_PER_SEC ) ;
stats - > ac_btime = clamp_t ( time64_t , btime , 0 , U32_MAX ) ;
2019-10-24 16:47:56 +02:00
stats - > ac_btime64 = btime ;
2019-10-24 16:34:25 +02:00
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 ;
2012-02-07 17:56:49 -08:00
stats - > ac_pid = task_pid_nr_ns ( tsk , pid_ns ) ;
2006-10-28 10:38:50 -07:00
rcu_read_lock ( ) ;
2008-11-14 10:39:19 +11:00
tcred = __task_cred ( tsk ) ;
2012-02-07 17:56:49 -08:00
stats - > ac_uid = from_kuid_munged ( user_ns , tcred - > uid ) ;
stats - > ac_gid = from_kgid_munged ( user_ns , tcred - > gid ) ;
2006-10-28 10:38:50 -07:00
stats - > ac_ppid = pid_alive ( tsk ) ?
2012-02-07 17:56:49 -08:00
task_tgid_nr_ns ( rcu_dereference ( tsk - > real_parent ) , pid_ns ) : 0 ;
2006-10-28 10:38:50 -07:00
rcu_read_unlock ( ) ;
2012-11-13 14:20:55 +01:00
2017-01-31 04:09:30 +01:00
task_cputime ( tsk , & utime , & stime ) ;
stats - > ac_utime = div_u64 ( utime , NSEC_PER_USEC ) ;
stats - > ac_stime = div_u64 ( stime , NSEC_PER_USEC ) ;
2012-11-13 14:20:55 +01:00
2017-01-31 04:09:30 +01:00
task_cputime_scaled ( tsk , & utimescaled , & stimescaled ) ;
stats - > ac_utimescaled = div_u64 ( utimescaled , NSEC_PER_USEC ) ;
stats - > ac_stimescaled = div_u64 ( stimescaled , NSEC_PER_USEC ) ;
2012-11-13 14:20:55 +01:00
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)
2011-09-19 17:10:57 -07:00
# define KB_MASK (~(KB-1))
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 ;
2016-02-10 20:08:24 -05:00
/* convert pages-nsec/1024 to Mbyte-usec, see __acct_update_integrals */
stats - > coremem = p - > acct_rss_mem1 * PAGE_SIZE ;
do_div ( stats - > coremem , 1000 * KB ) ;
stats - > virtmem = p - > acct_vm_mem1 * PAGE_SIZE ;
do_div ( stats - > virtmem , 1000 * KB ) ;
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
}
2011-09-19 17:10:57 -07:00
stats - > read_char = p - > ioac . rchar & KB_MASK ;
stats - > write_char = p - > ioac . wchar & KB_MASK ;
stats - > read_syscalls = p - > ioac . syscr & KB_MASK ;
stats - > write_syscalls = p - > ioac . syscw & KB_MASK ;
2006-12-10 02:19:53 -08:00
# ifdef CONFIG_TASK_IO_ACCOUNTING
2011-09-19 17:10:57 -07:00
stats - > read_bytes = p - > ioac . read_bytes & KB_MASK ;
stats - > write_bytes = p - > ioac . write_bytes & KB_MASK ;
stats - > cancelled_write_bytes = p - > ioac . cancelled_write_bytes & KB_MASK ;
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
2012-11-13 14:20:55 +01:00
static void __acct_update_integrals ( struct task_struct * tsk ,
2017-01-31 04:09:30 +01:00
u64 utime , u64 stime )
2006-09-30 23:28:59 -07:00
{
2017-01-31 04:09:30 +01:00
u64 time , delta ;
2016-02-10 20:08:25 -05:00
if ( ! likely ( tsk - > mm ) )
return ;
time = stime + utime ;
2017-01-31 04:09:30 +01:00
delta = time - tsk - > acct_timexpd ;
2016-02-10 20:08:25 -05:00
if ( delta < TICK_NSEC )
2016-02-10 20:08:26 -05:00
return ;
2016-02-10 20:08:25 -05:00
tsk - > acct_timexpd = time ;
/*
* Divide by 1024 to avoid overflow , and to avoid division .
* The final unit reported to userspace is Mbyte - usecs ,
* the rest of the math is done in xacct_add_tsk .
*/
tsk - > acct_rss_mem1 + = delta * get_mm_rss ( tsk - > mm ) > > 10 ;
2021-11-05 13:38:12 -07:00
tsk - > acct_vm_mem1 + = delta * READ_ONCE ( tsk - > mm - > total_vm ) > > 10 ;
2006-09-30 23:28:59 -07:00
}
2012-11-13 14:20:55 +01:00
/**
* acct_update_integrals - update mm integral fields in task_struct
* @ tsk : task_struct for accounting
*/
void acct_update_integrals ( struct task_struct * tsk )
{
2017-01-31 04:09:30 +01:00
u64 utime , stime ;
2016-02-10 20:08:26 -05:00
unsigned long flags ;
2012-11-13 14:20:55 +01:00
2016-02-10 20:08:26 -05:00
local_irq_save ( flags ) ;
2017-01-31 04:09:30 +01:00
task_cputime ( tsk , & utime , & stime ) ;
2012-11-13 14:20:55 +01:00
__acct_update_integrals ( tsk , utime , stime ) ;
2016-02-10 20:08:26 -05:00
local_irq_restore ( flags ) ;
2012-11-13 14:20:55 +01:00
}
/**
* acct_account_cputime - update mm integral after cputime update
* @ tsk : task_struct for accounting
*/
void acct_account_cputime ( struct task_struct * tsk )
{
__acct_update_integrals ( tsk , tsk - > utime , tsk - > stime ) ;
}
2006-09-30 23:28:59 -07:00
/**
* 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