在你选择的开发和运行环境下,写一个函数print_stackframe()
,用于获取当前位置的函数调用栈信息。实现获取以下一种或多种信息的功能:函数入口地址、函数名、参数调用参数、返回值
STACKFRAME_DEPTH = 10
NN = 4
void
print_stackframe(void) {
uint32_t ebp = read_ebp();
uint32_t eip = read_eip();
for (int i = 0; ebp != 0 && i < STACKFRAME_DEPTH; i ++) {
cprintf("ebp:0x%08x eip:0x%08x args:", ebp, eip);
uint32_t *arg = (uint32_t *)ebp + 2;
for (int j = 0; j < NN; j ++) {
cprintf("0x%08x ", arg[j]);
}
cprintf("\n");
print_debuginfo(eip - 1);
eip=((uint32_t *)ebp)[1];
ebp=((uint32_t *)ebp)[0];
}
}
每当进入一个新的函数,就会默认调用以下汇编代码:
pushl %ebp
movl %esp , %ebp
这是为了记录下来ebp的位置,便于函数调用结束的时候可以返回。
调用的时候,先将栈底指针ebp压栈,然后把栈顶指针赋给ebp,相当于此时ebp就是被调用函数的栈底。
需要注意的几点:
1.read_eip读取的不是当前eip寄存器的值,而是调用者的返回地址即ss:[ebp+4]处的值
2.ebp+8等是(可能的)参数。
3.print_debuginfo函数需要再进一步探究
参考:
https://www.cnblogs.com/xxrxxr/p/9527344.html
https://www.cnblogs.com/lsxkugou/p/14230375.html