2.9BSD/usr/net/sys/sys/acct.c
/*
* SCCS id @(#)acct.c 2.1 (Berkeley) 8/5/83
*/
#include "param.h"
#ifdef ACCT
#include <sys/systm.h>
#include <sys/acct.h>
#include <sys/dir.h>
#include <sys/user.h>
#include <sys/inode.h>
#include <sys/proc.h>
#include <sys/seg.h>
/*
* Perform process accounting functions.
*/
sysacct()
{
register struct inode *ip;
register struct a {
char *fname;
} *uap;
uap = (struct a *)u.u_ap;
if (suser()) {
if (uap->fname==NULL) {
if (acctp) {
plock(acctp);
iput(acctp);
acctp = NULL;
}
return;
}
if (acctp) {
u.u_error = EBUSY;
return;
}
#ifndef UCB_SYMLINKS
ip = namei(uchar, LOOKUP);
#else
ip = namei(uchar, LOOKUP, 1);
#endif
if(ip == NULL)
return;
if((ip->i_mode & IFMT) != IFREG) {
u.u_error = EACCES;
iput(ip);
return;
}
acctp = ip;
prele(ip);
}
}
/*
* On exit, write a record on the accounting file.
*/
acct()
{
register i;
register struct inode *ip;
off_t siz;
if ((ip=acctp)==NULL)
return;
plock(ip);
for (i=0; i<sizeof(acctbuf.ac_comm); i++)
acctbuf.ac_comm[i] = u.u_comm[i];
acctbuf.ac_utime = compress(u.u_utime);
acctbuf.ac_stime = compress(u.u_stime);
acctbuf.ac_etime = compress(time - u.u_start);
acctbuf.ac_btime = u.u_start;
acctbuf.ac_uid = u.u_ruid;
acctbuf.ac_gid = u.u_rgid;
acctbuf.ac_mem = 0;
acctbuf.ac_io = 0;
acctbuf.ac_tty = u.u_ttyd;
acctbuf.ac_flag = u.u_acflag;
#ifdef UCB_SUBM
if(u.u_procp->p_flag & SSUBM)
acctbuf.ac_flag |= ASUBM;
#endif
#ifdef UCB_LOGIN
for(i=0; i<sizeof(acctbuf.ac_crn); i++)
acctbuf.ac_crn[i] = u.u_crn[i];
acctbuf.ac_magic = AMAGIC;
#endif
siz = ip->i_size;
u.u_offset = siz;
u.u_base = (caddr_t)&acctbuf;
u.u_count = sizeof(acctbuf);
u.u_segflg = 1;
u.u_error = 0;
writei(ip);
if(u.u_error)
ip->i_size = siz;
prele(ip);
}
/*
* Produce a pseudo-floating point representation
* with 3 bits base-8 exponent, 13 bits fraction.
*/
compress(t)
register time_t t;
{
register exp = 0, round = 0;
while (t >= 8192) {
exp++;
round = t&04;
t >>= 3;
}
if (round) {
t++;
if (t >= 8192) {
t >>= 3;
exp++;
}
}
return((exp<<13) + t);
}
#endif ACCT