4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
23 * Copyright 2008 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
28 #include <stdio_ext.h>
38 #include <sys/types.h>
41 #include <sys/mkdev.h>
43 #include <sys/lgrp_user.h>
46 #include "pmap_common.h"
49 #define MEGABYTE (KILOBYTE * KILOBYTE)
50 #define GIGABYTE (KILOBYTE * KILOBYTE * KILOBYTE)
53 * Round up the value to the nearest kilobyte
55 #define ROUNDUP_KB(x) (((x) + (KILOBYTE - 1)) / KILOBYTE)
58 * The alignment should be a power of 2.
60 #define P2ALIGN(x, align) ((x) & -(align))
62 #define INVALID_ADDRESS (uintptr_t)(-1)
73 * -L option requires per-page information. The information is presented in an
74 * array of page_descr structures.
76 typedef struct page_descr
{
77 uintptr_t pd_start
; /* start address of a page */
78 size_t pd_pagesize
; /* page size in bytes */
79 lgrp_id_t pd_lgrp
; /* lgroup of memory backing the page */
80 int pd_valid
; /* valid page description if non-zero */
84 * Per-page information for a memory chunk.
85 * The meminfo(2) system call accepts up to MAX_MEMINFO_CNT pages at once.
86 * When we need to scan larger ranges we divide them in MAX_MEMINFO_CNT sized
87 * chunks. The chunk information is stored in the memory_chunk structure.
89 typedef struct memory_chunk
{
90 page_descr_t page_info
[MAX_MEMINFO_CNT
];
92 uintptr_t chunk_start
; /* Starting address */
93 uintptr_t chunk_end
; /* chunk_end is always <= end_addr */
95 int page_index
; /* Current page */
96 int page_count
; /* Number of pages */
99 static volatile int interrupt
;
101 typedef int proc_xmap_f(void *, const prxmap_t
*, const char *, int, int);
103 static int xmapping_iter(struct ps_prochandle
*, proc_xmap_f
*, void *,
105 static int rmapping_iter(struct ps_prochandle
*, proc_map_f
*, void *);
107 static int look_map(void *, const prmap_t
*, const char *);
108 static int look_smap(void *, const prxmap_t
*, const char *, int, int);
109 static int look_xmap(void *, const prxmap_t
*, const char *, int, int);
110 static int look_xmap_nopgsz(void *, const prxmap_t
*, const char *,
113 static int gather_map(void *, const prmap_t
*, const char *);
114 static int gather_xmap(void *, const prxmap_t
*, const char *, int, int);
115 static int iter_map(proc_map_f
*, void *);
116 static int iter_xmap(proc_xmap_f
*, void *);
117 static int parse_addr_range(char *, uintptr_t *, uintptr_t *);
118 static void mem_chunk_init(memory_chunk_t
*, uintptr_t, size_t);
120 static int perr(char *);
121 static void printK(long, int);
122 static char *mflags(uint_t
);
124 static size_t get_contiguous_region(memory_chunk_t
*, uintptr_t,
125 uintptr_t, size_t, lgrp_id_t
*);
126 static void mem_chunk_get(memory_chunk_t
*, uintptr_t);
127 static lgrp_id_t
addr_to_lgrp(memory_chunk_t
*, uintptr_t, size_t *);
128 static char *lgrp2str(lgrp_id_t
);
130 static int address_in_range(uintptr_t, uintptr_t, size_t);
131 static size_t adjust_addr_range(uintptr_t, uintptr_t, size_t,
132 uintptr_t *, uintptr_t *);
134 static int lflag
= 0;
135 static int Lflag
= 0;
136 static int aflag
= 0;
139 * The -A address range is represented as a pair of addresses
140 * <start_addr, end_addr>. Either one of these may be unspecified (set to
141 * INVALID_ADDRESS). If both are unspecified, no address range restrictions are
144 static uintptr_t start_addr
= INVALID_ADDRESS
;
145 static uintptr_t end_addr
= INVALID_ADDRESS
;
147 static int addr_width
, size_width
;
148 static char *command
;
149 static char *procname
;
150 static struct ps_prochandle
*Pr
;
152 static void intr(int);
162 static mapdata_t
*maps
;
163 static int map_count
;
164 static int map_alloc
;
166 static lwpstack_t
*stacks
= NULL
;
167 static uint_t nstacks
= 0;
172 getstack(void *data
, const lwpstatus_t
*lsp
)
174 int *np
= (int *)data
;
176 if (Plwp_alt_stack(Pr
, lsp
->pr_lwpid
, &stacks
[*np
].lwps_stack
) == 0) {
177 stacks
[*np
].lwps_stack
.ss_flags
|= SS_ONSTACK
;
178 stacks
[*np
].lwps_lwpid
= lsp
->pr_lwpid
;
182 if (Plwp_main_stack(Pr
, lsp
->pr_lwpid
, &stacks
[*np
].lwps_stack
) == 0) {
183 stacks
[*np
].lwps_lwpid
= lsp
->pr_lwpid
;
191 main(int argc
, char **argv
)
193 int rflag
= 0, sflag
= 0, xflag
= 0, Fflag
= 0;
194 int errflg
= 0, Sflag
= 0;
197 const char *bar8
= "-------";
198 const char *bar16
= "----------";
201 struct stat64 statbuf
;
204 int prg_gflags
= PGRAB_RDONLY
;
206 boolean_t use_agent_lwp
= B_FALSE
;
208 if ((command
= strrchr(argv
[0], '/')) != NULL
)
213 while ((opt
= getopt(argc
, argv
, "arsxSlLFA:")) != EOF
) {
215 case 'a': /* include shared mappings in -[xS] */
218 case 'r': /* show reserved mappings */
221 case 's': /* show hardware page sizes */
224 case 'S': /* show swap reservations */
227 case 'x': /* show extended mappings */
230 case 'l': /* show unresolved link map names */
233 case 'L': /* show lgroup information */
235 use_agent_lwp
= B_TRUE
;
237 case 'F': /* force grabbing (no O_EXCL) */
241 if (parse_addr_range(optarg
, &start_addr
, &end_addr
)
254 if ((Sflag
&& (xflag
|| rflag
|| sflag
)) || (xflag
&& rflag
) ||
255 (aflag
&& (!xflag
&& !Sflag
)) ||
256 (Lflag
&& (xflag
|| Sflag
))) {
260 if (errflg
|| argc
<= 0) {
261 (void) fprintf(stderr
,
262 "usage:\t%s [-rslF] [-A start[,end]] { pid | core } ...\n",
264 (void) fprintf(stderr
,
265 "\t\t(report process address maps)\n");
266 (void) fprintf(stderr
,
267 "\t%s -L [-rslF] [-A start[,end]] pid ...\n", command
);
268 (void) fprintf(stderr
,
269 "\t\t(report process address maps lgroups mappings)\n");
270 (void) fprintf(stderr
,
271 "\t%s -x [-aslF] [-A start[,end]] pid ...\n", command
);
272 (void) fprintf(stderr
,
273 "\t\t(show resident/anon/locked mapping details)\n");
274 (void) fprintf(stderr
,
275 "\t%s -S [-alF] [-A start[,end]] { pid | core } ...\n",
277 (void) fprintf(stderr
,
278 "\t\t(show swap reservations)\n\n");
279 (void) fprintf(stderr
,
280 "\t-a: include shared mappings in -[xS] summary\n");
281 (void) fprintf(stderr
,
282 "\t-r: show reserved address maps\n");
283 (void) fprintf(stderr
,
284 "\t-s: show hardware page sizes\n");
285 (void) fprintf(stderr
,
286 "\t-l: show unresolved dynamic linker map names\n");
287 (void) fprintf(stderr
,
288 "\t-F: force grabbing of the target process\n");
289 (void) fprintf(stderr
,
290 "\t-L: show lgroup mappings\n");
291 (void) fprintf(stderr
,
292 "\t-A start,end: limit output to the specified range\n");
297 * Make sure we'll have enough file descriptors to handle a target
298 * that has many many mappings.
300 if (getrlimit(RLIMIT_NOFILE
, &rlim
) == 0) {
301 rlim
.rlim_cur
= rlim
.rlim_max
;
302 (void) setrlimit(RLIMIT_NOFILE
, &rlim
);
303 (void) enable_extended_FILE_stdio(-1, -1);
307 * The implementation of -L option creates an agent LWP in the target
308 * process address space. The agent LWP issues meminfo(2) system calls
309 * on behalf of the target process. If we are interrupted prematurely,
310 * the target process remains in the stopped state with the agent still
311 * attached to it. To prevent such situation we catch signals from
312 * terminal and terminate gracefully.
316 * Buffer output to stdout, stderr while process is grabbed.
317 * Prevents infamous deadlocks due to pmap `pgrep xterm` and
320 (void) proc_initstdio();
322 prg_gflags
= PGRAB_RETAIN
| Fflag
;
323 prr_flags
= PRELEASE_RETAIN
;
325 if (sigset(SIGHUP
, SIG_IGN
) == SIG_DFL
)
326 (void) sigset(SIGHUP
, intr
);
327 if (sigset(SIGINT
, SIG_IGN
) == SIG_DFL
)
328 (void) sigset(SIGINT
, intr
);
329 if (sigset(SIGQUIT
, SIG_IGN
) == SIG_DFL
)
330 (void) sigset(SIGQUIT
, intr
);
331 (void) sigset(SIGPIPE
, intr
);
332 (void) sigset(SIGTERM
, intr
);
342 (void) proc_flushstdio();
344 if ((Pr
= proc_arg_grab(arg
= *argv
++, PR_ARG_ANY
,
345 prg_gflags
, &gcode
)) == NULL
) {
346 (void) fprintf(stderr
, "%s: cannot examine %s: %s\n",
347 command
, arg
, Pgrab_error(gcode
));
352 procname
= arg
; /* for perr() */
354 addr_width
= (Pstatus(Pr
)->pr_dmodel
== PR_MODEL_LP64
) ? 16 : 8;
355 size_width
= (Pstatus(Pr
)->pr_dmodel
== PR_MODEL_LP64
) ? 11 : 8;
356 bar
= addr_width
== 8 ? bar8
: bar16
;
357 (void) memcpy(&psinfo
, Ppsinfo(Pr
), sizeof (psinfo_t
));
358 proc_unctrl_psinfo(&psinfo
);
360 if (Pstate(Pr
) != PS_DEAD
) {
361 (void) snprintf(buf
, sizeof (buf
),
362 "/proc/%d/map", (int)psinfo
.pr_pid
);
363 if ((mapfd
= open(buf
, O_RDONLY
)) < 0) {
364 (void) fprintf(stderr
, "%s: cannot "
365 "examine %s: lost control of "
366 "process\n", command
, arg
);
368 Prelease(Pr
, prr_flags
);
378 if (Pstate(Pr
) == PS_DEAD
) {
379 (void) printf("core '%s' of %d:\t%.70s\n",
380 arg
, (int)psinfo
.pr_pid
, psinfo
.pr_psargs
);
382 if (rflag
|| sflag
|| xflag
|| Sflag
|| Lflag
) {
383 (void) printf(" -%c option is not compatible "
384 "with core files\n", xflag
? 'x' :
385 sflag
? 's' : rflag
? 'r' :
387 Prelease(Pr
, prr_flags
);
393 (void) printf("%d:\t%.70s\n",
394 (int)psinfo
.pr_pid
, psinfo
.pr_psargs
);
397 if (!(Pstatus(Pr
)->pr_flags
& PR_ISSYS
)) {
401 * Since we're grabbing the process readonly, we need
402 * to make sure the address space doesn't change during
405 if (Pstate(Pr
) != PS_DEAD
) {
406 if (tries
++ == MAX_TRIES
) {
407 Prelease(Pr
, prr_flags
);
409 (void) fprintf(stderr
, "%s: cannot "
410 "examine %s: address space is "
411 "changing\n", command
, arg
);
415 if (fstat64(mapfd
, &statbuf
) != 0) {
416 Prelease(Pr
, prr_flags
);
418 (void) fprintf(stderr
, "%s: cannot "
419 "examine %s: lost control of "
420 "process\n", command
, arg
);
425 nstacks
= psinfo
.pr_nlwp
* 2;
426 stacks
= calloc(nstacks
, sizeof (stacks
[0]));
427 if (stacks
!= NULL
) {
429 (void) Plwp_iter(Pr
, getstack
, &n
);
430 qsort(stacks
, nstacks
, sizeof (stacks
[0]),
434 (void) memset(&t
, 0, sizeof (t
));
436 if (Pgetauxval(Pr
, AT_BASE
) != -1L &&
437 Prd_agent(Pr
) == NULL
) {
438 (void) fprintf(stderr
, "%s: warning: "
439 "librtld_db failed to initialize; "
440 "shared library information will not be "
441 "available\n", command
);
448 rc
+= xmapping_iter(Pr
, gather_xmap
, NULL
, 0);
450 rc
+= xmapping_iter(Pr
, gather_xmap
, NULL
, 1);
453 rc
+= rmapping_iter(Pr
, gather_map
,
456 rc
+= xmapping_iter(Pr
, gather_xmap
,
459 rc
+= Pmapping_iter(Pr
,
462 rc
+= Pmapping_iter_resolved(Pr
,
467 * Ensure mappings are consistent.
469 if (Pstate(Pr
) != PS_DEAD
) {
470 struct stat64 newbuf
;
472 if (fstat64(mapfd
, &newbuf
) != 0 ||
473 memcmp(&newbuf
.st_mtim
, &statbuf
.st_mtim
,
474 sizeof (newbuf
.st_mtim
)) != 0) {
475 if (stacks
!= NULL
) {
487 (void) printf("%*s%*s%*s%*s%*s "
488 "%sMode Mapped File\n",
489 addr_width
, "Address",
490 size_width
, "Kbytes",
493 size_width
, "Locked",
494 sflag
? "Pgsz " : "");
496 rc
+= iter_xmap(sflag
? look_xmap
:
497 look_xmap_nopgsz
, &t
);
499 (void) printf("%s%s %s %s %s %s\n",
500 addr_width
== 8 ? "-" : "------",
501 bar
, bar
, bar
, bar
, bar
);
503 (void) printf("%stotal Kb", addr_width
== 16 ?
506 printK(t
.total_size
, size_width
);
507 printK(t
.total_rss
, size_width
);
508 printK(t
.total_anon
, size_width
);
509 printK(t
.total_locked
, size_width
);
514 (void) printf("%*s%*s%*s Mode"
516 addr_width
, "Address",
517 size_width
, "Kbytes",
520 rc
+= iter_xmap(look_xmap_nopgsz
, &t
);
522 (void) printf("%s%s %s %s\n",
523 addr_width
== 8 ? "-" : "------",
526 (void) printf("%stotal Kb", addr_width
== 16 ?
529 printK(t
.total_size
, size_width
);
530 printK(t
.total_swap
, size_width
);
537 rc
+= iter_map(look_map
, &t
);
540 (void) printf("%*s %*s %4s"
542 addr_width
, "Address",
544 "Bytes", "Pgsz", "Mode ",
545 "Lgrp", "Mapped File");
546 rc
+= iter_xmap(look_smap
, &t
);
548 (void) printf("%*s %*s %4s"
550 addr_width
, "Address",
552 "Bytes", "Pgsz", "Mode ",
554 rc
+= iter_xmap(look_smap
, &t
);
557 rc
+= iter_map(look_map
, &t
);
560 (void) printf(" %stotal %*luK\n",
563 size_width
, t
.total_size
);
566 if (stacks
!= NULL
) {
573 Prelease(Pr
, prr_flags
);
579 (void) proc_finistdio();
585 rmapping_iter(struct ps_prochandle
*Pr
, proc_map_f
*func
, void *cd
)
587 char mapname
[PATH_MAX
];
588 int mapfd
, nmap
, i
, rc
;
590 prmap_t
*prmapp
, *pmp
;
593 (void) snprintf(mapname
, sizeof (mapname
),
594 "/proc/%d/rmap", (int)Pstatus(Pr
)->pr_pid
);
596 if ((mapfd
= open(mapname
, O_RDONLY
)) < 0 || fstat(mapfd
, &st
) != 0) {
599 return (perr(mapname
));
602 nmap
= st
.st_size
/ sizeof (prmap_t
);
603 prmapp
= malloc((nmap
+ 1) * sizeof (prmap_t
));
605 if ((n
= pread(mapfd
, prmapp
, (nmap
+ 1) * sizeof (prmap_t
), 0L)) < 0) {
608 return (perr("read rmap"));
612 nmap
= n
/ sizeof (prmap_t
);
614 for (i
= 0, pmp
= prmapp
; i
< nmap
; i
++, pmp
++) {
615 if ((rc
= func(cd
, pmp
, NULL
)) != 0) {
626 xmapping_iter(struct ps_prochandle
*Pr
, proc_xmap_f
*func
, void *cd
, int doswap
)
628 char mapname
[PATH_MAX
];
629 int mapfd
, nmap
, i
, rc
;
631 prxmap_t
*prmapp
, *pmp
;
634 (void) snprintf(mapname
, sizeof (mapname
),
635 "/proc/%d/xmap", (int)Pstatus(Pr
)->pr_pid
);
637 if ((mapfd
= open(mapname
, O_RDONLY
)) < 0 || fstat(mapfd
, &st
) != 0) {
640 return (perr(mapname
));
643 nmap
= st
.st_size
/ sizeof (prxmap_t
);
646 prmapp
= malloc((nmap
+ 1) * sizeof (prxmap_t
));
648 if ((n
= pread(mapfd
, prmapp
, (nmap
+ 1) * sizeof (prxmap_t
), 0)) < 0) {
651 return (perr("read xmap"));
654 if (nmap
< n
/ sizeof (prxmap_t
)) {
661 nmap
= n
/ sizeof (prxmap_t
);
663 for (i
= 0, pmp
= prmapp
; i
< nmap
; i
++, pmp
++) {
664 if ((rc
= func(cd
, pmp
, NULL
, i
== nmap
- 1, doswap
)) != 0) {
671 * Mark the last element.
674 maps
[map_count
- 1].md_last
= B_TRUE
;
682 look_map(void *data
, const prmap_t
*pmp
, const char *object_name
)
684 struct totals
*t
= data
;
685 const pstatus_t
*Psp
= Pstatus(Pr
);
687 char mname
[PATH_MAX
];
689 size_t psz
= pmp
->pr_pagesize
;
690 uintptr_t vaddr
= pmp
->pr_vaddr
;
691 uintptr_t segment_end
= vaddr
+ pmp
->pr_size
;
693 memory_chunk_t mchunk
;
696 * If the mapping is not anon or not part of the heap, make a name
697 * for it. We don't want to report the heap as a.out's data.
699 if (!(pmp
->pr_mflags
& MA_ANON
) ||
700 segment_end
<= Psp
->pr_brkbase
||
701 pmp
->pr_vaddr
>= Psp
->pr_brkbase
+ Psp
->pr_brksize
) {
702 lname
= make_name(Pr
, lflag
, pmp
->pr_vaddr
, pmp
->pr_mapname
,
703 mname
, sizeof (mname
));
707 ((pmp
->pr_mflags
& MA_ANON
) || Pstate(Pr
) == PS_DEAD
)) {
708 lname
= anon_name(mname
, Psp
, stacks
, nstacks
, pmp
->pr_vaddr
,
709 pmp
->pr_size
, pmp
->pr_mflags
, pmp
->pr_shmid
, NULL
);
713 * Adjust the address range if -A is specified.
715 size
= adjust_addr_range(pmp
->pr_vaddr
, segment_end
, psz
,
716 &vaddr
, &segment_end
);
723 * Display the whole mapping
725 size
= ROUNDUP_KB(size
);
727 (void) printf(lname
?
728 "%.*lX %*luK %-6s %s\n" :
731 size_width
- 1, size
, mflags(pmp
->pr_mflags
), lname
);
733 t
->total_size
+= size
;
738 * We need to display lgroups backing physical memory, so we break the
739 * segment into individual pages and coalesce pages with the same lgroup
740 * into one "segment".
744 * Initialize address descriptions for the mapping.
746 mem_chunk_init(&mchunk
, segment_end
, psz
);
750 * Walk mapping (page by page) and display contiguous ranges of memory
751 * allocated to same lgroup.
757 * Get contiguous region of memory starting from vaddr allocated
758 * from the same lgroup.
760 size_contig
= get_contiguous_region(&mchunk
, vaddr
,
761 segment_end
, pmp
->pr_pagesize
, &lgrp
);
763 (void) printf(lname
? "%.*lX %*luK %-6s%s %s\n" :
764 "%.*lX %*luK %s %s\n",
766 size_width
- 1, size_contig
/ KILOBYTE
,
767 mflags(pmp
->pr_mflags
),
768 lgrp2str(lgrp
), lname
);
770 vaddr
+= size_contig
;
772 } while (vaddr
< segment_end
&& !interrupt
);
774 /* Update the total size */
775 t
->total_size
+= ROUNDUP_KB(size
);
780 printK(long value
, int width
)
783 (void) printf(width
== 8 ? " -" : " -");
785 (void) printf(" %*lu", width
- 1, value
);
789 pagesize(const prxmap_t
*pmp
)
791 int pagesize
= pmp
->pr_hatpagesize
;
795 return ("-"); /* no underlying HAT mapping */
798 if (pagesize
>= KILOBYTE
&& (pagesize
% KILOBYTE
) == 0) {
799 if ((pagesize
% GIGABYTE
) == 0)
800 (void) snprintf(buf
, sizeof (buf
), "%dG",
801 pagesize
/ GIGABYTE
);
802 else if ((pagesize
% MEGABYTE
) == 0)
803 (void) snprintf(buf
, sizeof (buf
), "%dM",
804 pagesize
/ MEGABYTE
);
806 (void) snprintf(buf
, sizeof (buf
), "%dK",
807 pagesize
/ KILOBYTE
);
809 (void) snprintf(buf
, sizeof (buf
), "%db", pagesize
);
816 look_smap(void *data
,
818 const char *object_name
,
819 int last
, int doswap
)
821 struct totals
*t
= data
;
822 const pstatus_t
*Psp
= Pstatus(Pr
);
824 char mname
[PATH_MAX
];
827 size_t psz
= pmp
->pr_pagesize
;
828 uintptr_t vaddr
= pmp
->pr_vaddr
;
829 uintptr_t segment_end
= vaddr
+ pmp
->pr_size
;
831 memory_chunk_t mchunk
;
834 * If the mapping is not anon or not part of the heap, make a name
835 * for it. We don't want to report the heap as a.out's data.
837 if (!(pmp
->pr_mflags
& MA_ANON
) ||
838 pmp
->pr_vaddr
+ pmp
->pr_size
<= Psp
->pr_brkbase
||
839 pmp
->pr_vaddr
>= Psp
->pr_brkbase
+ Psp
->pr_brksize
) {
840 lname
= make_name(Pr
, lflag
, pmp
->pr_vaddr
, pmp
->pr_mapname
,
841 mname
, sizeof (mname
));
845 ((pmp
->pr_mflags
& MA_ANON
) || Pstate(Pr
) == PS_DEAD
)) {
846 lname
= anon_name(mname
, Psp
, stacks
, nstacks
, pmp
->pr_vaddr
,
847 pmp
->pr_size
, pmp
->pr_mflags
, pmp
->pr_shmid
, NULL
);
851 * Adjust the address range if -A is specified.
853 size
= adjust_addr_range(pmp
->pr_vaddr
, segment_end
, psz
,
854 &vaddr
, &segment_end
);
861 * Display the whole mapping
864 format
= "%.*lX %*luK %4s %-6s %s\n";
866 format
= "%.*lX %*luK %4s %s\n";
868 size
= ROUNDUP_KB(size
);
870 (void) printf(format
, addr_width
, vaddr
, size_width
- 1, size
,
871 pagesize(pmp
), mflags(pmp
->pr_mflags
), lname
);
873 t
->total_size
+= size
;
878 format
= "%.*lX %*luK %4s %-6s%s %s\n";
880 format
= "%.*lX %*luK %4s%s %s\n";
883 * We need to display lgroups backing physical memory, so we break the
884 * segment into individual pages and coalesce pages with the same lgroup
885 * into one "segment".
889 * Initialize address descriptions for the mapping.
891 mem_chunk_init(&mchunk
, segment_end
, psz
);
895 * Walk mapping (page by page) and display contiguous ranges of memory
896 * allocated to same lgroup.
902 * Get contiguous region of memory starting from vaddr allocated
903 * from the same lgroup.
905 size_contig
= get_contiguous_region(&mchunk
, vaddr
,
906 segment_end
, pmp
->pr_pagesize
, &lgrp
);
908 (void) printf(format
, addr_width
, vaddr
,
909 size_width
- 1, size_contig
/ KILOBYTE
,
910 pagesize(pmp
), mflags(pmp
->pr_mflags
),
911 lgrp2str(lgrp
), lname
);
913 vaddr
+= size_contig
;
915 } while (vaddr
< segment_end
&& !interrupt
);
917 t
->total_size
+= ROUNDUP_KB(size
);
921 #define ANON(x) ((aflag || (((x)->pr_mflags & MA_SHARED) == 0)) ? \
926 look_xmap(void *data
,
928 const char *object_name
,
929 int last
, int doswap
)
931 struct totals
*t
= data
;
932 const pstatus_t
*Psp
= Pstatus(Pr
);
933 char mname
[PATH_MAX
];
938 * If the mapping is not anon or not part of the heap, make a name
939 * for it. We don't want to report the heap as a.out's data.
941 if (!(pmp
->pr_mflags
& MA_ANON
) ||
942 pmp
->pr_vaddr
+ pmp
->pr_size
<= Psp
->pr_brkbase
||
943 pmp
->pr_vaddr
>= Psp
->pr_brkbase
+ Psp
->pr_brksize
) {
944 lname
= make_name(Pr
, lflag
, pmp
->pr_vaddr
, pmp
->pr_mapname
,
945 mname
, sizeof (mname
));
949 if ((ln
= strrchr(lname
, '/')) != NULL
)
951 } else if ((pmp
->pr_mflags
& MA_ANON
) || Pstate(Pr
) == PS_DEAD
) {
952 lname
= anon_name(mname
, Psp
, stacks
, nstacks
, pmp
->pr_vaddr
,
953 pmp
->pr_size
, pmp
->pr_mflags
, pmp
->pr_shmid
, NULL
);
956 (void) printf("%.*lX", addr_width
, (ulong_t
)pmp
->pr_vaddr
);
958 printK(ROUNDUP_KB(pmp
->pr_size
), size_width
);
959 printK(pmp
->pr_rss
* (pmp
->pr_pagesize
/ KILOBYTE
), size_width
);
960 printK(ANON(pmp
) * (pmp
->pr_pagesize
/ KILOBYTE
), size_width
);
961 printK(pmp
->pr_locked
* (pmp
->pr_pagesize
/ KILOBYTE
), size_width
);
962 (void) printf(lname
? " %4s %-6s %s\n" : " %4s %s\n",
963 pagesize(pmp
), mflags(pmp
->pr_mflags
), lname
);
965 t
->total_size
+= ROUNDUP_KB(pmp
->pr_size
);
966 t
->total_rss
+= pmp
->pr_rss
* (pmp
->pr_pagesize
/ KILOBYTE
);
967 t
->total_anon
+= ANON(pmp
) * (pmp
->pr_pagesize
/ KILOBYTE
);
968 t
->total_locked
+= (pmp
->pr_locked
* (pmp
->pr_pagesize
/ KILOBYTE
));
975 look_xmap_nopgsz(void *data
,
977 const char *object_name
,
978 int last
, int doswap
)
980 struct totals
*t
= data
;
981 const pstatus_t
*Psp
= Pstatus(Pr
);
982 char mname
[PATH_MAX
];
985 static uintptr_t prev_vaddr
;
986 static size_t prev_size
;
987 static offset_t prev_offset
;
988 static int prev_mflags
;
989 static char *prev_lname
;
990 static char prev_mname
[PATH_MAX
];
991 static ulong_t prev_rss
;
992 static ulong_t prev_anon
;
993 static ulong_t prev_locked
;
994 static ulong_t prev_swap
;
996 static int first
= 1;
1001 * Calculate swap reservations
1003 if (pmp
->pr_mflags
& MA_SHARED
) {
1004 if (aflag
&& (pmp
->pr_mflags
& MA_NORESERVE
) == 0) {
1005 /* Swap reserved for entire non-ism SHM */
1006 swap
= pmp
->pr_size
/ pmp
->pr_pagesize
;
1008 } else if (pmp
->pr_mflags
& MA_NORESERVE
) {
1009 /* Swap reserved on fault for each anon page */
1010 swap
= pmp
->pr_anon
;
1011 } else if (pmp
->pr_mflags
& MA_WRITE
) {
1012 /* Swap reserve for entire writable segment */
1013 swap
= pmp
->pr_size
/ pmp
->pr_pagesize
;
1017 * If the mapping is not anon or not part of the heap, make a name
1018 * for it. We don't want to report the heap as a.out's data.
1020 if (!(pmp
->pr_mflags
& MA_ANON
) ||
1021 pmp
->pr_vaddr
+ pmp
->pr_size
<= Psp
->pr_brkbase
||
1022 pmp
->pr_vaddr
>= Psp
->pr_brkbase
+ Psp
->pr_brksize
) {
1023 lname
= make_name(Pr
, lflag
, pmp
->pr_vaddr
, pmp
->pr_mapname
,
1024 mname
, sizeof (mname
));
1027 if (lname
!= NULL
) {
1028 if ((ln
= strrchr(lname
, '/')) != NULL
)
1030 } else if ((pmp
->pr_mflags
& MA_ANON
) || Pstate(Pr
) == PS_DEAD
) {
1031 lname
= anon_name(mname
, Psp
, stacks
, nstacks
, pmp
->pr_vaddr
,
1032 pmp
->pr_size
, pmp
->pr_mflags
, pmp
->pr_shmid
, NULL
);
1035 kperpage
= pmp
->pr_pagesize
/ KILOBYTE
;
1037 t
->total_size
+= ROUNDUP_KB(pmp
->pr_size
);
1038 t
->total_rss
+= pmp
->pr_rss
* kperpage
;
1039 t
->total_anon
+= ANON(pmp
) * kperpage
;
1040 t
->total_locked
+= pmp
->pr_locked
* kperpage
;
1041 t
->total_swap
+= swap
* kperpage
;
1045 prev_vaddr
= pmp
->pr_vaddr
;
1046 prev_size
= pmp
->pr_size
;
1047 prev_offset
= pmp
->pr_offset
;
1048 prev_mflags
= pmp
->pr_mflags
;
1049 if (lname
== NULL
) {
1052 (void) strcpy(prev_mname
, lname
);
1053 prev_lname
= prev_mname
;
1055 prev_rss
= pmp
->pr_rss
* kperpage
;
1056 prev_anon
= ANON(pmp
) * kperpage
;
1057 prev_locked
= pmp
->pr_locked
* kperpage
;
1058 prev_swap
= swap
* kperpage
;
1063 } else if (prev_vaddr
+ prev_size
== pmp
->pr_vaddr
&&
1064 prev_mflags
== pmp
->pr_mflags
&&
1065 ((prev_mflags
& MA_ISM
) ||
1066 prev_offset
+ prev_size
== pmp
->pr_offset
) &&
1067 ((lname
== NULL
&& prev_lname
== NULL
) ||
1068 (lname
!= NULL
&& prev_lname
!= NULL
&&
1069 strcmp(lname
, prev_lname
) == 0))) {
1070 prev_size
+= pmp
->pr_size
;
1071 prev_rss
+= pmp
->pr_rss
* kperpage
;
1072 prev_anon
+= ANON(pmp
) * kperpage
;
1073 prev_locked
+= pmp
->pr_locked
* kperpage
;
1074 prev_swap
+= swap
* kperpage
;
1081 (void) printf("%.*lX", addr_width
, (ulong_t
)prev_vaddr
);
1082 printK(ROUNDUP_KB(prev_size
), size_width
);
1085 printK(prev_swap
, size_width
);
1087 printK(prev_rss
, size_width
);
1088 printK(prev_anon
, size_width
);
1089 printK(prev_locked
, size_width
);
1091 (void) printf(prev_lname
? " %-6s %s\n" : "%s\n",
1092 mflags(prev_mflags
), prev_lname
);
1095 prev_vaddr
= pmp
->pr_vaddr
;
1096 prev_size
= pmp
->pr_size
;
1097 prev_offset
= pmp
->pr_offset
;
1098 prev_mflags
= pmp
->pr_mflags
;
1099 if (lname
== NULL
) {
1102 (void) strcpy(prev_mname
, lname
);
1103 prev_lname
= prev_mname
;
1105 prev_rss
= pmp
->pr_rss
* kperpage
;
1106 prev_anon
= ANON(pmp
) * kperpage
;
1107 prev_locked
= pmp
->pr_locked
* kperpage
;
1108 prev_swap
= swap
* kperpage
;
1109 } else if (merged
== 0) {
1110 (void) printf("%.*lX", addr_width
, (ulong_t
)pmp
->pr_vaddr
);
1111 printK(ROUNDUP_KB(pmp
->pr_size
), size_width
);
1113 printK(swap
* kperpage
, size_width
);
1115 printK(pmp
->pr_rss
* kperpage
, size_width
);
1116 printK(ANON(pmp
) * kperpage
, size_width
);
1117 printK(pmp
->pr_locked
* kperpage
, size_width
);
1119 (void) printf(lname
? " %-6s %s\n" : " %s\n",
1120 mflags(pmp
->pr_mflags
), lname
);
1133 (void) fprintf(stderr
, "%s: ", procname
);
1143 static char code_buf
[80];
1144 char *str
= code_buf
;
1149 * r - segment is readable
1150 * w - segment is writable
1151 * x - segment is executable
1152 * s - segment is shared
1153 * R - segment is mapped MAP_NORESERVE
1156 (void) sprintf(str
, "%c%c%c%c%c%c",
1157 arg
& MA_READ
? 'r' : '-',
1158 arg
& MA_WRITE
? 'w' : '-',
1159 arg
& MA_EXEC
? 'x' : '-',
1160 arg
& MA_SHARED
? 's' : '-',
1161 arg
& MA_NORESERVE
? 'R' : '-',
1162 arg
& MA_RESERVED1
? '*' : ' ');
1173 if (map_count
== map_alloc
) {
1177 next
= map_alloc
* 2;
1179 newmaps
= realloc(maps
, next
* sizeof (mapdata_t
));
1180 if (newmaps
== NULL
) {
1181 (void) perr("failed to allocate maps");
1184 (void) memset(newmaps
+ map_alloc
, '\0',
1185 (next
- map_alloc
) * sizeof (mapdata_t
));
1191 return (&maps
[map_count
++]);
1196 gather_map(void *ignored
, const prmap_t
*map
, const char *objname
)
1200 /* Skip mappings which are outside the range specified by -A */
1201 if (!address_in_range(map
->pr_vaddr
,
1202 map
->pr_vaddr
+ map
->pr_size
, map
->pr_pagesize
))
1206 data
->md_map
= *map
;
1207 if (data
->md_objname
!= NULL
)
1208 free(data
->md_objname
);
1209 data
->md_objname
= objname
? strdup(objname
) : NULL
;
1216 gather_xmap(void *ignored
, const prxmap_t
*xmap
, const char *objname
,
1217 int last
, int doswap
)
1221 /* Skip mappings which are outside the range specified by -A */
1222 if (!address_in_range(xmap
->pr_vaddr
,
1223 xmap
->pr_vaddr
+ xmap
->pr_size
, xmap
->pr_pagesize
))
1227 data
->md_xmap
= *xmap
;
1228 if (data
->md_objname
!= NULL
)
1229 free(data
->md_objname
);
1230 data
->md_objname
= objname
? strdup(objname
) : NULL
;
1231 data
->md_last
= last
;
1232 data
->md_doswap
= doswap
;
1238 iter_map(proc_map_f
*func
, void *data
)
1243 for (i
= 0; i
< map_count
; i
++) {
1246 if ((ret
= func(data
, &maps
[i
].md_map
,
1247 maps
[i
].md_objname
)) != 0)
1255 iter_xmap(proc_xmap_f
*func
, void *data
)
1260 for (i
= 0; i
< map_count
; i
++) {
1263 if ((ret
= func(data
, &maps
[i
].md_xmap
, maps
[i
].md_objname
,
1264 maps
[i
].md_last
, maps
[i
].md_doswap
)) != 0)
1272 * Convert lgroup ID to string.
1273 * returns dash when lgroup ID is invalid.
1276 lgrp2str(lgrp_id_t lgrp
)
1278 static char lgrp_buf
[20];
1279 char *str
= lgrp_buf
;
1281 (void) sprintf(str
, lgrp
== LGRP_NONE
? " -" : "%4d", lgrp
);
1286 * Parse address range specification for -A option.
1287 * The address range may have the following forms:
1290 * start and end is set to address
1292 * start is set to address, end is set to INVALID_ADDRESS
1294 * start is set to 0, end is set to address
1296 * start is set to address1, end is set to address2
1300 parse_addr_range(char *input_str
, uintptr_t *start
, uintptr_t *end
)
1302 char *startp
= input_str
;
1303 char *endp
= strchr(input_str
, ',');
1304 ulong_t s
= (ulong_t
)INVALID_ADDRESS
;
1305 ulong_t e
= (ulong_t
)INVALID_ADDRESS
;
1309 * Comma is present. If there is nothing after comma, the end
1310 * remains set at INVALID_ADDRESS. Otherwise it is set to the
1311 * value after comma.
1316 if ((*endp
!= '\0') && sscanf(endp
, "%lx", &e
) != 1)
1320 if (startp
!= NULL
) {
1322 * Read the start address, if it is specified. If the address is
1323 * missing, start will be set to INVALID_ADDRESS.
1325 if ((*startp
!= '\0') && sscanf(startp
, "%lx", &s
) != 1)
1329 /* If there is no comma, end becomes equal to start */
1334 * ,end implies 0..end range
1336 if (e
!= INVALID_ADDRESS
&& s
== INVALID_ADDRESS
)
1339 *start
= (uintptr_t)s
;
1340 *end
= (uintptr_t)e
;
1342 /* Return error if neither start nor end address were specified */
1343 return (! (s
!= INVALID_ADDRESS
|| e
!= INVALID_ADDRESS
));
1347 * Check whether any portion of [start, end] segment is within the
1348 * [start_addr, end_addr] range.
1351 * 0 - address is outside the range
1352 * 1 - address is within the range
1355 address_in_range(uintptr_t start
, uintptr_t end
, size_t psz
)
1360 * Nothing to do if there is no address range specified with -A
1362 if (start_addr
!= INVALID_ADDRESS
|| end_addr
!= INVALID_ADDRESS
) {
1363 /* The segment end is below the range start */
1364 if ((start_addr
!= INVALID_ADDRESS
) &&
1365 (end
< P2ALIGN(start_addr
, psz
)))
1368 /* The segment start is above the range end */
1369 if ((end_addr
!= INVALID_ADDRESS
) &&
1370 (start
> P2ALIGN(end_addr
+ psz
, psz
)))
1377 * Returns an intersection of the [start, end] interval and the range specified
1378 * by -A flag [start_addr, end_addr]. Unspecified parts of the address range
1379 * have value INVALID_ADDRESS.
1381 * The start_addr address is rounded down to the beginning of page and end_addr
1382 * is rounded up to the end of page.
1384 * Returns the size of the resulting interval or zero if the interval is empty
1388 adjust_addr_range(uintptr_t start
, uintptr_t end
, size_t psz
,
1389 uintptr_t *new_start
, uintptr_t *new_end
)
1391 uintptr_t from
; /* start_addr rounded down */
1392 uintptr_t to
; /* end_addr rounded up */
1395 * Round down the lower address of the range to the beginning of page.
1397 if (start_addr
== INVALID_ADDRESS
) {
1399 * No start_addr specified by -A, the lower part of the interval
1404 from
= P2ALIGN(start_addr
, psz
);
1406 * If end address is outside the range, return an empty
1410 *new_start
= *new_end
= 0;
1414 * The adjusted start address is the maximum of requested start
1415 * and the aligned start_addr of the -A range.
1417 *new_start
= start
< from
? from
: start
;
1421 * Round up the higher address of the range to the end of page.
1423 if (end_addr
== INVALID_ADDRESS
) {
1425 * No end_addr specified by -A, the upper part of the interval
1431 * If only one address is specified and it is the beginning of a
1432 * segment, get information about the whole segment. This
1433 * function is called once per segment and the 'end' argument is
1434 * always the end of a segment, so just use the 'end' value.
1436 to
= (end_addr
== start_addr
&& start
== start_addr
) ?
1438 P2ALIGN(end_addr
+ psz
, psz
);
1440 * If start address is outside the range, return an empty
1444 *new_start
= *new_end
= 0;
1448 * The adjusted end address is the minimum of requested end
1449 * and the aligned end_addr of the -A range.
1451 *new_end
= end
> to
? to
: end
;
1455 * Make sure that the resulting interval is legal.
1457 if (*new_end
< *new_start
)
1458 *new_start
= *new_end
= 0;
1460 /* Return the size of the interval */
1461 return (*new_end
- *new_start
);
1465 * Initialize memory_info data structure with information about a new segment.
1468 mem_chunk_init(memory_chunk_t
*chunk
, uintptr_t end
, size_t psz
)
1470 chunk
->end_addr
= end
;
1471 chunk
->page_size
= psz
;
1472 chunk
->page_index
= 0;
1473 chunk
->chunk_start
= chunk
->chunk_end
= 0;
1477 * Create a new chunk of addresses starting from vaddr.
1478 * Pass the whole chunk to pr_meminfo to collect lgroup and page size
1479 * information for each page in the chunk.
1482 mem_chunk_get(memory_chunk_t
*chunk
, uintptr_t vaddr
)
1484 page_descr_t
*pdp
= chunk
->page_info
;
1485 size_t psz
= chunk
->page_size
;
1486 uintptr_t addr
= vaddr
;
1487 uint64_t inaddr
[MAX_MEMINFO_CNT
];
1488 uint64_t outdata
[2 * MAX_MEMINFO_CNT
];
1489 uint_t info
[2] = { MEMINFO_VLGRP
, MEMINFO_VPAGESIZE
};
1490 uint_t validity
[MAX_MEMINFO_CNT
];
1491 uint64_t *dataptr
= inaddr
;
1492 uint64_t *outptr
= outdata
;
1493 uint_t
*valptr
= validity
;
1496 chunk
->chunk_start
= vaddr
;
1497 chunk
->page_index
= 0; /* reset index for the new chunk */
1500 * Fill in MAX_MEMINFO_CNT wotrh of pages starting from vaddr. Also,
1501 * copy starting address of each page to inaddr array for pr_meminfo.
1503 for (i
= 0, pdp
= chunk
->page_info
;
1504 (i
< MAX_MEMINFO_CNT
) && (addr
<= chunk
->end_addr
);
1505 i
++, pdp
++, dataptr
++, addr
+= psz
) {
1506 *dataptr
= (uint64_t)addr
;
1507 pdp
->pd_start
= addr
;
1508 pdp
->pd_lgrp
= LGRP_NONE
;
1510 pdp
->pd_pagesize
= 0;
1513 /* Mark the number of entries in the chunk and the last address */
1514 chunk
->page_count
= i
;
1515 chunk
->chunk_end
= addr
- psz
;
1520 /* Call meminfo for all collected addresses */
1521 rc
= pr_meminfo(Pr
, inaddr
, i
, info
, 2, outdata
, validity
);
1523 (void) perr("can not get memory information");
1527 /* Verify validity of each result and fill in the addrs array */
1528 pdp
= chunk
->page_info
;
1529 for (j
= 0; j
< i
; j
++, pdp
++, valptr
++, outptr
+= 2) {
1530 /* Skip invalid address pointers */
1531 if ((*valptr
& 1) == 0) {
1535 /* Is lgroup information available? */
1536 if ((*valptr
& 2) != 0) {
1537 pdp
->pd_lgrp
= (lgrp_id_t
)*outptr
;
1541 /* Is page size informaion available? */
1542 if ((*valptr
& 4) != 0) {
1543 pdp
->pd_pagesize
= *(outptr
+ 1);
1549 * Starting from address 'vaddr' find the region with pages allocated from the
1553 * mchunk Initialized memory chunk structure
1554 * vaddr Starting address of the region
1555 * maxaddr Upper bound of the region
1556 * pagesize Default page size to use
1557 * ret_lgrp On exit contains the lgroup ID of all pages in the
1561 * Size of the contiguous region in bytes
1562 * The lgroup ID of all pages in the region in ret_lgrp argument.
1565 get_contiguous_region(memory_chunk_t
*mchunk
, uintptr_t vaddr
,
1566 uintptr_t maxaddr
, size_t pagesize
, lgrp_id_t
*ret_lgrp
)
1568 size_t size_contig
= 0;
1569 lgrp_id_t lgrp
; /* Lgroup of the region start */
1570 lgrp_id_t curr_lgrp
; /* Lgroup of the current page */
1571 size_t psz
= pagesize
; /* Pagesize to use */
1573 /* Set both lgroup IDs to the lgroup of the first page */
1574 curr_lgrp
= lgrp
= addr_to_lgrp(mchunk
, vaddr
, &psz
);
1577 * Starting from vaddr, walk page by page until either the end
1578 * of the segment is reached or a page is allocated from a different
1579 * lgroup. Also stop if interrupted from keyboard.
1581 while ((vaddr
< maxaddr
) && (curr_lgrp
== lgrp
) && !interrupt
) {
1583 * Get lgroup ID and the page size of the current page.
1585 curr_lgrp
= addr_to_lgrp(mchunk
, vaddr
, &psz
);
1586 /* If there is no page size information, use the default */
1590 if (curr_lgrp
== lgrp
) {
1592 * This page belongs to the contiguous region.
1593 * Increase the region size and advance to the new page.
1600 /* Return the region lgroup ID and the size */
1602 return (size_contig
);
1606 * Given a virtual address, return its lgroup and page size. If there is meminfo
1607 * information for an address, use it, otherwise shift the chunk window to the
1608 * vaddr and create a new chunk with known meminfo information.
1611 addr_to_lgrp(memory_chunk_t
*chunk
, uintptr_t vaddr
, size_t *psz
)
1614 lgrp_id_t lgrp
= LGRP_NONE
;
1617 *psz
= chunk
->page_size
;
1623 * Is there information about this address? If not, create a new chunk
1624 * starting from vaddr and apply pr_meminfo() to the whole chunk.
1626 if (vaddr
< chunk
->chunk_start
|| vaddr
> chunk
->chunk_end
) {
1628 * This address is outside the chunk, get the new chunk and
1629 * collect meminfo information for it.
1631 mem_chunk_get(chunk
, vaddr
);
1635 * Find information about the address.
1637 pdp
= &chunk
->page_info
[chunk
->page_index
];
1638 for (i
= chunk
->page_index
; i
< chunk
->page_count
; i
++, pdp
++) {
1639 if (pdp
->pd_start
== vaddr
) {
1640 if (pdp
->pd_valid
) {
1641 lgrp
= pdp
->pd_lgrp
;
1643 * Override page size information if it is
1646 if (pdp
->pd_pagesize
> 0)
1647 *psz
= pdp
->pd_pagesize
;
1653 * Remember where we ended - the next search will start here.
1654 * We can query for the lgrp for the same address again, so do not
1655 * advance index past the current value.
1657 chunk
->page_index
= i
;