#ifndef lint
static char copyright[] =
"@(#) Copyright 2002 Purdue Research Foundation.\nAll rights reserved.\n";
#endif
#include "LsofTest.h"
#include "lsof_fields.h"
#define OFFTST_STAT 1
#if defined(LT_DIAL_linux)
#undef OFFTST_STAT
#define OFFTST_STAT 0
_PROTOTYPE(static int ck_Linux_offset_support,(void));
#endif
#define TYTST_SZ 0
#define TYTST_0to 1
#define TYTST_0xo 2
#define TSTFSZ 32768
int Fd = -1;
pid_t MyPid = (pid_t)0;
char *Path = (char *)NULL;
char *Pn = (char *)NULL;
_PROTOTYPE(static void cleanup,(void));
_PROTOTYPE(static char *testlsof,(int tt, char *opt, char *xval));
int
main(argc, argv)
int argc;
char *argv[];
{
char buf[2048];
int do_offt = OFFTST_STAT;
char *em;
int ti;
char *tcp;
char *tstsz = (char *)NULL;
char *tst0to = (char *)NULL;
char *tst0xo = (char *)NULL;
int xv = 0;
char xbuf[64];
if ((Pn = strrchr(argv[0], '/')))
Pn++;
else
Pn = argv[0];
MyPid = getpid();
(void) printf("%s ... ", Pn);
(void) fflush(stdout);
PrtMsg((char *)NULL, Pn);
if (ScanArg(argc, argv, "hp:", Pn))
xv = 1;
if (xv || LTopt_h) {
(void) PrtMsg("usage: [-h] [-p path]", Pn);
PrtMsg (" -h print help (this panel)", Pn);
PrtMsgX (" -p path define test file path", Pn, cleanup, xv);
}
#if defined(LT_DIAL_linux)
do_offt = ck_Linux_offset_support();
#endif
if ((em = IsLsofExec()))
(void) PrtMsgX(em, Pn, cleanup, 1);
if ((em = CanRdKmem()))
(void) PrtMsgX(em, Pn, cleanup, 1);
if (!(Path = LTopt_p)) {
(void) snprintf(buf, sizeof(buf) - 1, "./config.LTszoff%ld",
(long)MyPid);
buf[sizeof(buf) - 1] = '\0';
Path = MkStrCpy(buf, &ti);
}
(void) unlink(Path);
if ((Fd = open(Path, O_RDWR|O_CREAT, 0600)) < 0) {
(void) fprintf(stderr, "ERROR!!! can't open %s\n", Path);
print_file_error:
MsgStat = 1;
(void) snprintf(buf, sizeof(buf) - 1, " Errno %d: %s",
errno, strerror(errno));
buf[sizeof(buf) - 1] = '\0';
(void) PrtMsgX(buf, Pn, cleanup, 1);
}
for (ti = 0; ti < sizeof(buf); ti++) {
buf[ti] = (char)(ti & 0xff);
}
for (ti = 0; ti < TSTFSZ; ti += sizeof(buf)) {
if (write(Fd, buf, sizeof(buf)) != sizeof(buf)) {
(void) fprintf(stderr, "ERROR!!! can't write %d bytes to %s\n",
(int)sizeof(buf), Path);
goto print_file_error;
}
}
if (fsync(Fd)) {
(void) fprintf(stderr, "ERROR!!! can't fsync %s\n", Path);
goto print_file_error;
}
(void) snprintf(xbuf, sizeof(xbuf) - 1, "%d", TSTFSZ);
xbuf[sizeof(xbuf) - 1] = '\0';
if ((tstsz = testlsof(TYTST_SZ, "-s", xbuf)))
(void) PrtMsg(tstsz, Pn);
if (do_offt) {
(void) snprintf(xbuf, sizeof(xbuf) - 1, "0t%d", TSTFSZ);
xbuf[sizeof(xbuf) - 1] = '\0';
if ((tst0to = testlsof(TYTST_0to, "-o", xbuf)))
(void) PrtMsg(tst0to, Pn);
(void) snprintf(xbuf, sizeof(xbuf) - 1, "0x%x", TSTFSZ);
xbuf[sizeof(xbuf) - 1] = '\0';
if ((tst0xo = testlsof(TYTST_0xo, "-oo2", xbuf)))
(void) PrtMsg(tst0to, Pn);
} else {
PrtMsg("WARNING!!! lsof can't return file offsets for this dialect,",
Pn);
PrtMsg(" so offset tests have been disabled.", Pn);
}
if (tstsz || tst0to || tst0xo) {
tcp = (char *)NULL;
xv = 1;
} else {
tcp = "OK";
xv = 0;
}
(void) PrtMsgX(tcp, Pn, cleanup, xv);
return(0);
}
#if defined(LT_DIAL_linux)
static int
ck_Linux_offset_support()
{
char buf[1024];
int bufl = sizeof(buf);
char *opv[5];
int rv = 1;
if (IsLsofExec())
return(0);
opv[0] = "-o";
snprintf(buf, bufl - 1, "-p%d", (int)getpid());
opv[1] = buf;
opv[2] = "-ad0";
opv[3] = "+w";
opv[4] = (char *)NULL;
if (ExecLsof(opv))
return(0);
while(fgets(buf, bufl - 1, LsofFs)) {
if (strstr(buf, "WARNING: can't report offset")) {
rv = 0;
break;
}
}
(void) StopLsof();
return(rv);
}
#endif
static void
cleanup()
{
if (Fd >= 0) {
(void) close(Fd);
Fd = -1;
if (Path) {
(void) unlink(Path);
Path = (char *)NULL;
}
}
}
static char *
testlsof(tt, opt, xval)
int tt;
char *opt;
char *xval;
{
char buf[2048];
char *cem;
LTfldo_t *cmdp;
LTfldo_t *devp;
int ff = 0;
LTfldo_t *fop;
char ibuf[64];
LTfldo_t *inop;
LTdev_t lsofdc;
int nf;
LTfldo_t *offp;
char *opv[4];
char *pem = (char *)NULL;
pid_t pid;
int pids = 0;
struct stat sb;
LTdev_t stdc;
LTfldo_t *szp;
char *tcp;
int ti;
char *tnm1, *tnm2;
int ts = 0;
LTfldo_t *typ;
switch (tt) {
case TYTST_SZ:
tnm1 = "";
tnm2 = " size";
break;
case TYTST_0to:
tnm1 = " 0t";
tnm2 = " offset";
break;
case TYTST_0xo:
tnm1 = " 0x";
tnm2 = " offset";
break;
default:
(void) snprintf(buf, sizeof(buf) - 1,
"ERROR!!! illegal test type: %d", tt);
buf[sizeof(buf) - 1] = '\0';
(void) PrtMsgX(buf, Pn, cleanup, 1);
}
if (stat(Path, &sb)) {
(void) snprintf(buf, sizeof(buf) - 1,
"ERROR!!! can't stat(2) %s: %s", Path, strerror(errno));
buf[sizeof(buf) - 1] = '\0';
PrtMsgX(buf, Pn, cleanup, 1);
}
if ((cem = ConvStatDev(&sb.st_dev, &stdc)))
PrtMsgX(buf, Pn, cleanup, 1);
(void) snprintf(ibuf, sizeof(ibuf) - 1, "%u", (unsigned int)sb.st_ino);
ibuf[sizeof(ibuf) - 1] = '\0';
ti = 0;
if (opt && *opt)
opv[ti++] = opt;
#if defined(USE_LSOF_C_OPT)
opv[ti++] = "-C";
#else
opv[ti++] = "--";
#endif
opv[ti++] = Path;
opv[ti] = (char *)NULL;
if ((cem = ExecLsof(opv)))
return(cem);
while (!ff && !cem && (fop = RdFrLsof(&nf, &cem))) {
if (cem) {
if (pem)
(void) PrtMsg(pem, Pn);
return(cem);
}
switch (fop->ft) {
case LSOF_FID_PID:
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 != MyPid))
pids = 0;
break;
case LSOF_FID_FD:
if (!pids)
break;
for (ti = 0, tcp = fop->v; *tcp; tcp++) {
if (*tcp == ' ')
continue;
if (((int)*tcp < (int)'0') || ((int)*tcp > (int)'9')) {
ti = -1;
break;
}
ti = (ti * 10) + (int)*tcp - (int)'0';
}
if (Fd != ti)
break;
devp = inop = offp = szp = typ = (LTfldo_t *)NULL;
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_OFFSET:
offp = fop;
break;
case LSOF_FID_SIZE:
szp = fop;
break;
case LSOF_FID_TYPE:
typ = fop;
break;
}
}
if (!devp || !inop || !typ)
break;
if (strcasecmp(typ->v, "reg") && strcasecmp(typ->v, "vreg"))
break;
if ((cem = ConvLsofDev(devp->v, &lsofdc))) {
if (pem)
(void) PrtMsg(pem, Pn);
pem = cem;
break;
}
if ((stdc.maj != lsofdc.maj)
|| (stdc.min != lsofdc.min)
|| (stdc.unit != lsofdc.unit)
|| strcmp(inop->v, ibuf)
) {
break;
}
ff = 1;
fop = (tt == TYTST_SZ) ? szp : offp;
if (!fop) {
(void) snprintf(buf, sizeof(buf) - 1,
"ERROR!!! %s%s test, but no lsof%s", tnm1, tnm2, tnm2);
ts = 1;
} else if (strcmp(fop->v, xval)) {
(void) snprintf(buf, sizeof(buf) - 1,
"ERROR!!! %s%s mismatch: expected %s, got %s",
tnm1, tnm2, xval, fop->v);
ts = 1;
}
if (ts) {
buf[sizeof(buf) - 1] = '\0';
cem = MkStrCpy(buf, &ti);
if (pem)
(void) PrtMsg(pem, Pn);
pem = cem;
}
break;
}
}
(void) StopLsof();
if (!ff) {
(void) snprintf(buf, sizeof(buf) - 1,
"ERROR!!! test file %s not found by lsof", Path);
buf[sizeof(buf) - 1] = '\0';
cem = MkStrCpy(buf, &ti);
if (pem)
(void) PrtMsg(pem, Pn);
return(cem);
}
return(pem);
}