xref: /freebsd/lib/libprocstat/libprocstat.c (revision 391bdfb835a79618b504d5405336068bea116f60)
10daf62d9SStanislav Sedov /*-
20daf62d9SStanislav Sedov  * Copyright (c) 2009 Stanislav Sedov <stas@FreeBSD.org>
30daf62d9SStanislav Sedov  * Copyright (c) 1988, 1993
40daf62d9SStanislav Sedov  *      The Regents of the University of California.  All rights reserved.
50daf62d9SStanislav Sedov  *
60daf62d9SStanislav Sedov  * Redistribution and use in source and binary forms, with or without
70daf62d9SStanislav Sedov  * modification, are permitted provided that the following conditions
80daf62d9SStanislav Sedov  * are met:
90daf62d9SStanislav Sedov  * 1. Redistributions of source code must retain the above copyright
100daf62d9SStanislav Sedov  *    notice, this list of conditions and the following disclaimer.
110daf62d9SStanislav Sedov  * 2. Redistributions in binary form must reproduce the above copyright
120daf62d9SStanislav Sedov  *    notice, this list of conditions and the following disclaimer in the
130daf62d9SStanislav Sedov  *    documentation and/or other materials provided with the distribution.
140daf62d9SStanislav Sedov  * 3. All advertising materials mentioning features or use of this software
150daf62d9SStanislav Sedov  *    must display the following acknowledgement:
160daf62d9SStanislav Sedov  *      This product includes software developed by the University of
170daf62d9SStanislav Sedov  *      California, Berkeley and its contributors.
180daf62d9SStanislav Sedov  * 4. Neither the name of the University nor the names of its contributors
190daf62d9SStanislav Sedov  *    may be used to endorse or promote products derived from this software
200daf62d9SStanislav Sedov  *    without specific prior written permission.
210daf62d9SStanislav Sedov  *
220daf62d9SStanislav Sedov  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
230daf62d9SStanislav Sedov  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
240daf62d9SStanislav Sedov  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
250daf62d9SStanislav Sedov  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
260daf62d9SStanislav Sedov  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
270daf62d9SStanislav Sedov  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
280daf62d9SStanislav Sedov  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
290daf62d9SStanislav Sedov  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
300daf62d9SStanislav Sedov  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
310daf62d9SStanislav Sedov  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
320daf62d9SStanislav Sedov  * SUCH DAMAGE.
330daf62d9SStanislav Sedov  */
340daf62d9SStanislav Sedov 
350daf62d9SStanislav Sedov #include <sys/cdefs.h>
360daf62d9SStanislav Sedov __FBSDID("$FreeBSD$");
370daf62d9SStanislav Sedov 
380daf62d9SStanislav Sedov #include <sys/param.h>
390daf62d9SStanislav Sedov #include <sys/time.h>
400daf62d9SStanislav Sedov #include <sys/proc.h>
410daf62d9SStanislav Sedov #include <sys/user.h>
420daf62d9SStanislav Sedov #include <sys/stat.h>
430daf62d9SStanislav Sedov #include <sys/vnode.h>
440daf62d9SStanislav Sedov #include <sys/socket.h>
450daf62d9SStanislav Sedov #include <sys/socketvar.h>
460daf62d9SStanislav Sedov #include <sys/domain.h>
470daf62d9SStanislav Sedov #include <sys/protosw.h>
480daf62d9SStanislav Sedov #include <sys/un.h>
490daf62d9SStanislav Sedov #include <sys/unpcb.h>
500daf62d9SStanislav Sedov #include <sys/sysctl.h>
510daf62d9SStanislav Sedov #include <sys/tty.h>
520daf62d9SStanislav Sedov #include <sys/filedesc.h>
530daf62d9SStanislav Sedov #include <sys/queue.h>
540daf62d9SStanislav Sedov #define	_WANT_FILE
550daf62d9SStanislav Sedov #include <sys/file.h>
560daf62d9SStanislav Sedov #include <sys/conf.h>
57e506e182SJohn Baldwin #include <sys/mman.h>
580daf62d9SStanislav Sedov #define	_KERNEL
590daf62d9SStanislav Sedov #include <sys/mount.h>
600daf62d9SStanislav Sedov #include <sys/pipe.h>
610daf62d9SStanislav Sedov #include <ufs/ufs/quota.h>
620daf62d9SStanislav Sedov #include <ufs/ufs/inode.h>
630daf62d9SStanislav Sedov #include <fs/devfs/devfs.h>
640daf62d9SStanislav Sedov #include <fs/devfs/devfs_int.h>
650daf62d9SStanislav Sedov #undef _KERNEL
660daf62d9SStanislav Sedov #include <nfs/nfsproto.h>
670daf62d9SStanislav Sedov #include <nfsclient/nfs.h>
680daf62d9SStanislav Sedov #include <nfsclient/nfsnode.h>
690daf62d9SStanislav Sedov 
700daf62d9SStanislav Sedov #include <vm/vm.h>
710daf62d9SStanislav Sedov #include <vm/vm_map.h>
720daf62d9SStanislav Sedov #include <vm/vm_object.h>
730daf62d9SStanislav Sedov 
740daf62d9SStanislav Sedov #include <net/route.h>
750daf62d9SStanislav Sedov #include <netinet/in.h>
760daf62d9SStanislav Sedov #include <netinet/in_systm.h>
770daf62d9SStanislav Sedov #include <netinet/ip.h>
780daf62d9SStanislav Sedov #include <netinet/in_pcb.h>
790daf62d9SStanislav Sedov 
800daf62d9SStanislav Sedov #include <assert.h>
810daf62d9SStanislav Sedov #include <ctype.h>
820daf62d9SStanislav Sedov #include <err.h>
830daf62d9SStanislav Sedov #include <fcntl.h>
840daf62d9SStanislav Sedov #include <kvm.h>
850daf62d9SStanislav Sedov #include <libutil.h>
860daf62d9SStanislav Sedov #include <limits.h>
870daf62d9SStanislav Sedov #include <paths.h>
880daf62d9SStanislav Sedov #include <pwd.h>
890daf62d9SStanislav Sedov #include <stdio.h>
900daf62d9SStanislav Sedov #include <stdlib.h>
910daf62d9SStanislav Sedov #include <stddef.h>
920daf62d9SStanislav Sedov #include <string.h>
930daf62d9SStanislav Sedov #include <unistd.h>
940daf62d9SStanislav Sedov #include <netdb.h>
950daf62d9SStanislav Sedov 
960daf62d9SStanislav Sedov #include <libprocstat.h>
970daf62d9SStanislav Sedov #include "libprocstat_internal.h"
980daf62d9SStanislav Sedov #include "common_kvm.h"
990daf62d9SStanislav Sedov 
1000daf62d9SStanislav Sedov int     statfs(const char *, struct statfs *);	/* XXX */
1010daf62d9SStanislav Sedov 
1020daf62d9SStanislav Sedov #define	PROCSTAT_KVM	1
1030daf62d9SStanislav Sedov #define	PROCSTAT_SYSCTL	2
1040daf62d9SStanislav Sedov 
1050daf62d9SStanislav Sedov static char	*getmnton(kvm_t *kd, struct mount *m);
1060daf62d9SStanislav Sedov static struct filestat_list	*procstat_getfiles_kvm(
1070daf62d9SStanislav Sedov     struct procstat *procstat, struct kinfo_proc *kp, int mmapped);
1080daf62d9SStanislav Sedov static struct filestat_list	*procstat_getfiles_sysctl(
1090daf62d9SStanislav Sedov     struct procstat *procstat, struct kinfo_proc *kp, int mmapped);
1100daf62d9SStanislav Sedov static int	procstat_get_pipe_info_sysctl(struct filestat *fst,
1110daf62d9SStanislav Sedov     struct pipestat *pipe, char *errbuf);
1120daf62d9SStanislav Sedov static int	procstat_get_pipe_info_kvm(kvm_t *kd, struct filestat *fst,
1130daf62d9SStanislav Sedov     struct pipestat *pipe, char *errbuf);
1140daf62d9SStanislav Sedov static int	procstat_get_pts_info_sysctl(struct filestat *fst,
1150daf62d9SStanislav Sedov     struct ptsstat *pts, char *errbuf);
1160daf62d9SStanislav Sedov static int	procstat_get_pts_info_kvm(kvm_t *kd, struct filestat *fst,
1170daf62d9SStanislav Sedov     struct ptsstat *pts, char *errbuf);
118e506e182SJohn Baldwin static int	procstat_get_shm_info_sysctl(struct filestat *fst,
119e506e182SJohn Baldwin     struct shmstat *shm, char *errbuf);
120e506e182SJohn Baldwin static int	procstat_get_shm_info_kvm(kvm_t *kd, struct filestat *fst,
121e506e182SJohn Baldwin     struct shmstat *shm, char *errbuf);
1220daf62d9SStanislav Sedov static int	procstat_get_socket_info_sysctl(struct filestat *fst,
1230daf62d9SStanislav Sedov     struct sockstat *sock, char *errbuf);
1240daf62d9SStanislav Sedov static int	procstat_get_socket_info_kvm(kvm_t *kd, struct filestat *fst,
1250daf62d9SStanislav Sedov     struct sockstat *sock, char *errbuf);
1260daf62d9SStanislav Sedov static int	to_filestat_flags(int flags);
1270daf62d9SStanislav Sedov static int	procstat_get_vnode_info_kvm(kvm_t *kd, struct filestat *fst,
1280daf62d9SStanislav Sedov     struct vnstat *vn, char *errbuf);
1290daf62d9SStanislav Sedov static int	procstat_get_vnode_info_sysctl(struct filestat *fst,
1300daf62d9SStanislav Sedov     struct vnstat *vn, char *errbuf);
1310daf62d9SStanislav Sedov static int	vntype2psfsttype(int type);
1320daf62d9SStanislav Sedov 
1330daf62d9SStanislav Sedov void
1340daf62d9SStanislav Sedov procstat_close(struct procstat *procstat)
1350daf62d9SStanislav Sedov {
1360daf62d9SStanislav Sedov 
1370daf62d9SStanislav Sedov 	assert(procstat);
1380daf62d9SStanislav Sedov 	if (procstat->type == PROCSTAT_KVM)
1390daf62d9SStanislav Sedov 		kvm_close(procstat->kd);
140d7b666aeSSergey Kandaurov 	free(procstat);
1410daf62d9SStanislav Sedov }
1420daf62d9SStanislav Sedov 
1430daf62d9SStanislav Sedov struct procstat *
1440daf62d9SStanislav Sedov procstat_open_sysctl(void)
1450daf62d9SStanislav Sedov {
1460daf62d9SStanislav Sedov 	struct procstat *procstat;
1470daf62d9SStanislav Sedov 
1480daf62d9SStanislav Sedov 	procstat = calloc(1, sizeof(*procstat));
1490daf62d9SStanislav Sedov 	if (procstat == NULL) {
1500daf62d9SStanislav Sedov 		warn("malloc()");
1510daf62d9SStanislav Sedov 		return (NULL);
1520daf62d9SStanislav Sedov 	}
1530daf62d9SStanislav Sedov 	procstat->type = PROCSTAT_SYSCTL;
1540daf62d9SStanislav Sedov 	return (procstat);
1550daf62d9SStanislav Sedov }
1560daf62d9SStanislav Sedov 
1570daf62d9SStanislav Sedov struct procstat *
1580daf62d9SStanislav Sedov procstat_open_kvm(const char *nlistf, const char *memf)
1590daf62d9SStanislav Sedov {
1600daf62d9SStanislav Sedov 	struct procstat *procstat;
1610daf62d9SStanislav Sedov 	kvm_t *kd;
1620daf62d9SStanislav Sedov 	char buf[_POSIX2_LINE_MAX];
1630daf62d9SStanislav Sedov 
1640daf62d9SStanislav Sedov 	procstat = calloc(1, sizeof(*procstat));
1650daf62d9SStanislav Sedov 	if (procstat == NULL) {
1660daf62d9SStanislav Sedov 		warn("malloc()");
1670daf62d9SStanislav Sedov 		return (NULL);
1680daf62d9SStanislav Sedov 	}
1690daf62d9SStanislav Sedov 	kd = kvm_openfiles(nlistf, memf, NULL, O_RDONLY, buf);
1700daf62d9SStanislav Sedov 	if (kd == NULL) {
1710daf62d9SStanislav Sedov 		warnx("kvm_openfiles(): %s", buf);
1720daf62d9SStanislav Sedov 		free(procstat);
1730daf62d9SStanislav Sedov 		return (NULL);
1740daf62d9SStanislav Sedov 	}
1750daf62d9SStanislav Sedov 	procstat->type = PROCSTAT_KVM;
1760daf62d9SStanislav Sedov 	procstat->kd = kd;
1770daf62d9SStanislav Sedov 	return (procstat);
1780daf62d9SStanislav Sedov }
1790daf62d9SStanislav Sedov 
1800daf62d9SStanislav Sedov struct kinfo_proc *
1810daf62d9SStanislav Sedov procstat_getprocs(struct procstat *procstat, int what, int arg,
1820daf62d9SStanislav Sedov     unsigned int *count)
1830daf62d9SStanislav Sedov {
1840daf62d9SStanislav Sedov 	struct kinfo_proc *p0, *p;
1850daf62d9SStanislav Sedov 	size_t len;
1860daf62d9SStanislav Sedov 	int name[4];
187*391bdfb8SAndriy Gapon 	int cnt;
1880daf62d9SStanislav Sedov 	int error;
1890daf62d9SStanislav Sedov 
1900daf62d9SStanislav Sedov 	assert(procstat);
1910daf62d9SStanislav Sedov 	assert(count);
1920daf62d9SStanislav Sedov 	p = NULL;
1930daf62d9SStanislav Sedov 	if (procstat->type == PROCSTAT_KVM) {
194*391bdfb8SAndriy Gapon 		*count = 0;
195*391bdfb8SAndriy Gapon 		p0 = kvm_getprocs(procstat->kd, what, arg, &cnt);
196*391bdfb8SAndriy Gapon 		if (p0 == NULL || cnt <= 0)
1970daf62d9SStanislav Sedov 			return (NULL);
198*391bdfb8SAndriy Gapon 		*count = cnt;
1990daf62d9SStanislav Sedov 		len = *count * sizeof(*p);
2000daf62d9SStanislav Sedov 		p = malloc(len);
2010daf62d9SStanislav Sedov 		if (p == NULL) {
20265869214SJilles Tjoelker 			warnx("malloc(%zu)", len);
2030daf62d9SStanislav Sedov 			goto fail;
2040daf62d9SStanislav Sedov 		}
2050daf62d9SStanislav Sedov 		bcopy(p0, p, len);
2060daf62d9SStanislav Sedov 		return (p);
2070daf62d9SStanislav Sedov 	} else if (procstat->type == PROCSTAT_SYSCTL) {
2080daf62d9SStanislav Sedov 		len = 0;
2090daf62d9SStanislav Sedov 		name[0] = CTL_KERN;
2100daf62d9SStanislav Sedov 		name[1] = KERN_PROC;
2110daf62d9SStanislav Sedov 		name[2] = what;
2120daf62d9SStanislav Sedov 		name[3] = arg;
2130daf62d9SStanislav Sedov 		error = sysctl(name, 4, NULL, &len, NULL, 0);
2140daf62d9SStanislav Sedov 		if (error < 0 && errno != EPERM) {
2150daf62d9SStanislav Sedov 			warn("sysctl(kern.proc)");
2160daf62d9SStanislav Sedov 			goto fail;
2170daf62d9SStanislav Sedov 		}
2180daf62d9SStanislav Sedov 		if (len == 0) {
2190daf62d9SStanislav Sedov 			warnx("no processes?");
2200daf62d9SStanislav Sedov 			goto fail;
2210daf62d9SStanislav Sedov 		}
2220daf62d9SStanislav Sedov 		p = malloc(len);
2230daf62d9SStanislav Sedov 		if (p == NULL) {
22465869214SJilles Tjoelker 			warnx("malloc(%zu)", len);
2250daf62d9SStanislav Sedov 			goto fail;
2260daf62d9SStanislav Sedov 		}
2270daf62d9SStanislav Sedov 		error = sysctl(name, 4, p, &len, NULL, 0);
2280daf62d9SStanislav Sedov 		if (error < 0 && errno != EPERM) {
2290daf62d9SStanislav Sedov 			warn("sysctl(kern.proc)");
2300daf62d9SStanislav Sedov 			goto fail;
2310daf62d9SStanislav Sedov 		}
2320daf62d9SStanislav Sedov 		/* Perform simple consistency checks. */
2330daf62d9SStanislav Sedov 		if ((len % sizeof(*p)) != 0 || p->ki_structsize != sizeof(*p)) {
2340daf62d9SStanislav Sedov 			warnx("kinfo_proc structure size mismatch");
2350daf62d9SStanislav Sedov 			goto fail;
2360daf62d9SStanislav Sedov 		}
2370daf62d9SStanislav Sedov 		*count = len / sizeof(*p);
2380daf62d9SStanislav Sedov 		return (p);
2390daf62d9SStanislav Sedov 	} else {
24080905c35SJilles Tjoelker 		warnx("unknown access method: %d", procstat->type);
2410daf62d9SStanislav Sedov 		return (NULL);
2420daf62d9SStanislav Sedov 	}
2430daf62d9SStanislav Sedov fail:
2440daf62d9SStanislav Sedov 	if (p)
2450daf62d9SStanislav Sedov 		free(p);
2460daf62d9SStanislav Sedov 	return (NULL);
2470daf62d9SStanislav Sedov }
2480daf62d9SStanislav Sedov 
2490daf62d9SStanislav Sedov void
2500daf62d9SStanislav Sedov procstat_freeprocs(struct procstat *procstat __unused, struct kinfo_proc *p)
2510daf62d9SStanislav Sedov {
2520daf62d9SStanislav Sedov 
2530daf62d9SStanislav Sedov 	if (p != NULL)
2540daf62d9SStanislav Sedov 		free(p);
2550daf62d9SStanislav Sedov 	p = NULL;
2560daf62d9SStanislav Sedov }
2570daf62d9SStanislav Sedov 
2580daf62d9SStanislav Sedov struct filestat_list *
2590daf62d9SStanislav Sedov procstat_getfiles(struct procstat *procstat, struct kinfo_proc *kp, int mmapped)
2600daf62d9SStanislav Sedov {
2610daf62d9SStanislav Sedov 
2620daf62d9SStanislav Sedov 	if (procstat->type == PROCSTAT_SYSCTL)
2630daf62d9SStanislav Sedov 		return (procstat_getfiles_sysctl(procstat, kp, mmapped));
2640daf62d9SStanislav Sedov 	else if (procstat->type == PROCSTAT_KVM)
2650daf62d9SStanislav Sedov 		return (procstat_getfiles_kvm(procstat, kp, mmapped));
2660daf62d9SStanislav Sedov 	else
2670daf62d9SStanislav Sedov 		return (NULL);
2680daf62d9SStanislav Sedov }
2690daf62d9SStanislav Sedov 
2700daf62d9SStanislav Sedov void
2710daf62d9SStanislav Sedov procstat_freefiles(struct procstat *procstat, struct filestat_list *head)
2720daf62d9SStanislav Sedov {
2730daf62d9SStanislav Sedov 	struct filestat *fst, *tmp;
2740daf62d9SStanislav Sedov 
2750daf62d9SStanislav Sedov 	STAILQ_FOREACH_SAFE(fst, head, next, tmp) {
2760daf62d9SStanislav Sedov 		if (fst->fs_path != NULL)
2770daf62d9SStanislav Sedov 			free(fst->fs_path);
2780daf62d9SStanislav Sedov 		free(fst);
2790daf62d9SStanislav Sedov 	}
2800daf62d9SStanislav Sedov 	free(head);
2810daf62d9SStanislav Sedov 	if (procstat->vmentries != NULL) {
2820daf62d9SStanislav Sedov 		free(procstat->vmentries);
2830daf62d9SStanislav Sedov 		procstat->vmentries = NULL;
2840daf62d9SStanislav Sedov 	}
2850daf62d9SStanislav Sedov 	if (procstat->files != NULL) {
2860daf62d9SStanislav Sedov 		free(procstat->files);
2870daf62d9SStanislav Sedov 		procstat->files = NULL;
2880daf62d9SStanislav Sedov 	}
2890daf62d9SStanislav Sedov }
2900daf62d9SStanislav Sedov 
2910daf62d9SStanislav Sedov static struct filestat *
2920daf62d9SStanislav Sedov filestat_new_entry(void *typedep, int type, int fd, int fflags, int uflags,
293d57486e2SRobert Watson     int refcount, off_t offset, char *path, cap_rights_t cap_rights)
2940daf62d9SStanislav Sedov {
2950daf62d9SStanislav Sedov 	struct filestat *entry;
2960daf62d9SStanislav Sedov 
2970daf62d9SStanislav Sedov 	entry = calloc(1, sizeof(*entry));
2980daf62d9SStanislav Sedov 	if (entry == NULL) {
2990daf62d9SStanislav Sedov 		warn("malloc()");
3000daf62d9SStanislav Sedov 		return (NULL);
3010daf62d9SStanislav Sedov 	}
3020daf62d9SStanislav Sedov 	entry->fs_typedep = typedep;
3030daf62d9SStanislav Sedov 	entry->fs_fflags = fflags;
3040daf62d9SStanislav Sedov 	entry->fs_uflags = uflags;
3050daf62d9SStanislav Sedov 	entry->fs_fd = fd;
3060daf62d9SStanislav Sedov 	entry->fs_type = type;
3070daf62d9SStanislav Sedov 	entry->fs_ref_count = refcount;
3080daf62d9SStanislav Sedov 	entry->fs_offset = offset;
3090daf62d9SStanislav Sedov 	entry->fs_path = path;
310d57486e2SRobert Watson 	entry->fs_cap_rights = cap_rights;
3110daf62d9SStanislav Sedov 	return (entry);
3120daf62d9SStanislav Sedov }
3130daf62d9SStanislav Sedov 
3140daf62d9SStanislav Sedov static struct vnode *
3150daf62d9SStanislav Sedov getctty(kvm_t *kd, struct kinfo_proc *kp)
3160daf62d9SStanislav Sedov {
3170daf62d9SStanislav Sedov 	struct pgrp pgrp;
3180daf62d9SStanislav Sedov 	struct proc proc;
3190daf62d9SStanislav Sedov 	struct session sess;
3200daf62d9SStanislav Sedov 	int error;
3210daf62d9SStanislav Sedov 
3220daf62d9SStanislav Sedov 	assert(kp);
3230daf62d9SStanislav Sedov 	error = kvm_read_all(kd, (unsigned long)kp->ki_paddr, &proc,
3240daf62d9SStanislav Sedov 	    sizeof(proc));
3250daf62d9SStanislav Sedov 	if (error == 0) {
3260daf62d9SStanislav Sedov 		warnx("can't read proc struct at %p for pid %d",
3270daf62d9SStanislav Sedov 		    kp->ki_paddr, kp->ki_pid);
3280daf62d9SStanislav Sedov 		return (NULL);
3290daf62d9SStanislav Sedov 	}
3300daf62d9SStanislav Sedov 	if (proc.p_pgrp == NULL)
3310daf62d9SStanislav Sedov 		return (NULL);
3320daf62d9SStanislav Sedov 	error = kvm_read_all(kd, (unsigned long)proc.p_pgrp, &pgrp,
3330daf62d9SStanislav Sedov 	    sizeof(pgrp));
3340daf62d9SStanislav Sedov 	if (error == 0) {
3350daf62d9SStanislav Sedov 		warnx("can't read pgrp struct at %p for pid %d",
3360daf62d9SStanislav Sedov 		    proc.p_pgrp, kp->ki_pid);
3370daf62d9SStanislav Sedov 		return (NULL);
3380daf62d9SStanislav Sedov 	}
3390daf62d9SStanislav Sedov 	error = kvm_read_all(kd, (unsigned long)pgrp.pg_session, &sess,
3400daf62d9SStanislav Sedov 	    sizeof(sess));
3410daf62d9SStanislav Sedov 	if (error == 0) {
3420daf62d9SStanislav Sedov 		warnx("can't read session struct at %p for pid %d",
3430daf62d9SStanislav Sedov 		    pgrp.pg_session, kp->ki_pid);
3440daf62d9SStanislav Sedov 		return (NULL);
3450daf62d9SStanislav Sedov 	}
3460daf62d9SStanislav Sedov 	return (sess.s_ttyvp);
3470daf62d9SStanislav Sedov }
3480daf62d9SStanislav Sedov 
3490daf62d9SStanislav Sedov static struct filestat_list *
3500daf62d9SStanislav Sedov procstat_getfiles_kvm(struct procstat *procstat, struct kinfo_proc *kp, int mmapped)
3510daf62d9SStanislav Sedov {
3520daf62d9SStanislav Sedov 	struct file file;
3530daf62d9SStanislav Sedov 	struct filedesc filed;
3540daf62d9SStanislav Sedov 	struct vm_map_entry vmentry;
3550daf62d9SStanislav Sedov 	struct vm_object object;
3560daf62d9SStanislav Sedov 	struct vmspace vmspace;
3570daf62d9SStanislav Sedov 	vm_map_entry_t entryp;
3580daf62d9SStanislav Sedov 	vm_map_t map;
3590daf62d9SStanislav Sedov 	vm_object_t objp;
3600daf62d9SStanislav Sedov 	struct vnode *vp;
3610daf62d9SStanislav Sedov 	struct file **ofiles;
3620daf62d9SStanislav Sedov 	struct filestat *entry;
3630daf62d9SStanislav Sedov 	struct filestat_list *head;
3640daf62d9SStanislav Sedov 	kvm_t *kd;
3650daf62d9SStanislav Sedov 	void *data;
3660daf62d9SStanislav Sedov 	int i, fflags;
3670daf62d9SStanislav Sedov 	int prot, type;
3680daf62d9SStanislav Sedov 	unsigned int nfiles;
3690daf62d9SStanislav Sedov 
3700daf62d9SStanislav Sedov 	assert(procstat);
3710daf62d9SStanislav Sedov 	kd = procstat->kd;
3720daf62d9SStanislav Sedov 	if (kd == NULL)
3730daf62d9SStanislav Sedov 		return (NULL);
3740daf62d9SStanislav Sedov 	if (kp->ki_fd == NULL)
3750daf62d9SStanislav Sedov 		return (NULL);
3760daf62d9SStanislav Sedov 	if (!kvm_read_all(kd, (unsigned long)kp->ki_fd, &filed,
3770daf62d9SStanislav Sedov 	    sizeof(filed))) {
3780daf62d9SStanislav Sedov 		warnx("can't read filedesc at %p", (void *)kp->ki_fd);
3790daf62d9SStanislav Sedov 		return (NULL);
3800daf62d9SStanislav Sedov 	}
3810daf62d9SStanislav Sedov 
3820daf62d9SStanislav Sedov 	/*
3830daf62d9SStanislav Sedov 	 * Allocate list head.
3840daf62d9SStanislav Sedov 	 */
3850daf62d9SStanislav Sedov 	head = malloc(sizeof(*head));
3860daf62d9SStanislav Sedov 	if (head == NULL)
3870daf62d9SStanislav Sedov 		return (NULL);
3880daf62d9SStanislav Sedov 	STAILQ_INIT(head);
3890daf62d9SStanislav Sedov 
3900daf62d9SStanislav Sedov 	/* root directory vnode, if one. */
3910daf62d9SStanislav Sedov 	if (filed.fd_rdir) {
3920daf62d9SStanislav Sedov 		entry = filestat_new_entry(filed.fd_rdir, PS_FST_TYPE_VNODE, -1,
393d57486e2SRobert Watson 		    PS_FST_FFLAG_READ, PS_FST_UFLAG_RDIR, 0, 0, NULL, 0);
3940daf62d9SStanislav Sedov 		if (entry != NULL)
3950daf62d9SStanislav Sedov 			STAILQ_INSERT_TAIL(head, entry, next);
3960daf62d9SStanislav Sedov 	}
3970daf62d9SStanislav Sedov 	/* current working directory vnode. */
3980daf62d9SStanislav Sedov 	if (filed.fd_cdir) {
3990daf62d9SStanislav Sedov 		entry = filestat_new_entry(filed.fd_cdir, PS_FST_TYPE_VNODE, -1,
400d57486e2SRobert Watson 		    PS_FST_FFLAG_READ, PS_FST_UFLAG_CDIR, 0, 0, NULL, 0);
4010daf62d9SStanislav Sedov 		if (entry != NULL)
4020daf62d9SStanislav Sedov 			STAILQ_INSERT_TAIL(head, entry, next);
4030daf62d9SStanislav Sedov 	}
4040daf62d9SStanislav Sedov 	/* jail root, if any. */
4050daf62d9SStanislav Sedov 	if (filed.fd_jdir) {
4060daf62d9SStanislav Sedov 		entry = filestat_new_entry(filed.fd_jdir, PS_FST_TYPE_VNODE, -1,
407d57486e2SRobert Watson 		    PS_FST_FFLAG_READ, PS_FST_UFLAG_JAIL, 0, 0, NULL, 0);
4080daf62d9SStanislav Sedov 		if (entry != NULL)
4090daf62d9SStanislav Sedov 			STAILQ_INSERT_TAIL(head, entry, next);
4100daf62d9SStanislav Sedov 	}
4110daf62d9SStanislav Sedov 	/* ktrace vnode, if one */
4120daf62d9SStanislav Sedov 	if (kp->ki_tracep) {
4130daf62d9SStanislav Sedov 		entry = filestat_new_entry(kp->ki_tracep, PS_FST_TYPE_VNODE, -1,
4140daf62d9SStanislav Sedov 		    PS_FST_FFLAG_READ | PS_FST_FFLAG_WRITE,
415d57486e2SRobert Watson 		    PS_FST_UFLAG_TRACE, 0, 0, NULL, 0);
4160daf62d9SStanislav Sedov 		if (entry != NULL)
4170daf62d9SStanislav Sedov 			STAILQ_INSERT_TAIL(head, entry, next);
4180daf62d9SStanislav Sedov 	}
4190daf62d9SStanislav Sedov 	/* text vnode, if one */
4200daf62d9SStanislav Sedov 	if (kp->ki_textvp) {
4210daf62d9SStanislav Sedov 		entry = filestat_new_entry(kp->ki_textvp, PS_FST_TYPE_VNODE, -1,
422d57486e2SRobert Watson 		    PS_FST_FFLAG_READ, PS_FST_UFLAG_TEXT, 0, 0, NULL, 0);
4230daf62d9SStanislav Sedov 		if (entry != NULL)
4240daf62d9SStanislav Sedov 			STAILQ_INSERT_TAIL(head, entry, next);
4250daf62d9SStanislav Sedov 	}
4260daf62d9SStanislav Sedov 	/* Controlling terminal. */
4270daf62d9SStanislav Sedov 	if ((vp = getctty(kd, kp)) != NULL) {
4280daf62d9SStanislav Sedov 		entry = filestat_new_entry(vp, PS_FST_TYPE_VNODE, -1,
4290daf62d9SStanislav Sedov 		    PS_FST_FFLAG_READ | PS_FST_FFLAG_WRITE,
430d57486e2SRobert Watson 		    PS_FST_UFLAG_CTTY, 0, 0, NULL, 0);
4310daf62d9SStanislav Sedov 		if (entry != NULL)
4320daf62d9SStanislav Sedov 			STAILQ_INSERT_TAIL(head, entry, next);
4330daf62d9SStanislav Sedov 	}
4340daf62d9SStanislav Sedov 
4350daf62d9SStanislav Sedov 	nfiles = filed.fd_lastfile + 1;
4360daf62d9SStanislav Sedov 	ofiles = malloc(nfiles * sizeof(struct file *));
4370daf62d9SStanislav Sedov 	if (ofiles == NULL) {
43865869214SJilles Tjoelker 		warn("malloc(%zu)", nfiles * sizeof(struct file *));
4390daf62d9SStanislav Sedov 		goto do_mmapped;
4400daf62d9SStanislav Sedov 	}
4410daf62d9SStanislav Sedov 	if (!kvm_read_all(kd, (unsigned long)filed.fd_ofiles, ofiles,
4420daf62d9SStanislav Sedov 	    nfiles * sizeof(struct file *))) {
4430daf62d9SStanislav Sedov 		warnx("cannot read file structures at %p",
4440daf62d9SStanislav Sedov 		    (void *)filed.fd_ofiles);
4450daf62d9SStanislav Sedov 		free(ofiles);
4460daf62d9SStanislav Sedov 		goto do_mmapped;
4470daf62d9SStanislav Sedov 	}
4480daf62d9SStanislav Sedov 	for (i = 0; i <= filed.fd_lastfile; i++) {
4490daf62d9SStanislav Sedov 		if (ofiles[i] == NULL)
4500daf62d9SStanislav Sedov 			continue;
4510daf62d9SStanislav Sedov 		if (!kvm_read_all(kd, (unsigned long)ofiles[i], &file,
4520daf62d9SStanislav Sedov 		    sizeof(struct file))) {
4530daf62d9SStanislav Sedov 			warnx("can't read file %d at %p", i,
4540daf62d9SStanislav Sedov 			    (void *)ofiles[i]);
4550daf62d9SStanislav Sedov 			continue;
4560daf62d9SStanislav Sedov 		}
4570daf62d9SStanislav Sedov 		switch (file.f_type) {
4580daf62d9SStanislav Sedov 		case DTYPE_VNODE:
4590daf62d9SStanislav Sedov 			type = PS_FST_TYPE_VNODE;
4600daf62d9SStanislav Sedov 			data = file.f_vnode;
4610daf62d9SStanislav Sedov 			break;
4620daf62d9SStanislav Sedov 		case DTYPE_SOCKET:
4630daf62d9SStanislav Sedov 			type = PS_FST_TYPE_SOCKET;
4640daf62d9SStanislav Sedov 			data = file.f_data;
4650daf62d9SStanislav Sedov 			break;
4660daf62d9SStanislav Sedov 		case DTYPE_PIPE:
4670daf62d9SStanislav Sedov 			type = PS_FST_TYPE_PIPE;
4680daf62d9SStanislav Sedov 			data = file.f_data;
4690daf62d9SStanislav Sedov 			break;
4700daf62d9SStanislav Sedov 		case DTYPE_FIFO:
4710daf62d9SStanislav Sedov 			type = PS_FST_TYPE_FIFO;
4720daf62d9SStanislav Sedov 			data = file.f_vnode;
4730daf62d9SStanislav Sedov 			break;
4740daf62d9SStanislav Sedov #ifdef DTYPE_PTS
4750daf62d9SStanislav Sedov 		case DTYPE_PTS:
4760daf62d9SStanislav Sedov 			type = PS_FST_TYPE_PTS;
4770daf62d9SStanislav Sedov 			data = file.f_data;
4780daf62d9SStanislav Sedov 			break;
4790daf62d9SStanislav Sedov #endif
480e506e182SJohn Baldwin 		case DTYPE_SHM:
481e506e182SJohn Baldwin 			type = PS_FST_TYPE_SHM;
482e506e182SJohn Baldwin 			data = file.f_data;
483e506e182SJohn Baldwin 			break;
4840daf62d9SStanislav Sedov 		default:
4850daf62d9SStanislav Sedov 			continue;
4860daf62d9SStanislav Sedov 		}
487d57486e2SRobert Watson 		/* XXXRW: No capability rights support for kvm yet. */
4880daf62d9SStanislav Sedov 		entry = filestat_new_entry(data, type, i,
489d57486e2SRobert Watson 		    to_filestat_flags(file.f_flag), 0, 0, 0, NULL, 0);
4900daf62d9SStanislav Sedov 		if (entry != NULL)
4910daf62d9SStanislav Sedov 			STAILQ_INSERT_TAIL(head, entry, next);
4920daf62d9SStanislav Sedov 	}
4930daf62d9SStanislav Sedov 	free(ofiles);
4940daf62d9SStanislav Sedov 
4950daf62d9SStanislav Sedov do_mmapped:
4960daf62d9SStanislav Sedov 
4970daf62d9SStanislav Sedov 	/*
4980daf62d9SStanislav Sedov 	 * Process mmapped files if requested.
4990daf62d9SStanislav Sedov 	 */
5000daf62d9SStanislav Sedov 	if (mmapped) {
5010daf62d9SStanislav Sedov 		if (!kvm_read_all(kd, (unsigned long)kp->ki_vmspace, &vmspace,
5020daf62d9SStanislav Sedov 		    sizeof(vmspace))) {
5030daf62d9SStanislav Sedov 			warnx("can't read vmspace at %p",
5040daf62d9SStanislav Sedov 			    (void *)kp->ki_vmspace);
5050daf62d9SStanislav Sedov 			goto exit;
5060daf62d9SStanislav Sedov 		}
5070daf62d9SStanislav Sedov 		map = &vmspace.vm_map;
5080daf62d9SStanislav Sedov 
5090daf62d9SStanislav Sedov 		for (entryp = map->header.next;
5100daf62d9SStanislav Sedov 		    entryp != &kp->ki_vmspace->vm_map.header;
5110daf62d9SStanislav Sedov 		    entryp = vmentry.next) {
5120daf62d9SStanislav Sedov 			if (!kvm_read_all(kd, (unsigned long)entryp, &vmentry,
5130daf62d9SStanislav Sedov 			    sizeof(vmentry))) {
5140daf62d9SStanislav Sedov 				warnx("can't read vm_map_entry at %p",
5150daf62d9SStanislav Sedov 				    (void *)entryp);
5160daf62d9SStanislav Sedov 				continue;
5170daf62d9SStanislav Sedov 			}
5180daf62d9SStanislav Sedov 			if (vmentry.eflags & MAP_ENTRY_IS_SUB_MAP)
5190daf62d9SStanislav Sedov 				continue;
5200daf62d9SStanislav Sedov 			if ((objp = vmentry.object.vm_object) == NULL)
5210daf62d9SStanislav Sedov 				continue;
5220daf62d9SStanislav Sedov 			for (; objp; objp = object.backing_object) {
5230daf62d9SStanislav Sedov 				if (!kvm_read_all(kd, (unsigned long)objp,
5240daf62d9SStanislav Sedov 				    &object, sizeof(object))) {
5250daf62d9SStanislav Sedov 					warnx("can't read vm_object at %p",
5260daf62d9SStanislav Sedov 					    (void *)objp);
5270daf62d9SStanislav Sedov 					break;
5280daf62d9SStanislav Sedov 				}
5290daf62d9SStanislav Sedov 			}
5300daf62d9SStanislav Sedov 
5310daf62d9SStanislav Sedov 			/* We want only vnode objects. */
5320daf62d9SStanislav Sedov 			if (object.type != OBJT_VNODE)
5330daf62d9SStanislav Sedov 				continue;
5340daf62d9SStanislav Sedov 
5350daf62d9SStanislav Sedov 			prot = vmentry.protection;
5360daf62d9SStanislav Sedov 			fflags = 0;
5370daf62d9SStanislav Sedov 			if (prot & VM_PROT_READ)
5380daf62d9SStanislav Sedov 				fflags = PS_FST_FFLAG_READ;
539279a233dSJilles Tjoelker 			if ((vmentry.eflags & MAP_ENTRY_COW) == 0 &&
540279a233dSJilles Tjoelker 			    prot & VM_PROT_WRITE)
5410daf62d9SStanislav Sedov 				fflags |= PS_FST_FFLAG_WRITE;
5420daf62d9SStanislav Sedov 
5430daf62d9SStanislav Sedov 			/*
5440daf62d9SStanislav Sedov 			 * Create filestat entry.
5450daf62d9SStanislav Sedov 			 */
5460daf62d9SStanislav Sedov 			entry = filestat_new_entry(object.handle,
5470daf62d9SStanislav Sedov 			    PS_FST_TYPE_VNODE, -1, fflags,
548d57486e2SRobert Watson 			    PS_FST_UFLAG_MMAP, 0, 0, NULL, 0);
5490daf62d9SStanislav Sedov 			if (entry != NULL)
5500daf62d9SStanislav Sedov 				STAILQ_INSERT_TAIL(head, entry, next);
5510daf62d9SStanislav Sedov 		}
5520daf62d9SStanislav Sedov 	}
5530daf62d9SStanislav Sedov exit:
5540daf62d9SStanislav Sedov 	return (head);
5550daf62d9SStanislav Sedov }
5560daf62d9SStanislav Sedov 
5570daf62d9SStanislav Sedov /*
5580daf62d9SStanislav Sedov  * kinfo types to filestat translation.
5590daf62d9SStanislav Sedov  */
5600daf62d9SStanislav Sedov static int
5610daf62d9SStanislav Sedov kinfo_type2fst(int kftype)
5620daf62d9SStanislav Sedov {
5630daf62d9SStanislav Sedov 	static struct {
5640daf62d9SStanislav Sedov 		int	kf_type;
5650daf62d9SStanislav Sedov 		int	fst_type;
5660daf62d9SStanislav Sedov 	} kftypes2fst[] = {
5670daf62d9SStanislav Sedov 		{ KF_TYPE_CRYPTO, PS_FST_TYPE_CRYPTO },
5680daf62d9SStanislav Sedov 		{ KF_TYPE_FIFO, PS_FST_TYPE_FIFO },
5690daf62d9SStanislav Sedov 		{ KF_TYPE_KQUEUE, PS_FST_TYPE_KQUEUE },
5700daf62d9SStanislav Sedov 		{ KF_TYPE_MQUEUE, PS_FST_TYPE_MQUEUE },
5710daf62d9SStanislav Sedov 		{ KF_TYPE_NONE, PS_FST_TYPE_NONE },
5720daf62d9SStanislav Sedov 		{ KF_TYPE_PIPE, PS_FST_TYPE_PIPE },
5730daf62d9SStanislav Sedov 		{ KF_TYPE_PTS, PS_FST_TYPE_PTS },
5740daf62d9SStanislav Sedov 		{ KF_TYPE_SEM, PS_FST_TYPE_SEM },
5750daf62d9SStanislav Sedov 		{ KF_TYPE_SHM, PS_FST_TYPE_SHM },
5760daf62d9SStanislav Sedov 		{ KF_TYPE_SOCKET, PS_FST_TYPE_SOCKET },
5770daf62d9SStanislav Sedov 		{ KF_TYPE_VNODE, PS_FST_TYPE_VNODE },
5780daf62d9SStanislav Sedov 		{ KF_TYPE_UNKNOWN, PS_FST_TYPE_UNKNOWN }
5790daf62d9SStanislav Sedov 	};
5800daf62d9SStanislav Sedov #define NKFTYPES	(sizeof(kftypes2fst) / sizeof(*kftypes2fst))
5810daf62d9SStanislav Sedov 	unsigned int i;
5820daf62d9SStanislav Sedov 
5830daf62d9SStanislav Sedov 	for (i = 0; i < NKFTYPES; i++)
5840daf62d9SStanislav Sedov 		if (kftypes2fst[i].kf_type == kftype)
5850daf62d9SStanislav Sedov 			break;
5860daf62d9SStanislav Sedov 	if (i == NKFTYPES)
5870daf62d9SStanislav Sedov 		return (PS_FST_TYPE_UNKNOWN);
5880daf62d9SStanislav Sedov 	return (kftypes2fst[i].fst_type);
5890daf62d9SStanislav Sedov }
5900daf62d9SStanislav Sedov 
5910daf62d9SStanislav Sedov /*
5920daf62d9SStanislav Sedov  * kinfo flags to filestat translation.
5930daf62d9SStanislav Sedov  */
5940daf62d9SStanislav Sedov static int
5950daf62d9SStanislav Sedov kinfo_fflags2fst(int kfflags)
5960daf62d9SStanislav Sedov {
5970daf62d9SStanislav Sedov 	static struct {
5980daf62d9SStanislav Sedov 		int	kf_flag;
5990daf62d9SStanislav Sedov 		int	fst_flag;
6000daf62d9SStanislav Sedov 	} kfflags2fst[] = {
6010daf62d9SStanislav Sedov 		{ KF_FLAG_APPEND, PS_FST_FFLAG_APPEND },
6020daf62d9SStanislav Sedov 		{ KF_FLAG_ASYNC, PS_FST_FFLAG_ASYNC },
603d57486e2SRobert Watson 		{ KF_FLAG_CAPABILITY, PS_FST_FFLAG_CAPABILITY },
6040daf62d9SStanislav Sedov 		{ KF_FLAG_CREAT, PS_FST_FFLAG_CREAT },
6050daf62d9SStanislav Sedov 		{ KF_FLAG_DIRECT, PS_FST_FFLAG_DIRECT },
6060daf62d9SStanislav Sedov 		{ KF_FLAG_EXCL, PS_FST_FFLAG_EXCL },
6070daf62d9SStanislav Sedov 		{ KF_FLAG_EXEC, PS_FST_FFLAG_EXEC },
6080daf62d9SStanislav Sedov 		{ KF_FLAG_EXLOCK, PS_FST_FFLAG_EXLOCK },
6090daf62d9SStanislav Sedov 		{ KF_FLAG_FSYNC, PS_FST_FFLAG_SYNC },
6100daf62d9SStanislav Sedov 		{ KF_FLAG_HASLOCK, PS_FST_FFLAG_HASLOCK },
6110daf62d9SStanislav Sedov 		{ KF_FLAG_NOFOLLOW, PS_FST_FFLAG_NOFOLLOW },
6120daf62d9SStanislav Sedov 		{ KF_FLAG_NONBLOCK, PS_FST_FFLAG_NONBLOCK },
6130daf62d9SStanislav Sedov 		{ KF_FLAG_READ, PS_FST_FFLAG_READ },
6140daf62d9SStanislav Sedov 		{ KF_FLAG_SHLOCK, PS_FST_FFLAG_SHLOCK },
6150daf62d9SStanislav Sedov 		{ KF_FLAG_TRUNC, PS_FST_FFLAG_TRUNC },
6160daf62d9SStanislav Sedov 		{ KF_FLAG_WRITE, PS_FST_FFLAG_WRITE }
6170daf62d9SStanislav Sedov 	};
6180daf62d9SStanislav Sedov #define NKFFLAGS	(sizeof(kfflags2fst) / sizeof(*kfflags2fst))
6190daf62d9SStanislav Sedov 	unsigned int i;
6200daf62d9SStanislav Sedov 	int flags;
6210daf62d9SStanislav Sedov 
6220daf62d9SStanislav Sedov 	flags = 0;
6230daf62d9SStanislav Sedov 	for (i = 0; i < NKFFLAGS; i++)
6240daf62d9SStanislav Sedov 		if ((kfflags & kfflags2fst[i].kf_flag) != 0)
6250daf62d9SStanislav Sedov 			flags |= kfflags2fst[i].fst_flag;
6260daf62d9SStanislav Sedov 	return (flags);
6270daf62d9SStanislav Sedov }
6280daf62d9SStanislav Sedov 
6290daf62d9SStanislav Sedov static int
6300daf62d9SStanislav Sedov kinfo_uflags2fst(int fd)
6310daf62d9SStanislav Sedov {
6320daf62d9SStanislav Sedov 
6330daf62d9SStanislav Sedov 	switch (fd) {
6340daf62d9SStanislav Sedov 	case KF_FD_TYPE_CTTY:
6350daf62d9SStanislav Sedov 		return (PS_FST_UFLAG_CTTY);
6360daf62d9SStanislav Sedov 	case KF_FD_TYPE_CWD:
6370daf62d9SStanislav Sedov 		return (PS_FST_UFLAG_CDIR);
6380daf62d9SStanislav Sedov 	case KF_FD_TYPE_JAIL:
6390daf62d9SStanislav Sedov 		return (PS_FST_UFLAG_JAIL);
6400daf62d9SStanislav Sedov 	case KF_FD_TYPE_TEXT:
6410daf62d9SStanislav Sedov 		return (PS_FST_UFLAG_TEXT);
6420daf62d9SStanislav Sedov 	case KF_FD_TYPE_TRACE:
6430daf62d9SStanislav Sedov 		return (PS_FST_UFLAG_TRACE);
6440daf62d9SStanislav Sedov 	case KF_FD_TYPE_ROOT:
6450daf62d9SStanislav Sedov 		return (PS_FST_UFLAG_RDIR);
6460daf62d9SStanislav Sedov 	}
6470daf62d9SStanislav Sedov 	return (0);
6480daf62d9SStanislav Sedov }
6490daf62d9SStanislav Sedov 
6500daf62d9SStanislav Sedov static struct filestat_list *
6510daf62d9SStanislav Sedov procstat_getfiles_sysctl(struct procstat *procstat, struct kinfo_proc *kp, int mmapped)
6520daf62d9SStanislav Sedov {
6530daf62d9SStanislav Sedov 	struct kinfo_file *kif, *files;
6540daf62d9SStanislav Sedov 	struct kinfo_vmentry *kve, *vmentries;
6550daf62d9SStanislav Sedov 	struct filestat_list *head;
6560daf62d9SStanislav Sedov 	struct filestat *entry;
6570daf62d9SStanislav Sedov 	char *path;
6580daf62d9SStanislav Sedov 	off_t offset;
6590daf62d9SStanislav Sedov 	int cnt, fd, fflags;
6600daf62d9SStanislav Sedov 	int i, type, uflags;
6610daf62d9SStanislav Sedov 	int refcount;
662d57486e2SRobert Watson 	cap_rights_t cap_rights;
6630daf62d9SStanislav Sedov 
6640daf62d9SStanislav Sedov 	assert(kp);
6650daf62d9SStanislav Sedov 	if (kp->ki_fd == NULL)
6660daf62d9SStanislav Sedov 		return (NULL);
6670daf62d9SStanislav Sedov 
6680daf62d9SStanislav Sedov 	files = kinfo_getfile(kp->ki_pid, &cnt);
6690daf62d9SStanislav Sedov 	if (files == NULL && errno != EPERM) {
6700daf62d9SStanislav Sedov 		warn("kinfo_getfile()");
6710daf62d9SStanislav Sedov 		return (NULL);
6720daf62d9SStanislav Sedov 	}
6730daf62d9SStanislav Sedov 	procstat->files = files;
6740daf62d9SStanislav Sedov 
6750daf62d9SStanislav Sedov 	/*
6760daf62d9SStanislav Sedov 	 * Allocate list head.
6770daf62d9SStanislav Sedov 	 */
6780daf62d9SStanislav Sedov 	head = malloc(sizeof(*head));
6790daf62d9SStanislav Sedov 	if (head == NULL)
6800daf62d9SStanislav Sedov 		return (NULL);
6810daf62d9SStanislav Sedov 	STAILQ_INIT(head);
6820daf62d9SStanislav Sedov 	for (i = 0; i < cnt; i++) {
6830daf62d9SStanislav Sedov 		kif = &files[i];
6840daf62d9SStanislav Sedov 
6850daf62d9SStanislav Sedov 		type = kinfo_type2fst(kif->kf_type);
6860daf62d9SStanislav Sedov 		fd = kif->kf_fd >= 0 ? kif->kf_fd : -1;
6870daf62d9SStanislav Sedov 		fflags = kinfo_fflags2fst(kif->kf_flags);
6880daf62d9SStanislav Sedov 		uflags = kinfo_uflags2fst(kif->kf_fd);
6890daf62d9SStanislav Sedov 		refcount = kif->kf_ref_count;
6900daf62d9SStanislav Sedov 		offset = kif->kf_offset;
6910daf62d9SStanislav Sedov 		if (*kif->kf_path != '\0')
6920daf62d9SStanislav Sedov 			path = strdup(kif->kf_path);
6930daf62d9SStanislav Sedov 		else
6940daf62d9SStanislav Sedov 			path = NULL;
695d57486e2SRobert Watson 		cap_rights = kif->kf_cap_rights;
6960daf62d9SStanislav Sedov 
6970daf62d9SStanislav Sedov 		/*
6980daf62d9SStanislav Sedov 		 * Create filestat entry.
6990daf62d9SStanislav Sedov 		 */
7000daf62d9SStanislav Sedov 		entry = filestat_new_entry(kif, type, fd, fflags, uflags,
701d57486e2SRobert Watson 		    refcount, offset, path, cap_rights);
7020daf62d9SStanislav Sedov 		if (entry != NULL)
7030daf62d9SStanislav Sedov 			STAILQ_INSERT_TAIL(head, entry, next);
7040daf62d9SStanislav Sedov 	}
7050daf62d9SStanislav Sedov 	if (mmapped != 0) {
7060daf62d9SStanislav Sedov 		vmentries = kinfo_getvmmap(kp->ki_pid, &cnt);
7070daf62d9SStanislav Sedov 		procstat->vmentries = vmentries;
7080daf62d9SStanislav Sedov 		if (vmentries == NULL || cnt == 0)
7090daf62d9SStanislav Sedov 			goto fail;
7100daf62d9SStanislav Sedov 		for (i = 0; i < cnt; i++) {
7110daf62d9SStanislav Sedov 			kve = &vmentries[i];
7120daf62d9SStanislav Sedov 			if (kve->kve_type != KVME_TYPE_VNODE)
7130daf62d9SStanislav Sedov 				continue;
7140daf62d9SStanislav Sedov 			fflags = 0;
7150daf62d9SStanislav Sedov 			if (kve->kve_protection & KVME_PROT_READ)
7160daf62d9SStanislav Sedov 				fflags = PS_FST_FFLAG_READ;
717279a233dSJilles Tjoelker 			if ((kve->kve_flags & KVME_FLAG_COW) == 0 &&
718279a233dSJilles Tjoelker 			    kve->kve_protection & KVME_PROT_WRITE)
7190daf62d9SStanislav Sedov 				fflags |= PS_FST_FFLAG_WRITE;
7200daf62d9SStanislav Sedov 			offset = kve->kve_offset;
7210daf62d9SStanislav Sedov 			refcount = kve->kve_ref_count;
7220daf62d9SStanislav Sedov 			if (*kve->kve_path != '\0')
7230daf62d9SStanislav Sedov 				path = strdup(kve->kve_path);
7240daf62d9SStanislav Sedov 			else
7250daf62d9SStanislav Sedov 				path = NULL;
7260daf62d9SStanislav Sedov 			entry = filestat_new_entry(kve, PS_FST_TYPE_VNODE, -1,
727d57486e2SRobert Watson 			    fflags, PS_FST_UFLAG_MMAP, refcount, offset, path,
728d57486e2SRobert Watson 			    0);
7290daf62d9SStanislav Sedov 			if (entry != NULL)
7300daf62d9SStanislav Sedov 				STAILQ_INSERT_TAIL(head, entry, next);
7310daf62d9SStanislav Sedov 		}
7320daf62d9SStanislav Sedov 	}
7330daf62d9SStanislav Sedov fail:
7340daf62d9SStanislav Sedov 	return (head);
7350daf62d9SStanislav Sedov }
7360daf62d9SStanislav Sedov 
7370daf62d9SStanislav Sedov int
7380daf62d9SStanislav Sedov procstat_get_pipe_info(struct procstat *procstat, struct filestat *fst,
7390daf62d9SStanislav Sedov     struct pipestat *ps, char *errbuf)
7400daf62d9SStanislav Sedov {
7410daf62d9SStanislav Sedov 
7420daf62d9SStanislav Sedov 	assert(ps);
7430daf62d9SStanislav Sedov 	if (procstat->type == PROCSTAT_KVM) {
7440daf62d9SStanislav Sedov 		return (procstat_get_pipe_info_kvm(procstat->kd, fst, ps,
7450daf62d9SStanislav Sedov 		    errbuf));
7460daf62d9SStanislav Sedov 	} else if (procstat->type == PROCSTAT_SYSCTL) {
7470daf62d9SStanislav Sedov 		return (procstat_get_pipe_info_sysctl(fst, ps, errbuf));
7480daf62d9SStanislav Sedov 	} else {
74980905c35SJilles Tjoelker 		warnx("unknown access method: %d", procstat->type);
7500daf62d9SStanislav Sedov 		snprintf(errbuf, _POSIX2_LINE_MAX, "error");
7510daf62d9SStanislav Sedov 		return (1);
7520daf62d9SStanislav Sedov 	}
7530daf62d9SStanislav Sedov }
7540daf62d9SStanislav Sedov 
7550daf62d9SStanislav Sedov static int
7560daf62d9SStanislav Sedov procstat_get_pipe_info_kvm(kvm_t *kd, struct filestat *fst,
7570daf62d9SStanislav Sedov     struct pipestat *ps, char *errbuf)
7580daf62d9SStanislav Sedov {
7590daf62d9SStanislav Sedov 	struct pipe pi;
7600daf62d9SStanislav Sedov 	void *pipep;
7610daf62d9SStanislav Sedov 
7620daf62d9SStanislav Sedov 	assert(kd);
7630daf62d9SStanislav Sedov 	assert(ps);
7640daf62d9SStanislav Sedov 	assert(fst);
7650daf62d9SStanislav Sedov 	bzero(ps, sizeof(*ps));
7660daf62d9SStanislav Sedov 	pipep = fst->fs_typedep;
7670daf62d9SStanislav Sedov 	if (pipep == NULL)
7680daf62d9SStanislav Sedov 		goto fail;
7690daf62d9SStanislav Sedov 	if (!kvm_read_all(kd, (unsigned long)pipep, &pi, sizeof(struct pipe))) {
7700daf62d9SStanislav Sedov 		warnx("can't read pipe at %p", (void *)pipep);
7710daf62d9SStanislav Sedov 		goto fail;
7720daf62d9SStanislav Sedov 	}
7730daf62d9SStanislav Sedov 	ps->addr = (uintptr_t)pipep;
7740daf62d9SStanislav Sedov 	ps->peer = (uintptr_t)pi.pipe_peer;
7750daf62d9SStanislav Sedov 	ps->buffer_cnt = pi.pipe_buffer.cnt;
7760daf62d9SStanislav Sedov 	return (0);
7770daf62d9SStanislav Sedov 
7780daf62d9SStanislav Sedov fail:
7790daf62d9SStanislav Sedov 	snprintf(errbuf, _POSIX2_LINE_MAX, "error");
7800daf62d9SStanislav Sedov 	return (1);
7810daf62d9SStanislav Sedov }
7820daf62d9SStanislav Sedov 
7830daf62d9SStanislav Sedov static int
7840daf62d9SStanislav Sedov procstat_get_pipe_info_sysctl(struct filestat *fst, struct pipestat *ps,
7850daf62d9SStanislav Sedov     char *errbuf __unused)
7860daf62d9SStanislav Sedov {
7870daf62d9SStanislav Sedov 	struct kinfo_file *kif;
7880daf62d9SStanislav Sedov 
7890daf62d9SStanislav Sedov 	assert(ps);
7900daf62d9SStanislav Sedov 	assert(fst);
7910daf62d9SStanislav Sedov 	bzero(ps, sizeof(*ps));
7920daf62d9SStanislav Sedov 	kif = fst->fs_typedep;
7930daf62d9SStanislav Sedov 	if (kif == NULL)
7940daf62d9SStanislav Sedov 		return (1);
7950daf62d9SStanislav Sedov 	ps->addr = kif->kf_un.kf_pipe.kf_pipe_addr;
7960daf62d9SStanislav Sedov 	ps->peer = kif->kf_un.kf_pipe.kf_pipe_peer;
7970daf62d9SStanislav Sedov 	ps->buffer_cnt = kif->kf_un.kf_pipe.kf_pipe_buffer_cnt;
7980daf62d9SStanislav Sedov 	return (0);
7990daf62d9SStanislav Sedov }
8000daf62d9SStanislav Sedov 
8010daf62d9SStanislav Sedov int
8020daf62d9SStanislav Sedov procstat_get_pts_info(struct procstat *procstat, struct filestat *fst,
8030daf62d9SStanislav Sedov     struct ptsstat *pts, char *errbuf)
8040daf62d9SStanislav Sedov {
8050daf62d9SStanislav Sedov 
8060daf62d9SStanislav Sedov 	assert(pts);
8070daf62d9SStanislav Sedov 	if (procstat->type == PROCSTAT_KVM) {
8080daf62d9SStanislav Sedov 		return (procstat_get_pts_info_kvm(procstat->kd, fst, pts,
8090daf62d9SStanislav Sedov 		    errbuf));
8100daf62d9SStanislav Sedov 	} else if (procstat->type == PROCSTAT_SYSCTL) {
8110daf62d9SStanislav Sedov 		return (procstat_get_pts_info_sysctl(fst, pts, errbuf));
8120daf62d9SStanislav Sedov 	} else {
81380905c35SJilles Tjoelker 		warnx("unknown access method: %d", procstat->type);
8140daf62d9SStanislav Sedov 		snprintf(errbuf, _POSIX2_LINE_MAX, "error");
8150daf62d9SStanislav Sedov 		return (1);
8160daf62d9SStanislav Sedov 	}
8170daf62d9SStanislav Sedov }
8180daf62d9SStanislav Sedov 
8190daf62d9SStanislav Sedov static int
8200daf62d9SStanislav Sedov procstat_get_pts_info_kvm(kvm_t *kd, struct filestat *fst,
8210daf62d9SStanislav Sedov     struct ptsstat *pts, char *errbuf)
8220daf62d9SStanislav Sedov {
8230daf62d9SStanislav Sedov 	struct tty tty;
8240daf62d9SStanislav Sedov 	void *ttyp;
8250daf62d9SStanislav Sedov 
8260daf62d9SStanislav Sedov 	assert(kd);
8270daf62d9SStanislav Sedov 	assert(pts);
8280daf62d9SStanislav Sedov 	assert(fst);
8290daf62d9SStanislav Sedov 	bzero(pts, sizeof(*pts));
8300daf62d9SStanislav Sedov 	ttyp = fst->fs_typedep;
8310daf62d9SStanislav Sedov 	if (ttyp == NULL)
8320daf62d9SStanislav Sedov 		goto fail;
8330daf62d9SStanislav Sedov 	if (!kvm_read_all(kd, (unsigned long)ttyp, &tty, sizeof(struct tty))) {
8340daf62d9SStanislav Sedov 		warnx("can't read tty at %p", (void *)ttyp);
8350daf62d9SStanislav Sedov 		goto fail;
8360daf62d9SStanislav Sedov 	}
8370daf62d9SStanislav Sedov 	pts->dev = dev2udev(kd, tty.t_dev);
8380daf62d9SStanislav Sedov 	(void)kdevtoname(kd, tty.t_dev, pts->devname);
8390daf62d9SStanislav Sedov 	return (0);
8400daf62d9SStanislav Sedov 
8410daf62d9SStanislav Sedov fail:
8420daf62d9SStanislav Sedov 	snprintf(errbuf, _POSIX2_LINE_MAX, "error");
8430daf62d9SStanislav Sedov 	return (1);
8440daf62d9SStanislav Sedov }
8450daf62d9SStanislav Sedov 
8460daf62d9SStanislav Sedov static int
8470daf62d9SStanislav Sedov procstat_get_pts_info_sysctl(struct filestat *fst, struct ptsstat *pts,
8480daf62d9SStanislav Sedov     char *errbuf __unused)
8490daf62d9SStanislav Sedov {
8500daf62d9SStanislav Sedov 	struct kinfo_file *kif;
8510daf62d9SStanislav Sedov 
8520daf62d9SStanislav Sedov 	assert(pts);
8530daf62d9SStanislav Sedov 	assert(fst);
8540daf62d9SStanislav Sedov 	bzero(pts, sizeof(*pts));
8550daf62d9SStanislav Sedov 	kif = fst->fs_typedep;
8560daf62d9SStanislav Sedov 	if (kif == NULL)
8570daf62d9SStanislav Sedov 		return (0);
8580daf62d9SStanislav Sedov 	pts->dev = kif->kf_un.kf_pts.kf_pts_dev;
8590daf62d9SStanislav Sedov 	strlcpy(pts->devname, kif->kf_path, sizeof(pts->devname));
8600daf62d9SStanislav Sedov 	return (0);
8610daf62d9SStanislav Sedov }
8620daf62d9SStanislav Sedov 
8630daf62d9SStanislav Sedov int
864e506e182SJohn Baldwin procstat_get_shm_info(struct procstat *procstat, struct filestat *fst,
865e506e182SJohn Baldwin     struct shmstat *shm, char *errbuf)
866e506e182SJohn Baldwin {
867e506e182SJohn Baldwin 
868e506e182SJohn Baldwin 	assert(shm);
869e506e182SJohn Baldwin 	if (procstat->type == PROCSTAT_KVM) {
870e506e182SJohn Baldwin 		return (procstat_get_shm_info_kvm(procstat->kd, fst, shm,
871e506e182SJohn Baldwin 		    errbuf));
872e506e182SJohn Baldwin 	} else if (procstat->type == PROCSTAT_SYSCTL) {
873e506e182SJohn Baldwin 		return (procstat_get_shm_info_sysctl(fst, shm, errbuf));
874e506e182SJohn Baldwin 	} else {
875e506e182SJohn Baldwin 		warnx("unknown access method: %d", procstat->type);
876e506e182SJohn Baldwin 		snprintf(errbuf, _POSIX2_LINE_MAX, "error");
877e506e182SJohn Baldwin 		return (1);
878e506e182SJohn Baldwin 	}
879e506e182SJohn Baldwin }
880e506e182SJohn Baldwin 
881e506e182SJohn Baldwin static int
882e506e182SJohn Baldwin procstat_get_shm_info_kvm(kvm_t *kd, struct filestat *fst,
883e506e182SJohn Baldwin     struct shmstat *shm, char *errbuf)
884e506e182SJohn Baldwin {
885e506e182SJohn Baldwin 	struct shmfd shmfd;
886e506e182SJohn Baldwin 	void *shmfdp;
887a9f5e425SJohn Baldwin 	char *path;
888a9f5e425SJohn Baldwin 	int i;
889e506e182SJohn Baldwin 
890e506e182SJohn Baldwin 	assert(kd);
891e506e182SJohn Baldwin 	assert(shm);
892e506e182SJohn Baldwin 	assert(fst);
893e506e182SJohn Baldwin 	bzero(shm, sizeof(*shm));
894e506e182SJohn Baldwin 	shmfdp = fst->fs_typedep;
895e506e182SJohn Baldwin 	if (shmfdp == NULL)
896e506e182SJohn Baldwin 		goto fail;
897e506e182SJohn Baldwin 	if (!kvm_read_all(kd, (unsigned long)shmfdp, &shmfd,
898e506e182SJohn Baldwin 	    sizeof(struct shmfd))) {
899e506e182SJohn Baldwin 		warnx("can't read shmfd at %p", (void *)shmfdp);
900e506e182SJohn Baldwin 		goto fail;
901e506e182SJohn Baldwin 	}
902e506e182SJohn Baldwin 	shm->mode = S_IFREG | shmfd.shm_mode;
903e506e182SJohn Baldwin 	shm->size = shmfd.shm_size;
904a9f5e425SJohn Baldwin 	if (fst->fs_path == NULL && shmfd.shm_path != NULL) {
905a9f5e425SJohn Baldwin 		path = malloc(MAXPATHLEN);
906a9f5e425SJohn Baldwin 		for (i = 0; i < MAXPATHLEN - 1; i++) {
907a9f5e425SJohn Baldwin 			if (!kvm_read_all(kd, (unsigned long)shmfd.shm_path + i,
908a9f5e425SJohn Baldwin 			    path + i, 1))
909a9f5e425SJohn Baldwin 				break;
910a9f5e425SJohn Baldwin 			if (path[i] == '\0')
911a9f5e425SJohn Baldwin 				break;
912a9f5e425SJohn Baldwin 		}
913a9f5e425SJohn Baldwin 		path[i] = '\0';
914a9f5e425SJohn Baldwin 		if (i == 0)
915a9f5e425SJohn Baldwin 			free(path);
916a9f5e425SJohn Baldwin 		else
917a9f5e425SJohn Baldwin 			fst->fs_path = path;
918a9f5e425SJohn Baldwin 	}
919e506e182SJohn Baldwin 	return (0);
920e506e182SJohn Baldwin 
921e506e182SJohn Baldwin fail:
922e506e182SJohn Baldwin 	snprintf(errbuf, _POSIX2_LINE_MAX, "error");
923e506e182SJohn Baldwin 	return (1);
924e506e182SJohn Baldwin }
925e506e182SJohn Baldwin 
926e506e182SJohn Baldwin static int
927e506e182SJohn Baldwin procstat_get_shm_info_sysctl(struct filestat *fst, struct shmstat *shm,
928e506e182SJohn Baldwin     char *errbuf __unused)
929e506e182SJohn Baldwin {
930e506e182SJohn Baldwin 	struct kinfo_file *kif;
931e506e182SJohn Baldwin 
932e506e182SJohn Baldwin 	assert(shm);
933e506e182SJohn Baldwin 	assert(fst);
934e506e182SJohn Baldwin 	bzero(shm, sizeof(*shm));
935e506e182SJohn Baldwin 	kif = fst->fs_typedep;
936e506e182SJohn Baldwin 	if (kif == NULL)
937e506e182SJohn Baldwin 		return (0);
938e506e182SJohn Baldwin 	shm->size = kif->kf_un.kf_file.kf_file_size;
939e506e182SJohn Baldwin 	shm->mode = kif->kf_un.kf_file.kf_file_mode;
940e506e182SJohn Baldwin 	return (0);
941e506e182SJohn Baldwin }
942e506e182SJohn Baldwin 
943e506e182SJohn Baldwin int
9440daf62d9SStanislav Sedov procstat_get_vnode_info(struct procstat *procstat, struct filestat *fst,
9450daf62d9SStanislav Sedov     struct vnstat *vn, char *errbuf)
9460daf62d9SStanislav Sedov {
9470daf62d9SStanislav Sedov 
9480daf62d9SStanislav Sedov 	assert(vn);
9490daf62d9SStanislav Sedov 	if (procstat->type == PROCSTAT_KVM) {
9500daf62d9SStanislav Sedov 		return (procstat_get_vnode_info_kvm(procstat->kd, fst, vn,
9510daf62d9SStanislav Sedov 		    errbuf));
9520daf62d9SStanislav Sedov 	} else if (procstat->type == PROCSTAT_SYSCTL) {
9530daf62d9SStanislav Sedov 		return (procstat_get_vnode_info_sysctl(fst, vn, errbuf));
9540daf62d9SStanislav Sedov 	} else {
95580905c35SJilles Tjoelker 		warnx("unknown access method: %d", procstat->type);
9560daf62d9SStanislav Sedov 		snprintf(errbuf, _POSIX2_LINE_MAX, "error");
9570daf62d9SStanislav Sedov 		return (1);
9580daf62d9SStanislav Sedov 	}
9590daf62d9SStanislav Sedov }
9600daf62d9SStanislav Sedov 
9610daf62d9SStanislav Sedov static int
9620daf62d9SStanislav Sedov procstat_get_vnode_info_kvm(kvm_t *kd, struct filestat *fst,
9630daf62d9SStanislav Sedov     struct vnstat *vn, char *errbuf)
9640daf62d9SStanislav Sedov {
9650daf62d9SStanislav Sedov 	/* Filesystem specific handlers. */
9660daf62d9SStanislav Sedov 	#define FSTYPE(fst)     {#fst, fst##_filestat}
9670daf62d9SStanislav Sedov 	struct {
9680daf62d9SStanislav Sedov 		const char	*tag;
9690daf62d9SStanislav Sedov 		int		(*handler)(kvm_t *kd, struct vnode *vp,
9700daf62d9SStanislav Sedov 		    struct vnstat *vn);
9710daf62d9SStanislav Sedov 	} fstypes[] = {
9720daf62d9SStanislav Sedov 		FSTYPE(devfs),
9730daf62d9SStanislav Sedov 		FSTYPE(isofs),
9740daf62d9SStanislav Sedov 		FSTYPE(msdosfs),
9750daf62d9SStanislav Sedov 		FSTYPE(nfs),
9760daf62d9SStanislav Sedov 		FSTYPE(ntfs),
97759ea47d0SStanislav Sedov #ifdef LIBPROCSTAT_NWFS
9780daf62d9SStanislav Sedov 		FSTYPE(nwfs),
97959ea47d0SStanislav Sedov #endif
9800daf62d9SStanislav Sedov 		FSTYPE(smbfs),
9810daf62d9SStanislav Sedov 		FSTYPE(udf),
9820daf62d9SStanislav Sedov 		FSTYPE(ufs),
98359ea47d0SStanislav Sedov #ifdef LIBPROCSTAT_ZFS
9840daf62d9SStanislav Sedov 		FSTYPE(zfs),
9850daf62d9SStanislav Sedov #endif
9860daf62d9SStanislav Sedov 	};
9870daf62d9SStanislav Sedov #define	NTYPES	(sizeof(fstypes) / sizeof(*fstypes))
9880daf62d9SStanislav Sedov 	struct vnode vnode;
9890daf62d9SStanislav Sedov 	char tagstr[12];
9900daf62d9SStanislav Sedov 	void *vp;
9910daf62d9SStanislav Sedov 	int error, found;
9920daf62d9SStanislav Sedov 	unsigned int i;
9930daf62d9SStanislav Sedov 
9940daf62d9SStanislav Sedov 	assert(kd);
9950daf62d9SStanislav Sedov 	assert(vn);
9960daf62d9SStanislav Sedov 	assert(fst);
9970daf62d9SStanislav Sedov 	vp = fst->fs_typedep;
9980daf62d9SStanislav Sedov 	if (vp == NULL)
9990daf62d9SStanislav Sedov 		goto fail;
10000daf62d9SStanislav Sedov 	error = kvm_read_all(kd, (unsigned long)vp, &vnode, sizeof(vnode));
10010daf62d9SStanislav Sedov 	if (error == 0) {
10020daf62d9SStanislav Sedov 		warnx("can't read vnode at %p", (void *)vp);
10030daf62d9SStanislav Sedov 		goto fail;
10040daf62d9SStanislav Sedov 	}
10050daf62d9SStanislav Sedov 	bzero(vn, sizeof(*vn));
10060daf62d9SStanislav Sedov 	vn->vn_type = vntype2psfsttype(vnode.v_type);
10070daf62d9SStanislav Sedov 	if (vnode.v_type == VNON || vnode.v_type == VBAD)
10080daf62d9SStanislav Sedov 		return (0);
10090daf62d9SStanislav Sedov 	error = kvm_read_all(kd, (unsigned long)vnode.v_tag, tagstr,
10100daf62d9SStanislav Sedov 	    sizeof(tagstr));
10110daf62d9SStanislav Sedov 	if (error == 0) {
10120daf62d9SStanislav Sedov 		warnx("can't read v_tag at %p", (void *)vp);
10130daf62d9SStanislav Sedov 		goto fail;
10140daf62d9SStanislav Sedov 	}
10150daf62d9SStanislav Sedov 	tagstr[sizeof(tagstr) - 1] = '\0';
10160daf62d9SStanislav Sedov 
10170daf62d9SStanislav Sedov 	/*
10180daf62d9SStanislav Sedov 	 * Find appropriate handler.
10190daf62d9SStanislav Sedov 	 */
10200daf62d9SStanislav Sedov 	for (i = 0, found = 0; i < NTYPES; i++)
10210daf62d9SStanislav Sedov 		if (!strcmp(fstypes[i].tag, tagstr)) {
10220daf62d9SStanislav Sedov 			if (fstypes[i].handler(kd, &vnode, vn) != 0) {
10230daf62d9SStanislav Sedov 				goto fail;
10240daf62d9SStanislav Sedov 			}
10250daf62d9SStanislav Sedov 			break;
10260daf62d9SStanislav Sedov 		}
10270daf62d9SStanislav Sedov 	if (i == NTYPES) {
10280daf62d9SStanislav Sedov 		snprintf(errbuf, _POSIX2_LINE_MAX, "?(%s)", tagstr);
10290daf62d9SStanislav Sedov 		return (1);
10300daf62d9SStanislav Sedov 	}
10310daf62d9SStanislav Sedov 	vn->vn_mntdir = getmnton(kd, vnode.v_mount);
10320daf62d9SStanislav Sedov 	if ((vnode.v_type == VBLK || vnode.v_type == VCHR) &&
10330daf62d9SStanislav Sedov 	    vnode.v_rdev != NULL){
10340daf62d9SStanislav Sedov 		vn->vn_dev = dev2udev(kd, vnode.v_rdev);
10350daf62d9SStanislav Sedov 		(void)kdevtoname(kd, vnode.v_rdev, vn->vn_devname);
10360daf62d9SStanislav Sedov 	} else {
10370daf62d9SStanislav Sedov 		vn->vn_dev = -1;
10380daf62d9SStanislav Sedov 	}
10390daf62d9SStanislav Sedov 	return (0);
10400daf62d9SStanislav Sedov 
10410daf62d9SStanislav Sedov fail:
10420daf62d9SStanislav Sedov 	snprintf(errbuf, _POSIX2_LINE_MAX, "error");
10430daf62d9SStanislav Sedov 	return (1);
10440daf62d9SStanislav Sedov }
10450daf62d9SStanislav Sedov 
10460daf62d9SStanislav Sedov /*
10470daf62d9SStanislav Sedov  * kinfo vnode type to filestat translation.
10480daf62d9SStanislav Sedov  */
10490daf62d9SStanislav Sedov static int
10500daf62d9SStanislav Sedov kinfo_vtype2fst(int kfvtype)
10510daf62d9SStanislav Sedov {
10520daf62d9SStanislav Sedov 	static struct {
10530daf62d9SStanislav Sedov 		int	kf_vtype;
10540daf62d9SStanislav Sedov 		int	fst_vtype;
10550daf62d9SStanislav Sedov 	} kfvtypes2fst[] = {
10560daf62d9SStanislav Sedov 		{ KF_VTYPE_VBAD, PS_FST_VTYPE_VBAD },
10570daf62d9SStanislav Sedov 		{ KF_VTYPE_VBLK, PS_FST_VTYPE_VBLK },
10580daf62d9SStanislav Sedov 		{ KF_VTYPE_VCHR, PS_FST_VTYPE_VCHR },
10590daf62d9SStanislav Sedov 		{ KF_VTYPE_VDIR, PS_FST_VTYPE_VDIR },
10600daf62d9SStanislav Sedov 		{ KF_VTYPE_VFIFO, PS_FST_VTYPE_VFIFO },
10610daf62d9SStanislav Sedov 		{ KF_VTYPE_VLNK, PS_FST_VTYPE_VLNK },
10620daf62d9SStanislav Sedov 		{ KF_VTYPE_VNON, PS_FST_VTYPE_VNON },
10630daf62d9SStanislav Sedov 		{ KF_VTYPE_VREG, PS_FST_VTYPE_VREG },
10640daf62d9SStanislav Sedov 		{ KF_VTYPE_VSOCK, PS_FST_VTYPE_VSOCK }
10650daf62d9SStanislav Sedov 	};
10660daf62d9SStanislav Sedov #define	NKFVTYPES	(sizeof(kfvtypes2fst) / sizeof(*kfvtypes2fst))
10670daf62d9SStanislav Sedov 	unsigned int i;
10680daf62d9SStanislav Sedov 
10690daf62d9SStanislav Sedov 	for (i = 0; i < NKFVTYPES; i++)
10700daf62d9SStanislav Sedov 		if (kfvtypes2fst[i].kf_vtype == kfvtype)
10710daf62d9SStanislav Sedov 			break;
10720daf62d9SStanislav Sedov 	if (i == NKFVTYPES)
10730daf62d9SStanislav Sedov 		return (PS_FST_VTYPE_UNKNOWN);
10740daf62d9SStanislav Sedov 	return (kfvtypes2fst[i].fst_vtype);
10750daf62d9SStanislav Sedov }
10760daf62d9SStanislav Sedov 
10770daf62d9SStanislav Sedov static int
10780daf62d9SStanislav Sedov procstat_get_vnode_info_sysctl(struct filestat *fst, struct vnstat *vn,
10790daf62d9SStanislav Sedov     char *errbuf)
10800daf62d9SStanislav Sedov {
10810daf62d9SStanislav Sedov 	struct statfs stbuf;
10820daf62d9SStanislav Sedov 	struct kinfo_file *kif;
10830daf62d9SStanislav Sedov 	struct kinfo_vmentry *kve;
10840daf62d9SStanislav Sedov 	uint64_t fileid;
10850daf62d9SStanislav Sedov 	uint64_t size;
10860daf62d9SStanislav Sedov 	char *name, *path;
10870daf62d9SStanislav Sedov 	uint32_t fsid;
10880daf62d9SStanislav Sedov 	uint16_t mode;
10890daf62d9SStanislav Sedov 	uint32_t rdev;
10900daf62d9SStanislav Sedov 	int vntype;
10910daf62d9SStanislav Sedov 	int status;
10920daf62d9SStanislav Sedov 
10930daf62d9SStanislav Sedov 	assert(fst);
10940daf62d9SStanislav Sedov 	assert(vn);
10950daf62d9SStanislav Sedov 	bzero(vn, sizeof(*vn));
10960daf62d9SStanislav Sedov 	if (fst->fs_typedep == NULL)
10970daf62d9SStanislav Sedov 		return (1);
10980daf62d9SStanislav Sedov 	if (fst->fs_uflags & PS_FST_UFLAG_MMAP) {
10990daf62d9SStanislav Sedov 		kve = fst->fs_typedep;
11000daf62d9SStanislav Sedov 		fileid = kve->kve_vn_fileid;
11010daf62d9SStanislav Sedov 		fsid = kve->kve_vn_fsid;
11020daf62d9SStanislav Sedov 		mode = kve->kve_vn_mode;
11030daf62d9SStanislav Sedov 		path = kve->kve_path;
11040daf62d9SStanislav Sedov 		rdev = kve->kve_vn_rdev;
11050daf62d9SStanislav Sedov 		size = kve->kve_vn_size;
11060daf62d9SStanislav Sedov 		vntype = kinfo_vtype2fst(kve->kve_vn_type);
11070daf62d9SStanislav Sedov 		status = kve->kve_status;
11080daf62d9SStanislav Sedov 	} else {
11090daf62d9SStanislav Sedov 		kif = fst->fs_typedep;
11100daf62d9SStanislav Sedov 		fileid = kif->kf_un.kf_file.kf_file_fileid;
11110daf62d9SStanislav Sedov 		fsid = kif->kf_un.kf_file.kf_file_fsid;
11120daf62d9SStanislav Sedov 		mode = kif->kf_un.kf_file.kf_file_mode;
11130daf62d9SStanislav Sedov 		path = kif->kf_path;
11140daf62d9SStanislav Sedov 		rdev = kif->kf_un.kf_file.kf_file_rdev;
11150daf62d9SStanislav Sedov 		size = kif->kf_un.kf_file.kf_file_size;
11160daf62d9SStanislav Sedov 		vntype = kinfo_vtype2fst(kif->kf_vnode_type);
11170daf62d9SStanislav Sedov 		status = kif->kf_status;
11180daf62d9SStanislav Sedov 	}
11190daf62d9SStanislav Sedov 	vn->vn_type = vntype;
11200daf62d9SStanislav Sedov 	if (vntype == PS_FST_VTYPE_VNON || vntype == PS_FST_VTYPE_VBAD)
11210daf62d9SStanislav Sedov 		return (0);
11220daf62d9SStanislav Sedov 	if ((status & KF_ATTR_VALID) == 0) {
11230daf62d9SStanislav Sedov 		snprintf(errbuf, _POSIX2_LINE_MAX, "? (no info available)");
11240daf62d9SStanislav Sedov 		return (1);
11250daf62d9SStanislav Sedov 	}
11260daf62d9SStanislav Sedov 	if (path && *path) {
11270daf62d9SStanislav Sedov 		statfs(path, &stbuf);
11280daf62d9SStanislav Sedov 		vn->vn_mntdir = strdup(stbuf.f_mntonname);
11290daf62d9SStanislav Sedov 	} else
11300daf62d9SStanislav Sedov 		vn->vn_mntdir = strdup("-");
11310daf62d9SStanislav Sedov 	vn->vn_dev = rdev;
11320daf62d9SStanislav Sedov 	if (vntype == PS_FST_VTYPE_VBLK) {
11330daf62d9SStanislav Sedov 		name = devname(rdev, S_IFBLK);
11340daf62d9SStanislav Sedov 		if (name != NULL)
11350daf62d9SStanislav Sedov 			strlcpy(vn->vn_devname, name,
11360daf62d9SStanislav Sedov 			    sizeof(vn->vn_devname));
11370daf62d9SStanislav Sedov 	} else if (vntype == PS_FST_VTYPE_VCHR) {
11380daf62d9SStanislav Sedov 		name = devname(vn->vn_dev, S_IFCHR);
11390daf62d9SStanislav Sedov 		if (name != NULL)
11400daf62d9SStanislav Sedov 			strlcpy(vn->vn_devname, name,
11410daf62d9SStanislav Sedov 			    sizeof(vn->vn_devname));
11420daf62d9SStanislav Sedov 	}
11430daf62d9SStanislav Sedov 	vn->vn_fsid = fsid;
11440daf62d9SStanislav Sedov 	vn->vn_fileid = fileid;
11450daf62d9SStanislav Sedov 	vn->vn_size = size;
11460daf62d9SStanislav Sedov 	vn->vn_mode = mode;
11470daf62d9SStanislav Sedov 	return (0);
11480daf62d9SStanislav Sedov }
11490daf62d9SStanislav Sedov 
11500daf62d9SStanislav Sedov int
11510daf62d9SStanislav Sedov procstat_get_socket_info(struct procstat *procstat, struct filestat *fst,
11520daf62d9SStanislav Sedov     struct sockstat *sock, char *errbuf)
11530daf62d9SStanislav Sedov {
11540daf62d9SStanislav Sedov 
11550daf62d9SStanislav Sedov 	assert(sock);
11560daf62d9SStanislav Sedov 	if (procstat->type == PROCSTAT_KVM) {
11570daf62d9SStanislav Sedov 		return (procstat_get_socket_info_kvm(procstat->kd, fst, sock,
11580daf62d9SStanislav Sedov 		    errbuf));
11590daf62d9SStanislav Sedov 	} else if (procstat->type == PROCSTAT_SYSCTL) {
11600daf62d9SStanislav Sedov 		return (procstat_get_socket_info_sysctl(fst, sock, errbuf));
11610daf62d9SStanislav Sedov 	} else {
116280905c35SJilles Tjoelker 		warnx("unknown access method: %d", procstat->type);
11630daf62d9SStanislav Sedov 		snprintf(errbuf, _POSIX2_LINE_MAX, "error");
11640daf62d9SStanislav Sedov 		return (1);
11650daf62d9SStanislav Sedov 	}
11660daf62d9SStanislav Sedov }
11670daf62d9SStanislav Sedov 
11680daf62d9SStanislav Sedov static int
11690daf62d9SStanislav Sedov procstat_get_socket_info_kvm(kvm_t *kd, struct filestat *fst,
11700daf62d9SStanislav Sedov     struct sockstat *sock, char *errbuf)
11710daf62d9SStanislav Sedov {
11720daf62d9SStanislav Sedov 	struct domain dom;
11730daf62d9SStanislav Sedov 	struct inpcb inpcb;
11740daf62d9SStanislav Sedov 	struct protosw proto;
11750daf62d9SStanislav Sedov 	struct socket s;
11760daf62d9SStanislav Sedov 	struct unpcb unpcb;
11770daf62d9SStanislav Sedov 	ssize_t len;
11780daf62d9SStanislav Sedov 	void *so;
11790daf62d9SStanislav Sedov 
11800daf62d9SStanislav Sedov 	assert(kd);
11810daf62d9SStanislav Sedov 	assert(sock);
11820daf62d9SStanislav Sedov 	assert(fst);
11830daf62d9SStanislav Sedov 	bzero(sock, sizeof(*sock));
11840daf62d9SStanislav Sedov 	so = fst->fs_typedep;
11850daf62d9SStanislav Sedov 	if (so == NULL)
11860daf62d9SStanislav Sedov 		goto fail;
11870daf62d9SStanislav Sedov 	sock->so_addr = (uintptr_t)so;
11880daf62d9SStanislav Sedov 	/* fill in socket */
11890daf62d9SStanislav Sedov 	if (!kvm_read_all(kd, (unsigned long)so, &s,
11900daf62d9SStanislav Sedov 	    sizeof(struct socket))) {
11910daf62d9SStanislav Sedov 		warnx("can't read sock at %p", (void *)so);
11920daf62d9SStanislav Sedov 		goto fail;
11930daf62d9SStanislav Sedov 	}
11940daf62d9SStanislav Sedov 	/* fill in protosw entry */
11950daf62d9SStanislav Sedov 	if (!kvm_read_all(kd, (unsigned long)s.so_proto, &proto,
11960daf62d9SStanislav Sedov 	    sizeof(struct protosw))) {
11970daf62d9SStanislav Sedov 		warnx("can't read protosw at %p", (void *)s.so_proto);
11980daf62d9SStanislav Sedov 		goto fail;
11990daf62d9SStanislav Sedov 	}
12000daf62d9SStanislav Sedov 	/* fill in domain */
12010daf62d9SStanislav Sedov 	if (!kvm_read_all(kd, (unsigned long)proto.pr_domain, &dom,
12020daf62d9SStanislav Sedov 	    sizeof(struct domain))) {
12030daf62d9SStanislav Sedov 		warnx("can't read domain at %p",
12040daf62d9SStanislav Sedov 		    (void *)proto.pr_domain);
12050daf62d9SStanislav Sedov 		goto fail;
12060daf62d9SStanislav Sedov 	}
12070daf62d9SStanislav Sedov 	if ((len = kvm_read(kd, (unsigned long)dom.dom_name, sock->dname,
12080daf62d9SStanislav Sedov 	    sizeof(sock->dname) - 1)) < 0) {
12090daf62d9SStanislav Sedov 		warnx("can't read domain name at %p", (void *)dom.dom_name);
12100daf62d9SStanislav Sedov 		sock->dname[0] = '\0';
12110daf62d9SStanislav Sedov 	}
12120daf62d9SStanislav Sedov 	else
12130daf62d9SStanislav Sedov 		sock->dname[len] = '\0';
12140daf62d9SStanislav Sedov 
12150daf62d9SStanislav Sedov 	/*
12160daf62d9SStanislav Sedov 	 * Fill in known data.
12170daf62d9SStanislav Sedov 	 */
12180daf62d9SStanislav Sedov 	sock->type = s.so_type;
12190daf62d9SStanislav Sedov 	sock->proto = proto.pr_protocol;
12200daf62d9SStanislav Sedov 	sock->dom_family = dom.dom_family;
12210daf62d9SStanislav Sedov 	sock->so_pcb = (uintptr_t)s.so_pcb;
12220daf62d9SStanislav Sedov 
12230daf62d9SStanislav Sedov 	/*
12240daf62d9SStanislav Sedov 	 * Protocol specific data.
12250daf62d9SStanislav Sedov 	 */
12260daf62d9SStanislav Sedov 	switch(dom.dom_family) {
12270daf62d9SStanislav Sedov 	case AF_INET:
12280daf62d9SStanislav Sedov 	case AF_INET6:
12290daf62d9SStanislav Sedov 		if (proto.pr_protocol == IPPROTO_TCP) {
12300daf62d9SStanislav Sedov 			if (s.so_pcb) {
12310daf62d9SStanislav Sedov 				if (kvm_read(kd, (u_long)s.so_pcb,
12320daf62d9SStanislav Sedov 				    (char *)&inpcb, sizeof(struct inpcb))
12330daf62d9SStanislav Sedov 				    != sizeof(struct inpcb)) {
12340daf62d9SStanislav Sedov 					warnx("can't read inpcb at %p",
12350daf62d9SStanislav Sedov 					    (void *)s.so_pcb);
12360daf62d9SStanislav Sedov 				} else
12370daf62d9SStanislav Sedov 					sock->inp_ppcb =
12380daf62d9SStanislav Sedov 					    (uintptr_t)inpcb.inp_ppcb;
12390daf62d9SStanislav Sedov 			}
12400daf62d9SStanislav Sedov 		}
12410daf62d9SStanislav Sedov 		break;
12420daf62d9SStanislav Sedov 	case AF_UNIX:
12430daf62d9SStanislav Sedov 		if (s.so_pcb) {
12440daf62d9SStanislav Sedov 			if (kvm_read(kd, (u_long)s.so_pcb, (char *)&unpcb,
12450daf62d9SStanislav Sedov 			    sizeof(struct unpcb)) != sizeof(struct unpcb)){
12460daf62d9SStanislav Sedov 				warnx("can't read unpcb at %p",
12470daf62d9SStanislav Sedov 				    (void *)s.so_pcb);
12480daf62d9SStanislav Sedov 			} else if (unpcb.unp_conn) {
12490daf62d9SStanislav Sedov 				sock->so_rcv_sb_state = s.so_rcv.sb_state;
12500daf62d9SStanislav Sedov 				sock->so_snd_sb_state = s.so_snd.sb_state;
12510daf62d9SStanislav Sedov 				sock->unp_conn = (uintptr_t)unpcb.unp_conn;
12520daf62d9SStanislav Sedov 			}
12530daf62d9SStanislav Sedov 		}
12540daf62d9SStanislav Sedov 		break;
12550daf62d9SStanislav Sedov 	default:
12560daf62d9SStanislav Sedov 		break;
12570daf62d9SStanislav Sedov 	}
12580daf62d9SStanislav Sedov 	return (0);
12590daf62d9SStanislav Sedov 
12600daf62d9SStanislav Sedov fail:
12610daf62d9SStanislav Sedov 	snprintf(errbuf, _POSIX2_LINE_MAX, "error");
12620daf62d9SStanislav Sedov 	return (1);
12630daf62d9SStanislav Sedov }
12640daf62d9SStanislav Sedov 
12650daf62d9SStanislav Sedov static int
12660daf62d9SStanislav Sedov procstat_get_socket_info_sysctl(struct filestat *fst, struct sockstat *sock,
12670daf62d9SStanislav Sedov     char *errbuf __unused)
12680daf62d9SStanislav Sedov {
12690daf62d9SStanislav Sedov 	struct kinfo_file *kif;
12700daf62d9SStanislav Sedov 
12710daf62d9SStanislav Sedov 	assert(sock);
12720daf62d9SStanislav Sedov 	assert(fst);
12730daf62d9SStanislav Sedov 	bzero(sock, sizeof(*sock));
12740daf62d9SStanislav Sedov 	kif = fst->fs_typedep;
12750daf62d9SStanislav Sedov 	if (kif == NULL)
12760daf62d9SStanislav Sedov 		return (0);
12770daf62d9SStanislav Sedov 
12780daf62d9SStanislav Sedov 	/*
12790daf62d9SStanislav Sedov 	 * Fill in known data.
12800daf62d9SStanislav Sedov 	 */
12810daf62d9SStanislav Sedov 	sock->type = kif->kf_sock_type;
12820daf62d9SStanislav Sedov 	sock->proto = kif->kf_sock_protocol;
12830daf62d9SStanislav Sedov 	sock->dom_family = kif->kf_sock_domain;
12840daf62d9SStanislav Sedov 	sock->so_pcb = kif->kf_un.kf_sock.kf_sock_pcb;
12850daf62d9SStanislav Sedov 	strlcpy(sock->dname, kif->kf_path, sizeof(sock->dname));
12860daf62d9SStanislav Sedov 	bcopy(&kif->kf_sa_local, &sock->sa_local, kif->kf_sa_local.ss_len);
12870daf62d9SStanislav Sedov 	bcopy(&kif->kf_sa_peer, &sock->sa_peer, kif->kf_sa_peer.ss_len);
12880daf62d9SStanislav Sedov 
12890daf62d9SStanislav Sedov 	/*
12900daf62d9SStanislav Sedov 	 * Protocol specific data.
12910daf62d9SStanislav Sedov 	 */
12920daf62d9SStanislav Sedov 	switch(sock->dom_family) {
12930daf62d9SStanislav Sedov 	case AF_INET:
12940daf62d9SStanislav Sedov 	case AF_INET6:
12950daf62d9SStanislav Sedov 		if (sock->proto == IPPROTO_TCP)
12960daf62d9SStanislav Sedov 			sock->inp_ppcb = kif->kf_un.kf_sock.kf_sock_inpcb;
12970daf62d9SStanislav Sedov 		break;
12980daf62d9SStanislav Sedov 	case AF_UNIX:
12990daf62d9SStanislav Sedov 		if (kif->kf_un.kf_sock.kf_sock_unpconn != 0) {
13000daf62d9SStanislav Sedov 				sock->so_rcv_sb_state =
13010daf62d9SStanislav Sedov 				    kif->kf_un.kf_sock.kf_sock_rcv_sb_state;
13020daf62d9SStanislav Sedov 				sock->so_snd_sb_state =
13030daf62d9SStanislav Sedov 				    kif->kf_un.kf_sock.kf_sock_snd_sb_state;
13040daf62d9SStanislav Sedov 				sock->unp_conn =
13050daf62d9SStanislav Sedov 				    kif->kf_un.kf_sock.kf_sock_unpconn;
13060daf62d9SStanislav Sedov 		}
13070daf62d9SStanislav Sedov 		break;
13080daf62d9SStanislav Sedov 	default:
13090daf62d9SStanislav Sedov 		break;
13100daf62d9SStanislav Sedov 	}
13110daf62d9SStanislav Sedov 	return (0);
13120daf62d9SStanislav Sedov }
13130daf62d9SStanislav Sedov 
13140daf62d9SStanislav Sedov /*
13150daf62d9SStanislav Sedov  * Descriptor flags to filestat translation.
13160daf62d9SStanislav Sedov  */
13170daf62d9SStanislav Sedov static int
13180daf62d9SStanislav Sedov to_filestat_flags(int flags)
13190daf62d9SStanislav Sedov {
13200daf62d9SStanislav Sedov 	static struct {
13210daf62d9SStanislav Sedov 		int flag;
13220daf62d9SStanislav Sedov 		int fst_flag;
13230daf62d9SStanislav Sedov 	} fstflags[] = {
13240daf62d9SStanislav Sedov 		{ FREAD, PS_FST_FFLAG_READ },
13250daf62d9SStanislav Sedov 		{ FWRITE, PS_FST_FFLAG_WRITE },
13260daf62d9SStanislav Sedov 		{ O_APPEND, PS_FST_FFLAG_APPEND },
13270daf62d9SStanislav Sedov 		{ O_ASYNC, PS_FST_FFLAG_ASYNC },
13280daf62d9SStanislav Sedov 		{ O_CREAT, PS_FST_FFLAG_CREAT },
13290daf62d9SStanislav Sedov 		{ O_DIRECT, PS_FST_FFLAG_DIRECT },
13300daf62d9SStanislav Sedov 		{ O_EXCL, PS_FST_FFLAG_EXCL },
13310daf62d9SStanislav Sedov 		{ O_EXEC, PS_FST_FFLAG_EXEC },
13320daf62d9SStanislav Sedov 		{ O_EXLOCK, PS_FST_FFLAG_EXLOCK },
13330daf62d9SStanislav Sedov 		{ O_NOFOLLOW, PS_FST_FFLAG_NOFOLLOW },
13340daf62d9SStanislav Sedov 		{ O_NONBLOCK, PS_FST_FFLAG_NONBLOCK },
13350daf62d9SStanislav Sedov 		{ O_SHLOCK, PS_FST_FFLAG_SHLOCK },
13360daf62d9SStanislav Sedov 		{ O_SYNC, PS_FST_FFLAG_SYNC },
13370daf62d9SStanislav Sedov 		{ O_TRUNC, PS_FST_FFLAG_TRUNC }
13380daf62d9SStanislav Sedov 	};
13390daf62d9SStanislav Sedov #define NFSTFLAGS	(sizeof(fstflags) / sizeof(*fstflags))
13400daf62d9SStanislav Sedov 	int fst_flags;
13410daf62d9SStanislav Sedov 	unsigned int i;
13420daf62d9SStanislav Sedov 
13430daf62d9SStanislav Sedov 	fst_flags = 0;
13440daf62d9SStanislav Sedov 	for (i = 0; i < NFSTFLAGS; i++)
13450daf62d9SStanislav Sedov 		if (flags & fstflags[i].flag)
13460daf62d9SStanislav Sedov 			fst_flags |= fstflags[i].fst_flag;
13470daf62d9SStanislav Sedov 	return (fst_flags);
13480daf62d9SStanislav Sedov }
13490daf62d9SStanislav Sedov 
13500daf62d9SStanislav Sedov /*
13510daf62d9SStanislav Sedov  * Vnode type to filestate translation.
13520daf62d9SStanislav Sedov  */
13530daf62d9SStanislav Sedov static int
13540daf62d9SStanislav Sedov vntype2psfsttype(int type)
13550daf62d9SStanislav Sedov {
13560daf62d9SStanislav Sedov 	static struct {
13570daf62d9SStanislav Sedov 		int	vtype;
13580daf62d9SStanislav Sedov 		int	fst_vtype;
13590daf62d9SStanislav Sedov 	} vt2fst[] = {
13600daf62d9SStanislav Sedov 		{ VBAD, PS_FST_VTYPE_VBAD },
13610daf62d9SStanislav Sedov 		{ VBLK, PS_FST_VTYPE_VBLK },
13620daf62d9SStanislav Sedov 		{ VCHR, PS_FST_VTYPE_VCHR },
13630daf62d9SStanislav Sedov 		{ VDIR, PS_FST_VTYPE_VDIR },
13640daf62d9SStanislav Sedov 		{ VFIFO, PS_FST_VTYPE_VFIFO },
13650daf62d9SStanislav Sedov 		{ VLNK, PS_FST_VTYPE_VLNK },
13660daf62d9SStanislav Sedov 		{ VNON, PS_FST_VTYPE_VNON },
13670daf62d9SStanislav Sedov 		{ VREG, PS_FST_VTYPE_VREG },
13680daf62d9SStanislav Sedov 		{ VSOCK, PS_FST_VTYPE_VSOCK }
13690daf62d9SStanislav Sedov 	};
13700daf62d9SStanislav Sedov #define	NVFTYPES	(sizeof(vt2fst) / sizeof(*vt2fst))
13710daf62d9SStanislav Sedov 	unsigned int i, fst_type;
13720daf62d9SStanislav Sedov 
13730daf62d9SStanislav Sedov 	fst_type = PS_FST_VTYPE_UNKNOWN;
13740daf62d9SStanislav Sedov 	for (i = 0; i < NVFTYPES; i++) {
13750daf62d9SStanislav Sedov 		if (type == vt2fst[i].vtype) {
13760daf62d9SStanislav Sedov 			fst_type = vt2fst[i].fst_vtype;
13770daf62d9SStanislav Sedov 			break;
13780daf62d9SStanislav Sedov 		}
13790daf62d9SStanislav Sedov 	}
13800daf62d9SStanislav Sedov 	return (fst_type);
13810daf62d9SStanislav Sedov }
13820daf62d9SStanislav Sedov 
13830daf62d9SStanislav Sedov static char *
13840daf62d9SStanislav Sedov getmnton(kvm_t *kd, struct mount *m)
13850daf62d9SStanislav Sedov {
13865f301949SBen Laurie 	struct mount mnt;
13870daf62d9SStanislav Sedov 	static struct mtab {
13880daf62d9SStanislav Sedov 		struct mtab *next;
13890daf62d9SStanislav Sedov 		struct mount *m;
13900daf62d9SStanislav Sedov 		char mntonname[MNAMELEN + 1];
13910daf62d9SStanislav Sedov 	} *mhead = NULL;
13920daf62d9SStanislav Sedov 	struct mtab *mt;
13930daf62d9SStanislav Sedov 
13940daf62d9SStanislav Sedov 	for (mt = mhead; mt != NULL; mt = mt->next)
13950daf62d9SStanislav Sedov 		if (m == mt->m)
13960daf62d9SStanislav Sedov 			return (mt->mntonname);
13970daf62d9SStanislav Sedov 	if (!kvm_read_all(kd, (unsigned long)m, &mnt, sizeof(struct mount))) {
13980daf62d9SStanislav Sedov 		warnx("can't read mount table at %p", (void *)m);
13990daf62d9SStanislav Sedov 		return (NULL);
14000daf62d9SStanislav Sedov 	}
14010daf62d9SStanislav Sedov 	if ((mt = malloc(sizeof (struct mtab))) == NULL)
14020daf62d9SStanislav Sedov 		err(1, NULL);
14030daf62d9SStanislav Sedov 	mt->m = m;
14040daf62d9SStanislav Sedov 	bcopy(&mnt.mnt_stat.f_mntonname[0], &mt->mntonname[0], MNAMELEN);
14055f301949SBen Laurie 	mt->mntonname[MNAMELEN] = '\0';
14060daf62d9SStanislav Sedov 	mt->next = mhead;
14070daf62d9SStanislav Sedov 	mhead = mt;
14080daf62d9SStanislav Sedov 	return (mt->mntonname);
14090daf62d9SStanislav Sedov }
1410