|
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 |
}
|