mmap 文件,其中包含一个不受该文件支持的额外页面

mmap file with one additional page that is not backed by the file

我需要以只读模式映射一个文件,但需要在映射末尾添加一个以 \0 编辑的页面。我最初的想法是映射一个额外的页面并通过写入声明它。

int file=::open(name,O_RDONLY);
size_t size=lseek(file,0,SEEK_END);
size_t pageSize=::sysconf( _SC_PAGESIZE);
int padding=size%pageSize;
size_t mapSize=size+padding+pageSize;
void* mapping=mmap(0,mapSize,PROT_READ|PROT_WRITE,MAP_PRIVATE,file,0);
*(static_cast<char*>(mapping)+size+padding+1)=0;

但是,如 mmap 文档中所述,当我写入附加页面(以及从那里读取时)时,我会收到 SIGBUS。

我的问题是,我可以在不修改实际文件的情况下以某种方式在文件后声明一个额外的页面吗?

Linux 支持强制 mmap() 段放置的 MAP_FIXED 标志,因此您所需要做的就是正确计算地址调用第二个 mmap() 将映射匿名页面(当前您计算的 padding 不正确)。

int file = ::open("mmap", O_RDONLY);
size_t size = ::lseek(file, 0, SEEK_END);
size_t pageSize = ::sysconf(_SC_PAGESIZE);

int padding = pageSize - size % pageSize;
size_t mapSize = size + padding;

void* mapping = ::mmap(0, mapSize, PROT_READ | PROT_WRITE, 
                     MAP_PRIVATE, file, 0);
void* lastPage = static_cast<char*>(mapping) + mapSize;

void* mapping2 = ::mmap(lastPage, pageSize, PROT_READ | PROT_WRITE, 
                        MAP_PRIVATE | MAP_ANONYMOUS | MAP_FIXED, -1, 0);

* static_cast<char*>(mapping ) = 1;
* static_cast<char*>(mapping2) = 1;