1 /*- 2 * Copyright (c) 2007 Robert N. M. Watson 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24 * SUCH DAMAGE. 25 * 26 * $FreeBSD$ 27 */ 28 29 #include <sys/types.h> 30 #include <sys/sysctl.h> 31 #include <sys/user.h> 32 33 #include <err.h> 34 #include <errno.h> 35 #include <stdio.h> 36 #include <stdlib.h> 37 38 #include "procstat.h" 39 40 void 41 procstat_vm(pid_t pid, struct kinfo_proc *kipp) 42 { 43 struct kinfo_vmentry *freep, *kve; 44 int error, i, name[4], ptrwidth; 45 const char *str; 46 size_t len; 47 48 ptrwidth = 2*sizeof(void *) + 2; 49 if (!hflag) 50 printf("%5s %*s %*s %3s %4s %4s %3s %3s %2s %-2s %-s\n", 51 "PID", ptrwidth, "START", ptrwidth, "END", "PRT", "RES", 52 "PRES", "REF", "SHD", "FL", "TP", "PATH"); 53 54 name[0] = CTL_KERN; 55 name[1] = KERN_PROC; 56 name[2] = KERN_PROC_VMMAP; 57 name[3] = pid; 58 59 len = 0; 60 error = sysctl(name, 4, NULL, &len, NULL, 0); 61 if (error < 0 && errno != ESRCH && errno != EPERM) { 62 warn("sysctl: kern.proc.vmmap: %d", pid); 63 return; 64 } 65 if (error < 0) 66 return; 67 68 /* 69 * Especially if running procstat -sv, we may need room for more 70 * mappings when printing than were present when we queried, so pad 71 * out the allocation a bit. 72 */ 73 len += sizeof(*kve) * 3; 74 freep = kve = malloc(len); 75 if (kve == NULL) 76 err(-1, "malloc"); 77 if (sysctl(name, 4, kve, &len, NULL, 0) < 0) { 78 warn("sysctl: kern.proc.vmmap: %d", pid); 79 free(freep); 80 return; 81 } 82 83 for (i = 0; i < (len / sizeof(*kve)); i++, kve++) { 84 if (kve->kve_structsize != sizeof(*kve)) 85 errx(-1, "kinfo_vmentry structure mismatch"); 86 printf("%5d ", pid); 87 printf("%*p ", ptrwidth, kve->kve_start); 88 printf("%*p ", ptrwidth, kve->kve_end); 89 printf("%s", kve->kve_protection & KVME_PROT_READ ? "r" : "-"); 90 printf("%s", kve->kve_protection & KVME_PROT_WRITE ? "w" : "-"); 91 printf("%s ", kve->kve_protection & KVME_PROT_EXEC ? "x" : "-"); 92 printf("%4d ", kve->kve_resident); 93 printf("%4d ", kve->kve_private_resident); 94 printf("%3d ", kve->kve_ref_count); 95 printf("%3d ", kve->kve_shadow_count); 96 printf("%-1s", kve->kve_flags & KVME_FLAG_COW ? "C" : "-"); 97 printf("%-1s ", kve->kve_flags & KVME_FLAG_NEEDS_COPY ? "N" : 98 "-"); 99 switch (kve->kve_type) { 100 case KVME_TYPE_NONE: 101 str = "--"; 102 break; 103 case KVME_TYPE_DEFAULT: 104 str = "df"; 105 break; 106 case KVME_TYPE_VNODE: 107 str = "vn"; 108 break; 109 case KVME_TYPE_SWAP: 110 str = "sw"; 111 break; 112 case KVME_TYPE_DEVICE: 113 str = "dv"; 114 break; 115 case KVME_TYPE_PHYS: 116 str = "ph"; 117 break; 118 case KVME_TYPE_DEAD: 119 str = "dd"; 120 break; 121 case KVME_TYPE_UNKNOWN: 122 default: 123 str = "??"; 124 break; 125 } 126 printf("%-2s ", str); 127 printf("%-s\n", kve->kve_path); 128 } 129 free(freep); 130 } 131