【发布时间】:2015-08-27 15:18:46
【问题描述】:
我正在尝试将一个函数加载到映射的内存缓冲区中并稍后调用它,因此我制作了一个测试用例进行尝试:
auto func() -> void{
asm(
"nop;"
"nop;"
"nop;"
"nop;"
);
}
auto main(int argc, char *argv[]) -> int{
void *exec_mem = mmap(nullptr, getpagesize(), PROT_READ | PROT_WRITE | PROT_EXEC, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
// check errors here
memcpy(exec_mem, reinterpret_cast<const void*>(func), 5); // size is known
(reinterpret_cast<void(*)()>(exec_mem))(); // function call
munmap(exec_mem, getpagesize());
}
这很好用,但是一旦我尝试做一些微不足道的事情,我就会遇到段错误。
我试着做一个像这样的简单变量赋值:
int x;
auto func() -> void{
x = 5;
}
现在我的函数调用段错误。我已经适当地更改了缓冲区大小,并且确定正在将正确的内存写入缓冲区。
我在这里遗漏了哪些重要信息?为什么我不能这样做?
附:请不要教我不安全的代码,这是一个简单的个人学习练习。
【问题讨论】:
-
即使考虑编写这种代码,你的工作也一定很不安全。
-
@EdHeal 是的,我不是为工作而编程的。
-
只是好奇,这个语法是什么:`auto / ->' ?以前没见过。
-
@EugeneSh。 C++11 尾随返回类型语法。在这种情况下使用它真的很简单;我只是随机决定使用它哈哈
-
无论如何。一个函数可能包含到其他内存位置的相对跳转,这些跳转不会被复制。另外,你说的大小是已知的,真的是已知的吗?包括参数处理和返回码?