我目前正在尝试编写自己的程序来镜像pmap命令,特别是在Solaris 9上。我无法解析库的名称和路径。 Solaris命令的输出看起来像这样,来自bash shell:
bash-2.05# pmap $$
2427: bash
00010000 496K r-x-- /usr/bin/bash
0009A000 80K rwx-- /usr/bin/bash
000AE000 120K rwx-- [ heap ]
FF100000 688K r-x-- /usr/lib/libc.so.1
FF1BC000 24K rwx-- /usr/lib/libc.so.1
FF1C2000 8K rwx-- /usr/lib/libc.so.1
FF200000 568K r-x-- /usr/lib/libnsl.so.1
FF29E000 32K rwx-- /usr/lib/libnsl.so.1
FF2A6000 32K rwx-- /usr/lib/libnsl.so.1
FF2F0000 8K rwx-- [ anon ]
FF300000 16K r-x-- /usr/lib/libmp.so.2
FF314000 8K rwx-- /usr/lib/libmp.so.2
FF320000 8K r-x-- /usr/platform/sun4u-us3/lib/libc_psr.so.1
FF330000 40K r-x-- /usr/lib/libsocket.so.1
FF34A000 8K rwx-- /usr/lib/libsocket.so.1
FF350000 168K r-x-- /usr/lib/libcurses.so.1
FF38A000 32K rwx-- /usr/lib/libcurses.so.1
FF392000 8K rwx-- /usr/lib/libcurses.so.1
FF3A0000 8K r-x-- /usr/lib/libdl.so.1
FF3B0000 8K rwx-- [ anon ]
FF3C0000 152K r-x-- /usr/lib/ld.so.1
FF3F6000 8K rwx-- /usr/lib/ld.so.1
FFBFC000 16K rw--- [ stack ]
total 2536K
我可以通过阅读/ proc / $$ / map来复制程序的基本功能,但剩下的就是找出如何解析库名,如右图所示。 / proc / $$ / map只给出/ proc / $$ / object中文件的名称,它们只是通用名称。在Solaris 10(我有一个盒子)上,我似乎能够使用/ proc / $$ / path,它包含符号链接,但我正在处理的框没有这些。有没有人对如何获取这些库名称有任何直截了当的想法?当我桁架程序时,它似乎打开/ proc / $$ / as并查看内存并以某种方式找到它们,但我无法弄清楚它在哪里或为什么还没有。
答案 0 :(得分:2)
pmap
的Solaris实现实际上是通过OpenSolaris pmap.c
以源代码形式提供的。但这是非常复杂的东西;基础知识更容易实现。
Solaris为地图提供了三个 /proc
句柄:
/proc/<PID>/map
,其中包含反映“通常”工作集大小的struct prmap
数据/proc/<PID>/rmap
,其中还包含struct prmap
数据但虚拟集大小(VSZ),即反映映射的VA范围,即使映射未提交。/proc/<PID>/xmap
,其中包含struct prxmap
个数据,实际上会拖拽地址空间以识别内存驻留区域。当您传递no args(pmap
),map
(-r
)或rmap
(-x
时,xmap
实用程序会查看这些内容}),但是你发现它不仅仅是打开/读取地图proc文件。这很大程度上是导致源代码难以解析的原因。
尽管如此,您可以构建一个简单的Solaris [rx]map
(如果您当然访问struct prxmap
解析器,请使用struct prmap
而不是xmap
,例如:
char mappath[MAXPATHLEN];
sprintf(mappath, "/proc/%d/map", getpid());
int fd = open(mappath, O_RDONLY);
size_t nread;
size_t mapsz = (1 << 20); /* start at 1MB */
struct prmap *cur*mapbuf = malloc(mapsz);
while ((nread = pread(fd, mapbuf, mapsz, 0)) == mapsz) {
free(mapbuf);
mapsz *= 2;
mapbuf = malloc(mapsz);
}
for (cur = mapbuf; nread; cur++, nread -= sizeof(*mapbuf))
prettyprint(cur);
free(mapbuf);
一些提示:
mmap()
map
文件,这将无效(procfs不允许)read()
序列。始终从头开始阅读(使用示例中的pread
或lseek(..., 0, SEEK_SET)
之前read
)。享受实验!
修改强>
由于您已经着手寻找映射背后的路径名,这是一项非常困难的任务。 pmap
不会处理这个问题,而是使用libproc
的工具来解析这些名称,正如您在truss
找到的那样,可以通过进程地址空间进行搜索这些,但也使用了一些其他技术。它本质上是libproc/Psymtab.c
中的“iter”函数。 Solaris 9和更高版本之间存在差异......太久以前不记得......