لماذا الفشل remap_file_pages () في هذا المثال؟
-
19-09-2019 - |
سؤال
يوضح رمز 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. بالنظر إلى مصدر kernel أستطيع أن أرى كل الشروط في 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;
لا تنتمي إلى StackOverflow