#ifndef lint
static char copyright[] =
"@(#) Copyright 1994 Purdue Research Foundation.\nAll rights reserved.\n";
static char *rcsid = "$Id: main.c,v 1.38 2001/11/01 20:19:44 abe Exp $";
#endif
#include "lsof.h"
static int GObk[] = { 1, 1 };
static char GOp;
static char *GOv = (char *)NULL;
static int GOx1 = 1;
static int GOx2 = 0;
_PROTOTYPE(static int GetOpt,(int ct, char *opt[], char *rules, int *err));
_PROTOTYPE(static char *sv_fmt_str,(char *f));
int
main(argc, argv)
int argc;
char *argv[];
{
int c, i, n, rv;
char *cp;
int err = 0;
int ev = 0;
int fh = 0;
long l;
MALLOC_S len;
struct lfile *lf;
struct nwad *np, *npn;
char options[128];
int rc = 0;
struct stat sb;
struct sfile *sfp;
struct lproc **slp = (struct lproc **)NULL;
int sp = 0;
struct str_lst *str;
int version = 0;
if ((Pn = strrchr(argv[0], '/')))
Pn++;
else
Pn = argv[0];
#if defined(HASSETLOCALE)
(void) setlocale(LC_CTYPE, "");
#endif
Mypid = getpid();
if ((Mygid = (gid_t)getgid()) != getegid())
Setgid = 1;
if ((Myuid = (uid_t)getuid()) && !geteuid())
Setuidroot = 1;
if (!(Namech = (char *)malloc(MAXPATHLEN + 1))) {
(void) fprintf(stderr, "%s: no space for name buffer\n", Pn);
Exit(1);
}
Namechl = (size_t)(MAXPATHLEN + 1);
(void) snpf(options, sizeof(options),
"?a%sbc:D:d:%sf:F:g:hi:%slL:%sMnNo:Op:Pr:%ssS:tT:u:UvVw%s",
#if defined(HAS_AFS) && defined(HASAOPT)
"A:",
#else
"",
#endif
#if defined(HASNCACHE)
"C",
#else
"",
#endif
#if defined(HASKOPT)
"k:",
#else
"",
#endif
#if defined(HASMOPT)
"m:",
#else
"",
#endif
#if defined(HASPPID)
"R",
#else
"",
#endif
#if defined(HASXOPT)
# if defined(HASXOPT_ROOT)
(Myuid == 0) ? "X" : ""
# else
"X"
# endif
#else
""
#endif
);
while ((c = GetOpt(argc, argv, options, &rv)) != EOF) {
if (rv) {
err = 1;
continue;
}
switch(c) {
case 'a':
Fand = 1;
break;
#if defined(HAS_AFS) && defined(HASAOPT)
case 'A':
if (!GOv || *GOv == '-' || *GOv == '+') {
(void) fprintf(stderr, "%s: -A not followed by path\n", Pn);
err = 1;
} else
AFSApath = GOv;
break;
#endif
case 'b':
Fblock = 1;
break;
case 'c':
if (GOv && (*GOv == '/')) {
if (enter_cmd_rx(GOv))
err = 1;
} else {
if (enter_str_lst("c", GOv, &Cmdl))
err = 1;
}
break;
#if defined(HASNCACHE)
case 'C':
Fncache = (GOp == '-') ? 0 : 1;
break;
#endif
case 'd':
if (GOp == '+') {
if (enter_dir(GOv, 0))
err = 1;
else
Selflags |= SELNM;
} else {
if (enter_fd(GOv))
err = 1;
}
break;
case 'D':
if (GOp == '+') {
if (enter_dir(GOv, 1))
err = 1;
else
Selflags |= SELNM;
} else {
#if defined(HASDCACHE)
if (ctrl_dcache(GOv))
err = 1;
#else
(void) fprintf(stderr, "%s: unsupported option: -D\n", Pn);
err = 1;
#endif
}
break;
case 'f':
if (!GOv || *GOv == '-' || *GOv == '+') {
Ffilesys = (GOp == '+') ? 2 : 1;
break;
}
#if defined(HASFSTRUCT)
for (; *GOv; GOv++) {
switch (*GOv) {
case 'c':
case 'C':
if (GOp == '+')
Fsv |= FSV_CT;
else
Fsv &= (unsigned char)~FSV_CT;
break;
case 'f':
case 'F':
if (GOp == '+')
Fsv |= FSV_FA;
else
Fsv &= (unsigned char)~FSV_FA;
break;
case 'g':
case 'G':
if (GOp == '+')
Fsv |= FSV_FG;
else
Fsv &= (unsigned char)~FSV_FG;
FsvFlagX = (*GOv == 'G') ? 1 : 0;
break;
case 'n':
case 'N':
if (GOp == '+')
Fsv |= FSV_NI;
else
Fsv &= (unsigned char)~FSV_NI;
break;
default:
(void) fprintf(stderr,
"%s: unknown file struct option: %c\n", Pn, *GOv);
err++;
}
}
#else
(void) fprintf(stderr,
"%s: unknown string for %cf: %s\n", Pn, GOp, GOv);
err++;
#endif
break;
case 'F':
if (!GOv || *GOv == '-' || *GOv == '+'
|| strcmp(GOv, "0") == 0) {
if (GOv) {
if (*GOv == '-' || *GOv == '+') {
GOx1 = GObk[0];
GOx2 = GObk[1];
} else if (*GOv == '0')
Terminator = '\0';
}
for (i = 0; FieldSel[i].nm; i++) {
#if !defined(HASPPID)
if (FieldSel[i].id == LSOF_FID_PPID)
continue;
#endif
#if !defined(HASFSTRUCT)
if (FieldSel[i].id == LSOF_FID_CT
|| FieldSel[i].id == LSOF_FID_FA
|| FieldSel[i].id == LSOF_FID_FG
|| FieldSel[i].id == LSOF_FID_NI)
continue;
#endif
if (FieldSel[i].id == LSOF_FID_RDEV)
continue;
FieldSel[i].st = 1;
if (FieldSel[i].opt && FieldSel[i].ov)
*(FieldSel[i].opt) |= FieldSel[i].ov;
}
#if defined(HASFSTRUCT)
Ffield = FsvFlagX = 1;
#else
Ffield = 1;
#endif
break;
}
if (strcmp(GOv, "?") == 0) {
fh = 1;
break;
}
for (; *GOv; GOv++) {
for (i = 0; FieldSel[i].nm; i++) {
#if !defined(HASPPID)
if (FieldSel[i].id == LSOF_FID_PPID)
continue;
#endif
#if !defined(HASFSTRUCT)
if (FieldSel[i].id == LSOF_FID_CT
|| FieldSel[i].id == LSOF_FID_FA
|| FieldSel[i].id == LSOF_FID_FG
|| FieldSel[i].id == LSOF_FID_NI)
continue;
#endif
if (FieldSel[i].id == *GOv) {
FieldSel[i].st = 1;
if (FieldSel[i].opt && FieldSel[i].ov)
*(FieldSel[i].opt) |= FieldSel[i].ov;
#if defined(HASFSTRUCT)
if (i == LSOF_FIX_FG)
FsvFlagX = 1;
#endif
if (i == LSOF_FIX_TERM)
Terminator = '\0';
break;
}
}
if ( ! FieldSel[i].nm) {
(void) fprintf(stderr,
"%s: unknown field: %c\n", Pn, *GOv);
err++;
}
}
Ffield = 1;
break;
case 'g':
if (GOv) {
if (*GOv == '-' || *GOv == '+') {
GOx1 = GObk[0];
GOx2 = GObk[1];
} else if (enter_id(PGID, GOv))
err = 1;
}
Fpgid = 1;
break;
case 'h':
case '?':
Fhelp = 1;
break;
case 'i':
if (!GOv || *GOv == '-' || *GOv == '+') {
Fnet = 1;
FnetTy = 0;
if (GOv) {
GOx1 = GObk[0];
GOx2 = GObk[1];
}
break;
}
if (enter_network_address(GOv))
err = 1;
break;
#if defined(HASKOPT)
case 'k':
if (!GOv || *GOv == '-' || *GOv == '+') {
(void) fprintf(stderr, "%s: -k not followed by path\n", Pn);
err = 1;
} else
Nmlst = GOv;
break;
#endif
case 'l':
Futol = 0;
break;
case 'L':
Fnlink = (GOp == '+') ? 1 : 0;
if (!GOv || *GOv == '-' || *GOv == '+') {
if (GOv) {
GOx1 = GObk[0];
GOx2 = GObk[1];
}
Nlink = 0l;
break;
}
for (cp = GOv, l = 0l, n = 0; *cp; cp++) {
if (!isdigit((unsigned char)*cp))
break;
l = (l * 10l) + ((long)*cp - (long)'0');
n++;
}
if (n) {
if (GOp != '+') {
(void) fprintf(stderr,
"%s: no number may follow -L\n", Pn);
err = 1;
} else {
Nlink = l;
Selflags |= SELNLINK;
}
} else
Nlink = 0l;
if (*cp) {
GOx1 = GObk[0];
GOx2 = GObk[1] + n;
}
break;
#if defined(HASMOPT)
case 'm':
if (!GOv || *GOv == '-' || *GOv == '+') {
(void) fprintf(stderr, "%s: -m not followed by path\n", Pn);
err = 1;
} else
Memory = GOv;
break;
#endif
case 'M':
FportMap = (GOp == '+') ? 1 : 0;
break;
case 'n':
Fhost = (GOp == '-') ? 0 : 1;
break;
case 'N':
Fnfs = 1;
break;
case 'o':
if (!GOv || *GOv == '-' || *GOv == '+') {
Foffset = 1;
if (GOv) {
GOx1 = GObk[0];
GOx2 = GObk[1];
}
break;
}
for (cp = GOv, i = n = 0; *cp; cp++) {
if (!isdigit((unsigned char)*cp))
break;
i = (i * 10) + ((int)*cp - '0');
n++;
}
if (n)
OffDecDig = i;
else
Foffset = 1;
if (*cp) {
GOx1 = GObk[0];
GOx2 = GObk[1] + n;
}
break;
case 'O':
Fovhd = (GOp == '-') ? 1 : 0;
break;
case 'p':
if (enter_id(PID, GOv))
err = 1;
break;
case 'P':
Fport = (GOp == '-') ? 0 : 1;
break;
case 'r':
if (GOp == '+')
ev = rc = 1;
if (!GOv || *GOv == '-' || *GOv == '+') {
if (GOv) {
GOx1 = GObk[0];
GOx2 = GObk[1];
}
RptTm = RPTTM;
break;
}
for (cp = GOv, i = n = 0; *cp; cp++) {
if (!isdigit((unsigned char)*cp))
break;
i = (i * 10) + ((int)*cp - '0');
n++;
}
if (n)
RptTm = i;
else
RptTm = RPTTM;
if (*cp) {
GOx1 = GObk[0];
GOx2 = GObk[1] + n;
}
break;
#if defined(HASPPID)
case 'R':
Fppid = 1;
break;
#endif
case 's':
Fsize = 1;
break;
case 'S':
if (!GOv || *GOv == '-' || *GOv == '+') {
if (GOv) {
GOx1 = GObk[0];
GOx2 = GObk[1];
}
TmLimit = TMLIMIT;
break;
}
for (cp = GOv, i = n = 0; *cp; cp++) {
if (!isdigit((unsigned char)*cp))
break;
i = (i * 10) + ((int)*cp - '0');
n++;
}
if (n)
TmLimit = i;
else
TmLimit = TMLIMIT;
if (*cp) {
GOx1 = GObk[0];
GOx2 = GObk[1] + n;
}
if (TmLimit < TMLIMMIN) {
(void) fprintf(stderr,
"%s: WARNING: -S time (%d) changed to %d\n",
Pn, TmLimit, TMLIMMIN);
TmLimit = TMLIMMIN;
}
break;
case 't':
Fterse = Fwarn = 1;
break;
case 'T':
if (!GOv || *GOv == '-' || *GOv == '+') {
if (GOv) {
GOx1 = GObk[0];
GOx2 = GObk[1];
}
Ftcptpi = (GOp == '-') ? 0 : TCPTPI_STATE;
break;
}
for (Ftcptpi = 0; *GOv; GOv++) {
switch (*GOv) {
#if defined(HASTCPTPIQ)
case 'q':
Ftcptpi |= TCPTPI_QUEUES;
break;
#endif
case 's':
Ftcptpi |= TCPTPI_STATE;
break;
#if defined(HASTCPTPIW)
case 'w':
Ftcptpi |= TCPTPI_WINDOWS;
break;
#endif
default:
(void) fprintf(stderr,
"%s: unsupported TCP/TPI info selection: %c\n",
Pn, *GOv);
err = 1;
}
}
break;
case 'u':
if (enter_uid(GOv))
err = 1;
break;
case 'U':
Funix = 1;
break;
case 'v':
version = 1;
break;
case 'V':
Fverbose = 1;
break;
case 'w':
Fwarn = (GOp == '+') ? 0 : 1;
break;
#if defined(HASXOPT)
case 'X':
Fxopt = Fxopt ? 0 : 1;
break;
#endif
default:
(void) fprintf(stderr, "%s: unknown option (%c)\n", Pn, c);
err = 1;
}
}
if (Fsize && Foffset) {
(void) fprintf(stderr, "%s: -o and -s are mutually exclusive\n",
Pn);
err++;
}
if (Ffield) {
if (Fterse) {
(void) fprintf(stderr,
"%s: -f and -t are mutually exclusive\n", Pn);
err++;
}
FieldSel[LSOF_FIX_PID].st = 1;
}
if (DChelp || err || Fhelp || fh || version)
usage(err ? 1 : 0, fh, version);
if (Suid && Nuid && Nuid < Mxuid) {
if (!(Suid = (struct seluid *)realloc((MALLOC_P *)Suid,
(MALLOC_S)(sizeof(struct seluid) * Nuid))))
{
(void) fprintf(stderr, "%s: can't realloc UID table\n", Pn);
Exit(1);
}
Mxuid = Nuid;
}
if (Cmdl || CmdRx)
Selflags |= SELCMD;
if (Fdl)
Selflags |= SELFD;
if (Fnet)
Selflags |= SELNET;
if (Fnfs)
Selflags |= SELNFS;
if (Funix)
Selflags |= SELUNX;
if (Npgid)
Selflags |= SELPGID;
if (Npid)
Selflags |= SELPID;
if (Nuid && Nuidincl)
Selflags |= SELUID;
if (Nwad)
Selflags |= SELNA;
if (GOx1 < argc)
Selflags |= SELNM;
if (Selflags == 0) {
if (Fand) {
(void) fprintf(stderr,
"%s: no select options to AND via -a\n", Pn);
usage(1, 0, 0);
}
Selflags = SELALL;
} else {
if (GOx1 >= argc && (Selflags & (SELNA|SELNET)) != 0
&& (Selflags & ~(SELNA|SELNET)) == 0)
Selinet = 1;
Selall = 0;
}
if (stat("/dev", &sb)) {
(void) fprintf(stderr, "%s: can't stat(/dev): %s\n", Pn,
strerror(errno));
Exit(1);
}
DevDev = sb.st_dev;
if (GOx1 < argc) {
if (ck_file_arg(GOx1, argc, argv, Ffilesys, 0, (struct stat *)NULL))
usage(1, 0, 0);
}
initialize();
if (Sfile)
(void) hashSfile();
#if defined(WILLDROPGID)
(void) dropgid();
#endif
#if defined(HASDCACHE)
if (DCstate)
readdev(0);
#endif
(void) snpf(options, sizeof(options), "0t%%%su", SZOFFPSPEC);
SzOffFmt_0t = sv_fmt_str(options);
(void) snpf(options, sizeof(options), "%%%su", SZOFFPSPEC);
SzOffFmt_d = sv_fmt_str(options);
(void) snpf(options, sizeof(options), "%%*%su", SZOFFPSPEC);
SzOffFmt_dv = sv_fmt_str(options);
(void) snpf(options, sizeof(options), "%%#%sx", SZOFFPSPEC);
SzOffFmt_x = sv_fmt_str(options);
if (RptTm)
CkPasswd = 1;
do {
gather_proc_info();
if (Nlproc > 1) {
if (Nlproc > sp) {
len = (MALLOC_S)(Nlproc * sizeof(struct lproc *));
sp = Nlproc;
if (!slp)
slp = (struct lproc **)malloc(len);
else
slp = (struct lproc **)realloc((MALLOC_P *)slp, len);
if (!slp) {
(void) fprintf(stderr,
"%s: no space for %d sort pointers\n", Pn, Nlproc);
Exit(1);
}
}
for (i = 0; i < Nlproc; i++) {
slp[i] = &Lproc[i];
}
(void) qsort((QSORT_P *)slp, (size_t)Nlproc,
(size_t)sizeof(struct lproc *), comppid);
}
if ((n = Nlproc)) {
#if defined(HASNCACHE)
NcacheReload = 1;
#endif
for (lf = Lf, print_init(); PrPass < 2; PrPass++) {
for (i = n = 0; i < Nlproc; i++) {
Lp = (Nlproc > 1) ? slp[i] : &Lproc[i];
if (Lp->pss) {
if (print_proc())
n++;
}
if (RptTm && PrPass)
(void) free_lproc(Lp);
}
}
Lf = lf;
}
if (RptTm) {
if (rc) {
if (!n)
break;
else
ev = 0;
}
if (Ffield) {
putchar(LSOF_FID_MARK);
putchar('\n');
} else
puts("=======");
(void) fflush(stdout);
(void) childx();
(void) sleep(RptTm);
Hdr = Nlproc = 0;
CkPasswd = 1;
}
} while (RptTm);
(void) childx();
rv = 0;
for (str = Cmdl; str; str = str->next) {
if (str->f)
continue;
rv = 1;
if (Fverbose) {
(void) printf("%s: command not located: ", Pn);
safestrprt(str->str, stdout, 1);
}
}
for (i = 0; i < NCmdRxU; i++) {
if (CmdRx[i].mc)
continue;
rv = 1;
if (Fverbose) {
(void) printf("%s: no command found for regex: ", Pn);
safestrprt(CmdRx[i].exp, stdout, 1);
}
}
for (sfp = Sfile; sfp; sfp = sfp->next) {
if (sfp->f)
continue;
rv = 1;
if (Fverbose) {
(void) printf("%s: no file%s use located: ", Pn,
sfp->type ? "" : " system");
safestrprt(sfp->aname, stdout, 1);
}
}
#if defined(HASPROCFS)
if (Procsrch && !Procfind) {
rv = 1;
if (Fverbose) {
(void) printf("%s: no file system use located: ", Pn);
safestrprt(Mtprocfs ? Mtprocfs->dir : HASPROCFS, stdout, 1);
}
}
{
struct procfsid *pfi;
for (pfi = Procfsid; pfi; pfi = pfi->next) {
if (!pfi->f) {
rv = 1;
if (Fverbose) {
(void) printf("%s: no file use located: ", Pn);
safestrprt(pfi->nm, stdout, 1);
}
}
}
}
#endif
if ((np = Nwad)) {
for (; np; np = np->next) {
if (!(cp = np->arg))
continue;
for (npn = np->next; npn; npn = npn->next) {
if (!npn->arg)
continue;
if (!strcmp(cp, npn->arg)) {
if (np->f)
npn->f = np->f;
else if (npn->f)
np->f = npn->f;
else
npn->f = 1;
}
}
}
for (np = Nwad; np; np = np->next) {
if (!np->f && (cp = np->arg)) {
rv = 1;
if (Fverbose) {
(void) printf("%s: Internet address not located: ", Pn);
safestrprt(cp ? cp : "(unknown)", stdout, 1);
}
}
}
}
if (Fnet && Fnet < 2) {
rv = 1;
if (Fverbose)
(void) printf("%s: no Internet files located\n", Pn);
}
if (Fnfs && Fnfs < 2) {
rv = 1;
if (Fverbose)
(void) printf("%s: no NFS files located\n", Pn);
}
for (i = 0; i < Npid; i++) {
if (Spid[i].f)
continue;
rv = 1;
if (Fverbose)
(void) printf("%s: process ID not located: %d\n",
Pn, Spid[i].i);
}
for (i = 0; i < Npgid; i++) {
if (Spgid[i].f)
continue;
rv = 1;
if (Fverbose)
(void) printf("%s: process group ID not located: %d\n",
Pn, Spgid[i].i);
}
for (i = 0; i < Nuid; i++) {
if (Suid[i].excl || Suid[i].f)
continue;
rv = 1;
if (Fverbose) {
if (Suid[i].lnm) {
(void) printf("%s: login name (UID %lu) not located: ",
Pn, (unsigned long)Suid[i].uid);
safestrprt(Suid[i].lnm, stdout, 1);
} else
(void) printf("%s: user ID not located: %lu\n", Pn,
(unsigned long)Suid[i].uid);
}
}
if (!rv && rc)
rv = ev;
if (!rv && ErrStat)
rv = 1;
Exit(rv);
return(rv);
}
static int
GetOpt(ct, opt, rules, err)
int ct;
char *opt[];
char *rules;
int *err;
{
register int c;
register char *cp = (char *)NULL;
if (GOx2 == 0) {
if (GOx1 >= ct
|| (opt[GOx1][0] != '-' && opt[GOx1][0] != '+')
|| !opt[GOx1][1])
return(EOF);
if (strcmp(opt[GOx1], "--") == 0 || strcmp(opt[GOx1], "++") == 0) {
GOx1++;
return(EOF);
}
GOp = opt[GOx1][0];
GOx2 = 1;
}
*err = 0;
if ((c = opt[GOx1][GOx2]) == ':') {
(void) fprintf(stderr,
"%s: colon is an illegal option character.\n", Pn);
*err = 1;
} else if (!(cp = strchr(rules, c))) {
(void) fprintf(stderr, "%s: illegal option character: %c\n", Pn, c);
*err = 2;
}
if (*err) {
if (opt[GOx1][++GOx2] == '\0') {
GOx1++;
GOx2 = 0;
}
return(c);
}
if (*(cp + 1) == ':') {
if(opt[GOx1][GOx2 + 1] != '\0') {
GObk[0] = GOx1;
GObk[1] = ++GOx2;
GOv = &opt[GOx1++][GOx2];
} else if (++GOx1 >= ct)
GOv = (char *)NULL;
else {
GObk[0] = GOx1;
GObk[1] = 0;
GOv = opt[GOx1];
if (strcmp(GOv, "--") == 0)
GOv = (char *)NULL;
else
GOx1++;
}
GOx2 = 0;
} else {
if (opt[GOx1][++GOx2] == '\0') {
GOx2 = 0;
GOx1++;
}
GOv = (char *)NULL;
}
return(c);
}
static char *
sv_fmt_str(f)
char *f;
{
char *cp;
MALLOC_S l;
l = (MALLOC_S)(strlen(f) + 1);
if (!(cp = (char *)malloc(l))) {
(void) fprintf(stderr,
"%s: can't allocate %d bytes for format: %s\n", Pn, (int)l, f);
Exit(1);
}
(void) snpf(cp, l, "%s", f);
return(cp);
}