blob: 436fd5acfaca758ffac57fb78b08f99c469cb9fb [file] [log] [blame]
/*
* dmnt.c -- Darwin mount support functions for libproc-based lsof
*/
/*
* Portions Copyright 2005 Apple Computer, Inc. All rights reserved.
*
* Copyright 2005 Purdue Research Foundation, West Lafayette, Indiana
* 47907. All rights reserved.
*
* Written by Allan Nathanson, Apple Computer, Inc., and Victor A.
* Abell, Purdue University.
*
* 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 Apple Computer, Inc. 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, Apple
* Computer, Inc. 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 2005 Apple Computer, Inc. and Purdue Research Foundation.\nAll rights reserved.\n";
static char *rcsid = "$Id: dmnt.c,v 1.5 2011/08/07 22:52:30 abe Exp $";
#endif
#include "lsof.h"
/*
* Local static information
*/
static struct mounts *Lmi = (struct mounts *)NULL; /* local mount info */
static int Lmist = 0; /* Lmi status */
/*
* readmnt() -- read mount table
*/
struct mounts *
readmnt()
{
char *dn = (char *)NULL;
char *ln;
struct statfs *mb = (struct statfs *)NULL;
struct mounts *mtp;
int n;
struct stat sb;
if (Lmi || Lmist)
return(Lmi);
/*
* Access mount information.
*/
if ((n = getmntinfo(&mb, MNT_NOWAIT)) <= 0) {
(void) fprintf(stderr, "%s: no mount information\n", Pn);
return(0);
}
/*
* Read mount information.
*/
for (; n; n--, mb++) {
if (!mb->f_type)
continue;
/*
* Avoid file systems that are not appropriate paths to
* user data (e.g., automount maps, triggers).
*/
if (mb->f_flags & MNT_AUTOMOUNTED) {
if (!strncmp(mb->f_mntfromname, "map ", 4)
|| !strcmp(mb->f_mntfromname, "trigger"))
continue;
}
/*
* Interpolate a possible symbolic directory link.
*/
if (dn)
(void) free((FREE_P *)dn);
if (!(dn = mkstrcpy(mb->f_mntonname, (MALLOC_S *)NULL))) {
no_space_for_mount:
(void) fprintf(stderr, "%s: no space for mount at ", Pn);
safestrprt(mb->f_mntonname, stderr, 0);
(void) fprintf(stderr, " (");
safestrprt(mb->f_mntfromname, stderr, 0);
(void) fprintf(stderr, ")\n");
Exit(1);
}
if (!(ln = Readlink(dn))) {
if (!Fwarn) {
(void) fprintf(stderr,
" Output information may be incomplete.\n");
}
continue;
}
if (ln != dn) {
(void) free((FREE_P *)dn);
dn = ln;
}
if (*dn != '/')
continue;
/*
* Stat() the directory.
*/
if (statsafely(dn, &sb)) {
if (!Fwarn) {
(void) fprintf(stderr, "%s: WARNING: can't stat() ", Pn);
safestrprt(mb->f_fstypename, stderr, 0);
(void) fprintf(stderr, " file system ");
safestrprt(mb->f_mntonname, stderr, 1);
(void) fprintf(stderr,
" Output information may be incomplete.\n");
}
(void) bzero((char *)&sb, sizeof(sb));
sb.st_dev = (dev_t)mb->f_fsid.val[0];
sb.st_mode = S_IFDIR | 0777;
if (!Fwarn) {
(void) fprintf(stderr,
" assuming \"dev=%x\" from mount table\n",
sb.st_dev);
}
}
/*
* Allocate and fill a local mount structure.
*/
if (!(mtp = (struct mounts *)malloc(sizeof(struct mounts))))
goto no_space_for_mount;
mtp->dir = dn;
dn = (char *)NULL;
mtp->next = Lmi;
mtp->dev = sb.st_dev;
mtp->rdev = sb.st_rdev;
mtp->inode = (INODETYPE)sb.st_ino;
mtp->mode = sb.st_mode;
mtp->is_nfs = strcasecmp(mb->f_fstypename, "nfs") ? 0 : 1;
/*
* Interpolate a possible file system (mounted-on) device name link.
*/
if (!(dn = mkstrcpy(mb->f_mntfromname, (MALLOC_S *)NULL)))
goto no_space_for_mount;
mtp->fsname = dn;
ln = Readlink(dn);
dn = (char *)NULL;
/*
* Stat() the file system (mounted-on) name and add file system
* information to the local mount table entry.
*/
if (!ln || statsafely(ln, &sb))
sb.st_mode = 0;
mtp->fsnmres = ln;
mtp->fs_mode = sb.st_mode;
Lmi = mtp;
}
/*
* Clean up and return the local mount info table address.
*/
if (dn)
(void) free((FREE_P *)dn);
Lmist = 1;
return(Lmi);
}