c - 为什么 remap_file_pages() 在此示例中失败?

标签 c linux mmap memory-mapped-files

以下 C 代码说明了我在 Linux 2.6.30.5-43.fc11.x86_64 上遇到的问题:

#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>

int main() {
    char buf[1024];
    void *base;
    int fd;
    size_t pagesz = sysconf(_SC_PAGE_SIZE);

    fd = open("<some file, at least 4*pagesz in length>", O_RDONLY);
    if (fd < 0) {
       perror("open");
       return 1;
    }

    base = mmap(0, 4*pagesz, PROT_READ, MAP_SHARED, fd, 0);
    if (base < 0) {
        perror("mmap");
        close(fd);
        return 1;
    }

    memcpy(buf, (char*)base + 2*pagesz, 1024);

    if (remap_file_pages(base, pagesz, 0, 2, 0) < 0) {
        perror("remap_file_pages");
        munmap(base, 4*pagesz);
        close(fd);
        return 1;
    }

    printf("%d\n", memcmp(buf, base, 1024));

    munmap(base, 4*pagesz);
    close(fd);
    return 0;
}

这总是失败,remap_file_pages() 返回 -1 并且 errno 设置为 EINVAL。查看内核源代码,我可以在 remap_file_pages() 中看到它可能会失败的所有条件,但它们似乎都不适用于我的示例。怎么回事?

最佳答案

这是由于打开文件 O_RDONLY 引起的。如果将打开模式更改为 O_RDWR,它会起作用(即使 mmap() 仍然仅指定 PROT_READ)。

do_mmap_pgoff 中的这段代码是根本原因 - 它仅在打开文件进行写入时将 vma 标记为 VM_SHARED:

vm_flags |= VM_SHARED | VM_MAYSHARE;
if (!(file->f_mode & FMODE_WRITE))
    vm_flags &= ~(VM_MAYWRITE | VM_SHARED);

所以在 remap_file_pages() 中,您在第一次检查时失败了:

if (!vma || !(vma->vm_flags & VM_SHARED))
    goto out;

关于c - 为什么 remap_file_pages() 在此示例中失败?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2221453/

相关文章:

java - 如何查找两个数字是否是格雷码序列中的连续数字

linux - GCC/ELF - 我的符号从哪里来?

c - 使用 mmap 的共享变量导致分段违规 (SIGSEGV)

linux - git checkout --patch,但失败,无法在@INC 中找到 Git.pm

c++ - Windows Linux C++编译问题: "no matching function for call"

c - NetBSD 上的 libmagic 关闭文件描述符

c++ - 将 mmap 偏移量的最后 12 位归零

c - Macro中struct的问题定义

c - 通过引用访问 va_arg 宏是错误的风格还是不正确的?

C:在 sizeof() 内转换数组时出现意外结果