2011-11-21 211 views
5

我想在用户空间中使用mmap来读取'mem_map'开始的物理内存。它是一个包含所有物理页面的数组。这是运行3.0内核的i386机器。mmap:不允许操作

的代码是这样的:

.... 

//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); 
    } 

我跑以此为根。输出是:

/dev/mem opened. 
mem_map is at physical addr: 0x356f2000 
Memory map failed. err num = 1 
mmap: Operation not permitted 

可以肯定的,我用Google搜索的问题,并添加下面一行到我的/etc/sysctl.conf文件:

vm.mmap_min_addr = 0 

但是,这也不行。

任何人都知道为什么这样的mem_map操作是不被允许的以及我如何解决它?

谢谢。

+2

仅供参考,它是不正确的使用'X - 〜PAGE_MASK'。在64位系统上,这会将地址截断为32位。在进行补充操作之前,您必须转换为'uintptr_t'或等效宽类型。 –

+0

你运行了sysctl命令来设置mmap_min_addr的值还是只编辑conf文件?你必须这样做。 –

+0

是的,之后我做了“sysctl -p”。 – user899159

回答

8

这听起来像内核已编译CONFIG_STRICT_DEVMEM启用。这是一项安全功能,可防止用户空间访问1MB以上(可能敏感)的物理内存(IIRC)。您可以使用sysctl dev.mem.restricted来禁用此功能。

+0

是的,我的.config有CONFIG_STRICT_DEVMEM = y。我如何使用“sysctl dev.mem.restricted”?我试过了,错误是:/ proc/sys/dev/mem/restricted:没有这样的文件或目录。 – user899159

+0

我想你必须在禁用选项的情况下重新编译内核。 –

+0

好吧,我重新编译了CONFIG_STRICT_DEVMEM关闭的内核。现在我有一个新的错误:打开/ dev/mem。 mem_map位于物理地址:0x356db000 内存映射失败。 err num = 22 mmap:无效的参数。如果我试图将物理地址映射到0,这不会发生。有什么建议吗? - 谢谢。 – user899159

0

我有一个类似的问题,当我尝试在Arch Linux的APU2c4电路板上使用flashrom时发生。

sysctl选项dev.mem.restricted在我的系统中不可用,并且使用自编译的内核对我来说没有选择。

我工作围绕这个问题通过通过蛴螬的iomem Kernelparameter设置为relaxed

# /boot/grub/grub.cfg 
linux /boot/vmlinuz-linux iomem=relaxed 

当然的重启对于所必要的这个解决方案。

参考:
https://www.reddit.com/r/libreboot/comments/6wvyry/flashrom_failures_to_access/
https://www.flashrom.org/FAQ
https://www.kernel.org/doc/Documentation/admin-guide/kernel-parameters.txt