【问题标题】:mmap: Operation not permittedmmap:不允许操作
【发布时间】: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”。

标签: c linux


【解决方案1】:

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

【讨论】:

  • 是的,我的 .config 有 CONFIG_STRICT_DEVMEM=y。如何使用“sysctl dev.mem.restricted”?我试过了,错误是:/proc/sys/dev/mem/restricted: No such file or directory.
  • 我猜你必须在禁用该选项的情况下重新编译内核。
  • 好的,我重新编译了内核并关闭了 CONFIG_STRICT_DEVMEM。现在我有一个新错误:/dev/mem 打开。 mem_map 位于物理地址:0x356db000 内存映射失败。 err num = 22 mmap:参数无效。如果我尝试将物理地址映射为 0,则不会发生这种情况。有什么建议吗? - 谢谢。
  • @user899159,这确实是一个不同的问题。我不知道你使用的地址来自哪里。
  • 在较新的内核上,CONFIG_STRICT_DEVMEM 由 HARDENED_USERCOPY 和 HAVE_HARDENED_USERCOPY_ALLOCATOR 隐含。所以一定要禁用它们。
【解决方案2】:

当我尝试在带有 Arch Linux 的 APU2c4 板上使用 flashrom 时,我遇到了类似的问题。

sysctl 选项 dev.mem.restricted 在我的系统中不可用,并且我无法选择使用自编译内核。

我通过 Grub 将 iomem 内核参数设置为 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

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2021-02-10
    • 2021-01-01
    • 2014-11-01
    • 2014-06-30
    • 2012-03-19
    • 2023-04-03
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多