我正在尝试将一个函数加载到映射的内存缓冲区并稍后调用它,所以我做了一个测试用例来尝试:
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
哪个工作正常,但一旦我尝试做一些甚至微不足道的事情我就会得到一个段错误.
我试着像这样做一个简单的变量赋值:
int x;
auto func() -> void{
x = 5;
}
现在我的函数调用segfaults.我已适当地更改了缓冲区大小,并确定正在将正确的内存写入缓冲区.
我在这里错过了哪些重要的信息?为什么我不能这样做?
附:请不要向我讲授不安全的代码,这是一个简单的个人学习练习.
最佳答案
忽略这是明显的未定义行为这一事实,如果您执行全局变量的赋值,生成的代码可能会使用relative addressing来引用某些体系结构上的变量.
也就是说,函数期望自己和x在给定的地址,如果你移动它,事情就会破裂.
这是我的GCC为您的测试功能生成的:
x:
.zero 4
.text
.globl _Z4funcv
.type _Z4funcv,@function
_Z4funcv:
.LFB2:
.cfi_startproc
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset 6,-16
movq %rsp,%rbp
.cfi_def_cfa_register 6
movl $5,x(%rip)
nop
popq %rbp
.cfi_def_cfa 7,8
ret
.cfi_endproc
注意movl $5,x(%rip),这意味着代码使用自己的地址(存储在%rip中)来计算x和store 5的位置.
所以简而言之,没有简单的方法可以做你想做的事情,除非你确保你的功能只有position-independent code.即便如此,它只是要求麻烦.
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。