Blame tests/LTnlink.c

Packit 6f02de
/*
Packit 6f02de
 * LTnlink.c -- Lsof Test nlink tests
Packit 6f02de
 *
Packit 6f02de
 * V. Abell
Packit 6f02de
 * Purdue University
Packit 6f02de
 */
Packit 6f02de
Packit 6f02de
Packit 6f02de
/*
Packit 6f02de
 * Copyright 2002 Purdue Research Foundation, West Lafayette, Indiana
Packit 6f02de
 * 47907.  All rights reserved.
Packit 6f02de
 *
Packit 6f02de
 * Written by V. Abell.
Packit 6f02de
 *
Packit 6f02de
 * This software is not subject to any license of the American Telephone
Packit 6f02de
 * and Telegraph Company or the Regents of the University of California.
Packit 6f02de
 *
Packit 6f02de
 * Permission is granted to anyone to use this software for any purpose on
Packit 6f02de
 * any computer system, and to alter it and redistribute it freely, subject
Packit 6f02de
 * to the following restrictions:
Packit 6f02de
 *
Packit 6f02de
 * 1. Neither the authors nor Purdue University are responsible for any
Packit 6f02de
 *    consequences of the use of this software.
Packit 6f02de
 *
Packit 6f02de
 * 2. The origin of this software must not be misrepresented, either by
Packit 6f02de
 *    explicit claim or by omission.  Credit to the authors and Purdue
Packit 6f02de
 *    University must appear in documentation and sources.
Packit 6f02de
 *
Packit 6f02de
 * 3. Altered versions must be plainly marked as such, and must not be
Packit 6f02de
 *    misrepresented as being the original software.
Packit 6f02de
 *
Packit 6f02de
 * 4. This notice may not be removed or altered.
Packit 6f02de
 */
Packit 6f02de
Packit 6f02de
#ifndef lint
Packit 6f02de
static char copyright[] =
Packit 6f02de
"@(#) Copyright 2002 Purdue Research Foundation.\nAll rights reserved.\n";
Packit 6f02de
#endif
Packit 6f02de
Packit 6f02de
#include "LsofTest.h"
Packit 6f02de
#include "lsof_fields.h"
Packit 6f02de
Packit 6f02de
Packit 6f02de
/*
Packit 6f02de
 * Pre-definitions that may be changed by specific dialects
Packit 6f02de
 */
Packit 6f02de
Packit 6f02de
#define	DO_TEST			/* do the test */
Packit 6f02de
Packit 6f02de
Packit 6f02de
/*
Packit 6f02de
 * Dialect-specific items
Packit 6f02de
 */
Packit 6f02de
Packit 6f02de
Packit 6f02de
#if	defined(LT_DIAL_darwin)
Packit 6f02de
/*
Packit 6f02de
 * Darwin-specific items
Packit 6f02de
 */
Packit 6f02de
Packit 6f02de
# if	defined(LT_KMEM)
Packit 6f02de
#undef	DO_TEST
Packit 6f02de
# endif	/* defined(LT_KMEM) */
Packit 6f02de
Packit 6f02de
#endif	/* defined(LT_DIAL_darwin) */
Packit 6f02de
Packit 6f02de
/*
Packit 6f02de
 * Globals
Packit 6f02de
 */
Packit 6f02de
Packit 6f02de
int Fd = -1;			/* test file descriptor; open if >= 0 */
Packit 6f02de
pid_t MyPid = (pid_t)0;		/* PID of this process */
Packit 6f02de
char *Path = (char *)NULL;	/* test file path; none if NULL */
Packit 6f02de
char *Pn = (char *)NULL;	/* program name */
Packit 6f02de
Packit 6f02de
Packit 6f02de
/*
Packit 6f02de
 * Local function prototypes
Packit 6f02de
 */
Packit 6f02de
Packit 6f02de
_PROTOTYPE(static void cleanup,(void));
Packit 6f02de
_PROTOTYPE(static char *FindFile,(char *opt, int *ff, int ie, LTdev_t *tfdc,
Packit 6f02de
				  char *ibuf, char *xlnk, char *szbuf));
Packit 6f02de
Packit 6f02de
Packit 6f02de
/*
Packit 6f02de
 * Main program
Packit 6f02de
 */
Packit 6f02de
Packit 6f02de
int
Packit 6f02de
main(argc, argv)
Packit 6f02de
    int argc;				/* argument count */
Packit 6f02de
    char *argv[];			/* arguments */
Packit 6f02de
{
Packit 6f02de
    char buf[2048];			/* temporary buffer */
Packit 6f02de
    int do_unlink = 1;			/* do the unlink test section */
Packit 6f02de
    char *em;				/* error message pointer */
Packit 6f02de
    int ff;				/* FindFile() file-found flag */
Packit 6f02de
    char ibuf[32];			/* inode number in ASCII */
Packit 6f02de
    char *opt;				/* lsof option */
Packit 6f02de
    int sz;				/* file size */
Packit 6f02de
    char szbuf[32];			/* file size in ASCII */
Packit 6f02de
    LTdev_t tfdc;			/* device components */
Packit 6f02de
    struct stat tfsb;			/* test file stat(2) buffer */
Packit 6f02de
    int ti, tj;				/* temporary indexes */
Packit 6f02de
    char xlnk[32];			/* expected link count in ASCII */
Packit 6f02de
    int xv = 0;				/* exit value */
Packit 6f02de
/*
Packit 6f02de
 * Get program name and PID, issue start message, and build space prefix.
Packit 6f02de
 */
Packit 6f02de
    if ((Pn = strrchr(argv[0], '/')))
Packit 6f02de
	Pn++;
Packit 6f02de
    else
Packit 6f02de
	Pn = argv[0];
Packit 6f02de
    MyPid = getpid();
Packit 6f02de
    (void) printf("%s ... ", Pn);
Packit 6f02de
    (void) fflush(stdout);
Packit 6f02de
    PrtMsg((char *)NULL, Pn);
Packit 6f02de
Packit 6f02de
#if	!defined(DO_TEST)
Packit 6f02de
/*
Packit 6f02de
 * Quit if lsof for this dialect doesn't support adequate nlink reporting.
Packit 6f02de
 */
Packit 6f02de
    (void) PrtMsgX(LT_DONT_DO_TEST, Pn, cleanup, 0);
Packit 6f02de
#endif	/* !defined(DO_TEST) */
Packit 6f02de
Packit 6f02de
/*
Packit 6f02de
 * Process arguments.
Packit 6f02de
 */
Packit 6f02de
    if (ScanArg(argc, argv, "hp:", Pn))
Packit 6f02de
	xv = 1;
Packit 6f02de
    if (xv || LTopt_h) {
Packit 6f02de
	(void) PrtMsg("usage: [-h] [-p path]", Pn);
Packit 6f02de
	PrtMsg       ("       -h       print help (this panel)", Pn);
Packit 6f02de
	PrtMsgX      ("       -p path  define test file path", Pn, cleanup, xv);
Packit 6f02de
    }
Packit 6f02de
/*
Packit 6f02de
 * See if lsof can be executed and can access kernel memory.
Packit 6f02de
 */
Packit 6f02de
    if ((em = IsLsofExec()))
Packit 6f02de
	(void) PrtMsgX(em, Pn, cleanup, 1);
Packit 6f02de
    if ((em = CanRdKmem()))
Packit 6f02de
	(void) PrtMsgX(em, Pn, cleanup, 1);
Packit 6f02de
/*
Packit 6f02de
 * Process the file path.
Packit 6f02de
 */
Packit 6f02de
    if (!(Path = LTopt_p)) {
Packit 6f02de
Packit 6f02de
    /*
Packit 6f02de
     * The file path was not supplied, so make one.
Packit 6f02de
     */
Packit 6f02de
	(void) snprintf(buf, sizeof(buf) - 1, "./config.LTnlink%ld",
Packit 6f02de
	    (long)MyPid);
Packit 6f02de
	buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
	Path = MkStrCpy(buf, &ti);
Packit 6f02de
    }
Packit 6f02de
/*
Packit 6f02de
 * Create the test file.
Packit 6f02de
 */
Packit 6f02de
    (void) unlink(Path);
Packit 6f02de
    if ((Fd = open(Path, O_RDWR|O_CREAT, 0600)) < 0) {
Packit 6f02de
	(void) fprintf(stderr, "ERROR!!!  can't create %s\n", Path);
Packit 6f02de
Packit 6f02de
print_file_error:
Packit 6f02de
Packit 6f02de
	MsgStat = 1;
Packit 6f02de
	(void) snprintf(buf, sizeof(buf) - 1, "      Errno %d: %s",
Packit 6f02de
	    errno, strerror(errno));
Packit 6f02de
	buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
	(void) PrtMsgX(buf, Pn, cleanup, 1);
Packit 6f02de
    }
Packit 6f02de
/*
Packit 6f02de
 * Write the test file to its expected size.
Packit 6f02de
 */
Packit 6f02de
    sz = sizeof(buf);
Packit 6f02de
    for (ti = 0; ti < sz; ti++) {
Packit 6f02de
	buf[ti] = (char)(ti & 0xff);
Packit 6f02de
    }
Packit 6f02de
    if (write(Fd, buf, sz) != sz) {
Packit 6f02de
	(void) fprintf(stderr, "ERROR!!!  can't write %d bytes to %s\n",
Packit 6f02de
	    sz, Path);
Packit 6f02de
	goto print_file_error;
Packit 6f02de
    }
Packit 6f02de
/*
Packit 6f02de
 * Fsync() the file.
Packit 6f02de
 */
Packit 6f02de
    if (fsync(Fd)) {
Packit 6f02de
	(void) fprintf(stderr, "ERROR!!!  can't fsync %s\n", Path);
Packit 6f02de
	goto print_file_error;
Packit 6f02de
    }
Packit 6f02de
/*
Packit 6f02de
 * Stat(2) the test file.
Packit 6f02de
 */
Packit 6f02de
    if (stat(Path, &tfsb)) {
Packit 6f02de
	(void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
	    "ERROR!!!  can't stat(2) %s: %s", Path, strerror(errno));
Packit 6f02de
	buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
	PrtMsgX(buf, Pn, cleanup, 1);
Packit 6f02de
    }
Packit 6f02de
/*
Packit 6f02de
 * Set the test file status to open and linked.
Packit 6f02de
 *
Packit 6f02de
 * Get the test file's parameters:
Packit 6f02de
 *
Packit 6f02de
 *    * device paramters in LTdev_t form;
Packit 6f02de
 *    * inode number in ASCII;
Packit 6f02de
 *    * link count in ASCII;
Packit 6f02de
 *    * file size in ASCII.
Packit 6f02de
 */
Packit 6f02de
    if ((em = ConvStatDev(&tfsb.st_dev, &tfdc)))
Packit 6f02de
	PrtMsgX(em, Pn, cleanup, 1);
Packit 6f02de
    (void) snprintf(ibuf, sizeof(ibuf) - 1, "%u", (unsigned int)tfsb.st_ino);
Packit 6f02de
    ibuf[sizeof(szbuf) - 1] = '\0';
Packit 6f02de
    (void) snprintf(xlnk, sizeof(xlnk) - 1, "%d", (int)tfsb.st_nlink);
Packit 6f02de
    ibuf[sizeof(szbuf) - 1] = '\0';
Packit 6f02de
    (void) snprintf(szbuf, sizeof(szbuf) - 1, "%d", sz);
Packit 6f02de
    szbuf[sizeof(szbuf) - 1] = '\0';
Packit 6f02de
/*
Packit 6f02de
 * See if the file is on an NFS file system.
Packit 6f02de
 */
Packit 6f02de
    (void) FindFile("-Na", &ff, 1, &tfdc, ibuf, xlnk, szbuf);
Packit 6f02de
    if (ff) {
Packit 6f02de
Packit 6f02de
    /*
Packit 6f02de
     * The file was found on an NFS file system.
Packit 6f02de
     */
Packit 6f02de
	(void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
	    "WARNING!!!  Test file %s is NFS mounted.", Path);
Packit 6f02de
	(void) PrtMsg(buf, Pn);
Packit 6f02de
	(void) PrtMsg(
Packit 6f02de
	    "  As a result this test probably won't be able to unlink it and",
Packit 6f02de
	    Pn);
Packit 6f02de
	(void) PrtMsg(
Packit 6f02de
	    "  find its open and unlinked instance with lsof's +L option.",
Packit 6f02de
	    Pn);
Packit 6f02de
	(void) PrtMsg(
Packit 6f02de
	    "  Therefore, that section of this test has been disabled.\n",
Packit 6f02de
	    Pn);
Packit 6f02de
	(void) PrtMsg(
Packit 6f02de
	    "  Hint: supply a path with the -p option to a file in a non-NFS",
Packit 6f02de
	    Pn);
Packit 6f02de
	(void) PrtMsg(
Packit 6f02de
	    "  file system that this test can write and unlink.\n",
Packit 6f02de
	    Pn);
Packit 6f02de
	(void) PrtMsg(
Packit 6f02de
	    "  See 00FAQ and 00TEST for more information.",
Packit 6f02de
	    Pn);
Packit 6f02de
	do_unlink = 0;
Packit 6f02de
    }
Packit 6f02de
/*
Packit 6f02de
 * Find the test file.
Packit 6f02de
 */
Packit 6f02de
    if ((em = FindFile("+L", &ff, 0, &tfdc, ibuf, xlnk, szbuf)))
Packit 6f02de
	(void) PrtMsgX(em, Pn, cleanup, 1);
Packit 6f02de
/*
Packit 6f02de
 * If the unlink test is enabled, do it.
Packit 6f02de
 */
Packit 6f02de
    if (do_unlink) {
Packit 6f02de
	if (unlink(Path)) {
Packit 6f02de
	    (void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
		"ERROR!!!  unlink(%s) failed: (%s).", Path, strerror(errno));
Packit 6f02de
	    buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
	    (void) PrtMsg(buf, Pn);
Packit 6f02de
	    (void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
		"  %s may be on a ZFS file system, where it", Path);
Packit 6f02de
	    buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
	    (void) PrtMsg(buf, Pn);
Packit 6f02de
	    (void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
	      "  is not possible for %s to unlink the file it has open.", Pn);
Packit 6f02de
	    buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
	    (void) PrtMsg(buf, Pn);
Packit 6f02de
	    (void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
	      "  To run the %s test, use the \"-p path\" option to specify",
Packit 6f02de
	      Pn);
Packit 6f02de
	    buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
	    (void) PrtMsg(buf, Pn);
Packit 6f02de
	    (void) PrtMsg(
Packit 6f02de
	      "  a file on a file system -- e.g., UFS -- that supports unlink",
Packit 6f02de
	      Pn);
Packit 6f02de
	    (void) PrtMsg(
Packit 6f02de
	      "  while the file is open.  Usually /tmp can do that -- e.g.,",
Packit 6f02de
	      Pn);
Packit 6f02de
	    (void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
	      "  run the test as \"./%s -p /tmp/<name>\".\n", Pn);
Packit 6f02de
	    buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
	    (void) PrtMsg(buf, Pn);
Packit 6f02de
	    (void) PrtMsgX( "  See 00FAQ and 00TEST for more information.",
Packit 6f02de
	      Pn, cleanup, 1);
Packit 6f02de
	}
Packit 6f02de
	for (opt = "+L1", ti = 0, tj = 30; ti < tj; ti++) {
Packit 6f02de
Packit 6f02de
	/*
Packit 6f02de
	 * Wait a while for the link count to be updated before concluding
Packit 6f02de
	 * lsof can't find the unlinked file.  Use "+L1" for only the first
Packit 6f02de
	 * third of the tries, then switch to "+L".
Packit 6f02de
	 */
Packit 6f02de
	    if ((ti + ti + ti) >= tj)
Packit 6f02de
		opt = "+L";
Packit 6f02de
	    if (!(em = FindFile(opt, &ff, 0, &tfdc, ibuf, "0", szbuf)))
Packit 6f02de
		break;
Packit 6f02de
	    if (ti)
Packit 6f02de
		(void) printf(".");
Packit 6f02de
	    else
Packit 6f02de
		(void) printf("waiting for link count update: .");
Packit 6f02de
	    (void) fflush(stdout);
Packit 6f02de
	    (void) sleep(2);
Packit 6f02de
	}
Packit 6f02de
	if (ti) {
Packit 6f02de
Packit 6f02de
	/*
Packit 6f02de
	 * End the delay message.
Packit 6f02de
	 */
Packit 6f02de
	    printf("\n");
Packit 6f02de
	    (void) fflush(stdout);
Packit 6f02de
	    MsgStat = 1;
Packit 6f02de
	}
Packit 6f02de
	if (em)
Packit 6f02de
	    (void) PrtMsgX(em, Pn, cleanup, 1);
Packit 6f02de
    }
Packit 6f02de
/*
Packit 6f02de
 * Exit successfully.
Packit 6f02de
 */
Packit 6f02de
    (void) PrtMsgX("OK", Pn, cleanup, 0);
Packit 6f02de
    return(0);
Packit 6f02de
}
Packit 6f02de
Packit 6f02de
Packit 6f02de
/*
Packit 6f02de
 * cleanup() -- release resources
Packit 6f02de
 */
Packit 6f02de
Packit 6f02de
static void
Packit 6f02de
cleanup()
Packit 6f02de
{
Packit 6f02de
    if (Fd >= 0) {
Packit 6f02de
	(void) close(Fd);
Packit 6f02de
	Fd = -1;
Packit 6f02de
    }
Packit 6f02de
    if (Path)
Packit 6f02de
	(void) unlink(Path);
Packit 6f02de
}
Packit 6f02de
Packit 6f02de
Packit 6f02de
/*
Packit 6f02de
 * FindFile() -- find a file with lsof
Packit 6f02de
 */
Packit 6f02de
Packit 6f02de
static char *
Packit 6f02de
FindFile(opt, ff, ie, tfdc, ibuf, xlnk, szbuf)
Packit 6f02de
    char *opt;				/* additional lsof options */
Packit 6f02de
    int *ff;				/* file-found response receptor */
Packit 6f02de
    int ie;				/* ignore errors if == 1 */
Packit 6f02de
    LTdev_t *tfdc;			/* test file device components */
Packit 6f02de
    char *ibuf;				/* inode number in ASCII */
Packit 6f02de
    char *xlnk;				/* expected link count */
Packit 6f02de
    char *szbuf;			/* file size in ASCII */
Packit 6f02de
{
Packit 6f02de
    char buf[2048];			/* temporary buffer */
Packit 6f02de
    char *cem;				/* current error message pointer */
Packit 6f02de
    LTfldo_t *cmdp;			/* command pointer */
Packit 6f02de
    LTfldo_t *devp;			/* device pointer */
Packit 6f02de
    LTfldo_t *fop;			/* field output pointer */
Packit 6f02de
    LTfldo_t *inop;			/* inode number pointer */
Packit 6f02de
    LTdev_t lsofdc;			/* lsof device components */
Packit 6f02de
    int nf;				/* number of fields */
Packit 6f02de
    LTfldo_t *nlkp;			/* nlink pointer */
Packit 6f02de
    char *opv[4];			/* option vector for ExecLsof() */
Packit 6f02de
    char *pem = (char *)NULL;		/* previous error message pointer */
Packit 6f02de
    pid_t pid;				/* PID */
Packit 6f02de
    int pids = 0;			/* PID found status */
Packit 6f02de
    LTfldo_t *szp;			/* size pointer */
Packit 6f02de
    char *tcp;				/* temporary character pointer */
Packit 6f02de
    int ti;				/* temporary integer */
Packit 6f02de
    LTfldo_t *typ;			/* file type pointer */
Packit 6f02de
/*
Packit 6f02de
 * Check the argument pointers.
Packit 6f02de
 *
Packit 6f02de
 * Set the file-found response false.
Packit 6f02de
 */
Packit 6f02de
    if (!ff || !ibuf || !szbuf || !tfdc || !xlnk)
Packit 6f02de
	(void) PrtMsgX("ERROR!!!  missing argument to FindFile()",
Packit 6f02de
		       Pn, cleanup, 1);
Packit 6f02de
    *ff = 0;
Packit 6f02de
/*
Packit 6f02de
 * Complete the option vector and start lsof execution.
Packit 6f02de
 */
Packit 6f02de
    ti = 0;
Packit 6f02de
    if (opt && *opt)
Packit 6f02de
	opv[ti++] = opt;
Packit 6f02de
Packit 6f02de
#if	defined(USE_LSOF_C_OPT)
Packit 6f02de
    opv[ti++] = "-C";
Packit 6f02de
#endif	/* defined(USE_LSOF_C_OPT) */
Packit 6f02de
Packit 6f02de
    if (strcmp(xlnk, "0"))
Packit 6f02de
	opv[ti++] = Path;
Packit 6f02de
    opv[ti] = (char *)NULL;
Packit 6f02de
    if ((cem = ExecLsof(opv))) {
Packit 6f02de
 	if (ie)
Packit 6f02de
	    return((char *)NULL);
Packit 6f02de
	return(cem);
Packit 6f02de
    }
Packit 6f02de
/*
Packit 6f02de
 * Read lsof output.
Packit 6f02de
 */
Packit 6f02de
    while (!*ff && (fop = RdFrLsof(&nf, &cem))) {
Packit 6f02de
	if (cem) {
Packit 6f02de
 	    if (ie)
Packit 6f02de
		return((char *)NULL);
Packit 6f02de
	    if (pem)
Packit 6f02de
		(void) PrtMsg(pem, Pn);
Packit 6f02de
	    return(cem);
Packit 6f02de
	}
Packit 6f02de
	switch (fop->ft) {
Packit 6f02de
	case LSOF_FID_PID:
Packit 6f02de
Packit 6f02de
	/*
Packit 6f02de
	 * This is a process information line.
Packit 6f02de
	 */
Packit 6f02de
	    pid = (pid_t)atoi(fop->v);
Packit 6f02de
	    pids = 1;
Packit 6f02de
	    cmdp = (LTfldo_t *)NULL;
Packit 6f02de
	    for (fop++, ti = 1; ti < nf; fop++, ti++) {
Packit 6f02de
		switch (fop->ft) {
Packit 6f02de
		case LSOF_FID_CMD:
Packit 6f02de
		    cmdp = fop;
Packit 6f02de
		    break;
Packit 6f02de
		}
Packit 6f02de
	    }
Packit 6f02de
	    if (!cmdp || (pid != MyPid))
Packit 6f02de
		pids = 0;
Packit 6f02de
	    break;
Packit 6f02de
	case LSOF_FID_FD:
Packit 6f02de
Packit 6f02de
	/*
Packit 6f02de
	 * This is a file descriptor line.  Make sure its number matches the
Packit 6f02de
	 * test file's descriptor number.
Packit 6f02de
	 */
Packit 6f02de
	    if (!pids)
Packit 6f02de
		break;
Packit 6f02de
	    for (ti = 0, tcp = fop->v; *tcp; tcp++) {
Packit 6f02de
Packit 6f02de
	    /*
Packit 6f02de
	     * Convert file descriptor to a number.
Packit 6f02de
	     */
Packit 6f02de
		if (*tcp == ' ')
Packit 6f02de
		    continue;
Packit 6f02de
		if (((int)*tcp < (int)'0') || ((int)*tcp > (int)'9')) {
Packit 6f02de
		    ti = -1;
Packit 6f02de
		    break;
Packit 6f02de
		}
Packit 6f02de
		ti = (ti * 10) + (int)*tcp - (int)'0'; 
Packit 6f02de
	    }
Packit 6f02de
	    if (Fd != ti)
Packit 6f02de
		break;
Packit 6f02de
	/*
Packit 6f02de
	 * Scan for device, inode, nlink, size and type fields.
Packit 6f02de
	 */
Packit 6f02de
	    devp = inop = nlkp = szp = typ = (LTfldo_t *)NULL;
Packit 6f02de
	    for (fop++, ti = 1; ti < nf; fop++, ti++) {
Packit 6f02de
		switch (fop->ft) {
Packit 6f02de
		case LSOF_FID_DEVN:
Packit 6f02de
		    devp = fop;
Packit 6f02de
		    break;
Packit 6f02de
		case LSOF_FID_INODE:
Packit 6f02de
		    inop = fop;
Packit 6f02de
		    break;
Packit 6f02de
		case LSOF_FID_NLINK:
Packit 6f02de
		    nlkp = fop;
Packit 6f02de
		    break;
Packit 6f02de
		case LSOF_FID_SIZE:
Packit 6f02de
		    szp = fop;
Packit 6f02de
		    break;
Packit 6f02de
		case LSOF_FID_TYPE:
Packit 6f02de
		    typ = fop;
Packit 6f02de
		    break;
Packit 6f02de
		}
Packit 6f02de
	    }
Packit 6f02de
	/*
Packit 6f02de
	 * Check the device, inode, and type of the file.
Packit 6f02de
	 */
Packit 6f02de
	    if (!devp || !inop || !szp || !typ)
Packit 6f02de
		break;
Packit 6f02de
	    if (strcasecmp(typ->v, "reg") && strcasecmp(typ->v, "vreg"))
Packit 6f02de
		break;
Packit 6f02de
	    if ((cem = ConvLsofDev(devp->v, &lsofdc))) {
Packit 6f02de
		if (pem)
Packit 6f02de
		    (void) PrtMsg(pem, Pn);
Packit 6f02de
		pem = cem;
Packit 6f02de
		break;
Packit 6f02de
	    }
Packit 6f02de
	    if ((tfdc->maj != lsofdc.maj)
Packit 6f02de
	    ||  (tfdc->min != lsofdc.min)
Packit 6f02de
	    ||  (tfdc->unit != lsofdc.unit)
Packit 6f02de
	    ||  strcmp(inop->v, ibuf)
Packit 6f02de
	    ) {
Packit 6f02de
		break;
Packit 6f02de
	    }
Packit 6f02de
	/*
Packit 6f02de
	 * Indicate the file was found.
Packit 6f02de
	 */
Packit 6f02de
	    *ff = 1;
Packit 6f02de
	/*
Packit 6f02de
	 * Check the size and link count.
Packit 6f02de
	 */
Packit 6f02de
	    if (!szp) {
Packit 6f02de
		(void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
		    "ERROR!!!  lsof didn't report a file size for %s", Path);
Packit 6f02de
		buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
		cem = MkStrCpy(buf, &ti);
Packit 6f02de
		if (pem)
Packit 6f02de
		    (void) PrtMsg(pem, Pn);
Packit 6f02de
		pem = cem;
Packit 6f02de
		break;
Packit 6f02de
	    }
Packit 6f02de
	    if (strcmp(szp->v, szbuf)) {
Packit 6f02de
		(void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
		    "ERROR!!!  wrong file size: expected %s, got %s",
Packit 6f02de
		    szbuf, szp->v);
Packit 6f02de
		buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
		cem = MkStrCpy(buf, &ti);
Packit 6f02de
		if (pem)
Packit 6f02de
		    (void) PrtMsg(pem, Pn);
Packit 6f02de
		pem = cem;
Packit 6f02de
		break;
Packit 6f02de
	    }
Packit 6f02de
	    if (!nlkp) {
Packit 6f02de
		if (strcmp(xlnk, "0")) {
Packit 6f02de
Packit 6f02de
		/*
Packit 6f02de
		 * If lsof returned no link count and the expected return is
Packit 6f02de
		 * not "0", it's an error.  Otherwise, interpret no link count
Packit 6f02de
		 * as equivalent to a "0" link count.
Packit 6f02de
		 */
Packit 6f02de
		    (void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
			"ERROR!!!  lsof didn't report a link count for %s",
Packit 6f02de
			Path);
Packit 6f02de
		    buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
		    cem = MkStrCpy(buf, &ti);
Packit 6f02de
		    if (pem)
Packit 6f02de
			(void) PrtMsg(pem, Pn);
Packit 6f02de
		    pem = cem;
Packit 6f02de
		    break;
Packit 6f02de
		}
Packit 6f02de
	    } else {
Packit 6f02de
		if (strcmp(nlkp->v, xlnk)) {
Packit 6f02de
		    (void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
			"ERROR!!!  wrong link count: expected %s, got %s",
Packit 6f02de
			xlnk, nlkp->v);
Packit 6f02de
		    buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
		    cem = MkStrCpy(buf, &ti);
Packit 6f02de
		    if (pem)
Packit 6f02de
			(void) PrtMsg(pem, Pn);
Packit 6f02de
		    pem = cem;
Packit 6f02de
		    break;
Packit 6f02de
		}
Packit 6f02de
	    }
Packit 6f02de
	/*
Packit 6f02de
	 * The requested file was located.  Return the previous error message
Packit 6f02de
	 * pointer unless errors are being ignored.  (The previous error
Packit 6f02de
	 * message pointer  will be NULL if no error was detected.)
Packit 6f02de
	 */
Packit 6f02de
	    (void) StopLsof();
Packit 6f02de
 	    if (ie)
Packit 6f02de
		return((char *)NULL);
Packit 6f02de
	    return(pem);
Packit 6f02de
	}
Packit 6f02de
    }
Packit 6f02de
/*
Packit 6f02de
 * Clean up and return.
Packit 6f02de
 */
Packit 6f02de
    (void) StopLsof();
Packit 6f02de
    if (!*ff && !ie) {
Packit 6f02de
	if (pem)
Packit 6f02de
	    (void) PrtMsg(pem, Pn);
Packit 6f02de
	(void) snprintf(buf, sizeof(buf) - 1,
Packit 6f02de
	    "ERROR!!!  %s test file %s not found by lsof",
Packit 6f02de
	    strcmp(xlnk, "0") ? "linked" : "unlinked",
Packit 6f02de
	    Path);
Packit 6f02de
	buf[sizeof(buf) - 1] = '\0';
Packit 6f02de
	pem = MkStrCpy(buf, &ti);
Packit 6f02de
    }
Packit 6f02de
    if (ie)
Packit 6f02de
	return((char *)NULL);
Packit 6f02de
    return(pem);
Packit 6f02de
}