【发布时间】:2012-01-03 01:28:50
【问题描述】:
我正在尝试在用户空间中使用 mmap 来读取“mem_map”开始的物理内存。它是一个包含所有物理页面的数组。这是一台运行 3.0 内核的 i386 机器。
代码是这样的:
....
//define page size
//
#define PAGE_SIZE 0x1000 //4096 bytes
#define PAGE_MASK (PAGE_SIZE - 1)
....
/* open /dev/mem file*/
if((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1) {
printf("/dev/mem could not be opened.\n");
perror("open");
exit(1);
} else {
printf("/dev/mem opened.\n");
}
/* Map one page */
printf(" mem_map is at physical addr: 0x%x\n", mem_map_phy_addr);
map_base = mmap(0, PAGE_SIZE, PROT_READ, MAP_SHARED, fd, (mem_map_phy_addr & ~PAGE_MASK)); //mem_map_phy_addr is at 0x356f2000
if(map_base == (void *) -1) {
printf("Memory map failed. err num = %d\n",errno);
perror("mmap"); //failed here
} else {
printf("Memory mapped at address %p.\n", map_base);
}
我以 root 身份运行它。输出是:
/dev/mem opened.
mem_map is at physical addr: 0x356f2000
Memory map failed. err num = 1
mmap: Operation not permitted
可以肯定的是,我搜索了这个问题并将以下行添加到我的 /etc/sysctl.conf 文件中:
vm.mmap_min_addr = 0
但这也不起作用。
任何人都知道为什么不允许这样的 mem_map 操作以及如何绕过它?
谢谢。
【问题讨论】:
-
仅供参考,使用
x & ~PAGE_MASK是不正确的。在 64 位系统上,这会将地址截断为 32 位。在取补码之前,您必须转换为uintptr_t或等效的宽类型。 -
您是运行 sysctl 命令来设置 mmap_min_addr 的值还是只是编辑 conf 文件?你必须两者都做。
-
是的,我后来做了“sysctl -p”。