/* * LTbasic.c -- Lsof Test basic tests * * The basic tests measure the finding by lsof of its own open CWD, open * executable (when possible), and open /dev/kmem files. * * V. Abell * Purdue University */ /* * Copyright 2002 Purdue Research Foundation, West Lafayette, Indiana * 47907. All rights reserved. * * Written by V. Abell. * * This software is not subject to any license of the American Telephone * and Telegraph Company or the Regents of the University of California. * * Permission is granted to anyone to use this software for any purpose on * any computer system, and to alter it and redistribute it freely, subject * to the following restrictions: * * 1. Neither the authors nor Purdue University are responsible for any * consequences of the use of this software. * * 2. The origin of this software must not be misrepresented, either by * explicit claim or by omission. Credit to the authors and Purdue * University must appear in documentation and sources. * * 3. Altered versions must be plainly marked as such, and must not be * misrepresented as being the original software. * * 4. This notice may not be removed or altered. */ #ifndef lint static char copyright[] = "@(#) Copyright 2002 Purdue Research Foundation.\nAll rights reserved.\n"; #endif #include "LsofTest.h" #include "lsof_fields.h" /* * Local definitions */ /* * Globals */ char *Pn = (char *)NULL; /* program name */ /* * Local function prototypes */ _PROTOTYPE(static void cleanup,(void)); _PROTOTYPE(static char *tstlsof,(char **texec, char **tkmem, char **tproc)); /* * Main program for dialects that support locking tests. */ int main(argc, argv) int argc; /* argument count */ char *argv[]; /* arguments */ { char buf[2048]; /* temporary buffer */ char *em; /* error message pointer */ char *texec = (char *)NULL; /* lsof executable test result */ char *tkmem = (char *)NULL; /* /dev/kmem test result */ char *tproc = (char *)NULL; /* lsof process test result */ int xv = 0; /* exit value */ /* * Get program name and PID, issue start message, and build space prefix. */ if ((Pn = strrchr(argv[0], '/'))) Pn++; else Pn = argv[0]; (void) printf("%s ... ", Pn); (void) fflush(stdout); PrtMsg((char *)NULL, Pn); /* * Process arguments. */ if (ScanArg(argc, argv, "h", Pn)) xv = 1; if (xv || LTopt_h) { (void) PrtMsg("usage: [-h]", Pn); PrtMsgX (" -h print help (this panel)", Pn, cleanup, xv); } /* * See if lsof can be executed and can access kernel memory. */ if ((em = IsLsofExec())) (void) PrtMsgX(em, Pn, cleanup, 1); if ((em = CanRdKmem())) (void) PrtMsgX(em, Pn, cleanup, 1); /* * Test lsof. */ if ((em = tstlsof(&texec, &tkmem, &tproc))) PrtMsg(em, Pn); if (texec) PrtMsg(texec, Pn); if (tkmem) PrtMsg(tkmem, Pn); if (tproc) PrtMsg(tproc, Pn); /* * Compute exit value and exit. */ if (em || texec || tkmem || tproc) { if (strcmp(LT_DEF_LSOF_PATH, LsofPath)) { PrtMsg (" ", Pn); PrtMsg ("Hint: you used the LT_LSOF_PATH environment variable to", Pn); PrtMsg (" specify this path to the lsof executable:\n", Pn); (void) snprintf(buf, sizeof(buf) - 1, " %s\n", LsofPath); buf[sizeof(buf) - 1] = '\0'; PrtMsg (buf, Pn); PrtMsgX(" Make sure its revision is 4.63 or higher.", Pn, cleanup, 1); } else PrtMsgX("", Pn, cleanup, 1); } (void) PrtMsgX("OK", Pn, cleanup, 0); return(0); } /* * cleanup() -- release resources */ static void cleanup() { } /* * tstlsof() -- test for the lsof process */ static char * tstlsof(texec, tkmem, tproc) char **texec; /* result of the executable test */ char **tkmem; /* result of the /dev/kmem test */ char **tproc; /* result of the lsof process test */ { char buf[2048]; /* temporary buffer */ char *cem; /* current error message pointer */ LTfldo_t *cmdp; /* command pointer */ LTdev_t cwddc; /* CWD device components */ struct stat cwdsb; /* CWD stat(2) buffer */ LTfldo_t *devp; /* device pointer */ int execs = 0; /* executable status */ int fdn; /* FD is a number */ LTfldo_t *fdp; /* file descriptor pointer */ LTfldo_t *fop; /* field output pointer */ char ibuf[64]; /* inode string buffer */ LTfldo_t *inop; /* inode number pointer */ LTdev_t kmemdc; /* /dev/kmem device components */ int kmems = 0; /* kmem status */ struct stat kmemsb; /* /dev/kmem stat(2) buffer */ LTdev_t lsofdc; /* lsof device components */ struct stat lsofsb; /* lsof stat(2) buffer */ int nf; /* number of fields */ char *opv[4]; /* option vector for ExecLsof() */ char *pem = (char *)NULL; /* previous error message */ pid_t pid; /* PID */ int pids = 0; /* PID found status */ int procs = 0; /* process status */ LTfldo_t *rdevp; /* raw device pointer */ char *tcp; /* temporary character pointer */ int ti; /* temporary integer */ LTdev_t tmpdc; /* temporary device components */ LTfldo_t *typ; /* file type pointer */ int xwhile; /* exit while() flag */ /* * Get lsof executable's stat(2) information. */ if (stat(LsofPath, &lsofsb)) { (void) snprintf(buf, sizeof(buf) - 1, "ERROR!!! stat(%s): %s", LsofPath, strerror(errno)); buf[sizeof(buf) - 1] = '\0'; cem = MkStrCpy(buf, &ti); if (pem) (void) PrtMsg(pem, Pn); pem = cem; execs = 1; } else if ((cem = ConvStatDev(&lsofsb.st_dev, &lsofdc))) { if (pem) (void) PrtMsg(pem, Pn); pem = cem; execs = 1; } #if defined(LT_KMEM) /* * Get /dev/kmem's stat(2) information. */ if (stat("/dev/kmem", &kmemsb)) { (void) snprintf(buf, sizeof(buf) - 1, "ERROR!!! can't stat(2) /dev/kmem: %s", strerror(errno)); buf[sizeof(buf) - 1] = '\0'; cem = MkStrCpy(buf, &ti); if (pem) (void) PrtMsg(pem, Pn); pem = cem; kmems = 1; } else if ((cem = ConvStatDev(&kmemsb.st_rdev, &kmemdc))) { if (pem) (void) PrtMsg(pem, Pn); pem = cem; kmems = 1; } #else /* !defined(LT_KMEM) */ kmems = 1; #endif /* defined(LT_KMEM) */ /* * Get CWD's stat(2) information. */ if (stat(".", &cwdsb)) { (void) snprintf(buf, sizeof(buf) - 1, "ERROR!!! stat(.): %s", strerror(errno)); buf[sizeof(buf) - 1] = '\0'; cem = MkStrCpy(buf, &ti); if (pem) (void) PrtMsg(pem, Pn); pem = cem; procs = 1; } else if ((cem = ConvStatDev(&cwdsb.st_dev, &cwddc))) { if (pem) (void) PrtMsg(pem, Pn); pem = cem; procs = 1; } /* * Complete the option vector and start lsof execution. */ ti = 0; #if defined(USE_LSOF_C_OPT) opv[ti++] = "-C"; #endif /* defined(USE_LSOF_C_OPT) */ #if defined(USE_LSOF_X_OPT) opv[ti++] = "-X"; #endif /* defined(USE_LSOF_X_OPT) */ opv[ti++] = "-clsof"; opv[ti] = (char *)NULL; if ((cem = ExecLsof(opv))) { if (pem) (void) PrtMsg(pem, Pn); return(cem); } /* * Read lsof output. */ xwhile = execs + kmems + procs; while ((xwhile < 3) && (fop = RdFrLsof(&nf, &cem))) { if (pem) (void) PrtMsg(pem, Pn); pem = cem; switch (fop->ft) { case LSOF_FID_PID: /* * This is a process information line. */ pid = (pid_t)atoi(fop->v); pids = 1; cmdp = (LTfldo_t *)NULL; for (fop++, ti = 1; ti < nf; fop++, ti++) { switch (fop->ft) { case LSOF_FID_CMD: cmdp = fop; break; } } if (!cmdp || (pid != LsofPid)) pids = 0; break; case LSOF_FID_FD: /* * This is a file descriptor line. Scan its fields. */ if (!pids) break; devp = inop = rdevp = typ = (LTfldo_t *)NULL; fdp = fop; for (fop++, ti = 1; ti < nf; fop++, ti++) { switch(fop->ft) { case LSOF_FID_DEVN: devp = fop; break; case LSOF_FID_INODE: inop = fop; break; case LSOF_FID_RDEV: rdevp = fop; break; case LSOF_FID_TYPE: typ = fop; break; } } /* * A file descriptor line has been processes. * * Set the descriptor's numeric status. * * Check descriptor by FD type. */ for (fdn = 0, tcp = fdp->v; *tcp; tcp++) { if (!isdigit((unsigned char)*tcp)) { fdn = -1; break; } fdn = (fdn * 10) + (int)(*tcp - '0'); } if (!procs && (fdn == -1) && !strcasecmp(fdp->v, "cwd") && typ && (!strcasecmp(typ->v, "DIR") || !strcasecmp(typ->v, "VDIR")) ) { /* * This is the CWD for the process. Make sure its information * matches what stat(2) said about the CWD. */ if (!devp || !inop) break; if ((cem = ConvLsofDev(devp->v, &tmpdc))) { if (pem) (void) PrtMsg(pem, Pn); pem = cem; break; } (void) snprintf(ibuf, sizeof(ibuf) - 1, "%u", (unsigned int)cwdsb.st_ino); ibuf[sizeof(ibuf) - 1] = '\0'; if ((tmpdc.maj == cwddc.maj) && (tmpdc.min == cwddc.min) && (tmpdc.unit == cwddc.unit) && !strcmp(inop->v, ibuf) ) { procs = 1; xwhile++; } break; } if (!kmems && (fdn >= 0) && typ && (!strcasecmp(typ->v, "CHR") || !strcasecmp(typ->v, "VCHR")) ) { /* * /dev/kmem hasn't been found and this is an open character device * file with a numeric descriptor. * * See if it is /dev/kmem. */ if (!inop || !rdevp) break; if ((cem = ConvLsofDev(rdevp->v, &tmpdc))) { if (pem) (void) PrtMsg(pem, Pn); pem = cem; break; } (void) snprintf(ibuf, sizeof(ibuf) - 1, "%u", (unsigned int)kmemsb.st_ino); ibuf[sizeof(ibuf) - 1] = '\0'; if ((tmpdc.maj == kmemdc.maj) && (tmpdc.min == kmemdc.min) && (tmpdc.unit == kmemdc.unit) && !strcmp(inop->v, ibuf) ) { kmems = 1; xwhile++; } break; } if (!execs && (fdn == -1) && typ && (!strcasecmp(typ->v, "REG") || !strcasecmp(typ->v, "VREG")) ) { /* * If this is a regular file with a non-numeric FD, it may be the * executable. */ if (!devp || !inop) break; if ((cem = ConvLsofDev(devp->v, &lsofdc))) { if (pem) (void) PrtMsg(pem, Pn); pem = cem; break; } (void) snprintf(ibuf, sizeof(ibuf) - 1, "%u", (unsigned int)lsofsb.st_ino); ibuf[sizeof(ibuf) - 1] = '\0'; if ((tmpdc.maj == lsofdc.maj) && (tmpdc.min == lsofdc.min) && (tmpdc.unit == lsofdc.unit) && !strcmp(inop->v, ibuf) ) { execs = 1; xwhile++; } } } } (void) StopLsof(); if (!execs) *texec = "ERROR!!! open lsof executable wasn't found."; if (!kmems) *tkmem = "ERROR!!! open lsof /dev/kmem usage wasn't found."; if (!procs) *tproc = "ERROR!!! lsof process wasn't found."; return(pem); }