|
|
researchv10 Norman
/*
* Disk usage by user
*/
char *dargv[] = {
"/dev/usr",
0
};
#include <stdio.h>
#include <ctype.h>
#include <pwd.h>
#include <sys/param.h>
#include <sys/ino.h>
#include <sys/inode.h>
#include <sys/filsys.h>
#include <sys/stat.h>
#define YEAR (24L*60L*60L*365L) /* a year in seconds */
#define ITABSZ 256
#define ISIZ(dev) (BSIZE(dev)/sizeof(struct dinode))
#define NUID 10000
struct filsys sblock;
struct dinode itab[ITABSZ];
struct du {
long blocks;
long nfiles;
double byteyears;
int uid;
char *name;
} du[NUID];
#define TSIZE 500
int sizes[TSIZE];
long overflow;
struct stat stb;
int dev;
long now;
int nflg;
int fflg;
int cflg;
int bflg;
int fi;
unsigned ino;
unsigned nfiles;
struct passwd *getpwent();
char *malloc();
char *copy();
main(argc, argv)
char **argv;
{
register int n;
register struct passwd *lp;
register char **p;
for(n=0; n<NUID; n++)
du[n].uid = n;
while((lp=getpwent()) != 0) {
n = lp->pw_uid;
if (n>NUID)
continue;
if(du[n].name)
continue;
du[n].name = copy(lp->pw_name);
}
now = time((long *) NULL);
if (argc == 1) {
for (p = dargv; *p;)
check(*p++);
report();
return(0);
}
while (--argc) {
argv++;
if (argv[0][0]=='-') {
if (argv[0][1]=='n')
nflg++;
else if (argv[0][1]=='f')
fflg++;
else if (argv[0][1]=='c')
cflg++;
else if (argv[0][1] == 'b')
bflg++;
} else
check(*argv);
}
report();
return(0);
}
check(file)
char *file;
{
register unsigned i, j;
register c;
fi = open(file, 0);
if (fi < 0) {
fprintf(stderr, "quot: cannot open %s\n", file);
return;
}
fstat(fi, &stb);
dev = stb.st_rdev;
printf("%s:\n", file);
sync();
bread(1, (char *)&sblock, sizeof sblock);
nfiles = (sblock.s_isize-2)*(BSIZE(dev)/sizeof(struct dinode));
ino = 0;
if (nflg) {
if (isdigit(c = getchar()))
ungetc(c, stdin);
else while (c!='\n' && c != EOF)
c = getchar();
}
for(i=2; ino<nfiles; i += ITABSZ/ISIZ(dev)) {
bread(i, (char *)itab, sizeof itab);
for (j=0; j<ITABSZ && ino<nfiles; j++) {
ino++;
acct(&itab[j]);
}
}
close(fi);
}
acct(ip)
register struct dinode *ip;
{
register n;
register char *np;
static fino;
if ((ip->di_mode&IFMT) == 0)
return;
n = ((ip->di_size+BSIZE(dev)-1)/BSIZE(dev)) * (BSIZE(dev)/1024);
if (cflg) {
if ((ip->di_mode&IFMT)!=IFDIR && (ip->di_mode&IFMT)!=IFREG)
return;
if (n >= TSIZE) {
overflow += n;
n = TSIZE-1;
}
sizes[n]++;
return;
}
if (ip->di_uid >= NUID || ip->di_uid<0)
return;
du[ip->di_uid].blocks += n;
du[ip->di_uid].nfiles++;
du[ip->di_uid].byteyears += (double) n * (double) (now - ip->di_mtime) / YEAR;
if (nflg) {
tryagain:
if (fino==0)
if (scanf("%d", &fino)<=0)
return;
if (fino > ino)
return;
if (fino<ino) {
while ((n=getchar())!='\n' && n!=EOF)
;
fino = 0;
goto tryagain;
}
if (np = du[ip->di_uid].name)
printf("%.7s ", np);
else
printf("%d ", ip->di_uid);
while ((n = getchar())==' ' || n=='\t')
;
putchar(n);
while (n!=EOF && n!='\n') {
n = getchar();
putchar(n);
}
fino = 0;
}
}
bread(bno, buf, cnt)
unsigned bno;
char *buf;
{
lseek(fi, (long)bno*BSIZE(dev), 0);
if (read(fi, buf, cnt) != cnt) {
printf("read error %u\n", bno);
exit(1);
}
}
qcmp(p1, p2)
register struct du *p1, *p2;
{
if (p1->blocks > p2->blocks)
return(-1);
if (p1->blocks < p2->blocks)
return(1);
return(strcmp(p1->name, p2->name));
}
report()
{
register i;
if (nflg)
return;
if (cflg) {
long t = 0;
for (i=0; i<TSIZE-1; i++)
if (sizes[i]) {
t += i*sizes[i];
printf("%d %d %ld\n", i, sizes[i], t);
}
printf("%d %d %ld\n", TSIZE-1, sizes[TSIZE-1], overflow+t);
return;
}
qsort(du, NUID, sizeof(du[0]), qcmp);
for (i=0; i<NUID; i++) {
if (du[i].blocks==0)
return;
printf("%6ld\t", du[i].blocks);
if (fflg)
printf("%5ld\t", du[i].nfiles);
if (bflg)
printf("%15.2f\t", du[i].byteyears);
if (du[i].name)
printf("%s\n", du[i].name);
else
printf("#%d\n", du[i].uid);
}
}
char *
copy(s)
char *s;
{
register char *p;
register n;
for(n=0; s[n]; n++)
;
p = malloc((unsigned)n+1);
for(n=0; p[n] = s[n]; n++)
;
return(p);
}
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.