当使用写时复制语义在进程之间共享内存时,如何测试内存页是否可写或是否标记为只读?这可以通过调用特定的汇编代码、读取内存中的某个位置或通过操作系统的 API 来完成吗?
4 回答
在 Linux 上,您可以检查 /proc/ pid /maps:
$ cat /proc/self/maps
002b3000-002cc000 r-xp 00000000 68:01 143009 /lib/ld-2.5.so
002cc000-002cd000 r-xp 00018000 68:01 143009 /lib/ld-2.5.so
002cd000-002ce000 rwxp 00019000 68:01 143009 /lib/ld-2.5.so
002d0000-00407000 r-xp 00000000 68:01 143010 /lib/libc-2.5.so
00407000-00409000 r-xp 00137000 68:01 143010 /lib/libc-2.5.so
00409000-0040a000 rwxp 00139000 68:01 143010 /lib/libc-2.5.so
0040a000-0040d000 rwxp 0040a000 00:00 0
00c6f000-00c70000 r-xp 00c6f000 00:00 0 [vdso]
08048000-0804d000 r-xp 00000000 68:01 379298 /bin/cat
0804d000-0804e000 rw-p 00004000 68:01 379298 /bin/cat
08326000-08347000 rw-p 08326000 00:00 0
b7d1b000-b7f1b000 r--p 00000000 68:01 226705 /usr/lib/locale/locale-archive
b7f1b000-b7f1c000 rw-p b7f1b000 00:00 0
b7f28000-b7f29000 rw-p b7f28000 00:00 0
bfe37000-bfe4d000 rw-p bfe37000 00:00 0 [stack]
第一列是虚拟内存地址范围,第二列包含权限(读、写、执行和私有),第 3-6 列包含偏移量、主要和次要设备号、inode 和映射的内存名称文件。
在 Win32 上,最好的方法是使用VirtualQuery。它为地址所在的页面返回 a MEMORY_BASIC_INFORMATION
。其中一个成员是Protect
,它是这些标志的某种组合,其中包含可能的保护模式。该函数还告诉您内存是空闲的、已提交的、保留的,以及它是私有的、图像的一部分还是共享内存部分。
操作系统的 API 是确定页面保护的最佳方式。CPU 从页面描述符中读取保护模式,该页面描述符只能从内核模式访问。
如果您使用的是 Win32,则有调用 IsBadReadPtr 和 IsBadWritePtr。但是,不鼓励使用它们:
“普遍的共识是 IsBad 系列函数(IsBadReadPtr、IsBadWritePtr 等)已损坏,不应用于验证指针。”
Raymond Chen 的标题说明了一切:“IsBadXxxPtr 真的应该被称为 CrashProgramRandomly”
Chen 在这里有一些关于如何处理这个问题的有用建议。
结果是,您不应该在运行时测试这种东西。编写代码,以便您知道您正在处理什么,如果它不是预期的,则将其视为错误。如果您真的别无选择,请查看 SEH 以处理异常。
您是在谈论通过 shmget(在 Unix 上)分配的各种共享内存吗?IE
int shmget(key_t, size_t, int);
如果是这样,您可以使用查询该内存
int shmctl(int, int, struct shmid_ds *);
例如:
key_t key = /* your choice of memory api */
int flag = /* set of flags for your app */
int shmid = shmget(key, 4096, flag);
struct shmid_ds buf;
int result = shmctl(shmid, IPC_STAT, &buf);
/* buf.ipc_perm.mode contains the permissions for the memory segment */