Blame tests/LTunix.c

Packit Service 603f59
/*
Packit Service 603f59
 * LTunix.c -- Lsof Test UNIX domain socket test
Packit Service 603f59
 *
Packit Service 603f59
 * V. Abell
Packit Service 603f59
 * Purdue University
Packit Service 603f59
 */
Packit Service 603f59
Packit Service 603f59
Packit Service 603f59
/*
Packit Service 603f59
 * Copyright 2002 Purdue Research Foundation, West Lafayette, Indiana
Packit Service 603f59
 * 47907.  All rights reserved.
Packit Service 603f59
 *
Packit Service 603f59
 * Written by V. Abell.
Packit Service 603f59
 *
Packit Service 603f59
 * This software is not subject to any license of the American Telephone
Packit Service 603f59
 * and Telegraph Company or the Regents of the University of California.
Packit Service 603f59
 *
Packit Service 603f59
 * Permission is granted to anyone to use this software for any purpose on
Packit Service 603f59
 * any computer system, and to alter it and redistribute it freely, subject
Packit Service 603f59
 * to the following restrictions:
Packit Service 603f59
 *
Packit Service 603f59
 * 1. Neither the authors nor Purdue University are responsible for any
Packit Service 603f59
 *    consequences of the use of this software.
Packit Service 603f59
 *
Packit Service 603f59
 * 2. The origin of this software must not be misrepresented, either by
Packit Service 603f59
 *    explicit claim or by omission.  Credit to the authors and Purdue
Packit Service 603f59
 *    University must appear in documentation and sources.
Packit Service 603f59
 *
Packit Service 603f59
 * 3. Altered versions must be plainly marked as such, and must not be
Packit Service 603f59
 *    misrepresented as being the original software.
Packit Service 603f59
 *
Packit Service 603f59
 * 4. This notice may not be removed or altered.
Packit Service 603f59
 */
Packit Service 603f59
Packit Service 603f59
#ifndef lint
Packit Service 603f59
static char copyright[] =
Packit Service 603f59
"@(#) Copyright 2002 Purdue Research Foundation.\nAll rights reserved.\n";
Packit Service 603f59
#endif
Packit Service 603f59
Packit Service 603f59
#include "LsofTest.h"
Packit Service 603f59
#include "lsof_fields.h"
Packit Service 603f59
Packit Service 603f59
#include <sys/socket.h>
Packit Service 603f59
#include <sys/un.h>
Packit Service 603f59
Packit Service 603f59
Packit Service 603f59
/*
Packit Service 603f59
 * Local definitions
Packit Service 603f59
 */
Packit Service 603f59
Packit Service 603f59
#if	!defined(MAXPATHLEN)
Packit Service 603f59
#define	MAXPATHLEN	1024		/* maximum path length */
Packit Service 603f59
#endif	/* !defined(MAXPATHLEN) */
Packit Service 603f59
Packit Service 603f59
Packit Service 603f59
/*
Packit Service 603f59
 * Globals
Packit Service 603f59
 */
Packit Service 603f59
Packit Service 603f59
pid_t MyPid = (pid_t)0;		/* PID of this process */
Packit Service 603f59
char *Pn = (char *)NULL;	/* program name */
Packit Service 603f59
int SpFd[2] = {-1,-1};		/* socket pair FDs */
Packit Service 603f59
char *Path[2] = {(char *)NULL, (char *)NULL};
Packit Service 603f59
				/* socket pair paths */
Packit Service 603f59
Packit Service 603f59
Packit Service 603f59
/*
Packit Service 603f59
 * Local function prototypes
Packit Service 603f59
 */
Packit Service 603f59
Packit Service 603f59
_PROTOTYPE(static void cleanup,(void));
Packit Service 603f59
_PROTOTYPE(static char *FindUsocks,(void));
Packit Service 603f59
Packit Service 603f59
Packit Service 603f59
/*
Packit Service 603f59
 * Main program
Packit Service 603f59
 */
Packit Service 603f59
Packit Service 603f59
int
Packit Service 603f59
main(argc, argv)
Packit Service 603f59
    int argc;				/* argument count */
Packit Service 603f59
    char *argv[];			/* arguments */
Packit Service 603f59
{
Packit Service 603f59
    char buf[2048];			/* temporary buffer */
Packit Service 603f59
    char cwd[MAXPATHLEN + 1];		/* CWD buffer */
Packit Service 603f59
    char *em;				/* error message pointer */
Packit Service 603f59
    int ti, tj;				/* temporary indexes */
Packit Service 603f59
    struct sockaddr_un ua;		/* UNIX socket address */
Packit Service 603f59
    int xv = 0;				/* exit value */
Packit Service 603f59
/*
Packit Service 603f59
 * Get program name and PID, issue start message, and build space prefix.
Packit Service 603f59
 */
Packit Service 603f59
    if ((Pn = strrchr(argv[0], '/')))
Packit Service 603f59
	Pn++;
Packit Service 603f59
    else
Packit Service 603f59
	Pn = argv[0];
Packit Service 603f59
    MyPid = getpid();
Packit Service 603f59
    (void) printf("%s ... ", Pn);
Packit Service 603f59
    (void) fflush(stdout);
Packit Service 603f59
    PrtMsg((char *)NULL, Pn);
Packit Service 603f59
/*
Packit Service 603f59
 * Process arguments.
Packit Service 603f59
 */
Packit Service 603f59
    if (ScanArg(argc, argv, "h", Pn))
Packit Service 603f59
	xv = 1;
Packit Service 603f59
    if (xv || LTopt_h) {
Packit Service 603f59
	(void) PrtMsg("usage: [-h]", Pn);
Packit Service 603f59
	PrtMsgX("       -h       print help (this panel)", Pn, cleanup, xv);
Packit Service 603f59
    }
Packit Service 603f59
/*
Packit Service 603f59
 * See if lsof can be executed and can access kernel memory.
Packit Service 603f59
 */
Packit Service 603f59
    if ((em = IsLsofExec()))
Packit Service 603f59
	(void) PrtMsgX(em, Pn, cleanup, 1);
Packit Service 603f59
    if ((em = CanRdKmem()))
Packit Service 603f59
	(void) PrtMsgX(em, Pn, cleanup, 1);
Packit Service 603f59
/*
Packit Service 603f59
 * Construct the socket paths.
Packit Service 603f59
 */
Packit Service 603f59
Packit Service 603f59
#if	defined(USE_GETCWD)
Packit Service 603f59
    if (!getcwd(cwd, sizeof(cwd)))
Packit Service 603f59
#else	/* ! defined(USE_GETCWD) */
Packit Service 603f59
    if (!getwd(cwd))
Packit Service 603f59
#endif	/* defined(USE_GETCWD) */
Packit Service 603f59
Packit Service 603f59
    {
Packit Service 603f59
	em = "ERROR!!!  can't get CWD";
Packit Service 603f59
	goto print_errno;
Packit Service 603f59
    }
Packit Service 603f59
    cwd[sizeof(cwd) - 1] = '\0';
Packit Service 603f59
    if ((strlen(cwd) + strlen("/config.LT#U9223372036854775807") + 1)
Packit Service 603f59
    > sizeof(ua.sun_path))
Packit Service 603f59
    {
Packit Service 603f59
	strncpy(cwd, "/tmp", sizeof(cwd) - 1);
Packit Service 603f59
    }
Packit Service 603f59
    for (ti = 0; ti < 2; ti++) {
Packit Service 603f59
	(void) snprintf(buf, sizeof(buf) - 1, "%s/config.LT%dU%ld", cwd, ti,
Packit Service 603f59
	    (long)MyPid);
Packit Service 603f59
	buf[sizeof(buf) - 1] = '\0';
Packit Service 603f59
	Path[ti] = MkStrCpy(buf, &tj;;
Packit Service 603f59
	(void) unlink(Path[ti]);
Packit Service 603f59
    }
Packit Service 603f59
/*
Packit Service 603f59
 * Get two UNIX domain socket FDs.
Packit Service 603f59
 */
Packit Service 603f59
    for (ti = 0; ti < 2; ti++) {
Packit Service 603f59
	if ((SpFd[ti] = socket(AF_UNIX, SOCK_STREAM, PF_UNSPEC)) < 0) {
Packit Service 603f59
	    em = "socket";
Packit Service 603f59
Packit Service 603f59
print_errno_by_ti:
Packit Service 603f59
Packit Service 603f59
	    (void) snprintf(buf, sizeof(buf) - 1, "ERROR!!!  %s(%s) failure",
Packit Service 603f59
		em, Path[ti]);
Packit Service 603f59
	    buf[sizeof(buf) - 1] = '\0';
Packit Service 603f59
	    em = buf;
Packit Service 603f59
Packit Service 603f59
print_errno:
Packit Service 603f59
Packit Service 603f59
	    PrtMsg(em, Pn);
Packit Service 603f59
	    (void) snprintf(buf, sizeof(buf) - 1, "    Errno %d: %s", errno,
Packit Service 603f59
		strerror(errno));
Packit Service 603f59
	    buf[sizeof(buf) - 1] = '\0';
Packit Service 603f59
	    PrtMsgX(buf, Pn, cleanup, 1);
Packit Service 603f59
	}
Packit Service 603f59
    }
Packit Service 603f59
/*
Packit Service 603f59
 * Bind file system names to the sockets.
Packit Service 603f59
 */
Packit Service 603f59
    for (ti = 0; ti < 2; ti++) {
Packit Service 603f59
	(void) memset((void *)&ua, 0, sizeof(ua));
Packit Service 603f59
	ua.sun_family = AF_UNIX;
Packit Service 603f59
	(void) strncpy(ua.sun_path, Path[ti], sizeof(ua.sun_path));
Packit Service 603f59
	ua.sun_path[sizeof(ua.sun_path) - 1] = '\0';
Packit Service 603f59
	if (bind(SpFd[ti], (struct sockaddr *)&ua, sizeof(ua)) < 0) {
Packit Service 603f59
	    em = "bind";
Packit Service 603f59
	    goto print_errno_by_ti;
Packit Service 603f59
	}
Packit Service 603f59
    }
Packit Service 603f59
/*
Packit Service 603f59
 * Look for the open UNIX domain socket files with lsof.
Packit Service 603f59
 */
Packit Service 603f59
    if ((em = FindUsocks()))
Packit Service 603f59
	(void) PrtMsgX(em, Pn, cleanup, 1);
Packit Service 603f59
/*
Packit Service 603f59
 * Exit successfully.
Packit Service 603f59
 */
Packit Service 603f59
    (void) PrtMsgX("OK", Pn, cleanup, 0);
Packit Service 603f59
    return(0);
Packit Service 603f59
}
Packit Service 603f59
Packit Service 603f59
Packit Service 603f59
/*
Packit Service 603f59
 * cleanup() -- release resources
Packit Service 603f59
 */
Packit Service 603f59
Packit Service 603f59
static void
Packit Service 603f59
cleanup()
Packit Service 603f59
{
Packit Service 603f59
    int ti;
Packit Service 603f59
Packit Service 603f59
    for (ti = 0; ti < 2; ti++) {
Packit Service 603f59
	if (SpFd[ti] >= 0) {
Packit Service 603f59
	    (void) close(SpFd[ti]);
Packit Service 603f59
	    SpFd[ti] = -1;
Packit Service 603f59
	}
Packit Service 603f59
	if (Path[ti]) {
Packit Service 603f59
	    (void) unlink(Path[ti]);
Packit Service 603f59
	    (void) free((void *)Path[ti]);
Packit Service 603f59
	    Path[ti] = (char *)NULL;
Packit Service 603f59
	}
Packit Service 603f59
    }
Packit Service 603f59
}
Packit Service 603f59
Packit Service 603f59
Packit Service 603f59
/*
Packit Service 603f59
 * FindUsocks() -- find UNIX sockets with lsof
Packit Service 603f59
 */
Packit Service 603f59
Packit Service 603f59
static char *
Packit Service 603f59
FindUsocks()
Packit Service 603f59
{
Packit Service 603f59
    char buf[2048];			/* temporary buffer */
Packit Service 603f59
    char *cem;				/* current error message pointer */
Packit Service 603f59
    LTfldo_t *cmdp;			/* command pointer */
Packit Service 603f59
    int ff[2];				/* file-found flags */
Packit Service 603f59
    LTfldo_t *fop;			/* field output pointer */
Packit Service 603f59
    int nf;				/* number of fields */
Packit Service 603f59
    int nl;				/* name length */
Packit Service 603f59
    LTfldo_t *nmp;			/* name pointer */
Packit Service 603f59
    char *opv[5];			/* option vector for ExecLsof() */
Packit Service 603f59
    char *pem = (char *)NULL;		/* previous error message pointer */
Packit Service 603f59
    pid_t pid;				/* PID */
Packit Service 603f59
    int pids = 0;			/* PID found status */
Packit Service 603f59
    char *tcp;				/* temporary character pointer */
Packit Service 603f59
    int ti, tj;				/* temporary integers */
Packit Service 603f59
    LTfldo_t *typ;			/* file type pointer */
Packit Service 603f59
/*
Packit Service 603f59
 * Build the option vector and start lsof execution.
Packit Service 603f59
 */
Packit Service 603f59
    ff[0] = ff[1] = ti = 0;
Packit Service 603f59
    opv[ti++] = "-aU";
Packit Service 603f59
    opv[ti++] = "-p";
Packit Service 603f59
    (void) snprintf(buf, sizeof(buf) - 1, "%ld", (long)MyPid);
Packit Service 603f59
    buf[sizeof(buf) - 1] = '\0';
Packit Service 603f59
    opv[ti++] = MkStrCpy(buf, &tj;;
Packit Service 603f59
Packit Service 603f59
#if	defined(USE_LSOF_C_OPT)
Packit Service 603f59
    opv[ti++] = "-C";
Packit Service 603f59
#endif	/* defined(USE_LSOF_C_OPT) */
Packit Service 603f59
Packit Service 603f59
    opv[ti] = (char *)NULL;
Packit Service 603f59
    if ((cem = ExecLsof(opv)))
Packit Service 603f59
	return(cem);
Packit Service 603f59
/*
Packit Service 603f59
 * Read lsof output.
Packit Service 603f59
 */
Packit Service 603f59
    while (((ff[0] + ff[1]) < 2) && (fop = RdFrLsof(&nf, &cem))) {
Packit Service 603f59
	if (cem) {
Packit Service 603f59
	    if (pem)
Packit Service 603f59
		(void) PrtMsg(pem, Pn);
Packit Service 603f59
	    return(cem);
Packit Service 603f59
	}
Packit Service 603f59
	switch (fop->ft) {
Packit Service 603f59
	case LSOF_FID_PID:
Packit Service 603f59
Packit Service 603f59
	/*
Packit Service 603f59
	 * This is a process information line.
Packit Service 603f59
	 */
Packit Service 603f59
	    pid = (pid_t)atoi(fop->v);
Packit Service 603f59
	    pids = 1;
Packit Service 603f59
	    cmdp = (LTfldo_t *)NULL;
Packit Service 603f59
	    for (fop++, ti = 1; ti < nf; fop++, ti++) {
Packit Service 603f59
		switch (fop->ft) {
Packit Service 603f59
		case LSOF_FID_CMD:
Packit Service 603f59
		    cmdp = fop;
Packit Service 603f59
		    break;
Packit Service 603f59
		}
Packit Service 603f59
	    }
Packit Service 603f59
	    if (!cmdp || (pid != MyPid))
Packit Service 603f59
		pids = 0;
Packit Service 603f59
	    break;
Packit Service 603f59
	case LSOF_FID_FD:
Packit Service 603f59
Packit Service 603f59
	/*
Packit Service 603f59
	 * This is a file descriptor line.  Make sure its number matches a
Packit Service 603f59
	 * test file descriptor number.
Packit Service 603f59
	 */
Packit Service 603f59
	    if (!pids)
Packit Service 603f59
		break;
Packit Service 603f59
	    for (ti = 0, tcp = fop->v; *tcp; tcp++) {
Packit Service 603f59
Packit Service 603f59
	    /*
Packit Service 603f59
	     * Convert file descriptor to a number.
Packit Service 603f59
	     */
Packit Service 603f59
		if (*tcp == ' ')
Packit Service 603f59
		    continue;
Packit Service 603f59
		if (((int)*tcp < (int)'0') || ((int)*tcp > (int)'9')) {
Packit Service 603f59
		    ti = -1;
Packit Service 603f59
		    break;
Packit Service 603f59
		}
Packit Service 603f59
		ti = (ti * 10) + (int)*tcp - (int)'0'; 
Packit Service 603f59
	    }
Packit Service 603f59
	    for (tj = 0; tj < 2; tj++) {
Packit Service 603f59
		if (ff[tj])
Packit Service 603f59
		    continue;
Packit Service 603f59
		if (SpFd[tj] == ti)
Packit Service 603f59
		    break;
Packit Service 603f59
	    }
Packit Service 603f59
	    if (tj >= 2)
Packit Service 603f59
		break;
Packit Service 603f59
	/*
Packit Service 603f59
	 * Scan for name and type.
Packit Service 603f59
	 */
Packit Service 603f59
	    nmp = typ  = (LTfldo_t *)NULL;
Packit Service 603f59
	    for (fop++, ti = 1; ti < nf; fop++, ti++) {
Packit Service 603f59
		switch (fop->ft) {
Packit Service 603f59
		case LSOF_FID_NAME:
Packit Service 603f59
		    nmp = fop;
Packit Service 603f59
		    break;
Packit Service 603f59
		case LSOF_FID_TYPE:
Packit Service 603f59
		    typ = fop;
Packit Service 603f59
		    break;
Packit Service 603f59
		}
Packit Service 603f59
	    }
Packit Service 603f59
	/*
Packit Service 603f59
	 * Check the type of the file.
Packit Service 603f59
	 */
Packit Service 603f59
	    if (!typ || strcasecmp(typ->v, "unix"))
Packit Service 603f59
		break;
Packit Service 603f59
	/*
Packit Service 603f59
	 * Look for the name.
Packit Service 603f59
	 */
Packit Service 603f59
	    if (!nmp)
Packit Service 603f59
		break;
Packit Service 603f59
	    nl = strlen(Path[tj]);
Packit Service 603f59
	    for (tcp = nmp->v; tcp; tcp = strchr(tcp + 1, '/')) {
Packit Service 603f59
		if (!strncmp(tcp, Path[tj], nl)) {
Packit Service 603f59
Packit Service 603f59
		/*
Packit Service 603f59
		 * Mark a file as found.
Packit Service 603f59
		 */
Packit Service 603f59
		    ff[tj] = 1;
Packit Service 603f59
		    break;
Packit Service 603f59
		}
Packit Service 603f59
	    }
Packit Service 603f59
	}
Packit Service 603f59
    }
Packit Service 603f59
/*
Packit Service 603f59
 * Clean up and return.
Packit Service 603f59
 */
Packit Service 603f59
    (void) StopLsof();
Packit Service 603f59
    for (ti = 0; ti < 2; ti++) {
Packit Service 603f59
	if (ff[tj])
Packit Service 603f59
	    continue;
Packit Service 603f59
	(void) snprintf(buf, sizeof(buf) - 1, "ERROR!!!  not found: %s",
Packit Service 603f59
	    Path[ti]);
Packit Service 603f59
	buf[sizeof(buf) - 1] = '\0';
Packit Service 603f59
	if (pem)
Packit Service 603f59
	    (void) PrtMsg(pem, Pn);
Packit Service 603f59
	pem = MkStrCpy(buf, &tj;;
Packit Service 603f59
    }
Packit Service 603f59
    return(pem);
Packit Service 603f59
}