gpt4 book ai didi

c++ - 多次调用后 mprotect 失败

转载 作者:行者123 更新时间:2023-11-28 02:02:23 25 4
gpt4 key购买 nike

我正在尝试编写一个程序,在其中分配一 block 内存,然后有选择地更改该 block 的页面大小和页面对齐子集的保护。但是,当我尝试对 block 中大于 8 页的内存部分调用 mprotect 时,mprotect 失败并显示错误“无法分配内存”。

这是重现问题的最小、完整且可验证的示例:

#include <sys/mman.h>
#include <unistd.h>
#include <stdlib.h>
#include <cerrno>
#include <cstring>
#include <iostream>

int main() {
const int n_pages = 12;
const int page_size = sysconf(_SC_PAGE_SIZE); // 4096
const int block_size = n_pages * page_size; // 65536

char* block_addr = (char*)aligned_alloc(page_size, block_size);
char* chunks[n_pages];
char* pointer = block_addr;

for (int n = 0; n < n_pages; n++) {
pointer = pointer + (n * page_size);
chunks[n] = pointer;
}
std::cout << "Set chunks read-only.\n";
for (int n = 0; n < n_pages; n++) {
if (mprotect(chunks[n], page_size, PROT_READ) != 0) {
std::cerr << n+1 << '/' << n_pages << ' '
<< "mprotect failed: " << std::strerror(errno) << '\n';
}
}
std::cout << "Set chunks read/write.\n";
for (int n = 0; n < n_pages; n++) {
if (mprotect(chunks[n], page_size, PROT_READ|PROT_WRITE) != 0) {
std::cerr << n+1 << '/' << n_pages << ' '
<< "mprotect failed: " << std::strerror(errno) << '\n';
}
}
free(block_addr);
}

这对于 block n>8 始终失败,并给出以下消息:

Set chunks read-only.
9/12 mprotect failed: Cannot allocate memory
10/12 mprotect failed: Cannot allocate memory
11/12 mprotect failed: Cannot allocate memory
12/12 mprotect failed: Cannot allocate memory
Set chunks read/write.
9/12 mprotect failed: Cannot allocate memory
10/12 mprotect failed: Cannot allocate memory
11/12 mprotect failed: Cannot allocate memory
12/12 mprotect failed: Cannot allocate memory

我找到了一个 question其中OP似乎得到与我相同的错误,其中David Hammen有用地为问题的根源提供了一些提示,但我真的不明白他在说什么。不幸的是,OP 不提供他们的代码,因此我们无法确切知道他们在做什么或如何修复它。

所以基本上我的问题是:为什么 mprotect 会产生该错误,我该如何修复它?

最佳答案

    pointer = pointer + (n * page_size);

这一行看起来很可疑。应该是吧

    pointer = block_addr + (n * page_size);

    chunks[n] = pointer;  // need to assign to chunks[n] first
pointer = pointer + page_size;

否则 pointer 将走开(步长为 0, 1 (= 0 + 1), 3 (= 0 + 1 + 2), 6 (= 0 + 1 + 2 + 3) , ...) 到不属于您的进程的内存中,因为您没有分配它(即它是“未映射的”)。这可以解释为什么 mprotect 对最后几个 block 失败。

关于c++ - 多次调用后 mprotect 失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38931295/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com