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> 570daf62d9SStanislav Sedov #define _KERNEL 580daf62d9SStanislav Sedov #include <sys/mount.h> 590daf62d9SStanislav Sedov #include <sys/pipe.h> 600daf62d9SStanislav Sedov #include <ufs/ufs/quota.h> 610daf62d9SStanislav Sedov #include <ufs/ufs/inode.h> 620daf62d9SStanislav Sedov #include <fs/devfs/devfs.h> 630daf62d9SStanislav Sedov #include <fs/devfs/devfs_int.h> 640daf62d9SStanislav Sedov #undef _KERNEL 650daf62d9SStanislav Sedov #include <nfs/nfsproto.h> 660daf62d9SStanislav Sedov #include <nfsclient/nfs.h> 670daf62d9SStanislav Sedov #include <nfsclient/nfsnode.h> 680daf62d9SStanislav Sedov 690daf62d9SStanislav Sedov #include <vm/vm.h> 700daf62d9SStanislav Sedov #include <vm/vm_map.h> 710daf62d9SStanislav Sedov #include <vm/vm_object.h> 720daf62d9SStanislav Sedov 730daf62d9SStanislav Sedov #include <net/route.h> 740daf62d9SStanislav Sedov #include <netinet/in.h> 750daf62d9SStanislav Sedov #include <netinet/in_systm.h> 760daf62d9SStanislav Sedov #include <netinet/ip.h> 770daf62d9SStanislav Sedov #include <netinet/in_pcb.h> 780daf62d9SStanislav Sedov 790daf62d9SStanislav Sedov #include <assert.h> 800daf62d9SStanislav Sedov #include <ctype.h> 810daf62d9SStanislav Sedov #include <err.h> 820daf62d9SStanislav Sedov #include <fcntl.h> 830daf62d9SStanislav Sedov #include <kvm.h> 840daf62d9SStanislav Sedov #include <libutil.h> 850daf62d9SStanislav Sedov #include <limits.h> 860daf62d9SStanislav Sedov #include <paths.h> 870daf62d9SStanislav Sedov #include <pwd.h> 880daf62d9SStanislav Sedov #include <stdio.h> 890daf62d9SStanislav Sedov #include <stdlib.h> 900daf62d9SStanislav Sedov #include <stddef.h> 910daf62d9SStanislav Sedov #include <string.h> 920daf62d9SStanislav Sedov #include <unistd.h> 930daf62d9SStanislav Sedov #include <netdb.h> 940daf62d9SStanislav Sedov 950daf62d9SStanislav Sedov #include <libprocstat.h> 960daf62d9SStanislav Sedov #include "libprocstat_internal.h" 970daf62d9SStanislav Sedov #include "common_kvm.h" 980daf62d9SStanislav Sedov 990daf62d9SStanislav Sedov int statfs(const char *, struct statfs *); /* XXX */ 1000daf62d9SStanislav Sedov 1010daf62d9SStanislav Sedov #define PROCSTAT_KVM 1 1020daf62d9SStanislav Sedov #define PROCSTAT_SYSCTL 2 1030daf62d9SStanislav Sedov 1040daf62d9SStanislav Sedov static char *getmnton(kvm_t *kd, struct mount *m); 1050daf62d9SStanislav Sedov static struct filestat_list *procstat_getfiles_kvm( 1060daf62d9SStanislav Sedov struct procstat *procstat, struct kinfo_proc *kp, int mmapped); 1070daf62d9SStanislav Sedov static struct filestat_list *procstat_getfiles_sysctl( 1080daf62d9SStanislav Sedov struct procstat *procstat, struct kinfo_proc *kp, int mmapped); 1090daf62d9SStanislav Sedov static int procstat_get_pipe_info_sysctl(struct filestat *fst, 1100daf62d9SStanislav Sedov struct pipestat *pipe, char *errbuf); 1110daf62d9SStanislav Sedov static int procstat_get_pipe_info_kvm(kvm_t *kd, struct filestat *fst, 1120daf62d9SStanislav Sedov struct pipestat *pipe, char *errbuf); 1130daf62d9SStanislav Sedov static int procstat_get_pts_info_sysctl(struct filestat *fst, 1140daf62d9SStanislav Sedov struct ptsstat *pts, char *errbuf); 1150daf62d9SStanislav Sedov static int procstat_get_pts_info_kvm(kvm_t *kd, struct filestat *fst, 1160daf62d9SStanislav Sedov struct ptsstat *pts, char *errbuf); 1170daf62d9SStanislav Sedov static int procstat_get_socket_info_sysctl(struct filestat *fst, 1180daf62d9SStanislav Sedov struct sockstat *sock, char *errbuf); 1190daf62d9SStanislav Sedov static int procstat_get_socket_info_kvm(kvm_t *kd, struct filestat *fst, 1200daf62d9SStanislav Sedov struct sockstat *sock, char *errbuf); 1210daf62d9SStanislav Sedov static int to_filestat_flags(int flags); 1220daf62d9SStanislav Sedov static int procstat_get_vnode_info_kvm(kvm_t *kd, struct filestat *fst, 1230daf62d9SStanislav Sedov struct vnstat *vn, char *errbuf); 1240daf62d9SStanislav Sedov static int procstat_get_vnode_info_sysctl(struct filestat *fst, 1250daf62d9SStanislav Sedov struct vnstat *vn, char *errbuf); 1260daf62d9SStanislav Sedov static int vntype2psfsttype(int type); 1270daf62d9SStanislav Sedov 1280daf62d9SStanislav Sedov void 1290daf62d9SStanislav Sedov procstat_close(struct procstat *procstat) 1300daf62d9SStanislav Sedov { 1310daf62d9SStanislav Sedov 1320daf62d9SStanislav Sedov assert(procstat); 1330daf62d9SStanislav Sedov if (procstat->type == PROCSTAT_KVM) 1340daf62d9SStanislav Sedov kvm_close(procstat->kd); 135d7b666aeSSergey Kandaurov free(procstat); 1360daf62d9SStanislav Sedov } 1370daf62d9SStanislav Sedov 1380daf62d9SStanislav Sedov struct procstat * 1390daf62d9SStanislav Sedov procstat_open_sysctl(void) 1400daf62d9SStanislav Sedov { 1410daf62d9SStanislav Sedov struct procstat *procstat; 1420daf62d9SStanislav Sedov 1430daf62d9SStanislav Sedov procstat = calloc(1, sizeof(*procstat)); 1440daf62d9SStanislav Sedov if (procstat == NULL) { 1450daf62d9SStanislav Sedov warn("malloc()"); 1460daf62d9SStanislav Sedov return (NULL); 1470daf62d9SStanislav Sedov } 1480daf62d9SStanislav Sedov procstat->type = PROCSTAT_SYSCTL; 1490daf62d9SStanislav Sedov return (procstat); 1500daf62d9SStanislav Sedov } 1510daf62d9SStanislav Sedov 1520daf62d9SStanislav Sedov struct procstat * 1530daf62d9SStanislav Sedov procstat_open_kvm(const char *nlistf, const char *memf) 1540daf62d9SStanislav Sedov { 1550daf62d9SStanislav Sedov struct procstat *procstat; 1560daf62d9SStanislav Sedov kvm_t *kd; 1570daf62d9SStanislav Sedov char buf[_POSIX2_LINE_MAX]; 1580daf62d9SStanislav Sedov 1590daf62d9SStanislav Sedov procstat = calloc(1, sizeof(*procstat)); 1600daf62d9SStanislav Sedov if (procstat == NULL) { 1610daf62d9SStanislav Sedov warn("malloc()"); 1620daf62d9SStanislav Sedov return (NULL); 1630daf62d9SStanislav Sedov } 1640daf62d9SStanislav Sedov kd = kvm_openfiles(nlistf, memf, NULL, O_RDONLY, buf); 1650daf62d9SStanislav Sedov if (kd == NULL) { 1660daf62d9SStanislav Sedov warnx("kvm_openfiles(): %s", buf); 1670daf62d9SStanislav Sedov free(procstat); 1680daf62d9SStanislav Sedov return (NULL); 1690daf62d9SStanislav Sedov } 1700daf62d9SStanislav Sedov procstat->type = PROCSTAT_KVM; 1710daf62d9SStanislav Sedov procstat->kd = kd; 1720daf62d9SStanislav Sedov return (procstat); 1730daf62d9SStanislav Sedov } 1740daf62d9SStanislav Sedov 1750daf62d9SStanislav Sedov struct kinfo_proc * 1760daf62d9SStanislav Sedov procstat_getprocs(struct procstat *procstat, int what, int arg, 1770daf62d9SStanislav Sedov unsigned int *count) 1780daf62d9SStanislav Sedov { 1790daf62d9SStanislav Sedov struct kinfo_proc *p0, *p; 1800daf62d9SStanislav Sedov size_t len; 1810daf62d9SStanislav Sedov int name[4]; 1820daf62d9SStanislav Sedov int error; 1830daf62d9SStanislav Sedov 1840daf62d9SStanislav Sedov assert(procstat); 1850daf62d9SStanislav Sedov assert(count); 1860daf62d9SStanislav Sedov p = NULL; 1870daf62d9SStanislav Sedov if (procstat->type == PROCSTAT_KVM) { 1880daf62d9SStanislav Sedov p0 = kvm_getprocs(procstat->kd, what, arg, count); 1890daf62d9SStanislav Sedov if (p0 == NULL || count == 0) 1900daf62d9SStanislav Sedov return (NULL); 1910daf62d9SStanislav Sedov len = *count * sizeof(*p); 1920daf62d9SStanislav Sedov p = malloc(len); 1930daf62d9SStanislav Sedov if (p == NULL) { 19465869214SJilles Tjoelker warnx("malloc(%zu)", len); 1950daf62d9SStanislav Sedov goto fail; 1960daf62d9SStanislav Sedov } 1970daf62d9SStanislav Sedov bcopy(p0, p, len); 1980daf62d9SStanislav Sedov return (p); 1990daf62d9SStanislav Sedov } else if (procstat->type == PROCSTAT_SYSCTL) { 2000daf62d9SStanislav Sedov len = 0; 2010daf62d9SStanislav Sedov name[0] = CTL_KERN; 2020daf62d9SStanislav Sedov name[1] = KERN_PROC; 2030daf62d9SStanislav Sedov name[2] = what; 2040daf62d9SStanislav Sedov name[3] = arg; 2050daf62d9SStanislav Sedov error = sysctl(name, 4, NULL, &len, NULL, 0); 2060daf62d9SStanislav Sedov if (error < 0 && errno != EPERM) { 2070daf62d9SStanislav Sedov warn("sysctl(kern.proc)"); 2080daf62d9SStanislav Sedov goto fail; 2090daf62d9SStanislav Sedov } 2100daf62d9SStanislav Sedov if (len == 0) { 2110daf62d9SStanislav Sedov warnx("no processes?"); 2120daf62d9SStanislav Sedov goto fail; 2130daf62d9SStanislav Sedov } 2140daf62d9SStanislav Sedov p = malloc(len); 2150daf62d9SStanislav Sedov if (p == NULL) { 21665869214SJilles Tjoelker warnx("malloc(%zu)", len); 2170daf62d9SStanislav Sedov goto fail; 2180daf62d9SStanislav Sedov } 2190daf62d9SStanislav Sedov error = sysctl(name, 4, p, &len, NULL, 0); 2200daf62d9SStanislav Sedov if (error < 0 && errno != EPERM) { 2210daf62d9SStanislav Sedov warn("sysctl(kern.proc)"); 2220daf62d9SStanislav Sedov goto fail; 2230daf62d9SStanislav Sedov } 2240daf62d9SStanislav Sedov /* Perform simple consistency checks. */ 2250daf62d9SStanislav Sedov if ((len % sizeof(*p)) != 0 || p->ki_structsize != sizeof(*p)) { 2260daf62d9SStanislav Sedov warnx("kinfo_proc structure size mismatch"); 2270daf62d9SStanislav Sedov goto fail; 2280daf62d9SStanislav Sedov } 2290daf62d9SStanislav Sedov *count = len / sizeof(*p); 2300daf62d9SStanislav Sedov return (p); 2310daf62d9SStanislav Sedov } else { 232*80905c35SJilles Tjoelker warnx("unknown access method: %d", procstat->type); 2330daf62d9SStanislav Sedov return (NULL); 2340daf62d9SStanislav Sedov } 2350daf62d9SStanislav Sedov fail: 2360daf62d9SStanislav Sedov if (p) 2370daf62d9SStanislav Sedov free(p); 2380daf62d9SStanislav Sedov return (NULL); 2390daf62d9SStanislav Sedov } 2400daf62d9SStanislav Sedov 2410daf62d9SStanislav Sedov void 2420daf62d9SStanislav Sedov procstat_freeprocs(struct procstat *procstat __unused, struct kinfo_proc *p) 2430daf62d9SStanislav Sedov { 2440daf62d9SStanislav Sedov 2450daf62d9SStanislav Sedov if (p != NULL) 2460daf62d9SStanislav Sedov free(p); 2470daf62d9SStanislav Sedov p = NULL; 2480daf62d9SStanislav Sedov } 2490daf62d9SStanislav Sedov 2500daf62d9SStanislav Sedov struct filestat_list * 2510daf62d9SStanislav Sedov procstat_getfiles(struct procstat *procstat, struct kinfo_proc *kp, int mmapped) 2520daf62d9SStanislav Sedov { 2530daf62d9SStanislav Sedov 2540daf62d9SStanislav Sedov if (procstat->type == PROCSTAT_SYSCTL) 2550daf62d9SStanislav Sedov return (procstat_getfiles_sysctl(procstat, kp, mmapped)); 2560daf62d9SStanislav Sedov else if (procstat->type == PROCSTAT_KVM) 2570daf62d9SStanislav Sedov return (procstat_getfiles_kvm(procstat, kp, mmapped)); 2580daf62d9SStanislav Sedov else 2590daf62d9SStanislav Sedov return (NULL); 2600daf62d9SStanislav Sedov } 2610daf62d9SStanislav Sedov 2620daf62d9SStanislav Sedov void 2630daf62d9SStanislav Sedov procstat_freefiles(struct procstat *procstat, struct filestat_list *head) 2640daf62d9SStanislav Sedov { 2650daf62d9SStanislav Sedov struct filestat *fst, *tmp; 2660daf62d9SStanislav Sedov 2670daf62d9SStanislav Sedov STAILQ_FOREACH_SAFE(fst, head, next, tmp) { 2680daf62d9SStanislav Sedov if (fst->fs_path != NULL) 2690daf62d9SStanislav Sedov free(fst->fs_path); 2700daf62d9SStanislav Sedov free(fst); 2710daf62d9SStanislav Sedov } 2720daf62d9SStanislav Sedov free(head); 2730daf62d9SStanislav Sedov if (procstat->vmentries != NULL) { 2740daf62d9SStanislav Sedov free(procstat->vmentries); 2750daf62d9SStanislav Sedov procstat->vmentries = NULL; 2760daf62d9SStanislav Sedov } 2770daf62d9SStanislav Sedov if (procstat->files != NULL) { 2780daf62d9SStanislav Sedov free(procstat->files); 2790daf62d9SStanislav Sedov procstat->files = NULL; 2800daf62d9SStanislav Sedov } 2810daf62d9SStanislav Sedov } 2820daf62d9SStanislav Sedov 2830daf62d9SStanislav Sedov static struct filestat * 2840daf62d9SStanislav Sedov filestat_new_entry(void *typedep, int type, int fd, int fflags, int uflags, 2850daf62d9SStanislav Sedov int refcount, off_t offset, char *path) 2860daf62d9SStanislav Sedov { 2870daf62d9SStanislav Sedov struct filestat *entry; 2880daf62d9SStanislav Sedov 2890daf62d9SStanislav Sedov entry = calloc(1, sizeof(*entry)); 2900daf62d9SStanislav Sedov if (entry == NULL) { 2910daf62d9SStanislav Sedov warn("malloc()"); 2920daf62d9SStanislav Sedov return (NULL); 2930daf62d9SStanislav Sedov } 2940daf62d9SStanislav Sedov entry->fs_typedep = typedep; 2950daf62d9SStanislav Sedov entry->fs_fflags = fflags; 2960daf62d9SStanislav Sedov entry->fs_uflags = uflags; 2970daf62d9SStanislav Sedov entry->fs_fd = fd; 2980daf62d9SStanislav Sedov entry->fs_type = type; 2990daf62d9SStanislav Sedov entry->fs_ref_count = refcount; 3000daf62d9SStanislav Sedov entry->fs_offset = offset; 3010daf62d9SStanislav Sedov entry->fs_path = path; 3020daf62d9SStanislav Sedov return (entry); 3030daf62d9SStanislav Sedov } 3040daf62d9SStanislav Sedov 3050daf62d9SStanislav Sedov static struct vnode * 3060daf62d9SStanislav Sedov getctty(kvm_t *kd, struct kinfo_proc *kp) 3070daf62d9SStanislav Sedov { 3080daf62d9SStanislav Sedov struct pgrp pgrp; 3090daf62d9SStanislav Sedov struct proc proc; 3100daf62d9SStanislav Sedov struct session sess; 3110daf62d9SStanislav Sedov int error; 3120daf62d9SStanislav Sedov 3130daf62d9SStanislav Sedov assert(kp); 3140daf62d9SStanislav Sedov error = kvm_read_all(kd, (unsigned long)kp->ki_paddr, &proc, 3150daf62d9SStanislav Sedov sizeof(proc)); 3160daf62d9SStanislav Sedov if (error == 0) { 3170daf62d9SStanislav Sedov warnx("can't read proc struct at %p for pid %d", 3180daf62d9SStanislav Sedov kp->ki_paddr, kp->ki_pid); 3190daf62d9SStanislav Sedov return (NULL); 3200daf62d9SStanislav Sedov } 3210daf62d9SStanislav Sedov if (proc.p_pgrp == NULL) 3220daf62d9SStanislav Sedov return (NULL); 3230daf62d9SStanislav Sedov error = kvm_read_all(kd, (unsigned long)proc.p_pgrp, &pgrp, 3240daf62d9SStanislav Sedov sizeof(pgrp)); 3250daf62d9SStanislav Sedov if (error == 0) { 3260daf62d9SStanislav Sedov warnx("can't read pgrp struct at %p for pid %d", 3270daf62d9SStanislav Sedov proc.p_pgrp, kp->ki_pid); 3280daf62d9SStanislav Sedov return (NULL); 3290daf62d9SStanislav Sedov } 3300daf62d9SStanislav Sedov error = kvm_read_all(kd, (unsigned long)pgrp.pg_session, &sess, 3310daf62d9SStanislav Sedov sizeof(sess)); 3320daf62d9SStanislav Sedov if (error == 0) { 3330daf62d9SStanislav Sedov warnx("can't read session struct at %p for pid %d", 3340daf62d9SStanislav Sedov pgrp.pg_session, kp->ki_pid); 3350daf62d9SStanislav Sedov return (NULL); 3360daf62d9SStanislav Sedov } 3370daf62d9SStanislav Sedov return (sess.s_ttyvp); 3380daf62d9SStanislav Sedov } 3390daf62d9SStanislav Sedov 3400daf62d9SStanislav Sedov static struct filestat_list * 3410daf62d9SStanislav Sedov procstat_getfiles_kvm(struct procstat *procstat, struct kinfo_proc *kp, int mmapped) 3420daf62d9SStanislav Sedov { 3430daf62d9SStanislav Sedov struct file file; 3440daf62d9SStanislav Sedov struct filedesc filed; 3450daf62d9SStanislav Sedov struct vm_map_entry vmentry; 3460daf62d9SStanislav Sedov struct vm_object object; 3470daf62d9SStanislav Sedov struct vmspace vmspace; 3480daf62d9SStanislav Sedov vm_map_entry_t entryp; 3490daf62d9SStanislav Sedov vm_map_t map; 3500daf62d9SStanislav Sedov vm_object_t objp; 3510daf62d9SStanislav Sedov struct vnode *vp; 3520daf62d9SStanislav Sedov struct file **ofiles; 3530daf62d9SStanislav Sedov struct filestat *entry; 3540daf62d9SStanislav Sedov struct filestat_list *head; 3550daf62d9SStanislav Sedov kvm_t *kd; 3560daf62d9SStanislav Sedov void *data; 3570daf62d9SStanislav Sedov int i, fflags; 3580daf62d9SStanislav Sedov int prot, type; 3590daf62d9SStanislav Sedov unsigned int nfiles; 3600daf62d9SStanislav Sedov 3610daf62d9SStanislav Sedov assert(procstat); 3620daf62d9SStanislav Sedov kd = procstat->kd; 3630daf62d9SStanislav Sedov if (kd == NULL) 3640daf62d9SStanislav Sedov return (NULL); 3650daf62d9SStanislav Sedov if (kp->ki_fd == NULL) 3660daf62d9SStanislav Sedov return (NULL); 3670daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)kp->ki_fd, &filed, 3680daf62d9SStanislav Sedov sizeof(filed))) { 3690daf62d9SStanislav Sedov warnx("can't read filedesc at %p", (void *)kp->ki_fd); 3700daf62d9SStanislav Sedov return (NULL); 3710daf62d9SStanislav Sedov } 3720daf62d9SStanislav Sedov 3730daf62d9SStanislav Sedov /* 3740daf62d9SStanislav Sedov * Allocate list head. 3750daf62d9SStanislav Sedov */ 3760daf62d9SStanislav Sedov head = malloc(sizeof(*head)); 3770daf62d9SStanislav Sedov if (head == NULL) 3780daf62d9SStanislav Sedov return (NULL); 3790daf62d9SStanislav Sedov STAILQ_INIT(head); 3800daf62d9SStanislav Sedov 3810daf62d9SStanislav Sedov /* root directory vnode, if one. */ 3820daf62d9SStanislav Sedov if (filed.fd_rdir) { 3830daf62d9SStanislav Sedov entry = filestat_new_entry(filed.fd_rdir, PS_FST_TYPE_VNODE, -1, 3840daf62d9SStanislav Sedov PS_FST_FFLAG_READ, PS_FST_UFLAG_RDIR, 0, 0, NULL); 3850daf62d9SStanislav Sedov if (entry != NULL) 3860daf62d9SStanislav Sedov STAILQ_INSERT_TAIL(head, entry, next); 3870daf62d9SStanislav Sedov } 3880daf62d9SStanislav Sedov /* current working directory vnode. */ 3890daf62d9SStanislav Sedov if (filed.fd_cdir) { 3900daf62d9SStanislav Sedov entry = filestat_new_entry(filed.fd_cdir, PS_FST_TYPE_VNODE, -1, 3910daf62d9SStanislav Sedov PS_FST_FFLAG_READ, PS_FST_UFLAG_CDIR, 0, 0, NULL); 3920daf62d9SStanislav Sedov if (entry != NULL) 3930daf62d9SStanislav Sedov STAILQ_INSERT_TAIL(head, entry, next); 3940daf62d9SStanislav Sedov } 3950daf62d9SStanislav Sedov /* jail root, if any. */ 3960daf62d9SStanislav Sedov if (filed.fd_jdir) { 3970daf62d9SStanislav Sedov entry = filestat_new_entry(filed.fd_jdir, PS_FST_TYPE_VNODE, -1, 3980daf62d9SStanislav Sedov PS_FST_FFLAG_READ, PS_FST_UFLAG_JAIL, 0, 0, NULL); 3990daf62d9SStanislav Sedov if (entry != NULL) 4000daf62d9SStanislav Sedov STAILQ_INSERT_TAIL(head, entry, next); 4010daf62d9SStanislav Sedov } 4020daf62d9SStanislav Sedov /* ktrace vnode, if one */ 4030daf62d9SStanislav Sedov if (kp->ki_tracep) { 4040daf62d9SStanislav Sedov entry = filestat_new_entry(kp->ki_tracep, PS_FST_TYPE_VNODE, -1, 4050daf62d9SStanislav Sedov PS_FST_FFLAG_READ | PS_FST_FFLAG_WRITE, 4060daf62d9SStanislav Sedov PS_FST_UFLAG_TRACE, 0, 0, NULL); 4070daf62d9SStanislav Sedov if (entry != NULL) 4080daf62d9SStanislav Sedov STAILQ_INSERT_TAIL(head, entry, next); 4090daf62d9SStanislav Sedov } 4100daf62d9SStanislav Sedov /* text vnode, if one */ 4110daf62d9SStanislav Sedov if (kp->ki_textvp) { 4120daf62d9SStanislav Sedov entry = filestat_new_entry(kp->ki_textvp, PS_FST_TYPE_VNODE, -1, 4130daf62d9SStanislav Sedov PS_FST_FFLAG_READ, PS_FST_UFLAG_TEXT, 0, 0, NULL); 4140daf62d9SStanislav Sedov if (entry != NULL) 4150daf62d9SStanislav Sedov STAILQ_INSERT_TAIL(head, entry, next); 4160daf62d9SStanislav Sedov } 4170daf62d9SStanislav Sedov /* Controlling terminal. */ 4180daf62d9SStanislav Sedov if ((vp = getctty(kd, kp)) != NULL) { 4190daf62d9SStanislav Sedov entry = filestat_new_entry(vp, PS_FST_TYPE_VNODE, -1, 4200daf62d9SStanislav Sedov PS_FST_FFLAG_READ | PS_FST_FFLAG_WRITE, 4210daf62d9SStanislav Sedov PS_FST_UFLAG_CTTY, 0, 0, NULL); 4220daf62d9SStanislav Sedov if (entry != NULL) 4230daf62d9SStanislav Sedov STAILQ_INSERT_TAIL(head, entry, next); 4240daf62d9SStanislav Sedov } 4250daf62d9SStanislav Sedov 4260daf62d9SStanislav Sedov nfiles = filed.fd_lastfile + 1; 4270daf62d9SStanislav Sedov ofiles = malloc(nfiles * sizeof(struct file *)); 4280daf62d9SStanislav Sedov if (ofiles == NULL) { 42965869214SJilles Tjoelker warn("malloc(%zu)", nfiles * sizeof(struct file *)); 4300daf62d9SStanislav Sedov goto do_mmapped; 4310daf62d9SStanislav Sedov } 4320daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)filed.fd_ofiles, ofiles, 4330daf62d9SStanislav Sedov nfiles * sizeof(struct file *))) { 4340daf62d9SStanislav Sedov warnx("cannot read file structures at %p", 4350daf62d9SStanislav Sedov (void *)filed.fd_ofiles); 4360daf62d9SStanislav Sedov free(ofiles); 4370daf62d9SStanislav Sedov goto do_mmapped; 4380daf62d9SStanislav Sedov } 4390daf62d9SStanislav Sedov for (i = 0; i <= filed.fd_lastfile; i++) { 4400daf62d9SStanislav Sedov if (ofiles[i] == NULL) 4410daf62d9SStanislav Sedov continue; 4420daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)ofiles[i], &file, 4430daf62d9SStanislav Sedov sizeof(struct file))) { 4440daf62d9SStanislav Sedov warnx("can't read file %d at %p", i, 4450daf62d9SStanislav Sedov (void *)ofiles[i]); 4460daf62d9SStanislav Sedov continue; 4470daf62d9SStanislav Sedov } 4480daf62d9SStanislav Sedov switch (file.f_type) { 4490daf62d9SStanislav Sedov case DTYPE_VNODE: 4500daf62d9SStanislav Sedov type = PS_FST_TYPE_VNODE; 4510daf62d9SStanislav Sedov data = file.f_vnode; 4520daf62d9SStanislav Sedov break; 4530daf62d9SStanislav Sedov case DTYPE_SOCKET: 4540daf62d9SStanislav Sedov type = PS_FST_TYPE_SOCKET; 4550daf62d9SStanislav Sedov data = file.f_data; 4560daf62d9SStanislav Sedov break; 4570daf62d9SStanislav Sedov case DTYPE_PIPE: 4580daf62d9SStanislav Sedov type = PS_FST_TYPE_PIPE; 4590daf62d9SStanislav Sedov data = file.f_data; 4600daf62d9SStanislav Sedov break; 4610daf62d9SStanislav Sedov case DTYPE_FIFO: 4620daf62d9SStanislav Sedov type = PS_FST_TYPE_FIFO; 4630daf62d9SStanislav Sedov data = file.f_vnode; 4640daf62d9SStanislav Sedov break; 4650daf62d9SStanislav Sedov #ifdef DTYPE_PTS 4660daf62d9SStanislav Sedov case DTYPE_PTS: 4670daf62d9SStanislav Sedov type = PS_FST_TYPE_PTS; 4680daf62d9SStanislav Sedov data = file.f_data; 4690daf62d9SStanislav Sedov break; 4700daf62d9SStanislav Sedov #endif 4710daf62d9SStanislav Sedov default: 4720daf62d9SStanislav Sedov continue; 4730daf62d9SStanislav Sedov } 4740daf62d9SStanislav Sedov entry = filestat_new_entry(data, type, i, 4750daf62d9SStanislav Sedov to_filestat_flags(file.f_flag), 0, 0, 0, NULL); 4760daf62d9SStanislav Sedov if (entry != NULL) 4770daf62d9SStanislav Sedov STAILQ_INSERT_TAIL(head, entry, next); 4780daf62d9SStanislav Sedov } 4790daf62d9SStanislav Sedov free(ofiles); 4800daf62d9SStanislav Sedov 4810daf62d9SStanislav Sedov do_mmapped: 4820daf62d9SStanislav Sedov 4830daf62d9SStanislav Sedov /* 4840daf62d9SStanislav Sedov * Process mmapped files if requested. 4850daf62d9SStanislav Sedov */ 4860daf62d9SStanislav Sedov if (mmapped) { 4870daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)kp->ki_vmspace, &vmspace, 4880daf62d9SStanislav Sedov sizeof(vmspace))) { 4890daf62d9SStanislav Sedov warnx("can't read vmspace at %p", 4900daf62d9SStanislav Sedov (void *)kp->ki_vmspace); 4910daf62d9SStanislav Sedov goto exit; 4920daf62d9SStanislav Sedov } 4930daf62d9SStanislav Sedov map = &vmspace.vm_map; 4940daf62d9SStanislav Sedov 4950daf62d9SStanislav Sedov for (entryp = map->header.next; 4960daf62d9SStanislav Sedov entryp != &kp->ki_vmspace->vm_map.header; 4970daf62d9SStanislav Sedov entryp = vmentry.next) { 4980daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)entryp, &vmentry, 4990daf62d9SStanislav Sedov sizeof(vmentry))) { 5000daf62d9SStanislav Sedov warnx("can't read vm_map_entry at %p", 5010daf62d9SStanislav Sedov (void *)entryp); 5020daf62d9SStanislav Sedov continue; 5030daf62d9SStanislav Sedov } 5040daf62d9SStanislav Sedov if (vmentry.eflags & MAP_ENTRY_IS_SUB_MAP) 5050daf62d9SStanislav Sedov continue; 5060daf62d9SStanislav Sedov if ((objp = vmentry.object.vm_object) == NULL) 5070daf62d9SStanislav Sedov continue; 5080daf62d9SStanislav Sedov for (; objp; objp = object.backing_object) { 5090daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)objp, 5100daf62d9SStanislav Sedov &object, sizeof(object))) { 5110daf62d9SStanislav Sedov warnx("can't read vm_object at %p", 5120daf62d9SStanislav Sedov (void *)objp); 5130daf62d9SStanislav Sedov break; 5140daf62d9SStanislav Sedov } 5150daf62d9SStanislav Sedov } 5160daf62d9SStanislav Sedov 5170daf62d9SStanislav Sedov /* We want only vnode objects. */ 5180daf62d9SStanislav Sedov if (object.type != OBJT_VNODE) 5190daf62d9SStanislav Sedov continue; 5200daf62d9SStanislav Sedov 5210daf62d9SStanislav Sedov prot = vmentry.protection; 5220daf62d9SStanislav Sedov fflags = 0; 5230daf62d9SStanislav Sedov if (prot & VM_PROT_READ) 5240daf62d9SStanislav Sedov fflags = PS_FST_FFLAG_READ; 5250daf62d9SStanislav Sedov if (prot & VM_PROT_WRITE) 5260daf62d9SStanislav Sedov fflags |= PS_FST_FFLAG_WRITE; 5270daf62d9SStanislav Sedov 5280daf62d9SStanislav Sedov /* 5290daf62d9SStanislav Sedov * Create filestat entry. 5300daf62d9SStanislav Sedov */ 5310daf62d9SStanislav Sedov entry = filestat_new_entry(object.handle, 5320daf62d9SStanislav Sedov PS_FST_TYPE_VNODE, -1, fflags, 5330daf62d9SStanislav Sedov PS_FST_UFLAG_MMAP, 0, 0, NULL); 5340daf62d9SStanislav Sedov if (entry != NULL) 5350daf62d9SStanislav Sedov STAILQ_INSERT_TAIL(head, entry, next); 5360daf62d9SStanislav Sedov } 5370daf62d9SStanislav Sedov } 5380daf62d9SStanislav Sedov exit: 5390daf62d9SStanislav Sedov return (head); 5400daf62d9SStanislav Sedov } 5410daf62d9SStanislav Sedov 5420daf62d9SStanislav Sedov /* 5430daf62d9SStanislav Sedov * kinfo types to filestat translation. 5440daf62d9SStanislav Sedov */ 5450daf62d9SStanislav Sedov static int 5460daf62d9SStanislav Sedov kinfo_type2fst(int kftype) 5470daf62d9SStanislav Sedov { 5480daf62d9SStanislav Sedov static struct { 5490daf62d9SStanislav Sedov int kf_type; 5500daf62d9SStanislav Sedov int fst_type; 5510daf62d9SStanislav Sedov } kftypes2fst[] = { 5520daf62d9SStanislav Sedov { KF_TYPE_CRYPTO, PS_FST_TYPE_CRYPTO }, 5530daf62d9SStanislav Sedov { KF_TYPE_FIFO, PS_FST_TYPE_FIFO }, 5540daf62d9SStanislav Sedov { KF_TYPE_KQUEUE, PS_FST_TYPE_KQUEUE }, 5550daf62d9SStanislav Sedov { KF_TYPE_MQUEUE, PS_FST_TYPE_MQUEUE }, 5560daf62d9SStanislav Sedov { KF_TYPE_NONE, PS_FST_TYPE_NONE }, 5570daf62d9SStanislav Sedov { KF_TYPE_PIPE, PS_FST_TYPE_PIPE }, 5580daf62d9SStanislav Sedov { KF_TYPE_PTS, PS_FST_TYPE_PTS }, 5590daf62d9SStanislav Sedov { KF_TYPE_SEM, PS_FST_TYPE_SEM }, 5600daf62d9SStanislav Sedov { KF_TYPE_SHM, PS_FST_TYPE_SHM }, 5610daf62d9SStanislav Sedov { KF_TYPE_SOCKET, PS_FST_TYPE_SOCKET }, 5620daf62d9SStanislav Sedov { KF_TYPE_VNODE, PS_FST_TYPE_VNODE }, 5630daf62d9SStanislav Sedov { KF_TYPE_UNKNOWN, PS_FST_TYPE_UNKNOWN } 5640daf62d9SStanislav Sedov }; 5650daf62d9SStanislav Sedov #define NKFTYPES (sizeof(kftypes2fst) / sizeof(*kftypes2fst)) 5660daf62d9SStanislav Sedov unsigned int i; 5670daf62d9SStanislav Sedov 5680daf62d9SStanislav Sedov for (i = 0; i < NKFTYPES; i++) 5690daf62d9SStanislav Sedov if (kftypes2fst[i].kf_type == kftype) 5700daf62d9SStanislav Sedov break; 5710daf62d9SStanislav Sedov if (i == NKFTYPES) 5720daf62d9SStanislav Sedov return (PS_FST_TYPE_UNKNOWN); 5730daf62d9SStanislav Sedov return (kftypes2fst[i].fst_type); 5740daf62d9SStanislav Sedov } 5750daf62d9SStanislav Sedov 5760daf62d9SStanislav Sedov /* 5770daf62d9SStanislav Sedov * kinfo flags to filestat translation. 5780daf62d9SStanislav Sedov */ 5790daf62d9SStanislav Sedov static int 5800daf62d9SStanislav Sedov kinfo_fflags2fst(int kfflags) 5810daf62d9SStanislav Sedov { 5820daf62d9SStanislav Sedov static struct { 5830daf62d9SStanislav Sedov int kf_flag; 5840daf62d9SStanislav Sedov int fst_flag; 5850daf62d9SStanislav Sedov } kfflags2fst[] = { 5860daf62d9SStanislav Sedov { KF_FLAG_APPEND, PS_FST_FFLAG_APPEND }, 5870daf62d9SStanislav Sedov { KF_FLAG_ASYNC, PS_FST_FFLAG_ASYNC }, 5880daf62d9SStanislav Sedov { KF_FLAG_CREAT, PS_FST_FFLAG_CREAT }, 5890daf62d9SStanislav Sedov { KF_FLAG_DIRECT, PS_FST_FFLAG_DIRECT }, 5900daf62d9SStanislav Sedov { KF_FLAG_EXCL, PS_FST_FFLAG_EXCL }, 5910daf62d9SStanislav Sedov { KF_FLAG_EXEC, PS_FST_FFLAG_EXEC }, 5920daf62d9SStanislav Sedov { KF_FLAG_EXLOCK, PS_FST_FFLAG_EXLOCK }, 5930daf62d9SStanislav Sedov { KF_FLAG_FSYNC, PS_FST_FFLAG_SYNC }, 5940daf62d9SStanislav Sedov { KF_FLAG_HASLOCK, PS_FST_FFLAG_HASLOCK }, 5950daf62d9SStanislav Sedov { KF_FLAG_NOFOLLOW, PS_FST_FFLAG_NOFOLLOW }, 5960daf62d9SStanislav Sedov { KF_FLAG_NONBLOCK, PS_FST_FFLAG_NONBLOCK }, 5970daf62d9SStanislav Sedov { KF_FLAG_READ, PS_FST_FFLAG_READ }, 5980daf62d9SStanislav Sedov { KF_FLAG_SHLOCK, PS_FST_FFLAG_SHLOCK }, 5990daf62d9SStanislav Sedov { KF_FLAG_TRUNC, PS_FST_FFLAG_TRUNC }, 6000daf62d9SStanislav Sedov { KF_FLAG_WRITE, PS_FST_FFLAG_WRITE } 6010daf62d9SStanislav Sedov }; 6020daf62d9SStanislav Sedov #define NKFFLAGS (sizeof(kfflags2fst) / sizeof(*kfflags2fst)) 6030daf62d9SStanislav Sedov unsigned int i; 6040daf62d9SStanislav Sedov int flags; 6050daf62d9SStanislav Sedov 6060daf62d9SStanislav Sedov flags = 0; 6070daf62d9SStanislav Sedov for (i = 0; i < NKFFLAGS; i++) 6080daf62d9SStanislav Sedov if ((kfflags & kfflags2fst[i].kf_flag) != 0) 6090daf62d9SStanislav Sedov flags |= kfflags2fst[i].fst_flag; 6100daf62d9SStanislav Sedov return (flags); 6110daf62d9SStanislav Sedov } 6120daf62d9SStanislav Sedov 6130daf62d9SStanislav Sedov static int 6140daf62d9SStanislav Sedov kinfo_uflags2fst(int fd) 6150daf62d9SStanislav Sedov { 6160daf62d9SStanislav Sedov 6170daf62d9SStanislav Sedov switch (fd) { 6180daf62d9SStanislav Sedov case KF_FD_TYPE_CTTY: 6190daf62d9SStanislav Sedov return (PS_FST_UFLAG_CTTY); 6200daf62d9SStanislav Sedov case KF_FD_TYPE_CWD: 6210daf62d9SStanislav Sedov return (PS_FST_UFLAG_CDIR); 6220daf62d9SStanislav Sedov case KF_FD_TYPE_JAIL: 6230daf62d9SStanislav Sedov return (PS_FST_UFLAG_JAIL); 6240daf62d9SStanislav Sedov case KF_FD_TYPE_TEXT: 6250daf62d9SStanislav Sedov return (PS_FST_UFLAG_TEXT); 6260daf62d9SStanislav Sedov case KF_FD_TYPE_TRACE: 6270daf62d9SStanislav Sedov return (PS_FST_UFLAG_TRACE); 6280daf62d9SStanislav Sedov case KF_FD_TYPE_ROOT: 6290daf62d9SStanislav Sedov return (PS_FST_UFLAG_RDIR); 6300daf62d9SStanislav Sedov } 6310daf62d9SStanislav Sedov return (0); 6320daf62d9SStanislav Sedov } 6330daf62d9SStanislav Sedov 6340daf62d9SStanislav Sedov static struct filestat_list * 6350daf62d9SStanislav Sedov procstat_getfiles_sysctl(struct procstat *procstat, struct kinfo_proc *kp, int mmapped) 6360daf62d9SStanislav Sedov { 6370daf62d9SStanislav Sedov struct kinfo_file *kif, *files; 6380daf62d9SStanislav Sedov struct kinfo_vmentry *kve, *vmentries; 6390daf62d9SStanislav Sedov struct filestat_list *head; 6400daf62d9SStanislav Sedov struct filestat *entry; 6410daf62d9SStanislav Sedov char *path; 6420daf62d9SStanislav Sedov off_t offset; 6430daf62d9SStanislav Sedov int cnt, fd, fflags; 6440daf62d9SStanislav Sedov int i, type, uflags; 6450daf62d9SStanislav Sedov int refcount; 6460daf62d9SStanislav Sedov 6470daf62d9SStanislav Sedov assert(kp); 6480daf62d9SStanislav Sedov if (kp->ki_fd == NULL) 6490daf62d9SStanislav Sedov return (NULL); 6500daf62d9SStanislav Sedov 6510daf62d9SStanislav Sedov files = kinfo_getfile(kp->ki_pid, &cnt); 6520daf62d9SStanislav Sedov if (files == NULL && errno != EPERM) { 6530daf62d9SStanislav Sedov warn("kinfo_getfile()"); 6540daf62d9SStanislav Sedov return (NULL); 6550daf62d9SStanislav Sedov } 6560daf62d9SStanislav Sedov procstat->files = files; 6570daf62d9SStanislav Sedov 6580daf62d9SStanislav Sedov /* 6590daf62d9SStanislav Sedov * Allocate list head. 6600daf62d9SStanislav Sedov */ 6610daf62d9SStanislav Sedov head = malloc(sizeof(*head)); 6620daf62d9SStanislav Sedov if (head == NULL) 6630daf62d9SStanislav Sedov return (NULL); 6640daf62d9SStanislav Sedov STAILQ_INIT(head); 6650daf62d9SStanislav Sedov for (i = 0; i < cnt; i++) { 6660daf62d9SStanislav Sedov kif = &files[i]; 6670daf62d9SStanislav Sedov 6680daf62d9SStanislav Sedov type = kinfo_type2fst(kif->kf_type); 6690daf62d9SStanislav Sedov fd = kif->kf_fd >= 0 ? kif->kf_fd : -1; 6700daf62d9SStanislav Sedov fflags = kinfo_fflags2fst(kif->kf_flags); 6710daf62d9SStanislav Sedov uflags = kinfo_uflags2fst(kif->kf_fd); 6720daf62d9SStanislav Sedov refcount = kif->kf_ref_count; 6730daf62d9SStanislav Sedov offset = kif->kf_offset; 6740daf62d9SStanislav Sedov if (*kif->kf_path != '\0') 6750daf62d9SStanislav Sedov path = strdup(kif->kf_path); 6760daf62d9SStanislav Sedov else 6770daf62d9SStanislav Sedov path = NULL; 6780daf62d9SStanislav Sedov 6790daf62d9SStanislav Sedov /* 6800daf62d9SStanislav Sedov * Create filestat entry. 6810daf62d9SStanislav Sedov */ 6820daf62d9SStanislav Sedov entry = filestat_new_entry(kif, type, fd, fflags, uflags, 6830daf62d9SStanislav Sedov refcount, offset, path); 6840daf62d9SStanislav Sedov if (entry != NULL) 6850daf62d9SStanislav Sedov STAILQ_INSERT_TAIL(head, entry, next); 6860daf62d9SStanislav Sedov } 6870daf62d9SStanislav Sedov if (mmapped != 0) { 6880daf62d9SStanislav Sedov vmentries = kinfo_getvmmap(kp->ki_pid, &cnt); 6890daf62d9SStanislav Sedov procstat->vmentries = vmentries; 6900daf62d9SStanislav Sedov if (vmentries == NULL || cnt == 0) 6910daf62d9SStanislav Sedov goto fail; 6920daf62d9SStanislav Sedov for (i = 0; i < cnt; i++) { 6930daf62d9SStanislav Sedov kve = &vmentries[i]; 6940daf62d9SStanislav Sedov if (kve->kve_type != KVME_TYPE_VNODE) 6950daf62d9SStanislav Sedov continue; 6960daf62d9SStanislav Sedov fflags = 0; 6970daf62d9SStanislav Sedov if (kve->kve_protection & KVME_PROT_READ) 6980daf62d9SStanislav Sedov fflags = PS_FST_FFLAG_READ; 6990daf62d9SStanislav Sedov if (kve->kve_protection & KVME_PROT_WRITE) 7000daf62d9SStanislav Sedov fflags |= PS_FST_FFLAG_WRITE; 7010daf62d9SStanislav Sedov offset = kve->kve_offset; 7020daf62d9SStanislav Sedov refcount = kve->kve_ref_count; 7030daf62d9SStanislav Sedov if (*kve->kve_path != '\0') 7040daf62d9SStanislav Sedov path = strdup(kve->kve_path); 7050daf62d9SStanislav Sedov else 7060daf62d9SStanislav Sedov path = NULL; 7070daf62d9SStanislav Sedov entry = filestat_new_entry(kve, PS_FST_TYPE_VNODE, -1, 7080daf62d9SStanislav Sedov fflags, PS_FST_UFLAG_MMAP, refcount, offset, path); 7090daf62d9SStanislav Sedov if (entry != NULL) 7100daf62d9SStanislav Sedov STAILQ_INSERT_TAIL(head, entry, next); 7110daf62d9SStanislav Sedov } 7120daf62d9SStanislav Sedov } 7130daf62d9SStanislav Sedov fail: 7140daf62d9SStanislav Sedov return (head); 7150daf62d9SStanislav Sedov } 7160daf62d9SStanislav Sedov 7170daf62d9SStanislav Sedov int 7180daf62d9SStanislav Sedov procstat_get_pipe_info(struct procstat *procstat, struct filestat *fst, 7190daf62d9SStanislav Sedov struct pipestat *ps, char *errbuf) 7200daf62d9SStanislav Sedov { 7210daf62d9SStanislav Sedov 7220daf62d9SStanislav Sedov assert(ps); 7230daf62d9SStanislav Sedov if (procstat->type == PROCSTAT_KVM) { 7240daf62d9SStanislav Sedov return (procstat_get_pipe_info_kvm(procstat->kd, fst, ps, 7250daf62d9SStanislav Sedov errbuf)); 7260daf62d9SStanislav Sedov } else if (procstat->type == PROCSTAT_SYSCTL) { 7270daf62d9SStanislav Sedov return (procstat_get_pipe_info_sysctl(fst, ps, errbuf)); 7280daf62d9SStanislav Sedov } else { 729*80905c35SJilles Tjoelker warnx("unknown access method: %d", procstat->type); 7300daf62d9SStanislav Sedov snprintf(errbuf, _POSIX2_LINE_MAX, "error"); 7310daf62d9SStanislav Sedov return (1); 7320daf62d9SStanislav Sedov } 7330daf62d9SStanislav Sedov } 7340daf62d9SStanislav Sedov 7350daf62d9SStanislav Sedov static int 7360daf62d9SStanislav Sedov procstat_get_pipe_info_kvm(kvm_t *kd, struct filestat *fst, 7370daf62d9SStanislav Sedov struct pipestat *ps, char *errbuf) 7380daf62d9SStanislav Sedov { 7390daf62d9SStanislav Sedov struct pipe pi; 7400daf62d9SStanislav Sedov void *pipep; 7410daf62d9SStanislav Sedov 7420daf62d9SStanislav Sedov assert(kd); 7430daf62d9SStanislav Sedov assert(ps); 7440daf62d9SStanislav Sedov assert(fst); 7450daf62d9SStanislav Sedov bzero(ps, sizeof(*ps)); 7460daf62d9SStanislav Sedov pipep = fst->fs_typedep; 7470daf62d9SStanislav Sedov if (pipep == NULL) 7480daf62d9SStanislav Sedov goto fail; 7490daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)pipep, &pi, sizeof(struct pipe))) { 7500daf62d9SStanislav Sedov warnx("can't read pipe at %p", (void *)pipep); 7510daf62d9SStanislav Sedov goto fail; 7520daf62d9SStanislav Sedov } 7530daf62d9SStanislav Sedov ps->addr = (uintptr_t)pipep; 7540daf62d9SStanislav Sedov ps->peer = (uintptr_t)pi.pipe_peer; 7550daf62d9SStanislav Sedov ps->buffer_cnt = pi.pipe_buffer.cnt; 7560daf62d9SStanislav Sedov return (0); 7570daf62d9SStanislav Sedov 7580daf62d9SStanislav Sedov fail: 7590daf62d9SStanislav Sedov snprintf(errbuf, _POSIX2_LINE_MAX, "error"); 7600daf62d9SStanislav Sedov return (1); 7610daf62d9SStanislav Sedov } 7620daf62d9SStanislav Sedov 7630daf62d9SStanislav Sedov static int 7640daf62d9SStanislav Sedov procstat_get_pipe_info_sysctl(struct filestat *fst, struct pipestat *ps, 7650daf62d9SStanislav Sedov char *errbuf __unused) 7660daf62d9SStanislav Sedov { 7670daf62d9SStanislav Sedov struct kinfo_file *kif; 7680daf62d9SStanislav Sedov 7690daf62d9SStanislav Sedov assert(ps); 7700daf62d9SStanislav Sedov assert(fst); 7710daf62d9SStanislav Sedov bzero(ps, sizeof(*ps)); 7720daf62d9SStanislav Sedov kif = fst->fs_typedep; 7730daf62d9SStanislav Sedov if (kif == NULL) 7740daf62d9SStanislav Sedov return (1); 7750daf62d9SStanislav Sedov ps->addr = kif->kf_un.kf_pipe.kf_pipe_addr; 7760daf62d9SStanislav Sedov ps->peer = kif->kf_un.kf_pipe.kf_pipe_peer; 7770daf62d9SStanislav Sedov ps->buffer_cnt = kif->kf_un.kf_pipe.kf_pipe_buffer_cnt; 7780daf62d9SStanislav Sedov return (0); 7790daf62d9SStanislav Sedov } 7800daf62d9SStanislav Sedov 7810daf62d9SStanislav Sedov int 7820daf62d9SStanislav Sedov procstat_get_pts_info(struct procstat *procstat, struct filestat *fst, 7830daf62d9SStanislav Sedov struct ptsstat *pts, char *errbuf) 7840daf62d9SStanislav Sedov { 7850daf62d9SStanislav Sedov 7860daf62d9SStanislav Sedov assert(pts); 7870daf62d9SStanislav Sedov if (procstat->type == PROCSTAT_KVM) { 7880daf62d9SStanislav Sedov return (procstat_get_pts_info_kvm(procstat->kd, fst, pts, 7890daf62d9SStanislav Sedov errbuf)); 7900daf62d9SStanislav Sedov } else if (procstat->type == PROCSTAT_SYSCTL) { 7910daf62d9SStanislav Sedov return (procstat_get_pts_info_sysctl(fst, pts, errbuf)); 7920daf62d9SStanislav Sedov } else { 793*80905c35SJilles Tjoelker warnx("unknown access method: %d", procstat->type); 7940daf62d9SStanislav Sedov snprintf(errbuf, _POSIX2_LINE_MAX, "error"); 7950daf62d9SStanislav Sedov return (1); 7960daf62d9SStanislav Sedov } 7970daf62d9SStanislav Sedov } 7980daf62d9SStanislav Sedov 7990daf62d9SStanislav Sedov static int 8000daf62d9SStanislav Sedov procstat_get_pts_info_kvm(kvm_t *kd, struct filestat *fst, 8010daf62d9SStanislav Sedov struct ptsstat *pts, char *errbuf) 8020daf62d9SStanislav Sedov { 8030daf62d9SStanislav Sedov struct tty tty; 8040daf62d9SStanislav Sedov void *ttyp; 8050daf62d9SStanislav Sedov 8060daf62d9SStanislav Sedov assert(kd); 8070daf62d9SStanislav Sedov assert(pts); 8080daf62d9SStanislav Sedov assert(fst); 8090daf62d9SStanislav Sedov bzero(pts, sizeof(*pts)); 8100daf62d9SStanislav Sedov ttyp = fst->fs_typedep; 8110daf62d9SStanislav Sedov if (ttyp == NULL) 8120daf62d9SStanislav Sedov goto fail; 8130daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)ttyp, &tty, sizeof(struct tty))) { 8140daf62d9SStanislav Sedov warnx("can't read tty at %p", (void *)ttyp); 8150daf62d9SStanislav Sedov goto fail; 8160daf62d9SStanislav Sedov } 8170daf62d9SStanislav Sedov pts->dev = dev2udev(kd, tty.t_dev); 8180daf62d9SStanislav Sedov (void)kdevtoname(kd, tty.t_dev, pts->devname); 8190daf62d9SStanislav Sedov return (0); 8200daf62d9SStanislav Sedov 8210daf62d9SStanislav Sedov fail: 8220daf62d9SStanislav Sedov snprintf(errbuf, _POSIX2_LINE_MAX, "error"); 8230daf62d9SStanislav Sedov return (1); 8240daf62d9SStanislav Sedov } 8250daf62d9SStanislav Sedov 8260daf62d9SStanislav Sedov static int 8270daf62d9SStanislav Sedov procstat_get_pts_info_sysctl(struct filestat *fst, struct ptsstat *pts, 8280daf62d9SStanislav Sedov char *errbuf __unused) 8290daf62d9SStanislav Sedov { 8300daf62d9SStanislav Sedov struct kinfo_file *kif; 8310daf62d9SStanislav Sedov 8320daf62d9SStanislav Sedov assert(pts); 8330daf62d9SStanislav Sedov assert(fst); 8340daf62d9SStanislav Sedov bzero(pts, sizeof(*pts)); 8350daf62d9SStanislav Sedov kif = fst->fs_typedep; 8360daf62d9SStanislav Sedov if (kif == NULL) 8370daf62d9SStanislav Sedov return (0); 8380daf62d9SStanislav Sedov pts->dev = kif->kf_un.kf_pts.kf_pts_dev; 8390daf62d9SStanislav Sedov strlcpy(pts->devname, kif->kf_path, sizeof(pts->devname)); 8400daf62d9SStanislav Sedov return (0); 8410daf62d9SStanislav Sedov } 8420daf62d9SStanislav Sedov 8430daf62d9SStanislav Sedov int 8440daf62d9SStanislav Sedov procstat_get_vnode_info(struct procstat *procstat, struct filestat *fst, 8450daf62d9SStanislav Sedov struct vnstat *vn, char *errbuf) 8460daf62d9SStanislav Sedov { 8470daf62d9SStanislav Sedov 8480daf62d9SStanislav Sedov assert(vn); 8490daf62d9SStanislav Sedov if (procstat->type == PROCSTAT_KVM) { 8500daf62d9SStanislav Sedov return (procstat_get_vnode_info_kvm(procstat->kd, fst, vn, 8510daf62d9SStanislav Sedov errbuf)); 8520daf62d9SStanislav Sedov } else if (procstat->type == PROCSTAT_SYSCTL) { 8530daf62d9SStanislav Sedov return (procstat_get_vnode_info_sysctl(fst, vn, errbuf)); 8540daf62d9SStanislav Sedov } else { 855*80905c35SJilles Tjoelker warnx("unknown access method: %d", procstat->type); 8560daf62d9SStanislav Sedov snprintf(errbuf, _POSIX2_LINE_MAX, "error"); 8570daf62d9SStanislav Sedov return (1); 8580daf62d9SStanislav Sedov } 8590daf62d9SStanislav Sedov } 8600daf62d9SStanislav Sedov 8610daf62d9SStanislav Sedov static int 8620daf62d9SStanislav Sedov procstat_get_vnode_info_kvm(kvm_t *kd, struct filestat *fst, 8630daf62d9SStanislav Sedov struct vnstat *vn, char *errbuf) 8640daf62d9SStanislav Sedov { 8650daf62d9SStanislav Sedov /* Filesystem specific handlers. */ 8660daf62d9SStanislav Sedov #define FSTYPE(fst) {#fst, fst##_filestat} 8670daf62d9SStanislav Sedov struct { 8680daf62d9SStanislav Sedov const char *tag; 8690daf62d9SStanislav Sedov int (*handler)(kvm_t *kd, struct vnode *vp, 8700daf62d9SStanislav Sedov struct vnstat *vn); 8710daf62d9SStanislav Sedov } fstypes[] = { 8720daf62d9SStanislav Sedov FSTYPE(devfs), 8730daf62d9SStanislav Sedov FSTYPE(isofs), 8740daf62d9SStanislav Sedov FSTYPE(msdosfs), 8750daf62d9SStanislav Sedov FSTYPE(nfs), 8760daf62d9SStanislav Sedov FSTYPE(ntfs), 87759ea47d0SStanislav Sedov #ifdef LIBPROCSTAT_NWFS 8780daf62d9SStanislav Sedov FSTYPE(nwfs), 87959ea47d0SStanislav Sedov #endif 8800daf62d9SStanislav Sedov FSTYPE(smbfs), 8810daf62d9SStanislav Sedov FSTYPE(udf), 8820daf62d9SStanislav Sedov FSTYPE(ufs), 88359ea47d0SStanislav Sedov #ifdef LIBPROCSTAT_ZFS 8840daf62d9SStanislav Sedov FSTYPE(zfs), 8850daf62d9SStanislav Sedov #endif 8860daf62d9SStanislav Sedov }; 8870daf62d9SStanislav Sedov #define NTYPES (sizeof(fstypes) / sizeof(*fstypes)) 8880daf62d9SStanislav Sedov struct vnode vnode; 8890daf62d9SStanislav Sedov char tagstr[12]; 8900daf62d9SStanislav Sedov void *vp; 8910daf62d9SStanislav Sedov int error, found; 8920daf62d9SStanislav Sedov unsigned int i; 8930daf62d9SStanislav Sedov 8940daf62d9SStanislav Sedov assert(kd); 8950daf62d9SStanislav Sedov assert(vn); 8960daf62d9SStanislav Sedov assert(fst); 8970daf62d9SStanislav Sedov vp = fst->fs_typedep; 8980daf62d9SStanislav Sedov if (vp == NULL) 8990daf62d9SStanislav Sedov goto fail; 9000daf62d9SStanislav Sedov error = kvm_read_all(kd, (unsigned long)vp, &vnode, sizeof(vnode)); 9010daf62d9SStanislav Sedov if (error == 0) { 9020daf62d9SStanislav Sedov warnx("can't read vnode at %p", (void *)vp); 9030daf62d9SStanislav Sedov goto fail; 9040daf62d9SStanislav Sedov } 9050daf62d9SStanislav Sedov bzero(vn, sizeof(*vn)); 9060daf62d9SStanislav Sedov vn->vn_type = vntype2psfsttype(vnode.v_type); 9070daf62d9SStanislav Sedov if (vnode.v_type == VNON || vnode.v_type == VBAD) 9080daf62d9SStanislav Sedov return (0); 9090daf62d9SStanislav Sedov error = kvm_read_all(kd, (unsigned long)vnode.v_tag, tagstr, 9100daf62d9SStanislav Sedov sizeof(tagstr)); 9110daf62d9SStanislav Sedov if (error == 0) { 9120daf62d9SStanislav Sedov warnx("can't read v_tag at %p", (void *)vp); 9130daf62d9SStanislav Sedov goto fail; 9140daf62d9SStanislav Sedov } 9150daf62d9SStanislav Sedov tagstr[sizeof(tagstr) - 1] = '\0'; 9160daf62d9SStanislav Sedov 9170daf62d9SStanislav Sedov /* 9180daf62d9SStanislav Sedov * Find appropriate handler. 9190daf62d9SStanislav Sedov */ 9200daf62d9SStanislav Sedov for (i = 0, found = 0; i < NTYPES; i++) 9210daf62d9SStanislav Sedov if (!strcmp(fstypes[i].tag, tagstr)) { 9220daf62d9SStanislav Sedov if (fstypes[i].handler(kd, &vnode, vn) != 0) { 9230daf62d9SStanislav Sedov goto fail; 9240daf62d9SStanislav Sedov } 9250daf62d9SStanislav Sedov break; 9260daf62d9SStanislav Sedov } 9270daf62d9SStanislav Sedov if (i == NTYPES) { 9280daf62d9SStanislav Sedov snprintf(errbuf, _POSIX2_LINE_MAX, "?(%s)", tagstr); 9290daf62d9SStanislav Sedov return (1); 9300daf62d9SStanislav Sedov } 9310daf62d9SStanislav Sedov vn->vn_mntdir = getmnton(kd, vnode.v_mount); 9320daf62d9SStanislav Sedov if ((vnode.v_type == VBLK || vnode.v_type == VCHR) && 9330daf62d9SStanislav Sedov vnode.v_rdev != NULL){ 9340daf62d9SStanislav Sedov vn->vn_dev = dev2udev(kd, vnode.v_rdev); 9350daf62d9SStanislav Sedov (void)kdevtoname(kd, vnode.v_rdev, vn->vn_devname); 9360daf62d9SStanislav Sedov } else { 9370daf62d9SStanislav Sedov vn->vn_dev = -1; 9380daf62d9SStanislav Sedov } 9390daf62d9SStanislav Sedov return (0); 9400daf62d9SStanislav Sedov 9410daf62d9SStanislav Sedov fail: 9420daf62d9SStanislav Sedov snprintf(errbuf, _POSIX2_LINE_MAX, "error"); 9430daf62d9SStanislav Sedov return (1); 9440daf62d9SStanislav Sedov } 9450daf62d9SStanislav Sedov 9460daf62d9SStanislav Sedov /* 9470daf62d9SStanislav Sedov * kinfo vnode type to filestat translation. 9480daf62d9SStanislav Sedov */ 9490daf62d9SStanislav Sedov static int 9500daf62d9SStanislav Sedov kinfo_vtype2fst(int kfvtype) 9510daf62d9SStanislav Sedov { 9520daf62d9SStanislav Sedov static struct { 9530daf62d9SStanislav Sedov int kf_vtype; 9540daf62d9SStanislav Sedov int fst_vtype; 9550daf62d9SStanislav Sedov } kfvtypes2fst[] = { 9560daf62d9SStanislav Sedov { KF_VTYPE_VBAD, PS_FST_VTYPE_VBAD }, 9570daf62d9SStanislav Sedov { KF_VTYPE_VBLK, PS_FST_VTYPE_VBLK }, 9580daf62d9SStanislav Sedov { KF_VTYPE_VCHR, PS_FST_VTYPE_VCHR }, 9590daf62d9SStanislav Sedov { KF_VTYPE_VDIR, PS_FST_VTYPE_VDIR }, 9600daf62d9SStanislav Sedov { KF_VTYPE_VFIFO, PS_FST_VTYPE_VFIFO }, 9610daf62d9SStanislav Sedov { KF_VTYPE_VLNK, PS_FST_VTYPE_VLNK }, 9620daf62d9SStanislav Sedov { KF_VTYPE_VNON, PS_FST_VTYPE_VNON }, 9630daf62d9SStanislav Sedov { KF_VTYPE_VREG, PS_FST_VTYPE_VREG }, 9640daf62d9SStanislav Sedov { KF_VTYPE_VSOCK, PS_FST_VTYPE_VSOCK } 9650daf62d9SStanislav Sedov }; 9660daf62d9SStanislav Sedov #define NKFVTYPES (sizeof(kfvtypes2fst) / sizeof(*kfvtypes2fst)) 9670daf62d9SStanislav Sedov unsigned int i; 9680daf62d9SStanislav Sedov 9690daf62d9SStanislav Sedov for (i = 0; i < NKFVTYPES; i++) 9700daf62d9SStanislav Sedov if (kfvtypes2fst[i].kf_vtype == kfvtype) 9710daf62d9SStanislav Sedov break; 9720daf62d9SStanislav Sedov if (i == NKFVTYPES) 9730daf62d9SStanislav Sedov return (PS_FST_VTYPE_UNKNOWN); 9740daf62d9SStanislav Sedov return (kfvtypes2fst[i].fst_vtype); 9750daf62d9SStanislav Sedov } 9760daf62d9SStanislav Sedov 9770daf62d9SStanislav Sedov static int 9780daf62d9SStanislav Sedov procstat_get_vnode_info_sysctl(struct filestat *fst, struct vnstat *vn, 9790daf62d9SStanislav Sedov char *errbuf) 9800daf62d9SStanislav Sedov { 9810daf62d9SStanislav Sedov struct statfs stbuf; 9820daf62d9SStanislav Sedov struct kinfo_file *kif; 9830daf62d9SStanislav Sedov struct kinfo_vmentry *kve; 9840daf62d9SStanislav Sedov uint64_t fileid; 9850daf62d9SStanislav Sedov uint64_t size; 9860daf62d9SStanislav Sedov char *name, *path; 9870daf62d9SStanislav Sedov uint32_t fsid; 9880daf62d9SStanislav Sedov uint16_t mode; 9890daf62d9SStanislav Sedov uint32_t rdev; 9900daf62d9SStanislav Sedov int vntype; 9910daf62d9SStanislav Sedov int status; 9920daf62d9SStanislav Sedov 9930daf62d9SStanislav Sedov assert(fst); 9940daf62d9SStanislav Sedov assert(vn); 9950daf62d9SStanislav Sedov bzero(vn, sizeof(*vn)); 9960daf62d9SStanislav Sedov if (fst->fs_typedep == NULL) 9970daf62d9SStanislav Sedov return (1); 9980daf62d9SStanislav Sedov if (fst->fs_uflags & PS_FST_UFLAG_MMAP) { 9990daf62d9SStanislav Sedov kve = fst->fs_typedep; 10000daf62d9SStanislav Sedov fileid = kve->kve_vn_fileid; 10010daf62d9SStanislav Sedov fsid = kve->kve_vn_fsid; 10020daf62d9SStanislav Sedov mode = kve->kve_vn_mode; 10030daf62d9SStanislav Sedov path = kve->kve_path; 10040daf62d9SStanislav Sedov rdev = kve->kve_vn_rdev; 10050daf62d9SStanislav Sedov size = kve->kve_vn_size; 10060daf62d9SStanislav Sedov vntype = kinfo_vtype2fst(kve->kve_vn_type); 10070daf62d9SStanislav Sedov status = kve->kve_status; 10080daf62d9SStanislav Sedov } else { 10090daf62d9SStanislav Sedov kif = fst->fs_typedep; 10100daf62d9SStanislav Sedov fileid = kif->kf_un.kf_file.kf_file_fileid; 10110daf62d9SStanislav Sedov fsid = kif->kf_un.kf_file.kf_file_fsid; 10120daf62d9SStanislav Sedov mode = kif->kf_un.kf_file.kf_file_mode; 10130daf62d9SStanislav Sedov path = kif->kf_path; 10140daf62d9SStanislav Sedov rdev = kif->kf_un.kf_file.kf_file_rdev; 10150daf62d9SStanislav Sedov size = kif->kf_un.kf_file.kf_file_size; 10160daf62d9SStanislav Sedov vntype = kinfo_vtype2fst(kif->kf_vnode_type); 10170daf62d9SStanislav Sedov status = kif->kf_status; 10180daf62d9SStanislav Sedov } 10190daf62d9SStanislav Sedov vn->vn_type = vntype; 10200daf62d9SStanislav Sedov if (vntype == PS_FST_VTYPE_VNON || vntype == PS_FST_VTYPE_VBAD) 10210daf62d9SStanislav Sedov return (0); 10220daf62d9SStanislav Sedov if ((status & KF_ATTR_VALID) == 0) { 10230daf62d9SStanislav Sedov snprintf(errbuf, _POSIX2_LINE_MAX, "? (no info available)"); 10240daf62d9SStanislav Sedov return (1); 10250daf62d9SStanislav Sedov } 10260daf62d9SStanislav Sedov if (path && *path) { 10270daf62d9SStanislav Sedov statfs(path, &stbuf); 10280daf62d9SStanislav Sedov vn->vn_mntdir = strdup(stbuf.f_mntonname); 10290daf62d9SStanislav Sedov } else 10300daf62d9SStanislav Sedov vn->vn_mntdir = strdup("-"); 10310daf62d9SStanislav Sedov vn->vn_dev = rdev; 10320daf62d9SStanislav Sedov if (vntype == PS_FST_VTYPE_VBLK) { 10330daf62d9SStanislav Sedov name = devname(rdev, S_IFBLK); 10340daf62d9SStanislav Sedov if (name != NULL) 10350daf62d9SStanislav Sedov strlcpy(vn->vn_devname, name, 10360daf62d9SStanislav Sedov sizeof(vn->vn_devname)); 10370daf62d9SStanislav Sedov } else if (vntype == PS_FST_VTYPE_VCHR) { 10380daf62d9SStanislav Sedov name = devname(vn->vn_dev, S_IFCHR); 10390daf62d9SStanislav Sedov if (name != NULL) 10400daf62d9SStanislav Sedov strlcpy(vn->vn_devname, name, 10410daf62d9SStanislav Sedov sizeof(vn->vn_devname)); 10420daf62d9SStanislav Sedov } 10430daf62d9SStanislav Sedov vn->vn_fsid = fsid; 10440daf62d9SStanislav Sedov vn->vn_fileid = fileid; 10450daf62d9SStanislav Sedov vn->vn_size = size; 10460daf62d9SStanislav Sedov vn->vn_mode = mode; 10470daf62d9SStanislav Sedov return (0); 10480daf62d9SStanislav Sedov } 10490daf62d9SStanislav Sedov 10500daf62d9SStanislav Sedov int 10510daf62d9SStanislav Sedov procstat_get_socket_info(struct procstat *procstat, struct filestat *fst, 10520daf62d9SStanislav Sedov struct sockstat *sock, char *errbuf) 10530daf62d9SStanislav Sedov { 10540daf62d9SStanislav Sedov 10550daf62d9SStanislav Sedov assert(sock); 10560daf62d9SStanislav Sedov if (procstat->type == PROCSTAT_KVM) { 10570daf62d9SStanislav Sedov return (procstat_get_socket_info_kvm(procstat->kd, fst, sock, 10580daf62d9SStanislav Sedov errbuf)); 10590daf62d9SStanislav Sedov } else if (procstat->type == PROCSTAT_SYSCTL) { 10600daf62d9SStanislav Sedov return (procstat_get_socket_info_sysctl(fst, sock, errbuf)); 10610daf62d9SStanislav Sedov } else { 1062*80905c35SJilles Tjoelker warnx("unknown access method: %d", procstat->type); 10630daf62d9SStanislav Sedov snprintf(errbuf, _POSIX2_LINE_MAX, "error"); 10640daf62d9SStanislav Sedov return (1); 10650daf62d9SStanislav Sedov } 10660daf62d9SStanislav Sedov } 10670daf62d9SStanislav Sedov 10680daf62d9SStanislav Sedov static int 10690daf62d9SStanislav Sedov procstat_get_socket_info_kvm(kvm_t *kd, struct filestat *fst, 10700daf62d9SStanislav Sedov struct sockstat *sock, char *errbuf) 10710daf62d9SStanislav Sedov { 10720daf62d9SStanislav Sedov struct domain dom; 10730daf62d9SStanislav Sedov struct inpcb inpcb; 10740daf62d9SStanislav Sedov struct protosw proto; 10750daf62d9SStanislav Sedov struct socket s; 10760daf62d9SStanislav Sedov struct unpcb unpcb; 10770daf62d9SStanislav Sedov ssize_t len; 10780daf62d9SStanislav Sedov void *so; 10790daf62d9SStanislav Sedov 10800daf62d9SStanislav Sedov assert(kd); 10810daf62d9SStanislav Sedov assert(sock); 10820daf62d9SStanislav Sedov assert(fst); 10830daf62d9SStanislav Sedov bzero(sock, sizeof(*sock)); 10840daf62d9SStanislav Sedov so = fst->fs_typedep; 10850daf62d9SStanislav Sedov if (so == NULL) 10860daf62d9SStanislav Sedov goto fail; 10870daf62d9SStanislav Sedov sock->so_addr = (uintptr_t)so; 10880daf62d9SStanislav Sedov /* fill in socket */ 10890daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)so, &s, 10900daf62d9SStanislav Sedov sizeof(struct socket))) { 10910daf62d9SStanislav Sedov warnx("can't read sock at %p", (void *)so); 10920daf62d9SStanislav Sedov goto fail; 10930daf62d9SStanislav Sedov } 10940daf62d9SStanislav Sedov /* fill in protosw entry */ 10950daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)s.so_proto, &proto, 10960daf62d9SStanislav Sedov sizeof(struct protosw))) { 10970daf62d9SStanislav Sedov warnx("can't read protosw at %p", (void *)s.so_proto); 10980daf62d9SStanislav Sedov goto fail; 10990daf62d9SStanislav Sedov } 11000daf62d9SStanislav Sedov /* fill in domain */ 11010daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)proto.pr_domain, &dom, 11020daf62d9SStanislav Sedov sizeof(struct domain))) { 11030daf62d9SStanislav Sedov warnx("can't read domain at %p", 11040daf62d9SStanislav Sedov (void *)proto.pr_domain); 11050daf62d9SStanislav Sedov goto fail; 11060daf62d9SStanislav Sedov } 11070daf62d9SStanislav Sedov if ((len = kvm_read(kd, (unsigned long)dom.dom_name, sock->dname, 11080daf62d9SStanislav Sedov sizeof(sock->dname) - 1)) < 0) { 11090daf62d9SStanislav Sedov warnx("can't read domain name at %p", (void *)dom.dom_name); 11100daf62d9SStanislav Sedov sock->dname[0] = '\0'; 11110daf62d9SStanislav Sedov } 11120daf62d9SStanislav Sedov else 11130daf62d9SStanislav Sedov sock->dname[len] = '\0'; 11140daf62d9SStanislav Sedov 11150daf62d9SStanislav Sedov /* 11160daf62d9SStanislav Sedov * Fill in known data. 11170daf62d9SStanislav Sedov */ 11180daf62d9SStanislav Sedov sock->type = s.so_type; 11190daf62d9SStanislav Sedov sock->proto = proto.pr_protocol; 11200daf62d9SStanislav Sedov sock->dom_family = dom.dom_family; 11210daf62d9SStanislav Sedov sock->so_pcb = (uintptr_t)s.so_pcb; 11220daf62d9SStanislav Sedov 11230daf62d9SStanislav Sedov /* 11240daf62d9SStanislav Sedov * Protocol specific data. 11250daf62d9SStanislav Sedov */ 11260daf62d9SStanislav Sedov switch(dom.dom_family) { 11270daf62d9SStanislav Sedov case AF_INET: 11280daf62d9SStanislav Sedov case AF_INET6: 11290daf62d9SStanislav Sedov if (proto.pr_protocol == IPPROTO_TCP) { 11300daf62d9SStanislav Sedov if (s.so_pcb) { 11310daf62d9SStanislav Sedov if (kvm_read(kd, (u_long)s.so_pcb, 11320daf62d9SStanislav Sedov (char *)&inpcb, sizeof(struct inpcb)) 11330daf62d9SStanislav Sedov != sizeof(struct inpcb)) { 11340daf62d9SStanislav Sedov warnx("can't read inpcb at %p", 11350daf62d9SStanislav Sedov (void *)s.so_pcb); 11360daf62d9SStanislav Sedov } else 11370daf62d9SStanislav Sedov sock->inp_ppcb = 11380daf62d9SStanislav Sedov (uintptr_t)inpcb.inp_ppcb; 11390daf62d9SStanislav Sedov } 11400daf62d9SStanislav Sedov } 11410daf62d9SStanislav Sedov break; 11420daf62d9SStanislav Sedov case AF_UNIX: 11430daf62d9SStanislav Sedov if (s.so_pcb) { 11440daf62d9SStanislav Sedov if (kvm_read(kd, (u_long)s.so_pcb, (char *)&unpcb, 11450daf62d9SStanislav Sedov sizeof(struct unpcb)) != sizeof(struct unpcb)){ 11460daf62d9SStanislav Sedov warnx("can't read unpcb at %p", 11470daf62d9SStanislav Sedov (void *)s.so_pcb); 11480daf62d9SStanislav Sedov } else if (unpcb.unp_conn) { 11490daf62d9SStanislav Sedov sock->so_rcv_sb_state = s.so_rcv.sb_state; 11500daf62d9SStanislav Sedov sock->so_snd_sb_state = s.so_snd.sb_state; 11510daf62d9SStanislav Sedov sock->unp_conn = (uintptr_t)unpcb.unp_conn; 11520daf62d9SStanislav Sedov } 11530daf62d9SStanislav Sedov } 11540daf62d9SStanislav Sedov break; 11550daf62d9SStanislav Sedov default: 11560daf62d9SStanislav Sedov break; 11570daf62d9SStanislav Sedov } 11580daf62d9SStanislav Sedov return (0); 11590daf62d9SStanislav Sedov 11600daf62d9SStanislav Sedov fail: 11610daf62d9SStanislav Sedov snprintf(errbuf, _POSIX2_LINE_MAX, "error"); 11620daf62d9SStanislav Sedov return (1); 11630daf62d9SStanislav Sedov } 11640daf62d9SStanislav Sedov 11650daf62d9SStanislav Sedov static int 11660daf62d9SStanislav Sedov procstat_get_socket_info_sysctl(struct filestat *fst, struct sockstat *sock, 11670daf62d9SStanislav Sedov char *errbuf __unused) 11680daf62d9SStanislav Sedov { 11690daf62d9SStanislav Sedov struct kinfo_file *kif; 11700daf62d9SStanislav Sedov 11710daf62d9SStanislav Sedov assert(sock); 11720daf62d9SStanislav Sedov assert(fst); 11730daf62d9SStanislav Sedov bzero(sock, sizeof(*sock)); 11740daf62d9SStanislav Sedov kif = fst->fs_typedep; 11750daf62d9SStanislav Sedov if (kif == NULL) 11760daf62d9SStanislav Sedov return (0); 11770daf62d9SStanislav Sedov 11780daf62d9SStanislav Sedov /* 11790daf62d9SStanislav Sedov * Fill in known data. 11800daf62d9SStanislav Sedov */ 11810daf62d9SStanislav Sedov sock->type = kif->kf_sock_type; 11820daf62d9SStanislav Sedov sock->proto = kif->kf_sock_protocol; 11830daf62d9SStanislav Sedov sock->dom_family = kif->kf_sock_domain; 11840daf62d9SStanislav Sedov sock->so_pcb = kif->kf_un.kf_sock.kf_sock_pcb; 11850daf62d9SStanislav Sedov strlcpy(sock->dname, kif->kf_path, sizeof(sock->dname)); 11860daf62d9SStanislav Sedov bcopy(&kif->kf_sa_local, &sock->sa_local, kif->kf_sa_local.ss_len); 11870daf62d9SStanislav Sedov bcopy(&kif->kf_sa_peer, &sock->sa_peer, kif->kf_sa_peer.ss_len); 11880daf62d9SStanislav Sedov 11890daf62d9SStanislav Sedov /* 11900daf62d9SStanislav Sedov * Protocol specific data. 11910daf62d9SStanislav Sedov */ 11920daf62d9SStanislav Sedov switch(sock->dom_family) { 11930daf62d9SStanislav Sedov case AF_INET: 11940daf62d9SStanislav Sedov case AF_INET6: 11950daf62d9SStanislav Sedov if (sock->proto == IPPROTO_TCP) 11960daf62d9SStanislav Sedov sock->inp_ppcb = kif->kf_un.kf_sock.kf_sock_inpcb; 11970daf62d9SStanislav Sedov break; 11980daf62d9SStanislav Sedov case AF_UNIX: 11990daf62d9SStanislav Sedov if (kif->kf_un.kf_sock.kf_sock_unpconn != 0) { 12000daf62d9SStanislav Sedov sock->so_rcv_sb_state = 12010daf62d9SStanislav Sedov kif->kf_un.kf_sock.kf_sock_rcv_sb_state; 12020daf62d9SStanislav Sedov sock->so_snd_sb_state = 12030daf62d9SStanislav Sedov kif->kf_un.kf_sock.kf_sock_snd_sb_state; 12040daf62d9SStanislav Sedov sock->unp_conn = 12050daf62d9SStanislav Sedov kif->kf_un.kf_sock.kf_sock_unpconn; 12060daf62d9SStanislav Sedov } 12070daf62d9SStanislav Sedov break; 12080daf62d9SStanislav Sedov default: 12090daf62d9SStanislav Sedov break; 12100daf62d9SStanislav Sedov } 12110daf62d9SStanislav Sedov return (0); 12120daf62d9SStanislav Sedov } 12130daf62d9SStanislav Sedov 12140daf62d9SStanislav Sedov /* 12150daf62d9SStanislav Sedov * Descriptor flags to filestat translation. 12160daf62d9SStanislav Sedov */ 12170daf62d9SStanislav Sedov static int 12180daf62d9SStanislav Sedov to_filestat_flags(int flags) 12190daf62d9SStanislav Sedov { 12200daf62d9SStanislav Sedov static struct { 12210daf62d9SStanislav Sedov int flag; 12220daf62d9SStanislav Sedov int fst_flag; 12230daf62d9SStanislav Sedov } fstflags[] = { 12240daf62d9SStanislav Sedov { FREAD, PS_FST_FFLAG_READ }, 12250daf62d9SStanislav Sedov { FWRITE, PS_FST_FFLAG_WRITE }, 12260daf62d9SStanislav Sedov { O_APPEND, PS_FST_FFLAG_APPEND }, 12270daf62d9SStanislav Sedov { O_ASYNC, PS_FST_FFLAG_ASYNC }, 12280daf62d9SStanislav Sedov { O_CREAT, PS_FST_FFLAG_CREAT }, 12290daf62d9SStanislav Sedov { O_DIRECT, PS_FST_FFLAG_DIRECT }, 12300daf62d9SStanislav Sedov { O_EXCL, PS_FST_FFLAG_EXCL }, 12310daf62d9SStanislav Sedov { O_EXEC, PS_FST_FFLAG_EXEC }, 12320daf62d9SStanislav Sedov { O_EXLOCK, PS_FST_FFLAG_EXLOCK }, 12330daf62d9SStanislav Sedov { O_NOFOLLOW, PS_FST_FFLAG_NOFOLLOW }, 12340daf62d9SStanislav Sedov { O_NONBLOCK, PS_FST_FFLAG_NONBLOCK }, 12350daf62d9SStanislav Sedov { O_SHLOCK, PS_FST_FFLAG_SHLOCK }, 12360daf62d9SStanislav Sedov { O_SYNC, PS_FST_FFLAG_SYNC }, 12370daf62d9SStanislav Sedov { O_TRUNC, PS_FST_FFLAG_TRUNC } 12380daf62d9SStanislav Sedov }; 12390daf62d9SStanislav Sedov #define NFSTFLAGS (sizeof(fstflags) / sizeof(*fstflags)) 12400daf62d9SStanislav Sedov int fst_flags; 12410daf62d9SStanislav Sedov unsigned int i; 12420daf62d9SStanislav Sedov 12430daf62d9SStanislav Sedov fst_flags = 0; 12440daf62d9SStanislav Sedov for (i = 0; i < NFSTFLAGS; i++) 12450daf62d9SStanislav Sedov if (flags & fstflags[i].flag) 12460daf62d9SStanislav Sedov fst_flags |= fstflags[i].fst_flag; 12470daf62d9SStanislav Sedov return (fst_flags); 12480daf62d9SStanislav Sedov } 12490daf62d9SStanislav Sedov 12500daf62d9SStanislav Sedov /* 12510daf62d9SStanislav Sedov * Vnode type to filestate translation. 12520daf62d9SStanislav Sedov */ 12530daf62d9SStanislav Sedov static int 12540daf62d9SStanislav Sedov vntype2psfsttype(int type) 12550daf62d9SStanislav Sedov { 12560daf62d9SStanislav Sedov static struct { 12570daf62d9SStanislav Sedov int vtype; 12580daf62d9SStanislav Sedov int fst_vtype; 12590daf62d9SStanislav Sedov } vt2fst[] = { 12600daf62d9SStanislav Sedov { VBAD, PS_FST_VTYPE_VBAD }, 12610daf62d9SStanislav Sedov { VBLK, PS_FST_VTYPE_VBLK }, 12620daf62d9SStanislav Sedov { VCHR, PS_FST_VTYPE_VCHR }, 12630daf62d9SStanislav Sedov { VDIR, PS_FST_VTYPE_VDIR }, 12640daf62d9SStanislav Sedov { VFIFO, PS_FST_VTYPE_VFIFO }, 12650daf62d9SStanislav Sedov { VLNK, PS_FST_VTYPE_VLNK }, 12660daf62d9SStanislav Sedov { VNON, PS_FST_VTYPE_VNON }, 12670daf62d9SStanislav Sedov { VREG, PS_FST_VTYPE_VREG }, 12680daf62d9SStanislav Sedov { VSOCK, PS_FST_VTYPE_VSOCK } 12690daf62d9SStanislav Sedov }; 12700daf62d9SStanislav Sedov #define NVFTYPES (sizeof(vt2fst) / sizeof(*vt2fst)) 12710daf62d9SStanislav Sedov unsigned int i, fst_type; 12720daf62d9SStanislav Sedov 12730daf62d9SStanislav Sedov fst_type = PS_FST_VTYPE_UNKNOWN; 12740daf62d9SStanislav Sedov for (i = 0; i < NVFTYPES; i++) { 12750daf62d9SStanislav Sedov if (type == vt2fst[i].vtype) { 12760daf62d9SStanislav Sedov fst_type = vt2fst[i].fst_vtype; 12770daf62d9SStanislav Sedov break; 12780daf62d9SStanislav Sedov } 12790daf62d9SStanislav Sedov } 12800daf62d9SStanislav Sedov return (fst_type); 12810daf62d9SStanislav Sedov } 12820daf62d9SStanislav Sedov 12830daf62d9SStanislav Sedov static char * 12840daf62d9SStanislav Sedov getmnton(kvm_t *kd, struct mount *m) 12850daf62d9SStanislav Sedov { 12865f301949SBen Laurie struct mount mnt; 12870daf62d9SStanislav Sedov static struct mtab { 12880daf62d9SStanislav Sedov struct mtab *next; 12890daf62d9SStanislav Sedov struct mount *m; 12900daf62d9SStanislav Sedov char mntonname[MNAMELEN + 1]; 12910daf62d9SStanislav Sedov } *mhead = NULL; 12920daf62d9SStanislav Sedov struct mtab *mt; 12930daf62d9SStanislav Sedov 12940daf62d9SStanislav Sedov for (mt = mhead; mt != NULL; mt = mt->next) 12950daf62d9SStanislav Sedov if (m == mt->m) 12960daf62d9SStanislav Sedov return (mt->mntonname); 12970daf62d9SStanislav Sedov if (!kvm_read_all(kd, (unsigned long)m, &mnt, sizeof(struct mount))) { 12980daf62d9SStanislav Sedov warnx("can't read mount table at %p", (void *)m); 12990daf62d9SStanislav Sedov return (NULL); 13000daf62d9SStanislav Sedov } 13010daf62d9SStanislav Sedov if ((mt = malloc(sizeof (struct mtab))) == NULL) 13020daf62d9SStanislav Sedov err(1, NULL); 13030daf62d9SStanislav Sedov mt->m = m; 13040daf62d9SStanislav Sedov bcopy(&mnt.mnt_stat.f_mntonname[0], &mt->mntonname[0], MNAMELEN); 13055f301949SBen Laurie mt->mntonname[MNAMELEN] = '\0'; 13060daf62d9SStanislav Sedov mt->next = mhead; 13070daf62d9SStanislav Sedov mhead = mt; 13080daf62d9SStanislav Sedov return (mt->mntonname); 13090daf62d9SStanislav Sedov } 1310